Skip to content
This repository has been archived by the owner on Jun 11, 2024. It is now read-only.

Unit test review - Interoperability genesis state initialization/finalization #8980

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -133,11 +133,8 @@ export abstract class BaseInteroperabilityModule extends BaseInteroperableModule
}

// activeValidators must be ordered lexicographically by blsKey property
const sortedByBlsKeys = [...activeValidators].sort((a, b) => a.blsKey.compare(b.blsKey));
for (let i = 0; i < activeValidators.length; i += 1) {
if (!activeValidators[i].blsKey.equals(sortedByBlsKeys[i].blsKey)) {
throw new Error('activeValidators must be ordered lexicographically by blsKey property.');
}
if (!objectUtils.isBufferArrayOrdered(activeValidators.map(v => v.blsKey))) {
throw new Error('activeValidators must be ordered lexicographically by blsKey property.');
}

// all blsKey properties must be pairwise distinct
Expand All @@ -147,7 +144,7 @@ export abstract class BaseInteroperabilityModule extends BaseInteroperableModule
}

// for each validator in activeValidators, validator.bftWeight > 0 must hold
if (activeValidators.filter(v => v.bftWeight <= 0).length > 0) {
if (activeValidators.filter(v => v.bftWeight <= BigInt(0)).length > 0) {
throw new Error(`validator.bftWeight must be > 0.`);
}

Expand Down Expand Up @@ -200,16 +197,15 @@ export abstract class BaseInteroperabilityModule extends BaseInteroperableModule
}

// terminatedStateAccounts is ordered lexicographically by stateAccount.chainID
const sortedByChainID = [...terminatedStateAccounts].sort((a, b) =>
a.chainID.compare(b.chainID),
);

for (let i = 0; i < terminatedStateAccounts.length; i += 1) {
const stateAccountWithChainID = terminatedStateAccounts[i];
if (!stateAccountWithChainID.chainID.equals(sortedByChainID[i].chainID)) {
throw new Error('terminatedStateAccounts must be ordered lexicographically by chainID.');
}
if (
!objectUtils.isBufferArrayOrdered(
terminatedStateAccounts.map(accountWithChainID => accountWithChainID.chainID),
)
) {
throw new Error('terminatedStateAccounts must be ordered lexicographically by chainID.');
}

for (const stateAccountWithChainID of terminatedStateAccounts) {
this._verifyChainID(stateAccountWithChainID.chainID, mainchainID, 'stateAccount.');
}
}
Expand Down
100 changes: 58 additions & 42 deletions framework/src/modules/interoperability/mainchain/module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -261,15 +261,7 @@ export class MainchainInteroperabilityModule extends BaseInteroperabilityModule
throw new Error(`ownChainName must be equal to ${CHAIN_NAME_MAINCHAIN}.`);
}

// if chainInfos is empty, then ownChainNonce == 0
// If chainInfos is non-empty, ownChainNonce > 0
if (chainInfos.length === 0 && ownChainNonce !== BigInt(0)) {
throw new Error(`ownChainNonce must be 0 if chainInfos is empty.`);
} else if (chainInfos.length !== 0 && ownChainNonce <= 0) {
throw new Error(`ownChainNonce must be positive if chainInfos is not empty.`);
}

this._verifyChainInfos(ctx, chainInfos);
this._verifyChainInfos(ctx, chainInfos, ownChainNonce);
gkoumout marked this conversation as resolved.
Show resolved Hide resolved
this._verifyTerminatedStateAccounts(chainInfos, terminatedStateAccounts, mainchainID);
this._verifyTerminatedOutboxAccounts(
chainInfos,
Expand All @@ -281,7 +273,19 @@ export class MainchainInteroperabilityModule extends BaseInteroperabilityModule
}

// https://github.com/LiskHQ/lips/blob/main/proposals/lip-0045.md#mainchain
private _verifyChainInfos(ctx: GenesisBlockExecuteContext, chainInfos: ChainInfo[]) {
private _verifyChainInfos(
ctx: GenesisBlockExecuteContext,
chainInfos: ChainInfo[],
ownChainNonce: bigint,
) {
// if chainInfos is empty, then ownChainNonce == 0
// If chainInfos is non-empty, ownChainNonce > 0
if (chainInfos.length === 0 && ownChainNonce !== BigInt(0)) {
throw new Error(`ownChainNonce must be 0 if chainInfos is empty.`);
} else if (chainInfos.length !== 0 && ownChainNonce <= 0) {
throw new Error(`ownChainNonce must be positive if chainInfos is not empty.`);
}

// Each entry chainInfo in chainInfos has a unique chainInfo.chainID
const chainIDs = chainInfos.map(info => info.chainID);
if (!objectUtils.bufferArrayUniqueItems(chainIDs)) {
Expand Down Expand Up @@ -341,9 +345,9 @@ export class MainchainInteroperabilityModule extends BaseInteroperabilityModule
mainchainID: Buffer,
) {
// Sanity check to fulfill if-and-only-if situation
for (const account of terminatedStateAccounts) {
for (const terminatedStateAccount of terminatedStateAccounts) {
const correspondingChainInfo = chainInfos.find(chainInfo =>
chainInfo.chainID.equals(account.chainID),
chainInfo.chainID.equals(terminatedStateAccount.chainID),
);
if (
!correspondingChainInfo ||
Expand All @@ -359,40 +363,52 @@ export class MainchainInteroperabilityModule extends BaseInteroperabilityModule
// For each entry chainInfo in chainInfos, chainInfo.chainData.status == CHAIN_STATUS_TERMINATED
// if and only if a corresponding entry (i.e., with chainID == chainInfo.chainID) exists in terminatedStateAccounts.
if (chainInfo.chainData.status === ChainStatus.TERMINATED) {
const terminatedAccount = terminatedStateAccounts.find(tAccount =>
tAccount.chainID.equals(chainInfo.chainID),
const correspondingTerminatedAccount = terminatedStateAccounts.find(
terminatedStateAccount => terminatedStateAccount.chainID.equals(chainInfo.chainID),
);
if (!terminatedAccount) {
if (!correspondingTerminatedAccount) {
throw new Error(
'For each chainInfo with status terminated there should be a corresponding entry in terminatedStateAccounts.',
);
}
}
}

this._verifyTerminatedStateAccountsCommon(terminatedStateAccounts, mainchainID);

// For each entry stateAccount in terminatedStateAccounts holds
// stateAccount.stateRoot == chainData.lastCertificate.stateRoot,
// stateAccount.mainchainStateRoot == EMPTY_HASH, and
// stateAccount.initialized == True.
// Here chainData is the corresponding entry (i.e., with chainID == stateAccount.chainID) in chainInfos.
const stateAccount = terminatedAccount.terminatedStateAccount;
if (stateAccount) {
if (!stateAccount.stateRoot.equals(chainInfo.chainData.lastCertificate.stateRoot)) {
throw new Error(
"stateAccount.stateRoot doesn't match chainInfo.chainData.lastCertificate.stateRoot.",
);
}

if (!stateAccount.mainchainStateRoot.equals(EMPTY_HASH)) {
throw new Error(
`stateAccount.mainchainStateRoot is not equal to ${EMPTY_HASH.toString('hex')}.`,
);
}

if (!stateAccount.initialized) {
throw new Error('stateAccount is not initialized.');
}
}
this._verifyTerminatedStateAccountsCommon(terminatedStateAccounts, mainchainID);

/**
ishantiw marked this conversation as resolved.
Show resolved Hide resolved
* For each entry stateAccount in terminatedStateAccounts holds
* stateAccount.terminatedStateAccount.mainchainStateRoot == EMPTY_HASH, and stateAccount.terminatedStateAccount.initialized == True.
*
* Moreover, let chainInfo be the corresponding entry in chainInfos (i.e., with chainInfo.chainID == stateAccount.chainID); then it holds that
* stateAccount.terminatedStateAccount.stateRoot == chainInfo.chainData.lastCertificate.stateRoot.
*/
for (const terminatedStateAccountWithChainID of terminatedStateAccounts) {
if (
!terminatedStateAccountWithChainID.terminatedStateAccount.mainchainStateRoot.equals(
EMPTY_HASH,
)
) {
throw new Error(
`stateAccount.mainchainStateRoot is not equal to ${EMPTY_HASH.toString('hex')}.`,
);
}
if (!terminatedStateAccountWithChainID.terminatedStateAccount.initialized) {
throw new Error('stateAccount is not initialized.');
}

const correspondingChainInfo = chainInfos.find(chainInfo =>
chainInfo.chainID.equals(terminatedStateAccountWithChainID.chainID),
) as ChainInfo; // at this point, it's not undefined, since similar check already applied above

if (
!terminatedStateAccountWithChainID.terminatedStateAccount.stateRoot.equals(
correspondingChainInfo.chainData.lastCertificate.stateRoot,
)
) {
throw new Error(
"stateAccount.stateRoot doesn't match chainInfo.chainData.lastCertificate.stateRoot.",
);
}
}
}
Expand Down Expand Up @@ -426,9 +442,9 @@ export class MainchainInteroperabilityModule extends BaseInteroperabilityModule
terminatedStateAccounts.find(a => a.chainID.equals(outboxAccount.chainID)) === undefined
) {
throw new Error(
`Each entry outboxAccount in terminatedOutboxAccounts must have a corresponding entry in terminatedStateAccount. outboxAccount with chainID: ${outboxAccount.chainID.toString(
`outboxAccount with chainID: ${outboxAccount.chainID.toString(
'hex',
)} does not exist in terminatedStateAccounts`,
)} must have a corresponding entry in terminatedStateAccounts.`,
);
}
}
Expand Down
20 changes: 14 additions & 6 deletions framework/src/modules/interoperability/sidechain/module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -237,8 +237,13 @@ export class SidechainInteroperabilityModule extends BaseInteroperabilityModule
ctx: GenesisBlockExecuteContext,
genesisInteroperability: GenesisInteroperability,
) {
const { ownChainName, ownChainNonce, chainInfos, terminatedStateAccounts } =
genesisInteroperability;
const {
ownChainName,
ownChainNonce,
chainInfos,
terminatedStateAccounts,
terminatedOutboxAccounts,
} = genesisInteroperability;

// If chainInfos is empty, then check that:
//
Expand All @@ -257,6 +262,9 @@ export class SidechainInteroperabilityModule extends BaseInteroperabilityModule
if (terminatedStateAccounts.length !== 0) {
throw new Error(`terminatedStateAccounts must be empty, ${ifChainInfosIsEmpty}.`);
}
if (terminatedOutboxAccounts.length !== 0) {
throw new Error(`terminatedOutboxAccounts must be empty, ${ifChainInfosIsEmpty}.`);
}
ishantiw marked this conversation as resolved.
Show resolved Hide resolved
} else {
// ownChainName
// has length between MIN_CHAIN_NAME_LENGTH and MAX_CHAIN_NAME_LENGTH,
Expand All @@ -267,7 +275,7 @@ export class SidechainInteroperabilityModule extends BaseInteroperabilityModule
ownChainName.length > MAX_CHAIN_NAME_LENGTH // will only run if not already applied in schema
) {
throw new Error(
`ownChainName.length must be between ${MIN_CHAIN_NAME_LENGTH} and ${MAX_CHAIN_NAME_LENGTH}`,
`ownChainName.length must be inclusively between ${MIN_CHAIN_NAME_LENGTH} and ${MAX_CHAIN_NAME_LENGTH}.`,
);
}
// CAUTION!
Expand All @@ -290,7 +298,7 @@ export class SidechainInteroperabilityModule extends BaseInteroperabilityModule
}
// mainchainInfo.chainID == getMainchainID();
const mainchainInfo = chainInfos[0];
const mainchainID = getMainchainID(mainchainInfo.chainID);
const mainchainID = getMainchainID(ctx.chainID);
if (!mainchainInfo.chainID.equals(mainchainID)) {
throw new Error(`mainchainInfo.chainID must be equal to ${mainchainID.toString('hex')}.`);
}
Expand Down Expand Up @@ -335,7 +343,7 @@ export class SidechainInteroperabilityModule extends BaseInteroperabilityModule
if (terminatedStateAccount.initialized) {
if (terminatedStateAccount.stateRoot.equals(EMPTY_HASH)) {
throw new Error(
`stateAccount.stateRoot mst be not equal to "${EMPTY_HASH.toString(
`stateAccount.stateRoot must not be equal to "${EMPTY_HASH.toString(
'hex',
)}", if initialized is true.`,
);
Expand All @@ -358,7 +366,7 @@ export class SidechainInteroperabilityModule extends BaseInteroperabilityModule
}
if (terminatedStateAccount.mainchainStateRoot.equals(EMPTY_HASH)) {
throw new Error(
`terminatedStateAccount.mainchainStateRoot must be not equal to "${EMPTY_HASH.toString(
`terminatedStateAccount.mainchainStateRoot must not be equal to "${EMPTY_HASH.toString(
'hex',
)}", if initialized is false.`,
);
Expand Down
Loading