Anyone who holds staking tokens called 'CCS' can share fees that are paid for transactions. It is implemented as a custom action in CodeChain and enabled for the Tendermint consensus engine.
STAKING_CUSTOM_ACTION_ID = 2;
makeKey(...fragments) = rlp([
"ActionData",
STAKING_CUSTOM_ACTION_ID,
[...fragments]
])
-
State Key :
makeKey("StakeholderAddresses")
-
Value:
rlp(list of account)
An
account
is an Account Id which is a blake160 hash of a public key. The list consists of accounts that have non-zero balance of CCS, which is the sum of both undelegated and delegated CCS. An RLP-encoded list of those accounts is saved in the state. The list should be sorted in ascending order, and everyaccount
should be unique.
-
State Key:
makeKey("Account", account)
An
account
is anAccountId
-
Value:
rlp(quantity)
quantity
is a non-zerou64
amount of undelegated CCS that an account is holding.
-
State Key:
makeKey("Delegation", delegator)
A
delegator
is anAccountID
. -
Value:
rlp(list of [delegatee, quantity])
A
delegatee
is anAccountId
, and thequantity
is a non-zerou64
amount of CCS that adelegator
delegated to adelegatee
. The RLP-encoded non-empty list should be sorted by adelegatee
in ascending order, and everydelegatee
should be unique.
-
State Key:
makeKey("Revocations")
-
Value:
rlp(list of [delegator, delegatee, endTime, quantity])
A
delegator
is anAccountId
who has delegated CCS to adelegatee
, which is also anAccountId
.endTime
is a Unix time in UTC that specifies when the revocation will finally end.quantity
is theu64
amount of CCC that is going to be revoked. The RLP-encoded non-empty list should be sorted byendTime
in ascending order. When multiple revocations have the sameendTime
, then the revocation created earlier (a block number is smaller, a transaction index is smaller) must have a smaller index.
You can send a RLP-encoded staking action as a payload to Action::Custom
by specifying the handler_id
as a STAKING_CUSTOM_ACTION_ID
Action::Custom {
handler_id: STAKING_CUSTOM_ACTION_ID,
bytes: rlp(action)
}
-
Format:
[ 1, receiver, quantity ]
- A
receiver
is anAccountId
. quantity
is au64
amount of CCS to transfer to areceiver
.
A
receiver
will be inserted to the list of CCS holders, and its amount of undelegated CCS will be increased byquantity
. The transaction sender's amount of undelegated CCS will be decreased byquantity
. A sender cannot transfer more than the amount of undelegated CCS it has. - A
state = {
stakeholders: [ "0xAB..CD" ],
balance: {
"0xAB..CD": 1000,
}
}
> "0xAB..CD" sends staking action [ 1, "0x01..23", 300 ]
state = {
stakeholders: [ "0x01..23", "0xAB..CD" ],
balance: {
"0x01..23": 300,
"0xAB..CD": 700,
}
}
> "0xAB..CD" sends staking action [ 1, "0x01..23", 100 ]
state = {
stakeholders: [ "0x01..23", "0xAB..CD" ],
balance: {
"0x01..23": 400,
"0xAB..CD": 600,
}
}
-
Format:
[ 2, delegatee, quantity ]
- A
delegatee
is anAccountId
. quantity
is au64
amount of CCS to delegate todelegatee
.
A
delegatee
must be one of the current candidates. The amount of undelegated CCS of the sender will be decreased byquantity
. However, instead of increasing the amount of undelegated CCS of thedelegatee
, [delegatee
,quantity
] will be inserted to the transaction sender's delegation list. A sender cannot delegate more than the amount of undelegated CCS it has. - A
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0x23..45": 500,
"0xAB..CD": 1000,
}
}
> "0xAB..CD" sends staking action [ 2, "0x23..45", 100 ]
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0x23..45": 500,
"0xAB..CD": 900,
},
delegation: {
"0xAB..CD": [ ["0x23..45", 100] ]
}
}
> "0xAB..CD" sends staking action [ 2, "0x23..45", 10 ]
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0x23..45": 500,
"0xAB..CD": 890,
},
delegation: {
"0xAB..CD": [ ["0x23..45", 110] ]
}
}
> "0xAB..CD" sends staking action [ 2, "0x01..23", 200 ]
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0x23..45": 500,
"0xAB..CD": 690,
},
delegation: {
"0xAB..CD": [ ["0x01..23", 200], ["0x23..45", 110] ]
}
}
> "0x23..45" sends staking action [ 2, "0x01..23", 500 ]
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0xAB..CD": 690,
},
delegation: {
"0x23..45": [ ["0x01..23", 500] ],
"0xAB..CD": [ ["0x01..23", 200], ["0x23..45", 110] ]
}
}
-
Format:
[ 3, delegatee, quantity ]
- A
delegatee
is anAccountId
. quantity
is a non-zerou64
amount of CCS to revoke fromdelegatee
.
This action will queue a pending revocation rather than revoke immediately. A pending revocation is
[delegator, delegatee, endTime, quantity]
, whereendTime
is set as thetimestamp of a block + REVOKE_PENDING_DURATION
. Adelegator
cannotRequestRevoke
more than the amount of delegated CCS to adelegatee
minus the sum of the pending revocations between them. - A
REVOKE_PENDING_DURATION
is 500 in this example.
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0xAB..CD": 690,
},
delegation: {
"0x23..45": [ ["0x01..23", 500] ],
"0xAB..CD": [ ["0x01..23", 200], ["0x23..45", 110] ]
}
}
> "0xAB..CD" sends staking action [3, "0x01..23", 100] at block timestamp 123400
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0xAB..CD": 690,
},
delegation: {
"0x23..45": [ ["0x01..23", 500] ],
"0xAB..CD": [ ["0x01..23", 200], ["0x23..45", 110] ]
},
revocations: [ ["0xAB..CD", "0x01..23", 123900, 100] ]
}
> "0xAB..CD" sends staking action [3, "0x01..23", 50"] at block timestamp 123500
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0xAB..CD": 690,
},
delegation: {
"0x23..45": [ ["0x01..23", 500] ],
"0xAB..CD": [ ["0x01..23", 200], ["0x23..45", 110] ]
},
revocations: [
["0xAB..CD", "0x01..23", 123900, 100],
["0xAB..CD", "0x01..23", 124000, 50],
]
}
> "0x23..45" sends staking action [3, "0x01..23", 500"] at block timestamp 123600
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0xAB..CD": 690,
},
delegation: {
"0x23..45": [ ["0x01..23", 500] ],
"0xAB..CD": [ ["0x01..23", 200], ["0x23..45", 110] ]
},
revocations: [
["0xAB..CD", "0x01..23", 123900, 100],
["0xAB..CD", "0x01..23", 124000, 50],
["0x23..45", "0x01..23", 124100, 500],
]
}
> after a block with timestamp 123900
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0xAB..CD": 790,
},
delegation: {
"0x23..45": [ ["0x01..23", 500] ],
"0xAB..CD": [ ["0x01..23", 100], ["0x23..45", 110] ]
},
revocations: [
["0xAB..CD", "0x01..23", 124000, 50],
["0x23..45", "0x01..23", 124100, 500],
]
}
> after a block with timestamp 124000
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0xAB..CD": 840,
},
delegation: {
"0x23..45": [ ["0x01..23", 500] ],
"0xAB..CD": [ ["0x01..23", 50], ["0x23..45", 110] ]
},
revocations: [
["0x23..45", "0x01..23", 124100, 500],
]
}
> after a block with timestamp 124100
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0x23..45": 500,
"0xAB..CD": 840,
},
delegation: {
"0xAB..CD": [ ["0x01..23", 50], ["0x23..45", 110] ]
}
}
- Format:
[6, prev_delegatee, next_delegatee, quantity]
- A
prev_delegatee
is anAccountId
. - A
next_delegatee
is anAccountId
. quantity
is au64
amount of CCS to redelegate tonext_delegatee
fromprev_delegatee
.
- A
Executing this action is the same as executing the revoke action and the delegate action. The quantity
should be less than the delegated quantity of prev_delegatee
from the sender.
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0x23..45": 500,
"0xAB..CD": 900,
},
delegation: {
"0xAB..CD": [ ["0x23..45", 100] ]
}
}
> "0xAB..CD" sends staking action [ 6, "0x23..45", "0x67..89", 10 ]
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0x23..45": 500,
"0xAB..CD": 900,
},
delegation: {
"0xAB..CD": [ ["0x23..45", 90], ["0x67..89", 10] ]
}
}
You pay fees to make a transaction. Fees should be greater than the specified minimum fee for the transaction type. Transaction fees that are within a block are collected, and CCS holders and the block author share this.
CCS Holders share the sum of the minimum fees for the transaction. They get fees in proportion to the CCS balance they have, which is the sum of the amount of undelegated CCS plus the total amount of CCS that are delegated to someone.
Fees distributed to CCS holders will be rounded down to an integer. The block author gets the rest, which is the sum of the remaining amount due to rounding and the amount that exceeds the minimum fee.
state = {
stakeholders: [ "0x23..45", "0xAB..CD" ],
balance: {
"0x23..45": 450,
"0xAB..CD": 690,
},
delegation: {
"0x23..45": [ ["0x01..23", 50] ],
"0xAB..CD": [ ["0x01..23", 200], ["0x23..45", 110] ]
},
revocations: [
["0xAB..CD", "0x01..23", 124000, 50],
}
}
In a such state, an account "0x23..45" has a balance of 500 (undelegated 450 + delegated 50), and "0xAB..CD" has a balance of 1000 (undelegated 690 + delegated 200 + delegated 110, pending revocation will not affect it). If a block has total fees of 110, and the sum of the minimum fees for it is 35, each account will share the following amount of fees:
share of "0x23..45" = Math.floor(35 * ( 500 / (500 + 1000))) = 11
share of "0xAB..CD" = Math.floor(35 * (1000 / (500 + 1000))) = 23
remaing share (of author) = 110 - (11 + 23) = 76
This transaction will change the common parameters when more than half of the stakeholders agree. It does not change other fields of the scheme file because there are fields related to the genesis block.
It also does not provide a voting feature. The vote initiator should collect the signatures through the off-chain.
This transaction increases the seq
of Metadata
and changes the params
of Metadata
.
The changed parameters are applied from the next block that the changing transaction is included.
The new parameters are used from the next block.
[ 0xFF, metadata_seq, new_parameters, ...signatures ]
The transaction fails if the metadata_seq is different from the seq
of Metadata
and is introduced to prevent replay attacks.
new_parameters := [ (value,)* ]
value := usize | u64 | boolean | string
It is the list of the values that the transaction changes. The stakeholder MUST NOT sign the transaction when the type of value is not a type that the key expected.
signatures
are the ECDSA signatures of stakeholders.
The stakeholders should send the signature of blake256(rlp_encode([ 0xFF, metadata_seq, new_parameters ]))
to the vote initiator if they agree to the change.
The transaction is valid only if more than half of the stakeholders agree.
RequestRevoke
will queue a pending revocation instead of revoking a delegation immediately.
The revocation will be delayed by a certain amount of time (REVOKE_PENDING_DURATION
) to prevent abuse.
It will be processed when a block whose timestamp is greater than or equal to the time when endTime
is created.
The amount of undelegated CCS of the delegator
will be increased by quantity
upon revocation, and the amount of delegated CCS that the delegator
has delegated to the delegatee
will be decreased by quantity
, and the pending revocation will be removed from the revocation queue.
Queues and delegations that become empty should be removed from the state.
It is 3 weeks in mainnet (1814400 seconds)
See Example of RequestRevoke in Staking Actions