user_test.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. package sqlstore
  2. import (
  3. "context"
  4. "fmt"
  5. "testing"
  6. "time"
  7. . "github.com/smartystreets/goconvey/convey"
  8. "github.com/grafana/grafana/pkg/models"
  9. )
  10. func TestUserDataAccess(t *testing.T) {
  11. Convey("Testing DB", t, func() {
  12. ss := InitTestDB(t)
  13. Convey("Creating a user", func() {
  14. cmd := &models.CreateUserCommand{
  15. Email: "usertest@test.com",
  16. Name: "user name",
  17. Login: "user_test_login",
  18. }
  19. err := CreateUser(context.Background(), cmd)
  20. So(err, ShouldBeNil)
  21. Convey("Loading a user", func() {
  22. query := models.GetUserByIdQuery{Id: cmd.Result.Id}
  23. err := GetUserById(&query)
  24. So(err, ShouldBeNil)
  25. So(query.Result.Email, ShouldEqual, "usertest@test.com")
  26. So(query.Result.Password, ShouldEqual, "")
  27. So(query.Result.Rands, ShouldHaveLength, 10)
  28. So(query.Result.Salt, ShouldHaveLength, 10)
  29. })
  30. })
  31. Convey("Given 5 users", func() {
  32. var err error
  33. var cmd *models.CreateUserCommand
  34. users := []models.User{}
  35. for i := 0; i < 5; i++ {
  36. cmd = &models.CreateUserCommand{
  37. Email: fmt.Sprint("user", i, "@test.com"),
  38. Name: fmt.Sprint("user", i),
  39. Login: fmt.Sprint("loginuser", i),
  40. }
  41. err = CreateUser(context.Background(), cmd)
  42. So(err, ShouldBeNil)
  43. users = append(users, cmd.Result)
  44. }
  45. Convey("Can return the first page of users and a total count", func() {
  46. query := models.SearchUsersQuery{Query: "", Page: 1, Limit: 3}
  47. err = SearchUsers(&query)
  48. So(err, ShouldBeNil)
  49. So(len(query.Result.Users), ShouldEqual, 3)
  50. So(query.Result.TotalCount, ShouldEqual, 5)
  51. })
  52. Convey("Can return the second page of users and a total count", func() {
  53. query := models.SearchUsersQuery{Query: "", Page: 2, Limit: 3}
  54. err = SearchUsers(&query)
  55. So(err, ShouldBeNil)
  56. So(len(query.Result.Users), ShouldEqual, 2)
  57. So(query.Result.TotalCount, ShouldEqual, 5)
  58. })
  59. Convey("Can return list of users matching query on user name", func() {
  60. query := models.SearchUsersQuery{Query: "use", Page: 1, Limit: 3}
  61. err = SearchUsers(&query)
  62. So(err, ShouldBeNil)
  63. So(len(query.Result.Users), ShouldEqual, 3)
  64. So(query.Result.TotalCount, ShouldEqual, 5)
  65. query = models.SearchUsersQuery{Query: "ser1", Page: 1, Limit: 3}
  66. err = SearchUsers(&query)
  67. So(err, ShouldBeNil)
  68. So(len(query.Result.Users), ShouldEqual, 1)
  69. So(query.Result.TotalCount, ShouldEqual, 1)
  70. query = models.SearchUsersQuery{Query: "USER1", Page: 1, Limit: 3}
  71. err = SearchUsers(&query)
  72. So(err, ShouldBeNil)
  73. So(len(query.Result.Users), ShouldEqual, 1)
  74. So(query.Result.TotalCount, ShouldEqual, 1)
  75. query = models.SearchUsersQuery{Query: "idontexist", Page: 1, Limit: 3}
  76. err = SearchUsers(&query)
  77. So(err, ShouldBeNil)
  78. So(len(query.Result.Users), ShouldEqual, 0)
  79. So(query.Result.TotalCount, ShouldEqual, 0)
  80. })
  81. Convey("Can return list of users matching query on email", func() {
  82. query := models.SearchUsersQuery{Query: "ser1@test.com", Page: 1, Limit: 3}
  83. err = SearchUsers(&query)
  84. So(err, ShouldBeNil)
  85. So(len(query.Result.Users), ShouldEqual, 1)
  86. So(query.Result.TotalCount, ShouldEqual, 1)
  87. })
  88. Convey("Can return list of users matching query on login name", func() {
  89. query := models.SearchUsersQuery{Query: "loginuser1", Page: 1, Limit: 3}
  90. err = SearchUsers(&query)
  91. So(err, ShouldBeNil)
  92. So(len(query.Result.Users), ShouldEqual, 1)
  93. So(query.Result.TotalCount, ShouldEqual, 1)
  94. })
  95. Convey("can return list users based on their auth type", func() {
  96. // add users to auth table
  97. for index, user := range users {
  98. authModule := "killa"
  99. // define every second user as ldap
  100. if index%2 == 0 {
  101. authModule = "ldap"
  102. }
  103. cmd2 := &models.SetAuthInfoCommand{
  104. UserId: user.Id,
  105. AuthModule: authModule,
  106. AuthId: "gorilla",
  107. }
  108. err = SetAuthInfo(cmd2)
  109. So(err, ShouldBeNil)
  110. }
  111. query := models.SearchUsersQuery{AuthModule: "ldap"}
  112. err = SearchUsers(&query)
  113. So(err, ShouldBeNil)
  114. So(query.Result.Users, ShouldHaveLength, 3)
  115. zero, second, fourth := false, false, false
  116. for _, user := range query.Result.Users {
  117. if user.Name == "user0" {
  118. zero = true
  119. }
  120. if user.Name == "user2" {
  121. second = true
  122. }
  123. if user.Name == "user4" {
  124. fourth = true
  125. }
  126. }
  127. So(zero, ShouldBeTrue)
  128. So(second, ShouldBeTrue)
  129. So(fourth, ShouldBeTrue)
  130. })
  131. Convey("when a user is an org member and has been assigned permissions", func() {
  132. err = AddOrgUser(&models.AddOrgUserCommand{LoginOrEmail: users[1].Login, Role: models.ROLE_VIEWER, OrgId: users[0].OrgId, UserId: users[1].Id})
  133. So(err, ShouldBeNil)
  134. testHelperUpdateDashboardAcl(1, models.DashboardAcl{DashboardId: 1, OrgId: users[0].OrgId, UserId: users[1].Id, Permission: models.PERMISSION_EDIT})
  135. So(err, ShouldBeNil)
  136. err = SavePreferences(&models.SavePreferencesCommand{UserId: users[1].Id, OrgId: users[0].OrgId, HomeDashboardId: 1, Theme: "dark"})
  137. So(err, ShouldBeNil)
  138. Convey("when the user is deleted", func() {
  139. err = DeleteUser(&models.DeleteUserCommand{UserId: users[1].Id})
  140. So(err, ShouldBeNil)
  141. Convey("Should delete connected org users and permissions", func() {
  142. query := &models.GetOrgUsersQuery{OrgId: users[0].OrgId}
  143. err = GetOrgUsersForTest(query)
  144. So(err, ShouldBeNil)
  145. So(len(query.Result), ShouldEqual, 1)
  146. permQuery := &models.GetDashboardAclInfoListQuery{DashboardId: 1, OrgId: users[0].OrgId}
  147. err = GetDashboardAclInfoList(permQuery)
  148. So(err, ShouldBeNil)
  149. So(len(permQuery.Result), ShouldEqual, 0)
  150. prefsQuery := &models.GetPreferencesQuery{OrgId: users[0].OrgId, UserId: users[1].Id}
  151. err = GetPreferences(prefsQuery)
  152. So(err, ShouldBeNil)
  153. So(prefsQuery.Result.OrgId, ShouldEqual, 0)
  154. So(prefsQuery.Result.UserId, ShouldEqual, 0)
  155. })
  156. })
  157. Convey("when retreiving signed in user for orgId=0 result should return active org id", func() {
  158. ss.CacheService.Flush()
  159. query := &models.GetSignedInUserQuery{OrgId: users[1].OrgId, UserId: users[1].Id}
  160. err := ss.GetSignedInUserWithCache(query)
  161. So(err, ShouldBeNil)
  162. So(query.Result, ShouldNotBeNil)
  163. So(query.OrgId, ShouldEqual, users[1].OrgId)
  164. err = SetUsingOrg(&models.SetUsingOrgCommand{UserId: users[1].Id, OrgId: users[0].OrgId})
  165. So(err, ShouldBeNil)
  166. query = &models.GetSignedInUserQuery{OrgId: 0, UserId: users[1].Id}
  167. err = ss.GetSignedInUserWithCache(query)
  168. So(err, ShouldBeNil)
  169. So(query.Result, ShouldNotBeNil)
  170. So(query.Result.OrgId, ShouldEqual, users[0].OrgId)
  171. cacheKey := newSignedInUserCacheKey(query.Result.OrgId, query.UserId)
  172. _, found := ss.CacheService.Get(cacheKey)
  173. So(found, ShouldBeTrue)
  174. })
  175. })
  176. Convey("When batch disabling users", func() {
  177. userIdsToDisable := []int64{}
  178. for i := 0; i < 3; i++ {
  179. userIdsToDisable = append(userIdsToDisable, users[i].Id)
  180. }
  181. disableCmd := models.BatchDisableUsersCommand{UserIds: userIdsToDisable, IsDisabled: true}
  182. err = BatchDisableUsers(&disableCmd)
  183. So(err, ShouldBeNil)
  184. Convey("Should disable all provided users", func() {
  185. query := models.SearchUsersQuery{}
  186. err = SearchUsers(&query)
  187. So(query.Result.TotalCount, ShouldEqual, 5)
  188. for _, user := range query.Result.Users {
  189. shouldBeDisabled := false
  190. // Check if user id is in the userIdsToDisable list
  191. for _, disabledUserId := range userIdsToDisable {
  192. if user.Id == disabledUserId {
  193. So(user.IsDisabled, ShouldBeTrue)
  194. shouldBeDisabled = true
  195. }
  196. }
  197. // Otherwise user shouldn't be disabled
  198. if !shouldBeDisabled {
  199. So(user.IsDisabled, ShouldBeFalse)
  200. }
  201. }
  202. })
  203. })
  204. Convey("When searching users", func() {
  205. // Find a user to set tokens on
  206. login := "loginuser0"
  207. // Calling GetUserByAuthInfoQuery on an existing user will populate an entry in the user_auth table
  208. // Make the first log-in during the past
  209. getTime = func() time.Time { return time.Now().AddDate(0, 0, -2) }
  210. query := &models.GetUserByAuthInfoQuery{Login: login, AuthModule: "test1", AuthId: "test1"}
  211. err = GetUserByAuthInfo(query)
  212. getTime = time.Now
  213. So(err, ShouldBeNil)
  214. So(query.Result.Login, ShouldEqual, login)
  215. // Add a second auth module for this user
  216. // Have this module's last log-in be more recent
  217. getTime = func() time.Time { return time.Now().AddDate(0, 0, -1) }
  218. query = &models.GetUserByAuthInfoQuery{Login: login, AuthModule: "test2", AuthId: "test2"}
  219. err = GetUserByAuthInfo(query)
  220. getTime = time.Now
  221. So(err, ShouldBeNil)
  222. So(query.Result.Login, ShouldEqual, login)
  223. Convey("Should return the only most recently used auth_module", func() {
  224. searchUserQuery := &models.SearchUsersQuery{}
  225. err = SearchUsers(searchUserQuery)
  226. So(err, ShouldBeNil)
  227. So(searchUserQuery.Result.Users, ShouldHaveLength, 5)
  228. for _, user := range searchUserQuery.Result.Users {
  229. if user.Login == login {
  230. So(user.AuthModule, ShouldHaveLength, 1)
  231. So(user.AuthModule[0], ShouldEqual, "test2")
  232. }
  233. }
  234. // "log in" again with the first auth module
  235. updateAuthCmd := &models.UpdateAuthInfoCommand{UserId: query.Result.Id, AuthModule: "test1", AuthId: "test1"}
  236. err = UpdateAuthInfo(updateAuthCmd)
  237. So(err, ShouldBeNil)
  238. searchUserQuery = &models.SearchUsersQuery{}
  239. err = SearchUsers(searchUserQuery)
  240. So(err, ShouldBeNil)
  241. for _, user := range searchUserQuery.Result.Users {
  242. if user.Login == login {
  243. So(user.AuthModule, ShouldHaveLength, 1)
  244. So(user.AuthModule[0], ShouldEqual, "test1")
  245. }
  246. }
  247. })
  248. })
  249. })
  250. Convey("Given one grafana admin user", func() {
  251. var err error
  252. createUserCmd := &models.CreateUserCommand{
  253. Email: fmt.Sprint("admin", "@test.com"),
  254. Name: fmt.Sprint("admin"),
  255. Login: fmt.Sprint("admin"),
  256. IsAdmin: true,
  257. }
  258. err = CreateUser(context.Background(), createUserCmd)
  259. So(err, ShouldBeNil)
  260. Convey("Cannot make themselves a non-admin", func() {
  261. updateUserPermsCmd := models.UpdateUserPermissionsCommand{IsGrafanaAdmin: false, UserId: 1}
  262. updatePermsError := UpdateUserPermissions(&updateUserPermsCmd)
  263. So(updatePermsError, ShouldEqual, models.ErrLastGrafanaAdmin)
  264. query := models.GetUserByIdQuery{Id: createUserCmd.Result.Id}
  265. getUserError := GetUserById(&query)
  266. So(getUserError, ShouldBeNil)
  267. So(query.Result.IsAdmin, ShouldEqual, true)
  268. })
  269. })
  270. })
  271. }
  272. func GetOrgUsersForTest(query *models.GetOrgUsersQuery) error {
  273. query.Result = make([]*models.OrgUserDTO, 0)
  274. sess := x.Table("org_user")
  275. sess.Join("LEFT ", x.Dialect().Quote("user"), fmt.Sprintf("org_user.user_id=%s.id", x.Dialect().Quote("user")))
  276. sess.Where("org_user.org_id=?", query.OrgId)
  277. sess.Cols("org_user.org_id", "org_user.user_id", "user.email", "user.login", "org_user.role")
  278. err := sess.Find(&query.Result)
  279. return err
  280. }