You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

index.go 7.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306
  1. // Copyright 2022 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package cargo
  4. import (
  5. "bytes"
  6. "context"
  7. "errors"
  8. "fmt"
  9. "io"
  10. "path"
  11. "strconv"
  12. "time"
  13. packages_model "code.gitea.io/gitea/models/packages"
  14. repo_model "code.gitea.io/gitea/models/repo"
  15. user_model "code.gitea.io/gitea/models/user"
  16. "code.gitea.io/gitea/modules/git"
  17. "code.gitea.io/gitea/modules/json"
  18. cargo_module "code.gitea.io/gitea/modules/packages/cargo"
  19. repo_module "code.gitea.io/gitea/modules/repository"
  20. "code.gitea.io/gitea/modules/setting"
  21. "code.gitea.io/gitea/modules/util"
  22. files_service "code.gitea.io/gitea/services/repository/files"
  23. )
  24. const (
  25. IndexRepositoryName = "_cargo-index"
  26. ConfigFileName = "config.json"
  27. )
  28. // https://doc.rust-lang.org/cargo/reference/registries.html#index-format
  29. func BuildPackagePath(name string) string {
  30. switch len(name) {
  31. case 0:
  32. panic("Cargo package name can not be empty")
  33. case 1:
  34. return path.Join("1", name)
  35. case 2:
  36. return path.Join("2", name)
  37. case 3:
  38. return path.Join("3", string(name[0]), name)
  39. default:
  40. return path.Join(name[0:2], name[2:4], name)
  41. }
  42. }
  43. func InitializeIndexRepository(ctx context.Context, doer, owner *user_model.User) error {
  44. repo, err := getOrCreateIndexRepository(ctx, doer, owner)
  45. if err != nil {
  46. return err
  47. }
  48. if err := createOrUpdateConfigFile(ctx, repo, doer, owner); err != nil {
  49. return fmt.Errorf("createOrUpdateConfigFile: %w", err)
  50. }
  51. return nil
  52. }
  53. func RebuildIndex(ctx context.Context, doer, owner *user_model.User) error {
  54. repo, err := getOrCreateIndexRepository(ctx, doer, owner)
  55. if err != nil {
  56. return err
  57. }
  58. ps, err := packages_model.GetPackagesByType(ctx, owner.ID, packages_model.TypeCargo)
  59. if err != nil {
  60. return fmt.Errorf("GetPackagesByType: %w", err)
  61. }
  62. return alterRepositoryContent(
  63. ctx,
  64. doer,
  65. repo,
  66. "Rebuild Cargo Index",
  67. func(t *files_service.TemporaryUploadRepository) error {
  68. // Remove all existing content but the Cargo config
  69. files, err := t.LsFiles()
  70. if err != nil {
  71. return err
  72. }
  73. for i, file := range files {
  74. if file == ConfigFileName {
  75. files[i] = files[len(files)-1]
  76. files = files[:len(files)-1]
  77. break
  78. }
  79. }
  80. if err := t.RemoveFilesFromIndex(files...); err != nil {
  81. return err
  82. }
  83. // Add all packages
  84. for _, p := range ps {
  85. if err := addOrUpdatePackageIndex(ctx, t, p); err != nil {
  86. return err
  87. }
  88. }
  89. return nil
  90. },
  91. )
  92. }
  93. func AddOrUpdatePackageIndex(ctx context.Context, doer, owner *user_model.User, packageID int64) error {
  94. repo, err := getOrCreateIndexRepository(ctx, doer, owner)
  95. if err != nil {
  96. return err
  97. }
  98. p, err := packages_model.GetPackageByID(ctx, packageID)
  99. if err != nil {
  100. return fmt.Errorf("GetPackageByID[%d]: %w", packageID, err)
  101. }
  102. return alterRepositoryContent(
  103. ctx,
  104. doer,
  105. repo,
  106. "Update "+p.Name,
  107. func(t *files_service.TemporaryUploadRepository) error {
  108. return addOrUpdatePackageIndex(ctx, t, p)
  109. },
  110. )
  111. }
  112. type IndexVersionEntry struct {
  113. Name string `json:"name"`
  114. Version string `json:"vers"`
  115. Dependencies []*cargo_module.Dependency `json:"deps"`
  116. FileChecksum string `json:"cksum"`
  117. Features map[string][]string `json:"features"`
  118. Yanked bool `json:"yanked"`
  119. Links string `json:"links,omitempty"`
  120. }
  121. func BuildPackageIndex(ctx context.Context, p *packages_model.Package) (*bytes.Buffer, error) {
  122. pvs, _, err := packages_model.SearchVersions(ctx, &packages_model.PackageSearchOptions{
  123. PackageID: p.ID,
  124. Sort: packages_model.SortVersionAsc,
  125. })
  126. if err != nil {
  127. return nil, fmt.Errorf("SearchVersions[%s]: %w", p.Name, err)
  128. }
  129. if len(pvs) == 0 {
  130. return nil, nil
  131. }
  132. pds, err := packages_model.GetPackageDescriptors(ctx, pvs)
  133. if err != nil {
  134. return nil, fmt.Errorf("GetPackageDescriptors[%s]: %w", p.Name, err)
  135. }
  136. var b bytes.Buffer
  137. for _, pd := range pds {
  138. metadata := pd.Metadata.(*cargo_module.Metadata)
  139. dependencies := metadata.Dependencies
  140. if dependencies == nil {
  141. dependencies = make([]*cargo_module.Dependency, 0)
  142. }
  143. features := metadata.Features
  144. if features == nil {
  145. features = make(map[string][]string)
  146. }
  147. yanked, _ := strconv.ParseBool(pd.VersionProperties.GetByName(cargo_module.PropertyYanked))
  148. entry, err := json.Marshal(&IndexVersionEntry{
  149. Name: pd.Package.Name,
  150. Version: pd.Version.Version,
  151. Dependencies: dependencies,
  152. FileChecksum: pd.Files[0].Blob.HashSHA256,
  153. Features: features,
  154. Yanked: yanked,
  155. Links: metadata.Links,
  156. })
  157. if err != nil {
  158. return nil, err
  159. }
  160. b.Write(entry)
  161. b.WriteString("\n")
  162. }
  163. return &b, nil
  164. }
  165. func addOrUpdatePackageIndex(ctx context.Context, t *files_service.TemporaryUploadRepository, p *packages_model.Package) error {
  166. b, err := BuildPackageIndex(ctx, p)
  167. if err != nil {
  168. return err
  169. }
  170. if b == nil {
  171. return nil
  172. }
  173. return writeObjectToIndex(t, BuildPackagePath(p.LowerName), b)
  174. }
  175. func getOrCreateIndexRepository(ctx context.Context, doer, owner *user_model.User) (*repo_model.Repository, error) {
  176. repo, err := repo_model.GetRepositoryByOwnerAndName(ctx, owner.Name, IndexRepositoryName)
  177. if err != nil {
  178. if errors.Is(err, util.ErrNotExist) {
  179. repo, err = repo_module.CreateRepository(doer, owner, repo_module.CreateRepoOptions{
  180. Name: IndexRepositoryName,
  181. })
  182. if err != nil {
  183. return nil, fmt.Errorf("CreateRepository: %w", err)
  184. }
  185. } else {
  186. return nil, fmt.Errorf("GetRepositoryByOwnerAndName: %w", err)
  187. }
  188. }
  189. return repo, nil
  190. }
  191. type Config struct {
  192. DownloadURL string `json:"dl"`
  193. APIURL string `json:"api"`
  194. }
  195. func BuildConfig(owner *user_model.User) *Config {
  196. return &Config{
  197. DownloadURL: setting.AppURL + "api/packages/" + owner.Name + "/cargo/api/v1/crates",
  198. APIURL: setting.AppURL + "api/packages/" + owner.Name + "/cargo",
  199. }
  200. }
  201. func createOrUpdateConfigFile(ctx context.Context, repo *repo_model.Repository, doer, owner *user_model.User) error {
  202. return alterRepositoryContent(
  203. ctx,
  204. doer,
  205. repo,
  206. "Initialize Cargo Config",
  207. func(t *files_service.TemporaryUploadRepository) error {
  208. var b bytes.Buffer
  209. err := json.NewEncoder(&b).Encode(BuildConfig(owner))
  210. if err != nil {
  211. return err
  212. }
  213. return writeObjectToIndex(t, ConfigFileName, &b)
  214. },
  215. )
  216. }
  217. // This is a shorter version of CreateOrUpdateRepoFile which allows to perform multiple actions on a git repository
  218. func alterRepositoryContent(ctx context.Context, doer *user_model.User, repo *repo_model.Repository, commitMessage string, fn func(*files_service.TemporaryUploadRepository) error) error {
  219. t, err := files_service.NewTemporaryUploadRepository(ctx, repo)
  220. if err != nil {
  221. return err
  222. }
  223. defer t.Close()
  224. var lastCommitID string
  225. if err := t.Clone(repo.DefaultBranch); err != nil {
  226. if !git.IsErrBranchNotExist(err) || !repo.IsEmpty {
  227. return err
  228. }
  229. if err := t.Init(); err != nil {
  230. return err
  231. }
  232. } else {
  233. if err := t.SetDefaultIndex(); err != nil {
  234. return err
  235. }
  236. commit, err := t.GetBranchCommit(repo.DefaultBranch)
  237. if err != nil {
  238. return err
  239. }
  240. lastCommitID = commit.ID.String()
  241. }
  242. if err := fn(t); err != nil {
  243. return err
  244. }
  245. treeHash, err := t.WriteTree()
  246. if err != nil {
  247. return err
  248. }
  249. now := time.Now()
  250. commitHash, err := t.CommitTreeWithDate(lastCommitID, doer, doer, treeHash, commitMessage, false, now, now)
  251. if err != nil {
  252. return err
  253. }
  254. return t.Push(doer, commitHash, repo.DefaultBranch)
  255. }
  256. func writeObjectToIndex(t *files_service.TemporaryUploadRepository, path string, r io.Reader) error {
  257. hash, err := t.HashObject(r)
  258. if err != nil {
  259. return err
  260. }
  261. return t.AddObjectToIndex("100644", hash, path)
  262. }