backup.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. package utils
  2. import (
  3. "bufio"
  4. "bytes"
  5. "compress/gzip"
  6. "database/sql"
  7. "errors"
  8. "fmt"
  9. "io"
  10. "os"
  11. "path/filepath"
  12. "github.com/schollz/sqlite3dump"
  13. log "github.com/sirupsen/logrus"
  14. )
  15. // Restore will attempt to restore the database using a specified backup file.
  16. func Restore(backupFile string, databaseFile string) error {
  17. log.Printf("Restoring database backup %s to %s", backupFile, databaseFile)
  18. data, err := os.ReadFile(backupFile) // nolint
  19. if err != nil {
  20. return fmt.Errorf("unable to read backup file %s", err)
  21. }
  22. gz, err := gzip.NewReader(bytes.NewBuffer(data))
  23. if err != nil {
  24. return fmt.Errorf("unable to read backup file %s", err)
  25. }
  26. defer gz.Close()
  27. var b bytes.Buffer
  28. if _, err := io.Copy(&b, gz); err != nil { // nolint
  29. return fmt.Errorf("unable to read backup file %s", err)
  30. }
  31. defer gz.Close()
  32. rawSQL := b.String()
  33. // nolint:gosec
  34. if _, err := os.Create(databaseFile); err != nil {
  35. return errors.New("unable to write restored database")
  36. }
  37. // Create a new database by executing the raw SQL
  38. db, err := sql.Open("sqlite3", databaseFile)
  39. if err != nil {
  40. return err
  41. }
  42. if _, err := db.Exec(rawSQL); err != nil {
  43. return err
  44. }
  45. return nil
  46. }
  47. // Backup will backup the provided instance of the database to the specified file.
  48. func Backup(db *sql.DB, backupFile string) {
  49. log.Traceln("Backing up database to", backupFile)
  50. backupDirectory := filepath.Dir(backupFile)
  51. if !DoesFileExists(backupDirectory) {
  52. err := os.MkdirAll(backupDirectory, 0o700)
  53. if err != nil {
  54. log.Errorln("unable to create backup directory. check permissions and ownership.", backupDirectory, err)
  55. return
  56. }
  57. }
  58. // Dump the entire database as plain text sql
  59. var b bytes.Buffer
  60. out := bufio.NewWriter(&b)
  61. if err := sqlite3dump.DumpDB(db, out); err != nil {
  62. handleError(err)
  63. return
  64. }
  65. _ = out.Flush()
  66. // Create a new backup file
  67. f, err := os.OpenFile(backupFile, os.O_WRONLY|os.O_CREATE, 0o600) // nolint
  68. if err != nil {
  69. handleError(err)
  70. return
  71. }
  72. // Create a gzip compression writer
  73. w, err := gzip.NewWriterLevel(f, gzip.BestCompression)
  74. if err != nil {
  75. handleError(err)
  76. return
  77. }
  78. // Write compressed data
  79. if _, err := w.Write(b.Bytes()); err != nil {
  80. handleError(err)
  81. return
  82. }
  83. if err := w.Close(); err != nil {
  84. handleError(err)
  85. return
  86. }
  87. }
  88. func handleError(err error) {
  89. log.Errorln("unable to backup owncast database to file", err)
  90. }