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.

173 lines
7.7 KiB

  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. "net/http"
  8. "testing"
  9. "time"
  10. "code.gitea.io/gitea/models"
  11. "code.gitea.io/gitea/modules/setting"
  12. api "code.gitea.io/sdk/gitea"
  13. "github.com/stretchr/testify/assert"
  14. )
  15. func TestAPILFSLocksNotStarted(t *testing.T) {
  16. prepareTestEnv(t)
  17. setting.LFS.StartServer = false
  18. user := models.AssertExistsAndLoadBean(t, &models.User{ID: 2}).(*models.User)
  19. repo := models.AssertExistsAndLoadBean(t, &models.Repository{ID: 1}).(*models.Repository)
  20. req := NewRequestf(t, "GET", "/%s/%s.git/info/lfs/locks", user.Name, repo.Name)
  21. MakeRequest(t, req, http.StatusNotFound)
  22. req = NewRequestf(t, "POST", "/%s/%s.git/info/lfs/locks", user.Name, repo.Name)
  23. MakeRequest(t, req, http.StatusNotFound)
  24. req = NewRequestf(t, "GET", "/%s/%s.git/info/lfs/locks/verify", user.Name, repo.Name)
  25. MakeRequest(t, req, http.StatusNotFound)
  26. req = NewRequestf(t, "GET", "/%s/%s.git/info/lfs/locks/10/unlock", user.Name, repo.Name)
  27. MakeRequest(t, req, http.StatusNotFound)
  28. }
  29. func TestAPILFSLocksNotLogin(t *testing.T) {
  30. prepareTestEnv(t)
  31. setting.LFS.StartServer = true
  32. user := models.AssertExistsAndLoadBean(t, &models.User{ID: 2}).(*models.User)
  33. repo := models.AssertExistsAndLoadBean(t, &models.Repository{ID: 1}).(*models.Repository)
  34. req := NewRequestf(t, "GET", "/%s/%s.git/info/lfs/locks", user.Name, repo.Name)
  35. req.Header.Set("Accept", "application/vnd.git-lfs+json")
  36. resp := MakeRequest(t, req, http.StatusUnauthorized)
  37. var lfsLockError api.LFSLockError
  38. DecodeJSON(t, resp, &lfsLockError)
  39. assert.Equal(t, "Unauthorized", lfsLockError.Message)
  40. }
  41. func TestAPILFSLocksLogged(t *testing.T) {
  42. prepareTestEnv(t)
  43. setting.LFS.StartServer = true
  44. user2 := models.AssertExistsAndLoadBean(t, &models.User{ID: 2}).(*models.User) //in org 3
  45. user4 := models.AssertExistsAndLoadBean(t, &models.User{ID: 4}).(*models.User) //in org 3
  46. repo1 := models.AssertExistsAndLoadBean(t, &models.Repository{ID: 1}).(*models.Repository)
  47. repo3 := models.AssertExistsAndLoadBean(t, &models.Repository{ID: 3}).(*models.Repository) // own by org 3
  48. tests := []struct {
  49. user *models.User
  50. repo *models.Repository
  51. path string
  52. httpResult int
  53. addTime []int
  54. }{
  55. {user: user2, repo: repo1, path: "foo/bar.zip", httpResult: http.StatusCreated, addTime: []int{0}},
  56. {user: user2, repo: repo1, path: "path/test", httpResult: http.StatusCreated, addTime: []int{0}},
  57. {user: user2, repo: repo1, path: "path/test", httpResult: http.StatusConflict},
  58. {user: user2, repo: repo1, path: "Foo/BaR.zip", httpResult: http.StatusConflict},
  59. {user: user2, repo: repo1, path: "Foo/Test/../subFOlder/../Relative/../BaR.zip", httpResult: http.StatusConflict},
  60. {user: user4, repo: repo1, path: "FoO/BaR.zip", httpResult: http.StatusUnauthorized},
  61. {user: user4, repo: repo1, path: "path/test-user4", httpResult: http.StatusUnauthorized},
  62. {user: user2, repo: repo1, path: "patH/Test-user4", httpResult: http.StatusCreated, addTime: []int{0}},
  63. {user: user2, repo: repo1, path: "some/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/long/path", httpResult: http.StatusCreated, addTime: []int{0}},
  64. {user: user2, repo: repo3, path: "test/foo/bar.zip", httpResult: http.StatusCreated, addTime: []int{1, 2}},
  65. {user: user4, repo: repo3, path: "test/foo/bar.zip", httpResult: http.StatusConflict},
  66. {user: user4, repo: repo3, path: "test/foo/bar.bin", httpResult: http.StatusCreated, addTime: []int{1, 2}},
  67. }
  68. resultsTests := []struct {
  69. user *models.User
  70. repo *models.Repository
  71. totalCount int
  72. oursCount int
  73. theirsCount int
  74. locksOwners []*models.User
  75. locksTimes []time.Time
  76. }{
  77. {user: user2, repo: repo1, totalCount: 4, oursCount: 4, theirsCount: 0, locksOwners: []*models.User{user2, user2, user2, user2}, locksTimes: []time.Time{}},
  78. {user: user2, repo: repo3, totalCount: 2, oursCount: 1, theirsCount: 1, locksOwners: []*models.User{user2, user4}, locksTimes: []time.Time{}},
  79. {user: user4, repo: repo3, totalCount: 2, oursCount: 1, theirsCount: 1, locksOwners: []*models.User{user2, user4}, locksTimes: []time.Time{}},
  80. }
  81. deleteTests := []struct {
  82. user *models.User
  83. repo *models.Repository
  84. lockID string
  85. }{}
  86. //create locks
  87. for _, test := range tests {
  88. session := loginUser(t, test.user.Name)
  89. req := NewRequestWithJSON(t, "POST", fmt.Sprintf("/%s.git/info/lfs/locks", test.repo.FullName()), map[string]string{"path": test.path})
  90. req.Header.Set("Accept", "application/vnd.git-lfs+json")
  91. req.Header.Set("Content-Type", "application/vnd.git-lfs+json")
  92. session.MakeRequest(t, req, test.httpResult)
  93. if len(test.addTime) > 0 {
  94. for _, id := range test.addTime {
  95. resultsTests[id].locksTimes = append(resultsTests[id].locksTimes, time.Now())
  96. }
  97. }
  98. }
  99. //check creation
  100. for _, test := range resultsTests {
  101. session := loginUser(t, test.user.Name)
  102. req := NewRequestf(t, "GET", "/%s.git/info/lfs/locks", test.repo.FullName())
  103. req.Header.Set("Accept", "application/vnd.git-lfs+json")
  104. resp := session.MakeRequest(t, req, http.StatusOK)
  105. var lfsLocks api.LFSLockList
  106. DecodeJSON(t, resp, &lfsLocks)
  107. assert.Len(t, lfsLocks.Locks, test.totalCount)
  108. for i, lock := range lfsLocks.Locks {
  109. assert.EqualValues(t, test.locksOwners[i].DisplayName(), lock.Owner.Name)
  110. assert.WithinDuration(t, test.locksTimes[i], lock.LockedAt, 3*time.Second)
  111. }
  112. req = NewRequestWithJSON(t, "POST", fmt.Sprintf("/%s.git/info/lfs/locks/verify", test.repo.FullName()), map[string]string{})
  113. req.Header.Set("Accept", "application/vnd.git-lfs+json")
  114. req.Header.Set("Content-Type", "application/vnd.git-lfs+json")
  115. resp = session.MakeRequest(t, req, http.StatusOK)
  116. var lfsLocksVerify api.LFSLockListVerify
  117. DecodeJSON(t, resp, &lfsLocksVerify)
  118. assert.Len(t, lfsLocksVerify.Ours, test.oursCount)
  119. assert.Len(t, lfsLocksVerify.Theirs, test.theirsCount)
  120. for _, lock := range lfsLocksVerify.Ours {
  121. assert.EqualValues(t, test.user.DisplayName(), lock.Owner.Name)
  122. deleteTests = append(deleteTests, struct {
  123. user *models.User
  124. repo *models.Repository
  125. lockID string
  126. }{test.user, test.repo, lock.ID})
  127. }
  128. for _, lock := range lfsLocksVerify.Theirs {
  129. assert.NotEqual(t, test.user.DisplayName(), lock.Owner.Name)
  130. }
  131. }
  132. //remove all locks
  133. for _, test := range deleteTests {
  134. session := loginUser(t, test.user.Name)
  135. req := NewRequestWithJSON(t, "POST", fmt.Sprintf("/%s.git/info/lfs/locks/%s/unlock", test.repo.FullName(), test.lockID), map[string]string{})
  136. req.Header.Set("Accept", "application/vnd.git-lfs+json")
  137. req.Header.Set("Content-Type", "application/vnd.git-lfs+json")
  138. resp := session.MakeRequest(t, req, http.StatusOK)
  139. var lfsLockRep api.LFSLockResponse
  140. DecodeJSON(t, resp, &lfsLockRep)
  141. assert.Equal(t, test.lockID, lfsLockRep.Lock.ID)
  142. assert.Equal(t, test.user.DisplayName(), lfsLockRep.Lock.Owner.Name)
  143. }
  144. // check that we don't have any lock
  145. for _, test := range resultsTests {
  146. session := loginUser(t, test.user.Name)
  147. req := NewRequestf(t, "GET", "/%s.git/info/lfs/locks", test.repo.FullName())
  148. req.Header.Set("Accept", "application/vnd.git-lfs+json")
  149. resp := session.MakeRequest(t, req, http.StatusOK)
  150. var lfsLocks api.LFSLockList
  151. DecodeJSON(t, resp, &lfsLocks)
  152. assert.Len(t, lfsLocks.Locks, 0)
  153. }
  154. }