restic/cmd/restic/cmd_cat.go

179 lines
2.8 KiB
Go
Raw Normal View History

2014-10-05 12:44:59 +00:00
package main
import (
"encoding/json"
"errors"
"fmt"
"os"
2014-12-05 20:45:49 +00:00
"github.com/restic/restic"
"github.com/restic/restic/backend"
2014-10-05 12:44:59 +00:00
)
2014-12-07 15:30:52 +00:00
type CmdCat struct{}
2014-11-30 21:39:58 +00:00
func init() {
2014-12-07 15:30:52 +00:00
_, err := parser.AddCommand("cat",
"dump something",
"The cat command dumps data structures or data from a repository",
&CmdCat{})
if err != nil {
panic(err)
}
}
func (cmd CmdCat) Usage() string {
return "[blob|tree|snapshot|key|lock] ID"
2014-11-30 21:39:58 +00:00
}
2014-12-07 15:30:52 +00:00
func (cmd CmdCat) Execute(args []string) error {
2014-10-05 12:44:59 +00:00
if len(args) != 2 {
2014-12-07 15:30:52 +00:00
return fmt.Errorf("type or ID not specified, Usage: %s", cmd.Usage())
}
be, key, err := OpenRepo()
if err != nil {
return err
2014-10-05 12:44:59 +00:00
}
tpe := args[0]
id, err := backend.ParseID(args[1])
if err != nil {
id = nil
if tpe != "snapshot" {
return err
}
// find snapshot id with prefix
id, err = backend.Find(be, backend.Snapshot, args[1])
if err != nil {
return err
}
2014-10-05 12:44:59 +00:00
}
2014-12-05 20:45:49 +00:00
ch, err := restic.NewContentHandler(be, key)
2014-10-05 12:44:59 +00:00
if err != nil {
return err
}
2014-11-23 21:26:01 +00:00
err = ch.LoadAllMaps()
2014-10-05 12:44:59 +00:00
if err != nil {
return err
}
switch tpe {
case "blob":
// try id
data, err := ch.Load(backend.Data, id)
2014-10-05 12:44:59 +00:00
if err == nil {
_, err = os.Stdout.Write(data)
return err
}
// try storage id
buf, err := be.Get(backend.Data, id)
2014-10-05 12:44:59 +00:00
if err != nil {
return err
}
// decrypt
buf, err = key.Decrypt(buf)
if err != nil {
return err
}
_, err = os.Stdout.Write(buf)
2014-10-05 12:44:59 +00:00
return err
case "tree":
2014-12-05 20:45:49 +00:00
var tree restic.Tree
2014-10-05 12:44:59 +00:00
// try id
err := ch.LoadJSON(backend.Tree, id, &tree)
if err != nil {
// try storage id
buf, err := be.Get(backend.Tree, id)
if err != nil {
return err
}
// decrypt
buf, err = key.Decrypt(buf)
if err != nil {
return err
}
// unmarshal
err = json.Unmarshal(backend.Uncompress(buf), &tree)
if err != nil {
return err
}
}
buf, err := json.MarshalIndent(&tree, "", " ")
if err != nil {
return err
}
fmt.Println(string(buf))
2014-11-23 21:26:01 +00:00
return nil
case "map":
2014-12-05 20:45:49 +00:00
var bl restic.BlobList
2014-11-23 21:26:01 +00:00
err := ch.LoadJSONRaw(backend.Map, id, &bl)
if err != nil {
return err
}
buf, err := json.MarshalIndent(&bl, "", " ")
if err != nil {
return err
}
fmt.Println(string(buf))
2014-10-05 12:44:59 +00:00
return nil
case "snapshot":
2014-12-05 20:45:49 +00:00
var sn restic.Snapshot
err = ch.LoadJSONRaw(backend.Snapshot, id, &sn)
2014-10-05 12:44:59 +00:00
if err != nil {
return err
}
buf, err := json.MarshalIndent(&sn, "", " ")
if err != nil {
return err
}
fmt.Println(string(buf))
return nil
case "key":
data, err := be.Get(backend.Key, id)
if err != nil {
return err
}
2014-12-05 20:45:49 +00:00
var key restic.Key
2014-10-05 12:44:59 +00:00
err = json.Unmarshal(data, &key)
if err != nil {
return err
}
buf, err := json.MarshalIndent(&key, "", " ")
if err != nil {
return err
}
fmt.Println(string(buf))
return nil
case "lock":
return errors.New("not yet implemented")
default:
return errors.New("invalid type")
}
}