eth-accounts-index/solidity/AccountsIndex.sol

155 lines
4.2 KiB
Solidity
Raw Normal View History

pragma solidity >0.6.12;
2020-11-13 18:33:39 +01:00
// SPDX-License-Identifier: GPL-3.0-or-later
// File-Version: 2
2020-11-13 18:33:39 +01:00
contract CustodialAccountIndex {
2023-02-12 08:51:58 +01:00
uint256 constant blockedField = 1 << 128;
address[] entryList;
mapping(address => uint256) public entryIndex;
2023-03-21 14:26:55 +01:00
mapping(address => bool) public isWriter;
2021-04-30 11:36:56 +02:00
address public owner;
address newOwner;
2020-11-13 18:33:39 +01:00
2023-02-12 08:51:58 +01:00
event AddressAdded(address indexed _executor, address _account); // AccountsIndex
event AddressActive(address indexed _executor, address _account);
event AddressInactive(address indexed _executor, address _account);
event AddressRemoved(address indexed _executor, address _account);
2021-04-30 11:36:56 +02:00
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); // EIP173
2023-03-21 14:26:55 +01:00
event WriterAdded(address indexed _executor, address _account); // AccountsIndex
event WriterDeleted(address indexed _executor, address _account);
2020-11-13 18:33:39 +01:00
2023-03-17 22:35:12 +01:00
constructor() {
2020-11-13 18:33:39 +01:00
owner = msg.sender;
2023-02-12 08:51:58 +01:00
entryList.push(address(0));
2023-02-12 07:07:14 +01:00
}
2023-03-17 22:35:12 +01:00
function entryCount() external view returns (uint256) {
2023-02-12 08:51:58 +01:00
return entryList.length - 1;
2020-11-13 18:33:39 +01:00
}
function addWriter(address _writer) public returns (bool) {
2020-11-13 18:33:39 +01:00
require(owner == msg.sender);
2023-03-21 14:26:55 +01:00
isWriter[_writer] = true;
emit WriterAdded(msg.sender, _writer);
return true;
2020-11-13 18:33:39 +01:00
}
function deleteWriter(address _writer) public returns (bool) {
2020-11-13 18:33:39 +01:00
require(owner == msg.sender);
2023-03-21 14:26:55 +01:00
delete isWriter[_writer];
emit WriterDeleted(msg.sender, _writer);
return true;
2020-11-13 18:33:39 +01:00
}
// Implements AccountsIndex
function add(address _account) external returns (bool) {
uint256 i;
2023-02-12 08:51:58 +01:00
uint256 _entry;
2023-03-21 14:26:55 +01:00
require(isWriter[msg.sender]);
2021-04-30 11:36:56 +02:00
require(entryIndex[_account] == 0);
2023-02-12 08:51:58 +01:00
require(entryList.length < (1 << 64));
i = entryList.length;
entryList.push(_account);
_entry = uint64(i);
_entry |= block.timestamp << 64;
entryIndex[_account] = _entry;
emit AddressAdded(msg.sender, _account);
return true;
}
// Implements AccountsIndex
2023-02-12 08:51:58 +01:00
function remove(address _account) external returns (bool) {
uint256 i;
uint256 l;
2023-03-21 14:26:55 +01:00
require(isWriter[msg.sender]);
2023-02-12 08:51:58 +01:00
require(this.have(_account));
l = entryList.length - 1;
i = entryIndex[_account];
if (i < l) {
entryList[i] = entryList[l];
}
entryList.pop();
entryIndex[_account] = 0;
emit AddressRemoved(msg.sender, _account);
return true;
}
// Implements AccountsIndex
// Activate previously deactivated account. Will not affect the entry count.
function activate(address _account) external returns (bool) {
2023-03-21 14:26:55 +01:00
require(isWriter[msg.sender]);
require(entryIndex[_account] > 0 && entryIndex[_account] & blockedField == blockedField);
2023-02-12 08:51:58 +01:00
entryIndex[_account] >>= 129;
emit AddressActive(msg.sender, _account);
2023-03-17 22:35:12 +01:00
return true;
}
2023-02-12 08:51:58 +01:00
// Implements AccountsIndex
// Deactivate account, without removing the entry. The entry will still be part of the entry count.
function deactivate(address _account) external returns (bool) {
2023-03-21 14:26:55 +01:00
require(isWriter[msg.sender]);
2023-02-12 08:51:58 +01:00
require(entryIndex[_account] > 0 && entryIndex[_account] & blockedField == 0);
entryIndex[_account] <<= 129;
entryIndex[_account] |= blockedField;
emit AddressInactive(msg.sender, _account);
return true;
}
// Implements AccountsIndex
2023-03-17 22:35:12 +01:00
function entry(uint256 _i) external view returns (address) {
2023-02-12 08:51:58 +01:00
return entryList[_i + 1];
}
// Implements AccountsIndex
2023-03-17 22:35:12 +01:00
function time(address _account) external view returns (uint256) {
2023-02-12 08:51:58 +01:00
require(entryIndex[_account] > 0);
return entryIndex[_account] >> 64;
}
// Implements AccountsIndex
2023-02-05 05:54:50 +01:00
function have(address _account) external view returns (bool) {
return entryIndex[_account] > 0;
2021-04-30 11:36:56 +02:00
}
// Implements AccountsIndex
function isActive(address _account) external view returns (bool) {
2023-02-12 07:07:14 +01:00
return this.have(_account) && entryIndex[_account] & blockedField != blockedField;
}
// Implements EIP173
2021-04-30 11:36:56 +02:00
function transferOwnership(address _newOwner) public returns (bool) {
address oldOwner;
2021-04-30 11:36:56 +02:00
require(msg.sender == owner);
oldOwner = owner;
owner = _newOwner;
2021-04-30 11:36:56 +02:00
emit OwnershipTransferred(oldOwner, owner);
2023-03-17 22:35:12 +01:00
return true;
2021-04-30 11:36:56 +02:00
}
// Implements EIP165
2021-04-30 11:36:56 +02:00
function supportsInterface(bytes4 _sum) public pure returns (bool) {
if (_sum == 0xcbdb05c7) { // AccountsIndex
return true;
}
if (_sum == 0x01ffc9a7) { // EIP165
return true;
}
if (_sum == 0x9493f8b2) { // EIP173
return true;
}
2021-10-24 15:25:33 +02:00
if (_sum == 0x80c84bd6) { // Writer
return true;
}
2021-04-30 11:36:56 +02:00
return false;
2020-11-13 18:33:39 +01:00
}
}