user_test.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499
  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("Creates 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. So(query.Result.IsDisabled, ShouldBeFalse)
  30. })
  31. })
  32. Convey("Creates disabled user", func() {
  33. cmd := &models.CreateUserCommand{
  34. Email: "usertest@test.com",
  35. Name: "user name",
  36. Login: "user_test_login",
  37. IsDisabled: true,
  38. }
  39. err := CreateUser(context.Background(), cmd)
  40. So(err, ShouldBeNil)
  41. Convey("Loading a user", func() {
  42. query := models.GetUserByIdQuery{Id: cmd.Result.Id}
  43. err := GetUserById(&query)
  44. So(err, ShouldBeNil)
  45. So(query.Result.Email, ShouldEqual, "usertest@test.com")
  46. So(query.Result.Password, ShouldEqual, "")
  47. So(query.Result.Rands, ShouldHaveLength, 10)
  48. So(query.Result.Salt, ShouldHaveLength, 10)
  49. So(query.Result.IsDisabled, ShouldBeTrue)
  50. })
  51. })
  52. Convey("Given 5 users", func() {
  53. users := createFiveTestUsers(func(i int) *models.CreateUserCommand {
  54. return &models.CreateUserCommand{
  55. Email: fmt.Sprint("user", i, "@test.com"),
  56. Name: fmt.Sprint("user", i),
  57. Login: fmt.Sprint("loginuser", i),
  58. IsDisabled: false,
  59. }
  60. })
  61. Convey("Can return the first page of users and a total count", func() {
  62. query := models.SearchUsersQuery{Query: "", Page: 1, Limit: 3}
  63. err := SearchUsers(&query)
  64. So(err, ShouldBeNil)
  65. So(len(query.Result.Users), ShouldEqual, 3)
  66. So(query.Result.TotalCount, ShouldEqual, 5)
  67. })
  68. Convey("Can return the second page of users and a total count", func() {
  69. query := models.SearchUsersQuery{Query: "", Page: 2, Limit: 3}
  70. err := SearchUsers(&query)
  71. So(err, ShouldBeNil)
  72. So(len(query.Result.Users), ShouldEqual, 2)
  73. So(query.Result.TotalCount, ShouldEqual, 5)
  74. })
  75. Convey("Can return list of users matching query on user name", func() {
  76. query := models.SearchUsersQuery{Query: "use", Page: 1, Limit: 3}
  77. err := SearchUsers(&query)
  78. So(err, ShouldBeNil)
  79. So(len(query.Result.Users), ShouldEqual, 3)
  80. So(query.Result.TotalCount, ShouldEqual, 5)
  81. query = models.SearchUsersQuery{Query: "ser1", Page: 1, Limit: 3}
  82. err = SearchUsers(&query)
  83. So(err, ShouldBeNil)
  84. So(len(query.Result.Users), ShouldEqual, 1)
  85. So(query.Result.TotalCount, ShouldEqual, 1)
  86. query = models.SearchUsersQuery{Query: "USER1", Page: 1, Limit: 3}
  87. err = SearchUsers(&query)
  88. So(err, ShouldBeNil)
  89. So(len(query.Result.Users), ShouldEqual, 1)
  90. So(query.Result.TotalCount, ShouldEqual, 1)
  91. query = models.SearchUsersQuery{Query: "idontexist", Page: 1, Limit: 3}
  92. err = SearchUsers(&query)
  93. So(err, ShouldBeNil)
  94. So(len(query.Result.Users), ShouldEqual, 0)
  95. So(query.Result.TotalCount, ShouldEqual, 0)
  96. })
  97. Convey("Can return list of users matching query on email", func() {
  98. query := models.SearchUsersQuery{Query: "ser1@test.com", Page: 1, Limit: 3}
  99. err := SearchUsers(&query)
  100. So(err, ShouldBeNil)
  101. So(len(query.Result.Users), ShouldEqual, 1)
  102. So(query.Result.TotalCount, ShouldEqual, 1)
  103. })
  104. Convey("Can return list of users matching query on login name", func() {
  105. query := models.SearchUsersQuery{Query: "loginuser1", Page: 1, Limit: 3}
  106. err := SearchUsers(&query)
  107. So(err, ShouldBeNil)
  108. So(len(query.Result.Users), ShouldEqual, 1)
  109. So(query.Result.TotalCount, ShouldEqual, 1)
  110. })
  111. Convey("Can return list users based on their auth type", func() {
  112. // add users to auth table
  113. for index, user := range users {
  114. authModule := "killa"
  115. // define every second user as ldap
  116. if index%2 == 0 {
  117. authModule = "ldap"
  118. }
  119. cmd2 := &models.SetAuthInfoCommand{
  120. UserId: user.Id,
  121. AuthModule: authModule,
  122. AuthId: "gorilla",
  123. }
  124. err := SetAuthInfo(cmd2)
  125. So(err, ShouldBeNil)
  126. }
  127. query := models.SearchUsersQuery{AuthModule: "ldap"}
  128. err := SearchUsers(&query)
  129. So(err, ShouldBeNil)
  130. So(query.Result.Users, ShouldHaveLength, 3)
  131. zero, second, fourth := false, false, false
  132. for _, user := range query.Result.Users {
  133. if user.Name == "user0" {
  134. zero = true
  135. }
  136. if user.Name == "user2" {
  137. second = true
  138. }
  139. if user.Name == "user4" {
  140. fourth = true
  141. }
  142. }
  143. So(zero, ShouldBeTrue)
  144. So(second, ShouldBeTrue)
  145. So(fourth, ShouldBeTrue)
  146. })
  147. Convey("Can return list users based on their is_disabled flag", func() {
  148. ss = InitTestDB(t)
  149. createFiveTestUsers(func(i int) *models.CreateUserCommand {
  150. return &models.CreateUserCommand{
  151. Email: fmt.Sprint("user", i, "@test.com"),
  152. Name: fmt.Sprint("user", i),
  153. Login: fmt.Sprint("loginuser", i),
  154. IsDisabled: i%2 == 0,
  155. }
  156. })
  157. isDisabled := false
  158. query := models.SearchUsersQuery{IsDisabled: &isDisabled}
  159. err := SearchUsers(&query)
  160. So(err, ShouldBeNil)
  161. So(query.Result.Users, ShouldHaveLength, 2)
  162. first, third := false, false
  163. for _, user := range query.Result.Users {
  164. if user.Name == "user1" {
  165. first = true
  166. }
  167. if user.Name == "user3" {
  168. third = true
  169. }
  170. }
  171. So(first, ShouldBeTrue)
  172. So(third, ShouldBeTrue)
  173. ss = InitTestDB(t)
  174. users = createFiveTestUsers(func(i int) *models.CreateUserCommand {
  175. return &models.CreateUserCommand{
  176. Email: fmt.Sprint("user", i, "@test.com"),
  177. Name: fmt.Sprint("user", i),
  178. Login: fmt.Sprint("loginuser", i),
  179. IsDisabled: false,
  180. }
  181. })
  182. })
  183. Convey("when a user is an org member and has been assigned permissions", func() {
  184. err := AddOrgUser(&models.AddOrgUserCommand{LoginOrEmail: users[1].Login, Role: models.ROLE_VIEWER, OrgId: users[0].OrgId, UserId: users[1].Id})
  185. So(err, ShouldBeNil)
  186. testHelperUpdateDashboardAcl(1, models.DashboardAcl{DashboardId: 1, OrgId: users[0].OrgId, UserId: users[1].Id, Permission: models.PERMISSION_EDIT})
  187. So(err, ShouldBeNil)
  188. err = SavePreferences(&models.SavePreferencesCommand{UserId: users[1].Id, OrgId: users[0].OrgId, HomeDashboardId: 1, Theme: "dark"})
  189. So(err, ShouldBeNil)
  190. Convey("when the user is deleted", func() {
  191. err = DeleteUser(&models.DeleteUserCommand{UserId: users[1].Id})
  192. So(err, ShouldBeNil)
  193. Convey("Should delete connected org users and permissions", func() {
  194. query := &models.GetOrgUsersQuery{OrgId: users[0].OrgId}
  195. err = GetOrgUsersForTest(query)
  196. So(err, ShouldBeNil)
  197. So(len(query.Result), ShouldEqual, 1)
  198. permQuery := &models.GetDashboardAclInfoListQuery{DashboardId: 1, OrgId: users[0].OrgId}
  199. err = GetDashboardAclInfoList(permQuery)
  200. So(err, ShouldBeNil)
  201. So(len(permQuery.Result), ShouldEqual, 0)
  202. prefsQuery := &models.GetPreferencesQuery{OrgId: users[0].OrgId, UserId: users[1].Id}
  203. err = GetPreferences(prefsQuery)
  204. So(err, ShouldBeNil)
  205. So(prefsQuery.Result.OrgId, ShouldEqual, 0)
  206. So(prefsQuery.Result.UserId, ShouldEqual, 0)
  207. })
  208. })
  209. Convey("when retreiving signed in user for orgId=0 result should return active org id", func() {
  210. ss.CacheService.Flush()
  211. query := &models.GetSignedInUserQuery{OrgId: users[1].OrgId, UserId: users[1].Id}
  212. err := ss.GetSignedInUserWithCache(query)
  213. So(err, ShouldBeNil)
  214. So(query.Result, ShouldNotBeNil)
  215. So(query.OrgId, ShouldEqual, users[1].OrgId)
  216. err = SetUsingOrg(&models.SetUsingOrgCommand{UserId: users[1].Id, OrgId: users[0].OrgId})
  217. So(err, ShouldBeNil)
  218. query = &models.GetSignedInUserQuery{OrgId: 0, UserId: users[1].Id}
  219. err = ss.GetSignedInUserWithCache(query)
  220. So(err, ShouldBeNil)
  221. So(query.Result, ShouldNotBeNil)
  222. So(query.Result.OrgId, ShouldEqual, users[0].OrgId)
  223. cacheKey := newSignedInUserCacheKey(query.Result.OrgId, query.UserId)
  224. _, found := ss.CacheService.Get(cacheKey)
  225. So(found, ShouldBeTrue)
  226. })
  227. })
  228. Convey("When batch disabling users", func() {
  229. Convey("Should disable all users", func() {
  230. disableCmd := models.BatchDisableUsersCommand{
  231. UserIds: []int64{1, 2, 3, 4, 5},
  232. IsDisabled: true,
  233. }
  234. err := BatchDisableUsers(&disableCmd)
  235. So(err, ShouldBeNil)
  236. isDisabled := true
  237. query := &models.SearchUsersQuery{IsDisabled: &isDisabled}
  238. err = SearchUsers(query)
  239. So(err, ShouldBeNil)
  240. So(query.Result.TotalCount, ShouldEqual, 5)
  241. })
  242. Convey("Should enable all users", func() {
  243. ss = InitTestDB(t)
  244. createFiveTestUsers(func(i int) *models.CreateUserCommand {
  245. return &models.CreateUserCommand{
  246. Email: fmt.Sprint("user", i, "@test.com"),
  247. Name: fmt.Sprint("user", i),
  248. Login: fmt.Sprint("loginuser", i),
  249. IsDisabled: true,
  250. }
  251. })
  252. disableCmd := models.BatchDisableUsersCommand{
  253. UserIds: []int64{1, 2, 3, 4, 5},
  254. IsDisabled: false,
  255. }
  256. err := BatchDisableUsers(&disableCmd)
  257. So(err, ShouldBeNil)
  258. isDisabled := false
  259. query := &models.SearchUsersQuery{IsDisabled: &isDisabled}
  260. err = SearchUsers(query)
  261. So(err, ShouldBeNil)
  262. So(query.Result.TotalCount, ShouldEqual, 5)
  263. })
  264. Convey("Should disable only specific users", func() {
  265. ss = InitTestDB(t)
  266. users = createFiveTestUsers(func(i int) *models.CreateUserCommand {
  267. return &models.CreateUserCommand{
  268. Email: fmt.Sprint("user", i, "@test.com"),
  269. Name: fmt.Sprint("user", i),
  270. Login: fmt.Sprint("loginuser", i),
  271. IsDisabled: false,
  272. }
  273. })
  274. userIdsToDisable := []int64{}
  275. for i := 0; i < 3; i++ {
  276. userIdsToDisable = append(userIdsToDisable, users[i].Id)
  277. }
  278. disableCmd := models.BatchDisableUsersCommand{
  279. UserIds: userIdsToDisable,
  280. IsDisabled: true,
  281. }
  282. err := BatchDisableUsers(&disableCmd)
  283. So(err, ShouldBeNil)
  284. query := models.SearchUsersQuery{}
  285. err = SearchUsers(&query)
  286. So(err, ShouldBeNil)
  287. So(query.Result.TotalCount, ShouldEqual, 5)
  288. for _, user := range query.Result.Users {
  289. shouldBeDisabled := false
  290. // Check if user id is in the userIdsToDisable list
  291. for _, disabledUserId := range userIdsToDisable {
  292. if user.Id == disabledUserId {
  293. So(user.IsDisabled, ShouldBeTrue)
  294. shouldBeDisabled = true
  295. }
  296. }
  297. // Otherwise user shouldn't be disabled
  298. if !shouldBeDisabled {
  299. So(user.IsDisabled, ShouldBeFalse)
  300. }
  301. }
  302. })
  303. // Since previous tests were destructive
  304. ss = InitTestDB(t)
  305. users = createFiveTestUsers(func(i int) *models.CreateUserCommand {
  306. return &models.CreateUserCommand{
  307. Email: fmt.Sprint("user", i, "@test.com"),
  308. Name: fmt.Sprint("user", i),
  309. Login: fmt.Sprint("loginuser", i),
  310. IsDisabled: false,
  311. }
  312. })
  313. })
  314. Convey("When searching users", func() {
  315. // Find a user to set tokens on
  316. login := "loginuser0"
  317. // Calling GetUserByAuthInfoQuery on an existing user will populate an entry in the user_auth table
  318. // Make the first log-in during the past
  319. getTime = func() time.Time { return time.Now().AddDate(0, 0, -2) }
  320. query := &models.GetUserByAuthInfoQuery{Login: login, AuthModule: "test1", AuthId: "test1"}
  321. err := GetUserByAuthInfo(query)
  322. getTime = time.Now
  323. So(err, ShouldBeNil)
  324. So(query.Result.Login, ShouldEqual, login)
  325. // Add a second auth module for this user
  326. // Have this module's last log-in be more recent
  327. getTime = func() time.Time { return time.Now().AddDate(0, 0, -1) }
  328. query = &models.GetUserByAuthInfoQuery{Login: login, AuthModule: "test2", AuthId: "test2"}
  329. err = GetUserByAuthInfo(query)
  330. getTime = time.Now
  331. So(err, ShouldBeNil)
  332. So(query.Result.Login, ShouldEqual, login)
  333. Convey("Should return the only most recently used auth_module", func() {
  334. searchUserQuery := &models.SearchUsersQuery{}
  335. err = SearchUsers(searchUserQuery)
  336. So(err, ShouldBeNil)
  337. So(searchUserQuery.Result.Users, ShouldHaveLength, 5)
  338. for _, user := range searchUserQuery.Result.Users {
  339. if user.Login == login {
  340. So(user.AuthModule, ShouldHaveLength, 1)
  341. So(user.AuthModule[0], ShouldEqual, "test2")
  342. }
  343. }
  344. // "log in" again with the first auth module
  345. updateAuthCmd := &models.UpdateAuthInfoCommand{UserId: query.Result.Id, AuthModule: "test1", AuthId: "test1"}
  346. err = UpdateAuthInfo(updateAuthCmd)
  347. So(err, ShouldBeNil)
  348. searchUserQuery = &models.SearchUsersQuery{}
  349. err = SearchUsers(searchUserQuery)
  350. So(err, ShouldBeNil)
  351. for _, user := range searchUserQuery.Result.Users {
  352. if user.Login == login {
  353. So(user.AuthModule, ShouldHaveLength, 1)
  354. So(user.AuthModule[0], ShouldEqual, "test1")
  355. }
  356. }
  357. })
  358. })
  359. })
  360. Convey("Given one grafana admin user", func() {
  361. var err error
  362. createUserCmd := &models.CreateUserCommand{
  363. Email: fmt.Sprint("admin", "@test.com"),
  364. Name: fmt.Sprint("admin"),
  365. Login: fmt.Sprint("admin"),
  366. IsAdmin: true,
  367. }
  368. err = CreateUser(context.Background(), createUserCmd)
  369. So(err, ShouldBeNil)
  370. Convey("Cannot make themselves a non-admin", func() {
  371. updateUserPermsCmd := models.UpdateUserPermissionsCommand{IsGrafanaAdmin: false, UserId: 1}
  372. updatePermsError := UpdateUserPermissions(&updateUserPermsCmd)
  373. So(updatePermsError, ShouldEqual, models.ErrLastGrafanaAdmin)
  374. query := models.GetUserByIdQuery{Id: createUserCmd.Result.Id}
  375. getUserError := GetUserById(&query)
  376. So(getUserError, ShouldBeNil)
  377. So(query.Result.IsAdmin, ShouldEqual, true)
  378. })
  379. })
  380. })
  381. }
  382. func GetOrgUsersForTest(query *models.GetOrgUsersQuery) error {
  383. query.Result = make([]*models.OrgUserDTO, 0)
  384. sess := x.Table("org_user")
  385. sess.Join("LEFT ", x.Dialect().Quote("user"), fmt.Sprintf("org_user.user_id=%s.id", x.Dialect().Quote("user")))
  386. sess.Where("org_user.org_id=?", query.OrgId)
  387. sess.Cols("org_user.org_id", "org_user.user_id", "user.email", "user.login", "org_user.role")
  388. err := sess.Find(&query.Result)
  389. return err
  390. }
  391. func createFiveTestUsers(fn func(i int) *models.CreateUserCommand) []models.User {
  392. var err error
  393. var cmd *models.CreateUserCommand
  394. users := []models.User{}
  395. for i := 0; i < 5; i++ {
  396. cmd = fn(i)
  397. err = CreateUser(context.Background(), cmd)
  398. users = append(users, cmd.Result)
  399. So(err, ShouldBeNil)
  400. }
  401. return users
  402. }