mirror of
https://github.com/jech/galene.git
synced 2024-11-23 00:55:58 +01:00
295 lines
6.1 KiB
Go
295 lines
6.1 KiB
Go
|
package token
|
||
|
|
||
|
import (
|
||
|
"encoding/json"
|
||
|
"io"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
"reflect"
|
||
|
"sort"
|
||
|
"testing"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
func equal(a, b *Stateful) bool {
|
||
|
if a.Token != b.Token || a.Group != b.Group ||
|
||
|
!reflect.DeepEqual(a.Username, b.Username) ||
|
||
|
!reflect.DeepEqual(a.Permissions, b.Permissions) {
|
||
|
return false
|
||
|
}
|
||
|
if a.Expires != nil && b.Expires != nil {
|
||
|
return (*a.Expires).Equal(*b.Expires)
|
||
|
}
|
||
|
if (a.Expires != nil) != (b.Expires != nil) {
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
if a.NotBefore != nil && b.NotBefore != nil {
|
||
|
return (*a.NotBefore).Equal(*b.NotBefore)
|
||
|
}
|
||
|
return (a.NotBefore != nil) == (b.NotBefore != nil)
|
||
|
}
|
||
|
|
||
|
func TestStatefulCheck(t *testing.T) {
|
||
|
now := time.Now()
|
||
|
past := now.Add(-time.Hour)
|
||
|
nearFuture := now.Add(time.Hour / 2)
|
||
|
future := now.Add(time.Hour)
|
||
|
user := "user"
|
||
|
user2 := "user2"
|
||
|
token1 := &Stateful{
|
||
|
Token: "token",
|
||
|
Group: "group",
|
||
|
Username: &user,
|
||
|
Permissions: []string{"present"},
|
||
|
Expires: &future,
|
||
|
}
|
||
|
token2 := &Stateful{
|
||
|
Token: "token",
|
||
|
Group: "group",
|
||
|
Permissions: []string{"present"},
|
||
|
Expires: &future,
|
||
|
}
|
||
|
token3 := &Stateful{
|
||
|
Token: "token",
|
||
|
Group: "group",
|
||
|
Username: &user,
|
||
|
Permissions: []string{"present"},
|
||
|
Expires: &past,
|
||
|
}
|
||
|
token4 := &Stateful{
|
||
|
Token: "token",
|
||
|
Group: "group",
|
||
|
Username: &user,
|
||
|
Permissions: []string{"present"},
|
||
|
Expires: &future,
|
||
|
NotBefore: &nearFuture,
|
||
|
}
|
||
|
|
||
|
success := []struct {
|
||
|
token *Stateful
|
||
|
group string
|
||
|
username *string
|
||
|
expUsername string
|
||
|
expPermissions []string
|
||
|
}{
|
||
|
{
|
||
|
token: token1,
|
||
|
group: "group",
|
||
|
username: &user,
|
||
|
expUsername: user,
|
||
|
expPermissions: []string{"present"},
|
||
|
},
|
||
|
{
|
||
|
token: token1,
|
||
|
group: "group",
|
||
|
username: &user2,
|
||
|
expUsername: user,
|
||
|
expPermissions: []string{"present"},
|
||
|
},
|
||
|
{
|
||
|
token: token1,
|
||
|
group: "group",
|
||
|
expUsername: user,
|
||
|
expPermissions: []string{"present"},
|
||
|
},
|
||
|
{
|
||
|
token: token2,
|
||
|
group: "group",
|
||
|
username: &user,
|
||
|
expUsername: "",
|
||
|
expPermissions: []string{"present"},
|
||
|
},
|
||
|
}
|
||
|
|
||
|
for i, s := range success {
|
||
|
u, p, err := s.token.Check("", s.group, s.username)
|
||
|
if err != nil || u != s.expUsername ||
|
||
|
!reflect.DeepEqual(p, s.expPermissions) {
|
||
|
t.Errorf("Check %v failed: %v %v %v -> %v %v %v",
|
||
|
i, s.token, s.group, s.username,
|
||
|
u, p, err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
failure := []struct {
|
||
|
token *Stateful
|
||
|
group string
|
||
|
username *string
|
||
|
}{
|
||
|
{
|
||
|
token: token1,
|
||
|
group: "group2",
|
||
|
username: &user,
|
||
|
},
|
||
|
{
|
||
|
token: token3,
|
||
|
group: "group",
|
||
|
username: &user,
|
||
|
},
|
||
|
{
|
||
|
token: token4,
|
||
|
group: "group",
|
||
|
username: &user,
|
||
|
},
|
||
|
}
|
||
|
|
||
|
for i, s := range failure {
|
||
|
u, p, err := s.token.Check("", s.group, s.username)
|
||
|
if err == nil {
|
||
|
t.Errorf("Check %v succeded: %v %v %v -> %v %v %v",
|
||
|
i, s.token, s.group, s.username,
|
||
|
u, p, err)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func readTokenFile(filename string) []*Stateful {
|
||
|
f, err := os.Open(filename)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
defer f.Close()
|
||
|
|
||
|
a := make([]*Stateful, 0)
|
||
|
decoder := json.NewDecoder(f)
|
||
|
for {
|
||
|
var t Stateful
|
||
|
err := decoder.Decode(&t)
|
||
|
if err == io.EOF {
|
||
|
break
|
||
|
} else if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
a = append(a, &t)
|
||
|
}
|
||
|
return a
|
||
|
}
|
||
|
|
||
|
func expectTokenArray(t *testing.T, a, b []*Stateful) {
|
||
|
if len(a) != len(b) {
|
||
|
t.Errorf("Bad length: %v != %v", len(a), len(b))
|
||
|
}
|
||
|
aa := append([]*Stateful(nil), a...)
|
||
|
sort.Slice(aa, func(i, j int) bool {
|
||
|
return aa[i].Token < aa[j].Token
|
||
|
})
|
||
|
bb := append([]*Stateful(nil), b...)
|
||
|
sort.Slice(bb, func(i, j int) bool {
|
||
|
return bb[i].Token < bb[j].Token
|
||
|
})
|
||
|
|
||
|
if len(aa) != len(bb) {
|
||
|
t.Errorf("Not equal: %v != %v", len(aa), len(bb))
|
||
|
}
|
||
|
|
||
|
for i, ta := range aa {
|
||
|
tb := bb[i]
|
||
|
if !equal(ta, tb) {
|
||
|
t.Errorf("Not equal: %v != %v", ta, tb)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func expectTokens(t *testing.T, tokens map[string]*Stateful, value []*Stateful) {
|
||
|
a := make([]*Stateful, 0, len(tokens))
|
||
|
for tok, token := range tokens {
|
||
|
if tok != token.Token {
|
||
|
t.Errorf("Inconsistent token: %v != %v",
|
||
|
tok, token.Token)
|
||
|
}
|
||
|
a = append(a, token)
|
||
|
}
|
||
|
expectTokenArray(t, a, value)
|
||
|
}
|
||
|
|
||
|
func expectTokenFile(t *testing.T, filename string, value []*Stateful) {
|
||
|
a := readTokenFile(filename)
|
||
|
expectTokenArray(t, a, value)
|
||
|
}
|
||
|
|
||
|
func TestTokenStorage(t *testing.T) {
|
||
|
d := t.TempDir()
|
||
|
s := state{
|
||
|
filename: filepath.Join(d, "test.jsonl"),
|
||
|
}
|
||
|
now := time.Now()
|
||
|
past := now.Add(-time.Hour)
|
||
|
nearFuture := now.Add(time.Hour / 2)
|
||
|
future := now.Add(time.Hour)
|
||
|
user1 := "user1"
|
||
|
user2 := "user2"
|
||
|
user3 := "user3"
|
||
|
tokens := []*Stateful{
|
||
|
&Stateful{
|
||
|
Token: "tok1",
|
||
|
Group: "test",
|
||
|
Username: &user1,
|
||
|
Permissions: []string{"present"},
|
||
|
Expires: &future,
|
||
|
},
|
||
|
&Stateful{
|
||
|
Token: "tok2",
|
||
|
Group: "test",
|
||
|
Username: &user2,
|
||
|
Permissions: []string{"present", "record"},
|
||
|
Expires: &nearFuture,
|
||
|
NotBefore: &past,
|
||
|
},
|
||
|
&Stateful{
|
||
|
Token: "tok3",
|
||
|
Group: "test",
|
||
|
Username: &user3,
|
||
|
Permissions: []string{"present"},
|
||
|
Expires: &nearFuture,
|
||
|
},
|
||
|
}
|
||
|
for i, token := range tokens {
|
||
|
new, err := s.Add(token)
|
||
|
if err != nil {
|
||
|
t.Errorf("Add: %v", err)
|
||
|
}
|
||
|
if !equal(new, token) {
|
||
|
t.Errorf("Add: got %v, expected %v", new, token)
|
||
|
}
|
||
|
expectTokens(t, s.tokens, tokens[:i+1])
|
||
|
expectTokenFile(t, s.filename, tokens[:i+1])
|
||
|
}
|
||
|
|
||
|
s.modTime = time.Time{}
|
||
|
err := s.load()
|
||
|
if err != nil {
|
||
|
t.Errorf("Load: %v", err)
|
||
|
}
|
||
|
expectTokens(t, s.tokens, tokens)
|
||
|
|
||
|
_, err = s.Edit("test2", tokens[1].Token, now.Add(time.Hour))
|
||
|
if err == nil {
|
||
|
t.Errorf("Edit succeeded with wrong group")
|
||
|
}
|
||
|
new, err := s.Edit("test", tokens[1].Token, now.Add(time.Hour))
|
||
|
if err != nil {
|
||
|
t.Errorf("Edit: %v", err)
|
||
|
}
|
||
|
tokens[1].Expires = &future
|
||
|
if !equal(new, tokens[1]) {
|
||
|
t.Errorf("Edit: got %v, expected %v", tokens[1], new)
|
||
|
}
|
||
|
expectTokens(t, s.tokens, tokens)
|
||
|
expectTokenFile(t, s.filename, tokens)
|
||
|
|
||
|
for t := range s.tokens {
|
||
|
delete(s.tokens, t)
|
||
|
}
|
||
|
|
||
|
err = s.rewrite()
|
||
|
if err != nil {
|
||
|
t.Errorf("rewrite(empty): %v", err)
|
||
|
}
|
||
|
|
||
|
_, err = os.Stat(s.filename)
|
||
|
if !os.IsNotExist(err) {
|
||
|
t.Errorf("existence check: %v", err)
|
||
|
}
|
||
|
}
|