From 783d282de7110c574cec34769393299da77c5354 Mon Sep 17 00:00:00 2001 From: Andrew Farries Date: Thu, 2 Jan 2025 09:06:06 +0000 Subject: [PATCH] Fix `ALTER TABLE ADD COLUMN` default nullability (#558) Columns added by `ALTER TABLE ADD COLUMN` are nullable by default. For the `pgroll` `Column` type, these means we need to set `Nullable` to `true` by default when converting such SQL statements with `sql2pgroll`. --- pkg/sql2pgroll/alter_table.go | 8 +++-- pkg/sql2pgroll/alter_table_test.go | 2 +- pkg/sql2pgroll/expect/add_column.go | 49 +++++++++++++++++++---------- 3 files changed, 40 insertions(+), 19 deletions(-) diff --git a/pkg/sql2pgroll/alter_table.go b/pkg/sql2pgroll/alter_table.go index 1380f12a..6f581017 100644 --- a/pkg/sql2pgroll/alter_table.go +++ b/pkg/sql2pgroll/alter_table.go @@ -418,8 +418,9 @@ func convertAlterTableAddColumn(stmt *pgq.AlterTableStmt, cmd *pgq.AlterTableCmd operation := &migrations.OpAddColumn{ Column: migrations.Column{ - Name: columnDef.GetColname(), - Type: columnType, + Name: columnDef.GetColname(), + Type: columnType, + Nullable: true, }, Table: getQualifiedRelationName(stmt.GetRelation()), Up: PlaceHolderSQL, @@ -430,8 +431,11 @@ func convertAlterTableAddColumn(stmt *pgq.AlterTableStmt, cmd *pgq.AlterTableCmd switch constraint.GetConstraint().GetContype() { case pgq.ConstrType_CONSTR_NULL: operation.Column.Nullable = true + case pgq.ConstrType_CONSTR_NOTNULL: + operation.Column.Nullable = false case pgq.ConstrType_CONSTR_PRIMARY: operation.Column.Pk = true + operation.Column.Nullable = false case pgq.ConstrType_CONSTR_UNIQUE: operation.Column.Unique = true case pgq.ConstrType_CONSTR_CHECK: diff --git a/pkg/sql2pgroll/alter_table_test.go b/pkg/sql2pgroll/alter_table_test.go index f74eace5..96f90348 100644 --- a/pkg/sql2pgroll/alter_table_test.go +++ b/pkg/sql2pgroll/alter_table_test.go @@ -148,7 +148,7 @@ func TestConvertAlterTableStatements(t *testing.T) { }, { sql: "ALTER TABLE foo ADD COLUMN bar int NOT NULL", - expectedOp: expect.AddColumnOp1, + expectedOp: expect.AddColumnOp8, }, { sql: "ALTER TABLE schema.foo ADD COLUMN bar int", diff --git a/pkg/sql2pgroll/expect/add_column.go b/pkg/sql2pgroll/expect/add_column.go index 76de2583..2df82c28 100644 --- a/pkg/sql2pgroll/expect/add_column.go +++ b/pkg/sql2pgroll/expect/add_column.go @@ -11,8 +11,9 @@ var AddColumnOp1 = &migrations.OpAddColumn{ Table: "foo", Up: sql2pgroll.PlaceHolderSQL, Column: migrations.Column{ - Name: "bar", - Type: "int", + Name: "bar", + Type: "int", + Nullable: true, }, } @@ -20,8 +21,9 @@ var AddColumnOp2 = &migrations.OpAddColumn{ Table: "schema.foo", Up: sql2pgroll.PlaceHolderSQL, Column: migrations.Column{ - Name: "bar", - Type: "int", + Name: "bar", + Type: "int", + Nullable: true, }, } @@ -30,9 +32,10 @@ func AddColumnOp1WithDefault(def *string) *migrations.OpAddColumn { Table: "foo", Up: sql2pgroll.PlaceHolderSQL, Column: migrations.Column{ - Name: "bar", - Type: "int", - Default: def, + Name: "bar", + Type: "int", + Default: def, + Nullable: true, }, } } @@ -51,9 +54,10 @@ var AddColumnOp4 = &migrations.OpAddColumn{ Table: "foo", Up: sql2pgroll.PlaceHolderSQL, Column: migrations.Column{ - Name: "bar", - Type: "int", - Unique: true, + Name: "bar", + Type: "int", + Unique: true, + Nullable: true, }, } @@ -71,8 +75,9 @@ var AddColumnOp6 = &migrations.OpAddColumn{ Table: "foo", Up: sql2pgroll.PlaceHolderSQL, Column: migrations.Column{ - Name: "bar", - Type: "int", + Name: "bar", + Type: "int", + Nullable: true, Check: &migrations.CheckConstraint{ Constraint: "bar > 0", Name: "", @@ -84,8 +89,9 @@ var AddColumnOp7 = &migrations.OpAddColumn{ Table: "foo", Up: sql2pgroll.PlaceHolderSQL, Column: migrations.Column{ - Name: "bar", - Type: "int", + Name: "bar", + Type: "int", + Nullable: true, Check: &migrations.CheckConstraint{ Constraint: "bar > 0", Name: "check_bar", @@ -93,13 +99,24 @@ var AddColumnOp7 = &migrations.OpAddColumn{ }, } +var AddColumnOp8 = &migrations.OpAddColumn{ + Table: "foo", + Up: sql2pgroll.PlaceHolderSQL, + Column: migrations.Column{ + Name: "bar", + Type: "int", + Nullable: false, + }, +} + func AddColumnOp8WithOnDeleteAction(action migrations.ForeignKeyReferenceOnDelete) *migrations.OpAddColumn { return &migrations.OpAddColumn{ Table: "foo", Up: sql2pgroll.PlaceHolderSQL, Column: migrations.Column{ - Name: "bar", - Type: "int", + Name: "bar", + Type: "int", + Nullable: true, References: &migrations.ForeignKeyReference{ Column: "bar", Name: "fk_baz",