Install project packages:
npm install
Running Hardhat tests:
npx hardhat test
Running code coverage tests:
npx hardhat coverage
Running deployment to a new chain:
npx hardhat setup --network {network name}
The previous approach of the project has multiple inefficient solutions. At the moment of OtoCo conception it makes sense but due to increased demand of gas at Ethereum, we are required to drastically change our approach to make Dapp feasible to work on mainnet.
- Charging for spin-up usually requires the user to exchange their ETH for DAI before proceeding with approval step during spin-up.
- The DAI approval and transfer steps cost at least 90K additional gas for the user.
- Multiple bugs happen on different wallets during the approval step. New users also doesn't not comprehend very well that the approval step doesn't transfer the tokens, only approve for the deployer to consume them after.
- The activation phase that creates the new entity instantiates a new contract on the blockchain. It required an absurd amount of 300K gas for a simple Ownable contract.
- Entity > Ownership relation couldn't be tracked by the Master smart-contract requiring us to rely on third party services like TheGraph to keep tracking Series and his Owner.
- Having multiple Master contracts sometimes confuses users that don't recognize those different contracts for each jurisdiction.
- Master Registry create a unecessary complexity requiring different structures to store different plugins.
- The first decision was get rid of DAI payment since that make so much friction to the process.
- We also decide to charge a percentage of fees in ETH during creation. We are targeting entity creation at 100k in gas. So for a 50gwei gas price a company creation would cost 15USD at max (ETH at 3.000 USD).
- Based on multiple projects around like ENS, KaliDAO and The Graph that uses ERC721 to define ownership, we decided to switch our Ownables contracts to a single ERC721 contract to define ownership of entities making simpler to list the this entities owned by a single address, without requiring a third-party service.
- Otoco values paid for plugins are transferred to OtoCoMaster instantly requiring a single transaction to withdraw all values paid for company creation and plugins.
- Get rid of Master Registry contract, now entity owners call directly the plugins they want to add to their company. The informations are stored directly on plugin storage.
- Allow add and tweak jurisdictions without require upgrade Master contract.
- ADMINISTRATOR: Master contract owner.
- USER: Dapp user.
- MAINTAINER: The Plugin contract owner.
- Unique contracts that hold information related to one specific jurisdiction.
- Pre-existent jurisdiction has to be deployed before the master.
- Return the name formatting according to the requirements of the jurisdiction.
- Return badges(NFT Images) related to the jurisdiction.
- Has 2 different badges, the Default and the gold badge. The gold one is for the series created before migration.
- For unincorporated entities there's no change in the names, for DE entities a " LLC" suffix is added to the name of new series and for WY a " - Series ##" suffix is added to the name.
- The contract is created passing the pre-existent jurisdictions addresses.
- The contract has a function to migrate all previous entities (DE, WY and Unincorporated) as new ERC721 tokens in chronological order.
- The migration function has to be called multiple times due to high consumption of gas that surpasses the max size of the blocks. Once migration completes, a flag is set defining the last migrated entity. For all migrated entities a Gold badge is fetched when NFT info is returned.
- Allow USER to create an entity in a single transaction passing the OtoCoFee(10% for Ethereum Mainnet) of transaction fees in ETH.
- At the moment of creation a USER could create a Series on behalf of ANY address.
- Allow USER to close a LLC that he owns passing OtoCoFee(10% for Ethereum Mainnet) in ETH to transaction value.
- Retain value paid for creation/closing of series for all jurisdictions, even Unincorporated.
- Allow ANYONE to request tokenURI(badges)
- Allow ANYONE to list user owned entities based on "transfer" events + token ownership.
- The name formatting for the series happens at the moment of creation, based on Jurisdiction rules.
- Jurisdiction addresses(contracts) could be added and updated by the ADMINISTRATOR not requiring a redeployed master.
- Base fees OtoCoFee(10% for Ethereum Mainnet) could be changed by the ADMINISTRATOR.
- Fees withdrawn could be requested by the ADMINISTRATOR.
- Transfer payments to OtoCoMaster before add/attach/remove plugins.
- Plugins will always request the OtoCoMaster the current baseFees to add/attach/remove.
- Do not transfer any value in cases where baseFees are 0, or msg.sender is the OtoCoMaster(In future implementation where OtoCoMaster will create LLC along with plugins).
- Only allow USER to add/attach/remove plugins to a series that he controls.
- Plugins could have more functions but it is obligatory to MAINTAINER declare the add/attach/remove functions.
- MAINTAINER could tweak some parameters of the plugin but not the BaseFees(that is requested to OtoCoMaster) or change the OtoCoMaster reference during maintainance.
Contract with series name rules definition and the TokenURI badges related to it. Used to add new jurisdiction with their own naming rules to Master without requiring to upgrade master contract.
- getSeriesNameFormatted(uint256 count, string name): Format the series name to the requirements of the jurisdiction. Receive current series count and return name formatted.
- getJurisdictionName(): Return the name of the jurisdiction.
- getJurisdictionBadge(): Return the default badge URI for the Jurisdiction.
- getJurisdictionGoldBadge(): Return the gold badge URI for the Jurisdiction.
Responsible to receive payments and deploy new companies. Contains a list of deployed Series as NFTs and its details.
- initialize: Initialize the implementation contract and create initial jurisdictions.
- createBatchSeries: Function to Migrate previous entities.
- changeBaseFees: Set the new base fees to be paid in ETH along with the Dapp transactions.
- addJurisdiction: Add a new jurisdiction to the network.
- updateJurisdiction: Update a pre-existing jurisdiction to the network.
- withdrawFees: Withdraw the ETH stored on the contract.
- seriesCount(): Return the total number of series joining all jurisdictions.
- series(uint256 tokenId): Mapping of the series created, each of them are structs with the following properties:
(uint16)jurisdiction
,(uint16)entityType
,(uint64)creation
,(string)name
. - jurisdictionCount(): Return the number jurisdiction registered at this Master.
- jurisdictionsAddress(uint16 JurisdictionIndex): A mapping to point jurisdiction addresses that are OtocoJurisdiction contracts, with their own rules to name entities and NFT URIs.
- baseFee(): Base fees charged for company deployment/closing and plugins attach/removal. This amount is in percentage and represent a value in ETH that needs to be send along with transaction. To know exact amount to send along with transaction the calculation require is the following:
- createSeries(uint16 jurisdiction, address controller, string name): Create a new entity and set his controller. To create a new series is required to set the jurisdiction, name and the controller of the entity. GasLimit to send this transaction take into account if is the first entity created by the controller and the size of the name of the entity after jurisdiction modification requirements. It usually fluctuates between 130k and 250k gas.
- closeSeries(uint256 tokenId): Close the desired series. Cold only be called by the controller of the entity. The gas limit required to close an entity is 50k gas.
- tokenURI(uint256 tokenId): Return the JSON URI of the desired entity index. This is user to fetch ERC-721 metadata from tokens.
The OtoCo Plugins are reworked and simplified. Not require to link a Registry to fetch their informations. Basically all plugins has its own initializers.
To initialize a plugin a message should be encoded to pass parameters to addPlugin
, attachPlugin
, removePlugin
function. For each different plugin you should check the specific parameters on Plugins.
How to encode pluginData
parameters using Ethers:
let encoded = ethers.utils.defaultAbiCoder.encode(
['uint256', 'string', 'string', 'address'],
[ethers.utils.parseEther('8000000'), 'Test Token', 'TTOK', wallet2.address]
);
-
otocoMaster(): reference to the OtoCoMaster contract to transfer ETH paid for services. This function appears on all plugins, so it's possible to verify the correct OtoCo Master attachment.
-
addPlugin(uint256 tokenId, bytes pluginData) - Add the plugin to the entity selected according to the parameters selected. To call this function is required to add a ETH value to transaction referent to OtoCo Fee related to the gas spend.
-
attachPlugin(uint256 tokenId, bytes pluginData) - Attach a pre-existing plugin to the entity. To call this function is required to add a ETH value to transaction referent to OtoCo Fee related to the gas spend.
-
removePlugin(uint256 tokenId, bytes pluginData) - Add the plugin to the entity selected according to the parameters selected. To call this function is required to add a ETH value to transaction referent to OtoCo Fee related to the gas spend.
Here there are functions related to the possible functions to call and its required parameters.
-
DocumentTimestamped(uint256 indexed seriesId, uint256 timestamp, string filename, string cid) - Fetch this events to check specific entity timestamps.
-
addPlugin(uint256 tokenId, bytes pluginData) - Attach a timestamped document CID to a series.
pluginData
parameters:[string, string]
that representfileCID
andfilename
.gasLimit
recommended to use is 50k gas.
-
tokenContract() - Returns the token implementation that serves as source to clone to deploy new tokens.
-
tokensPerEntity(uint256 tokenId) - Return the current amount of tokens attached to the respective entity.
-
tokensDeployed(uint256 tokenId, uint256 index) - Address of the token index respective to a specific entity.
-
addPlugin(uint256 tokenId, bytes pluginData) - Deploy a new ERC-20 Token on behalf of the series.
pluginData
parameters:[uint256,string,string,address]
that representtotalSupply
,name
,symbol
andholder
, being holder the address that will receive the total supply of the tokens after deployment.gasLimit
recommended to use is 300k gas. -
attachPlugin(uint256 tokenId, bytes pluginData) - Attach a pre-existing token to the entity.
pluginData
parameters:[address]
the address of the token to be attached.gasLimit
recommended to use 130k. -
removePlugin(uint256 tokenId, bytes pluginData) - Remove a pre-existing token to the entity.
pluginData
parameters:[uint256]
the index of the token to be removed.gasLimit
recommended to use 70k.
-
gnosisMasterCopy() - Returns the current mastercopy used by Gnosis Proxy Factory to deploy.
-
gnosisProxyFactory() - Returns the current Gnosis Proxy Factory used to deploy new multisig wallets.
-
multisigPerEntity(uint256 tokenId) - Return the current amount of multisig wallets attached to the respective entity.
-
multisigDeployed(uint256 tokenId, uint256 index) - Address of the multisig wallet at given index respective to a specific entity.
-
addPlugin(uint256 tokenId, bytes pluginData) - Deploy a new Gnosis-Safe Multisig on behalf of the series.
pluginData
parameters:[address[],uint256,address,bytes,address,address,address,uint256,address]
that represent Setup Parameters for Multisig.gasLimit
recommended to use is 400k gas. -
attachPlugin(uint256 tokenId, bytes pluginData) - Attach a pre-existing multisig to the entity.
pluginData
parameters:[address]
the address of the multisig to be attached.gasLimit
recommended to use 130k.
-
launchpoolDeployed(uint256 tokenId) - Returns the current launchpool assigned to an entity.
-
addPlugin(uint256 tokenId, bytes pluginData) - Deploy a new OtoGo Launchpool on behalf of the series.
pluginData
parameters:[ address[], uint256[], string, address, uint16,address]
that represent Setup Parameters for Launchpool.gasLimit
recommended to use is 1000k gas.
-
domainsPerEntity(uint256 tokenId) - Return the current amount of domains attached to the respective entity.
-
seriesDomains(uint256 tokenId, uint256 index) - return the subdomain name at given index respective to a specific entity.
-
addPlugin(uint256 tokenId, bytes pluginData) - Assign a
otoco.eth
subdomain to the manager or multisig address associated to the entity.pluginData
parameters:[string, address]
that representsubdomain
andtarget
. Where subdomains is a simple string name and target should be a valid address to subdomain points to.gasLimit
recommended to use is 250k gas.
A contract that allows LLC members to transform their LLC into a Tokenized LLC with a different Agreement that allow it to have multi owners and also a MANAGER that no need to have shares.
- PARTNERS: Accounts that have tokens shares from the entity.
- MANAGER: A manager that could interact with the Allowed contracts
Governor contract where Tokenized LLC resides is based on OpenZeppelin Governor. Here is the Governor base features:
- Possibility to any PARTNER that have at least proposalThreshold(defaults to 1) tokens create a proposal. It means that any person with any amount of tokens bigger than 0 could propose.
- Proposals have a votingDelay defined in blocks to be voted (defaults to 1).
- Proposal have a votingPeriod defined in blocks to PARTNERS vote.
- Proposals need a minimum quorum to pass (defaults to 4%)
- A proposal could be executed after the votingPeriod ends, voting has enough quorum and vote FOR is bigger than votes AGAINST.
- It is possible to vote using EIP-712 signatures.
- The ERC20 token should have snapshot voting mechanism.
- It is possible to propose a change to governance settings like voting delay, voting period and quorum required.
- Votes could be casted with reason attached.
Here is a list of feature that OtoCoGovernor add:
- Set default quorum to 51%.
- Add a list of allowed contracts to interact.
- Add the MANAGER role to the governance.
- Manager could create any kind of proposal without requiring any token.
- Proposals created by the MANAGER that only deals with allowed contracts, is flagged as manager proposals.
- Proposals flagged as manager proposals not require any quorum, and if voter FOR is equals to votes AGAINST it will be considered valid. This means that nobody needs to vote, the MANAGER only need to wait for the votingPeriod finish to execute the proposal. If any partner decide to veto the proposal it could vote AGAINST. In those cases the validity of the proposal will be defined by the votes FOR and AGAINST.
- The MANAGER could be changed by a proposal.
- The allowed contracts could be changed(Added/Removed) by proposal.
- The MANAGER could resign as MANAGER without requiring a proposal.
Tokenization contract functions:
-
governorsDeployed(uint256 tokenId) - Returns the current governor assigned to an entity.
-
addPlugin(uint256 tokenId, bytes pluginData) - Deploy a new OtoCo Govenor on behalf of the series.
pluginData
parameters:[ string, string, address[], address[], uint256[]]
that representname of the token
,symbol of the token
,allowed contracts array
,addresses([0] manager address, [1] token source to be cloned, [2..n] partner addresses)
,settings ([0] partners quantity, [1] Voting period in block, [2..n] member shares)
.gasLimit
recommended to use is 1200k gas. -
attachPlugin(uint256 tokenId, bytes pluginData) - Deploy a new OtoCo Govenor using a pre-existent ERC20Votes token.
pluginData
parameters:[ address[], address[], uint256[]]
that representallowed contract array, addresses([0] Manager address,[1] Token Address), settings([0] Members size, [1] Voting period in blocks)
.gasLimit
recommended to use is 1000k gas.
In May 2022, OtoCo engaged Coinspect to perform a source code review of OtoCo Smart Contracts. The objective of the project was to evaluate the security of the smart contracts.
The documentation generated by this audit is present at audit
folder.
Ethereum Mainnet
Name | Address | Explorer |
---|---|---|
OtoCo Master | 0x752B0073422A7F9Cda7f71B5fE7F12a1789e6506 |
link |
Timestamp Plugin | 0xaA6e8D5baB906B5d1F270f95d38e92d51a7a9C8A |
link |
Token Plugin | 0x0650601Bc69e8BBf7Dc9ca0F032fB057135103e3 |
link |
Multisig Plugin | 0xD9a0927345244F2c45511CFaA5bDD00db7a8decc |
link |
Launchpool Plugin | 0x1477E9f01B78Ec20405542e5ee07469a7c93c83e |
link |
ENS Plugin | 0x355894349219dccf4825B66b2f383877C77304c9 |
link |