Skip to content

Commit

Permalink
Fix progress bar for "get" command
Browse files Browse the repository at this point in the history
License: MIT
Signed-off-by: rht <rhtbot@gmail.com>
  • Loading branch information
rht authored and rht committed Jun 15, 2015
1 parent 125717e commit 703d6de
Show file tree
Hide file tree
Showing 2 changed files with 58 additions and 8 deletions.
24 changes: 16 additions & 8 deletions core/commands/get.go
Original file line number Diff line number Diff line change
Expand Up @@ -63,11 +63,12 @@ may also specify the level of compression by specifying '-l=<1-9>'.
return
}

reader, err := get(req.Context().Context, node, req.Arguments()[0], cmplvl)
reader, length, err := get(req.Context().Context, node, req.Arguments()[0], cmplvl)
if err != nil {
res.SetError(err, cmds.ErrNormal)
return
}
res.SetLength(length)
res.SetOutput(reader)
},
PostRun: func(req cmds.Request, res cmds.Response) {
Expand Down Expand Up @@ -105,6 +106,8 @@ may also specify the level of compression by specifying '-l=<1-9>'.
}
defer file.Close()

// bar length is set to 0 because there is no deterministic way to
// pre-determine the size of the .tar.gz output
bar := pb.New(0).SetUnits(pb.U_BYTES)
bar.Output = os.Stderr
pbReader := bar.NewProxyReader(outReader)
Expand All @@ -122,8 +125,7 @@ may also specify the level of compression by specifying '-l=<1-9>'.

fmt.Printf("Saving file(s) to %s\n", outPath)

// TODO: get total length of files
bar := pb.New(0).SetUnits(pb.U_BYTES)
bar := pb.New64(int64(res.Length())).SetUnits(pb.U_BYTES)
bar.Output = os.Stderr

// wrap the reader with the progress bar proxy reader
Expand All @@ -145,8 +147,7 @@ may also specify the level of compression by specifying '-l=<1-9>'.
defer bar.Finish()

extractor := &tar.Extractor{outPath}
err = extractor.Extract(reader)
if err != nil {
if err = extractor.Extract(reader); err != nil {
res.SetError(err, cmds.ErrNormal)
}
},
Expand All @@ -166,12 +167,19 @@ func getCompressOptions(req cmds.Request) (int, error) {
return gzip.NoCompression, nil
}

func get(ctx context.Context, node *core.IpfsNode, p string, compression int) (io.Reader, error) {
func get(ctx context.Context, node *core.IpfsNode, p string, compression int) (io.Reader, uint64, error) {
pathToResolve := path.Path(p)
dagnode, err := core.Resolve(ctx, node, pathToResolve)
if err != nil {
return nil, err
return nil, 0, err
}

reader, err := utar.NewReader(pathToResolve, node.DAG, dagnode, compression)

length, err1 := utar.GetTarSize(node.DAG, dagnode)
if err1 != nil {
return nil, 0, err1
}

return utar.NewReader(pathToResolve, node.DAG, dagnode, compression)
return reader, length, err
}
42 changes: 42 additions & 0 deletions unixfs/tar/reader.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,8 @@ import (
upb "github.com/ipfs/go-ipfs/unixfs/pb"
)

const tarBlockSize = 512

type Reader struct {
buf bytes.Buffer
closed bool
Expand Down Expand Up @@ -202,3 +204,43 @@ func (r *Reader) syncCopy(reader io.Reader) error {
}
return nil
}

func GetTarSize(dag mdag.DAGService, dagnode *mdag.Node) (uint64, error) {
return _GetTarSize(dag, dagnode, true)
}

func _GetTarSize(dag mdag.DAGService, dagnode *mdag.Node, isparent bool) (uint64, error) {
size := uint64(0)

pb := new(upb.Data)
if err := proto.Unmarshal(dagnode.Data, pb); err != nil {
return 0, err
}

if pb.GetType() == upb.Data_Directory {
//size += tarBlockSize

ctx, cancel := context.WithTimeout(context.TODO(), time.Second*60)
defer cancel()
for _, ng := range dag.GetDAG(ctx, dagnode) {
childNode, err := ng.Get(ctx)
if err != nil {
return 0, err
}
childSize, err := _GetTarSize(dag, childNode, false)
if err != nil {
return 0, err
}
size += childSize
}
} else {
unixSize := pb.GetFilesize()
// tar header + file size + round up to nearest 512 bytes
size = tarBlockSize + unixSize + (tarBlockSize - unixSize%tarBlockSize)
}

if isparent {
size += 2 * tarBlockSize // tar parent padding
}
return size, nil
}

0 comments on commit 703d6de

Please sign in to comment.