backup.go 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. package cmd
  2. import (
  3. "context"
  4. "fmt"
  5. "os"
  6. "path"
  7. "path/filepath"
  8. "strconv"
  9. "time"
  10. "github.com/cockroachdb/errors"
  11. "github.com/unknwon/cae/zip"
  12. "github.com/urfave/cli"
  13. "gopkg.in/ini.v1"
  14. log "unknwon.dev/clog/v2"
  15. "gogs.io/gogs/internal/conf"
  16. "gogs.io/gogs/internal/database"
  17. "gogs.io/gogs/internal/osutil"
  18. )
  19. var Backup = cli.Command{
  20. Name: "backup",
  21. Usage: "Backup files and database",
  22. Description: `Backup dumps and compresses all related files and database into zip file,
  23. which can be used for migrating Gogs to another server. The output format is meant to be
  24. portable among all supported database engines.`,
  25. Action: runBackup,
  26. Flags: []cli.Flag{
  27. stringFlag("config, c", "", "Custom configuration file path"),
  28. boolFlag("verbose, v", "Show process details"),
  29. stringFlag("tempdir, t", os.TempDir(), "Temporary directory path"),
  30. stringFlag("target", "./", "Target directory path to save backup archive"),
  31. stringFlag("archive-name", fmt.Sprintf("gogs-backup-%s.zip", time.Now().Format("20060102150405")), "Name of backup archive"),
  32. boolFlag("database-only", "Only dump database"),
  33. boolFlag("exclude-mirror-repos", "Exclude mirror repositories"),
  34. boolFlag("exclude-repos", "Exclude repositories"),
  35. },
  36. }
  37. const (
  38. currentBackupFormatVersion = 1
  39. archiveRootDir = "gogs-backup"
  40. )
  41. func runBackup(c *cli.Context) error {
  42. zip.Verbose = c.Bool("verbose")
  43. err := conf.Init(c.String("config"))
  44. if err != nil {
  45. return errors.Wrap(err, "init configuration")
  46. }
  47. conf.InitLogging(true)
  48. conn, err := database.SetEngine()
  49. if err != nil {
  50. return errors.Wrap(err, "set engine")
  51. }
  52. tmpDir := c.String("tempdir")
  53. if !osutil.Exist(tmpDir) {
  54. log.Fatal("'--tempdir' does not exist: %s", tmpDir)
  55. }
  56. rootDir, err := os.MkdirTemp(tmpDir, "gogs-backup-")
  57. if err != nil {
  58. log.Fatal("Failed to create backup root directory '%s': %v", rootDir, err)
  59. }
  60. log.Info("Backup root directory: %s", rootDir)
  61. // Metadata
  62. metaFile := path.Join(rootDir, "metadata.ini")
  63. metadata := ini.Empty()
  64. metadata.Section("").Key("VERSION").SetValue(strconv.Itoa(currentBackupFormatVersion))
  65. metadata.Section("").Key("DATE_TIME").SetValue(time.Now().String())
  66. metadata.Section("").Key("GOGS_VERSION").SetValue(conf.App.Version)
  67. if err = metadata.SaveTo(metaFile); err != nil {
  68. log.Fatal("Failed to save metadata '%s': %v", metaFile, err)
  69. }
  70. archiveName := filepath.Join(c.String("target"), c.String("archive-name"))
  71. log.Info("Packing backup files to: %s", archiveName)
  72. z, err := zip.Create(archiveName)
  73. if err != nil {
  74. log.Fatal("Failed to create backup archive '%s': %v", archiveName, err)
  75. }
  76. if err = z.AddFile(archiveRootDir+"/metadata.ini", metaFile); err != nil {
  77. log.Fatal("Failed to include 'metadata.ini': %v", err)
  78. }
  79. // Database
  80. dbDir := filepath.Join(rootDir, "db")
  81. if err = database.DumpDatabase(context.Background(), conn, dbDir, c.Bool("verbose")); err != nil {
  82. log.Fatal("Failed to dump database: %v", err)
  83. }
  84. if err = z.AddDir(archiveRootDir+"/db", dbDir); err != nil {
  85. log.Fatal("Failed to include 'db': %v", err)
  86. }
  87. if !c.Bool("database-only") {
  88. // Custom files
  89. err = addCustomDirToBackup(z)
  90. if err != nil {
  91. log.Fatal("Failed to add custom directory to backup: %v", err)
  92. }
  93. // Data files
  94. for _, dir := range []string{"ssh", "attachments", "avatars", "repo-avatars"} {
  95. dirPath := filepath.Join(conf.Server.AppDataPath, dir)
  96. if !osutil.IsDir(dirPath) {
  97. continue
  98. }
  99. if err = z.AddDir(path.Join(archiveRootDir+"/data", dir), dirPath); err != nil {
  100. log.Fatal("Failed to include 'data': %v", err)
  101. }
  102. }
  103. }
  104. // Repositories
  105. if !c.Bool("exclude-repos") && !c.Bool("database-only") {
  106. reposDump := filepath.Join(rootDir, "repositories.zip")
  107. log.Info("Dumping repositories in %q", conf.Repository.Root)
  108. if c.Bool("exclude-mirror-repos") {
  109. repos, err := database.GetNonMirrorRepositories()
  110. if err != nil {
  111. log.Fatal("Failed to get non-mirror repositories: %v", err)
  112. }
  113. reposZip, err := zip.Create(reposDump)
  114. if err != nil {
  115. log.Fatal("Failed to create %q: %v", reposDump, err)
  116. }
  117. baseDir := filepath.Base(conf.Repository.Root)
  118. for _, r := range repos {
  119. name := r.FullName() + ".git"
  120. if err := reposZip.AddDir(filepath.Join(baseDir, name), filepath.Join(conf.Repository.Root, name)); err != nil {
  121. log.Fatal("Failed to add %q: %v", name, err)
  122. }
  123. }
  124. if err = reposZip.Close(); err != nil {
  125. log.Fatal("Failed to save %q: %v", reposDump, err)
  126. }
  127. } else {
  128. if err = zip.PackTo(conf.Repository.Root, reposDump, true); err != nil {
  129. log.Fatal("Failed to dump repositories: %v", err)
  130. }
  131. }
  132. log.Info("Repositories dumped to: %s", reposDump)
  133. if err = z.AddFile(archiveRootDir+"/repositories.zip", reposDump); err != nil {
  134. log.Fatal("Failed to include %q: %v", reposDump, err)
  135. }
  136. }
  137. if err = z.Close(); err != nil {
  138. log.Fatal("Failed to save backup archive '%s': %v", archiveName, err)
  139. }
  140. _ = os.RemoveAll(rootDir)
  141. log.Info("Backup succeed! Archive is located at: %s", archiveName)
  142. log.Stop()
  143. return nil
  144. }
  145. func addCustomDirToBackup(z *zip.ZipArchive) error {
  146. customDir := conf.CustomDir()
  147. entries, err := os.ReadDir(customDir)
  148. if err != nil {
  149. return errors.Wrap(err, "list custom directory entries")
  150. }
  151. for _, e := range entries {
  152. if e.Name() == "data" {
  153. // Skip the "data" directory because it lives under the "custom" directory in
  154. // the Docker setup and will be backed up separately.
  155. log.Trace(`Skipping "data" directory in custom directory`)
  156. continue
  157. }
  158. add := z.AddFile
  159. if e.IsDir() {
  160. add = z.AddDir
  161. }
  162. err = add(
  163. fmt.Sprintf("%s/custom/%s", archiveRootDir, e.Name()),
  164. filepath.Join(customDir, e.Name()),
  165. )
  166. if err != nil {
  167. return errors.Wrapf(err, "add %q", e.Name())
  168. }
  169. }
  170. return nil
  171. }