all repos

onasty @ abf0cb5582f45940ba0ad8e6312986a02a63c769

a one-time notes service

onasty/e2e/e2e_test.go (view raw)

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