restic/cmd/restic/cmd_key.go

148 lines
3.0 KiB
Go
Raw Normal View History

2014-11-25 21:52:53 +00:00
package main
import (
"errors"
"fmt"
2014-12-05 20:45:49 +00:00
"github.com/restic/restic/backend"
"github.com/restic/restic/repository"
2014-11-25 21:52:53 +00:00
)
type CmdKey struct {
2015-06-21 13:01:52 +00:00
global *GlobalOptions
newPassword string
}
2014-12-07 15:30:52 +00:00
2014-11-30 21:39:58 +00:00
func init() {
2014-12-07 15:30:52 +00:00
_, err := parser.AddCommand("key",
"manage keys",
"The key command manages keys (passwords) of a repository",
&CmdKey{global: &globalOpts})
2014-12-07 15:30:52 +00:00
if err != nil {
panic(err)
}
2014-11-30 21:39:58 +00:00
}
func (cmd CmdKey) listKeys(s *repository.Repository) error {
2014-11-25 21:52:53 +00:00
tab := NewTable()
2014-11-27 22:26:19 +00:00
tab.Header = fmt.Sprintf(" %-10s %-10s %-10s %s", "ID", "User", "Host", "Created")
tab.RowFormat = "%s%-10s %-10s %-10s %s"
2014-11-25 21:52:53 +00:00
2014-12-21 17:10:19 +00:00
plen, err := s.PrefixLength(backend.Key)
2014-11-25 21:52:53 +00:00
if err != nil {
return err
}
2015-03-28 10:50:23 +00:00
done := make(chan struct{})
defer close(done)
for id := range s.List(backend.Key, done) {
k, err := repository.LoadKey(s, id.String())
2014-11-25 21:52:53 +00:00
if err != nil {
2015-06-21 11:27:56 +00:00
cmd.global.Warnf("LoadKey() failed: %v\n", err)
2015-03-28 10:50:23 +00:00
continue
2014-11-25 21:52:53 +00:00
}
2014-11-27 22:26:19 +00:00
var current string
if id.String() == s.KeyName() {
2014-11-27 22:26:19 +00:00
current = "*"
} else {
current = " "
}
tab.Rows = append(tab.Rows, []interface{}{current, id.String()[:plen],
2014-11-25 21:52:53 +00:00
k.Username, k.Hostname, k.Created.Format(TimeFormat)})
2015-03-28 10:50:23 +00:00
}
2014-11-25 21:52:53 +00:00
2015-06-21 11:27:56 +00:00
return tab.Write(cmd.global.stdout)
2014-11-25 21:52:53 +00:00
}
2015-06-21 13:01:52 +00:00
func (cmd CmdKey) getNewPassword() string {
if cmd.newPassword != "" {
return cmd.newPassword
2014-11-25 22:07:00 +00:00
}
2015-06-21 13:01:52 +00:00
return cmd.global.ReadPasswordTwice(
"enter password for new key: ",
"enter password again: ")
}
func (cmd CmdKey) addKey(repo *repository.Repository) error {
2015-06-21 13:01:52 +00:00
id, err := repository.AddKey(repo, cmd.getNewPassword(), repo.Key())
2014-11-25 22:07:00 +00:00
if err != nil {
return fmt.Errorf("creating new key failed: %v\n", err)
}
2015-06-21 11:27:56 +00:00
cmd.global.Verbosef("saved new key as %s\n", id)
2014-11-25 22:07:00 +00:00
return nil
}
func (cmd CmdKey) deleteKey(repo *repository.Repository, name string) error {
2015-05-09 11:32:52 +00:00
if name == repo.KeyName() {
2014-11-25 22:18:02 +00:00
return errors.New("refusing to remove key currently used to access repository")
}
2015-05-17 18:52:31 +00:00
err := repo.Backend().Remove(backend.Key, name)
2014-11-25 22:18:02 +00:00
if err != nil {
return err
}
2015-06-21 11:27:56 +00:00
cmd.global.Verbosef("removed key %v\n", name)
2014-11-25 22:18:02 +00:00
return nil
}
func (cmd CmdKey) changePassword(repo *repository.Repository) error {
2015-06-21 13:01:52 +00:00
id, err := repository.AddKey(repo, cmd.getNewPassword(), repo.Key())
if err != nil {
return fmt.Errorf("creating new key failed: %v\n", err)
}
err = repo.Backend().Remove(backend.Key, repo.KeyName())
if err != nil {
return err
}
2015-06-21 11:27:56 +00:00
cmd.global.Verbosef("saved new key as %s\n", id)
return nil
}
2014-12-07 15:30:52 +00:00
func (cmd CmdKey) Usage() string {
2015-04-25 06:42:52 +00:00
return "[list|add|rm|passwd] [ID]"
2014-12-07 15:30:52 +00:00
}
func (cmd CmdKey) Execute(args []string) error {
2014-11-25 22:18:02 +00:00
if len(args) < 1 || (args[0] == "rm" && len(args) != 2) {
2014-12-07 15:30:52 +00:00
return fmt.Errorf("wrong number of arguments, Usage: %s", cmd.Usage())
}
repo, err := cmd.global.OpenRepository()
2014-12-07 15:30:52 +00:00
if err != nil {
return err
2014-11-25 21:52:53 +00:00
}
lock, err := lockRepoExclusive(repo)
defer unlockRepo(lock)
2015-06-27 12:40:18 +00:00
if err != nil {
return err
}
2014-11-25 21:52:53 +00:00
switch args[0] {
case "list":
return cmd.listKeys(repo)
2014-11-25 22:07:00 +00:00
case "add":
return cmd.addKey(repo)
2014-11-25 22:18:02 +00:00
case "rm":
id, err := backend.Find(repo.Backend(), backend.Key, args[1])
2014-11-25 22:18:02 +00:00
if err != nil {
return err
}
return cmd.deleteKey(repo, id)
2015-04-25 06:42:52 +00:00
case "passwd":
return cmd.changePassword(repo)
2014-11-25 21:52:53 +00:00
}
return nil
}