2015-01-23 16:25:12 -08:00
|
|
|
package lock
|
|
|
|
|
2015-01-24 22:19:22 -08:00
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"gopkg.in/authboss.v0"
|
|
|
|
"gopkg.in/authboss.v0/internal/mocks"
|
|
|
|
)
|
2015-01-23 16:25:12 -08:00
|
|
|
|
|
|
|
func TestStorage(t *testing.T) {
|
2015-02-26 23:09:37 -08:00
|
|
|
l := &Lock{}
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.NewConfig()
|
2015-02-26 23:09:37 -08:00
|
|
|
storage := l.Storage()
|
2015-02-16 13:31:26 -08:00
|
|
|
if _, ok := storage[StoreAttemptNumber]; !ok {
|
2015-01-23 16:25:12 -08:00
|
|
|
t.Error("Expected attempt number storage option.")
|
|
|
|
}
|
2015-02-16 13:31:26 -08:00
|
|
|
if _, ok := storage[StoreAttemptTime]; !ok {
|
2015-01-23 16:25:12 -08:00
|
|
|
t.Error("Expected attempt number time option.")
|
|
|
|
}
|
|
|
|
}
|
2015-01-24 22:19:22 -08:00
|
|
|
|
|
|
|
func TestBeforeAuth(t *testing.T) {
|
2015-02-26 23:09:37 -08:00
|
|
|
l := &Lock{}
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.NewConfig()
|
2015-01-24 22:19:22 -08:00
|
|
|
ctx := authboss.NewContext()
|
|
|
|
|
2015-02-26 23:09:37 -08:00
|
|
|
if interrupt, err := l.BeforeAuth(ctx); err != errUserMissing {
|
2015-02-22 12:55:09 -08:00
|
|
|
t.Error("Expected an error because of missing user:", err)
|
2015-02-22 12:43:28 -08:00
|
|
|
} else if interrupt != authboss.InterruptNone {
|
|
|
|
t.Error("Interrupt should not be set:", interrupt)
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
ctx.User = authboss.Attributes{"locked": true}
|
|
|
|
|
2015-02-26 23:09:37 -08:00
|
|
|
if interrupt, err := l.BeforeAuth(ctx); err != nil {
|
2015-02-22 12:43:28 -08:00
|
|
|
t.Error(err)
|
|
|
|
} else if interrupt != authboss.InterruptAccountLocked {
|
|
|
|
t.Error("Expected a locked interrupt:", interrupt)
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAfterAuth(t *testing.T) {
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.NewConfig()
|
2015-01-24 22:19:22 -08:00
|
|
|
lock := Lock{}
|
|
|
|
ctx := authboss.NewContext()
|
|
|
|
|
2015-02-22 12:55:09 -08:00
|
|
|
if err := lock.AfterAuth(ctx); err != errUserMissing {
|
|
|
|
t.Error("Expected an error because of missing user:", err)
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
storer := mocks.NewMockStorer()
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.Cfg.Storer = storer
|
2015-02-22 13:16:11 -08:00
|
|
|
ctx.User = authboss.Attributes{authboss.Cfg.PrimaryID: "john@john.com"}
|
2015-01-24 22:19:22 -08:00
|
|
|
|
2015-02-22 12:43:28 -08:00
|
|
|
if err := lock.AfterAuth(ctx); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
2015-03-03 22:34:37 -08:00
|
|
|
if storer.Users["john@john.com"][StoreAttemptNumber].(int64) != int64(0) {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Error("StoreAttemptNumber set incorrectly.")
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
2015-02-22 13:16:11 -08:00
|
|
|
if _, ok := storer.Users["john@john.com"][StoreAttemptTime].(time.Time); !ok {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Error("StoreAttemptTime not set.")
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAfterAuthFail_Lock(t *testing.T) {
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.NewConfig()
|
2015-01-24 22:19:22 -08:00
|
|
|
var old, current time.Time
|
|
|
|
var ok bool
|
|
|
|
|
|
|
|
ctx := authboss.NewContext()
|
|
|
|
storer := mocks.NewMockStorer()
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.Cfg.Storer = storer
|
2015-01-24 22:19:22 -08:00
|
|
|
lock := Lock{}
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.Cfg.LockWindow = 30 * time.Minute
|
|
|
|
authboss.Cfg.LockAfter = 3
|
2015-01-24 22:19:22 -08:00
|
|
|
|
2015-02-22 13:16:11 -08:00
|
|
|
email := "john@john.com"
|
|
|
|
|
|
|
|
ctx.User = map[string]interface{}{authboss.Cfg.PrimaryID: email}
|
2015-01-24 22:19:22 -08:00
|
|
|
|
|
|
|
old = time.Now().UTC().Add(-1 * time.Hour)
|
|
|
|
|
|
|
|
for i := 0; i < 3; i++ {
|
2015-02-22 13:16:11 -08:00
|
|
|
if lockedIntf, ok := storer.Users["john@john.com"][StoreLocked]; ok && lockedIntf.(bool) {
|
2015-01-24 22:19:22 -08:00
|
|
|
t.Errorf("%d: User should not be locked.", i)
|
|
|
|
}
|
|
|
|
|
2015-02-22 12:43:28 -08:00
|
|
|
if err := lock.AfterAuthFail(ctx); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
2015-03-03 22:34:37 -08:00
|
|
|
if val := storer.Users[email][StoreAttemptNumber].(int64); val != int64(i+1) {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Errorf("%d: StoreAttemptNumber set incorrectly: %v", i, val)
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
2015-02-22 13:16:11 -08:00
|
|
|
if current, ok = storer.Users[email][StoreAttemptTime].(time.Time); !ok || old.After(current) {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Error("%d: StoreAttemptTime not set correctly: %v", i, current)
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
current = old
|
|
|
|
}
|
|
|
|
|
2015-02-22 13:16:11 -08:00
|
|
|
if !storer.Users[email][StoreLocked].(bool) {
|
2015-01-24 22:19:22 -08:00
|
|
|
t.Error("User should be locked.")
|
|
|
|
}
|
2015-03-03 22:34:37 -08:00
|
|
|
if val := storer.Users[email][StoreAttemptNumber].(int64); val != int64(3) {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Error("StoreAttemptNumber set incorrectly:", val)
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
2015-02-22 13:16:11 -08:00
|
|
|
if _, ok = storer.Users[email][StoreAttemptTime].(time.Time); !ok {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Error("StoreAttemptTime not set correctly.")
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAfterAuthFail_Reset(t *testing.T) {
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.NewConfig()
|
2015-01-24 22:19:22 -08:00
|
|
|
var old, current time.Time
|
|
|
|
var ok bool
|
|
|
|
|
|
|
|
ctx := authboss.NewContext()
|
|
|
|
storer := mocks.NewMockStorer()
|
|
|
|
lock := Lock{}
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.Cfg.LockWindow = 30 * time.Minute
|
|
|
|
authboss.Cfg.Storer = storer
|
2015-01-24 22:19:22 -08:00
|
|
|
|
|
|
|
old = time.Now().UTC().Add(-time.Hour)
|
|
|
|
|
2015-02-22 13:16:11 -08:00
|
|
|
email := "john@john.com"
|
2015-01-24 22:19:22 -08:00
|
|
|
ctx.User = map[string]interface{}{
|
2015-02-22 13:16:11 -08:00
|
|
|
authboss.Cfg.PrimaryID: email,
|
2015-03-03 22:34:37 -08:00
|
|
|
StoreAttemptNumber: int64(2),
|
2015-02-22 13:16:11 -08:00
|
|
|
StoreAttemptTime: old,
|
|
|
|
StoreLocked: false,
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
lock.AfterAuthFail(ctx)
|
2015-03-03 22:34:37 -08:00
|
|
|
if val := storer.Users[email][StoreAttemptNumber].(int64); val != int64(0) {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Error("StoreAttemptNumber set incorrectly:", val)
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
2015-02-22 13:16:11 -08:00
|
|
|
if current, ok = storer.Users[email][StoreAttemptTime].(time.Time); !ok || current.Before(old) {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Error("StoreAttemptTime not set correctly.")
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
2015-02-22 13:16:11 -08:00
|
|
|
if locked := storer.Users[email][StoreLocked].(bool); locked {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Error("StoreLocked not set correctly:", locked)
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAfterAuthFail_Errors(t *testing.T) {
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.NewConfig()
|
2015-01-24 22:19:22 -08:00
|
|
|
lock := Lock{}
|
|
|
|
ctx := authboss.NewContext()
|
|
|
|
|
|
|
|
lock.AfterAuthFail(ctx)
|
2015-02-16 13:31:26 -08:00
|
|
|
if _, ok := ctx.User[StoreAttemptNumber]; ok {
|
2015-01-24 22:19:22 -08:00
|
|
|
t.Error("Expected nothing to be set, missing user.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestLock(t *testing.T) {
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.NewConfig()
|
2015-01-24 22:19:22 -08:00
|
|
|
storer := mocks.NewMockStorer()
|
2015-02-22 12:43:28 -08:00
|
|
|
authboss.Cfg.Storer = storer
|
2015-01-24 22:19:22 -08:00
|
|
|
lock := Lock{}
|
|
|
|
|
2015-02-22 13:16:11 -08:00
|
|
|
email := "john@john.com"
|
|
|
|
storer.Users[email] = map[string]interface{}{
|
|
|
|
authboss.Cfg.PrimaryID: email,
|
|
|
|
"password": "password",
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
|
|
|
|
2015-02-22 13:16:11 -08:00
|
|
|
err := lock.Lock(email)
|
2015-01-24 22:19:22 -08:00
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
|
2015-02-22 13:16:11 -08:00
|
|
|
if locked := storer.Users[email][StoreLocked].(bool); !locked {
|
2015-01-24 22:19:22 -08:00
|
|
|
t.Error("User should be locked.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUnlock(t *testing.T) {
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.NewConfig()
|
2015-01-24 22:19:22 -08:00
|
|
|
storer := mocks.NewMockStorer()
|
2015-02-22 12:43:28 -08:00
|
|
|
authboss.Cfg.Storer = storer
|
2015-01-24 22:19:22 -08:00
|
|
|
lock := Lock{}
|
2015-02-15 20:07:36 -08:00
|
|
|
authboss.Cfg.LockWindow = 1 * time.Hour
|
2015-01-24 22:19:22 -08:00
|
|
|
|
2015-02-22 13:16:11 -08:00
|
|
|
email := "john@john.com"
|
|
|
|
storer.Users[email] = map[string]interface{}{
|
|
|
|
authboss.Cfg.PrimaryID: email,
|
|
|
|
"password": "password",
|
|
|
|
"locked": true,
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
|
|
|
|
2015-02-22 13:16:11 -08:00
|
|
|
err := lock.Unlock(email)
|
2015-01-24 22:19:22 -08:00
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
|
2015-02-22 13:16:11 -08:00
|
|
|
attemptTime := storer.Users[email][StoreAttemptTime].(time.Time)
|
2015-02-15 20:07:36 -08:00
|
|
|
if attemptTime.After(time.Now().UTC().Add(-authboss.Cfg.LockWindow)) {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Error("StoreLocked not set correctly:", attemptTime)
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
2015-03-03 22:34:37 -08:00
|
|
|
if number := storer.Users[email][StoreAttemptNumber].(int64); number != int64(0) {
|
2015-02-16 13:31:26 -08:00
|
|
|
t.Error("StoreLocked not set correctly:", number)
|
2015-01-24 22:19:22 -08:00
|
|
|
}
|
2015-02-22 13:16:11 -08:00
|
|
|
if locked := storer.Users[email][StoreLocked].(bool); locked {
|
2015-01-24 22:19:22 -08:00
|
|
|
t.Error("User should not be locked.")
|
|
|
|
}
|
|
|
|
}
|