-
Notifications
You must be signed in to change notification settings - Fork 472
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'master' into update-bold-dec-16
- Loading branch information
Showing
11 changed files
with
525 additions
and
66 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Submodule contracts
updated
33 files
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Submodule go-ethereum
updated
5 files
+1 −1 | core/state_transition.go | |
+41 −2 | core/types/transaction.go | |
+49 −0 | core/types/transaction_arbitrum_test.go | |
+1 −0 | ethdb/pebble/extraoptions.go | |
+1 −1 | ethdb/pebble/pebble.go |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,271 @@ | ||
// Copyright 2021-2024, Offchain Labs, Inc. | ||
// For license information, see https://github.com/OffchainLabs/nitro/blob/master/LICENSE | ||
|
||
package arbtest | ||
|
||
import ( | ||
"context" | ||
"math/big" | ||
"strings" | ||
"testing" | ||
"time" | ||
|
||
"github.com/ethereum/go-ethereum/accounts/abi/bind" | ||
"github.com/ethereum/go-ethereum/common" | ||
"github.com/ethereum/go-ethereum/core/types" | ||
|
||
"github.com/offchainlabs/nitro/arbnode" | ||
"github.com/offchainlabs/nitro/arbos/arbosState" | ||
"github.com/offchainlabs/nitro/execution/gethexec" | ||
"github.com/offchainlabs/nitro/solgen/go/mocksgen" | ||
"github.com/offchainlabs/nitro/solgen/go/precompilesgen" | ||
) | ||
|
||
func TestScheduleArbosUpgrade(t *testing.T) { | ||
ctx, cancel := context.WithCancel(context.Background()) | ||
defer cancel() | ||
|
||
builder := NewNodeBuilder(ctx).DefaultConfig(t, false) | ||
cleanup := builder.Build(t) | ||
defer cleanup() | ||
|
||
auth := builder.L2Info.GetDefaultTransactOpts("Owner", ctx) | ||
|
||
arbOwnerPublic, err := precompilesgen.NewArbOwnerPublic(common.HexToAddress("0x6b"), builder.L2.Client) | ||
Require(t, err, "could not bind ArbOwner contract") | ||
|
||
arbOwner, err := precompilesgen.NewArbOwner(common.HexToAddress("0x70"), builder.L2.Client) | ||
Require(t, err, "could not bind ArbOwner contract") | ||
|
||
callOpts := &bind.CallOpts{Context: ctx} | ||
scheduled, err := arbOwnerPublic.GetScheduledUpgrade(callOpts) | ||
Require(t, err, "failed to call GetScheduledUpgrade before scheduling upgrade") | ||
if scheduled.ArbosVersion != 0 || scheduled.ScheduledForTimestamp != 0 { | ||
t.Errorf("expected no upgrade to be scheduled, got version %v timestamp %v", scheduled.ArbosVersion, scheduled.ScheduledForTimestamp) | ||
} | ||
|
||
// Schedule a noop upgrade, which should test GetScheduledUpgrade in the same way an already completed upgrade would. | ||
tx, err := arbOwner.ScheduleArbOSUpgrade(&auth, 1, 1) | ||
Require(t, err) | ||
_, err = builder.L2.EnsureTxSucceeded(tx) | ||
Require(t, err) | ||
|
||
scheduled, err = arbOwnerPublic.GetScheduledUpgrade(callOpts) | ||
Require(t, err, "failed to call GetScheduledUpgrade after scheduling noop upgrade") | ||
if scheduled.ArbosVersion != 0 || scheduled.ScheduledForTimestamp != 0 { | ||
t.Errorf("expected completed scheduled upgrade to be ignored, got version %v timestamp %v", scheduled.ArbosVersion, scheduled.ScheduledForTimestamp) | ||
} | ||
|
||
// We can't test 11 -> 20 because 11 doesn't have the GetScheduledUpgrade method we want to test | ||
var testVersion uint64 = 100 | ||
var testTimestamp uint64 = 1 << 62 | ||
tx, err = arbOwner.ScheduleArbOSUpgrade(&auth, 100, 1<<62) | ||
Require(t, err) | ||
_, err = builder.L2.EnsureTxSucceeded(tx) | ||
Require(t, err) | ||
|
||
scheduled, err = arbOwnerPublic.GetScheduledUpgrade(callOpts) | ||
Require(t, err, "failed to call GetScheduledUpgrade after scheduling upgrade") | ||
if scheduled.ArbosVersion != testVersion || scheduled.ScheduledForTimestamp != testTimestamp { | ||
t.Errorf("expected upgrade to be scheduled for version %v timestamp %v, got version %v timestamp %v", testVersion, testTimestamp, scheduled.ArbosVersion, scheduled.ScheduledForTimestamp) | ||
} | ||
} | ||
|
||
func checkArbOSVersion(t *testing.T, testClient *TestClient, expectedVersion uint64, scenario string) { | ||
statedb, err := testClient.ExecNode.Backend.ArbInterface().BlockChain().State() | ||
Require(t, err, "could not get statedb", scenario) | ||
state, err := arbosState.OpenSystemArbosState(statedb, nil, true) | ||
Require(t, err, "could not open ArbOS state", scenario) | ||
if state.ArbOSVersion() != expectedVersion { | ||
t.Errorf("%s: expected ArbOS version %v, got %v", scenario, expectedVersion, state.ArbOSVersion()) | ||
} | ||
|
||
} | ||
|
||
func TestArbos11To32UpgradeWithMcopy(t *testing.T) { | ||
t.Parallel() | ||
|
||
ctx, cancel := context.WithCancel(context.Background()) | ||
defer cancel() | ||
|
||
initialVersion := uint64(11) | ||
finalVersion := uint64(32) | ||
|
||
builder := NewNodeBuilder(ctx). | ||
DefaultConfig(t, true). | ||
WithArbOSVersion(initialVersion) | ||
cleanup := builder.Build(t) | ||
defer cleanup() | ||
seqTestClient := builder.L2 | ||
|
||
auth := builder.L2Info.GetDefaultTransactOpts("Owner", ctx) | ||
auth.GasLimit = 32000000 | ||
|
||
// makes Owner a chain owner | ||
arbDebug, err := precompilesgen.NewArbDebug(types.ArbDebugAddress, seqTestClient.Client) | ||
Require(t, err) | ||
tx, err := arbDebug.BecomeChainOwner(&auth) | ||
Require(t, err) | ||
_, err = EnsureTxSucceeded(ctx, seqTestClient.Client, tx) | ||
Require(t, err) | ||
|
||
// deploys test contract | ||
_, tx, contract, err := mocksgen.DeployArbOS11To32UpgradeTest(&auth, seqTestClient.Client) | ||
Require(t, err) | ||
_, err = EnsureTxSucceeded(ctx, seqTestClient.Client, tx) | ||
Require(t, err) | ||
|
||
// build replica node | ||
replicaConfig := arbnode.ConfigDefaultL1Test() | ||
replicaConfig.BatchPoster.Enable = false | ||
replicaTestClient, replicaCleanup := builder.Build2ndNode(t, &SecondNodeParams{nodeConfig: replicaConfig}) | ||
defer replicaCleanup() | ||
|
||
checkArbOSVersion(t, seqTestClient, initialVersion, "initial sequencer") | ||
checkArbOSVersion(t, replicaTestClient, initialVersion, "initial replica") | ||
|
||
// mcopy should fail since arbos 11 doesn't support it | ||
tx, err = contract.Mcopy(&auth) | ||
Require(t, err) | ||
_, err = seqTestClient.EnsureTxSucceeded(tx) | ||
if (err == nil) || !strings.Contains(err.Error(), "invalid opcode: MCOPY") { | ||
t.Errorf("expected MCOPY to fail, got %v", err) | ||
} | ||
_, err = WaitForTx(ctx, replicaTestClient.Client, tx.Hash(), time.Second*15) | ||
Require(t, err) | ||
|
||
// upgrade arbos to final version | ||
arbOwner, err := precompilesgen.NewArbOwner(types.ArbOwnerAddress, seqTestClient.Client) | ||
Require(t, err) | ||
tx, err = arbOwner.ScheduleArbOSUpgrade(&auth, finalVersion, 0) | ||
Require(t, err) | ||
_, err = seqTestClient.EnsureTxSucceeded(tx) | ||
Require(t, err) | ||
_, err = WaitForTx(ctx, replicaTestClient.Client, tx.Hash(), time.Second*15) | ||
Require(t, err) | ||
|
||
// checks upgrade worked | ||
tx, err = contract.Mcopy(&auth) | ||
Require(t, err) | ||
_, err = seqTestClient.EnsureTxSucceeded(tx) | ||
Require(t, err) | ||
_, err = WaitForTx(ctx, replicaTestClient.Client, tx.Hash(), time.Second*15) | ||
Require(t, err) | ||
|
||
checkArbOSVersion(t, seqTestClient, finalVersion, "final sequencer") | ||
checkArbOSVersion(t, replicaTestClient, finalVersion, "final replica") | ||
|
||
// generates more blocks | ||
builder.L2Info.GenerateAccount("User2") | ||
for i := 0; i < 3; i++ { | ||
tx = builder.L2Info.PrepareTx("Owner", "User2", builder.L2Info.TransferGas, big.NewInt(1e12), nil) | ||
err = seqTestClient.Client.SendTransaction(ctx, tx) | ||
Require(t, err) | ||
_, err = seqTestClient.EnsureTxSucceeded(tx) | ||
Require(t, err) | ||
_, err = WaitForTx(ctx, replicaTestClient.Client, tx.Hash(), time.Second*15) | ||
Require(t, err) | ||
} | ||
|
||
blockNumberSeq, err := seqTestClient.Client.BlockNumber(ctx) | ||
Require(t, err) | ||
blockNumberReplica, err := replicaTestClient.Client.BlockNumber(ctx) | ||
Require(t, err) | ||
if blockNumberSeq != blockNumberReplica { | ||
t.Errorf("expected sequencer and replica to have same block number, got %v and %v", blockNumberSeq, blockNumberReplica) | ||
} | ||
// #nosec G115 | ||
blockNumber := big.NewInt(int64(blockNumberSeq)) | ||
|
||
blockSeq, err := seqTestClient.Client.BlockByNumber(ctx, blockNumber) | ||
Require(t, err) | ||
blockReplica, err := replicaTestClient.Client.BlockByNumber(ctx, blockNumber) | ||
Require(t, err) | ||
if blockSeq.Hash() != blockReplica.Hash() { | ||
t.Errorf("expected sequencer and replica to have same block hash, got %v and %v", blockSeq.Hash(), blockReplica.Hash()) | ||
} | ||
} | ||
|
||
func TestArbos11To32UpgradeWithCalldata(t *testing.T) { | ||
t.Parallel() | ||
|
||
ctx, cancel := context.WithCancel(context.Background()) | ||
defer cancel() | ||
|
||
initialVersion := uint64(11) | ||
finalVersion := uint64(32) | ||
|
||
builder := NewNodeBuilder(ctx). | ||
DefaultConfig(t, true). | ||
WithArbOSVersion(initialVersion) | ||
builder.execConfig.TxPreChecker.Strictness = gethexec.TxPreCheckerStrictnessLikelyCompatible | ||
cleanup := builder.Build(t) | ||
defer cleanup() | ||
seqTestClient := builder.L2 | ||
|
||
auth := builder.L2Info.GetDefaultTransactOpts("Owner", ctx) | ||
auth.GasLimit = 32000000 | ||
|
||
// makes Owner a chain owner | ||
arbDebug, err := precompilesgen.NewArbDebug(types.ArbDebugAddress, seqTestClient.Client) | ||
Require(t, err) | ||
tx, err := arbDebug.BecomeChainOwner(&auth) | ||
Require(t, err) | ||
_, err = EnsureTxSucceeded(ctx, seqTestClient.Client, tx) | ||
Require(t, err) | ||
|
||
// build replica node | ||
replicaConfig := arbnode.ConfigDefaultL1Test() | ||
replicaConfig.BatchPoster.Enable = false | ||
replicaTestClient, replicaCleanup := builder.Build2ndNode(t, &SecondNodeParams{nodeConfig: replicaConfig}) | ||
defer replicaCleanup() | ||
|
||
checkArbOSVersion(t, seqTestClient, initialVersion, "initial sequencer") | ||
checkArbOSVersion(t, replicaTestClient, initialVersion, "initial replica") | ||
|
||
// upgrade arbos to final version | ||
arbOwner, err := precompilesgen.NewArbOwner(types.ArbOwnerAddress, seqTestClient.Client) | ||
Require(t, err) | ||
tx, err = arbOwner.ScheduleArbOSUpgrade(&auth, finalVersion, 0) | ||
Require(t, err) | ||
_, err = seqTestClient.EnsureTxSucceeded(tx) | ||
Require(t, err) | ||
_, err = WaitForTx(ctx, replicaTestClient.Client, tx.Hash(), time.Second*15) | ||
Require(t, err) | ||
|
||
// checks upgrade worked | ||
var data []byte | ||
for i := range 10 { | ||
for range 100 { | ||
data = append(data, byte(i)) | ||
} | ||
} | ||
tx = builder.L2Info.PrepareTx("Owner", "Owner", builder.L2Info.TransferGas, big.NewInt(1e12), data) | ||
err = seqTestClient.Client.SendTransaction(ctx, tx) | ||
Require(t, err) | ||
_, err = seqTestClient.EnsureTxSucceeded(tx) | ||
Require(t, err) | ||
_, err = WaitForTx(ctx, replicaTestClient.Client, tx.Hash(), time.Second*15) | ||
Require(t, err) | ||
|
||
checkArbOSVersion(t, seqTestClient, finalVersion, "final sequencer") | ||
checkArbOSVersion(t, replicaTestClient, finalVersion, "final replica") | ||
|
||
blockNumberSeq, err := seqTestClient.Client.BlockNumber(ctx) | ||
Require(t, err) | ||
blockNumberReplica, err := replicaTestClient.Client.BlockNumber(ctx) | ||
Require(t, err) | ||
if blockNumberSeq != blockNumberReplica { | ||
t.Errorf("expected sequencer and replica to have same block number, got %v and %v", blockNumberSeq, blockNumberReplica) | ||
} | ||
// #nosec G115 | ||
blockNumber := big.NewInt(int64(blockNumberSeq)) | ||
|
||
blockSeq, err := seqTestClient.Client.BlockByNumber(ctx, blockNumber) | ||
Require(t, err) | ||
blockReplica, err := replicaTestClient.Client.BlockByNumber(ctx, blockNumber) | ||
Require(t, err) | ||
if blockSeq.Hash() != blockReplica.Hash() { | ||
t.Errorf("expected sequencer and replica to have same block hash, got %v and %v", blockSeq.Hash(), blockReplica.Hash()) | ||
} | ||
} |
Oops, something went wrong.