From 661a2857b6268989a352475131c93da64faf8628 Mon Sep 17 00:00:00 2001 From: Mustafa Elbehery Date: Mon, 22 Apr 2024 20:43:25 +0200 Subject: [PATCH] add pageId flag to check cmd Signed-off-by: Mustafa Elbehery --- cmd/bbolt/command_check.go | 23 +++++++++-- cmd/bbolt/command_check_test.go | 69 ++++++++++++++++++++++++--------- 2 files changed, 70 insertions(+), 22 deletions(-) diff --git a/cmd/bbolt/command_check.go b/cmd/bbolt/command_check.go index 237e02f91..4cba6c25e 100644 --- a/cmd/bbolt/command_check.go +++ b/cmd/bbolt/command_check.go @@ -2,6 +2,7 @@ package main import ( "fmt" + "github.com/spf13/pflag" "github.com/spf13/cobra" @@ -9,20 +10,30 @@ import ( "go.etcd.io/bbolt/internal/guts_cli" ) +type checkPageOptions struct { + pageId uint64 +} + +func (o *checkPageOptions) AddFlags(fs *pflag.FlagSet) { + fs.Uint64VarP(&o.pageId, "pageId", "", o.pageId, "check db integrity starting from the given pageId") +} + func newCheckCommand() *cobra.Command { + var o checkPageOptions checkCmd := &cobra.Command{ Use: "check ", Short: "verify integrity of bbolt database data", - Args: cobra.ExactArgs(1), + Args: cobra.MinimumNArgs(1), RunE: func(cmd *cobra.Command, args []string) error { - return checkFunc(cmd, args[0]) + return checkFunc(cmd, args[0], o) }, } + o.AddFlags(checkCmd.Flags()) return checkCmd } -func checkFunc(cmd *cobra.Command, dbPath string) error { +func checkFunc(cmd *cobra.Command, dbPath string, cfg checkPageOptions) error { if _, err := checkSourceDBPath(dbPath); err != nil { return err } @@ -37,10 +48,14 @@ func checkFunc(cmd *cobra.Command, dbPath string) error { } defer db.Close() + opts := []bolt.CheckOption{bolt.WithKVStringer(CmdKvStringer())} + if cfg.pageId != 0 { + opts = append(opts, bolt.WithPageId(cfg.pageId)) + } // Perform consistency check. return db.View(func(tx *bolt.Tx) error { var count int - for err := range tx.Check(bolt.WithKVStringer(CmdKvStringer())) { + for err := range tx.Check(opts...) { fmt.Fprintln(cmd.OutOrStdout(), err) count++ } diff --git a/cmd/bbolt/command_check_test.go b/cmd/bbolt/command_check_test.go index 02500745c..476211081 100644 --- a/cmd/bbolt/command_check_test.go +++ b/cmd/bbolt/command_check_test.go @@ -9,25 +9,58 @@ import ( main "go.etcd.io/bbolt/cmd/bbolt" "go.etcd.io/bbolt/internal/btesting" + "go.etcd.io/bbolt/internal/guts_cli" ) func TestCheckCommand_Run(t *testing.T) { - db := btesting.MustCreateDB(t) - db.Close() - defer requireDBNoChange(t, dbData(t, db.Path()), db.Path()) - - rootCmd := main.NewRootCommand() - // capture output for assertion - outputBuf := bytes.NewBufferString("") - rootCmd.SetOut(outputBuf) - - rootCmd.SetArgs([]string{ - "check", db.Path(), - }) - err := rootCmd.Execute() - require.NoError(t, err) - - output, err := io.ReadAll(outputBuf) - require.NoError(t, err) - require.Equalf(t, "OK\n", string(output), "unexpected stdout:\n\n%s", string(output)) + testCases := []struct { + name string + args []string + expErr error + expOutput string + }{ + { + name: "check whole db", + args: []string{"check", "path"}, + expErr: nil, + expOutput: "OK\n", + }, + { + name: "check valid pageId", + args: []string{"check", "path", "--pageId", "3"}, + expErr: nil, + expOutput: "OK\n", + }, + { + name: "check invalid pageId", + args: []string{"check", "path", "--pageId", "1"}, + expErr: guts_cli.ErrCorrupt, + expOutput: "page ID (1) out of range [2, 4)", + }, + } + + for _, tc := range testCases { + t.Run(tc.name, func(t *testing.T) { + + t.Log("Creating sample DB") + db := btesting.MustCreateDB(t) + db.Close() + defer requireDBNoChange(t, dbData(t, db.Path()), db.Path()) + + t.Log("Running check cmd") + rootCmd := main.NewRootCommand() + outputBuf := bytes.NewBufferString("") // capture output for assertion + rootCmd.SetOut(outputBuf) + + tc.args[1] = db.Path() // path to be replaced with db.Path() + rootCmd.SetArgs(tc.args) + err := rootCmd.Execute() + require.Equal(t, tc.expErr, err) + + t.Log("Checking output") + output, err := io.ReadAll(outputBuf) + require.NoError(t, err) + require.Containsf(t, string(output), tc.expOutput, "unexpected stdout:\n\n%s", string(output)) + }) + } }