Skip to content

lukso-network/rICO-smart-contracts

Repository files navigation

Reversible ICO smart contracts

The first ever Reversible ICO smart contract. Reversible ICO is a way to collect funding over time. While conventional ICOs collect ETH and return a token. Collecting a lot of ETH within a short amount of time can lead to scams or other issues with the project team.

The rICO tries to solve that by keeping investors in control of their funds over a defined period of time, while buying tokens gradually.

Basic ways of interaction for participants

  • Committing ETH: call commit() (data: 0x3c7a3aff) and send ETH along to the rICO.
  • Cancel pending ETH (Before participant is whitelisted): Send less than minContribution (e.g. 0.1) to the rICO, or call cancel() (data: 0xea8a1af0)
  • Withdrawing ETH (Before participant is whitelisted): Send tokens back to the rICO.

Reversible ICO Functionality

The rICO is set in stages that can be defined in the init function.

Stage 0 is seen as the "commit phase": In this period no ETH buys tokens, and no ETH is allocated to the project, investors can commit or reverse their commitment by sending back the tokens.

Stage 1-x is called "buy phase", this is where the committed ETH gradually buys the token over time. ETH that already bought tokens is available to the project for withdraw.

Tokens that are still returnable (meaning they haven't been unlocked for the participant), can be returned to the rICO smart contract, to withdraw their corresponding ETH by any investor at any point in time.

Each stage has a price increase, so that committing early is rewarded. The scheme looks a little bit as follows:

alt text

How do I reserve tokens?

To reserve tokens at any stage, call the commit() function (ABI: "0x3C7A3AFF") and send the amount of ETH along for which you want to reserve tokens with.

What happens if somebody commits during stage 1-x

Committing during stage 1-x means that the rICO period of buying of tokens over time is shorter for yourself, as the period starts for everyone at the point in time of their commitment.

E.g. If you come in at month 3 of 10, your rICO will take place over 7 months and end with everyone else.

Why do I see the full balance of tokens, if I have not bought them yet?

This is done so that normal token wallets can function as an interface to the rICO. You will see the full token balance at your address, but will only be able to move the amount you have actually bought by the current point in time.

While in reverse you will only be able to send back the amount of tokens that are returnable. Should you send a higher than the returnable token balance you will only return the amount of ETH matching your total returnable token balance, and get the rest tokens returned to your address automatically.

What is the pending state?

When you commit ETH your address needs to be whitelisted first. This is done by the projects whitelisting address by calling whitelist(address[], bool) on the rICO smart contract. Should your address be rejected (the project calling whitelist([0xyouraddress], false)), you will receive your committed ETH back. Should it be approved you will see your full token balance at your committing address.

What if I want to withdraw ETH while my address is still pending?

This can be done by sending an ETH transaction with 0 value, or smaller than minContribution (default 0.001 ether) to the rICO smart contract address. Or call the cancel() function directly. This will trigger a cancel of all your pending ETH.

Security Measurements

The rICO as well as the token contract was audited by (ConsenSys Diligence)[https://github.com/ConsenSys] in April 2020. No severe issues were found. You can find the audit report here: diligence.consensys.net/audits/2020/04/rico/

Escape hatch

The current version includes a security escape hatch in case of malfunction. The security is split between two entities:

  • The freezerAddress, who can freeze the rICO as well as the token contract, to prevent any interaction with it. (The rICO freezer can be different from the token freezer)
  • The rescuerAddress, who should be held by an escrow company, only to be used in the severe case of emergency. (The rICO rescuer can be different from the token rescuer) It can when frozen move the tokens of the rICO and ETH funds to another address (e.g. return contract, or updated rICO contract). It can also point the token contract to another rICO contract, but only when the token contract is frozen as well.

The freezerAddress has the ability to to deactivate the freeze function in both token, and rICO contract, should the contracts deemed fully secure. This will deactivate the ability to rescue funds or freeze the contracts all together.

Token functionality

The token contract is a basic ERC777 with some modifications.

For the balance the the contract checks with the rICOs getParticipantReservedTokens(address) function to see how much tokens are still returnable, and prevents them from being movable.

Migration address

The project can set a migration address at any point in time, which is - besides the rICO address, the only address where locked (returnable) tokens can also be sent to.

This is there so that when the project migrates the token to e.g. a native token on a Blockchain, it can provide a smart contract that handles such an migration. This can happen even before the rICO is finished.

Security Measurements

The token has a similar security scheme like the rICO contract (seperate freezerAddress and rescuerAddress), but when frozen only the rICO address can be set to a new one. This is there to be able to upgrade to a different rICO in case the old one needs to be replaced.

It has the same

Development

Building Requirements

$ sudo apt-get update
$ sudo apt-get install nodejs
$ sudo apt-get install npm

Recommended

$ nvm install v8.17.0
$ nvm use v8.17.0
$ npm install
$ npm run test

Running Tests

$ npm test

// or to run the solc tests
$ npm run test-solc

// to only run the random tests
$ npm run test-random-standalone

Merging contracts for deployment

$ npm run merge-contracts

Outputs the merged contracts in ./contracts-merged

Deployment

You can edit the rICO settings in deployment/rICO-config-deployment.js before deploying.

To test deploy on a local simulated development network. The addresses will not be the one from the config file, but local generate ones.

$ npm run deploy-test

To deploy live, using the addresses from the config file:

$ npm run deploy-live

Gas costs

  • Contribution before whitelisting: ~60-130k GAS (first contribution is more expensive ~130k GAS)
  • Contribution after whitelisting: ~190k GAS
  • User Withdraw: ~160K GAS
  • Project Withdraw: ~70k GAS
  • Whitelisting: ~200k-650k GAS (with max 9 stages) (depends in how much stages were contributions)

NOTE Its not recommended to choose more than 50 stages! 9 stages require ~650k GAS when whitelisting contributions, the whitelisting function could run out of gas with a high number of stages, preventing accepting contributions.

Test before using the npm run test-random-standalone and adjust /test/solc_tests/0_standalone_random_tests.js

Main Contributors

LICENSE

Apache 2.0

Past Test Deployment on Rinkeby

// RINKEBY deployment
// rICO: 0x706ce3A2D48F73A879CCC5aB648d3f578BDF34F8
// Token: 0xA0610e2BE6A66f7AbA8E979a83C46ae9941D7F81
// ----------------------------------------------------------------
//     Step 3 - Deploy contracts
// ----------------------------------------------------------------
//     - Contract deployed: ReversibleICOToken
// Hash:             0x18f996b63215b92c376aa6074b42b8733436a2ca1bc4ce2d23cf8481579a49c5
// Gas used:         2542617
// Contract Address: 0xA0610e2BE6A66f7AbA8E979a83C46ae9941D7F81
// - Contract deployed: ReversibleICO
// Hash:             0xaae3a662767e7dac5674430025fdd9e759866514cea210d0a1a1a1b801f0797f
// Gas used:         3252326
// Contract Address: 0x706ce3A2D48F73A879CCC5aB648d3f578BDF34F8
// ----------------------------------------------------------------
//     Step 4 - Initialise Token Contract
// ----------------------------------------------------------------
//     - Settings:
// - ReversibleICOAddress: 0x706ce3A2D48F73A879CCC5aB648d3f578BDF34F8
// - freezerAddress:       0x4b302ceFF48E2212a761086667195149b3861230
// - rescuerAddress:       0x29D869F89f3DC193C17EeE8D2656e69A257E01E9
// - projectAddress:       0x01d934D2D2D8a4AF45532B330e67610E7697eca6
// - initialSupply:        100000000 tokens
// - Caller: 0x52b333c238Bf73888fDDe266E9D2A39B75752807
// - Hash: 0x536bed6ab05f5230bb423d4a536651cf49e747714b350f307b5af1541e3f6cc1
// - Gas used: 141426
// - Done
// ----------------------------------------------------------------
//     Step 5 - Initialise rICO Contract
// ----------------------------------------------------------------
//     - Settings:
// - TokenContractAddress:     0xA0610e2BE6A66f7AbA8E979a83C46ae9941D7F81
// - whitelistingAddress       0xc9e82F820F69C67263c0abd220f31bD0Cf73D28E
// - freezerAddress:           0x4b302ceFF48E2212a761086667195149b3861230
// - rescuerAddress:           0x29D869F89f3DC193C17EeE8D2656e69A257E01E9
// - projectAddress:           0x01d934D2D2D8a4AF45532B330e67610E7697eca6
// - commitPhaseStartBlock:    6445769
// - commitPhaseBlockCount:    23040
// - commitPhasePrice:         1000000000000000 wei
// - stagePriceIncrease:       500000000000000 wei
// - stageCount:               8
// - stageBlockCount:          5760
// - Caller: 0x52b333c238Bf73888fDDe266E9D2A39B75752807
// - Hash: 0x70c428aad19e1aa9f311208e617cf9a42220a3cc4fa24deb9395bf25b50fba33
// - Gas used: 704054