2015-01-17 02:42:42 -08:00
|
|
|
package authboss
|
|
|
|
|
2015-01-24 16:07:41 -08:00
|
|
|
import (
|
2017-02-24 16:45:47 -08:00
|
|
|
"bytes"
|
2017-02-20 16:08:19 -08:00
|
|
|
"context"
|
2017-02-23 16:13:25 -08:00
|
|
|
"encoding/json"
|
2017-03-05 10:01:46 -08:00
|
|
|
"fmt"
|
2015-01-24 16:07:41 -08:00
|
|
|
"net/http"
|
2015-08-02 13:00:16 -07:00
|
|
|
"net/url"
|
|
|
|
"strings"
|
2015-01-24 16:07:41 -08:00
|
|
|
)
|
|
|
|
|
2015-01-17 02:42:42 -08:00
|
|
|
type mockUser struct {
|
|
|
|
Email string
|
|
|
|
Password string
|
|
|
|
}
|
|
|
|
|
2018-03-09 13:11:08 -08:00
|
|
|
func newMockServerStorer() *mockServerStorer {
|
|
|
|
return &mockServerStorer{
|
|
|
|
Users: make(map[string]*mockUser),
|
|
|
|
Tokens: make(map[string][]string),
|
|
|
|
}
|
|
|
|
}
|
2015-01-17 02:42:42 -08:00
|
|
|
|
2018-03-09 13:11:08 -08:00
|
|
|
type mockServerStorer struct {
|
|
|
|
Users map[string]*mockUser
|
|
|
|
Tokens map[string][]string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockServerStorer) Load(ctx context.Context, key string) (User, error) {
|
|
|
|
u, ok := m.Users[key]
|
2017-02-21 15:04:30 -08:00
|
|
|
if !ok {
|
|
|
|
return nil, ErrUserNotFound
|
|
|
|
}
|
|
|
|
|
2018-02-01 11:51:43 -08:00
|
|
|
return u, nil
|
2017-02-21 15:04:30 -08:00
|
|
|
}
|
|
|
|
|
2018-03-09 13:11:08 -08:00
|
|
|
func (m *mockServerStorer) Save(ctx context.Context, user User) error {
|
|
|
|
u := user.(*mockUser)
|
|
|
|
m.Users[u.Email] = u
|
2017-02-21 15:04:30 -08:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-03-09 13:11:08 -08:00
|
|
|
func (m *mockServerStorer) AddRememberToken(pid, token string) error {
|
|
|
|
m.Tokens[pid] = append(m.Tokens[pid], token)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockServerStorer) DelRememberTokens(pid string) error {
|
|
|
|
delete(m.Tokens, pid)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockServerStorer) UseRememberToken(pid, token string) error {
|
|
|
|
arr, ok := m.Tokens[pid]
|
|
|
|
if !ok {
|
|
|
|
return ErrTokenNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, tok := range arr {
|
|
|
|
if tok == token {
|
|
|
|
arr[i] = arr[len(arr)-1]
|
|
|
|
m.Tokens[pid] = arr[:len(arr)-2]
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ErrTokenNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockUser) PutPID(email string) {
|
2017-02-20 16:08:19 -08:00
|
|
|
m.Email = email
|
2015-01-17 02:42:42 -08:00
|
|
|
}
|
|
|
|
|
2018-03-09 13:11:08 -08:00
|
|
|
func (m *mockUser) PutPassword(password string) {
|
2017-02-20 16:08:19 -08:00
|
|
|
m.Password = password
|
2015-03-13 16:23:43 -07:00
|
|
|
}
|
|
|
|
|
2018-03-09 13:11:08 -08:00
|
|
|
func (m *mockUser) GetPID() (email string) {
|
2018-02-14 15:16:44 -08:00
|
|
|
return m.Email
|
2017-02-21 15:04:30 -08:00
|
|
|
}
|
|
|
|
|
2018-03-09 13:11:08 -08:00
|
|
|
func (m *mockUser) GetPassword() (password string) {
|
2018-02-14 15:16:44 -08:00
|
|
|
return m.Password
|
2015-03-13 16:23:43 -07:00
|
|
|
}
|
|
|
|
|
2017-02-24 16:45:47 -08:00
|
|
|
type mockClientStateReadWriter struct {
|
|
|
|
state mockClientState
|
2017-02-23 16:13:25 -08:00
|
|
|
}
|
2015-01-17 02:42:42 -08:00
|
|
|
|
2017-02-24 16:45:47 -08:00
|
|
|
type mockClientState map[string]string
|
|
|
|
|
|
|
|
func newMockClientStateRW(keyValue ...string) mockClientStateReadWriter {
|
|
|
|
state := mockClientState{}
|
|
|
|
for i := 0; i < len(keyValue); i += 2 {
|
|
|
|
key, value := keyValue[i], keyValue[i+1]
|
|
|
|
state[key] = value
|
2017-02-23 16:13:25 -08:00
|
|
|
}
|
2017-02-24 16:45:47 -08:00
|
|
|
|
|
|
|
return mockClientStateReadWriter{state}
|
2017-02-23 16:13:25 -08:00
|
|
|
}
|
|
|
|
|
2018-03-07 16:21:37 -08:00
|
|
|
func (m mockClientStateReadWriter) ReadState(r *http.Request) (ClientState, error) {
|
2017-02-24 16:45:47 -08:00
|
|
|
return m.state, nil
|
2015-01-17 02:42:42 -08:00
|
|
|
}
|
2017-02-24 16:45:47 -08:00
|
|
|
|
|
|
|
func (m mockClientStateReadWriter) WriteState(w http.ResponseWriter, cs ClientState, evs []ClientStateEvent) error {
|
|
|
|
var state mockClientState
|
|
|
|
|
|
|
|
if cs != nil {
|
|
|
|
state = cs.(mockClientState)
|
|
|
|
} else {
|
|
|
|
state = mockClientState{}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, ev := range evs {
|
|
|
|
switch ev.Kind {
|
|
|
|
case ClientStateEventPut:
|
|
|
|
state[ev.Key] = ev.Value
|
|
|
|
case ClientStateEventDel:
|
|
|
|
delete(state, ev.Key)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err := json.Marshal(state)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2015-02-20 22:02:55 -08:00
|
|
|
}
|
2017-02-24 16:45:47 -08:00
|
|
|
|
|
|
|
w.Header().Set("test_session", string(b))
|
|
|
|
return nil
|
2015-02-20 22:02:55 -08:00
|
|
|
}
|
2015-01-24 16:07:41 -08:00
|
|
|
|
2017-02-24 16:45:47 -08:00
|
|
|
func (m mockClientState) Get(key string) (string, bool) {
|
|
|
|
val, ok := m[key]
|
|
|
|
return val, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
func newMockRequest(postKeyValues ...string) *http.Request {
|
2015-08-02 13:00:16 -07:00
|
|
|
urlValues := make(url.Values)
|
2015-01-24 16:07:41 -08:00
|
|
|
for i := 0; i < len(postKeyValues); i += 2 {
|
2015-08-02 13:00:16 -07:00
|
|
|
urlValues.Set(postKeyValues[i], postKeyValues[i+1])
|
2015-01-24 16:07:41 -08:00
|
|
|
}
|
|
|
|
|
2015-08-02 13:00:16 -07:00
|
|
|
req, err := http.NewRequest("POST", "http://localhost", strings.NewReader(urlValues.Encode()))
|
2015-01-24 16:07:41 -08:00
|
|
|
if err != nil {
|
|
|
|
panic(err.Error())
|
|
|
|
}
|
|
|
|
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
|
|
|
|
2015-08-02 13:00:16 -07:00
|
|
|
return req
|
2015-01-24 16:07:41 -08:00
|
|
|
}
|
|
|
|
|
2017-02-24 16:45:47 -08:00
|
|
|
func newMockAPIRequest(postKeyValues ...string) *http.Request {
|
|
|
|
kv := map[string]string{}
|
|
|
|
for i := 0; i < len(postKeyValues); i += 2 {
|
|
|
|
key, value := postKeyValues[i], postKeyValues[i+1]
|
|
|
|
kv[key] = value
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err := json.Marshal(kv)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
req, err := http.NewRequest("POST", "http://localhost", bytes.NewReader(b))
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
|
|
|
|
return req
|
|
|
|
}
|
|
|
|
|
2017-03-05 10:01:46 -08:00
|
|
|
type mockRenderer struct {
|
|
|
|
expectName string
|
|
|
|
}
|
2017-02-23 16:13:25 -08:00
|
|
|
|
2018-02-01 15:42:48 -08:00
|
|
|
func (m mockRenderer) Load(names ...string) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-02-23 16:13:25 -08:00
|
|
|
func (m mockRenderer) Render(ctx context.Context, name string, data HTMLData) ([]byte, string, error) {
|
2017-03-05 10:01:46 -08:00
|
|
|
if len(m.expectName) != 0 && m.expectName != name {
|
|
|
|
panic(fmt.Sprintf("want template name: %s, but got: %s", m.expectName, name))
|
2017-02-23 16:13:25 -08:00
|
|
|
}
|
|
|
|
|
2017-03-05 10:01:46 -08:00
|
|
|
b, err := json.Marshal(data)
|
|
|
|
return b, "application/json", err
|
|
|
|
}
|
|
|
|
|
|
|
|
type mockEmailRenderer struct{}
|
|
|
|
|
|
|
|
func (m mockEmailRenderer) Render(ctx context.Context, name string, data HTMLData) ([]byte, string, error) {
|
|
|
|
switch name {
|
|
|
|
case "text":
|
|
|
|
return []byte("a development text e-mail template"), "text/plain", nil
|
|
|
|
case "html":
|
|
|
|
return []byte("a development html e-mail template"), "text/html", nil
|
|
|
|
default:
|
|
|
|
panic("shouldn't get here")
|
|
|
|
}
|
2017-02-23 16:13:25 -08:00
|
|
|
}
|
2018-04-30 18:21:28 -07:00
|
|
|
|
|
|
|
type mockLogger struct{}
|
|
|
|
|
|
|
|
func (m mockLogger) Info(s string) {}
|
|
|
|
func (m mockLogger) Error(s string) {}
|