| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | package api | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2021-11-04 18:17:07 +08:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2020-08-13 20:38:54 +08:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2021-11-29 17:18:01 +08:00
										 |  |  | 	"net/http" | 
					
						
							| 
									
										
										
										
											2022-11-14 20:11:26 +08:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2020-08-13 20:38:54 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/api/dtos" | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/api/response" | 
					
						
							| 
									
										
										
										
											2015-06-08 23:56:56 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/events" | 
					
						
							| 
									
										
										
										
											2019-02-24 06:35:26 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/infra/metrics" | 
					
						
							| 
									
										
										
										
											2023-01-27 15:50:36 +08:00
										 |  |  | 	contextmodel "github.com/grafana/grafana/pkg/services/contexthandler/model" | 
					
						
							| 
									
										
										
										
											2023-01-06 16:02:05 +08:00
										 |  |  | 	tempuser "github.com/grafana/grafana/pkg/services/temp_user" | 
					
						
							| 
									
										
										
										
											2022-06-28 20:32:25 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/services/user" | 
					
						
							| 
									
										
										
										
											2015-02-05 17:37:13 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/setting" | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/util" | 
					
						
							| 
									
										
										
										
											2021-11-29 17:18:01 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/web" | 
					
						
							| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | // GET /api/user/signup/options
 | 
					
						
							| 
									
										
										
										
											2023-02-27 17:23:38 +08:00
										 |  |  | func (hs *HTTPServer) GetSignUpOptions(c *contextmodel.ReqContext) response.Response { | 
					
						
							| 
									
										
										
										
											2022-04-15 20:01:58 +08:00
										 |  |  | 	return response.JSON(http.StatusOK, util.DynMap{ | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 		"verifyEmailEnabled": setting.VerifyEmailEnabled, | 
					
						
							| 
									
										
										
										
											2023-02-27 17:23:38 +08:00
										 |  |  | 		"autoAssignOrg":      hs.Cfg.AutoAssignOrg, | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 22:14:53 +08:00
										 |  |  | // POST /api/user/signup
 | 
					
						
							| 
									
										
										
										
											2023-01-27 15:50:36 +08:00
										 |  |  | func (hs *HTTPServer) SignUp(c *contextmodel.ReqContext) response.Response { | 
					
						
							| 
									
										
										
										
											2021-11-29 17:18:01 +08:00
										 |  |  | 	form := dtos.SignUpForm{} | 
					
						
							| 
									
										
										
										
											2022-11-14 20:11:26 +08:00
										 |  |  | 	var err error | 
					
						
							|  |  |  | 	if err = web.Bind(c.Req, &form); err != nil { | 
					
						
							| 
									
										
										
										
											2021-11-29 17:18:01 +08:00
										 |  |  | 		return response.Error(http.StatusBadRequest, "bad request data", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-11 23:19:29 +08:00
										 |  |  | 	if !setting.AllowUserSignUp { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return response.Error(401, "User signup is disabled", nil) | 
					
						
							| 
									
										
										
										
											2015-01-29 22:46:54 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-14 20:11:26 +08:00
										 |  |  | 	form.Email, err = ValidateAndNormalizeEmail(form.Email) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return response.Error(http.StatusBadRequest, "Invalid email address", nil) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 19:22:43 +08:00
										 |  |  | 	existing := user.GetUserByLoginQuery{LoginOrEmail: form.Email} | 
					
						
							| 
									
										
										
										
											2022-11-14 20:11:26 +08:00
										 |  |  | 	_, err = hs.userService.GetByLogin(c.Req.Context(), &existing) | 
					
						
							| 
									
										
										
										
											2022-08-04 19:22:43 +08:00
										 |  |  | 	if err == nil { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return response.Error(422, "User with same email address already exists", nil) | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 16:02:05 +08:00
										 |  |  | 	cmd := tempuser.CreateTempUserCommand{} | 
					
						
							|  |  |  | 	cmd.OrgID = -1 | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 	cmd.Email = form.Email | 
					
						
							| 
									
										
										
										
											2023-01-06 16:02:05 +08:00
										 |  |  | 	cmd.Status = tempuser.TmpUserSignUpStarted | 
					
						
							|  |  |  | 	cmd.InvitedByUserID = c.UserID | 
					
						
							| 
									
										
										
										
											2019-10-23 16:40:12 +08:00
										 |  |  | 	cmd.Code, err = util.GetRandomString(20) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return response.Error(500, "Failed to generate random string", err) | 
					
						
							| 
									
										
										
										
											2019-10-23 16:40:12 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-12-13 22:33:05 +08:00
										 |  |  | 	cmd.RemoteAddr = c.RemoteAddr() | 
					
						
							| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 16:02:05 +08:00
										 |  |  | 	if _, err := hs.tempUserService.CreateTempUser(c.Req.Context(), &cmd); err != nil { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return response.Error(500, "Failed to create signup", err) | 
					
						
							| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-14 22:07:41 +08:00
										 |  |  | 	if err := hs.bus.Publish(c.Req.Context(), &events.SignUpStarted{ | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 		Email: form.Email, | 
					
						
							|  |  |  | 		Code:  cmd.Code, | 
					
						
							| 
									
										
										
										
											2019-10-09 00:57:53 +08:00
										 |  |  | 	}); err != nil { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return response.Error(500, "Failed to publish event", err) | 
					
						
							| 
									
										
										
										
											2019-10-09 00:57:53 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-01-21 16:52:40 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-16 22:58:46 +08:00
										 |  |  | 	metrics.MApiUserSignUpStarted.Inc() | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-15 20:01:58 +08:00
										 |  |  | 	return response.JSON(http.StatusOK, util.DynMap{"status": "SignUpCreated"}) | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-27 15:50:36 +08:00
										 |  |  | func (hs *HTTPServer) SignUpStep2(c *contextmodel.ReqContext) response.Response { | 
					
						
							| 
									
										
										
										
											2021-11-29 17:18:01 +08:00
										 |  |  | 	form := dtos.SignUpStep2Form{} | 
					
						
							|  |  |  | 	if err := web.Bind(c.Req, &form); err != nil { | 
					
						
							|  |  |  | 		return response.Error(http.StatusBadRequest, "bad request data", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 	if !setting.AllowUserSignUp { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return response.Error(401, "User signup is disabled", nil) | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-14 20:11:26 +08:00
										 |  |  | 	form.Email = strings.TrimSpace(form.Email) | 
					
						
							|  |  |  | 	form.Username = strings.TrimSpace(form.Username) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-28 20:32:25 +08:00
										 |  |  | 	createUserCmd := user.CreateUserCommand{ | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 		Email:    form.Email, | 
					
						
							|  |  |  | 		Login:    form.Username, | 
					
						
							|  |  |  | 		Name:     form.Name, | 
					
						
							|  |  |  | 		Password: form.Password, | 
					
						
							|  |  |  | 		OrgName:  form.OrgName, | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-01 18:35:06 +08:00
										 |  |  | 	// verify email
 | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 	if setting.VerifyEmailEnabled { | 
					
						
							| 
									
										
										
										
											2022-01-27 17:33:02 +08:00
										 |  |  | 		if ok, rsp := hs.verifyUserSignUpEmail(c.Req.Context(), form.Email, form.Code); !ok { | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 			return rsp | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		createUserCmd.EmailVerified = true | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-29 17:20:44 +08:00
										 |  |  | 	usr, err := hs.userService.Create(c.Req.Context(), &createUserCmd) | 
					
						
							| 
									
										
										
										
											2021-03-19 00:16:56 +08:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2022-07-20 20:50:06 +08:00
										 |  |  | 		if errors.Is(err, user.ErrUserAlreadyExists) { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 			return response.Error(401, "User with same email address already exists", nil) | 
					
						
							| 
									
										
										
										
											2020-08-13 20:38:54 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return response.Error(500, "Failed to create user", err) | 
					
						
							| 
									
										
										
										
											2015-08-28 21:14:24 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 00:56:53 +08:00
										 |  |  | 	// publish signup event
 | 
					
						
							| 
									
										
										
										
											2022-06-14 22:07:41 +08:00
										 |  |  | 	if err := hs.bus.Publish(c.Req.Context(), &events.SignUpCompleted{ | 
					
						
							| 
									
										
										
										
											2022-06-28 20:32:25 +08:00
										 |  |  | 		Email: usr.Email, | 
					
						
							|  |  |  | 		Name:  usr.NameOrFallback(), | 
					
						
							| 
									
										
										
										
											2019-10-09 00:57:53 +08:00
										 |  |  | 	}); err != nil { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return response.Error(500, "Failed to publish event", err) | 
					
						
							| 
									
										
										
										
											2019-10-09 00:57:53 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-08-28 21:14:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 	// mark temp user as completed
 | 
					
						
							| 
									
										
										
										
											2023-01-06 16:02:05 +08:00
										 |  |  | 	if ok, rsp := hs.updateTempUserStatus(c.Req.Context(), form.Code, tempuser.TmpUserCompleted); !ok { | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 		return rsp | 
					
						
							| 
									
										
										
										
											2015-08-31 00:56:53 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-28 21:14:24 +08:00
										 |  |  | 	// check for pending invites
 | 
					
						
							| 
									
										
										
										
											2023-01-06 16:02:05 +08:00
										 |  |  | 	invitesQuery := tempuser.GetTempUsersQuery{Email: form.Email, Status: tempuser.TmpUserInvitePending} | 
					
						
							|  |  |  | 	invitesQueryResult, err := hs.tempUserService.GetTempUsersQuery(c.Req.Context(), &invitesQuery) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return response.Error(500, "Failed to query database for invites", err) | 
					
						
							| 
									
										
										
										
											2015-08-28 21:14:24 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-13 08:01:03 +08:00
										 |  |  | 	apiResponse := util.DynMap{"message": "User sign up completed successfully", "code": "redirect-to-landing-page"} | 
					
						
							| 
									
										
										
										
											2023-01-06 16:02:05 +08:00
										 |  |  | 	for _, invite := range invitesQueryResult { | 
					
						
							| 
									
										
										
										
											2022-06-28 20:32:25 +08:00
										 |  |  | 		if ok, rsp := hs.applyUserInvite(c.Req.Context(), usr, invite, false); !ok { | 
					
						
							| 
									
										
										
										
											2015-08-31 00:56:53 +08:00
										 |  |  | 			return rsp | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		apiResponse["code"] = "redirect-to-select-org" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-28 20:32:25 +08:00
										 |  |  | 	err = hs.loginUserWithUser(usr, c) | 
					
						
							| 
									
										
										
										
											2020-03-23 20:37:53 +08:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return response.Error(500, "failed to login user", err) | 
					
						
							| 
									
										
										
										
											2020-03-23 20:37:53 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-16 22:58:46 +08:00
										 |  |  | 	metrics.MApiUserSignUpCompleted.Inc() | 
					
						
							| 
									
										
										
										
											2015-08-31 00:56:53 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-15 20:01:58 +08:00
										 |  |  | 	return response.JSON(http.StatusOK, apiResponse) | 
					
						
							| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-27 17:33:02 +08:00
										 |  |  | func (hs *HTTPServer) verifyUserSignUpEmail(ctx context.Context, email string, code string) (bool, response.Response) { | 
					
						
							| 
									
										
										
										
											2023-01-06 16:02:05 +08:00
										 |  |  | 	query := tempuser.GetTempUserByCodeQuery{Code: code} | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 16:02:05 +08:00
										 |  |  | 	queryResult, err := hs.tempUserService.GetTempUserByCode(ctx, &query) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		if errors.Is(err, tempuser.ErrTempUserNotFound) { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 			return false, response.Error(404, "Invalid email verification code", nil) | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return false, response.Error(500, "Failed to read temp user", err) | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 16:02:05 +08:00
										 |  |  | 	tempUser := queryResult | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 	if tempUser.Email != email { | 
					
						
							| 
									
										
										
										
											2021-01-15 21:43:20 +08:00
										 |  |  | 		return false, response.Error(404, "Email verification code does not match email", nil) | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return true, nil | 
					
						
							|  |  |  | } |