diff --git a/conf/defaults.ini b/conf/defaults.ini
index 8eafec90e0d..6ebf5fac01d 100644
--- a/conf/defaults.ini
+++ b/conf/defaults.ini
@@ -134,6 +134,9 @@ auto_assign_org = true
# Default role new users will be automatically assigned (if auto_assign_org above is set to true)
auto_assign_org_role = Viewer
+# Require email validation before sign up completes
+verify_email_enabled = false
+
#################################### Anonymous Auth ##########################
[auth.anonymous]
# enable anonymous access
diff --git a/grafana b/grafana
deleted file mode 100755
index d363cea6a70..00000000000
Binary files a/grafana and /dev/null differ
diff --git a/pkg/api/api.go b/pkg/api/api.go
index fe165fe9307..1a876c3bc70 100644
--- a/pkg/api/api.go
+++ b/pkg/api/api.go
@@ -43,6 +43,7 @@ func Register(r *macaron.Macaron) {
// sign up
r.Get("/signup", Index)
+ r.Get("/api/user/signup/options", wrap(GetSignUpOptions))
r.Post("/api/user/signup", bind(dtos.SignUpForm{}), wrap(SignUp))
r.Post("/api/user/signup/step2", bind(dtos.SignUpStep2Form{}), wrap(SignUpStep2))
diff --git a/pkg/api/signup.go b/pkg/api/signup.go
index f8b47e1dfe5..fa450b93df2 100644
--- a/pkg/api/signup.go
+++ b/pkg/api/signup.go
@@ -11,6 +11,14 @@ import (
"github.com/grafana/grafana/pkg/util"
)
+// GET /api/user/signup/options
+func GetSignUpOptions(c *middleware.Context) Response {
+ return Json(200, util.DynMap{
+ "verifyEmailEnabled": setting.VerifyEmailEnabled,
+ "autoAssignOrg": setting.AutoAssignOrg,
+ })
+}
+
// POST /api/user/signup
func SignUp(c *middleware.Context, form dtos.SignUpForm) Response {
if !setting.AllowUserSignUp {
@@ -19,7 +27,7 @@ func SignUp(c *middleware.Context, form dtos.SignUpForm) Response {
existing := m.GetUserByLoginQuery{LoginOrEmail: form.Email}
if err := bus.Dispatch(&existing); err == nil {
- return ApiError(401, "User with same email address already exists", nil)
+ return ApiError(422, "User with same email address already exists", nil)
}
cmd := m.CreateTempUserCommand{}
@@ -49,64 +57,49 @@ func SignUpStep2(c *middleware.Context, form dtos.SignUpStep2Form) Response {
return ApiError(401, "User signup is disabled", nil)
}
- query := m.GetTempUserByCodeQuery{Code: form.Code}
-
- if err := bus.Dispatch(&query); err != nil {
- if err == m.ErrTempUserNotFound {
- return ApiError(404, "Invalid email verification code", nil)
- }
- return ApiError(500, "Failed to read temp user", err)
- }
-
- tempUser := query.Result
- if tempUser.Email != form.Email {
- return ApiError(404, "Email verification code does not match email", nil)
- }
-
- existing := m.GetUserByLoginQuery{LoginOrEmail: tempUser.Email}
- if err := bus.Dispatch(&existing); err == nil {
- return ApiError(401, "User with same email address already exists", nil)
- }
-
- // create user
createUserCmd := m.CreateUserCommand{
- Email: tempUser.Email,
+ Email: form.Email,
Login: form.Username,
Name: form.Name,
Password: form.Password,
OrgName: form.OrgName,
}
+ if setting.VerifyEmailEnabled {
+ if ok, rsp := verifyUserSignUpEmail(form.Email, form.Code); !ok {
+ return rsp
+ }
+ createUserCmd.EmailVerified = true
+ }
+
+ existing := m.GetUserByLoginQuery{LoginOrEmail: form.Email}
+ if err := bus.Dispatch(&existing); err == nil {
+ return ApiError(401, "User with same email address already exists", nil)
+ }
+
if err := bus.Dispatch(&createUserCmd); err != nil {
return ApiError(500, "Failed to create user", err)
}
// publish signup event
user := &createUserCmd.Result
-
bus.Publish(&events.SignUpCompleted{
Email: user.Email,
Name: user.NameOrFallback(),
})
- // update tempuser
- updateTempUserCmd := m.UpdateTempUserStatusCommand{
- Code: tempUser.Code,
- Status: m.TmpUserCompleted,
- }
-
- if err := bus.Dispatch(&updateTempUserCmd); err != nil {
- return ApiError(500, "Failed to update temp user", err)
+ // mark temp user as completed
+ if ok, rsp := updateTempUserStatus(form.Code, m.TmpUserCompleted); !ok {
+ return rsp
}
// check for pending invites
- invitesQuery := m.GetTempUsersQuery{Email: tempUser.Email, Status: m.TmpUserInvitePending}
+ invitesQuery := m.GetTempUsersQuery{Email: form.Email, Status: m.TmpUserInvitePending}
if err := bus.Dispatch(&invitesQuery); err != nil {
return ApiError(500, "Failed to query database for invites", err)
}
apiResponse := util.DynMap{"message": "User sign up completed succesfully", "code": "redirect-to-landing-page"}
-
for _, invite := range invitesQuery.Result {
if ok, rsp := applyUserInvite(user, invite, false); !ok {
return rsp
@@ -119,3 +112,21 @@ func SignUpStep2(c *middleware.Context, form dtos.SignUpStep2Form) Response {
return Json(200, apiResponse)
}
+
+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
+}
diff --git a/pkg/models/user.go b/pkg/models/user.go
index a7393eacf68..30ab34db9a6 100644
--- a/pkg/models/user.go
+++ b/pkg/models/user.go
@@ -44,15 +44,16 @@ func (u *User) NameOrFallback() string {
// COMMANDS
type CreateUserCommand struct {
- Email string `json:"email" binding:"Required"`
- Login string `json:"login"`
- Name string `json:"name"`
- Company string `json:"compay"`
- OrgName string `json:"orgName"`
- Password string `json:"password" binding:"Required"`
- IsAdmin bool `json:"-"`
+ Email string
+ Login string
+ Name string
+ Company string
+ OrgName string
+ Password string
+ EmailVerified bool
+ IsAdmin bool
- Result User `json:"-"`
+ Result User
}
type UpdateUserCommand struct {
diff --git a/pkg/services/notifications/notifications.go b/pkg/services/notifications/notifications.go
index 2b041e9e4b7..3d2c95b7c23 100644
--- a/pkg/services/notifications/notifications.go
+++ b/pkg/services/notifications/notifications.go
@@ -123,6 +123,10 @@ func validateResetPasswordCode(query *m.ValidateResetPasswordCodeQuery) error {
}
func signUpStartedHandler(evt *events.SignUpStarted) error {
+ if !setting.VerifyEmailEnabled {
+ return nil
+ }
+
log.Info("User signup started: %s", evt.Email)
if evt.Email == "" {
diff --git a/pkg/services/sqlstore/user.go b/pkg/services/sqlstore/user.go
index 4b75f1be26a..b26000673d6 100644
--- a/pkg/services/sqlstore/user.go
+++ b/pkg/services/sqlstore/user.go
@@ -80,14 +80,15 @@ func CreateUser(cmd *m.CreateUserCommand) error {
// create user
user := m.User{
- Email: cmd.Email,
- Name: cmd.Name,
- Login: cmd.Login,
- Company: cmd.Company,
- IsAdmin: cmd.IsAdmin,
- OrgId: orgId,
- Created: time.Now(),
- Updated: time.Now(),
+ Email: cmd.Email,
+ Name: cmd.Name,
+ Login: cmd.Login,
+ Company: cmd.Company,
+ IsAdmin: cmd.IsAdmin,
+ OrgId: orgId,
+ EmailVerified: cmd.EmailVerified,
+ Created: time.Now(),
+ Updated: time.Now(),
}
if len(cmd.Password) > 0 {
diff --git a/pkg/setting/setting.go b/pkg/setting/setting.go
index ae69e65192a..a29d54eb662 100644
--- a/pkg/setting/setting.go
+++ b/pkg/setting/setting.go
@@ -75,11 +75,11 @@ var (
EmailCodeValidMinutes int
// User settings
- AllowUserSignUp bool
- AllowUserOrgCreate bool
- AutoAssignOrg bool
- AutoAssignOrgRole string
- RequireEmailValidation bool
+ AllowUserSignUp bool
+ AllowUserOrgCreate bool
+ AutoAssignOrg bool
+ AutoAssignOrgRole string
+ VerifyEmailEnabled bool
// Http auth
AdminUser string
@@ -394,6 +394,7 @@ func NewConfigContext(args *CommandLineArgs) {
AllowUserOrgCreate = users.Key("allow_org_create").MustBool(true)
AutoAssignOrg = users.Key("auto_assign_org").MustBool(true)
AutoAssignOrgRole = users.Key("auto_assign_org_role").In("Editor", []string{"Editor", "Admin", "Read Only Editor", "Viewer"})
+ VerifyEmailEnabled = users.Key("verify_email_enabled").MustBool(false)
// anonymous access
AnonymousEnabled = Cfg.Section("auth.anonymous").Key("enabled").MustBool(false)
@@ -425,6 +426,10 @@ func NewConfigContext(args *CommandLineArgs) {
readSessionConfig()
readSmtpSettings()
+
+ if VerifyEmailEnabled && !Smtp.Enabled {
+ log.Warn("require_email_validation is enabled but smpt is disabled")
+ }
}
func readSessionConfig() {
diff --git a/public/app/controllers/signupCtrl.js b/public/app/controllers/signupCtrl.js
index d7a7b318b32..c23e7b151c6 100644
--- a/public/app/controllers/signupCtrl.js
+++ b/public/app/controllers/signupCtrl.js
@@ -19,10 +19,18 @@ function (angular, config) {
$scope.formModel.email = params.email;
$scope.formModel.username = params.email;
$scope.formModel.code = params.code;
+
+ $scope.verifyEmailEnabled = false;
+ $scope.autoAssignOrg = false;
+
+ backendSrv.get('/api/user/signup/options').then(function(options) {
+ $scope.verifyEmailEnabled = options.verifyEmailEnabled;
+ $scope.autoAssignOrg = options.autoAssignOrg;
+ });
};
$scope.submit = function() {
- if (!$scope.signupForm.$valid) {
+ if (!$scope.signUpForm.$valid) {
return;
}
diff --git a/public/app/partials/signup_step2.html b/public/app/partials/signup_step2.html
index 5343c73258c..a604d216372 100644
--- a/public/app/partials/signup_step2.html
+++ b/public/app/partials/signup_step2.html
@@ -27,9 +27,9 @@
-