Update integration test suite (formally e2e) (#713)

pull/723/head
Michael Fridman 2024-03-14 09:35:52 -04:00 committed by GitHub
parent 73e2f8897f
commit b2c483ada4
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
91 changed files with 1519 additions and 2204 deletions

View File

@ -1,4 +1,4 @@
name: Goose end-end tests
name: Goose integration tests
on:
push:
@ -12,13 +12,9 @@ concurrency:
jobs:
test:
name: Run e2e tests
name: Run integration tests
timeout-minutes: 10
runs-on: ubuntu-latest
strategy:
max-parallel: 2
matrix:
dialect: ["postgres", "mysql", "vertica", "clickhouse", "ydb", "turso"]
steps:
- name: Checkout code
@ -27,6 +23,6 @@ jobs:
uses: actions/setup-go@v5
with:
go-version: "stable"
- name: Run e2e ${{ matrix.dialect }} tests
- name: Run full integration tests
run: |
make test-e2e-${{ matrix.dialect }}
make test-integration

View File

@ -1,4 +1,5 @@
name: golangci
on:
push:
branches:
@ -6,7 +7,7 @@ on:
- main
pull_request:
concurrency:
concurrency:
group: ${{ github.workflow }}-${{ github.ref }}
cancel-in-progress: true

3
.gitignore vendored
View File

@ -13,3 +13,6 @@
dist/
release_notes.txt
go.work
go.work.sum

View File

@ -1,4 +1,4 @@
GO_TEST_FLAGS ?= -race -count=1 -v -timeout=10m
GO_TEST_FLAGS ?= -race -count=1 -v -timeout=5m
# These are the default values for the test database. They can be overridden
DB_USER ?= dbuser
@ -42,25 +42,50 @@ test-packages:
test-packages-short:
go test -test.short $(GO_TEST_FLAGS) $$(go list ./... | grep -v -e /tests -e /bin -e /cmd -e /examples)
test-e2e: test-e2e-postgres test-e2e-mysql test-e2e-clickhouse test-e2e-vertica test-e2e-ydb test-e2e-turso
#
# Integration-related targets
#
add-gowork:
@[ -f go.work ] || go work init
@[ -f go.work.sum ] || go work use -r .
test-e2e-postgres:
go test $(GO_TEST_FLAGS) ./tests/e2e -dialect=postgres
remove-gowork:
rm -rf go.work go.work.sum
test-e2e-mysql:
go test $(GO_TEST_FLAGS) ./tests/e2e -dialect=mysql
gomod:
go mod tidy
cd ./internal/testing/integration && go mod tidy
test-e2e-clickhouse:
go test $(GO_TEST_FLAGS) ./tests/clickhouse -test.short
upgrade-integration-deps:
cd ./internal/testing/integration && go get -u ./... && go mod tidy
test-e2e-vertica:
go test $(GO_TEST_FLAGS) ./tests/vertica
test-postgres-long: add-gowork test-postgres
go test $(GO_TEST_FLAGS) ./internal/testing/integration -run='(TestPostgresProviderLocking|TestPostgresSessionLocker)'
test-e2e-ydb:
go test $(GO_TEST_FLAGS) -parallel=1 ./tests/e2e -dialect=ydb
test-postgres: add-gowork
go test $(GO_TEST_FLAGS) ./internal/testing/integration -run='TestPostgres'
test-e2e-turso:
go test $(GO_TEST_FLAGS) -parallel=1 ./tests/e2e -dialect=turso
test-clickhouse: add-gowork
go test $(GO_TEST_FLAGS) ./internal/testing/integration -run='(TestClickhouse|TestClickhouseRemote)'
test-mysql: add-gowork
go test $(GO_TEST_FLAGS) ./internal/testing/integration -run='TestMySQL'
test-turso: add-gowork
go test $(GO_TEST_FLAGS) ./internal/testing/integration -run='TestTurso'
test-vertica: add-gowork
go test $(GO_TEST_FLAGS) ./internal/testing/integration -run='TestVertica'
test-ydb: add-gowork
go test $(GO_TEST_FLAGS) ./internal/testing/integration -run='TestYDB'
test-integration: add-gowork
go test $(GO_TEST_FLAGS) ./internal/testing/integration/...
#
# Docker-related targets
#
docker-cleanup:
docker stop -t=0 $$(docker ps --filter="label=goose_test" -aq)

View File

@ -7,10 +7,8 @@ import (
"path/filepath"
"testing"
"github.com/jackc/pgx/v5/pgconn"
"github.com/pressly/goose/v3/database"
"github.com/pressly/goose/v3/internal/check"
"github.com/pressly/goose/v3/internal/testdb"
"go.uber.org/multierr"
"modernc.org/sqlite"
)
@ -32,21 +30,6 @@ func TestDialectStore(t *testing.T) {
_, err = database.NewStore("", "foo")
check.HasError(t, err)
})
t.Run("postgres", func(t *testing.T) {
if testing.Short() {
t.Skip("skip long-running test")
}
// Test postgres specific behavior.
db, cleanup, err := testdb.NewPostgres()
check.NoError(t, err)
t.Cleanup(cleanup)
testStore(context.Background(), t, database.DialectPostgres, db, func(t *testing.T, err error) {
var pgErr *pgconn.PgError
ok := errors.As(err, &pgErr)
check.Bool(t, ok, true)
check.Equal(t, pgErr.Code, "42P07") // duplicate_table
})
})
// Test generic behavior.
t.Run("sqlite3", func(t *testing.T) {
db, err := sql.Open("sqlite", ":memory:")

25
go.mod
View File

@ -8,7 +8,6 @@ require (
github.com/jackc/pgx/v5 v5.5.5
github.com/mfridman/interpolate v0.0.2
github.com/microsoft/go-mssqldb v1.7.0
github.com/ory/dockertest/v3 v3.10.0
github.com/sethvargo/go-retry v0.2.4
github.com/tursodatabase/libsql-client-go v0.0.0-20240220085343-4ae0eb9d0898
github.com/vertica/vertica-sql-go v1.3.3
@ -21,32 +20,20 @@ require (
require (
filippo.io/edwards25519 v1.1.0 // indirect
github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 // indirect
github.com/ClickHouse/ch-go v0.58.2 // indirect
github.com/Microsoft/go-winio v0.6.1 // indirect
github.com/Nvveen/Gotty v0.0.0-20120604004816-cd527374f1e5 // indirect
github.com/andybalholm/brotli v1.0.6 // indirect
github.com/antlr/antlr4/runtime/Go/antlr/v4 v4.0.0-20230512164433-5d1fd1a340c9 // indirect
github.com/cenkalti/backoff/v4 v4.2.1 // indirect
github.com/containerd/continuity v0.4.3 // indirect
github.com/docker/cli v24.0.7+incompatible // indirect
github.com/docker/docker v24.0.7+incompatible // indirect
github.com/docker/go-connections v0.4.0 // indirect
github.com/docker/go-units v0.5.0 // indirect
github.com/dustin/go-humanize v1.0.1 // indirect
github.com/elastic/go-sysinfo v1.11.2 // indirect
github.com/elastic/go-windows v1.0.1 // indirect
github.com/go-faster/city v1.0.1 // indirect
github.com/go-faster/errors v0.6.1 // indirect
github.com/gogo/protobuf v1.3.2 // indirect
github.com/golang-jwt/jwt/v4 v4.5.0 // indirect
github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9 // indirect
github.com/golang-sql/sqlexp v0.1.0 // indirect
github.com/golang/protobuf v1.5.3 // indirect
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 // indirect
github.com/google/uuid v1.6.0 // indirect
github.com/hashicorp/golang-lru/v2 v2.0.7 // indirect
github.com/imdario/mergo v0.3.16 // indirect
github.com/jackc/pgpassfile v1.0.0 // indirect
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect
github.com/jackc/puddle/v2 v2.2.1 // indirect
@ -55,12 +42,7 @@ require (
github.com/klauspost/compress v1.17.2 // indirect
github.com/libsql/sqlite-antlr4-parser v0.0.0-20230802215326-5cb5bb604475 // indirect
github.com/mattn/go-isatty v0.0.20 // indirect
github.com/mitchellh/mapstructure v1.5.0 // indirect
github.com/moby/term v0.5.0 // indirect
github.com/ncruces/go-strftime v0.1.9 // indirect
github.com/opencontainers/go-digest v1.0.0 // indirect
github.com/opencontainers/image-spec v1.1.0-rc5 // indirect
github.com/opencontainers/runc v1.1.12 // indirect
github.com/paulmach/orb v0.10.0 // indirect
github.com/pierrec/lz4/v4 v4.1.18 // indirect
github.com/pkg/errors v0.9.1 // indirect
@ -68,24 +50,17 @@ require (
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec // indirect
github.com/segmentio/asm v1.2.0 // indirect
github.com/shopspring/decimal v1.3.1 // indirect
github.com/sirupsen/logrus v1.9.3 // indirect
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb // indirect
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 // indirect
github.com/xeipuuv/gojsonschema v1.2.0 // indirect
github.com/ydb-platform/ydb-go-genproto v0.0.0-20240126124512-dbb0e1720dbf // indirect
go.opentelemetry.io/otel v1.20.0 // indirect
go.opentelemetry.io/otel/trace v1.20.0 // indirect
golang.org/x/crypto v0.18.0 // indirect
golang.org/x/exp v0.0.0-20231108232855-2478ac86f678 // indirect
golang.org/x/mod v0.14.0 // indirect
golang.org/x/net v0.20.0 // indirect
golang.org/x/sys v0.16.0 // indirect
golang.org/x/text v0.14.0 // indirect
golang.org/x/tools v0.17.0 // indirect
google.golang.org/genproto/googleapis/rpc v0.0.0-20231106174013-bbf56f31fb17 // indirect
google.golang.org/grpc v1.59.0 // indirect
google.golang.org/protobuf v1.33.0 // indirect
gopkg.in/yaml.v2 v2.4.0 // indirect
gopkg.in/yaml.v3 v3.0.1 // indirect
howett.net/plist v1.0.0 // indirect
modernc.org/gc/v3 v3.0.0-20240107210532-573471604cb6 // indirect

45
go.sum
View File

@ -7,8 +7,6 @@ github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.5.1 h1:sO0/P7g68FrryJzljemN+
github.com/Azure/azure-sdk-for-go/sdk/internal v1.5.1 h1:6oNBlSdi1QqM1PNW7FPA6xOGA5UNsXnkaYZz9vdPGhA=
github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/azkeys v1.0.1 h1:MyVTgWR8qd/Jw1Le0NZebGBUCLbtak3bJ3z1OlqZBpw=
github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/internal v1.0.0 h1:D3occbWoio4EBLkbkevetNMAVX197GkzbUMtqjGWn80=
github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 h1:L/gRVlceqvL25UVaW/CKtUDjefjrs0SPonmDGUVOYP0=
github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E=
github.com/AzureAD/microsoft-authentication-library-for-go v1.2.1 h1:DzHpqpoJVaCgOUdVHxE8QB52S6NiVdDQvGlny1qvPqA=
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
github.com/ClickHouse/ch-go v0.58.2 h1:jSm2szHbT9MCAB1rJ3WuCJqmGLi5UTjlNu+f530UTS0=
@ -16,16 +14,11 @@ github.com/ClickHouse/ch-go v0.58.2/go.mod h1:Ap/0bEmiLa14gYjCiRkYGbXvbe8vwdrfTY
github.com/ClickHouse/clickhouse-go/v2 v2.17.1 h1:ZCmAYWpu75IyEi7+Yrs/uaAjiCGY5wfW5kXo64exkX4=
github.com/ClickHouse/clickhouse-go/v2 v2.17.1/go.mod h1:rkGTvFDTLqLIm0ma+13xmcCfr/08Gvs7KmFt1tgiWHQ=
github.com/Microsoft/go-winio v0.6.1 h1:9/kr64B9VUZrLm5YYwbGtUJnMgqWVOdUAXu6Migciow=
github.com/Microsoft/go-winio v0.6.1/go.mod h1:LRdKpFKfdobln8UmuiYcKPot9D2v6svN5+sAH+4kjUM=
github.com/Nvveen/Gotty v0.0.0-20120604004816-cd527374f1e5 h1:TngWCqHvy9oXAN6lEVMRuU21PR1EtLVZJmdB18Gu3Rw=
github.com/Nvveen/Gotty v0.0.0-20120604004816-cd527374f1e5/go.mod h1:lmUJ/7eu/Q8D7ML55dXQrVaamCz2vxCfdQBasLZfHKk=
github.com/andybalholm/brotli v1.0.6 h1:Yf9fFpf49Zrxb9NlQaluyE92/+X7UVHlhMNJN2sxfOI=
github.com/andybalholm/brotli v1.0.6/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig=
github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY=
github.com/antlr/antlr4/runtime/Go/antlr/v4 v4.0.0-20230512164433-5d1fd1a340c9 h1:goHVqTbFX3AIo0tzGr14pgfAW2ZfPChKO21Z9MGf/gk=
github.com/antlr/antlr4/runtime/Go/antlr/v4 v4.0.0-20230512164433-5d1fd1a340c9/go.mod h1:pSwJ0fSY5KhvocuWSx4fz3BA8OrA1bQn+K1Eli3BRwM=
github.com/cenkalti/backoff/v4 v4.2.1 h1:y4OZtCnogmCPw98Zjyt5a6+QwPLGkiQsYW5oUqylYbM=
github.com/cenkalti/backoff/v4 v4.2.1/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE=
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
@ -35,21 +28,13 @@ github.com/cncf/udpa/go v0.0.0-20210930031921-04548b0d99d4/go.mod h1:6pvJx4me5XP
github.com/cncf/xds/go v0.0.0-20210922020428-25de7278fc84/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/containerd/continuity v0.4.3 h1:6HVkalIp+2u1ZLH1J/pYX2oBVXlJZvh1X1A7bEZ9Su8=
github.com/containerd/continuity v0.4.3/go.mod h1:F6PTNCKepoxEaXLQp3wDAjygEnImnZ/7o4JzpodfroQ=
github.com/creack/pty v1.1.18 h1:n56/Zwd5o6whRC5PMGretI4IdRLlmBXYNjScPaBgsbY=
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/docker/cli v24.0.7+incompatible h1:wa/nIwYFW7BVTGa7SWPVyyXU9lgORqUb1xfI36MSkFg=
github.com/docker/cli v24.0.7+incompatible/go.mod h1:JLrzqnKDaYBop7H2jaqPtU4hHvMKP+vjCwu2uszcLI8=
github.com/docker/distribution v2.8.2+incompatible h1:T3de5rq0dB1j30rp0sA2rER+m322EBzniBPB6ZIzuh8=
github.com/docker/docker v24.0.7+incompatible h1:Wo6l37AuwP3JaMnZa226lzVXGA3F9Ig1seQen0cKYlM=
github.com/docker/docker v24.0.7+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk=
github.com/docker/go-connections v0.4.0 h1:El9xVISelRB7BuFusrZozjnkIM5YnzCViNKohAFqRJQ=
github.com/docker/go-connections v0.4.0/go.mod h1:Gbd7IOopHjR8Iph03tsViu4nIes5XhDvyHbTtUxmeec=
github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4=
github.com/docker/go-units v0.5.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY=
github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto=
github.com/elastic/go-sysinfo v1.8.1/go.mod h1:JfllUnzoQV/JRYymbH3dO1yggI3mV2oTKSXsDHM+uIM=
@ -127,8 +112,6 @@ github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
github.com/google/pprof v0.0.0-20221118152302-e6195bd50e26 h1:Xim43kblpZXfIBQsbuBVKCudVG457BR2GZFIz3uw3hQ=
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 h1:El6M4kTTCOh6aBiKaUGG7oYTSPP8MxqL4YI3kZKwcP4=
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510/go.mod h1:pupxD2MaaD3pAXIBCelhxNneeOaAeabZDe5s4K6zSpQ=
github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
@ -137,8 +120,6 @@ github.com/gorilla/websocket v1.4.2 h1:+/TMaTYc4QFitKJxsQ7Yye35DkWvkdLcvGKqM+x0U
github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw=
github.com/hashicorp/golang-lru/v2 v2.0.7 h1:a+bsQ5rvGLjzHuww6tVxozPZFVghXaHOwFs4luLUK2k=
github.com/hashicorp/golang-lru/v2 v2.0.7/go.mod h1:QeFd9opnmA6QUJc5vARoKUSoFhyfM2/ZepoAG6RGpeM=
github.com/imdario/mergo v0.3.16 h1:wwQJbIsHYGMUyLSPrEq1CT16AhnhNJQ51+4fdHUnCl4=
github.com/imdario/mergo v0.3.16/go.mod h1:WBLT9ZmE3lPoWsEzCh9LPo3TiwVN+ZKEjmz+hD27ysY=
github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM=
github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg=
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a h1:bbPeKD0xmW/Y25WS6cokEszi5g+S0QxI/d45PkRi7Nk=
@ -168,7 +149,6 @@ github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc=
github.com/leodido/go-urn v1.2.0 h1:hpXL4XnriNwQ/ABnpepYM/1vCLWNDfUNts8dX3xTG6Y=
github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII=
github.com/lib/pq v0.0.0-20180327071824-d34b9ff171c2 h1:hRGSmZu7j271trc9sneMrpOW7GN5ngLm8YUZIPzf394=
github.com/libsql/sqlite-antlr4-parser v0.0.0-20230802215326-5cb5bb604475 h1:6PfEMwfInASh9hkN83aR0j4W/eKaAZt/AURtXAXlas0=
github.com/libsql/sqlite-antlr4-parser v0.0.0-20230802215326-5cb5bb604475/go.mod h1:20nXSmcf0nAscrzqsXeC2/tA3KkV2eCiJqYuyAgl+ss=
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
@ -179,10 +159,6 @@ github.com/mfridman/interpolate v0.0.2 h1:pnuTK7MQIxxFz1Gr+rjSIx9u7qVjf5VOoM/u6B
github.com/mfridman/interpolate v0.0.2/go.mod h1:p+7uk6oE07mpE/Ik1b8EckO0O4ZXiGAfshKBWLUM9Xg=
github.com/microsoft/go-mssqldb v1.7.0 h1:sgMPW0HA6Ihd37Yx0MzHyKD726C2kY/8KJsQtXHNaAs=
github.com/microsoft/go-mssqldb v1.7.0/go.mod h1:kOvZKUdrhhFQmxLZqbwUV0rHkNkZpthMITIb2Ko1IoA=
github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY=
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
github.com/moby/term v0.5.0 h1:xt8Q1nalod/v7BqbG21f8mQPqH+xAaC9C3N3wfWbVP0=
github.com/moby/term v0.5.0/go.mod h1:8FzsFHVUBGZdbDsJw/ot+X+d5HLUbvklYLJ9uGfcI3Y=
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742 h1:Esafd1046DLDQ0W1YjYsBW+p8U2u7vzgW2SQVmlNazg=
@ -191,13 +167,7 @@ github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe/go.mod h1:wL8QJ
github.com/ncruces/go-strftime v0.1.9 h1:bY0MQC28UADQmHmaF5dgpLmImcShSi2kHU9XLdhx/f4=
github.com/ncruces/go-strftime v0.1.9/go.mod h1:Fwc5htZGVVkseilnfgOVb9mKy6w1naJmn9CehxcKcls=
github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8Oi/yOhh5U=
github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM=
github.com/opencontainers/image-spec v1.1.0-rc5 h1:Ygwkfw9bpDvs+c9E34SdgGOj41dX/cbdlwvlWt0pnFI=
github.com/opencontainers/image-spec v1.1.0-rc5/go.mod h1:X4pATf0uXsnn3g5aiGIsVnJBR4mxhKzfwmvK/B2NTm8=
github.com/opencontainers/runc v1.1.12 h1:BOIssBaW1La0/qbNZHXOOa71dZfZEQOzW7dqQf3phss=
github.com/opencontainers/runc v1.1.12/go.mod h1:S+lQwSfncpBha7XTy/5lBwWgm5+y5Ma/O44Ekby9FK8=
github.com/ory/dockertest/v3 v3.10.0 h1:4K3z2VMe8Woe++invjaTB7VRyQXQy5UY+loujO4aNE4=
github.com/ory/dockertest/v3 v3.10.0/go.mod h1:nr57ZbRWMqfsdGdFNLHz5jjNdDb7VVFnzAeW1n5N1Lg=
github.com/paulmach/orb v0.10.0 h1:guVYVqzxHE/CQ1KpfGO077TR0ATHSNjp4s6XGLn3W9s=
github.com/paulmach/orb v0.10.0/go.mod h1:5mULz1xQfs3bmQm63QEJA6lNGujuRafwA5S/EnuLaLU=
github.com/paulmach/protoscan v0.2.1/go.mod h1:SpcSwydNLrxUGSDvXvO0P7g7AuhJ7lcKfDlhJCDw2gY=
@ -223,8 +193,6 @@ github.com/sethvargo/go-retry v0.2.4 h1:T+jHEQy/zKJf5s95UkguisicE0zuF9y7+/vgz08O
github.com/sethvargo/go-retry v0.2.4/go.mod h1:1afjQuvh7s4gflMObvjLPaWgluLLyhA1wmVZ6KLpICw=
github.com/shopspring/decimal v1.3.1 h1:2Usl1nmF/WZucqkFZhnfFYxxxu8LG21F6nPQBE5gKV8=
github.com/shopspring/decimal v1.3.1/go.mod h1:DKyhrW/HYNuLGql+MJL6WCR6knT2jwCFRcu2hWCYk4o=
github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ=
github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
@ -244,13 +212,6 @@ github.com/vertica/vertica-sql-go v1.3.3/go.mod h1:jnn2GFuv+O2Jcjktb7zyc4Utlbu9Y
github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI=
github.com/xdg-go/scram v1.1.1/go.mod h1:RaEWvsqvNKKvBPvcKeFjrG2cJqOkHTiyTpzz23ni57g=
github.com/xdg-go/stringprep v1.0.3/go.mod h1:W3f5j4i+9rC0kuIEJL0ky1VpHXQU3ocBgklLGvcBnW8=
github.com/xeipuuv/gojsonpointer v0.0.0-20180127040702-4e3ac2762d5f/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU=
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb h1:zGWFAtiMcyryUHoUjUJX0/lt1H2+i2Ka2n+D3DImSNo=
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU=
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 h1:EzJWgHovont7NscjpAxXsDA8S8BMYve8Y5+7cuRE7R0=
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415/go.mod h1:GwrjFmJcFw6At/Gs6z4yjiIwzuJ1/+UwLxMQDVQXShQ=
github.com/xeipuuv/gojsonschema v1.2.0 h1:LhYJRs+L4fBtjZUfuSZIKGeVu0QRy8e5Xi7D17UxZ74=
github.com/xeipuuv/gojsonschema v1.2.0/go.mod h1:anYRn/JVcOK2ZgGU+IjEV4nwlhoK5sQluxsYJ78Id3Y=
github.com/ydb-platform/ydb-go-genproto v0.0.0-20240126124512-dbb0e1720dbf h1:ckwNHVo4bv2tqNkgx3W3HANh3ta1j6TR5qw08J1A7Tw=
github.com/ydb-platform/ydb-go-genproto v0.0.0-20240126124512-dbb0e1720dbf/go.mod h1:Er+FePu1dNUieD+XTMDduGpQuCPssK5Q4BjF+IIXJ3I=
github.com/ydb-platform/ydb-go-sdk/v3 v3.55.1 h1:Ebo6J5AMXgJ3A438ECYotA0aK7ETqjQx9WoZvVxzKBE=
@ -284,7 +245,6 @@ golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHl
golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
golang.org/x/mod v0.14.0 h1:dGoOF9QVLYng8IHTm7BAyWqCqSheQ5pYWGhzW00YJr0=
golang.org/x/mod v0.14.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
@ -320,9 +280,7 @@ golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7w
golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.16.0 h1:xWw16ngr6ZMtmxDyKyIgsE93KNKz5HKmMa3b8ALHidU=
golang.org/x/sys v0.16.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
@ -344,7 +302,6 @@ golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtn
golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
golang.org/x/tools v0.17.0 h1:FvmRgNOcs3kOa+T20R1uhfP9F6HgG2mfxDv1vrx1Htc=
golang.org/x/tools v0.17.0/go.mod h1:xsh6VxdV005rRVaS6SSAf9oiAqljS7UZUacMZ8Bnsps=
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
@ -390,11 +347,9 @@ gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gotest.tools/v3 v3.4.0 h1:ZazjZUfuVeZGLAmlKKuyv3IKP5orXcwtOwDQH6YVr6o=
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
howett.net/plist v0.0.0-20181124034731-591f970eefbb/go.mod h1:vMygbs4qMhSZSc4lCUl2OEE+rDiIIJAIdR4m7MiMcm0=

87
internal/testing/go.mod Normal file
View File

@ -0,0 +1,87 @@
module github.com/pressly/goose/v3/internal/testing
go 1.22.1
require (
github.com/ClickHouse/clickhouse-go/v2 v2.22.0
github.com/go-sql-driver/mysql v1.8.0
github.com/jackc/pgx/v5 v5.5.5
github.com/ory/dockertest/v3 v3.10.0
github.com/pressly/goose/v3 v3.19.2
github.com/sethvargo/go-retry v0.2.4
github.com/stretchr/testify v1.9.0
github.com/tursodatabase/libsql-client-go v0.0.0-20240220085343-4ae0eb9d0898
github.com/vertica/vertica-sql-go v1.3.3
github.com/ydb-platform/ydb-go-sdk/v3 v3.57.4
golang.org/x/sync v0.6.0
)
require (
dario.cat/mergo v1.0.0 // indirect
filippo.io/edwards25519 v1.1.0 // indirect
github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 // indirect
github.com/ClickHouse/ch-go v0.61.5 // indirect
github.com/Microsoft/go-winio v0.6.1 // indirect
github.com/Nvveen/Gotty v0.0.0-20120604004816-cd527374f1e5 // indirect
github.com/andybalholm/brotli v1.1.0 // indirect
github.com/antlr/antlr4/runtime/Go/antlr/v4 v4.0.0-20230512164433-5d1fd1a340c9 // indirect
github.com/cenkalti/backoff/v4 v4.2.1 // indirect
github.com/containerd/continuity v0.4.3 // indirect
github.com/davecgh/go-spew v1.1.1 // indirect
github.com/docker/cli v25.0.4+incompatible // indirect
github.com/docker/docker v25.0.4+incompatible // indirect
github.com/docker/go-connections v0.5.0 // indirect
github.com/docker/go-units v0.5.0 // indirect
github.com/elastic/go-sysinfo v1.13.1 // indirect
github.com/elastic/go-windows v1.0.1 // indirect
github.com/go-faster/city v1.0.1 // indirect
github.com/go-faster/errors v0.7.1 // indirect
github.com/gogo/protobuf v1.3.2 // indirect
github.com/golang-jwt/jwt/v4 v4.5.0 // indirect
github.com/golang/protobuf v1.5.4 // indirect
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 // indirect
github.com/google/uuid v1.6.0 // indirect
github.com/jackc/pgpassfile v1.0.0 // indirect
github.com/jackc/pgservicefile v0.0.0-20231201235250-de7065d80cb9 // indirect
github.com/jackc/puddle/v2 v2.2.1 // indirect
github.com/joeshaw/multierror v0.0.0-20140124173710-69b34d4ec901 // indirect
github.com/jonboulle/clockwork v0.4.0 // indirect
github.com/klauspost/compress v1.17.7 // indirect
github.com/kr/text v0.2.0 // indirect
github.com/libsql/sqlite-antlr4-parser v0.0.0-20230802215326-5cb5bb604475 // indirect
github.com/mfridman/interpolate v0.0.2 // indirect
github.com/mitchellh/mapstructure v1.5.0 // indirect
github.com/moby/term v0.5.0 // indirect
github.com/opencontainers/go-digest v1.0.0 // indirect
github.com/opencontainers/image-spec v1.1.0 // indirect
github.com/opencontainers/runc v1.1.12 // indirect
github.com/paulmach/orb v0.11.1 // indirect
github.com/pierrec/lz4/v4 v4.1.21 // indirect
github.com/pkg/errors v0.9.1 // indirect
github.com/pmezard/go-difflib v1.0.0 // indirect
github.com/prometheus/procfs v0.13.0 // indirect
github.com/segmentio/asm v1.2.0 // indirect
github.com/shopspring/decimal v1.3.1 // indirect
github.com/sirupsen/logrus v1.9.3 // indirect
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb // indirect
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 // indirect
github.com/xeipuuv/gojsonschema v1.2.0 // indirect
github.com/ydb-platform/ydb-go-genproto v0.0.0-20240313174312-cec53bca0668 // indirect
go.opentelemetry.io/otel v1.24.0 // indirect
go.opentelemetry.io/otel/trace v1.24.0 // indirect
go.uber.org/multierr v1.11.0 // indirect
golang.org/x/crypto v0.21.0 // indirect
golang.org/x/exp v0.0.0-20240222234643-814bf88cf225 // indirect
golang.org/x/mod v0.16.0 // indirect
golang.org/x/net v0.22.0 // indirect
golang.org/x/sys v0.18.0 // indirect
golang.org/x/text v0.14.0 // indirect
golang.org/x/tools v0.19.0 // indirect
google.golang.org/genproto/googleapis/rpc v0.0.0-20240311173647-c811ad7063a7 // indirect
google.golang.org/grpc v1.62.1 // indirect
google.golang.org/protobuf v1.33.0 // indirect
gopkg.in/yaml.v2 v2.4.0 // indirect
gopkg.in/yaml.v3 v3.0.1 // indirect
howett.net/plist v1.0.1 // indirect
nhooyr.io/websocket v1.8.10 // indirect
)

384
internal/testing/go.sum Normal file
View File

@ -0,0 +1,384 @@
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
dario.cat/mergo v1.0.0 h1:AGCNq9Evsj31mOgNPcLyXc+4PNABt905YmuqPYYpBWk=
dario.cat/mergo v1.0.0/go.mod h1:uNxQE+84aUszobStD9th8a29P2fMDhsBdgRYvZOxGmk=
filippo.io/edwards25519 v1.1.0 h1:FNf4tywRC1HmFuKW5xopWpigGjJKiJSV0Cqo0cJWDaA=
filippo.io/edwards25519 v1.1.0/go.mod h1:BxyFTGdWcka3PhytdK4V28tE5sGfRvvvRV7EaN4VDT4=
github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 h1:L/gRVlceqvL25UVaW/CKtUDjefjrs0SPonmDGUVOYP0=
github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E=
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
github.com/ClickHouse/ch-go v0.61.5 h1:zwR8QbYI0tsMiEcze/uIMK+Tz1D3XZXLdNrlaOpeEI4=
github.com/ClickHouse/ch-go v0.61.5/go.mod h1:s1LJW/F/LcFs5HJnuogFMta50kKDO0lf9zzfrbl0RQg=
github.com/ClickHouse/clickhouse-go/v2 v2.22.0 h1:LAdk0qT125PpSPnYepFQs5X5z1EwpAtIX10SUELPgi0=
github.com/ClickHouse/clickhouse-go/v2 v2.22.0/go.mod h1:tBhdF3f3RdP7sS59+oBAtTyhWpy0024ZxDMhgxra0QE=
github.com/Microsoft/go-winio v0.6.1 h1:9/kr64B9VUZrLm5YYwbGtUJnMgqWVOdUAXu6Migciow=
github.com/Microsoft/go-winio v0.6.1/go.mod h1:LRdKpFKfdobln8UmuiYcKPot9D2v6svN5+sAH+4kjUM=
github.com/Nvveen/Gotty v0.0.0-20120604004816-cd527374f1e5 h1:TngWCqHvy9oXAN6lEVMRuU21PR1EtLVZJmdB18Gu3Rw=
github.com/Nvveen/Gotty v0.0.0-20120604004816-cd527374f1e5/go.mod h1:lmUJ/7eu/Q8D7ML55dXQrVaamCz2vxCfdQBasLZfHKk=
github.com/andybalholm/brotli v1.1.0 h1:eLKJA0d02Lf0mVpIDgYnqXcUn0GqVmEFny3VuID1U3M=
github.com/andybalholm/brotli v1.1.0/go.mod h1:sms7XGricyQI9K10gOSf56VKKWS4oLer58Q+mhRPtnY=
github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY=
github.com/antlr/antlr4/runtime/Go/antlr/v4 v4.0.0-20230512164433-5d1fd1a340c9 h1:goHVqTbFX3AIo0tzGr14pgfAW2ZfPChKO21Z9MGf/gk=
github.com/antlr/antlr4/runtime/Go/antlr/v4 v4.0.0-20230512164433-5d1fd1a340c9/go.mod h1:pSwJ0fSY5KhvocuWSx4fz3BA8OrA1bQn+K1Eli3BRwM=
github.com/cenkalti/backoff/v4 v4.2.1 h1:y4OZtCnogmCPw98Zjyt5a6+QwPLGkiQsYW5oUqylYbM=
github.com/cenkalti/backoff/v4 v4.2.1/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE=
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc=
github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk=
github.com/cncf/udpa/go v0.0.0-20210930031921-04548b0d99d4/go.mod h1:6pvJx4me5XPnfI9Z40ddWsdw2W/uZgQLFXToKeRcDiI=
github.com/cncf/xds/go v0.0.0-20210922020428-25de7278fc84/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/containerd/continuity v0.4.3 h1:6HVkalIp+2u1ZLH1J/pYX2oBVXlJZvh1X1A7bEZ9Su8=
github.com/containerd/continuity v0.4.3/go.mod h1:F6PTNCKepoxEaXLQp3wDAjygEnImnZ/7o4JzpodfroQ=
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
github.com/creack/pty v1.1.18 h1:n56/Zwd5o6whRC5PMGretI4IdRLlmBXYNjScPaBgsbY=
github.com/creack/pty v1.1.18/go.mod h1:MOBLtS5ELjhRRrroQr9kyvTxUAFNvYEK993ew/Vr4O4=
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/distribution/reference v0.5.0 h1:/FUIFXtfc/x2gpa5/VGfiGLuOIdYa1t65IKK2OFGvA0=
github.com/distribution/reference v0.5.0/go.mod h1:BbU0aIcezP1/5jX/8MP0YiH4SdvB5Y4f/wlDRiLyi3E=
github.com/docker/cli v25.0.4+incompatible h1:DatRkJ+nrFoYL2HZUzjM5Z5sAmcA5XGp+AW0oEw2+cA=
github.com/docker/cli v25.0.4+incompatible/go.mod h1:JLrzqnKDaYBop7H2jaqPtU4hHvMKP+vjCwu2uszcLI8=
github.com/docker/docker v25.0.4+incompatible h1:XITZTrq+52tZyZxUOtFIahUf3aH367FLxJzt9vZeAF8=
github.com/docker/docker v25.0.4+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk=
github.com/docker/go-connections v0.5.0 h1:USnMq7hx7gwdVZq1L49hLXaFtUdTADjXGp+uj1Br63c=
github.com/docker/go-connections v0.5.0/go.mod h1:ov60Kzw0kKElRwhNs9UlUHAE/F9Fe6GLaXnqyDdmEXc=
github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4=
github.com/docker/go-units v0.5.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY=
github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto=
github.com/elastic/go-sysinfo v1.8.1/go.mod h1:JfllUnzoQV/JRYymbH3dO1yggI3mV2oTKSXsDHM+uIM=
github.com/elastic/go-sysinfo v1.13.1 h1:U5Jlx6c/rLkR72O8wXXXo1abnGlWGJU/wbzNJ2AfQa4=
github.com/elastic/go-sysinfo v1.13.1/go.mod h1:GKqR8bbMK/1ITnez9NIsIfXQr25aLhRJa7AfT8HpBFQ=
github.com/elastic/go-windows v1.0.0/go.mod h1:TsU0Nrp7/y3+VwE82FoZF8gC/XFg/Elz6CcloAxnPgU=
github.com/elastic/go-windows v1.0.1 h1:AlYZOldA+UJ0/2nBuqWdo90GFCgG9xuyw9SYzGUtJm0=
github.com/elastic/go-windows v1.0.1/go.mod h1:FoVvqWSun28vaDQPbj2Elfc0JahhPB7WQEGa3c814Ss=
github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98=
github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk=
github.com/envoyproxy/go-control-plane v0.10.2-0.20220325020618-49ff273808a1/go.mod h1:KJwIaB5Mv44NWtYuAOFCVOjcI94vtpEz2JU/D2v6IjE=
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
github.com/felixge/httpsnoop v1.0.3 h1:s/nj+GCswXYzN5v2DpNMuMQYe+0DDwt5WVCU6CWBdXk=
github.com/felixge/httpsnoop v1.0.3/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U=
github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
github.com/go-faster/city v1.0.1 h1:4WAxSZ3V2Ws4QRDrscLEDcibJY8uf41H6AhXDrNDcGw=
github.com/go-faster/city v1.0.1/go.mod h1:jKcUJId49qdW3L1qKHH/3wPeUstCVpVSXTM6vO3VcTw=
github.com/go-faster/errors v0.7.1 h1:MkJTnDoEdi9pDabt1dpWf7AA8/BaSYZqibYyhZ20AYg=
github.com/go-faster/errors v0.7.1/go.mod h1:5ySTjWFiphBs07IKuiL69nxdfd5+fzh1u7FPGZP2quo=
github.com/go-logr/logr v1.4.1 h1:pKouT5E8xu9zeFC39JXRDukb6JFQPXM5p5I91188VAQ=
github.com/go-logr/logr v1.4.1/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag=
github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE=
github.com/go-sql-driver/mysql v1.8.0 h1:UtktXaU2Nb64z/pLiGIxY4431SJ4/dR5cjMmlVHgnT4=
github.com/go-sql-driver/mysql v1.8.0/go.mod h1:wEBSXgmK//2ZFJyE+qWnIsVGmvmEKlqwuVSjsCm7DZg=
github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q=
github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q=
github.com/golang-jwt/jwt/v4 v4.5.0 h1:7cYmW1XlMY7h7ii7UhUyChSgS5wUJEnm9uZVTGqOWzg=
github.com/golang-jwt/jwt/v4 v4.5.0/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0=
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8=
github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA=
github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs=
github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w=
github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0=
github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY=
github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek=
github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps=
github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 h1:El6M4kTTCOh6aBiKaUGG7oYTSPP8MxqL4YI3kZKwcP4=
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510/go.mod h1:pupxD2MaaD3pAXIBCelhxNneeOaAeabZDe5s4K6zSpQ=
github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw=
github.com/hashicorp/golang-lru/v2 v2.0.7 h1:a+bsQ5rvGLjzHuww6tVxozPZFVghXaHOwFs4luLUK2k=
github.com/hashicorp/golang-lru/v2 v2.0.7/go.mod h1:QeFd9opnmA6QUJc5vARoKUSoFhyfM2/ZepoAG6RGpeM=
github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM=
github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg=
github.com/jackc/pgservicefile v0.0.0-20231201235250-de7065d80cb9 h1:L0QtFUgDarD7Fpv9jeVMgy/+Ec0mtnmYuImjTz6dtDA=
github.com/jackc/pgservicefile v0.0.0-20231201235250-de7065d80cb9/go.mod h1:5TJZWKEWniPve33vlWYSoGYefn3gLQRzjfDlhSJ9ZKM=
github.com/jackc/pgx/v5 v5.5.5 h1:amBjrZVmksIdNjxGW/IiIMzxMKZFelXbUoPNb+8sjQw=
github.com/jackc/pgx/v5 v5.5.5/go.mod h1:ez9gk+OAat140fv9ErkZDYFWmXLfV+++K0uAOiwgm1A=
github.com/jackc/puddle/v2 v2.2.1 h1:RhxXJtFG022u4ibrCSMSiu5aOq1i77R3OHKNJj77OAk=
github.com/jackc/puddle/v2 v2.2.1/go.mod h1:vriiEXHvEE654aYKXXjOvZM39qJ0q+azkZFrfEOc3H4=
github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI=
github.com/joeshaw/multierror v0.0.0-20140124173710-69b34d4ec901 h1:rp+c0RAYOWj8l6qbCUTSiRLG/iKnW3K3/QfPPuSsBt4=
github.com/joeshaw/multierror v0.0.0-20140124173710-69b34d4ec901/go.mod h1:Z86h9688Y0wesXCyonoVr47MasHilkuLMqGhRZ4Hpak=
github.com/jonboulle/clockwork v0.4.0 h1:p4Cf1aMWXnXAUh8lVfewRBx1zaTSYKrKMF2g3ST4RZ4=
github.com/jonboulle/clockwork v0.4.0/go.mod h1:xgRqUGwRcjKCO1vbZUEtSLrqKoPSsUpK7fnezOII0kc=
github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8=
github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
github.com/klauspost/compress v1.13.6/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk=
github.com/klauspost/compress v1.17.7 h1:ehO88t2UGzQK66LMdE8tibEd1ErmzZjNEqWkjLAKQQg=
github.com/klauspost/compress v1.17.7/go.mod h1:Di0epgTjJY877eYKx5yC51cX2A2Vl2ibi7bDH9ttBbw=
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
github.com/lib/pq v0.0.0-20180327071824-d34b9ff171c2 h1:hRGSmZu7j271trc9sneMrpOW7GN5ngLm8YUZIPzf394=
github.com/lib/pq v0.0.0-20180327071824-d34b9ff171c2/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo=
github.com/libsql/sqlite-antlr4-parser v0.0.0-20230802215326-5cb5bb604475 h1:6PfEMwfInASh9hkN83aR0j4W/eKaAZt/AURtXAXlas0=
github.com/libsql/sqlite-antlr4-parser v0.0.0-20230802215326-5cb5bb604475/go.mod h1:20nXSmcf0nAscrzqsXeC2/tA3KkV2eCiJqYuyAgl+ss=
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
github.com/mfridman/interpolate v0.0.2 h1:pnuTK7MQIxxFz1Gr+rjSIx9u7qVjf5VOoM/u6BbAxPY=
github.com/mfridman/interpolate v0.0.2/go.mod h1:p+7uk6oE07mpE/Ik1b8EckO0O4ZXiGAfshKBWLUM9Xg=
github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY=
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
github.com/moby/term v0.5.0 h1:xt8Q1nalod/v7BqbG21f8mQPqH+xAaC9C3N3wfWbVP0=
github.com/moby/term v0.5.0/go.mod h1:8FzsFHVUBGZdbDsJw/ot+X+d5HLUbvklYLJ9uGfcI3Y=
github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe/go.mod h1:wL8QJuTMNUDYhXwkmfOly8iTdp5TEcJFWZD2D7SIkUc=
github.com/ncruces/go-strftime v0.1.9 h1:bY0MQC28UADQmHmaF5dgpLmImcShSi2kHU9XLdhx/f4=
github.com/ncruces/go-strftime v0.1.9/go.mod h1:Fwc5htZGVVkseilnfgOVb9mKy6w1naJmn9CehxcKcls=
github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8Oi/yOhh5U=
github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM=
github.com/opencontainers/image-spec v1.1.0 h1:8SG7/vwALn54lVB/0yZ/MMwhFrPYtpEHQb2IpWsCzug=
github.com/opencontainers/image-spec v1.1.0/go.mod h1:W4s4sFTMaBeK1BQLXbG4AdM2szdn85PY75RI83NrTrM=
github.com/opencontainers/runc v1.1.12 h1:BOIssBaW1La0/qbNZHXOOa71dZfZEQOzW7dqQf3phss=
github.com/opencontainers/runc v1.1.12/go.mod h1:S+lQwSfncpBha7XTy/5lBwWgm5+y5Ma/O44Ekby9FK8=
github.com/ory/dockertest/v3 v3.10.0 h1:4K3z2VMe8Woe++invjaTB7VRyQXQy5UY+loujO4aNE4=
github.com/ory/dockertest/v3 v3.10.0/go.mod h1:nr57ZbRWMqfsdGdFNLHz5jjNdDb7VVFnzAeW1n5N1Lg=
github.com/paulmach/orb v0.11.1 h1:3koVegMC4X/WeiXYz9iswopaTwMem53NzTJuTF20JzU=
github.com/paulmach/orb v0.11.1/go.mod h1:5mULz1xQfs3bmQm63QEJA6lNGujuRafwA5S/EnuLaLU=
github.com/paulmach/protoscan v0.2.1/go.mod h1:SpcSwydNLrxUGSDvXvO0P7g7AuhJ7lcKfDlhJCDw2gY=
github.com/pierrec/lz4/v4 v4.1.21 h1:yOVMLb6qSIDP67pl/5F7RepeKYu/VmTyEXvuMI5d9mQ=
github.com/pierrec/lz4/v4 v4.1.21/go.mod h1:gZWDp/Ze/IJXGXf23ltt2EXimqmTUXEy0GFuRQyBid4=
github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/pressly/goose/v3 v3.19.2 h1:z1yuD41jS4iaqLkyjkzGkKBz4rgyz/BYtCyMMGHlgzQ=
github.com/pressly/goose/v3 v3.19.2/go.mod h1:BHkf3LzSBmO8E5FTMPupUYIpMTIh/ZuQVy+YTfhZLD4=
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
github.com/prometheus/procfs v0.0.0-20190425082905-87a4384529e0/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
github.com/prometheus/procfs v0.13.0 h1:GqzLlQyfsPbaEHaQkO7tbDlriv/4o5Hudv6OXHGKX7o=
github.com/prometheus/procfs v0.13.0/go.mod h1:cd4PFCR54QLnGKPaKGA6l+cfuNXtht43ZKY6tow0Y1g=
github.com/rekby/fixenv v0.6.1 h1:jUFiSPpajT4WY2cYuc++7Y1zWrnCxnovGCIX72PZniM=
github.com/rekby/fixenv v0.6.1/go.mod h1:/b5LRc06BYJtslRtHKxsPWFT/ySpHV+rWvzTg+XWk4c=
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE=
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo=
github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ=
github.com/rogpeppe/go-internal v1.10.0 h1:TMyTOH3F/DB16zRVcYyreMH6GnZZrwQVAoYjRBZyWFQ=
github.com/rogpeppe/go-internal v1.10.0/go.mod h1:UQnix2H7Ngw/k4C5ijL5+65zddjncjaFoBhdsK/akog=
github.com/segmentio/asm v1.2.0 h1:9BQrFxC+YOHJlTlHGkTrFWf59nbL3XnCoFLTwDCI7ys=
github.com/segmentio/asm v1.2.0/go.mod h1:BqMnlJP91P8d+4ibuonYZw9mfnzI9HfxselHZr5aAcs=
github.com/sethvargo/go-retry v0.2.4 h1:T+jHEQy/zKJf5s95UkguisicE0zuF9y7+/vgz08Ocec=
github.com/sethvargo/go-retry v0.2.4/go.mod h1:1afjQuvh7s4gflMObvjLPaWgluLLyhA1wmVZ6KLpICw=
github.com/shopspring/decimal v1.3.1 h1:2Usl1nmF/WZucqkFZhnfFYxxxu8LG21F6nPQBE5gKV8=
github.com/shopspring/decimal v1.3.1/go.mod h1:DKyhrW/HYNuLGql+MJL6WCR6knT2jwCFRcu2hWCYk4o=
github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ=
github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA=
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg=
github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk=
github.com/tursodatabase/libsql-client-go v0.0.0-20240220085343-4ae0eb9d0898 h1:1MvEhzI5pvP27e9Dzz861mxk9WzXZLSJwzOU67cKTbU=
github.com/tursodatabase/libsql-client-go v0.0.0-20240220085343-4ae0eb9d0898/go.mod h1:9bKuHS7eZh/0mJndbUOrCx8Ej3PlsRDszj4L7oVYMPQ=
github.com/vertica/vertica-sql-go v1.3.3 h1:fL+FKEAEy5ONmsvya2WH5T8bhkvY27y/Ik3ReR2T+Qw=
github.com/vertica/vertica-sql-go v1.3.3/go.mod h1:jnn2GFuv+O2Jcjktb7zyc4Utlbu9YVqpHH/lx63+1M4=
github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI=
github.com/xdg-go/scram v1.1.1/go.mod h1:RaEWvsqvNKKvBPvcKeFjrG2cJqOkHTiyTpzz23ni57g=
github.com/xdg-go/stringprep v1.0.3/go.mod h1:W3f5j4i+9rC0kuIEJL0ky1VpHXQU3ocBgklLGvcBnW8=
github.com/xeipuuv/gojsonpointer v0.0.0-20180127040702-4e3ac2762d5f/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU=
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb h1:zGWFAtiMcyryUHoUjUJX0/lt1H2+i2Ka2n+D3DImSNo=
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU=
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 h1:EzJWgHovont7NscjpAxXsDA8S8BMYve8Y5+7cuRE7R0=
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415/go.mod h1:GwrjFmJcFw6At/Gs6z4yjiIwzuJ1/+UwLxMQDVQXShQ=
github.com/xeipuuv/gojsonschema v1.2.0 h1:LhYJRs+L4fBtjZUfuSZIKGeVu0QRy8e5Xi7D17UxZ74=
github.com/xeipuuv/gojsonschema v1.2.0/go.mod h1:anYRn/JVcOK2ZgGU+IjEV4nwlhoK5sQluxsYJ78Id3Y=
github.com/ydb-platform/ydb-go-genproto v0.0.0-20240313174312-cec53bca0668 h1:R7jnhzn/38Rerzeb4i4+zprtJZEUw0i+p0R0+l4dtnc=
github.com/ydb-platform/ydb-go-genproto v0.0.0-20240313174312-cec53bca0668/go.mod h1:Er+FePu1dNUieD+XTMDduGpQuCPssK5Q4BjF+IIXJ3I=
github.com/ydb-platform/ydb-go-sdk/v3 v3.57.4 h1:Tu9kNhWLMnbkxThc+sRJvSXIauP4K0hhL6C5m4hoaLs=
github.com/ydb-platform/ydb-go-sdk/v3 v3.57.4/go.mod h1:7nt3l5aNZvbsstGGnlJZ8n7lPcKPiIUcPglADPwn0n4=
github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d/go.mod h1:rHwXgn7JulP+udvsHwJoVG1YGAP6VLg4y9I5dyZdqmA=
github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
go.mongodb.org/mongo-driver v1.11.4/go.mod h1:PTSz5yu21bkT/wXpkS7WR5f0ddqw5quethTUn9WM+2g=
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.45.0 h1:x8Z78aZx8cOF0+Kkazoc7lwUNMGy0LrzEMxTm4BbTxg=
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.45.0/go.mod h1:62CPTSry9QZtOaSsE3tOzhx6LzDhHnXJ6xHeMNNiM6Q=
go.opentelemetry.io/otel v1.24.0 h1:0LAOdjNmQeSTzGBzduGe/rU4tZhMwL5rWgtp9Ku5Jfo=
go.opentelemetry.io/otel v1.24.0/go.mod h1:W7b9Ozg4nkF5tWI5zsXkaKKDjdVjpD4oAt9Qi/MArHo=
go.opentelemetry.io/otel/metric v1.24.0 h1:6EhoGWWK28x1fbpA4tYTOWBkPefTDQnb8WSGXlc88kI=
go.opentelemetry.io/otel/metric v1.24.0/go.mod h1:VYhLe1rFfxuTXLgj4CBiyz+9WYBA8pNGJgDcSFRKBco=
go.opentelemetry.io/otel/trace v1.24.0 h1:CsKnnL4dUAr/0llH9FKuc698G04IrpWV0MQA/Y1YELI=
go.opentelemetry.io/otel/trace v1.24.0/go.mod h1:HPc3Xr/cOApsBI154IU0OI0HJexz+aw5uPdbs3UCjNU=
go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI=
go.uber.org/mock v0.4.0 h1:VcM4ZOtdbR4f6VXfiOpwpVJDL6lCReaZ6mw31wqh7KU=
go.uber.org/mock v0.4.0/go.mod h1:a6FSlNadKUHUa9IP5Vyt1zh4fC7uAwxMutEAscFbkZc=
go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
golang.org/x/crypto v0.0.0-20220622213112-05595931fe9d/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
golang.org/x/crypto v0.21.0 h1:X31++rzVUdKhX5sWmSOFZxx8UW/ldWx55cbf08iNAMA=
golang.org/x/crypto v0.21.0/go.mod h1:0BP7YvVV9gBbVKyeTG0Gyn+gZm94bibOW5BjDEYAOMs=
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
golang.org/x/exp v0.0.0-20240222234643-814bf88cf225 h1:LfspQV/FYTatPTr/3HzIcmiUFH7PGP+OQ6mgDYo3yuQ=
golang.org/x/exp v0.0.0-20240222234643-814bf88cf225/go.mod h1:CxmFvTBINI24O/j8iY7H1xHzx2i4OsyguNBmN/uPtqc=
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
golang.org/x/mod v0.16.0 h1:QX4fJ0Rr5cPQCF7O9lh9Se4pmwfwskqZfq5moyldzic=
golang.org/x/mod v0.16.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
golang.org/x/net v0.22.0 h1:9sGLhx7iRIHEiX0oAJ3MRZMUCElJgy7Br1nO+AMN3Tc=
golang.org/x/net v0.22.0/go.mod h1:JKghWKKOSdJwpW2GEx0Ja7fmaKnMsbu+MWVZTokSYmg=
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.6.0 h1:5BMeUDZ7vkXGfEr1x9B4bRcTH4lpkTkpdh0T/J+qjbQ=
golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.18.0 h1:DBdB3niSjOA/O0blCZBqDefyWNYveAYMNF1Wum0DYQ4=
golang.org/x/sys v0.18.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
golang.org/x/text v0.14.0 h1:ScX5w1eTa3QqT8oi6+ziP7dTV1S2+ALU0bI+0zXKWiQ=
golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
golang.org/x/tools v0.19.0 h1:tfGCXNR1OsFG+sVdLAitlpjAvD/I6dHDKnYrpEZUHkw=
golang.org/x/tools v0.19.0/go.mod h1:qoJWxmGSIBmAeriMx19ogtrEPrGtDbPK634QFIcLAhc=
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo=
google.golang.org/genproto/googleapis/rpc v0.0.0-20240311173647-c811ad7063a7 h1:8EeVk1VKMD+GD/neyEHGmz7pFblqPjHoi+PGQIlLx2s=
google.golang.org/genproto/googleapis/rpc v0.0.0-20240311173647-c811ad7063a7/go.mod h1:WtryC6hu0hhx87FDGxWCDptyssuo68sk10vYjF+T9fY=
google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg=
google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY=
google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk=
google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0=
google.golang.org/grpc v1.36.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU=
google.golang.org/grpc v1.47.0/go.mod h1:vN9eftEi1UMyUsIF80+uQXhHjbXYbm0uXoFCACuMGWk=
google.golang.org/grpc v1.62.1 h1:B4n+nfKzOICUXMgyrNd19h/I9oH0L1pizfk1d4zSgTk=
google.golang.org/grpc v1.62.1/go.mod h1:IWTG0VlJLCh1SkC58F7np9ka9mx/WNkjl4PGJaiq+QE=
google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8=
google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0=
google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM=
google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE=
google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo=
google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
google.golang.org/protobuf v1.28.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
google.golang.org/protobuf v1.33.0 h1:uNO2rsAINq/JlFpSdYEKIZ0uKD/R9cpdv0T+yoGwGmI=
google.golang.org/protobuf v1.33.0/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos=
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
gopkg.in/yaml.v1 v1.0.0-20140924161607-9f9df34309c0/go.mod h1:WDnlLJ4WF5VGsH/HVa3CI79GS0ol3YnhVnKP89i0kNg=
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gotest.tools/v3 v3.4.0 h1:ZazjZUfuVeZGLAmlKKuyv3IKP5orXcwtOwDQH6YVr6o=
gotest.tools/v3 v3.4.0/go.mod h1:CtbdzLSsqVhDgMtKsx03ird5YTGB3ar27v0u/yKBW5g=
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
howett.net/plist v0.0.0-20181124034731-591f970eefbb/go.mod h1:vMygbs4qMhSZSc4lCUl2OEE+rDiIIJAIdR4m7MiMcm0=
howett.net/plist v1.0.1 h1:37GdZ8tP09Q35o9ych3ehygcsL+HqKSwzctveSlarvM=
howett.net/plist v1.0.1/go.mod h1:lqaXoTrLY4hg8tnEzNru53gicrbv7rrk+2xJA/7hw9g=
modernc.org/gc/v3 v3.0.0-20240107210532-573471604cb6 h1:5D53IMaUuA5InSeMu9eJtlQXS2NxAhyWQvkKEgXZhHI=
modernc.org/gc/v3 v3.0.0-20240107210532-573471604cb6/go.mod h1:Qz0X07sNOR1jWYCrJMEnbW/X55x206Q7Vt4mz6/wHp4=
modernc.org/libc v1.41.0 h1:g9YAc6BkKlgORsUWj+JwqoB1wU3o4DE3bM3yvA3k+Gk=
modernc.org/libc v1.41.0/go.mod h1:w0eszPsiXoOnoMJgrXjglgLuDy/bt5RR4y3QzUUeodY=
modernc.org/mathutil v1.6.0 h1:fRe9+AmYlaej+64JsEEhoWuAYBkOtQiMEU7n/XgfYi4=
modernc.org/mathutil v1.6.0/go.mod h1:Ui5Q9q1TR2gFm0AQRqQUaBWFLAhQpCwNcuhBOSedWPo=
modernc.org/memory v1.7.2 h1:Klh90S215mmH8c9gO98QxQFsY+W451E8AnzjoE2ee1E=
modernc.org/memory v1.7.2/go.mod h1:NO4NVCQy0N7ln+T9ngWqOQfi7ley4vpwvARR+Hjw95E=
modernc.org/sqlite v1.29.5 h1:8l/SQKAjDtZFo9lkJLdk8g9JEOeYRG4/ghStDCCTiTE=
modernc.org/sqlite v1.29.5/go.mod h1:S02dvcmm7TnTRvGhv8IGYyLnIt7AS2KPaB1F/71p75U=
modernc.org/strutil v1.2.0 h1:agBi9dp1I+eOnxXeiZawM8F4LawKv4NzGWSaLfyeNZA=
modernc.org/strutil v1.2.0/go.mod h1:/mdcBmfOibveCTBxUl5B5l6W+TTH1FXPLHZE6bTosX0=
modernc.org/token v1.1.0 h1:Xl7Ap9dKaEs5kLoOQeQmPWevfnk/DM5qcLcYlA8ys6Y=
modernc.org/token v1.1.0/go.mod h1:UGzOrNV1mAFSEB63lOFHIpNRUVMvYTc6yu1SMY/XTDM=
nhooyr.io/websocket v1.8.10 h1:mv4p+MnGrLDcPlBoWsvPP7XCzTYMXP9F9eIGoKbgx7Q=
nhooyr.io/websocket v1.8.10/go.mod h1:rN9OFWIUwuxg4fR5tELlYC04bXYowCP9GX47ivo2l+c=

View File

@ -0,0 +1,58 @@
# Integration tests
This directory contains integration tests for the [pressly/goose/v3][goose_module] Go module. An
integration test is a test that runs against a real database (docker container) and exercises the
same driver used by the CLI.
## Why is this a separate module?
There are separate `go.mod` and `go.sum` files in this directory to allow for the use of different
dependencies. We leverage [multi-module workspaces](https://go.dev/doc/tutorial/workspaces) to glue
things together.
Namely, we want to avoid dependencies on docker, sql drivers, and other dependencies **that are**
not necessary for the core functionality of the goose library.
## Overview
There are separate migration files for each database that we support, see the [migrations
directory][migrations_dir]. Databases typically have different SQL syntax and features, so the
migration files are different.
A good set of migrations should be representative of the types of migrations users will write
typically write. This should include:
- Creating and dropping tables
- Adding and removing columns
- Creating and dropping indexes
- Inserting and deleting data
- Complex SQL statements that require special handling with `StatementBegin` and `StatementEnd`
annotations
- Statements that must run outside a transaction, annotated with `-- +goose NO TRANSACTION`
There is a common test function that applies migrations up, down and then up again.
The gold standard is the PostgreSQL migration files. We try to make other migration files as close
to the PostgreSQL files as possible, but this is not always possible or desirable.
Lastly, some tests will assert for database state after migrations are applied.
To add a new `.sql` file, you can use the following command:
```
goose -s -dir testdata/migrations/<database_name> create <filename> sql
```
- Update the database name (e.g. `postgres`)
- Update the filename name (e.g. `b`) as needed
## Limitation
Note, the integration tests are not exhaustive.
They are meantto ensure that the goose library works with the various databases that we support and
the chosen drivers. We do not test every possible combination of operations, nor do we test every
possible edge case. We rely on the unit tests in the goose package to cover library-specific logic.
[goose_module]: https://pkg.go.dev/github.com/pressly/goose/v3
[migrations_dir]: ./testdata/migrations

View File

@ -0,0 +1,195 @@
package integration
import (
"testing"
"time"
"github.com/pressly/goose/v3/database"
"github.com/pressly/goose/v3/internal/testing/testdb"
"github.com/stretchr/testify/require"
)
func TestPostgres(t *testing.T) {
t.Parallel()
db, cleanup, err := testdb.NewPostgres()
require.NoError(t, err)
t.Cleanup(cleanup)
require.NoError(t, db.Ping())
testDatabase(t, database.DialectPostgres, db, "testdata/migrations/postgres")
}
func TestClickhouse(t *testing.T) {
t.Parallel()
db, cleanup, err := testdb.NewClickHouse()
require.NoError(t, err)
t.Cleanup(cleanup)
require.NoError(t, db.Ping())
testDatabase(t, database.DialectClickHouse, db, "testdata/migrations/clickhouse")
type result struct {
customerID string `db:"customer_id"`
timestamp time.Time `db:"time_stamp"`
clickEventType string `db:"click_event_type"`
countryCode string `db:"country_code"`
sourceID int64 `db:"source_id"`
}
rows, err := db.Query(`SELECT * FROM clickstream ORDER BY customer_id`)
require.NoError(t, err)
var results []result
for rows.Next() {
var r result
err = rows.Scan(&r.customerID, &r.timestamp, &r.clickEventType, &r.countryCode, &r.sourceID)
require.NoError(t, err)
results = append(results, r)
}
require.Equal(t, len(results), 3)
require.NoError(t, rows.Close())
require.NoError(t, rows.Err())
parseTime := func(t *testing.T, s string) time.Time {
t.Helper()
tm, err := time.Parse("2006-01-02", s)
require.NoError(t, err)
return tm
}
want := []result{
{"customer1", parseTime(t, "2021-10-02"), "add_to_cart", "US", 568239},
{"customer2", parseTime(t, "2021-10-30"), "remove_from_cart", "", 0},
{"customer3", parseTime(t, "2021-11-07"), "checkout", "", 307493},
}
for i, result := range results {
require.Equal(t, result.customerID, want[i].customerID)
require.Equal(t, result.timestamp, want[i].timestamp)
require.Equal(t, result.clickEventType, want[i].clickEventType)
if result.countryCode != "" && want[i].countryCode != "" {
require.Equal(t, result.countryCode, want[i].countryCode)
}
require.Equal(t, result.sourceID, want[i].sourceID)
}
}
func TestClickhouseRemote(t *testing.T) {
t.Parallel()
db, cleanup, err := testdb.NewClickHouse()
require.NoError(t, err)
t.Cleanup(cleanup)
require.NoError(t, db.Ping())
testDatabase(t, database.DialectClickHouse, db, "testdata/migrations/clickhouse-remote")
// assert that the taxi_zone_dictionary table has been created and populated
var count int
err = db.QueryRow(`SELECT count(*) FROM taxi_zone_dictionary`).Scan(&count)
require.NoError(t, err)
require.Equal(t, 265, count)
}
func TestMySQL(t *testing.T) {
t.Parallel()
db, cleanup, err := testdb.NewMariaDB()
require.NoError(t, err)
t.Cleanup(cleanup)
require.NoError(t, db.Ping())
testDatabase(t, database.DialectMySQL, db, "testdata/migrations/mysql")
}
func TestTurso(t *testing.T) {
t.Parallel()
db, cleanup, err := testdb.NewTurso()
require.NoError(t, err)
t.Cleanup(cleanup)
require.NoError(t, db.Ping())
testDatabase(t, database.DialectTurso, db, "testdata/migrations/turso")
}
func TestVertica(t *testing.T) {
t.Parallel()
db, cleanup, err := testdb.NewVertica()
require.NoError(t, err)
t.Cleanup(cleanup)
require.NoError(t, db.Ping())
testDatabase(t, database.DialectVertica, db, "testdata/migrations/vertica")
type result struct {
TestKey int64 `db:"test_key"`
TestID string `db:"test_id"`
ValidFrom time.Time `db:"valid_from"`
ValidTo time.Time `db:"valid_to"`
IsCurrent bool `db:"is_current"`
ExternalID string `db:"external_id"`
}
rows, err := db.Query(`SELECT * FROM testing.dim_test_scd ORDER BY test_key`)
require.NoError(t, err)
var results []result
for rows.Next() {
var r result
err = rows.Scan(&r.TestKey, &r.TestID, &r.ValidFrom, &r.ValidTo, &r.IsCurrent, &r.ExternalID)
require.NoError(t, err)
results = append(results, r)
}
require.Equal(t, len(results), 3)
require.NoError(t, rows.Close())
require.NoError(t, rows.Err())
parseTime := func(t *testing.T, s string) time.Time {
t.Helper()
tm, err := time.Parse("2006-01-02", s)
require.NoError(t, err)
return tm
}
want := []result{
{
TestKey: 1,
TestID: "575a0dd4-bd97-44ac-aae0-987090181da8",
ValidFrom: parseTime(t, "2021-10-02"),
ValidTo: parseTime(t, "2021-10-03"),
IsCurrent: false,
ExternalID: "123",
},
{
TestKey: 2,
TestID: "575a0dd4-bd97-44ac-aae0-987090181da8",
ValidFrom: parseTime(t, "2021-10-03"),
ValidTo: parseTime(t, "2021-10-04"),
IsCurrent: false,
ExternalID: "456",
},
{
TestKey: 3,
TestID: "575a0dd4-bd97-44ac-aae0-987090181da8",
ValidFrom: parseTime(t, "2021-10-04"),
ValidTo: parseTime(t, "9999-12-31"),
IsCurrent: true,
ExternalID: "789",
},
}
for i, result := range results {
require.Equal(t, result.TestKey, want[i].TestKey)
require.Equal(t, result.TestID, want[i].TestID)
require.Equal(t, result.ValidFrom, want[i].ValidFrom)
require.Equal(t, result.ValidTo, want[i].ValidTo)
require.Equal(t, result.IsCurrent, want[i].IsCurrent)
require.Equal(t, result.ExternalID, want[i].ExternalID)
}
}
func TestYDB(t *testing.T) {
t.Parallel()
db, cleanup, err := testdb.NewYdb()
require.NoError(t, err)
t.Cleanup(cleanup)
require.NoError(t, db.Ping())
testDatabase(t, database.DialectYdB, db, "testdata/migrations/ydb")
}

View File

@ -0,0 +1,82 @@
package integration
import (
"context"
"database/sql"
"errors"
"os"
"path/filepath"
"testing"
"github.com/pressly/goose/v3"
"github.com/pressly/goose/v3/database"
"github.com/stretchr/testify/require"
)
type collected struct {
fullpath string
version int64
}
func collectMigrations(t *testing.T, dir string) []collected {
t.Helper()
files, err := os.ReadDir(dir)
require.NoError(t, err)
all := make([]collected, 0, len(files))
for _, f := range files {
require.False(t, f.IsDir())
v, err := goose.NumericComponent(f.Name())
require.NoError(t, err)
all = append(all, collected{
fullpath: filepath.Base(f.Name()),
version: v,
})
}
return all
}
func testDatabase(t *testing.T, dialect database.Dialect, db *sql.DB, migrationsDir string) {
t.Helper()
ctx := context.Background()
// collect all migration files from the testdata directory
wantFiles := collectMigrations(t, migrationsDir)
// initialize a new goose provider
p, err := goose.NewProvider(dialect, db, os.DirFS(migrationsDir))
require.NoError(t, err)
require.Equal(t, len(wantFiles), len(p.ListSources()), "number of migrations")
// run all up migrations
results, err := p.Up(ctx)
require.NoError(t, err)
require.Equal(t, len(wantFiles), len(results), "number of migrations applied")
for i, r := range results {
require.Equal(t, wantFiles[i].fullpath, r.Source.Path, "migration file")
require.Equal(t, wantFiles[i].version, r.Source.Version, "migration version")
}
// check the current version
currentVersion, err := p.GetDBVersion(ctx)
require.NoError(t, err)
require.Equal(t, len(wantFiles), int(currentVersion), "current version")
// run all down migrations
results, err = p.DownTo(ctx, 0)
require.NoError(t, err)
require.Equal(t, len(wantFiles), len(results), "number of migrations rolled back")
// check the current version
currentVersion, err = p.GetDBVersion(ctx)
require.NoError(t, err)
require.Equal(t, 0, int(currentVersion), "current version")
// run all up migrations one by one
for i := range len(wantFiles) {
result, err := p.UpByOne(ctx)
require.NoError(t, err)
if errors.Is(err, goose.ErrNoNextVersion) {
break
}
require.Equal(t, wantFiles[i].fullpath, result.Source.Path, "migration file")
}
// check the current version
currentVersion, err = p.GetDBVersion(ctx)
require.NoError(t, err)
require.Equal(t, len(wantFiles), int(currentVersion), "current version")
}

View File

@ -0,0 +1,417 @@
package integration
import (
"context"
"database/sql"
"errors"
"math/rand"
"os"
"sort"
"sync"
"testing"
"time"
"github.com/pressly/goose/v3"
"github.com/pressly/goose/v3/internal/testing/testdb"
"github.com/pressly/goose/v3/lock"
"github.com/stretchr/testify/require"
"golang.org/x/sync/errgroup"
)
func TestPostgresSessionLocker(t *testing.T) {
t.Parallel()
if testing.Short() {
t.Skip("skipping test in short mode.")
}
db, cleanup, err := testdb.NewPostgres()
require.NoError(t, err)
t.Cleanup(cleanup)
// Do not run subtests in parallel, because they are using the same database.
t.Run("lock_and_unlock", func(t *testing.T) {
const (
lockID int64 = 123456789
)
locker, err := lock.NewPostgresSessionLocker(
lock.WithLockID(lockID),
lock.WithLockTimeout(1, 4), // 4 second timeout
lock.WithUnlockTimeout(1, 4), // 4 second timeout
)
require.NoError(t, err)
ctx := context.Background()
conn, err := db.Conn(ctx)
require.NoError(t, err)
t.Cleanup(func() {
require.NoError(t, conn.Close())
})
err = locker.SessionLock(ctx, conn)
require.NoError(t, err)
// Check that the lock was acquired.
exists, err := existsPgLock(ctx, db, lockID)
require.NoError(t, err)
require.True(t, exists)
// Check that the lock is released.
err = locker.SessionUnlock(ctx, conn)
require.NoError(t, err)
exists, err = existsPgLock(ctx, db, lockID)
require.NoError(t, err)
require.False(t, exists)
})
t.Run("lock_close_conn_unlock", func(t *testing.T) {
locker, err := lock.NewPostgresSessionLocker(
lock.WithLockTimeout(1, 4), // 4 second timeout
lock.WithUnlockTimeout(1, 4), // 4 second timeout
)
require.NoError(t, err)
ctx := context.Background()
conn, err := db.Conn(ctx)
require.NoError(t, err)
err = locker.SessionLock(ctx, conn)
require.NoError(t, err)
exists, err := existsPgLock(ctx, db, lock.DefaultLockID)
require.NoError(t, err)
require.True(t, exists)
// Simulate a connection close.
err = conn.Close()
require.NoError(t, err)
// Check an error is returned when unlocking, because the connection is already closed.
err = locker.SessionUnlock(ctx, conn)
require.Error(t, err)
require.True(t, errors.Is(err, sql.ErrConnDone))
})
t.Run("multiple_connections", func(t *testing.T) {
const (
workers = 5
)
ch := make(chan error)
var wg sync.WaitGroup
for i := 0; i < workers; i++ {
wg.Add(1)
go func() {
defer wg.Done()
ctx := context.Background()
conn, err := db.Conn(ctx)
require.NoError(t, err)
t.Cleanup(func() {
require.NoError(t, conn.Close())
})
// Exactly one connection should acquire the lock. While the other connections
// should fail to acquire the lock and timeout.
locker, err := lock.NewPostgresSessionLocker(
lock.WithLockTimeout(1, 4), // 4 second timeout
lock.WithUnlockTimeout(1, 4), // 4 second timeout
)
require.NoError(t, err)
// NOTE, we are not unlocking the lock, because we want to test that the lock is
// released when the connection is closed.
ch <- locker.SessionLock(ctx, conn)
}()
}
go func() {
wg.Wait()
close(ch)
}()
var errors []error
for err := range ch {
if err != nil {
errors = append(errors, err)
}
}
require.Equal(t, len(errors), workers-1) // One worker succeeds, the rest fail.
for _, err := range errors {
require.Error(t, err)
require.Equal(t, err.Error(), "failed to acquire lock")
}
exists, err := existsPgLock(context.Background(), db, lock.DefaultLockID)
require.NoError(t, err)
require.True(t, exists)
})
t.Run("unlock_with_different_connection_error", func(t *testing.T) {
rng := rand.New(rand.NewSource(time.Now().UnixNano()))
randomLockID := rng.Int63n(90000) + 10000
ctx := context.Background()
locker, err := lock.NewPostgresSessionLocker(
lock.WithLockID(randomLockID),
lock.WithLockTimeout(1, 4), // 4 second timeout
lock.WithUnlockTimeout(1, 4), // 4 second timeout
)
require.NoError(t, err)
conn1, err := db.Conn(ctx)
require.NoError(t, err)
err = locker.SessionLock(ctx, conn1)
require.NoError(t, err)
t.Cleanup(func() {
// Defer the unlock with the same connection.
err = locker.SessionUnlock(ctx, conn1)
require.NoError(t, err)
require.NoError(t, conn1.Close())
})
exists, err := existsPgLock(ctx, db, randomLockID)
require.NoError(t, err)
require.True(t, exists)
// Unlock with a different connection.
conn2, err := db.Conn(ctx)
require.NoError(t, err)
t.Cleanup(func() {
require.NoError(t, conn2.Close())
})
// Check an error is returned when unlocking with a different connection.
err = locker.SessionUnlock(ctx, conn2)
require.Error(t, err)
})
}
func TestPostgresProviderLocking(t *testing.T) {
t.Parallel()
if testing.Short() {
t.Skip("skipping test in short mode.")
}
// The migrations are written in such a way they cannot be applied in parallel, they will fail
// 99.9999% of the time. This test ensures that the advisory session lock mode works as
// expected.
// TODO(mf): small improvement here is to use the SAME postgres instance but different databases
// created from a template. This will speed up the test.
db, cleanup, err := testdb.NewPostgres()
require.NoError(t, err)
t.Cleanup(cleanup)
newProvider := func() *goose.Provider {
sessionLocker, err := lock.NewPostgresSessionLocker(
lock.WithLockTimeout(5, 60), // Timeout 5min. Try every 5s up to 60 times.
)
require.NoError(t, err)
p, err := goose.NewProvider(
goose.DialectPostgres,
db,
os.DirFS("testdata/migrations/postgres"),
goose.WithSessionLocker(sessionLocker), // Use advisory session lock mode.
)
require.NoError(t, err)
return p
}
provider1 := newProvider()
provider2 := newProvider()
sources := provider1.ListSources()
maxVersion := sources[len(sources)-1].Version
// Since the lock mode is advisory session, only one of these providers is expected to apply ALL
// the migrations. The other provider should apply NO migrations. The test MUST fail if both
// providers apply migrations.
t.Run("up", func(t *testing.T) {
var g errgroup.Group
var res1, res2 int
g.Go(func() error {
ctx := context.Background()
results, err := provider1.Up(ctx)
require.NoError(t, err)
res1 = len(results)
currentVersion, err := provider1.GetDBVersion(ctx)
require.NoError(t, err)
require.Equal(t, currentVersion, maxVersion)
return nil
})
g.Go(func() error {
ctx := context.Background()
results, err := provider2.Up(ctx)
require.NoError(t, err)
res2 = len(results)
currentVersion, err := provider2.GetDBVersion(ctx)
require.NoError(t, err)
require.Equal(t, currentVersion, maxVersion)
return nil
})
require.NoError(t, g.Wait())
// One of the providers should have applied all migrations and the other should have applied
// no migrations, but with no error.
if res1 == 0 && res2 == 0 {
t.Fatal("both providers applied no migrations")
}
if res1 > 0 && res2 > 0 {
t.Fatal("both providers applied migrations")
}
})
// Reset the database and run the same test with the advisory lock mode, but apply migrations
// one-by-one.
{
_, err := provider1.DownTo(context.Background(), 0)
require.NoError(t, err)
currentVersion, err := provider1.GetDBVersion(context.Background())
require.NoError(t, err)
require.Equal(t, currentVersion, int64(0))
}
t.Run("up_by_one", func(t *testing.T) {
var g errgroup.Group
var (
mu sync.Mutex
applied []int64
)
g.Go(func() error {
for {
result, err := provider1.UpByOne(context.Background())
if err != nil {
if errors.Is(err, goose.ErrNoNextVersion) {
return nil
}
return err
}
require.NoError(t, err)
require.NotNil(t, result)
mu.Lock()
applied = append(applied, result.Source.Version)
mu.Unlock()
}
})
g.Go(func() error {
for {
result, err := provider2.UpByOne(context.Background())
if err != nil {
if errors.Is(err, goose.ErrNoNextVersion) {
return nil
}
return err
}
require.NoError(t, err)
require.NotNil(t, result)
mu.Lock()
applied = append(applied, result.Source.Version)
mu.Unlock()
}
})
require.NoError(t, g.Wait())
require.Equal(t, len(applied), len(sources))
sort.Slice(applied, func(i, j int) bool {
return applied[i] < applied[j]
})
// Each migration should have been applied up exactly once.
for i := 0; i < len(sources); i++ {
require.Equal(t, applied[i], sources[i].Version)
}
})
// Restore the database state by applying all migrations and run the same test with the advisory
// lock mode, but apply down migrations in parallel.
{
_, err := provider1.Up(context.Background())
require.NoError(t, err)
currentVersion, err := provider1.GetDBVersion(context.Background())
require.NoError(t, err)
require.Equal(t, currentVersion, maxVersion)
}
t.Run("down_to", func(t *testing.T) {
var g errgroup.Group
var res1, res2 int
g.Go(func() error {
ctx := context.Background()
results, err := provider1.DownTo(ctx, 0)
require.NoError(t, err)
res1 = len(results)
currentVersion, err := provider1.GetDBVersion(ctx)
require.NoError(t, err)
require.Equal(t, int64(0), currentVersion)
return nil
})
g.Go(func() error {
ctx := context.Background()
results, err := provider2.DownTo(ctx, 0)
require.NoError(t, err)
res2 = len(results)
currentVersion, err := provider2.GetDBVersion(ctx)
require.NoError(t, err)
require.Equal(t, int64(0), currentVersion)
return nil
})
require.NoError(t, g.Wait())
if res1 == 0 && res2 == 0 {
t.Fatal("both providers applied no migrations")
}
if res1 > 0 && res2 > 0 {
t.Fatal("both providers applied migrations")
}
})
// Restore the database state by applying all migrations and run the same test with the advisory
// lock mode, but apply down migrations one-by-one.
{
_, err := provider1.Up(context.Background())
require.NoError(t, err)
currentVersion, err := provider1.GetDBVersion(context.Background())
require.NoError(t, err)
require.Equal(t, currentVersion, maxVersion)
}
t.Run("down_by_one", func(t *testing.T) {
var g errgroup.Group
var (
mu sync.Mutex
applied []int64
)
g.Go(func() error {
for {
result, err := provider1.Down(context.Background())
if err != nil {
if errors.Is(err, goose.ErrNoNextVersion) {
return nil
}
return err
}
require.NoError(t, err)
require.NotNil(t, result)
mu.Lock()
applied = append(applied, result.Source.Version)
mu.Unlock()
}
})
g.Go(func() error {
for {
result, err := provider2.Down(context.Background())
if err != nil {
if errors.Is(err, goose.ErrNoNextVersion) {
return nil
}
return err
}
require.NoError(t, err)
require.NotNil(t, result)
mu.Lock()
applied = append(applied, result.Source.Version)
mu.Unlock()
}
})
require.NoError(t, g.Wait())
require.Equal(t, len(applied), len(sources))
sort.Slice(applied, func(i, j int) bool {
return applied[i] < applied[j]
})
// Each migration should have been applied down exactly once. Since this is sequential the
// applied down migrations should be in reverse order.
for i := len(sources) - 1; i >= 0; i-- {
require.Equal(t, applied[i], sources[i].Version)
}
})
}
func existsPgLock(ctx context.Context, db *sql.DB, lockID int64) (bool, error) {
q := `SELECT EXISTS(SELECT 1 FROM pg_locks WHERE locktype='advisory' AND ((classid::bigint<<32)|objid::bigint)=$1)`
row := db.QueryRowContext(ctx, q, lockID)
var exists bool
if err := row.Scan(&exists); err != nil {
return false, err
}
return exists, nil
}

View File

@ -0,0 +1,23 @@
-- +goose Up
-- +goose StatementBegin
CREATE TABLE owners (
owner_id BIGINT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
owner_name VARCHAR(255) NOT NULL,
owner_type ENUM('user', 'organization') NOT NULL
);
CREATE TABLE IF NOT EXISTS repos (
repo_id BIGINT UNSIGNED AUTO_INCREMENT NOT NULL,
repo_full_name VARCHAR(255) NOT NULL,
repo_owner_id BIGINT UNSIGNED NOT NULL,
PRIMARY KEY (repo_id),
FOREIGN KEY (repo_owner_id) REFERENCES owners(owner_id) ON DELETE CASCADE
);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DROP TABLE IF EXISTS repos;
DROP TABLE IF EXISTS owners;
-- +goose StatementEnd

View File

@ -0,0 +1,22 @@
-- +goose Up
-- +goose StatementBegin
INSERT INTO owners (owner_name, owner_type)
VALUES
('lucas', 'user'),
('space', 'organization');
-- +goose StatementEnd
INSERT INTO owners (owner_name, owner_type)
VALUES
('james', 'user'),
('pressly', 'organization');
INSERT INTO repos (repo_full_name, repo_owner_id)
VALUES
('james/rover', (SELECT owner_id FROM owners WHERE owner_name = 'james')),
('pressly/goose', (SELECT owner_id FROM owners WHERE owner_name = 'pressly'));
-- +goose Down
-- +goose StatementBegin
DELETE FROM owners;
-- +goose StatementEnd

View File

@ -0,0 +1,17 @@
-- +goose Up
-- +goose StatementBegin
ALTER TABLE repos
ADD COLUMN IF NOT EXISTS homepage_url TEXT;
ALTER TABLE repos
ADD COLUMN is_private BOOLEAN NOT NULL DEFAULT FALSE;
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
ALTER TABLE repos
DROP COLUMN IF EXISTS homepage_url;
ALTER TABLE repos
DROP COLUMN IF EXISTS is_private;
-- +goose StatementEnd

View File

@ -0,0 +1 @@
-- +goose Up

View File

@ -0,0 +1,8 @@
-- +goose NO TRANSACTION
-- +goose Up
CREATE UNIQUE INDEX owners_owner_name_idx ON owners(owner_name);
-- +goose Down
DROP INDEX IF EXISTS owners_owner_name_idx ON owners;

View File

@ -0,0 +1,36 @@
-- +goose up
-- +goose statementbegin
CREATE OR REPLACE PROCEDURE insert_repository(
IN p_repo_full_name VARCHAR(255),
IN p_owner_name VARCHAR(255),
IN p_owner_type VARCHAR(20)
)
BEGIN
DECLARE v_owner_id BIGINT;
DECLARE v_repo_id BIGINT;
-- Check if the owner already exists
SELECT owner_id INTO v_owner_id
FROM owners
WHERE owner_name = p_owner_name AND owner_type = p_owner_type;
-- If the owner does not exist, insert a new owner
IF v_owner_id IS NULL THEN
INSERT INTO owners (owner_name, owner_type)
VALUES (p_owner_name, p_owner_type);
SET v_owner_id = LAST_INSERT_ID();
END IF;
-- Insert the repository using the obtained owner_id
INSERT INTO repos (repo_full_name, repo_owner_id)
VALUES (p_repo_full_name, v_owner_id);
-- No explicit return needed in procedures
END;
-- +goose statementend
-- +goose down
DROP PROCEDURE IF EXISTS insert_repository;

View File

@ -22,4 +22,4 @@ CREATE TABLE IF NOT EXISTS repos (
DROP TABLE IF EXISTS repos;
DROP TABLE IF EXISTS owners;
DROP TYPE owner_type;
-- +goose StatementEnd
-- +goose StatementEnd

View File

@ -1,13 +1,16 @@
-- +goose Up
-- +goose StatementBegin
INSERT INTO owners(owner_name, owner_type)
VALUES ('lucas', 'user'), ('space', 'organization');
-- +goose StatementEnd
INSERT INTO owners(owner_name, owner_type)
VALUES ('james', 'user'), ('pressly', 'organization');
INSERT INTO repos(repo_full_name, repo_owner_id)
VALUES ('james/rover', 3), ('pressly/goose', 4);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DELETE FROM owners WHERE owner_name IN ('james', 'pressly');
DELETE FROM owners;
-- +goose StatementEnd

View File

@ -0,0 +1 @@
-- +goose Up

View File

@ -1,7 +1,7 @@
-- +goose NO TRANSACTION
-- +goose Up
DROP INDEX IF EXISTS owners_owner_name_idx;
CREATE UNIQUE INDEX CONCURRENTLY ON owners(owner_name);
-- +goose Down
CREATE UNIQUE INDEX CONCURRENTLY ON owners(owner_name);
DROP INDEX IF EXISTS owners_owner_name_idx;

View File

@ -0,0 +1,36 @@
-- +goose up
-- +goose statementbegin
CREATE OR REPLACE FUNCTION insert_repository(
p_repo_full_name TEXT,
p_owner_name TEXT,
p_owner_type OWNER_TYPE
) RETURNS VOID AS $$
DECLARE
v_owner_id BIGINT;
v_repo_id BIGINT;
BEGIN
-- Check if the owner already exists
SELECT owner_id INTO v_owner_id
FROM owners
WHERE owner_name = p_owner_name AND owner_type = p_owner_type;
-- If the owner does not exist, insert a new owner
IF v_owner_id IS NULL THEN
INSERT INTO owners (owner_name, owner_type)
VALUES (p_owner_name, p_owner_type)
RETURNING owner_id INTO v_owner_id;
END IF;
-- Insert the repository using the obtained owner_id
INSERT INTO repos (repo_full_name, repo_owner_id)
VALUES (p_repo_full_name, v_owner_id)
RETURNING repo_id INTO v_repo_id;
-- Commit the transaction
COMMIT;
END;
$$ LANGUAGE plpgsql;
-- +goose statementend
-- +goose down
DROP FUNCTION IF EXISTS insert_repository(TEXT, TEXT, OWNER_TYPE);

View File

@ -0,0 +1,20 @@
-- +goose Up
-- +goose StatementBegin
CREATE TABLE owners (
owner_id INTEGER PRIMARY KEY AUTOINCREMENT,
owner_name TEXT NOT NULL,
owner_type TEXT CHECK(owner_type IN ('user', 'organization')) NOT NULL
);
CREATE TABLE IF NOT EXISTS repos (
repo_id INTEGER PRIMARY KEY AUTOINCREMENT,
repo_full_name TEXT NOT NULL,
repo_owner_id INTEGER NOT NULL REFERENCES owners(owner_id) ON DELETE CASCADE
);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DROP TABLE IF EXISTS repos;
DROP TABLE IF EXISTS owners;
-- +goose StatementEnd

View File

@ -0,0 +1,19 @@
-- +goose Up
-- +goose StatementBegin
INSERT INTO owners(owner_name, owner_type)
VALUES
('lucas', 'user'),
('space', 'organization'),
('james', 'user'),
('pressly', 'organization');
-- +goose StatementEnd
INSERT INTO repos(repo_full_name, repo_owner_id)
VALUES
('james/rover', (SELECT owner_id FROM owners WHERE owner_name = 'james')),
('pressly/goose', (SELECT owner_id FROM owners WHERE owner_name = 'pressly'));
-- +goose Down
-- +goose StatementBegin
DELETE FROM owners;
-- +goose StatementEnd

View File

@ -0,0 +1,17 @@
-- +goose Up
-- +goose StatementBegin
ALTER TABLE repos
ADD COLUMN homepage_url TEXT;
ALTER TABLE repos
ADD COLUMN is_private BOOLEAN DEFAULT 0;
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
ALTER TABLE repos
DROP COLUMN homepage_url;
ALTER TABLE repos
DROP COLUMN is_private;
-- +goose StatementEnd

View File

@ -0,0 +1 @@
-- +goose Up

View File

@ -0,0 +1,9 @@
-- +goose NO TRANSACTION
-- +goose Up
CREATE UNIQUE INDEX IF NOT EXISTS idx_owners_owner_name ON owners(owner_name);
-- +goose Down
DROP INDEX IF EXISTS idx_owners_owner_name;

View File

@ -1,3 +1,4 @@
-- +goose Up
-- +goose StatementBegin
INSERT INTO owners(owner_id, owner_name, owner_type)

View File

@ -14,7 +14,7 @@ import (
const (
// https://hub.docker.com/_/postgres
POSTGRES_IMAGE = "postgres"
POSTGRES_VERSION = "14-alpine"
POSTGRES_VERSION = "16-alpine"
POSTGRES_DB = "testdb"
POSTGRES_USER = "postgres"

View File

@ -12,9 +12,9 @@ import (
)
const (
// ghcr.io/tursodatabase/libsql-server:v0.22.10
// ghcr.io/tursodatabase/libsql-server:v0.23.7
TURSO_IMAGE = "ghcr.io/tursodatabase/libsql-server"
TURSO_VERSION = "v0.22.10"
TURSO_VERSION = "v0.23.7"
TURSO_PORT = "8080"
)

View File

@ -1,6 +1,7 @@
package testdb
import (
"context"
"database/sql"
"fmt"
"log"
@ -9,13 +10,14 @@ import (
"github.com/ory/dockertest/v3"
"github.com/ory/dockertest/v3/docker"
"github.com/sethvargo/go-retry"
_ "github.com/vertica/vertica-sql-go"
)
const (
// https://hub.docker.com/r/vertica/vertica-ce
VERTICA_IMAGE = "vertica/vertica-ce"
VERTICA_VERSION = "23.3.0-0"
VERTICA_VERSION = "24.1.0-0"
VERTICA_DB = "testdb"
)
@ -75,21 +77,21 @@ func newVertica(opts ...OptionsFunc) (*sql.DB, func(), error) {
)
var db *sql.DB
// Give vertica a head start since the container takes a little bit to start up.
time.Sleep(time.Second * 15)
// Exponential backoff-retry, because the application in the container
// might not be ready to accept connections yet.
if err := pool.Retry(
func() error {
var err error
db, err = sql.Open("vertica", verticaInfo)
if err != nil {
return err
}
return db.Ping()
},
); err != nil {
backoff := retry.WithMaxDuration(1*time.Minute, retry.NewConstant(2*time.Second))
if err := retry.Do(context.Background(), backoff, func(ctx context.Context) error {
var err error
db, err = sql.Open("vertica", verticaInfo)
if err != nil {
return retry.RetryableError(fmt.Errorf("failed to open vertica connection: %v", err))
}
if err := db.Ping(); err != nil {
return retry.RetryableError(fmt.Errorf("failed to ping vertica: %v", err))
}
return nil
}); err != nil {
return nil, cleanup, fmt.Errorf("could not connect to docker database: %v", err)
}
return db, cleanup, nil

View File

@ -13,7 +13,7 @@ import (
"github.com/ory/dockertest/v3/docker"
"github.com/ydb-platform/ydb-go-sdk/v3"
"github.com/ydb-platform/ydb-go-sdk/v3/balancers"
ydbLog "github.com/ydb-platform/ydb-go-sdk/v3/log"
ydblog "github.com/ydb-platform/ydb-go-sdk/v3/log"
"github.com/ydb-platform/ydb-go-sdk/v3/trace"
)
@ -102,7 +102,7 @@ func newYdb(opts ...OptionsFunc) (*sql.DB, func(), error) {
}
if option.debug {
opts = append(opts, ydb.WithLogger(ydbLog.Default(os.Stdout), trace.DetailsAll, ydbLog.WithLogQuery()))
opts = append(opts, ydb.WithLogger(ydblog.Default(os.Stdout), trace.DetailsAll, ydblog.WithLogQuery()))
}
nativeDriver, err := ydb.Open(ctx, dsn, opts...)

View File

@ -1,172 +0,0 @@
package lock_test
import (
"context"
"database/sql"
"errors"
"math/rand"
"sync"
"testing"
"time"
"github.com/pressly/goose/v3/internal/check"
"github.com/pressly/goose/v3/internal/testdb"
"github.com/pressly/goose/v3/lock"
)
func TestPostgresSessionLocker(t *testing.T) {
t.Parallel()
if testing.Short() {
t.Skip("skip long running test")
}
db, cleanup, err := testdb.NewPostgres()
check.NoError(t, err)
t.Cleanup(cleanup)
// Do not run tests in parallel, because they are using the same database.
t.Run("lock_and_unlock", func(t *testing.T) {
const (
lockID int64 = 123456789
)
locker, err := lock.NewPostgresSessionLocker(
lock.WithLockID(lockID),
lock.WithLockTimeout(1, 4), // 4 second timeout
lock.WithUnlockTimeout(1, 4), // 4 second timeout
)
check.NoError(t, err)
ctx := context.Background()
conn, err := db.Conn(ctx)
check.NoError(t, err)
t.Cleanup(func() {
check.NoError(t, conn.Close())
})
err = locker.SessionLock(ctx, conn)
check.NoError(t, err)
// Check that the lock was acquired.
exists, err := existsPgLock(ctx, db, lockID)
check.NoError(t, err)
check.Bool(t, exists, true)
// Check that the lock is released.
err = locker.SessionUnlock(ctx, conn)
check.NoError(t, err)
exists, err = existsPgLock(ctx, db, lockID)
check.NoError(t, err)
check.Bool(t, exists, false)
})
t.Run("lock_close_conn_unlock", func(t *testing.T) {
locker, err := lock.NewPostgresSessionLocker(
lock.WithLockTimeout(1, 4), // 4 second timeout
lock.WithUnlockTimeout(1, 4), // 4 second timeout
)
check.NoError(t, err)
ctx := context.Background()
conn, err := db.Conn(ctx)
check.NoError(t, err)
err = locker.SessionLock(ctx, conn)
check.NoError(t, err)
exists, err := existsPgLock(ctx, db, lock.DefaultLockID)
check.NoError(t, err)
check.Bool(t, exists, true)
// Simulate a connection close.
err = conn.Close()
check.NoError(t, err)
// Check an error is returned when unlocking, because the connection is already closed.
err = locker.SessionUnlock(ctx, conn)
check.HasError(t, err)
check.Bool(t, errors.Is(err, sql.ErrConnDone), true)
})
t.Run("multiple_connections", func(t *testing.T) {
const (
workers = 5
)
ch := make(chan error)
var wg sync.WaitGroup
for i := 0; i < workers; i++ {
wg.Add(1)
go func() {
defer wg.Done()
ctx := context.Background()
conn, err := db.Conn(ctx)
check.NoError(t, err)
t.Cleanup(func() {
check.NoError(t, conn.Close())
})
// Exactly one connection should acquire the lock. While the other connections
// should fail to acquire the lock and timeout.
locker, err := lock.NewPostgresSessionLocker(
lock.WithLockTimeout(1, 4), // 4 second timeout
lock.WithUnlockTimeout(1, 4), // 4 second timeout
)
check.NoError(t, err)
// NOTE, we are not unlocking the lock, because we want to test that the lock is
// released when the connection is closed.
ch <- locker.SessionLock(ctx, conn)
}()
}
go func() {
wg.Wait()
close(ch)
}()
var errors []error
for err := range ch {
if err != nil {
errors = append(errors, err)
}
}
check.Equal(t, len(errors), workers-1) // One worker succeeds, the rest fail.
for _, err := range errors {
check.HasError(t, err)
check.Equal(t, err.Error(), "failed to acquire lock")
}
exists, err := existsPgLock(context.Background(), db, lock.DefaultLockID)
check.NoError(t, err)
check.Bool(t, exists, true)
})
t.Run("unlock_with_different_connection_error", func(t *testing.T) {
rng := rand.New(rand.NewSource(time.Now().UnixNano()))
randomLockID := rng.Int63n(90000) + 10000
ctx := context.Background()
locker, err := lock.NewPostgresSessionLocker(
lock.WithLockID(randomLockID),
lock.WithLockTimeout(1, 4), // 4 second timeout
lock.WithUnlockTimeout(1, 4), // 4 second timeout
)
check.NoError(t, err)
conn1, err := db.Conn(ctx)
check.NoError(t, err)
err = locker.SessionLock(ctx, conn1)
check.NoError(t, err)
t.Cleanup(func() {
// Defer the unlock with the same connection.
err = locker.SessionUnlock(ctx, conn1)
check.NoError(t, err)
check.NoError(t, conn1.Close())
})
exists, err := existsPgLock(ctx, db, randomLockID)
check.NoError(t, err)
check.Bool(t, exists, true)
// Unlock with a different connection.
conn2, err := db.Conn(ctx)
check.NoError(t, err)
t.Cleanup(func() {
check.NoError(t, conn2.Close())
})
// Check an error is returned when unlocking with a different connection.
err = locker.SessionUnlock(ctx, conn2)
check.HasError(t, err)
})
}
func existsPgLock(ctx context.Context, db *sql.DB, lockID int64) (bool, error) {
q := `SELECT EXISTS(SELECT 1 FROM pg_locks WHERE locktype='advisory' AND ((classid::bigint<<32)|objid::bigint)=$1)`
row := db.QueryRowContext(ctx, q, lockID)
var exists bool
if err := row.Scan(&exists); err != nil {
return false, err
}
return exists, nil
}

View File

@ -11,7 +11,6 @@ import (
"os"
"path/filepath"
"reflect"
"sort"
"sync"
"testing"
"testing/fstest"
@ -19,9 +18,6 @@ import (
"github.com/pressly/goose/v3"
"github.com/pressly/goose/v3/database"
"github.com/pressly/goose/v3/internal/check"
"github.com/pressly/goose/v3/internal/testdb"
"github.com/pressly/goose/v3/lock"
"golang.org/x/sync/errgroup"
)
func TestProviderRun(t *testing.T) {
@ -924,246 +920,6 @@ func TestGoOnly(t *testing.T) {
})
}
func TestLockModeAdvisorySession(t *testing.T) {
t.Parallel()
if testing.Short() {
t.Skip("skip long running test")
}
// The migrations are written in such a way that they cannot be applied concurrently, they will
// fail 99.9999% of the time. This test ensures that the advisory session lock mode works as
// expected.
// TODO(mf): small improvement here is to use the SAME postgres instance but different databases
// created from a template. This will speed up the test.
db, cleanup, err := testdb.NewPostgres()
check.NoError(t, err)
t.Cleanup(cleanup)
newProvider := func() *goose.Provider {
sessionLocker, err := lock.NewPostgresSessionLocker(
lock.WithLockTimeout(5, 60), // Timeout 5min. Try every 5s up to 60 times.
)
check.NoError(t, err)
p, err := goose.NewProvider(
goose.DialectPostgres,
db,
os.DirFS("testdata/migrations"),
goose.WithSessionLocker(sessionLocker), // Use advisory session lock mode.
)
check.NoError(t, err)
return p
}
provider1 := newProvider()
provider2 := newProvider()
sources := provider1.ListSources()
maxVersion := sources[len(sources)-1].Version
// Since the lock mode is advisory session, only one of these providers is expected to apply ALL
// the migrations. The other provider should apply NO migrations. The test MUST fail if both
// providers apply migrations.
t.Run("up", func(t *testing.T) {
var g errgroup.Group
var res1, res2 int
g.Go(func() error {
ctx := context.Background()
results, err := provider1.Up(ctx)
check.NoError(t, err)
res1 = len(results)
currentVersion, err := provider1.GetDBVersion(ctx)
check.NoError(t, err)
check.Number(t, currentVersion, maxVersion)
return nil
})
g.Go(func() error {
ctx := context.Background()
results, err := provider2.Up(ctx)
check.NoError(t, err)
res2 = len(results)
currentVersion, err := provider2.GetDBVersion(ctx)
check.NoError(t, err)
check.Number(t, currentVersion, maxVersion)
return nil
})
check.NoError(t, g.Wait())
// One of the providers should have applied all migrations and the other should have applied
// no migrations, but with no error.
if res1 == 0 && res2 == 0 {
t.Fatal("both providers applied no migrations")
}
if res1 > 0 && res2 > 0 {
t.Fatal("both providers applied migrations")
}
})
// Reset the database and run the same test with the advisory lock mode, but apply migrations
// one-by-one.
{
_, err := provider1.DownTo(context.Background(), 0)
check.NoError(t, err)
currentVersion, err := provider1.GetDBVersion(context.Background())
check.NoError(t, err)
check.Number(t, currentVersion, 0)
}
t.Run("up_by_one", func(t *testing.T) {
var g errgroup.Group
var (
mu sync.Mutex
applied []int64
)
g.Go(func() error {
for {
result, err := provider1.UpByOne(context.Background())
if err != nil {
if errors.Is(err, goose.ErrNoNextVersion) {
return nil
}
return err
}
check.NoError(t, err)
check.Bool(t, result != nil, true)
mu.Lock()
applied = append(applied, result.Source.Version)
mu.Unlock()
}
})
g.Go(func() error {
for {
result, err := provider2.UpByOne(context.Background())
if err != nil {
if errors.Is(err, goose.ErrNoNextVersion) {
return nil
}
return err
}
check.NoError(t, err)
check.Bool(t, result != nil, true)
mu.Lock()
applied = append(applied, result.Source.Version)
mu.Unlock()
}
})
check.NoError(t, g.Wait())
check.Number(t, len(applied), len(sources))
sort.Slice(applied, func(i, j int) bool {
return applied[i] < applied[j]
})
// Each migration should have been applied up exactly once.
for i := 0; i < len(sources); i++ {
check.Number(t, applied[i], sources[i].Version)
}
})
// Restore the database state by applying all migrations and run the same test with the advisory
// lock mode, but apply down migrations in parallel.
{
_, err := provider1.Up(context.Background())
check.NoError(t, err)
currentVersion, err := provider1.GetDBVersion(context.Background())
check.NoError(t, err)
check.Number(t, currentVersion, maxVersion)
}
t.Run("down_to", func(t *testing.T) {
var g errgroup.Group
var res1, res2 int
g.Go(func() error {
ctx := context.Background()
results, err := provider1.DownTo(ctx, 0)
check.NoError(t, err)
res1 = len(results)
currentVersion, err := provider1.GetDBVersion(ctx)
check.NoError(t, err)
check.Number(t, currentVersion, 0)
return nil
})
g.Go(func() error {
ctx := context.Background()
results, err := provider2.DownTo(ctx, 0)
check.NoError(t, err)
res2 = len(results)
currentVersion, err := provider2.GetDBVersion(ctx)
check.NoError(t, err)
check.Number(t, currentVersion, 0)
return nil
})
check.NoError(t, g.Wait())
if res1 == 0 && res2 == 0 {
t.Fatal("both providers applied no migrations")
}
if res1 > 0 && res2 > 0 {
t.Fatal("both providers applied migrations")
}
})
// Restore the database state by applying all migrations and run the same test with the advisory
// lock mode, but apply down migrations one-by-one.
{
_, err := provider1.Up(context.Background())
check.NoError(t, err)
currentVersion, err := provider1.GetDBVersion(context.Background())
check.NoError(t, err)
check.Number(t, currentVersion, maxVersion)
}
t.Run("down_by_one", func(t *testing.T) {
var g errgroup.Group
var (
mu sync.Mutex
applied []int64
)
g.Go(func() error {
for {
result, err := provider1.Down(context.Background())
if err != nil {
if errors.Is(err, goose.ErrNoNextVersion) {
return nil
}
return err
}
check.NoError(t, err)
check.Bool(t, result != nil, true)
mu.Lock()
applied = append(applied, result.Source.Version)
mu.Unlock()
}
})
g.Go(func() error {
for {
result, err := provider2.Down(context.Background())
if err != nil {
if errors.Is(err, goose.ErrNoNextVersion) {
return nil
}
return err
}
check.NoError(t, err)
check.Bool(t, result != nil, true)
mu.Lock()
applied = append(applied, result.Source.Version)
mu.Unlock()
}
})
check.NoError(t, g.Wait())
check.Number(t, len(applied), len(sources))
sort.Slice(applied, func(i, j int) bool {
return applied[i] < applied[j]
})
// Each migration should have been applied down exactly once. Since this is sequential the
// applied down migrations should be in reverse order.
for i := len(sources) - 1; i >= 0; i-- {
check.Number(t, applied[i], sources[i].Version)
}
})
}
func newDBFn(query string) func(context.Context, *sql.DB) error {
return func(ctx context.Context, db *sql.DB) error {
_, err := db.ExecContext(ctx, query)

View File

@ -1,157 +0,0 @@
package clickhouse_test
import (
"log"
"os"
"path/filepath"
"testing"
"time"
"github.com/pressly/goose/v3"
"github.com/pressly/goose/v3/internal/check"
"github.com/pressly/goose/v3/internal/testdb"
)
func TestMain(m *testing.M) {
if err := goose.SetDialect("clickhouse"); err != nil {
log.Fatal(err)
}
os.Exit(m.Run())
}
func TestClickUpDownAll(t *testing.T) {
t.Parallel()
migrationDir := filepath.Join("testdata", "migrations")
db, cleanup, err := testdb.NewClickHouse()
check.NoError(t, err)
t.Cleanup(cleanup)
/*
This test applies all up migrations, asserts we have all the entries in
the versions table, applies all down migration and asserts we have zero
migrations applied.
ClickHouse performs UPDATES and DELETES asynchronously,
but we can best-effort check mutations and their progress.
This is especially important for down migrations where rows are deleted
from the versions table.
For the sake of testing, there might be a way to modifying the server
(or queries) to perform all operations synchronously?
Ref: https://clickhouse.com/docs/en/operations/system-tables/mutations/
Ref: https://clickhouse.com/docs/en/sql-reference/statements/alter/#mutations
Ref: https://clickhouse.com/blog/how-to-update-data-in-click-house/
*/
// Collect migrations so we don't have to hard-code the currentVersion
// in an assertion later in the test.
migrations, err := goose.CollectMigrations(migrationDir, 0, goose.MaxVersion)
check.NoError(t, err)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 0)
err = goose.Up(db, migrationDir)
check.NoError(t, err)
currentVersion, err = goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, len(migrations))
err = goose.DownTo(db, migrationDir, 0)
check.NoError(t, err)
currentVersion, err = goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 0)
}
func TestClickHouseFirstThree(t *testing.T) {
t.Parallel()
migrationDir := filepath.Join("testdata", "migrations")
db, cleanup, err := testdb.NewClickHouse()
check.NoError(t, err)
t.Cleanup(cleanup)
err = goose.Up(db, migrationDir)
check.NoError(t, err)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 3)
type result struct {
customerID string `db:"customer_id"`
timestamp time.Time `db:"time_stamp"`
clickEventType string `db:"click_event_type"`
countryCode string `db:"country_code"`
sourceID int64 `db:"source_id"`
}
rows, err := db.Query(`SELECT * FROM clickstream ORDER BY customer_id`)
check.NoError(t, err)
var results []result
for rows.Next() {
var r result
err = rows.Scan(&r.customerID, &r.timestamp, &r.clickEventType, &r.countryCode, &r.sourceID)
check.NoError(t, err)
results = append(results, r)
}
check.Number(t, len(results), 3)
check.NoError(t, rows.Close())
check.NoError(t, rows.Err())
parseTime := func(t *testing.T, s string) time.Time {
t.Helper()
tm, err := time.Parse("2006-01-02", s)
check.NoError(t, err)
return tm
}
want := []result{
{"customer1", parseTime(t, "2021-10-02"), "add_to_cart", "US", 568239},
{"customer2", parseTime(t, "2021-10-30"), "remove_from_cart", "", 0},
{"customer3", parseTime(t, "2021-11-07"), "checkout", "", 307493},
}
for i, result := range results {
check.Equal(t, result.customerID, want[i].customerID)
check.Equal(t, result.timestamp, want[i].timestamp)
check.Equal(t, result.clickEventType, want[i].clickEventType)
if result.countryCode != "" && want[i].countryCode != "" {
check.Equal(t, result.countryCode, want[i].countryCode)
}
check.Number(t, result.sourceID, want[i].sourceID)
}
}
func TestRemoteImportMigration(t *testing.T) {
t.Parallel()
// TODO(mf): use TestMain and create a proper "long" or "remote" flag.
if !testing.Short() {
t.Skip("skipping test")
}
// This test is using a remote dataset from an s3 bucket:
// https://datasets-documentation.s3.eu-west-3.amazonaws.com/nyc-taxi/taxi_zone_lookup.csv
// From this tutorial: https://clickhouse.com/docs/en/tutorial/
// Note, these files are backed up in this repository in:
// tests/clickhouse/testdata/backup-files/taxi_zone_lookup.csv
// We may want to host this ourselves. Or just don't bother with SOURCE(HTTP(URL..
// and craft a long INSERT statement.
migrationDir := filepath.Join("testdata", "migrations-remote")
db, cleanup, err := testdb.NewClickHouse()
check.NoError(t, err)
t.Cleanup(cleanup)
err = goose.Up(db, migrationDir)
check.NoError(t, err)
_, err = goose.GetDBVersion(db)
check.NoError(t, err)
var count int
err = db.QueryRow(`SELECT count(*) FROM taxi_zone_dictionary`).Scan(&count)
check.NoError(t, err)
check.Number(t, count, 265)
}

View File

@ -1,380 +0,0 @@
package e2e
import (
"database/sql"
"testing"
"github.com/pressly/goose/v3"
"github.com/pressly/goose/v3/internal/check"
)
func TestNotAllowMissing(t *testing.T) {
t.Parallel()
// Create and apply first 5 migrations.
db := setupTestDB(t, 5)
// Developer A and B check out the "main" branch which is currently
// on version 5. Developer A mistakenly creates migration 7 and commits.
// Developer B did not pull the latest changes and commits migration 6. Oops.
// Developer A - migration 7 (mistakenly applied)
migrations, err := goose.CollectMigrations(migrationsDir, 0, 7)
check.NoError(t, err)
err = migrations[6].Up(db)
check.NoError(t, err)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, current, 7)
// Developer B - migration 6 (missing) and 8 (new)
// This should raise an error. By default goose does not allow missing (out-of-order)
// migrations, which means halt if a missing migration is detected.
err = goose.Up(db, migrationsDir)
check.HasError(t, err)
check.Contains(t, err.Error(), "missing migrations")
// Confirm db version is unchanged.
current, err = goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, current, 7)
}
func TestAllowMissingUpWithRedo(t *testing.T) {
t.Parallel()
// Create and apply first 5 migrations.
db := setupTestDB(t, 5)
migrations, err := goose.CollectMigrations(migrationsDir, 0, goose.MaxVersion)
check.NoError(t, err)
if len(migrations) == 0 {
t.Fatalf("got zero migrations")
}
// Migration 7
{
migrations, err := goose.CollectMigrations(migrationsDir, 0, 7)
check.NoError(t, err)
err = migrations[6].Up(db)
check.NoError(t, err)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, current, 7)
// Redo the previous Up migration and re-apply it.
err = goose.Redo(db, migrationsDir)
check.NoError(t, err)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, migrations[6].Version)
}
// Migration 6
{
err = goose.UpByOne(db, migrationsDir, goose.WithAllowMissing())
check.NoError(t, err)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 6)
err = goose.Redo(db, migrationsDir)
check.NoError(t, err)
currentVersion, err = goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 6)
}
}
func TestNowAllowMissingUpByOne(t *testing.T) {
t.Parallel()
// Create and apply first 5 migrations.
db := setupTestDB(t, 5)
/*
Developer A and B simultaneously check out the "main" currently on version 5.
Developer A mistakenly creates migration 7 and commits.
Developer B did not pull the latest changes and commits migration 6. Oops.
If goose is set to allow missing migrations, then 6 should be applied
after 7.
*/
// Developer A - migration 7 (mistakenly applied)
{
migrations, err := goose.CollectMigrations(migrationsDir, 0, 7)
check.NoError(t, err)
err = migrations[6].Up(db)
check.NoError(t, err)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, current, 7)
}
// Developer B - migration 6
{
// By default, this should raise an error.
err := goose.UpByOne(db, migrationsDir)
// error: found 1 missing migrations
check.HasError(t, err)
check.Contains(t, err.Error(), "missing migrations")
count, err := getGooseVersionCount(db, goose.TableName())
check.NoError(t, err)
check.Number(t, count, 6)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
// Expecting max(version_id) to be 7
check.Number(t, current, 7)
}
}
func TestAllowMissingUpWithReset(t *testing.T) {
t.Parallel()
// Create and apply first 5 migrations.
db := setupTestDB(t, 5)
/*
Developer A and B simultaneously check out the "main" currently on version 5.
Developer A mistakenly creates migration 7 and commits.
Developer B did not pull the latest changes and commits migration 6. Oops.
If goose is set to allow missing migrations, then 6 should be applied
after 7.
*/
// Developer A - migration 7 (mistakenly applied)
{
migrations, err := goose.CollectMigrations(migrationsDir, 0, 7)
check.NoError(t, err)
err = migrations[6].Up(db)
check.NoError(t, err)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, current, 7)
}
// Developer B - migration 6 (missing) and 8 (new)
{
// By default, attempting to apply this migration will raise an error.
// If goose is set to "allow missing" migrations then it should get applied.
err := goose.Up(db, migrationsDir, goose.WithAllowMissing())
// Applying missing migration should return no error when allow-missing=true
check.NoError(t, err)
// Avoid hard-coding total and max, instead resolve it from the testdata migrations.
// In other words, we applied 1..5,7,6,8 and this test shouldn't care
// about migration 9 and onwards.
allMigrations, err := goose.CollectMigrations(migrationsDir, 0, goose.MaxVersion)
check.NoError(t, err)
maxVersionID := allMigrations[len(allMigrations)-1].Version
count, err := getGooseVersionCount(db, goose.TableName())
check.NoError(t, err)
// Count should be all testdata migrations (all applied)
check.Number(t, count, len(allMigrations))
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
// Expecting max(version_id) to be highest version in testdata
check.Number(t, current, maxVersionID)
}
// Migrate everything down using Reset.
err := goose.Reset(db, migrationsDir)
check.NoError(t, err)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 0)
}
func TestAllowMissingUpByOne(t *testing.T) {
t.Parallel()
// Create and apply first 5 migrations.
db := setupTestDB(t, 5)
/*
Developer A and B simultaneously check out the "main" currently on version 5.
Developer A mistakenly creates migration 7 and commits.
Developer B did not pull the latest changes and commits migration 6. Oops.
If goose is set to allow missing migrations, then 6 should be applied
after 7.
*/
// Developer A - migration 7 (mistakenly applied)
{
migrations, err := goose.CollectMigrations(migrationsDir, 0, 7)
check.NoError(t, err)
err = migrations[6].Up(db)
check.NoError(t, err)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, current, 7)
}
// Developer B - migration 6
{
err := goose.UpByOne(db, migrationsDir, goose.WithAllowMissing())
check.NoError(t, err)
count, err := getGooseVersionCount(db, goose.TableName())
check.NoError(t, err)
// Expecting count of migrations to be 7
check.Number(t, count, 7)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
// Expecting max(version_id) to be 6
check.Number(t, current, 6)
}
// Developer B - migration 8
{
// By default, this should raise an error.
err := goose.UpByOne(db, migrationsDir, goose.WithAllowMissing())
check.NoError(t, err)
count, err := getGooseVersionCount(db, goose.TableName())
check.NoError(t, err)
// Expecting count of migrations to be 8
check.Number(t, count, 8)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
// Expecting max(version_id) to be 8
check.Number(t, current, 8)
}
}
func TestMigrateAllowMissingDown(t *testing.T) {
t.Parallel()
const (
maxVersion = 8
)
// Create and apply first 5 migrations.
db := setupTestDB(t, 5)
// Developer A - migration 7 (mistakenly applied)
{
migrations, err := goose.CollectMigrations(migrationsDir, 0, maxVersion-1)
check.NoError(t, err)
err = migrations[6].Up(db)
check.NoError(t, err)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, current, maxVersion-1)
}
// Developer B - migration 6 (missing) and 8 (new)
{
// 6
err := goose.UpByOne(db, migrationsDir, goose.WithAllowMissing())
check.NoError(t, err)
// 8
err = goose.UpByOne(db, migrationsDir, goose.WithAllowMissing())
check.NoError(t, err)
count, err := getGooseVersionCount(db, goose.TableName())
check.NoError(t, err)
check.Number(t, count, maxVersion)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
// Expecting max(version_id) to be 8
check.Number(t, current, maxVersion)
}
// The order in the database is expected to be:
// 1,2,3,4,5,7,6,8
// So migrating down should be the reverse order:
// 8,6,7,5,4,3,2,1
//
// Migrate down by one. Expecting 6.
{
err := goose.Down(db, migrationsDir)
check.NoError(t, err)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
// Expecting max(version) to be 6
check.Number(t, current, 6)
}
// Migrate down by one. Expecting 7.
{
err := goose.Down(db, migrationsDir)
check.NoError(t, err)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
// Expecting max(version) to be 7
check.Number(t, current, 7)
}
// Migrate down by one. Expecting 5.
{
err := goose.Down(db, migrationsDir)
check.NoError(t, err)
current, err := goose.GetDBVersion(db)
check.NoError(t, err)
// Expecting max(version) to be 5
check.Number(t, current, 5)
}
}
func TestWithWithoutAllowMissing(t *testing.T) {
// Test for https://github.com/pressly/goose/issues/521
// Apply 1,2,4,3 then run up without allow missing. If the next requested migration is
// 4 then it should not raise an error because it has already been applied.
// If goose attempts to apply 4 again then it will raise an error (SQLSTATE 42701) because it
// has already been applied. Specifically it will raise a error.
db := setupTestDB(t, 2)
migrations, err := goose.CollectMigrations(migrationsDir, 0, 4)
check.NoError(t, err)
err = migrations[3].Up(db) // version 4
check.NoError(t, err)
err = migrations[2].Up(db) // version 3
check.NoError(t, err)
err = goose.UpTo(db, migrationsDir, 4)
check.NoError(t, err)
err = goose.UpTo(db, migrationsDir, 4, goose.WithAllowMissing())
check.NoError(t, err)
// Rollback migration 3 because it is the last applied migration.
// But, we want to change this behaviour to apply rollback migration 4.
// See these issues for more details:
// https://github.com/pressly/goose/issues/523
// https://github.com/pressly/goose/issues/402
//
// Adding this test to ensure the behaviour is updated and captured in a test.
err = goose.Down(db, migrationsDir)
check.NoError(t, err)
version, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, version, 4)
}
// setupTestDB is helper to setup a DB and apply migrations
// up to the specified version.
func setupTestDB(t *testing.T, version int64) *sql.DB {
db, err := newDockerDB(t)
check.NoError(t, err)
// Create goose table.
current, err := goose.EnsureDBVersion(db)
check.NoError(t, err)
check.Number(t, current, 0)
// Collect first 5 migrations.
migrations, err := goose.CollectMigrations(migrationsDir, 0, version)
check.NoError(t, err)
check.Number(t, len(migrations), version)
// Apply n migrations manually.
for _, m := range migrations {
err := m.Up(db)
check.NoError(t, err)
}
// Verify the current DB version is the Nth migration. This will only
// work for sqeuentially applied migrations.
current, err = goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, current, version)
return db
}

View File

@ -1,126 +0,0 @@
package e2e
import (
"database/sql"
"flag"
"fmt"
"log"
"os"
"os/signal"
"path/filepath"
"syscall"
"testing"
"github.com/pressly/goose/v3"
"github.com/pressly/goose/v3/internal/check"
"github.com/pressly/goose/v3/internal/testdb"
)
const (
dialectPostgres = "postgres"
dialectMySQL = "mysql"
dialectYdb = "ydb"
dialectTurso = "turso"
)
// Flags.
var (
debug = flag.Bool(
"debug",
false,
"Debug traps the test suite: useful for debugging running containers",
)
dialect = flag.String(
"dialect",
dialectPostgres,
"Dialect defines which docker container to run tests against (default: postgres)",
)
// bindPort is useful if you want to pin a database port instead of relying
// on the randomly assigned one from Docker. It is mainly used for debugging
// locally and will normally be set to 0.
bindPort = flag.Int(
"port",
0,
"Port is an optional bind port. Left empty will let Docker assign a random port (recommended)",
)
)
var (
// migrationsDir is a global that points to a ./testdata/{dialect}/migrations folder.
// It is set in TestMain based on the current dialect.
migrationsDir = ""
// seedDir is similar to migrationsDir but contains seed data
seedDir = ""
// known tables are the tables (including goose table) created by
// running all migration files. If you add a table, make sure to
// add to this list and keep it in order.
knownTables = []string{
"goose_db_version",
"issues",
"owners",
"repos",
"stargazers",
}
)
func TestMain(m *testing.M) {
flag.Parse()
switch *dialect {
case dialectPostgres, dialectMySQL, dialectYdb, dialectTurso:
default:
log.Printf("dialect not supported: %q", *dialect)
os.Exit(1)
}
migrationsDir = filepath.Join("testdata", *dialect, "migrations")
seedDir = filepath.Join("testdata", *dialect, "seed")
if err := goose.SetDialect(*dialect); err != nil {
log.Printf("failed to set dialect %q: %v\n", *dialect, err)
os.Exit(1)
}
exitCode := m.Run()
// Useful for debugging test services.
if *debug {
sigs := make(chan os.Signal, 1)
done := make(chan bool, 1)
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
go func() {
<-sigs
done <- true
}()
log.Printf("entering debug mode: must exit (CTRL+C) and cleanup containers manually. Exit code: %d)", exitCode)
<-done
}
os.Exit(exitCode)
}
// newDockerDB starts a database container and returns a usable SQL connection.
func newDockerDB(t *testing.T) (*sql.DB, error) {
options := []testdb.OptionsFunc{
testdb.WithBindPort(*bindPort),
testdb.WithDebug(*debug),
}
var (
db *sql.DB
cleanup func()
err error
)
switch *dialect {
case dialectPostgres:
db, cleanup, err = testdb.NewPostgres(options...)
case dialectMySQL:
db, cleanup, err = testdb.NewMariaDB(options...)
case dialectYdb:
db, cleanup, err = testdb.NewYdb(options...)
case dialectTurso:
db, cleanup, err = testdb.NewTurso(options...)
default:
return nil, fmt.Errorf("unsupported dialect: %q", *dialect)
}
check.NoError(t, err)
t.Cleanup(cleanup)
return db, nil
}

View File

@ -1,373 +0,0 @@
package e2e
import (
"context"
"database/sql"
"errors"
"fmt"
"os"
"path/filepath"
"reflect"
"testing"
"time"
"github.com/pressly/goose/v3"
"github.com/pressly/goose/v3/internal/check"
)
func TestMigrateUpWithReset(t *testing.T) {
t.Parallel()
db, err := newDockerDB(t)
check.NoError(t, err)
migrations, err := goose.CollectMigrations(migrationsDir, 0, goose.MaxVersion)
check.NoError(t, err)
check.NumberNotZero(t, len(migrations))
// Migrate all
err = goose.Up(db, migrationsDir)
check.NoError(t, err)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, migrations[len(migrations)-1].Version)
// Validate the db migration version actually matches what goose claims it is
gotVersion, err := getCurrentGooseVersion(db, goose.TableName())
check.NoError(t, err)
// incorrect database version
check.Number(t, gotVersion, currentVersion)
// Migrate everything down using Reset.
err = goose.Reset(db, migrationsDir)
check.NoError(t, err)
currentVersion, err = goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 0)
}
func TestMigrateUpWithRedo(t *testing.T) {
t.Parallel()
db, err := newDockerDB(t)
check.NoError(t, err)
migrations, err := goose.CollectMigrations(migrationsDir, 0, goose.MaxVersion)
check.NoError(t, err)
check.NumberNotZero(t, len(migrations))
startingVersion, err := goose.EnsureDBVersion(db)
check.NoError(t, err)
check.Number(t, startingVersion, 0)
// Migrate all
for _, migration := range migrations {
err = migration.Up(db)
check.NoError(t, err)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, migration.Version)
// Redo the previous Up migration and re-apply it.
err = goose.Redo(db, migrationsDir)
check.NoError(t, err)
currentVersion, err = goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, migration.Version)
}
// Once everything is tested the version should match the highest testdata version
maxVersion := migrations[len(migrations)-1].Version
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, maxVersion)
}
func TestMigrateUpTo(t *testing.T) {
t.Parallel()
const (
upToVersion int64 = 2
)
db, err := newDockerDB(t)
check.NoError(t, err)
migrations, err := goose.CollectMigrations(migrationsDir, 0, goose.MaxVersion)
check.NoError(t, err)
check.NumberNotZero(t, len(migrations))
// Migrate up to the second migration
err = goose.UpTo(db, migrationsDir, upToVersion)
check.NoError(t, err)
// Fetch the goose version from DB
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, upToVersion)
// Validate the version actually matches what goose claims it is
gotVersion, err := getCurrentGooseVersion(db, goose.TableName())
check.NoError(t, err)
check.Number(t, gotVersion, upToVersion) // incorrect database version
}
func writeFile(t *testing.T, dir, name, content string) {
t.Helper()
if err := os.WriteFile(filepath.Join(dir, name), []byte(content), 0644); err != nil {
t.Fatalf("failed to write file %q: %v", name, err)
}
}
func TestMigrateUpTimeout(t *testing.T) {
t.Parallel()
if *dialect != dialectPostgres {
t.Skipf("skipping test for dialect: %q", *dialect)
}
dir := t.TempDir()
writeFile(t, dir, "00001_a.sql", `
-- +goose Up
SELECT 1;
`)
writeFile(t, dir, "00002_a.sql", `
-- +goose Up
SELECT pg_sleep(10);
`)
db, err := newDockerDB(t)
check.NoError(t, err)
// Simulate timeout midway through a set of migrations. This should leave the
// database in a state where it has applied some migrations, but not all.
ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
defer cancel()
migrations, err := goose.CollectMigrations(dir, 0, goose.MaxVersion)
check.NoError(t, err)
check.NumberNotZero(t, len(migrations))
// Apply all migrations.
err = goose.UpContext(ctx, db, dir)
check.HasError(t, err) // expect it to timeout.
check.Bool(t, errors.Is(err, context.DeadlineExceeded), true)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 1)
// Validate the db migration version actually matches what goose claims it is
gotVersion, err := getCurrentGooseVersion(db, goose.TableName())
check.NoError(t, err)
check.Number(t, gotVersion, 1)
}
func TestMigrateDownTimeout(t *testing.T) {
t.Parallel()
if *dialect != dialectPostgres {
t.Skipf("skipping test for dialect: %q", *dialect)
}
dir := t.TempDir()
writeFile(t, dir, "00001_a.sql", `
-- +goose Up
SELECT 1;
-- +goose Down
SELECT pg_sleep(10);
`)
writeFile(t, dir, "00002_a.sql", `
-- +goose Up
SELECT 1;
`)
db, err := newDockerDB(t)
check.NoError(t, err)
// Simulate timeout midway through a set of migrations. This should leave the
// database in a state where it has applied some migrations, but not all.
ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
defer cancel()
migrations, err := goose.CollectMigrations(dir, 0, goose.MaxVersion)
check.NoError(t, err)
check.NumberNotZero(t, len(migrations))
// Apply all up migrations.
err = goose.UpContext(ctx, db, dir)
check.NoError(t, err)
// Applly all down migrations.
err = goose.DownToContext(ctx, db, dir, 0)
check.HasError(t, err) // expect it to timeout.
check.Bool(t, errors.Is(err, context.DeadlineExceeded), true)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 1)
// Validate the db migration version actually matches what goose claims it is
gotVersion, err := getCurrentGooseVersion(db, goose.TableName())
check.NoError(t, err)
check.Number(t, gotVersion, 1)
}
func TestMigrateUpByOne(t *testing.T) {
t.Parallel()
db, err := newDockerDB(t)
check.NoError(t, err)
migrations, err := goose.CollectMigrations(migrationsDir, 0, goose.MaxVersion)
check.NoError(t, err)
check.NumberNotZero(t, len(migrations))
// Apply all migrations one-by-one.
var counter int
for {
err := goose.UpByOne(db, migrationsDir)
counter++
if counter > len(migrations) {
if !errors.Is(err, goose.ErrNoNextVersion) {
t.Fatalf("incorrect error: got:%v want:%v", err, goose.ErrNoNextVersion)
}
break
}
check.NoError(t, err)
}
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, migrations[len(migrations)-1].Version)
// Validate the db migration version actually matches what goose claims it is
gotVersion, err := getCurrentGooseVersion(db, goose.TableName())
check.NoError(t, err)
check.Number(t, gotVersion, currentVersion) // incorrect database version
}
func TestMigrateFull(t *testing.T) {
t.Parallel()
db, err := newDockerDB(t)
check.NoError(t, err)
migrations, err := goose.CollectMigrations(migrationsDir, 0, goose.MaxVersion)
check.NoError(t, err)
check.NumberNotZero(t, len(migrations))
// test retrieving invalid current goose migrations. goose cannot return a migration
// if the supplied "current" version is non-existent or 0.
_, err = migrations.Current(20160813)
if !errors.Is(err, goose.ErrNoCurrentVersion) {
t.Fatalf("incorrect error: got:%v want:%v", err, goose.ErrNoCurrentVersion)
}
_, err = migrations.Current(0)
if !errors.Is(err, goose.ErrNoCurrentVersion) {
t.Fatalf("incorrect error: got:%v want:%v", err, goose.ErrNoCurrentVersion)
}
// verify the first migration1. This should not change if more migrations are added
// in the future.
migration1, err := migrations.Current(1)
check.NoError(t, err)
check.Number(t, migration1.Version, 1)
if migration1.Source != filepath.Join(migrationsDir, "00001_a.sql") {
t.Fatalf("failed to get correct migration file:\ngot:%s\nwant:%s",
migration1.Source,
filepath.Join(migrationsDir, "00001_a.sql"),
)
}
// expecting false for .sql migrations
check.Bool(t, migration1.Registered, false)
check.Number(t, migration1.Previous, -1)
check.Number(t, migration1.Next, 2)
{
// Apply all up migrations
err = goose.Up(db, migrationsDir)
check.NoError(t, err)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, migrations[len(migrations)-1].Version)
// Validate the db migration version actually matches what goose claims it is
gotVersion, err := getCurrentGooseVersion(db, goose.TableName())
check.NoError(t, err)
check.Number(t, gotVersion, currentVersion) // incorrect database version
tables, err := getTableNames(db)
check.NoError(t, err)
if !reflect.DeepEqual(tables, knownTables) {
t.Logf("got tables: %v", tables)
t.Logf("known tables: %v", knownTables)
t.Fatal("failed to match tables")
}
}
{
// Apply 1 down migration
err := goose.Down(db, migrationsDir)
check.NoError(t, err)
gotVersion, err := getCurrentGooseVersion(db, goose.TableName())
check.NoError(t, err)
check.Number(t, gotVersion, migrations[len(migrations)-1].Version-1) // incorrect database version
}
{
// Migrate all remaining migrations down. Should only be left with a single table:
// the default goose table
err := goose.DownTo(db, migrationsDir, 0)
check.NoError(t, err)
gotVersion, err := getCurrentGooseVersion(db, goose.TableName())
check.NoError(t, err)
check.Number(t, gotVersion, 0)
tables, err := getTableNames(db)
check.NoError(t, err)
knownTables := []string{goose.TableName()}
if !reflect.DeepEqual(tables, knownTables) {
t.Logf("got tables: %v", tables)
t.Logf("known tables: %v", knownTables)
t.Fatal("failed to match tables")
}
}
}
func getCurrentGooseVersion(db *sql.DB, gooseTable string) (int64, error) {
var gotVersion int64
if err := db.QueryRow(
fmt.Sprintf("select max(version_id) from %s", gooseTable),
).Scan(&gotVersion); err != nil {
return 0, err
}
return gotVersion, nil
}
func getGooseVersionCount(db *sql.DB, gooseTable string) (int64, error) {
var gotVersion int64
if err := db.QueryRow(
fmt.Sprintf("SELECT count(*) FROM %s WHERE version_id > 0", gooseTable),
).Scan(&gotVersion); err != nil {
return 0, err
}
return gotVersion, nil
}
func getTableNamesThroughQuery(db *sql.DB, query string) ([]string, error) {
rows, err := db.Query(query)
if err != nil {
return nil, err
}
defer rows.Close()
var tableNames []string
for rows.Next() {
var name string
if err := rows.Scan(&name); err != nil {
return nil, err
}
tableNames = append(tableNames, name)
}
return tableNames, nil
}
func getTableNames(db *sql.DB) (tableNames []string, _ error) {
switch *dialect {
case dialectPostgres:
return getTableNamesThroughQuery(db,
`SELECT table_name FROM information_schema.tables WHERE table_schema='public' ORDER BY table_name`,
)
case dialectMySQL:
return getTableNamesThroughQuery(db,
`SELECT table_name FROM INFORMATION_SCHEMA.tables WHERE TABLE_TYPE='BASE TABLE' ORDER BY table_name`,
)
case dialectYdb:
conn, err := db.Conn(context.Background())
if err != nil {
return nil, err
}
if err = conn.Raw(func(rawConn any) error {
if tables, has := rawConn.(interface {
GetTables(ctx context.Context, folder string, recursive bool, excludeSysDirs bool) (tables []string, err error)
}); has {
tableNames, err = tables.GetTables(context.Background(), ".", true, true)
if err != nil {
return err
}
return nil
}
return fmt.Errorf("%T not implemented GetTables interface", rawConn)
}); err != nil {
return nil, err
}
return tableNames, nil
case dialectTurso:
return getTableNamesThroughQuery(db, `SELECT NAME FROM sqlite_master where type='table' and name!='sqlite_sequence' ORDER BY NAME;`)
default:
return nil, fmt.Errorf("getTableNames not supported with dialect %q", *dialect)
}
}

View File

@ -1,149 +0,0 @@
package e2e
import (
"database/sql"
"testing"
"github.com/pressly/goose/v3"
"github.com/pressly/goose/v3/internal/check"
)
func TestNoVersioning(t *testing.T) {
if *dialect != dialectPostgres {
t.SkipNow()
}
const (
// Total owners created by the seed files.
wantSeedOwnerCount = 250
// These are owners created by migration files.
wantOwnerCount = 4
)
db, err := newDockerDB(t)
check.NoError(t, err)
err = goose.Up(db, migrationsDir)
check.NoError(t, err)
baseVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
t.Run("seed-up-down-to-zero", func(t *testing.T) {
// Run (all) up migrations from the seed dir
{
err = goose.Up(db, seedDir, goose.WithNoVersioning())
check.NoError(t, err)
// Confirm no changes to the versioned schema in the DB
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, baseVersion, currentVersion)
seedOwnerCount, err := countSeedOwners(db)
check.NoError(t, err)
check.Number(t, seedOwnerCount, wantSeedOwnerCount)
}
// Run (all) down migrations from the seed dir
{
err = goose.DownTo(db, seedDir, 0, goose.WithNoVersioning())
check.NoError(t, err)
// Confirm no changes to the versioned schema in the DB
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, baseVersion, currentVersion)
seedOwnerCount, err := countSeedOwners(db)
check.NoError(t, err)
check.Number(t, seedOwnerCount, 0)
}
// The migrations added 4 non-seed owners, they must remain
// in the database afterwards
ownerCount, err := countOwners(db)
check.NoError(t, err)
check.Number(t, ownerCount, wantOwnerCount)
})
t.Run("test-seed-up-reset", func(t *testing.T) {
// Run (all) up migrations from the seed dir
{
err = goose.Up(db, seedDir, goose.WithNoVersioning())
check.NoError(t, err)
// Confirm no changes to the versioned schema in the DB
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, baseVersion, currentVersion)
seedOwnerCount, err := countSeedOwners(db)
check.NoError(t, err)
check.Number(t, seedOwnerCount, wantSeedOwnerCount)
}
// Run reset (effectively the same as down-to 0)
{
err = goose.Reset(db, seedDir, goose.WithNoVersioning())
check.NoError(t, err)
// Confirm no changes to the versioned schema in the DB
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, baseVersion, currentVersion)
seedOwnerCount, err := countSeedOwners(db)
check.NoError(t, err)
check.Number(t, seedOwnerCount, 0)
}
// The migrations added 4 non-seed owners, they must remain
// in the database afterwards
ownerCount, err := countOwners(db)
check.NoError(t, err)
check.Number(t, ownerCount, wantOwnerCount)
})
t.Run("test-seed-up-redo", func(t *testing.T) {
// Run (all) up migrations from the seed dir
{
err = goose.Up(db, seedDir, goose.WithNoVersioning())
check.NoError(t, err)
// Confirm no changes to the versioned schema in the DB
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, baseVersion, currentVersion)
seedOwnerCount, err := countSeedOwners(db)
check.NoError(t, err)
check.Number(t, seedOwnerCount, wantSeedOwnerCount)
}
// Run reset (effectively the same as down-to 0)
{
err = goose.Redo(db, seedDir, goose.WithNoVersioning())
check.NoError(t, err)
// Confirm no changes to the versioned schema in the DB
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, baseVersion, currentVersion)
seedOwnerCount, err := countSeedOwners(db)
check.NoError(t, err)
check.Number(t, seedOwnerCount, wantSeedOwnerCount) // owners should be unchanged
}
// The migrations added 4 non-seed owners, they must remain
// in the database afterwards along with the 250 seed owners for a
// total of 254.
ownerCount, err := countOwners(db)
check.NoError(t, err)
check.Number(t, ownerCount, wantOwnerCount+wantSeedOwnerCount)
})
}
func countSeedOwners(db *sql.DB) (int, error) {
q := `SELECT count(*)FROM owners WHERE owner_name LIKE'seed-user-%'`
var count int
if err := db.QueryRow(q).Scan(&count); err != nil {
return 0, err
}
return count, nil
}
func countOwners(db *sql.DB) (int, error) {
q := `SELECT count(*)FROM owners`
var count int
if err := db.QueryRow(q).Scan(&count); err != nil {
return 0, err
}
return count, nil
}

View File

@ -1,20 +0,0 @@
-- +goose Up
-- +goose StatementBegin
CREATE TABLE owners (
owner_id BIGINT PRIMARY KEY AUTO_INCREMENT,
owner_name varchar(255) NOT NULL,
owner_type ENUM('user', 'organization') NOT NULL
);
CREATE TABLE repos (
repo_id BIGINT PRIMARY KEY AUTO_INCREMENT,
repo_owner_id BIGINT NOT NULL,
repo_full_name VARCHAR(255) NOT NULL,
FOREIGN KEY (repo_owner_id) REFERENCES owners (owner_id) ON DELETE CASCADE
);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DROP TABLE IF EXISTS repos;
DROP TABLE IF EXISTS owners;
-- +goose StatementEnd

View File

@ -1,10 +0,0 @@
-- +goose Up
-- +goose StatementBegin
INSERT INTO owners(owner_name, owner_type)
VALUES ('lucas', 'user'), ('space', 'organization');
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DELETE FROM owners;
-- +goose StatementEnd

View File

@ -1,13 +0,0 @@
-- +goose Up
-- +goose StatementBegin
INSERT INTO owners(owner_name, owner_type)
VALUES ('james', 'user'), ('pressly', 'organization');
INSERT INTO repos(repo_full_name, repo_owner_id)
VALUES ('james/rover', 3), ('pressly/goose', 4);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DELETE FROM owners WHERE owner_name IN ('james', 'pressly');
-- +goose StatementEnd

View File

@ -1,13 +0,0 @@
-- +goose Up
-- +goose StatementBegin
ALTER TABLE repos
ADD COLUMN IF NOT EXISTS homepage_url text NOT NULL DEFAULT '',
ADD COLUMN is_private boolean NOT NULL DEFAULT false;
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
ALTER TABLE repos
DROP COLUMN IF EXISTS homepage_url,
DROP COLUMN is_private;
-- +goose StatementEnd

View File

@ -1,17 +0,0 @@
-- +goose Up
-- +goose StatementBegin
CREATE TABLE IF NOT EXISTS stargazers (
stargazer_repo_id bigint NOT NULL REFERENCES repos(repo_id) ON DELETE CASCADE,
stargazer_owner_id bigint NOT NULL REFERENCES owners(owner_id) ON DELETE CASCADE,
stargazer_starred_at timestamp NOT NULL,
stargazer_location text NOT NULL
);
ALTER TABLE IF EXISTS stargazers
ADD CONSTRAINT stargazers_repo_id_owner_id_key PRIMARY KEY (stargazer_repo_id, stargazer_owner_id);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DROP TABLE stargazers;
-- +goose StatementEnd

View File

@ -1,15 +0,0 @@
-- +goose Up
-- +goose StatementBegin
CREATE TABLE issues (
issue_id BIGINT PRIMARY KEY AUTO_INCREMENT,
issue_created_by bigint NOT NULL REFERENCES owners(owner_id) ON DELETE CASCADE,
issue_repo_id bigint NOT NULL REFERENCES repos(repo_id) ON DELETE CASCADE,
issue_created_at timestamp NOT NULL,
issue_description text NOT NULL
);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DROP TABLE issues;
-- +goose StatementEnd

View File

@ -1,10 +0,0 @@
-- +goose Up
-- +goose StatementBegin
-- This migration intentionally depends on 00006_f.sql
ALTER TABLE stargazers DROP COLUMN stargazer_location;
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
ALTER TABLE stargazers ADD COLUMN stargazer_location text NOT NULL;
-- +goose StatementEnd

View File

@ -1,10 +0,0 @@
-- +goose Up
-- +goose StatementBegin
INSERT INTO owners(owner_name, owner_type)
VALUES ('lucas', 'user'), ('space', 'organization');
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DELETE FROM owners;
-- +goose StatementEnd

View File

@ -1,11 +0,0 @@
-- +goose Up
-- +goose StatementBegin
-- NOTE: intentionally left blank to verify migration logic.
SELECT 'up SQL query';
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
-- NOTE: intentionally left blank to verify migration logic.
SELECT 'down SQL query';
-- +goose StatementEnd

View File

@ -1,17 +0,0 @@
-- +goose Up
-- +goose StatementBegin
CREATE TABLE IF NOT EXISTS stargazers (
stargazer_repo_id bigint NOT NULL REFERENCES repos(repo_id) ON DELETE CASCADE DEFERRABLE,
stargazer_owner_id bigint NOT NULL REFERENCES owners(owner_id) ON DELETE CASCADE DEFERRABLE,
stargazer_starred_at timestamp NOT NULL,
stargazer_location text NOT NULL
);
ALTER TABLE IF EXISTS stargazers
ADD CONSTRAINT stargazers_repo_id_owner_id_key PRIMARY KEY (stargazer_repo_id, stargazer_owner_id);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DROP TABLE stargazers;
-- +goose StatementEnd

View File

@ -1,15 +0,0 @@
-- +goose Up
-- +goose StatementBegin
CREATE TABLE issues (
issue_id BIGSERIAL PRIMARY KEY,
issue_created_by bigint NOT NULL REFERENCES owners(owner_id) ON DELETE CASCADE DEFERRABLE,
issue_repo_id bigint NOT NULL REFERENCES repos(repo_id) ON DELETE CASCADE DEFERRABLE,
issue_created_at timestamp NOT NULL,
issue_description text NOT NULL
);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DROP TABLE issues;
-- +goose StatementEnd

View File

@ -1,10 +0,0 @@
-- +goose Up
-- +goose StatementBegin
-- This migration intentionally depends on 00006_f.sql
ALTER TABLE stargazers DROP COLUMN stargazer_location;
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
ALTER TABLE stargazers ADD COLUMN stargazer_location text NOT NULL;
-- +goose StatementEnd

View File

@ -1,9 +0,0 @@
-- +goose NO TRANSACTION
-- +goose Up
-- +goose StatementBegin
CREATE UNIQUE INDEX CONCURRENTLY ON owners(owner_name);
-- +goose StatementEnd
-- +goose Down
DROP INDEX IF EXISTS owners_owner_name_idx;

View File

@ -1,26 +0,0 @@
-- +goose Up
CREATE MATERIALIZED VIEW IF NOT EXISTS matview_stargazers_day AS
SELECT
t.*,
repo_full_name,
repo_owner_id
FROM (
SELECT
date_trunc('day', stargazer_starred_at)::date AS stars_day,
count(*) AS total,
stargazer_repo_id
FROM
stargazers
GROUP BY
stars_day,
stargazer_repo_id) AS t
JOIN repos ON stargazer_repo_id = repo_id
ORDER BY
stars_day;
CREATE UNIQUE INDEX ON matview_stargazers_day (stargazer_repo_id, stars_day, repo_owner_id, repo_full_name);
REFRESH MATERIALIZED VIEW CONCURRENTLY matview_stargazers_day WITH DATA;
-- +goose Down
DROP MATERIALIZED VIEW IF EXISTS matview_stargazers_day;

View File

@ -1,18 +0,0 @@
-- +goose Up
-- +goose StatementBegin
-- Insert 100 owners.
INSERT INTO owners (owner_name, owner_type)
SELECT
'seed-user-' || i,
(SELECT('{user,organization}'::owner_type []) [MOD(i, 2)+1])
FROM
generate_series(1, 100) s (i);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
-- NOTE: there are 4 existing users from the migrations, that's why owner_id starts at 5
DELETE FROM owners where owner_name LIKE 'seed-user-%' AND owner_id BETWEEN 5 AND 104;
SELECT setval('owners_owner_id_seq', COALESCE((SELECT MAX(owner_id)+1 FROM owners), 1), false);
-- +goose StatementEnd

View File

@ -1,14 +0,0 @@
-- +goose Up
-- Insert 150 more owners.
INSERT INTO owners (owner_name, owner_type)
SELECT
'seed-user-' || i,
(SELECT('{user,organization}'::owner_type []) [MOD(i, 2)+1])
FROM
generate_series(101, 250) s (i);
-- +goose Down
-- NOTE: there are 4 migration owners and 100 seed owners, that's why owner_id starts at 105
DELETE FROM owners where owner_name LIKE 'seed-user-%' AND owner_id BETWEEN 105 AND 254;
SELECT setval('owners_owner_id_seq', max(owner_id)) FROM owners;

View File

@ -1,21 +0,0 @@
-- +goose Up
-- +goose StatementBegin
CREATE TABLE owners (
owner_id integer,
owner_name integer,
owner_type integer,
PRIMARY KEY (owner_id)
);
CREATE TABLE repos (
repo_id integer,
repo_owner_id integer,
repo_full_name integer,
PRIMARY KEY (repo_id)
);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DROP TABLE repos;
DROP TABLE owners;
-- +goose StatementEnd

View File

@ -1,13 +0,0 @@
-- +goose Up
-- +goose StatementBegin
INSERT INTO owners(owner_id, owner_name, owner_type)
VALUES (3, 'james', 'user'), (4, 'pressly', 'organization');
INSERT INTO repos(repo_id, repo_full_name, repo_owner_id)
VALUES (1, 'james/rover', 3), (2, 'pressly/goose', 4);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DELETE FROM owners WHERE (owner_id = 3 OR owner_id = 4);
DELETE FROM repos WHERE (repo_id = 1 OR repo_id = 2);
-- +goose StatementEnd

View File

@ -1,15 +0,0 @@
-- +goose Up
-- +goose StatementBegin
ALTER TABLE repos
ADD COLUMN homepage_url text;
ALTER TABLE repos
ADD COLUMN is_private integer;
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
ALTER TABLE repos
DROP COLUMN homepage_url;
ALTER TABLE repos
DROP COLUMN is_private;
-- +goose StatementEnd

View File

@ -1,11 +0,0 @@
-- +goose Up
-- +goose StatementBegin
-- NOTE: intentionally left blank to verify migration logic.
SELECT 'up SQL query';
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
-- NOTE: intentionally left blank to verify migration logic.
SELECT 'down SQL query';
-- +goose StatementEnd

View File

@ -1,15 +0,0 @@
-- +goose Up
-- +goose StatementBegin
CREATE TABLE stargazers (
stargazer_repo_id integer,
stargazer_owner_id integer,
stargazer_starred_at integer,
stargazer_location text,
PRIMARY KEY (stargazer_repo_id, stargazer_owner_id)
);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DROP TABLE stargazers;
-- +goose StatementEnd

View File

@ -1,16 +0,0 @@
-- +goose Up
-- +goose StatementBegin
CREATE TABLE issues (
issue_id integer,
issue_created_by integer,
issue_repo_id integer,
issue_created_at integer,
issue_description text,
PRIMARY KEY (issue_id)
);
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DROP TABLE issues;
-- +goose StatementEnd

View File

@ -1,9 +0,0 @@
-- +goose Up
-- +goose StatementBegin
ALTER TABLE stargazers DROP COLUMN stargazer_location;
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
ALTER TABLE stargazers ADD COLUMN stargazer_location text;
-- +goose StatementEnd

View File

@ -1,10 +0,0 @@
-- +goose Up
-- +goose StatementBegin
INSERT INTO owners(owner_id, owner_name, owner_type)
VALUES (1, 'lucas', 'user'), (2, 'space', 'organization');
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
DELETE FROM owners;
-- +goose StatementEnd

View File

@ -1,11 +0,0 @@
-- +goose Up
-- +goose StatementBegin
-- NOTE: intentionally left blank to verify migration logic.
SELECT 'up SQL query';
-- +goose StatementEnd
-- +goose Down
-- +goose StatementBegin
-- NOTE: intentionally left blank to verify migration logic.
SELECT 'down SQL query';
-- +goose StatementEnd

View File

@ -1,19 +1,22 @@
package gomigrations
import (
"database/sql"
"path/filepath"
"testing"
"github.com/pressly/goose/v3"
"github.com/pressly/goose/v3/internal/check"
"github.com/pressly/goose/v3/internal/testdb"
_ "github.com/pressly/goose/v3/tests/gomigrations/error/testdata"
_ "modernc.org/sqlite"
)
func TestGoMigrationByOne(t *testing.T) {
db, cleanup, err := testdb.NewPostgres()
tempDir := t.TempDir()
db, err := sql.Open("sqlite", filepath.Join(tempDir, "test.db"))
check.NoError(t, err)
err = goose.SetDialect(string(goose.DialectSQLite3))
check.NoError(t, err)
t.Cleanup(cleanup)
// Create goose table.
current, err := goose.EnsureDBVersion(db)
check.NoError(t, err)

View File

@ -11,7 +11,7 @@ func init() {
}
func up001(db *sql.DB) error {
q := "CREATE TABLE foo (id INT)"
q := "CREATE TABLE foo (id INTEGER)"
_, err := db.Exec(q)
return err
}

View File

@ -11,7 +11,7 @@ func init() {
}
func up003(tx *sql.Tx) error {
q := "TRUNCATE TABLE foo"
q := "DELETE FROM foo"
_, err := tx.Exec(q)
return err
}

View File

@ -1,115 +0,0 @@
package vertica_test
import (
"path/filepath"
"testing"
"time"
"github.com/pressly/goose/v3"
"github.com/pressly/goose/v3/internal/check"
"github.com/pressly/goose/v3/internal/testdb"
)
/*
This test applies all up migrations, asserts we have all the entries in
the versions table, applies all down migration and asserts we have zero
migrations applied.
Limitations:
1) Only one instance of Vertica can be running on a host at any time.
*/
func TestVerticaUpDownAll(t *testing.T) {
t.Parallel()
migrationDir := filepath.Join("testdata", "migrations")
db, cleanup, err := testdb.NewVertica()
check.NoError(t, err)
t.Cleanup(cleanup)
check.NoError(t, goose.SetDialect("vertica"))
goose.SetTableName("goose_db_version")
migrations, err := goose.CollectMigrations(migrationDir, 0, goose.MaxVersion)
check.NoError(t, err)
currentVersion, err := goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 0)
err = goose.Up(db, migrationDir)
check.NoError(t, err)
currentVersion, err = goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, len(migrations))
type result struct {
TestKey int64 `db:"test_key"`
TestID string `db:"test_id"`
ValidFrom time.Time `db:"valid_from"`
ValidTo time.Time `db:"valid_to"`
IsCurrent bool `db:"is_current"`
ExternalID string `db:"external_id"`
}
rows, err := db.Query(`SELECT * FROM testing.dim_test_scd ORDER BY test_key`)
check.NoError(t, err)
var results []result
for rows.Next() {
var r result
err = rows.Scan(&r.TestKey, &r.TestID, &r.ValidFrom, &r.ValidTo, &r.IsCurrent, &r.ExternalID)
check.NoError(t, err)
results = append(results, r)
}
check.Number(t, len(results), 3)
check.NoError(t, rows.Close())
check.NoError(t, rows.Err())
parseTime := func(t *testing.T, s string) time.Time {
t.Helper()
tm, err := time.Parse("2006-01-02", s)
check.NoError(t, err)
return tm
}
want := []result{
{
TestKey: 1,
TestID: "575a0dd4-bd97-44ac-aae0-987090181da8",
ValidFrom: parseTime(t, "2021-10-02"),
ValidTo: parseTime(t, "2021-10-03"),
IsCurrent: false,
ExternalID: "123",
},
{
TestKey: 2,
TestID: "575a0dd4-bd97-44ac-aae0-987090181da8",
ValidFrom: parseTime(t, "2021-10-03"),
ValidTo: parseTime(t, "2021-10-04"),
IsCurrent: false,
ExternalID: "456",
},
{
TestKey: 3,
TestID: "575a0dd4-bd97-44ac-aae0-987090181da8",
ValidFrom: parseTime(t, "2021-10-04"),
ValidTo: parseTime(t, "9999-12-31"),
IsCurrent: true,
ExternalID: "789",
},
}
for i, result := range results {
check.Equal(t, result.TestKey, want[i].TestKey)
check.Equal(t, result.TestID, want[i].TestID)
check.Equal(t, result.ValidFrom, want[i].ValidFrom)
check.Equal(t, result.ValidTo, want[i].ValidTo)
check.Equal(t, result.IsCurrent, want[i].IsCurrent)
check.Equal(t, result.ExternalID, want[i].ExternalID)
}
err = goose.DownTo(db, migrationDir, 0)
check.NoError(t, err)
check.NoError(t, err)
currentVersion, err = goose.GetDBVersion(db)
check.NoError(t, err)
check.Number(t, currentVersion, 0)
}