2016-05-22 13:04:37 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"path"
|
2016-06-25 23:33:28 +00:00
|
|
|
"regexp"
|
2016-05-22 13:04:37 +00:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
2016-05-23 09:59:23 +00:00
|
|
|
"github.com/Luzifer/go_helpers/str"
|
2016-05-22 13:04:37 +00:00
|
|
|
"github.com/Luzifer/go_helpers/which"
|
|
|
|
"github.com/Luzifer/rconfig"
|
|
|
|
"github.com/mitchellh/go-homedir"
|
|
|
|
"github.com/nightlyone/lockfile"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
cfg = struct {
|
|
|
|
ConfigFile string `flag:"config-file,f" default:"~/.config/duplicity-backup.yaml" description:"Configuration for this duplicity wrapper"`
|
|
|
|
LockFile string `flag:"lock-file,l" default:"~/.config/duplicity-backup.lock" description:"File to hold the lock for this wrapper execution"`
|
|
|
|
|
|
|
|
RestoreTime string `flag:"time,t" description:"The time from which to restore or list files"`
|
|
|
|
|
2016-05-30 05:32:46 +00:00
|
|
|
DryRun bool `flag:"dry-run,n" default:"false" description:"Do a test-run without changes"`
|
|
|
|
Debug bool `flag:"debug,d" default:"false" description:"Print duplicity commands to output"`
|
2018-10-08 14:06:28 +00:00
|
|
|
Silent bool `flag:"silent,s" default:"false" description:"Do not print to stdout, only write to logfile (for example useful for crons)"`
|
2016-05-30 05:32:46 +00:00
|
|
|
|
2016-05-22 13:04:37 +00:00
|
|
|
VersionAndExit bool `flag:"version" default:"false" description:"Print version and exit"`
|
|
|
|
}{}
|
|
|
|
|
|
|
|
duplicityBinary string
|
|
|
|
logFile *os.File
|
|
|
|
|
|
|
|
version = "dev"
|
|
|
|
)
|
|
|
|
|
|
|
|
func initCFG() {
|
|
|
|
var err error
|
|
|
|
if err = rconfig.Parse(&cfg); err != nil {
|
|
|
|
log.Fatalf("Error while parsing arguments: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.VersionAndExit {
|
|
|
|
fmt.Printf("duplicity-backup %s\n", version)
|
|
|
|
os.Exit(0)
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.ConfigFile, err = homedir.Expand(cfg.ConfigFile); err != nil {
|
|
|
|
log.Fatalf("Unable to expand config-file: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.LockFile, err = homedir.Expand(cfg.LockFile); err != nil {
|
|
|
|
log.Fatalf("Unable to expand lock: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if duplicityBinary, err = which.FindInPath("duplicity"); err != nil {
|
|
|
|
log.Fatalf("Did not find duplicity binary in $PATH, please install it")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func logf(pattern string, fields ...interface{}) {
|
|
|
|
t := time.Now().Format("2006-01-02 15:04:05")
|
|
|
|
pattern = fmt.Sprintf("(%s) ", t) + pattern + "\n"
|
|
|
|
fmt.Fprintf(logFile, pattern, fields...)
|
2016-05-30 05:32:46 +00:00
|
|
|
if !cfg.Silent {
|
|
|
|
fmt.Printf(pattern, fields...)
|
|
|
|
}
|
2016-05-22 13:04:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
initCFG()
|
|
|
|
|
|
|
|
var (
|
|
|
|
err error
|
|
|
|
config *configFile
|
|
|
|
)
|
|
|
|
|
|
|
|
lock, err := lockfile.New(cfg.LockFile)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Could not initialize lockfile: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// If no command is passed assume we're requesting "help"
|
|
|
|
argv := rconfig.Args()
|
|
|
|
if len(argv) == 1 || argv[1] == "help" {
|
2018-10-08 14:06:28 +00:00
|
|
|
helptext, _ := Asset("help.txt") // #nosec G104
|
2016-05-22 13:04:37 +00:00
|
|
|
fmt.Println(string(helptext))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get configuration
|
|
|
|
configSource, err := os.Open(cfg.ConfigFile)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Unable to open configuration file %s: %s", cfg.ConfigFile, err)
|
|
|
|
}
|
|
|
|
defer configSource.Close()
|
|
|
|
config, err = loadConfigFile(configSource)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Unable to read configuration file: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize logfile
|
2018-10-08 14:06:28 +00:00
|
|
|
if err := os.MkdirAll(config.LogDirectory, 0750); err != nil {
|
|
|
|
log.Fatalf("Unable to create log dir: %s", err)
|
|
|
|
}
|
|
|
|
|
2016-05-22 13:04:37 +00:00
|
|
|
logFilePath := path.Join(config.LogDirectory, time.Now().Format("duplicity-backup_2006-01-02_15-04-05.txt"))
|
|
|
|
if logFile, err = os.Create(logFilePath); err != nil {
|
|
|
|
log.Fatalf("Unable to open logfile %s: %s", logFilePath, err)
|
|
|
|
}
|
|
|
|
defer logFile.Close()
|
|
|
|
|
2016-05-23 08:41:49 +00:00
|
|
|
logf("++++ duplicity-backup %s started with command '%s'", version, argv[1])
|
2016-05-22 13:04:37 +00:00
|
|
|
|
|
|
|
if err := lock.TryLock(); err != nil {
|
2018-10-08 14:06:28 +00:00
|
|
|
logf("Could not acquire lock: %s", err)
|
2016-05-22 13:04:37 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
defer lock.Unlock()
|
|
|
|
|
|
|
|
if err := execute(config, argv[1:]); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-05-23 09:59:23 +00:00
|
|
|
if config.Cleanup.Type != "none" && str.StringInSlice(argv[1], removeCommands) {
|
2016-05-22 13:04:37 +00:00
|
|
|
logf("++++ Starting removal of old backups")
|
|
|
|
|
2016-05-23 09:54:45 +00:00
|
|
|
if err := execute(config, []string{commandRemove}); err != nil {
|
2016-05-22 13:04:37 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-23 09:54:45 +00:00
|
|
|
if err := config.Notify(argv[1], true, nil); err != nil {
|
2016-05-23 08:41:20 +00:00
|
|
|
logf("[ERR] Error sending notifications: %s", err)
|
|
|
|
} else {
|
|
|
|
logf("[INF] Notifications sent")
|
|
|
|
}
|
|
|
|
|
2016-05-22 13:04:37 +00:00
|
|
|
logf("++++ Backup finished successfully")
|
|
|
|
}
|
|
|
|
|
|
|
|
func execute(config *configFile, argv []string) error {
|
|
|
|
var (
|
2016-05-23 08:41:20 +00:00
|
|
|
err error
|
|
|
|
commandLine, tmpEnv []string
|
2016-06-25 23:33:28 +00:00
|
|
|
logFilter *regexp.Regexp
|
2016-05-22 13:04:37 +00:00
|
|
|
)
|
2016-06-25 23:33:28 +00:00
|
|
|
commandLine, tmpEnv, logFilter, err = config.GenerateCommand(argv, cfg.RestoreTime)
|
2016-05-22 13:04:37 +00:00
|
|
|
if err != nil {
|
|
|
|
logf("[ERR] %s", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2016-05-23 08:41:20 +00:00
|
|
|
env := envListToMap(os.Environ())
|
|
|
|
for k, v := range envListToMap(tmpEnv) {
|
|
|
|
env[k] = v
|
|
|
|
}
|
|
|
|
|
2016-05-22 13:04:37 +00:00
|
|
|
// Ensure duplicity is talking to us
|
|
|
|
commandLine = append([]string{"-v3"}, commandLine...)
|
|
|
|
|
|
|
|
if cfg.DryRun {
|
|
|
|
commandLine = append([]string{"--dry-run"}, commandLine...)
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.Debug {
|
|
|
|
logf("[DBG] Command: %s %s", duplicityBinary, strings.Join(commandLine, " "))
|
|
|
|
}
|
|
|
|
|
2016-05-23 11:19:25 +00:00
|
|
|
msgChan := make(chan string, 10)
|
2016-06-25 23:33:28 +00:00
|
|
|
go func(c chan string, logFilter *regexp.Regexp) {
|
2016-05-23 11:19:25 +00:00
|
|
|
for l := range c {
|
2016-06-25 23:33:28 +00:00
|
|
|
if logFilter == nil || logFilter.MatchString(l) {
|
|
|
|
logf(l)
|
|
|
|
}
|
2016-05-23 11:19:25 +00:00
|
|
|
}
|
2016-06-25 23:33:28 +00:00
|
|
|
}(msgChan, logFilter)
|
2016-05-23 11:19:25 +00:00
|
|
|
|
2016-05-23 14:04:50 +00:00
|
|
|
output := newMessageChanWriter(msgChan)
|
2018-10-08 14:06:28 +00:00
|
|
|
cmd := exec.Command(duplicityBinary, commandLine...) // #nosec G204
|
2016-05-22 13:04:37 +00:00
|
|
|
cmd.Stdout = output
|
|
|
|
cmd.Stderr = output
|
2016-05-23 08:41:20 +00:00
|
|
|
cmd.Env = envMapToList(env)
|
2016-05-22 13:04:37 +00:00
|
|
|
err = cmd.Run()
|
|
|
|
|
2016-05-23 11:19:25 +00:00
|
|
|
close(msgChan)
|
|
|
|
|
2016-05-22 13:04:37 +00:00
|
|
|
if err != nil {
|
|
|
|
logf("[ERR] Execution of duplicity command was unsuccessful! (exit-code was non-zero)")
|
|
|
|
} else {
|
|
|
|
logf("[INF] Execution of duplicity command was successful.")
|
|
|
|
}
|
|
|
|
|
2016-05-23 09:54:45 +00:00
|
|
|
if err != nil {
|
|
|
|
if nErr := config.Notify(argv[0], false, fmt.Errorf("Could not create backup: %s", err)); nErr != nil {
|
|
|
|
logf("[ERR] Error sending notifications: %s", nErr)
|
|
|
|
} else {
|
|
|
|
logf("[INF] Notifications sent")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-22 13:04:37 +00:00
|
|
|
return err
|
|
|
|
}
|
2016-05-23 08:41:20 +00:00
|
|
|
|
|
|
|
func envListToMap(list []string) map[string]string {
|
|
|
|
out := map[string]string{}
|
|
|
|
for _, entry := range list {
|
|
|
|
if len(entry) == 0 || entry[0] == '#' {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
parts := strings.SplitN(entry, "=", 2)
|
|
|
|
out[parts[0]] = parts[1]
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
func envMapToList(envMap map[string]string) []string {
|
|
|
|
out := []string{}
|
|
|
|
for k, v := range envMap {
|
|
|
|
out = append(out, k+"="+v)
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|