Skip to content

Commit

Permalink
add sharded storage (sstorage) cmd (ethereum#83)
Browse files Browse the repository at this point in the history
* add cmd/storage

* add cmd/storage read sub-cmd

* add cmd/storage write sub-cmd

* add data_shard.go

* add shard_read/write for cmd/storage

* move sstorage lib files to /sstorage

Co-authored-by: Qi Zhou <[email protected]>
  • Loading branch information
qizhou and Qi Zhou authored May 16, 2022
1 parent d1dd0e1 commit bd2455a
Show file tree
Hide file tree
Showing 5 changed files with 835 additions and 0 deletions.
242 changes: 242 additions & 0 deletions cmd/sstorage/main.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,242 @@
package main

import (
"bufio"
"fmt"
"io"
"os"

"github.com/ethereum/go-ethereum/log"
"github.com/ethereum/go-ethereum/sstorage"
"github.com/mattn/go-colorable"
"github.com/mattn/go-isatty"
"github.com/spf13/cobra"
)

var (
chunkIdxLen *uint64
filenames *[]string

verbosity *int

chunkIdx *uint64
readLen *uint64
readMasked *bool

shardIdx *uint64
kvSize *uint64
kvEntries *uint64
kvIdx *uint64
)

var CreateCmd = &cobra.Command{
Use: "create",
Short: "Create a data file",
Run: runCreate,
}

var ChunkReadCmd = &cobra.Command{
Use: "chunk_read",
Short: "Read a chunk from a data file",
Run: runChunkRead,
}

var ChunkWriteCmd = &cobra.Command{
Use: "chunk_write",
Short: "Write a chunk from a data file",
Run: runChunkWrite,
}

var ShardReadCmd = &cobra.Command{
Use: "shard_read",
Short: "Read a KV from a data shard",
Run: runShardRead,
}

var ShardWriteCmd = &cobra.Command{
Use: "shard_write",
Short: "Write a value to a data shard",
Run: runShardWrite,
}

func init() {
chunkIdxLen = CreateCmd.Flags().Uint64("len", 0, "Chunk idx len to create")

filenames = rootCmd.PersistentFlags().StringArray("filename", []string{}, "Data filename")
verbosity = rootCmd.PersistentFlags().Int("verbosity", 3, "Logging verbosity: 0=silent, 1=error, 2=warn, 3=info, 4=debug, 5=detail")
chunkIdx = rootCmd.PersistentFlags().Uint64("chunk_idx", 0, "Chunk idx to start/read/write")

shardIdx = rootCmd.PersistentFlags().Uint64("shard_idx", 0, "Shard idx to read/write")
kvSize = rootCmd.PersistentFlags().Uint64("kv_size", 4096, "Shard KV size to read/write")
kvIdx = rootCmd.PersistentFlags().Uint64("kv_idx", 0, "Shard KV index to read/write")
kvEntries = rootCmd.PersistentFlags().Uint64("kv_entries", 0, "Number of KV entries in the shard")

readMasked = rootCmd.PersistentFlags().Bool("masked", false, "Read masked or not")
readLen = rootCmd.PersistentFlags().Uint64("readlen", 0, "Bytes to read (only for unmasked read)")
}

func setupLogger() {
glogger := log.NewGlogHandler(log.StreamHandler(os.Stderr, log.TerminalFormat(false)))
glogger.Verbosity(log.Lvl(*verbosity))
log.Root().SetHandler(glogger)

// setup logger
var ostream log.Handler
output := io.Writer(os.Stderr)

usecolor := (isatty.IsTerminal(os.Stderr.Fd()) || isatty.IsCygwinTerminal(os.Stderr.Fd())) && os.Getenv("TERM") != "dumb"
if usecolor {
output = colorable.NewColorableStderr()
}
ostream = log.StreamHandler(output, log.TerminalFormat(usecolor))

glogger.SetHandler(ostream)
}

func runCreate(cmd *cobra.Command, args []string) {
setupLogger()

if len(*filenames) != 1 {
log.Crit("must provide single filename")
}

_, err := sstorage.Create((*filenames)[0], *chunkIdx, *chunkIdxLen, sstorage.MASK_KECCAK_256)
if err != nil {
log.Crit("create failed", "error", err)
}
}

func runChunkRead(cmd *cobra.Command, args []string) {
setupLogger()

if len(*filenames) != 1 {
log.Crit("must provide a filename")
}

var err error
var df *sstorage.DataFile
df, err = sstorage.OpenDataFile((*filenames)[0])
if err != nil {
log.Crit("open failed", "error", err)
}

var b []byte
if *readMasked {
b, err = df.ReadMasked(*chunkIdx)
} else {
b, err = df.ReadUnmasked(*chunkIdx, int(*readLen))
}
if err != nil {
log.Crit("open failed", "error", err)
}
os.Stdout.Write(b)
}

func readInputBytes() []byte {
in := bufio.NewReader(os.Stdin)
b := make([]byte, 0)
for {
c, err := in.ReadByte()
if err == io.EOF {
break
}
b = append(b, c)
}
return b
}

func runChunkWrite(cmd *cobra.Command, args []string) {
setupLogger()

if len(*filenames) != 1 {
log.Crit("must provide a filename")
}

var err error
var df *sstorage.DataFile
df, err = sstorage.OpenDataFile((*filenames)[0])
if err != nil {
log.Crit("open failed", "error", err)
}

err = df.WriteUnmasked(*chunkIdx, readInputBytes())
if err != nil {
log.Crit("write failed", "error", err)
}
}

func runShardRead(cmd *cobra.Command, args []string) {
setupLogger()

ds := sstorage.NewDataShard(*shardIdx, *kvSize, *kvEntries)
for _, filename := range *filenames {
var err error
var df *sstorage.DataFile
df, err = sstorage.OpenDataFile(filename)
if err != nil {
log.Crit("open failed", "error", err)
}
ds.AddDataFile(df)
}

if !ds.IsComplete() {
log.Warn("shard is not completed")
}

var b []byte
var err error
if *readMasked {
b, err = ds.ReadMasked(*kvIdx)
} else {
b, err = ds.ReadUnmasked(*kvIdx, int(*readLen))
}
if err != nil {
log.Crit("read failed", "error", err)
}
os.Stdout.Write(b)
}

func runShardWrite(cmd *cobra.Command, args []string) {
setupLogger()

ds := sstorage.NewDataShard(*shardIdx, *kvSize, *kvEntries)
for _, filename := range *filenames {
var err error
var df *sstorage.DataFile
df, err = sstorage.OpenDataFile(filename)
if err != nil {
log.Crit("open failed", "error", err)
}
ds.AddDataFile(df)
}

if !ds.IsComplete() {
log.Warn("shard is not completed")
}

err := ds.WriteUnmasked(*kvIdx, readInputBytes())
if err != nil {
log.Crit("write failed", "error", err)
}
}

// rootCmd represents the base command when called without any subcommands
var rootCmd = &cobra.Command{
Use: "sstorage",
Short: "Sharded storage tools",
}

func init() {
rootCmd.AddCommand(CreateCmd)
rootCmd.AddCommand(ChunkReadCmd)
rootCmd.AddCommand(ChunkWriteCmd)
rootCmd.AddCommand(ShardReadCmd)
rootCmd.AddCommand(ShardWriteCmd)
}

func main() {
if err := rootCmd.Execute(); err != nil {
fmt.Println(err)
os.Exit(1)
}
}
Loading

0 comments on commit bd2455a

Please sign in to comment.