From 050a9fd6cde766fffd7d59d68d901daba2cc72f4 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 28 Jun 2024 18:44:27 +0200 Subject: [PATCH 1/4] support auto-detection of database engine --- cmd/conf/database.go | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/cmd/conf/database.go b/cmd/conf/database.go index 6fde00579f..2835f01319 100644 --- a/cmd/conf/database.go +++ b/cmd/conf/database.go @@ -32,7 +32,7 @@ var PersistentConfigDefault = PersistentConfig{ LogDir: "", Handles: 512, Ancient: "", - DBEngine: "leveldb", + DBEngine: "", // auto detect database type based on the db dir contents Pebble: PebbleConfigDefault, } @@ -42,7 +42,7 @@ func PersistentConfigAddOptions(prefix string, f *flag.FlagSet) { f.String(prefix+".log-dir", PersistentConfigDefault.LogDir, "directory to store log file") f.Int(prefix+".handles", PersistentConfigDefault.Handles, "number of file descriptor handles to use for the database") f.String(prefix+".ancient", PersistentConfigDefault.Ancient, "directory of ancient where the chain freezer can be opened") - f.String(prefix+".db-engine", PersistentConfigDefault.DBEngine, "backing database implementation to use ('leveldb' or 'pebble')") + f.String(prefix+".db-engine", PersistentConfigDefault.DBEngine, "backing database implementation to use. If set to empty string the database type will be autodetected and if no pre-existing database is found it will default to creating new pebble database ('leveldb', 'pebble' or '' (empty string = auto-detect))") PebbleConfigAddOptions(prefix+".pebble", f) } @@ -97,11 +97,12 @@ func DatabaseInDirectory(path string) bool { } func (c *PersistentConfig) Validate() error { - // we are validating .db-engine here to avoid unintended behaviour as empty string value also has meaning in geth's node.Config.DBEngine - if c.DBEngine != "leveldb" && c.DBEngine != "pebble" { - return fmt.Errorf(`invalid .db-engine choice: %q, allowed "leveldb" or "pebble"`, c.DBEngine) + if c.DBEngine != "leveldb" && c.DBEngine != "pebble" && c.DBEngine != "" { + return fmt.Errorf(`invalid .db-engine choice: %q, allowed "leveldb", "pebble" or ""`, c.DBEngine) } - if c.DBEngine == "pebble" { + // if DBEngine == "" then we may end up opening pebble database, so we want to validate the Pebble config + // if pre-existing database base is leveldb backed, then user shouldn't change the Pebble config defaults => this check should also succeed + if c.DBEngine == "pebble" || c.DBEngine == "" { if err := c.Pebble.Validate(); err != nil { return err } From 234207b2653cb1c52ce5ca1de15e305175cc5bd6 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 28 Jun 2024 19:22:58 +0200 Subject: [PATCH 2/4] check if database opening error is NotExists error, otherwise fail early --- cmd/conf/database.go | 6 +++--- cmd/nitro/init.go | 11 +++++++++++ 2 files changed, 14 insertions(+), 3 deletions(-) diff --git a/cmd/conf/database.go b/cmd/conf/database.go index 2835f01319..a75cca77d5 100644 --- a/cmd/conf/database.go +++ b/cmd/conf/database.go @@ -32,7 +32,7 @@ var PersistentConfigDefault = PersistentConfig{ LogDir: "", Handles: 512, Ancient: "", - DBEngine: "", // auto detect database type based on the db dir contents + DBEngine: "", // auto-detect database type based on the db dir contents Pebble: PebbleConfigDefault, } @@ -42,7 +42,7 @@ func PersistentConfigAddOptions(prefix string, f *flag.FlagSet) { f.String(prefix+".log-dir", PersistentConfigDefault.LogDir, "directory to store log file") f.Int(prefix+".handles", PersistentConfigDefault.Handles, "number of file descriptor handles to use for the database") f.String(prefix+".ancient", PersistentConfigDefault.Ancient, "directory of ancient where the chain freezer can be opened") - f.String(prefix+".db-engine", PersistentConfigDefault.DBEngine, "backing database implementation to use. If set to empty string the database type will be autodetected and if no pre-existing database is found it will default to creating new pebble database ('leveldb', 'pebble' or '' (empty string = auto-detect))") + f.String(prefix+".db-engine", PersistentConfigDefault.DBEngine, "backing database implementation to use. If set to empty string the database type will be autodetected and if no pre-existing database is found it will default to creating new pebble database ('leveldb', 'pebble' or '' = auto-detect)") PebbleConfigAddOptions(prefix+".pebble", f) } @@ -101,7 +101,7 @@ func (c *PersistentConfig) Validate() error { return fmt.Errorf(`invalid .db-engine choice: %q, allowed "leveldb", "pebble" or ""`, c.DBEngine) } // if DBEngine == "" then we may end up opening pebble database, so we want to validate the Pebble config - // if pre-existing database base is leveldb backed, then user shouldn't change the Pebble config defaults => this check should also succeed + // if pre-existing database is leveldb backed, then user shouldn't change the Pebble config defaults => this check should also succeed if c.DBEngine == "pebble" || c.DBEngine == "" { if err := c.Pebble.Validate(); err != nil { return err diff --git a/cmd/nitro/init.go b/cmd/nitro/init.go index 49d30fd59c..9a871bdf03 100644 --- a/cmd/nitro/init.go +++ b/cmd/nitro/init.go @@ -17,6 +17,7 @@ import ( "os" "path" "path/filepath" + "regexp" "runtime" "strings" "sync" @@ -396,6 +397,16 @@ func openInitializeChainDb(ctx context.Context, stack *node.Node, config *NodeCo return chainDb, l2BlockChain, nil } readOnlyDb.Close() + } else { // err != nil + isPebbleNotExistError, matchErr := regexp.MatchString("pebble: database .* does not exist", err.Error()) + if matchErr != nil { + return nil, nil, fmt.Errorf("Failed pattern matching database opening error string (\"%v\"): %w", err.Error(), matchErr) + } + isLeveldbNotExistError := os.IsNotExist(err) + // we only want to continue if the error is pebble or leveldb not exist error + if !isLeveldbNotExistError && !isPebbleNotExistError { + return nil, nil, fmt.Errorf("Failed to open database: %w", err) + } } } From 30a3e8ac74892465c6d96e603a768fcdcaff0600 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 28 Jun 2024 19:51:36 +0200 Subject: [PATCH 3/4] test if not exist error did not change --- cmd/nitro/init.go | 17 +++++++++++------ cmd/nitro/init_test.go | 33 +++++++++++++++++++++++++++++++++ 2 files changed, 44 insertions(+), 6 deletions(-) diff --git a/cmd/nitro/init.go b/cmd/nitro/init.go index 9a871bdf03..efaa2e95b1 100644 --- a/cmd/nitro/init.go +++ b/cmd/nitro/init.go @@ -327,6 +327,16 @@ func dirExists(path string) bool { return info.IsDir() } +var pebbleNotExistErrorRegex = regexp.MustCompile("pebble: database .* does not exist") + +func isPebbleNotExistError(err error) bool { + return pebbleNotExistErrorRegex.MatchString(err.Error()) +} + +func isLeveldbNotExistError(err error) bool { + return os.IsNotExist(err) +} + func openInitializeChainDb(ctx context.Context, stack *node.Node, config *NodeConfig, chainId *big.Int, cacheConfig *core.CacheConfig, persistentConfig *conf.PersistentConfig, l1Client arbutil.L1Interface, rollupAddrs chaininfo.RollupAddresses) (ethdb.Database, *core.BlockChain, error) { if !config.Init.Force { if readOnlyDb, err := stack.OpenDatabaseWithFreezerWithExtraOptions("l2chaindata", 0, 0, "", "l2chaindata/", true, persistentConfig.Pebble.ExtraOptions("l2chaindata")); err == nil { @@ -398,13 +408,8 @@ func openInitializeChainDb(ctx context.Context, stack *node.Node, config *NodeCo } readOnlyDb.Close() } else { // err != nil - isPebbleNotExistError, matchErr := regexp.MatchString("pebble: database .* does not exist", err.Error()) - if matchErr != nil { - return nil, nil, fmt.Errorf("Failed pattern matching database opening error string (\"%v\"): %w", err.Error(), matchErr) - } - isLeveldbNotExistError := os.IsNotExist(err) // we only want to continue if the error is pebble or leveldb not exist error - if !isLeveldbNotExistError && !isPebbleNotExistError { + if !isLeveldbNotExistError(err) && !isPebbleNotExistError(err) { return nil, nil, fmt.Errorf("Failed to open database: %w", err) } } diff --git a/cmd/nitro/init_test.go b/cmd/nitro/init_test.go index 17bac3d670..47ab4b4491 100644 --- a/cmd/nitro/init_test.go +++ b/cmd/nitro/init_test.go @@ -17,6 +17,7 @@ import ( "testing" "time" + "github.com/ethereum/go-ethereum/node" "github.com/offchainlabs/nitro/cmd/conf" "github.com/offchainlabs/nitro/util/testhelpers" ) @@ -177,3 +178,35 @@ func startFileServer(t *testing.T, ctx context.Context, dir string) string { }() return addr } + +func testIsNotExistError(t *testing.T, dbEngine string, isNotExist func(error) bool) { + stackConf := node.DefaultConfig + stackConf.DataDir = t.TempDir() + stackConf.DBEngine = dbEngine + stack, err := node.New(&stackConf) + if err != nil { + t.Fatalf("Failed to created test stack: %v", err) + } + defer stack.Close() + readonly := true + _, err = stack.OpenDatabaseWithExtraOptions("test", 16, 16, "", readonly, nil) + if err == nil { + t.Fatal("Opening non-existent database did not fail") + } + if !isNotExist(err) { + t.Fatalf("Failed to classify error as not exist error - internal implementation of OpenDatabaseWithExtraOptions might have changed, err: %v", err) + } + err = errors.New("some other error") + if isNotExist(err) { + t.Fatalf("Classified other error as not exist, err: %v", err) + } +} + +func TestIsNotExistError(t *testing.T) { + t.Run("TestIsPebbleNotExistError", func(t *testing.T) { + testIsNotExistError(t, "pebble", isPebbleNotExistError) + }) + t.Run("TestIsLeveldbNotExistError", func(t *testing.T) { + testIsNotExistError(t, "leveldb", isLeveldbNotExistError) + }) +} From 83d940568eaabcd7ff3c5a19c9c55068e86c0459 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 28 Jun 2024 22:30:52 +0200 Subject: [PATCH 4/4] fix lint --- cmd/nitro/init.go | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/cmd/nitro/init.go b/cmd/nitro/init.go index efaa2e95b1..9c23cdf852 100644 --- a/cmd/nitro/init.go +++ b/cmd/nitro/init.go @@ -407,11 +407,9 @@ func openInitializeChainDb(ctx context.Context, stack *node.Node, config *NodeCo return chainDb, l2BlockChain, nil } readOnlyDb.Close() - } else { // err != nil + } else if !isLeveldbNotExistError(err) && !isPebbleNotExistError(err) { // we only want to continue if the error is pebble or leveldb not exist error - if !isLeveldbNotExistError(err) && !isPebbleNotExistError(err) { - return nil, nil, fmt.Errorf("Failed to open database: %w", err) - } + return nil, nil, fmt.Errorf("Failed to open database: %w", err) } }