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.

user_test.go 9.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339
  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 models
  5. import (
  6. "fmt"
  7. "math/rand"
  8. "strings"
  9. "testing"
  10. "code.gitea.io/gitea/modules/setting"
  11. "code.gitea.io/gitea/modules/util"
  12. "github.com/stretchr/testify/assert"
  13. )
  14. func TestUserIsPublicMember(t *testing.T) {
  15. assert.NoError(t, PrepareTestDatabase())
  16. tt := []struct {
  17. uid int64
  18. orgid int64
  19. expected bool
  20. }{
  21. {2, 3, true},
  22. {4, 3, false},
  23. {5, 6, true},
  24. {5, 7, false},
  25. }
  26. for _, v := range tt {
  27. t.Run(fmt.Sprintf("UserId%dIsPublicMemberOf%d", v.uid, v.orgid), func(t *testing.T) {
  28. testUserIsPublicMember(t, v.uid, v.orgid, v.expected)
  29. })
  30. }
  31. }
  32. func testUserIsPublicMember(t *testing.T, uid int64, orgID int64, expected bool) {
  33. user, err := GetUserByID(uid)
  34. assert.NoError(t, err)
  35. assert.Equal(t, expected, user.IsPublicMember(orgID))
  36. }
  37. func TestIsUserOrgOwner(t *testing.T) {
  38. assert.NoError(t, PrepareTestDatabase())
  39. tt := []struct {
  40. uid int64
  41. orgid int64
  42. expected bool
  43. }{
  44. {2, 3, true},
  45. {4, 3, false},
  46. {5, 6, true},
  47. {5, 7, true},
  48. }
  49. for _, v := range tt {
  50. t.Run(fmt.Sprintf("UserId%dIsOrgOwnerOf%d", v.uid, v.orgid), func(t *testing.T) {
  51. testIsUserOrgOwner(t, v.uid, v.orgid, v.expected)
  52. })
  53. }
  54. }
  55. func testIsUserOrgOwner(t *testing.T, uid int64, orgID int64, expected bool) {
  56. user, err := GetUserByID(uid)
  57. assert.NoError(t, err)
  58. assert.Equal(t, expected, user.IsUserOrgOwner(orgID))
  59. }
  60. func TestGetUserEmailsByNames(t *testing.T) {
  61. assert.NoError(t, PrepareTestDatabase())
  62. // ignore none active user email
  63. assert.Equal(t, []string{"user8@example.com"}, GetUserEmailsByNames([]string{"user8", "user9"}))
  64. assert.Equal(t, []string{"user8@example.com", "user5@example.com"}, GetUserEmailsByNames([]string{"user8", "user5"}))
  65. }
  66. func TestUser_APIFormat(t *testing.T) {
  67. user, err := GetUserByID(1)
  68. assert.NoError(t, err)
  69. assert.True(t, user.IsAdmin)
  70. apiUser := user.APIFormat()
  71. assert.True(t, apiUser.IsAdmin)
  72. user, err = GetUserByID(2)
  73. assert.NoError(t, err)
  74. assert.False(t, user.IsAdmin)
  75. apiUser = user.APIFormat()
  76. assert.False(t, apiUser.IsAdmin)
  77. }
  78. func TestCanCreateOrganization(t *testing.T) {
  79. assert.NoError(t, PrepareTestDatabase())
  80. admin := AssertExistsAndLoadBean(t, &User{ID: 1}).(*User)
  81. assert.True(t, admin.CanCreateOrganization())
  82. user := AssertExistsAndLoadBean(t, &User{ID: 2}).(*User)
  83. assert.True(t, user.CanCreateOrganization())
  84. // Disable user create organization permission.
  85. user.AllowCreateOrganization = false
  86. assert.False(t, user.CanCreateOrganization())
  87. setting.Admin.DisableRegularOrgCreation = true
  88. user.AllowCreateOrganization = true
  89. assert.True(t, admin.CanCreateOrganization())
  90. assert.False(t, user.CanCreateOrganization())
  91. }
  92. func TestSearchUsers(t *testing.T) {
  93. assert.NoError(t, PrepareTestDatabase())
  94. testSuccess := func(opts *SearchUserOptions, expectedUserOrOrgIDs []int64) {
  95. users, _, err := SearchUsers(opts)
  96. assert.NoError(t, err)
  97. if assert.Len(t, users, len(expectedUserOrOrgIDs)) {
  98. for i, expectedID := range expectedUserOrOrgIDs {
  99. assert.EqualValues(t, expectedID, users[i].ID)
  100. }
  101. }
  102. }
  103. // test orgs
  104. testOrgSuccess := func(opts *SearchUserOptions, expectedOrgIDs []int64) {
  105. opts.Type = UserTypeOrganization
  106. testSuccess(opts, expectedOrgIDs)
  107. }
  108. testOrgSuccess(&SearchUserOptions{OrderBy: "id ASC", Page: 1, PageSize: 2},
  109. []int64{3, 6})
  110. testOrgSuccess(&SearchUserOptions{OrderBy: "id ASC", Page: 2, PageSize: 2},
  111. []int64{7, 17})
  112. testOrgSuccess(&SearchUserOptions{OrderBy: "id ASC", Page: 3, PageSize: 2},
  113. []int64{19, 25})
  114. testOrgSuccess(&SearchUserOptions{Page: 4, PageSize: 2},
  115. []int64{})
  116. // test users
  117. testUserSuccess := func(opts *SearchUserOptions, expectedUserIDs []int64) {
  118. opts.Type = UserTypeIndividual
  119. testSuccess(opts, expectedUserIDs)
  120. }
  121. testUserSuccess(&SearchUserOptions{OrderBy: "id ASC", Page: 1},
  122. []int64{1, 2, 4, 5, 8, 9, 10, 11, 12, 13, 14, 15, 16, 18, 20, 21, 24})
  123. testUserSuccess(&SearchUserOptions{Page: 1, IsActive: util.OptionalBoolFalse},
  124. []int64{9})
  125. testUserSuccess(&SearchUserOptions{OrderBy: "id ASC", Page: 1, IsActive: util.OptionalBoolTrue},
  126. []int64{1, 2, 4, 5, 8, 10, 11, 12, 13, 14, 15, 16, 18, 20, 21, 24})
  127. testUserSuccess(&SearchUserOptions{Keyword: "user1", OrderBy: "id ASC", Page: 1, IsActive: util.OptionalBoolTrue},
  128. []int64{1, 10, 11, 12, 13, 14, 15, 16, 18})
  129. // order by name asc default
  130. testUserSuccess(&SearchUserOptions{Keyword: "user1", Page: 1, IsActive: util.OptionalBoolTrue},
  131. []int64{1, 10, 11, 12, 13, 14, 15, 16, 18})
  132. }
  133. func TestDeleteUser(t *testing.T) {
  134. test := func(userID int64) {
  135. assert.NoError(t, PrepareTestDatabase())
  136. user := AssertExistsAndLoadBean(t, &User{ID: userID}).(*User)
  137. ownedRepos := make([]*Repository, 0, 10)
  138. assert.NoError(t, x.Find(&ownedRepos, &Repository{OwnerID: userID}))
  139. if len(ownedRepos) > 0 {
  140. err := DeleteUser(user)
  141. assert.Error(t, err)
  142. assert.True(t, IsErrUserOwnRepos(err))
  143. return
  144. }
  145. orgUsers := make([]*OrgUser, 0, 10)
  146. assert.NoError(t, x.Find(&orgUsers, &OrgUser{UID: userID}))
  147. for _, orgUser := range orgUsers {
  148. if err := RemoveOrgUser(orgUser.OrgID, orgUser.UID); err != nil {
  149. assert.True(t, IsErrLastOrgOwner(err))
  150. return
  151. }
  152. }
  153. assert.NoError(t, DeleteUser(user))
  154. AssertNotExistsBean(t, &User{ID: userID})
  155. CheckConsistencyFor(t, &User{}, &Repository{})
  156. }
  157. test(2)
  158. test(4)
  159. test(8)
  160. test(11)
  161. }
  162. func TestHashPasswordDeterministic(t *testing.T) {
  163. b := make([]byte, 16)
  164. rand.Read(b)
  165. u := &User{Salt: string(b)}
  166. algos := []string{"pbkdf2", "argon2", "scrypt", "bcrypt"}
  167. for j := 0; j < len(algos); j++ {
  168. u.PasswdHashAlgo = algos[j]
  169. for i := 0; i < 50; i++ {
  170. // generate a random password
  171. rand.Read(b)
  172. pass := string(b)
  173. // save the current password in the user - hash it and store the result
  174. u.HashPassword(pass)
  175. r1 := u.Passwd
  176. // run again
  177. u.HashPassword(pass)
  178. r2 := u.Passwd
  179. // assert equal (given the same salt+pass, the same result is produced) except bcrypt
  180. if u.PasswdHashAlgo == "bcrypt" {
  181. assert.NotEqual(t, r1, r2)
  182. } else {
  183. assert.Equal(t, r1, r2)
  184. }
  185. }
  186. }
  187. }
  188. func BenchmarkHashPassword(b *testing.B) {
  189. // BenchmarkHashPassword ensures that it takes a reasonable amount of time
  190. // to hash a password - in order to protect from brute-force attacks.
  191. pass := "password1337"
  192. bs := make([]byte, 16)
  193. rand.Read(bs)
  194. u := &User{Salt: string(bs), Passwd: pass}
  195. b.ResetTimer()
  196. for i := 0; i < b.N; i++ {
  197. u.HashPassword(pass)
  198. }
  199. }
  200. func TestGetOrgRepositoryIDs(t *testing.T) {
  201. assert.NoError(t, PrepareTestDatabase())
  202. user2 := AssertExistsAndLoadBean(t, &User{ID: 2}).(*User)
  203. user4 := AssertExistsAndLoadBean(t, &User{ID: 4}).(*User)
  204. user5 := AssertExistsAndLoadBean(t, &User{ID: 5}).(*User)
  205. accessibleRepos, err := user2.GetOrgRepositoryIDs()
  206. assert.NoError(t, err)
  207. // User 2's team has access to private repos 3, 5, repo 32 is a public repo of the organization
  208. assert.Equal(t, []int64{3, 5, 23, 24, 32}, accessibleRepos)
  209. accessibleRepos, err = user4.GetOrgRepositoryIDs()
  210. assert.NoError(t, err)
  211. // User 4's team has access to private repo 3, repo 32 is a public repo of the organization
  212. assert.Equal(t, []int64{3, 32}, accessibleRepos)
  213. accessibleRepos, err = user5.GetOrgRepositoryIDs()
  214. assert.NoError(t, err)
  215. // User 5's team has no access to any repo
  216. assert.Len(t, accessibleRepos, 0)
  217. }
  218. func TestNewGitSig(t *testing.T) {
  219. users := make([]*User, 0, 20)
  220. sess := x.NewSession()
  221. defer sess.Close()
  222. sess.Find(&users)
  223. for _, user := range users {
  224. sig := user.NewGitSig()
  225. assert.NotContains(t, sig.Name, "<")
  226. assert.NotContains(t, sig.Name, ">")
  227. assert.NotContains(t, sig.Name, "\n")
  228. assert.NotEqual(t, len(strings.TrimSpace(sig.Name)), 0)
  229. }
  230. }
  231. func TestDisplayName(t *testing.T) {
  232. users := make([]*User, 0, 20)
  233. sess := x.NewSession()
  234. defer sess.Close()
  235. sess.Find(&users)
  236. for _, user := range users {
  237. displayName := user.DisplayName()
  238. assert.Equal(t, strings.TrimSpace(displayName), displayName)
  239. if len(strings.TrimSpace(user.FullName)) == 0 {
  240. assert.Equal(t, user.Name, displayName)
  241. }
  242. assert.NotEqual(t, len(strings.TrimSpace(displayName)), 0)
  243. }
  244. }
  245. func TestCreateUser(t *testing.T) {
  246. user := &User{
  247. Name: "GiteaBot",
  248. Email: "GiteaBot@gitea.io",
  249. Passwd: ";p['////..-++']",
  250. IsAdmin: false,
  251. Theme: setting.UI.DefaultTheme,
  252. MustChangePassword: false,
  253. }
  254. assert.NoError(t, CreateUser(user))
  255. assert.NoError(t, DeleteUser(user))
  256. }
  257. func TestCreateUser_Issue5882(t *testing.T) {
  258. // Init settings
  259. _ = setting.Admin
  260. passwd := ".//.;1;;//.,-=_"
  261. tt := []struct {
  262. user *User
  263. disableOrgCreation bool
  264. }{
  265. {&User{Name: "GiteaBot", Email: "GiteaBot@gitea.io", Passwd: passwd, MustChangePassword: false}, false},
  266. {&User{Name: "GiteaBot2", Email: "GiteaBot2@gitea.io", Passwd: passwd, MustChangePassword: false}, true},
  267. }
  268. setting.Service.DefaultAllowCreateOrganization = true
  269. for _, v := range tt {
  270. setting.Admin.DisableRegularOrgCreation = v.disableOrgCreation
  271. assert.NoError(t, CreateUser(v.user))
  272. u, err := GetUserByEmail(v.user.Email)
  273. assert.NoError(t, err)
  274. assert.Equal(t, !u.AllowCreateOrganization, v.disableOrgCreation)
  275. assert.NoError(t, DeleteUser(v.user))
  276. }
  277. }