751 lines
23 KiB
Solidity
Raw Normal View History

pragma solidity ^0.4.21;
import "./ERC725.sol";
import "./ERC735.sol";
import "../common/MessageSigned.sol";
/**
* @title Self sovereign Identity
* @author Ricardo Guilherme Schmidt (Status Research & Development GmbH)
*/
contract Identity is ERC725, ERC735, MessageSigned {
uint256 public nonce;
2018-05-10 00:35:53 -03:00
address public recoveryContract;
uint256 salt;
2018-05-10 00:35:53 -03:00
mapping (bytes32 => uint256) indexes;
mapping (bytes32 => Key) keys;
2018-05-09 17:12:43 -03:00
mapping (bytes32 => bool) isKeyPurpose;
mapping (bytes32 => bytes32[]) keysByPurpose;
2018-03-24 05:30:07 -03:00
mapping (uint256 => Transaction) pendingTx;
2018-03-24 05:58:04 -03:00
mapping (uint256 => uint256) purposeThreshold;
2018-05-10 00:35:53 -03:00
mapping (bytes32 => Claim) claims;
mapping (uint256 => bytes32[]) claimsByType;
struct Transaction {
uint256 approverCount;
address to;
2018-03-24 05:58:04 -03:00
uint256 value;
bytes data;
mapping(bytes32 => bool) approvals;
}
/**
* @notice requires called by identity itself, otherwise forward to execute process
*/
modifier managementOnly {
if(msg.sender == address(this)) {
_;
} else {
_execute(keccak256(msg.sender), address(this), 0, msg.data);
}
}
2018-05-08 20:29:42 -03:00
/**
* @notices requires called by recovery address
*/
modifier recoveryOnly {
require(
2018-03-21 13:13:04 -03:00
recoveryContract != address(0) &&
msg.sender == recoveryContract
);
_;
}
/**
* @notice requires `_signature` from a `_key` with `_messageHash`
* @param _key key expected out from `_signature` of `_messageHash`
* @param _messageHash message signed in `_signature` by `_key`
* @param _signature `_messageHash` signed by `_key`
*/
modifier keyMessageSigned (
2018-03-03 00:00:11 -03:00
bytes32 _key,
bytes32 _messageHash,
bytes _signature
) {
2018-03-19 23:15:22 -03:00
require(
_key == keccak256(
recoverAddress(
getSignHash(_messageHash),
_signature
),
salt
)
);
2018-03-03 00:00:11 -03:00
_;
}
2018-01-08 19:00:07 -06:00
/**
* @notice constructor builds identity with first key as `msg.sender`
*/
constructor(bytes32 _key) public {
_constructIdentity(_key);
}
2018-03-03 00:00:11 -03:00
/**
* @notice default function allows deposit of ETH
*/
2018-03-03 00:00:11 -03:00
function ()
public
payable
{
}
2018-05-09 01:34:33 -03:00
////////////////
// Execute calls and multisig approval
////////////////
/**
* @notice execute (or request) call
* @param _to destination of call
* @param _value amount of ETH in call
* @param _data data
*/
2018-05-09 01:34:33 -03:00
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-05-09 01:34:33 -03:00
txId = _execute(keccak256(msg.sender), _to, _value, _data);
}
/**
* @notice approve a multisigned execution
* @param _txId unique id multisig transaction
* @param _approval approve (true) or reject (false)
*/
function approve(uint256 _txId, bool _approval)
2018-05-09 01:34:33 -03:00
public
returns (bool success)
{
return _approveRequest(keccak256(msg.sender), _txId, _approval);
}
2018-05-09 01:34:33 -03:00
////////////////
// Message Signed functions
////////////////
/**
* @notice execute (or request) call using ethereum signed message as authorization
* @param _to destination of call
* @param _value amount of ETH in call
* @param _data data
* @param _nonce current nonce
* @param _key key authorizing the call
* @param _signature signature of key
*/
2018-05-09 01:34:33 -03:00
function executeMessageSigned(
address _to,
uint256 _value,
bytes _data,
uint256 _nonce,
2018-05-09 01:34:33 -03:00
bytes32 _key,
bytes _signature
)
public
2018-05-09 01:34:33 -03:00
keyMessageSigned(
_key,
keccak256(
address(this),
bytes4(keccak256("execute(address,uint256,bytes)")),
_to,
_value,
_data,
_nonce
2018-05-09 01:34:33 -03:00
),
_signature
)
returns (uint256 txId)
{
require(_nonce == nonce);
2018-05-09 01:34:33 -03:00
txId = _execute(_key, _to, _value, _data);
2018-05-09 01:34:33 -03:00
}
/**
* @notice approve a multisigned execution using ethereum signed message as authorization
* @param _txId unique id multisig transaction
* @param _approval approve (true) or reject (false)
* @param _key key authorizing the call
* @param _signature signature of key
*/
2018-05-09 01:34:33 -03:00
function approveMessageSigned(
uint256 _txId,
2018-05-09 01:34:33 -03:00
bool _approval,
bytes32 _key,
bytes _signature
)
public
keyMessageSigned(
_key,
keccak256(
address(this),
bytes4(keccak256("approve(uint256,bool)")),
_txId,
2018-05-09 01:34:33 -03:00
_approval
),
_signature
)
returns (bool success)
{
return _approveRequest(_key, _txId, _approval);
}
2018-05-09 01:34:33 -03:00
2018-05-09 01:34:33 -03:00
////////////////
// Management functions
////////////////
2018-01-08 19:00:07 -06:00
/**
* @notice Adds a _key to the identity. The `_purpose`
* @param _key key hash being added
* @param _purpose specifies the purpose of key.
* @param _type inform type of key
*/
2018-02-22 05:05:10 -03:00
function addKey(
bytes32 _key,
uint256 _purpose,
uint256 _type
)
public
managementOnly
2018-02-22 05:05:10 -03:00
returns (bool success)
{
_addKey(_key, _purpose, _type, salt);
return true;
}
2018-01-08 19:00:07 -06:00
/**
2018-05-11 01:53:22 -03:00
* @notice Replaces one `_oldKey` with other `_newKey`
2018-05-09 02:52:05 -03:00
* @param _purpose what purpose being replaced
* @param _oldKey key to remove
2018-05-11 01:53:22 -03:00
* @param _newKey key to add
* @param _newType inform type of `_newKey`
*/
function replaceKey(
bytes32 _oldKey,
2018-05-11 01:53:22 -03:00
bytes32 _newKey,
uint256 _newType
)
public
managementOnly
returns (bool success)
2018-05-11 01:53:22 -03:00
{
return _replaceKey(_oldKey, _newKey, _newType, salt);
}
/**
* @notice Removes `_purpose` of `_key`
* @param _key key to remove
* @param _purpose purpose to remove
*/
2018-02-22 05:05:10 -03:00
function removeKey(
bytes32 _key,
uint256 _purpose
)
public
managementOnly
2018-02-22 05:05:10 -03:00
returns (bool success)
{
_removeKey(_key, _purpose, salt);
return true;
}
2018-01-08 19:00:07 -06:00
/**
* @notice Defines minimum approval required by key type
* @param _purpose select purpose
* @param _minimumApprovals select how much signatures needed
*/
function setMinimumApprovalsByKeyType(
uint256 _purpose,
uint256 _minimumApprovals
2018-02-22 05:16:06 -03:00
)
public
managementOnly
2018-02-22 05:16:06 -03:00
{
require(_minimumApprovals > 0);
require(_minimumApprovals <= keysByPurpose[keccak256(_purpose, salt)].length);
2018-03-24 05:58:04 -03:00
purposeThreshold[_purpose] = _minimumApprovals;
}
/**
* @notice Defines recovery address. This is one time only action.
* @param _recoveryContract address of recovery contract
*/
2018-05-09 01:34:33 -03:00
function setupRecovery(address _recoveryContract)
public
managementOnly
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 _topic,
2018-02-22 05:05:10 -03:00
uint256 _scheme,
address _issuer,
bytes _signature,
bytes _data,
string _uri
)
public
2018-02-23 14:02:53 -03:00
returns (bytes32 claimHash)
{
claimHash = keccak256(_issuer, _topic);
2018-02-23 14:02:53 -03:00
if (msg.sender == address(this)) {
if (claims[claimHash].topic > 0) {
_modifyClaim(claimHash, _topic, _scheme, _issuer, _signature, _data, _uri);
2018-02-23 14:02:53 -03:00
} else {
_includeClaim(claimHash, _topic, _scheme, _issuer, _signature, _data, _uri);
2018-02-23 14:02:53 -03:00
}
} else {
2018-05-09 17:12:43 -03:00
require(hasKeyPurpose(keccak256(msg.sender), CLAIM_SIGNER_KEY));
_requestApproval(0, address(this), 0, msg.data);
2018-03-19 23:15:22 -03:00
emit ClaimRequested(
2018-02-23 14:02:53 -03:00
claimHash,
_topic,
2018-02-22 05:05:10 -03:00
_scheme,
_issuer,
_signature,
_data,
2018-02-23 14:02:53 -03:00
_uri
);
2018-01-08 19:00:07 -06:00
}
}
2018-03-03 00:00:11 -03:00
function removeClaim(bytes32 _claimId)
public
returns (bool success)
{
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
);
// 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.
uint256 claimIdTopicPos = indexes[_claimId];
delete indexes[_claimId];
bytes32[] storage claimsTopicArr = claimsByType[c.topic];
bytes32 replacer = claimsTopicArr[claimsTopicArr.length-1];
claimsTopicArr[claimIdTopicPos] = replacer;
indexes[replacer] = claimIdTopicPos;
delete claims[_claimId];
claimsTopicArr.length--;
emit ClaimRemoved(_claimId, c.topic, c.scheme, c.issuer, c.signature, c.data, c.uri);
return true;
}
2018-01-08 19:00:07 -06:00
2018-05-09 01:34:33 -03:00
////////////////
// Recovery methods
////////////////
2018-05-10 01:24:04 -03:00
/**
* @notice Increase salt for hashing storage pointer of keys and add `_recoveryNewKey`
2018-05-10 01:24:04 -03:00
* @param _recoveryNewKey new key being defined
*/
function recoveryReset(bytes32 _recoveryNewKey)
2018-05-09 01:34:33 -03:00
public
recoveryOnly
{
salt++;
_addKey(_recoveryNewKey, MANAGEMENT_KEY, 0, salt);
_addKey(_recoveryNewKey, ACTION_KEY, 0, salt);
purposeThreshold[MANAGEMENT_KEY] = 1;
2018-05-09 01:34:33 -03:00
}
////////////////
// Public Views
////////////////
2018-02-22 05:05:10 -03:00
function getKey(
bytes32 _key,
uint256 _purpose
)
public
view
2018-05-09 17:12:43 -03:00
returns(uint256[] purposes, uint256 keyType, bytes32 key)
2018-02-22 05:05:10 -03:00
{
Key storage myKey = keys[keccak256(_key, salt)];
2018-05-09 17:12:43 -03:00
return (myKey.purposes, myKey.keyType, myKey.key);
}
2018-02-21 19:45:25 -03:00
2018-05-09 17:12:43 -03:00
function hasKeyPurpose(bytes32 _key, uint256 _purpose)
2018-02-22 05:05:10 -03:00
public
view
2018-02-22 05:05:10 -03:00
returns (bool)
{
2018-05-09 17:12:43 -03:00
return isKeyPurpose[keccak256(_key, _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
view
2018-02-22 05:05:10 -03:00
returns(uint256[] purpose)
{
return keys[keccak256(_key, salt)].purposes;
}
2018-02-22 05:05:10 -03:00
function getKeysByPurpose(uint256 _purpose)
public
view
2018-03-02 23:48:15 -03:00
returns(bytes32[])
2018-02-22 05:05:10 -03:00
{
return keysByPurpose[keccak256(_purpose, salt)];
}
2018-02-22 05:05:10 -03:00
function getClaim(bytes32 _claimId)
public
view
2018-03-03 00:00:11 -03:00
returns(
uint256 topic,
2018-03-03 00:00:11 -03:00
uint256 scheme,
address issuer,
bytes signature,
bytes data,
string uri
)
2018-02-22 05:05:10 -03:00
{
Claim memory _claim = claims[_claimId];
return (_claim.topic, _claim.scheme, _claim.issuer, _claim.signature, _claim.data, _claim.uri);
}
function getClaimIdsByTopic(uint256 _topic)
2018-02-22 05:05:10 -03:00
public
view
2018-02-22 05:05:10 -03:00
returns(bytes32[] claimIds)
{
return claimsByType[_topic];
2018-01-08 19:00:07 -06:00
}
2018-05-09 01:34:33 -03:00
////////////////
// Internal methods
////////////////
function _constructIdentity(bytes32 _managerKey)
internal
{
uint256 _salt = salt;
require(keysByPurpose[keccak256(MANAGEMENT_KEY, _salt)].length == 0);
2018-03-24 05:58:04 -03:00
require(purposeThreshold[MANAGEMENT_KEY] == 0);
_addKey(_managerKey, MANAGEMENT_KEY, 0, _salt);
_addKey(_managerKey, ACTION_KEY, 0, _salt);
2018-03-24 05:58:04 -03:00
purposeThreshold[MANAGEMENT_KEY] = 1;
purposeThreshold[ACTION_KEY] = 1;
}
function _execute(
bytes32 _key,
address _to,
uint256 _value,
bytes _data
)
internal
returns (uint256 txId)
{
uint256 requiredPurpose = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY;
2018-05-09 17:12:43 -03:00
require(hasKeyPurpose(_key, requiredPurpose));
if (purposeThreshold[requiredPurpose] == 1) {
txId = nonce++;
_commitCall(txId, _to, _value, _data);
} else {
txId = _requestApproval(_key, _to, _value, _data);
}
}
2018-05-09 01:34:33 -03:00
function _commitCall(
uint256 _txId,
address _to,
uint256 _value,
bytes _data
)
internal
returns(bool success)
{
success = _to.call.value(_value)(_data);
if (success) {
emit Executed(_txId, _to, _value, _data);
} else {
emit ExecutionFailed(_txId, _to, _value, _data);
}
}
function _requestApproval(
bytes32 _key,
address _to,
uint256 _value,
bytes _data
)
internal
returns (uint256 txId)
{
txId = nonce++;
pendingTx[txId] = Transaction({
approverCount: _key == 0 ? 0 : 1,
2018-03-19 23:15:22 -03:00
to: _to,
value: _value,
data: _data
});
if (_key != 0) {
pendingTx[txId].approvals[_key] = true;
}
emit ExecutionRequested(txId, _to, _value, _data);
}
2018-05-09 01:34:33 -03:00
////////////////
// Private methods
////////////////
function _approveRequest(
bytes32 _key,
uint256 _txId,
bool _approval
)
private
returns(bool success) //(?) should return approved instead of success?
{
Transaction memory approvedTx = pendingTx[_txId];
require(approvedTx.approverCount > 0);
uint256 requiredKeyPurpose = approvedTx.to == address(this) ? MANAGEMENT_KEY : ACTION_KEY;
2018-05-09 17:12:43 -03:00
require(hasKeyPurpose(_key, requiredKeyPurpose));
require(pendingTx[_txId].approvals[_key] != _approval);
2018-03-24 05:53:35 -03:00
if (_approval) {
if (approvedTx.approverCount + 1 == purposeThreshold[requiredKeyPurpose]) {
delete pendingTx[_txId];
emit Approved(_txId, _approval);
return _commitCall(_txId, approvedTx.to, approvedTx.value, approvedTx.data);
} else {
pendingTx[_txId].approvals[_key] = true;
pendingTx[_txId].approverCount++;
}
} else {
delete pendingTx[_txId].approvals[_key];
if (pendingTx[_txId].approverCount == 1) {
delete pendingTx[_txId];
emit Approved(_txId, _approval);
} else {
pendingTx[_txId].approverCount--;
}
}
}
2018-03-24 05:30:07 -03:00
2018-03-03 00:00:11 -03:00
function _addKey(
bytes32 _key,
uint256 _purpose,
uint256 _type,
uint256 _salt
2018-03-03 00:00:11 -03:00
)
private
2018-03-03 00:00:11 -03:00
{
2018-05-09 01:20:36 -03:00
require(_purpose > 0);
2018-05-09 17:12:43 -03:00
bytes32 keySaltedHash = keccak256(_key, _salt); //key storage pointer
bytes32 saltedKeyPurposeHash = keccak256(keySaltedHash, _purpose); // accounts by purpose hash element index pointer
require(!isKeyPurpose[saltedKeyPurposeHash]); //cannot add a key already added
isKeyPurpose[saltedKeyPurposeHash] = true; //set authorization
uint256 keyElementIndex = keysByPurpose[saltedKeyPurposeHash].push(_key) - 1; //add key to list by purpose
indexes[saltedKeyPurposeHash] = keyElementIndex; //save index of key in list by purpose
if (keys[keySaltedHash].key == 0) { //is a new key
uint256[] memory purposes = new uint256[](_purpose); //create new array with first purpose
keys[keySaltedHash] = Key(purposes,_type,_key); //add new key
2018-05-09 17:12:43 -03:00
} else {
uint256 addedPurposeElementIndex = keys[keySaltedHash].purposes.push(_purpose) - 1; //add purpose to key
bytes32 keyPurposeSaltedHash = keccak256(_key, _purpose, _salt); //index of purpose in key pointer
indexes[keyPurposeSaltedHash] = addedPurposeElementIndex; //save index
2018-05-09 17:12:43 -03:00
}
2018-03-24 05:16:54 -03:00
emit KeyAdded(_key, _purpose, _type);
}
2018-05-10 01:24:04 -03:00
2018-03-03 00:00:11 -03:00
function _removeKey(
bytes32 _key,
uint256 _purpose,
uint256 _salt
2018-03-03 00:00:11 -03:00
)
private
{
bytes32 purposeSaltedHash = keccak256(_purpose, _salt); // salted accounts by purpose array index pointer
// forbidden to remove last management key
2018-03-24 05:30:07 -03:00
if (_purpose == MANAGEMENT_KEY) {
require(keysByPurpose[purposeSaltedHash].length > purposeThreshold[MANAGEMENT_KEY]);
2018-03-24 05:30:07 -03:00
}
bytes32 keySaltedHash = keccak256(_key, _salt); // key storage pointer
bytes32 saltedKeyPurposeHash = keccak256(keySaltedHash, _purpose); // accounts by purpose hash element index pointer
require(isKeyPurpose[saltedKeyPurposeHash]); //not possible to remove what not exists
delete isKeyPurpose[saltedKeyPurposeHash]; //remove authorization
// remove keys by purpose array key element
uint256 removedKeyIndex = indexes[saltedKeyPurposeHash]; // read old key element index
delete indexes[saltedKeyPurposeHash]; // delete key index
uint256 replacerKeyIndex = keysByPurpose[purposeSaltedHash].length - 1; // replacer is last element
if (removedKeyIndex != replacerKeyIndex) { // deleted not the last element, replace deleted by last element
bytes32 replacerKey = keysByPurpose[purposeSaltedHash][replacerKeyIndex]; // get replacer key
keysByPurpose[purposeSaltedHash][removedKeyIndex] = replacerKey; // overwrite removed index by replacer
indexes[keccak256(keccak256(replacerKey, _salt), _purpose)] = removedKeyIndex; // update saltedKeyPurposeHash index of replacer
2018-05-09 17:12:43 -03:00
}
keysByPurpose[purposeSaltedHash].length--; // remove last element
//remove key purposes array purpose element
Key storage myKey = keys[keySaltedHash]; //load Key storage pointer
uint256 _type = myKey.keyType; //save type for case key deleted
uint256 replacerPurposeIndex = myKey.purposes.length; //load amount of purposes
bytes32 keyPurposeSaltedHash = keccak256(_key, _purpose, _salt); //account purpose array element index
uint256 removedPurposeIndex = indexes[keyPurposeSaltedHash]; //read old index
delete indexes[keyPurposeSaltedHash]; //delete key's purpose index
if (replacerPurposeIndex > 1) { //is not the last key
replacerPurposeIndex--; //move to last element pos
if(removedPurposeIndex != replacerPurposeIndex) { //removed element is not last element
uint256 replacerPurpose = myKey.purposes[replacerPurposeIndex]; //take last element
myKey.purposes[removedPurposeIndex] = replacerPurpose; //replace removed element with replacer element
indexes[keccak256(_key, replacerPurpose, _salt)] = removedPurposeIndex; //update index
2018-05-09 17:12:43 -03:00
}
myKey.purposes.length--; //remove last element
} else { //is the last purpose
delete keys[keySaltedHash]; //drop this Key
2018-05-09 17:12:43 -03:00
}
2018-03-24 05:30:07 -03:00
2018-05-09 17:12:43 -03:00
emit KeyRemoved(_key, _purpose, _type);
}
2018-05-11 01:53:22 -03:00
/**
* @notice Replaces one `_oldKey` with other `_newKey`
* @param _oldKey key to remove
* @param _newKey key to add
* @param _newType inform type of `_newKey`
* @param _salt current salt
*/
function _replaceKey(
bytes32 _oldKey,
bytes32 _newKey,
uint256 _newType,
uint256 _salt
)
private
returns (bool success)
{
bytes32 newKeySaltedHash = keccak256(_newKey, _salt); // key storage pointer
if (_oldKey == _newKey) { //not replacing key, just keyType
keys[newKeySaltedHash].keyType == _newType;
return true;
}
bytes32 oldKeySaltedHash = keccak256(_oldKey, _salt); // key storage pointer
Key memory oldKey = keys[oldKeySaltedHash];
delete keys[oldKeySaltedHash];
uint256 len = oldKey.purposes.length;
for (uint i = 0; i < len; i++) {
uint256 _purpose = oldKey.purposes[i];
bytes32 purposeSaltedHash = keccak256(_purpose, _salt); // salted accounts by purpose array index pointer
bytes32 saltedOldKeyPurposeHash = keccak256(oldKeySaltedHash, _purpose); // accounts by purpose hash element index pointer
bytes32 saltedNewKeyPurposeHash = keccak256(newKeySaltedHash, _purpose); // accounts by purpose hash element index pointer
bytes32 oldKeyPurposeSaltedHash = keccak256(_oldKey, _purpose, _salt); //account purpose array element index
bytes32 newKeyPurposeSaltedHash = keccak256(_newKey, _purpose, _salt); //account purpose array element index
delete isKeyPurpose[saltedOldKeyPurposeHash]; //clear oldKey auth
isKeyPurpose[saltedNewKeyPurposeHash] = true; //set newKey auth
uint256 replacedKeyElementIndex = indexes[saltedOldKeyPurposeHash];
delete indexes[saltedOldKeyPurposeHash];
keysByPurpose[purposeSaltedHash][replacedKeyElementIndex] = _newKey; //replace key at list by purpose
indexes[saltedNewKeyPurposeHash] = replacedKeyElementIndex; // save index
indexes[newKeyPurposeSaltedHash] = indexes[oldKeyPurposeSaltedHash]; //transfer key purposes list index
delete indexes[oldKeyPurposeSaltedHash];
}
keys[newKeySaltedHash] = Key(oldKey.purposes, _newType, _newKey); //add new key
return true;
}
function _includeClaim(
bytes32 _claimHash,
uint256 _topic,
uint256 _scheme,
address _issuer,
bytes _signature,
bytes _data,
string _uri
)
private
{
claims[_claimHash] = Claim(
{
topic: _topic,
2018-03-21 13:13:04 -03:00
scheme: _scheme,
issuer: _issuer,
signature: _signature,
data: _data,
uri: _uri
}
);
indexes[_claimHash] = claimsByType[_topic].length;
claimsByType[_topic].push(_claimHash);
2018-03-19 23:15:22 -03:00
emit ClaimAdded(
_claimHash,
_topic,
_scheme,
_issuer,
_signature,
_data,
_uri
);
}
function _modifyClaim(
bytes32 _claimHash,
uint256 _topic,
uint256 _scheme,
address _issuer,
bytes _signature,
bytes _data,
string _uri
)
private
{
require(msg.sender == _issuer);
claims[_claimHash] = Claim({
topic: _topic,
scheme: _scheme,
issuer: _issuer,
signature: _signature,
data: _data,
uri: _uri
});
2018-03-24 05:16:54 -03:00
emit ClaimChanged(
_claimHash,
_topic,
2018-03-24 05:16:54 -03:00
_scheme,
_issuer,
_signature,
_data,
_uri
);
}
}
2018-01-08 19:00:07 -06:00