| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535 |
- package auth
- import (
- "context"
- "encoding/json"
- "testing"
- "time"
- "github.com/grafana/grafana/pkg/components/simplejson"
- "github.com/grafana/grafana/pkg/setting"
- "github.com/grafana/grafana/pkg/infra/log"
- "github.com/grafana/grafana/pkg/models"
- "github.com/grafana/grafana/pkg/services/sqlstore"
- . "github.com/smartystreets/goconvey/convey"
- )
- func TestUserAuthToken(t *testing.T) {
- Convey("Test user auth token", t, func() {
- ctx := createTestContext(t)
- userAuthTokenService := ctx.tokenService
- userID := int64(10)
- t := time.Date(2018, 12, 13, 13, 45, 0, 0, time.UTC)
- getTime = func() time.Time {
- return t
- }
- Convey("When creating token", func() {
- userToken, err := userAuthTokenService.CreateToken(context.Background(), userID, "192.168.10.11:1234", "some user agent")
- So(err, ShouldBeNil)
- So(userToken, ShouldNotBeNil)
- So(userToken.AuthTokenSeen, ShouldBeFalse)
- Convey("Can count active tokens", func() {
- count, err := userAuthTokenService.ActiveTokenCount(context.Background())
- So(err, ShouldBeNil)
- So(count, ShouldEqual, 1)
- })
- Convey("When lookup unhashed token should return user auth token", func() {
- userToken, err := userAuthTokenService.LookupToken(context.Background(), userToken.UnhashedToken)
- So(err, ShouldBeNil)
- So(userToken, ShouldNotBeNil)
- So(userToken.UserId, ShouldEqual, userID)
- So(userToken.AuthTokenSeen, ShouldBeTrue)
- storedAuthToken, err := ctx.getAuthTokenByID(userToken.Id)
- So(err, ShouldBeNil)
- So(storedAuthToken, ShouldNotBeNil)
- So(storedAuthToken.AuthTokenSeen, ShouldBeTrue)
- })
- Convey("When lookup hashed token should return user auth token not found error", func() {
- userToken, err := userAuthTokenService.LookupToken(context.Background(), userToken.AuthToken)
- So(err, ShouldEqual, models.ErrUserTokenNotFound)
- So(userToken, ShouldBeNil)
- })
- Convey("revoking existing token should delete token", func() {
- err = userAuthTokenService.RevokeToken(context.Background(), userToken)
- So(err, ShouldBeNil)
- model, err := ctx.getAuthTokenByID(userToken.Id)
- So(err, ShouldBeNil)
- So(model, ShouldBeNil)
- })
- Convey("revoking nil token should return error", func() {
- err = userAuthTokenService.RevokeToken(context.Background(), nil)
- So(err, ShouldEqual, models.ErrUserTokenNotFound)
- })
- Convey("revoking non-existing token should return error", func() {
- userToken.Id = 1000
- err = userAuthTokenService.RevokeToken(context.Background(), userToken)
- So(err, ShouldEqual, models.ErrUserTokenNotFound)
- })
- Convey("When creating an additional token", func() {
- userToken2, err := userAuthTokenService.CreateToken(context.Background(), userID, "192.168.10.11:1234", "some user agent")
- So(err, ShouldBeNil)
- So(userToken2, ShouldNotBeNil)
- Convey("Can get first user token", func() {
- token, err := userAuthTokenService.GetUserToken(context.Background(), userID, userToken.Id)
- So(err, ShouldBeNil)
- So(token, ShouldNotBeNil)
- So(token.Id, ShouldEqual, userToken.Id)
- })
- Convey("Can get second user token", func() {
- token, err := userAuthTokenService.GetUserToken(context.Background(), userID, userToken2.Id)
- So(err, ShouldBeNil)
- So(token, ShouldNotBeNil)
- So(token.Id, ShouldEqual, userToken2.Id)
- })
- Convey("Can get user tokens", func() {
- tokens, err := userAuthTokenService.GetUserTokens(context.Background(), userID)
- So(err, ShouldBeNil)
- So(tokens, ShouldHaveLength, 2)
- So(tokens[0].Id, ShouldEqual, userToken.Id)
- So(tokens[1].Id, ShouldEqual, userToken2.Id)
- })
- Convey("Can revoke all user tokens", func() {
- err := userAuthTokenService.RevokeAllUserTokens(context.Background(), userID)
- So(err, ShouldBeNil)
- model, err := ctx.getAuthTokenByID(userToken.Id)
- So(err, ShouldBeNil)
- So(model, ShouldBeNil)
- model2, err := ctx.getAuthTokenByID(userToken2.Id)
- So(err, ShouldBeNil)
- So(model2, ShouldBeNil)
- })
- })
- })
- Convey("expires correctly", func() {
- userToken, err := userAuthTokenService.CreateToken(context.Background(), userID, "192.168.10.11:1234", "some user agent")
- So(err, ShouldBeNil)
- userToken, err = userAuthTokenService.LookupToken(context.Background(), userToken.UnhashedToken)
- So(err, ShouldBeNil)
- getTime = func() time.Time {
- return t.Add(time.Hour)
- }
- rotated, err := userAuthTokenService.TryRotateToken(context.Background(), userToken, "192.168.10.11:1234", "some user agent")
- So(err, ShouldBeNil)
- So(rotated, ShouldBeTrue)
- userToken, err = userAuthTokenService.LookupToken(context.Background(), userToken.UnhashedToken)
- So(err, ShouldBeNil)
- stillGood, err := userAuthTokenService.LookupToken(context.Background(), userToken.UnhashedToken)
- So(err, ShouldBeNil)
- So(stillGood, ShouldNotBeNil)
- model, err := ctx.getAuthTokenByID(userToken.Id)
- So(err, ShouldBeNil)
- Convey("when rotated_at is 6:59:59 ago should find token", func() {
- getTime = func() time.Time {
- return time.Unix(model.RotatedAt, 0).Add(24 * 7 * time.Hour).Add(-time.Second)
- }
- stillGood, err = userAuthTokenService.LookupToken(context.Background(), stillGood.UnhashedToken)
- So(err, ShouldBeNil)
- So(stillGood, ShouldNotBeNil)
- })
- Convey("when rotated_at is 7:00:00 ago should not find token", func() {
- getTime = func() time.Time {
- return time.Unix(model.RotatedAt, 0).Add(24 * 7 * time.Hour)
- }
- notGood, err := userAuthTokenService.LookupToken(context.Background(), userToken.UnhashedToken)
- So(err, ShouldEqual, models.ErrUserTokenNotFound)
- So(notGood, ShouldBeNil)
- Convey("should not find active token when expired", func() {
- count, err := userAuthTokenService.ActiveTokenCount(context.Background())
- So(err, ShouldBeNil)
- So(count, ShouldEqual, 0)
- })
- })
- Convey("when rotated_at is 5 days ago and created_at is 29 days and 23:59:59 ago should not find token", func() {
- updated, err := ctx.updateRotatedAt(model.Id, time.Unix(model.CreatedAt, 0).Add(24*25*time.Hour).Unix())
- So(err, ShouldBeNil)
- So(updated, ShouldBeTrue)
- getTime = func() time.Time {
- return time.Unix(model.CreatedAt, 0).Add(24 * 30 * time.Hour).Add(-time.Second)
- }
- stillGood, err = userAuthTokenService.LookupToken(context.Background(), stillGood.UnhashedToken)
- So(err, ShouldBeNil)
- So(stillGood, ShouldNotBeNil)
- })
- Convey("when rotated_at is 5 days ago and created_at is 30 days ago should not find token", func() {
- updated, err := ctx.updateRotatedAt(model.Id, time.Unix(model.CreatedAt, 0).Add(24*25*time.Hour).Unix())
- So(err, ShouldBeNil)
- So(updated, ShouldBeTrue)
- getTime = func() time.Time {
- return time.Unix(model.CreatedAt, 0).Add(24 * 30 * time.Hour)
- }
- notGood, err := userAuthTokenService.LookupToken(context.Background(), userToken.UnhashedToken)
- So(err, ShouldEqual, models.ErrUserTokenNotFound)
- So(notGood, ShouldBeNil)
- })
- })
- Convey("can properly rotate tokens", func() {
- userToken, err := userAuthTokenService.CreateToken(context.Background(), userID, "192.168.10.11:1234", "some user agent")
- So(err, ShouldBeNil)
- prevToken := userToken.AuthToken
- unhashedPrev := userToken.UnhashedToken
- rotated, err := userAuthTokenService.TryRotateToken(context.Background(), userToken, "192.168.10.12:1234", "a new user agent")
- So(err, ShouldBeNil)
- So(rotated, ShouldBeFalse)
- updated, err := ctx.markAuthTokenAsSeen(userToken.Id)
- So(err, ShouldBeNil)
- So(updated, ShouldBeTrue)
- model, err := ctx.getAuthTokenByID(userToken.Id)
- So(err, ShouldBeNil)
- var tok models.UserToken
- err = model.toUserToken(&tok)
- So(err, ShouldBeNil)
- getTime = func() time.Time {
- return t.Add(time.Hour)
- }
- rotated, err = userAuthTokenService.TryRotateToken(context.Background(), &tok, "192.168.10.12:1234", "a new user agent")
- So(err, ShouldBeNil)
- So(rotated, ShouldBeTrue)
- unhashedToken := tok.UnhashedToken
- model, err = ctx.getAuthTokenByID(tok.Id)
- So(err, ShouldBeNil)
- model.UnhashedToken = unhashedToken
- So(model.RotatedAt, ShouldEqual, getTime().Unix())
- So(model.ClientIp, ShouldEqual, "192.168.10.12")
- So(model.UserAgent, ShouldEqual, "a new user agent")
- So(model.AuthTokenSeen, ShouldBeFalse)
- So(model.SeenAt, ShouldEqual, 0)
- So(model.PrevAuthToken, ShouldEqual, prevToken)
- // ability to auth using an old token
- lookedUpUserToken, err := userAuthTokenService.LookupToken(context.Background(), model.UnhashedToken)
- So(err, ShouldBeNil)
- So(lookedUpUserToken, ShouldNotBeNil)
- So(lookedUpUserToken.AuthTokenSeen, ShouldBeTrue)
- So(lookedUpUserToken.SeenAt, ShouldEqual, getTime().Unix())
- lookedUpUserToken, err = userAuthTokenService.LookupToken(context.Background(), unhashedPrev)
- So(err, ShouldBeNil)
- So(lookedUpUserToken, ShouldNotBeNil)
- So(lookedUpUserToken.Id, ShouldEqual, model.Id)
- So(lookedUpUserToken.AuthTokenSeen, ShouldBeTrue)
- getTime = func() time.Time {
- return t.Add(time.Hour + (2 * time.Minute))
- }
- lookedUpUserToken, err = userAuthTokenService.LookupToken(context.Background(), unhashedPrev)
- So(err, ShouldBeNil)
- So(lookedUpUserToken, ShouldNotBeNil)
- So(lookedUpUserToken.AuthTokenSeen, ShouldBeTrue)
- lookedUpModel, err := ctx.getAuthTokenByID(lookedUpUserToken.Id)
- So(err, ShouldBeNil)
- So(lookedUpModel, ShouldNotBeNil)
- So(lookedUpModel.AuthTokenSeen, ShouldBeFalse)
- rotated, err = userAuthTokenService.TryRotateToken(context.Background(), userToken, "192.168.10.12:1234", "a new user agent")
- So(err, ShouldBeNil)
- So(rotated, ShouldBeTrue)
- model, err = ctx.getAuthTokenByID(userToken.Id)
- So(err, ShouldBeNil)
- So(model, ShouldNotBeNil)
- So(model.SeenAt, ShouldEqual, 0)
- })
- Convey("keeps prev token valid for 1 minute after it is confirmed", func() {
- userToken, err := userAuthTokenService.CreateToken(context.Background(), userID, "192.168.10.11:1234", "some user agent")
- So(err, ShouldBeNil)
- So(userToken, ShouldNotBeNil)
- lookedUpUserToken, err := userAuthTokenService.LookupToken(context.Background(), userToken.UnhashedToken)
- So(err, ShouldBeNil)
- So(lookedUpUserToken, ShouldNotBeNil)
- getTime = func() time.Time {
- return t.Add(10 * time.Minute)
- }
- prevToken := userToken.UnhashedToken
- rotated, err := userAuthTokenService.TryRotateToken(context.Background(), userToken, "1.1.1.1", "firefox")
- So(err, ShouldBeNil)
- So(rotated, ShouldBeTrue)
- getTime = func() time.Time {
- return t.Add(20 * time.Minute)
- }
- currentUserToken, err := userAuthTokenService.LookupToken(context.Background(), userToken.UnhashedToken)
- So(err, ShouldBeNil)
- So(currentUserToken, ShouldNotBeNil)
- prevUserToken, err := userAuthTokenService.LookupToken(context.Background(), prevToken)
- So(err, ShouldBeNil)
- So(prevUserToken, ShouldNotBeNil)
- })
- Convey("will not mark token unseen when prev and current are the same", func() {
- userToken, err := userAuthTokenService.CreateToken(context.Background(), userID, "192.168.10.11:1234", "some user agent")
- So(err, ShouldBeNil)
- So(userToken, ShouldNotBeNil)
- lookedUpUserToken, err := userAuthTokenService.LookupToken(context.Background(), userToken.UnhashedToken)
- So(err, ShouldBeNil)
- So(lookedUpUserToken, ShouldNotBeNil)
- lookedUpUserToken, err = userAuthTokenService.LookupToken(context.Background(), userToken.UnhashedToken)
- So(err, ShouldBeNil)
- So(lookedUpUserToken, ShouldNotBeNil)
- lookedUpModel, err := ctx.getAuthTokenByID(lookedUpUserToken.Id)
- So(err, ShouldBeNil)
- So(lookedUpModel, ShouldNotBeNil)
- So(lookedUpModel.AuthTokenSeen, ShouldBeTrue)
- })
- Convey("Rotate token", func() {
- userToken, err := userAuthTokenService.CreateToken(context.Background(), userID, "192.168.10.11:1234", "some user agent")
- So(err, ShouldBeNil)
- So(userToken, ShouldNotBeNil)
- prevToken := userToken.AuthToken
- Convey("Should rotate current token and previous token when auth token seen", func() {
- updated, err := ctx.markAuthTokenAsSeen(userToken.Id)
- So(err, ShouldBeNil)
- So(updated, ShouldBeTrue)
- getTime = func() time.Time {
- return t.Add(10 * time.Minute)
- }
- rotated, err := userAuthTokenService.TryRotateToken(context.Background(), userToken, "1.1.1.1", "firefox")
- So(err, ShouldBeNil)
- So(rotated, ShouldBeTrue)
- storedToken, err := ctx.getAuthTokenByID(userToken.Id)
- So(err, ShouldBeNil)
- So(storedToken, ShouldNotBeNil)
- So(storedToken.AuthTokenSeen, ShouldBeFalse)
- So(storedToken.PrevAuthToken, ShouldEqual, prevToken)
- So(storedToken.AuthToken, ShouldNotEqual, prevToken)
- prevToken = storedToken.AuthToken
- updated, err = ctx.markAuthTokenAsSeen(userToken.Id)
- So(err, ShouldBeNil)
- So(updated, ShouldBeTrue)
- getTime = func() time.Time {
- return t.Add(20 * time.Minute)
- }
- rotated, err = userAuthTokenService.TryRotateToken(context.Background(), userToken, "1.1.1.1", "firefox")
- So(err, ShouldBeNil)
- So(rotated, ShouldBeTrue)
- storedToken, err = ctx.getAuthTokenByID(userToken.Id)
- So(err, ShouldBeNil)
- So(storedToken, ShouldNotBeNil)
- So(storedToken.AuthTokenSeen, ShouldBeFalse)
- So(storedToken.PrevAuthToken, ShouldEqual, prevToken)
- So(storedToken.AuthToken, ShouldNotEqual, prevToken)
- })
- Convey("Should rotate current token, but keep previous token when auth token not seen", func() {
- userToken.RotatedAt = getTime().Add(-2 * time.Minute).Unix()
- getTime = func() time.Time {
- return t.Add(2 * time.Minute)
- }
- rotated, err := userAuthTokenService.TryRotateToken(context.Background(), userToken, "1.1.1.1", "firefox")
- So(err, ShouldBeNil)
- So(rotated, ShouldBeTrue)
- storedToken, err := ctx.getAuthTokenByID(userToken.Id)
- So(err, ShouldBeNil)
- So(storedToken, ShouldNotBeNil)
- So(storedToken.AuthTokenSeen, ShouldBeFalse)
- So(storedToken.PrevAuthToken, ShouldEqual, prevToken)
- So(storedToken.AuthToken, ShouldNotEqual, prevToken)
- })
- })
- Convey("When populating userAuthToken from UserToken should copy all properties", func() {
- ut := models.UserToken{
- Id: 1,
- UserId: 2,
- AuthToken: "a",
- PrevAuthToken: "b",
- UserAgent: "c",
- ClientIp: "d",
- AuthTokenSeen: true,
- SeenAt: 3,
- RotatedAt: 4,
- CreatedAt: 5,
- UpdatedAt: 6,
- UnhashedToken: "e",
- }
- utBytes, err := json.Marshal(ut)
- So(err, ShouldBeNil)
- utJSON, err := simplejson.NewJson(utBytes)
- So(err, ShouldBeNil)
- utMap := utJSON.MustMap()
- var uat userAuthToken
- uat.fromUserToken(&ut)
- uatBytes, err := json.Marshal(uat)
- So(err, ShouldBeNil)
- uatJSON, err := simplejson.NewJson(uatBytes)
- So(err, ShouldBeNil)
- uatMap := uatJSON.MustMap()
- So(uatMap, ShouldResemble, utMap)
- })
- Convey("When populating userToken from userAuthToken should copy all properties", func() {
- uat := userAuthToken{
- Id: 1,
- UserId: 2,
- AuthToken: "a",
- PrevAuthToken: "b",
- UserAgent: "c",
- ClientIp: "d",
- AuthTokenSeen: true,
- SeenAt: 3,
- RotatedAt: 4,
- CreatedAt: 5,
- UpdatedAt: 6,
- UnhashedToken: "e",
- }
- uatBytes, err := json.Marshal(uat)
- So(err, ShouldBeNil)
- uatJSON, err := simplejson.NewJson(uatBytes)
- So(err, ShouldBeNil)
- uatMap := uatJSON.MustMap()
- var ut models.UserToken
- err = uat.toUserToken(&ut)
- So(err, ShouldBeNil)
- utBytes, err := json.Marshal(ut)
- So(err, ShouldBeNil)
- utJSON, err := simplejson.NewJson(utBytes)
- So(err, ShouldBeNil)
- utMap := utJSON.MustMap()
- So(utMap, ShouldResemble, uatMap)
- })
- Reset(func() {
- getTime = time.Now
- })
- })
- }
- func createTestContext(t *testing.T) *testContext {
- t.Helper()
- sqlstore := sqlstore.InitTestDB(t)
- tokenService := &UserAuthTokenService{
- SQLStore: sqlstore,
- Cfg: &setting.Cfg{
- LoginMaxInactiveLifetimeDays: 7,
- LoginMaxLifetimeDays: 30,
- TokenRotationIntervalMinutes: 10,
- },
- log: log.New("test-logger"),
- }
- return &testContext{
- sqlstore: sqlstore,
- tokenService: tokenService,
- }
- }
- type testContext struct {
- sqlstore *sqlstore.SqlStore
- tokenService *UserAuthTokenService
- }
- func (c *testContext) getAuthTokenByID(id int64) (*userAuthToken, error) {
- sess := c.sqlstore.NewSession()
- var t userAuthToken
- found, err := sess.ID(id).Get(&t)
- if err != nil || !found {
- return nil, err
- }
- return &t, nil
- }
- func (c *testContext) markAuthTokenAsSeen(id int64) (bool, error) {
- sess := c.sqlstore.NewSession()
- res, err := sess.Exec("UPDATE user_auth_token SET auth_token_seen = ? WHERE id = ?", c.sqlstore.Dialect.BooleanStr(true), id)
- if err != nil {
- return false, err
- }
- rowsAffected, err := res.RowsAffected()
- if err != nil {
- return false, err
- }
- return rowsAffected == 1, nil
- }
- func (c *testContext) updateRotatedAt(id, rotatedAt int64) (bool, error) {
- sess := c.sqlstore.NewSession()
- res, err := sess.Exec("UPDATE user_auth_token SET rotated_at = ? WHERE id = ?", rotatedAt, id)
- if err != nil {
- return false, err
- }
- rowsAffected, err := res.RowsAffected()
- if err != nil {
- return false, err
- }
- return rowsAffected == 1, nil
- }
|