Skip to content

Commit

Permalink
Merge pull request #2427 from OffchainLabs/das-backends-conf
Browse files Browse the repository at this point in the history
[config change] Make DAS backends and keyset easier to configure, fixes: NIT-2594
  • Loading branch information
PlasmaPower authored Jun 28, 2024
2 parents 9085e2b + c9ac4e0 commit 3767b1f
Show file tree
Hide file tree
Showing 8 changed files with 87 additions and 41 deletions.
6 changes: 5 additions & 1 deletion cmd/datool/datool.go
Original file line number Diff line number Diff line change
Expand Up @@ -316,6 +316,10 @@ func parseDumpKeyset(args []string) (*DumpKeysetConfig, error) {
return nil, err
}

if err = das.FixKeysetCLIParsing("keyset.backends", k); err != nil {
return nil, err
}

var config DumpKeysetConfig
if err := confighelpers.EndCommonParse(k, &config); err != nil {
return nil, err
Expand All @@ -334,7 +338,7 @@ func parseDumpKeyset(args []string) (*DumpKeysetConfig, error) {
if config.Keyset.AssumedHonest == 0 {
return nil, errors.New("--keyset.assumed-honest must be set")
}
if config.Keyset.Backends == "" {
if config.Keyset.Backends == nil {
return nil, errors.New("--keyset.backends must be set")
}

Expand Down
5 changes: 4 additions & 1 deletion cmd/nitro/config_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import (

"github.com/offchainlabs/nitro/cmd/genericconf"
"github.com/offchainlabs/nitro/cmd/util/confighelpers"
"github.com/offchainlabs/nitro/das"
"github.com/offchainlabs/nitro/util/colors"
"github.com/offchainlabs/nitro/util/testhelpers"

Expand All @@ -28,6 +29,8 @@ func TestEmptyCliConfig(t *testing.T) {
NodeConfigAddOptions(f)
k, err := confighelpers.BeginCommonParse(f, []string{})
Require(t, err)
err = das.FixKeysetCLIParsing("node.data-availability.rpc-aggregator.backends", k)
Require(t, err)
var emptyCliNodeConfig NodeConfig
err = confighelpers.EndCommonParse(k, &emptyCliNodeConfig)
Require(t, err)
Expand Down Expand Up @@ -57,7 +60,7 @@ func TestValidatorConfig(t *testing.T) {
}

func TestAggregatorConfig(t *testing.T) {
args := strings.Split("--persistent.chain /tmp/data --init.dev-init --node.parent-chain-reader.enable=false --parent-chain.id 5 --chain.id 421613 --node.batch-poster.parent-chain-wallet.pathname /l1keystore --node.batch-poster.parent-chain-wallet.password passphrase --http.addr 0.0.0.0 --ws.addr 0.0.0.0 --node.sequencer --execution.sequencer.enable --node.feed.output.enable --node.feed.output.port 9642 --node.data-availability.enable --node.data-availability.rpc-aggregator.backends {[\"url\":\"http://localhost:8547\",\"pubkey\":\"abc==\",\"signerMask\":0x1]}", " ")
args := strings.Split("--persistent.chain /tmp/data --init.dev-init --node.parent-chain-reader.enable=false --parent-chain.id 5 --chain.id 421613 --node.batch-poster.parent-chain-wallet.pathname /l1keystore --node.batch-poster.parent-chain-wallet.password passphrase --http.addr 0.0.0.0 --ws.addr 0.0.0.0 --node.sequencer --execution.sequencer.enable --node.feed.output.enable --node.feed.output.port 9642 --node.data-availability.enable --node.data-availability.rpc-aggregator.backends [{\"url\":\"http://localhost:8547\",\"pubkey\":\"abc==\"}]", " ")
_, _, err := ParseNode(context.Background(), args)
Require(t, err)
}
Expand Down
5 changes: 5 additions & 0 deletions cmd/nitro/nitro.go
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,7 @@ import (
"github.com/offchainlabs/nitro/cmd/genericconf"
"github.com/offchainlabs/nitro/cmd/util"
"github.com/offchainlabs/nitro/cmd/util/confighelpers"
"github.com/offchainlabs/nitro/das"
"github.com/offchainlabs/nitro/execution/gethexec"
_ "github.com/offchainlabs/nitro/execution/nodeInterface"
"github.com/offchainlabs/nitro/solgen/go/bridgegen"
Expand Down Expand Up @@ -879,6 +880,10 @@ func ParseNode(ctx context.Context, args []string) (*NodeConfig, *genericconf.Wa
return nil, nil, err
}

if err = das.FixKeysetCLIParsing("node.data-availability.rpc-aggregator.backends", k); err != nil {
return nil, nil, err
}

var nodeConfig NodeConfig
if err := confighelpers.EndCommonParse(k, &nodeConfig); err != nil {
return nil, nil, err
Expand Down
14 changes: 8 additions & 6 deletions das/aggregator.go
Original file line number Diff line number Diff line change
Expand Up @@ -37,22 +37,24 @@ var (
)

type AggregatorConfig struct {
Enable bool `koanf:"enable"`
AssumedHonest int `koanf:"assumed-honest"`
Backends string `koanf:"backends"`
MaxStoreChunkBodySize int `koanf:"max-store-chunk-body-size"`
Enable bool `koanf:"enable"`
AssumedHonest int `koanf:"assumed-honest"`
Backends BackendConfigList `koanf:"backends"`
MaxStoreChunkBodySize int `koanf:"max-store-chunk-body-size"`
}

var DefaultAggregatorConfig = AggregatorConfig{
AssumedHonest: 0,
Backends: "",
Backends: nil,
MaxStoreChunkBodySize: 512 * 1024,
}

var parsedBackendsConf BackendConfigList

func AggregatorConfigAddOptions(prefix string, f *flag.FlagSet) {
f.Bool(prefix+".enable", DefaultAggregatorConfig.Enable, "enable storage of sequencer batch data from a list of RPC endpoints; this should only be used by the batch poster and not in combination with other DAS storage types")
f.Int(prefix+".assumed-honest", DefaultAggregatorConfig.AssumedHonest, "Number of assumed honest backends (H). If there are N backends, K=N+1-H valid responses are required to consider an Store request to be successful.")
f.String(prefix+".backends", DefaultAggregatorConfig.Backends, "JSON RPC backend configuration")
f.Var(&parsedBackendsConf, prefix+".backends", "JSON RPC backend configuration. This can be specified on the command line as a JSON array, eg: [{\"url\": \"...\", \"pubkey\": \"...\"},...], or as a JSON array in the config file.")
f.Int(prefix+".max-store-chunk-body-size", DefaultAggregatorConfig.MaxStoreChunkBodySize, "maximum HTTP POST body size to use for individual batch chunks, including JSON RPC overhead and an estimated overhead of 512B of headers")
}

Expand Down
65 changes: 53 additions & 12 deletions das/rpc_aggregator.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,8 @@ import (
"math/bits"
"net/url"

"github.com/knadh/koanf"
"github.com/knadh/koanf/providers/confmap"
"github.com/offchainlabs/nitro/arbstate/daprovider"
"github.com/offchainlabs/nitro/blsSignatures"
"github.com/offchainlabs/nitro/solgen/go/bridgegen"
Expand All @@ -23,9 +25,54 @@ import (
)

type BackendConfig struct {
URL string `json:"url"`
PubKeyBase64Encoded string `json:"pubkey"`
SignerMask uint64 `json:"signermask"`
URL string `koanf:"url" json:"url"`
Pubkey string `koanf:"pubkey" json:"pubkey"`
}

type BackendConfigList []BackendConfig

func (l *BackendConfigList) String() string {
b, _ := json.Marshal(*l)
return string(b)
}

func (l *BackendConfigList) Set(value string) error {
return l.UnmarshalJSON([]byte(value))
}

func (l *BackendConfigList) UnmarshalJSON(data []byte) error {
var tmp []BackendConfig
if err := json.Unmarshal(data, &tmp); err != nil {
return err
}
*l = tmp
return nil
}

func (l *BackendConfigList) Type() string {
return "backendConfigList"
}

func FixKeysetCLIParsing(path string, k *koanf.Koanf) error {
rawBackends := k.Get(path)
if bk, ok := rawBackends.(string); ok {
err := parsedBackendsConf.UnmarshalJSON([]byte(bk))
if err != nil {
return err
}

// Create a map with the parsed backend configurations
tempMap := map[string]interface{}{
path: parsedBackendsConf,
}

// Load the map into koanf
if err = k.Load(confmap.Provider(tempMap, "."), nil); err != nil {
return err
}

}
return nil
}

func NewRPCAggregator(ctx context.Context, config DataAvailabilityConfig, signer signature.DataSignerFunc) (*Aggregator, error) {
Expand Down Expand Up @@ -53,15 +100,9 @@ func NewRPCAggregatorWithSeqInboxCaller(config DataAvailabilityConfig, seqInboxC
}

func ParseServices(config AggregatorConfig, signer signature.DataSignerFunc) ([]ServiceDetails, error) {
var cs []BackendConfig
err := json.Unmarshal([]byte(config.Backends), &cs)
if err != nil {
return nil, err
}

var services []ServiceDetails

for _, b := range cs {
for i, b := range config.Backends {
url, err := url.Parse(b.URL)
if err != nil {
return nil, err
Expand All @@ -73,12 +114,12 @@ func ParseServices(config AggregatorConfig, signer signature.DataSignerFunc) ([]
return nil, err
}

pubKey, err := DecodeBase64BLSPublicKey([]byte(b.PubKeyBase64Encoded))
pubKey, err := DecodeBase64BLSPublicKey([]byte(b.Pubkey))
if err != nil {
return nil, err
}

d, err := NewServiceDetails(service, *pubKey, b.SignerMask, metricName)
d, err := NewServiceDetails(service, *pubKey, 1<<uint64(i), metricName)
if err != nil {
return nil, err
}
Expand Down
13 changes: 5 additions & 8 deletions das/rpc_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,6 @@ import (
"context"
"encoding/base64"
"encoding/hex"
"encoding/json"
"net"
"sync"
"testing"
Expand Down Expand Up @@ -73,18 +72,16 @@ func testRpcImpl(t *testing.T, size, times int, concurrent bool) {
}
}()
testhelpers.RequireImpl(t, err)
beConfig := BackendConfig{
URL: "http://" + lis.Addr().String(),
PubKeyBase64Encoded: blsPubToBase64(pubkey),
SignerMask: 1,
}
beConfigs := BackendConfigList{BackendConfig{
URL: "http://" + lis.Addr().String(),
Pubkey: blsPubToBase64(pubkey),
}}

backendsJsonByte, err := json.Marshal([]BackendConfig{beConfig})
testhelpers.RequireImpl(t, err)
aggConf := DataAvailabilityConfig{
RPCAggregator: AggregatorConfig{
AssumedHonest: 1,
Backends: string(backendsJsonByte),
Backends: beConfigs,
MaxStoreChunkBodySize: (chunkSize * 2) + len(sendChunkJSONBoilerplate),
},
RequestTimeout: time.Minute,
Expand Down
5 changes: 2 additions & 3 deletions system_tests/common_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1144,9 +1144,8 @@ func setupConfigWithDAS(
Require(t, err)

beConfigA := das.BackendConfig{
URL: "http://" + rpcLis.Addr().String(),
PubKeyBase64Encoded: blsPubToBase64(dasSignerKey),
SignerMask: 1,
URL: "http://" + rpcLis.Addr().String(),
Pubkey: blsPubToBase64(dasSignerKey),
}
l1NodeConfigA.DataAvailability.RPCAggregator = aggConfigForBackend(t, beConfigA)
l1NodeConfigA.DataAvailability.Enable = true
Expand Down
15 changes: 5 additions & 10 deletions system_tests/das_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@ package arbtest
import (
"context"
"encoding/base64"
"encoding/json"
"io"
"math/big"
"net"
Expand Down Expand Up @@ -80,9 +79,8 @@ func startLocalDASServer(
restServer, err := das.NewRestfulDasServerOnListener(restLis, genericconf.HTTPServerTimeoutConfigDefault, storageService, storageService)
Require(t, err)
beConfig := das.BackendConfig{
URL: "http://" + rpcLis.Addr().String(),
PubKeyBase64Encoded: blsPubToBase64(pubkey),
SignerMask: 1,
URL: "http://" + rpcLis.Addr().String(),
Pubkey: blsPubToBase64(pubkey),
}
return rpcServer, pubkey, beConfig, restServer, "http://" + restLis.Addr().String()
}
Expand All @@ -95,12 +93,10 @@ func blsPubToBase64(pubkey *blsSignatures.PublicKey) string {
}

func aggConfigForBackend(t *testing.T, backendConfig das.BackendConfig) das.AggregatorConfig {
backendsJsonByte, err := json.Marshal([]das.BackendConfig{backendConfig})
Require(t, err)
return das.AggregatorConfig{
Enable: true,
AssumedHonest: 1,
Backends: string(backendsJsonByte),
Backends: das.BackendConfigList{backendConfig},
MaxStoreChunkBodySize: 512 * 1024,
}
}
Expand Down Expand Up @@ -301,9 +297,8 @@ func TestDASComplexConfigAndRestMirror(t *testing.T) {
RequestTimeout: 5 * time.Second,
}
beConfigA := das.BackendConfig{
URL: "http://" + rpcLis.Addr().String(),
PubKeyBase64Encoded: blsPubToBase64(pubkey),
SignerMask: 1,
URL: "http://" + rpcLis.Addr().String(),
Pubkey: blsPubToBase64(pubkey),
}
l1NodeConfigA.DataAvailability.RPCAggregator = aggConfigForBackend(t, beConfigA)
l1NodeConfigA.DataAvailability.RestAggregator = das.DefaultRestfulClientAggregatorConfig
Expand Down

0 comments on commit 3767b1f

Please sign in to comment.