Skip to content

Commit

Permalink
Allow some named inline column constraints in CREATE TABLE statemen…
Browse files Browse the repository at this point in the history
…ts (#559)

Allow some named inline column constraints in `CREATE TABLE` statements.

SQL like this can be converted to a pgroll `OpCreateTable`:

```sql
CREATE TABLE foo(a int CONSTRAINT my_check CHECK (a > 0))
CREATE TABLE foo(a int CONSTRAINT my_fk REFERENCES bar(b))
```

The previous behaviour was to allow only the unnamed forms of these
constraints:

```sql
CREATE TABLE foo(a int CHECK (a > 0))
CREATE TABLE foo(a int REFERENCES bar(b))
```

Named inline column constraints remain unsupported for `DEFAULT`,
`NULL`, `NOT NULL`, `UNIQUE` and `PRIMARY KEY` constraints and will fall
back to a raw SQL operation.
  • Loading branch information
andrew-farries authored Jan 2, 2025
1 parent 783d282 commit ca094f5
Show file tree
Hide file tree
Showing 3 changed files with 80 additions and 14 deletions.
41 changes: 32 additions & 9 deletions pkg/sql2pgroll/create_table.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@ package sql2pgroll

import (
"fmt"
"slices"

pgq "github.com/xataio/pg_query_go/v6"

Expand Down Expand Up @@ -87,14 +86,6 @@ func convertColumnDef(tableName string, col *pgq.ColumnDef) (*migrations.Column,
return nil, fmt.Errorf("error deparsing column type: %w", err)
}

// Named inline constraints are not supported
anyNamed := slices.ContainsFunc(col.GetConstraints(), func(c *pgq.Node) bool {
return c.GetConstraint().GetConname() != ""
})
if anyNamed {
return nil, nil
}

// Convert column constraints
var notNull, pk, unique bool
var check *migrations.CheckConstraint
Expand All @@ -103,15 +94,31 @@ func convertColumnDef(tableName string, col *pgq.ColumnDef) (*migrations.Column,
for _, c := range col.GetConstraints() {
switch c.GetConstraint().GetContype() {
case pgq.ConstrType_CONSTR_NULL:
// named NULL constraints are not supported
if isConstraintNamed(c.GetConstraint()) {
return nil, nil
}
notNull = false
case pgq.ConstrType_CONSTR_NOTNULL:
// named NOT NULL constraints are not supported
if isConstraintNamed(c.GetConstraint()) {
return nil, nil
}
notNull = true
case pgq.ConstrType_CONSTR_UNIQUE:
// named UNIQUE constraints are not supported
if isConstraintNamed(c.GetConstraint()) {
return nil, nil
}
if !canConvertUniqueConstraint(c.GetConstraint()) {
return nil, nil
}
unique = true
case pgq.ConstrType_CONSTR_PRIMARY:
// named PRIMARY KEY constraints are not supported
if isConstraintNamed(c.GetConstraint()) {
return nil, nil
}
if !canConvertPrimaryKeyConstraint(c.GetConstraint()) {
return nil, nil
}
Expand All @@ -126,6 +133,10 @@ func convertColumnDef(tableName string, col *pgq.ColumnDef) (*migrations.Column,
return nil, nil
}
case pgq.ConstrType_CONSTR_DEFAULT:
// named DEFAULT constraints are not supported
if isConstraintNamed(c.GetConstraint()) {
return nil, nil
}
d, err := pgq.DeparseExpr(c.GetConstraint().GetRawExpr())
if err != nil {
return nil, fmt.Errorf("error deparsing default value: %w", err)
Expand Down Expand Up @@ -254,3 +265,15 @@ func convertInlineForeignKeyConstraint(tableName, columnName string, constraint
Table: getQualifiedRelationName(constraint.GetPktable()),
}, nil
}

// isConstraintNamed returns true iff `constraint` has a name.
// Column constraints defined inline in CREATE TABLE statements can be either
// named or unnamed, for example:
// - CREATE TABLE t (a INT PRIMARY KEY);
// - CREATE TABLE t (a INT CONSTRAINT my_pk PRIMARY KEY);
// Likewise, table constraints can also be either named or unnamed, for example:
// - CREATE TABLE foo(a int, CONSTRAINT foo_check CHECK (a > 0)),
// - CREATE TABLE foo(a int, CHECK (a > 0)),
func isConstraintNamed(constraint *pgq.Constraint) bool {
return constraint.GetConname() != ""
}
21 changes: 16 additions & 5 deletions pkg/sql2pgroll/create_table_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -56,10 +56,22 @@ func TestConvertCreateTableStatements(t *testing.T) {
sql: "CREATE TABLE foo(a int CHECK (a > 0))",
expectedOp: expect.CreateTableOp10,
},
{
sql: "CREATE TABLE foo(a int CONSTRAINT my_check CHECK (a > 0))",
expectedOp: expect.CreateTableOp18,
},
{
sql: "CREATE TABLE foo(a timestamptz DEFAULT now())",
expectedOp: expect.CreateTableOp11,
},
{
sql: "CREATE TABLE foo(a int CONSTRAINT my_fk REFERENCES bar(b))",
expectedOp: expect.CreateTableOp19,
},
{
sql: "CREATE TABLE foo(a int REFERENCES bar(b))",
expectedOp: expect.CreateTableOp12,
},
{
sql: "CREATE TABLE foo(a int REFERENCES bar(b) NOT DEFERRABLE)",
expectedOp: expect.CreateTableOp12,
Expand Down Expand Up @@ -227,14 +239,13 @@ func TestUnconvertableCreateTableStatements(t *testing.T) {
"CREATE TABLE foo(a int REFERENCES bar (b) ON UPDATE SET DEFAULT)",
"CREATE TABLE foo(a int REFERENCES bar (b) MATCH FULL)",

// Named inline constraints are not supported
"CREATE TABLE foo(a int CONSTRAINT foo_check CHECK (a > 0))",
"CREATE TABLE foo(a int CONSTRAINT foo_unique UNIQUE)",
"CREATE TABLE foo(a int CONSTRAINT foo_pk PRIMARY KEY)",
"CREATE TABLE foo(a int CONSTRAINT foo_fk REFERENCES bar(b))",
// Named inline constraints are not supported for DEFAULT, NULL, NOT NULL,
// UNIQUE or PRIMARY KEY constraints
"CREATE TABLE foo(a int CONSTRAINT foo_default DEFAULT 0)",
"CREATE TABLE foo(a int CONSTRAINT foo_null NULL)",
"CREATE TABLE foo(a int CONSTRAINT foo_notnull NOT NULL)",
"CREATE TABLE foo(a int CONSTRAINT foo_unique UNIQUE)",
"CREATE TABLE foo(a int CONSTRAINT foo_pk PRIMARY KEY)",

// Generated columns are not supported
"CREATE TABLE foo(a int GENERATED ALWAYS AS (1) STORED)",
Expand Down
32 changes: 32 additions & 0 deletions pkg/sql2pgroll/expect/create_table.go
Original file line number Diff line number Diff line change
Expand Up @@ -225,3 +225,35 @@ var CreateTableOp17 = &migrations.OpCreateTable{
},
},
}

var CreateTableOp18 = &migrations.OpCreateTable{
Name: "foo",
Columns: []migrations.Column{
{
Name: "a",
Type: "int",
Nullable: true,
Check: &migrations.CheckConstraint{
Name: "my_check",
Constraint: "a > 0",
},
},
},
}

var CreateTableOp19 = &migrations.OpCreateTable{
Name: "foo",
Columns: []migrations.Column{
{
Name: "a",
Type: "int",
Nullable: true,
References: &migrations.ForeignKeyReference{
Name: "my_fk",
Table: "bar",
Column: "b",
OnDelete: migrations.ForeignKeyReferenceOnDeleteNOACTION,
},
},
},
}

0 comments on commit ca094f5

Please sign in to comment.