aboutsummaryrefslogtreecommitdiff
path: root/app/models/user.go
blob: eb0ccbfacd54d0a5513cd54912a7f2aecdfc42a7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
package models

import (
	"context"
	"time"

	"code.crute.us/mcrute/golib/db/mongodb"
	"go.mongodb.org/mongo-driver/bson"
	"go.mongodb.org/mongo-driver/bson/primitive"
	"golang.org/x/oauth2"
)

const userCol = "users"

type UserStore interface {
	List(context.Context) ([]*User, error)
	Get(context.Context, string) (*User, error) // Error on not found
	Put(context.Context, *User) error
	Delete(context.Context, *User) error
}

type User struct {
	Username   string                   `bson:"_id" json:"username"`
	IsAdmin    bool                     `json:"is_admin"`
	IsService  bool                     `json:"is_service"`
	Keys       map[string]*SessionKey   `json:"keys,omitempty"`        // kid  -> key
	AuthTokens map[string]*oauth2.Token `json:"auth_tokens,omitempty"` // kind -> token
	Deleted    *time.Time               `json:"deleted,omitempty"`
}

// GCKeys garbage collects keys that are no longer valid
func (u *User) GCKeys() {
	for k, v := range u.Keys {
		if v.IsGarbage() {
			delete(u.Keys, k)
		}
	}
}

// GetKey returns a key for a key ID. It will only return valid keys.
func (u *User) GetKey(kid string) *SessionKey {
	if u.Keys != nil {
		if k := u.Keys[kid]; k != nil && k.IsValid() {
			return k
		}
	}
	return nil
}

func (u *User) AddKey(k *SessionKey) {
	if u.Keys == nil {
		u.Keys = map[string]*SessionKey{}
	}
	u.Keys[k.KeyId] = k
}

func (u *User) AddToken(name string, t *oauth2.Token) {
	if u.AuthTokens == nil {
		u.AuthTokens = map[string]*oauth2.Token{}
	}
	u.AuthTokens[name] = t
}

type MongoDbUserStore struct {
	Db *mongodb.Mongo

	// ReturnDeleted will allow all methods to return deleted items. By default
	// items where the Deleted field is set will not be returned. This should
	// be the common cast for most code using this store but in some Admin
	// use-cases it would be useful to show deleted accounts.
	ReturnDeleted bool
}

func (s *MongoDbUserStore) List(ctx context.Context) ([]*User, error) {
	var out []*User

	filter := bson.M{}
	if !s.ReturnDeleted {
		filter["deleted"] = primitive.Null{}
	}

	if err := s.Db.FindAllByFilter(ctx, userCol, filter, &out); err != nil {
		return nil, err
	}

	return out, nil
}

func (s *MongoDbUserStore) Get(ctx context.Context, username string) (*User, error) {
	var u User

	filter := bson.M{"_id": username}
	if !s.ReturnDeleted {
		filter["deleted"] = primitive.Null{}
	}

	if err := s.Db.FindOneByFilter(ctx, userCol, filter, &u); err != nil {
		return nil, err
	}

	return &u, nil
}

func (s *MongoDbUserStore) Put(ctx context.Context, u *User) error {
	if err := s.Db.ReplaceOneById(ctx, userCol, u.Username, u); err != nil {
		return err
	}
	return nil
}

func (s *MongoDbUserStore) Delete(ctx context.Context, u *User) error {
	u, err := s.Get(ctx, u.Username)
	if err != nil {
		return err
	}

	now := time.Now()
	u.Deleted = &now

	return s.Put(ctx, u)
}

var _ UserStore = (*MongoDbUserStore)(nil)