spg_sql/ast.rs
1//! AST for the PG-dialect subset SPG accepts in v0.2.
2//!
3//! `Display` is implemented so that for any AST `a` produced by [`crate::parser`],
4//! re-parsing `format!("{a}")` yields a structurally equal AST. Binary and
5//! unary operators always emit parentheses to remove any precedence
6//! ambiguity — round-trip safety wins over prettiness.
7
8use alloc::boxed::Box;
9use alloc::format;
10use alloc::string::{String, ToString};
11use alloc::vec::Vec;
12use core::fmt;
13
14#[derive(Debug, Clone, PartialEq)]
15#[allow(clippy::large_enum_variant)] // Statement::Select dominates; Boxing would touch every match site
16pub enum Statement {
17 /// v7.14.0 — `DROP TABLE [IF EXISTS] name [, name…]
18 /// [CASCADE | RESTRICT]`. Engine removes the matching tables
19 /// (each one) from the catalog; IF EXISTS makes the drop
20 /// idempotent. CASCADE / RESTRICT trailers parsed silently
21 /// (SPG always cascades index drops on table drop).
22 DropTable {
23 names: Vec<String>,
24 if_exists: bool,
25 },
26 /// v7.14.0 — `DROP INDEX [IF EXISTS] name`. Removes the
27 /// matching index across whichever table holds it.
28 DropIndex {
29 name: String,
30 if_exists: bool,
31 },
32 /// v7.14.0 — empty / comment-only statement. The lexer strips
33 /// `--` line comments and `/* … */` block comments (including
34 /// the MySQL conditional `/*!NNNNN … */` form) before the
35 /// parser ever sees them; a SQL chunk that contains nothing
36 /// else lands here. Engine returns CommandOk no-op so
37 /// pg_dump / mysqldump preambles (`SET NAMES utf8mb4`
38 /// wrapped in conditional comments, etc.) load cleanly.
39 Empty,
40 Select(SelectStatement),
41 CreateTable(CreateTableStatement),
42 /// v7.9.15 — `CREATE EXTENSION [IF NOT EXISTS] <name>
43 /// [WITH SCHEMA <s>] [VERSION <v>] [CASCADE]` accepted as a
44 /// no-op so PG dumps that include extension declarations
45 /// (notably `pgvector`) load against SPG without splitting
46 /// init scripts. mailrs migration follow-up F3.
47 CreateExtension(String),
48 /// v7.9.27 → v7.16.2 — PG `DO $$ … $$ [LANGUAGE plpgsql];`
49 /// block. The body is now CAPTURED as a [`PlPgSqlBlock`] and
50 /// the engine executes it at top level (mailrs round-10
51 /// A.2). Pre-v7.16.2 the parser discarded the body and the
52 /// engine returned CommandOk — a SEV-1 silent no-op that
53 /// turned mailrs's `DO BEGIN IF EXISTS … THEN ALTER … END
54 /// $$` idempotent migrations into invisible no-ops.
55 DoBlock(PlPgSqlBlock),
56 CreateIndex(CreateIndexStatement),
57 Insert(InsertStatement),
58 /// v4.4 — `UPDATE <table> SET col=expr [, ...] [WHERE cond]`.
59 Update(UpdateStatement),
60 /// v4.4 — `DELETE FROM <table> [WHERE cond]`.
61 Delete(DeleteStatement),
62 /// v7.17.0 Phase 3.P0-42 — SQL:2003 / PG 15+ `MERGE` statement.
63 /// `MERGE INTO target [alias] USING source [alias] ON cond
64 /// WHEN MATCHED [AND cond] THEN { UPDATE SET … | DELETE | DO NOTHING }
65 /// WHEN NOT MATCHED [AND cond] THEN { INSERT (cols) VALUES (vals) | DO NOTHING }
66 /// [WHEN …]`. SPG v7.17 supports table-based source (subquery
67 /// source is a follow-up); BY SOURCE / BY TARGET and RETURNING
68 /// are also follow-ups.
69 Merge(MergeStatement),
70 Begin,
71 Commit,
72 Rollback,
73 /// `SAVEPOINT <name>` — push a named savepoint onto the active TX's
74 /// stack so a later `ROLLBACK TO <name>` can undo just the work
75 /// since this point.
76 Savepoint(String),
77 /// `ROLLBACK TO [SAVEPOINT] <name>` — restore catalog state to the
78 /// named savepoint and discard later savepoints. Does not end the
79 /// transaction.
80 RollbackToSavepoint(String),
81 /// `RELEASE [SAVEPOINT] <name>` — discard a savepoint without
82 /// rolling back. Keeps the work done since then.
83 ReleaseSavepoint(String),
84 /// `SHOW TABLES` — return the list of tables in the catalog.
85 ShowTables,
86 /// v7.17.0 Phase 3.P0-58 — MySQL `SHOW DATABASES` /
87 /// `SHOW SCHEMAS`. SPG is single-database; the executor
88 /// returns the canonical MySQL set so the mysql / MariaDB
89 /// client populates its database selector.
90 ShowDatabases,
91 /// v7.17.0 Phase 3.P0-59 — MySQL `SHOW CREATE TABLE <t>`
92 /// returns a 2-column row `(Table, "Create Table")` carrying
93 /// the synthesized DDL. mysqldump emits this for every
94 /// table at scrape time.
95 ShowCreateTable(String),
96 /// v7.17.0 Phase 3.P0-60 — MySQL `SHOW INDEXES FROM <t>`
97 /// (also `SHOW INDEX`, `SHOW KEYS`).
98 ShowIndexes(String),
99 /// v7.17.0 Phase 3.P0-61 — MySQL `SHOW STATUS`.
100 ShowStatus,
101 /// v7.17.0 Phase 3.P0-61 — MySQL `SHOW VARIABLES`.
102 ShowVariables,
103 /// v7.17.0 Phase 3.P0-62 — MySQL `SHOW PROCESSLIST`.
104 ShowProcesslist,
105 /// `SHOW COLUMNS FROM <table>` — return one row per column with
106 /// its declared name / type / nullability.
107 ShowColumns(String),
108 /// `CREATE USER 'name' WITH PASSWORD 'pw' ROLE 'admin'` (v4.1).
109 /// Role is optional; defaults to `readonly` when omitted.
110 CreateUser(CreateUserStatement),
111 /// `DROP USER 'name'` (v4.1).
112 DropUser(String),
113 /// `SHOW USERS` (v4.1) — admin-only listing of (name, role).
114 ShowUsers,
115 /// v4.26 — `EXPLAIN [ANALYZE] <select>`. The engine returns a
116 /// single-column text table describing the rewritten plan tree
117 /// for `inner`. `analyze` triggers an actual exec to attach
118 /// observed row counts and elapsed micros to each node.
119 Explain(ExplainStatement),
120 /// v6.0.4 — `ALTER INDEX <name> REBUILD [WITH (encoding = ...)]`.
121 /// Synchronous rebuild of an NSW index. With the optional
122 /// encoding clause, every stored cell at the indexed column is
123 /// also re-encoded through `coerce_value` before the new graph
124 /// builds.
125 AlterIndex(AlterIndexStatement),
126 /// v6.7.2 — `ALTER TABLE <name> SET <setting> = <value>`.
127 /// The only setting in v6.7.2 is `hot_tier_bytes`, which
128 /// overrides the global `SPG_HOT_TIER_BYTES` freezer trigger
129 /// for the named table.
130 AlterTable(AlterTableStatement),
131 /// v6.1.2 — `CREATE PUBLICATION <name> [FOR ALL TABLES]`.
132 /// The catalog row lives in `spg_publications`. Publisher-side
133 /// WAL filtering arrives in v6.1.5.
134 CreatePublication(CreatePublicationStatement),
135 /// v6.1.2 — `DROP PUBLICATION <name>`. PG-compatible silent
136 /// no-op when the publication does not exist.
137 DropPublication(String),
138 /// v6.1.3 — `SHOW PUBLICATIONS`. Returns one row per
139 /// publication ordered by name with `(name, scope_summary,
140 /// table_count)` columns. The scope summary is the human-
141 /// readable form `ALL TABLES` / `FOR TABLE …` / `FOR ALL
142 /// TABLES EXCEPT …`; `table_count` is `NULL` for the
143 /// `AllTables` scope and the table-list length otherwise.
144 ShowPublications,
145 /// v6.1.4 — `CREATE SUBSCRIPTION <name> CONNECTION '<conn>'
146 /// PUBLICATION <pub_name> [, <pub_name> …]`. Catalog lands
147 /// in `spg_subscriptions`; when the subscription is
148 /// `enabled = true` (default) the server spawns a
149 /// background worker that connects to `conn` and drains the
150 /// requested publication(s) into the local engine.
151 CreateSubscription(CreateSubscriptionStatement),
152 /// v6.1.4 — `DROP SUBSCRIPTION <name>`. Like DROP
153 /// PUBLICATION, silent no-op when absent. Stops the
154 /// associated worker thread before removing the row.
155 DropSubscription(String),
156 /// v6.1.4 — `SHOW SUBSCRIPTIONS`. Returns one row per
157 /// subscription ordered by name with `(name, conn_str,
158 /// publications, enabled, last_received_pos)`.
159 ShowSubscriptions,
160 /// v6.1.7 — `WAIT FOR WAL POSITION <pos> [WITH TIMEOUT <ms>]`.
161 /// Blocks until the local server's apply position reaches
162 /// `<pos>` or `<ms>` elapses. Server-layer command: the
163 /// engine refuses it (`EngineError::Unsupported`) since
164 /// `lag_state` lives in `spg-server`'s `ServerState`.
165 WaitForWalPosition {
166 pos: u64,
167 /// `None` → wait forever; `Some(ms)` → return after `ms`
168 /// milliseconds even if the target isn't reached.
169 timeout_ms: Option<u64>,
170 },
171 /// v6.2.0 — `ANALYZE [<table>]`. Bare form walks every user
172 /// table; `ANALYZE <name>` re-stats just one. Populates
173 /// `spg_statistic` with per-column null_frac + n_distinct +
174 /// 100-bucket equi-depth histogram.
175 Analyze(Option<String>),
176 /// v6.7.3 — `COMPACT COLD SEGMENTS`. Walks every user table's
177 /// BTree-cold indices and merges small cold-tier segments
178 /// (size below `SPG_COMPACTION_TARGET_SEGMENT_BYTES`, default
179 /// 4 MiB) into a single larger segment per (table, index).
180 /// `WHERE` predicate filtering on which tables to compact is
181 /// carved out of v6.7.3 (per V6_7_DESIGN.md STABILITY entry);
182 /// v6.7.3 only supports the bare form.
183 CompactColdSegments,
184 /// v7.12.1 — `SET <name> [TO|=] <value>`. Records a session
185 /// parameter on the engine; v7.12.1 honours
186 /// `default_text_search_config` (consumed by `to_tsvector` /
187 /// `plainto_tsquery` family when called without an explicit
188 /// config arg). All other names are accepted as a no-op so PG
189 /// dumps with `SET client_encoding`, `SET search_path` etc.
190 /// load cleanly.
191 SetParameter {
192 name: String,
193 value: SetValue,
194 },
195 /// v7.14.0 — `SET a = 1, b = 2, …` MySQL-flavoured
196 /// multi-assignment (mysqldump preamble uses
197 /// `SET @OLD_FOREIGN_KEY_CHECKS = @@FOREIGN_KEY_CHECKS,
198 /// FOREIGN_KEY_CHECKS=0`). Engine applies each pair in
199 /// source order. Pairs whose LHS is a MySQL session/user
200 /// variable (`@VAR` / `@@VAR`) are recorded with the raw
201 /// name so the engine can ignore them; pairs whose LHS is
202 /// a recognised engine parameter (e.g. `FOREIGN_KEY_CHECKS`)
203 /// go through the regular `set_session_param` path.
204 SetParameterList(Vec<(String, SetValue)>),
205 /// v7.12.1 — `RESET <name>` / `RESET ALL`. Restores parameter
206 /// to its default. No-op for parameters SPG does not track.
207 ResetParameter(Option<String>),
208 /// v7.12.4 — `CREATE [OR REPLACE] FUNCTION name(args) RETURNS
209 /// <type> [LANGUAGE <lang>] AS $$ body $$ [LANGUAGE <lang>]`.
210 /// v7.12.4 ships `plpgsql` for `RETURNS TRIGGER` bodies (the
211 /// CREATE TRIGGER + AFTER/BEFORE row-level pipeline). Other
212 /// languages parse but error at exec time with a clear
213 /// unsupported message.
214 CreateFunction(CreateFunctionStatement),
215 /// v7.12.4 — `CREATE [OR REPLACE] TRIGGER name {BEFORE|AFTER}
216 /// {INSERT|UPDATE|DELETE} [OR ...] ON tbl FOR EACH ROW
217 /// EXECUTE {FUNCTION|PROCEDURE} fn_name()`. STATEMENT-level
218 /// triggers and column-list / WHEN clauses are out of scope
219 /// for v7.12.4.
220 CreateTrigger(CreateTriggerStatement),
221 /// v7.12.4 — `DROP TRIGGER [IF EXISTS] name ON tbl`. Silent
222 /// no-op when missing if `IF EXISTS` is set.
223 DropTrigger {
224 name: String,
225 table: String,
226 if_exists: bool,
227 },
228 /// v7.12.4 — `DROP FUNCTION [IF EXISTS] name`. Same shape as
229 /// DROP TRIGGER but global (no table scope).
230 DropFunction {
231 name: String,
232 if_exists: bool,
233 },
234 /// v7.17.0 — `CREATE [TEMPORARY] SEQUENCE [IF NOT EXISTS] name
235 /// [AS data_type]
236 /// [INCREMENT [BY] n]
237 /// [MINVALUE n | NO MINVALUE]
238 /// [MAXVALUE n | NO MAXVALUE]
239 /// [START [WITH] n]
240 /// [CACHE n]
241 /// [[NO] CYCLE]
242 /// [OWNED BY {table.col | NONE}]`.
243 /// Closes the round-7+ silent-no-op SEQUENCE story so pg_dump
244 /// emits + nextval/currval/setval downstream all work.
245 CreateSequence(CreateSequenceStatement),
246 /// v7.17.0 — `ALTER SEQUENCE [IF EXISTS] name <options>` with
247 /// the same option grammar as CREATE SEQUENCE, plus
248 /// `RESTART [WITH n]` and `OWNED BY ...` re-attach.
249 AlterSequence(AlterSequenceStatement),
250 /// v7.17.0 — `DROP SEQUENCE [IF EXISTS] name [, name…]
251 /// [CASCADE | RESTRICT]`. CASCADE / RESTRICT trailers parsed
252 /// silently (no FK on sequences).
253 DropSequence {
254 names: Vec<String>,
255 if_exists: bool,
256 },
257 /// v7.17.0 Phase 1.2 — `CREATE [OR REPLACE] [TEMPORARY] VIEW
258 /// [IF NOT EXISTS] name [(col, …)] AS <SELECT …>`. Closes the
259 /// silent-no-op VIEW story from the v7.17 customer-readiness
260 /// audit: pre-v7.17 SPG parsed CREATE VIEW as Statement::Empty
261 /// so any downstream `SELECT FROM v` errored with table-not-
262 /// found. The view body is stored verbatim; SELECT FROM <v>
263 /// rewrites at exec-time by prepending the view body as a
264 /// synthetic CTE.
265 CreateView(CreateViewStatement),
266 /// v7.17.0 Phase 1.2 — `DROP VIEW [IF EXISTS] name [, name…]
267 /// [CASCADE | RESTRICT]`. Removes the matching view from the
268 /// catalog; CASCADE/RESTRICT parsed silently.
269 DropView {
270 names: Vec<String>,
271 if_exists: bool,
272 },
273 /// v7.17.0 Phase 1.3 — `CREATE MATERIALIZED VIEW [IF NOT
274 /// EXISTS] name [(col, …)] AS <SELECT …> [WITH [NO] DATA]`.
275 /// Closes the silent-no-op MATERIALIZED VIEW story. Storage
276 /// model: the materialised result lives as a regular table
277 /// with the matching name + a parallel
278 /// `materialized_views` registry mapping name → body source
279 /// (used by REFRESH).
280 CreateMaterializedView(CreateMaterializedViewStatement),
281 /// v7.17.0 Phase 1.3 — `REFRESH MATERIALIZED VIEW name [WITH
282 /// [NO] DATA]`. Re-runs the stored body and replaces the
283 /// cached rows. `WITH NO DATA` truncates without re-running.
284 RefreshMaterializedView {
285 name: String,
286 with_data: bool,
287 },
288 /// v7.17.0 Phase 1.3 — `DROP MATERIALIZED VIEW [IF EXISTS]
289 /// name [, name…] [CASCADE | RESTRICT]`. Drops both the
290 /// backing table and the source registry entry.
291 DropMaterializedView {
292 names: Vec<String>,
293 if_exists: bool,
294 },
295 /// v7.17.0 Phase 1.4 — `CREATE TYPE name AS ENUM ('a', 'b',
296 /// …)`. Closes the silent-no-op CREATE TYPE story so PG
297 /// dumps that declare enum types load with real constraints
298 /// instead of becoming free-form TEXT. Future kinds
299 /// (composite / range / domain) extend the inner `kind`
300 /// enum.
301 CreateType(CreateTypeStatement),
302 /// v7.17.0 Phase 1.4 — `DROP TYPE [IF EXISTS] name [, name…]
303 /// [CASCADE | RESTRICT]`. Removes the matching enum/domain
304 /// from the catalog.
305 DropType {
306 names: Vec<String>,
307 if_exists: bool,
308 },
309 /// v7.17.0 Phase 1.5 — `CREATE DOMAIN name AS base_type
310 /// [DEFAULT expr] [NOT NULL | NULL] [CHECK (expr)]*`.
311 /// A DOMAIN is a named CHECK-constrained alias over a built-
312 /// in type. The CHECK + NOT NULL + DEFAULT clauses apply to
313 /// every column declared with the domain. Closes the
314 /// silent-no-op CREATE DOMAIN story so PG dumps that ship
315 /// validated identifier types (email, positive_int, …) keep
316 /// their guarantees.
317 CreateDomain(CreateDomainStatement),
318 /// v7.17.0 Phase 1.5 — `DROP DOMAIN [IF EXISTS] name
319 /// [, name…] [CASCADE | RESTRICT]`. Removes the matching
320 /// domain from the catalog.
321 DropDomain {
322 names: Vec<String>,
323 if_exists: bool,
324 },
325 /// v7.17.0 Phase 1.6 — `CREATE SCHEMA [IF NOT EXISTS]
326 /// name [AUTHORIZATION user]`. SPG is single-database;
327 /// schemas are tracked as a namespace registry so pg_dump
328 /// multi-schema declarations land cleanly and `SELECT *
329 /// FROM information_schema.schemata` returns real entries.
330 /// Schema-qualified `schema.table` references still strip
331 /// the prefix at lookup time per PG (schemas are not
332 /// isolation boundaries in v7.17 — see project-next-docket
333 /// for the v7.18+ isolation tracking).
334 CreateSchema {
335 name: String,
336 if_not_exists: bool,
337 },
338 /// v7.17.0 Phase 1.6 — `DROP SCHEMA [IF EXISTS] name
339 /// [, name…] [CASCADE | RESTRICT]`. Removes the schema
340 /// from the registry; built-in `public` / `pg_catalog` /
341 /// `information_schema` cannot be dropped.
342 DropSchema {
343 names: Vec<String>,
344 if_exists: bool,
345 },
346}
347
348/// v7.17.0 Phase 1.5 — `CREATE DOMAIN` AST.
349#[derive(Debug, Clone, PartialEq)]
350pub struct CreateDomainStatement {
351 pub name: String,
352 /// Base type for the domain (one of the built-in
353 /// `ColumnTypeName` variants). User-defined enum / domain
354 /// bases are deferred to Phase 1.5b.
355 pub base_type: ColumnTypeName,
356 /// Optional `DEFAULT <expr>`. Resolved at engine-side
357 /// CREATE TABLE time when a column is bound to this domain.
358 pub default: Option<Expr>,
359 /// `NOT NULL` from the domain definition. Engine ORs this
360 /// with the column-level nullability so the strictest of the
361 /// two wins (i.e. the column is non-nullable if either side
362 /// says so).
363 pub not_null: bool,
364 /// Zero-or-more `CHECK (expr)` predicates. Each one is
365 /// enforced as part of the column's CHECK list at INSERT /
366 /// UPDATE time, with `VALUE` substituted for the column's
367 /// current cell value.
368 pub checks: Vec<Expr>,
369}
370
371/// v7.17.0 Phase 1.4 — `CREATE TYPE` AST.
372#[derive(Debug, Clone, PartialEq, Eq)]
373pub struct CreateTypeStatement {
374 pub name: String,
375 pub kind: TypeKind,
376}
377
378/// v7.17.0 Phase 1.4 — flavour of the new type. Only ENUM is
379/// implemented; the variant set is open so Phase 1.5 (DOMAIN)
380/// and later (COMPOSITE, RANGE) can land without an AST shape
381/// migration.
382#[derive(Debug, Clone, PartialEq, Eq)]
383pub enum TypeKind {
384 /// `AS ENUM ('a', 'b', …)`. Order is preserved (PG enum
385 /// labels are ordered).
386 Enum { labels: Vec<String> },
387}
388
389/// v7.12.1 — payload of a SET right-hand side. PG syntax accepts
390/// a string literal, an identifier (often a config name), an
391/// integer/float, or the bare `DEFAULT` keyword.
392#[derive(Debug, Clone, PartialEq)]
393pub enum SetValue {
394 String(String),
395 Ident(String),
396 Number(String),
397 Default,
398}
399
400/// v6.1.4 — `CREATE SUBSCRIPTION` AST node. v6.1.4 ships a
401/// single fixed-shape DDL; the WITH-clause options PG supports
402/// (`enabled`, `slot_name`, `streaming`, `binary`) are out of
403/// scope for v6.1.4 — `enabled` defaults to true and there are
404/// no other knobs to set in v6.1.x.
405#[derive(Debug, Clone, PartialEq, Eq)]
406pub struct CreateSubscriptionStatement {
407 pub name: String,
408 /// Connection string in PG keyword=value form (e.g.
409 /// `host=127.0.0.1 port=20002`). v6.1.4 only consumes the
410 /// `host` and `port` fields; the rest is reserved for
411 /// future v6.1.x options.
412 pub conn_str: String,
413 /// One or more publications on the remote side. Order is
414 /// preserved verbatim from the DDL; the worker requests them
415 /// in this order. v6.1.4 records the list; v6.1.5
416 /// publisher-side filtering enforces it.
417 pub publications: Vec<String>,
418}
419
420/// v7.17.0 — `CREATE SEQUENCE` AST node. See [`Statement::CreateSequence`].
421#[derive(Debug, Clone, PartialEq, Eq)]
422pub struct CreateSequenceStatement {
423 pub name: String,
424 pub if_not_exists: bool,
425 pub temporary: bool,
426 /// Optional `AS data_type`. Default in PG is BIGINT; SPG matches.
427 pub data_type: Option<SequenceDataType>,
428 pub options: SequenceOptions,
429}
430
431/// v7.17.0 — narrow type for `AS` clause of CREATE SEQUENCE.
432#[derive(Debug, Clone, Copy, PartialEq, Eq)]
433pub enum SequenceDataType {
434 SmallInt,
435 Int,
436 BigInt,
437}
438
439/// v7.17.0 — option grammar shared by CREATE / ALTER SEQUENCE.
440/// All fields are optional. `min_value`/`max_value` carry
441/// `Some(SeqBound::NoBound)` for `NO MINVALUE` / `NO MAXVALUE`.
442#[derive(Debug, Clone, Default, PartialEq, Eq)]
443pub struct SequenceOptions {
444 pub increment: Option<i64>,
445 pub min_value: Option<SeqBound>,
446 pub max_value: Option<SeqBound>,
447 pub start: Option<i64>,
448 /// `RESTART [WITH n]` — ALTER-only. `Some(None)` = bare
449 /// RESTART, `Some(Some(n))` = RESTART WITH n.
450 pub restart: Option<Option<i64>>,
451 pub cache: Option<i64>,
452 pub cycle: Option<bool>,
453 pub owned_by: Option<SequenceOwnedBy>,
454}
455
456/// v7.17.0 — `MINVALUE n` / `NO MINVALUE`.
457#[derive(Debug, Clone, Copy, PartialEq, Eq)]
458pub enum SeqBound {
459 Value(i64),
460 NoBound,
461}
462
463/// v7.17.0 — `OWNED BY {table.col | NONE}`.
464#[derive(Debug, Clone, PartialEq, Eq)]
465pub enum SequenceOwnedBy {
466 None,
467 Column { table: String, column: String },
468}
469
470/// v7.17.0 Phase 1.3 — `CREATE MATERIALIZED VIEW` AST node.
471#[derive(Debug, Clone, PartialEq)]
472pub struct CreateMaterializedViewStatement {
473 pub name: String,
474 pub if_not_exists: bool,
475 /// Optional `(col, col, …)` rename list. Applies to the
476 /// backing table at CREATE / REFRESH time.
477 pub columns: Vec<String>,
478 /// Underlying SELECT. Re-parsed at REFRESH time to rebuild
479 /// the cached rows.
480 pub body: SelectStatement,
481 /// `WITH DATA` (default) = materialise the rows at CREATE
482 /// time. `WITH NO DATA` = create an empty backing table;
483 /// callers must REFRESH before SELECT returns rows.
484 pub with_data: bool,
485}
486
487/// v7.17.0 Phase 1.2 — `CREATE VIEW` AST node.
488#[derive(Debug, Clone, PartialEq)]
489pub struct CreateViewStatement {
490 pub name: String,
491 pub or_replace: bool,
492 pub if_not_exists: bool,
493 pub temporary: bool,
494 /// Optional `(col, col, …)` rename list. When non-empty,
495 /// these override the body's projected column names per-
496 /// position at SELECT-from-view time.
497 pub columns: Vec<String>,
498 /// Underlying SELECT. Re-parsed lazily at SELECT-from-view
499 /// time to materialise the view as a synthetic CTE.
500 pub body: SelectStatement,
501}
502
503/// v7.17.0 — `ALTER SEQUENCE` AST node.
504#[derive(Debug, Clone, PartialEq, Eq)]
505pub struct AlterSequenceStatement {
506 pub name: String,
507 pub if_exists: bool,
508 pub options: SequenceOptions,
509}
510
511/// v6.1.2 — `CREATE PUBLICATION` AST node. The `scope` field uses
512/// the [`PublicationScope`] shape. v6.1.2 only accepted
513/// `AllTables`; v6.1.3 unlocks the `ForTables` / `AllTablesExcept`
514/// variants by flipping the parser gate (no AST migration).
515#[derive(Debug, Clone, PartialEq, Eq)]
516pub struct CreatePublicationStatement {
517 pub name: String,
518 pub scope: PublicationScope,
519}
520
521/// v6.1.2 — Which tables a publication covers. v6.1.3 (this commit)
522/// flips the parser gate for the `ForTables` / `AllTablesExcept`
523/// variants — the on-disk shape, snapshot serialisation, and the
524/// AST round-trip Display path were already in place in v6.1.2
525/// so this is a parser-only widening.
526#[derive(Debug, Clone, PartialEq, Eq)]
527pub enum PublicationScope {
528 AllTables,
529 ForTables(Vec<String>),
530 AllTablesExcept(Vec<String>),
531}
532
533#[derive(Debug, Clone, PartialEq, Eq)]
534pub struct AlterIndexStatement {
535 pub name: String,
536 pub target: AlterIndexTarget,
537}
538
539#[derive(Debug, Clone, PartialEq, Eq)]
540pub enum AlterIndexTarget {
541 /// `REBUILD [WITH (encoding = <enc>)]`. `encoding = None`
542 /// rebuilds the existing graph in place without touching the
543 /// column encoding; `Some(enc)` re-encodes every cell first.
544 Rebuild { encoding: Option<VecEncoding> },
545 /// v7.16.2 — `[IF EXISTS] RENAME TO <new>`. mailrs migrate-042
546 /// uses this; PG drops the IF EXISTS noisily as ERROR, mailrs
547 /// uses it to make the migration idempotent (re-running on a
548 /// DB where the rename already happened is a no-op rather
549 /// than an error).
550 Rename { new: String, if_exists: bool },
551}
552
553/// v6.7.2 — `ALTER TABLE t SET <setting> = <value>`. v6.7.2 ships
554/// the single `hot_tier_bytes` setting; later v6.7.x sub-versions
555/// can add more SET subjects without changing the dispatch shape.
556#[derive(Debug, Clone, PartialEq)]
557pub struct AlterTableStatement {
558 pub name: String,
559 /// v7.13.2 — mailrs round-6 S1. One or more subactions
560 /// separated by commas in the source SQL. PG-semantic apply
561 /// is sequential; engine bails on first error (no
562 /// transactional rollback of completed subactions in v7.13).
563 /// Single-subaction shape stays a 1-element vec.
564 pub targets: Vec<AlterTableTarget>,
565}
566
567#[derive(Debug, Clone, PartialEq)]
568#[allow(clippy::large_enum_variant)]
569pub enum AlterTableTarget {
570 /// Per-table hot-tier byte budget override. The freezer
571 /// reads this before falling back to `SPG_HOT_TIER_BYTES`.
572 SetHotTierBytes(u64),
573 /// v7.6.8 — `ALTER TABLE t ADD CONSTRAINT name FOREIGN KEY
574 /// (cols) REFERENCES parent[(pcols)] [ON DELETE/UPDATE …]`.
575 /// Engine validates existing rows against the new constraint
576 /// before installing it.
577 AddForeignKey(ForeignKeyConstraint),
578 /// v7.6.8 — `ALTER TABLE t DROP CONSTRAINT [IF EXISTS] name`.
579 /// `if_exists` (v7.13.2 mailrs round-6 S7) makes the drop a
580 /// no-op when no FK with that name exists; otherwise raises.
581 DropForeignKey { name: String, if_exists: bool },
582 /// v7.13.0 — `ALTER TABLE t ADD [COLUMN] [IF NOT EXISTS] <col>
583 /// <type> [DEFAULT <expr>] [NOT NULL]`. mailrs round-5 G1
584 /// (20 migrate-*.sql hits). Engine appends the column to the
585 /// schema and back-fills every existing row with the DEFAULT
586 /// (or NULL when no DEFAULT and the column is nullable).
587 AddColumn {
588 column: ColumnDef,
589 if_not_exists: bool,
590 },
591 /// v7.13.0 — `ALTER TABLE t ALTER COLUMN <col> TYPE <ty>
592 /// [USING <expr>]` (mailrs round-5 G8). Engine rewrites every
593 /// existing row's column value by evaluating the optional
594 /// USING expression (default `col::<ty>`) and re-coercing
595 /// against the new column type.
596 AlterColumnType {
597 column: String,
598 new_type: ColumnTypeName,
599 using: Option<Expr>,
600 },
601 /// v7.13.3 — `ALTER TABLE t DROP [COLUMN] [IF EXISTS] <col>
602 /// [CASCADE | RESTRICT]` (mailrs round-7 S8). The column +
603 /// every row's value at that position is removed; any index
604 /// on the column is dropped. `if_exists` makes the drop a
605 /// no-op when the column is missing. `cascade` removes
606 /// dependents (FKs referencing the column, partial indexes
607 /// whose predicate names the column); without it, the engine
608 /// rejects when dependents exist.
609 DropColumn {
610 column: String,
611 if_exists: bool,
612 cascade: bool,
613 },
614 /// v7.14.0 — `ALTER TABLE t ADD CONSTRAINT name PRIMARY KEY
615 /// (cols)` / `ADD CONSTRAINT name UNIQUE (cols)` / `ADD
616 /// CONSTRAINT name CHECK (expr)` — table-level constraints
617 /// installed post-CREATE-TABLE. pg_dump emits PKs as a
618 /// separate ALTER TABLE statement, so this surface lets the
619 /// dump load straight through.
620 AddTableConstraint(TableConstraint),
621 /// v7.15.0 — `ALTER TABLE t RENAME [COLUMN] old TO new`.
622 /// Renames the column in the schema and propagates the rename
623 /// to every stored source string that references it as a
624 /// (potentially-qualified) column identifier: CHECK predicates,
625 /// partial-index predicates, runtime DEFAULT expressions, and
626 /// triggers' `UPDATE OF` column lists. Function bodies and
627 /// trigger bodies are NOT auto-rewritten — they're loose
628 /// source text and may contain references SPG can't statically
629 /// resolve to this column (NEW./OLD. + dynamic SQL). Renames
630 /// the column even if dependents exist; users renaming a
631 /// column referenced by a function body update the function
632 /// body separately.
633 RenameColumn { old: String, new: String },
634 /// v7.22 (round-13 T2) — mark a column auto-incrementing.
635 /// pg_dump splits SERIAL/IDENTITY columns into a plain integer
636 /// column plus either `ALTER COLUMN c SET DEFAULT nextval(…)`
637 /// (serial) or `ALTER COLUMN c ADD GENERATED … AS IDENTITY (…)`
638 /// (identity); both lower to this. SPG's auto-increment is
639 /// max+1-scan based, so the dump's `setval(…)` calls stay
640 /// no-ops without losing the sequence position.
641 SetColumnAutoIncrement {
642 column: String,
643 /// The implicit sequence pg_dump names for an identity
644 /// column (`ADD GENERATED … ( SEQUENCE NAME s … )`) or the
645 /// nextval target for a serial default. The engine creates
646 /// it if absent so the dump's later `setval(s, …)` lands.
647 seq_name: Option<String>,
648 },
649 /// v7.16.2 — `ALTER TABLE old RENAME TO new`. Renames the
650 /// table itself (mailrs round-10 A.5 carve-out — mailrs's
651 /// migrate-042 uses it). The engine moves the table entry
652 /// in the catalog under the new name; child catalog state
653 /// (FKs pointing at this table, triggers watching this
654 /// table) tracks the rename through the storage layer.
655 RenameTable { new: String },
656 /// v7.16.1 — `ALTER TABLE t { ENABLE | DISABLE } TRIGGER
657 /// { ALL | <name> }`. Toggles whether row-level triggers
658 /// fire on subsequent INSERT/UPDATE/DELETE on the table.
659 /// `pg_dump --disable-triggers` emits a DISABLE wrapper +
660 /// ENABLE epilogue around every table's data block so the
661 /// rows already-computed in prod don't get re-rewritten
662 /// (and so trigger-driven side effects like
663 /// audit/queueing don't re-fire during a bulk reload).
664 /// `which == TriggerSelector::All` toggles every trigger
665 /// on the table; `Named(name)` toggles one trigger. The
666 /// engine persists the disabled state on `TriggerDef.enabled`
667 /// (catalog FILE_VERSION 25+) and the row-write paths skip
668 /// the trigger when `!enabled`.
669 SetTriggerEnabled {
670 which: TriggerSelector,
671 enabled: bool,
672 },
673}
674
675/// v7.16.1 — target of `ALTER TABLE … { ENABLE | DISABLE }
676/// TRIGGER …`. PG also accepts `USER`, `REPLICA`, `ALWAYS`
677/// modifiers; v7.16.1 ships the two shapes pg_dump actually
678/// emits (`ALL` + per-name) — the rest parse-accept as `Named`
679/// shouldn't surface from a dump.
680#[derive(Debug, Clone, PartialEq, Eq)]
681pub enum TriggerSelector {
682 /// Every trigger on the table.
683 All,
684 /// A specific trigger by name.
685 Named(String),
686}
687
688#[derive(Debug, Clone, PartialEq)]
689pub struct ExplainStatement {
690 pub analyze: bool,
691 pub inner: Box<SelectStatement>,
692 /// v6.8.3 — `EXPLAIN (SUGGEST) <SELECT>` enables the index
693 /// advisor pass: after the regular plan tree, the engine
694 /// emits one suggestion line per column referenced in the
695 /// query's WHERE / JOIN that has no covering index on the
696 /// owning table.
697 pub suggest: bool,
698}
699
700#[derive(Debug, Clone, PartialEq, Eq)]
701pub struct CreateUserStatement {
702 pub name: String,
703 pub password: String,
704 /// One of `admin` / `readwrite` / `readonly`. Stored verbatim from
705 /// the parser; the engine validates against `Role::parse` so a
706 /// typo lands as a runtime error with a clear message rather than
707 /// a parse failure.
708 pub role: String,
709}
710
711/// v7.12.4 — `CREATE [OR REPLACE] FUNCTION`. v7.12.4 ships
712/// `RETURNS TRIGGER LANGUAGE plpgsql` as the primary use case
713/// (the row-level trigger body the CREATE TRIGGER below references).
714/// Non-trigger user-defined functions parse but error at execution
715/// time with a clear unsupported message; that surface lands in
716/// v7.12.5+.
717#[derive(Debug, Clone, PartialEq)]
718pub struct CreateFunctionStatement {
719 pub name: String,
720 /// `OR REPLACE` was present; an existing function with the
721 /// same name is overwritten instead of erroring.
722 pub or_replace: bool,
723 /// `(arg1 type1, ...)` — v7.12.4 only accepts the empty arg
724 /// list `()` (sufficient for trigger functions). Other shapes
725 /// parse and store the args but the executor refuses to call
726 /// them.
727 pub args: Vec<FunctionArg>,
728 /// `RETURNS <type>` — `trigger` is the supported shape for
729 /// v7.12.4; arbitrary return types parse to
730 /// [`FunctionReturn::Other`].
731 pub returns: FunctionReturn,
732 /// `LANGUAGE <lang>` clause. PG accepts the clause on either
733 /// side of `AS $$...$$`; the parser canonicalises to one slot.
734 /// `plpgsql` and `sql` are the two interesting values.
735 pub language: String,
736 /// `AS $$ ... $$` body. v7.12.4 parses PL/pgSQL bodies into
737 /// a structured AST; non-trigger / non-plpgsql bodies stay as
738 /// the raw source text so the v7.12.5+ executor can pick them
739 /// up without a parser rev.
740 pub body: FunctionBody,
741}
742
743/// v7.12.4 — one positional argument to a `CREATE FUNCTION`.
744#[derive(Debug, Clone, PartialEq)]
745pub struct FunctionArg {
746 /// `IN` / `OUT` / `INOUT` mode. v7.12.4 only accepts `IN`
747 /// (the default); `OUT` / `INOUT` parse but the executor
748 /// refuses them.
749 pub mode: FunctionArgMode,
750 /// Optional arg name. Trigger functions traditionally don't
751 /// name their args (they read NEW/OLD instead), so `None` is
752 /// the common case.
753 pub name: Option<String>,
754 /// Declared type, normalised to the SPG `DataType` mapping
755 /// where one exists. Unknown / extension types parse as a
756 /// raw string under [`FunctionArgType::Raw`].
757 pub ty: FunctionArgType,
758}
759
760#[derive(Debug, Clone, Copy, PartialEq, Eq)]
761pub enum FunctionArgMode {
762 In,
763 Out,
764 InOut,
765}
766
767#[derive(Debug, Clone, PartialEq)]
768pub enum FunctionArgType {
769 Typed(ColumnTypeName),
770 /// Unknown / extension types — kept as the parser-side raw
771 /// identifier so error messages can name them precisely.
772 Raw(String),
773}
774
775#[derive(Debug, Clone, PartialEq)]
776pub enum FunctionReturn {
777 /// `RETURNS TRIGGER` — the row-level trigger function shape.
778 /// v7.12.4 ships exactly this for execution.
779 Trigger,
780 /// `RETURNS VOID`. Parses; executor rejects in v7.12.4 unless
781 /// the function is unused (since v7.12.4 doesn't ship scalar
782 /// function invocation).
783 Void,
784 /// `RETURNS <type>` for any concrete data type. Reserved for
785 /// v7.12.5+'s scalar UDF surface.
786 Type(ColumnTypeName),
787 /// `RETURNS <ident>` for types SPG doesn't know — extension
788 /// types, RETURNS SETOF rows, RETURNS TABLE(...), etc.
789 Other(String),
790}
791
792#[derive(Debug, Clone, PartialEq)]
793pub enum FunctionBody {
794 /// v7.12.4 — parsed PL/pgSQL `BEGIN … END` block. The
795 /// trigger-function executor walks this directly without
796 /// re-parsing.
797 PlPgSql(PlPgSqlBlock),
798 /// Raw source text — parser couldn't (or didn't try to)
799 /// structure-parse the body. Used for `LANGUAGE sql`
800 /// functions and any PL/pgSQL body that contains v7.12.5+
801 /// features the v7.12.4 parser doesn't yet recognise. The
802 /// executor returns an unsupported error when invoked.
803 Raw(String),
804}
805
806/// v7.12.4 — PL/pgSQL `BEGIN ... END;` block. v7.12.6 widens
807/// from assignment + return to a real-PL/pgSQL surface:
808/// `DECLARE`-block local variables, `IF/ELSIF/ELSE/END IF`
809/// control flow, `RAISE` diagnostics, and embedded SQL
810/// statements that execute through the regular engine path.
811/// The remaining v7.12.x carve-out is loops (`LOOP/WHILE/FOR`),
812/// which mailrs's trigger doesn't need but other PG customers
813/// may; deferred to a future minor release.
814#[derive(Debug, Clone, PartialEq)]
815pub struct PlPgSqlBlock {
816 /// v7.12.6 — `DECLARE var TYPE [:= init_expr];` declarations
817 /// preceding `BEGIN`. Empty when the body opens directly with
818 /// `BEGIN`. Declarations execute in order; each may reference
819 /// earlier-declared locals in its init expression.
820 pub declarations: Vec<PlPgSqlDeclare>,
821 pub statements: Vec<PlPgSqlStmt>,
822}
823
824/// v7.12.6 — single `DECLARE` entry: variable name + declared
825/// type + optional initialiser. Variables default to SQL NULL
826/// when no init is given (matches PG).
827#[derive(Debug, Clone, PartialEq)]
828pub struct PlPgSqlDeclare {
829 pub name: String,
830 /// Declared SQL type (mapped to [`ColumnTypeName`] where SPG
831 /// knows it; raw text otherwise).
832 pub ty: FunctionArgType,
833 pub default: Option<Expr>,
834}
835
836#[derive(Debug, Clone, PartialEq)]
837pub enum PlPgSqlStmt {
838 /// `NEW.col := expr;` or `OLD.col := expr;`. OLD is parsed
839 /// for clarity in error reporting (PG also forbids it) — the
840 /// executor errors with a clear "OLD is read-only" message.
841 Assign { target: AssignTarget, value: Expr },
842 /// v7.16.2 — plpgsql `SELECT <projection> INTO <var>
843 /// [FROM …]` (mailrs round-10 migrate-042). The `body` is
844 /// the SELECT statement with the INTO clause stripped; the
845 /// engine runs it via `Engine::execute`, takes the first
846 /// row's first column, and assigns to the local variable
847 /// in the DECLARE scope. Single-column / single-row
848 /// queries only at v7.16.2; multi-target (`INTO a, b`) is
849 /// a v7.16.x follow-up.
850 SelectInto {
851 var: String,
852 body: Box<SelectStatement>,
853 },
854 /// `RETURN <target>;` — trigger functions canonically return
855 /// `NEW` / `OLD` / `NULL`; v7.12.4 also accepts a bare
856 /// expression for forward compatibility with scalar UDFs.
857 Return(ReturnTarget),
858 /// v7.12.6 — `IF cond THEN body [ELSIF cond THEN body]*
859 /// [ELSE body] END IF;`. Branches are tried in order; first
860 /// truthy condition wins; the optional ELSE runs when no
861 /// condition matched.
862 If {
863 branches: Vec<(Expr, Vec<PlPgSqlStmt>)>,
864 else_branch: Vec<PlPgSqlStmt>,
865 },
866 /// v7.12.6 — `RAISE <level> '<fmt>' [, args]*;`. Level is one
867 /// of `NOTICE` / `WARNING` / `INFO` / `LOG` / `DEBUG`
868 /// (logging — observable side effect only) or `EXCEPTION`
869 /// (aborts the trigger and propagates as an error). v7.12.6
870 /// supports the basic format-string substitution PG uses
871 /// (`%` placeholders consumed positionally).
872 Raise {
873 level: RaiseLevel,
874 message: String,
875 args: Vec<Expr>,
876 },
877 /// v7.12.6 — embedded SQL statement inside the trigger body
878 /// (`INSERT INTO …`, `UPDATE …`, `DELETE FROM …`, `SELECT …`).
879 /// NEW.col / OLD.col references inside the embedded
880 /// statement's expression tree are substituted with the
881 /// current trigger context before the engine re-executes the
882 /// statement. Recursion depth into nested triggers is
883 /// bounded by the engine's existing trigger-fire guard.
884 EmbeddedSql(Box<Statement>),
885}
886
887#[derive(Debug, Clone, Copy, PartialEq, Eq)]
888pub enum RaiseLevel {
889 /// `RAISE NOTICE` — diagnostic message, observable in the
890 /// server log. Does not affect the trigger's outcome.
891 Notice,
892 /// `RAISE WARNING` — like NOTICE, slightly louder severity.
893 Warning,
894 /// `RAISE INFO` — like NOTICE, slightly quieter.
895 Info,
896 /// `RAISE LOG` — like NOTICE, lower priority.
897 Log,
898 /// `RAISE DEBUG` — like NOTICE, lowest priority.
899 Debug,
900 /// `RAISE EXCEPTION` — aborts the trigger function with the
901 /// given message, propagating up to the caller as a query-
902 /// level error.
903 Exception,
904}
905
906#[derive(Debug, Clone, PartialEq)]
907pub enum AssignTarget {
908 NewColumn(String),
909 OldColumn(String),
910 /// Reserved for v7.12.5 DECLARE'd local variables.
911 Local(String),
912}
913
914#[derive(Debug, Clone, PartialEq)]
915pub enum ReturnTarget {
916 /// `RETURN NEW;` — for BEFORE triggers, this is the row that
917 /// actually gets written (possibly with NEW.col mutations
918 /// applied). For AFTER triggers, the return value is ignored.
919 New,
920 /// `RETURN OLD;` — pass-through. For BEFORE DELETE this lets
921 /// the delete proceed; for BEFORE UPDATE / INSERT it's
922 /// equivalent to dropping the write.
923 Old,
924 /// `RETURN NULL;` — for BEFORE triggers, skips the write
925 /// entirely. For AFTER, the return value is ignored.
926 Null,
927 /// `RETURN <expr>;` — non-row return shape; reserved for the
928 /// scalar UDF surface in v7.12.5+. Executor errors when used
929 /// inside a trigger function.
930 Expr(Expr),
931}
932
933/// v7.12.4 — `CREATE [OR REPLACE] TRIGGER`. Always row-level
934/// (`FOR EACH ROW`) in v7.12.4 — statement-level triggers parse
935/// but the executor refuses them. `WHEN (cond)` clauses are out
936/// of scope; the trigger function can short-circuit on a leading
937/// IF inside its body once v7.12.5 lands IF.
938#[derive(Debug, Clone, PartialEq)]
939pub struct CreateTriggerStatement {
940 pub name: String,
941 pub or_replace: bool,
942 pub timing: TriggerTiming,
943 /// At least one event; `INSERT OR UPDATE OR DELETE` parses to
944 /// three entries in order.
945 pub events: Vec<TriggerEvent>,
946 pub table: String,
947 /// `FOR EACH ROW` vs `FOR EACH STATEMENT`. v7.12.4 ships
948 /// only `Row`; `Statement` parses but the executor refuses.
949 pub for_each: TriggerForEach,
950 /// Name of the function to invoke. v7.12.4 requires the
951 /// function to be `CREATE FUNCTION`'d earlier; forward
952 /// references (PG accepts) are deferred to v7.12.5.
953 pub function: String,
954 /// v7.13.0 — `UPDATE OF col, col, …` column-list filter
955 /// (mailrs round-5 G7). Non-empty only when the events list
956 /// contains UPDATE and the user wrote the column-list filter.
957 /// PG fires the trigger only when at least one of these
958 /// columns appears in the SET clause; SPG conservatively
959 /// fires on any UPDATE matching the listed columns or
960 /// rewriting them at the row level. Empty vec = no filter
961 /// (fire on every UPDATE).
962 pub update_columns: Vec<String>,
963}
964
965#[derive(Debug, Clone, Copy, PartialEq, Eq)]
966pub enum TriggerTiming {
967 /// Fires before the row is written; the trigger function's
968 /// return value (NEW or NULL) decides the row content and
969 /// whether the write proceeds at all.
970 Before,
971 /// Fires after the row is written; the return value is
972 /// ignored.
973 After,
974 /// `INSTEAD OF` is PG-VIEW-trigger-only and out of scope for
975 /// v7.12.4 (SPG has no updatable-view surface).
976 InsteadOf,
977}
978
979#[derive(Debug, Clone, Copy, PartialEq, Eq)]
980pub enum TriggerEvent {
981 Insert,
982 Update,
983 Delete,
984 /// `TRUNCATE` event parses; SPG has no TRUNCATE statement
985 /// so the trigger never fires.
986 Truncate,
987}
988
989#[derive(Debug, Clone, Copy, PartialEq, Eq)]
990pub enum TriggerForEach {
991 Row,
992 Statement,
993}
994
995#[derive(Debug, Clone, PartialEq)]
996pub struct CreateIndexStatement {
997 pub name: String,
998 pub table: String,
999 pub column: String,
1000 /// Optional `USING <method>` clause. v2.0 recognises `hnsw` (NSW
1001 /// graph for vector kNN); unspecified is the default B-tree index.
1002 pub method: IndexMethod,
1003 /// `IF NOT EXISTS` — engine returns `CommandOk` no-op when the
1004 /// index name already exists, instead of raising `DuplicateIndex`.
1005 pub if_not_exists: bool,
1006 /// v6.8.0 — `INCLUDE (col1, col2, …)` columns. Identifies the
1007 /// non-key columns the planner should treat as "covered" by
1008 /// this index when checking whether a query can run as an
1009 /// index-only scan. Empty when no `INCLUDE` clause was given.
1010 pub included_columns: Vec<String>,
1011 /// v6.8.1 — `WHERE <expr>` partial-index predicate. Only rows
1012 /// for which `<expr>` evaluates truthy enter the index;
1013 /// queries whose `WHERE` clause's canonical Display form
1014 /// matches this expression's Display form can be served by the
1015 /// partial index. Stored as a parsed `Expr` so the engine
1016 /// re-uses the existing evaluation path; storage persists the
1017 /// Display form on the catalog snapshot.
1018 pub partial_predicate: Option<Expr>,
1019 /// v6.8.2 — expression-based index. When `Some(expr)`, the
1020 /// index key is the result of `expr` evaluated on each row
1021 /// (e.g. `CREATE INDEX … (lower(name))`). The `column`
1022 /// field still names the *primary* column the expression
1023 /// touches so existing planner shortcuts that resolve a
1024 /// column position stay valid. `None` = plain
1025 /// column-reference index (the legacy shape).
1026 pub expression: Option<Expr>,
1027 /// v7.9.14 — extra column names after the leading column in a
1028 /// multi-column `CREATE INDEX … (a, b, c)`. mailrs F2. The
1029 /// planner today still only uses the leading column for index
1030 /// seeks; the extras are tracked verbatim so the same DDL
1031 /// round-trips through WAL replay + catalog snapshot, and so
1032 /// the engine can emit a clear warning at INDEX CREATE time
1033 /// that only the leading column is currently honoured.
1034 /// Composite BTree index keys land in v7.10.
1035 pub extra_columns: Vec<String>,
1036 /// v7.9.29 — `CREATE UNIQUE INDEX …`. When true the engine
1037 /// enforces uniqueness on the indexed key (combined with the
1038 /// `partial_predicate` filter — only rows where the predicate
1039 /// evaluates truthy enter the uniqueness check). Standard SQL
1040 /// and PG's canonical way to express conditional uniqueness.
1041 /// mailrs K1.
1042 pub is_unique: bool,
1043 /// v7.15.0 — operator class on the leading column, when the
1044 /// CREATE INDEX named one (`(col vector_cosine_ops)` shape).
1045 /// Lower-cased. Most opclasses are still informational; the
1046 /// engine routes on `gin_trgm_ops` specifically to build a
1047 /// trigram-shingle GIN over a TEXT column, and otherwise
1048 /// keeps the current "accepted and discarded" behaviour for
1049 /// pg_dump compatibility.
1050 pub opclass: Option<String>,
1051}
1052
1053#[derive(Debug, Clone, Copy, PartialEq, Eq)]
1054pub enum IndexMethod {
1055 /// Default — B-tree over `IndexKey`. Used for equality / range
1056 /// lookups on scalar columns.
1057 BTree,
1058 /// `USING hnsw` — NSW graph for kNN over a vector column.
1059 Hnsw,
1060 /// v6.7.1 — `USING brin` — Block Range INdex. Per-segment
1061 /// metadata that records (min_key, max_key) for each page in a
1062 /// cold-tier segment, on the indexed column. The optimizer
1063 /// can use these summaries to skip pages whose range does NOT
1064 /// overlap a query's WHERE predicate. BRIN indexes carry no
1065 /// in-memory data — the summaries live in the segment v2
1066 /// envelope's sidecar. Created via the standard
1067 /// `CREATE INDEX … USING brin (col)` syntax.
1068 Brin,
1069 /// v7.12.3 — `USING gin` — inverted index over a `tsvector`
1070 /// column. Posting lists map `lexeme word` → row locators; the
1071 /// planner uses them to narrow `WHERE col @@ tsquery` to the
1072 /// candidate rows whose vectors contain a matching term, then
1073 /// re-evaluates the full `@@` semantics on each candidate.
1074 /// Replaces the v7.9.26b `USING gin` → BTree fallback that
1075 /// silently degraded to a full scan at query time.
1076 Gin,
1077}
1078
1079#[derive(Debug, Clone, PartialEq)]
1080pub struct CreateTableStatement {
1081 pub name: String,
1082 pub columns: Vec<ColumnDef>,
1083 /// `IF NOT EXISTS` — engine returns `CommandOk` no-op when the
1084 /// table name already exists, instead of raising `DuplicateTable`.
1085 pub if_not_exists: bool,
1086 /// v7.6.0 — table-level `FOREIGN KEY (...) REFERENCES ...`
1087 /// constraints. Column-level `REFERENCES` (single-column inline
1088 /// form) is normalised into this vec at parse time so the engine
1089 /// sees one uniform list.
1090 pub foreign_keys: Vec<ForeignKeyConstraint>,
1091 /// v7.9.18 — table-level constraints: `PRIMARY KEY (a, b)` and
1092 /// `UNIQUE (a, b, ...)`. mailrs migration follow-up G1 + G6.
1093 /// Engine resolves each into a BTree index named after the
1094 /// constraint's leading column at CREATE TABLE time; INSERT
1095 /// path enforces composite uniqueness via row scan on the
1096 /// leading column index.
1097 pub table_constraints: Vec<TableConstraint>,
1098}
1099
1100/// v7.9.18 — table-level constraint at the end of a CREATE TABLE
1101/// column list. Either a composite PRIMARY KEY or a UNIQUE
1102/// (single- or multi-column).
1103#[derive(Debug, Clone, PartialEq)]
1104pub enum TableConstraint {
1105 /// `PRIMARY KEY (col1, col2, ...)`. Implies NOT NULL on each
1106 /// referenced column. Engine builds a BTree index named
1107 /// `<table>_pkey` and enforces composite uniqueness on INSERT.
1108 PrimaryKey {
1109 name: Option<String>,
1110 columns: Vec<String>,
1111 },
1112 /// `UNIQUE (col1, col2, ...)`. Engine builds a BTree index
1113 /// named `<table>_<leading_col>_key` (single-column) or
1114 /// `<table>_<leading_col>_<…>_key` (composite) and enforces
1115 /// uniqueness on INSERT.
1116 Unique {
1117 name: Option<String>,
1118 columns: Vec<String>,
1119 /// v7.13.0 — `NULLS NOT DISTINCT` modifier (mailrs round-5
1120 /// G10). PG 15+ flips the NULL handling so any number of
1121 /// NULL rows collide on the constraint. Default is
1122 /// `false` (NULLS DISTINCT, standard SQL behaviour).
1123 nulls_not_distinct: bool,
1124 },
1125 /// v7.13.0 — `CHECK (<expr>)` table-level constraint
1126 /// (mailrs round-5 G3). Column-level inline CHECKs fold into
1127 /// this same variant at parse time. Engine evaluates the
1128 /// predicate against each INSERT/UPDATE candidate row; a
1129 /// false / NULL result rejects the mutation.
1130 Check { name: Option<String>, expr: Expr },
1131 /// v7.15.0 — MySQL `KEY name (cols)` / `INDEX name (cols)`
1132 /// non-unique secondary-index declaration inline in CREATE
1133 /// TABLE. Engine builds a BTree index on the leading column
1134 /// (composite columns parse but only the leading column is
1135 /// honoured at v7.15 — matches the existing
1136 /// `CreateIndexStatement::extra_columns` semantics). Useful
1137 /// for `mysql/blog`-style schemas that lean on routine
1138 /// secondary indexes for ORM lookups.
1139 Index {
1140 name: Option<String>,
1141 columns: Vec<String>,
1142 },
1143 /// v7.17.0 Phase 2.2 — MySQL `FULLTEXT KEY/INDEX [name]
1144 /// (cols)` inline declaration. Pre-v7.17 the parser
1145 /// silently dropped these so MyISAM-imported FULLTEXT
1146 /// indexes vanished; v7.17 routes them through the
1147 /// existing tsvector-GIN engine path so MATCH AGAINST
1148 /// queries get a real inverted index instead of falling
1149 /// back to a full scan. Multi-column FULLTEXT KEYs build
1150 /// one GIN per column at v7.17 (per-column posting lists);
1151 /// the leading column drives query planning.
1152 FulltextIndex {
1153 name: Option<String>,
1154 columns: Vec<String>,
1155 },
1156}
1157
1158#[derive(Debug, Clone, PartialEq)]
1159#[allow(clippy::struct_excessive_bools)] // grammar-driven; each flag maps to a distinct PG column-constraint keyword
1160pub struct ColumnDef {
1161 pub name: String,
1162 pub ty: ColumnTypeName,
1163 pub nullable: bool,
1164 /// `DEFAULT <expr>` literal supplied at CREATE TABLE. Engine
1165 /// evaluates this once (with an empty row) and caches the resulting
1166 /// `Value` on the column schema.
1167 pub default: Option<Expr>,
1168 /// MySQL-style `AUTO_INCREMENT` — the engine maintains a counter
1169 /// per such column and fills the slot when INSERT leaves it
1170 /// unbound (omitted from a column-list INSERT or explicitly NULL).
1171 pub auto_increment: bool,
1172 /// v7.9.13 — inline `PRIMARY KEY` column constraint. mailrs
1173 /// migration follow-up F1. Implies `NOT NULL`. Engine creates
1174 /// an implicit BTree index named `<table>_pkey` over this
1175 /// column at CREATE TABLE time, satisfying the parent-side
1176 /// index requirement for any FOREIGN KEY pointing at it.
1177 pub is_primary_key: bool,
1178 /// v7.13.0 — inline `UNIQUE` column constraint
1179 /// (mailrs round-5 G2). The CREATE TABLE handler folds this
1180 /// into a single-column `TableConstraint::Unique` so the
1181 /// engine path stays uniform with table-level UNIQUE.
1182 pub is_unique: bool,
1183 /// v7.13.0 — inline `CHECK (<expr>)` column constraint
1184 /// (mailrs round-5 G3). Stored alongside the column so the
1185 /// CREATE TABLE handler can fold these into table-level
1186 /// CHECK constraints. Multiple inline CHECKs on the same
1187 /// column are concatenated with AND at the table level.
1188 pub check: Option<Expr>,
1189 /// v7.17.0 Phase 1.4 — user-defined type reference. When the
1190 /// parser sees an unknown column-type ident (anything not in
1191 /// the built-in `parse_column_type_name` table), it sets
1192 /// `ty = ColumnTypeName::Text` and records the original name
1193 /// here. The engine resolves at CREATE TABLE time: if a
1194 /// catalog enum/domain with this name exists, the column is
1195 /// bound to it (label-checked on INSERT for enums; CHECK-
1196 /// constrained for domains); otherwise the CREATE TABLE
1197 /// errors with "unknown type".
1198 pub user_type_ref: Option<String>,
1199 /// v7.17.0 Phase 2.1 — MySQL-style `ON UPDATE
1200 /// CURRENT_TIMESTAMP` column attribute. When set, an
1201 /// UPDATE that does NOT explicitly bind this column
1202 /// overrides the new value with `now()` (engine clock).
1203 /// Pre-v7.17 SPG silently accepted the syntax and never
1204 /// fired the override — `updated_at` columns from mysqldump
1205 /// stayed pinned at their initial DEFAULT forever, an
1206 /// audit Tier-S silent-failure. Generalised as a stored
1207 /// expression source so future shapes (`ON UPDATE
1208 /// CURRENT_TIMESTAMP(6)`, `ON UPDATE LOCALTIMESTAMP`) reuse
1209 /// the same field; v7.17 only accepts CURRENT_TIMESTAMP.
1210 pub on_update_runtime: Option<Expr>,
1211 /// v7.17.0 Phase 2.5 — text collation derived from the
1212 /// post-fix `COLLATE <name>` clause (and / or the table-level
1213 /// `COLLATE=<name>` for MySQL dumps that don't repeat it
1214 /// per column). Pre-2.5 SPG accepted the clause and
1215 /// discarded the name, leaving every column byte-compared
1216 /// — a Tier-S silent failure when the customer expected
1217 /// `_ci` / `case_insensitive` semantics. Parser normalises
1218 /// the raw collation name into the variants in `Collation`.
1219 /// Default `Binary` preserves the legacy compare path.
1220 pub collation: Collation,
1221 /// v7.17.0 Phase 4.4 — MySQL `UNSIGNED` modifier flag. Pre-
1222 /// 4.4 SPG accepted and discarded the keyword, leaving
1223 /// negative values silently accepted on a column the
1224 /// customer declared `INT UNSIGNED NOT NULL`. Now: the engine
1225 /// rejects negative INSERT / UPDATE values on UNSIGNED int
1226 /// columns. SPG widening to `u64`-shaped storage is out of
1227 /// v7.17 scope; the upper bound remains the signed-type max
1228 /// (i64::MAX for BIGINT UNSIGNED), which still strictly
1229 /// exceeds what every mailrs / Rails app actually uses.
1230 pub is_unsigned: bool,
1231 /// v7.17.0 Phase 3.P0-36 — MySQL inline `ENUM('a','b','c')`
1232 /// value list captured at parse time. When `Some`, the parser
1233 /// recognised `ENUM(...)` in the type slot; the engine
1234 /// validates INSERT cells against this list at
1235 /// column_def_to_schema time and persists the variants on
1236 /// `ColumnSchema.inline_enum_variants`. None for all
1237 /// non-ENUM columns.
1238 pub inline_enum_variants: Option<Vec<String>>,
1239 /// v7.17.0 Phase 3.P0-37 — MySQL inline `SET('a','b','c')`
1240 /// value list. Distinct from ENUM (subset semantics rather
1241 /// than pick-one). None for all non-SET columns.
1242 pub inline_set_variants: Option<Vec<String>>,
1243}
1244
1245/// v7.17.0 Phase 2.5 — text collation classification surfaced
1246/// from the SQL parser. Mirrors `spg_storage::Collation`; the
1247/// engine bridges between the two at CREATE TABLE time.
1248///
1249/// Recognised collation-name patterns (case-insensitive):
1250/// * `case_insensitive`, `*_ci`, `*_ai_ci`, `nocase` → CaseInsensitive
1251/// * Everything else (`C`, `POSIX`, `default`,
1252/// `pg_catalog.default`, `*_cs`, `*_bin`, unknown names) → Binary
1253#[derive(Debug, Clone, Copy, PartialEq, Eq)]
1254pub enum Collation {
1255 Binary,
1256 CaseInsensitive,
1257}
1258
1259#[allow(clippy::derivable_impls)]
1260impl Default for Collation {
1261 fn default() -> Self {
1262 Self::Binary
1263 }
1264}
1265
1266impl Collation {
1267 /// Classify a `COLLATE <name>` ident into one of the supported
1268 /// variants. Empty / unknown names fall back to `Binary` —
1269 /// matches the pre-2.5 silent-accept behaviour for snapshots
1270 /// that load through but don't actually depend on the
1271 /// collation semantics.
1272 #[must_use]
1273 pub fn from_collation_name(name: &str) -> Self {
1274 let lc = name.trim().to_ascii_lowercase();
1275 // Strip any quotes / schema-qualifier the parser left on
1276 // (e.g. `pg_catalog.default`).
1277 let bare = lc
1278 .trim_matches(|c: char| c == '"' || c == '\'')
1279 .rsplit('.')
1280 .next()
1281 .unwrap_or("");
1282 if bare.is_empty() {
1283 return Self::Binary;
1284 }
1285 if bare == "case_insensitive" || bare == "nocase" {
1286 return Self::CaseInsensitive;
1287 }
1288 // MySQL `_ci` suffix (covers `utf8mb4_general_ci`,
1289 // `utf8mb4_unicode_ci`, `utf8mb4_0900_ai_ci`, …).
1290 if bare.ends_with("_ci") {
1291 return Self::CaseInsensitive;
1292 }
1293 Self::Binary
1294 }
1295}
1296
1297/// v7.6.0 — A single FOREIGN KEY constraint. Both column-level
1298/// `REFERENCES` and table-level `FOREIGN KEY (...) REFERENCES ...`
1299/// parse into this shape — the column-level form has a single-entry
1300/// `columns` / `parent_columns`.
1301#[derive(Debug, Clone, PartialEq)]
1302pub struct ForeignKeyConstraint {
1303 /// Optional `CONSTRAINT <name>` prefix. Engine ignores the name
1304 /// today but parses + stores it so a future ALTER TABLE DROP
1305 /// CONSTRAINT can target by name (v7.6.8).
1306 pub name: Option<String>,
1307 /// Local columns participating in the FK (≥ 1).
1308 pub columns: Vec<String>,
1309 /// Referenced parent table.
1310 pub parent_table: String,
1311 /// Referenced parent columns. Must have the same arity as
1312 /// `columns`; engine validates parent has a PK / UNIQUE index
1313 /// on exactly this column set (v7.6.1).
1314 pub parent_columns: Vec<String>,
1315 /// `ON DELETE` action. Defaults to `Restrict` if absent.
1316 pub on_delete: FkAction,
1317 /// `ON UPDATE` action. Defaults to `Restrict` if absent.
1318 pub on_update: FkAction,
1319}
1320
1321/// v7.6.0 — Referential action for `ON DELETE` / `ON UPDATE`.
1322#[derive(Debug, Clone, Copy, PartialEq, Eq)]
1323pub enum FkAction {
1324 /// Reject the parent mutation if any child row references it.
1325 /// SQL spec default; SPG default when no clause is given.
1326 Restrict,
1327 /// Recursively propagate the parent's delete / update to the
1328 /// child rows. Same TX.
1329 Cascade,
1330 /// Set the child FK column(s) to NULL. Requires the FK columns
1331 /// to be NULL-able.
1332 SetNull,
1333 /// Set the child FK column(s) to their declared DEFAULT.
1334 /// Requires the child column(s) to have DEFAULT.
1335 SetDefault,
1336 /// SQL spec `NO ACTION` (deferred check). SPG treats this as
1337 /// `Restrict` because the single-writer model has no deferred
1338 /// constraint window; the keyword is accepted for compatibility.
1339 NoAction,
1340}
1341
1342/// In-cell encoding for a `VECTOR(N)` column. v6.0.1 added the
1343/// optional `USING <encoding>` clause; omitting it keeps the
1344/// pre-v6 `F32` default. `Sq8` quantises each cell to a per-vector
1345/// affine `(min, max, [u8; dim])` triple (4× compression). `F16`
1346/// (v6.0.3, DDL keyword `HALF`) stores each element as IEEE-754
1347/// binary16 (2× compression, ~3 decimal digits of precision).
1348#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
1349pub enum VecEncoding {
1350 /// IEEE-754 binary32. Pre-v6 default; matches pgvector's
1351 /// uncompressed `vector` type wire / storage layout.
1352 #[default]
1353 F32,
1354 /// v6.0.1 SQ8 — per-vector affine 8-bit quantisation. See
1355 /// `spg_storage::quantize::Sq8Vector` for the math + recall
1356 /// envelope (≥ 0.95 on Gaussian / unit-sphere corpora at
1357 /// dim ≥ 32).
1358 Sq8,
1359 /// v6.0.3 halfvec — IEEE-754 binary16 (half-precision)
1360 /// per-element. DDL keyword `HALF` (pgvector convention).
1361 /// Bit-exact dequantise to f32 at the storage layer; no
1362 /// rerank pass needed for kNN search.
1363 F16,
1364}
1365
1366impl fmt::Display for VecEncoding {
1367 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1368 match self {
1369 Self::F32 => f.write_str("F32"),
1370 Self::Sq8 => f.write_str("SQ8"),
1371 // pgvector convention: DDL keyword is `HALF`, not `F16`.
1372 Self::F16 => f.write_str("HALF"),
1373 }
1374 }
1375}
1376
1377/// SQL-level type names. The mapping to the storage runtime's `DataType`
1378/// happens in `spg-engine` — keeping `spg-sql` free of storage deps.
1379#[derive(Debug, Clone, Copy, PartialEq, Eq)]
1380pub enum ColumnTypeName {
1381 SmallInt,
1382 Int,
1383 BigInt,
1384 Float,
1385 Text,
1386 /// `VARCHAR(N)` — TEXT capped at N Unicode characters.
1387 Varchar(u32),
1388 /// `CHAR(N)` — TEXT right-padded with spaces to exactly N characters.
1389 Char(u32),
1390 Bool,
1391 /// pgvector fixed-dimension `VECTOR(N)`. v6.0.1 added the
1392 /// `USING <encoding>` clause; omitting it surfaces as
1393 /// `encoding = VecEncoding::F32` (the pre-v6 default).
1394 Vector {
1395 dim: u32,
1396 encoding: VecEncoding,
1397 },
1398 /// `NUMERIC` / `NUMERIC(p)` / `NUMERIC(p, s)` — exact decimal.
1399 /// Bare `NUMERIC` and `NUMERIC(p)` both surface with `scale=0`.
1400 Numeric(u8, u8),
1401 /// `DATE` — calendar day, no time-of-day component.
1402 Date,
1403 /// `TIMESTAMP` / `MySQL` `DATETIME` — instant with microsecond
1404 /// precision.
1405 Timestamp,
1406 /// v7.9.2 `TIMESTAMPTZ` / `TIMESTAMP WITH TIME ZONE`. SPG
1407 /// stores all timestamps as UTC microseconds-since-epoch and
1408 /// does not carry per-row offset (PG's internal representation
1409 /// is the same — TZ is a display convention). The distinction
1410 /// from `TIMESTAMP` exists for the PG-wire layer to advertise
1411 /// OID 1184 so sqlx-style clients decode into
1412 /// `chrono::DateTime<Utc>` instead of `NaiveDateTime`.
1413 Timestamptz,
1414 /// v4.9 `JSON` — text-backed JSON document. No parse-time
1415 /// validation; the engine round-trips the literal verbatim.
1416 /// PG OID 114 on the wire.
1417 Json,
1418 /// v7.9.0 `JSONB` — same storage shape as Json, advertised as
1419 /// PG OID 3802 on the wire so sqlx-style binary-typed clients
1420 /// decode without a custom type registration.
1421 Jsonb,
1422 /// v7.10.4 `BYTES` / `BYTEA` — raw binary blob. PG wire OID 17.
1423 /// Literal forms (decoded by the engine at coercion time):
1424 /// - PG hex form: `'\xDEADBEEF'`
1425 /// - Escape form: `'foo\\000bar'` (backslash octal triples)
1426 Bytes,
1427 /// v7.10.10 `TEXT[]` — single-dimension TEXT array. PG wire
1428 /// OID 1009. Literal forms accepted by the parser:
1429 /// - `ARRAY['a', 'b', NULL]`
1430 /// - `'{a,b,NULL}'::TEXT[]` (engine decodes the external
1431 /// form at coerce time)
1432 TextArray,
1433 /// v7.11.13 `INT[]` — single-dimension i32 array. PG wire OID
1434 /// 1007. Same literal forms as TEXT[] (substituting integer
1435 /// elements).
1436 IntArray,
1437 /// v7.11.13 `BIGINT[]` — single-dimension i64 array. PG wire
1438 /// OID 1016.
1439 BigIntArray,
1440 /// v7.12.0 `tsvector` — PG full-text search lexeme set. PG
1441 /// wire OID 3614. Literal: `'foo:1 bar:2'::tsvector` (PG
1442 /// external form). G-CRIT-3.
1443 TsVector,
1444 /// v7.12.0 `tsquery` — PG full-text search parse tree. PG
1445 /// wire OID 3615.
1446 TsQuery,
1447 /// v7.17.0 `UUID` — 128-bit identifier. PG wire OID 2950.
1448 /// Literal input accepts canonical hyphenated, unhyphenated,
1449 /// uppercase, and `{...}`-braced forms; display normalises to
1450 /// canonical lowercase 8-4-4-4-12. The drop-in PG surface for
1451 /// Django / Rails / Hibernate `id UUID PRIMARY KEY DEFAULT
1452 /// gen_random_uuid()`.
1453 Uuid,
1454 /// v7.17.0 Phase 3.P0-32 `TIME` (without time zone) — i64
1455 /// microseconds since 00:00:00. PG wire OID 1083. Literal
1456 /// input is `'HH:MM:SS'` with an optional `.fraction` suffix
1457 /// (6-digit microsecond precision). Display normalises to
1458 /// the canonical `HH:MM:SS[.ffffff]`.
1459 Time,
1460 /// v7.17.0 Phase 3.P0-33 MySQL `YEAR` — u16 in range
1461 /// 1901..=2155 plus the zero-year sentinel 0. No dedicated
1462 /// PG OID; advertised as INT4 on the wire. Display always
1463 /// 4 digits zero-padded.
1464 Year,
1465 /// v7.17.0 Phase 3.P0-34 PG `TIME WITH TIME ZONE` (TIMETZ) —
1466 /// i64 us since 00:00:00 (local) + i32 offset_secs from UTC.
1467 /// Wire OID 1266. Literal input is `'HH:MM:SS[.ffffff]±HH[:MM]'`.
1468 /// Offset range: ±14 hours.
1469 TimeTz,
1470 /// v7.17.0 Phase 3.P0-35 PG `MONEY` — i64 cents
1471 /// (locale-independent storage). Wire OID 790. Literal input
1472 /// accepts `$N.NN`, `$N,NNN.NN`, bare integer (treated as
1473 /// major units), optional leading `-`. Display: en_US locale.
1474 Money,
1475 /// v7.17.0 Phase 3.P0-38 PG range types. Pair stores the
1476 /// element kind tag (Int4 / Int8 / Num / Ts / TsTz / Date)
1477 /// — the engine bridges to `DataType::Range(RangeKind)`.
1478 Range(RangeKindAst),
1479 /// v7.17.0 Phase 3.P0-39 PG `hstore` extension type — flat
1480 /// `text => text` map with NULL value support.
1481 Hstore,
1482 /// v7.17.0 Phase 3.P0-40 — 2D arrays for INT / TEXT / BIGINT.
1483 IntArray2D,
1484 BigIntArray2D,
1485 TextArray2D,
1486}
1487
1488/// v7.17.0 Phase 3.P0-38 — PG range element kind. Mirrors
1489/// `spg_storage::RangeKind`; we keep it spg-sql-local so the AST
1490/// crate doesn't depend on storage. Bridged at engine boundary.
1491#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
1492pub enum RangeKindAst {
1493 Int4,
1494 Int8,
1495 Num,
1496 Ts,
1497 TsTz,
1498 Date,
1499}
1500
1501impl fmt::Display for ColumnTypeName {
1502 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1503 match self {
1504 Self::SmallInt => f.write_str("SMALLINT"),
1505 Self::Int => f.write_str("INT"),
1506 Self::BigInt => f.write_str("BIGINT"),
1507 Self::Float => f.write_str("FLOAT"),
1508 Self::Text => f.write_str("TEXT"),
1509 Self::Varchar(n) => write!(f, "VARCHAR({n})"),
1510 Self::Char(n) => write!(f, "CHAR({n})"),
1511 Self::Bool => f.write_str("BOOL"),
1512 Self::Vector { dim, encoding } => match encoding {
1513 VecEncoding::F32 => write!(f, "VECTOR({dim})"),
1514 VecEncoding::Sq8 => write!(f, "VECTOR({dim}) USING SQ8"),
1515 VecEncoding::F16 => write!(f, "VECTOR({dim}) USING HALF"),
1516 },
1517 Self::Json => f.write_str("JSON"),
1518 Self::Jsonb => f.write_str("JSONB"),
1519 Self::Bytes => f.write_str("BYTEA"),
1520 Self::TextArray => f.write_str("TEXT[]"),
1521 Self::IntArray => f.write_str("INT[]"),
1522 Self::BigIntArray => f.write_str("BIGINT[]"),
1523 Self::TsVector => f.write_str("TSVECTOR"),
1524 Self::TsQuery => f.write_str("TSQUERY"),
1525 Self::Uuid => f.write_str("UUID"),
1526 Self::Numeric(p, s) => {
1527 if *s == 0 {
1528 write!(f, "NUMERIC({p})")
1529 } else {
1530 write!(f, "NUMERIC({p}, {s})")
1531 }
1532 }
1533 Self::Date => f.write_str("DATE"),
1534 Self::Timestamp => f.write_str("TIMESTAMP"),
1535 Self::Timestamptz => f.write_str("TIMESTAMPTZ"),
1536 Self::Time => f.write_str("TIME"),
1537 Self::Year => f.write_str("YEAR"),
1538 Self::TimeTz => f.write_str("TIMETZ"),
1539 Self::Money => f.write_str("MONEY"),
1540 Self::Range(k) => f.write_str(match k {
1541 RangeKindAst::Int4 => "INT4RANGE",
1542 RangeKindAst::Int8 => "INT8RANGE",
1543 RangeKindAst::Num => "NUMRANGE",
1544 RangeKindAst::Ts => "TSRANGE",
1545 RangeKindAst::TsTz => "TSTZRANGE",
1546 RangeKindAst::Date => "DATERANGE",
1547 }),
1548 Self::Hstore => f.write_str("HSTORE"),
1549 Self::IntArray2D => f.write_str("INT[][]"),
1550 Self::BigIntArray2D => f.write_str("BIGINT[][]"),
1551 Self::TextArray2D => f.write_str("TEXT[][]"),
1552 }
1553 }
1554}
1555
1556/// `UPDATE <table> SET col = expr [, ...] [WHERE cond]`. v4.4 — the
1557/// engine evaluates `expr` per matched row in the table's row order
1558/// and rewrites cells in place. Indexed columns are dropped + re-
1559/// inserted into the affected B-tree on each row change.
1560#[derive(Debug, Clone, PartialEq)]
1561pub struct UpdateStatement {
1562 pub table: String,
1563 pub assignments: Vec<(String, Expr)>,
1564 pub where_: Option<Expr>,
1565 /// v7.9.4 — `RETURNING <projection>`. None = no RETURNING
1566 /// clause (legacy CommandComplete path). Some = engine
1567 /// evaluates the projection over each mutated row and
1568 /// streams the result as a Rows QueryResult.
1569 pub returning: Option<Vec<SelectItem>>,
1570}
1571
1572/// `DELETE FROM <table> [WHERE cond]`. v4.4 — removes matched rows
1573/// from the active catalog and prunes them from every index.
1574#[derive(Debug, Clone, PartialEq)]
1575pub struct DeleteStatement {
1576 pub table: String,
1577 pub where_: Option<Expr>,
1578 /// v7.9.4 — `RETURNING <projection>`.
1579 pub returning: Option<Vec<SelectItem>>,
1580}
1581
1582/// v7.17.0 Phase 3.P0-42 — SQL:2003 / PG 15+ MERGE statement.
1583/// One WHEN clause fires per source row depending on whether the
1584/// `on` condition matched any target row(s); the executor walks
1585/// `clauses` in declaration order and fires the first whose
1586/// `matched` kind and optional `condition` are both satisfied.
1587#[derive(Debug, Clone, PartialEq)]
1588pub struct MergeStatement {
1589 pub target: String,
1590 pub target_alias: Option<String>,
1591 pub source: String,
1592 pub source_alias: Option<String>,
1593 pub on: Expr,
1594 pub clauses: Vec<MergeWhenClause>,
1595}
1596
1597#[derive(Debug, Clone, PartialEq)]
1598pub struct MergeWhenClause {
1599 pub matched: MergeMatched,
1600 /// Optional `AND <expr>` filter — when present, the clause
1601 /// only fires for the source rows whose match-pair satisfies
1602 /// the predicate.
1603 pub condition: Option<Expr>,
1604 pub action: MergeAction,
1605}
1606
1607#[derive(Debug, Clone, Copy, PartialEq, Eq)]
1608pub enum MergeMatched {
1609 Matched,
1610 NotMatched,
1611}
1612
1613#[derive(Debug, Clone, PartialEq)]
1614pub enum MergeAction {
1615 /// `INSERT (cols) VALUES (vals)`. SPG v7.17 requires the
1616 /// explicit column list (the bare `INSERT VALUES (vals)`
1617 /// shape lands later).
1618 Insert {
1619 columns: Vec<String>,
1620 values: Vec<Expr>,
1621 },
1622 /// `UPDATE SET col = expr [, …]` — applied to every matched
1623 /// target row for the firing source row.
1624 Update { assignments: Vec<(String, Expr)> },
1625 /// `DELETE` — drop every matched target row.
1626 Delete,
1627 /// `DO NOTHING` — explicit no-op (the SQL standard accepts
1628 /// the clause and SPG mirrors so a customer-side MERGE that
1629 /// uses it for branch-control doesn't error).
1630 DoNothing,
1631}
1632
1633#[derive(Debug, Clone, PartialEq)]
1634pub struct InsertStatement {
1635 pub table: String,
1636 /// Optional column list — `INSERT INTO t (a, b) VALUES (...)`. When
1637 /// `None`, every tuple is positional and must match the table arity.
1638 /// When `Some`, the engine maps each tuple slot to the named column and
1639 /// fills the rest with NULL (must be nullable).
1640 pub columns: Option<Vec<String>>,
1641 /// One or more `(expr, expr, ...)` tuples — the multi-row VALUES form.
1642 /// v1.3+ accepts `INSERT INTO t VALUES (a), (b)`. Empty when
1643 /// `select_source` is `Some` (the engine builds rows from the
1644 /// inner SELECT result set instead).
1645 pub rows: Vec<Vec<Expr>>,
1646 /// v7.13.0 — `INSERT INTO t [(cols)] SELECT …` (mailrs
1647 /// round-5 G4). When present, `rows` is empty and the engine
1648 /// materialises the SELECT result, coerces each output tuple to
1649 /// the target column types, and inserts as a single batch.
1650 pub select_source: Option<Box<SelectStatement>>,
1651 /// v7.9.7 — `ON CONFLICT (cols) DO { NOTHING | UPDATE SET … }`
1652 /// upsert clause. None = legacy INSERT (conflict raises a
1653 /// DuplicateKey error). mailrs migration blocker #2.
1654 pub on_conflict: Option<OnConflictClause>,
1655 /// v7.9.4 — `RETURNING <projection>`.
1656 pub returning: Option<Vec<SelectItem>>,
1657}
1658
1659/// v7.9.7 — INSERT upsert clause: `ON CONFLICT (target) DO action`.
1660#[derive(Debug, Clone, PartialEq)]
1661pub struct OnConflictClause {
1662 /// Local columns that identify the conflict (must match a
1663 /// UNIQUE / PRIMARY KEY index on the target table). Empty
1664 /// list means the user wrote `ON CONFLICT DO …` without a
1665 /// target — engine picks the table's first BTree index by
1666 /// convention.
1667 pub target_columns: Vec<String>,
1668 /// The action on conflict.
1669 pub action: OnConflictAction,
1670}
1671
1672/// v7.9.7 — action on conflict.
1673#[derive(Debug, Clone, PartialEq)]
1674pub enum OnConflictAction {
1675 /// `DO NOTHING` — INSERT proceeds for non-conflicting rows,
1676 /// silently skips conflicting ones.
1677 Nothing,
1678 /// `DO UPDATE SET col = expr [, …] [WHERE cond]`. `assignments`
1679 /// may reference `EXCLUDED.col` to read the incoming row's
1680 /// value (engine wires `EXCLUDED` as a virtual table).
1681 Update {
1682 assignments: Vec<(String, Expr)>,
1683 where_: Option<Expr>,
1684 },
1685}
1686
1687#[derive(Debug, Clone, PartialEq)]
1688pub struct SelectStatement {
1689 /// v4.11: `WITH name AS (SELECT ...) [, ...]` common-table
1690 /// expressions, materialised once at query start before the
1691 /// body SELECT runs. Empty for a regular SELECT. Non-recursive
1692 /// only — no `WITH RECURSIVE` for v4.x.
1693 pub ctes: Vec<Cte>,
1694 pub distinct: bool,
1695 pub items: Vec<SelectItem>,
1696 pub from: Option<FromClause>,
1697 pub where_: Option<Expr>,
1698 pub group_by: Option<Vec<Expr>>,
1699 /// v6.4.1 — `GROUP BY ALL` shortcut: when true, the planner
1700 /// expands `group_by` to every non-aggregate SELECT-list item
1701 /// before the executor runs. Mutually exclusive with an
1702 /// explicit `group_by` list (the parser sets exactly one).
1703 pub group_by_all: bool,
1704 /// `HAVING <expr>` — filter applied *after* `GROUP BY` aggregation.
1705 /// Supports aggregate calls (e.g. `HAVING count(*) > 1`); the
1706 /// aggregate executor resolves them through the same synthetic
1707 /// schema used for the SELECT items.
1708 pub having: Option<Expr>,
1709 /// UNION / UNION ALL chain. Empty for a plain SELECT. Each peer is
1710 /// itself a `SelectStatement` with `order_by = None` and `limit =
1711 /// None` (the parser enforces that — ORDER BY / LIMIT belong to the
1712 /// top of the chain).
1713 pub unions: Vec<(UnionKind, SelectStatement)>,
1714 /// v6.4.0 — multi-key ORDER BY. Empty `Vec` means no ORDER BY.
1715 /// Keys are matched left-to-right: first key decides, ties break
1716 /// to the second, etc.
1717 pub order_by: Vec<OrderBy>,
1718 /// `LIMIT <n>` — bound on row output. `n` is an integer
1719 /// literal **or** (v7.9.24) a placeholder `$N` resolved
1720 /// against the prepared-statement Bind values. mailrs
1721 /// migration follow-up H2.
1722 pub limit: Option<LimitExpr>,
1723 /// `OFFSET <n>` — drop the first `n` rows after ORDER BY but
1724 /// before LIMIT (so `LIMIT 10 OFFSET 5` keeps rows 6..=15).
1725 pub offset: Option<LimitExpr>,
1726 /// v7.17.0 Phase 3.P0-49 — `FETCH FIRST <n> ROWS WITH TIES`
1727 /// (SQL:2008). When true and an ORDER BY is present, the
1728 /// executor extends past the LIMIT-truncated tail to include
1729 /// every row whose ORDER BY key equals the last-kept row's
1730 /// key. Requires an ORDER BY; the executor errors otherwise
1731 /// (matching PG's `WITH TIES` rule). The parser was already
1732 /// accepting `WITH TIES` since Phase 5.1; this field captures
1733 /// the choice so the executor can act on it.
1734 pub limit_with_ties: bool,
1735}
1736
1737/// v7.9.24 — LIMIT / OFFSET value. Integer literal at parse
1738/// time or a placeholder `$N` resolved during extended-query
1739/// Bind. mailrs migration follow-up H2.
1740#[derive(Debug, Clone, Copy, PartialEq, Eq)]
1741pub enum LimitExpr {
1742 /// `LIMIT 10` — value known at parse time.
1743 Literal(u32),
1744 /// `LIMIT $N` — the 1-based parameter index, resolved against
1745 /// the bind values when the prepared statement executes.
1746 Placeholder(u16),
1747}
1748
1749impl fmt::Display for LimitExpr {
1750 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1751 match self {
1752 Self::Literal(n) => write!(f, "{n}"),
1753 Self::Placeholder(n) => write!(f, "${n}"),
1754 }
1755 }
1756}
1757
1758impl LimitExpr {
1759 /// Convenience for the simple-query path where no placeholders
1760 /// can possibly exist. Returns the literal value or `None` if
1761 /// this is a placeholder (caller must surface as Unsupported).
1762 pub fn as_literal(self) -> Option<u32> {
1763 match self {
1764 Self::Literal(n) => Some(n),
1765 Self::Placeholder(_) => None,
1766 }
1767 }
1768}
1769
1770/// v7.9.24 — extract LIMIT / OFFSET as a `u32` literal. After
1771/// the engine's `substitute_placeholders` pass these are
1772/// always Literal; in the simple-query path a Placeholder
1773/// shape returns None (executor surfaces as
1774/// "LIMIT/OFFSET ${n} requires prepared-statement binding").
1775impl SelectStatement {
1776 #[must_use]
1777 pub fn limit_literal(&self) -> Option<u32> {
1778 self.limit.and_then(LimitExpr::as_literal)
1779 }
1780 #[must_use]
1781 pub fn offset_literal(&self) -> Option<u32> {
1782 self.offset.and_then(LimitExpr::as_literal)
1783 }
1784}
1785
1786#[derive(Debug, Clone, PartialEq)]
1787pub struct Cte {
1788 pub name: String,
1789 pub body: SelectStatement,
1790 /// v4.22: `WITH RECURSIVE` — set when the WITH clause had the
1791 /// RECURSIVE keyword. Applies to every CTE in the clause per
1792 /// PG semantics. A non-recursive body in a RECURSIVE WITH is
1793 /// allowed; the engine just runs it once.
1794 pub recursive: bool,
1795 /// v4.22: optional `WITH name(a, b, c)` column-name list. When
1796 /// non-empty, these override the body's output column names
1797 /// position-by-position; the engine errors out if the count
1798 /// doesn't match the body's projection width.
1799 pub column_overrides: Vec<String>,
1800}
1801
1802#[derive(Debug, Clone, PartialEq)]
1803pub struct OrderBy {
1804 pub expr: Expr,
1805 /// `false` = ASC (default), `true` = DESC.
1806 pub desc: bool,
1807}
1808
1809#[derive(Debug, Clone, Copy, PartialEq, Eq)]
1810pub enum UnionKind {
1811 /// `UNION` — dedupes the combined set.
1812 Distinct,
1813 /// `UNION ALL` — concatenates without dedup.
1814 All,
1815}
1816
1817#[derive(Debug, Clone, PartialEq)]
1818pub enum SelectItem {
1819 Wildcard,
1820 Expr { expr: Expr, alias: Option<String> },
1821}
1822
1823#[derive(Debug, Clone, PartialEq)]
1824pub struct TableRef {
1825 pub name: String,
1826 pub alias: Option<String>,
1827 /// v6.10.2 — `AS OF SEGMENT '<id>'` cold-tier time-travel.
1828 /// When `Some(id)`, the scan restricts to rows that live in
1829 /// segment `<id>` only — useful for forensic inspection of a
1830 /// specific freezer-emitted segment without exposing the hot
1831 /// tier. `AS OF TIMESTAMP <ts>` (PG-flavoured time travel)
1832 /// is STABILITY carve-out for v6.10 — needs the freezer to
1833 /// stamp each segment with a wall-clock at creation time.
1834 pub as_of_segment: Option<u32>,
1835 /// v7.11.7 — `FROM unnest(<expr>) [AS] <alias>` set-returning
1836 /// source. When `Some`, `name` is the alias (defaulting to
1837 /// `"unnest"` when no `AS` is given) and the engine builds a
1838 /// synthetic single-column table by evaluating the expression
1839 /// once at SELECT entry. Each TEXT[] element becomes one row;
1840 /// NULL elements become NULL cells. v7.11 supported
1841 /// uncorrelated UNNEST only as the FROM primary; v7.13.2
1842 /// (mailrs round-6 S5) widens to UNNEST in any FROM-list
1843 /// position (cross-join with regular tables).
1844 pub unnest_expr: Option<Box<Expr>>,
1845 /// v7.13.2 — mailrs round-6 S5. PG-standard
1846 /// `UNNEST(<arr>) AS alias(col_name)` column-list aliasing:
1847 /// when non-empty, the first entry overrides the projected
1848 /// column name for the unnested column. Empty = fall back to
1849 /// the table alias (pre-v7.13.2 behaviour).
1850 pub unnest_column_aliases: Vec<String>,
1851 /// v7.17.0 Phase 3.10 — `FROM generate_series(start, stop
1852 /// [, step])` set-returning source. When `Some`, the engine
1853 /// materialises a single-column virtual table by stepping
1854 /// `start` to `stop` inclusive. Args are the literal arg list
1855 /// (2 for default-step, 3 for explicit-step). Supports:
1856 /// * SmallInt / Int / BigInt with integer step (default = 1)
1857 /// * Timestamp with INTERVAL step (PG date-range pattern)
1858 /// Mutually exclusive with `unnest_expr` — both populate the
1859 /// same downstream dispatch slot. `name` defaults to
1860 /// `"generate_series"` when no alias is provided.
1861 pub generate_series_args: Option<Vec<Expr>>,
1862 /// v7.17.0 Phase 3.P0-41 — `LATERAL ( SELECT … )` derived
1863 /// table. When `Some`, the TableRef is a parenthesised SELECT
1864 /// that may reference columns from the preceding FROM items
1865 /// (correlated derived table). The executor materialises the
1866 /// subquery per left-row, substituting outer-column references
1867 /// against the current join row's values before running the
1868 /// inner SELECT, then cross-joins the result back.
1869 /// Mutually exclusive with `name` / `unnest_expr` /
1870 /// `generate_series_args`.
1871 pub lateral_subquery: Option<Box<SelectStatement>>,
1872}
1873
1874/// FROM clause shape. v1.10 accepts a primary table plus a flat list of
1875/// joined peers — `FROM a [, b]* [INNER|LEFT] JOIN c ON expr ...`. The
1876/// joins evaluate left-associatively in nested-loop order.
1877#[derive(Debug, Clone, PartialEq)]
1878pub struct FromClause {
1879 pub primary: TableRef,
1880 pub joins: Vec<FromJoin>,
1881}
1882
1883#[derive(Debug, Clone, PartialEq)]
1884pub struct FromJoin {
1885 pub kind: JoinKind,
1886 pub table: TableRef,
1887 /// Required for INNER/LEFT; must be `None` for CROSS / comma-list.
1888 pub on: Option<Expr>,
1889}
1890
1891#[derive(Debug, Clone, Copy, PartialEq, Eq)]
1892pub enum JoinKind {
1893 Inner,
1894 Left,
1895 Cross,
1896}
1897
1898#[derive(Debug, Clone, PartialEq)]
1899pub enum Expr {
1900 Literal(Literal),
1901 Column(ColumnName),
1902 /// v6.1.1 — `$N` parameter placeholder for the extended query
1903 /// protocol. The number is 1-based per PostgreSQL convention.
1904 /// Evaluation looks up `params[N-1]` from the prepared-statement
1905 /// bind buffer; out-of-range indices raise a runtime error
1906 /// (same shape as a column-not-found miss).
1907 Placeholder(u16),
1908 Binary {
1909 lhs: Box<Expr>,
1910 op: BinOp,
1911 rhs: Box<Expr>,
1912 },
1913 Unary {
1914 op: UnOp,
1915 expr: Box<Expr>,
1916 },
1917 /// PG-style `expr::TYPE` cast. v1.3 supports VECTOR, INT, BIGINT, FLOAT,
1918 /// TEXT, BOOL targets; engine coerces at evaluation time.
1919 Cast {
1920 expr: Box<Expr>,
1921 target: CastTarget,
1922 },
1923 /// Postfix `IS NULL` / `IS NOT NULL`. Returns BOOL.
1924 IsNull {
1925 expr: Box<Expr>,
1926 negated: bool,
1927 },
1928 /// Function call `name(args...)`. v1.4 supports a small built-in set
1929 /// (length, upper, lower, abs, coalesce); unknown names error at eval
1930 /// time so the parser stays open for v1.5 aggregates.
1931 FunctionCall {
1932 name: String,
1933 args: Vec<Expr>,
1934 },
1935 /// SQL `LIKE` predicate. `pattern` evaluates to text at runtime;
1936 /// wildcards are `%` (any run) and `_` (one char), backslash escapes
1937 /// the next char (so `\%` matches a literal `%`).
1938 Like {
1939 expr: Box<Expr>,
1940 pattern: Box<Expr>,
1941 negated: bool,
1942 },
1943 /// v4.12 window function call: `name(args) OVER (PARTITION BY
1944 /// ... ORDER BY ...)`. Supports `ROW_NUMBER` / `RANK` /
1945 /// `DENSE_RANK` and the partition-aware aggregates `SUM` /
1946 /// `AVG` / `COUNT` / `MIN` / `MAX`. The window frame defaults to "entire partition" for
1947 /// unordered windows and "from start of partition through
1948 /// current row" for ordered windows — no explicit ROWS /
1949 /// RANGE clause in v4.12 MVP.
1950 WindowFunction {
1951 name: String,
1952 args: Vec<Expr>,
1953 partition_by: Vec<Expr>,
1954 order_by: Vec<(Expr, bool /* desc */)>,
1955 /// v4.20 explicit frame. `None` means "use the default":
1956 /// whole-partition when unordered, running aggregate from
1957 /// partition start through current row when ordered.
1958 frame: Option<WindowFrame>,
1959 /// v6.4.2 — `IGNORE NULLS` / `RESPECT NULLS` modifier on
1960 /// LAG / LEAD / FIRST_VALUE / LAST_VALUE. Default is
1961 /// `Respect` (PG / ANSI default — NULLs participate). Other
1962 /// window functions ignore this flag.
1963 null_treatment: NullTreatment,
1964 },
1965 /// v4.10 scalar subquery — `(SELECT ...)` used in expression
1966 /// position. Must return exactly one row × one column at eval
1967 /// time; the engine errors out otherwise. Uncorrelated only —
1968 /// the inner SELECT cannot reference outer columns.
1969 ScalarSubquery(Box<SelectStatement>),
1970 /// v4.10 `[NOT] EXISTS (SELECT ...)`. Returns Bool. Inner
1971 /// projection is ignored; only row-count matters.
1972 Exists {
1973 subquery: Box<SelectStatement>,
1974 negated: bool,
1975 },
1976 /// v4.10 `expr [NOT] IN (SELECT ...)`. Inner SELECT must
1977 /// project exactly one column; membership is tested by Eq
1978 /// against each row's value (NULL handling follows ANSI:
1979 /// NULL ∈ list ⇒ NULL ; otherwise present ⇒ true).
1980 InSubquery {
1981 expr: Box<Expr>,
1982 subquery: Box<SelectStatement>,
1983 negated: bool,
1984 },
1985 /// `EXTRACT(<field> FROM <source>)` — pull an integer component
1986 /// out of a `DATE` or `TIMESTAMP`. Parsed as its own AST node
1987 /// because the `FROM` keyword is what separates the two halves,
1988 /// not a comma.
1989 Extract {
1990 field: ExtractField,
1991 source: Box<Expr>,
1992 },
1993 /// v7.10.10 — `ARRAY[expr, expr, …]` array constructor. Each
1994 /// element is evaluated independently; NULLs are allowed.
1995 /// v7.10 supports only single-dimension TEXT[] semantically;
1996 /// non-text elements coerce at engine evaluation time when
1997 /// the surrounding context (column type / cast) makes the
1998 /// target clear.
1999 Array(Vec<Expr>),
2000 /// v7.10.10 — array subscript `arr[i]`. PG 1-based; the
2001 /// engine returns NULL for out-of-range indices.
2002 ArraySubscript {
2003 target: Box<Expr>,
2004 index: Box<Expr>,
2005 },
2006 /// v7.10.12 — `expr op ANY(arr)` and `expr op ALL(arr)`. The
2007 /// operator is the comparison binary op (Eq / Ne / Lt / …);
2008 /// the engine desugars: `ANY` returns true if any element
2009 /// satisfies; `ALL` returns true only if every element does.
2010 /// NULL handling follows PG's three-valued logic.
2011 AnyAll {
2012 expr: Box<Expr>,
2013 op: BinOp,
2014 array: Box<Expr>,
2015 /// `true` = ANY, `false` = ALL.
2016 is_any: bool,
2017 },
2018 /// v7.13.0 — `CASE WHEN <cond> THEN <val> ... ELSE <val> END`
2019 /// (searched form, `operand` is None) and
2020 /// `CASE <expr> WHEN <val> THEN <val> ... END` (simple form,
2021 /// `operand` is the lead expression compared against each
2022 /// branch's match). Each `(when_expr, then_expr)` branch
2023 /// stays as written; engine short-circuits on the first match.
2024 /// `else_branch` is `None` when no ELSE; evaluates to NULL.
2025 /// mailrs round-5 G9.
2026 Case {
2027 operand: Option<Box<Expr>>,
2028 branches: Vec<(Expr, Expr)>,
2029 else_branch: Option<Box<Expr>>,
2030 },
2031}
2032
2033/// v6.4.2 — null treatment on `LAG` / `LEAD` / `FIRST_VALUE` /
2034/// `LAST_VALUE`. PG / ANSI default is `Respect` — NULLs participate
2035/// in the offset walk. `Ignore` causes the function to skip NULL
2036/// values in the argument expression, returning the next non-NULL.
2037#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
2038pub enum NullTreatment {
2039 #[default]
2040 Respect,
2041 Ignore,
2042}
2043
2044/// v4.20 explicit window frame: `ROWS|RANGE BETWEEN <bound> AND
2045/// <bound>`. `end` is `None` for the shorthand "ROWS <bound>"
2046/// where end implicitly = CURRENT ROW.
2047#[derive(Debug, Clone, PartialEq, Eq)]
2048pub struct WindowFrame {
2049 pub kind: FrameKind,
2050 pub start: FrameBound,
2051 pub end: Option<FrameBound>,
2052}
2053
2054#[derive(Debug, Clone, Copy, PartialEq, Eq)]
2055pub enum FrameKind {
2056 Rows,
2057 Range,
2058}
2059
2060#[derive(Debug, Clone, PartialEq, Eq)]
2061pub enum FrameBound {
2062 UnboundedPreceding,
2063 OffsetPreceding(u64),
2064 CurrentRow,
2065 OffsetFollowing(u64),
2066 UnboundedFollowing,
2067}
2068
2069impl fmt::Display for FrameBound {
2070 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2071 match self {
2072 Self::UnboundedPreceding => f.write_str("UNBOUNDED PRECEDING"),
2073 Self::OffsetPreceding(n) => write!(f, "{n} PRECEDING"),
2074 Self::CurrentRow => f.write_str("CURRENT ROW"),
2075 Self::OffsetFollowing(n) => write!(f, "{n} FOLLOWING"),
2076 Self::UnboundedFollowing => f.write_str("UNBOUNDED FOLLOWING"),
2077 }
2078 }
2079}
2080
2081#[derive(Debug, Clone, Copy, PartialEq, Eq)]
2082pub enum ExtractField {
2083 Year,
2084 Month,
2085 Day,
2086 Hour,
2087 Minute,
2088 Second,
2089 Microsecond,
2090 /// Seconds since 1970-01-01 00:00:00 UTC (PG returns numeric;
2091 /// SPG keeps the integer convention — truncated seconds).
2092 Epoch,
2093}
2094
2095impl fmt::Display for ExtractField {
2096 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2097 f.write_str(match self {
2098 Self::Year => "YEAR",
2099 Self::Month => "MONTH",
2100 Self::Day => "DAY",
2101 Self::Hour => "HOUR",
2102 Self::Minute => "MINUTE",
2103 Self::Second => "SECOND",
2104 Self::Microsecond => "MICROSECOND",
2105 Self::Epoch => "EPOCH",
2106 })
2107 }
2108}
2109
2110#[derive(Debug, Clone, Copy, PartialEq, Eq)]
2111pub enum CastTarget {
2112 Int,
2113 BigInt,
2114 Float,
2115 Text,
2116 Bool,
2117 Vector,
2118 Date,
2119 Timestamp,
2120 /// v7.9.25 — `::INTERVAL` and `::TIMESTAMPTZ`. mailrs follow-up
2121 /// H3a. Engine reuses the existing runtime-interval / timestamp
2122 /// paths (parse the text input, return the matching Value).
2123 Interval,
2124 Timestamptz,
2125 /// v7.9.25 — `::JSON` and `::JSONB`. SPG already has both
2126 /// types (v7.9.0); the cast just routes Text→Json with the
2127 /// requested OID for the wire layer.
2128 Json,
2129 Jsonb,
2130 /// v7.9.26 — `::regtype` / `::regclass`. Parsed for PG dump
2131 /// compatibility; engine surfaces as Unsupported with a
2132 /// hint to use `SHOW TABLES` or `spg_table_ddl`. mailrs F3b.
2133 RegType,
2134 RegClass,
2135 /// v7.10.11 — `::TEXT[]`. Engine decodes the LHS Text into
2136 /// the PG external array form `{a,b,NULL}`.
2137 TextArray,
2138 /// v7.11.13 — `::INT[]` / `::BIGINT[]`. Decodes PG external
2139 /// `{1,2,3}` or widens a `TextArray` whose elements are
2140 /// integer-shaped.
2141 IntArray,
2142 BigIntArray,
2143 /// v7.12.0 — `::tsvector` / `::tsquery`. Decodes the PG
2144 /// external form text representation. Used by pg_dump output
2145 /// and by `WHERE col @@ 'term'::tsquery` literal patterns.
2146 TsVector,
2147 TsQuery,
2148 /// v7.17.0 — `::uuid`. Decodes the LHS Text via
2149 /// `spg_storage::parse_uuid_str` (accepts canonical hyphenated,
2150 /// unhyphenated, uppercase, and brace-wrapped forms); malformed
2151 /// input is a SQL error.
2152 Uuid,
2153 /// v7.18 — `::bytea`. Decodes the LHS Text via PG's hex form
2154 /// (`'\xdeadbeef'`) or escape form (`'\x05\x00'`); Bytes
2155 /// inputs pass through unchanged. Closes the mailrs D-pre #3
2156 /// reverse-acceptance gap — anywhere a PG schema writes
2157 /// `expr::bytea`, SPG now matches.
2158 Bytea,
2159}
2160
2161impl fmt::Display for CastTarget {
2162 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2163 f.write_str(match self {
2164 Self::Int => "int",
2165 Self::BigInt => "bigint",
2166 Self::Float => "float",
2167 Self::Text => "text",
2168 Self::Bool => "bool",
2169 Self::Vector => "vector",
2170 Self::Interval => "interval",
2171 Self::Timestamptz => "timestamptz",
2172 Self::Json => "json",
2173 Self::Jsonb => "jsonb",
2174 Self::RegType => "regtype",
2175 Self::RegClass => "regclass",
2176 Self::Date => "date",
2177 Self::Timestamp => "timestamp",
2178 Self::TextArray => "TEXT[]",
2179 Self::IntArray => "INT[]",
2180 Self::BigIntArray => "BIGINT[]",
2181 Self::TsVector => "tsvector",
2182 Self::TsQuery => "tsquery",
2183 Self::Uuid => "uuid",
2184 Self::Bytea => "bytea",
2185 })
2186 }
2187}
2188
2189#[derive(Debug, Clone, PartialEq)]
2190pub enum Literal {
2191 Integer(i64),
2192 Float(f64),
2193 String(String),
2194 Bool(bool),
2195 Null,
2196 /// pgvector-style array literal, e.g. `[1, 2.5, -3]`.
2197 Vector(Vec<f32>),
2198 /// TEXT[] value carried through the prepared-bind path
2199 /// (`= ANY($1)` has no column context to re-parse a `{a,b}`
2200 /// text form, so the array rides the AST natively).
2201 TextArray(Vec<Option<String>>),
2202 /// INT[] value carried through the prepared-bind path.
2203 IntArray(Vec<Option<i32>>),
2204 /// BIGINT[] value carried through the prepared-bind path.
2205 BigIntArray(Vec<Option<i64>>),
2206 /// `INTERVAL '<n> <unit> [<n> <unit> ...]'` — calendar-aware span.
2207 /// Split into a months part (because a month is not a fixed number of
2208 /// days) and a microseconds part (everything sub-month). `text` keeps
2209 /// the original spelling so Display round-trips byte-for-byte.
2210 Interval {
2211 months: i32,
2212 micros: i64,
2213 text: String,
2214 },
2215}
2216
2217#[derive(Debug, Clone, PartialEq, Eq)]
2218pub struct ColumnName {
2219 pub qualifier: Option<String>,
2220 pub name: String,
2221}
2222
2223#[derive(Debug, Clone, Copy, PartialEq, Eq)]
2224pub enum BinOp {
2225 Or,
2226 And,
2227 Eq,
2228 NotEq,
2229 /// v7.9.27b — PG `a IS DISTINCT FROM b` / `a IS NOT DISTINCT
2230 /// FROM b`. NULL-safe equality: NULL IS NOT DISTINCT FROM
2231 /// NULL → true, NULL IS DISTINCT FROM NULL → false. The
2232 /// non-NULL behaviour matches `<>` / `=` exactly. Common in
2233 /// PG-style JOIN ON predicates and pg_dump output.
2234 IsDistinctFrom,
2235 IsNotDistinctFrom,
2236 Lt,
2237 LtEq,
2238 Gt,
2239 GtEq,
2240 Add,
2241 Sub,
2242 Mul,
2243 Div,
2244 /// pgvector L2 (Euclidean) distance `<->`. Defined for two vector
2245 /// operands of equal dimension; engine returns `Value::Float(d)`.
2246 L2Distance,
2247 /// pgvector inner-product `<#>` — returns `-Σ aᵢ bᵢ` so "smaller =
2248 /// more similar" remains true (matches pgvector's published convention).
2249 InnerProduct,
2250 /// pgvector cosine distance `<=>` — `1 - (a·b)/(|a| |b|)`.
2251 CosineDistance,
2252 /// SQL string concatenation `||`. NULL propagates.
2253 Concat,
2254 /// Bitwise OR `|` on integers.
2255 BitOr,
2256 /// Bitwise AND `&` on integers.
2257 BitAnd,
2258 /// v4.14 `json -> key` — element access by string key (object)
2259 /// or integer index (array). Returns a JSON value.
2260 JsonGet,
2261 /// v4.14 `json ->> key` — same access, returns the result as
2262 /// TEXT (unwraps a top-level JSON string; renders other scalars
2263 /// as their canonical text).
2264 JsonGetText,
2265 /// v6.4.5 `json #> path_text` — walk the path encoded as a PG
2266 /// text array literal like `'{a,0,b}'`. Returns JSON.
2267 JsonGetPath,
2268 /// v6.4.5 `json #>> path_text` — same walk, returns TEXT.
2269 JsonGetPathText,
2270 /// v6.4.5 `json @> sub_json` — containment. Returns BOOL; true
2271 /// when every key/value in `sub_json` is structurally present in
2272 /// the left side. Matches PG semantics (top-level + recursive).
2273 JsonContains,
2274 /// v7.12.2 `tsvector @@ tsquery` — FTS match. Returns BOOL;
2275 /// 3VL on NULL. Symmetric: PG also accepts `tsquery @@
2276 /// tsvector` and engine eval normalises either ordering.
2277 TsMatch,
2278 /// v7.17.0 Phase 3.P0-47 — PG INET / CIDR strict contained-in
2279 /// `<<`. LHS network is strictly inside RHS network (no equality).
2280 InetContainedBy,
2281 /// v7.17.0 Phase 3.P0-47 — PG INET / CIDR contained-in-or-equal
2282 /// `<<=`. LHS network ⊆ RHS network.
2283 InetContainedByEq,
2284 /// v7.17.0 Phase 3.P0-47 — PG INET / CIDR strict contains `>>`.
2285 /// LHS network strictly contains RHS network.
2286 InetContains,
2287 /// v7.17.0 Phase 3.P0-47 — PG INET / CIDR contains-or-equal `>>=`.
2288 /// LHS network ⊇ RHS network.
2289 InetContainsEq,
2290 /// v7.17.0 Phase 3.P0-47 — PG INET / CIDR network overlap `&&`.
2291 /// True iff either network contains any address of the other.
2292 InetOverlap,
2293}
2294
2295#[derive(Debug, Clone, Copy, PartialEq, Eq)]
2296pub enum UnOp {
2297 Not,
2298 Neg,
2299 /// Bitwise NOT `~` on integers.
2300 BitNot,
2301}
2302
2303// --- Display impls (round-trip-safe) --------------------------------------
2304
2305impl Statement {
2306 /// v7.18 — classify whether the statement is read-only at
2307 /// engine level. Used by `spg-sqlx`'s `SpgConnection` to
2308 /// route SELECT-shaped traffic through the fan-out
2309 /// `AsyncReadHandle` (no writer-lock contention) while
2310 /// keeping DML / DDL / TX-control on the single-writer path.
2311 ///
2312 /// The classification matches what
2313 /// `Engine::execute_readonly_with_cancel` accepts: anything
2314 /// that does NOT mutate catalog, statistics, session state,
2315 /// or transaction state. WaitForWalPosition is included
2316 /// (engine returns `Unsupported`, but the classification is
2317 /// semantically read-only — no mutation). Empty is excluded
2318 /// out of an abundance of caution — the no-op routes
2319 /// through the writer so any future side effect lands
2320 /// uniformly.
2321 ///
2322 /// **Not connection-state aware**. `SET LOCAL` / `RESET`
2323 /// affect session parameters and must run on the writer
2324 /// engine that owns the session state; they classify as
2325 /// writer-path here. Same for `BEGIN` / `COMMIT` /
2326 /// `ROLLBACK` / `SAVEPOINT` — transaction control is
2327 /// always writer-path.
2328 #[must_use]
2329 pub fn is_readonly(&self) -> bool {
2330 match self {
2331 Statement::Select(_)
2332 | Statement::Explain(_)
2333 | Statement::ShowTables
2334 | Statement::ShowDatabases
2335 | Statement::ShowCreateTable(_)
2336 | Statement::ShowIndexes(_)
2337 | Statement::ShowStatus
2338 | Statement::ShowVariables
2339 | Statement::ShowProcesslist
2340 | Statement::ShowColumns(_)
2341 | Statement::ShowUsers
2342 | Statement::ShowPublications
2343 | Statement::ShowSubscriptions
2344 | Statement::WaitForWalPosition { .. } => true,
2345 // Everything else mutates catalog, statistics,
2346 // session state, or transaction state — writer path.
2347 // Listed explicitly so a new Statement variant fails
2348 // the match exhaustiveness check and forces a
2349 // classification decision at add-site.
2350 Statement::Empty
2351 | Statement::DropTable { .. }
2352 | Statement::DropIndex { .. }
2353 | Statement::CreateTable(_)
2354 | Statement::CreateExtension(_)
2355 | Statement::DoBlock(_)
2356 | Statement::CreateIndex(_)
2357 | Statement::Insert(_)
2358 | Statement::Update(_)
2359 | Statement::Delete(_)
2360 | Statement::Merge(_)
2361 | Statement::Begin
2362 | Statement::Commit
2363 | Statement::Rollback
2364 | Statement::Savepoint(_)
2365 | Statement::RollbackToSavepoint(_)
2366 | Statement::ReleaseSavepoint(_)
2367 | Statement::CreateUser(_)
2368 | Statement::DropUser(_)
2369 | Statement::AlterIndex(_)
2370 | Statement::AlterTable(_)
2371 | Statement::CreatePublication(_)
2372 | Statement::DropPublication(_)
2373 | Statement::CreateSubscription(_)
2374 | Statement::DropSubscription(_)
2375 | Statement::Analyze(_)
2376 | Statement::CompactColdSegments
2377 | Statement::SetParameter { .. }
2378 | Statement::SetParameterList(_)
2379 | Statement::ResetParameter(_)
2380 | Statement::CreateFunction(_)
2381 | Statement::CreateTrigger(_)
2382 | Statement::DropTrigger { .. }
2383 | Statement::DropFunction { .. }
2384 | Statement::CreateSequence(_)
2385 | Statement::AlterSequence(_)
2386 | Statement::DropSequence { .. }
2387 | Statement::CreateView(_)
2388 | Statement::DropView { .. }
2389 | Statement::CreateMaterializedView(_)
2390 | Statement::RefreshMaterializedView { .. }
2391 | Statement::DropMaterializedView { .. }
2392 | Statement::CreateType(_)
2393 | Statement::DropType { .. }
2394 | Statement::CreateDomain(_)
2395 | Statement::DropDomain { .. }
2396 | Statement::CreateSchema { .. }
2397 | Statement::DropSchema { .. } => false,
2398 }
2399 }
2400}
2401
2402impl fmt::Display for Statement {
2403 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2404 match self {
2405 Self::Empty => Ok(()),
2406 Self::DropTable { names, if_exists } => {
2407 f.write_str("DROP TABLE ")?;
2408 if *if_exists {
2409 f.write_str("IF EXISTS ")?;
2410 }
2411 for (i, n) in names.iter().enumerate() {
2412 if i > 0 {
2413 f.write_str(", ")?;
2414 }
2415 write!(f, "{}", quote_ident(n))?;
2416 }
2417 Ok(())
2418 }
2419 Self::DropIndex { name, if_exists } => {
2420 f.write_str("DROP INDEX ")?;
2421 if *if_exists {
2422 f.write_str("IF EXISTS ")?;
2423 }
2424 write!(f, "{}", quote_ident(name))
2425 }
2426 Self::Select(s) => s.fmt(f),
2427 Self::CreateTable(s) => s.fmt(f),
2428 Self::CreateIndex(s) => s.fmt(f),
2429 Self::Insert(s) => s.fmt(f),
2430 Self::Update(s) => s.fmt(f),
2431 Self::Delete(s) => s.fmt(f),
2432 Self::Merge(s) => {
2433 // v7.17.0 Phase 3.P0-42 — MERGE display is approximate
2434 // (it round-trips for the cases tests cover, not for
2435 // round-tripping every edge of the surface).
2436 f.write_str("MERGE INTO ")?;
2437 write!(f, "{}", quote_ident(&s.target))?;
2438 if let Some(a) = &s.target_alias {
2439 write!(f, " {}", quote_ident(a))?;
2440 }
2441 f.write_str(" USING ")?;
2442 write!(f, "{}", quote_ident(&s.source))?;
2443 if let Some(a) = &s.source_alias {
2444 write!(f, " {}", quote_ident(a))?;
2445 }
2446 write!(f, " ON {}", s.on)?;
2447 for clause in &s.clauses {
2448 f.write_str(" WHEN ")?;
2449 f.write_str(match clause.matched {
2450 MergeMatched::Matched => "MATCHED",
2451 MergeMatched::NotMatched => "NOT MATCHED",
2452 })?;
2453 if let Some(c) = &clause.condition {
2454 write!(f, " AND {c}")?;
2455 }
2456 f.write_str(" THEN ")?;
2457 match &clause.action {
2458 MergeAction::Insert { columns, values } => {
2459 f.write_str("INSERT (")?;
2460 for (i, c) in columns.iter().enumerate() {
2461 if i > 0 {
2462 f.write_str(", ")?;
2463 }
2464 write!(f, "{}", quote_ident(c))?;
2465 }
2466 f.write_str(") VALUES (")?;
2467 for (i, v) in values.iter().enumerate() {
2468 if i > 0 {
2469 f.write_str(", ")?;
2470 }
2471 write!(f, "{v}")?;
2472 }
2473 f.write_str(")")?;
2474 }
2475 MergeAction::Update { assignments } => {
2476 f.write_str("UPDATE SET ")?;
2477 for (i, (c, e)) in assignments.iter().enumerate() {
2478 if i > 0 {
2479 f.write_str(", ")?;
2480 }
2481 write!(f, "{} = {e}", quote_ident(c))?;
2482 }
2483 }
2484 MergeAction::Delete => f.write_str("DELETE")?,
2485 MergeAction::DoNothing => f.write_str("DO NOTHING")?,
2486 }
2487 }
2488 Ok(())
2489 }
2490 Self::Begin => f.write_str("BEGIN"),
2491 Self::Commit => f.write_str("COMMIT"),
2492 Self::Rollback => f.write_str("ROLLBACK"),
2493 Self::Savepoint(n) => write!(f, "SAVEPOINT {}", quote_ident(n)),
2494 Self::RollbackToSavepoint(n) => write!(f, "ROLLBACK TO SAVEPOINT {}", quote_ident(n)),
2495 Self::ReleaseSavepoint(n) => write!(f, "RELEASE SAVEPOINT {}", quote_ident(n)),
2496 Self::ShowTables => f.write_str("SHOW TABLES"),
2497 Self::ShowDatabases => f.write_str("SHOW DATABASES"),
2498 Self::ShowCreateTable(t) => write!(f, "SHOW CREATE TABLE {}", quote_ident(t)),
2499 Self::ShowIndexes(t) => write!(f, "SHOW INDEXES FROM {}", quote_ident(t)),
2500 Self::ShowStatus => f.write_str("SHOW STATUS"),
2501 Self::ShowVariables => f.write_str("SHOW VARIABLES"),
2502 Self::ShowProcesslist => f.write_str("SHOW PROCESSLIST"),
2503 Self::ShowColumns(t) => write!(f, "SHOW COLUMNS FROM {}", quote_ident(t)),
2504 Self::CreateUser(s) => write!(
2505 f,
2506 "CREATE USER {} WITH PASSWORD '<redacted>' ROLE '{}'",
2507 quote_ident(&s.name),
2508 s.role
2509 ),
2510 Self::DropUser(n) => write!(f, "DROP USER {}", quote_ident(n)),
2511 Self::ShowUsers => f.write_str("SHOW USERS"),
2512 Self::ShowPublications => f.write_str("SHOW PUBLICATIONS"),
2513 Self::ShowSubscriptions => f.write_str("SHOW SUBSCRIPTIONS"),
2514 Self::CreateSubscription(s) => {
2515 write!(
2516 f,
2517 "CREATE SUBSCRIPTION {} CONNECTION '{}' PUBLICATION ",
2518 quote_ident(&s.name),
2519 s.conn_str.replace('\'', "''")
2520 )?;
2521 for (i, p) in s.publications.iter().enumerate() {
2522 if i > 0 {
2523 f.write_str(", ")?;
2524 }
2525 write!(f, "{}", quote_ident(p))?;
2526 }
2527 Ok(())
2528 }
2529 Self::DropSubscription(name) => {
2530 write!(f, "DROP SUBSCRIPTION {}", quote_ident(name))
2531 }
2532 Self::WaitForWalPosition { pos, timeout_ms } => {
2533 write!(f, "WAIT FOR WAL POSITION {pos}")?;
2534 if let Some(ms) = timeout_ms {
2535 write!(f, " WITH TIMEOUT {ms}")?;
2536 }
2537 Ok(())
2538 }
2539 Self::Analyze(None) => f.write_str("ANALYZE"),
2540 Self::Analyze(Some(t)) => write!(f, "ANALYZE {}", quote_ident(t)),
2541 Self::CompactColdSegments => f.write_str("COMPACT COLD SEGMENTS"),
2542 Self::Explain(e) => {
2543 if e.suggest {
2544 write!(f, "EXPLAIN (SUGGEST) {}", e.inner)
2545 } else if e.analyze {
2546 write!(f, "EXPLAIN ANALYZE {}", e.inner)
2547 } else {
2548 write!(f, "EXPLAIN {}", e.inner)
2549 }
2550 }
2551 Self::AlterIndex(a) => {
2552 write!(f, "ALTER INDEX ")?;
2553 match &a.target {
2554 AlterIndexTarget::Rebuild { encoding } => {
2555 write!(f, "{} REBUILD", quote_ident(&a.name))?;
2556 if let Some(enc) = encoding {
2557 write!(f, " WITH (encoding = {enc})")?;
2558 }
2559 Ok(())
2560 }
2561 AlterIndexTarget::Rename { new, if_exists } => {
2562 if *if_exists {
2563 f.write_str("IF EXISTS ")?;
2564 }
2565 write!(f, "{} RENAME TO {}", quote_ident(&a.name), quote_ident(new))
2566 }
2567 }
2568 }
2569 Self::AlterTable(a) => {
2570 write!(f, "ALTER TABLE {} ", quote_ident(&a.name))?;
2571 for (i, t) in a.targets.iter().enumerate() {
2572 if i > 0 {
2573 f.write_str(", ")?;
2574 }
2575 fmt_alter_target(f, t)?;
2576 }
2577 Ok(())
2578 }
2579 Self::CreatePublication(p) => {
2580 write!(f, "CREATE PUBLICATION {}", quote_ident(&p.name))?;
2581 match &p.scope {
2582 PublicationScope::AllTables => f.write_str(" FOR ALL TABLES"),
2583 PublicationScope::ForTables(ts) => {
2584 f.write_str(" FOR TABLE ")?;
2585 for (i, t) in ts.iter().enumerate() {
2586 if i > 0 {
2587 f.write_str(", ")?;
2588 }
2589 write!(f, "{}", quote_ident(t))?;
2590 }
2591 Ok(())
2592 }
2593 PublicationScope::AllTablesExcept(ts) => {
2594 f.write_str(" FOR ALL TABLES EXCEPT ")?;
2595 for (i, t) in ts.iter().enumerate() {
2596 if i > 0 {
2597 f.write_str(", ")?;
2598 }
2599 write!(f, "{}", quote_ident(t))?;
2600 }
2601 Ok(())
2602 }
2603 }
2604 }
2605 Self::CreateExtension(name) => {
2606 write!(f, "CREATE EXTENSION IF NOT EXISTS {}", quote_ident(name))
2607 }
2608 Self::DoBlock(body) => write!(f, "DO $$ {body} $$"),
2609 Self::DropPublication(name) => {
2610 write!(f, "DROP PUBLICATION {}", quote_ident(name))
2611 }
2612 Self::SetParameter { name, value } => {
2613 write!(f, "SET {name} = ")?;
2614 match value {
2615 SetValue::String(s) => write!(f, "'{}'", s.replace('\'', "''")),
2616 SetValue::Ident(s) | SetValue::Number(s) => f.write_str(s),
2617 SetValue::Default => f.write_str("DEFAULT"),
2618 }
2619 }
2620 Self::SetParameterList(pairs) => {
2621 f.write_str("SET ")?;
2622 for (i, (name, value)) in pairs.iter().enumerate() {
2623 if i > 0 {
2624 f.write_str(", ")?;
2625 }
2626 write!(f, "{name} = ")?;
2627 match value {
2628 SetValue::String(s) => write!(f, "'{}'", s.replace('\'', "''"))?,
2629 SetValue::Ident(s) | SetValue::Number(s) => f.write_str(s)?,
2630 SetValue::Default => f.write_str("DEFAULT")?,
2631 }
2632 }
2633 Ok(())
2634 }
2635 Self::ResetParameter(None) => f.write_str("RESET ALL"),
2636 Self::ResetParameter(Some(name)) => write!(f, "RESET {name}"),
2637 Self::CreateFunction(s) => s.fmt(f),
2638 Self::CreateTrigger(s) => s.fmt(f),
2639 Self::DropTrigger {
2640 name,
2641 table,
2642 if_exists,
2643 } => {
2644 f.write_str("DROP TRIGGER ")?;
2645 if *if_exists {
2646 f.write_str("IF EXISTS ")?;
2647 }
2648 write!(f, "{} ON {}", quote_ident(name), quote_ident(table))
2649 }
2650 Self::DropFunction { name, if_exists } => {
2651 f.write_str("DROP FUNCTION ")?;
2652 if *if_exists {
2653 f.write_str("IF EXISTS ")?;
2654 }
2655 write!(f, "{}", quote_ident(name))
2656 }
2657 Self::CreateSequence(s) => s.fmt(f),
2658 Self::AlterSequence(s) => s.fmt(f),
2659 Self::DropSequence { names, if_exists } => {
2660 f.write_str("DROP SEQUENCE ")?;
2661 if *if_exists {
2662 f.write_str("IF EXISTS ")?;
2663 }
2664 for (i, n) in names.iter().enumerate() {
2665 if i > 0 {
2666 f.write_str(", ")?;
2667 }
2668 write!(f, "{}", quote_ident(n))?;
2669 }
2670 Ok(())
2671 }
2672 Self::CreateView(v) => v.fmt(f),
2673 Self::DropView { names, if_exists } => {
2674 f.write_str("DROP VIEW ")?;
2675 if *if_exists {
2676 f.write_str("IF EXISTS ")?;
2677 }
2678 for (i, n) in names.iter().enumerate() {
2679 if i > 0 {
2680 f.write_str(", ")?;
2681 }
2682 write!(f, "{}", quote_ident(n))?;
2683 }
2684 Ok(())
2685 }
2686 Self::CreateMaterializedView(v) => v.fmt(f),
2687 Self::RefreshMaterializedView { name, with_data } => {
2688 write!(f, "REFRESH MATERIALIZED VIEW {}", quote_ident(name))?;
2689 if !*with_data {
2690 f.write_str(" WITH NO DATA")?;
2691 }
2692 Ok(())
2693 }
2694 Self::DropMaterializedView { names, if_exists } => {
2695 f.write_str("DROP MATERIALIZED VIEW ")?;
2696 if *if_exists {
2697 f.write_str("IF EXISTS ")?;
2698 }
2699 for (i, n) in names.iter().enumerate() {
2700 if i > 0 {
2701 f.write_str(", ")?;
2702 }
2703 write!(f, "{}", quote_ident(n))?;
2704 }
2705 Ok(())
2706 }
2707 Self::CreateType(t) => t.fmt(f),
2708 Self::DropType { names, if_exists } => {
2709 f.write_str("DROP TYPE ")?;
2710 if *if_exists {
2711 f.write_str("IF EXISTS ")?;
2712 }
2713 for (i, n) in names.iter().enumerate() {
2714 if i > 0 {
2715 f.write_str(", ")?;
2716 }
2717 write!(f, "{}", quote_ident(n))?;
2718 }
2719 Ok(())
2720 }
2721 Self::CreateDomain(d) => d.fmt(f),
2722 Self::DropDomain { names, if_exists } => {
2723 f.write_str("DROP DOMAIN ")?;
2724 if *if_exists {
2725 f.write_str("IF EXISTS ")?;
2726 }
2727 for (i, n) in names.iter().enumerate() {
2728 if i > 0 {
2729 f.write_str(", ")?;
2730 }
2731 write!(f, "{}", quote_ident(n))?;
2732 }
2733 Ok(())
2734 }
2735 Self::CreateSchema {
2736 name,
2737 if_not_exists,
2738 } => {
2739 f.write_str("CREATE SCHEMA ")?;
2740 if *if_not_exists {
2741 f.write_str("IF NOT EXISTS ")?;
2742 }
2743 write!(f, "{}", quote_ident(name))
2744 }
2745 Self::DropSchema { names, if_exists } => {
2746 f.write_str("DROP SCHEMA ")?;
2747 if *if_exists {
2748 f.write_str("IF EXISTS ")?;
2749 }
2750 for (i, n) in names.iter().enumerate() {
2751 if i > 0 {
2752 f.write_str(", ")?;
2753 }
2754 write!(f, "{}", quote_ident(n))?;
2755 }
2756 Ok(())
2757 }
2758 }
2759 }
2760}
2761
2762impl fmt::Display for CreateDomainStatement {
2763 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2764 write!(
2765 f,
2766 "CREATE DOMAIN {} AS {}",
2767 quote_ident(&self.name),
2768 self.base_type
2769 )?;
2770 if let Some(d) = &self.default {
2771 write!(f, " DEFAULT {d}")?;
2772 }
2773 if self.not_null {
2774 f.write_str(" NOT NULL")?;
2775 }
2776 for c in &self.checks {
2777 write!(f, " CHECK ({c})")?;
2778 }
2779 Ok(())
2780 }
2781}
2782
2783impl fmt::Display for CreateTypeStatement {
2784 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2785 write!(f, "CREATE TYPE {} AS ", quote_ident(&self.name))?;
2786 match &self.kind {
2787 TypeKind::Enum { labels } => {
2788 f.write_str("ENUM (")?;
2789 for (i, l) in labels.iter().enumerate() {
2790 if i > 0 {
2791 f.write_str(", ")?;
2792 }
2793 write!(f, "'{}'", l.replace('\'', "''"))?;
2794 }
2795 f.write_str(")")
2796 }
2797 }
2798 }
2799}
2800
2801impl fmt::Display for CreateMaterializedViewStatement {
2802 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2803 f.write_str("CREATE MATERIALIZED VIEW ")?;
2804 if self.if_not_exists {
2805 f.write_str("IF NOT EXISTS ")?;
2806 }
2807 write!(f, "{}", quote_ident(&self.name))?;
2808 if !self.columns.is_empty() {
2809 f.write_str(" (")?;
2810 for (i, c) in self.columns.iter().enumerate() {
2811 if i > 0 {
2812 f.write_str(", ")?;
2813 }
2814 write!(f, "{}", quote_ident(c))?;
2815 }
2816 f.write_str(")")?;
2817 }
2818 write!(f, " AS {}", self.body)?;
2819 if !self.with_data {
2820 f.write_str(" WITH NO DATA")?;
2821 }
2822 Ok(())
2823 }
2824}
2825
2826impl fmt::Display for CreateViewStatement {
2827 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2828 f.write_str("CREATE ")?;
2829 if self.or_replace {
2830 f.write_str("OR REPLACE ")?;
2831 }
2832 if self.temporary {
2833 f.write_str("TEMPORARY ")?;
2834 }
2835 f.write_str("VIEW ")?;
2836 if self.if_not_exists {
2837 f.write_str("IF NOT EXISTS ")?;
2838 }
2839 write!(f, "{}", quote_ident(&self.name))?;
2840 if !self.columns.is_empty() {
2841 f.write_str(" (")?;
2842 for (i, c) in self.columns.iter().enumerate() {
2843 if i > 0 {
2844 f.write_str(", ")?;
2845 }
2846 write!(f, "{}", quote_ident(c))?;
2847 }
2848 f.write_str(")")?;
2849 }
2850 write!(f, " AS {}", self.body)
2851 }
2852}
2853
2854impl fmt::Display for CreateSequenceStatement {
2855 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2856 f.write_str("CREATE ")?;
2857 if self.temporary {
2858 f.write_str("TEMPORARY ")?;
2859 }
2860 f.write_str("SEQUENCE ")?;
2861 if self.if_not_exists {
2862 f.write_str("IF NOT EXISTS ")?;
2863 }
2864 write!(f, "{}", quote_ident(&self.name))?;
2865 if let Some(dt) = self.data_type {
2866 write!(f, " AS {dt}")?;
2867 }
2868 write_sequence_options(f, &self.options)
2869 }
2870}
2871
2872impl fmt::Display for AlterSequenceStatement {
2873 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2874 f.write_str("ALTER SEQUENCE ")?;
2875 if self.if_exists {
2876 f.write_str("IF EXISTS ")?;
2877 }
2878 write!(f, "{}", quote_ident(&self.name))?;
2879 write_sequence_options(f, &self.options)
2880 }
2881}
2882
2883impl fmt::Display for SequenceDataType {
2884 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2885 f.write_str(match self {
2886 Self::SmallInt => "smallint",
2887 Self::Int => "integer",
2888 Self::BigInt => "bigint",
2889 })
2890 }
2891}
2892
2893fn write_sequence_options(f: &mut fmt::Formatter<'_>, o: &SequenceOptions) -> fmt::Result {
2894 if let Some(n) = o.increment {
2895 write!(f, " INCREMENT BY {n}")?;
2896 }
2897 match o.min_value {
2898 Some(SeqBound::Value(n)) => write!(f, " MINVALUE {n}")?,
2899 Some(SeqBound::NoBound) => f.write_str(" NO MINVALUE")?,
2900 None => {}
2901 }
2902 match o.max_value {
2903 Some(SeqBound::Value(n)) => write!(f, " MAXVALUE {n}")?,
2904 Some(SeqBound::NoBound) => f.write_str(" NO MAXVALUE")?,
2905 None => {}
2906 }
2907 if let Some(n) = o.start {
2908 write!(f, " START WITH {n}")?;
2909 }
2910 match o.restart {
2911 Some(Some(n)) => write!(f, " RESTART WITH {n}")?,
2912 Some(None) => f.write_str(" RESTART")?,
2913 None => {}
2914 }
2915 if let Some(n) = o.cache {
2916 write!(f, " CACHE {n}")?;
2917 }
2918 match o.cycle {
2919 Some(true) => f.write_str(" CYCLE")?,
2920 Some(false) => f.write_str(" NO CYCLE")?,
2921 None => {}
2922 }
2923 if let Some(ob) = &o.owned_by {
2924 match ob {
2925 SequenceOwnedBy::None => f.write_str(" OWNED BY NONE")?,
2926 SequenceOwnedBy::Column { table, column } => {
2927 write!(
2928 f,
2929 " OWNED BY {}.{}",
2930 quote_ident(table),
2931 quote_ident(column)
2932 )?;
2933 }
2934 }
2935 }
2936 Ok(())
2937}
2938
2939impl fmt::Display for CreateFunctionStatement {
2940 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2941 f.write_str("CREATE ")?;
2942 if self.or_replace {
2943 f.write_str("OR REPLACE ")?;
2944 }
2945 write!(f, "FUNCTION {}(", quote_ident(&self.name))?;
2946 for (i, arg) in self.args.iter().enumerate() {
2947 if i > 0 {
2948 f.write_str(", ")?;
2949 }
2950 match arg.mode {
2951 FunctionArgMode::In => {}
2952 FunctionArgMode::Out => f.write_str("OUT ")?,
2953 FunctionArgMode::InOut => f.write_str("INOUT ")?,
2954 }
2955 if let Some(name) = &arg.name {
2956 write!(f, "{} ", quote_ident(name))?;
2957 }
2958 match &arg.ty {
2959 FunctionArgType::Typed(t) => write!(f, "{t}")?,
2960 FunctionArgType::Raw(s) => f.write_str(s)?,
2961 }
2962 }
2963 f.write_str(") RETURNS ")?;
2964 match &self.returns {
2965 FunctionReturn::Trigger => f.write_str("TRIGGER")?,
2966 FunctionReturn::Void => f.write_str("VOID")?,
2967 FunctionReturn::Type(t) => write!(f, "{t}")?,
2968 FunctionReturn::Other(s) => f.write_str(s)?,
2969 }
2970 write!(f, " LANGUAGE {} AS $$", self.language)?;
2971 match &self.body {
2972 FunctionBody::PlPgSql(b) => write!(f, "\n{b}\n")?,
2973 FunctionBody::Raw(s) => f.write_str(s)?,
2974 }
2975 f.write_str("$$")
2976 }
2977}
2978
2979impl fmt::Display for PlPgSqlBlock {
2980 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2981 if !self.declarations.is_empty() {
2982 f.write_str("DECLARE\n")?;
2983 for d in &self.declarations {
2984 write!(f, " {} ", quote_ident(&d.name))?;
2985 match &d.ty {
2986 FunctionArgType::Typed(t) => write!(f, "{t}")?,
2987 FunctionArgType::Raw(s) => f.write_str(s)?,
2988 }
2989 if let Some(e) = &d.default {
2990 write!(f, " := {e}")?;
2991 }
2992 f.write_str(";\n")?;
2993 }
2994 }
2995 f.write_str("BEGIN\n")?;
2996 for stmt in &self.statements {
2997 writeln!(f, " {stmt};")?;
2998 }
2999 f.write_str("END")
3000 }
3001}
3002
3003impl fmt::Display for PlPgSqlStmt {
3004 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3005 match self {
3006 Self::Assign { target, value } => write!(f, "{target} := {value}"),
3007 Self::SelectInto { var, body } => write!(f, "{body} INTO {var}"),
3008 Self::Return(t) => match t {
3009 ReturnTarget::New => f.write_str("RETURN NEW"),
3010 ReturnTarget::Old => f.write_str("RETURN OLD"),
3011 ReturnTarget::Null => f.write_str("RETURN NULL"),
3012 ReturnTarget::Expr(e) => write!(f, "RETURN {e}"),
3013 },
3014 Self::If {
3015 branches,
3016 else_branch,
3017 } => {
3018 for (i, (cond, body)) in branches.iter().enumerate() {
3019 if i == 0 {
3020 write!(f, "IF {cond} THEN ")?;
3021 } else {
3022 write!(f, " ELSIF {cond} THEN ")?;
3023 }
3024 for (j, s) in body.iter().enumerate() {
3025 if j > 0 {
3026 f.write_str("; ")?;
3027 }
3028 write!(f, "{s}")?;
3029 }
3030 }
3031 if !else_branch.is_empty() {
3032 f.write_str(" ELSE ")?;
3033 for (j, s) in else_branch.iter().enumerate() {
3034 if j > 0 {
3035 f.write_str("; ")?;
3036 }
3037 write!(f, "{s}")?;
3038 }
3039 }
3040 f.write_str(" END IF")
3041 }
3042 Self::Raise {
3043 level,
3044 message,
3045 args,
3046 } => {
3047 let lvl = match level {
3048 RaiseLevel::Notice => "NOTICE",
3049 RaiseLevel::Warning => "WARNING",
3050 RaiseLevel::Info => "INFO",
3051 RaiseLevel::Log => "LOG",
3052 RaiseLevel::Debug => "DEBUG",
3053 RaiseLevel::Exception => "EXCEPTION",
3054 };
3055 write!(f, "RAISE {lvl} '{}'", message.replace('\'', "''"))?;
3056 for a in args {
3057 write!(f, ", {a}")?;
3058 }
3059 Ok(())
3060 }
3061 Self::EmbeddedSql(s) => write!(f, "{s}"),
3062 }
3063 }
3064}
3065
3066impl fmt::Display for AssignTarget {
3067 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3068 match self {
3069 Self::NewColumn(c) => write!(f, "NEW.{}", quote_ident(c)),
3070 Self::OldColumn(c) => write!(f, "OLD.{}", quote_ident(c)),
3071 Self::Local(n) => f.write_str(n),
3072 }
3073 }
3074}
3075
3076impl fmt::Display for CreateTriggerStatement {
3077 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3078 f.write_str("CREATE ")?;
3079 if self.or_replace {
3080 f.write_str("OR REPLACE ")?;
3081 }
3082 write!(f, "TRIGGER {} ", quote_ident(&self.name))?;
3083 match self.timing {
3084 TriggerTiming::Before => f.write_str("BEFORE")?,
3085 TriggerTiming::After => f.write_str("AFTER")?,
3086 TriggerTiming::InsteadOf => f.write_str("INSTEAD OF")?,
3087 }
3088 for (i, e) in self.events.iter().enumerate() {
3089 if i == 0 {
3090 f.write_str(" ")?;
3091 } else {
3092 f.write_str(" OR ")?;
3093 }
3094 match e {
3095 TriggerEvent::Insert => f.write_str("INSERT")?,
3096 TriggerEvent::Update => {
3097 f.write_str("UPDATE")?;
3098 if !self.update_columns.is_empty() {
3099 f.write_str(" OF ")?;
3100 for (j, col) in self.update_columns.iter().enumerate() {
3101 if j > 0 {
3102 f.write_str(", ")?;
3103 }
3104 f.write_str("e_ident(col))?;
3105 }
3106 }
3107 }
3108 TriggerEvent::Delete => f.write_str("DELETE")?,
3109 TriggerEvent::Truncate => f.write_str("TRUNCATE")?,
3110 }
3111 }
3112 write!(f, " ON {} FOR EACH ", quote_ident(&self.table))?;
3113 match self.for_each {
3114 TriggerForEach::Row => f.write_str("ROW")?,
3115 TriggerForEach::Statement => f.write_str("STATEMENT")?,
3116 }
3117 write!(f, " EXECUTE FUNCTION {}()", quote_ident(&self.function))
3118 }
3119}
3120
3121impl fmt::Display for CreateIndexStatement {
3122 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3123 if self.is_unique {
3124 f.write_str("CREATE UNIQUE INDEX ")?;
3125 } else {
3126 f.write_str("CREATE INDEX ")?;
3127 }
3128 if self.if_not_exists {
3129 f.write_str("IF NOT EXISTS ")?;
3130 }
3131 write!(
3132 f,
3133 "{} ON {} ",
3134 quote_ident(&self.name),
3135 quote_ident(&self.table)
3136 )?;
3137 match self.method {
3138 IndexMethod::Hnsw => f.write_str("USING hnsw ")?,
3139 IndexMethod::Brin => f.write_str("USING brin ")?,
3140 IndexMethod::Gin => f.write_str("USING gin ")?,
3141 IndexMethod::BTree => {}
3142 }
3143 if let Some(expr) = &self.expression {
3144 write!(f, "({})", expr)?;
3145 } else if self.extra_columns.is_empty() {
3146 // v7.15.0 — preserve operator class on round-trip
3147 // (`(col opclass)`) so WAL replay reconstructs the
3148 // engine-routing intent (e.g. `gin_trgm_ops` →
3149 // trigram-GIN build path).
3150 if let Some(op) = &self.opclass {
3151 write!(f, "({} {})", quote_ident(&self.column), op)?;
3152 } else {
3153 write!(f, "({})", quote_ident(&self.column))?;
3154 }
3155 } else {
3156 // v7.9.14 — multi-column key. Emit each column quoted
3157 // so the round-tripped form re-parses to identical AST.
3158 f.write_str("(")?;
3159 write!(f, "{}", quote_ident(&self.column))?;
3160 for c in &self.extra_columns {
3161 write!(f, ", {}", quote_ident(c))?;
3162 }
3163 f.write_str(")")?;
3164 }
3165 if !self.included_columns.is_empty() {
3166 f.write_str(" INCLUDE (")?;
3167 for (i, c) in self.included_columns.iter().enumerate() {
3168 if i > 0 {
3169 f.write_str(", ")?;
3170 }
3171 write!(f, "{}", quote_ident(c))?;
3172 }
3173 f.write_str(")")?;
3174 }
3175 if let Some(pred) = &self.partial_predicate {
3176 write!(f, " WHERE {}", pred)?;
3177 }
3178 Ok(())
3179 }
3180}
3181
3182impl fmt::Display for CreateTableStatement {
3183 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3184 f.write_str("CREATE TABLE ")?;
3185 if self.if_not_exists {
3186 f.write_str("IF NOT EXISTS ")?;
3187 }
3188 write!(f, "{} (", quote_ident(&self.name))?;
3189 for (i, col) in self.columns.iter().enumerate() {
3190 if i > 0 {
3191 f.write_str(", ")?;
3192 }
3193 write!(f, "{col}")?;
3194 }
3195 // v7.6.0 — render FK constraints in table-level form, after
3196 // the column list. WAL replay round-trips through Display, so
3197 // every FK must serialise here for replay to reconstruct the
3198 // schema bit-for-bit.
3199 for fk in &self.foreign_keys {
3200 f.write_str(", ")?;
3201 write!(f, "{fk}")?;
3202 }
3203 // v7.13.0 — render table-level constraints (PRIMARY KEY /
3204 // UNIQUE / CHECK) so WAL replay reconstructs them. Inline
3205 // column-level UNIQUE / CHECK get lifted to this list at
3206 // parse time, so emitting only here avoids double-counting.
3207 for tc in &self.table_constraints {
3208 f.write_str(", ")?;
3209 write!(f, "{tc}")?;
3210 }
3211 f.write_str(")")
3212 }
3213}
3214
3215fn fmt_alter_target(f: &mut fmt::Formatter<'_>, t: &AlterTableTarget) -> fmt::Result {
3216 match t {
3217 AlterTableTarget::SetHotTierBytes(n) => {
3218 write!(f, "SET hot_tier_bytes = {n}")
3219 }
3220 AlterTableTarget::AddForeignKey(fk) => write!(f, "ADD {fk}"),
3221 AlterTableTarget::DropForeignKey { name, if_exists } => {
3222 f.write_str("DROP CONSTRAINT ")?;
3223 if *if_exists {
3224 f.write_str("IF EXISTS ")?;
3225 }
3226 write!(f, "{}", quote_ident(name))
3227 }
3228 AlterTableTarget::AddColumn {
3229 column,
3230 if_not_exists,
3231 } => {
3232 f.write_str("ADD COLUMN ")?;
3233 if *if_not_exists {
3234 f.write_str("IF NOT EXISTS ")?;
3235 }
3236 write!(f, "{} {}", quote_ident(&column.name), column.ty)?;
3237 if !column.nullable {
3238 f.write_str(" NOT NULL")?;
3239 }
3240 if let Some(d) = &column.default {
3241 write!(f, " DEFAULT {d}")?;
3242 }
3243 if column.auto_increment {
3244 f.write_str(" AUTO_INCREMENT")?;
3245 }
3246 if column.is_primary_key {
3247 f.write_str(" PRIMARY KEY")?;
3248 }
3249 Ok(())
3250 }
3251 AlterTableTarget::AlterColumnType {
3252 column,
3253 new_type,
3254 using,
3255 } => {
3256 write!(f, "ALTER COLUMN {} TYPE {new_type}", quote_ident(column))?;
3257 if let Some(u) = using {
3258 write!(f, " USING {u}")?;
3259 }
3260 Ok(())
3261 }
3262 AlterTableTarget::DropColumn {
3263 column,
3264 if_exists,
3265 cascade,
3266 } => {
3267 f.write_str("DROP COLUMN ")?;
3268 if *if_exists {
3269 f.write_str("IF EXISTS ")?;
3270 }
3271 write!(f, "{}", quote_ident(column))?;
3272 if *cascade {
3273 f.write_str(" CASCADE")?;
3274 }
3275 Ok(())
3276 }
3277 AlterTableTarget::AddTableConstraint(tc) => {
3278 write!(f, "ADD {tc}")
3279 }
3280 AlterTableTarget::SetColumnAutoIncrement { column, seq_name } => {
3281 // Round-trip-safe spelling: re-parsing this form lowers
3282 // back to SetColumnAutoIncrement (the nextval default is
3283 // how pg_dump says "serial").
3284 let seq = seq_name
3285 .clone()
3286 .unwrap_or_else(|| alloc::format!("{column}_seq"));
3287 write!(
3288 f,
3289 "ALTER COLUMN {} SET DEFAULT nextval('{seq}')",
3290 quote_ident(column)
3291 )
3292 }
3293 AlterTableTarget::RenameColumn { old, new } => {
3294 write!(
3295 f,
3296 "RENAME COLUMN {} TO {}",
3297 quote_ident(old),
3298 quote_ident(new)
3299 )
3300 }
3301 AlterTableTarget::RenameTable { new } => {
3302 write!(f, "RENAME TO {}", quote_ident(new))
3303 }
3304 AlterTableTarget::SetTriggerEnabled { which, enabled } => {
3305 f.write_str(if *enabled {
3306 "ENABLE TRIGGER "
3307 } else {
3308 "DISABLE TRIGGER "
3309 })?;
3310 match which {
3311 TriggerSelector::All => f.write_str("ALL"),
3312 TriggerSelector::Named(n) => f.write_str("e_ident(n)),
3313 }
3314 }
3315 }
3316}
3317
3318impl fmt::Display for TableConstraint {
3319 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3320 match self {
3321 Self::PrimaryKey { name, columns } => {
3322 if let Some(n) = name {
3323 write!(f, "CONSTRAINT {} ", quote_ident(n))?;
3324 }
3325 f.write_str("PRIMARY KEY (")?;
3326 for (i, c) in columns.iter().enumerate() {
3327 if i > 0 {
3328 f.write_str(", ")?;
3329 }
3330 f.write_str("e_ident(c))?;
3331 }
3332 f.write_str(")")
3333 }
3334 Self::Unique {
3335 name,
3336 columns,
3337 nulls_not_distinct,
3338 } => {
3339 if let Some(n) = name {
3340 write!(f, "CONSTRAINT {} ", quote_ident(n))?;
3341 }
3342 f.write_str("UNIQUE ")?;
3343 if *nulls_not_distinct {
3344 f.write_str("NULLS NOT DISTINCT ")?;
3345 }
3346 f.write_str("(")?;
3347 for (i, c) in columns.iter().enumerate() {
3348 if i > 0 {
3349 f.write_str(", ")?;
3350 }
3351 f.write_str("e_ident(c))?;
3352 }
3353 f.write_str(")")
3354 }
3355 Self::Check { name, expr } => {
3356 if let Some(n) = name {
3357 write!(f, "CONSTRAINT {} ", quote_ident(n))?;
3358 }
3359 write!(f, "CHECK ({expr})")
3360 }
3361 Self::Index { name, columns } => {
3362 f.write_str("KEY ")?;
3363 if let Some(n) = name {
3364 write!(f, "{} ", quote_ident(n))?;
3365 }
3366 f.write_str("(")?;
3367 for (i, c) in columns.iter().enumerate() {
3368 if i > 0 {
3369 f.write_str(", ")?;
3370 }
3371 f.write_str("e_ident(c))?;
3372 }
3373 f.write_str(")")
3374 }
3375 Self::FulltextIndex { name, columns } => {
3376 // Mysqldump emits `FULLTEXT KEY name (cols)` —
3377 // Display rounds back to that shape so dump
3378 // replay reproduces the input verbatim.
3379 f.write_str("FULLTEXT KEY ")?;
3380 if let Some(n) = name {
3381 write!(f, "{} ", quote_ident(n))?;
3382 }
3383 f.write_str("(")?;
3384 for (i, c) in columns.iter().enumerate() {
3385 if i > 0 {
3386 f.write_str(", ")?;
3387 }
3388 f.write_str("e_ident(c))?;
3389 }
3390 f.write_str(")")
3391 }
3392 }
3393 }
3394}
3395
3396impl fmt::Display for ForeignKeyConstraint {
3397 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3398 if let Some(name) = &self.name {
3399 write!(f, "CONSTRAINT {} ", quote_ident(name))?;
3400 }
3401 f.write_str("FOREIGN KEY (")?;
3402 for (i, c) in self.columns.iter().enumerate() {
3403 if i > 0 {
3404 f.write_str(", ")?;
3405 }
3406 f.write_str("e_ident(c))?;
3407 }
3408 write!(f, ") REFERENCES {}", quote_ident(&self.parent_table))?;
3409 if !self.parent_columns.is_empty() {
3410 f.write_str(" (")?;
3411 for (i, c) in self.parent_columns.iter().enumerate() {
3412 if i > 0 {
3413 f.write_str(", ")?;
3414 }
3415 f.write_str("e_ident(c))?;
3416 }
3417 f.write_str(")")?;
3418 }
3419 // Only render non-default actions to keep Display output
3420 // close to user input. SPG's default is RESTRICT (matches
3421 // SQL spec).
3422 if self.on_delete != FkAction::Restrict {
3423 write!(f, " ON DELETE {}", self.on_delete)?;
3424 }
3425 if self.on_update != FkAction::Restrict {
3426 write!(f, " ON UPDATE {}", self.on_update)?;
3427 }
3428 Ok(())
3429 }
3430}
3431
3432impl fmt::Display for FkAction {
3433 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3434 match self {
3435 Self::Restrict => f.write_str("RESTRICT"),
3436 Self::Cascade => f.write_str("CASCADE"),
3437 Self::SetNull => f.write_str("SET NULL"),
3438 Self::SetDefault => f.write_str("SET DEFAULT"),
3439 Self::NoAction => f.write_str("NO ACTION"),
3440 }
3441 }
3442}
3443
3444impl fmt::Display for ColumnDef {
3445 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3446 write!(f, "{} {}", quote_ident(&self.name), self.ty)?;
3447 // v7.17.0 Phase 2.5 — render COLLATE for round-trippable
3448 // DDL. Only emits when non-default so the typical output
3449 // stays unchanged.
3450 match self.collation {
3451 Collation::Binary => {}
3452 Collation::CaseInsensitive => f.write_str(" COLLATE \"case_insensitive\"")?,
3453 }
3454 if let Some(d) = &self.default {
3455 write!(f, " DEFAULT {d}")?;
3456 }
3457 if self.auto_increment {
3458 f.write_str(" AUTO_INCREMENT")?;
3459 }
3460 if !self.nullable {
3461 f.write_str(" NOT NULL")?;
3462 }
3463 Ok(())
3464 }
3465}
3466
3467impl fmt::Display for InsertStatement {
3468 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3469 write!(f, "INSERT INTO {}", quote_ident(&self.table))?;
3470 if let Some(cols) = &self.columns {
3471 f.write_str(" (")?;
3472 for (i, c) in cols.iter().enumerate() {
3473 if i > 0 {
3474 f.write_str(", ")?;
3475 }
3476 f.write_str("e_ident(c))?;
3477 }
3478 f.write_str(")")?;
3479 }
3480 // v7.13.0 — INSERT…SELECT renders as `... SELECT …`,
3481 // skipping the VALUES list (mailrs round-5 G4).
3482 if let Some(sel) = &self.select_source {
3483 write!(f, " {sel}")?;
3484 } else {
3485 f.write_str(" VALUES ")?;
3486 for (ri, row) in self.rows.iter().enumerate() {
3487 if ri > 0 {
3488 f.write_str(", ")?;
3489 }
3490 f.write_str("(")?;
3491 for (i, v) in row.iter().enumerate() {
3492 if i > 0 {
3493 f.write_str(", ")?;
3494 }
3495 write!(f, "{v}")?;
3496 }
3497 f.write_str(")")?;
3498 }
3499 }
3500 Ok(())
3501 }
3502}
3503
3504impl fmt::Display for UpdateStatement {
3505 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3506 write!(f, "UPDATE {} SET ", quote_ident(&self.table))?;
3507 for (i, (col, expr)) in self.assignments.iter().enumerate() {
3508 if i > 0 {
3509 f.write_str(", ")?;
3510 }
3511 write!(f, "{} = {expr}", quote_ident(col))?;
3512 }
3513 if let Some(w) = &self.where_ {
3514 write!(f, " WHERE {w}")?;
3515 }
3516 Ok(())
3517 }
3518}
3519
3520impl fmt::Display for DeleteStatement {
3521 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3522 write!(f, "DELETE FROM {}", quote_ident(&self.table))?;
3523 if let Some(w) = &self.where_ {
3524 write!(f, " WHERE {w}")?;
3525 }
3526 Ok(())
3527 }
3528}
3529
3530impl fmt::Display for SelectStatement {
3531 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3532 write_bare_select(self, f)?;
3533 for (kind, peer) in &self.unions {
3534 f.write_str(match kind {
3535 UnionKind::Distinct => " UNION ",
3536 UnionKind::All => " UNION ALL ",
3537 })?;
3538 write_bare_select(peer, f)?;
3539 }
3540 if !self.order_by.is_empty() {
3541 f.write_str(" ORDER BY ")?;
3542 for (i, o) in self.order_by.iter().enumerate() {
3543 if i > 0 {
3544 f.write_str(", ")?;
3545 }
3546 write!(f, "{}", o.expr)?;
3547 if o.desc {
3548 f.write_str(" DESC")?;
3549 }
3550 }
3551 }
3552 if let Some(n) = &self.limit {
3553 write!(f, " LIMIT {n}")?;
3554 }
3555 if let Some(o) = &self.offset {
3556 write!(f, " OFFSET {o}")?;
3557 }
3558 Ok(())
3559 }
3560}
3561
3562fn write_bare_select(s: &SelectStatement, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3563 f.write_str("SELECT ")?;
3564 if s.distinct {
3565 f.write_str("DISTINCT ")?;
3566 }
3567 write_bare_select_body(s, f)
3568}
3569
3570fn write_bare_select_body(s: &SelectStatement, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3571 for (i, item) in s.items.iter().enumerate() {
3572 if i > 0 {
3573 f.write_str(", ")?;
3574 }
3575 write!(f, "{item}")?;
3576 }
3577 if let Some(t) = &s.from {
3578 write!(f, " FROM {t}")?;
3579 }
3580 if let Some(e) = &s.where_ {
3581 write!(f, " WHERE {e}")?;
3582 }
3583 if let Some(gs) = &s.group_by {
3584 f.write_str(" GROUP BY ")?;
3585 for (i, g) in gs.iter().enumerate() {
3586 if i > 0 {
3587 f.write_str(", ")?;
3588 }
3589 write!(f, "{g}")?;
3590 }
3591 }
3592 if let Some(h) = &s.having {
3593 write!(f, " HAVING {h}")?;
3594 }
3595 Ok(())
3596}
3597
3598impl fmt::Display for SelectItem {
3599 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3600 match self {
3601 Self::Wildcard => f.write_str("*"),
3602 Self::Expr { expr, alias } => {
3603 write!(f, "{expr}")?;
3604 if let Some(a) = alias {
3605 write!(f, " AS {}", quote_ident(a))?;
3606 }
3607 Ok(())
3608 }
3609 }
3610 }
3611}
3612
3613impl fmt::Display for FromClause {
3614 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3615 write!(f, "{}", self.primary)?;
3616 for j in &self.joins {
3617 match j.kind {
3618 JoinKind::Inner => write!(f, " INNER JOIN {}", j.table)?,
3619 JoinKind::Left => write!(f, " LEFT JOIN {}", j.table)?,
3620 JoinKind::Cross => write!(f, " CROSS JOIN {}", j.table)?,
3621 }
3622 if let Some(on) = &j.on {
3623 write!(f, " ON {on}")?;
3624 }
3625 }
3626 Ok(())
3627 }
3628}
3629
3630impl fmt::Display for TableRef {
3631 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3632 write!(f, "{}", quote_ident(&self.name))?;
3633 if let Some(a) = &self.alias {
3634 write!(f, " AS {}", quote_ident(a))?;
3635 }
3636 Ok(())
3637 }
3638}
3639
3640impl fmt::Display for ColumnName {
3641 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3642 if let Some(q) = &self.qualifier {
3643 write!(f, "{}.{}", quote_ident(q), quote_ident(&self.name))
3644 } else {
3645 write!(f, "{}", quote_ident(&self.name))
3646 }
3647 }
3648}
3649
3650impl fmt::Display for Expr {
3651 #[allow(clippy::too_many_lines)]
3652 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3653 match self {
3654 Self::Literal(l) => write!(f, "{l}"),
3655 Self::Column(c) => write!(f, "{c}"),
3656 Self::Placeholder(n) => write!(f, "${n}"),
3657 Self::Binary { lhs, op, rhs } => write!(f, "({lhs} {op} {rhs})"),
3658 Self::Unary { op, expr } => match op {
3659 UnOp::Not => write!(f, "(NOT {expr})"),
3660 UnOp::Neg => write!(f, "(-{expr})"),
3661 UnOp::BitNot => write!(f, "(~{expr})"),
3662 },
3663 Self::Cast { expr, target } => write!(f, "({expr}::{target})"),
3664 Self::IsNull { expr, negated } => {
3665 if *negated {
3666 write!(f, "({expr} IS NOT NULL)")
3667 } else {
3668 write!(f, "({expr} IS NULL)")
3669 }
3670 }
3671 Self::FunctionCall { name, args } => {
3672 write!(f, "{name}(")?;
3673 for (i, a) in args.iter().enumerate() {
3674 if i > 0 {
3675 f.write_str(", ")?;
3676 }
3677 write!(f, "{a}")?;
3678 }
3679 f.write_str(")")
3680 }
3681 Self::Like {
3682 expr,
3683 pattern,
3684 negated,
3685 } => {
3686 if *negated {
3687 write!(f, "({expr} NOT LIKE {pattern})")
3688 } else {
3689 write!(f, "({expr} LIKE {pattern})")
3690 }
3691 }
3692 Self::Extract { field, source } => write!(f, "EXTRACT({field} FROM {source})"),
3693 Self::WindowFunction {
3694 name,
3695 args,
3696 partition_by,
3697 order_by,
3698 frame,
3699 null_treatment: _,
3700 } => {
3701 write!(f, "{name}(")?;
3702 for (i, a) in args.iter().enumerate() {
3703 if i > 0 {
3704 f.write_str(", ")?;
3705 }
3706 write!(f, "{a}")?;
3707 }
3708 f.write_str(") OVER (")?;
3709 if !partition_by.is_empty() {
3710 f.write_str("PARTITION BY ")?;
3711 for (i, p) in partition_by.iter().enumerate() {
3712 if i > 0 {
3713 f.write_str(", ")?;
3714 }
3715 write!(f, "{p}")?;
3716 }
3717 }
3718 if !order_by.is_empty() {
3719 if !partition_by.is_empty() {
3720 f.write_str(" ")?;
3721 }
3722 f.write_str("ORDER BY ")?;
3723 for (i, (e, desc)) in order_by.iter().enumerate() {
3724 if i > 0 {
3725 f.write_str(", ")?;
3726 }
3727 write!(f, "{e}")?;
3728 if *desc {
3729 f.write_str(" DESC")?;
3730 }
3731 }
3732 }
3733 if let Some(fr) = frame {
3734 if !partition_by.is_empty() || !order_by.is_empty() {
3735 f.write_str(" ")?;
3736 }
3737 let k = match fr.kind {
3738 FrameKind::Rows => "ROWS",
3739 FrameKind::Range => "RANGE",
3740 };
3741 if let Some(end) = &fr.end {
3742 write!(f, "{k} BETWEEN {} AND {}", fr.start, end)?;
3743 } else {
3744 write!(f, "{k} {}", fr.start)?;
3745 }
3746 }
3747 f.write_str(")")
3748 }
3749 Self::ScalarSubquery(s) => write!(f, "({s})"),
3750 Self::Exists { subquery, negated } => {
3751 if *negated {
3752 write!(f, "NOT EXISTS ({subquery})")
3753 } else {
3754 write!(f, "EXISTS ({subquery})")
3755 }
3756 }
3757 Self::InSubquery {
3758 expr,
3759 subquery,
3760 negated,
3761 } => {
3762 if *negated {
3763 write!(f, "({expr} NOT IN ({subquery}))")
3764 } else {
3765 write!(f, "({expr} IN ({subquery}))")
3766 }
3767 }
3768 Self::Array(items) => {
3769 f.write_str("ARRAY[")?;
3770 for (i, e) in items.iter().enumerate() {
3771 if i > 0 {
3772 f.write_str(", ")?;
3773 }
3774 write!(f, "{e}")?;
3775 }
3776 f.write_str("]")
3777 }
3778 Self::ArraySubscript { target, index } => write!(f, "({target}[{index}])"),
3779 Self::AnyAll {
3780 expr,
3781 op,
3782 array,
3783 is_any,
3784 } => {
3785 let kw = if *is_any { "ANY" } else { "ALL" };
3786 write!(f, "({expr} {op} {kw}({array}))")
3787 }
3788 Self::Case {
3789 operand,
3790 branches,
3791 else_branch,
3792 } => {
3793 f.write_str("CASE")?;
3794 if let Some(op) = operand {
3795 write!(f, " {op}")?;
3796 }
3797 for (w, t) in branches {
3798 write!(f, " WHEN {w} THEN {t}")?;
3799 }
3800 if let Some(e) = else_branch {
3801 write!(f, " ELSE {e}")?;
3802 }
3803 f.write_str(" END")
3804 }
3805 }
3806 }
3807}
3808
3809impl fmt::Display for Literal {
3810 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3811 match self {
3812 Self::Integer(n) => write!(f, "{n}"),
3813 Self::Float(x) => {
3814 let s = format!("{x}");
3815 // Default Display for an integral f64 (e.g. 1.0) emits "1",
3816 // which would round-trip back to Integer. Force a dot.
3817 if s.contains('.') || s.contains('e') || s.contains('E') {
3818 f.write_str(&s)
3819 } else {
3820 write!(f, "{s}.0")
3821 }
3822 }
3823 Self::String(s) => {
3824 f.write_str("'")?;
3825 for c in s.chars() {
3826 if c == '\'' {
3827 f.write_str("''")?;
3828 } else {
3829 write!(f, "{c}")?;
3830 }
3831 }
3832 f.write_str("'")
3833 }
3834 Self::Bool(b) => f.write_str(if *b { "TRUE" } else { "FALSE" }),
3835 Self::Null => f.write_str("NULL"),
3836 // PG external array form. Display round-trip re-enters
3837 // through the column-typed text coerce, same as pgwire.
3838 Self::TextArray(items) => {
3839 f.write_str("'{")?;
3840 for (i, it) in items.iter().enumerate() {
3841 if i > 0 {
3842 f.write_str(",")?;
3843 }
3844 match it {
3845 None => f.write_str("NULL")?,
3846 Some(s) => {
3847 f.write_str("\"")?;
3848 for c in s.chars() {
3849 match c {
3850 // array-element escapes
3851 '"' | '\\' => write!(f, "\\{c}")?,
3852 // the OUTER wrapper is a SQL string
3853 // literal — embedded quotes must
3854 // double, or the rendered form
3855 // (WAL replay parses it back) is
3856 // invalid SQL
3857 '\'' => f.write_str("''")?,
3858 _ => write!(f, "{c}")?,
3859 }
3860 }
3861 f.write_str("\"")?;
3862 }
3863 }
3864 }
3865 f.write_str("}'")
3866 }
3867 Self::IntArray(items) => {
3868 f.write_str("'{")?;
3869 for (i, it) in items.iter().enumerate() {
3870 if i > 0 {
3871 f.write_str(",")?;
3872 }
3873 match it {
3874 None => f.write_str("NULL")?,
3875 Some(n) => write!(f, "{n}")?,
3876 }
3877 }
3878 f.write_str("}'")
3879 }
3880 Self::BigIntArray(items) => {
3881 f.write_str("'{")?;
3882 for (i, it) in items.iter().enumerate() {
3883 if i > 0 {
3884 f.write_str(",")?;
3885 }
3886 match it {
3887 None => f.write_str("NULL")?,
3888 Some(n) => write!(f, "{n}")?,
3889 }
3890 }
3891 f.write_str("}'")
3892 }
3893 Self::Vector(v) => {
3894 f.write_str("[")?;
3895 for (i, x) in v.iter().enumerate() {
3896 if i > 0 {
3897 f.write_str(", ")?;
3898 }
3899 let s = format!("{x}");
3900 // Mirror Float Display: force a dot so re-parse stays
3901 // numerically literal.
3902 if s.contains('.') || s.contains('e') || s.contains('E') {
3903 f.write_str(&s)?;
3904 } else {
3905 write!(f, "{s}.0")?;
3906 }
3907 }
3908 f.write_str("]")
3909 }
3910 Self::Interval { text, .. } => {
3911 f.write_str("INTERVAL '")?;
3912 for c in text.chars() {
3913 if c == '\'' {
3914 f.write_str("''")?;
3915 } else {
3916 write!(f, "{c}")?;
3917 }
3918 }
3919 f.write_str("'")
3920 }
3921 }
3922 }
3923}
3924
3925impl fmt::Display for BinOp {
3926 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
3927 f.write_str(match self {
3928 Self::Or => "OR",
3929 Self::And => "AND",
3930 Self::Eq => "=",
3931 Self::NotEq => "<>",
3932 Self::IsDistinctFrom => "IS DISTINCT FROM",
3933 Self::IsNotDistinctFrom => "IS NOT DISTINCT FROM",
3934 Self::Lt => "<",
3935 Self::LtEq => "<=",
3936 Self::Gt => ">",
3937 Self::GtEq => ">=",
3938 Self::Add => "+",
3939 Self::Sub => "-",
3940 Self::Mul => "*",
3941 Self::Div => "/",
3942 Self::L2Distance => "<->",
3943 Self::InnerProduct => "<#>",
3944 Self::CosineDistance => "<=>",
3945 Self::Concat => "||",
3946 Self::BitOr => "|",
3947 Self::BitAnd => "&",
3948 Self::JsonGet => "->",
3949 Self::JsonGetText => "->>",
3950 Self::JsonGetPath => "#>",
3951 Self::JsonGetPathText => "#>>",
3952 Self::JsonContains => "@>",
3953 Self::TsMatch => "@@",
3954 Self::InetContainedBy => "<<",
3955 Self::InetContainedByEq => "<<=",
3956 Self::InetContains => ">>",
3957 Self::InetContainsEq => ">>=",
3958 Self::InetOverlap => "&&",
3959 })
3960 }
3961}
3962
3963/// Quote `s` as a PG double-quoted identifier when required (keyword,
3964/// non-folded case, leading digit, embedded non-`[A-Za-z0-9_]`, empty).
3965/// Otherwise return it as-is. Returns an owned `String` to keep the call site
3966/// uniform.
3967fn quote_ident(s: &str) -> String {
3968 let needs_quote = match s.chars().next() {
3969 None => true,
3970 Some(c) if !c.is_ascii_alphabetic() && c != '_' => true,
3971 _ => {
3972 s.chars().any(|c| !(c.is_ascii_alphanumeric() || c == '_'))
3973 || s.chars().any(|c| c.is_ascii_uppercase())
3974 || is_keyword(s)
3975 }
3976 };
3977 if !needs_quote {
3978 return s.to_string();
3979 }
3980 let mut out = String::with_capacity(s.len() + 2);
3981 out.push('"');
3982 for c in s.chars() {
3983 if c == '"' {
3984 out.push_str("\"\"");
3985 } else {
3986 out.push(c);
3987 }
3988 }
3989 out.push('"');
3990 out
3991}
3992
3993fn is_keyword(s: &str) -> bool {
3994 matches!(
3995 &*s.to_ascii_lowercase(),
3996 "select"
3997 | "from"
3998 | "where"
3999 | "as"
4000 | "null"
4001 | "true"
4002 | "false"
4003 | "and"
4004 | "or"
4005 | "not"
4006 | "create"
4007 | "table"
4008 | "insert"
4009 | "into"
4010 | "values"
4011 | "index"
4012 | "on"
4013 | "begin"
4014 | "commit"
4015 | "rollback"
4016 | "is"
4017 | "between"
4018 | "in"
4019 | "like"
4020 | "group"
4021 | "distinct"
4022 | "union"
4023 | "all"
4024 | "join"
4025 | "inner"
4026 | "left"
4027 | "cross"
4028 | "outer"
4029 | "default"
4030 | "savepoint"
4031 | "release"
4032 | "to"
4033 | "having"
4034 | "show"
4035 | "extract"
4036 | "offset"
4037 | "asc"
4038 | "desc"
4039 | "interval"
4040 )
4041}
4042
4043#[cfg(test)]
4044mod tests {
4045 use super::*;
4046 use alloc::vec;
4047
4048 #[test]
4049 fn integer_literal_renders_without_dot() {
4050 assert_eq!(Literal::Integer(42).to_string(), "42");
4051 }
4052
4053 #[test]
4054 fn integral_float_keeps_dot() {
4055 assert_eq!(Literal::Float(1.0).to_string(), "1.0");
4056 assert_eq!(Literal::Float(1.5).to_string(), "1.5");
4057 assert_eq!(Literal::Float(2.5e-3).to_string(), "0.0025");
4058 }
4059
4060 #[test]
4061 fn string_literal_doubles_quote() {
4062 assert_eq!(Literal::String("it's".into()).to_string(), "'it''s'");
4063 }
4064
4065 #[test]
4066 fn bool_and_null_render_uppercase() {
4067 assert_eq!(Literal::Bool(true).to_string(), "TRUE");
4068 assert_eq!(Literal::Bool(false).to_string(), "FALSE");
4069 assert_eq!(Literal::Null.to_string(), "NULL");
4070 }
4071
4072 #[test]
4073 fn binary_op_always_parenthesised() {
4074 let e = Expr::Binary {
4075 lhs: Box::new(Expr::Literal(Literal::Integer(1))),
4076 op: BinOp::Add,
4077 rhs: Box::new(Expr::Literal(Literal::Integer(2))),
4078 };
4079 assert_eq!(e.to_string(), "(1 + 2)");
4080 }
4081
4082 #[test]
4083 fn select_star_from_table() {
4084 let s = SelectStatement {
4085 items: vec![SelectItem::Wildcard],
4086 from: Some(FromClause {
4087 primary: TableRef {
4088 name: "users".into(),
4089 alias: None,
4090 as_of_segment: None,
4091 unnest_expr: None,
4092 unnest_column_aliases: Vec::new(),
4093 generate_series_args: None,
4094 lateral_subquery: None,
4095 },
4096 joins: vec![],
4097 }),
4098 where_: None,
4099 group_by: None,
4100 group_by_all: false,
4101 having: None,
4102 unions: vec![],
4103 order_by: Vec::new(),
4104 limit: None,
4105 offset: None,
4106 limit_with_ties: false,
4107 distinct: false,
4108 ctes: vec![],
4109 };
4110 assert_eq!(s.to_string(), "SELECT * FROM users");
4111 }
4112
4113 #[test]
4114 fn quote_ident_for_uppercase_and_keyword() {
4115 assert_eq!(quote_ident("foo"), "foo");
4116 assert_eq!(quote_ident("Foo"), "\"Foo\"");
4117 assert_eq!(quote_ident("select"), "\"select\"");
4118 assert_eq!(quote_ident(""), "\"\"");
4119 assert_eq!(quote_ident("a\"b"), "\"a\"\"b\"");
4120 }
4121}