| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | package api | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/api/dtos" | 
					
						
							| 
									
										
										
										
											2015-02-05 17:37:13 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/bus" | 
					
						
							| 
									
										
										
										
											2015-06-08 23:56:56 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/events" | 
					
						
							| 
									
										
										
										
											2015-03-23 03:14:00 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/metrics" | 
					
						
							| 
									
										
										
										
											2015-02-05 17:37:13 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/middleware" | 
					
						
							|  |  |  | 	m "github.com/grafana/grafana/pkg/models" | 
					
						
							|  |  |  | 	"github.com/grafana/grafana/pkg/setting" | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/util" | 
					
						
							| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | // GET /api/user/signup/options
 | 
					
						
							|  |  |  | func GetSignUpOptions(c *middleware.Context) Response { | 
					
						
							|  |  |  | 	return Json(200, util.DynMap{ | 
					
						
							|  |  |  | 		"verifyEmailEnabled": setting.VerifyEmailEnabled, | 
					
						
							|  |  |  | 		"autoAssignOrg":      setting.AutoAssignOrg, | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 22:14:53 +08:00
										 |  |  | // POST /api/user/signup
 | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | func SignUp(c *middleware.Context, form dtos.SignUpForm) Response { | 
					
						
							| 
									
										
										
										
											2015-03-11 23:19:29 +08:00
										 |  |  | 	if !setting.AllowUserSignUp { | 
					
						
							| 
									
										
										
										
											2015-06-10 16:15:42 +08:00
										 |  |  | 		return ApiError(401, "User signup is disabled", nil) | 
					
						
							| 
									
										
										
										
											2015-01-29 22:46:54 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 	existing := m.GetUserByLoginQuery{LoginOrEmail: form.Email} | 
					
						
							|  |  |  | 	if err := bus.Dispatch(&existing); err == nil { | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 		return ApiError(422, "User with same email address already exists", nil) | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cmd := m.CreateTempUserCommand{} | 
					
						
							|  |  |  | 	cmd.OrgId = -1 | 
					
						
							|  |  |  | 	cmd.Email = form.Email | 
					
						
							|  |  |  | 	cmd.Status = m.TmpUserSignUpStarted | 
					
						
							|  |  |  | 	cmd.InvitedByUserId = c.UserId | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 	cmd.Code = util.GetRandomString(20) | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 	cmd.RemoteAddr = c.Req.RemoteAddr | 
					
						
							| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-08 16:00:00 +08:00
										 |  |  | 	if err := bus.Dispatch(&cmd); err != nil { | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 		return ApiError(500, "Failed to create signup", err) | 
					
						
							| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 	bus.Publish(&events.SignUpStarted{ | 
					
						
							|  |  |  | 		Email: form.Email, | 
					
						
							|  |  |  | 		Code:  cmd.Code, | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2015-01-21 16:52:40 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-27 19:59:58 +08:00
										 |  |  | 	metrics.M_Api_User_SignUpStarted.Inc(1) | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return Json(200, util.DynMap{"status": "SignUpCreated"}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func SignUpStep2(c *middleware.Context, form dtos.SignUpStep2Form) Response { | 
					
						
							|  |  |  | 	if !setting.AllowUserSignUp { | 
					
						
							|  |  |  | 		return ApiError(401, "User signup is disabled", nil) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 	createUserCmd := m.CreateUserCommand{ | 
					
						
							|  |  |  | 		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 { | 
					
						
							|  |  |  | 		if ok, rsp := verifyUserSignUpEmail(form.Email, form.Code); !ok { | 
					
						
							|  |  |  | 			return rsp | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		createUserCmd.EmailVerified = true | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-01 18:35:06 +08:00
										 |  |  | 	// check if user exists
 | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 	existing := m.GetUserByLoginQuery{LoginOrEmail: form.Email} | 
					
						
							| 
									
										
										
										
											2015-08-28 15:24:30 +08:00
										 |  |  | 	if err := bus.Dispatch(&existing); err == nil { | 
					
						
							|  |  |  | 		return ApiError(401, "User with same email address already exists", nil) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-01 18:35:06 +08:00
										 |  |  | 	// dispatch create command
 | 
					
						
							| 
									
										
										
										
											2015-08-28 21:14:24 +08:00
										 |  |  | 	if err := bus.Dispatch(&createUserCmd); err != nil { | 
					
						
							|  |  |  | 		return ApiError(500, "Failed to create user", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 00:56:53 +08:00
										 |  |  | 	// publish signup event
 | 
					
						
							|  |  |  | 	user := &createUserCmd.Result | 
					
						
							| 
									
										
										
										
											2015-08-28 21:14:24 +08:00
										 |  |  | 	bus.Publish(&events.SignUpCompleted{ | 
					
						
							|  |  |  | 		Email: user.Email, | 
					
						
							|  |  |  | 		Name:  user.NameOrFallback(), | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 	// mark temp user as completed
 | 
					
						
							|  |  |  | 	if ok, rsp := updateTempUserStatus(form.Code, m.TmpUserCompleted); !ok { | 
					
						
							|  |  |  | 		return rsp | 
					
						
							| 
									
										
										
										
											2015-08-31 00:56:53 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-28 21:14:24 +08:00
										 |  |  | 	// check for pending invites
 | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 	invitesQuery := m.GetTempUsersQuery{Email: form.Email, Status: m.TmpUserInvitePending} | 
					
						
							| 
									
										
										
										
											2015-08-28 21:14:24 +08:00
										 |  |  | 	if err := bus.Dispatch(&invitesQuery); err != nil { | 
					
						
							|  |  |  | 		return ApiError(500, "Failed to query database for invites", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-13 08:01:03 +08:00
										 |  |  | 	apiResponse := util.DynMap{"message": "User sign up completed successfully", "code": "redirect-to-landing-page"} | 
					
						
							| 
									
										
										
										
											2015-08-31 00:56:53 +08:00
										 |  |  | 	for _, invite := range invitesQuery.Result { | 
					
						
							|  |  |  | 		if ok, rsp := applyUserInvite(user, invite, false); !ok { | 
					
						
							|  |  |  | 			return rsp | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		apiResponse["code"] = "redirect-to-select-org" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	loginUserWithUser(user, c) | 
					
						
							| 
									
										
										
										
											2015-08-28 21:14:24 +08:00
										 |  |  | 	metrics.M_Api_User_SignUpCompleted.Inc(1) | 
					
						
							| 
									
										
										
										
											2015-08-31 00:56:53 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return Json(200, apiResponse) | 
					
						
							| 
									
										
										
										
											2014-12-29 20:36:08 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-08-31 17:35:07 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func verifyUserSignUpEmail(email string, code string) (bool, Response) { | 
					
						
							|  |  |  | 	query := m.GetTempUserByCodeQuery{Code: code} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err := bus.Dispatch(&query); err != nil { | 
					
						
							|  |  |  | 		if err == m.ErrTempUserNotFound { | 
					
						
							|  |  |  | 			return false, ApiError(404, "Invalid email verification code", nil) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return false, ApiError(500, "Failed to read temp user", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tempUser := query.Result | 
					
						
							|  |  |  | 	if tempUser.Email != email { | 
					
						
							|  |  |  | 		return false, ApiError(404, "Email verification code does not match email", nil) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return true, nil | 
					
						
							|  |  |  | } |