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 7.8KB

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