func runClusterBackup(args *docopt.Args) error { client, err := getClusterClient() if err != nil { return err } var bar *pb.ProgressBar if term.IsTerminal(os.Stderr.Fd()) { bar = pb.New(0) bar.SetUnits(pb.U_BYTES) bar.ShowBar = false bar.ShowSpeed = true bar.Output = os.Stderr bar.Start() } var dest io.Writer = os.Stdout if filename := args.String["--file"]; filename != "" { f, err := os.Create(filename) if err != nil { return err } defer f.Close() dest = f } fmt.Fprintln(os.Stderr, "Creating cluster backup...") if bar != nil { dest = io.MultiWriter(dest, bar) } if err := backup.Run(client, dest); err != nil { return err } if bar != nil { bar.Finish() } fmt.Fprintln(os.Stderr, "Backup complete.") return nil }
func runExport(args *docopt.Args, client *controller.Client) error { var dest io.Writer = os.Stdout if filename := args.String["--file"]; filename != "" { f, err := os.Create(filename) if err != nil { return fmt.Errorf("error creating export file: %s", err) } defer f.Close() dest = f } app, err := client.GetApp(mustApp()) if err != nil { return fmt.Errorf("error getting app: %s", err) } tw := backup.NewTarWriter(app.Name, dest) defer tw.Close() if err := tw.WriteJSON("app.json", app); err != nil { return fmt.Errorf("error exporting app: %s", err) } routes, err := client.RouteList(mustApp()) if err != nil { return fmt.Errorf("error getting routes: %s", err) } if err := tw.WriteJSON("routes.json", routes); err != nil { return fmt.Errorf("error exporting routes: %s", err) } release, err := client.GetAppRelease(mustApp()) if err != nil && err != controller.ErrNotFound { return fmt.Errorf("error retrieving app: %s", err) } else if err == nil { // Do not allow the exporting of passwords. delete(release.Env, "REDIS_PASSWORD") if err := tw.WriteJSON("release.json", release); err != nil { return fmt.Errorf("error exporting release: %s", err) } } artifact, err := client.GetArtifact(release.ArtifactID) if err != nil && err != controller.ErrNotFound { return fmt.Errorf("error retrieving artifact: %s", err) } else if err == nil { if err := tw.WriteJSON("artifact.json", artifact); err != nil { return fmt.Errorf("error exporting artifact: %s", err) } } formation, err := client.GetFormation(mustApp(), release.ID) if err != nil && err != controller.ErrNotFound { return fmt.Errorf("error retrieving formation: %s", err) } else if err == nil { if err := tw.WriteJSON("formation.json", formation); err != nil { return fmt.Errorf("error exporting formation: %s", err) } } var bar *pb.ProgressBar if !args.Bool["--quiet"] && term.IsTerminal(os.Stderr.Fd()) { bar = pb.New(0) bar.SetUnits(pb.U_BYTES) bar.ShowBar = false bar.ShowSpeed = true bar.Output = os.Stderr bar.Start() defer bar.Finish() } if slug, ok := release.Env["SLUG_URL"]; ok { reqR, reqW := io.Pipe() config := runConfig{ App: mustApp(), Release: release.ID, DisableLog: true, Entrypoint: []string{"curl"}, Args: []string{"--include", "--raw", slug}, Stdout: reqW, Stderr: ioutil.Discard, } if bar != nil { config.Stdout = io.MultiWriter(config.Stdout, bar) } go func() { if err := runJob(client, config); err != nil { shutdown.Fatalf("error retrieving slug: %s", err) } }() res, err := http.ReadResponse(bufio.NewReader(reqR), nil) if err != nil { return fmt.Errorf("error reading slug response: %s", err) } if res.StatusCode != 200 { return fmt.Errorf("unexpected status getting slug: %d", res.StatusCode) } length, err := strconv.Atoi(res.Header.Get("Content-Length")) if err != nil { return fmt.Errorf("slug has missing or malformed Content-Length") } if err := tw.WriteHeader("slug.tar.gz", length); err != nil { return fmt.Errorf("error writing slug header: %s", err) } if _, err := io.Copy(tw, res.Body); err != nil { return fmt.Errorf("error writing slug: %s", err) } res.Body.Close() } if config, err := getAppPgRunConfig(client); err == nil { configPgDump(config) if err := tw.WriteCommandOutput(client, "postgres.dump", config.App, &ct.NewJob{ ReleaseID: config.Release, Entrypoint: config.Entrypoint, Cmd: config.Args, Env: config.Env, DisableLog: config.DisableLog, }); err != nil { return fmt.Errorf("error creating postgres dump: %s", err) } } return nil }
func runExport(args *docopt.Args, client *controller.Client) error { var dest io.Writer = os.Stdout if filename := args.String["--file"]; filename != "" { f, err := os.Create(filename) if err != nil { return fmt.Errorf("error creating export file: %s", err) } defer f.Close() dest = f } tw := tar.NewWriter(dest) defer tw.Close() uid := syscall.Getuid() header := func(name string, length int) error { return tw.WriteHeader(&tar.Header{ Name: path.Join(mustApp(), name), Mode: 0644, Size: int64(length), ModTime: time.Now(), Typeflag: tar.TypeReg, Uid: uid, Gid: uid, }) } writeJSON := func(name string, v interface{}) error { data, err := json.MarshalIndent(v, "", " ") if err != nil { return err } if err := header(name, len(data)+1); err != nil { return err } if _, err := tw.Write(data); err != nil { return err } if _, err := tw.Write([]byte("\n")); err != nil { return err } return nil } app, err := client.GetApp(mustApp()) if err != nil { return fmt.Errorf("error getting app: %s", err) } if err := writeJSON("app.json", app); err != nil { return fmt.Errorf("error exporting app: %s", err) } release, err := client.GetAppRelease(mustApp()) if err != nil && err != controller.ErrNotFound { return fmt.Errorf("error retrieving app: %s", err) } else if err == nil { if err := writeJSON("release.json", release); err != nil { return fmt.Errorf("error exporting release: %s", err) } } artifact, err := client.GetArtifact(release.ArtifactID) if err != nil && err != controller.ErrNotFound { return fmt.Errorf("error retrieving artifact: %s", err) } else if err == nil { if err := writeJSON("artifact.json", artifact); err != nil { return fmt.Errorf("error exporting artifact: %s", err) } } formation, err := client.GetFormation(mustApp(), release.ID) if err != nil && err != controller.ErrNotFound { return fmt.Errorf("error retrieving formation: %s", err) } else if err == nil { if err := writeJSON("formation.json", formation); err != nil { return fmt.Errorf("error exporting formation: %s", err) } } var bar *pb.ProgressBar if !args.Bool["--quiet"] && term.IsTerminal(os.Stderr.Fd()) { bar = pb.New(0) bar.SetUnits(pb.U_BYTES) bar.ShowBar = false bar.ShowSpeed = true bar.Output = os.Stderr bar.Start() defer bar.Finish() } if slug, ok := release.Env["SLUG_URL"]; ok { reqR, reqW := io.Pipe() config := runConfig{ App: mustApp(), Release: release.ID, DisableLog: true, Entrypoint: []string{"curl"}, Args: []string{"--include", "--raw", slug}, Stdout: reqW, Stderr: ioutil.Discard, } if bar != nil { config.Stdout = io.MultiWriter(config.Stdout, bar) } go func() { if err := runJob(client, config); err != nil { shutdown.Fatalf("error retrieving slug: %s", err) } }() res, err := http.ReadResponse(bufio.NewReader(reqR), nil) if err != nil { return fmt.Errorf("error reading slug response: %s", err) } if res.StatusCode != 200 { return fmt.Errorf("unexpected status getting slug: %d", res.StatusCode) } length, err := strconv.Atoi(res.Header.Get("Content-Length")) if err != nil { return fmt.Errorf("slug has missing or malformed Content-Length") } if err := header("slug.tar.gz", length); err != nil { return fmt.Errorf("error writing slug header: %s", err) } if _, err := io.Copy(tw, res.Body); err != nil { return fmt.Errorf("error writing slug: %s", err) } res.Body.Close() } if config, err := getAppPgRunConfig(client); err == nil { f, err := ioutil.TempFile("", "postgres.dump") if err != nil { return fmt.Errorf("error creating db temp file: %s", err) } defer f.Close() defer os.Remove(f.Name()) config.Stdout = f config.Exit = false if bar != nil { config.Stdout = io.MultiWriter(config.Stdout, bar) } if err := pgDump(client, config); err != nil { return fmt.Errorf("error dumping database: %s", err) } length, err := f.Seek(0, os.SEEK_CUR) if err != nil { return fmt.Errorf("error getting db size: %s", err) } if err := header("postgres.dump", int(length)); err != nil { return fmt.Errorf("error writing db header: %s", err) } if _, err := f.Seek(0, os.SEEK_SET); err != nil { return fmt.Errorf("error seeking db dump: %s", err) } if _, err := io.Copy(tw, f); err != nil { return fmt.Errorf("error exporting db: %s", err) } } return nil }
func runClusterBackup(args *docopt.Args) error { client, err := getClusterClient() if err != nil { return err } var bar *pb.ProgressBar if term.IsTerminal(os.Stderr.Fd()) { bar = pb.New(0) bar.SetUnits(pb.U_BYTES) bar.ShowBar = false bar.ShowSpeed = true bar.Output = os.Stderr bar.Start() } var dest io.Writer = os.Stdout if filename := args.String["--file"]; filename != "" { f, err := os.Create(filename) if err != nil { return err } defer f.Close() dest = f } fmt.Fprintln(os.Stderr, "Creating cluster backup...") tw := NewTarWriter("flynn-backup-"+time.Now().UTC().Format("2006-01-02_150405"), dest) defer tw.Close() // get app and release details for key apps data := make(map[string]*ct.ExpandedFormation, 4) for _, name := range []string{"postgres", "discoverd", "flannel", "controller"} { app, err := client.GetApp(name) if err != nil { return fmt.Errorf("error getting %s app details: %s", name, err) } release, err := client.GetAppRelease(app.ID) if err != nil { return fmt.Errorf("error getting %s app release: %s", name, err) } formation, err := client.GetFormation(app.ID, release.ID) if err != nil { return fmt.Errorf("error getting %s app formation: %s", name, err) } artifact, err := client.GetArtifact(release.ArtifactID) if err != nil { return fmt.Errorf("error getting %s app artifact: %s", name, err) } data[name] = &ct.ExpandedFormation{ App: app, Release: release, Artifact: artifact, Processes: formation.Processes, } } if err := tw.WriteJSON("flynn.json", data); err != nil { return err } config := &runConfig{ App: "postgres", Release: data["postgres"].Release.ID, Entrypoint: []string{"sh"}, Args: []string{"-c", "pg_dumpall --clean --if-exists | gzip -9"}, Env: map[string]string{ "PGHOST": "leader.postgres.discoverd", "PGUSER": "******", "PGPASSWORD": data["postgres"].Release.Env["PGPASSWORD"], }, DisableLog: true, } if err := tw.WriteCommandOutput(client, "postgres.sql.gz", config, bar); err != nil { return fmt.Errorf("error dumping database: %s", err) } if bar != nil { bar.Finish() } fmt.Fprintln(os.Stderr, "Backup complete.") return nil }