Skip to content

Commit

Permalink
Add testcase to ensure new columns can't be used as the the identity …
Browse files Browse the repository at this point in the history
…column in a backfill. (#298)

Add a test to ensure that new columns can't be used as the identity for
a backfill, even if they satisfy the conditions for an identity column.

A newly added `NOT NULL` and `UNIQUE` column could in theory be used as
the identity column for a backfill. However, it shouldn't be used as the
identity column fora backfill because it's a newly added column whose
temporary column will be full of NULLs for any existing rows in the
table.

Currently the column won't be selected as an identity for backfills
because the nullability and uniqueness for new column are not populated
when adding it to the virtual schema in the add_column operation:


https://github.com/xataio/pgroll/blob/c08ef7065cdbce965110e64f58e66eddf605eaee/pkg/migrations/op_add_column.go#L61-L63

Following on from the discussion here:
#289 (comment)
  • Loading branch information
andrew-farries authored Mar 1, 2024
1 parent 52fb532 commit 431b951
Showing 1 changed file with 121 additions and 78 deletions.
199 changes: 121 additions & 78 deletions pkg/migrations/op_add_column_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,98 +15,141 @@ import (
func TestAddColumn(t *testing.T) {
t.Parallel()

ExecuteTests(t, TestCases{{
name: "add column",
migrations: []migrations.Migration{
{
Name: "01_add_table",
Operations: migrations.Operations{
&migrations.OpCreateTable{
Name: "users",
Columns: []migrations.Column{
{
Name: "id",
Type: "serial",
Pk: ptr(true),
},
{
Name: "name",
Type: "varchar(255)",
Unique: ptr(true),
ExecuteTests(t, TestCases{
{
name: "add column",
migrations: []migrations.Migration{
{
Name: "01_add_table",
Operations: migrations.Operations{
&migrations.OpCreateTable{
Name: "users",
Columns: []migrations.Column{
{
Name: "id",
Type: "serial",
Pk: ptr(true),
},
{
Name: "name",
Type: "varchar(255)",
Unique: ptr(true),
},
},
},
},
},
},
{
Name: "02_add_column",
Operations: migrations.Operations{
&migrations.OpAddColumn{
Table: "users",
Column: migrations.Column{
Name: "age",
Type: "integer",
Nullable: ptr(false),
Default: ptr("0"),
Comment: ptr("the age of the user"),
{
Name: "02_add_column",
Operations: migrations.Operations{
&migrations.OpAddColumn{
Table: "users",
Column: migrations.Column{
Name: "age",
Type: "integer",
Nullable: ptr(false),
Default: ptr("0"),
Comment: ptr("the age of the user"),
},
},
},
},
},
},
afterStart: func(t *testing.T, db *sql.DB, schema string) {
// old and new views of the table should exist
ViewMustExist(t, db, schema, "01_add_table", "users")
ViewMustExist(t, db, schema, "02_add_column", "users")
afterStart: func(t *testing.T, db *sql.DB, schema string) {
// old and new views of the table should exist
ViewMustExist(t, db, schema, "01_add_table", "users")
ViewMustExist(t, db, schema, "02_add_column", "users")

// inserting via both the old and the new views works
MustInsert(t, db, schema, "01_add_table", "users", map[string]string{
"name": "Alice",
})
MustInsert(t, db, schema, "02_add_column", "users", map[string]string{
"name": "Bob",
"age": "21",
})
// inserting via both the old and the new views works
MustInsert(t, db, schema, "01_add_table", "users", map[string]string{
"name": "Alice",
})
MustInsert(t, db, schema, "02_add_column", "users", map[string]string{
"name": "Bob",
"age": "21",
})

// selecting from both the old and the new views works
resOld := MustSelect(t, db, schema, "01_add_table", "users")
assert.Equal(t, []map[string]any{
{"id": 1, "name": "Alice"},
{"id": 2, "name": "Bob"},
}, resOld)
resNew := MustSelect(t, db, schema, "02_add_column", "users")
assert.Equal(t, []map[string]any{
{"id": 1, "name": "Alice", "age": 0},
{"id": 2, "name": "Bob", "age": 21},
}, resNew)
},
afterRollback: func(t *testing.T, db *sql.DB, schema string) {
// The new column has been dropped from the underlying table
columnName := migrations.TemporaryName("age")
ColumnMustNotExist(t, db, schema, "users", columnName)
// selecting from both the old and the new views works
resOld := MustSelect(t, db, schema, "01_add_table", "users")
assert.Equal(t, []map[string]any{
{"id": 1, "name": "Alice"},
{"id": 2, "name": "Bob"},
}, resOld)
resNew := MustSelect(t, db, schema, "02_add_column", "users")
assert.Equal(t, []map[string]any{
{"id": 1, "name": "Alice", "age": 0},
{"id": 2, "name": "Bob", "age": 21},
}, resNew)
},
afterRollback: func(t *testing.T, db *sql.DB, schema string) {
// The new column has been dropped from the underlying table
columnName := migrations.TemporaryName("age")
ColumnMustNotExist(t, db, schema, "users", columnName)

// The table's column count reflects the drop of the new column
TableMustHaveColumnCount(t, db, schema, "users", 2)
},
afterComplete: func(t *testing.T, db *sql.DB, schema string) {
// The new view still exists
ViewMustExist(t, db, schema, "02_add_column", "users")
// The table's column count reflects the drop of the new column
TableMustHaveColumnCount(t, db, schema, "users", 2)
},
afterComplete: func(t *testing.T, db *sql.DB, schema string) {
// The new view still exists
ViewMustExist(t, db, schema, "02_add_column", "users")

// Inserting into the new view still works
MustInsert(t, db, schema, "02_add_column", "users", map[string]string{
"name": "Carl",
"age": "31",
})
// Inserting into the new view still works
MustInsert(t, db, schema, "02_add_column", "users", map[string]string{
"name": "Carl",
"age": "31",
})

// Selecting from the new view still works
res := MustSelect(t, db, schema, "02_add_column", "users")
assert.Equal(t, []map[string]any{
{"id": 1, "name": "Alice", "age": 0},
{"id": 2, "name": "Bob", "age": 0},
{"id": 3, "name": "Carl", "age": 31},
}, res)
// Selecting from the new view still works
res := MustSelect(t, db, schema, "02_add_column", "users")
assert.Equal(t, []map[string]any{
{"id": 1, "name": "Alice", "age": 0},
{"id": 2, "name": "Bob", "age": 0},
{"id": 3, "name": "Carl", "age": 31},
}, res)
},
},
}})
{
name: "a newly added column can't be used as the identity column for a backfill",
migrations: []migrations.Migration{
{
Name: "01_add_table",
Operations: migrations.Operations{
&migrations.OpCreateTable{
Name: "users",
Columns: []migrations.Column{
{
Name: "name",
Type: "varchar(255)",
},
},
},
},
},
{
// This column is NOT NULL and UNIQUE and is added to a table without
// a PK, so it could in theory be used as the identity column for a
// backfill. However, it shouldn't be used as the identity column for
// a backfill because it's a newly added column whose temporary
// `_pgroll_new_description` column will be full of NULLs for any
// existing rows in the table.
Name: "02_add_column",
Operations: migrations.Operations{
&migrations.OpAddColumn{
Table: "users",
Column: migrations.Column{
Name: "description",
Type: "integer",
Nullable: ptr(false),
Unique: ptr(true),
},
Up: ptr("'this is a description'"),
},
},
},
},
wantStartErr: migrations.BackfillNotPossibleError{Table: "users"},
},
})
}

func TestAddForeignKeyColumn(t *testing.T) {
Expand Down

0 comments on commit 431b951

Please sign in to comment.