job.go 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. package worker
  2. import (
  3. "errors"
  4. "fmt"
  5. tunasync "github.com/tuna/tunasync/internal"
  6. )
  7. // this file contains the workflow of a mirror jb
  8. type ctrlAction uint8
  9. const (
  10. jobStart ctrlAction = iota
  11. jobStop // stop syncing keep the job
  12. jobDisable // disable the job (stops goroutine)
  13. jobRestart // restart syncing
  14. jobPing // ensure the goroutine is alive
  15. )
  16. type jobMessage struct {
  17. status tunasync.SyncStatus
  18. name string
  19. msg string
  20. }
  21. type mirrorJob struct {
  22. provider mirrorProvider
  23. ctrlChan chan ctrlAction
  24. stopped chan empty
  25. enabled bool
  26. }
  27. func newMirrorJob(provider mirrorProvider) *mirrorJob {
  28. return &mirrorJob{
  29. provider: provider,
  30. ctrlChan: make(chan ctrlAction, 1),
  31. enabled: false,
  32. }
  33. }
  34. func (m *mirrorJob) Name() string {
  35. return m.provider.Name()
  36. }
  37. // runMirrorJob is the goroutine where syncing job runs in
  38. // arguments:
  39. // provider: mirror provider object
  40. // ctrlChan: receives messages from the manager
  41. // managerChan: push messages to the manager, this channel should have a larger buffer
  42. // sempaphore: make sure the concurrent running syncing job won't explode
  43. // TODO: message struct for managerChan
  44. func (m *mirrorJob) Run(managerChan chan<- jobMessage, semaphore chan empty) error {
  45. m.stopped = make(chan empty)
  46. defer close(m.stopped)
  47. provider := m.provider
  48. // to make code shorter
  49. runHooks := func(Hooks []jobHook, action func(h jobHook) error, hookname string) error {
  50. for _, hook := range Hooks {
  51. if err := action(hook); err != nil {
  52. logger.Error(
  53. "failed at %s hooks for %s: %s",
  54. hookname, m.Name(), err.Error(),
  55. )
  56. managerChan <- jobMessage{
  57. tunasync.Failed, m.Name(),
  58. fmt.Sprintf("error exec hook %s: %s", hookname, err.Error()),
  59. }
  60. return err
  61. }
  62. }
  63. return nil
  64. }
  65. runJobWrapper := func(kill <-chan empty, jobDone chan<- empty) error {
  66. defer close(jobDone)
  67. managerChan <- jobMessage{tunasync.PreSyncing, m.Name(), ""}
  68. logger.Info("start syncing: %s", m.Name())
  69. Hooks := provider.Hooks()
  70. rHooks := []jobHook{}
  71. for i := len(Hooks); i > 0; i-- {
  72. rHooks = append(rHooks, Hooks[i-1])
  73. }
  74. logger.Debug("hooks: pre-job")
  75. err := runHooks(Hooks, func(h jobHook) error { return h.preJob() }, "pre-job")
  76. if err != nil {
  77. return err
  78. }
  79. for retry := 0; retry < maxRetry; retry++ {
  80. stopASAP := false // stop job as soon as possible
  81. if retry > 0 {
  82. logger.Info("retry syncing: %s, retry: %d", m.Name(), retry)
  83. }
  84. err := runHooks(Hooks, func(h jobHook) error { return h.preExec() }, "pre-exec")
  85. if err != nil {
  86. return err
  87. }
  88. // start syncing
  89. managerChan <- jobMessage{tunasync.Syncing, m.Name(), ""}
  90. var syncErr error
  91. syncDone := make(chan error, 1)
  92. go func() {
  93. err := provider.Run()
  94. if !stopASAP {
  95. syncDone <- err
  96. }
  97. }()
  98. select {
  99. case syncErr = <-syncDone:
  100. logger.Debug("syncing done")
  101. case <-kill:
  102. logger.Debug("received kill")
  103. stopASAP = true
  104. err := provider.Terminate()
  105. if err != nil {
  106. logger.Error("failed to terminate provider %s: %s", m.Name(), err.Error())
  107. return err
  108. }
  109. syncErr = errors.New("killed by manager")
  110. }
  111. // post-exec hooks
  112. herr := runHooks(rHooks, func(h jobHook) error { return h.postExec() }, "post-exec")
  113. if herr != nil {
  114. return herr
  115. }
  116. if syncErr == nil {
  117. // syncing success
  118. logger.Info("succeeded syncing %s", m.Name())
  119. managerChan <- jobMessage{tunasync.Success, m.Name(), ""}
  120. // post-success hooks
  121. err := runHooks(rHooks, func(h jobHook) error { return h.postSuccess() }, "post-success")
  122. if err != nil {
  123. return err
  124. }
  125. return nil
  126. }
  127. // syncing failed
  128. logger.Warning("failed syncing %s: %s", m.Name(), syncErr.Error())
  129. managerChan <- jobMessage{tunasync.Failed, m.Name(), syncErr.Error()}
  130. // post-fail hooks
  131. logger.Debug("post-fail hooks")
  132. err = runHooks(rHooks, func(h jobHook) error { return h.postFail() }, "post-fail")
  133. if err != nil {
  134. return err
  135. }
  136. // gracefully exit
  137. if stopASAP {
  138. logger.Debug("No retry, exit directly")
  139. return nil
  140. }
  141. // continue to next retry
  142. } // for retry
  143. return nil
  144. }
  145. runJob := func(kill <-chan empty, jobDone chan<- empty) {
  146. select {
  147. case semaphore <- empty{}:
  148. defer func() { <-semaphore }()
  149. runJobWrapper(kill, jobDone)
  150. case <-kill:
  151. jobDone <- empty{}
  152. return
  153. }
  154. }
  155. for {
  156. if m.enabled {
  157. kill := make(chan empty)
  158. jobDone := make(chan empty)
  159. go runJob(kill, jobDone)
  160. _wait_for_job:
  161. select {
  162. case <-jobDone:
  163. logger.Debug("job done")
  164. case ctrl := <-m.ctrlChan:
  165. switch ctrl {
  166. case jobStop:
  167. m.enabled = false
  168. close(kill)
  169. <-jobDone
  170. case jobDisable:
  171. close(kill)
  172. <-jobDone
  173. return nil
  174. case jobRestart:
  175. m.enabled = true
  176. close(kill)
  177. <-jobDone
  178. continue
  179. case jobStart:
  180. m.enabled = true
  181. goto _wait_for_job
  182. default:
  183. // TODO: implement this
  184. close(kill)
  185. return nil
  186. }
  187. }
  188. }
  189. ctrl := <-m.ctrlChan
  190. switch ctrl {
  191. case jobStop:
  192. m.enabled = false
  193. case jobDisable:
  194. return nil
  195. case jobRestart:
  196. m.enabled = true
  197. case jobStart:
  198. m.enabled = true
  199. default:
  200. // TODO
  201. return nil
  202. }
  203. }
  204. }