mirror of
https://github.com/volatiletech/authboss.git
synced 2025-01-10 04:17:59 +02:00
334 lines
8.4 KiB
Go
334 lines
8.4 KiB
Go
package defaults
|
|
|
|
import (
|
|
"encoding/json"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"net/url"
|
|
"regexp"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/volatiletech/authboss"
|
|
)
|
|
|
|
// FormValue types
|
|
const (
|
|
FormValueEmail = "email"
|
|
FormValuePassword = "password"
|
|
FormValueUsername = "username"
|
|
|
|
FormValueConfirm = "cnf"
|
|
FormValueToken = "token"
|
|
FormValueCode = "code"
|
|
FormValueRecoveryCode = "recovery_code"
|
|
FormValuePhoneNumber = "phone_number"
|
|
)
|
|
|
|
// UserValues from the login form
|
|
type UserValues struct {
|
|
HTTPFormValidator
|
|
|
|
PID string
|
|
Password string
|
|
|
|
Arbitrary map[string]string
|
|
}
|
|
|
|
// GetPID from the values
|
|
func (u UserValues) GetPID() string {
|
|
return u.PID
|
|
}
|
|
|
|
// GetPassword from the values
|
|
func (u UserValues) GetPassword() string {
|
|
return u.Password
|
|
}
|
|
|
|
// GetValues from the form.
|
|
func (u UserValues) GetValues() map[string]string {
|
|
return u.Arbitrary
|
|
}
|
|
|
|
// ConfirmValues retrieves values on the confirm page.
|
|
type ConfirmValues struct {
|
|
HTTPFormValidator
|
|
|
|
Token string
|
|
}
|
|
|
|
// GetToken from the confirm values
|
|
func (c ConfirmValues) GetToken() string {
|
|
return c.Token
|
|
}
|
|
|
|
// RecoverStartValues for recover_start page
|
|
type RecoverStartValues struct {
|
|
HTTPFormValidator
|
|
|
|
PID string
|
|
}
|
|
|
|
// GetPID for recovery
|
|
func (r RecoverStartValues) GetPID() string { return r.PID }
|
|
|
|
// RecoverMiddleValues for recover_middle page
|
|
type RecoverMiddleValues struct {
|
|
HTTPFormValidator
|
|
|
|
Token string
|
|
}
|
|
|
|
// GetToken for recovery
|
|
func (r RecoverMiddleValues) GetToken() string { return r.Token }
|
|
|
|
// RecoverEndValues for recover_end page
|
|
type RecoverEndValues struct {
|
|
HTTPFormValidator
|
|
|
|
Token string
|
|
NewPassword string
|
|
}
|
|
|
|
// GetToken for recovery
|
|
func (r RecoverEndValues) GetToken() string { return r.Token }
|
|
|
|
// GetPassword for recovery
|
|
func (r RecoverEndValues) GetPassword() string { return r.NewPassword }
|
|
|
|
// TwoFA for totp2fa_validate page
|
|
type TwoFA struct {
|
|
HTTPFormValidator
|
|
|
|
Code string
|
|
RecoveryCode string
|
|
}
|
|
|
|
// GetCode from authenticator
|
|
func (t TwoFA) GetCode() string { return t.Code }
|
|
|
|
// GetRecoveryCode for authenticator
|
|
func (t TwoFA) GetRecoveryCode() string { return t.RecoveryCode }
|
|
|
|
// SMSTwoFA for sms2fa_validate page
|
|
type SMSTwoFA struct {
|
|
HTTPFormValidator
|
|
|
|
Code string
|
|
RecoveryCode string
|
|
PhoneNumber string
|
|
}
|
|
|
|
// GetCode from sms
|
|
func (s SMSTwoFA) GetCode() string { return s.Code }
|
|
|
|
// GetRecoveryCode from sms
|
|
func (s SMSTwoFA) GetRecoveryCode() string { return s.RecoveryCode }
|
|
|
|
// GetPhoneNumber from authenticator
|
|
func (s SMSTwoFA) GetPhoneNumber() string { return s.PhoneNumber }
|
|
|
|
// HTTPBodyReader reads forms from various pages and decodes
|
|
// them.
|
|
type HTTPBodyReader struct {
|
|
// ReadJSON if turned on reads json from the http request
|
|
// instead of a encoded form.
|
|
ReadJSON bool
|
|
|
|
// UseUsername instead of e-mail address
|
|
UseUsername bool
|
|
|
|
// Rulesets for each page.
|
|
Rulesets map[string][]Rules
|
|
// Confirm fields for each page.
|
|
Confirms map[string][]string
|
|
// Whitelist values for each page through the html forms
|
|
// this is for security so that we can properly protect the
|
|
// arbitrary user API. In reality this really only needs to be set
|
|
// for the register page since everything else is expecting
|
|
// a hardcoded set of values.
|
|
Whitelist map[string][]string
|
|
}
|
|
|
|
// NewHTTPBodyReader creates a form reader with default validation rules
|
|
// and fields for each page. If no defaults are required, simply construct
|
|
// this using the struct members itself for more control.
|
|
func NewHTTPBodyReader(readJSON, useUsernameNotEmail bool) *HTTPBodyReader {
|
|
var pid string
|
|
var pidRules Rules
|
|
|
|
if useUsernameNotEmail {
|
|
pid = "username"
|
|
pidRules = Rules{
|
|
FieldName: pid, Required: true,
|
|
MatchError: "Usernames must only start with letters, and contain letters and numbers",
|
|
MustMatch: regexp.MustCompile(`(?i)[a-z][a-z0-9]?`),
|
|
}
|
|
} else {
|
|
pid = "email"
|
|
pidRules = Rules{
|
|
FieldName: pid, Required: true,
|
|
MatchError: "Must be a valid e-mail address",
|
|
MustMatch: regexp.MustCompile(`.*@.*\.[a-z]{1,}`),
|
|
}
|
|
}
|
|
|
|
passwordRule := Rules{
|
|
FieldName: "password",
|
|
MinLength: 8,
|
|
MinNumeric: 1,
|
|
MinSymbols: 1,
|
|
MinUpper: 1,
|
|
MinLower: 1,
|
|
}
|
|
|
|
return &HTTPBodyReader{
|
|
UseUsername: useUsernameNotEmail,
|
|
ReadJSON: readJSON,
|
|
Rulesets: map[string][]Rules{
|
|
"login": {pidRules},
|
|
"register": {pidRules, passwordRule},
|
|
"confirm": {Rules{FieldName: FormValueConfirm, Required: true}},
|
|
"recover_start": {pidRules},
|
|
"recover_end": {passwordRule},
|
|
|
|
"twofactor_verify_end": {Rules{FieldName: FormValueToken, Required: true}},
|
|
},
|
|
Confirms: map[string][]string{
|
|
"register": {FormValuePassword, authboss.ConfirmPrefix + FormValuePassword},
|
|
"recover_end": {FormValuePassword, authboss.ConfirmPrefix + FormValuePassword},
|
|
},
|
|
Whitelist: map[string][]string{
|
|
"register": []string{FormValueEmail, FormValuePassword},
|
|
},
|
|
}
|
|
}
|
|
|
|
// Read the form pages
|
|
func (h HTTPBodyReader) Read(page string, r *http.Request) (authboss.Validator, error) {
|
|
var values map[string]string
|
|
|
|
if h.ReadJSON {
|
|
b, err := ioutil.ReadAll(r.Body)
|
|
r.Body.Close()
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to read http body")
|
|
}
|
|
|
|
if err = json.Unmarshal(b, &values); err != nil {
|
|
return nil, errors.Wrap(err, "failed to parse json http body")
|
|
}
|
|
} else {
|
|
if err := r.ParseForm(); err != nil {
|
|
return nil, errors.Wrapf(err, "failed to parse form on page: %s", page)
|
|
}
|
|
values = URLValuesToMap(r.Form)
|
|
}
|
|
|
|
rules := h.Rulesets[page]
|
|
confirms := h.Confirms[page]
|
|
whitelist := h.Whitelist[page]
|
|
|
|
switch page {
|
|
case "confirm":
|
|
return ConfirmValues{
|
|
HTTPFormValidator: HTTPFormValidator{Values: values, Ruleset: rules},
|
|
Token: values[FormValueConfirm],
|
|
}, nil
|
|
case "login":
|
|
var pid string
|
|
if h.UseUsername {
|
|
pid = values[FormValueUsername]
|
|
} else {
|
|
pid = values[FormValueEmail]
|
|
}
|
|
|
|
return UserValues{
|
|
HTTPFormValidator: HTTPFormValidator{Values: values, Ruleset: rules, ConfirmFields: confirms},
|
|
PID: pid,
|
|
Password: values[FormValuePassword],
|
|
}, nil
|
|
case "recover_start":
|
|
var pid string
|
|
if h.UseUsername {
|
|
pid = values[FormValueUsername]
|
|
} else {
|
|
pid = values[FormValueEmail]
|
|
}
|
|
|
|
return RecoverStartValues{
|
|
HTTPFormValidator: HTTPFormValidator{Values: values, Ruleset: rules, ConfirmFields: confirms},
|
|
PID: pid,
|
|
}, nil
|
|
case "recover_middle":
|
|
return RecoverMiddleValues{
|
|
HTTPFormValidator: HTTPFormValidator{Values: values, Ruleset: rules, ConfirmFields: confirms},
|
|
Token: values[FormValueToken],
|
|
}, nil
|
|
case "recover_end":
|
|
return RecoverEndValues{
|
|
HTTPFormValidator: HTTPFormValidator{Values: values, Ruleset: rules, ConfirmFields: confirms},
|
|
Token: values[FormValueToken],
|
|
NewPassword: values[FormValuePassword],
|
|
}, nil
|
|
case "twofactor_verify_end":
|
|
// Reuse ConfirmValues here, it's the same values we need
|
|
return ConfirmValues{
|
|
HTTPFormValidator: HTTPFormValidator{Values: values, Ruleset: rules, ConfirmFields: confirms},
|
|
Token: values[FormValueToken],
|
|
}, nil
|
|
case "totp2fa_confirm", "totp2fa_remove", "totp2fa_validate":
|
|
return TwoFA{
|
|
HTTPFormValidator: HTTPFormValidator{Values: values, Ruleset: rules, ConfirmFields: confirms},
|
|
Code: values[FormValueCode],
|
|
RecoveryCode: values[FormValueRecoveryCode],
|
|
}, nil
|
|
case "sms2fa_setup", "sms2fa_remove", "sms2fa_confirm", "sms2fa_validate":
|
|
return SMSTwoFA{
|
|
HTTPFormValidator: HTTPFormValidator{Values: values, Ruleset: rules, ConfirmFields: confirms},
|
|
Code: values[FormValueCode],
|
|
PhoneNumber: values[FormValuePhoneNumber],
|
|
RecoveryCode: values[FormValueRecoveryCode],
|
|
}, nil
|
|
case "register":
|
|
arbitrary := make(map[string]string)
|
|
|
|
for k, v := range values {
|
|
for _, w := range whitelist {
|
|
if k == w {
|
|
arbitrary[k] = v
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
var pid string
|
|
if h.UseUsername {
|
|
pid = values[FormValueUsername]
|
|
} else {
|
|
pid = values[FormValueEmail]
|
|
}
|
|
|
|
return UserValues{
|
|
HTTPFormValidator: HTTPFormValidator{Values: values, Ruleset: rules, ConfirmFields: confirms},
|
|
PID: pid,
|
|
Password: values[FormValuePassword],
|
|
Arbitrary: arbitrary,
|
|
}, nil
|
|
default:
|
|
return nil, errors.Errorf("failed to parse unknown page's form: %s", page)
|
|
}
|
|
}
|
|
|
|
// URLValuesToMap helps create a map from url.Values
|
|
func URLValuesToMap(form url.Values) map[string]string {
|
|
values := make(map[string]string)
|
|
|
|
for k, v := range form {
|
|
if len(v) != 0 {
|
|
values[k] = v[0]
|
|
}
|
|
}
|
|
|
|
return values
|
|
}
|