Skip to content

Commit 21859d6

Browse files
committed
Backwards-compatible WithdrawalNetwork Go type.
1 parent 90111ed commit 21859d6

7 files changed

+220
-28
lines changed

op-chain-ops/genesis/config.go

+10-10
Original file line numberDiff line numberDiff line change
@@ -99,7 +99,7 @@ type DeployConfig struct {
9999
// L2GenesisRegolithTimeOffset is the number of seconds after genesis block that Regolith hard fork activates.
100100
// Set it to 0 to activate at genesis. Nil to disable regolith.
101101
L2GenesisRegolithTimeOffset *hexutil.Uint64 `json:"l2GenesisRegolithTimeOffset,omitempty"`
102-
// Configurable extradata. Will default to []byte("BEDROCK") if left unspecified.
102+
// L2GenesisBlockExtraData is configurable extradata. Will default to []byte("BEDROCK") if left unspecified.
103103
L2GenesisBlockExtraData []byte `json:"l2GenesisBlockExtraData"`
104104
// ProxyAdminOwner represents the owner of the ProxyAdmin predeploy on L2.
105105
ProxyAdminOwner common.Address `json:"proxyAdminOwner"`
@@ -124,11 +124,11 @@ type DeployConfig struct {
124124
// SequencerFeeVaultMinimumWithdrawalAmount represents the minimum withdrawal amount for the SequencerFeeVault.
125125
SequencerFeeVaultMinimumWithdrawalAmount *hexutil.Big `json:"sequencerFeeVaultMinimumWithdrawalAmount"`
126126
// BaseFeeVaultWithdrawalNetwork represents the withdrawal network for the BaseFeeVault.
127-
BaseFeeVaultWithdrawalNetwork uint8 `json:"baseFeeVaultWithdrawalNetwork"`
127+
BaseFeeVaultWithdrawalNetwork WithdrawalNetwork `json:"baseFeeVaultWithdrawalNetwork"`
128128
// L1FeeVaultWithdrawalNetwork represents the withdrawal network for the L1FeeVault.
129-
L1FeeVaultWithdrawalNetwork uint8 `json:"l1FeeVaultWithdrawalNetwork"`
129+
L1FeeVaultWithdrawalNetwork WithdrawalNetwork `json:"l1FeeVaultWithdrawalNetwork"`
130130
// SequencerFeeVaultWithdrawalNetwork represents the withdrawal network for the SequencerFeeVault.
131-
SequencerFeeVaultWithdrawalNetwork uint8 `json:"sequencerFeeVaultWithdrawalNetwork"`
131+
SequencerFeeVaultWithdrawalNetwork WithdrawalNetwork `json:"sequencerFeeVaultWithdrawalNetwork"`
132132
// L1StandardBridgeProxy represents the address of the L1StandardBridgeProxy on L1 and is used
133133
// as part of building the L2 genesis state.
134134
L1StandardBridgeProxy common.Address `json:"l1StandardBridgeProxy"`
@@ -253,13 +253,13 @@ func (d *DeployConfig) Check() error {
253253
if d.SequencerFeeVaultRecipient == (common.Address{}) {
254254
return fmt.Errorf("%w: SequencerFeeVaultRecipient cannot be address(0)", ErrInvalidDeployConfig)
255255
}
256-
if d.BaseFeeVaultWithdrawalNetwork >= 2 {
256+
if !d.BaseFeeVaultWithdrawalNetwork.Valid() {
257257
return fmt.Errorf("%w: BaseFeeVaultWithdrawalNetwork can only be 0 (L1) or 1 (L2)", ErrInvalidDeployConfig)
258258
}
259-
if d.L1FeeVaultWithdrawalNetwork >= 2 {
259+
if !d.L1FeeVaultWithdrawalNetwork.Valid() {
260260
return fmt.Errorf("%w: L1FeeVaultWithdrawalNetwork can only be 0 (L1) or 1 (L2)", ErrInvalidDeployConfig)
261261
}
262-
if d.SequencerFeeVaultWithdrawalNetwork >= 2 {
262+
if !d.SequencerFeeVaultWithdrawalNetwork.Valid() {
263263
return fmt.Errorf("%w: SequencerFeeVaultWithdrawalNetwork can only be 0 (L1) or 1 (L2)", ErrInvalidDeployConfig)
264264
}
265265
if d.GasPriceOracleOverhead == 0 {
@@ -536,17 +536,17 @@ func NewL2ImmutableConfig(config *DeployConfig, block *types.Block) (immutables.
536536
immutable["SequencerFeeVault"] = immutables.ImmutableValues{
537537
"recipient": config.SequencerFeeVaultRecipient,
538538
"minimumWithdrawalAmount": config.SequencerFeeVaultMinimumWithdrawalAmount,
539-
"withdrawalNetwork": config.SequencerFeeVaultWithdrawalNetwork,
539+
"withdrawalNetwork": config.SequencerFeeVaultWithdrawalNetwork.ToUint8(),
540540
}
541541
immutable["L1FeeVault"] = immutables.ImmutableValues{
542542
"recipient": config.L1FeeVaultRecipient,
543543
"minimumWithdrawalAmount": config.L1FeeVaultMinimumWithdrawalAmount,
544-
"withdrawalNetwork": config.L1FeeVaultWithdrawalNetwork,
544+
"withdrawalNetwork": config.L1FeeVaultWithdrawalNetwork.ToUint8(),
545545
}
546546
immutable["BaseFeeVault"] = immutables.ImmutableValues{
547547
"recipient": config.BaseFeeVaultRecipient,
548548
"minimumWithdrawalAmount": config.BaseFeeVaultMinimumWithdrawalAmount,
549-
"withdrawalNetwork": config.BaseFeeVaultWithdrawalNetwork,
549+
"withdrawalNetwork": config.BaseFeeVaultWithdrawalNetwork.ToUint8(),
550550
}
551551

552552
return immutable, nil

op-chain-ops/genesis/testdata/test-deploy-config-devnet-l1.json

+3-3
Original file line numberDiff line numberDiff line change
@@ -27,9 +27,9 @@
2727
"baseFeeVaultMinimumWithdrawalAmount": "0x8ac7230489e80000",
2828
"l1FeeVaultMinimumWithdrawalAmount": "0x8ac7230489e80000",
2929
"sequencerFeeVaultMinimumWithdrawalAmount": "0x8ac7230489e80000",
30-
"baseFeeVaultWithdrawalNetwork": 0,
31-
"l1FeeVaultWithdrawalNetwork": 0,
32-
"sequencerFeeVaultWithdrawalNetwork": 0,
30+
"baseFeeVaultWithdrawalNetwork": "local",
31+
"l1FeeVaultWithdrawalNetwork": "local",
32+
"sequencerFeeVaultWithdrawalNetwork": "local",
3333

3434
"l1ERC721BridgeProxy": "0xff000000000000000000000000000000000000ff",
3535
"l1StandardBridgeProxy": "0xff000000000000000000000000000000000000fd",

op-chain-ops/genesis/testdata/test-deploy-config-full.json

+3-3
Original file line numberDiff line numberDiff line change
@@ -44,9 +44,9 @@
4444
"baseFeeVaultMinimumWithdrawalAmount": "0x8ac7230489e80000",
4545
"l1FeeVaultMinimumWithdrawalAmount": "0x8ac7230489e80000",
4646
"sequencerFeeVaultMinimumWithdrawalAmount": "0x8ac7230489e80000",
47-
"baseFeeVaultWithdrawalNetwork": 0,
48-
"l1FeeVaultWithdrawalNetwork": 0,
49-
"sequencerFeeVaultWithdrawalNetwork": 0,
47+
"baseFeeVaultWithdrawalNetwork": "remote",
48+
"l1FeeVaultWithdrawalNetwork": "local",
49+
"sequencerFeeVaultWithdrawalNetwork": "local",
5050
"l1StandardBridgeProxy": "0x42000000000000000000000000000000000000f8",
5151
"l1CrossDomainMessengerProxy": "0x42000000000000000000000000000000000000f9",
5252
"l1ERC721BridgeProxy": "0x4200000000000000000000000000000000000060",
+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
package genesis
2+
3+
import (
4+
"encoding/json"
5+
"fmt"
6+
"strconv"
7+
)
8+
9+
// WithdrawalNetwork represents the network that withdrawals are sent to.
10+
// Its value when marshalled in json is intended to be a consistent with its
11+
// internal string type but is backwards-compatible with uint8 values.
12+
// That is, WithdrawalNetwork can be unmarshalled from a JSON field into a uint8.
13+
type WithdrawalNetwork string
14+
15+
// Valid returns if the withdrawal network is valid.
16+
func (w *WithdrawalNetwork) Valid() bool {
17+
switch *w {
18+
case "local", "remote":
19+
return true
20+
default:
21+
return false
22+
}
23+
}
24+
25+
// ToUint8 converts a WithdrawalNetwork to a uint8.
26+
func (w *WithdrawalNetwork) ToUint8() uint8 {
27+
switch *w {
28+
case "remote":
29+
return 0
30+
default:
31+
return 1
32+
}
33+
}
34+
35+
// FromUint8 converts a uint8 to a WithdrawalNetwork.
36+
func FromUint8(i uint8) WithdrawalNetwork {
37+
switch i {
38+
case 0:
39+
return WithdrawalNetwork("remote")
40+
case 1:
41+
return WithdrawalNetwork("local")
42+
default:
43+
return WithdrawalNetwork(strconv.Itoa(int(i)))
44+
}
45+
}
46+
47+
// UnmarshalJSON implements the json.Unmarshaler interface, which
48+
// allows us to ingest values of any json type as an int and run our custom conversion
49+
func (w *WithdrawalNetwork) UnmarshalJSON(b []byte) error {
50+
var s WithdrawalNetwork
51+
if b[0] == '"' {
52+
if err := json.Unmarshal(b, (*string)(&s)); err != nil {
53+
return err
54+
}
55+
} else {
56+
var i uint8
57+
if err := json.Unmarshal(b, &i); err != nil {
58+
return err
59+
}
60+
s = FromUint8(i)
61+
}
62+
if !s.Valid() {
63+
return fmt.Errorf("invalid withdrawal network: %v", s)
64+
}
65+
*w = s
66+
return nil
67+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,125 @@
1+
package genesis
2+
3+
import (
4+
"encoding/json"
5+
"testing"
6+
7+
"github.com/stretchr/testify/require"
8+
)
9+
10+
// TestWithdrawalNetworkValid checks that valid withdrawal networks are detected.
11+
func TestWithdrawalNetworkValid(t *testing.T) {
12+
localWithdrawalNetwork := WithdrawalNetwork("local")
13+
require.True(t, localWithdrawalNetwork.Valid())
14+
15+
remoteWithdrawalNetwork := WithdrawalNetwork("remote")
16+
require.True(t, remoteWithdrawalNetwork.Valid())
17+
18+
invalidWithdrawalNetwork := WithdrawalNetwork("invalid")
19+
require.False(t, invalidWithdrawalNetwork.Valid())
20+
}
21+
22+
// TestWithdrawalNetworkToUint8 checks that withdrawal networks are converted to uint8 correctly.
23+
func TestWithdrawalNetworkToUint8(t *testing.T) {
24+
localWithdrawalNetwork := WithdrawalNetwork("local")
25+
require.EqualValues(t, 1, localWithdrawalNetwork.ToUint8())
26+
27+
remoteWithdrawalNetwork := WithdrawalNetwork("remote")
28+
require.EqualValues(t, 0, remoteWithdrawalNetwork.ToUint8())
29+
30+
invalidWithdrawalNetwork := WithdrawalNetwork("invalid")
31+
require.EqualValues(t, 1, invalidWithdrawalNetwork.ToUint8())
32+
}
33+
34+
// TestWithdrawalNetworkFromUint8 checks that uint8s are converted to withdrawal networks correctly.
35+
func TestWithdrawalNetworkFromUint8(t *testing.T) {
36+
require.EqualValues(t, "local", FromUint8(1))
37+
require.EqualValues(t, "remote", FromUint8(0))
38+
// invalid uint8s are converted to their uint8 string representation
39+
// this will be caught by the Valid() check
40+
require.EqualValues(t, "2", FromUint8(2))
41+
}
42+
43+
// TestWithdrawalNetworkUnmarshalJSON checks that withdrawal networks are unmarshalled correctly.
44+
func TestWithdrawalNetworkUnmarshalJSON(t *testing.T) {
45+
t.Run("LocalInt", func(t *testing.T) {
46+
var w WithdrawalNetwork
47+
require.NoError(t, json.Unmarshal([]byte(`1`), &w))
48+
require.EqualValues(t, "local", w)
49+
})
50+
51+
t.Run("RemoteInt", func(t *testing.T) {
52+
var w WithdrawalNetwork
53+
require.NoError(t, json.Unmarshal([]byte(`0`), &w))
54+
require.EqualValues(t, "remote", w)
55+
})
56+
57+
t.Run("InvalidInt", func(t *testing.T) {
58+
var w WithdrawalNetwork
59+
require.Error(t, json.Unmarshal([]byte(`2`), &w))
60+
})
61+
62+
t.Run("LocalString", func(t *testing.T) {
63+
var w WithdrawalNetwork
64+
require.NoError(t, json.Unmarshal([]byte(`"local"`), &w))
65+
require.EqualValues(t, "local", w)
66+
})
67+
68+
t.Run("RemoteString", func(t *testing.T) {
69+
var w WithdrawalNetwork
70+
require.NoError(t, json.Unmarshal([]byte(`"remote"`), &w))
71+
require.EqualValues(t, "remote", w)
72+
})
73+
74+
t.Run("InvalidString", func(t *testing.T) {
75+
var w WithdrawalNetwork
76+
require.Error(t, json.Unmarshal([]byte(`"invalid"`), &w))
77+
})
78+
}
79+
80+
// TestWithdrawalNetworkInlineJSON tests unmarshalling of withdrawal networks in inline JSON.
81+
func TestWithdrawalNetworkInlineJSON(t *testing.T) {
82+
type tempNetworks struct {
83+
BaseFeeVaultWithdrawalNetwork WithdrawalNetwork `json:"baseFeeVaultWithdrawalNetwork"`
84+
L1FeeVaultWithdrawalNetwork WithdrawalNetwork `json:"l1FeeVaultWithdrawalNetwork"`
85+
SequencerFeeVaultWithdrawalNetwork WithdrawalNetwork `json:"sequencerFeeVaultWithdrawalNetwork"`
86+
}
87+
88+
jsonString := `{
89+
"baseFeeVaultWithdrawalNetwork": "remote",
90+
"l1FeeVaultWithdrawalNetwork": "local",
91+
"sequencerFeeVaultWithdrawalNetwork": "local"
92+
}`
93+
94+
t.Run("StringMarshaling", func(t *testing.T) {
95+
decoded := new(tempNetworks)
96+
require.NoError(t, json.Unmarshal([]byte(jsonString), decoded))
97+
98+
require.Equal(t, WithdrawalNetwork("remote"), decoded.BaseFeeVaultWithdrawalNetwork)
99+
require.Equal(t, WithdrawalNetwork("local"), decoded.L1FeeVaultWithdrawalNetwork)
100+
require.Equal(t, WithdrawalNetwork("local"), decoded.SequencerFeeVaultWithdrawalNetwork)
101+
102+
encoded, err := json.Marshal(decoded)
103+
require.NoError(t, err)
104+
require.JSONEq(t, jsonString, string(encoded))
105+
})
106+
107+
t.Run("IntMarshaling", func(t *testing.T) {
108+
intJsonString := `{
109+
"baseFeeVaultWithdrawalNetwork": 0,
110+
"l1FeeVaultWithdrawalNetwork": 1,
111+
"sequencerFeeVaultWithdrawalNetwork": 1
112+
}`
113+
114+
decoded := new(tempNetworks)
115+
require.NoError(t, json.Unmarshal([]byte(intJsonString), decoded))
116+
117+
require.Equal(t, WithdrawalNetwork("remote"), decoded.BaseFeeVaultWithdrawalNetwork)
118+
require.Equal(t, WithdrawalNetwork("local"), decoded.L1FeeVaultWithdrawalNetwork)
119+
require.Equal(t, WithdrawalNetwork("local"), decoded.SequencerFeeVaultWithdrawalNetwork)
120+
121+
encoded, err := json.Marshal(decoded)
122+
require.NoError(t, err)
123+
require.JSONEq(t, jsonString, string(encoded))
124+
})
125+
}

op-e2e/e2eutils/setup.go

+6-6
Original file line numberDiff line numberDiff line change
@@ -105,12 +105,12 @@ func MakeDeployParams(t require.TestingT, tp *TestParams) *DeployParams {
105105
SequencerFeeVaultRecipient: common.Address{19: 1},
106106
BaseFeeVaultRecipient: common.Address{19: 2},
107107
L1FeeVaultRecipient: common.Address{19: 3},
108-
BaseFeeVaultMinimumWithdrawalAmount: uint64ToBig(1000_000_000), // 1 gwei
109-
L1FeeVaultMinimumWithdrawalAmount: uint64ToBig(1000_000_000), // 1 gwei
110-
SequencerFeeVaultMinimumWithdrawalAmount: uint64ToBig(1000_000_000), // 1 gwei
111-
BaseFeeVaultWithdrawalNetwork: uint8(1), // L2 withdrawal network
112-
L1FeeVaultWithdrawalNetwork: uint8(1), // L2 withdrawal network
113-
SequencerFeeVaultWithdrawalNetwork: uint8(1), // L2 withdrawal network
108+
BaseFeeVaultMinimumWithdrawalAmount: uint64ToBig(1000_000_000), // 1 gwei
109+
L1FeeVaultMinimumWithdrawalAmount: uint64ToBig(1000_000_000), // 1 gwei
110+
SequencerFeeVaultMinimumWithdrawalAmount: uint64ToBig(1000_000_000), // 1 gwei
111+
BaseFeeVaultWithdrawalNetwork: genesis.WithdrawalNetwork("local"), // L2 withdrawal network
112+
L1FeeVaultWithdrawalNetwork: genesis.WithdrawalNetwork("local"), // L2 withdrawal network
113+
SequencerFeeVaultWithdrawalNetwork: genesis.WithdrawalNetwork("local"), // L2 withdrawal network
114114

115115
EIP1559Elasticity: 10,
116116
EIP1559Denominator: 50,

op-e2e/setup.go

+6-6
Original file line numberDiff line numberDiff line change
@@ -128,12 +128,12 @@ func DefaultSystemConfig(t *testing.T) SystemConfig {
128128
SequencerFeeVaultRecipient: common.Address{19: 1},
129129
BaseFeeVaultRecipient: common.Address{19: 2},
130130
L1FeeVaultRecipient: common.Address{19: 3},
131-
BaseFeeVaultMinimumWithdrawalAmount: uint642big(1000_000_000), // 1 gwei
132-
L1FeeVaultMinimumWithdrawalAmount: uint642big(1000_000_000), // 1 gwei
133-
SequencerFeeVaultMinimumWithdrawalAmount: uint642big(1000_000_000), // 1 gwei
134-
BaseFeeVaultWithdrawalNetwork: uint8(1), // L2 withdrawal network
135-
L1FeeVaultWithdrawalNetwork: uint8(1), // L2 withdrawal network
136-
SequencerFeeVaultWithdrawalNetwork: uint8(1), // L2 withdrawal network
131+
BaseFeeVaultMinimumWithdrawalAmount: uint642big(1000_000_000), // 1 gwei
132+
L1FeeVaultMinimumWithdrawalAmount: uint642big(1000_000_000), // 1 gwei
133+
SequencerFeeVaultMinimumWithdrawalAmount: uint642big(1000_000_000), // 1 gwei
134+
BaseFeeVaultWithdrawalNetwork: genesis.WithdrawalNetwork("local"), // L2 withdrawal network
135+
L1FeeVaultWithdrawalNetwork: genesis.WithdrawalNetwork("local"), // L2 withdrawal network
136+
SequencerFeeVaultWithdrawalNetwork: genesis.WithdrawalNetwork("local"), // L2 withdrawal network
137137

138138
DeploymentWaitConfirmations: 1,
139139

0 commit comments

Comments
 (0)