2018-03-30 02:31:08 -03:00
|
|
|
pragma solidity ^0.4.21;
|
2018-02-21 17:17:38 -04:00
|
|
|
|
|
|
|
import "./ERC725.sol";
|
|
|
|
import "./ERC735.sol";
|
2018-05-08 20:48:18 -03:00
|
|
|
import "../common/MessageSigned.sol";
|
2018-02-21 17:17:38 -04:00
|
|
|
|
2018-05-08 20:48:18 -03:00
|
|
|
contract Identity is ERC725, ERC735, MessageSigned {
|
2018-02-21 17:17:38 -04:00
|
|
|
|
|
|
|
mapping (bytes32 => Key) keys;
|
|
|
|
mapping (uint256 => bytes32[]) keysByPurpose;
|
2018-03-24 05:30:07 -03:00
|
|
|
mapping (bytes32 => Claim) claims;
|
2018-02-21 17:17:38 -04:00
|
|
|
mapping (uint256 => bytes32[]) claimsByType;
|
2018-03-24 05:30:07 -03:00
|
|
|
|
2018-02-21 17:17:38 -04:00
|
|
|
mapping (bytes32 => uint256) indexes;
|
2018-05-09 01:18:35 -03:00
|
|
|
mapping (uint256 => Transaction) multisigTx;
|
2018-03-24 05:58:04 -03:00
|
|
|
mapping (uint256 => uint256) purposeThreshold;
|
2018-03-24 05:53:35 -03:00
|
|
|
|
2018-05-09 01:18:35 -03:00
|
|
|
uint256 txCount;
|
2018-03-24 05:58:04 -03:00
|
|
|
uint256 nonce;
|
2018-02-27 13:23:00 -04:00
|
|
|
address recoveryContract;
|
2018-04-23 08:04:03 -03:00
|
|
|
bytes32 recoveryManager;
|
2018-02-21 17:17:38 -04:00
|
|
|
|
|
|
|
struct Transaction {
|
2018-03-24 05:53:35 -03:00
|
|
|
bool valid;
|
2018-02-21 17:17:38 -04:00
|
|
|
address to;
|
2018-03-24 05:58:04 -03:00
|
|
|
uint256 value;
|
2018-02-21 17:17:38 -04:00
|
|
|
bytes data;
|
2018-03-24 05:58:04 -03:00
|
|
|
uint256 approverCount;
|
2018-02-21 17:17:38 -04:00
|
|
|
mapping(bytes32 => bool) approvals;
|
|
|
|
}
|
2018-05-09 01:18:35 -03:00
|
|
|
|
2018-05-09 01:37:38 -03:00
|
|
|
modifier selfOrSenderKey(uint256 keyPurpose) {
|
2018-03-21 23:30:54 -03:00
|
|
|
if(msg.sender == address(this)) {
|
|
|
|
_;
|
|
|
|
} else {
|
2018-05-08 20:29:42 -03:00
|
|
|
require(isKeyPurpose(keccak256(msg.sender), keyPurpose));
|
|
|
|
if (purposeThreshold[keyPurpose] == 1) {
|
2018-03-21 23:30:54 -03:00
|
|
|
_;
|
|
|
|
} else {
|
|
|
|
execute(address(this), 0, msg.data);
|
|
|
|
}
|
|
|
|
}
|
2018-02-21 17:17:38 -04:00
|
|
|
}
|
2018-05-08 20:29:42 -03:00
|
|
|
|
2018-03-02 11:32:00 -03:00
|
|
|
modifier recoveryOnly {
|
|
|
|
require(
|
2018-03-21 13:13:04 -03:00
|
|
|
recoveryContract != address(0) &&
|
2018-04-23 08:04:03 -03:00
|
|
|
msg.sender == recoveryContract
|
2018-02-27 13:23:00 -04:00
|
|
|
);
|
2018-02-21 17:17:38 -04:00
|
|
|
_;
|
|
|
|
}
|
|
|
|
|
2018-05-09 01:18:35 -03:00
|
|
|
modifier keyMessageSigned (
|
2018-03-03 00:00:11 -03:00
|
|
|
bytes32 _key,
|
2018-05-08 20:48:18 -03:00
|
|
|
bytes32 _messageHash,
|
|
|
|
bytes _signature
|
|
|
|
) {
|
2018-03-19 23:15:22 -03:00
|
|
|
require(
|
2018-04-23 08:04:03 -03:00
|
|
|
_key == keccak256(
|
2018-05-08 20:48:18 -03:00
|
|
|
recoverAddress(
|
|
|
|
getSignHash(_messageHash),
|
|
|
|
_signature
|
2018-04-23 08:04:03 -03:00
|
|
|
)
|
2018-03-19 23:15:22 -03:00
|
|
|
)
|
|
|
|
);
|
2018-03-03 00:00:11 -03:00
|
|
|
require(keys[_key].purpose != 0);
|
|
|
|
_;
|
|
|
|
}
|
2018-01-08 19:00:07 -06:00
|
|
|
|
2018-04-23 08:04:03 -03:00
|
|
|
constructor() public {
|
|
|
|
_constructIdentity(keccak256(msg.sender));
|
2018-03-03 17:16:32 -03:00
|
|
|
}
|
2018-03-03 00:00:11 -03:00
|
|
|
|
|
|
|
function ()
|
|
|
|
public
|
|
|
|
payable
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-05-09 01:34:33 -03:00
|
|
|
////////////////
|
|
|
|
// Execute calls and multisig approval
|
|
|
|
////////////////
|
|
|
|
|
|
|
|
function execute(
|
|
|
|
address _to,
|
|
|
|
uint256 _value,
|
|
|
|
bytes _data
|
|
|
|
)
|
2018-03-03 00:00:11 -03:00
|
|
|
public
|
2018-05-09 01:34:33 -03:00
|
|
|
returns (uint256 txId)
|
2018-03-02 11:32:00 -03:00
|
|
|
{
|
2018-05-09 01:34:33 -03:00
|
|
|
txId = _execute(keccak256(msg.sender), _to, _value, _data);
|
|
|
|
}
|
|
|
|
|
|
|
|
function approve(uint256 _id, bool _approval)
|
|
|
|
public
|
|
|
|
returns (bool success)
|
|
|
|
{
|
|
|
|
return _approveRequest(keccak256(msg.sender), _id, _approval);
|
2018-03-02 11:32:00 -03:00
|
|
|
}
|
2018-05-09 01:34:33 -03:00
|
|
|
|
|
|
|
////////////////
|
|
|
|
// Message Signed functions
|
|
|
|
////////////////
|
2018-03-02 11:32:00 -03:00
|
|
|
|
2018-05-09 01:34:33 -03:00
|
|
|
function executeMessageSigned(
|
|
|
|
address _to,
|
|
|
|
uint256 _value,
|
|
|
|
bytes _data,
|
|
|
|
uint256 _nonce,
|
|
|
|
bytes32 _key,
|
|
|
|
bytes _signature
|
|
|
|
)
|
2018-03-02 15:00:46 -04:00
|
|
|
public
|
2018-05-09 01:34:33 -03:00
|
|
|
keyMessageSigned(
|
|
|
|
_key,
|
|
|
|
keccak256(
|
|
|
|
address(this),
|
|
|
|
bytes4(keccak256("execute(address,uint256,bytes)")),
|
|
|
|
_to,
|
|
|
|
_value,
|
|
|
|
_data,
|
|
|
|
_nonce
|
|
|
|
),
|
|
|
|
_signature
|
|
|
|
)
|
|
|
|
returns (uint256 txId)
|
2018-03-02 11:32:00 -03:00
|
|
|
{
|
2018-05-09 01:34:33 -03:00
|
|
|
txId = _execute(_key, _to, _value, _data);
|
|
|
|
}
|
2018-03-02 15:00:46 -04:00
|
|
|
|
2018-05-09 01:34:33 -03:00
|
|
|
function approveMessageSigned(
|
|
|
|
uint256 _id,
|
|
|
|
bool _approval,
|
|
|
|
bytes32 _key,
|
|
|
|
bytes _signature
|
|
|
|
)
|
|
|
|
public
|
|
|
|
keyMessageSigned(
|
|
|
|
_key,
|
|
|
|
keccak256(
|
|
|
|
address(this),
|
|
|
|
bytes4(keccak256("approve(uint256,bool)")),
|
|
|
|
_id,
|
|
|
|
_approval
|
|
|
|
),
|
|
|
|
_signature
|
|
|
|
)
|
|
|
|
returns (bool success)
|
|
|
|
{
|
|
|
|
return _approveRequest(_key, _id, _approval);
|
2018-02-21 17:17:38 -04:00
|
|
|
}
|
2018-05-09 01:34:33 -03:00
|
|
|
|
|
|
|
////////////////
|
|
|
|
// Management functions
|
|
|
|
////////////////
|
2018-01-08 19:00:07 -06:00
|
|
|
|
2018-02-22 05:05:10 -03:00
|
|
|
function addKey(
|
|
|
|
bytes32 _key,
|
|
|
|
uint256 _purpose,
|
|
|
|
uint256 _type
|
|
|
|
)
|
|
|
|
public
|
2018-05-09 01:37:38 -03:00
|
|
|
selfOrSenderKey(MANAGEMENT_KEY)
|
2018-02-22 05:05:10 -03:00
|
|
|
returns (bool success)
|
2018-03-02 11:32:00 -03:00
|
|
|
{
|
2018-02-21 17:17:38 -04:00
|
|
|
_addKey(_key, _purpose, _type);
|
|
|
|
return true;
|
|
|
|
}
|
2018-01-08 19:00:07 -06:00
|
|
|
|
2018-03-02 11:32:00 -03:00
|
|
|
function replaceKey(
|
|
|
|
bytes32 _oldKey,
|
|
|
|
bytes32 _newKey,
|
|
|
|
uint256 _newType
|
|
|
|
)
|
|
|
|
public
|
2018-05-09 01:37:38 -03:00
|
|
|
selfOrSenderKey(MANAGEMENT_KEY)
|
2018-03-02 11:32:00 -03:00
|
|
|
returns (bool success)
|
|
|
|
{
|
2018-03-02 15:00:46 -04:00
|
|
|
uint256 purpose = keys[_oldKey].purpose;
|
2018-03-02 11:32:00 -03:00
|
|
|
_addKey(_newKey, purpose, _newType);
|
|
|
|
_removeKey(_oldKey, purpose);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-02-22 05:05:10 -03:00
|
|
|
function removeKey(
|
|
|
|
bytes32 _key,
|
|
|
|
uint256 _purpose
|
|
|
|
)
|
|
|
|
public
|
2018-05-09 01:37:38 -03:00
|
|
|
selfOrSenderKey(MANAGEMENT_KEY)
|
2018-02-22 05:05:10 -03:00
|
|
|
returns (bool success)
|
|
|
|
{
|
2018-02-21 17:17:38 -04:00
|
|
|
_removeKey(_key, _purpose);
|
|
|
|
return true;
|
|
|
|
}
|
2018-01-08 19:00:07 -06:00
|
|
|
|
2018-03-01 11:42:44 -04:00
|
|
|
function setMinimumApprovalsByKeyType(
|
2018-03-02 15:00:46 -04:00
|
|
|
uint256 _purpose,
|
|
|
|
uint256 _minimumApprovals
|
2018-02-22 05:16:06 -03:00
|
|
|
)
|
|
|
|
public
|
2018-05-09 01:37:38 -03:00
|
|
|
selfOrSenderKey(MANAGEMENT_KEY)
|
2018-02-22 05:16:06 -03:00
|
|
|
{
|
2018-02-28 01:36:48 -03:00
|
|
|
require(_minimumApprovals > 0);
|
2018-03-02 15:00:46 -04:00
|
|
|
require(_minimumApprovals <= keysByPurpose[_purpose].length);
|
2018-03-24 05:58:04 -03:00
|
|
|
purposeThreshold[_purpose] = _minimumApprovals;
|
2018-02-21 17:17:38 -04:00
|
|
|
}
|
2018-02-22 14:58:25 -04:00
|
|
|
|
2018-05-09 01:34:33 -03:00
|
|
|
function setupRecovery(address _recoveryContract)
|
|
|
|
public
|
2018-05-09 01:37:38 -03:00
|
|
|
selfOrSenderKey(MANAGEMENT_KEY)
|
2018-05-09 01:34:33 -03:00
|
|
|
{
|
|
|
|
require(recoveryContract == address(0));
|
|
|
|
recoveryContract = _recoveryContract;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////
|
|
|
|
// Claim related
|
|
|
|
////////////////
|
|
|
|
|
2018-02-22 05:05:10 -03:00
|
|
|
function addClaim(
|
|
|
|
uint256 _claimType,
|
|
|
|
uint256 _scheme,
|
|
|
|
address _issuer,
|
|
|
|
bytes _signature,
|
|
|
|
bytes _data,
|
|
|
|
string _uri
|
|
|
|
)
|
2018-02-21 17:17:38 -04:00
|
|
|
public
|
2018-02-23 14:02:53 -03:00
|
|
|
returns (bytes32 claimHash)
|
2018-02-21 17:17:38 -04:00
|
|
|
{
|
2018-02-23 14:02:53 -03:00
|
|
|
claimHash = keccak256(_issuer, _claimType);
|
|
|
|
if (msg.sender == address(this)) {
|
|
|
|
if (claims[claimHash].claimType > 0) {
|
|
|
|
_modifyClaim(claimHash, _claimType, _scheme, _issuer, _signature, _data, _uri);
|
|
|
|
} else {
|
|
|
|
_includeClaim(claimHash, _claimType, _scheme, _issuer, _signature, _data, _uri);
|
|
|
|
}
|
2018-02-21 17:17:38 -04:00
|
|
|
} else {
|
2018-04-23 08:04:03 -03:00
|
|
|
require(isKeyPurpose(keccak256(msg.sender), CLAIM_SIGNER_KEY));
|
2018-05-09 01:18:35 -03:00
|
|
|
_requestApproval(address(this), 0, msg.data);
|
2018-03-19 23:15:22 -03:00
|
|
|
emit ClaimRequested(
|
2018-02-23 14:02:53 -03:00
|
|
|
claimHash,
|
2018-02-22 05:05:10 -03:00
|
|
|
_claimType,
|
|
|
|
_scheme,
|
|
|
|
_issuer,
|
|
|
|
_signature,
|
|
|
|
_data,
|
2018-02-23 14:02:53 -03:00
|
|
|
_uri
|
|
|
|
);
|
2018-01-08 19:00:07 -06:00
|
|
|
}
|
2018-02-21 17:17:38 -04:00
|
|
|
}
|
2018-05-09 01:18:35 -03:00
|
|
|
|
2018-03-03 00:00:11 -03:00
|
|
|
function removeClaim(bytes32 _claimId)
|
|
|
|
public
|
|
|
|
returns (bool success)
|
|
|
|
{
|
2018-02-21 17:17:38 -04:00
|
|
|
Claim memory c = claims[_claimId];
|
|
|
|
|
2018-02-22 05:05:10 -03:00
|
|
|
require(
|
|
|
|
msg.sender == c.issuer ||
|
2018-02-23 14:17:57 -03:00
|
|
|
msg.sender == address(this)
|
2018-02-22 05:05:10 -03:00
|
|
|
);
|
2018-02-21 17:17:38 -04:00
|
|
|
|
|
|
|
// MUST only be done by the issuer of the claim, or KEYS OF PURPOSE 1, or the identity itself.
|
|
|
|
// TODO If its the identity itself, the approval process will determine its approval.
|
|
|
|
|
2018-03-24 05:58:04 -03:00
|
|
|
uint256 claimIdTypePos = indexes[_claimId];
|
2018-02-21 17:17:38 -04:00
|
|
|
delete indexes[_claimId];
|
|
|
|
bytes32[] storage claimsTypeArr = claimsByType[c.claimType];
|
|
|
|
bytes32 replacer = claimsTypeArr[claimsTypeArr.length-1];
|
|
|
|
claimsTypeArr[claimIdTypePos] = replacer;
|
|
|
|
indexes[replacer] = claimIdTypePos;
|
|
|
|
delete claims[_claimId];
|
|
|
|
claimsTypeArr.length--;
|
2018-04-24 15:58:04 -03:00
|
|
|
emit ClaimRemoved(_claimId, c.claimType, c.scheme, c.issuer, c.signature, c.data, c.uri);
|
2018-02-21 17:17:38 -04:00
|
|
|
return true;
|
|
|
|
}
|
2018-01-08 19:00:07 -06:00
|
|
|
|
2018-05-09 01:34:33 -03:00
|
|
|
////////////////
|
|
|
|
// Recovery methods
|
|
|
|
////////////////
|
|
|
|
|
|
|
|
function managerReset(bytes32 _newKey)
|
|
|
|
public
|
|
|
|
recoveryOnly
|
|
|
|
{
|
|
|
|
recoveryManager = _newKey;
|
|
|
|
_addKey(keccak256(recoveryManager), MANAGEMENT_KEY, 0);
|
|
|
|
purposeThreshold[MANAGEMENT_KEY] = keysByPurpose[MANAGEMENT_KEY].length;
|
|
|
|
}
|
|
|
|
|
|
|
|
function processManagerReset(uint256 _limit)
|
|
|
|
public
|
|
|
|
{
|
|
|
|
require(recoveryManager != 0);
|
|
|
|
uint256 limit = _limit;
|
|
|
|
bytes32 newKey = recoveryManager;
|
|
|
|
bytes32[] memory managers = keysByPurpose[MANAGEMENT_KEY];
|
|
|
|
uint256 totalManagers = managers.length;
|
|
|
|
|
|
|
|
if (limit == 0) {
|
|
|
|
limit = totalManagers;
|
|
|
|
}
|
|
|
|
|
|
|
|
purposeThreshold[MANAGEMENT_KEY] = totalManagers - limit + 1;
|
|
|
|
for (uint256 i = 0; i < limit; i++) {
|
|
|
|
bytes32 manager = managers[i];
|
|
|
|
if (manager != newKey) {
|
|
|
|
_removeKey(manager, MANAGEMENT_KEY);
|
|
|
|
totalManagers--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (totalManagers == 1) {
|
|
|
|
delete recoveryManager;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////
|
|
|
|
// Public Views
|
|
|
|
////////////////
|
|
|
|
|
2018-02-22 05:05:10 -03:00
|
|
|
function getKey(
|
|
|
|
bytes32 _key,
|
|
|
|
uint256 _purpose
|
|
|
|
)
|
|
|
|
public
|
2018-03-30 02:31:08 -03:00
|
|
|
view
|
2018-02-22 05:05:10 -03:00
|
|
|
returns(uint256 purpose, uint256 keyType, bytes32 key)
|
|
|
|
{
|
2018-02-21 17:17:38 -04:00
|
|
|
Key storage myKey = keys[keccak256(_key, _purpose)];
|
|
|
|
return (myKey.purpose, myKey.keyType, myKey.key);
|
|
|
|
}
|
2018-02-21 19:45:25 -03:00
|
|
|
|
2018-03-24 05:58:49 -03:00
|
|
|
function isKeyPurpose(bytes32 _key, uint256 _purpose)
|
2018-02-22 05:05:10 -03:00
|
|
|
public
|
2018-03-30 02:31:08 -03:00
|
|
|
view
|
2018-02-22 05:05:10 -03:00
|
|
|
returns (bool)
|
|
|
|
{
|
2018-03-24 05:58:49 -03:00
|
|
|
return keys[keccak256(_key, _purpose)].purpose == _purpose;
|
2018-02-21 19:45:25 -03:00
|
|
|
}
|
2018-01-08 19:00:07 -06:00
|
|
|
|
2018-02-22 05:05:10 -03:00
|
|
|
function getKeyPurpose(bytes32 _key)
|
|
|
|
public
|
2018-03-30 02:31:08 -03:00
|
|
|
view
|
2018-02-22 05:05:10 -03:00
|
|
|
returns(uint256[] purpose)
|
|
|
|
{
|
2018-02-21 17:17:38 -04:00
|
|
|
|
|
|
|
uint256[] memory purposeHolder = new uint256[](4);
|
|
|
|
uint8 counter = 0;
|
|
|
|
|
2018-03-21 12:50:03 -03:00
|
|
|
if (isKeyPurpose(_key, MANAGEMENT_KEY)) {
|
2018-02-21 17:17:38 -04:00
|
|
|
purposeHolder[counter] = MANAGEMENT_KEY;
|
|
|
|
counter++;
|
2018-01-08 19:00:07 -06:00
|
|
|
}
|
2018-02-21 17:17:38 -04:00
|
|
|
|
2018-03-21 12:50:03 -03:00
|
|
|
if (isKeyPurpose(_key, ACTION_KEY)) {
|
2018-02-21 17:17:38 -04:00
|
|
|
purposeHolder[counter] = ACTION_KEY;
|
|
|
|
counter++;
|
2018-01-08 19:00:07 -06:00
|
|
|
}
|
2018-02-21 17:17:38 -04:00
|
|
|
|
2018-03-21 12:50:03 -03:00
|
|
|
if (isKeyPurpose(_key, CLAIM_SIGNER_KEY)) {
|
2018-02-21 17:17:38 -04:00
|
|
|
purposeHolder[counter] = CLAIM_SIGNER_KEY;
|
|
|
|
counter++;
|
2018-01-08 19:00:07 -06:00
|
|
|
}
|
2018-02-21 17:17:38 -04:00
|
|
|
|
2018-03-21 12:50:03 -03:00
|
|
|
if (isKeyPurpose(_key, ENCRYPTION_KEY)) {
|
2018-02-21 17:17:38 -04:00
|
|
|
purposeHolder[counter] = ENCRYPTION_KEY;
|
|
|
|
counter++;
|
2018-01-08 19:00:07 -06:00
|
|
|
}
|
2018-02-21 17:17:38 -04:00
|
|
|
|
|
|
|
uint256[] memory result = new uint256[](counter);
|
2018-02-22 05:05:10 -03:00
|
|
|
for (uint8 i = 0; i < counter; i++) {
|
2018-02-21 17:17:38 -04:00
|
|
|
result[i] = purposeHolder[i];
|
2018-02-22 05:05:10 -03:00
|
|
|
}
|
2018-02-21 17:17:38 -04:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-02-22 05:05:10 -03:00
|
|
|
function getKeysByPurpose(uint256 _purpose)
|
|
|
|
public
|
2018-03-30 02:31:08 -03:00
|
|
|
view
|
2018-03-02 23:48:15 -03:00
|
|
|
returns(bytes32[])
|
2018-02-22 05:05:10 -03:00
|
|
|
{
|
2018-02-21 17:17:38 -04:00
|
|
|
return keysByPurpose[_purpose];
|
|
|
|
}
|
|
|
|
|
2018-02-22 05:05:10 -03:00
|
|
|
function getClaim(bytes32 _claimId)
|
|
|
|
public
|
2018-03-30 02:31:08 -03:00
|
|
|
view
|
2018-03-03 00:00:11 -03:00
|
|
|
returns(
|
|
|
|
uint256 claimType,
|
|
|
|
uint256 scheme,
|
|
|
|
address issuer,
|
|
|
|
bytes signature,
|
|
|
|
bytes data,
|
|
|
|
string uri
|
|
|
|
)
|
2018-02-22 05:05:10 -03:00
|
|
|
{
|
2018-02-21 17:17:38 -04:00
|
|
|
Claim memory _claim = claims[_claimId];
|
|
|
|
return (_claim.claimType, _claim.scheme, _claim.issuer, _claim.signature, _claim.data, _claim.uri);
|
|
|
|
}
|
|
|
|
|
2018-02-22 05:05:10 -03:00
|
|
|
function getClaimIdsByType(uint256 _claimType)
|
|
|
|
public
|
2018-03-30 02:31:08 -03:00
|
|
|
view
|
2018-02-22 05:05:10 -03:00
|
|
|
returns(bytes32[] claimIds)
|
|
|
|
{
|
2018-02-21 17:17:38 -04:00
|
|
|
return claimsByType[_claimType];
|
2018-01-08 19:00:07 -06:00
|
|
|
}
|
|
|
|
|
2018-05-09 01:34:33 -03:00
|
|
|
////////////////
|
|
|
|
// Internal methods
|
|
|
|
////////////////
|
2018-02-26 17:26:14 -04:00
|
|
|
|
2018-04-23 08:04:03 -03:00
|
|
|
function _constructIdentity(bytes32 _managerKey)
|
2018-03-03 17:16:32 -03:00
|
|
|
internal
|
|
|
|
{
|
2018-03-20 22:29:37 -03:00
|
|
|
require(keysByPurpose[MANAGEMENT_KEY].length == 0);
|
2018-03-24 05:58:04 -03:00
|
|
|
require(purposeThreshold[MANAGEMENT_KEY] == 0);
|
2018-04-23 08:04:03 -03:00
|
|
|
_addKey(_managerKey, MANAGEMENT_KEY, 0);
|
|
|
|
_addKey(_managerKey, ACTION_KEY, 0);
|
2018-03-03 17:16:32 -03:00
|
|
|
|
2018-03-24 05:58:04 -03:00
|
|
|
purposeThreshold[MANAGEMENT_KEY] = 1;
|
|
|
|
purposeThreshold[ACTION_KEY] = 1;
|
2018-03-03 17:16:32 -03:00
|
|
|
}
|
2018-02-26 17:26:14 -04:00
|
|
|
|
2018-03-02 23:54:36 -03:00
|
|
|
function _execute(
|
2018-05-09 01:18:35 -03:00
|
|
|
bytes32 _key,
|
2018-03-02 23:54:36 -03:00
|
|
|
address _to,
|
|
|
|
uint256 _value,
|
|
|
|
bytes _data
|
|
|
|
)
|
2018-05-09 01:18:35 -03:00
|
|
|
internal
|
|
|
|
returns (uint256 txId)
|
2018-03-02 23:54:36 -03:00
|
|
|
{
|
2018-05-09 01:18:35 -03:00
|
|
|
uint256 requiredPurpose = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY;
|
|
|
|
require(isKeyPurpose(_key, requiredPurpose));
|
|
|
|
if (purposeThreshold[requiredPurpose] == 1) {
|
|
|
|
txId = txCount++;
|
|
|
|
_commitCall(txId, _to, _value, _data);
|
|
|
|
} else {
|
|
|
|
txId = _requestApproval(_to, _value, _data);
|
|
|
|
_approveRequest(_key, txId, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-09 01:34:33 -03:00
|
|
|
function _commitCall(
|
|
|
|
uint256 _txId,
|
|
|
|
address _to,
|
|
|
|
uint256 _value,
|
|
|
|
bytes _data
|
|
|
|
)
|
|
|
|
internal
|
|
|
|
returns(bool success)
|
|
|
|
{
|
|
|
|
nonce++;
|
|
|
|
success = _to.call.value(_value)(_data);
|
|
|
|
if (success) {
|
|
|
|
emit Executed(_txId, _to, _value, _data);
|
|
|
|
} else {
|
|
|
|
emit ExecutionFailed(_txId, _to, _value, _data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-09 01:18:35 -03:00
|
|
|
function _requestApproval(
|
|
|
|
address _to,
|
|
|
|
uint256 _value,
|
|
|
|
bytes _data
|
|
|
|
)
|
|
|
|
internal
|
|
|
|
returns (uint256 txId)
|
|
|
|
{
|
|
|
|
txId = txCount++;
|
|
|
|
multisigTx[txCount] = Transaction({
|
2018-03-24 05:53:35 -03:00
|
|
|
valid: true,
|
2018-03-19 23:15:22 -03:00
|
|
|
to: _to,
|
|
|
|
value: _value,
|
|
|
|
data: _data,
|
|
|
|
approverCount: 0
|
2018-05-09 01:18:35 -03:00
|
|
|
});
|
|
|
|
emit ExecutionRequested(txId, _to, _value, _data);
|
2018-03-02 23:54:36 -03:00
|
|
|
}
|
2018-05-09 01:18:35 -03:00
|
|
|
|
2018-05-09 01:34:33 -03:00
|
|
|
////////////////
|
|
|
|
// Private methods
|
|
|
|
////////////////
|
|
|
|
|
2018-05-09 01:18:35 -03:00
|
|
|
function _approveRequest(
|
2018-03-02 23:54:36 -03:00
|
|
|
bytes32 _key,
|
|
|
|
uint256 _id,
|
|
|
|
bool _approval
|
|
|
|
)
|
|
|
|
private
|
2018-03-21 23:30:54 -03:00
|
|
|
returns(bool success) //(?) should return approved instead of success?
|
2018-03-02 23:54:36 -03:00
|
|
|
{
|
|
|
|
|
2018-05-09 01:18:35 -03:00
|
|
|
Transaction memory approvedTx = multisigTx[_id];
|
|
|
|
require(approvedTx.valid);
|
|
|
|
uint256 requiredKeyPurpose = approvedTx.to == address(this) ? MANAGEMENT_KEY : ACTION_KEY;
|
2018-03-24 05:53:35 -03:00
|
|
|
require(isKeyPurpose(_key, requiredKeyPurpose));
|
2018-05-09 01:18:35 -03:00
|
|
|
require(multisigTx[_id].approvals[_key] != _approval);
|
|
|
|
|
|
|
|
emit Approved(_id, _approval);
|
2018-03-02 23:54:36 -03:00
|
|
|
|
2018-03-24 05:53:35 -03:00
|
|
|
if (_approval) {
|
2018-05-09 01:18:35 -03:00
|
|
|
if (approvedTx.approverCount + 1 == purposeThreshold[requiredKeyPurpose]) {
|
|
|
|
delete multisigTx[_id];
|
|
|
|
return _commitCall(_id, approvedTx.to, approvedTx.value, approvedTx.data);
|
|
|
|
} else {
|
|
|
|
multisigTx[_id].approverCount++;
|
|
|
|
}
|
2018-03-02 23:54:36 -03:00
|
|
|
} else {
|
2018-05-09 01:18:35 -03:00
|
|
|
multisigTx[_id].approverCount--;
|
2018-03-02 23:54:36 -03:00
|
|
|
}
|
2018-03-24 05:53:35 -03:00
|
|
|
|
2018-05-09 01:18:35 -03:00
|
|
|
multisigTx[_id].approvals[_key] = _approval;
|
|
|
|
}
|
2018-03-24 05:30:07 -03:00
|
|
|
|
2018-03-03 00:00:11 -03:00
|
|
|
function _addKey(
|
|
|
|
bytes32 _key,
|
|
|
|
uint256 _purpose,
|
|
|
|
uint256 _type
|
|
|
|
)
|
2018-03-03 17:16:32 -03:00
|
|
|
private
|
2018-03-03 00:00:11 -03:00
|
|
|
{
|
2018-05-09 01:20:36 -03:00
|
|
|
require(_purpose > 0);
|
2018-03-02 23:54:36 -03:00
|
|
|
bytes32 keyHash = keccak256(_key, _purpose);
|
|
|
|
require(keys[keyHash].purpose == 0);
|
|
|
|
keys[keyHash] = Key(_purpose, _type, _key);
|
|
|
|
indexes[keyHash] = keysByPurpose[_purpose].push(_key) - 1;
|
2018-03-24 05:16:54 -03:00
|
|
|
emit KeyAdded(_key, _purpose, _type);
|
2018-03-02 23:54:36 -03:00
|
|
|
}
|
|
|
|
|
2018-03-03 00:00:11 -03:00
|
|
|
function _removeKey(
|
|
|
|
bytes32 _key,
|
|
|
|
uint256 _purpose
|
|
|
|
)
|
|
|
|
private
|
|
|
|
{
|
2018-03-24 05:30:07 -03:00
|
|
|
if (_purpose == MANAGEMENT_KEY) {
|
2018-03-24 05:58:04 -03:00
|
|
|
require(keysByPurpose[MANAGEMENT_KEY].length > purposeThreshold[MANAGEMENT_KEY]);
|
2018-03-24 05:30:07 -03:00
|
|
|
}
|
|
|
|
|
2018-03-02 23:54:36 -03:00
|
|
|
bytes32 keyHash = keccak256(_key, _purpose);
|
2018-03-24 05:16:54 -03:00
|
|
|
Key memory myKey = keys[keyHash];
|
2018-03-24 05:58:04 -03:00
|
|
|
uint256 index = indexes[keyHash];
|
2018-03-24 05:30:07 -03:00
|
|
|
bytes32 indexReplacer = keysByPurpose[_purpose][keysByPurpose[_purpose].length - 1];
|
|
|
|
|
|
|
|
keysByPurpose[_purpose][index] = indexReplacer;
|
|
|
|
indexes[keccak256(indexReplacer, _purpose)] = index;
|
2018-03-02 23:54:36 -03:00
|
|
|
keysByPurpose[_purpose].length--;
|
|
|
|
|
2018-03-24 05:30:07 -03:00
|
|
|
delete indexes[keyHash];
|
2018-03-02 23:54:36 -03:00
|
|
|
delete keys[keyHash];
|
|
|
|
|
2018-03-24 05:53:35 -03:00
|
|
|
emit KeyRemoved(myKey.key, myKey.purpose, myKey.keyType);
|
2018-03-02 23:54:36 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
function _includeClaim(
|
|
|
|
bytes32 _claimHash,
|
|
|
|
uint256 _claimType,
|
|
|
|
uint256 _scheme,
|
|
|
|
address _issuer,
|
|
|
|
bytes _signature,
|
|
|
|
bytes _data,
|
|
|
|
string _uri
|
|
|
|
)
|
|
|
|
private
|
|
|
|
{
|
|
|
|
claims[_claimHash] = Claim(
|
|
|
|
{
|
2018-03-21 13:13:04 -03:00
|
|
|
claimType: _claimType,
|
|
|
|
scheme: _scheme,
|
|
|
|
issuer: _issuer,
|
|
|
|
signature: _signature,
|
|
|
|
data: _data,
|
|
|
|
uri: _uri
|
2018-03-02 23:54:36 -03:00
|
|
|
}
|
|
|
|
);
|
|
|
|
indexes[_claimHash] = claimsByType[_claimType].length;
|
|
|
|
claimsByType[_claimType].push(_claimHash);
|
2018-03-19 23:15:22 -03:00
|
|
|
emit ClaimAdded(
|
2018-03-02 23:54:36 -03:00
|
|
|
_claimHash,
|
|
|
|
_claimType,
|
|
|
|
_scheme,
|
|
|
|
_issuer,
|
|
|
|
_signature,
|
|
|
|
_data,
|
|
|
|
_uri
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function _modifyClaim(
|
|
|
|
bytes32 _claimHash,
|
|
|
|
uint256 _claimType,
|
|
|
|
uint256 _scheme,
|
|
|
|
address _issuer,
|
|
|
|
bytes _signature,
|
|
|
|
bytes _data,
|
|
|
|
string _uri
|
|
|
|
)
|
|
|
|
private
|
|
|
|
{
|
|
|
|
require(msg.sender == _issuer);
|
|
|
|
claims[_claimHash] = Claim({
|
|
|
|
claimType: _claimType,
|
|
|
|
scheme: _scheme,
|
|
|
|
issuer: _issuer,
|
|
|
|
signature: _signature,
|
|
|
|
data: _data,
|
|
|
|
uri: _uri
|
|
|
|
});
|
2018-03-24 05:16:54 -03:00
|
|
|
emit ClaimChanged(
|
|
|
|
_claimHash,
|
|
|
|
_claimType,
|
|
|
|
_scheme,
|
|
|
|
_issuer,
|
|
|
|
_signature,
|
|
|
|
_data,
|
|
|
|
_uri
|
|
|
|
);
|
2018-03-02 23:54:36 -03:00
|
|
|
}
|
|
|
|
|
2018-05-09 01:18:35 -03:00
|
|
|
|
2018-02-21 17:17:38 -04:00
|
|
|
}
|
2018-01-08 19:00:07 -06:00
|
|
|
|