user.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. package api
  2. import (
  3. "github.com/grafana/grafana/pkg/bus"
  4. "github.com/grafana/grafana/pkg/middleware"
  5. m "github.com/grafana/grafana/pkg/models"
  6. "github.com/grafana/grafana/pkg/setting"
  7. "github.com/grafana/grafana/pkg/util"
  8. )
  9. // GET /api/user (current authenticated user)
  10. func GetSignedInUser(c *middleware.Context) Response {
  11. return getUserUserProfile(c.UserId)
  12. }
  13. // GET /api/users/:id
  14. func GetUserById(c *middleware.Context) Response {
  15. return getUserUserProfile(c.ParamsInt64(":id"))
  16. }
  17. func getUserUserProfile(userId int64) Response {
  18. query := m.GetUserProfileQuery{UserId: userId}
  19. if err := bus.Dispatch(&query); err != nil {
  20. if err == m.ErrUserNotFound {
  21. return ApiError(404, m.ErrUserNotFound.Error(), nil)
  22. }
  23. return ApiError(500, "Failed to get user", err)
  24. }
  25. return Json(200, query.Result)
  26. }
  27. // GET /api/users/lookup
  28. func GetUserByLoginOrEmail(c *middleware.Context) Response {
  29. query := m.GetUserByLoginQuery{LoginOrEmail: c.Query("loginOrEmail")}
  30. if err := bus.Dispatch(&query); err != nil {
  31. if err == m.ErrUserNotFound {
  32. return ApiError(404, m.ErrUserNotFound.Error(), nil)
  33. }
  34. return ApiError(500, "Failed to get user", err)
  35. }
  36. user := query.Result
  37. result := m.UserProfileDTO{
  38. Name: user.Name,
  39. Email: user.Email,
  40. Login: user.Login,
  41. Theme: user.Theme,
  42. IsGrafanaAdmin: user.IsAdmin,
  43. OrgId: user.OrgId,
  44. }
  45. return Json(200, &result)
  46. }
  47. // POST /api/user
  48. func UpdateSignedInUser(c *middleware.Context, cmd m.UpdateUserCommand) Response {
  49. if setting.AuthProxyEnabled {
  50. if setting.AuthProxyHeaderProperty == "email" && cmd.Email != c.Email {
  51. return ApiError(400, "Not allowed to change email when auth proxy is using email property", nil)
  52. }
  53. if setting.AuthProxyHeaderProperty == "username" && cmd.Login != c.Login {
  54. return ApiError(400, "Not allowed to change username when auth proxy is using username property", nil)
  55. }
  56. }
  57. cmd.UserId = c.UserId
  58. return handleUpdateUser(cmd)
  59. }
  60. // POST /api/users/:id
  61. func UpdateUser(c *middleware.Context, cmd m.UpdateUserCommand) Response {
  62. cmd.UserId = c.ParamsInt64(":id")
  63. return handleUpdateUser(cmd)
  64. }
  65. //POST /api/users/:id/using/:orgId
  66. func UpdateUserActiveOrg(c *middleware.Context) Response {
  67. userId := c.ParamsInt64(":id")
  68. orgId := c.ParamsInt64(":orgId")
  69. if !validateUsingOrg(userId, orgId) {
  70. return ApiError(401, "Not a valid organization", nil)
  71. }
  72. cmd := m.SetUsingOrgCommand{UserId: userId, OrgId: orgId}
  73. if err := bus.Dispatch(&cmd); err != nil {
  74. return ApiError(500, "Failed to change active organization", err)
  75. }
  76. return ApiSuccess("Active organization changed")
  77. }
  78. func handleUpdateUser(cmd m.UpdateUserCommand) Response {
  79. if len(cmd.Login) == 0 {
  80. cmd.Login = cmd.Email
  81. if len(cmd.Login) == 0 {
  82. return ApiError(400, "Validation error, need to specify either username or email", nil)
  83. }
  84. }
  85. if err := bus.Dispatch(&cmd); err != nil {
  86. return ApiError(500, "Failed to update user", err)
  87. }
  88. return ApiSuccess("User updated")
  89. }
  90. // GET /api/user/orgs
  91. func GetSignedInUserOrgList(c *middleware.Context) Response {
  92. return getUserOrgList(c.UserId)
  93. }
  94. // GET /api/user/:id/orgs
  95. func GetUserOrgList(c *middleware.Context) Response {
  96. return getUserOrgList(c.ParamsInt64(":id"))
  97. }
  98. func getUserOrgList(userId int64) Response {
  99. query := m.GetUserOrgListQuery{UserId: userId}
  100. if err := bus.Dispatch(&query); err != nil {
  101. return ApiError(500, "Failed to get user organizations", err)
  102. }
  103. return Json(200, query.Result)
  104. }
  105. func validateUsingOrg(userId int64, orgId int64) bool {
  106. query := m.GetUserOrgListQuery{UserId: userId}
  107. if err := bus.Dispatch(&query); err != nil {
  108. return false
  109. }
  110. // validate that the org id in the list
  111. valid := false
  112. for _, other := range query.Result {
  113. if other.OrgId == orgId {
  114. valid = true
  115. }
  116. }
  117. return valid
  118. }
  119. // POST /api/user/using/:id
  120. func UserSetUsingOrg(c *middleware.Context) Response {
  121. orgId := c.ParamsInt64(":id")
  122. if !validateUsingOrg(c.UserId, orgId) {
  123. return ApiError(401, "Not a valid organization", nil)
  124. }
  125. cmd := m.SetUsingOrgCommand{UserId: c.UserId, OrgId: orgId}
  126. if err := bus.Dispatch(&cmd); err != nil {
  127. return ApiError(500, "Failed to change active organization", err)
  128. }
  129. return ApiSuccess("Active organization changed")
  130. }
  131. // GET /profile/switch-org/:id
  132. func ChangeActiveOrgAndRedirectToHome(c *middleware.Context) {
  133. orgId := c.ParamsInt64(":id")
  134. if !validateUsingOrg(c.UserId, orgId) {
  135. NotFoundHandler(c)
  136. }
  137. cmd := m.SetUsingOrgCommand{UserId: c.UserId, OrgId: orgId}
  138. if err := bus.Dispatch(&cmd); err != nil {
  139. NotFoundHandler(c)
  140. }
  141. c.Redirect(setting.AppSubUrl + "/")
  142. }
  143. func ChangeUserPassword(c *middleware.Context, cmd m.ChangeUserPasswordCommand) Response {
  144. if setting.LdapEnabled || setting.AuthProxyEnabled {
  145. return ApiError(400, "Not allowed to change password when LDAP or Auth Proxy is enabled", nil)
  146. }
  147. userQuery := m.GetUserByIdQuery{Id: c.UserId}
  148. if err := bus.Dispatch(&userQuery); err != nil {
  149. return ApiError(500, "Could not read user from database", err)
  150. }
  151. passwordHashed := util.EncodePassword(cmd.OldPassword, userQuery.Result.Salt)
  152. if passwordHashed != userQuery.Result.Password {
  153. return ApiError(401, "Invalid old password", nil)
  154. }
  155. password := m.Password(cmd.NewPassword)
  156. if password.IsWeak() {
  157. return ApiError(400, "New password is too short", nil)
  158. }
  159. cmd.UserId = c.UserId
  160. cmd.NewPassword = util.EncodePassword(cmd.NewPassword, userQuery.Result.Salt)
  161. if err := bus.Dispatch(&cmd); err != nil {
  162. return ApiError(500, "Failed to change user password", err)
  163. }
  164. return ApiSuccess("User password changed")
  165. }
  166. // GET /api/users
  167. func SearchUsers(c *middleware.Context) Response {
  168. query := m.SearchUsersQuery{Query: "", Page: 0, Limit: 1000}
  169. if err := bus.Dispatch(&query); err != nil {
  170. return ApiError(500, "Failed to fetch users", err)
  171. }
  172. return Json(200, query.Result)
  173. }
  174. func SetHelpFlag(c *middleware.Context) Response {
  175. flag := c.ParamsInt64(":id")
  176. bitmask := &c.HelpFlags1
  177. bitmask.AddFlag(m.HelpFlags1(flag))
  178. cmd := m.SetUserHelpFlagCommand{
  179. UserId: c.UserId,
  180. HelpFlags1: *bitmask,
  181. }
  182. if err := bus.Dispatch(&cmd); err != nil {
  183. return ApiError(500, "Failed to update help flag", err)
  184. }
  185. return Json(200, &util.DynMap{"message": "Help flag set", "helpFlags1": cmd.HelpFlags1})
  186. }
  187. func ClearHelpFlags(c *middleware.Context) Response {
  188. cmd := m.SetUserHelpFlagCommand{
  189. UserId: c.UserId,
  190. HelpFlags1: m.HelpFlags1(0),
  191. }
  192. if err := bus.Dispatch(&cmd); err != nil {
  193. return ApiError(500, "Failed to update help flag", err)
  194. }
  195. return Json(200, &util.DynMap{"message": "Help flag set", "helpFlags1": cmd.HelpFlags1})
  196. }