two_factors_test.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. // Copyright 2020 The Gogs Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package db
  5. import (
  6. "context"
  7. "testing"
  8. "time"
  9. "github.com/stretchr/testify/assert"
  10. "github.com/stretchr/testify/require"
  11. "gorm.io/gorm"
  12. "gogs.io/gogs/internal/dbtest"
  13. "gogs.io/gogs/internal/errutil"
  14. )
  15. func TestTwoFactor_BeforeCreate(t *testing.T) {
  16. now := time.Now()
  17. db := &gorm.DB{
  18. Config: &gorm.Config{
  19. SkipDefaultTransaction: true,
  20. NowFunc: func() time.Time {
  21. return now
  22. },
  23. },
  24. }
  25. t.Run("CreatedUnix has been set", func(t *testing.T) {
  26. tf := &TwoFactor{CreatedUnix: 1}
  27. _ = tf.BeforeCreate(db)
  28. assert.Equal(t, int64(1), tf.CreatedUnix)
  29. })
  30. t.Run("CreatedUnix has not been set", func(t *testing.T) {
  31. tf := &TwoFactor{}
  32. _ = tf.BeforeCreate(db)
  33. assert.Equal(t, db.NowFunc().Unix(), tf.CreatedUnix)
  34. })
  35. }
  36. func TestTwoFactors(t *testing.T) {
  37. if testing.Short() {
  38. t.Skip()
  39. }
  40. t.Parallel()
  41. tables := []interface{}{new(TwoFactor), new(TwoFactorRecoveryCode)}
  42. db := &twoFactors{
  43. DB: dbtest.NewDB(t, "twoFactors", tables...),
  44. }
  45. for _, tc := range []struct {
  46. name string
  47. test func(*testing.T, *twoFactors)
  48. }{
  49. {"Create", twoFactorsCreate},
  50. {"GetByUserID", twoFactorsGetByUserID},
  51. {"IsUserEnabled", twoFactorsIsUserEnabled},
  52. } {
  53. t.Run(tc.name, func(t *testing.T) {
  54. t.Cleanup(func() {
  55. err := clearTables(t, db.DB, tables...)
  56. require.NoError(t, err)
  57. })
  58. tc.test(t, db)
  59. })
  60. if t.Failed() {
  61. break
  62. }
  63. }
  64. }
  65. func twoFactorsCreate(t *testing.T, db *twoFactors) {
  66. ctx := context.Background()
  67. // Create a 2FA token
  68. err := db.Create(ctx, 1, "secure-key", "secure-secret")
  69. require.NoError(t, err)
  70. // Get it back and check the Created field
  71. tf, err := db.GetByUserID(ctx, 1)
  72. require.NoError(t, err)
  73. assert.Equal(t, db.NowFunc().Format(time.RFC3339), tf.Created.UTC().Format(time.RFC3339))
  74. // Verify there are 10 recover codes generated
  75. var count int64
  76. err = db.Model(new(TwoFactorRecoveryCode)).Count(&count).Error
  77. require.NoError(t, err)
  78. assert.Equal(t, int64(10), count)
  79. }
  80. func twoFactorsGetByUserID(t *testing.T, db *twoFactors) {
  81. ctx := context.Background()
  82. // Create a 2FA token for user 1
  83. err := db.Create(ctx, 1, "secure-key", "secure-secret")
  84. require.NoError(t, err)
  85. // We should be able to get it back
  86. _, err = db.GetByUserID(ctx, 1)
  87. require.NoError(t, err)
  88. // Try to get a non-existent 2FA token
  89. _, err = db.GetByUserID(ctx, 2)
  90. wantErr := ErrTwoFactorNotFound{args: errutil.Args{"userID": int64(2)}}
  91. assert.Equal(t, wantErr, err)
  92. }
  93. func twoFactorsIsUserEnabled(t *testing.T, db *twoFactors) {
  94. ctx := context.Background()
  95. // Create a 2FA token for user 1
  96. err := db.Create(ctx, 1, "secure-key", "secure-secret")
  97. require.NoError(t, err)
  98. assert.True(t, db.IsUserEnabled(ctx, 1))
  99. assert.False(t, db.IsUserEnabled(ctx, 2))
  100. }