jet/postgres/dialect.go

185 lines
3.4 KiB
Go
Raw Normal View History

2019-08-03 14:10:47 +02:00
package postgres
import (
2025-02-28 18:23:15 +01:00
"encoding/hex"
2024-10-17 14:12:21 +02:00
"fmt"
"strconv"
"github.com/go-jet/jet/v2/internal/jet"
2019-08-03 14:10:47 +02:00
)
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
dialectParams := jet.DialectParams{
2019-08-15 13:54:05 +02:00
Name: "PostgreSQL",
PackageName: "postgres",
2026-02-02 13:21:35 +01:00
OperatorSerializeOverrides: nil,
2019-08-15 13:54:05 +02:00
AliasQuoteChar: '"',
IdentifierQuoteChar: '"',
2019-08-03 14:10:47 +02:00
ArgumentPlaceholder: func(ord int) string {
return "$" + strconv.Itoa(ord)
},
2025-02-28 18:23:15 +01:00
ArgumentToString: argumentToString,
ReservedWords: reservedWords,
2024-10-17 14:12:21 +02:00
ValuesDefaultColumnName: func(index int) string {
return fmt.Sprintf("column%d", index+1)
},
JsonValueEncode: func(expr Expression) Expression {
switch e := expr.(type) {
case ByteaExpression:
return ENCODE(e, Base64)
// CustomExpression used bellow (instead TO_CHAR function) so that only expr is parametrized
case TimeExpression:
return CustomExpression(Token("'0000-01-01T' || to_char('2000-10-10'::date + "), e, Token(`, 'HH24:MI:SS.USZ')`))
case TimezExpression:
return CustomExpression(Token("'0000-01-01T' || to_char('2000-10-10'::date + "), e, Token(`, 'HH24:MI:SS.USTZH:TZM')`))
case TimestampExpression:
return jet.AtomicCustomExpression(Token("to_char("), e, Token(`, 'YYYY-MM-DD"T"HH24:MI:SS.USZ')`))
case DateExpression:
return CustomExpression(Token("to_char("), e, Token(`::timestamp, 'YYYY-MM-DD') || 'T00:00:00Z'`))
}
return expr
},
2026-02-02 13:21:35 +01:00
RegexpLike: regexpLike,
2019-08-03 14:10:47 +02:00
}
return jet.NewDialect(dialectParams)
}
2025-02-28 18:23:15 +01:00
func argumentToString(value any) (string, bool) {
switch bindVal := value.(type) {
case []byte:
return fmt.Sprintf("'\\x%s'", hex.EncodeToString(bindVal)), true
}
return "", false
}
2026-02-02 13:21:35 +01:00
func regexpLike(str jet.StringExpression, not bool, pattern jet.StringExpression, caseSensitive bool) jet.SerializerFunc {
2019-08-17 18:32:01 +02:00
return func(statement jet.StatementType, out *jet.SQLBuilder, options ...jet.SerializeOption) {
2026-02-02 13:21:35 +01:00
jet.Serialize(str, statement, out, options...)
2019-08-15 11:10:02 +02:00
2026-02-02 13:21:35 +01:00
var notOperator string
2019-08-15 11:10:02 +02:00
2026-02-02 13:21:35 +01:00
if not {
notOperator = "!"
2019-08-15 11:10:02 +02:00
}
if caseSensitive {
2026-02-02 13:21:35 +01:00
out.WriteString(notOperator + "~")
2019-08-15 11:10:02 +02:00
} else {
2026-02-02 13:21:35 +01:00
out.WriteString(notOperator + "~*")
2019-08-15 11:10:02 +02:00
}
2026-02-02 13:21:35 +01:00
jet.Serialize(pattern, statement, out, options...)
2019-08-15 11:10:02 +02:00
}
}
var reservedWords = []string{
"ALL",
"ANALYSE",
"ANALYZE",
"AND",
"ANY",
"ARRAY",
"AS",
"ASC",
"ASYMMETRIC",
2024-12-23 13:22:29 +03:00
"AUTHORIZATION",
"BINARY",
"BOTH",
"CASE",
"CAST",
"CHECK",
"COLLATE",
"COLLATION",
"COLUMN",
"CONCURRENTLY",
"CONSTRAINT",
"CREATE",
"CROSS",
"CURRENT_CATALOG",
"CURRENT_DATE",
"CURRENT_ROLE",
"CURRENT_SCHEMA",
"CURRENT_TIME",
"CURRENT_TIMESTAMP",
"CURRENT_USER",
"DEFAULT",
"DEFERRABLE",
"DESC",
"DISTINCT",
"DO",
"ELSE",
"END",
"EXCEPT",
"FALSE",
"FETCH",
"FOR",
"FOREIGN",
"FREEZE",
"FROM",
"FULL",
"GRANT",
"GROUP",
"HAVING",
"ILIKE",
"IN",
"INITIALLY",
"INNER",
"INTERSECT",
"INTO",
"IS",
"ISNULL",
"JOIN",
"LATERAL",
"LEADING",
"LEFT",
"LIKE",
"LIMIT",
"LOCALTIME",
"LOCALTIMESTAMP",
"NATURAL",
"NOT",
"NOTNULL",
"NULL",
"OFFSET",
"ON",
"ONLY",
"OR",
"ORDER",
"OUTER",
"OVERLAPS",
"PLACING",
"PRIMARY",
"REFERENCES",
"RETURNING",
"RIGHT",
"SELECT",
"SESSION_USER",
"SIMILAR",
"SOME",
"SYMMETRIC",
"SYSTEM_USER",
"TABLE",
"TABLESAMPLE",
"THEN",
"TO",
"TRAILING",
"TRUE",
"UNION",
"UNIQUE",
"USER",
"USING",
"VARIADIC",
"VERBOSE",
"WHEN",
"WHERE",
"WINDOW",
"WITH",
}