From 1f9ad9b135e1a0710132799b0cd73e165bc2dc35 Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Wed, 21 Mar 2018 18:02:05 -0300 Subject: [PATCH 01/13] add economic abstraction property to Identities --- contracts/identity/IdentityGasRelay.sol | 247 ++++++++++++++++++++++++ 1 file changed, 247 insertions(+) create mode 100644 contracts/identity/IdentityGasRelay.sol diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol new file mode 100644 index 0000000..bb5eb02 --- /dev/null +++ b/contracts/identity/IdentityGasRelay.sol @@ -0,0 +1,247 @@ +pragma solidity ^0.4.17; + +import "./Identity.sol"; +import "../token/ERC20Token.sol"; + +contract IdentityGasRelay is Identity { + + bytes4 public constant EXECUTE_PREFIX = bytes4(keccak256("executeGasRelayed")); + + event ExecutedGasRelayed(bytes32 signHash); + + function executeSigned( + address _to, + uint256 _value, + bytes _data, + uint _nonce, + uint _gasPrice, + uint _gasMinimum, + address _gasToken, + bytes _messageSignature + ) + external + { + uint startGas = gasleft(); + require(startGas > _gasMinimum); + uint256 requiredKey = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY; + require(minimumApprovalsByKeyPurpose[requiredKey] == 1); + require(_nonce == nonce); + nonce++; + + bytes32 _signedHash = getSignHash( + executeHash( + _to, + _value, + keccak256(_data), + _nonce, + _gasPrice, + _gasMinimum, + _gasToken + ) + ); + + require( + isKeyPurpose( + recoverKey( + _signedHash, + _messageSignature, + 0 + ), + requiredKey + ) + ); + + if (_to.call.value(_value)(_data)) { + emit ExecutedGasRelayed(_signedHash); + } + + if(_gasPrice > 0) { + payInclusionFee( + startGas - gasleft(), + _gasPrice, + msg.sender, + _gasToken + ); + } + } + + function executeMultiSigned( + address _to, + uint256 _value, + bytes _data, + uint _nonce, + uint _gasPrice, + uint _gasMinimum, + address _gasToken, + bytes _messageSignatures + ) + external + { + uint startGas = gasleft(); + require(startGas > _gasMinimum); + require(_nonce == nonce); + nonce++; + _executeMultiSigned(_to, _value, _data, _nonce, _gasPrice, _gasMinimum, _gasToken, _messageSignatures); + if (_gasPrice > 0) { + payInclusionFee( + startGas - gasleft(), + _gasPrice, + msg.sender, + _gasToken + ); + } + } + + function executeHash( + address _to, + uint256 _value, + bytes32 _dataHash, + uint _nonce, + uint256 _gasPrice, + uint256 _gasMinimum, + address _gasToken + ) + public + view + returns (bytes32) + { + return keccak256( + address(this), + EXECUTE_PREFIX, + _to, + _value, + _dataHash, + _nonce, + _gasPrice, + _gasMinimum, + _gasToken + ); + } + + function recoverKey ( + bytes32 _signHash, + bytes _messageSignature, + uint256 _pos + ) + pure + public + returns(bytes32) + { + uint8 v; + bytes32 r; + bytes32 s; + (v,r,s) = signatureSplit(_messageSignature, _pos); + return bytes32( + ecrecover( + _signHash, + v, + r, + s + ) + ); + } + + /** + * @dev divides bytes signature into `uint8 v, bytes32 r, bytes32 s` + */ + function signatureSplit(bytes _signature, uint256 _pos) + pure + public + returns (uint8 v, bytes32 r, bytes32 s) + { + uint pos = _pos + 1; + // The signature format is a compact form of: + // {bytes32 r}{bytes32 s}{uint8 v} + // Compact means, uint8 is not padded to 32 bytes. + assembly { + r := mload(add(_signature, mul(32,pos))) + s := mload(add(_signature, mul(64,pos))) + // Here we are loading the last 32 bytes, including 31 bytes + // of 's'. There is no 'mload8' to do this. + // + // 'byte' is not working due to the Solidity parser, so lets + // use the second best option, 'and' + v := and(mload(add(_signature, mul(65,pos))), 0xff) + } + + require(v == 27 || v == 28); + } + + /** + * @notice Hash a hash with `"\x19Ethereum Signed Message:\n32"` + * @param _hash Sign to hash. + * @return signHash Hash to be signed. + */ + function getSignHash( + bytes32 _hash + ) + pure + public + returns(bytes32 signHash) + { + signHash = keccak256("\x19Ethereum Signed Message:\n32", _hash); + } + + + function _executeMultiSigned( + address _to, + uint256 _value, + bytes _data, + uint _nonce, + uint _gasPrice, + uint _gasMinimum, + address _gasToken, + bytes _messageSignatures + ) + private + { + uint256 requiredKey = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY; + uint256 len = _messageSignatures.length / 72; + require(len == minimumApprovalsByKeyPurpose[requiredKey]); + + bytes32 _signedHash = getSignHash( + executeHash( + _to, + _value, + keccak256(_data), + _nonce, + _gasPrice, + _gasMinimum, + _gasToken + ) + ); + + bytes32 _lastKey = 0; + for (uint256 i = 0; i < len; i++) { + bytes32 _key = recoverKey( + _signedHash, + _messageSignatures, + i + ); + require(_key > _lastKey); //assert keys are different + require(isKeyPurpose(_key, requiredKey)); + _lastKey = _key; + } + + if (_to.call.value(_value)(_data)) { + emit ExecutedGasRelayed(_signedHash); + } + } + + function payInclusionFee( + uint256 _gasUsed, + uint256 _gasPrice, + address _msgIncluder, + address _gasToken + ) + private + { + uint256 _amount = (21000 + _gasUsed) * _gasPrice; + if (_gasToken == address(0)) { + address(_msgIncluder).transfer(_amount); + } else { + ERC20Token(_gasToken).transfer(_msgIncluder, _amount); + } + } + +} \ No newline at end of file From e13ac0315dc8006b9c84065023f6e990e3cc70e8 Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Wed, 21 Mar 2018 18:43:26 -0300 Subject: [PATCH 02/13] update names --- contracts/identity/IdentityGasRelay.sol | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index bb5eb02..4fde2f7 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -5,11 +5,11 @@ import "../token/ERC20Token.sol"; contract IdentityGasRelay is Identity { - bytes4 public constant EXECUTE_PREFIX = bytes4(keccak256("executeGasRelayed")); + bytes4 public constant EXECUTE_PREFIX = bytes4(keccak256("executeGasRelayed(address,uint256,bytes32,uint256,uint256,address)")); event ExecutedGasRelayed(bytes32 signHash); - function executeSigned( + function executeGasRelayed( address _to, uint256 _value, bytes _data, @@ -29,7 +29,7 @@ contract IdentityGasRelay is Identity { nonce++; bytes32 _signedHash = getSignHash( - executeHash( + executeGasRelayedHash( _to, _value, keccak256(_data), @@ -65,7 +65,7 @@ contract IdentityGasRelay is Identity { } } - function executeMultiSigned( + function executeGasRelayedMultiSigned( address _to, uint256 _value, bytes _data, @@ -81,7 +81,7 @@ contract IdentityGasRelay is Identity { require(startGas > _gasMinimum); require(_nonce == nonce); nonce++; - _executeMultiSigned(_to, _value, _data, _nonce, _gasPrice, _gasMinimum, _gasToken, _messageSignatures); + _executeGasRelayedMultiSigned(_to, _value, _data, _nonce, _gasPrice, _gasMinimum, _gasToken, _messageSignatures); if (_gasPrice > 0) { payInclusionFee( startGas - gasleft(), @@ -92,7 +92,7 @@ contract IdentityGasRelay is Identity { } } - function executeHash( + function executeGasRelayedHash( address _to, uint256 _value, bytes32 _dataHash, @@ -183,7 +183,7 @@ contract IdentityGasRelay is Identity { } - function _executeMultiSigned( + function _executeGasRelayedMultiSigned( address _to, uint256 _value, bytes _data, @@ -200,7 +200,7 @@ contract IdentityGasRelay is Identity { require(len == minimumApprovalsByKeyPurpose[requiredKey]); bytes32 _signedHash = getSignHash( - executeHash( + executeGasRelayedHash( _to, _value, keccak256(_data), From cfc3c17c6bf1fe379b081934332820a0384885e1 Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Wed, 21 Mar 2018 19:36:23 -0300 Subject: [PATCH 03/13] new snt controller with economic abstraction --- contracts/status/SNTController.sol | 297 +++++++++++++++++++++++++++++ 1 file changed, 297 insertions(+) create mode 100644 contracts/status/SNTController.sol diff --git a/contracts/status/SNTController.sol b/contracts/status/SNTController.sol new file mode 100644 index 0000000..3af76ca --- /dev/null +++ b/contracts/status/SNTController.sol @@ -0,0 +1,297 @@ +pragma solidity ^0.4.17; + +import "../token/TokenController.sol"; +import "../common/Owned.sol"; +import "../token/ERC20Token.sol"; +import "../token/MiniMeToken.sol"; + +/** + * @title SNTController + * @author Ricardo Guilherme Schmidt + */ +contract SNTController is TokenController, Owned { + + + bytes4 public constant TRANSFER_PREFIX = bytes4(keccak256("transferSNT(address,uint256,uint256,uint256)")); + bytes4 public constant EXECUTE_PREFIX = bytes4(keccak256("executeGasRelayed(address,bytes,uint256,uint256,uint256)")); + + MiniMeToken public snt; + mapping (address => uint256) public signNonce; + mapping (address => bool) public allowPublicExecution; + + event PublicExecutionEnabled(address indexed contractAddress, bool enabled); + event GasRelayedExecution(address indexed msgSigner, bytes32 signedHash, bool executed); + event ClaimedTokens(address indexed _token, address indexed _controller, uint256 _amount); + event ControllerChanged(address indexed _newController); + + /** + * @notice Constructor + * @param _owner Authority address + * @param _snt SNT token + */ + function SNTController(address _owner, address _snt) public { + owner = _owner; + snt = MiniMeToken(_snt); + } + + /** + * @notice allows externally owned address sign a message to transfer SNT and pay + * @param _to address receving the tokens from message signer + * @param _amount total being transfered + * @param _nonce current signNonce of message signer + * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used + * @param _signature concatenated rsv of message + */ + function transferSNT( + address _to, + uint256 _amount, + uint256 _nonce, + uint256 _gasPrice, + bytes _signature + ) + external + { + uint256 startGas = gasleft(); + bytes32 msgSigned = getSignHash( + getTransferSNTHash( + _to, + _amount, + _nonce, + _gasPrice + ) + ); + + address msgSigner = recoverAddress(msgSigned, _signature); + require(signNonce[msgSigner] == _nonce); + signNonce[msgSigner]++; + if (snt.transferFrom(msgSigner, _to, _amount)) { + require(snt.transferFrom(msgSigner, msg.sender, (21000 + startGas-gasleft()) * _gasPrice)); + } + } + + /** + * @notice allows externally owned address sign a message to offer SNT for a execution + * @param _allowedContract address of a contracts in execution trust list; + * @param _data msg.data to be sent to `_allowedContract` + * @param _nonce current signNonce of message signer + * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used + * @param _gasMinimal minimal amount of gas needed to complete the execution + * @param _signature concatenated rsv of message + */ + function executeGasRelayed( + address _allowedContract, + bytes _data, + uint256 _nonce, + uint256 _gasPrice, + uint256 _gasMinimal, + bytes _signature + ) + external + { + uint256 startGas = gasleft(); + require(startGas >= _gasMinimal); + bytes32 msgSigned = getSignHash( + getExecuteGasRelayedHash( + _allowedContract, + _data, + _nonce, + _gasPrice, + _gasMinimal + ) + ); + + address msgSigner = recoverAddress(msgSigned, _signature); + require(signNonce[msgSigner] == _nonce); + signNonce[msgSigner]++; + bool success = _allowedContract.call(_data); + emit GasRelayedExecution(msgSigner, msgSigned, success); + require( + snt.transferFrom( + msgSigner, + msg.sender, + (21000 + startGas-gasleft()) * _gasPrice + ) + ); + } + + /** + * @notice The owner of this contract can change the controller of the SNT token + * Please, be sure that the owner is a trusted agent or 0x0 address. + * @param _newController The address of the new controller + */ + function changeController(address _newController) public onlyOwner { + snt.changeController(_newController); + emit ControllerChanged(_newController); + } + + function enablePublicExecution(address _contract, bool _enable) public onlyOwner { + allowPublicExecution[_contract] = _enable; + emit PublicExecutionEnabled(_contract, _enable); + } + + ////////// + // Safety Methods + ////////// + + /** + * @notice This method can be used by the controller to extract mistakenly + * sent tokens to this contract. + * @param _token The address of the token contract that you want to recover + * set to 0 in case you want to extract ether. + */ + function claimTokens(address _token) public onlyOwner { + if (snt.controller() == address(this)) { + snt.claimTokens(_token); + } + if (_token == 0x0) { + address(owner).transfer(this.balance); + return; + } + + ERC20Token token = ERC20Token(_token); + uint256 balance = token.balanceOf(this); + token.transfer(owner, balance); + emit ClaimedTokens(_token, owner, balance); + } + + + ////////// + // MiniMe Controller Interface functions + ////////// + + // In between the offering and the network. Default settings for allowing token transfers. + function proxyPayment(address) public payable returns (bool) { + return false; + } + + function onTransfer(address, address, uint256) public returns (bool) { + return true; + } + + function onApprove(address, address, uint256) public returns (bool) { + return true; + } + + /** + * @notice get execution hash + * @param _allowedContract address of a contracts in execution trust list; + * @param _data msg.data to be sent to `_allowedContract` + * @param _nonce current signNonce of message signer + * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used + * @param _gasMinimal minimal amount of gas needed to complete the execution + */ + function getExecuteGasRelayedHash( + address _allowedContract, + bytes _data, + uint256 _nonce, + uint256 _gasPrice, + uint256 _gasMinimal + ) + public + view + returns (bytes32 execHash) + { + execHash = keccak256( + address(this), + EXECUTE_PREFIX, + _allowedContract, + keccak256(_data), + _nonce, + _gasPrice, + _gasMinimal + ); + } + + /** + * @notice get transfer hash + * @param _to address receving the tokens from message signer + * @param _amount total being transfered + * @param _nonce current signNonce of message signer + * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used + */ + function getTransferSNTHash( + address _to, + uint256 _amount, + uint256 _nonce, + uint256 _gasPrice + ) + public + view + returns (bytes32 txHash) + { + txHash = keccak256( + address(this), + TRANSFER_PREFIX, + _to, + _amount, + _nonce, + _gasPrice + ); + } + + /** + * @notice recovers address who signed the message + * @param _signHash operation ethereum signed message hash + * @param _messageSignature message `_signHash` signature + */ + function recoverAddress( + bytes32 _signHash, + bytes _messageSignature + ) + pure + public + returns(address) + { + uint8 v; + bytes32 r; + bytes32 s; + (v,r,s) = signatureSplit(_messageSignature); + return ecrecover( + _signHash, + v, + r, + s + ); + } + + /** + * @dev divides bytes signature into `uint8 v, bytes32 r, bytes32 s` + */ + function signatureSplit(bytes _signature) + pure + public + returns (uint8 v, bytes32 r, bytes32 s) + { + // The signature format is a compact form of: + // {bytes32 r}{bytes32 s}{uint8 v} + // Compact means, uint8 is not padded to 32 bytes. + assembly { + r := mload(add(_signature, 32)) + s := mload(add(_signature, 64)) + // Here we are loading the last 32 bytes, including 31 bytes + // of 's'. There is no 'mload8' to do this. + // + // 'byte' is not working due to the Solidity parser, so lets + // use the second best option, 'and' + v := and(mload(add(_signature, 65)), 0xff) + } + + require(v == 27 || v == 28); + } + + /** + * @notice Hash a hash with `"\x19Ethereum Signed Message:\n32"` + * @param _hash Sign to hash. + * @return signHash Hash to be signed. + */ + function getSignHash( + bytes32 _hash + ) + pure + public + returns (bytes32 signHash) + { + signHash = keccak256("\x19Ethereum Signed Message:\n32", _hash); + } + +} \ No newline at end of file From 770a6383243843906335f899e7f1a271ad348496 Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Wed, 21 Mar 2018 19:58:19 -0300 Subject: [PATCH 04/13] natspec --- contracts/identity/IdentityGasRelay.sol | 97 +++++++++++++++++++------ contracts/status/SNTController.sol | 3 +- 2 files changed, 78 insertions(+), 22 deletions(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index 4fde2f7..07cb168 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -3,39 +3,56 @@ pragma solidity ^0.4.17; import "./Identity.sol"; import "../token/ERC20Token.sol"; +/** + * @title IdentityGasRelay + * @author Ricardo Guilherme Schmidt (Status Research & Development GmbH) + * @notice enables economic abstraction for Identity + */ contract IdentityGasRelay is Identity { - bytes4 public constant EXECUTE_PREFIX = bytes4(keccak256("executeGasRelayed(address,uint256,bytes32,uint256,uint256,address)")); + bytes4 public constant CALL_PREFIX = bytes4(keccak256("callGasRelayed(address,uint256,bytes32,uint256,uint256,address)")); event ExecutedGasRelayed(bytes32 signHash); - function executeGasRelayed( + /** + * @notice include ethereum signed callHash in return of gas proportional amount multiplied by `_gasPrice` of `_gasToken` + * allows identity of being controlled without requiring ether in key balace + * @param _to destination of call + * @param _value call value (ether) + * @param _data call data + * @param _nonce current identity nonce + * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used + * @param _gasMinimal minimal amount of gas needed to complete the execution + * @param _gasToken token being used for paying `msg.sender` + * @param _messageSignature rsv concatenated ethereum signed message signature + */ + function callGasRelayed( address _to, uint256 _value, bytes _data, uint _nonce, uint _gasPrice, - uint _gasMinimum, + uint _gasMinimal, address _gasToken, bytes _messageSignature ) external { uint startGas = gasleft(); - require(startGas > _gasMinimum); + require(startGas > _gasMinimal); uint256 requiredKey = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY; require(minimumApprovalsByKeyPurpose[requiredKey] == 1); require(_nonce == nonce); nonce++; bytes32 _signedHash = getSignHash( - executeGasRelayedHash( + callGasRelayedHash( _to, _value, keccak256(_data), _nonce, _gasPrice, - _gasMinimum, + _gasMinimal, _gasToken ) ); @@ -65,7 +82,19 @@ contract IdentityGasRelay is Identity { } } - function executeGasRelayedMultiSigned( + /** + * @notice include ethereum signed callHash in return of gas proportional amount multiplied by `_gasPrice` of `_gasToken` + * allows identity of being controlled without requiring ether in key balace + * @param _to destination of call + * @param _value call value (ether) + * @param _data call data + * @param _nonce current identity nonce + * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used + * @param _gasMinimal minimal amount of gas needed to complete the execution + * @param _gasToken token being used for paying `msg.sender` + * @param _messageSignatures rsv concatenated ethereum signed message signatures + */ + function callGasRelayedMultiSigned( address _to, uint256 _value, bytes _data, @@ -81,7 +110,7 @@ contract IdentityGasRelay is Identity { require(startGas > _gasMinimum); require(_nonce == nonce); nonce++; - _executeGasRelayedMultiSigned(_to, _value, _data, _nonce, _gasPrice, _gasMinimum, _gasToken, _messageSignatures); + _callGasRelayedMultiSigned(_to, _value, _data, _nonce, _gasPrice, _gasMinimum, _gasToken, _messageSignatures); if (_gasPrice > 0) { payInclusionFee( startGas - gasleft(), @@ -92,7 +121,17 @@ contract IdentityGasRelay is Identity { } } - function executeGasRelayedHash( + /** + * @notice get callHash + * @param _to destination of call + * @param _value call value (ether) + * @param _data call data + * @param _nonce current identity nonce + * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used + * @param _gasMinimal minimal amount of gas needed to complete the execution + * @param _gasToken token being used for paying `msg.sender` + */ + function callGasRelayedHash( address _to, uint256 _value, bytes32 _dataHash, @@ -103,11 +142,11 @@ contract IdentityGasRelay is Identity { ) public view - returns (bytes32) + returns (bytes32 callHash) { - return keccak256( + callHash = keccak256( address(this), - EXECUTE_PREFIX, + CALL_PREFIX, _to, _value, _dataHash, @@ -117,7 +156,12 @@ contract IdentityGasRelay is Identity { _gasToken ); } - + /** + * @notice recovers address who signed the message + * @param _signHash operation ethereum signed message hash + * @param _messageSignature message `_signHash` signature + * @param _pos which signature to read + */ function recoverKey ( bytes32 _signHash, bytes _messageSignature, @@ -142,9 +186,11 @@ contract IdentityGasRelay is Identity { } /** - * @dev divides bytes signature into `uint8 v, bytes32 r, bytes32 s` + * @dev divides bytes signature into `uint8 v, bytes32 r, bytes32 s` + * @param _pos which signature to read + * @param _signatures concatenated vrs signatures */ - function signatureSplit(bytes _signature, uint256 _pos) + function signatureSplit(bytes _signatures, uint256 _pos) pure public returns (uint8 v, bytes32 r, bytes32 s) @@ -154,14 +200,14 @@ contract IdentityGasRelay is Identity { // {bytes32 r}{bytes32 s}{uint8 v} // Compact means, uint8 is not padded to 32 bytes. assembly { - r := mload(add(_signature, mul(32,pos))) - s := mload(add(_signature, mul(64,pos))) + r := mload(add(_signatures, mul(32,pos))) + s := mload(add(_signatures, mul(64,pos))) // Here we are loading the last 32 bytes, including 31 bytes // of 's'. There is no 'mload8' to do this. // // 'byte' is not working due to the Solidity parser, so lets // use the second best option, 'and' - v := and(mload(add(_signature, mul(65,pos))), 0xff) + v := and(mload(add(_signatures, mul(65,pos))), 0xff) } require(v == 27 || v == 28); @@ -182,8 +228,10 @@ contract IdentityGasRelay is Identity { signHash = keccak256("\x19Ethereum Signed Message:\n32", _hash); } - - function _executeGasRelayedMultiSigned( + /** + * @dev needed function to avoid "too much variables, stack too deep" + */ + function _callGasRelayedMultiSigned( address _to, uint256 _value, bytes _data, @@ -200,7 +248,7 @@ contract IdentityGasRelay is Identity { require(len == minimumApprovalsByKeyPurpose[requiredKey]); bytes32 _signedHash = getSignHash( - executeGasRelayedHash( + callGasRelayedHash( _to, _value, keccak256(_data), @@ -228,6 +276,13 @@ contract IdentityGasRelay is Identity { } } + /** + * @dev performs the gas payment in the selected token + * @param _gasUsed the amount of gas used + * @param _gasPrice selected gas price + * @param _msgIncluder address who included the message + * @param _gasToken ERC20Token to transfer, or if 0x0 uses ether in balance. + */ function payInclusionFee( uint256 _gasUsed, uint256 _gasPrice, diff --git a/contracts/status/SNTController.sol b/contracts/status/SNTController.sol index 3af76ca..f65e7ab 100644 --- a/contracts/status/SNTController.sol +++ b/contracts/status/SNTController.sol @@ -7,7 +7,8 @@ import "../token/MiniMeToken.sol"; /** * @title SNTController - * @author Ricardo Guilherme Schmidt + * @author Ricardo Guilherme Schmidt (Status Research & Development GmbH) + * @notice enables economic abstraction for SNT */ contract SNTController is TokenController, Owned { From 3037d4310917933003acb6d63e30643a8e4840fe Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Wed, 21 Mar 2018 22:10:41 -0300 Subject: [PATCH 05/13] standarize to _gasMinimal var name, and _gasMinimal >= startGas --- contracts/identity/IdentityGasRelay.sol | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index 07cb168..6b8068b 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -39,7 +39,7 @@ contract IdentityGasRelay is Identity { external { uint startGas = gasleft(); - require(startGas > _gasMinimal); + require(startGas >= _gasMinimal); uint256 requiredKey = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY; require(minimumApprovalsByKeyPurpose[requiredKey] == 1); require(_nonce == nonce); @@ -100,17 +100,17 @@ contract IdentityGasRelay is Identity { bytes _data, uint _nonce, uint _gasPrice, - uint _gasMinimum, + uint _gasMinimal, address _gasToken, bytes _messageSignatures ) external { uint startGas = gasleft(); - require(startGas > _gasMinimum); + require(startGas >= _gasMinimal); require(_nonce == nonce); nonce++; - _callGasRelayedMultiSigned(_to, _value, _data, _nonce, _gasPrice, _gasMinimum, _gasToken, _messageSignatures); + _callGasRelayedMultiSigned(_to, _value, _data, _nonce, _gasPrice, _gasMinimal, _gasToken, _messageSignatures); if (_gasPrice > 0) { payInclusionFee( startGas - gasleft(), @@ -137,7 +137,7 @@ contract IdentityGasRelay is Identity { bytes32 _dataHash, uint _nonce, uint256 _gasPrice, - uint256 _gasMinimum, + uint256 _gasMinimal, address _gasToken ) public @@ -152,7 +152,7 @@ contract IdentityGasRelay is Identity { _dataHash, _nonce, _gasPrice, - _gasMinimum, + _gasMinimal, _gasToken ); } @@ -237,7 +237,7 @@ contract IdentityGasRelay is Identity { bytes _data, uint _nonce, uint _gasPrice, - uint _gasMinimum, + uint _gasMinimal, address _gasToken, bytes _messageSignatures ) @@ -254,7 +254,7 @@ contract IdentityGasRelay is Identity { keccak256(_data), _nonce, _gasPrice, - _gasMinimum, + _gasMinimal, _gasToken ) ); From 39fcf1a05537d2faf20a196ea93db9fbbca86d32 Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Wed, 21 Mar 2018 22:13:51 -0300 Subject: [PATCH 06/13] fix natspec --- contracts/identity/IdentityGasRelay.sol | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index 6b8068b..8610d12 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -125,7 +125,7 @@ contract IdentityGasRelay is Identity { * @notice get callHash * @param _to destination of call * @param _value call value (ether) - * @param _data call data + * @param _dataHash call data hash * @param _nonce current identity nonce * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used * @param _gasMinimal minimal amount of gas needed to complete the execution From 789252ba97f763cca308a9ff64f784736f6c9103 Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Wed, 21 Mar 2018 23:36:31 -0300 Subject: [PATCH 07/13] executed event always fire with executin result bool --- contracts/identity/IdentityGasRelay.sol | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index 8610d12..814dc6d 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -12,7 +12,7 @@ contract IdentityGasRelay is Identity { bytes4 public constant CALL_PREFIX = bytes4(keccak256("callGasRelayed(address,uint256,bytes32,uint256,uint256,address)")); - event ExecutedGasRelayed(bytes32 signHash); + event ExecutedGasRelayed(bytes32 signHash, bool success); /** * @notice include ethereum signed callHash in return of gas proportional amount multiplied by `_gasPrice` of `_gasToken` @@ -68,10 +68,9 @@ contract IdentityGasRelay is Identity { ) ); - if (_to.call.value(_value)(_data)) { - emit ExecutedGasRelayed(_signedHash); - } - + bool success = _to.call.value(_value)(_data); + emit ExecutedGasRelayed(_signedHash, success); + if(_gasPrice > 0) { payInclusionFee( startGas - gasleft(), From 3bdba827c9bcef784af47e95165b4be1952662df Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Thu, 22 Mar 2018 13:09:02 -0300 Subject: [PATCH 08/13] move nonce check as first step (gas relayer pays this gas) --- contracts/identity/IdentityGasRelay.sol | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index 814dc6d..9dd2e2b 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -38,11 +38,11 @@ contract IdentityGasRelay is Identity { ) external { + require(_nonce == nonce); uint startGas = gasleft(); require(startGas >= _gasMinimal); uint256 requiredKey = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY; require(minimumApprovalsByKeyPurpose[requiredKey] == 1); - require(_nonce == nonce); nonce++; bytes32 _signedHash = getSignHash( @@ -105,9 +105,9 @@ contract IdentityGasRelay is Identity { ) external { + require(_nonce == nonce); uint startGas = gasleft(); require(startGas >= _gasMinimal); - require(_nonce == nonce); nonce++; _callGasRelayedMultiSigned(_to, _value, _data, _nonce, _gasPrice, _gasMinimal, _gasToken, _messageSignatures); if (_gasPrice > 0) { From 17b6b7357ce4052cf2bf8ae4d5e07a34f5ebae0b Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Thu, 22 Mar 2018 23:12:12 -0300 Subject: [PATCH 09/13] simplify contract for easier readibility --- contracts/identity/IdentityGasRelay.sol | 260 +++++++++--------------- 1 file changed, 92 insertions(+), 168 deletions(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index 9dd2e2b..cd544d8 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -1,4 +1,4 @@ -pragma solidity ^0.4.17; +pragma solidity ^0.4.21; import "./Identity.sol"; import "../token/ERC20Token.sol"; @@ -10,77 +10,10 @@ import "../token/ERC20Token.sol"; */ contract IdentityGasRelay is Identity { - bytes4 public constant CALL_PREFIX = bytes4(keccak256("callGasRelayed(address,uint256,bytes32,uint256,uint256,address)")); + bytes4 public constant CALL_PREFIX = bytes4(keccak256("callGasRelay(address,uint256,bytes32,uint256,uint256,address)")); event ExecutedGasRelayed(bytes32 signHash, bool success); - /** - * @notice include ethereum signed callHash in return of gas proportional amount multiplied by `_gasPrice` of `_gasToken` - * allows identity of being controlled without requiring ether in key balace - * @param _to destination of call - * @param _value call value (ether) - * @param _data call data - * @param _nonce current identity nonce - * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used - * @param _gasMinimal minimal amount of gas needed to complete the execution - * @param _gasToken token being used for paying `msg.sender` - * @param _messageSignature rsv concatenated ethereum signed message signature - */ - function callGasRelayed( - address _to, - uint256 _value, - bytes _data, - uint _nonce, - uint _gasPrice, - uint _gasMinimal, - address _gasToken, - bytes _messageSignature - ) - external - { - require(_nonce == nonce); - uint startGas = gasleft(); - require(startGas >= _gasMinimal); - uint256 requiredKey = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY; - require(minimumApprovalsByKeyPurpose[requiredKey] == 1); - nonce++; - - bytes32 _signedHash = getSignHash( - callGasRelayedHash( - _to, - _value, - keccak256(_data), - _nonce, - _gasPrice, - _gasMinimal, - _gasToken - ) - ); - - require( - isKeyPurpose( - recoverKey( - _signedHash, - _messageSignature, - 0 - ), - requiredKey - ) - ); - - bool success = _to.call.value(_value)(_data); - emit ExecutedGasRelayed(_signedHash, success); - - if(_gasPrice > 0) { - payInclusionFee( - startGas - gasleft(), - _gasPrice, - msg.sender, - _gasToken - ); - } - } - /** * @notice include ethereum signed callHash in return of gas proportional amount multiplied by `_gasPrice` of `_gasToken` * allows identity of being controlled without requiring ether in key balace @@ -91,9 +24,9 @@ contract IdentityGasRelay is Identity { * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used * @param _gasMinimal minimal amount of gas needed to complete the execution * @param _gasToken token being used for paying `msg.sender` - * @param _messageSignatures rsv concatenated ethereum signed message signatures + * @param _messageSignatures rsv concatenated ethereum signed message signatures required */ - function callGasRelayedMultiSigned( + function callGasRelayed( address _to, uint256 _value, bytes _data, @@ -105,21 +38,81 @@ contract IdentityGasRelay is Identity { ) external { + //verify transaction parameters require(_nonce == nonce); uint startGas = gasleft(); require(startGas >= _gasMinimal); - nonce++; - _callGasRelayedMultiSigned(_to, _value, _data, _nonce, _gasPrice, _gasMinimal, _gasToken, _messageSignatures); - if (_gasPrice > 0) { - payInclusionFee( - startGas - gasleft(), + + // calculates signHash + bytes32 signHash = getSignHash( + callGasRelayHash( + _to, + _value, + keccak256(_data), + _nonce, _gasPrice, - msg.sender, - _gasToken - ); + _gasMinimal, + _gasToken + ) + ); + + //verify if signatures are valid and came from correct actors; + uint256 requiredKey = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY; + verifySignatures(requiredKey, signHash, _messageSignatures); + + //executes transaction + nonce++; + emit ExecutedGasRelayed( + signHash, + _to.call.value(_value)(_data) + ); + + //refund gas used using contract held ERC20 tokens or ETH + if (_gasPrice > 0) { + uint256 _amount = 21000 + (startGas - gasleft()); + _amount = _amount * _gasPrice; + if (_gasToken == address(0)) { + address(msg.sender).transfer(_amount); + } else { + ERC20Token(_gasToken).transfer(msg.sender, _amount); + } } } + + /** + * @notice reverts if signatures are not valid for the signed hash and required key type. + * @param _requiredKey key required to call, if _to from payload is the identity itself, is `MANAGEMENT_KEY`, else `ACTION_KEY` + * @param _signHash ethereum signable callGasRelayHash message provided for the payload + * @param _messageSignatures ethereum signed `_signHash` messages + * @return true case valid + */ + function verifySignatures( + uint256 _requiredKey, + bytes32 _signHash, + bytes _messageSignatures + ) + public + view + returns(bool) + { + uint _amountSignatures = _messageSignatures.length / 72; + require(_amountSignatures == minimumApprovalsByKeyPurpose[_requiredKey]); + bytes32 _lastKey = 0; + for (uint256 i = 0; i < _amountSignatures; i++) { + bytes32 _currentKey = recoverKey( + _signHash, + _messageSignatures, + i + ); + require(_currentKey > _lastKey); //assert keys are different + require(isKeyPurpose(_currentKey, _requiredKey)); + _lastKey = _currentKey; + } + return true; + } + + /** * @notice get callHash * @param _to destination of call @@ -129,8 +122,9 @@ contract IdentityGasRelay is Identity { * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used * @param _gasMinimal minimal amount of gas needed to complete the execution * @param _gasToken token being used for paying `msg.sender` + * @return callGasRelayHash the hash to be signed by wallet */ - function callGasRelayedHash( + function callGasRelayHash( address _to, uint256 _value, bytes32 _dataHash, @@ -141,9 +135,9 @@ contract IdentityGasRelay is Identity { ) public view - returns (bytes32 callHash) + returns (bytes32 _callGasRelayHash) { - callHash = keccak256( + _callGasRelayHash = keccak256( address(this), CALL_PREFIX, _to, @@ -155,6 +149,22 @@ contract IdentityGasRelay is Identity { _gasToken ); } + + /** + * @notice Hash a hash with `"\x19Ethereum Signed Message:\n32"` + * @param _hash Sign to hash. + * @return signHash Hash ethereum wallet signs. + */ + function getSignHash( + bytes32 _hash + ) + pure + public + returns(bytes32 signHash) + { + signHash = keccak256("\x19Ethereum Signed Message:\n32", _hash); + } + /** * @notice recovers address who signed the message * @param _signHash operation ethereum signed message hash @@ -212,90 +222,4 @@ contract IdentityGasRelay is Identity { require(v == 27 || v == 28); } - /** - * @notice Hash a hash with `"\x19Ethereum Signed Message:\n32"` - * @param _hash Sign to hash. - * @return signHash Hash to be signed. - */ - function getSignHash( - bytes32 _hash - ) - pure - public - returns(bytes32 signHash) - { - signHash = keccak256("\x19Ethereum Signed Message:\n32", _hash); - } - - /** - * @dev needed function to avoid "too much variables, stack too deep" - */ - function _callGasRelayedMultiSigned( - address _to, - uint256 _value, - bytes _data, - uint _nonce, - uint _gasPrice, - uint _gasMinimal, - address _gasToken, - bytes _messageSignatures - ) - private - { - uint256 requiredKey = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY; - uint256 len = _messageSignatures.length / 72; - require(len == minimumApprovalsByKeyPurpose[requiredKey]); - - bytes32 _signedHash = getSignHash( - callGasRelayedHash( - _to, - _value, - keccak256(_data), - _nonce, - _gasPrice, - _gasMinimal, - _gasToken - ) - ); - - bytes32 _lastKey = 0; - for (uint256 i = 0; i < len; i++) { - bytes32 _key = recoverKey( - _signedHash, - _messageSignatures, - i - ); - require(_key > _lastKey); //assert keys are different - require(isKeyPurpose(_key, requiredKey)); - _lastKey = _key; - } - - if (_to.call.value(_value)(_data)) { - emit ExecutedGasRelayed(_signedHash); - } - } - - /** - * @dev performs the gas payment in the selected token - * @param _gasUsed the amount of gas used - * @param _gasPrice selected gas price - * @param _msgIncluder address who included the message - * @param _gasToken ERC20Token to transfer, or if 0x0 uses ether in balance. - */ - function payInclusionFee( - uint256 _gasUsed, - uint256 _gasPrice, - address _msgIncluder, - address _gasToken - ) - private - { - uint256 _amount = (21000 + _gasUsed) * _gasPrice; - if (_gasToken == address(0)) { - address(_msgIncluder).transfer(_amount); - } else { - ERC20Token(_gasToken).transfer(_msgIncluder, _amount); - } - } - } \ No newline at end of file From fe4a10dfcded78901cdf305d3647a218f9f4b378 Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Thu, 22 Mar 2018 23:31:20 -0300 Subject: [PATCH 10/13] makes "call" more explicit --- contracts/identity/IdentityGasRelay.sol | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index cd544d8..5254cee 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -57,16 +57,20 @@ contract IdentityGasRelay is Identity { ); //verify if signatures are valid and came from correct actors; - uint256 requiredKey = _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY; - verifySignatures(requiredKey, signHash, _messageSignatures); + verifySignatures( + _to == address(this) ? MANAGEMENT_KEY : ACTION_KEY, + signHash, + _messageSignatures + ); //executes transaction nonce++; + bool success = _to.call.value(_value)(_data); emit ExecutedGasRelayed( signHash, - _to.call.value(_value)(_data) + success ); - + //refund gas used using contract held ERC20 tokens or ETH if (_gasPrice > 0) { uint256 _amount = 21000 + (startGas - gasleft()); From 4c5cb43da05fe94ed83f0700108fa7e2cf89e4f4 Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Sat, 24 Mar 2018 01:26:06 -0300 Subject: [PATCH 11/13] added approve and call gas relayed for ERC20 --- contracts/identity/IdentityGasRelay.sol | 138 +++++++++++++++++++++++- 1 file changed, 137 insertions(+), 1 deletion(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index 5254cee..2651282 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -11,11 +11,12 @@ import "../token/ERC20Token.sol"; contract IdentityGasRelay is Identity { bytes4 public constant CALL_PREFIX = bytes4(keccak256("callGasRelay(address,uint256,bytes32,uint256,uint256,address)")); + bytes4 public constant APPROVEANDCALL_PREFIX = bytes4(keccak256("approveAndCallGasRelay(address,address,uint256,bytes32,uint256,uint256)")); event ExecutedGasRelayed(bytes32 signHash, bool success); /** - * @notice include ethereum signed callHash in return of gas proportional amount multiplied by `_gasPrice` of `_gasToken` + * @notice include ethereum signed callHash in return of gas proportional amount multiplied by `_gasPrice` of `_gasToken` * allows identity of being controlled without requiring ether in key balace * @param _to destination of call * @param _value call value (ether) @@ -83,6 +84,82 @@ contract IdentityGasRelay is Identity { } } + /** + * @notice include ethereum signed approve ERC20 and call hash + * (`ERC20Token(baseToken).approve(_to, _value)` + `_to.call(_data)`). + * in return of gas proportional amount multiplied by `_gasPrice` of `_gasToken` + * fixes race condition in double transaction for ERC20. + * @param _baseToken token approved for `_to` + * @param _to destination of call + * @param _value call value (ether) + * @param _data call data + * @param _nonce current identity nonce + * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used + * @param _gasMinimal minimal amount of gas needed to complete the execution + * @param _gasToken token being used for paying `msg.sender` + * @param _messageSignatures rsv concatenated ethereum signed message signatures required + */ + function approveAndCallGasRelayed( + address _baseToken, + address _to, + uint256 _value, + bytes _data, + uint _nonce, + uint _gasPrice, + uint _gasMinimal, + address _gasToken, + bytes _messageSignatures + ) + external + { + //verify transaction parameters + require(_nonce == nonce); + uint startGas = gasleft(); + require(startGas >= _gasMinimal); + require(_baseToken != address(0)); //_baseToken should be something! + require(_to != address(this)); //no management with approveAndCall + + // calculates signHash + bytes32 signHash = getSignHash( + approveAndCallGasRelayHash( + _baseToken, + _to, + _value, + keccak256(_data), + _nonce, + _gasPrice, + _gasMinimal, + _gasToken + ) + ); + + //verify if signatures are valid and came from correct actors; + verifySignatures( + ACTION_KEY, //no management with approveAndCall + signHash, + _messageSignatures + ); + + approveAndCall( + signHash, + _baseToken, + _to, + _value, + _data + ); + + //refund gas used using contract held ERC20 tokens or ETH + if (_gasPrice > 0) { + uint256 _amount = 21000 + (startGas - gasleft()); + _amount = _amount * _gasPrice; + if (_gasToken == address(0)) { + address(msg.sender).transfer(_amount); + } else { + ERC20Token(_gasToken).transfer(msg.sender, _amount); + } + } + + } /** * @notice reverts if signatures are not valid for the signed hash and required key type. @@ -154,6 +231,46 @@ contract IdentityGasRelay is Identity { ); } + + /** + * @notice get callHash + * @param _to destination of call + * @param _value call value (ether) + * @param _dataHash call data hash + * @param _nonce current identity nonce + * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used + * @param _gasMinimal minimal amount of gas needed to complete the execution + * @param _gasToken token being used for paying `msg.sender` + * @return callGasRelayHash the hash to be signed by wallet + */ + function approveAndCallGasRelayHash( + address _baseToken, + address _to, + uint256 _value, + bytes32 _dataHash, + uint _nonce, + uint256 _gasPrice, + uint256 _gasMinimal, + address _gasToken + ) + public + view + returns (bytes32 _callGasRelayHash) + { + _callGasRelayHash = keccak256( + address(this), + APPROVEANDCALL_PREFIX, + _baseToken, + _to, + _value, + _dataHash, + _nonce, + _gasPrice, + _gasMinimal, + _gasToken + ); + } + /** * @notice Hash a hash with `"\x19Ethereum Signed Message:\n32"` * @param _hash Sign to hash. @@ -226,4 +343,23 @@ contract IdentityGasRelay is Identity { require(v == 27 || v == 28); } + function approveAndCall( + bytes32 _signHash, + address _token, + address _to, + uint256 _value, + bytes _data + ) + private + { + //executes transaction + nonce++; + ERC20Token(_token).approve(_to, _value); + emit ExecutedGasRelayed( + _signHash, + _to.call(_data) + ); + + } + } \ No newline at end of file From 922fea958f2aaa2359e8633aedda391d703a9af5 Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Tue, 17 Apr 2018 11:18:23 -0300 Subject: [PATCH 12/13] change from gasMinimal to gasLimit --- contracts/identity/IdentityGasRelay.sol | 34 ++++++++++++------------- 1 file changed, 17 insertions(+), 17 deletions(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index 2651282..b29759f 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -23,7 +23,7 @@ contract IdentityGasRelay is Identity { * @param _data call data * @param _nonce current identity nonce * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used - * @param _gasMinimal minimal amount of gas needed to complete the execution + * @param _gasLimit minimal gasLimit required to execute this call * @param _gasToken token being used for paying `msg.sender` * @param _messageSignatures rsv concatenated ethereum signed message signatures required */ @@ -33,17 +33,17 @@ contract IdentityGasRelay is Identity { bytes _data, uint _nonce, uint _gasPrice, - uint _gasMinimal, + uint _gasLimit, address _gasToken, bytes _messageSignatures ) external { + uint startGas = gasleft(); + require(startGas >= _gasLimit); //verify transaction parameters require(_nonce == nonce); - uint startGas = gasleft(); - require(startGas >= _gasMinimal); - + ; // calculates signHash bytes32 signHash = getSignHash( callGasRelayHash( @@ -52,7 +52,7 @@ contract IdentityGasRelay is Identity { keccak256(_data), _nonce, _gasPrice, - _gasMinimal, + _gasLimit, _gasToken ) ); @@ -95,7 +95,7 @@ contract IdentityGasRelay is Identity { * @param _data call data * @param _nonce current identity nonce * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used - * @param _gasMinimal minimal amount of gas needed to complete the execution + * @param _gasLimit minimal gasLimit required to execute this call * @param _gasToken token being used for paying `msg.sender` * @param _messageSignatures rsv concatenated ethereum signed message signatures required */ @@ -106,16 +106,16 @@ contract IdentityGasRelay is Identity { bytes _data, uint _nonce, uint _gasPrice, - uint _gasMinimal, + uint _gasLimit, address _gasToken, bytes _messageSignatures ) external { + uint startGas = gasleft(); + require(startGas >= _gasLimit); //verify transaction parameters require(_nonce == nonce); - uint startGas = gasleft(); - require(startGas >= _gasMinimal); require(_baseToken != address(0)); //_baseToken should be something! require(_to != address(this)); //no management with approveAndCall @@ -128,7 +128,7 @@ contract IdentityGasRelay is Identity { keccak256(_data), _nonce, _gasPrice, - _gasMinimal, + _gasLimit, _gasToken ) ); @@ -201,7 +201,7 @@ contract IdentityGasRelay is Identity { * @param _dataHash call data hash * @param _nonce current identity nonce * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used - * @param _gasMinimal minimal amount of gas needed to complete the execution + * @param _gasLimit minimal gasLimit required to execute this call * @param _gasToken token being used for paying `msg.sender` * @return callGasRelayHash the hash to be signed by wallet */ @@ -211,7 +211,7 @@ contract IdentityGasRelay is Identity { bytes32 _dataHash, uint _nonce, uint256 _gasPrice, - uint256 _gasMinimal, + uint256 _gasLimit, address _gasToken ) public @@ -226,7 +226,7 @@ contract IdentityGasRelay is Identity { _dataHash, _nonce, _gasPrice, - _gasMinimal, + _gasLimit, _gasToken ); } @@ -239,7 +239,7 @@ contract IdentityGasRelay is Identity { * @param _dataHash call data hash * @param _nonce current identity nonce * @param _gasPrice price in SNT paid back to msg.sender for each gas unit used - * @param _gasMinimal minimal amount of gas needed to complete the execution + * @param _gasLimit minimal gasLimit required to execute this call * @param _gasToken token being used for paying `msg.sender` * @return callGasRelayHash the hash to be signed by wallet */ @@ -250,7 +250,7 @@ contract IdentityGasRelay is Identity { bytes32 _dataHash, uint _nonce, uint256 _gasPrice, - uint256 _gasMinimal, + uint256 _gasLimit, address _gasToken ) public @@ -266,7 +266,7 @@ contract IdentityGasRelay is Identity { _dataHash, _nonce, _gasPrice, - _gasMinimal, + _gasLimit, _gasToken ); } From be024406d2ffe884b1074a55ca885bef0298fea2 Mon Sep 17 00:00:00 2001 From: Ricardo Guilherme Schmidt <3esmit@gmail.com> Date: Tue, 17 Apr 2018 22:24:45 -0300 Subject: [PATCH 13/13] small fixes --- contracts/identity/IdentityGasRelay.sol | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/contracts/identity/IdentityGasRelay.sol b/contracts/identity/IdentityGasRelay.sol index b29759f..b02b284 100644 --- a/contracts/identity/IdentityGasRelay.sol +++ b/contracts/identity/IdentityGasRelay.sol @@ -40,10 +40,9 @@ contract IdentityGasRelay is Identity { external { uint startGas = gasleft(); - require(startGas >= _gasLimit); //verify transaction parameters + require(startGas >= _gasLimit); require(_nonce == nonce); - ; // calculates signHash bytes32 signHash = getSignHash( callGasRelayHash( @@ -113,8 +112,8 @@ contract IdentityGasRelay is Identity { external { uint startGas = gasleft(); - require(startGas >= _gasLimit); //verify transaction parameters + require(startGas >= _gasLimit); require(_nonce == nonce); require(_baseToken != address(0)); //_baseToken should be something! require(_to != address(this)); //no management with approveAndCall @@ -178,7 +177,7 @@ contract IdentityGasRelay is Identity { returns(bool) { uint _amountSignatures = _messageSignatures.length / 72; - require(_amountSignatures == minimumApprovalsByKeyPurpose[_requiredKey]); + require(_amountSignatures == purposeThreshold[_requiredKey]); bytes32 _lastKey = 0; for (uint256 i = 0; i < _amountSignatures; i++) { bytes32 _currentKey = recoverKey(