Skip to content

Commit

Permalink
deployment/ccip/changeset: add active/candidate test
Browse files Browse the repository at this point in the history
  • Loading branch information
makramkd committed Jan 8, 2025
1 parent 9bcb3db commit b7fe3b6
Showing 1 changed file with 251 additions and 0 deletions.
251 changes: 251 additions & 0 deletions deployment/ccip/changeset/cs_active_candidate_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,251 @@
package changeset

import (
"math/big"
"testing"

"github.com/ethereum/go-ethereum/accounts/abi/bind"
"github.com/ethereum/go-ethereum/common"
"github.com/smartcontractkit/chainlink-testing-framework/lib/utils/testcontext"
"github.com/smartcontractkit/chainlink/deployment/ccip/changeset/internal"
commonchangeset "github.com/smartcontractkit/chainlink/deployment/common/changeset"
"golang.org/x/exp/maps"

"github.com/stretchr/testify/require"

"github.com/smartcontractkit/chainlink/v2/core/capabilities/ccip/types"
"github.com/smartcontractkit/chainlink/v2/core/gethwrappers/ccip/generated/fee_quoter"
"github.com/smartcontractkit/chainlink/v2/core/gethwrappers/ccip/generated/router"
"github.com/smartcontractkit/chainlink/v2/core/logger"
)

func Test_ActiveCandidate(t *testing.T) {
// Setup an environment with 2 chains, a source and a dest.
// We want to have the active instance execute a few messages
// and then setup a candidate instance. The candidate instance
// should not be able to transmit anything until we make it active.
tenv := NewMemoryEnvironment(t,
WithChains(2),
WithNodes(4))
state, err := LoadOnchainState(tenv.Env)
require.NoError(t, err)

// Deploy to all chains.
allChains := maps.Keys(tenv.Env.Chains)
source := allChains[0]
dest := allChains[1]

// Connect source to dest
sourceState := state.Chains[source]
tenv.Env, err = commonchangeset.ApplyChangesets(t, tenv.Env, tenv.TimelockContracts(t), []commonchangeset.ChangesetApplication{
{
Changeset: commonchangeset.WrapChangeSet(UpdateOnRampsDests),
Config: UpdateOnRampDestsConfig{
UpdatesByChain: map[uint64]map[uint64]OnRampDestinationUpdate{
source: {
dest: {
IsEnabled: true,
AllowListEnabled: false,
},
},
},
},
},
{
Changeset: commonchangeset.WrapChangeSet(UpdateFeeQuoterPricesCS),
Config: UpdateFeeQuoterPricesConfig{
PricesByChain: map[uint64]FeeQuoterPriceUpdatePerSource{
source: {
TokenPrices: map[common.Address]*big.Int{
sourceState.LinkToken.Address(): DefaultLinkPrice,
sourceState.Weth9.Address(): DefaultWethPrice,
},
GasPrices: map[uint64]*big.Int{
dest: DefaultGasPrice,
},
},
},
},
},
{
Changeset: commonchangeset.WrapChangeSet(UpdateFeeQuoterDests),
Config: UpdateFeeQuoterDestsConfig{
UpdatesByChain: map[uint64]map[uint64]fee_quoter.FeeQuoterDestChainConfig{
source: {
dest: DefaultFeeQuoterDestChainConfig(),
},
},
},
},
{
Changeset: commonchangeset.WrapChangeSet(UpdateOffRampSources),
Config: UpdateOffRampSourcesConfig{
UpdatesByChain: map[uint64]map[uint64]OffRampSourceUpdate{
dest: {
source: {
IsEnabled: true,
},
},
},
},
},
{
Changeset: commonchangeset.WrapChangeSet(UpdateRouterRamps),
Config: UpdateRouterRampsConfig{
UpdatesByChain: map[uint64]RouterUpdates{
// onRamp update on source chain
source: {
OnRampUpdates: map[uint64]bool{
dest: true,
},
},
// offramp update on dest chain
dest: {
OffRampUpdates: map[uint64]bool{
source: true,
},
},
},
},
},
})
require.NoError(t, err)

// check that source router has dest enabled
onRamp, err := sourceState.Router.GetOnRamp(&bind.CallOpts{
Context: testcontext.Get(t),
}, dest)
require.NoError(t, err)
require.NotEqual(t, common.HexToAddress("0x0"), onRamp, "expected onRamp to be set")

// Transfer ownership so that we can set new candidate configs
// and set new config digest on the offramp.
_, err = commonchangeset.ApplyChangesets(t, tenv.Env, tenv.TimelockContracts(t), []commonchangeset.ChangesetApplication{
{
Changeset: commonchangeset.WrapChangeSet(commonchangeset.TransferToMCMSWithTimelock),
Config: genTestTransferOwnershipConfig(tenv, allChains, state),
},
})
require.NoError(t, err)
assertTimelockOwnership(t, tenv, allChains, state)

sendMsg := func() {
latesthdr, err := tenv.Env.Chains[dest].Client.HeaderByNumber(testcontext.Get(t), nil)
require.NoError(t, err)
block := latesthdr.Number.Uint64()
msgSentEvent := TestSendRequest(t, tenv.Env, state, source, dest, false, router.ClientEVM2AnyMessage{
Receiver: common.LeftPadBytes(state.Chains[dest].Receiver.Address().Bytes(), 32),
Data: []byte("hello world"),
TokenAmounts: nil,
FeeToken: common.HexToAddress("0x0"),
ExtraArgs: nil,
})

var (
startBlocks = map[uint64]*uint64{
dest: &block,
}
expectedSeqNum = map[SourceDestPair]uint64{
{
SourceChainSelector: source,
DestChainSelector: dest,
}: msgSentEvent.SequenceNumber,
}
expectedSeqNumExec = map[SourceDestPair][]uint64{
{
SourceChainSelector: source,
DestChainSelector: dest,
}: {msgSentEvent.SequenceNumber},
}
)

// Confirm execution of the message
ConfirmCommitForAllWithExpectedSeqNums(t, tenv.Env, state, expectedSeqNum, startBlocks)
ConfirmExecWithSeqNrsForAll(t, tenv.Env, state, expectedSeqNumExec, startBlocks)
}

// send a message from source to dest and ensure that it gets executed
sendMsg()

var (
capReg = state.Chains[tenv.HomeChainSel].CapabilityRegistry
ccipHome = state.Chains[tenv.HomeChainSel].CCIPHome
)
donID, err := internal.DonIDForChain(capReg, ccipHome, dest)
require.NoError(t, err)
candidateDigestCommitBefore, err := ccipHome.GetCandidateDigest(&bind.CallOpts{
Context: testcontext.Get(t),
}, donID, uint8(types.PluginTypeCCIPCommit))
require.NoError(t, err)
require.Equal(t, [32]byte{}, candidateDigestCommitBefore)
candidateDigestExecBefore, err := ccipHome.GetCandidateDigest(&bind.CallOpts{
Context: testcontext.Get(t),
}, donID, uint8(types.PluginTypeCCIPExec))
require.NoError(t, err)
require.Equal(t, [32]byte{}, candidateDigestExecBefore)

// Now we can add a candidate config, send another request, and observe behavior.
// The candidate config should not be able to execute messages.
tokenConfig := NewTestTokenConfig(state.Chains[tenv.FeedChainSel].USDFeeds)
_, err = commonchangeset.ApplyChangesets(t, tenv.Env, tenv.TimelockContracts(t), []commonchangeset.ChangesetApplication{
{
Changeset: commonchangeset.WrapChangeSet(SetCandidateChangeset),
Config: SetCandidateChangesetConfig{
SetCandidateConfigBase: SetCandidateConfigBase{
HomeChainSelector: tenv.HomeChainSel,
FeedChainSelector: tenv.FeedChainSel,
// NOTE: this is technically not a new chain, but needed for validation.
OCRConfigPerRemoteChainSelector: map[uint64]CCIPOCRParams{
dest: DefaultOCRParams(
tenv.FeedChainSel,
tokenConfig.GetTokenInfo(logger.TestLogger(t), state.Chains[dest].LinkToken, state.Chains[dest].Weth9),
nil,
),
},
PluginType: types.PluginTypeCCIPCommit,
MCMS: &MCMSConfig{
MinDelay: 0,
},
},
},
},
{
Changeset: commonchangeset.WrapChangeSet(SetCandidateChangeset),
Config: SetCandidateChangesetConfig{
SetCandidateConfigBase: SetCandidateConfigBase{
HomeChainSelector: tenv.HomeChainSel,
FeedChainSelector: tenv.FeedChainSel,
// NOTE: this is technically not a new chain, but needed for validation.
OCRConfigPerRemoteChainSelector: map[uint64]CCIPOCRParams{
dest: DefaultOCRParams(
tenv.FeedChainSel,
tokenConfig.GetTokenInfo(logger.TestLogger(t), state.Chains[dest].LinkToken, state.Chains[dest].Weth9),
nil,
),
},
PluginType: types.PluginTypeCCIPExec,
MCMS: &MCMSConfig{
MinDelay: 0,
},
},
},
},
})
require.NoError(t, err)

// check that CCIPHome state is updated with the new candidate configs
// for the dest chain DON.
candidateDigestCommit, err := ccipHome.GetCandidateDigest(&bind.CallOpts{
Context: testcontext.Get(t),
}, donID, uint8(types.PluginTypeCCIPCommit))
require.NoError(t, err)
require.NotEqual(t, candidateDigestCommit, candidateDigestCommitBefore)
candidateDigestExec, err := ccipHome.GetCandidateDigest(&bind.CallOpts{
Context: testcontext.Get(t),
}, donID, uint8(types.PluginTypeCCIPExec))
require.NoError(t, err)
require.NotEqual(t, candidateDigestExec, candidateDigestExecBefore)

// send a message from source to dest and ensure that it gets executed after the candidate config is set
sendMsg()
}

0 comments on commit b7fe3b6

Please sign in to comment.