mirror of https://github.com/perkeep/perkeep.git
241 lines
5.8 KiB
Go
241 lines
5.8 KiB
Go
/*
|
|
Copyright 2011 Google Inc.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
// Usage:
|
|
//
|
|
// Writes to stdout by default:
|
|
// camget BLOBREF
|
|
//
|
|
// Like curl, lets you set output file/directory with -o:
|
|
// camget -o dir BLOBREF (if dir exists and is directory, BLOBREF must be a directory, and -f to overwrite any files)
|
|
// camget -o file BLOBREF
|
|
//
|
|
// Should be possible to get a directory JSON blob without recursively
|
|
// fetching an entire directory. Likewise with files. But default
|
|
// should be sensitive on the type of the listed blob. Maybe --blob
|
|
// just to get the blob? Seems consistent.
|
|
|
|
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"flag"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
|
|
"camlistore.org/pkg/blobref"
|
|
"camlistore.org/pkg/client"
|
|
"camlistore.org/pkg/index"
|
|
"camlistore.org/pkg/schema"
|
|
)
|
|
|
|
var (
|
|
flagVerbose = flag.Bool("verbose", false, "be verbose")
|
|
flagCheck = flag.Bool("check", false, "just check for the existence of listed blobs; returning 0 if all our present")
|
|
flagOutput = flag.String("o", "-", "Output file/directory to create. Use -f to overwrite.")
|
|
flagVia = flag.String("via", "", "Fetch the blob via the given comma-separated sharerefs (dev only).")
|
|
flagGraph = flag.Bool("graph", false, "Output a graphviz directed graph .dot file of the provided root schema blob, to be rendered with 'dot -Tsvg -o graph.svg graph.dot'")
|
|
)
|
|
|
|
var viaRefs []*blobref.BlobRef
|
|
|
|
func main() {
|
|
client.AddFlags()
|
|
flag.Parse()
|
|
|
|
if len(*flagVia) > 0 {
|
|
vs := strings.Split(*flagVia, ",")
|
|
viaRefs = make([]*blobref.BlobRef, len(vs))
|
|
for i, sbr := range vs {
|
|
viaRefs[i] = blobref.Parse(sbr)
|
|
if viaRefs[i] == nil {
|
|
log.Fatalf("Invalid -via blobref: %q", sbr)
|
|
}
|
|
if *flagVerbose {
|
|
log.Printf("via: %s", sbr)
|
|
}
|
|
}
|
|
}
|
|
|
|
if *flagGraph && flag.NArg() != 1 {
|
|
log.Fatalf("The --graph option requires exactly one parameter.")
|
|
}
|
|
|
|
cl := client.NewOrFail()
|
|
|
|
for n := 0; n < flag.NArg(); n++ {
|
|
arg := flag.Arg(n)
|
|
br := blobref.Parse(arg)
|
|
if br == nil {
|
|
log.Fatalf("Failed to parse argument %q as a blobref.", arg)
|
|
}
|
|
if *flagGraph {
|
|
printGraph(cl, br)
|
|
return
|
|
}
|
|
if *flagCheck {
|
|
// TODO: do HEAD requests checking if the blobs exists.
|
|
log.Fatal("not implemented")
|
|
return
|
|
}
|
|
if *flagOutput == "-" {
|
|
rc, err := fetch(cl, br)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
defer rc.Close()
|
|
if _, err := io.Copy(os.Stdout, rc); err != nil {
|
|
log.Fatalf("Failed reading %q: %v", br, err)
|
|
}
|
|
return
|
|
}
|
|
if err := smartFetch(cl, *flagOutput, br); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func fetch(cl *client.Client, br *blobref.BlobRef) (r io.ReadCloser, err error) {
|
|
if *flagVerbose {
|
|
log.Printf("Fetching %s", br.String())
|
|
}
|
|
if len(viaRefs) > 0 {
|
|
r, _, err = cl.FetchVia(br, viaRefs)
|
|
} else {
|
|
r, _, err = cl.FetchStreaming(br)
|
|
}
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to fetch %q: %s", br, err)
|
|
}
|
|
return r, err
|
|
}
|
|
|
|
// A little less than the sniffer will take, so we don't truncate.
|
|
const sniffSize = 900 * 1024
|
|
|
|
// smartFetch the things that blobs point to, not just blobs. (wow)
|
|
func smartFetch(cl *client.Client, targ string, br *blobref.BlobRef) error {
|
|
if *flagVerbose {
|
|
log.Printf("Fetching %v into %q", br, targ)
|
|
}
|
|
|
|
rc, err := fetch(cl, br)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer rc.Close()
|
|
|
|
sniffer := new(index.BlobSniffer)
|
|
_, err = io.CopyN(sniffer, rc, sniffSize)
|
|
if err != nil && err != io.EOF {
|
|
return err
|
|
}
|
|
|
|
sniffer.Parse()
|
|
sc, ok := sniffer.Superset()
|
|
|
|
if !ok {
|
|
// opaque data - put it in a file
|
|
f, err := os.Create(targ)
|
|
if err != nil {
|
|
return fmt.Errorf("opaque: %v", err)
|
|
}
|
|
defer f.Close()
|
|
body, _ := sniffer.Body()
|
|
r := io.MultiReader(bytes.NewBuffer(body), rc)
|
|
_, err = io.Copy(f, r)
|
|
return err
|
|
}
|
|
|
|
sc.BlobRef = br
|
|
|
|
switch sc.Type {
|
|
case "directory":
|
|
dir := filepath.Join(targ, sc.FileName)
|
|
if err := os.MkdirAll(dir, sc.FileMode()); err != nil {
|
|
return err
|
|
}
|
|
if err := setFileMeta(dir, sc); err != nil {
|
|
log.Print(err)
|
|
}
|
|
entries := blobref.Parse(sc.Entries)
|
|
if entries == nil {
|
|
return fmt.Errorf("bad entries blobref: %v", sc.Entries)
|
|
}
|
|
return smartFetch(cl, dir, entries)
|
|
case "static-set":
|
|
// directory entries
|
|
for _, m := range sc.Members {
|
|
dref := blobref.Parse(m)
|
|
if dref == nil {
|
|
return fmt.Errorf("bad member blobref: %v", m)
|
|
}
|
|
if err := smartFetch(cl, targ, dref); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
case "file":
|
|
name := filepath.Join(targ, sc.FileName)
|
|
f, err := os.Create(name)
|
|
if err != nil {
|
|
return fmt.Errorf("file type: %v", err)
|
|
}
|
|
defer f.Close()
|
|
for _, p := range sc.Parts {
|
|
if p.BytesRef != nil {
|
|
panic("don't know how to handle BytesRef")
|
|
}
|
|
rc, err := fetch(cl, p.BlobRef)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = io.Copy(f, rc)
|
|
rc.Close()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if err := setFileMeta(name, sc); err != nil {
|
|
log.Print(err)
|
|
}
|
|
return nil
|
|
default:
|
|
return errors.New("unknown blob type: " + sc.Type)
|
|
}
|
|
panic("unreachable")
|
|
}
|
|
|
|
func setFileMeta(name string, sc *schema.Superset) error {
|
|
if err := os.Chmod(name, sc.FileMode()); err != nil {
|
|
return err
|
|
}
|
|
if err := os.Chown(name, sc.UnixOwnerId, sc.UnixGroupId); err != nil {
|
|
return err
|
|
}
|
|
t, err := time.Parse(time.RFC3339, sc.UnixMtime)
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
return os.Chtimes(name, t, t)
|
|
}
|