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.

api_repo_test.go 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412
  1. // Copyright 2017 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 integrations
  5. import (
  6. "fmt"
  7. "io/ioutil"
  8. "net/http"
  9. "net/url"
  10. "os"
  11. "testing"
  12. "code.gitea.io/gitea/models"
  13. api "code.gitea.io/gitea/modules/structs"
  14. "github.com/stretchr/testify/assert"
  15. )
  16. func TestAPIUserReposNotLogin(t *testing.T) {
  17. defer prepareTestEnv(t)()
  18. user := models.AssertExistsAndLoadBean(t, &models.User{ID: 2}).(*models.User)
  19. req := NewRequestf(t, "GET", "/api/v1/users/%s/repos", user.Name)
  20. resp := MakeRequest(t, req, http.StatusOK)
  21. var apiRepos []api.Repository
  22. DecodeJSON(t, resp, &apiRepos)
  23. expectedLen := models.GetCount(t, models.Repository{OwnerID: user.ID},
  24. models.Cond("is_private = ?", false))
  25. assert.Len(t, apiRepos, expectedLen)
  26. for _, repo := range apiRepos {
  27. assert.EqualValues(t, user.ID, repo.Owner.ID)
  28. assert.False(t, repo.Private)
  29. }
  30. }
  31. func TestAPISearchRepo(t *testing.T) {
  32. defer prepareTestEnv(t)()
  33. const keyword = "test"
  34. req := NewRequestf(t, "GET", "/api/v1/repos/search?q=%s", keyword)
  35. resp := MakeRequest(t, req, http.StatusOK)
  36. var body api.SearchResults
  37. DecodeJSON(t, resp, &body)
  38. assert.NotEmpty(t, body.Data)
  39. for _, repo := range body.Data {
  40. assert.Contains(t, repo.Name, keyword)
  41. assert.False(t, repo.Private)
  42. }
  43. user := models.AssertExistsAndLoadBean(t, &models.User{ID: 15}).(*models.User)
  44. user2 := models.AssertExistsAndLoadBean(t, &models.User{ID: 16}).(*models.User)
  45. user3 := models.AssertExistsAndLoadBean(t, &models.User{ID: 18}).(*models.User)
  46. user4 := models.AssertExistsAndLoadBean(t, &models.User{ID: 20}).(*models.User)
  47. orgUser := models.AssertExistsAndLoadBean(t, &models.User{ID: 17}).(*models.User)
  48. // Map of expected results, where key is user for login
  49. type expectedResults map[*models.User]struct {
  50. count int
  51. repoOwnerID int64
  52. repoName string
  53. includesPrivate bool
  54. }
  55. testCases := []struct {
  56. name, requestURL string
  57. expectedResults
  58. }{
  59. {name: "RepositoriesMax50", requestURL: "/api/v1/repos/search?limit=50&private=false", expectedResults: expectedResults{
  60. nil: {count: 27},
  61. user: {count: 27},
  62. user2: {count: 27}},
  63. },
  64. {name: "RepositoriesMax10", requestURL: "/api/v1/repos/search?limit=10&private=false", expectedResults: expectedResults{
  65. nil: {count: 10},
  66. user: {count: 10},
  67. user2: {count: 10}},
  68. },
  69. {name: "RepositoriesDefaultMax10", requestURL: "/api/v1/repos/search?default&private=false", expectedResults: expectedResults{
  70. nil: {count: 10},
  71. user: {count: 10},
  72. user2: {count: 10}},
  73. },
  74. {name: "RepositoriesByName", requestURL: fmt.Sprintf("/api/v1/repos/search?q=%s&private=false", "big_test_"), expectedResults: expectedResults{
  75. nil: {count: 7, repoName: "big_test_"},
  76. user: {count: 7, repoName: "big_test_"},
  77. user2: {count: 7, repoName: "big_test_"}},
  78. },
  79. {name: "RepositoriesAccessibleAndRelatedToUser", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d", user.ID), expectedResults: expectedResults{
  80. nil: {count: 5},
  81. user: {count: 9, includesPrivate: true},
  82. user2: {count: 6, includesPrivate: true}},
  83. },
  84. {name: "RepositoriesAccessibleAndRelatedToUser2", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d", user2.ID), expectedResults: expectedResults{
  85. nil: {count: 1},
  86. user: {count: 2, includesPrivate: true},
  87. user2: {count: 2, includesPrivate: true},
  88. user4: {count: 1}},
  89. },
  90. {name: "RepositoriesAccessibleAndRelatedToUser3", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d", user3.ID), expectedResults: expectedResults{
  91. nil: {count: 1},
  92. user: {count: 4, includesPrivate: true},
  93. user2: {count: 3, includesPrivate: true},
  94. user3: {count: 4, includesPrivate: true}},
  95. },
  96. {name: "RepositoriesOwnedByOrganization", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d", orgUser.ID), expectedResults: expectedResults{
  97. nil: {count: 1, repoOwnerID: orgUser.ID},
  98. user: {count: 2, repoOwnerID: orgUser.ID, includesPrivate: true},
  99. user2: {count: 1, repoOwnerID: orgUser.ID}},
  100. },
  101. {name: "RepositoriesAccessibleAndRelatedToUser4", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d", user4.ID), expectedResults: expectedResults{
  102. nil: {count: 3},
  103. user: {count: 4, includesPrivate: true},
  104. user4: {count: 7, includesPrivate: true}}},
  105. {name: "RepositoriesAccessibleAndRelatedToUser4/SearchModeSource", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d&mode=%s", user4.ID, "source"), expectedResults: expectedResults{
  106. nil: {count: 0},
  107. user: {count: 1, includesPrivate: true},
  108. user4: {count: 1, includesPrivate: true}}},
  109. {name: "RepositoriesAccessibleAndRelatedToUser4/SearchModeFork", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d&mode=%s", user4.ID, "fork"), expectedResults: expectedResults{
  110. nil: {count: 1},
  111. user: {count: 1},
  112. user4: {count: 2, includesPrivate: true}}},
  113. {name: "RepositoriesAccessibleAndRelatedToUser4/SearchModeFork/Exclusive", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d&mode=%s&exclusive=1", user4.ID, "fork"), expectedResults: expectedResults{
  114. nil: {count: 1},
  115. user: {count: 1},
  116. user4: {count: 2, includesPrivate: true}}},
  117. {name: "RepositoriesAccessibleAndRelatedToUser4/SearchModeMirror", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d&mode=%s", user4.ID, "mirror"), expectedResults: expectedResults{
  118. nil: {count: 2},
  119. user: {count: 2},
  120. user4: {count: 4, includesPrivate: true}}},
  121. {name: "RepositoriesAccessibleAndRelatedToUser4/SearchModeMirror/Exclusive", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d&mode=%s&exclusive=1", user4.ID, "mirror"), expectedResults: expectedResults{
  122. nil: {count: 1},
  123. user: {count: 1},
  124. user4: {count: 2, includesPrivate: true}}},
  125. {name: "RepositoriesAccessibleAndRelatedToUser4/SearchModeCollaborative", requestURL: fmt.Sprintf("/api/v1/repos/search?uid=%d&mode=%s", user4.ID, "collaborative"), expectedResults: expectedResults{
  126. nil: {count: 0},
  127. user: {count: 1, includesPrivate: true},
  128. user4: {count: 1, includesPrivate: true}}},
  129. }
  130. for _, testCase := range testCases {
  131. t.Run(testCase.name, func(t *testing.T) {
  132. for userToLogin, expected := range testCase.expectedResults {
  133. var session *TestSession
  134. var testName string
  135. var userID int64
  136. var token string
  137. if userToLogin != nil && userToLogin.ID > 0 {
  138. testName = fmt.Sprintf("LoggedUser%d", userToLogin.ID)
  139. session = loginUser(t, userToLogin.Name)
  140. token = getTokenForLoggedInUser(t, session)
  141. userID = userToLogin.ID
  142. } else {
  143. testName = "AnonymousUser"
  144. session = emptyTestSession(t)
  145. }
  146. t.Run(testName, func(t *testing.T) {
  147. request := NewRequest(t, "GET", testCase.requestURL+"&token="+token)
  148. response := session.MakeRequest(t, request, http.StatusOK)
  149. var body api.SearchResults
  150. DecodeJSON(t, response, &body)
  151. repoNames := make([]string, 0, len(body.Data))
  152. for _, repo := range body.Data {
  153. repoNames = append(repoNames, fmt.Sprintf("%d:%s:%t", repo.ID, repo.FullName, repo.Private))
  154. }
  155. assert.Len(t, repoNames, expected.count)
  156. for _, repo := range body.Data {
  157. r := getRepo(t, repo.ID)
  158. hasAccess, err := models.HasAccess(userID, r)
  159. assert.NoError(t, err, "Error when checking if User: %d has access to %s: %v", userID, repo.FullName, err)
  160. assert.True(t, hasAccess, "User: %d does not have access to %s", userID, repo.FullName)
  161. assert.NotEmpty(t, repo.Name)
  162. assert.Equal(t, repo.Name, r.Name)
  163. if len(expected.repoName) > 0 {
  164. assert.Contains(t, repo.Name, expected.repoName)
  165. }
  166. if expected.repoOwnerID > 0 {
  167. assert.Equal(t, expected.repoOwnerID, repo.Owner.ID)
  168. }
  169. if !expected.includesPrivate {
  170. assert.False(t, repo.Private, "User: %d not expecting private repository: %s", userID, repo.FullName)
  171. }
  172. }
  173. })
  174. }
  175. })
  176. }
  177. }
  178. var repoCache = make(map[int64]*models.Repository)
  179. func getRepo(t *testing.T, repoID int64) *models.Repository {
  180. if _, ok := repoCache[repoID]; !ok {
  181. repoCache[repoID] = models.AssertExistsAndLoadBean(t, &models.Repository{ID: repoID}).(*models.Repository)
  182. }
  183. return repoCache[repoID]
  184. }
  185. func TestAPIViewRepo(t *testing.T) {
  186. defer prepareTestEnv(t)()
  187. var repo api.Repository
  188. req := NewRequest(t, "GET", "/api/v1/repos/user2/repo1")
  189. resp := MakeRequest(t, req, http.StatusOK)
  190. DecodeJSON(t, resp, &repo)
  191. assert.EqualValues(t, 1, repo.ID)
  192. assert.EqualValues(t, "repo1", repo.Name)
  193. assert.EqualValues(t, 1, repo.Releases)
  194. assert.EqualValues(t, 1, repo.OpenIssues)
  195. assert.EqualValues(t, 2, repo.OpenPulls)
  196. req = NewRequest(t, "GET", "/api/v1/repos/user12/repo10")
  197. resp = MakeRequest(t, req, http.StatusOK)
  198. DecodeJSON(t, resp, &repo)
  199. assert.EqualValues(t, 10, repo.ID)
  200. assert.EqualValues(t, "repo10", repo.Name)
  201. assert.EqualValues(t, 1, repo.OpenPulls)
  202. assert.EqualValues(t, 1, repo.Forks)
  203. req = NewRequest(t, "GET", "/api/v1/repos/user5/repo4")
  204. resp = MakeRequest(t, req, http.StatusOK)
  205. DecodeJSON(t, resp, &repo)
  206. assert.EqualValues(t, 4, repo.ID)
  207. assert.EqualValues(t, "repo4", repo.Name)
  208. assert.EqualValues(t, 1, repo.Stars)
  209. }
  210. func TestAPIOrgRepos(t *testing.T) {
  211. defer prepareTestEnv(t)()
  212. user := models.AssertExistsAndLoadBean(t, &models.User{ID: 2}).(*models.User)
  213. user2 := models.AssertExistsAndLoadBean(t, &models.User{ID: 1}).(*models.User)
  214. user3 := models.AssertExistsAndLoadBean(t, &models.User{ID: 5}).(*models.User)
  215. // User3 is an Org. Check their repos.
  216. sourceOrg := models.AssertExistsAndLoadBean(t, &models.User{ID: 3}).(*models.User)
  217. expectedResults := map[*models.User]struct {
  218. count int
  219. includesPrivate bool
  220. }{
  221. nil: {count: 1},
  222. user: {count: 2, includesPrivate: true},
  223. user2: {count: 3, includesPrivate: true},
  224. user3: {count: 1},
  225. }
  226. for userToLogin, expected := range expectedResults {
  227. var session *TestSession
  228. var testName string
  229. var token string
  230. if userToLogin != nil && userToLogin.ID > 0 {
  231. testName = fmt.Sprintf("LoggedUser%d", userToLogin.ID)
  232. session = loginUser(t, userToLogin.Name)
  233. token = getTokenForLoggedInUser(t, session)
  234. } else {
  235. testName = "AnonymousUser"
  236. session = emptyTestSession(t)
  237. }
  238. t.Run(testName, func(t *testing.T) {
  239. req := NewRequestf(t, "GET", "/api/v1/orgs/%s/repos?token="+token, sourceOrg.Name)
  240. resp := session.MakeRequest(t, req, http.StatusOK)
  241. var apiRepos []*api.Repository
  242. DecodeJSON(t, resp, &apiRepos)
  243. assert.Len(t, apiRepos, expected.count)
  244. for _, repo := range apiRepos {
  245. if !expected.includesPrivate {
  246. assert.False(t, repo.Private)
  247. }
  248. }
  249. })
  250. }
  251. }
  252. func TestAPIGetRepoByIDUnauthorized(t *testing.T) {
  253. defer prepareTestEnv(t)()
  254. user := models.AssertExistsAndLoadBean(t, &models.User{ID: 4}).(*models.User)
  255. session := loginUser(t, user.Name)
  256. token := getTokenForLoggedInUser(t, session)
  257. req := NewRequestf(t, "GET", "/api/v1/repositories/2?token="+token)
  258. session.MakeRequest(t, req, http.StatusNotFound)
  259. }
  260. func TestAPIRepoMigrate(t *testing.T) {
  261. testCases := []struct {
  262. ctxUserID, userID int64
  263. cloneURL, repoName string
  264. expectedStatus int
  265. }{
  266. {ctxUserID: 1, userID: 2, cloneURL: "https://github.com/go-gitea/git.git", repoName: "git-admin", expectedStatus: http.StatusCreated},
  267. {ctxUserID: 2, userID: 2, cloneURL: "https://github.com/go-gitea/git.git", repoName: "git-own", expectedStatus: http.StatusCreated},
  268. {ctxUserID: 2, userID: 1, cloneURL: "https://github.com/go-gitea/git.git", repoName: "git-bad", expectedStatus: http.StatusForbidden},
  269. {ctxUserID: 2, userID: 3, cloneURL: "https://github.com/go-gitea/git.git", repoName: "git-org", expectedStatus: http.StatusCreated},
  270. {ctxUserID: 2, userID: 6, cloneURL: "https://github.com/go-gitea/git.git", repoName: "git-bad-org", expectedStatus: http.StatusForbidden},
  271. }
  272. defer prepareTestEnv(t)()
  273. for _, testCase := range testCases {
  274. user := models.AssertExistsAndLoadBean(t, &models.User{ID: testCase.ctxUserID}).(*models.User)
  275. session := loginUser(t, user.Name)
  276. token := getTokenForLoggedInUser(t, session)
  277. req := NewRequestWithJSON(t, "POST", "/api/v1/repos/migrate?token="+token, &api.MigrateRepoOption{
  278. CloneAddr: testCase.cloneURL,
  279. UID: int(testCase.userID),
  280. RepoName: testCase.repoName,
  281. })
  282. session.MakeRequest(t, req, testCase.expectedStatus)
  283. }
  284. }
  285. func TestAPIRepoMigrateConflict(t *testing.T) {
  286. onGiteaRun(t, testAPIRepoMigrateConflict)
  287. }
  288. func testAPIRepoMigrateConflict(t *testing.T, u *url.URL) {
  289. username := "user2"
  290. baseAPITestContext := NewAPITestContext(t, username, "repo1")
  291. u.Path = baseAPITestContext.GitPath()
  292. t.Run("Existing", func(t *testing.T) {
  293. httpContext := baseAPITestContext
  294. httpContext.Reponame = "repo-tmp-17"
  295. dstPath, err := ioutil.TempDir("", httpContext.Reponame)
  296. assert.NoError(t, err)
  297. defer os.RemoveAll(dstPath)
  298. t.Run("CreateRepo", doAPICreateRepository(httpContext, false))
  299. user, err := models.GetUserByName(httpContext.Username)
  300. assert.NoError(t, err)
  301. userID := user.ID
  302. cloneURL := "https://github.com/go-gitea/git.git"
  303. req := NewRequestWithJSON(t, "POST", "/api/v1/repos/migrate?token="+httpContext.Token,
  304. &api.MigrateRepoOption{
  305. CloneAddr: cloneURL,
  306. UID: int(userID),
  307. RepoName: httpContext.Reponame,
  308. })
  309. resp := httpContext.Session.MakeRequest(t, req, http.StatusConflict)
  310. respJSON := map[string]string{}
  311. DecodeJSON(t, resp, &respJSON)
  312. assert.Equal(t, "The repository with the same name already exists.", respJSON["message"])
  313. })
  314. }
  315. func TestAPIOrgRepoCreate(t *testing.T) {
  316. testCases := []struct {
  317. ctxUserID int64
  318. orgName, repoName string
  319. expectedStatus int
  320. }{
  321. {ctxUserID: 1, orgName: "user3", repoName: "repo-admin", expectedStatus: http.StatusCreated},
  322. {ctxUserID: 2, orgName: "user3", repoName: "repo-own", expectedStatus: http.StatusCreated},
  323. {ctxUserID: 2, orgName: "user6", repoName: "repo-bad-org", expectedStatus: http.StatusForbidden},
  324. {ctxUserID: 28, orgName: "user3", repoName: "repo-creator", expectedStatus: http.StatusCreated},
  325. {ctxUserID: 28, orgName: "user6", repoName: "repo-not-creator", expectedStatus: http.StatusForbidden},
  326. }
  327. defer prepareTestEnv(t)()
  328. for _, testCase := range testCases {
  329. user := models.AssertExistsAndLoadBean(t, &models.User{ID: testCase.ctxUserID}).(*models.User)
  330. session := loginUser(t, user.Name)
  331. token := getTokenForLoggedInUser(t, session)
  332. req := NewRequestWithJSON(t, "POST", fmt.Sprintf("/api/v1/org/%s/repos?token="+token, testCase.orgName), &api.CreateRepoOption{
  333. Name: testCase.repoName,
  334. })
  335. session.MakeRequest(t, req, testCase.expectedStatus)
  336. }
  337. }
  338. func TestAPIRepoCreateConflict(t *testing.T) {
  339. onGiteaRun(t, testAPIRepoCreateConflict)
  340. }
  341. func testAPIRepoCreateConflict(t *testing.T, u *url.URL) {
  342. username := "user2"
  343. baseAPITestContext := NewAPITestContext(t, username, "repo1")
  344. u.Path = baseAPITestContext.GitPath()
  345. t.Run("Existing", func(t *testing.T) {
  346. httpContext := baseAPITestContext
  347. httpContext.Reponame = "repo-tmp-17"
  348. dstPath, err := ioutil.TempDir("", httpContext.Reponame)
  349. assert.NoError(t, err)
  350. defer os.RemoveAll(dstPath)
  351. t.Run("CreateRepo", doAPICreateRepository(httpContext, false))
  352. req := NewRequestWithJSON(t, "POST", "/api/v1/user/repos?token="+httpContext.Token,
  353. &api.CreateRepoOption{
  354. Name: httpContext.Reponame,
  355. })
  356. resp := httpContext.Session.MakeRequest(t, req, http.StatusConflict)
  357. respJSON := map[string]string{}
  358. DecodeJSON(t, resp, &respJSON)
  359. assert.Equal(t, respJSON["message"], "The repository with the same name already exists.")
  360. })
  361. }