-
Notifications
You must be signed in to change notification settings - Fork 160
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* Test cases for data store * Rebase conflicts resolved * wip ds proxy * DS deployment * Fixed create instance helper * tests fixed * Added KYC TM module * Added array operation * Added features * Added batch functions * Removed multi delete * Removed overloaded external functions * Added test cases * Code cleanup and comments * Added test cases for security token * Added more test cases for data store * minor fixes * Small update to setSecurityToken * Fix
- Loading branch information
Showing
16 changed files
with
1,279 additions
and
7 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,363 @@ | ||
pragma solidity ^0.5.0; | ||
|
||
import "../interfaces/ISecurityToken.sol"; | ||
import "../interfaces/IOwnable.sol"; | ||
import "../interfaces/IDataStore.sol"; | ||
import "./DataStoreStorage.sol"; | ||
|
||
/** | ||
* @title Data store contract that stores data for all the modules in a central contract. | ||
*/ | ||
contract DataStore is DataStoreStorage, IDataStore { | ||
//NB To modify a specific element of an array, First push a new element to the array and then delete the old element. | ||
//Whenver an element is deleted from an Array, last element of that array is moved to the index of deleted element. | ||
//Delegate with MANAGEDATA permission can modify data. | ||
|
||
event SecurityTokenChanged(address indexed _oldSecurityToken, address indexed _newSecurityToken); | ||
|
||
modifier onlyAuthorized() { | ||
bool isOwner = msg.sender == IOwnable(address(securityToken)).owner(); | ||
require(isOwner || | ||
securityToken.isModule(msg.sender, DATA_KEY) || | ||
securityToken.checkPermission(msg.sender, address(this), MANAGEDATA), | ||
"Unauthorized" | ||
); | ||
_; | ||
} | ||
|
||
modifier validKey(bytes32 _key) { | ||
require(_key != bytes32(0), "Missing key"); | ||
_; | ||
} | ||
|
||
modifier validArrayLength(uint256 _keyLength, uint256 _dataLength) { | ||
require(_keyLength == _dataLength, "Array length mismatch"); | ||
_; | ||
} | ||
|
||
modifier onlyOwner() { | ||
require(msg.sender == IOwnable(address(securityToken)).owner(), "Unauthorized"); | ||
_; | ||
} | ||
|
||
/** | ||
* @dev Changes security token atatched to this data store | ||
* @param _securityToken address of the security token | ||
*/ | ||
function setSecurityToken(address _securityToken) external onlyOwner { | ||
require(_securityToken != address(0), "Invalid address"); | ||
emit SecurityTokenChanged(address(securityToken), _securityToken); | ||
securityToken = ISecurityToken(_securityToken); | ||
} | ||
|
||
/** | ||
* @dev Stores a uint256 data against a key | ||
* @param _key Unique key to identify the data | ||
* @param _data Data to be stored against the key | ||
*/ | ||
function setUint256(bytes32 _key, uint256 _data) external onlyAuthorized { | ||
_setData(_key, _data); | ||
} | ||
|
||
function setBytes32(bytes32 _key, bytes32 _data) external onlyAuthorized { | ||
_setData(_key, _data); | ||
} | ||
|
||
function setAddress(bytes32 _key, address _data) external onlyAuthorized { | ||
_setData(_key, _data); | ||
} | ||
|
||
function setString(bytes32 _key, string calldata _data) external onlyAuthorized { | ||
_setData(_key, _data); | ||
} | ||
|
||
function setBytes(bytes32 _key, bytes calldata _data) external onlyAuthorized { | ||
_setData(_key, _data); | ||
} | ||
|
||
function setBool(bytes32 _key, bool _data) external onlyAuthorized { | ||
_setData(_key, _data); | ||
} | ||
|
||
/** | ||
* @dev Stores a uint256 array against a key | ||
* @param _key Unique key to identify the array | ||
* @param _data Array to be stored against the key | ||
*/ | ||
function setUint256Array(bytes32 _key, uint256[] calldata _data) external onlyAuthorized { | ||
_setData(_key, _data); | ||
} | ||
|
||
function setBytes32Array(bytes32 _key, bytes32[] calldata _data) external onlyAuthorized { | ||
_setData(_key, _data); | ||
} | ||
|
||
function setAddressArray(bytes32 _key, address[] calldata _data) external onlyAuthorized { | ||
_setData(_key, _data); | ||
} | ||
|
||
function setBoolArray(bytes32 _key, bool[] calldata _data) external onlyAuthorized { | ||
_setData(_key, _data); | ||
} | ||
|
||
/** | ||
* @dev Inserts a uint256 element to the array identified by the key | ||
* @param _key Unique key to identify the array | ||
* @param _data Element to push into the array | ||
*/ | ||
function insertUint256(bytes32 _key, uint256 _data) external onlyAuthorized { | ||
_insertData(_key, _data); | ||
} | ||
|
||
function insertBytes32(bytes32 _key, bytes32 _data) external onlyAuthorized { | ||
_insertData(_key, _data); | ||
} | ||
|
||
function insertAddress(bytes32 _key, address _data) external onlyAuthorized { | ||
_insertData(_key, _data); | ||
} | ||
|
||
function insertBool(bytes32 _key, bool _data) external onlyAuthorized { | ||
_insertData(_key, _data); | ||
} | ||
|
||
/** | ||
* @dev Deletes an element from the array identified by the key. | ||
* When an element is deleted from an Array, last element of that array is moved to the index of deleted element. | ||
* @param _key Unique key to identify the array | ||
* @param _index Index of the element to delete | ||
*/ | ||
function deleteUint256(bytes32 _key, uint256 _index) external onlyAuthorized { | ||
_deleteUint(_key, _index); | ||
} | ||
|
||
function deleteBytes32(bytes32 _key, uint256 _index) external onlyAuthorized { | ||
_deleteBytes32(_key, _index); | ||
} | ||
|
||
function deleteAddress(bytes32 _key, uint256 _index) external onlyAuthorized { | ||
_deleteAddress(_key, _index); | ||
} | ||
|
||
function deleteBool(bytes32 _key, uint256 _index) external onlyAuthorized { | ||
_deleteBool(_key, _index); | ||
} | ||
|
||
/** | ||
* @dev Stores multiple uint256 data against respective keys | ||
* @param _keys Array of keys to identify the data | ||
* @param _data Array of data to be stored against the respective keys | ||
*/ | ||
function setUint256Multi(bytes32[] calldata _keys, uint256[] calldata _data) external onlyAuthorized validArrayLength(_keys.length, _data.length) { | ||
for (uint256 i = 0; i < _keys.length; i++) { | ||
_setData(_keys[i], _data[i]); | ||
} | ||
} | ||
|
||
function setBytes32Multi(bytes32[] calldata _keys, bytes32[] calldata _data) external onlyAuthorized validArrayLength(_keys.length, _data.length) { | ||
for (uint256 i = 0; i < _keys.length; i++) { | ||
_setData(_keys[i], _data[i]); | ||
} | ||
} | ||
|
||
function setAddressMulti(bytes32[] calldata _keys, address[] calldata _data) external onlyAuthorized validArrayLength(_keys.length, _data.length) { | ||
for (uint256 i = 0; i < _keys.length; i++) { | ||
_setData(_keys[i], _data[i]); | ||
} | ||
} | ||
|
||
function setBoolMulti(bytes32[] calldata _keys, bool[] calldata _data) external onlyAuthorized validArrayLength(_keys.length, _data.length) { | ||
for (uint256 i = 0; i < _keys.length; i++) { | ||
_setData(_keys[i], _data[i]); | ||
} | ||
} | ||
|
||
/** | ||
* @dev Inserts multiple uint256 elements to the array identified by the respective keys | ||
* @param _keys Array of keys to identify the data | ||
* @param _data Array of data to be inserted in arrays of the respective keys | ||
*/ | ||
function insertUint256Multi(bytes32[] calldata _keys, uint256[] calldata _data) external onlyAuthorized validArrayLength(_keys.length, _data.length) { | ||
for (uint256 i = 0; i < _keys.length; i++) { | ||
_insertData(_keys[i], _data[i]); | ||
} | ||
} | ||
|
||
function insertBytes32Multi(bytes32[] calldata _keys, bytes32[] calldata _data) external onlyAuthorized validArrayLength(_keys.length, _data.length) { | ||
for (uint256 i = 0; i < _keys.length; i++) { | ||
_insertData(_keys[i], _data[i]); | ||
} | ||
} | ||
|
||
function insertAddressMulti(bytes32[] calldata _keys, address[] calldata _data) external onlyAuthorized validArrayLength(_keys.length, _data.length) { | ||
for (uint256 i = 0; i < _keys.length; i++) { | ||
_insertData(_keys[i], _data[i]); | ||
} | ||
} | ||
|
||
function insertBoolMulti(bytes32[] calldata _keys, bool[] calldata _data) external onlyAuthorized validArrayLength(_keys.length, _data.length) { | ||
for (uint256 i = 0; i < _keys.length; i++) { | ||
_insertData(_keys[i], _data[i]); | ||
} | ||
} | ||
|
||
function getUint256(bytes32 _key) external view returns(uint256) { | ||
return uintData[_key]; | ||
} | ||
|
||
function getBytes32(bytes32 _key) external view returns(bytes32) { | ||
return bytes32Data[_key]; | ||
} | ||
|
||
function getAddress(bytes32 _key) external view returns(address) { | ||
return addressData[_key]; | ||
} | ||
|
||
function getString(bytes32 _key) external view returns(string memory) { | ||
return stringData[_key]; | ||
} | ||
|
||
function getBytes(bytes32 _key) external view returns(bytes memory) { | ||
return bytesData[_key]; | ||
} | ||
|
||
function getBool(bytes32 _key) external view returns(bool) { | ||
return boolData[_key]; | ||
} | ||
|
||
function getUint256Array(bytes32 _key) external view returns(uint256[] memory) { | ||
return uintArrayData[_key]; | ||
} | ||
|
||
function getBytes32Array(bytes32 _key) external view returns(bytes32[] memory) { | ||
return bytes32ArrayData[_key]; | ||
} | ||
|
||
function getAddressArray(bytes32 _key) external view returns(address[] memory) { | ||
return addressArrayData[_key]; | ||
} | ||
|
||
function getBoolArray(bytes32 _key) external view returns(bool[] memory) { | ||
return boolArrayData[_key]; | ||
} | ||
|
||
function getUint256ArrayLength(bytes32 _key) external view returns(uint256) { | ||
return uintArrayData[_key].length; | ||
} | ||
|
||
function getBytes32ArrayLength(bytes32 _key) external view returns(uint256) { | ||
return bytes32ArrayData[_key].length; | ||
} | ||
|
||
function getAddressArrayLength(bytes32 _key) external view returns(uint256) { | ||
return addressArrayData[_key].length; | ||
} | ||
|
||
function getBoolArrayLength(bytes32 _key) external view returns(uint256) { | ||
return boolArrayData[_key].length; | ||
} | ||
|
||
function getUint256ArrayElement(bytes32 _key, uint256 _index) external view returns(uint256) { | ||
return uintArrayData[_key][_index]; | ||
} | ||
|
||
function getBytes32ArrayElement(bytes32 _key, uint256 _index) external view returns(bytes32) { | ||
return bytes32ArrayData[_key][_index]; | ||
} | ||
|
||
function getAddressArrayElement(bytes32 _key, uint256 _index) external view returns(address) { | ||
return addressArrayData[_key][_index]; | ||
} | ||
|
||
function getBoolArrayElement(bytes32 _key, uint256 _index) external view returns(bool) { | ||
return boolArrayData[_key][_index]; | ||
} | ||
|
||
function _setData(bytes32 _key, uint256 _data) internal validKey(_key) { | ||
uintData[_key] = _data; | ||
} | ||
|
||
function _setData(bytes32 _key, bytes32 _data) internal validKey(_key) { | ||
bytes32Data[_key] = _data; | ||
} | ||
|
||
function _setData(bytes32 _key, address _data) internal validKey(_key) { | ||
addressData[_key] = _data; | ||
} | ||
|
||
function _setData(bytes32 _key, string memory _data) internal validKey(_key) { | ||
stringData[_key] = _data; | ||
} | ||
|
||
function _setData(bytes32 _key, bytes memory _data) internal validKey(_key) { | ||
bytesData[_key] = _data; | ||
} | ||
|
||
function _setData(bytes32 _key, bool _data) internal validKey(_key) { | ||
boolData[_key] = _data; | ||
} | ||
|
||
function _setData(bytes32 _key, uint256[] memory _data) internal validKey(_key) { | ||
uintArrayData[_key] = _data; | ||
} | ||
|
||
function _setData(bytes32 _key, bytes32[] memory _data) internal validKey(_key) { | ||
bytes32ArrayData[_key] = _data; | ||
} | ||
|
||
function _setData(bytes32 _key, address[] memory _data) internal validKey(_key) { | ||
addressArrayData[_key] = _data; | ||
} | ||
|
||
function _setData(bytes32 _key, bool[] memory _data) internal validKey(_key) { | ||
boolArrayData[_key] = _data; | ||
} | ||
|
||
function _insertData(bytes32 _key, uint256 _data) internal validKey(_key) { | ||
uintArrayData[_key].push(_data); | ||
} | ||
|
||
function _insertData(bytes32 _key, bytes32 _data) internal validKey(_key) { | ||
bytes32ArrayData[_key].push(_data); | ||
} | ||
|
||
function _insertData(bytes32 _key, address _data) internal validKey(_key) { | ||
addressArrayData[_key].push(_data); | ||
} | ||
|
||
function _insertData(bytes32 _key, bool _data) internal validKey(_key) { | ||
boolArrayData[_key].push(_data); | ||
} | ||
|
||
function _deleteUint(bytes32 _key, uint256 _index) internal validKey(_key) { | ||
require(uintArrayData[_key].length > _index, "Invalid Index"); //Also prevents undeflow | ||
if(uintArrayData[_key].length - 1 != _index) { | ||
uintArrayData[_key][_index] = uintArrayData[_key][uintArrayData[_key].length - 1]; | ||
} | ||
uintArrayData[_key].length--; | ||
} | ||
|
||
function _deleteBytes32(bytes32 _key, uint256 _index) internal validKey(_key) { | ||
require(bytes32ArrayData[_key].length > _index, "Invalid Index"); //Also prevents undeflow | ||
if(bytes32ArrayData[_key].length - 1 != _index) { | ||
bytes32ArrayData[_key][_index] = bytes32ArrayData[_key][bytes32ArrayData[_key].length - 1]; | ||
} | ||
bytes32ArrayData[_key].length--; | ||
} | ||
|
||
function _deleteAddress(bytes32 _key, uint256 _index) internal validKey(_key) { | ||
require(addressArrayData[_key].length > _index, "Invalid Index"); //Also prevents undeflow | ||
if(addressArrayData[_key].length - 1 != _index) { | ||
addressArrayData[_key][_index] = addressArrayData[_key][addressArrayData[_key].length - 1]; | ||
} | ||
addressArrayData[_key].length--; | ||
} | ||
|
||
function _deleteBool(bytes32 _key, uint256 _index) internal validKey(_key) { | ||
require(boolArrayData[_key].length > _index, "Invalid Index"); //Also prevents undeflow | ||
if(boolArrayData[_key].length - 1 != _index) { | ||
boolArrayData[_key][_index] = boolArrayData[_key][boolArrayData[_key].length - 1]; | ||
} | ||
boolArrayData[_key].length--; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
pragma solidity ^0.5.0; | ||
|
||
import "../proxy/DataStoreProxy.sol"; | ||
|
||
contract DataStoreFactory { | ||
|
||
address public implementation; | ||
|
||
constructor(address _implementation) public { | ||
require(_implementation != address(0), "Address should not be 0x"); | ||
implementation = _implementation; | ||
} | ||
|
||
function generateDataStore(address _securityToken) public returns (address) { | ||
DataStoreProxy dsProxy = new DataStoreProxy(_securityToken, implementation); | ||
return address(dsProxy); | ||
} | ||
} |
Oops, something went wrong.