-
Notifications
You must be signed in to change notification settings - Fork 28
/
Copy pathnullability.go
42 lines (38 loc) · 1.4 KB
/
nullability.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
package pginfer
import (
"github.com/jschaf/pggen/internal/ast"
"github.com/jschaf/pggen/internal/pg"
"strings"
"unicode"
)
// isColNullable tries to prove the column is not nullable. Strive for
// correctness here: it's better to assume a column is nullable when we can't
// know for sure.
func isColNullable(query *ast.SourceQuery, plan Plan, out string, column pg.Column) bool {
switch {
case len(out) == 0:
// No output? Not sure what this means but do the check here so that we
// don't have to do it in each case below.
return false
case strings.HasPrefix(out, "'"):
return false // literal string can't be null
case unicode.IsDigit(rune(out[0])):
return false // literal number can't be null
default:
// try below
}
// A plain select query (possibly with a LIMIT clause) with no joins where
// the column comes from a table and has a not-null constraint. Not full
// proof because of cross-join with comma syntax.
if (plan.Type == PlanResult || plan.Type == PlanLimit) &&
!strings.Contains(strings.ToLower(query.PreparedSQL), "join") &&
!column.Null {
return false
}
// A returning clause in an insert, update, or delete statement. The column
// must come from the underlying table and must have a not null constraint.
if plan.Type == PlanModifyTable && plan.Relation == column.TableName && !column.Null {
return false
}
return true // we can't figure it out; assume nullable
}