mirror of
https://source.quilibrium.com/quilibrium/ceremonyclient.git
synced 2024-12-27 09:05:18 +00:00
133 lines
3.2 KiB
Go
133 lines
3.2 KiB
Go
|
// Copyright 2023 The LevelDB-Go and Pebble Authors. All rights reserved. Use
|
||
|
// of this source code is governed by a BSD-style license that can be found in
|
||
|
// the LICENSE file.
|
||
|
|
||
|
package tool
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"slices"
|
||
|
|
||
|
"github.com/cockroachdb/pebble"
|
||
|
"github.com/cockroachdb/pebble/internal/base"
|
||
|
"github.com/cockroachdb/pebble/objstorage"
|
||
|
"github.com/cockroachdb/pebble/objstorage/objstorageprovider/remoteobjcat"
|
||
|
"github.com/cockroachdb/pebble/record"
|
||
|
"github.com/spf13/cobra"
|
||
|
)
|
||
|
|
||
|
// remoteCatalogT implements tools for the remote object catalog.
|
||
|
type remoteCatalogT struct {
|
||
|
Root *cobra.Command
|
||
|
Dump *cobra.Command
|
||
|
|
||
|
verbose bool
|
||
|
opts *pebble.Options
|
||
|
}
|
||
|
|
||
|
func newRemoteCatalog(opts *pebble.Options) *remoteCatalogT {
|
||
|
m := &remoteCatalogT{
|
||
|
opts: opts,
|
||
|
}
|
||
|
|
||
|
m.Root = &cobra.Command{
|
||
|
Use: "remotecat",
|
||
|
Short: "remote object catalog introspection tools",
|
||
|
}
|
||
|
|
||
|
// Add dump command
|
||
|
m.Dump = &cobra.Command{
|
||
|
Use: "dump <remote-catalog-files>",
|
||
|
Short: "print remote object catalog contents",
|
||
|
Long: `
|
||
|
Print the contents of the REMOTE-OBJ-CATALOG files.
|
||
|
`,
|
||
|
Args: cobra.MinimumNArgs(1),
|
||
|
Run: m.runDump,
|
||
|
}
|
||
|
m.Dump.Flags().BoolVarP(&m.verbose, "verbose", "v", false, "show each record in the catalog")
|
||
|
m.Root.AddCommand(m.Dump)
|
||
|
|
||
|
return m
|
||
|
}
|
||
|
|
||
|
func (m *remoteCatalogT) runDump(cmd *cobra.Command, args []string) {
|
||
|
for _, arg := range args {
|
||
|
err := m.runDumpOne(cmd.OutOrStdout(), arg)
|
||
|
if err != nil {
|
||
|
fmt.Fprintf(cmd.OutOrStderr(), "%s\n", err)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (m *remoteCatalogT) runDumpOne(stdout io.Writer, filename string) error {
|
||
|
f, err := m.opts.FS.Open(filename)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
var creatorID objstorage.CreatorID
|
||
|
objects := make(map[base.DiskFileNum]remoteobjcat.RemoteObjectMetadata)
|
||
|
|
||
|
fmt.Fprintf(stdout, "%s\n", filename)
|
||
|
var editIdx int
|
||
|
rr := record.NewReader(f, 0 /* logNum */)
|
||
|
for {
|
||
|
offset := rr.Offset()
|
||
|
r, err := rr.Next()
|
||
|
if err == io.EOF {
|
||
|
break
|
||
|
} else if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
var ve remoteobjcat.VersionEdit
|
||
|
err = ve.Decode(r)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
if m.verbose {
|
||
|
fmt.Fprintf(stdout, "%d/%d\n", offset, editIdx)
|
||
|
if ve.CreatorID.IsSet() {
|
||
|
fmt.Fprintf(stdout, " CreatorID: %s\n", ve.CreatorID)
|
||
|
}
|
||
|
if len(ve.NewObjects) > 0 {
|
||
|
fmt.Fprintf(stdout, " NewObjects:\n")
|
||
|
for _, m := range ve.NewObjects {
|
||
|
fmt.Fprintf(
|
||
|
stdout, " %s CreatorID: %s CreatorFileNum: %s Locator: %q CustomObjectName: %q\n",
|
||
|
m.FileNum, m.CreatorID, m.CreatorFileNum, m.Locator, m.CustomObjectName,
|
||
|
)
|
||
|
}
|
||
|
}
|
||
|
if len(ve.DeletedObjects) > 0 {
|
||
|
fmt.Fprintf(stdout, " DeletedObjects:\n")
|
||
|
for _, n := range ve.DeletedObjects {
|
||
|
fmt.Fprintf(stdout, " %s\n", n)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
editIdx++
|
||
|
if err := ve.Apply(&creatorID, objects); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
fmt.Fprintf(stdout, "CreatorID: %v\n", creatorID)
|
||
|
var filenums []base.DiskFileNum
|
||
|
for n := range objects {
|
||
|
filenums = append(filenums, n)
|
||
|
}
|
||
|
slices.Sort(filenums)
|
||
|
fmt.Fprintf(stdout, "Objects:\n")
|
||
|
for _, n := range filenums {
|
||
|
m := objects[n]
|
||
|
fmt.Fprintf(
|
||
|
stdout, " %s CreatorID: %s CreatorFileNum: %s Locator: %q CustomObjectName: %q\n",
|
||
|
n, m.CreatorID, m.CreatorFileNum, m.Locator, m.CustomObjectName,
|
||
|
)
|
||
|
}
|
||
|
return nil
|
||
|
}
|