2019-08-03 14:10:47 +02:00
|
|
|
package postgres
|
|
|
|
|
|
|
|
|
|
import (
|
2020-06-27 18:48:19 +02:00
|
|
|
"github.com/go-jet/jet/v2/internal/jet"
|
2019-08-03 14:10:47 +02:00
|
|
|
"strconv"
|
|
|
|
|
)
|
|
|
|
|
|
2019-08-17 14:49:35 +02:00
|
|
|
// Dialect is implementation of postgres dialect for SQL Builder serialisation.
|
|
|
|
|
var Dialect = newDialect()
|
2019-08-03 14:10:47 +02:00
|
|
|
|
2019-08-17 14:49:35 +02:00
|
|
|
func newDialect() jet.Dialect {
|
2019-08-03 14:10:47 +02:00
|
|
|
|
2019-08-15 13:54:05 +02:00
|
|
|
operatorSerializeOverrides := map[string]jet.SerializeOverride{}
|
2019-09-20 19:54:56 +02:00
|
|
|
operatorSerializeOverrides[jet.StringRegexpLikeOperator] = postgresREGEXPLIKEoperator
|
|
|
|
|
operatorSerializeOverrides[jet.StringNotRegexpLikeOperator] = postgresNOTREGEXPLIKEoperator
|
2019-08-15 13:54:05 +02:00
|
|
|
operatorSerializeOverrides["CAST"] = postgresCAST
|
2019-08-06 10:29:04 +02:00
|
|
|
|
2019-08-03 14:10:47 +02:00
|
|
|
dialectParams := jet.DialectParams{
|
2019-08-15 13:54:05 +02:00
|
|
|
Name: "PostgreSQL",
|
|
|
|
|
PackageName: "postgres",
|
|
|
|
|
OperatorSerializeOverrides: operatorSerializeOverrides,
|
|
|
|
|
AliasQuoteChar: '"',
|
|
|
|
|
IdentifierQuoteChar: '"',
|
2019-08-03 14:10:47 +02:00
|
|
|
ArgumentPlaceholder: func(ord int) string {
|
|
|
|
|
return "$" + strconv.Itoa(ord)
|
|
|
|
|
},
|
2020-02-16 10:25:21 +01:00
|
|
|
ReservedWords: reservedWords,
|
2019-08-03 14:10:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return jet.NewDialect(dialectParams)
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-01 18:26:01 +01:00
|
|
|
func postgresCAST(expressions ...jet.Serializer) jet.SerializerFunc {
|
2019-08-17 18:32:01 +02:00
|
|
|
return func(statement jet.StatementType, out *jet.SQLBuilder, options ...jet.SerializeOption) {
|
2019-08-06 12:12:48 +02:00
|
|
|
if len(expressions) < 2 {
|
2019-08-13 13:57:26 +02:00
|
|
|
panic("jet: invalid number of expressions for operator")
|
2019-08-06 12:12:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
expression := expressions[0]
|
|
|
|
|
|
|
|
|
|
litExpr, ok := expressions[1].(jet.LiteralExpression)
|
|
|
|
|
|
|
|
|
|
if !ok {
|
2019-08-13 13:57:26 +02:00
|
|
|
panic("jet: cast invalid cast type")
|
2019-08-06 12:12:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
castType, ok := litExpr.Value().(string)
|
|
|
|
|
|
|
|
|
|
if !ok {
|
2019-08-13 13:57:26 +02:00
|
|
|
panic("jet: cast type is not string")
|
2019-08-06 12:12:48 +02:00
|
|
|
}
|
|
|
|
|
|
2019-08-13 13:57:26 +02:00
|
|
|
jet.Serialize(expression, statement, out, options...)
|
2019-08-03 14:10:47 +02:00
|
|
|
out.WriteString("::" + castType)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-01 18:26:01 +01:00
|
|
|
func postgresREGEXPLIKEoperator(expressions ...jet.Serializer) jet.SerializerFunc {
|
2019-08-17 18:32:01 +02:00
|
|
|
return func(statement jet.StatementType, out *jet.SQLBuilder, options ...jet.SerializeOption) {
|
2019-08-06 10:29:04 +02:00
|
|
|
if len(expressions) < 2 {
|
2019-08-13 13:57:26 +02:00
|
|
|
panic("jet: invalid number of expressions for operator")
|
2019-08-06 10:29:04 +02:00
|
|
|
}
|
|
|
|
|
|
2019-08-13 13:57:26 +02:00
|
|
|
jet.Serialize(expressions[0], statement, out, options...)
|
2019-08-06 10:29:04 +02:00
|
|
|
|
|
|
|
|
caseSensitive := false
|
|
|
|
|
|
|
|
|
|
if len(expressions) >= 3 {
|
|
|
|
|
if stringLiteral, ok := expressions[2].(jet.LiteralExpression); ok {
|
2019-08-15 11:10:02 +02:00
|
|
|
caseSensitive = stringLiteral.Value().(bool)
|
2019-08-06 10:29:04 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if caseSensitive {
|
|
|
|
|
out.WriteString("~")
|
|
|
|
|
} else {
|
|
|
|
|
out.WriteString("~*")
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-13 13:57:26 +02:00
|
|
|
jet.Serialize(expressions[1], statement, out, options...)
|
2019-08-06 10:29:04 +02:00
|
|
|
}
|
|
|
|
|
}
|
2019-08-15 11:10:02 +02:00
|
|
|
|
2019-12-01 18:26:01 +01:00
|
|
|
func postgresNOTREGEXPLIKEoperator(expressions ...jet.Serializer) jet.SerializerFunc {
|
2019-08-17 18:32:01 +02:00
|
|
|
return func(statement jet.StatementType, out *jet.SQLBuilder, options ...jet.SerializeOption) {
|
2019-08-15 11:10:02 +02:00
|
|
|
if len(expressions) < 2 {
|
|
|
|
|
panic("jet: invalid number of expressions for operator")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jet.Serialize(expressions[0], statement, out, options...)
|
|
|
|
|
|
|
|
|
|
caseSensitive := false
|
|
|
|
|
|
|
|
|
|
if len(expressions) >= 3 {
|
|
|
|
|
if stringLiteral, ok := expressions[2].(jet.LiteralExpression); ok {
|
|
|
|
|
caseSensitive = stringLiteral.Value().(bool)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if caseSensitive {
|
|
|
|
|
out.WriteString("!~")
|
|
|
|
|
} else {
|
|
|
|
|
out.WriteString("!~*")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jet.Serialize(expressions[1], statement, out, options...)
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-02-16 10:25:21 +01:00
|
|
|
|
|
|
|
|
var reservedWords = []string{
|
|
|
|
|
"ALL",
|
|
|
|
|
"ANALYSE",
|
|
|
|
|
"ANALYZE",
|
|
|
|
|
"AND",
|
|
|
|
|
"ANY",
|
|
|
|
|
"ARRAY",
|
|
|
|
|
"AS",
|
|
|
|
|
"ASC",
|
|
|
|
|
"ASYMMETRIC",
|
|
|
|
|
"BOTH",
|
|
|
|
|
"CASE",
|
|
|
|
|
"CAST",
|
|
|
|
|
"CHECK",
|
|
|
|
|
"COLLATE",
|
|
|
|
|
"COLUMN",
|
|
|
|
|
"CONSTRAINT",
|
|
|
|
|
"CREATE",
|
|
|
|
|
"CURRENT_CATALOG",
|
|
|
|
|
"CURRENT_DATE",
|
|
|
|
|
"CURRENT_ROLE",
|
|
|
|
|
"CURRENT_TIME",
|
|
|
|
|
"CURRENT_TIMESTAMP",
|
|
|
|
|
"CURRENT_USER",
|
|
|
|
|
"DEFAULT",
|
|
|
|
|
"DEFERRABLE",
|
|
|
|
|
"DESC",
|
|
|
|
|
"DISTINCT",
|
|
|
|
|
"DO",
|
|
|
|
|
"ELSE",
|
|
|
|
|
"END",
|
|
|
|
|
"EXCEPT",
|
|
|
|
|
"FALSE",
|
|
|
|
|
"FETCH",
|
|
|
|
|
"FOR",
|
|
|
|
|
"FOREIGN",
|
|
|
|
|
"FROM",
|
|
|
|
|
"GRANT",
|
|
|
|
|
"GROUP",
|
|
|
|
|
"HAVING",
|
|
|
|
|
"IN",
|
|
|
|
|
"INITIALLY",
|
|
|
|
|
"INTERSECT",
|
|
|
|
|
"INTO",
|
|
|
|
|
"LATERAL",
|
|
|
|
|
"LEADING",
|
|
|
|
|
"LIMIT",
|
|
|
|
|
"LOCALTIME",
|
|
|
|
|
"LOCALTIMESTAMP",
|
|
|
|
|
"NOT",
|
|
|
|
|
"NULL",
|
|
|
|
|
"OFFSET",
|
|
|
|
|
"ON",
|
|
|
|
|
"ONLY",
|
|
|
|
|
"OR",
|
|
|
|
|
"ORDER",
|
|
|
|
|
"PLACING",
|
|
|
|
|
"PRIMARY",
|
|
|
|
|
"REFERENCES",
|
|
|
|
|
"RETURNING",
|
|
|
|
|
"SELECT",
|
|
|
|
|
"SESSION_USER",
|
|
|
|
|
"SOME",
|
|
|
|
|
"SYMMETRIC",
|
|
|
|
|
"TABLE",
|
|
|
|
|
"THEN",
|
|
|
|
|
"TO",
|
|
|
|
|
"TRAILING",
|
|
|
|
|
"TRUE",
|
|
|
|
|
"UNION",
|
|
|
|
|
"UNIQUE",
|
|
|
|
|
"USER",
|
|
|
|
|
"USING",
|
|
|
|
|
"VARIADIC",
|
|
|
|
|
"WHEN",
|
|
|
|
|
"WHERE",
|
|
|
|
|
"WINDOW",
|
|
|
|
|
"WITH",
|
|
|
|
|
}
|