storj/examples/piecestore-cli/main.go
Alexander Leitner 900f67e3d0 Implement psclient interface (#107)
* Implement psclient interface

* Add string method to pieceID type

* try to fix linter errors

* Whoops missed an error

* More linter errors

* Typo

* Lol double typo

*  Get everything working, begin adding tests for psclient rpc

* goimports

* Forgot to change the piecestore cli when changed the piecestore code

* Fix CLI

* remove ID length, added validator to pieceID

* Move grpc ranger to client
Change client PUT api to take a reader rather than return a writer

* GRPCRanger -> PieceRanger; Make PieceRanger a RangeCloser

* Forgot to remove offset

* Added message upon successful store

* Do that thing dennis and kaloyan wanted

* goimports

* Make closeConn a part of the interface for psclient

* Use interface

* Removed uneccessary new lines

* goimport

* Whoops

* Actually we don't want to use the interface in Piece Ranger

* Renamed piecestore in examples to piecestore-client; moved piecestore-cli to examples

* Make comments look nicer
2018-06-27 21:42:54 +03:00

148 lines
3.1 KiB
Go

// Copyright (C) 2018 Storj Labs, Inc.
// See LICENSE for copying information.
package main
import (
"context"
"flag"
"fmt"
"io"
"os"
"sort"
"github.com/spf13/cobra"
"github.com/urfave/cli"
"github.com/zeebo/errs"
"storj.io/storj/pkg/piecestore"
"storj.io/storj/pkg/process"
)
var argError = errs.Class("argError")
func run(ctx context.Context, _ *cobra.Command, _ []string) error {
app := cli.NewApp()
app.Name = "Piece Store CLI"
app.Usage = "Store data in hash folder structure"
app.Version = "1.0.0"
app.Flags = []cli.Flag{}
app.Commands = []cli.Command{
{
Name: "store",
Aliases: []string{"s"},
Usage: "Store data by id",
ArgsUsage: "[id] [dataPath] [storeDir]",
Action: func(c *cli.Context) error {
if c.Args().Get(0) == "" {
return argError.New("No id specified")
}
id := c.Args().Get(0)
if c.Args().Get(1) == "" {
return argError.New("No input file specified")
}
path := c.Args().Get(1)
if c.Args().Get(2) == "" {
return argError.New("No output directory specified")
}
outputDir := c.Args().Get(2)
file, err := os.Open(path)
if err != nil {
return err
}
// Close the file when we are done
defer file.Close()
fileInfo, err := os.Stat(path)
if err != nil {
return err
}
if fileInfo.IsDir() {
return argError.New(fmt.Sprintf("Path (%s) is a directory, not a file", path))
}
dataFileChunk, err := pstore.StoreWriter(id, outputDir)
if err != nil {
return err
}
// Close when finished
defer dataFileChunk.Close()
_, err = io.Copy(dataFileChunk, file)
return err
},
},
{
Name: "retrieve",
Aliases: []string{"r"},
Usage: "Retrieve data by id and print to Stdout",
ArgsUsage: "[id] [storeDir]",
Action: func(c *cli.Context) error {
if c.Args().Get(0) == "" {
return argError.New("Missing data id")
}
if c.Args().Get(1) == "" {
return argError.New("Missing file path")
}
fileInfo, err := os.Stat(c.Args().Get(1))
if err != nil {
return err
}
if fileInfo.IsDir() != true {
return argError.New(fmt.Sprintf("Path (%s) is a file, not a directory", c.Args().Get(1)))
}
dataFileChunk, err := pstore.RetrieveReader(context.Background(),
c.Args().Get(0), 0, -1, c.Args().Get(1))
if err != nil {
return err
}
// Close when finished
defer dataFileChunk.Close()
_, err = io.Copy(os.Stdout, dataFileChunk)
return err
},
},
{
Name: "delete",
Aliases: []string{"d"},
Usage: "Delete data by id",
ArgsUsage: "[id] [storeDir]",
Action: func(c *cli.Context) error {
if c.Args().Get(0) == "" {
return argError.New("Missing data id")
}
if c.Args().Get(1) == "" {
return argError.New("No directory specified")
}
err := pstore.Delete(c.Args().Get(0), c.Args().Get(1))
return err
},
},
}
sort.Sort(cli.FlagsByName(app.Flags))
sort.Sort(cli.CommandsByName(app.Commands))
return app.Run(append([]string{os.Args[0]}, flag.Args()...))
}
func main() { process.Must(process.Main(process.ServiceFunc(run))) }