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"
|
2025-03-08 19:01:37 +01:00
|
|
|
"strconv"
|
2026-02-02 13:18:19 +01:00
|
|
|
|
2026-05-09 01:43:14 +00:00
|
|
|
"github.com/Gleipnir-Technology/jet/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)
|
|
|
|
|
},
|
2025-03-08 19:01:37 +01:00
|
|
|
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:
|
2026-02-02 13:18:19 +01:00
|
|
|
return jet.AtomicCustomExpression(Token("to_char("), e, Token(`, 'YYYY-MM-DD"T"HH24:MI:SS.USZ')`))
|
2025-03-08 19:01:37 +01:00
|
|
|
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
|
|
|
}
|
|
|
|
|
}
|
2020-02-16 10:25:21 +01:00
|
|
|
|
|
|
|
|
var reservedWords = []string{
|
|
|
|
|
"ALL",
|
|
|
|
|
"ANALYSE",
|
|
|
|
|
"ANALYZE",
|
|
|
|
|
"AND",
|
|
|
|
|
"ANY",
|
|
|
|
|
"ARRAY",
|
|
|
|
|
"AS",
|
|
|
|
|
"ASC",
|
|
|
|
|
"ASYMMETRIC",
|
2024-12-23 13:22:29 +03:00
|
|
|
"AUTHORIZATION",
|
2025-01-13 10:58:48 +03:00
|
|
|
"BINARY",
|
2020-02-16 10:25:21 +01:00
|
|
|
"BOTH",
|
|
|
|
|
"CASE",
|
|
|
|
|
"CAST",
|
|
|
|
|
"CHECK",
|
|
|
|
|
"COLLATE",
|
2025-01-13 10:58:48 +03:00
|
|
|
"COLLATION",
|
2020-02-16 10:25:21 +01:00
|
|
|
"COLUMN",
|
2025-01-13 10:58:48 +03:00
|
|
|
"CONCURRENTLY",
|
2020-02-16 10:25:21 +01:00
|
|
|
"CONSTRAINT",
|
|
|
|
|
"CREATE",
|
2025-01-13 10:58:48 +03:00
|
|
|
"CROSS",
|
2020-02-16 10:25:21 +01:00
|
|
|
"CURRENT_CATALOG",
|
|
|
|
|
"CURRENT_DATE",
|
|
|
|
|
"CURRENT_ROLE",
|
2025-01-13 10:58:48 +03:00
|
|
|
"CURRENT_SCHEMA",
|
2020-02-16 10:25:21 +01:00
|
|
|
"CURRENT_TIME",
|
|
|
|
|
"CURRENT_TIMESTAMP",
|
|
|
|
|
"CURRENT_USER",
|
|
|
|
|
"DEFAULT",
|
|
|
|
|
"DEFERRABLE",
|
|
|
|
|
"DESC",
|
|
|
|
|
"DISTINCT",
|
|
|
|
|
"DO",
|
|
|
|
|
"ELSE",
|
|
|
|
|
"END",
|
|
|
|
|
"EXCEPT",
|
|
|
|
|
"FALSE",
|
|
|
|
|
"FETCH",
|
|
|
|
|
"FOR",
|
|
|
|
|
"FOREIGN",
|
2025-01-13 10:58:48 +03:00
|
|
|
"FREEZE",
|
2020-02-16 10:25:21 +01:00
|
|
|
"FROM",
|
2025-01-13 10:58:48 +03:00
|
|
|
"FULL",
|
2020-02-16 10:25:21 +01:00
|
|
|
"GRANT",
|
|
|
|
|
"GROUP",
|
|
|
|
|
"HAVING",
|
2025-01-13 10:58:48 +03:00
|
|
|
"ILIKE",
|
2020-02-16 10:25:21 +01:00
|
|
|
"IN",
|
|
|
|
|
"INITIALLY",
|
2025-01-13 10:58:48 +03:00
|
|
|
"INNER",
|
2020-02-16 10:25:21 +01:00
|
|
|
"INTERSECT",
|
|
|
|
|
"INTO",
|
2025-01-13 10:58:48 +03:00
|
|
|
"IS",
|
|
|
|
|
"ISNULL",
|
|
|
|
|
"JOIN",
|
2020-02-16 10:25:21 +01:00
|
|
|
"LATERAL",
|
|
|
|
|
"LEADING",
|
2025-01-13 10:58:48 +03:00
|
|
|
"LEFT",
|
|
|
|
|
"LIKE",
|
2020-02-16 10:25:21 +01:00
|
|
|
"LIMIT",
|
|
|
|
|
"LOCALTIME",
|
|
|
|
|
"LOCALTIMESTAMP",
|
2025-01-13 10:58:48 +03:00
|
|
|
"NATURAL",
|
2020-02-16 10:25:21 +01:00
|
|
|
"NOT",
|
2025-01-13 10:58:48 +03:00
|
|
|
"NOTNULL",
|
2020-02-16 10:25:21 +01:00
|
|
|
"NULL",
|
|
|
|
|
"OFFSET",
|
|
|
|
|
"ON",
|
|
|
|
|
"ONLY",
|
|
|
|
|
"OR",
|
|
|
|
|
"ORDER",
|
2025-01-13 10:58:48 +03:00
|
|
|
"OUTER",
|
|
|
|
|
"OVERLAPS",
|
2020-02-16 10:25:21 +01:00
|
|
|
"PLACING",
|
|
|
|
|
"PRIMARY",
|
|
|
|
|
"REFERENCES",
|
|
|
|
|
"RETURNING",
|
2022-09-07 00:51:24 +09:00
|
|
|
"RIGHT",
|
2020-02-16 10:25:21 +01:00
|
|
|
"SELECT",
|
|
|
|
|
"SESSION_USER",
|
2025-01-13 10:58:48 +03:00
|
|
|
"SIMILAR",
|
2020-02-16 10:25:21 +01:00
|
|
|
"SOME",
|
|
|
|
|
"SYMMETRIC",
|
2025-01-13 10:58:48 +03:00
|
|
|
"SYSTEM_USER",
|
2020-02-16 10:25:21 +01:00
|
|
|
"TABLE",
|
2025-01-13 10:58:48 +03:00
|
|
|
"TABLESAMPLE",
|
2020-02-16 10:25:21 +01:00
|
|
|
"THEN",
|
|
|
|
|
"TO",
|
|
|
|
|
"TRAILING",
|
|
|
|
|
"TRUE",
|
|
|
|
|
"UNION",
|
|
|
|
|
"UNIQUE",
|
|
|
|
|
"USER",
|
|
|
|
|
"USING",
|
|
|
|
|
"VARIADIC",
|
2025-01-13 10:58:48 +03:00
|
|
|
"VERBOSE",
|
2020-02-16 10:25:21 +01:00
|
|
|
"WHEN",
|
|
|
|
|
"WHERE",
|
|
|
|
|
"WINDOW",
|
|
|
|
|
"WITH",
|
|
|
|
|
}
|