summaryrefslogtreecommitdiffstats
path: root/modules
diff options
context:
space:
mode:
Diffstat (limited to 'modules')
-rw-r--r--modules/password/password.go63
-rw-r--r--modules/password/password_test.go75
-rw-r--r--modules/setting/setting.go24
3 files changed, 120 insertions, 42 deletions
diff --git a/modules/password/password.go b/modules/password/password.go
index 54131b9641..92986977ec 100644
--- a/modules/password/password.go
+++ b/modules/password/password.go
@@ -7,45 +7,60 @@ package password
import (
"crypto/rand"
"math/big"
- "regexp"
+ "strings"
"sync"
"code.gitea.io/gitea/modules/setting"
)
-var matchComplexities = map[string]regexp.Regexp{}
-var matchComplexityOnce sync.Once
-var validChars string
-var validComplexities = map[string]string{
- "lower": "abcdefghijklmnopqrstuvwxyz",
- "upper": "ABCDEFGHIJKLMNOPQRSTUVWXYZ",
- "digit": "0123456789",
- "spec": `][ !"#$%&'()*+,./:;<=>?@\^_{|}~` + "`-",
-}
+var (
+ matchComplexityOnce sync.Once
+ validChars string
+ requiredChars []string
+
+ charComplexities = map[string]string{
+ "lower": `abcdefghijklmnopqrstuvwxyz`,
+ "upper": `ABCDEFGHIJKLMNOPQRSTUVWXYZ`,
+ "digit": `0123456789`,
+ "spec": ` !"#$%&'()*+,-./:;<=>?@[\]^_{|}~` + "`",
+ }
+)
// NewComplexity for preparation
func NewComplexity() {
matchComplexityOnce.Do(func() {
- if len(setting.PasswordComplexity) > 0 {
- for key, val := range setting.PasswordComplexity {
- matchComplexity := regexp.MustCompile(val)
- matchComplexities[key] = *matchComplexity
- validChars += validComplexities[key]
+ setupComplexity(setting.PasswordComplexity)
+ })
+}
+
+func setupComplexity(values []string) {
+ if len(values) != 1 || values[0] != "off" {
+ for _, val := range values {
+ if chars, ok := charComplexities[val]; ok {
+ validChars += chars
+ requiredChars = append(requiredChars, chars)
}
- } else {
- for _, val := range validComplexities {
- validChars += val
+ }
+ if len(requiredChars) == 0 {
+ // No valid character classes found; use all classes as default
+ for _, chars := range charComplexities {
+ validChars += chars
+ requiredChars = append(requiredChars, chars)
}
}
- })
+ }
+ if validChars == "" {
+ // No complexities to check; provide a sensible default for password generation
+ validChars = charComplexities["lower"] + charComplexities["upper"] + charComplexities["digit"]
+ }
}
-// IsComplexEnough return True if password is Complexity
+// IsComplexEnough return True if password meets complexity settings
func IsComplexEnough(pwd string) bool {
- if len(setting.PasswordComplexity) > 0 {
- NewComplexity()
- for _, val := range matchComplexities {
- if !val.MatchString(pwd) {
+ NewComplexity()
+ if len(validChars) > 0 {
+ for _, req := range requiredChars {
+ if !strings.ContainsAny(req, pwd) {
return false
}
}
diff --git a/modules/password/password_test.go b/modules/password/password_test.go
new file mode 100644
index 0000000000..d46a6d1571
--- /dev/null
+++ b/modules/password/password_test.go
@@ -0,0 +1,75 @@
+// Copyright 2019 The Gitea Authors. All rights reserved.
+// Use of this source code is governed by a MIT-style
+// license that can be found in the LICENSE file.
+
+package password
+
+import (
+ "testing"
+
+ "github.com/stretchr/testify/assert"
+)
+
+func TestComplexity_IsComplexEnough(t *testing.T) {
+ matchComplexityOnce.Do(func() {})
+
+ testlist := []struct {
+ complexity []string
+ truevalues []string
+ falsevalues []string
+ }{
+ {[]string{"lower"}, []string{"abc", "abc!"}, []string{"ABC", "123", "=!$", ""}},
+ {[]string{"upper"}, []string{"ABC"}, []string{"abc", "123", "=!$", "abc!", ""}},
+ {[]string{"digit"}, []string{"123"}, []string{"abc", "ABC", "=!$", "abc!", ""}},
+ {[]string{"spec"}, []string{"=!$", "abc!"}, []string{"abc", "ABC", "123", ""}},
+ {[]string{"off"}, []string{"abc", "ABC", "123", "=!$", "abc!", ""}, nil},
+ {[]string{"lower", "spec"}, []string{"abc!"}, []string{"abc", "ABC", "123", "=!$", "abcABC123", ""}},
+ {[]string{"lower", "upper", "digit"}, []string{"abcABC123"}, []string{"abc", "ABC", "123", "=!$", "abc!", ""}},
+ }
+
+ for _, test := range testlist {
+ testComplextity(test.complexity)
+ for _, val := range test.truevalues {
+ assert.True(t, IsComplexEnough(val))
+ }
+ for _, val := range test.falsevalues {
+ assert.False(t, IsComplexEnough(val))
+ }
+ }
+
+ // Remove settings for other tests
+ testComplextity([]string{"off"})
+}
+
+func TestComplexity_Generate(t *testing.T) {
+ matchComplexityOnce.Do(func() {})
+
+ const maxCount = 50
+ const pwdLen = 50
+
+ test := func(t *testing.T, modes []string) {
+ testComplextity(modes)
+ for i := 0; i < maxCount; i++ {
+ pwd, err := Generate(pwdLen)
+ assert.NoError(t, err)
+ assert.Equal(t, pwdLen, len(pwd))
+ assert.True(t, IsComplexEnough(pwd), "Failed complexities with modes %+v for generated: %s", modes, pwd)
+ }
+ }
+
+ test(t, []string{"lower"})
+ test(t, []string{"upper"})
+ test(t, []string{"lower", "upper", "spec"})
+ test(t, []string{"off"})
+ test(t, []string{""})
+
+ // Remove settings for other tests
+ testComplextity([]string{"off"})
+}
+
+func testComplextity(values []string) {
+ // Cleanup previous values
+ validChars = ""
+ requiredChars = make([]string, 0, len(values))
+ setupComplexity(values)
+}
diff --git a/modules/setting/setting.go b/modules/setting/setting.go
index afcb943d00..d05c52fea0 100644
--- a/modules/setting/setting.go
+++ b/modules/setting/setting.go
@@ -149,7 +149,7 @@ var (
MinPasswordLength int
ImportLocalPaths bool
DisableGitHooks bool
- PasswordComplexity map[string]string
+ PasswordComplexity []string
PasswordHashAlgo string
// UI settings
@@ -781,26 +781,14 @@ func NewContext() {
InternalToken = loadInternalToken(sec)
- var dictPC = map[string]string{
- "lower": "[a-z]+",
- "upper": "[A-Z]+",
- "digit": "[0-9]+",
- "spec": `][ !"#$%&'()*+,./:;<=>?@\\^_{|}~` + "`-",
- }
- PasswordComplexity = make(map[string]string)
cfgdata := sec.Key("PASSWORD_COMPLEXITY").Strings(",")
- for _, y := range cfgdata {
- ts := strings.TrimSpace(y)
- for a := range dictPC {
- if strings.ToLower(ts) == a {
- PasswordComplexity[ts] = dictPC[ts]
- break
- }
+ PasswordComplexity = make([]string, 0, len(cfgdata))
+ for _, name := range cfgdata {
+ name := strings.ToLower(strings.Trim(name, `"`))
+ if name != "" {
+ PasswordComplexity = append(PasswordComplexity, name)
}
}
- if len(PasswordComplexity) == 0 {
- PasswordComplexity = dictPC
- }
sec = Cfg.Section("attachment")
AttachmentPath = sec.Key("PATH").MustString(path.Join(AppDataPath, "attachments"))