repo.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. // Copyright 2014 The Gogs 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. "net/http"
  7. "path"
  8. api "github.com/gogs/go-gogs-client"
  9. "github.com/pkg/errors"
  10. log "unknwon.dev/clog/v2"
  11. "gogs.io/gogs/internal/conf"
  12. "gogs.io/gogs/internal/context"
  13. "gogs.io/gogs/internal/db"
  14. "gogs.io/gogs/internal/form"
  15. "gogs.io/gogs/internal/route/api/v1/convert"
  16. )
  17. func Search(c *context.APIContext) {
  18. opts := &db.SearchRepoOptions{
  19. Keyword: path.Base(c.Query("q")),
  20. OwnerID: c.QueryInt64("uid"),
  21. PageSize: convert.ToCorrectPageSize(c.QueryInt("limit")),
  22. Page: c.QueryInt("page"),
  23. }
  24. // Check visibility.
  25. if c.IsLogged && opts.OwnerID > 0 {
  26. if c.User.ID == opts.OwnerID {
  27. opts.Private = true
  28. } else {
  29. u, err := db.Users.GetByID(c.Req.Context(), opts.OwnerID)
  30. if err != nil {
  31. c.JSON(http.StatusInternalServerError, map[string]any{
  32. "ok": false,
  33. "error": err.Error(),
  34. })
  35. return
  36. }
  37. if u.IsOrganization() && u.IsOwnedBy(c.User.ID) {
  38. opts.Private = true
  39. }
  40. // FIXME: how about collaborators?
  41. }
  42. }
  43. repos, count, err := db.SearchRepositoryByName(opts)
  44. if err != nil {
  45. c.JSON(http.StatusInternalServerError, map[string]any{
  46. "ok": false,
  47. "error": err.Error(),
  48. })
  49. return
  50. }
  51. if err = db.RepositoryList(repos).LoadAttributes(); err != nil {
  52. c.JSON(http.StatusInternalServerError, map[string]any{
  53. "ok": false,
  54. "error": err.Error(),
  55. })
  56. return
  57. }
  58. results := make([]*api.Repository, len(repos))
  59. for i := range repos {
  60. results[i] = repos[i].APIFormatLegacy(nil)
  61. }
  62. c.SetLinkHeader(int(count), opts.PageSize)
  63. c.JSONSuccess(map[string]any{
  64. "ok": true,
  65. "data": results,
  66. })
  67. }
  68. func listUserRepositories(c *context.APIContext, username string) {
  69. user, err := db.Users.GetByUsername(c.Req.Context(), username)
  70. if err != nil {
  71. c.NotFoundOrError(err, "get user by name")
  72. return
  73. }
  74. // Only list public repositories if user requests someone else's repository list,
  75. // or an organization isn't a member of.
  76. var ownRepos []*db.Repository
  77. if user.IsOrganization() {
  78. ownRepos, _, err = db.Orgs.AccessibleRepositoriesByUser(
  79. c.Req.Context(),
  80. user.ID,
  81. c.User.ID,
  82. 1,
  83. user.NumRepos,
  84. db.AccessibleRepositoriesByUserOptions{},
  85. )
  86. } else {
  87. ownRepos, err = db.GetUserRepositories(&db.UserRepoOptions{
  88. UserID: user.ID,
  89. Private: c.User.ID == user.ID,
  90. Page: 1,
  91. PageSize: user.NumRepos,
  92. })
  93. }
  94. if err != nil {
  95. c.Error(err, "get user repositories")
  96. return
  97. }
  98. if err = db.RepositoryList(ownRepos).LoadAttributes(); err != nil {
  99. c.Error(err, "load attributes")
  100. return
  101. }
  102. // Early return for querying other user's repositories
  103. if c.User.ID != user.ID {
  104. repos := make([]*api.Repository, len(ownRepos))
  105. for i := range ownRepos {
  106. repos[i] = ownRepos[i].APIFormatLegacy(&api.Permission{Admin: true, Push: true, Pull: true})
  107. }
  108. c.JSONSuccess(&repos)
  109. return
  110. }
  111. accessibleRepos, err := db.Repos.GetByCollaboratorIDWithAccessMode(c.Req.Context(), user.ID)
  112. if err != nil {
  113. c.Error(err, "get repositories accesses by collaborator")
  114. return
  115. }
  116. numOwnRepos := len(ownRepos)
  117. repos := make([]*api.Repository, 0, numOwnRepos+len(accessibleRepos))
  118. for _, r := range ownRepos {
  119. repos = append(repos, r.APIFormatLegacy(&api.Permission{Admin: true, Push: true, Pull: true}))
  120. }
  121. for repo, access := range accessibleRepos {
  122. repos = append(repos,
  123. repo.APIFormatLegacy(&api.Permission{
  124. Admin: access >= db.AccessModeAdmin,
  125. Push: access >= db.AccessModeWrite,
  126. Pull: true,
  127. }),
  128. )
  129. }
  130. c.JSONSuccess(&repos)
  131. }
  132. func ListMyRepos(c *context.APIContext) {
  133. listUserRepositories(c, c.User.Name)
  134. }
  135. func ListUserRepositories(c *context.APIContext) {
  136. listUserRepositories(c, c.Params(":username"))
  137. }
  138. func ListOrgRepositories(c *context.APIContext) {
  139. listUserRepositories(c, c.Params(":org"))
  140. }
  141. func CreateUserRepo(c *context.APIContext, owner *db.User, opt api.CreateRepoOption) {
  142. repo, err := db.CreateRepository(c.User, owner, db.CreateRepoOptionsLegacy{
  143. Name: opt.Name,
  144. Description: opt.Description,
  145. Gitignores: opt.Gitignores,
  146. License: opt.License,
  147. Readme: opt.Readme,
  148. IsPrivate: opt.Private,
  149. AutoInit: opt.AutoInit,
  150. })
  151. if err != nil {
  152. if db.IsErrRepoAlreadyExist(err) ||
  153. db.IsErrNameNotAllowed(err) {
  154. c.ErrorStatus(http.StatusUnprocessableEntity, err)
  155. } else {
  156. if repo != nil {
  157. if err = db.DeleteRepository(c.User.ID, repo.ID); err != nil {
  158. log.Error("Failed to delete repository: %v", err)
  159. }
  160. }
  161. c.Error(err, "create repository")
  162. }
  163. return
  164. }
  165. c.JSON(201, repo.APIFormatLegacy(&api.Permission{Admin: true, Push: true, Pull: true}))
  166. }
  167. func Create(c *context.APIContext, opt api.CreateRepoOption) {
  168. // Shouldn't reach this condition, but just in case.
  169. if c.User.IsOrganization() {
  170. c.ErrorStatus(http.StatusUnprocessableEntity, errors.New("Not allowed to create repository for organization."))
  171. return
  172. }
  173. CreateUserRepo(c, c.User, opt)
  174. }
  175. func CreateOrgRepo(c *context.APIContext, opt api.CreateRepoOption) {
  176. org, err := db.Orgs.GetByName(c.Req.Context(), c.Params(":org"))
  177. if err != nil {
  178. c.NotFoundOrError(err, "get organization by name")
  179. return
  180. }
  181. if !org.IsOwnedBy(c.User.ID) {
  182. c.ErrorStatus(http.StatusForbidden, errors.New("Given user is not owner of organization."))
  183. return
  184. }
  185. CreateUserRepo(c, org, opt)
  186. }
  187. func Migrate(c *context.APIContext, f form.MigrateRepo) {
  188. ctxUser := c.User
  189. // Not equal means context user is an organization,
  190. // or is another user/organization if current user is admin.
  191. if f.Uid != ctxUser.ID {
  192. org, err := db.Users.GetByID(c.Req.Context(), f.Uid)
  193. if err != nil {
  194. if db.IsErrUserNotExist(err) {
  195. c.ErrorStatus(http.StatusUnprocessableEntity, err)
  196. } else {
  197. c.Error(err, "get user by ID")
  198. }
  199. return
  200. } else if !org.IsOrganization() && !c.User.IsAdmin {
  201. c.ErrorStatus(http.StatusForbidden, errors.New("Given user is not an organization."))
  202. return
  203. }
  204. ctxUser = org
  205. }
  206. if c.HasError() {
  207. c.ErrorStatus(http.StatusUnprocessableEntity, errors.New(c.GetErrMsg()))
  208. return
  209. }
  210. if ctxUser.IsOrganization() && !c.User.IsAdmin {
  211. // Check ownership of organization.
  212. if !ctxUser.IsOwnedBy(c.User.ID) {
  213. c.ErrorStatus(http.StatusForbidden, errors.New("Given user is not owner of organization."))
  214. return
  215. }
  216. }
  217. remoteAddr, err := f.ParseRemoteAddr(c.User)
  218. if err != nil {
  219. if db.IsErrInvalidCloneAddr(err) {
  220. addrErr := err.(db.ErrInvalidCloneAddr)
  221. switch {
  222. case addrErr.IsURLError:
  223. c.ErrorStatus(http.StatusUnprocessableEntity, err)
  224. case addrErr.IsPermissionDenied:
  225. c.ErrorStatus(http.StatusUnprocessableEntity, errors.New("You are not allowed to import local repositories."))
  226. case addrErr.IsInvalidPath:
  227. c.ErrorStatus(http.StatusUnprocessableEntity, errors.New("Invalid local path, it does not exist or not a directory."))
  228. case addrErr.IsBlockedLocalAddress:
  229. c.ErrorStatus(http.StatusUnprocessableEntity, errors.New("Clone address resolved to a local network address that is implicitly blocked."))
  230. default:
  231. c.Error(err, "unexpected error")
  232. }
  233. } else {
  234. c.Error(err, "parse remote address")
  235. }
  236. return
  237. }
  238. repo, err := db.MigrateRepository(c.User, ctxUser, db.MigrateRepoOptions{
  239. Name: f.RepoName,
  240. Description: f.Description,
  241. IsPrivate: f.Private || conf.Repository.ForcePrivate,
  242. IsMirror: f.Mirror,
  243. RemoteAddr: remoteAddr,
  244. })
  245. if err != nil {
  246. if repo != nil {
  247. if errDelete := db.DeleteRepository(ctxUser.ID, repo.ID); errDelete != nil {
  248. log.Error("DeleteRepository: %v", errDelete)
  249. }
  250. }
  251. if db.IsErrReachLimitOfRepo(err) {
  252. c.ErrorStatus(http.StatusUnprocessableEntity, err)
  253. } else {
  254. c.Error(errors.New(db.HandleMirrorCredentials(err.Error(), true)), "migrate repository")
  255. }
  256. return
  257. }
  258. log.Trace("Repository migrated: %s/%s", ctxUser.Name, f.RepoName)
  259. c.JSON(201, repo.APIFormatLegacy(&api.Permission{Admin: true, Push: true, Pull: true}))
  260. }
  261. // FIXME: inject in the handler chain
  262. func parseOwnerAndRepo(c *context.APIContext) (*db.User, *db.Repository) {
  263. owner, err := db.Users.GetByUsername(c.Req.Context(), c.Params(":username"))
  264. if err != nil {
  265. if db.IsErrUserNotExist(err) {
  266. c.ErrorStatus(http.StatusUnprocessableEntity, err)
  267. } else {
  268. c.Error(err, "get user by name")
  269. }
  270. return nil, nil
  271. }
  272. repo, err := db.GetRepositoryByName(owner.ID, c.Params(":reponame"))
  273. if err != nil {
  274. c.NotFoundOrError(err, "get repository by name")
  275. return nil, nil
  276. }
  277. return owner, repo
  278. }
  279. func Get(c *context.APIContext) {
  280. _, repo := parseOwnerAndRepo(c)
  281. if c.Written() {
  282. return
  283. }
  284. c.JSONSuccess(repo.APIFormatLegacy(&api.Permission{
  285. Admin: c.Repo.IsAdmin(),
  286. Push: c.Repo.IsWriter(),
  287. Pull: true,
  288. }))
  289. }
  290. func Delete(c *context.APIContext) {
  291. owner, repo := parseOwnerAndRepo(c)
  292. if c.Written() {
  293. return
  294. }
  295. if owner.IsOrganization() && !owner.IsOwnedBy(c.User.ID) {
  296. c.ErrorStatus(http.StatusForbidden, errors.New("Given user is not owner of organization."))
  297. return
  298. }
  299. if err := db.DeleteRepository(owner.ID, repo.ID); err != nil {
  300. c.Error(err, "delete repository")
  301. return
  302. }
  303. log.Trace("Repository deleted: %s/%s", owner.Name, repo.Name)
  304. c.NoContent()
  305. }
  306. func ListForks(c *context.APIContext) {
  307. forks, err := c.Repo.Repository.GetForks()
  308. if err != nil {
  309. c.Error(err, "get forks")
  310. return
  311. }
  312. apiForks := make([]*api.Repository, len(forks))
  313. for i := range forks {
  314. if err := forks[i].GetOwner(); err != nil {
  315. c.Error(err, "get owner")
  316. return
  317. }
  318. accessMode := db.Perms.AccessMode(
  319. c.Req.Context(),
  320. c.User.ID,
  321. forks[i].ID,
  322. db.AccessModeOptions{
  323. OwnerID: forks[i].OwnerID,
  324. Private: forks[i].IsPrivate,
  325. },
  326. )
  327. apiForks[i] = forks[i].APIFormatLegacy(
  328. &api.Permission{
  329. Admin: accessMode >= db.AccessModeAdmin,
  330. Push: accessMode >= db.AccessModeWrite,
  331. Pull: true,
  332. },
  333. )
  334. }
  335. c.JSONSuccess(&apiForks)
  336. }
  337. func IssueTracker(c *context.APIContext, form api.EditIssueTrackerOption) {
  338. _, repo := parseOwnerAndRepo(c)
  339. if c.Written() {
  340. return
  341. }
  342. if form.EnableIssues != nil {
  343. repo.EnableIssues = *form.EnableIssues
  344. }
  345. if form.EnableExternalTracker != nil {
  346. repo.EnableExternalTracker = *form.EnableExternalTracker
  347. }
  348. if form.ExternalTrackerURL != nil {
  349. repo.ExternalTrackerURL = *form.ExternalTrackerURL
  350. }
  351. if form.TrackerURLFormat != nil {
  352. repo.ExternalTrackerFormat = *form.TrackerURLFormat
  353. }
  354. if form.TrackerIssueStyle != nil {
  355. repo.ExternalTrackerStyle = *form.TrackerIssueStyle
  356. }
  357. if err := db.UpdateRepository(repo, false); err != nil {
  358. c.Error(err, "update repository")
  359. return
  360. }
  361. c.NoContent()
  362. }
  363. func Wiki(c *context.APIContext, form api.EditWikiOption) {
  364. _, repo := parseOwnerAndRepo(c)
  365. if c.Written() {
  366. return
  367. }
  368. if form.AllowPublicWiki != nil {
  369. repo.AllowPublicWiki = *form.AllowPublicWiki
  370. }
  371. if form.EnableExternalWiki != nil {
  372. repo.EnableExternalWiki = *form.EnableExternalWiki
  373. }
  374. if form.EnableWiki != nil {
  375. repo.EnableWiki = *form.EnableWiki
  376. }
  377. if form.ExternalWikiURL != nil {
  378. repo.ExternalWikiURL = *form.ExternalWikiURL
  379. }
  380. if err := db.UpdateRepository(repo, false); err != nil {
  381. c.Error(err, "update repository")
  382. return
  383. }
  384. c.NoContent()
  385. }
  386. func MirrorSync(c *context.APIContext) {
  387. _, repo := parseOwnerAndRepo(c)
  388. if c.Written() {
  389. return
  390. } else if !repo.IsMirror {
  391. c.NotFound()
  392. return
  393. }
  394. go db.MirrorQueue.Add(repo.ID)
  395. c.Status(http.StatusAccepted)
  396. }
  397. func Releases(c *context.APIContext) {
  398. _, repo := parseOwnerAndRepo(c)
  399. releases, err := db.GetReleasesByRepoID(repo.ID)
  400. if err != nil {
  401. c.Error(err, "get releases by repository ID")
  402. return
  403. }
  404. apiReleases := make([]*api.Release, 0, len(releases))
  405. for _, r := range releases {
  406. publisher, err := db.Users.GetByID(c.Req.Context(), r.PublisherID)
  407. if err != nil {
  408. c.Error(err, "get release publisher")
  409. return
  410. }
  411. r.Publisher = publisher
  412. }
  413. for _, r := range releases {
  414. apiReleases = append(apiReleases, r.APIFormat())
  415. }
  416. c.JSONSuccess(&apiReleases)
  417. }