mirror of
https://github.com/hasura/graphql-engine.git
synced 2024-12-16 18:42:30 +03:00
50a5462857
PR-URL: https://github.com/hasura/graphql-engine-mono/pull/7363 GitOrigin-RevId: 2b58511c374ee4d93a202e23c87c48b54ec363a6
1210 lines
37 KiB
Go
1210 lines
37 KiB
Go
package migrate
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/hasura/graphql-engine/cli/v2/internal/errors"
|
|
"github.com/hasura/graphql-engine/cli/v2/internal/hasura"
|
|
"github.com/hasura/graphql-engine/cli/v2/internal/testutil"
|
|
"github.com/hasura/graphql-engine/cli/v2/util"
|
|
)
|
|
|
|
func TestProjectMigrate_ApplyConfig_v3(t *testing.T) {
|
|
port, teardown := testutil.StartHasuraWithMetadataDatabase(t, testutil.HasuraDockerImage)
|
|
hasuraEndpoint := fmt.Sprintf("%s:%s", testutil.BaseURL, port)
|
|
connectionStringSource1, teardownPG1 := testutil.StartPGContainer(t)
|
|
connectionStringSource2, teardownPG2 := testutil.StartPGContainer(t)
|
|
testutil.AddPGSourceToHasura(t, hasuraEndpoint, connectionStringSource1, "s1")
|
|
testutil.AddPGSourceToHasura(t, hasuraEndpoint, connectionStringSource2, "s2")
|
|
defer func() {
|
|
teardownPG2()
|
|
teardownPG1()
|
|
teardown()
|
|
}()
|
|
hgeEndpoint := fmt.Sprintf("http://localhost:%s", port)
|
|
type fields struct {
|
|
projectDirectory string
|
|
endpointString string
|
|
}
|
|
type args struct {
|
|
opts []ProjectMigrationApplierOption
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
args args
|
|
want []ApplyResult
|
|
wantErr bool
|
|
assertErr require.ErrorAssertionFunc
|
|
}{
|
|
{
|
|
"can apply migrations in config v3 project",
|
|
fields{
|
|
projectDirectory: "testdata/projectv3",
|
|
endpointString: hgeEndpoint,
|
|
},
|
|
args{
|
|
[]ProjectMigrationApplierOption{ApplyOnAllDatabases()},
|
|
},
|
|
[]ApplyResult{
|
|
{
|
|
"s1",
|
|
"migrations applied on database: s1",
|
|
nil,
|
|
},
|
|
{
|
|
"s2",
|
|
"migrations applied on database: s2",
|
|
nil,
|
|
},
|
|
},
|
|
false,
|
|
require.NoError,
|
|
},
|
|
{
|
|
"can apply a version in config v3 project",
|
|
fields{
|
|
projectDirectory: "testdata/projectv3",
|
|
endpointString: hgeEndpoint,
|
|
},
|
|
args{
|
|
[]ProjectMigrationApplierOption{ApplyOnDatabaseName("s1"), ApplyVersion("1623841477474", MigrationDirectionDown)},
|
|
},
|
|
[]ApplyResult{
|
|
{
|
|
"s1",
|
|
"migrations applied",
|
|
nil,
|
|
},
|
|
},
|
|
false,
|
|
require.NoError,
|
|
},
|
|
{
|
|
"can apply a version in config v3 project",
|
|
fields{
|
|
projectDirectory: "testdata/projectv3",
|
|
endpointString: hgeEndpoint,
|
|
},
|
|
args{
|
|
[]ProjectMigrationApplierOption{ApplyOnDatabaseName("s1"), ApplyVersion("1623841477474", MigrationDirectionUp)},
|
|
},
|
|
[]ApplyResult{
|
|
{
|
|
"s1",
|
|
"migrations applied",
|
|
nil,
|
|
},
|
|
},
|
|
false,
|
|
require.NoError,
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
p, err := NewProjectMigrate(tt.fields.projectDirectory, WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(tt.fields.endpointString))
|
|
require.NoError(t, err)
|
|
got, err := p.Apply(tt.args.opts...)
|
|
tt.assertErr(t, err)
|
|
if tt.wantErr {
|
|
return
|
|
}
|
|
require.NoError(t, err)
|
|
require.Equal(t, tt.want, got)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestProjectMigrate_Apply_Configv2(t *testing.T) {
|
|
port, teardown := testutil.StartHasura(t, testutil.HasuraDockerImage)
|
|
defer teardown()
|
|
hgeEndpoint := fmt.Sprintf("http://localhost:%s", port)
|
|
type fields struct {
|
|
projectDirectory string
|
|
adminSecret string
|
|
endpointString string
|
|
}
|
|
type args struct {
|
|
opts []ProjectMigrationApplierOption
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
args args
|
|
want []ApplyResult
|
|
wantErr bool
|
|
assertErr require.ErrorAssertionFunc
|
|
}{
|
|
{
|
|
"can apply migrations in config v2 project",
|
|
fields{
|
|
projectDirectory: "testdata/projectv2",
|
|
adminSecret: "",
|
|
endpointString: hgeEndpoint,
|
|
},
|
|
args{
|
|
[]ProjectMigrationApplierOption{ApplyOnAllDatabases()},
|
|
},
|
|
[]ApplyResult{
|
|
{
|
|
Message: "migrations applied",
|
|
},
|
|
},
|
|
false,
|
|
require.NoError,
|
|
},
|
|
{
|
|
"can apply down migration on a version in config v2 project",
|
|
fields{
|
|
projectDirectory: "testdata/projectv2",
|
|
adminSecret: "",
|
|
endpointString: hgeEndpoint,
|
|
},
|
|
args{
|
|
[]ProjectMigrationApplierOption{ApplyVersion("1623842054907", MigrationDirectionDown)},
|
|
},
|
|
[]ApplyResult{
|
|
{
|
|
Message: "migrations applied",
|
|
},
|
|
},
|
|
false,
|
|
require.NoError,
|
|
},
|
|
{
|
|
"throws error when trying to do a down migration which is not applied",
|
|
fields{
|
|
projectDirectory: "testdata/projectv2",
|
|
adminSecret: "",
|
|
endpointString: hgeEndpoint,
|
|
},
|
|
args{
|
|
[]ProjectMigrationApplierOption{ApplyVersion("1623842054907", MigrationDirectionDown)},
|
|
},
|
|
[]ApplyResult{
|
|
{
|
|
Error: fmt.Errorf("skipping applying migrations on database '', encountered: \nMigration not applied in database"),
|
|
},
|
|
},
|
|
false,
|
|
require.NoError,
|
|
},
|
|
{
|
|
"can apply up migrations of a version on a config v2 project",
|
|
fields{
|
|
projectDirectory: "testdata/projectv2",
|
|
adminSecret: "",
|
|
endpointString: hgeEndpoint,
|
|
},
|
|
args{
|
|
[]ProjectMigrationApplierOption{ApplyVersion("1623842054907", MigrationDirectionUp)},
|
|
},
|
|
[]ApplyResult{
|
|
{
|
|
Message: "migrations applied",
|
|
},
|
|
},
|
|
false,
|
|
require.NoError,
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
p, err := NewProjectMigrate(tt.fields.projectDirectory, WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(tt.fields.endpointString))
|
|
require.NoError(t, err)
|
|
got, err := p.Apply(tt.args.opts...)
|
|
tt.assertErr(t, err)
|
|
if tt.wantErr {
|
|
return
|
|
}
|
|
for idx, want := range tt.want {
|
|
if idx >= len(got) {
|
|
t.Errorf("expected to got to have equal number of elements: want %v got %v", len(tt.want), len(got))
|
|
}
|
|
if len(want.Message) > 0 {
|
|
assert.Equal(t, want.Message, got[idx].Message)
|
|
}
|
|
if want.Error != nil {
|
|
assert.Equal(t, want.Error.Error(), got[idx].Error.Error())
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestProjectMigrate_Status_ConfigV2(t *testing.T) {
|
|
port, teardown := testutil.StartHasura(t, testutil.HasuraDockerImage)
|
|
defer teardown()
|
|
hgeEndpoint := fmt.Sprintf("http://localhost:%s", port)
|
|
type fields struct {
|
|
projectDirectory string
|
|
adminSecret string
|
|
endpointString string
|
|
}
|
|
type args struct {
|
|
opts []ProjectMigrationStatusOption
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
args args
|
|
want string
|
|
wantErr bool
|
|
assertErr require.ErrorAssertionFunc
|
|
before func(t *testing.T, p *ProjectMigrate)
|
|
}{
|
|
{
|
|
"can get status of migrations",
|
|
fields{
|
|
projectDirectory: "testdata/projectv2",
|
|
adminSecret: "",
|
|
endpointString: hgeEndpoint,
|
|
},
|
|
args{
|
|
opts: []ProjectMigrationStatusOption{},
|
|
},
|
|
`
|
|
[
|
|
{
|
|
"databaseName": "default",
|
|
"status": {
|
|
"migrations": [
|
|
1623842054907,
|
|
1623842062104,
|
|
1623842069725,
|
|
1623842076537,
|
|
1623842087940
|
|
],
|
|
"status": {
|
|
"1623842054907": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623842062104": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623842069725": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623842076537": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623842087940": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]`,
|
|
false,
|
|
require.NoError,
|
|
func(t *testing.T, p *ProjectMigrate) {},
|
|
},
|
|
{
|
|
"can get status of migrations",
|
|
fields{
|
|
projectDirectory: "testdata/projectv2",
|
|
adminSecret: "",
|
|
endpointString: hgeEndpoint,
|
|
},
|
|
args{
|
|
opts: []ProjectMigrationStatusOption{},
|
|
},
|
|
`
|
|
[
|
|
{
|
|
"databaseName": "default",
|
|
"status": {
|
|
"migrations": [
|
|
1623842054907,
|
|
1623842062104,
|
|
1623842069725,
|
|
1623842076537,
|
|
1623842087940
|
|
],
|
|
"status": {
|
|
"1623842054907": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842062104": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842069725": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842076537": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842087940": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]`,
|
|
false,
|
|
require.NoError,
|
|
func(t *testing.T, p *ProjectMigrate) {
|
|
_, err := p.Apply(ApplyOnAllDatabases())
|
|
assert.NoError(t, err)
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
p, err := NewProjectMigrate(tt.fields.projectDirectory, WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(tt.fields.endpointString))
|
|
require.NoError(t, err)
|
|
applier, err := NewProjectMigrate(tt.fields.projectDirectory, WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(tt.fields.endpointString))
|
|
require.NoError(t, err)
|
|
tt.before(t, applier)
|
|
got, err := p.status(tt.args.opts...)
|
|
tt.assertErr(t, err)
|
|
if tt.wantErr {
|
|
return
|
|
}
|
|
require.NoError(t, err)
|
|
gotJSON, err := json.Marshal(got)
|
|
require.NoError(t, err)
|
|
require.JSONEq(t, tt.want, string(gotJSON))
|
|
|
|
statusJson, err := p.StatusJSON(tt.args.opts...)
|
|
require.NoError(t, err)
|
|
statusJsonb, err := ioutil.ReadAll(statusJson)
|
|
require.NoError(t, err)
|
|
require.JSONEq(t, tt.want, string(statusJsonb))
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestProjectMigrate_Status_ConfigV3(t *testing.T) {
|
|
port, teardown := testutil.StartHasuraWithMetadataDatabase(t, testutil.HasuraDockerImage)
|
|
hasuraEndpoint := fmt.Sprintf("%s:%s", testutil.BaseURL, port)
|
|
connectionStringSource1, teardownPG1 := testutil.StartPGContainer(t)
|
|
connectionStringSource2, teardownPG2 := testutil.StartPGContainer(t)
|
|
testutil.AddPGSourceToHasura(t, hasuraEndpoint, connectionStringSource1, "s1")
|
|
testutil.AddPGSourceToHasura(t, hasuraEndpoint, connectionStringSource2, "s2")
|
|
defer func() {
|
|
teardownPG2()
|
|
teardownPG1()
|
|
teardown()
|
|
}()
|
|
hgeEndpoint := fmt.Sprintf("http://localhost:%s", port)
|
|
type fields struct {
|
|
projectDirectory string
|
|
adminSecret string
|
|
}
|
|
type args struct {
|
|
opts []ProjectMigrationStatusOption
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
args args
|
|
want string
|
|
wantErr bool
|
|
assertErr require.ErrorAssertionFunc
|
|
testSetup func() (hgeEndpoint string, teardown func())
|
|
before func(t *testing.T, p *ProjectMigrate)
|
|
}{
|
|
{
|
|
"can get status of migrations",
|
|
fields{
|
|
projectDirectory: "testdata/projectv3",
|
|
adminSecret: "",
|
|
},
|
|
args{
|
|
opts: []ProjectMigrationStatusOption{},
|
|
},
|
|
`[
|
|
{
|
|
"databaseName": "s1",
|
|
"status": {
|
|
"migrations": [
|
|
1623841477474,
|
|
1623841485323,
|
|
1623841492743,
|
|
1623841500466,
|
|
1623841510619
|
|
],
|
|
"status": {
|
|
"1623841477474": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623841485323": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623841492743": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623841500466": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623841510619": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"databaseName": "s2",
|
|
"status": {
|
|
"migrations": [
|
|
1623841477474,
|
|
1623841485323,
|
|
1623841492743,
|
|
1623841500466,
|
|
1623841510619
|
|
],
|
|
"status": {
|
|
"1623841477474": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623841485323": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623841492743": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623841500466": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623841510619": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]`,
|
|
false,
|
|
require.NoError,
|
|
func() (string, func()) { return hgeEndpoint, func() {} },
|
|
func(t *testing.T, p *ProjectMigrate) {},
|
|
},
|
|
{
|
|
"can get status of migrations",
|
|
fields{
|
|
projectDirectory: "testdata/projectv3",
|
|
adminSecret: "",
|
|
},
|
|
args{
|
|
opts: []ProjectMigrationStatusOption{},
|
|
},
|
|
`
|
|
[
|
|
{
|
|
"databaseName": "s1",
|
|
"status": {
|
|
"migrations": [
|
|
1623841477474,
|
|
1623841485323,
|
|
1623841492743,
|
|
1623841500466,
|
|
1623841510619
|
|
],
|
|
"status": {
|
|
"1623841477474": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841485323": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841492743": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841500466": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841510619": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"databaseName": "s2",
|
|
"status": {
|
|
"migrations": [
|
|
1623841477474,
|
|
1623841485323,
|
|
1623841492743,
|
|
1623841500466,
|
|
1623841510619
|
|
],
|
|
"status": {
|
|
"1623841477474": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841485323": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841492743": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841500466": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841510619": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]`,
|
|
false,
|
|
require.NoError,
|
|
func() (string, func()) { return hgeEndpoint, func() {} },
|
|
func(t *testing.T, p *ProjectMigrate) {
|
|
_, err := p.Apply(ApplyOnAllDatabases())
|
|
assert.NoError(t, err)
|
|
},
|
|
},
|
|
{
|
|
"can throw an error when no databases are connected to hge",
|
|
fields{
|
|
projectDirectory: "testdata/projectv3",
|
|
adminSecret: "",
|
|
},
|
|
args{
|
|
opts: []ProjectMigrationStatusOption{},
|
|
},
|
|
``,
|
|
true,
|
|
func(tt require.TestingT, err error, i ...interface{}) {
|
|
require.IsType(t, &errors.Error{}, err)
|
|
e := err.(*errors.Error)
|
|
require.Equal(t, errors.Op("migrate.ProjectMigrate.status"), e.Op)
|
|
},
|
|
func() (string, func()) {
|
|
port, teardown := testutil.StartHasuraWithMetadataDatabase(t, testutil.HasuraDockerImage)
|
|
return fmt.Sprintf("http://%s:%s", testutil.Hostname, port), teardown
|
|
},
|
|
func(t *testing.T, p *ProjectMigrate) {
|
|
_, err := p.Apply(ApplyOnAllDatabases())
|
|
assert.NoError(t, err)
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
hgeEndpoint, setupTeardown := tt.testSetup()
|
|
defer setupTeardown()
|
|
p, err := NewProjectMigrate(tt.fields.projectDirectory, WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(hgeEndpoint))
|
|
require.NoError(t, err)
|
|
applier, err := NewProjectMigrate(tt.fields.projectDirectory, WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(hgeEndpoint))
|
|
require.NoError(t, err)
|
|
tt.before(t, applier)
|
|
got, err := p.status(tt.args.opts...)
|
|
tt.assertErr(t, err)
|
|
if tt.wantErr {
|
|
return
|
|
}
|
|
gotJSON, err := json.Marshal(got)
|
|
require.NoError(t, err)
|
|
require.JSONEq(t, tt.want, string(gotJSON))
|
|
|
|
statusJson, err := p.StatusJSON(tt.args.opts...)
|
|
require.NoError(t, err)
|
|
statusJsonb, err := ioutil.ReadAll(statusJson)
|
|
require.NoError(t, err)
|
|
require.JSONEq(t, tt.want, string(statusJsonb))
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestProjectMigrate_SkipExecution_Configv3(t *testing.T) {
|
|
port, teardown := testutil.StartHasuraWithMetadataDatabase(t, testutil.HasuraDockerImage)
|
|
hasuraEndpoint := fmt.Sprintf("%s:%s", testutil.BaseURL, port)
|
|
connectionStringSource1, teardownPG1 := testutil.StartPGContainer(t)
|
|
testutil.AddPGSourceToHasura(t, hasuraEndpoint, connectionStringSource1, "s1")
|
|
defer func() {
|
|
teardownPG1()
|
|
teardown()
|
|
}()
|
|
hgeEndpoint := fmt.Sprintf("http://localhost:%s", port)
|
|
p, err := NewProjectMigrate("testdata/projectv3", WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(hgeEndpoint))
|
|
require.NoError(t, err)
|
|
_, err = p.Apply(ApplyOnAllDatabases())
|
|
require.NoError(t, err)
|
|
|
|
type args struct {
|
|
opts []ProjectMigrationApplierOption
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
args args
|
|
want string
|
|
wantErr bool
|
|
assertErr require.ErrorAssertionFunc
|
|
}{
|
|
{
|
|
"mark migration as unapplied",
|
|
args{
|
|
opts: []ProjectMigrationApplierOption{
|
|
ApplyOnDatabaseName("s1"), ApplyVersion("1623841477474", MigrationDirectionDown), ApplyWithSkipExecution(),
|
|
},
|
|
},
|
|
`
|
|
[
|
|
{
|
|
"databaseName": "s1",
|
|
"status": {
|
|
"migrations": [
|
|
1623841477474,
|
|
1623841485323,
|
|
1623841492743,
|
|
1623841500466,
|
|
1623841510619
|
|
],
|
|
"status": {
|
|
"1623841477474": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623841485323": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841492743": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841500466": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841510619": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
`,
|
|
false,
|
|
require.NoError,
|
|
},
|
|
{
|
|
"mark migration as applied",
|
|
args{
|
|
opts: []ProjectMigrationApplierOption{
|
|
ApplyOnDatabaseName("s1"), ApplyVersion("1623841477474", MigrationDirectionUp), ApplyWithSkipExecution(),
|
|
},
|
|
},
|
|
`
|
|
[
|
|
{
|
|
"databaseName": "s1",
|
|
"status": {
|
|
"migrations": [
|
|
1623841477474,
|
|
1623841485323,
|
|
1623841492743,
|
|
1623841500466,
|
|
1623841510619
|
|
],
|
|
"status": {
|
|
"1623841477474": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841485323": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841492743": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841500466": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623841510619": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
`,
|
|
false,
|
|
require.NoError,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
p1, err := NewProjectMigrate("testdata/projectv3", WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(hgeEndpoint))
|
|
require.NoError(t, err)
|
|
_, err = p1.Apply(tt.args.opts...)
|
|
require.NoError(t, err)
|
|
|
|
status, err := p.StatusJSON()
|
|
tt.assertErr(t, err)
|
|
if tt.wantErr {
|
|
return
|
|
}
|
|
statusJsonb, err := ioutil.ReadAll(status)
|
|
assert.NoError(t, err)
|
|
|
|
assert.JSONEq(t, tt.want, string(statusJsonb))
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestProjectMigrate_SkipExecution_Configv2(t *testing.T) {
|
|
port, teardown := testutil.StartHasura(t, testutil.HasuraDockerImage)
|
|
defer teardown()
|
|
hgeEndpoint := fmt.Sprintf("http://localhost:%s", port)
|
|
|
|
p, err := NewProjectMigrate("testdata/projectv2", WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(hgeEndpoint))
|
|
require.NoError(t, err)
|
|
_, err = p.Apply(ApplyOnAllDatabases())
|
|
require.NoError(t, err)
|
|
|
|
type args struct {
|
|
opts []ProjectMigrationApplierOption
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
args args
|
|
want string
|
|
wantErr bool
|
|
assertErr require.ErrorAssertionFunc
|
|
}{
|
|
{
|
|
"mark migration as unapplied",
|
|
args{
|
|
opts: []ProjectMigrationApplierOption{
|
|
ApplyVersion("1623842054907", MigrationDirectionDown), ApplyWithSkipExecution(),
|
|
},
|
|
},
|
|
`
|
|
[
|
|
{
|
|
"databaseName": "default",
|
|
"status": {
|
|
"migrations": [
|
|
1623842054907,
|
|
1623842062104,
|
|
1623842069725,
|
|
1623842076537,
|
|
1623842087940
|
|
],
|
|
"status": {
|
|
"1623842054907": {
|
|
"database_status": false,
|
|
"source_status": true
|
|
},
|
|
"1623842062104": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842069725": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842076537": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842087940": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
`,
|
|
false,
|
|
require.NoError,
|
|
},
|
|
{
|
|
"mark migration as applied",
|
|
args{
|
|
opts: []ProjectMigrationApplierOption{
|
|
ApplyVersion("1623842054907", MigrationDirectionUp), ApplyWithSkipExecution(),
|
|
},
|
|
},
|
|
`
|
|
[
|
|
{
|
|
"databaseName": "default",
|
|
"status": {
|
|
"migrations": [
|
|
1623842054907,
|
|
1623842062104,
|
|
1623842069725,
|
|
1623842076537,
|
|
1623842087940
|
|
],
|
|
"status": {
|
|
"1623842054907": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842062104": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842069725": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842076537": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
},
|
|
"1623842087940": {
|
|
"database_status": true,
|
|
"source_status": true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
`,
|
|
false,
|
|
require.NoError,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
p1, err := NewProjectMigrate("testdata/projectv2", WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(hgeEndpoint))
|
|
require.NoError(t, err)
|
|
_, err = p1.Apply(tt.args.opts...)
|
|
require.NoError(t, err)
|
|
|
|
status, err := p1.StatusJSON()
|
|
tt.assertErr(t, err)
|
|
if tt.wantErr {
|
|
return
|
|
}
|
|
statusJsonb, err := ioutil.ReadAll(status)
|
|
assert.NoError(t, err)
|
|
|
|
assert.JSONEq(t, tt.want, string(statusJsonb))
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestProjectMigrate_Delete_Configv3(t *testing.T) {
|
|
startHasura := func(t *testing.T, databaseName1 string, databaseName2 string) (port string, endpoint string, teardown func(), teardownPG1 func(), teardownPG2 func()) {
|
|
t.Helper()
|
|
port, teardown = testutil.StartHasuraWithMetadataDatabase(t, testutil.HasuraDockerImage)
|
|
endpoint = fmt.Sprintf("%s:%s", testutil.BaseURL, port)
|
|
connectionStringSource1, teardownPG1 := testutil.StartPGContainer(t)
|
|
connectionStringSource2, teardownPG2 := testutil.StartPGContainer(t)
|
|
testutil.AddPGSourceToHasura(t, endpoint, connectionStringSource1, databaseName1)
|
|
testutil.AddPGSourceToHasura(t, endpoint, connectionStringSource2, databaseName2)
|
|
return port, endpoint, teardown, teardownPG1, teardownPG2
|
|
}
|
|
|
|
setupTestEnv := func(t *testing.T, endpoint string) (p *ProjectMigrate, tempDir, projectDir string) {
|
|
t.Helper()
|
|
// create temp test dir
|
|
tempDir = copyTestdataToTempDir(t)
|
|
projectDir = fmt.Sprintf("%s/projectv3", tempDir)
|
|
// apply migrations
|
|
p, err := NewProjectMigrate(projectDir, WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(endpoint))
|
|
require.NoError(t, err)
|
|
_, err = p.Apply(ApplyOnAllDatabases())
|
|
require.NoError(t, err)
|
|
return p, tempDir, projectDir
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
deleteMigrationOpts []ProjectMigrationDeleterOption
|
|
configVersion int
|
|
want string
|
|
wantErr bool
|
|
assertErr require.ErrorAssertionFunc
|
|
}{
|
|
{
|
|
name: "can delete all migrations",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnDatabase("s1", hasura.SourceKindPG),
|
|
DeleteAllMigrations(),
|
|
},
|
|
configVersion: 3,
|
|
want: `[{"databaseName":"s1","status":{"migrations":[],"status":{}}},{"databaseName":"s2","status":{"migrations":[1623841477474,1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841477474":{"database_status":true,"source_status":true},"1623841485323":{"database_status":true,"source_status":true},"1623841492743":{"database_status":true,"source_status":true},"1623841500466":{"database_status":true,"source_status":true},"1623841510619":{"database_status":true,"source_status":true}}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
{
|
|
name: "can delete specific migration",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnDatabase("s1", hasura.SourceKindPG),
|
|
DeleteVersion(1623841477474),
|
|
},
|
|
configVersion: 3,
|
|
want: `[{"databaseName":"s1","status":{"migrations":[1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841485323":{"database_status":true,"source_status":true},"1623841492743":{"database_status":true,"source_status":true},"1623841500466":{"database_status":true,"source_status":true},"1623841510619":{"database_status":true,"source_status":true}}}},{"databaseName":"s2","status":{"migrations":[1623841477474,1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841477474":{"database_status":true,"source_status":true},"1623841485323":{"database_status":true,"source_status":true},"1623841492743":{"database_status":true,"source_status":true},"1623841500466":{"database_status":true,"source_status":true},"1623841510619":{"database_status":true,"source_status":true}}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
{
|
|
name: "can delete all migrations --server",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnDatabase("s2", hasura.SourceKindPG),
|
|
DeleteAllMigrations(),
|
|
DeleteOnlyOnServer(),
|
|
},
|
|
configVersion: 3,
|
|
want: `[{"databaseName":"s1","status":{"migrations":[1623841477474,1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841477474":{"database_status":true,"source_status":true},"1623841485323":{"database_status":true,"source_status":true},"1623841492743":{"database_status":true,"source_status":true},"1623841500466":{"database_status":true,"source_status":true},"1623841510619":{"database_status":true,"source_status":true}}}},{"databaseName":"s2","status":{"migrations":[1623841477474,1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841477474":{"database_status":false,"source_status":true},"1623841485323":{"database_status":false,"source_status":true},"1623841492743":{"database_status":false,"source_status":true},"1623841500466":{"database_status":false,"source_status":true},"1623841510619":{"database_status":false,"source_status":true}}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
{
|
|
name: "can delete specific migration --server",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnDatabase("s1", hasura.SourceKindPG),
|
|
DeleteVersion(1623841477474),
|
|
DeleteOnlyOnServer(),
|
|
},
|
|
configVersion: 3,
|
|
want: `[{"databaseName":"s1","status":{"migrations":[1623841477474,1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841477474":{"database_status":false,"source_status":true},"1623841485323":{"database_status":true,"source_status":true},"1623841492743":{"database_status":true,"source_status":true},"1623841500466":{"database_status":true,"source_status":true},"1623841510619":{"database_status":true,"source_status":true}}}},{"databaseName":"s2","status":{"migrations":[1623841477474,1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841477474":{"database_status":true,"source_status":true},"1623841485323":{"database_status":true,"source_status":true},"1623841492743":{"database_status":true,"source_status":true},"1623841500466":{"database_status":true,"source_status":true},"1623841510619":{"database_status":true,"source_status":true}}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
{
|
|
name: "can delete all migrations on all databases",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnAllDatabases(),
|
|
DeleteAllMigrations(),
|
|
},
|
|
configVersion: 3,
|
|
want: `[{"databaseName":"s1","status":{"migrations":[],"status":{}}},{"databaseName":"s2","status":{"migrations":[],"status":{}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
{
|
|
name: "can delete all migrations on all databases --server",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnAllDatabases(),
|
|
DeleteAllMigrations(),
|
|
DeleteOnlyOnServer(),
|
|
},
|
|
configVersion: 3,
|
|
want: `[{"databaseName":"s1","status":{"migrations":[1623841477474,1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841477474":{"database_status":false,"source_status":true},"1623841485323":{"database_status":false,"source_status":true},"1623841492743":{"database_status":false,"source_status":true},"1623841500466":{"database_status":false,"source_status":true},"1623841510619":{"database_status":false,"source_status":true}}}},{"databaseName":"s2","status":{"migrations":[1623841477474,1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841477474":{"database_status":false,"source_status":true},"1623841485323":{"database_status":false,"source_status":true},"1623841492743":{"database_status":false,"source_status":true},"1623841500466":{"database_status":false,"source_status":true},"1623841510619":{"database_status":false,"source_status":true}}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
{
|
|
name: "can delete specific migration on all databases config",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnAllDatabases(),
|
|
DeleteVersion(1623841477474),
|
|
},
|
|
configVersion: 3,
|
|
want: `[{"databaseName":"s1","status":{"migrations":[1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841485323":{"database_status":true,"source_status":true},"1623841492743":{"database_status":true,"source_status":true},"1623841500466":{"database_status":true,"source_status":true},"1623841510619":{"database_status":true,"source_status":true}}}},{"databaseName":"s2","status":{"migrations":[1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841485323":{"database_status":true,"source_status":true},"1623841492743":{"database_status":true,"source_status":true},"1623841500466":{"database_status":true,"source_status":true},"1623841510619":{"database_status":true,"source_status":true}}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
{
|
|
name: "can delete specific migration on all databases --server config",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnAllDatabases(),
|
|
DeleteVersion(1623841477474),
|
|
DeleteOnlyOnServer(),
|
|
},
|
|
configVersion: 3,
|
|
want: `[{"databaseName":"s1","status":{"migrations":[1623841477474,1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841477474":{"database_status":false,"source_status":true},"1623841485323":{"database_status":true,"source_status":true},"1623841492743":{"database_status":true,"source_status":true},"1623841500466":{"database_status":true,"source_status":true},"1623841510619":{"database_status":true,"source_status":true}}}},{"databaseName":"s2","status":{"migrations":[1623841477474,1623841485323,1623841492743,1623841500466,1623841510619],"status":{"1623841477474":{"database_status":false,"source_status":true},"1623841485323":{"database_status":true,"source_status":true},"1623841492743":{"database_status":true,"source_status":true},"1623841500466":{"database_status":true,"source_status":true},"1623841510619":{"database_status":true,"source_status":true}}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
}
|
|
|
|
for _, tc := range tests {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
_, endpoint, teardown, teardownPG1, teardownPG2 := startHasura(t, "s1", "s2")
|
|
defer func() {
|
|
teardownPG1()
|
|
teardownPG2()
|
|
teardown()
|
|
}()
|
|
p, tempDir, _ := setupTestEnv(t, endpoint)
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
// applied migrations status
|
|
// got, err := p.status([]ProjectMigrationStatusOption{}...)
|
|
// require.NoError(t, err)
|
|
// gotJSON, err := json.Marshal(got)
|
|
// require.NoError(t, err)
|
|
// log.Printf("Json Pre Del: %s\n", gotJSON)
|
|
|
|
err := p.Delete(tc.deleteMigrationOpts...)
|
|
tc.assertErr(t, err)
|
|
if tc.wantErr {
|
|
return
|
|
}
|
|
|
|
got, err := p.status([]ProjectMigrationStatusOption{}...)
|
|
require.NoError(t, err)
|
|
gotJSON, err := json.Marshal(got)
|
|
require.NoError(t, err)
|
|
|
|
var pretty_got, pretty_want bytes.Buffer
|
|
err = json.Indent(&pretty_got, gotJSON, "", " ")
|
|
require.NoError(t, err)
|
|
err = json.Indent(&pretty_want, []byte(tc.want), "", " ")
|
|
require.NoError(t, err)
|
|
// log.Printf("Json Post Del: %s\n", pretty_got.String())
|
|
assert.Equal(t, pretty_want.String(), pretty_got.String())
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestProjectMigrate_Delete_Configv2(t *testing.T) {
|
|
startHasura := func(t *testing.T) (port string, endpoint string, teardown func()) {
|
|
t.Helper()
|
|
port, teardown = testutil.StartHasura(t, testutil.HasuraDockerImage)
|
|
endpoint = fmt.Sprintf("http://localhost:%s", port)
|
|
return port, endpoint, teardown
|
|
}
|
|
setupTestEnv := func(t *testing.T, endpoint string) (p *ProjectMigrate, tempDir, projectDir string) {
|
|
t.Helper()
|
|
// create temp test dir
|
|
tempDir = copyTestdataToTempDir(t)
|
|
projectDir = fmt.Sprintf("%s/projectv2", tempDir)
|
|
// apply migrations
|
|
p, err := NewProjectMigrate(projectDir, WithAdminSecret(testutil.TestAdminSecret), WithEndpoint(endpoint))
|
|
require.NoError(t, err)
|
|
_, err = p.Apply(ApplyOnAllDatabases())
|
|
require.NoError(t, err)
|
|
return p, tempDir, projectDir
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
deleteMigrationOpts []ProjectMigrationDeleterOption
|
|
want string
|
|
wantErr bool
|
|
assertErr require.ErrorAssertionFunc
|
|
}{
|
|
{
|
|
name: "can delete all migrations",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnDatabase("", hasura.SourceKindPG),
|
|
DeleteAllMigrations(),
|
|
},
|
|
want: `[{"databaseName":"default","status":{"migrations":[],"status":{}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
{
|
|
name: "can delete specific migration",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnDatabase("", hasura.SourceKindPG),
|
|
DeleteVersion(1623842069725),
|
|
},
|
|
want: `[{"databaseName":"default","status":{"migrations":[1623842054907,1623842062104,1623842076537,1623842087940],"status":{"1623842054907":{"database_status":true,"source_status":true},"1623842062104":{"database_status":true,"source_status":true},"1623842076537":{"database_status":true,"source_status":true},"1623842087940":{"database_status":true,"source_status":true}}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
{
|
|
name: "can delete all migrations --server",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnDatabase("", hasura.SourceKindPG),
|
|
DeleteAllMigrations(),
|
|
DeleteOnlyOnServer(),
|
|
},
|
|
want: `[{"databaseName":"default","status":{"migrations":[1623842054907,1623842062104,1623842069725,1623842076537,1623842087940],"status":{"1623842054907":{"database_status":false,"source_status":true},"1623842062104":{"database_status":false,"source_status":true},"1623842069725":{"database_status":false,"source_status":true},"1623842076537":{"database_status":false,"source_status":true},"1623842087940":{"database_status":false,"source_status":true}}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
{
|
|
name: "can delete specific migration --server",
|
|
deleteMigrationOpts: []ProjectMigrationDeleterOption{
|
|
DeleteOnDatabase("", hasura.SourceKindPG),
|
|
DeleteVersion(1623842069725),
|
|
DeleteOnlyOnServer(),
|
|
},
|
|
want: `[{"databaseName":"default","status":{"migrations":[1623842054907,1623842062104,1623842069725,1623842076537,1623842087940],"status":{"1623842054907":{"database_status":true,"source_status":true},"1623842062104":{"database_status":true,"source_status":true},"1623842069725":{"database_status":false,"source_status":true},"1623842076537":{"database_status":true,"source_status":true},"1623842087940":{"database_status":true,"source_status":true}}}}]`,
|
|
wantErr: false,
|
|
assertErr: require.NoError,
|
|
},
|
|
}
|
|
for _, tc := range tests {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
_, endpoint, teardown := startHasura(t)
|
|
defer teardown()
|
|
p, tempDir, _ := setupTestEnv(t, endpoint)
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
// applied migrations status
|
|
// got, err := p.status([]ProjectMigrationStatusOption{}...)
|
|
// require.NoError(t, err)
|
|
// gotJSON, err := json.Marshal(got)
|
|
// require.NoError(t, err)
|
|
// log.Printf("Json Pre Del: %s\n", gotJSON)
|
|
|
|
err := p.Delete(tc.deleteMigrationOpts...)
|
|
tc.assertErr(t, err)
|
|
if tc.wantErr {
|
|
return
|
|
}
|
|
|
|
got, err := p.status([]ProjectMigrationStatusOption{}...)
|
|
require.NoError(t, err)
|
|
gotJSON, err := json.Marshal(got)
|
|
require.NoError(t, err)
|
|
|
|
var pretty_got, pretty_want bytes.Buffer
|
|
err = json.Indent(&pretty_got, gotJSON, "", " ")
|
|
require.NoError(t, err)
|
|
err = json.Indent(&pretty_want, []byte(tc.want), "", " ")
|
|
require.NoError(t, err)
|
|
// log.Printf("Json Post Del: %s\n", pretty_got.String())
|
|
assert.Equal(t, pretty_want.String(), pretty_got.String())
|
|
})
|
|
}
|
|
}
|
|
|
|
func copyTestdataToTempDir(t *testing.T) string {
|
|
t.Helper()
|
|
dir := testutil.RandDirName()
|
|
err := util.CopyDir("testdata", dir)
|
|
require.NoError(t, err)
|
|
return dir
|
|
}
|