all repos

onasty @ e771742

a one-time notes service

onasty/e2e/e2e_test.go (view raw)

Olexandr Smirnov Olexandr Smirnov
ss2316544@gmail.com
fix: use separate ratelimiter for some endpoints (#166)..., 10 months ago
1
package e2e_test
2
3
import (
4
	"context"
5
	"fmt"
6
	"net/http"
7
	"testing"
8
	"time"
9
10
	"github.com/gin-gonic/gin"
11
	"github.com/golang-migrate/migrate/v4"
12
	"github.com/golang-migrate/migrate/v4/database/pgx"
13
	"github.com/jackc/pgx/v5/stdlib"
14
	"github.com/olexsmir/onasty/internal/config"
15
	"github.com/olexsmir/onasty/internal/hasher"
16
	"github.com/olexsmir/onasty/internal/jwtutil"
17
	"github.com/olexsmir/onasty/internal/logger"
18
	"github.com/olexsmir/onasty/internal/service/notesrv"
19
	"github.com/olexsmir/onasty/internal/service/usersrv"
20
	"github.com/olexsmir/onasty/internal/store/psql/noterepo"
21
	"github.com/olexsmir/onasty/internal/store/psql/passwordtokrepo"
22
	"github.com/olexsmir/onasty/internal/store/psql/sessionrepo"
23
	"github.com/olexsmir/onasty/internal/store/psql/userepo"
24
	"github.com/olexsmir/onasty/internal/store/psql/vertokrepo"
25
	"github.com/olexsmir/onasty/internal/store/psqlutil"
26
	"github.com/olexsmir/onasty/internal/store/rdb"
27
	"github.com/olexsmir/onasty/internal/store/rdb/notecache"
28
	"github.com/olexsmir/onasty/internal/store/rdb/usercache"
29
	httptransport "github.com/olexsmir/onasty/internal/transport/http"
30
	"github.com/olexsmir/onasty/internal/transport/http/ratelimit"
31
	"github.com/redis/go-redis/v9"
32
	"github.com/stretchr/testify/require"
33
	"github.com/stretchr/testify/suite"
34
	"github.com/testcontainers/testcontainers-go"
35
	tcpostgres "github.com/testcontainers/testcontainers-go/modules/postgres"
36
	tcredis "github.com/testcontainers/testcontainers-go/modules/redis"
37
	"github.com/testcontainers/testcontainers-go/wait"
38
39
	_ "github.com/golang-migrate/migrate/v4/source/file"
40
)
41
42
type (
43
	stopFunc     func()
44
	AppTestSuite struct {
45
		suite.Suite
46
47
		ctx     context.Context
48
		require *require.Assertions
49
50
		postgresDB   *psqlutil.DB
51
		stopPostgres stopFunc
52
53
		redisDB   *rdb.DB
54
		stopRedis stopFunc
55
56
		router       http.Handler
57
		hasher       hasher.Hasher
58
		jwtTokenizer jwtutil.JWTTokenizer
59
	}
60
	errorResponse struct {
61
		Message string `json:"message"`
62
	}
63
)
64
65
func TestAppSuite(t *testing.T) {
66
	if testing.Short() {
67
		t.Skip()
68
	}
69
70
	// gin output is too verbose(and annoying) in tests
71
	gin.SetMode(gin.TestMode)
72
73
	suite.Run(t, new(AppTestSuite))
74
}
75
76
func (e *AppTestSuite) SetupSuite() {
77
	e.ctx = context.Background()
78
	e.require = e.Require()
79
80
	e.postgresDB, e.stopPostgres = e.prepPostgres()
81
	e.redisDB, e.stopRedis = e.prepRedis()
82
83
	e.initDeps()
84
}
85
86
func (e *AppTestSuite) TearDownSuite() {
87
	e.stopPostgres()
88
	e.stopRedis()
89
}
90
91
// initDeps initializes the dependencies for the app
92
// and sets up the router for tests
93
func (e *AppTestSuite) initDeps() {
94
	cfg := e.getConfig()
95
96
	err := logger.SetDefault(cfg.LogLevel, cfg.LogFormat, cfg.LogShowLine)
97
	e.require.NoError(err)
98
99
	e.hasher = hasher.NewSHA256Hasher(cfg.PasswordSalt)
100
	e.jwtTokenizer = jwtutil.NewJWTUtil(cfg.JwtSigningKey, time.Hour)
101
102
	sessionrepo := sessionrepo.New(e.postgresDB)
103
	vertokrepo := vertokrepo.New(e.postgresDB)
104
	pwdtokrepo := passwordtokrepo.NewPasswordResetTokenRepo(e.postgresDB)
105
106
	stubOAuthProvider := newOauthProviderStub()
107
108
	notecache := notecache.New(e.redisDB, cfg.CacheUsersTTL)
109
	noterepo := noterepo.New(e.postgresDB)
110
	notesrv := notesrv.New(noterepo, e.hasher, notecache)
111
112
	userepo := userepo.New(e.postgresDB)
113
	usercache := usercache.New(e.redisDB, cfg.CacheUsersTTL)
114
	usersrv := usersrv.New(
115
		userepo,
116
		sessionrepo,
117
		vertokrepo,
118
		pwdtokrepo,
119
		noterepo,
120
		e.hasher,
121
		e.jwtTokenizer,
122
		newMailerMockService(),
123
		usercache,
124
		stubOAuthProvider,
125
		stubOAuthProvider,
126
		cfg.JwtRefreshTokenTTL,
127
		cfg.VerificationTokenTTL,
128
		cfg.ResetPasswordTokenTTL,
129
	)
130
131
	// for testing purposes, it's ok to have high values ig
132
	ratelimitCfg := ratelimit.Config{
133
		RPS:   1000,
134
		TTL:   time.Millisecond,
135
		Burst: 1000,
136
	}
137
138
	handler := httptransport.NewTransport(
139
		usersrv,
140
		notesrv,
141
		cfg.AppEnv,
142
		cfg.AppURL,
143
		cfg.CORSAllowedOrigins,
144
		cfg.CORSMaxAge,
145
		ratelimitCfg,
146
		ratelimitCfg,
147
	)
148
	e.router = handler.Handler()
149
}
150
151
func (e *AppTestSuite) prepPostgres() (*psqlutil.DB, stopFunc) {
152
	dbCredential := "testing"
153
	postgresContainer, err := tcpostgres.Run(e.ctx,
154
		"postgres:16-alpine",
155
		tcpostgres.WithUsername(dbCredential),
156
		tcpostgres.WithPassword(dbCredential),
157
		tcpostgres.WithDatabase(dbCredential),
158
		testcontainers.WithWaitStrategy(wait.ForListeningPort("5432/tcp")))
159
	e.require.NoError(err)
160
161
	stop := func() { e.require.NoError(postgresContainer.Terminate(e.ctx)) }
162
163
	// connect to the db
164
	host, err := postgresContainer.Host(e.ctx)
165
	e.require.NoError(err)
166
167
	port, err := postgresContainer.MappedPort(e.ctx, "5432/tcp")
168
	e.require.NoError(err)
169
170
	db, err := psqlutil.Connect(e.ctx, fmt.Sprintf( //nolint:nosprintfhostport
171
		"postgres://%s:%s@%s:%s/%s",
172
		dbCredential,
173
		dbCredential,
174
		host,
175
		port.Port(),
176
		dbCredential,
177
	))
178
	e.require.NoError(err)
179
180
	// run migrations
181
	sdb := stdlib.OpenDBFromPool(db.Pool)
182
	driver, err := pgx.WithInstance(sdb, &pgx.Config{})
183
	e.require.NoError(err)
184
185
	m, err := migrate.NewWithDatabaseInstance(
186
		"file://../migrations/",
187
		"pgxv5", driver,
188
	)
189
	e.require.NoError(err)
190
191
	e.require.NoError(m.Up())
192
	e.require.NoError(driver.Close())
193
194
	return db, stop
195
}
196
197
func (e *AppTestSuite) prepRedis() (*rdb.DB, stopFunc) {
198
	redisContainer, err := tcredis.Run(e.ctx, "redis:7.4-alpine")
199
	e.require.NoError(err)
200
201
	stop := func() { e.require.NoError(redisContainer.Terminate(e.ctx)) }
202
203
	uri, err := redisContainer.ConnectionString(e.ctx)
204
	e.require.NoError(err)
205
206
	connOpts, err := redis.ParseURL(uri)
207
	e.require.NoError(err)
208
209
	redis, err := rdb.Connect(e.ctx, connOpts.Addr, connOpts.Password, connOpts.DB)
210
	e.require.NoError(err)
211
212
	return redis, stop
213
}
214
215
func (e *AppTestSuite) getConfig() *config.Config {
216
	e.T().Setenv("APP_ENV", "test")
217
	e.T().Setenv("APP_URL", "localhost")
218
	e.T().Setenv("PASSWORD_SALT", "salty-password")
219
	e.T().Setenv("NOTE_PASSWORD_SALT", "salty-noted-password")
220
	e.T().Setenv("JWT_SIGNING_KEY", "jwt-key")
221
	e.T().Setenv("LOG_SHOW_LINE", "true")
222
	e.T().Setenv("LOG_FORMAT", "text")
223
	e.T().Setenv("LOG_LEVEL", "debug")
224
225
	return config.NewConfig()
226
}