gitea源码

mirror_push.go 7.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. // Copyright 2021 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package mirror
  4. import (
  5. "context"
  6. "errors"
  7. "fmt"
  8. "io"
  9. "regexp"
  10. "time"
  11. "code.gitea.io/gitea/models/db"
  12. repo_model "code.gitea.io/gitea/models/repo"
  13. "code.gitea.io/gitea/modules/git"
  14. "code.gitea.io/gitea/modules/gitrepo"
  15. "code.gitea.io/gitea/modules/lfs"
  16. "code.gitea.io/gitea/modules/log"
  17. "code.gitea.io/gitea/modules/process"
  18. "code.gitea.io/gitea/modules/proxy"
  19. "code.gitea.io/gitea/modules/repository"
  20. "code.gitea.io/gitea/modules/setting"
  21. "code.gitea.io/gitea/modules/timeutil"
  22. "code.gitea.io/gitea/modules/util"
  23. repo_service "code.gitea.io/gitea/services/repository"
  24. )
  25. var stripExitStatus = regexp.MustCompile(`exit status \d+ - `)
  26. // AddPushMirrorRemote registers the push mirror remote.
  27. func AddPushMirrorRemote(ctx context.Context, m *repo_model.PushMirror, addr string) error {
  28. addRemoteAndConfig := func(storageRepo gitrepo.Repository, addr string) error {
  29. if err := gitrepo.GitRemoteAdd(ctx, storageRepo, m.RemoteName, addr, gitrepo.RemoteOptionMirrorPush); err != nil {
  30. return err
  31. }
  32. if err := gitrepo.GitConfigAdd(ctx, storageRepo, "remote."+m.RemoteName+".push", "+refs/heads/*:refs/heads/*"); err != nil {
  33. return err
  34. }
  35. return gitrepo.GitConfigAdd(ctx, storageRepo, "remote."+m.RemoteName+".push", "+refs/tags/*:refs/tags/*")
  36. }
  37. if err := addRemoteAndConfig(m.Repo, addr); err != nil {
  38. return err
  39. }
  40. if repo_service.HasWiki(ctx, m.Repo) {
  41. wikiRemoteURL := repository.WikiRemoteURL(ctx, addr)
  42. if len(wikiRemoteURL) > 0 {
  43. if err := addRemoteAndConfig(m.Repo.WikiStorageRepo(), wikiRemoteURL); err != nil {
  44. return err
  45. }
  46. }
  47. }
  48. return nil
  49. }
  50. // RemovePushMirrorRemote removes the push mirror remote.
  51. func RemovePushMirrorRemote(ctx context.Context, m *repo_model.PushMirror) error {
  52. _ = m.GetRepository(ctx)
  53. if err := gitrepo.GitRemoteRemove(ctx, m.Repo, m.RemoteName); err != nil {
  54. return err
  55. }
  56. if repo_service.HasWiki(ctx, m.Repo) {
  57. if err := gitrepo.GitRemoteRemove(ctx, m.Repo.WikiStorageRepo(), m.RemoteName); err != nil {
  58. // The wiki remote may not exist
  59. log.Warn("Wiki Remote[%d] could not be removed: %v", m.ID, err)
  60. }
  61. }
  62. return nil
  63. }
  64. // SyncPushMirror starts the sync of the push mirror and schedules the next run.
  65. func SyncPushMirror(ctx context.Context, mirrorID int64) bool {
  66. log.Trace("SyncPushMirror [mirror: %d]", mirrorID)
  67. defer func() {
  68. err := recover()
  69. if err == nil {
  70. return
  71. }
  72. // There was a panic whilst syncPushMirror...
  73. log.Error("PANIC whilst syncPushMirror[%d] Panic: %v\nStacktrace: %s", mirrorID, err, log.Stack(2))
  74. }()
  75. // TODO: Handle "!exist" better
  76. m, exist, err := db.GetByID[repo_model.PushMirror](ctx, mirrorID)
  77. if err != nil || !exist {
  78. log.Error("GetPushMirrorByID [%d]: %v", mirrorID, err)
  79. return false
  80. }
  81. _ = m.GetRepository(ctx)
  82. m.LastError = ""
  83. ctx, _, finished := process.GetManager().AddContext(ctx, fmt.Sprintf("Syncing PushMirror %s/%s to %s", m.Repo.OwnerName, m.Repo.Name, m.RemoteName))
  84. defer finished()
  85. log.Trace("SyncPushMirror [mirror: %d][repo: %-v]: Running Sync", m.ID, m.Repo)
  86. err = runPushSync(ctx, m)
  87. if err != nil {
  88. log.Error("SyncPushMirror [mirror: %d][repo: %-v]: %v", m.ID, m.Repo, err)
  89. m.LastError = stripExitStatus.ReplaceAllLiteralString(err.Error(), "")
  90. }
  91. m.LastUpdateUnix = timeutil.TimeStampNow()
  92. if err := repo_model.UpdatePushMirror(ctx, m); err != nil {
  93. log.Error("UpdatePushMirror [%d]: %v", m.ID, err)
  94. return false
  95. }
  96. log.Trace("SyncPushMirror [mirror: %d][repo: %-v]: Finished", m.ID, m.Repo)
  97. return err == nil
  98. }
  99. func runPushSync(ctx context.Context, m *repo_model.PushMirror) error {
  100. timeout := time.Duration(setting.Git.Timeout.Mirror) * time.Second
  101. performPush := func(repo *repo_model.Repository, isWiki bool) error {
  102. var storageRepo gitrepo.Repository = repo
  103. path := repo.RepoPath()
  104. if isWiki {
  105. storageRepo = repo.WikiStorageRepo()
  106. path = repo.WikiPath()
  107. }
  108. remoteURL, err := gitrepo.GitRemoteGetURL(ctx, storageRepo, m.RemoteName)
  109. if err != nil {
  110. log.Error("GetRemoteURL(%s) Error %v", path, err)
  111. return errors.New("Unexpected error")
  112. }
  113. if setting.LFS.StartServer {
  114. log.Trace("SyncMirrors [repo: %-v]: syncing LFS objects...", m.Repo)
  115. gitRepo, err := gitrepo.OpenRepository(ctx, storageRepo)
  116. if err != nil {
  117. log.Error("OpenRepository: %v", err)
  118. return errors.New("Unexpected error")
  119. }
  120. defer gitRepo.Close()
  121. endpoint := lfs.DetermineEndpoint(remoteURL.String(), "")
  122. lfsClient := lfs.NewClient(endpoint, nil)
  123. if err := pushAllLFSObjects(ctx, gitRepo, lfsClient); err != nil {
  124. return util.SanitizeErrorCredentialURLs(err)
  125. }
  126. }
  127. log.Trace("Pushing %s mirror[%d] remote %s", path, m.ID, m.RemoteName)
  128. envs := proxy.EnvWithProxy(remoteURL.URL)
  129. if err := git.Push(ctx, path, git.PushOptions{
  130. Remote: m.RemoteName,
  131. Force: true,
  132. Mirror: true,
  133. Timeout: timeout,
  134. Env: envs,
  135. }); err != nil {
  136. log.Error("Error pushing %s mirror[%d] remote %s: %v", path, m.ID, m.RemoteName, err)
  137. return util.SanitizeErrorCredentialURLs(err)
  138. }
  139. return nil
  140. }
  141. err := performPush(m.Repo, false)
  142. if err != nil {
  143. return err
  144. }
  145. if repo_service.HasWiki(ctx, m.Repo) {
  146. if _, err := gitrepo.GitRemoteGetURL(ctx, m.Repo.WikiStorageRepo(), m.RemoteName); err == nil {
  147. err := performPush(m.Repo, true)
  148. if err != nil {
  149. return err
  150. }
  151. } else if !errors.Is(err, util.ErrNotExist) {
  152. log.Error("GetRemote of wiki failed: %v", err)
  153. }
  154. }
  155. return nil
  156. }
  157. func pushAllLFSObjects(ctx context.Context, gitRepo *git.Repository, lfsClient lfs.Client) error {
  158. contentStore := lfs.NewContentStore()
  159. pointerChan := make(chan lfs.PointerBlob)
  160. errChan := make(chan error, 1)
  161. go lfs.SearchPointerBlobs(ctx, gitRepo, pointerChan, errChan)
  162. uploadObjects := func(pointers []lfs.Pointer) error {
  163. err := lfsClient.Upload(ctx, pointers, func(p lfs.Pointer, objectError error) (io.ReadCloser, error) {
  164. if objectError != nil {
  165. return nil, objectError
  166. }
  167. content, err := contentStore.Get(p)
  168. if err != nil {
  169. log.Error("Error reading LFS object %v: %v", p, err)
  170. }
  171. return content, err
  172. })
  173. if err != nil {
  174. select {
  175. case <-ctx.Done():
  176. return nil
  177. default:
  178. }
  179. }
  180. return err
  181. }
  182. var batch []lfs.Pointer
  183. for pointerBlob := range pointerChan {
  184. exists, err := contentStore.Exists(pointerBlob.Pointer)
  185. if err != nil {
  186. log.Error("Error checking if LFS object %v exists: %v", pointerBlob.Pointer, err)
  187. return err
  188. }
  189. if !exists {
  190. log.Trace("Skipping missing LFS object %v", pointerBlob.Pointer)
  191. continue
  192. }
  193. batch = append(batch, pointerBlob.Pointer)
  194. if len(batch) >= lfsClient.BatchSize() {
  195. if err := uploadObjects(batch); err != nil {
  196. return err
  197. }
  198. batch = nil
  199. }
  200. }
  201. if len(batch) > 0 {
  202. if err := uploadObjects(batch); err != nil {
  203. return err
  204. }
  205. }
  206. err, has := <-errChan
  207. if has {
  208. log.Error("Error enumerating LFS objects for repository: %v", err)
  209. return err
  210. }
  211. return nil
  212. }
  213. func syncPushMirrorWithSyncOnCommit(ctx context.Context, repoID int64) {
  214. pushMirrors, err := repo_model.GetPushMirrorsSyncedOnCommit(ctx, repoID)
  215. if err != nil {
  216. log.Error("repo_model.GetPushMirrorsSyncedOnCommit failed: %v", err)
  217. return
  218. }
  219. for _, mirror := range pushMirrors {
  220. AddPushMirrorToQueue(mirror.ID)
  221. }
  222. }