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

tree.go 4.1 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. // Copyright 2018 The Gitea Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package repo
  5. import (
  6. "fmt"
  7. "strings"
  8. "code.gitea.io/git"
  9. "code.gitea.io/gitea/modules/context"
  10. "code.gitea.io/gitea/modules/setting"
  11. "code.gitea.io/sdk/gitea"
  12. )
  13. // GetTree get the tree of a repository.
  14. func GetTree(ctx *context.APIContext) {
  15. // swagger:operation GET /repos/{owner}/{repo}/git/trees/{sha} repository GetTree
  16. // ---
  17. // summary: Gets the tree of a repository.
  18. // produces:
  19. // - application/json
  20. // parameters:
  21. // - name: owner
  22. // in: path
  23. // description: owner of the repo
  24. // type: string
  25. // required: true
  26. // - name: repo
  27. // in: path
  28. // description: name of the repo
  29. // type: string
  30. // required: true
  31. // - name: sha
  32. // in: path
  33. // description: sha of the commit
  34. // type: string
  35. // required: true
  36. // - name: recursive
  37. // in: query
  38. // description: show all directories and files
  39. // required: false
  40. // type: boolean
  41. // - name: page
  42. // in: query
  43. // description: page number; the 'truncated' field in the response will be true if there are still more items after this page, false if the last page
  44. // required: false
  45. // type: integer
  46. // - name: per_page
  47. // in: query
  48. // description: number of items per page; default is 1000 or what is set in app.ini as DEFAULT_GIT_TREES_PER_PAGE
  49. // required: false
  50. // type: integer
  51. // responses:
  52. // "200":
  53. // "$ref": "#/responses/GitTreeResponse"
  54. sha := ctx.Params("sha")
  55. if len(sha) == 0 {
  56. ctx.Error(400, "", "sha not provided")
  57. return
  58. }
  59. tree := GetTreeBySHA(ctx, sha)
  60. if tree != nil {
  61. ctx.JSON(200, tree)
  62. } else {
  63. ctx.Error(400, "", "sha invalid")
  64. }
  65. }
  66. // GetTreeBySHA get the GitTreeResponse of a repository using a sha hash.
  67. func GetTreeBySHA(ctx *context.APIContext, sha string) *gitea.GitTreeResponse {
  68. gitTree, err := ctx.Repo.GitRepo.GetTree(sha)
  69. if err != nil || gitTree == nil {
  70. return nil
  71. }
  72. tree := new(gitea.GitTreeResponse)
  73. repoID := strings.TrimRight(setting.AppURL, "/") + "/api/v1/repos/" + ctx.Repo.Repository.Owner.Name + "/" + ctx.Repo.Repository.Name
  74. tree.SHA = gitTree.ID.String()
  75. tree.URL = repoID + "/git/trees/" + tree.SHA
  76. var entries git.Entries
  77. if ctx.QueryBool("recursive") {
  78. entries, err = gitTree.ListEntriesRecursive()
  79. } else {
  80. entries, err = gitTree.ListEntries()
  81. }
  82. if err != nil {
  83. return tree
  84. }
  85. repoIDLen := len(repoID)
  86. // 51 is len(sha1) + len("/git/blobs/"). 40 + 11.
  87. blobURL := make([]byte, repoIDLen+51)
  88. copy(blobURL[:], repoID)
  89. copy(blobURL[repoIDLen:], "/git/blobs/")
  90. // 51 is len(sha1) + len("/git/trees/"). 40 + 11.
  91. treeURL := make([]byte, repoIDLen+51)
  92. copy(treeURL[:], repoID)
  93. copy(treeURL[repoIDLen:], "/git/trees/")
  94. // 40 is the size of the sha1 hash in hexadecimal format.
  95. copyPos := len(treeURL) - 40
  96. page := ctx.QueryInt("page")
  97. perPage := ctx.QueryInt("per_page")
  98. if perPage <= 0 || perPage > setting.API.DefaultGitTreesPerPage {
  99. perPage = setting.API.DefaultGitTreesPerPage
  100. }
  101. if page <= 0 {
  102. page = 1
  103. }
  104. tree.Page = page
  105. tree.TotalCount = len(entries)
  106. rangeStart := perPage * (page - 1)
  107. if rangeStart >= len(entries) {
  108. return tree
  109. }
  110. var rangeEnd int
  111. if len(entries) > perPage {
  112. tree.Truncated = true
  113. }
  114. if rangeStart+perPage < len(entries) {
  115. rangeEnd = rangeStart + perPage
  116. } else {
  117. rangeEnd = len(entries)
  118. }
  119. tree.Entries = make([]gitea.GitEntry, rangeEnd-rangeStart)
  120. for e := rangeStart; e < rangeEnd; e++ {
  121. i := e - rangeStart
  122. tree.Entries[i].Path = entries[e].Name()
  123. tree.Entries[i].Mode = fmt.Sprintf("%06x", entries[e].Mode())
  124. tree.Entries[i].Type = string(entries[e].Type)
  125. tree.Entries[i].Size = entries[e].Size()
  126. tree.Entries[i].SHA = entries[e].ID.String()
  127. if entries[e].IsDir() {
  128. copy(treeURL[copyPos:], entries[e].ID.String())
  129. tree.Entries[i].URL = string(treeURL[:])
  130. } else {
  131. copy(blobURL[copyPos:], entries[e].ID.String())
  132. tree.Entries[i].URL = string(blobURL[:])
  133. }
  134. }
  135. return tree
  136. }