main.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. package main
  2. import (
  3. "backup-x/client"
  4. "backup-x/util"
  5. "backup-x/web"
  6. "embed"
  7. "flag"
  8. "net"
  9. "os"
  10. "log"
  11. "net/http"
  12. "time"
  13. "github.com/kardianos/service"
  14. )
  15. // 监听地址
  16. var listen = flag.String("l", ":9977", "监听地址")
  17. // 服务管理
  18. var serviceType = flag.String("s", "", "服务管理, 支持install, uninstall")
  19. // 默认备份路径当前运行目录
  20. var backupDirDefault, _ = os.Getwd()
  21. // 配置文件路径
  22. var backupDir = flag.String("d", backupDirDefault, "自定义备份目录地址")
  23. //go:embed static
  24. var staticEmbededFiles embed.FS
  25. //go:embed favicon.ico
  26. var faviconEmbededFile embed.FS
  27. // version
  28. var version = "DEV"
  29. func main() {
  30. flag.Parse()
  31. if _, err := net.ResolveTCPAddr("tcp", *listen); err != nil {
  32. log.Fatalf("解析监听地址异常,%s", err)
  33. }
  34. os.Setenv(web.VersionEnv, version)
  35. switch *serviceType {
  36. case "install":
  37. installService()
  38. case "uninstall":
  39. uninstallService()
  40. default:
  41. if util.IsRunInDocker() {
  42. run(100 * time.Millisecond)
  43. } else {
  44. s := getService()
  45. status, _ := s.Status()
  46. if status != service.StatusUnknown {
  47. // 以服务方式运行
  48. s.Run()
  49. } else {
  50. // 非服务方式运行
  51. switch s.Platform() {
  52. case "windows-service":
  53. log.Println("可使用 .\\backup-x.exe -s install 安装服务运行")
  54. default:
  55. log.Println("可使用 ./backup-x -s install 安装服务运行")
  56. }
  57. run(100 * time.Millisecond)
  58. }
  59. }
  60. }
  61. }
  62. func staticFsFunc(writer http.ResponseWriter, request *http.Request) {
  63. http.FileServer(http.FS(staticEmbededFiles)).ServeHTTP(writer, request)
  64. }
  65. func faviconFsFunc(writer http.ResponseWriter, request *http.Request) {
  66. http.FileServer(http.FS(faviconEmbededFile)).ServeHTTP(writer, request)
  67. }
  68. func run(firstDelay time.Duration) {
  69. // 启动静态文件服务
  70. http.HandleFunc("/static/", web.BasicAuth(staticFsFunc))
  71. http.HandleFunc("/favicon.ico", web.BasicAuth(faviconFsFunc))
  72. http.HandleFunc("/", web.BasicAuth(web.WritingConfig))
  73. http.HandleFunc("/save", web.BasicAuth(web.Save))
  74. http.HandleFunc("/logs", web.BasicAuth(web.Logs))
  75. http.HandleFunc("/clearLog", web.BasicAuth(web.ClearLog))
  76. http.HandleFunc("/webhookTest", web.BasicAuth(web.WebhookTest))
  77. // 改变工作目录
  78. os.Chdir(*backupDir)
  79. // 运行
  80. go client.DeleteOldBackup()
  81. go client.RunLoop(firstDelay)
  82. err := http.ListenAndServe(*listen, nil)
  83. if err != nil {
  84. log.Println("启动端口发生异常, 请检查端口是否被占用", err)
  85. time.Sleep(time.Minute)
  86. }
  87. }
  88. type program struct{}
  89. func (p *program) Start(s service.Service) error {
  90. // Start should not block. Do the actual work async.
  91. go p.run()
  92. return nil
  93. }
  94. func (p *program) run() {
  95. // 服务运行,延时20秒运行,等待网络
  96. run(20 * time.Second)
  97. }
  98. func (p *program) Stop(s service.Service) error {
  99. // Stop should not block. Return with a few seconds.
  100. return nil
  101. }
  102. func getService() service.Service {
  103. options := make(service.KeyValue)
  104. var depends []string
  105. // 确保服务等待网络就绪后再启动
  106. switch service.ChosenSystem().String() {
  107. case "windows-service":
  108. // 将 Windows 服务的启动类型设为自动(延迟启动)
  109. options["DelayedAutoStart"] = true
  110. default:
  111. // 向 Systemd 添加网络依赖
  112. depends = append(depends, "Requires=network.target",
  113. "After=network-online.target")
  114. }
  115. svcConfig := &service.Config{
  116. Name: "backup-x",
  117. DisplayName: "backup-x",
  118. Description: "带Web界面的数据库/文件备份增强工具",
  119. Arguments: []string{"-l", *listen, "-d", *backupDir},
  120. Dependencies: depends,
  121. Option: options,
  122. }
  123. prg := &program{}
  124. s, err := service.New(prg, svcConfig)
  125. if err != nil {
  126. log.Fatalln(err)
  127. }
  128. return s
  129. }
  130. // 卸载服务
  131. func uninstallService() {
  132. s := getService()
  133. status, _ := s.Status()
  134. // 处理卸载
  135. if status != service.StatusUnknown {
  136. s.Stop()
  137. if err := s.Uninstall(); err == nil {
  138. log.Println("backup-x 服务卸载成功!")
  139. } else {
  140. log.Printf("backup-x 服务卸载失败, ERR: %s\n", err)
  141. }
  142. } else {
  143. log.Printf("backup-x 服务未安装")
  144. }
  145. }
  146. // 安装服务
  147. func installService() {
  148. s := getService()
  149. status, err := s.Status()
  150. if err != nil && status == service.StatusUnknown {
  151. // 服务未知,创建服务
  152. if err = s.Install(); err == nil {
  153. s.Start()
  154. log.Println("安装 backup-x 服务成功! 程序会一直运行, 包括重启后。")
  155. return
  156. }
  157. log.Printf("安装 backup-x 服务失败, ERR: %s\n", err)
  158. switch s.Platform() {
  159. case "windows-service":
  160. log.Println("请确保使用如下命令: .\\backup-x.exe -s install")
  161. default:
  162. log.Println("请确保使用如下命令: ./backup-x -s install")
  163. }
  164. }
  165. if status != service.StatusUnknown {
  166. log.Println("backup-x 服务已安装, 无需再次安装")
  167. }
  168. }