use std::collections::BTreeMap;
use std::fmt;
use enum_kinds::EnumKind;
use serde::{Deserialize, Serialize};
use smallvec::{smallvec, SmallVec};
use crate::ast::display::{self, AstDisplay, AstFormatter, WithOptionName};
use crate::ast::{
AstInfo, ColumnDef, ConnectionOption, ConnectionOptionName, CreateConnectionOption,
CreateConnectionType, CreateSinkConnection, CreateSourceConnection, CreateSourceOption,
CreateSourceOptionName, DeferredItemName, Expr, Format, FormatSpecifier, Ident, IntervalValue,
KeyConstraint, MaterializedViewOption, Query, SelectItem, SinkEnvelope, SourceEnvelope,
SourceIncludeMetadata, SubscribeOutput, TableAlias, TableConstraint, TableWithJoins,
UnresolvedDatabaseName, UnresolvedItemName, UnresolvedObjectName, UnresolvedSchemaName, Value,
};
#[allow(clippy::large_enum_variant)]
#[derive(Debug, Clone, PartialEq, Eq, Hash, EnumKind)]
#[enum_kind(StatementKind, derive(Serialize, Deserialize))]
pub enum Statement<T: AstInfo> {
Select(SelectStatement<T>),
Insert(InsertStatement<T>),
Copy(CopyStatement<T>),
Update(UpdateStatement<T>),
Delete(DeleteStatement<T>),
CreateConnection(CreateConnectionStatement<T>),
CreateDatabase(CreateDatabaseStatement),
CreateSchema(CreateSchemaStatement),
CreateWebhookSource(CreateWebhookSourceStatement<T>),
CreateSource(CreateSourceStatement<T>),
CreateSubsource(CreateSubsourceStatement<T>),
CreateSink(CreateSinkStatement<T>),
CreateView(CreateViewStatement<T>),
CreateMaterializedView(CreateMaterializedViewStatement<T>),
CreateTable(CreateTableStatement<T>),
CreateTableFromSource(CreateTableFromSourceStatement<T>),
CreateIndex(CreateIndexStatement<T>),
CreateType(CreateTypeStatement<T>),
CreateRole(CreateRoleStatement),
CreateCluster(CreateClusterStatement<T>),
CreateClusterReplica(CreateClusterReplicaStatement<T>),
CreateSecret(CreateSecretStatement<T>),
AlterCluster(AlterClusterStatement<T>),
AlterOwner(AlterOwnerStatement<T>),
AlterObjectRename(AlterObjectRenameStatement),
AlterObjectSwap(AlterObjectSwapStatement),
AlterRetainHistory(AlterRetainHistoryStatement<T>),
AlterIndex(AlterIndexStatement<T>),
AlterSecret(AlterSecretStatement<T>),
AlterSetCluster(AlterSetClusterStatement<T>),
AlterSink(AlterSinkStatement<T>),
AlterSource(AlterSourceStatement<T>),
AlterSystemSet(AlterSystemSetStatement),
AlterSystemReset(AlterSystemResetStatement),
AlterSystemResetAll(AlterSystemResetAllStatement),
AlterConnection(AlterConnectionStatement<T>),
AlterRole(AlterRoleStatement<T>),
AlterTableAddColumn(AlterTableAddColumnStatement<T>),
Discard(DiscardStatement),
DropObjects(DropObjectsStatement),
DropOwned(DropOwnedStatement<T>),
SetVariable(SetVariableStatement),
ResetVariable(ResetVariableStatement),
Show(ShowStatement<T>),
StartTransaction(StartTransactionStatement),
SetTransaction(SetTransactionStatement),
Commit(CommitStatement),
Rollback(RollbackStatement),
Subscribe(SubscribeStatement<T>),
ExplainPlan(ExplainPlanStatement<T>),
ExplainPushdown(ExplainPushdownStatement<T>),
ExplainTimestamp(ExplainTimestampStatement<T>),
ExplainSinkSchema(ExplainSinkSchemaStatement<T>),
Declare(DeclareStatement<T>),
Fetch(FetchStatement<T>),
Close(CloseStatement),
Prepare(PrepareStatement<T>),
Execute(ExecuteStatement<T>),
Deallocate(DeallocateStatement),
Raise(RaiseStatement),
GrantRole(GrantRoleStatement<T>),
RevokeRole(RevokeRoleStatement<T>),
GrantPrivileges(GrantPrivilegesStatement<T>),
RevokePrivileges(RevokePrivilegesStatement<T>),
AlterDefaultPrivileges(AlterDefaultPrivilegesStatement<T>),
ReassignOwned(ReassignOwnedStatement<T>),
ValidateConnection(ValidateConnectionStatement<T>),
Comment(CommentStatement<T>),
}
impl<T: AstInfo> AstDisplay for Statement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
Statement::Select(stmt) => f.write_node(stmt),
Statement::Insert(stmt) => f.write_node(stmt),
Statement::Copy(stmt) => f.write_node(stmt),
Statement::Update(stmt) => f.write_node(stmt),
Statement::Delete(stmt) => f.write_node(stmt),
Statement::CreateConnection(stmt) => f.write_node(stmt),
Statement::CreateDatabase(stmt) => f.write_node(stmt),
Statement::CreateSchema(stmt) => f.write_node(stmt),
Statement::CreateWebhookSource(stmt) => f.write_node(stmt),
Statement::CreateSource(stmt) => f.write_node(stmt),
Statement::CreateSubsource(stmt) => f.write_node(stmt),
Statement::CreateSink(stmt) => f.write_node(stmt),
Statement::CreateView(stmt) => f.write_node(stmt),
Statement::CreateMaterializedView(stmt) => f.write_node(stmt),
Statement::CreateTable(stmt) => f.write_node(stmt),
Statement::CreateTableFromSource(stmt) => f.write_node(stmt),
Statement::CreateIndex(stmt) => f.write_node(stmt),
Statement::CreateRole(stmt) => f.write_node(stmt),
Statement::CreateSecret(stmt) => f.write_node(stmt),
Statement::CreateType(stmt) => f.write_node(stmt),
Statement::CreateCluster(stmt) => f.write_node(stmt),
Statement::CreateClusterReplica(stmt) => f.write_node(stmt),
Statement::AlterCluster(stmt) => f.write_node(stmt),
Statement::AlterOwner(stmt) => f.write_node(stmt),
Statement::AlterObjectRename(stmt) => f.write_node(stmt),
Statement::AlterRetainHistory(stmt) => f.write_node(stmt),
Statement::AlterObjectSwap(stmt) => f.write_node(stmt),
Statement::AlterIndex(stmt) => f.write_node(stmt),
Statement::AlterSetCluster(stmt) => f.write_node(stmt),
Statement::AlterSecret(stmt) => f.write_node(stmt),
Statement::AlterSink(stmt) => f.write_node(stmt),
Statement::AlterSource(stmt) => f.write_node(stmt),
Statement::AlterSystemSet(stmt) => f.write_node(stmt),
Statement::AlterSystemReset(stmt) => f.write_node(stmt),
Statement::AlterSystemResetAll(stmt) => f.write_node(stmt),
Statement::AlterConnection(stmt) => f.write_node(stmt),
Statement::AlterRole(stmt) => f.write_node(stmt),
Statement::AlterTableAddColumn(stmt) => f.write_node(stmt),
Statement::Discard(stmt) => f.write_node(stmt),
Statement::DropObjects(stmt) => f.write_node(stmt),
Statement::DropOwned(stmt) => f.write_node(stmt),
Statement::SetVariable(stmt) => f.write_node(stmt),
Statement::ResetVariable(stmt) => f.write_node(stmt),
Statement::Show(stmt) => f.write_node(stmt),
Statement::StartTransaction(stmt) => f.write_node(stmt),
Statement::SetTransaction(stmt) => f.write_node(stmt),
Statement::Commit(stmt) => f.write_node(stmt),
Statement::Rollback(stmt) => f.write_node(stmt),
Statement::Subscribe(stmt) => f.write_node(stmt),
Statement::ExplainPlan(stmt) => f.write_node(stmt),
Statement::ExplainPushdown(stmt) => f.write_node(stmt),
Statement::ExplainTimestamp(stmt) => f.write_node(stmt),
Statement::ExplainSinkSchema(stmt) => f.write_node(stmt),
Statement::Declare(stmt) => f.write_node(stmt),
Statement::Close(stmt) => f.write_node(stmt),
Statement::Fetch(stmt) => f.write_node(stmt),
Statement::Prepare(stmt) => f.write_node(stmt),
Statement::Execute(stmt) => f.write_node(stmt),
Statement::Deallocate(stmt) => f.write_node(stmt),
Statement::Raise(stmt) => f.write_node(stmt),
Statement::GrantRole(stmt) => f.write_node(stmt),
Statement::RevokeRole(stmt) => f.write_node(stmt),
Statement::GrantPrivileges(stmt) => f.write_node(stmt),
Statement::RevokePrivileges(stmt) => f.write_node(stmt),
Statement::AlterDefaultPrivileges(stmt) => f.write_node(stmt),
Statement::ReassignOwned(stmt) => f.write_node(stmt),
Statement::ValidateConnection(stmt) => f.write_node(stmt),
Statement::Comment(stmt) => f.write_node(stmt),
}
}
}
impl_display_t!(Statement);
pub fn statement_kind_label_value(kind: StatementKind) -> &'static str {
match kind {
StatementKind::Select => "select",
StatementKind::Insert => "insert",
StatementKind::Copy => "copy",
StatementKind::Update => "update",
StatementKind::Delete => "delete",
StatementKind::CreateConnection => "create_connection",
StatementKind::CreateDatabase => "create_database",
StatementKind::CreateSchema => "create_schema",
StatementKind::CreateWebhookSource => "create_webhook",
StatementKind::CreateSource => "create_source",
StatementKind::CreateSubsource => "create_subsource",
StatementKind::CreateSink => "create_sink",
StatementKind::CreateView => "create_view",
StatementKind::CreateMaterializedView => "create_materialized_view",
StatementKind::CreateTable => "create_table",
StatementKind::CreateTableFromSource => "create_table_from_source",
StatementKind::CreateIndex => "create_index",
StatementKind::CreateType => "create_type",
StatementKind::CreateRole => "create_role",
StatementKind::CreateCluster => "create_cluster",
StatementKind::CreateClusterReplica => "create_cluster_replica",
StatementKind::CreateSecret => "create_secret",
StatementKind::AlterCluster => "alter_cluster",
StatementKind::AlterObjectRename => "alter_object_rename",
StatementKind::AlterRetainHistory => "alter_retain_history",
StatementKind::AlterObjectSwap => "alter_object_swap",
StatementKind::AlterIndex => "alter_index",
StatementKind::AlterRole => "alter_role",
StatementKind::AlterSecret => "alter_secret",
StatementKind::AlterSetCluster => "alter_set_cluster",
StatementKind::AlterSink => "alter_sink",
StatementKind::AlterSource => "alter_source",
StatementKind::AlterSystemSet => "alter_system_set",
StatementKind::AlterSystemReset => "alter_system_reset",
StatementKind::AlterSystemResetAll => "alter_system_reset_all",
StatementKind::AlterOwner => "alter_owner",
StatementKind::AlterConnection => "alter_connection",
StatementKind::AlterTableAddColumn => "alter_table",
StatementKind::Discard => "discard",
StatementKind::DropObjects => "drop_objects",
StatementKind::DropOwned => "drop_owned",
StatementKind::SetVariable => "set_variable",
StatementKind::ResetVariable => "reset_variable",
StatementKind::Show => "show",
StatementKind::StartTransaction => "start_transaction",
StatementKind::SetTransaction => "set_transaction",
StatementKind::Commit => "commit",
StatementKind::Rollback => "rollback",
StatementKind::Subscribe => "subscribe",
StatementKind::ExplainPlan => "explain_plan",
StatementKind::ExplainPushdown => "explain_pushdown",
StatementKind::ExplainTimestamp => "explain_timestamp",
StatementKind::ExplainSinkSchema => "explain_sink_schema",
StatementKind::Declare => "declare",
StatementKind::Fetch => "fetch",
StatementKind::Close => "close",
StatementKind::Prepare => "prepare",
StatementKind::Execute => "execute",
StatementKind::Deallocate => "deallocate",
StatementKind::Raise => "raise",
StatementKind::GrantRole => "grant_role",
StatementKind::RevokeRole => "revoke_role",
StatementKind::GrantPrivileges => "grant_privileges",
StatementKind::RevokePrivileges => "revoke_privileges",
StatementKind::AlterDefaultPrivileges => "alter_default_privileges",
StatementKind::ReassignOwned => "reassign_owned",
StatementKind::ValidateConnection => "validate_connection",
StatementKind::Comment => "comment",
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct SelectStatement<T: AstInfo> {
pub query: Query<T>,
pub as_of: Option<AsOf<T>>,
}
impl<T: AstInfo> AstDisplay for SelectStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_node(&self.query);
if let Some(as_of) = &self.as_of {
f.write_str(" ");
f.write_node(as_of);
}
}
}
impl_display_t!(SelectStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct InsertStatement<T: AstInfo> {
pub table_name: T::ItemName,
pub columns: Vec<Ident>,
pub source: InsertSource<T>,
pub returning: Vec<SelectItem<T>>,
}
impl<T: AstInfo> AstDisplay for InsertStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("INSERT INTO ");
f.write_node(&self.table_name);
if !self.columns.is_empty() {
f.write_str(" (");
f.write_node(&display::comma_separated(&self.columns));
f.write_str(")");
}
f.write_str(" ");
f.write_node(&self.source);
if !self.returning.is_empty() {
f.write_str(" RETURNING ");
f.write_node(&display::comma_separated(&self.returning));
}
}
}
impl_display_t!(InsertStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum CopyRelation<T: AstInfo> {
Named {
name: T::ItemName,
columns: Vec<Ident>,
},
Select(SelectStatement<T>),
Subscribe(SubscribeStatement<T>),
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum CopyDirection {
To,
From,
}
impl AstDisplay for CopyDirection {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
CopyDirection::To => "TO",
CopyDirection::From => "FROM",
})
}
}
impl_display!(CopyDirection);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum CopyTarget<T: AstInfo> {
Stdin,
Stdout,
Expr(Expr<T>),
}
impl<T: AstInfo> AstDisplay for CopyTarget<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
CopyTarget::Stdin => f.write_str("STDIN"),
CopyTarget::Stdout => f.write_str("STDOUT"),
CopyTarget::Expr(expr) => f.write_node(expr),
}
}
}
impl_display_t!(CopyTarget);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum CopyOptionName {
Format,
Delimiter,
Null,
Escape,
Quote,
Header,
AwsConnection,
MaxFileSize,
}
impl AstDisplay for CopyOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
CopyOptionName::Format => "FORMAT",
CopyOptionName::Delimiter => "DELIMITER",
CopyOptionName::Null => "NULL",
CopyOptionName::Escape => "ESCAPE",
CopyOptionName::Quote => "QUOTE",
CopyOptionName::Header => "HEADER",
CopyOptionName::AwsConnection => "AWS CONNECTION",
CopyOptionName::MaxFileSize => "MAX FILE SIZE",
})
}
}
impl WithOptionName for CopyOptionName {
fn redact_value(&self) -> bool {
match self {
CopyOptionName::Format
| CopyOptionName::Delimiter
| CopyOptionName::Null
| CopyOptionName::Escape
| CopyOptionName::Quote
| CopyOptionName::Header
| CopyOptionName::AwsConnection
| CopyOptionName::MaxFileSize => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct CopyOption<T: AstInfo> {
pub name: CopyOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(CopyOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CopyStatement<T: AstInfo> {
pub relation: CopyRelation<T>,
pub direction: CopyDirection,
pub target: CopyTarget<T>,
pub options: Vec<CopyOption<T>>,
}
impl<T: AstInfo> AstDisplay for CopyStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("COPY ");
match &self.relation {
CopyRelation::Named { name, columns } => {
f.write_node(name);
if !columns.is_empty() {
f.write_str("(");
f.write_node(&display::comma_separated(columns));
f.write_str(")");
}
}
CopyRelation::Select(query) => {
f.write_str("(");
f.write_node(query);
f.write_str(")");
}
CopyRelation::Subscribe(query) => {
f.write_str("(");
f.write_node(query);
f.write_str(")");
}
};
f.write_str(" ");
f.write_node(&self.direction);
f.write_str(" ");
f.write_node(&self.target);
if !self.options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.options));
f.write_str(")");
}
}
}
impl_display_t!(CopyStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct UpdateStatement<T: AstInfo> {
pub table_name: T::ItemName,
pub alias: Option<TableAlias>,
pub assignments: Vec<Assignment<T>>,
pub selection: Option<Expr<T>>,
}
impl<T: AstInfo> AstDisplay for UpdateStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("UPDATE ");
f.write_node(&self.table_name);
if let Some(alias) = &self.alias {
f.write_str(" AS ");
f.write_node(alias);
}
if !self.assignments.is_empty() {
f.write_str(" SET ");
f.write_node(&display::comma_separated(&self.assignments));
}
if let Some(selection) = &self.selection {
f.write_str(" WHERE ");
f.write_node(selection);
}
}
}
impl_display_t!(UpdateStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct DeleteStatement<T: AstInfo> {
pub table_name: T::ItemName,
pub alias: Option<TableAlias>,
pub using: Vec<TableWithJoins<T>>,
pub selection: Option<Expr<T>>,
}
impl<T: AstInfo> AstDisplay for DeleteStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("DELETE FROM ");
f.write_node(&self.table_name);
if let Some(alias) = &self.alias {
f.write_str(" AS ");
f.write_node(alias);
}
if !self.using.is_empty() {
f.write_str(" USING ");
f.write_node(&display::comma_separated(&self.using));
}
if let Some(selection) = &self.selection {
f.write_str(" WHERE ");
f.write_node(selection);
}
}
}
impl_display_t!(DeleteStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateDatabaseStatement {
pub name: UnresolvedDatabaseName,
pub if_not_exists: bool,
}
impl AstDisplay for CreateDatabaseStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE DATABASE ");
if self.if_not_exists {
f.write_str("IF NOT EXISTS ");
}
f.write_node(&self.name);
}
}
impl_display!(CreateDatabaseStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateSchemaStatement {
pub name: UnresolvedSchemaName,
pub if_not_exists: bool,
}
impl AstDisplay for CreateSchemaStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE SCHEMA ");
if self.if_not_exists {
f.write_str("IF NOT EXISTS ");
}
f.write_node(&self.name);
}
}
impl_display!(CreateSchemaStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ConnectionDefaultAwsPrivatelink<T: AstInfo> {
pub connection: T::ItemName,
pub port: Option<u16>,
}
impl<T: AstInfo> AstDisplay for ConnectionDefaultAwsPrivatelink<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_node(&self.connection);
if let Some(port) = self.port {
f.write_str(" (PORT ");
f.write_node(&display::escape_single_quote_string(&port.to_string()));
f.write_str(")");
}
}
}
impl_display_t!(ConnectionDefaultAwsPrivatelink);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct KafkaBroker<T: AstInfo> {
pub address: String,
pub tunnel: KafkaBrokerTunnel<T>,
}
impl<T: AstInfo> AstDisplay for KafkaBroker<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("'");
f.write_node(&display::escape_single_quote_string(&self.address));
f.write_str("'");
f.write_node(&self.tunnel);
}
}
impl_display_t!(KafkaBroker);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum KafkaBrokerTunnel<T: AstInfo> {
Direct,
AwsPrivatelink(KafkaBrokerAwsPrivatelink<T>),
SshTunnel(T::ItemName),
}
impl<T: AstInfo> AstDisplay for KafkaBrokerTunnel<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
use KafkaBrokerTunnel::*;
match self {
Direct => {}
AwsPrivatelink(aws) => {
f.write_str(" ");
f.write_node(aws);
}
Self::SshTunnel(connection) => {
f.write_str("USING SSH TUNNEL ");
f.write_node(connection);
}
}
}
}
impl_display_t!(KafkaBrokerTunnel);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum KafkaBrokerAwsPrivatelinkOptionName {
AvailabilityZone,
Port,
}
impl AstDisplay for KafkaBrokerAwsPrivatelinkOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
Self::AvailabilityZone => f.write_str("AVAILABILITY ZONE"),
Self::Port => f.write_str("PORT"),
}
}
}
impl_display!(KafkaBrokerAwsPrivatelinkOptionName);
impl WithOptionName for KafkaBrokerAwsPrivatelinkOptionName {
fn redact_value(&self) -> bool {
match self {
Self::AvailabilityZone | Self::Port => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct KafkaBrokerAwsPrivatelinkOption<T: AstInfo> {
pub name: KafkaBrokerAwsPrivatelinkOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(KafkaBrokerAwsPrivatelinkOption);
impl_display_t!(KafkaBrokerAwsPrivatelinkOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct KafkaBrokerAwsPrivatelink<T: AstInfo> {
pub connection: T::ItemName,
pub options: Vec<KafkaBrokerAwsPrivatelinkOption<T>>,
}
impl<T: AstInfo> AstDisplay for KafkaBrokerAwsPrivatelink<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("USING AWS PRIVATELINK ");
f.write_node(&self.connection);
if !self.options.is_empty() {
f.write_str(" (");
f.write_node(&display::comma_separated(&self.options));
f.write_str(")");
}
}
}
impl_display_t!(KafkaBrokerAwsPrivatelink);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateConnectionStatement<T: AstInfo> {
pub name: UnresolvedItemName,
pub connection_type: CreateConnectionType,
pub if_not_exists: bool,
pub values: Vec<ConnectionOption<T>>,
pub with_options: Vec<CreateConnectionOption<T>>,
}
impl<T: AstInfo> AstDisplay for CreateConnectionStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE CONNECTION ");
if self.if_not_exists {
f.write_str("IF NOT EXISTS ");
}
f.write_node(&self.name);
f.write_str(" TO ");
self.connection_type.fmt(f);
f.write_str(" (");
f.write_node(&display::comma_separated(&self.values));
f.write_str(")");
if !self.with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.with_options));
f.write_str(")");
}
}
}
impl_display_t!(CreateConnectionStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct ValidateConnectionStatement<T: AstInfo> {
pub name: T::ItemName,
}
impl<T: AstInfo> AstDisplay for ValidateConnectionStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("VALIDATE CONNECTION ");
f.write_node(&self.name);
}
}
impl_display_t!(ValidateConnectionStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateWebhookSourceStatement<T: AstInfo> {
pub name: UnresolvedItemName,
pub if_not_exists: bool,
pub body_format: Format<T>,
pub include_headers: CreateWebhookSourceIncludeHeaders,
pub validate_using: Option<CreateWebhookSourceCheck<T>>,
pub in_cluster: Option<T::ClusterName>,
}
impl<T: AstInfo> AstDisplay for CreateWebhookSourceStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE SOURCE ");
if self.if_not_exists {
f.write_str("IF NOT EXISTS ");
}
f.write_node(&self.name);
if let Some(cluster_name) = &self.in_cluster {
f.write_str(" IN CLUSTER ");
f.write_node(cluster_name);
}
f.write_str(" FROM WEBHOOK ");
f.write_str("BODY FORMAT ");
f.write_node(&self.body_format);
f.write_node(&self.include_headers);
if let Some(validate) = &self.validate_using {
f.write_str(" ");
f.write_node(validate);
}
}
}
impl_display_t!(CreateWebhookSourceStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateWebhookSourceCheck<T: AstInfo> {
pub options: Option<CreateWebhookSourceCheckOptions<T>>,
pub using: Expr<T>,
}
impl<T: AstInfo> AstDisplay for CreateWebhookSourceCheck<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CHECK (");
if let Some(options) = &self.options {
f.write_node(options);
f.write_str(" ");
}
f.write_node(&self.using);
f.write_str(")");
}
}
impl_display_t!(CreateWebhookSourceCheck);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateWebhookSourceCheckOptions<T: AstInfo> {
pub secrets: Vec<CreateWebhookSourceSecret<T>>,
pub headers: Vec<CreateWebhookSourceHeader>,
pub bodies: Vec<CreateWebhookSourceBody>,
}
impl<T: AstInfo> AstDisplay for CreateWebhookSourceCheckOptions<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("WITH (");
let mut delim = "";
if !self.headers.is_empty() {
f.write_node(&display::comma_separated(&self.headers[..]));
delim = ", ";
}
if !self.bodies.is_empty() {
f.write_str(delim);
f.write_node(&display::comma_separated(&self.bodies[..]));
delim = ", ";
}
if !self.secrets.is_empty() {
f.write_str(delim);
f.write_node(&display::comma_separated(&self.secrets[..]));
}
f.write_str(")");
}
}
impl_display_t!(CreateWebhookSourceCheckOptions);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateWebhookSourceSecret<T: AstInfo> {
pub secret: T::ItemName,
pub alias: Option<Ident>,
pub use_bytes: bool,
}
impl<T: AstInfo> AstDisplay for CreateWebhookSourceSecret<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SECRET ");
f.write_node(&self.secret);
if let Some(alias) = &self.alias {
f.write_str(" AS ");
f.write_node(alias);
}
if self.use_bytes {
f.write_str(" BYTES");
}
}
}
impl_display_t!(CreateWebhookSourceSecret);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateWebhookSourceHeader {
pub alias: Option<Ident>,
pub use_bytes: bool,
}
impl AstDisplay for CreateWebhookSourceHeader {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("HEADERS");
if let Some(alias) = &self.alias {
f.write_str(" AS ");
f.write_node(alias);
}
if self.use_bytes {
f.write_str(" BYTES");
}
}
}
impl_display!(CreateWebhookSourceHeader);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateWebhookSourceBody {
pub alias: Option<Ident>,
pub use_bytes: bool,
}
impl AstDisplay for CreateWebhookSourceBody {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("BODY");
if let Some(alias) = &self.alias {
f.write_str(" AS ");
f.write_node(alias);
}
if self.use_bytes {
f.write_str(" BYTES");
}
}
}
impl_display!(CreateWebhookSourceBody);
#[derive(Default, Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateWebhookSourceIncludeHeaders {
pub mappings: Vec<CreateWebhookSourceMapHeader>,
pub column: Option<Vec<CreateWebhookSourceFilterHeader>>,
}
impl AstDisplay for CreateWebhookSourceIncludeHeaders {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
if !self.mappings.is_empty() {
f.write_str(" ");
}
f.write_node(&display::separated(&self.mappings[..], " "));
if let Some(column) = &self.column {
f.write_str(" INCLUDE HEADERS");
if !column.is_empty() {
f.write_str(" ");
f.write_str("(");
f.write_node(&display::comma_separated(&column[..]));
f.write_str(")");
}
}
}
}
impl_display!(CreateWebhookSourceIncludeHeaders);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateWebhookSourceFilterHeader {
pub block: bool,
pub header_name: String,
}
impl AstDisplay for CreateWebhookSourceFilterHeader {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
if self.block {
f.write_str("NOT ");
}
f.write_node(&display::escaped_string_literal(&self.header_name));
}
}
impl_display!(CreateWebhookSourceFilterHeader);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateWebhookSourceMapHeader {
pub header_name: String,
pub column_name: Ident,
pub use_bytes: bool,
}
impl AstDisplay for CreateWebhookSourceMapHeader {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("INCLUDE HEADER ");
f.write_node(&display::escaped_string_literal(&self.header_name));
f.write_str(" AS ");
f.write_node(&self.column_name);
if self.use_bytes {
f.write_str(" BYTES");
}
}
}
impl_display!(CreateWebhookSourceMapHeader);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateSourceStatement<T: AstInfo> {
pub name: UnresolvedItemName,
pub in_cluster: Option<T::ClusterName>,
pub col_names: Vec<Ident>,
pub connection: CreateSourceConnection<T>,
pub include_metadata: Vec<SourceIncludeMetadata>,
pub format: Option<FormatSpecifier<T>>,
pub envelope: Option<SourceEnvelope>,
pub if_not_exists: bool,
pub key_constraint: Option<KeyConstraint>,
pub with_options: Vec<CreateSourceOption<T>>,
pub external_references: Option<ExternalReferences>,
pub progress_subsource: Option<DeferredItemName<T>>,
}
impl<T: AstInfo> AstDisplay for CreateSourceStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE SOURCE ");
if self.if_not_exists {
f.write_str("IF NOT EXISTS ");
}
f.write_node(&self.name);
if !self.col_names.is_empty() {
f.write_str(" (");
f.write_node(&display::comma_separated(&self.col_names));
if self.key_constraint.is_some() {
f.write_str(", ");
f.write_node(self.key_constraint.as_ref().unwrap());
}
f.write_str(")");
} else if self.key_constraint.is_some() {
f.write_str(" (");
f.write_node(self.key_constraint.as_ref().unwrap());
f.write_str(")")
}
if let Some(cluster) = &self.in_cluster {
f.write_str(" IN CLUSTER ");
f.write_node(cluster);
}
f.write_str(" FROM ");
f.write_node(&self.connection);
if let Some(format) = &self.format {
f.write_str(" ");
f.write_node(format);
}
if !self.include_metadata.is_empty() {
f.write_str(" INCLUDE ");
f.write_node(&display::comma_separated(&self.include_metadata));
}
if let Some(envelope) = &self.envelope {
f.write_str(" ENVELOPE ");
f.write_node(envelope);
}
if let Some(subsources) = &self.external_references {
f.write_str(" ");
f.write_node(subsources);
}
if let Some(progress) = &self.progress_subsource {
f.write_str(" EXPOSE PROGRESS AS ");
f.write_node(progress);
}
if !self.with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.with_options));
f.write_str(")");
}
}
}
impl_display_t!(CreateSourceStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ExternalReferenceExport {
pub reference: UnresolvedItemName,
pub alias: Option<UnresolvedItemName>,
}
impl AstDisplay for ExternalReferenceExport {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_node(&self.reference);
if let Some(alias) = &self.alias {
f.write_str(" AS ");
f.write_node(alias);
}
}
}
impl_display!(ExternalReferenceExport);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum ExternalReferences {
SubsetTables(Vec<ExternalReferenceExport>),
SubsetSchemas(Vec<Ident>),
All,
}
impl AstDisplay for ExternalReferences {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
Self::SubsetTables(tables) => {
f.write_str("FOR TABLES (");
f.write_node(&display::comma_separated(tables));
f.write_str(")");
}
Self::SubsetSchemas(schemas) => {
f.write_str("FOR SCHEMAS (");
f.write_node(&display::comma_separated(schemas));
f.write_str(")");
}
Self::All => f.write_str("FOR ALL TABLES"),
}
}
}
impl_display!(ExternalReferences);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum CreateSubsourceOptionName {
Progress,
ExternalReference,
TextColumns,
ExcludeColumns,
Details,
}
impl AstDisplay for CreateSubsourceOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
CreateSubsourceOptionName::Progress => "PROGRESS",
CreateSubsourceOptionName::ExternalReference => "EXTERNAL REFERENCE",
CreateSubsourceOptionName::TextColumns => "TEXT COLUMNS",
CreateSubsourceOptionName::ExcludeColumns => "EXCLUDE COLUMNS",
CreateSubsourceOptionName::Details => "DETAILS",
})
}
}
impl WithOptionName for CreateSubsourceOptionName {
fn redact_value(&self) -> bool {
match self {
CreateSubsourceOptionName::Progress
| CreateSubsourceOptionName::ExternalReference
| CreateSubsourceOptionName::Details
| CreateSubsourceOptionName::TextColumns
| CreateSubsourceOptionName::ExcludeColumns => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateSubsourceOption<T: AstInfo> {
pub name: CreateSubsourceOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(CreateSubsourceOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateSubsourceStatement<T: AstInfo> {
pub name: UnresolvedItemName,
pub columns: Vec<ColumnDef<T>>,
pub of_source: Option<T::ItemName>,
pub constraints: Vec<TableConstraint<T>>,
pub if_not_exists: bool,
pub with_options: Vec<CreateSubsourceOption<T>>,
}
impl<T: AstInfo> AstDisplay for CreateSubsourceStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE SUBSOURCE ");
if self.if_not_exists {
f.write_str("IF NOT EXISTS ");
}
f.write_node(&self.name);
f.write_str(" (");
f.write_node(&display::comma_separated(&self.columns));
if !self.constraints.is_empty() {
f.write_str(", ");
f.write_node(&display::comma_separated(&self.constraints));
}
f.write_str(")");
if let Some(of_source) = &self.of_source {
f.write_str(" OF SOURCE ");
f.write_node(of_source);
}
if !self.with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.with_options));
f.write_str(")");
}
}
}
impl_display_t!(CreateSubsourceStatement);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum CreateSinkOptionName {
Snapshot,
Version,
PartitionStrategy,
}
impl AstDisplay for CreateSinkOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
CreateSinkOptionName::Snapshot => {
f.write_str("SNAPSHOT");
}
CreateSinkOptionName::Version => {
f.write_str("VERSION");
}
CreateSinkOptionName::PartitionStrategy => {
f.write_str("PARTITION STRATEGY");
}
}
}
}
impl WithOptionName for CreateSinkOptionName {
fn redact_value(&self) -> bool {
match self {
CreateSinkOptionName::Snapshot => false,
CreateSinkOptionName::Version => false,
CreateSinkOptionName::PartitionStrategy => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct CreateSinkOption<T: AstInfo> {
pub name: CreateSinkOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(CreateSinkOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateSinkStatement<T: AstInfo> {
pub name: Option<UnresolvedItemName>,
pub in_cluster: Option<T::ClusterName>,
pub if_not_exists: bool,
pub from: T::ItemName,
pub connection: CreateSinkConnection<T>,
pub format: Option<FormatSpecifier<T>>,
pub envelope: Option<SinkEnvelope>,
pub with_options: Vec<CreateSinkOption<T>>,
}
impl<T: AstInfo> AstDisplay for CreateSinkStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE SINK ");
if self.if_not_exists {
f.write_str("IF NOT EXISTS ");
}
if let Some(name) = &self.name {
f.write_node(&name);
f.write_str(" ");
}
if let Some(cluster) = &self.in_cluster {
f.write_str("IN CLUSTER ");
f.write_node(cluster);
f.write_str(" ");
}
f.write_str("FROM ");
f.write_node(&self.from);
f.write_str(" INTO ");
f.write_node(&self.connection);
if let Some(format) = &self.format {
f.write_str(" ");
f.write_node(format);
}
if let Some(envelope) = &self.envelope {
f.write_str(" ENVELOPE ");
f.write_node(envelope);
}
if !self.with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.with_options));
f.write_str(")");
}
}
}
impl_display_t!(CreateSinkStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct ViewDefinition<T: AstInfo> {
pub name: UnresolvedItemName,
pub columns: Vec<Ident>,
pub query: Query<T>,
}
impl<T: AstInfo> AstDisplay for ViewDefinition<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_node(&self.name);
if !self.columns.is_empty() {
f.write_str(" (");
f.write_node(&display::comma_separated(&self.columns));
f.write_str(")");
}
f.write_str(" AS ");
f.write_node(&self.query);
}
}
impl_display_t!(ViewDefinition);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateViewStatement<T: AstInfo> {
pub if_exists: IfExistsBehavior,
pub temporary: bool,
pub definition: ViewDefinition<T>,
}
impl<T: AstInfo> AstDisplay for CreateViewStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE");
if self.if_exists == IfExistsBehavior::Replace {
f.write_str(" OR REPLACE");
}
if self.temporary {
f.write_str(" TEMPORARY");
}
f.write_str(" VIEW");
if self.if_exists == IfExistsBehavior::Skip {
f.write_str(" IF NOT EXISTS");
}
f.write_str(" ");
f.write_node(&self.definition);
}
}
impl_display_t!(CreateViewStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateMaterializedViewStatement<T: AstInfo> {
pub if_exists: IfExistsBehavior,
pub name: UnresolvedItemName,
pub columns: Vec<Ident>,
pub in_cluster: Option<T::ClusterName>,
pub query: Query<T>,
pub as_of: Option<u64>,
pub with_options: Vec<MaterializedViewOption<T>>,
}
impl<T: AstInfo> AstDisplay for CreateMaterializedViewStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE");
if self.if_exists == IfExistsBehavior::Replace {
f.write_str(" OR REPLACE");
}
f.write_str(" MATERIALIZED VIEW");
if self.if_exists == IfExistsBehavior::Skip {
f.write_str(" IF NOT EXISTS");
}
f.write_str(" ");
f.write_node(&self.name);
if !self.columns.is_empty() {
f.write_str(" (");
f.write_node(&display::comma_separated(&self.columns));
f.write_str(")");
}
if let Some(cluster) = &self.in_cluster {
f.write_str(" IN CLUSTER ");
f.write_node(cluster);
}
if !self.with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.with_options));
f.write_str(")");
}
f.write_str(" AS ");
f.write_node(&self.query);
if let Some(time) = &self.as_of {
f.write_str(" AS OF ");
f.write_str(time);
}
}
}
impl_display_t!(CreateMaterializedViewStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterSetClusterStatement<T: AstInfo> {
pub if_exists: bool,
pub name: UnresolvedItemName,
pub object_type: ObjectType,
pub set_cluster: T::ClusterName,
}
impl<T: AstInfo> AstDisplay for AlterSetClusterStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER ");
f.write_node(&self.object_type);
if self.if_exists {
f.write_str(" IF EXISTS");
}
f.write_str(" ");
f.write_node(&self.name);
f.write_str(" SET CLUSTER ");
f.write_node(&self.set_cluster);
}
}
impl_display_t!(AlterSetClusterStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateTableStatement<T: AstInfo> {
pub name: UnresolvedItemName,
pub columns: Vec<ColumnDef<T>>,
pub constraints: Vec<TableConstraint<T>>,
pub if_not_exists: bool,
pub temporary: bool,
pub with_options: Vec<TableOption<T>>,
}
impl<T: AstInfo> AstDisplay for CreateTableStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
let Self {
name,
columns,
constraints,
if_not_exists,
temporary,
with_options,
} = self;
f.write_str("CREATE ");
if *temporary {
f.write_str("TEMPORARY ");
}
f.write_str("TABLE ");
if *if_not_exists {
f.write_str("IF NOT EXISTS ");
}
f.write_node(name);
f.write_str(" (");
f.write_node(&display::comma_separated(columns));
if !self.constraints.is_empty() {
f.write_str(", ");
f.write_node(&display::comma_separated(constraints));
}
f.write_str(")");
if !with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.with_options));
f.write_str(")");
}
}
}
impl_display_t!(CreateTableStatement);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum TableOptionName {
RetainHistory,
RedactedTest,
}
impl AstDisplay for TableOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
TableOptionName::RetainHistory => {
f.write_str("RETAIN HISTORY");
}
TableOptionName::RedactedTest => {
f.write_str("REDACTED");
}
}
}
}
impl WithOptionName for TableOptionName {
fn redact_value(&self) -> bool {
match self {
TableOptionName::RetainHistory => false,
TableOptionName::RedactedTest => true,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct TableOption<T: AstInfo> {
pub name: TableOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(TableOption);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum TableFromSourceOptionName {
TextColumns,
ExcludeColumns,
Details,
}
impl AstDisplay for TableFromSourceOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
TableFromSourceOptionName::TextColumns => "TEXT COLUMNS",
TableFromSourceOptionName::ExcludeColumns => "EXCLUDE COLUMNS",
TableFromSourceOptionName::Details => "DETAILS",
})
}
}
impl_display!(TableFromSourceOptionName);
impl WithOptionName for TableFromSourceOptionName {
fn redact_value(&self) -> bool {
match self {
TableFromSourceOptionName::Details
| TableFromSourceOptionName::TextColumns
| TableFromSourceOptionName::ExcludeColumns => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct TableFromSourceOption<T: AstInfo> {
pub name: TableFromSourceOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(TableFromSourceOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateTableFromSourceStatement<T: AstInfo> {
pub name: UnresolvedItemName,
pub columns: Vec<ColumnDef<T>>,
pub constraints: Vec<TableConstraint<T>>,
pub if_not_exists: bool,
pub source: T::ItemName,
pub external_reference: Option<UnresolvedItemName>,
pub with_options: Vec<TableFromSourceOption<T>>,
pub include_metadata: Vec<SourceIncludeMetadata>,
pub format: Option<FormatSpecifier<T>>,
pub envelope: Option<SourceEnvelope>,
}
impl<T: AstInfo> AstDisplay for CreateTableFromSourceStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
let Self {
name,
columns,
constraints,
source,
external_reference,
if_not_exists,
with_options,
include_metadata,
format,
envelope,
} = self;
f.write_str("CREATE TABLE ");
if *if_not_exists {
f.write_str("IF NOT EXISTS ");
}
f.write_node(name);
if !columns.is_empty() || !constraints.is_empty() {
f.write_str(" (");
f.write_node(&display::comma_separated(columns));
if !constraints.is_empty() {
f.write_str(", ");
f.write_node(&display::comma_separated(constraints));
}
f.write_str(")");
}
f.write_str(" FROM SOURCE ");
f.write_node(source);
if let Some(external_reference) = external_reference {
f.write_str(" (REFERENCE = ");
f.write_node(external_reference);
f.write_str(")");
}
if let Some(format) = &format {
f.write_str(" ");
f.write_node(format);
}
if !include_metadata.is_empty() {
f.write_str(" INCLUDE ");
f.write_node(&display::comma_separated(include_metadata));
}
if let Some(envelope) = &envelope {
f.write_str(" ENVELOPE ");
f.write_node(envelope);
}
if !with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(with_options));
f.write_str(")");
}
}
}
impl_display_t!(CreateTableFromSourceStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateIndexStatement<T: AstInfo> {
pub name: Option<Ident>,
pub in_cluster: Option<T::ClusterName>,
pub on_name: T::ItemName,
pub key_parts: Option<Vec<Expr<T>>>,
pub with_options: Vec<IndexOption<T>>,
pub if_not_exists: bool,
}
impl<T: AstInfo> AstDisplay for CreateIndexStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE ");
if self.key_parts.is_none() {
f.write_str("DEFAULT ");
}
f.write_str("INDEX ");
if self.if_not_exists {
f.write_str("IF NOT EXISTS ");
}
if let Some(name) = &self.name {
f.write_node(name);
f.write_str(" ");
}
if let Some(cluster) = &self.in_cluster {
f.write_str("IN CLUSTER ");
f.write_node(cluster);
f.write_str(" ");
}
f.write_str("ON ");
f.write_node(&self.on_name);
if let Some(key_parts) = &self.key_parts {
f.write_str(" (");
f.write_node(&display::comma_separated(key_parts));
f.write_str(")");
}
if !self.with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.with_options));
f.write_str(")");
}
}
}
impl_display_t!(CreateIndexStatement);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum IndexOptionName {
RetainHistory,
}
impl AstDisplay for IndexOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
IndexOptionName::RetainHistory => {
f.write_str("RETAIN HISTORY");
}
}
}
}
impl WithOptionName for IndexOptionName {
fn redact_value(&self) -> bool {
match self {
IndexOptionName::RetainHistory => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct IndexOption<T: AstInfo> {
pub name: IndexOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(IndexOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateRoleStatement {
pub name: Ident,
pub options: Vec<RoleAttribute>,
}
impl AstDisplay for CreateRoleStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE ");
f.write_str("ROLE ");
f.write_node(&self.name);
for option in &self.options {
f.write_str(" ");
option.fmt(f)
}
}
}
impl_display!(CreateRoleStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum RoleAttribute {
Inherit,
NoInherit,
Login,
NoLogin,
SuperUser,
NoSuperUser,
CreateCluster,
NoCreateCluster,
CreateDB,
NoCreateDB,
CreateRole,
NoCreateRole,
}
impl AstDisplay for RoleAttribute {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
RoleAttribute::SuperUser => f.write_str("SUPERUSER"),
RoleAttribute::NoSuperUser => f.write_str("NOSUPERUSER"),
RoleAttribute::Login => f.write_str("LOGIN"),
RoleAttribute::NoLogin => f.write_str("NOLOGIN"),
RoleAttribute::Inherit => f.write_str("INHERIT"),
RoleAttribute::NoInherit => f.write_str("NOINHERIT"),
RoleAttribute::CreateCluster => f.write_str("CREATECLUSTER"),
RoleAttribute::NoCreateCluster => f.write_str("NOCREATECLUSTER"),
RoleAttribute::CreateDB => f.write_str("CREATEDB"),
RoleAttribute::NoCreateDB => f.write_str("NOCREATEDB"),
RoleAttribute::CreateRole => f.write_str("CREATEROLE"),
RoleAttribute::NoCreateRole => f.write_str("NOCREATEROLE"),
}
}
}
impl_display!(RoleAttribute);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum SetRoleVar {
Set { name: Ident, value: SetVariableTo },
Reset { name: Ident },
}
impl AstDisplay for SetRoleVar {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
SetRoleVar::Set { name, value } => {
f.write_str("SET ");
f.write_node(name);
f.write_str(" = ");
f.write_node(value);
}
SetRoleVar::Reset { name } => {
f.write_str("RESET ");
f.write_node(name);
}
}
}
}
impl_display!(SetRoleVar);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateSecretStatement<T: AstInfo> {
pub name: UnresolvedItemName,
pub if_not_exists: bool,
pub value: Expr<T>,
}
impl<T: AstInfo> AstDisplay for CreateSecretStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE SECRET ");
if self.if_not_exists {
f.write_str("IF NOT EXISTS ");
}
f.write_node(&self.name);
f.write_str(" AS ");
if f.redacted() {
f.write_str("'<REDACTED>'");
} else {
f.write_node(&self.value);
}
}
}
impl_display_t!(CreateSecretStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateTypeStatement<T: AstInfo> {
pub name: UnresolvedItemName,
pub as_type: CreateTypeAs<T>,
}
impl<T: AstInfo> AstDisplay for CreateTypeStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE TYPE ");
f.write_node(&self.name);
f.write_str(" AS ");
match &self.as_type {
CreateTypeAs::List { options } => {
f.write_str(&self.as_type);
f.write_str("(");
if !options.is_empty() {
f.write_node(&display::comma_separated(options));
}
f.write_str(")");
}
CreateTypeAs::Map { options } => {
f.write_str(&self.as_type);
f.write_str("(");
if !options.is_empty() {
f.write_node(&display::comma_separated(options));
}
f.write_str(")");
}
CreateTypeAs::Record { column_defs } => {
f.write_str("(");
if !column_defs.is_empty() {
f.write_node(&display::comma_separated(column_defs));
}
f.write_str(")");
}
};
}
}
impl_display_t!(CreateTypeStatement);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum ClusterOptionName {
AvailabilityZones,
Disk,
IntrospectionInterval,
IntrospectionDebugging,
Managed,
Replicas,
ReplicationFactor,
Size,
Schedule,
WorkloadClass,
}
impl AstDisplay for ClusterOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
ClusterOptionName::AvailabilityZones => f.write_str("AVAILABILITY ZONES"),
ClusterOptionName::Disk => f.write_str("DISK"),
ClusterOptionName::IntrospectionDebugging => f.write_str("INTROSPECTION DEBUGGING"),
ClusterOptionName::IntrospectionInterval => f.write_str("INTROSPECTION INTERVAL"),
ClusterOptionName::Managed => f.write_str("MANAGED"),
ClusterOptionName::Replicas => f.write_str("REPLICAS"),
ClusterOptionName::ReplicationFactor => f.write_str("REPLICATION FACTOR"),
ClusterOptionName::Size => f.write_str("SIZE"),
ClusterOptionName::Schedule => f.write_str("SCHEDULE"),
ClusterOptionName::WorkloadClass => f.write_str("WORKLOAD CLASS"),
}
}
}
impl WithOptionName for ClusterOptionName {
fn redact_value(&self) -> bool {
match self {
ClusterOptionName::AvailabilityZones
| ClusterOptionName::Disk
| ClusterOptionName::IntrospectionDebugging
| ClusterOptionName::IntrospectionInterval
| ClusterOptionName::Managed
| ClusterOptionName::Replicas
| ClusterOptionName::ReplicationFactor
| ClusterOptionName::Size
| ClusterOptionName::Schedule
| ClusterOptionName::WorkloadClass => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct ClusterOption<T: AstInfo> {
pub name: ClusterOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(ClusterOption);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum ClusterAlterUntilReadyOptionName {
Timeout,
OnTimeout,
}
impl AstDisplay for ClusterAlterUntilReadyOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
Self::Timeout => f.write_str("TIMEOUT"),
Self::OnTimeout => f.write_str("ON TIMEOUT"),
}
}
}
impl WithOptionName for ClusterAlterUntilReadyOptionName {
fn redact_value(&self) -> bool {
match self {
ClusterAlterUntilReadyOptionName::Timeout
| ClusterAlterUntilReadyOptionName::OnTimeout => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct ClusterAlterUntilReadyOption<T: AstInfo> {
pub name: ClusterAlterUntilReadyOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(ClusterAlterUntilReadyOption);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum ClusterAlterOptionName {
Wait,
}
impl AstDisplay for ClusterAlterOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
ClusterAlterOptionName::Wait => f.write_str("WAIT"),
}
}
}
impl WithOptionName for ClusterAlterOptionName {
fn redact_value(&self) -> bool {
match self {
ClusterAlterOptionName::Wait => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum ClusterAlterOptionValue<T: AstInfo> {
For(Value),
UntilReady(Vec<ClusterAlterUntilReadyOption<T>>),
}
impl<T: AstInfo> AstDisplay for ClusterAlterOptionValue<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
ClusterAlterOptionValue::For(duration) => {
f.write_str("FOR ");
f.write_node(duration);
}
ClusterAlterOptionValue::UntilReady(options) => {
f.write_str("UNTIL READY (");
f.write_node(&display::comma_separated(options));
f.write_str(")");
}
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct ClusterAlterOption<T: AstInfo> {
pub name: ClusterAlterOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(ClusterAlterOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum ClusterFeatureName {
ReoptimizeImportedViews,
EnableNewOuterJoinLowering,
EnableEagerDeltaJoins,
EnableVariadicLeftJoinLowering,
EnableLetrecFixpointAnalysis,
EnableOuterJoinNullFilter,
EnableValueWindowFunctionFusion,
EnableReduceUnnestListFusion,
}
impl WithOptionName for ClusterFeatureName {
fn redact_value(&self) -> bool {
match self {
Self::ReoptimizeImportedViews
| Self::EnableNewOuterJoinLowering
| Self::EnableEagerDeltaJoins
| Self::EnableVariadicLeftJoinLowering
| Self::EnableLetrecFixpointAnalysis
| Self::EnableOuterJoinNullFilter
| Self::EnableValueWindowFunctionFusion
| Self::EnableReduceUnnestListFusion => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ClusterFeature<T: AstInfo> {
pub name: ClusterFeatureName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(ClusterFeature);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateClusterStatement<T: AstInfo> {
pub name: Ident,
pub options: Vec<ClusterOption<T>>,
pub features: Vec<ClusterFeature<T>>,
}
impl<T: AstInfo> AstDisplay for CreateClusterStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE CLUSTER ");
f.write_node(&self.name);
if !self.options.is_empty() {
f.write_str(" (");
f.write_node(&display::comma_separated(&self.options));
f.write_str(")");
}
if !self.features.is_empty() {
f.write_str(" FEATURES (");
f.write_node(&display::comma_separated(&self.features));
f.write_str(")");
}
}
}
impl_display_t!(CreateClusterStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ReplicaDefinition<T: AstInfo> {
pub name: Ident,
pub options: Vec<ReplicaOption<T>>,
}
impl<T: AstInfo> AstDisplay for ReplicaDefinition<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_node(&self.name);
f.write_str(" (");
f.write_node(&display::comma_separated(&self.options));
f.write_str(")");
}
}
impl_display_t!(ReplicaDefinition);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum AlterClusterAction<T: AstInfo> {
SetOptions {
options: Vec<ClusterOption<T>>,
with_options: Vec<ClusterAlterOption<T>>,
},
ResetOptions(Vec<ClusterOptionName>),
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterClusterStatement<T: AstInfo> {
pub if_exists: bool,
pub name: Ident,
pub action: AlterClusterAction<T>,
}
impl<T: AstInfo> AstDisplay for AlterClusterStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER CLUSTER ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&self.name);
f.write_str(" ");
match &self.action {
AlterClusterAction::SetOptions {
options,
with_options,
} => {
f.write_str("SET (");
f.write_node(&display::comma_separated(options));
f.write_str(")");
if !with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(with_options));
f.write_str(")");
}
}
AlterClusterAction::ResetOptions(options) => {
f.write_str("RESET (");
f.write_node(&display::comma_separated(options));
f.write_str(")");
}
}
}
}
impl_display_t!(AlterClusterStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CreateClusterReplicaStatement<T: AstInfo> {
pub of_cluster: Ident,
pub definition: ReplicaDefinition<T>,
}
impl<T: AstInfo> AstDisplay for CreateClusterReplicaStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CREATE CLUSTER REPLICA ");
f.write_node(&self.of_cluster);
f.write_str(".");
f.write_node(&self.definition.name);
f.write_str(" (");
f.write_node(&display::comma_separated(&self.definition.options));
f.write_str(")");
}
}
impl_display_t!(CreateClusterReplicaStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum ReplicaOptionName {
BilledAs,
Size,
AvailabilityZone,
StorageAddresses,
StoragectlAddresses,
ComputectlAddresses,
ComputeAddresses,
Workers,
Internal,
IntrospectionInterval,
IntrospectionDebugging,
Disk,
}
impl AstDisplay for ReplicaOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
ReplicaOptionName::BilledAs => f.write_str("BILLED AS"),
ReplicaOptionName::Size => f.write_str("SIZE"),
ReplicaOptionName::AvailabilityZone => f.write_str("AVAILABILITY ZONE"),
ReplicaOptionName::StorageAddresses => f.write_str("STORAGE ADDRESSES"),
ReplicaOptionName::StoragectlAddresses => f.write_str("STORAGECTL ADDRESSES"),
ReplicaOptionName::ComputectlAddresses => f.write_str("COMPUTECTL ADDRESSES"),
ReplicaOptionName::ComputeAddresses => f.write_str("COMPUTE ADDRESSES"),
ReplicaOptionName::Workers => f.write_str("WORKERS"),
ReplicaOptionName::Internal => f.write_str("INTERNAL"),
ReplicaOptionName::IntrospectionInterval => f.write_str("INTROSPECTION INTERVAL"),
ReplicaOptionName::IntrospectionDebugging => f.write_str("INTROSPECTION DEBUGGING"),
ReplicaOptionName::Disk => f.write_str("DISK"),
}
}
}
impl WithOptionName for ReplicaOptionName {
fn redact_value(&self) -> bool {
match self {
ReplicaOptionName::BilledAs
| ReplicaOptionName::Size
| ReplicaOptionName::AvailabilityZone
| ReplicaOptionName::StorageAddresses
| ReplicaOptionName::StoragectlAddresses
| ReplicaOptionName::ComputectlAddresses
| ReplicaOptionName::ComputeAddresses
| ReplicaOptionName::Workers
| ReplicaOptionName::Internal
| ReplicaOptionName::IntrospectionInterval
| ReplicaOptionName::IntrospectionDebugging
| ReplicaOptionName::Disk => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ReplicaOption<T: AstInfo> {
pub name: ReplicaOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(ReplicaOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum CreateTypeAs<T: AstInfo> {
List {
options: Vec<CreateTypeListOption<T>>,
},
Map {
options: Vec<CreateTypeMapOption<T>>,
},
Record {
column_defs: Vec<ColumnDef<T>>,
},
}
impl<T: AstInfo> AstDisplay for CreateTypeAs<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
CreateTypeAs::List { .. } => f.write_str("LIST "),
CreateTypeAs::Map { .. } => f.write_str("MAP "),
CreateTypeAs::Record { .. } => f.write_str("RECORD "),
}
}
}
impl_display_t!(CreateTypeAs);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum CreateTypeListOptionName {
ElementType,
}
impl AstDisplay for CreateTypeListOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
CreateTypeListOptionName::ElementType => "ELEMENT TYPE",
})
}
}
impl WithOptionName for CreateTypeListOptionName {
fn redact_value(&self) -> bool {
match self {
CreateTypeListOptionName::ElementType => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct CreateTypeListOption<T: AstInfo> {
pub name: CreateTypeListOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(CreateTypeListOption);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum CreateTypeMapOptionName {
KeyType,
ValueType,
}
impl AstDisplay for CreateTypeMapOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
CreateTypeMapOptionName::KeyType => "KEY TYPE",
CreateTypeMapOptionName::ValueType => "VALUE TYPE",
})
}
}
impl WithOptionName for CreateTypeMapOptionName {
fn redact_value(&self) -> bool {
match self {
CreateTypeMapOptionName::KeyType | CreateTypeMapOptionName::ValueType => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct CreateTypeMapOption<T: AstInfo> {
pub name: CreateTypeMapOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(CreateTypeMapOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterOwnerStatement<T: AstInfo> {
pub object_type: ObjectType,
pub if_exists: bool,
pub name: UnresolvedObjectName,
pub new_owner: T::RoleName,
}
impl<T: AstInfo> AstDisplay for AlterOwnerStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER ");
f.write_node(&self.object_type);
f.write_str(" ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&self.name);
f.write_str(" OWNER TO ");
f.write_node(&self.new_owner);
}
}
impl_display_t!(AlterOwnerStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterObjectRenameStatement {
pub object_type: ObjectType,
pub if_exists: bool,
pub name: UnresolvedObjectName,
pub to_item_name: Ident,
}
impl AstDisplay for AlterObjectRenameStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER ");
f.write_node(&self.object_type);
f.write_str(" ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&self.name);
f.write_str(" RENAME TO ");
f.write_node(&self.to_item_name);
}
}
impl_display!(AlterObjectRenameStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterRetainHistoryStatement<T: AstInfo> {
pub object_type: ObjectType,
pub if_exists: bool,
pub name: UnresolvedObjectName,
pub history: Option<WithOptionValue<T>>,
}
impl<T: AstInfo> AstDisplay for AlterRetainHistoryStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER ");
f.write_node(&self.object_type);
f.write_str(" ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&self.name);
if let Some(history) = &self.history {
f.write_str(" SET (RETAIN HISTORY ");
f.write_node(history);
} else {
f.write_str(" RESET (RETAIN HISTORY");
}
f.write_str(")");
}
}
impl_display_t!(AlterRetainHistoryStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterObjectSwapStatement {
pub object_type: ObjectType,
pub name_a: UnresolvedObjectName,
pub name_b: Ident,
}
impl AstDisplay for AlterObjectSwapStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER ");
f.write_node(&self.object_type);
f.write_str(" ");
f.write_node(&self.name_a);
f.write_str(" SWAP WITH ");
f.write_node(&self.name_b);
}
}
impl_display!(AlterObjectSwapStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum AlterIndexAction<T: AstInfo> {
SetOptions(Vec<IndexOption<T>>),
ResetOptions(Vec<IndexOptionName>),
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterIndexStatement<T: AstInfo> {
pub index_name: UnresolvedItemName,
pub if_exists: bool,
pub action: AlterIndexAction<T>,
}
impl<T: AstInfo> AstDisplay for AlterIndexStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER INDEX ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&self.index_name);
f.write_str(" ");
match &self.action {
AlterIndexAction::SetOptions(options) => {
f.write_str("SET (");
f.write_node(&display::comma_separated(options));
f.write_str(")");
}
AlterIndexAction::ResetOptions(options) => {
f.write_str("RESET (");
f.write_node(&display::comma_separated(options));
f.write_str(")");
}
}
}
}
impl_display_t!(AlterIndexStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum AlterSinkAction<T: AstInfo> {
SetOptions(Vec<CreateSinkOption<T>>),
ResetOptions(Vec<CreateSinkOptionName>),
ChangeRelation(T::ItemName),
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterSinkStatement<T: AstInfo> {
pub sink_name: UnresolvedItemName,
pub if_exists: bool,
pub action: AlterSinkAction<T>,
}
impl<T: AstInfo> AstDisplay for AlterSinkStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER SINK ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&self.sink_name);
f.write_str(" ");
match &self.action {
AlterSinkAction::ChangeRelation(from) => {
f.write_str("SET FROM ");
f.write_node(from);
}
AlterSinkAction::SetOptions(options) => {
f.write_str("SET (");
f.write_node(&display::comma_separated(options));
f.write_str(")");
}
AlterSinkAction::ResetOptions(options) => {
f.write_str("RESET (");
f.write_node(&display::comma_separated(options));
f.write_str(")");
}
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum AlterSourceAddSubsourceOptionName {
TextColumns,
ExcludeColumns,
Details,
}
impl AstDisplay for AlterSourceAddSubsourceOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
AlterSourceAddSubsourceOptionName::TextColumns => "TEXT COLUMNS",
AlterSourceAddSubsourceOptionName::ExcludeColumns => "EXCLUDE COLUMNS",
AlterSourceAddSubsourceOptionName::Details => "DETAILS",
})
}
}
impl_display!(AlterSourceAddSubsourceOptionName);
impl WithOptionName for AlterSourceAddSubsourceOptionName {
fn redact_value(&self) -> bool {
match self {
AlterSourceAddSubsourceOptionName::Details
| AlterSourceAddSubsourceOptionName::TextColumns
| AlterSourceAddSubsourceOptionName::ExcludeColumns => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct AlterSourceAddSubsourceOption<T: AstInfo> {
pub name: AlterSourceAddSubsourceOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(AlterSourceAddSubsourceOption);
impl_display_t!(AlterSourceAddSubsourceOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum AlterSourceAction<T: AstInfo> {
SetOptions(Vec<CreateSourceOption<T>>),
ResetOptions(Vec<CreateSourceOptionName>),
AddSubsources {
external_references: Vec<ExternalReferenceExport>,
options: Vec<AlterSourceAddSubsourceOption<T>>,
},
DropSubsources {
if_exists: bool,
cascade: bool,
names: Vec<UnresolvedItemName>,
},
}
impl<T: AstInfo> AstDisplay for AlterSourceAction<T> {
fn fmt<W>(&self, f: &mut AstFormatter<W>)
where
W: fmt::Write,
{
match &self {
AlterSourceAction::SetOptions(options) => {
f.write_str("SET (");
f.write_node(&display::comma_separated(options));
f.write_str(")");
}
AlterSourceAction::ResetOptions(options) => {
f.write_str("RESET (");
f.write_node(&display::comma_separated(options));
f.write_str(")");
}
AlterSourceAction::DropSubsources {
if_exists,
cascade,
names,
} => {
f.write_str("DROP SUBSOURCE ");
if *if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&display::comma_separated(names));
if *cascade {
f.write_str(" CASCADE");
}
}
AlterSourceAction::AddSubsources {
external_references: subsources,
options,
} => {
f.write_str("ADD SUBSOURCE ");
f.write_node(&display::comma_separated(subsources));
if !options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(options));
f.write_str(")");
}
}
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterSourceStatement<T: AstInfo> {
pub source_name: UnresolvedItemName,
pub if_exists: bool,
pub action: AlterSourceAction<T>,
}
impl<T: AstInfo> AstDisplay for AlterSourceStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER SOURCE ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&self.source_name);
f.write_str(" ");
f.write_node(&self.action)
}
}
impl_display_t!(AlterSourceStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterSecretStatement<T: AstInfo> {
pub name: UnresolvedItemName,
pub if_exists: bool,
pub value: Expr<T>,
}
impl<T: AstInfo> AstDisplay for AlterSecretStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER SECRET ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&self.name);
f.write_str(" AS ");
if f.redacted() {
f.write_str("'<REDACTED>'");
} else {
f.write_node(&self.value);
}
}
}
impl_display_t!(AlterSecretStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum AlterConnectionAction<T: AstInfo> {
RotateKeys,
SetOption(ConnectionOption<T>),
DropOption(ConnectionOptionName),
}
impl<T: AstInfo> AstDisplay for AlterConnectionAction<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
AlterConnectionAction::RotateKeys => f.write_str("ROTATE KEYS"),
AlterConnectionAction::SetOption(option) => {
f.write_str("SET (");
f.write_node(option);
f.write_str(")");
}
AlterConnectionAction::DropOption(option) => {
f.write_str("DROP (");
f.write_node(option);
f.write_str(")");
}
}
}
}
impl_display_t!(AlterConnectionAction);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum AlterConnectionOptionName {
Validate,
}
impl AstDisplay for AlterConnectionOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
AlterConnectionOptionName::Validate => "VALIDATE",
})
}
}
impl_display!(AlterConnectionOptionName);
impl WithOptionName for AlterConnectionOptionName {
fn redact_value(&self) -> bool {
match self {
AlterConnectionOptionName::Validate => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct AlterConnectionOption<T: AstInfo> {
pub name: AlterConnectionOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(AlterConnectionOption);
impl_display_t!(AlterConnectionOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterConnectionStatement<T: AstInfo> {
pub name: UnresolvedItemName,
pub if_exists: bool,
pub actions: Vec<AlterConnectionAction<T>>,
pub with_options: Vec<AlterConnectionOption<T>>,
}
impl<T: AstInfo> AstDisplay for AlterConnectionStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER CONNECTION ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&self.name);
f.write_str(" ");
f.write_node(&display::comma_separated(&self.actions));
if !self.with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.with_options));
f.write_str(")");
}
}
}
impl_display_t!(AlterConnectionStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterRoleStatement<T: AstInfo> {
pub name: T::RoleName,
pub option: AlterRoleOption,
}
impl<T: AstInfo> AstDisplay for AlterRoleStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER ROLE ");
f.write_node(&self.name);
f.write_node(&self.option);
}
}
impl_display_t!(AlterRoleStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum AlterRoleOption {
Attributes(Vec<RoleAttribute>),
Variable(SetRoleVar),
}
impl AstDisplay for AlterRoleOption {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
AlterRoleOption::Attributes(attrs) => {
for attr in attrs {
f.write_str(" ");
attr.fmt(f)
}
}
AlterRoleOption::Variable(var) => {
f.write_str(" ");
f.write_node(var);
}
}
}
}
impl_display!(AlterRoleOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterTableAddColumnStatement<T: AstInfo> {
pub if_exists: bool,
pub name: UnresolvedItemName,
pub if_col_not_exist: bool,
pub column_name: Ident,
pub data_type: T::DataType,
}
impl<T: AstInfo> AstDisplay for AlterTableAddColumnStatement<T> {
fn fmt<W>(&self, f: &mut AstFormatter<W>)
where
W: fmt::Write,
{
f.write_str("ALTER TABLE ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&self.name);
f.write_str(" ADD COLUMN ");
if self.if_col_not_exist {
f.write_str("IF NOT EXISTS ");
}
f.write_node(&self.column_name);
f.write_str(" ");
f.write_node(&self.data_type);
}
}
impl_display_t!(AlterTableAddColumnStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct DiscardStatement {
pub target: DiscardTarget,
}
impl AstDisplay for DiscardStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("DISCARD ");
f.write_node(&self.target);
}
}
impl_display!(DiscardStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum DiscardTarget {
Plans,
Sequences,
Temp,
All,
}
impl AstDisplay for DiscardTarget {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
DiscardTarget::Plans => f.write_str("PLANS"),
DiscardTarget::Sequences => f.write_str("SEQUENCES"),
DiscardTarget::Temp => f.write_str("TEMP"),
DiscardTarget::All => f.write_str("ALL"),
}
}
}
impl_display!(DiscardTarget);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct DropObjectsStatement {
pub object_type: ObjectType,
pub if_exists: bool,
pub names: Vec<UnresolvedObjectName>,
pub cascade: bool,
}
impl AstDisplay for DropObjectsStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("DROP ");
f.write_node(&self.object_type);
f.write_str(" ");
if self.if_exists {
f.write_str("IF EXISTS ");
}
f.write_node(&display::comma_separated(&self.names));
if self.cascade && self.object_type != ObjectType::Database {
f.write_str(" CASCADE");
} else if !self.cascade && self.object_type == ObjectType::Database {
f.write_str(" RESTRICT");
}
}
}
impl_display!(DropObjectsStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct DropOwnedStatement<T: AstInfo> {
pub role_names: Vec<T::RoleName>,
pub cascade: Option<bool>,
}
impl<T: AstInfo> DropOwnedStatement<T> {
pub fn cascade(&self) -> bool {
self.cascade == Some(true)
}
}
impl<T: AstInfo> AstDisplay for DropOwnedStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("DROP OWNED BY ");
f.write_node(&display::comma_separated(&self.role_names));
if let Some(true) = self.cascade {
f.write_str(" CASCADE");
} else if let Some(false) = self.cascade {
f.write_str(" RESTRICT");
}
}
}
impl_display_t!(DropOwnedStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct QualifiedReplica {
pub cluster: Ident,
pub replica: Ident,
}
impl AstDisplay for QualifiedReplica {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_node(&self.cluster);
f.write_str(".");
f.write_node(&self.replica);
}
}
impl_display!(QualifiedReplica);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct SetVariableStatement {
pub local: bool,
pub variable: Ident,
pub to: SetVariableTo,
}
impl AstDisplay for SetVariableStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SET ");
if self.local {
f.write_str("LOCAL ");
}
f.write_node(&self.variable);
f.write_str(" = ");
f.write_node(&self.to);
}
}
impl_display!(SetVariableStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct ResetVariableStatement {
pub variable: Ident,
}
impl AstDisplay for ResetVariableStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("RESET ");
f.write_node(&self.variable);
}
}
impl_display!(ResetVariableStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowVariableStatement {
pub variable: Ident,
}
impl AstDisplay for ShowVariableStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW ");
f.write_node(&self.variable);
}
}
impl_display!(ShowVariableStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct InspectShardStatement {
pub id: String,
}
impl AstDisplay for InspectShardStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("INSPECT SHARD ");
f.write_str("'");
f.write_node(&display::escape_single_quote_string(&self.id));
f.write_str("'");
}
}
impl_display!(InspectShardStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum ShowObjectType<T: AstInfo> {
MaterializedView {
in_cluster: Option<T::ClusterName>,
},
Index {
in_cluster: Option<T::ClusterName>,
on_object: Option<T::ItemName>,
},
Table,
View,
Source {
in_cluster: Option<T::ClusterName>,
},
Sink {
in_cluster: Option<T::ClusterName>,
},
Type,
Role,
Cluster,
ClusterReplica,
Object,
Secret,
Connection,
Database,
Schema {
from: Option<T::DatabaseName>,
},
Subsource {
on_source: Option<T::ItemName>,
},
Privileges {
object_type: Option<SystemObjectType>,
role: Option<T::RoleName>,
},
DefaultPrivileges {
object_type: Option<ObjectType>,
role: Option<T::RoleName>,
},
RoleMembership {
role: Option<T::RoleName>,
},
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowObjectsStatement<T: AstInfo> {
pub object_type: ShowObjectType<T>,
pub from: Option<T::SchemaName>,
pub filter: Option<ShowStatementFilter<T>>,
}
impl<T: AstInfo> AstDisplay for ShowObjectsStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW");
f.write_str(" ");
f.write_str(match &self.object_type {
ShowObjectType::Table => "TABLES",
ShowObjectType::View => "VIEWS",
ShowObjectType::Source { .. } => "SOURCES",
ShowObjectType::Sink { .. } => "SINKS",
ShowObjectType::Type => "TYPES",
ShowObjectType::Role => "ROLES",
ShowObjectType::Cluster => "CLUSTERS",
ShowObjectType::ClusterReplica => "CLUSTER REPLICAS",
ShowObjectType::Object => "OBJECTS",
ShowObjectType::Secret => "SECRETS",
ShowObjectType::Connection => "CONNECTIONS",
ShowObjectType::MaterializedView { .. } => "MATERIALIZED VIEWS",
ShowObjectType::Index { .. } => "INDEXES",
ShowObjectType::Database => "DATABASES",
ShowObjectType::Schema { .. } => "SCHEMAS",
ShowObjectType::Subsource { .. } => "SUBSOURCES",
ShowObjectType::Privileges { .. } => "PRIVILEGES",
ShowObjectType::DefaultPrivileges { .. } => "DEFAULT PRIVILEGES",
ShowObjectType::RoleMembership { .. } => "ROLE MEMBERSHIP",
});
if let ShowObjectType::Index { on_object, .. } = &self.object_type {
if let Some(on_object) = on_object {
f.write_str(" ON ");
f.write_node(on_object);
}
}
if let ShowObjectType::Schema { from: Some(from) } = &self.object_type {
f.write_str(" FROM ");
f.write_node(from);
}
if let Some(from) = &self.from {
f.write_str(" FROM ");
f.write_node(from);
}
match &self.object_type {
ShowObjectType::MaterializedView { in_cluster }
| ShowObjectType::Index { in_cluster, .. }
| ShowObjectType::Sink { in_cluster }
| ShowObjectType::Source { in_cluster } => {
if let Some(cluster) = in_cluster {
f.write_str(" IN CLUSTER ");
f.write_node(cluster);
}
}
_ => (),
}
if let ShowObjectType::Subsource { on_source } = &self.object_type {
if let Some(on_source) = on_source {
f.write_str(" ON ");
f.write_node(on_source);
}
}
if let ShowObjectType::Privileges { object_type, role } = &self.object_type {
if let Some(object_type) = object_type {
f.write_str(" ON ");
f.write_node(object_type);
if let SystemObjectType::Object(_) = object_type {
f.write_str("S");
}
}
if let Some(role) = role {
f.write_str(" FOR ");
f.write_node(role);
}
}
if let ShowObjectType::DefaultPrivileges { object_type, role } = &self.object_type {
if let Some(object_type) = object_type {
f.write_str(" ON ");
f.write_node(object_type);
f.write_str("S");
}
if let Some(role) = role {
f.write_str(" FOR ");
f.write_node(role);
}
}
if let ShowObjectType::RoleMembership {
role: Some(role), ..
} = &self.object_type
{
f.write_str(" FOR ");
f.write_node(role);
}
if let Some(filter) = &self.filter {
f.write_str(" ");
f.write_node(filter);
}
}
}
impl_display_t!(ShowObjectsStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowColumnsStatement<T: AstInfo> {
pub table_name: T::ItemName,
pub filter: Option<ShowStatementFilter<T>>,
}
impl<T: AstInfo> AstDisplay for ShowColumnsStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW ");
f.write_str("COLUMNS FROM ");
f.write_node(&self.table_name);
if let Some(filter) = &self.filter {
f.write_str(" ");
f.write_node(filter);
}
}
}
impl_display_t!(ShowColumnsStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowCreateViewStatement<T: AstInfo> {
pub view_name: T::ItemName,
}
impl<T: AstInfo> AstDisplay for ShowCreateViewStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW CREATE VIEW ");
f.write_node(&self.view_name);
}
}
impl_display_t!(ShowCreateViewStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowCreateMaterializedViewStatement<T: AstInfo> {
pub materialized_view_name: T::ItemName,
}
impl<T: AstInfo> AstDisplay for ShowCreateMaterializedViewStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW CREATE MATERIALIZED VIEW ");
f.write_node(&self.materialized_view_name);
}
}
impl_display_t!(ShowCreateMaterializedViewStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowCreateSourceStatement<T: AstInfo> {
pub source_name: T::ItemName,
}
impl<T: AstInfo> AstDisplay for ShowCreateSourceStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW CREATE SOURCE ");
f.write_node(&self.source_name);
}
}
impl_display_t!(ShowCreateSourceStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowCreateTableStatement<T: AstInfo> {
pub table_name: T::ItemName,
}
impl<T: AstInfo> AstDisplay for ShowCreateTableStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW CREATE TABLE ");
f.write_node(&self.table_name);
}
}
impl_display_t!(ShowCreateTableStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowCreateSinkStatement<T: AstInfo> {
pub sink_name: T::ItemName,
}
impl<T: AstInfo> AstDisplay for ShowCreateSinkStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW CREATE SINK ");
f.write_node(&self.sink_name);
}
}
impl_display_t!(ShowCreateSinkStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowCreateIndexStatement<T: AstInfo> {
pub index_name: T::ItemName,
}
impl<T: AstInfo> AstDisplay for ShowCreateIndexStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW CREATE INDEX ");
f.write_node(&self.index_name);
}
}
impl_display_t!(ShowCreateIndexStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowCreateConnectionStatement<T: AstInfo> {
pub connection_name: T::ItemName,
}
impl<T: AstInfo> AstDisplay for ShowCreateConnectionStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW CREATE CONNECTION ");
f.write_node(&self.connection_name);
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ShowCreateClusterStatement<T: AstInfo> {
pub cluster_name: T::ClusterName,
}
impl<T: AstInfo> AstDisplay for ShowCreateClusterStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SHOW CREATE CLUSTER ");
f.write_node(&self.cluster_name);
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct StartTransactionStatement {
pub modes: Vec<TransactionMode>,
}
impl AstDisplay for StartTransactionStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("START TRANSACTION");
if !self.modes.is_empty() {
f.write_str(" ");
f.write_node(&display::comma_separated(&self.modes));
}
}
}
impl_display!(StartTransactionStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct SetTransactionStatement {
pub local: bool,
pub modes: Vec<TransactionMode>,
}
impl AstDisplay for SetTransactionStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SET ");
if !self.local {
f.write_str("SESSION CHARACTERISTICS AS ");
}
f.write_str("TRANSACTION");
if !self.modes.is_empty() {
f.write_str(" ");
f.write_node(&display::comma_separated(&self.modes));
}
}
}
impl_display!(SetTransactionStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CommitStatement {
pub chain: bool,
}
impl AstDisplay for CommitStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("COMMIT");
if self.chain {
f.write_str(" AND CHAIN");
}
}
}
impl_display!(CommitStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct RollbackStatement {
pub chain: bool,
}
impl AstDisplay for RollbackStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ROLLBACK");
if self.chain {
f.write_str(" AND CHAIN");
}
}
}
impl_display!(RollbackStatement);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum SubscribeOptionName {
Snapshot,
Progress,
}
impl AstDisplay for SubscribeOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
SubscribeOptionName::Snapshot => f.write_str("SNAPSHOT"),
SubscribeOptionName::Progress => f.write_str("PROGRESS"),
}
}
}
impl_display!(SubscribeOptionName);
impl WithOptionName for SubscribeOptionName {
fn redact_value(&self) -> bool {
match self {
SubscribeOptionName::Snapshot | SubscribeOptionName::Progress => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct SubscribeOption<T: AstInfo> {
pub name: SubscribeOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(SubscribeOption);
impl_display_t!(SubscribeOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct SubscribeStatement<T: AstInfo> {
pub relation: SubscribeRelation<T>,
pub options: Vec<SubscribeOption<T>>,
pub as_of: Option<AsOf<T>>,
pub up_to: Option<Expr<T>>,
pub output: SubscribeOutput<T>,
}
impl<T: AstInfo> AstDisplay for SubscribeStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("SUBSCRIBE ");
f.write_node(&self.relation);
if !self.options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.options));
f.write_str(")");
}
if let Some(as_of) = &self.as_of {
f.write_str(" ");
f.write_node(as_of);
}
if let Some(up_to) = &self.up_to {
f.write_str(" UP TO ");
f.write_node(up_to);
}
f.write_str(&self.output);
}
}
impl_display_t!(SubscribeStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum SubscribeRelation<T: AstInfo> {
Name(T::ItemName),
Query(Query<T>),
}
impl<T: AstInfo> AstDisplay for SubscribeRelation<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
SubscribeRelation::Name(name) => f.write_node(name),
SubscribeRelation::Query(query) => {
f.write_str("(");
f.write_node(query);
f.write_str(")");
}
}
}
}
impl_display_t!(SubscribeRelation);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct ExplainPlanStatement<T: AstInfo> {
pub stage: Option<ExplainStage>,
pub with_options: Vec<ExplainPlanOption<T>>,
pub format: Option<ExplainFormat>,
pub explainee: Explainee<T>,
}
impl<T: AstInfo> ExplainPlanStatement<T> {
pub fn stage(&self) -> ExplainStage {
self.stage.unwrap_or(ExplainStage::GlobalPlan)
}
pub fn format(&self) -> ExplainFormat {
self.format.unwrap_or(ExplainFormat::Text)
}
}
impl<T: AstInfo> AstDisplay for ExplainPlanStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("EXPLAIN");
if let Some(stage) = &self.stage {
f.write_str(" ");
f.write_node(stage);
}
if !self.with_options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.with_options));
f.write_str(")");
}
if let Some(format) = &self.format {
f.write_str(" AS ");
f.write_node(format);
}
if self.stage.is_some() {
f.write_str(" FOR");
}
f.write_str(" ");
f.write_node(&self.explainee);
}
}
impl_display_t!(ExplainPlanStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum ExplainPlanOptionName {
Arity,
Cardinality,
ColumnNames,
FilterPushdown,
HumanizedExpressions,
JoinImplementations,
Keys,
LinearChains,
NonNegative,
NoFastPath,
NoNotices,
NodeIdentifiers,
RawPlans,
RawSyntax,
Raw, Redacted,
SubtreeSize,
Timing,
Types,
ReoptimizeImportedViews,
EnableNewOuterJoinLowering,
EnableEagerDeltaJoins,
EnableVariadicLeftJoinLowering,
EnableLetrecFixpointAnalysis,
EnableOuterJoinNullFilter,
EnableValueWindowFunctionFusion,
EnableReduceUnnestListFusion,
}
impl WithOptionName for ExplainPlanOptionName {
fn redact_value(&self) -> bool {
match self {
Self::Arity
| Self::Cardinality
| Self::ColumnNames
| Self::FilterPushdown
| Self::HumanizedExpressions
| Self::JoinImplementations
| Self::Keys
| Self::LinearChains
| Self::NonNegative
| Self::NoFastPath
| Self::NoNotices
| Self::NodeIdentifiers
| Self::RawPlans
| Self::RawSyntax
| Self::Raw
| Self::Redacted
| Self::SubtreeSize
| Self::Timing
| Self::Types
| Self::ReoptimizeImportedViews
| Self::EnableNewOuterJoinLowering
| Self::EnableEagerDeltaJoins
| Self::EnableVariadicLeftJoinLowering
| Self::EnableLetrecFixpointAnalysis
| Self::EnableOuterJoinNullFilter
| Self::EnableValueWindowFunctionFusion
| Self::EnableReduceUnnestListFusion => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct ExplainPlanOption<T: AstInfo> {
pub name: ExplainPlanOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(ExplainPlanOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum ExplainSinkSchemaFor {
Key,
Value,
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct ExplainSinkSchemaStatement<T: AstInfo> {
pub schema_for: ExplainSinkSchemaFor,
pub format: Option<ExplainFormat>,
pub statement: CreateSinkStatement<T>,
}
impl<T: AstInfo> AstDisplay for ExplainSinkSchemaStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("EXPLAIN ");
match &self.schema_for {
ExplainSinkSchemaFor::Key => f.write_str("KEY"),
ExplainSinkSchemaFor::Value => f.write_str("VALUE"),
}
f.write_str(" SCHEMA");
if let Some(format) = &self.format {
f.write_str(" AS ");
f.write_node(format);
}
f.write_str(" FOR ");
f.write_node(&self.statement);
}
}
impl_display_t!(ExplainSinkSchemaStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct ExplainPushdownStatement<T: AstInfo> {
pub explainee: Explainee<T>,
}
impl<T: AstInfo> AstDisplay for ExplainPushdownStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("EXPLAIN FILTER PUSHDOWN FOR ");
f.write_node(&self.explainee);
}
}
impl_display_t!(ExplainPushdownStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct ExplainTimestampStatement<T: AstInfo> {
pub format: Option<ExplainFormat>,
pub select: SelectStatement<T>,
}
impl<T: AstInfo> ExplainTimestampStatement<T> {
pub fn format(&self) -> ExplainFormat {
self.format.unwrap_or(ExplainFormat::Text)
}
}
impl<T: AstInfo> AstDisplay for ExplainTimestampStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("EXPLAIN TIMESTAMP");
if let Some(format) = &self.format {
f.write_str(" AS ");
f.write_node(format);
}
f.write_str(" FOR ");
f.write_node(&self.select);
}
}
impl_display_t!(ExplainTimestampStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum InsertSource<T: AstInfo> {
Query(Query<T>),
DefaultValues,
}
impl<T: AstInfo> AstDisplay for InsertSource<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
InsertSource::Query(query) => f.write_node(query),
InsertSource::DefaultValues => f.write_str("DEFAULT VALUES"),
}
}
}
impl_display_t!(InsertSource);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord, Copy)]
pub enum ObjectType {
Table,
View,
MaterializedView,
Source,
Sink,
Index,
Type,
Role,
Cluster,
ClusterReplica,
Secret,
Connection,
Database,
Schema,
Func,
Subsource,
}
impl ObjectType {
pub fn lives_in_schema(&self) -> bool {
match self {
ObjectType::Table
| ObjectType::View
| ObjectType::MaterializedView
| ObjectType::Source
| ObjectType::Sink
| ObjectType::Index
| ObjectType::Type
| ObjectType::Secret
| ObjectType::Connection
| ObjectType::Func
| ObjectType::Subsource => true,
ObjectType::Database
| ObjectType::Schema
| ObjectType::Cluster
| ObjectType::ClusterReplica
| ObjectType::Role => false,
}
}
}
impl AstDisplay for ObjectType {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
ObjectType::Table => "TABLE",
ObjectType::View => "VIEW",
ObjectType::MaterializedView => "MATERIALIZED VIEW",
ObjectType::Source => "SOURCE",
ObjectType::Sink => "SINK",
ObjectType::Index => "INDEX",
ObjectType::Type => "TYPE",
ObjectType::Role => "ROLE",
ObjectType::Cluster => "CLUSTER",
ObjectType::ClusterReplica => "CLUSTER REPLICA",
ObjectType::Secret => "SECRET",
ObjectType::Connection => "CONNECTION",
ObjectType::Database => "DATABASE",
ObjectType::Schema => "SCHEMA",
ObjectType::Func => "FUNCTION",
ObjectType::Subsource => "SUBSOURCE",
})
}
}
impl_display!(ObjectType);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord, Copy)]
pub enum SystemObjectType {
System,
Object(ObjectType),
}
impl AstDisplay for SystemObjectType {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
SystemObjectType::System => f.write_str("SYSTEM"),
SystemObjectType::Object(object) => f.write_node(object),
}
}
}
impl_display!(SystemObjectType);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum ShowStatementFilter<T: AstInfo> {
Like(String),
Where(Expr<T>),
}
impl<T: AstInfo> AstDisplay for ShowStatementFilter<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
use ShowStatementFilter::*;
match self {
Like(pattern) => {
f.write_str("LIKE '");
f.write_node(&display::escape_single_quote_string(pattern));
f.write_str("'");
}
Where(expr) => {
f.write_str("WHERE ");
f.write_node(expr);
}
}
}
}
impl_display_t!(ShowStatementFilter);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum WithOptionValue<T: AstInfo> {
Value(Value),
DataType(T::DataType),
Secret(T::ItemName),
Item(T::ItemName),
UnresolvedItemName(UnresolvedItemName),
Ident(Ident),
Sequence(Vec<WithOptionValue<T>>),
Map(BTreeMap<String, WithOptionValue<T>>),
Expr(Expr<T>),
ClusterReplicas(Vec<ReplicaDefinition<T>>),
ConnectionKafkaBroker(KafkaBroker<T>),
ConnectionAwsPrivatelink(ConnectionDefaultAwsPrivatelink<T>),
RetainHistoryFor(Value),
Refresh(RefreshOptionValue<T>),
ClusterScheduleOptionValue(ClusterScheduleOptionValue),
ClusterAlterStrategy(ClusterAlterOptionValue<T>),
}
impl<T: AstInfo> AstDisplay for WithOptionValue<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
if f.redacted() {
match self {
WithOptionValue::Value(_)
| WithOptionValue::Sequence(_)
| WithOptionValue::Map(_)
| WithOptionValue::RetainHistoryFor(_)
| WithOptionValue::Refresh(_)
| WithOptionValue::Expr(_) => {
}
WithOptionValue::Secret(_) | WithOptionValue::ConnectionKafkaBroker(_) => {
f.write_str("'<REDACTED>'");
return;
}
WithOptionValue::DataType(_)
| WithOptionValue::Item(_)
| WithOptionValue::UnresolvedItemName(_)
| WithOptionValue::Ident(_)
| WithOptionValue::ConnectionAwsPrivatelink(_)
| WithOptionValue::ClusterReplicas(_)
| WithOptionValue::ClusterScheduleOptionValue(_)
| WithOptionValue::ClusterAlterStrategy(_) => {
}
}
}
match self {
WithOptionValue::Sequence(values) => {
f.write_str("(");
f.write_node(&display::comma_separated(values));
f.write_str(")");
}
WithOptionValue::Map(values) => {
f.write_str("MAP[");
let len = values.len();
for (i, (key, value)) in values.iter().enumerate() {
f.write_str("'");
f.write_node(&display::escape_single_quote_string(key));
f.write_str("' => ");
f.write_node(value);
if i + 1 < len {
f.write_str(", ");
}
}
f.write_str("]");
}
WithOptionValue::Expr(e) => f.write_node(e),
WithOptionValue::Value(value) => f.write_node(value),
WithOptionValue::DataType(typ) => f.write_node(typ),
WithOptionValue::Secret(name) => {
f.write_str("SECRET ");
f.write_node(name)
}
WithOptionValue::Item(obj) => f.write_node(obj),
WithOptionValue::UnresolvedItemName(r) => f.write_node(r),
WithOptionValue::Ident(r) => f.write_node(r),
WithOptionValue::ClusterReplicas(replicas) => {
f.write_str("(");
f.write_node(&display::comma_separated(replicas));
f.write_str(")");
}
WithOptionValue::ConnectionAwsPrivatelink(aws_privatelink) => {
f.write_node(aws_privatelink);
}
WithOptionValue::ConnectionKafkaBroker(broker) => {
f.write_node(broker);
}
WithOptionValue::RetainHistoryFor(value) => {
f.write_str("FOR ");
f.write_node(value);
}
WithOptionValue::Refresh(opt) => f.write_node(opt),
WithOptionValue::ClusterScheduleOptionValue(value) => f.write_node(value),
WithOptionValue::ClusterAlterStrategy(value) => f.write_node(value),
}
}
}
impl_display_t!(WithOptionValue);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum RefreshOptionValue<T: AstInfo> {
OnCommit,
AtCreation,
At(RefreshAtOptionValue<T>),
Every(RefreshEveryOptionValue<T>),
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct RefreshAtOptionValue<T: AstInfo> {
pub time: Expr<T>,
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub struct RefreshEveryOptionValue<T: AstInfo> {
pub interval: IntervalValue,
pub aligned_to: Option<Expr<T>>,
}
impl<T: AstInfo> AstDisplay for RefreshOptionValue<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
RefreshOptionValue::OnCommit => {
f.write_str("ON COMMIT");
}
RefreshOptionValue::AtCreation => {
f.write_str("AT CREATION");
}
RefreshOptionValue::At(RefreshAtOptionValue { time }) => {
f.write_str("AT ");
f.write_node(time);
}
RefreshOptionValue::Every(RefreshEveryOptionValue {
interval,
aligned_to,
}) => {
f.write_str("EVERY '");
f.write_node(interval);
if let Some(aligned_to) = aligned_to {
f.write_str(" ALIGNED TO ");
f.write_node(aligned_to)
}
}
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord, Deserialize, Serialize)]
pub enum ClusterScheduleOptionValue {
Manual,
Refresh {
hydration_time_estimate: Option<IntervalValue>,
},
}
impl Default for ClusterScheduleOptionValue {
fn default() -> Self {
ClusterScheduleOptionValue::Manual
}
}
impl AstDisplay for ClusterScheduleOptionValue {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
ClusterScheduleOptionValue::Manual => {
f.write_str("MANUAL");
}
ClusterScheduleOptionValue::Refresh {
hydration_time_estimate,
} => {
f.write_str("ON REFRESH");
if let Some(hydration_time_estimate) = hydration_time_estimate {
f.write_str(" (HYDRATION TIME ESTIMATE = '");
f.write_node(hydration_time_estimate);
f.write_str(")");
}
}
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum TransactionMode {
AccessMode(TransactionAccessMode),
IsolationLevel(TransactionIsolationLevel),
}
impl AstDisplay for TransactionMode {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
use TransactionMode::*;
match self {
AccessMode(access_mode) => f.write_node(access_mode),
IsolationLevel(iso_level) => {
f.write_str("ISOLATION LEVEL ");
f.write_node(iso_level);
}
}
}
}
impl_display!(TransactionMode);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum TransactionAccessMode {
ReadOnly,
ReadWrite,
}
impl AstDisplay for TransactionAccessMode {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
use TransactionAccessMode::*;
f.write_str(match self {
ReadOnly => "READ ONLY",
ReadWrite => "READ WRITE",
})
}
}
impl_display!(TransactionAccessMode);
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum TransactionIsolationLevel {
ReadUncommitted,
ReadCommitted,
RepeatableRead,
Serializable,
StrongSessionSerializable,
StrictSerializable,
}
impl AstDisplay for TransactionIsolationLevel {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
use TransactionIsolationLevel::*;
f.write_str(match self {
ReadUncommitted => "READ UNCOMMITTED",
ReadCommitted => "READ COMMITTED",
RepeatableRead => "REPEATABLE READ",
Serializable => "SERIALIZABLE",
StrongSessionSerializable => "STRONG SESSION SERIALIZABLE",
StrictSerializable => "STRICT SERIALIZABLE",
})
}
}
impl_display!(TransactionIsolationLevel);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum SetVariableTo {
Default,
Values(Vec<SetVariableValue>),
}
impl AstDisplay for SetVariableTo {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
use SetVariableTo::*;
match self {
Values(values) => f.write_node(&display::comma_separated(values)),
Default => f.write_str("DEFAULT"),
}
}
}
impl_display!(SetVariableTo);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum SetVariableValue {
Ident(Ident),
Literal(Value),
}
impl AstDisplay for SetVariableValue {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
use SetVariableValue::*;
match self {
Ident(ident) => f.write_node(ident),
Literal(literal) => f.write_node(literal),
}
}
}
impl_display!(SetVariableValue);
impl SetVariableValue {
pub fn into_unquoted_value(self) -> String {
match self {
SetVariableValue::Literal(Value::String(s)) => s,
SetVariableValue::Literal(lit) => lit.to_string(),
SetVariableValue::Ident(ident) => ident.into_string(),
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct Assignment<T: AstInfo> {
pub id: Ident,
pub value: Expr<T>,
}
impl<T: AstInfo> AstDisplay for Assignment<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_node(&self.id);
f.write_str(" = ");
f.write_node(&self.value);
}
}
impl_display_t!(Assignment);
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub enum ExplainStage {
RawPlan,
DecorrelatedPlan,
LocalPlan,
GlobalPlan,
PhysicalPlan,
Trace,
PlanInsights,
}
impl ExplainStage {
pub fn paths(&self) -> Option<SmallVec<[NamedPlan; 4]>> {
use NamedPlan::*;
match self {
Self::RawPlan => Some(smallvec![Raw]),
Self::DecorrelatedPlan => Some(smallvec![Decorrelated]),
Self::LocalPlan => Some(smallvec![Local]),
Self::GlobalPlan => Some(smallvec![Global]),
Self::PhysicalPlan => Some(smallvec![Physical]),
Self::Trace => None,
Self::PlanInsights => Some(smallvec![Raw, Global, FastPath]),
}
}
pub fn show_fast_path(&self) -> bool {
match self {
Self::RawPlan => false,
Self::DecorrelatedPlan => false,
Self::LocalPlan => false,
Self::GlobalPlan => true,
Self::PhysicalPlan => true,
Self::Trace => false,
Self::PlanInsights => false,
}
}
}
impl AstDisplay for ExplainStage {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
Self::RawPlan => f.write_str("RAW PLAN"),
Self::DecorrelatedPlan => f.write_str("DECORRELATED PLAN"),
Self::LocalPlan => f.write_str("LOCALLY OPTIMIZED PLAN"),
Self::GlobalPlan => f.write_str("OPTIMIZED PLAN"),
Self::PhysicalPlan => f.write_str("PHYSICAL PLAN"),
Self::Trace => f.write_str("OPTIMIZER TRACE"),
Self::PlanInsights => f.write_str("PLAN INSIGHTS"),
}
}
}
impl_display!(ExplainStage);
#[derive(Clone)]
pub enum NamedPlan {
Raw,
Decorrelated,
Local,
Global,
Physical,
FastPath,
}
impl NamedPlan {
pub fn of_path(value: &str) -> Option<Self> {
match value {
"optimize/raw" => Some(Self::Raw),
"optimize/hir_to_mir" => Some(Self::Decorrelated),
"optimize/local" => Some(Self::Local),
"optimize/global" => Some(Self::Global),
"optimize/finalize_dataflow" => Some(Self::Physical),
"optimize/fast_path" => Some(Self::FastPath),
_ => None,
}
}
pub fn path(&self) -> &'static str {
match self {
Self::Raw => "optimize/raw",
Self::Decorrelated => "optimize/hir_to_mir",
Self::Local => "optimize/local",
Self::Global => "optimize/global",
Self::Physical => "optimize/finalize_dataflow",
Self::FastPath => "optimize/fast_path",
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum Explainee<T: AstInfo> {
View(T::ItemName),
MaterializedView(T::ItemName),
Index(T::ItemName),
ReplanView(T::ItemName),
ReplanMaterializedView(T::ItemName),
ReplanIndex(T::ItemName),
Select(Box<SelectStatement<T>>, bool),
CreateView(Box<CreateViewStatement<T>>, bool),
CreateMaterializedView(Box<CreateMaterializedViewStatement<T>>, bool),
CreateIndex(Box<CreateIndexStatement<T>>, bool),
}
impl<T: AstInfo> Explainee<T> {
pub fn is_view(&self) -> bool {
use Explainee::*;
matches!(self, View(_) | ReplanView(_) | CreateView(_, _))
}
}
impl<T: AstInfo> AstDisplay for Explainee<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
Self::View(name) => {
f.write_str("VIEW ");
f.write_node(name);
}
Self::MaterializedView(name) => {
f.write_str("MATERIALIZED VIEW ");
f.write_node(name);
}
Self::Index(name) => {
f.write_str("INDEX ");
f.write_node(name);
}
Self::ReplanView(name) => {
f.write_str("REPLAN VIEW ");
f.write_node(name);
}
Self::ReplanMaterializedView(name) => {
f.write_str("REPLAN MATERIALIZED VIEW ");
f.write_node(name);
}
Self::ReplanIndex(name) => {
f.write_str("REPLAN INDEX ");
f.write_node(name);
}
Self::Select(select, broken) => {
if *broken {
f.write_str("BROKEN ");
}
f.write_node(select);
}
Self::CreateView(statement, broken) => {
if *broken {
f.write_str("BROKEN ");
}
f.write_node(statement);
}
Self::CreateMaterializedView(statement, broken) => {
if *broken {
f.write_str("BROKEN ");
}
f.write_node(statement);
}
Self::CreateIndex(statement, broken) => {
if *broken {
f.write_str("BROKEN ");
}
f.write_node(statement);
}
}
}
}
impl_display_t!(Explainee);
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
pub enum ExplainFormat {
Text,
Json,
Dot,
}
impl AstDisplay for ExplainFormat {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
Self::Text => f.write_str("TEXT"),
Self::Json => f.write_str("JSON"),
Self::Dot => f.write_str("DOT"),
}
}
}
impl_display!(ExplainFormat);
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub enum IfExistsBehavior {
Error,
Skip,
Replace,
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct DeclareStatement<T: AstInfo> {
pub name: Ident,
pub stmt: Box<T::NestedStatement>,
pub sql: String,
}
impl<T: AstInfo> AstDisplay for DeclareStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("DECLARE ");
f.write_node(&self.name);
f.write_str(" CURSOR FOR ");
f.write_node(&self.stmt);
}
}
impl_display_t!(DeclareStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CloseStatement {
pub name: Ident,
}
impl AstDisplay for CloseStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("CLOSE ");
f.write_node(&self.name);
}
}
impl_display!(CloseStatement);
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum FetchOptionName {
Timeout,
}
impl AstDisplay for FetchOptionName {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
FetchOptionName::Timeout => "TIMEOUT",
})
}
}
impl WithOptionName for FetchOptionName {
fn redact_value(&self) -> bool {
match self {
FetchOptionName::Timeout => false,
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct FetchOption<T: AstInfo> {
pub name: FetchOptionName,
pub value: Option<WithOptionValue<T>>,
}
impl_display_for_with_option!(FetchOption);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct FetchStatement<T: AstInfo> {
pub name: Ident,
pub count: Option<FetchDirection>,
pub options: Vec<FetchOption<T>>,
}
impl<T: AstInfo> AstDisplay for FetchStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("FETCH ");
if let Some(ref count) = self.count {
f.write_str(format!("{} ", count));
}
f.write_node(&self.name);
if !self.options.is_empty() {
f.write_str(" WITH (");
f.write_node(&display::comma_separated(&self.options));
f.write_str(")");
}
}
}
impl_display_t!(FetchStatement);
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub enum FetchDirection {
ForwardAll,
ForwardCount(u64),
}
impl AstDisplay for FetchDirection {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
FetchDirection::ForwardAll => f.write_str("ALL"),
FetchDirection::ForwardCount(count) => f.write_str(format!("{}", count)),
}
}
}
impl_display!(FetchDirection);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct PrepareStatement<T: AstInfo> {
pub name: Ident,
pub stmt: Box<T::NestedStatement>,
pub sql: String,
}
impl<T: AstInfo> AstDisplay for PrepareStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("PREPARE ");
f.write_node(&self.name);
f.write_str(" AS ");
f.write_node(&self.stmt);
}
}
impl_display_t!(PrepareStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct ExecuteStatement<T: AstInfo> {
pub name: Ident,
pub params: Vec<Expr<T>>,
}
impl<T: AstInfo> AstDisplay for ExecuteStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("EXECUTE ");
f.write_node(&self.name);
if !self.params.is_empty() {
f.write_str(" (");
f.write_node(&display::comma_separated(&self.params));
f.write_str(")");
}
}
}
impl_display_t!(ExecuteStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct DeallocateStatement {
pub name: Option<Ident>,
}
impl AstDisplay for DeallocateStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("DEALLOCATE ");
match &self.name {
Some(name) => f.write_node(name),
None => f.write_str("ALL"),
};
}
}
impl_display!(DeallocateStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct RaiseStatement {
pub severity: NoticeSeverity,
}
impl AstDisplay for RaiseStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("RAISE ");
f.write_node(&self.severity);
}
}
impl_display!(RaiseStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum NoticeSeverity {
Debug,
Info,
Log,
Notice,
Warning,
}
impl AstDisplay for NoticeSeverity {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
NoticeSeverity::Debug => "DEBUG",
NoticeSeverity::Info => "INFO",
NoticeSeverity::Log => "LOG",
NoticeSeverity::Notice => "NOTICE",
NoticeSeverity::Warning => "WARNING",
})
}
}
impl_display!(NoticeSeverity);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterSystemSetStatement {
pub name: Ident,
pub to: SetVariableTo,
}
impl AstDisplay for AlterSystemSetStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER SYSTEM SET ");
f.write_node(&self.name);
f.write_str(" = ");
f.write_node(&self.to);
}
}
impl_display!(AlterSystemSetStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterSystemResetStatement {
pub name: Ident,
}
impl AstDisplay for AlterSystemResetStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER SYSTEM RESET ");
f.write_node(&self.name);
}
}
impl_display!(AlterSystemResetStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterSystemResetAllStatement {}
impl AstDisplay for AlterSystemResetAllStatement {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER SYSTEM RESET ALL");
}
}
impl_display!(AlterSystemResetAllStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum AsOf<T: AstInfo> {
At(Expr<T>),
AtLeast(Expr<T>),
}
impl<T: AstInfo> AstDisplay for AsOf<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("AS OF ");
match self {
AsOf::At(expr) => f.write_node(expr),
AsOf::AtLeast(expr) => {
f.write_str("AT LEAST ");
f.write_node(expr);
}
}
}
}
impl_display_t!(AsOf);
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum ShowStatement<T: AstInfo> {
ShowObjects(ShowObjectsStatement<T>),
ShowColumns(ShowColumnsStatement<T>),
ShowCreateView(ShowCreateViewStatement<T>),
ShowCreateMaterializedView(ShowCreateMaterializedViewStatement<T>),
ShowCreateSource(ShowCreateSourceStatement<T>),
ShowCreateTable(ShowCreateTableStatement<T>),
ShowCreateSink(ShowCreateSinkStatement<T>),
ShowCreateIndex(ShowCreateIndexStatement<T>),
ShowCreateConnection(ShowCreateConnectionStatement<T>),
ShowCreateCluster(ShowCreateClusterStatement<T>),
ShowVariable(ShowVariableStatement),
InspectShard(InspectShardStatement),
}
impl<T: AstInfo> AstDisplay for ShowStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
ShowStatement::ShowObjects(stmt) => f.write_node(stmt),
ShowStatement::ShowColumns(stmt) => f.write_node(stmt),
ShowStatement::ShowCreateView(stmt) => f.write_node(stmt),
ShowStatement::ShowCreateMaterializedView(stmt) => f.write_node(stmt),
ShowStatement::ShowCreateSource(stmt) => f.write_node(stmt),
ShowStatement::ShowCreateTable(stmt) => f.write_node(stmt),
ShowStatement::ShowCreateSink(stmt) => f.write_node(stmt),
ShowStatement::ShowCreateIndex(stmt) => f.write_node(stmt),
ShowStatement::ShowCreateConnection(stmt) => f.write_node(stmt),
ShowStatement::ShowCreateCluster(stmt) => f.write_node(stmt),
ShowStatement::ShowVariable(stmt) => f.write_node(stmt),
ShowStatement::InspectShard(stmt) => f.write_node(stmt),
}
}
}
impl_display_t!(ShowStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct GrantRoleStatement<T: AstInfo> {
pub role_names: Vec<T::RoleName>,
pub member_names: Vec<T::RoleName>,
}
impl<T: AstInfo> AstDisplay for GrantRoleStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("GRANT ");
f.write_node(&display::comma_separated(&self.role_names));
f.write_str(" TO ");
f.write_node(&display::comma_separated(&self.member_names));
}
}
impl_display_t!(GrantRoleStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct RevokeRoleStatement<T: AstInfo> {
pub role_names: Vec<T::RoleName>,
pub member_names: Vec<T::RoleName>,
}
impl<T: AstInfo> AstDisplay for RevokeRoleStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("REVOKE ");
f.write_node(&display::comma_separated(&self.role_names));
f.write_str(" FROM ");
f.write_node(&display::comma_separated(&self.member_names));
}
}
impl_display_t!(RevokeRoleStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum Privilege {
SELECT,
INSERT,
UPDATE,
DELETE,
USAGE,
CREATE,
CREATEROLE,
CREATEDB,
CREATECLUSTER,
}
impl AstDisplay for Privilege {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str(match self {
Privilege::SELECT => "SELECT",
Privilege::INSERT => "INSERT",
Privilege::UPDATE => "UPDATE",
Privilege::DELETE => "DELETE",
Privilege::CREATE => "CREATE",
Privilege::USAGE => "USAGE",
Privilege::CREATEROLE => "CREATEROLE",
Privilege::CREATEDB => "CREATEDB",
Privilege::CREATECLUSTER => "CREATECLUSTER",
});
}
}
impl_display!(Privilege);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum PrivilegeSpecification {
All,
Privileges(Vec<Privilege>),
}
impl AstDisplay for PrivilegeSpecification {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
PrivilegeSpecification::All => f.write_str("ALL"),
PrivilegeSpecification::Privileges(privileges) => {
f.write_node(&display::comma_separated(privileges))
}
}
}
}
impl_display!(PrivilegeSpecification);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum GrantTargetSpecification<T: AstInfo> {
Object {
object_type: ObjectType,
object_spec_inner: GrantTargetSpecificationInner<T>,
},
System,
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum GrantTargetSpecificationInner<T: AstInfo> {
All(GrantTargetAllSpecification<T>),
Objects { names: Vec<T::ObjectName> },
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum GrantTargetAllSpecification<T: AstInfo> {
All,
AllDatabases { databases: Vec<T::DatabaseName> },
AllSchemas { schemas: Vec<T::SchemaName> },
}
impl<T: AstInfo> GrantTargetAllSpecification<T> {
pub fn len(&self) -> usize {
match self {
GrantTargetAllSpecification::All => 1,
GrantTargetAllSpecification::AllDatabases { databases } => databases.len(),
GrantTargetAllSpecification::AllSchemas { schemas } => schemas.len(),
}
}
}
impl<T: AstInfo> AstDisplay for GrantTargetSpecification<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
GrantTargetSpecification::Object {
object_type,
object_spec_inner,
} => match object_spec_inner {
GrantTargetSpecificationInner::All(all_spec) => match all_spec {
GrantTargetAllSpecification::All => {
f.write_str("ALL ");
f.write_node(object_type);
f.write_str("S");
}
GrantTargetAllSpecification::AllDatabases { databases } => {
f.write_str("ALL ");
f.write_node(object_type);
f.write_str("S IN DATABASE ");
f.write_node(&display::comma_separated(databases));
}
GrantTargetAllSpecification::AllSchemas { schemas } => {
f.write_str("ALL ");
f.write_node(object_type);
f.write_str("S IN SCHEMA ");
f.write_node(&display::comma_separated(schemas));
}
},
GrantTargetSpecificationInner::Objects { names } => {
f.write_node(object_type);
f.write_str(" ");
f.write_node(&display::comma_separated(names));
}
},
GrantTargetSpecification::System => f.write_str("SYSTEM"),
}
}
}
impl_display_t!(GrantTargetSpecification);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct GrantPrivilegesStatement<T: AstInfo> {
pub privileges: PrivilegeSpecification,
pub target: GrantTargetSpecification<T>,
pub roles: Vec<T::RoleName>,
}
impl<T: AstInfo> AstDisplay for GrantPrivilegesStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("GRANT ");
f.write_node(&self.privileges);
f.write_str(" ON ");
f.write_node(&self.target);
f.write_str(" TO ");
f.write_node(&display::comma_separated(&self.roles));
}
}
impl_display_t!(GrantPrivilegesStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct RevokePrivilegesStatement<T: AstInfo> {
pub privileges: PrivilegeSpecification,
pub target: GrantTargetSpecification<T>,
pub roles: Vec<T::RoleName>,
}
impl<T: AstInfo> AstDisplay for RevokePrivilegesStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("REVOKE ");
f.write_node(&self.privileges);
f.write_str(" ON ");
f.write_node(&self.target);
f.write_str(" FROM ");
f.write_node(&display::comma_separated(&self.roles));
}
}
impl_display_t!(RevokePrivilegesStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum TargetRoleSpecification<T: AstInfo> {
Roles(Vec<T::RoleName>),
AllRoles,
}
impl<T: AstInfo> AstDisplay for TargetRoleSpecification<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
TargetRoleSpecification::Roles(roles) => f.write_node(&display::comma_separated(roles)),
TargetRoleSpecification::AllRoles => f.write_str("ALL ROLES"),
}
}
}
impl_display_t!(TargetRoleSpecification);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AbbreviatedGrantStatement<T: AstInfo> {
pub privileges: PrivilegeSpecification,
pub object_type: ObjectType,
pub grantees: Vec<T::RoleName>,
}
impl<T: AstInfo> AstDisplay for AbbreviatedGrantStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("GRANT ");
f.write_node(&self.privileges);
f.write_str(" ON ");
f.write_node(&self.object_type);
f.write_str("S TO ");
f.write_node(&display::comma_separated(&self.grantees));
}
}
impl_display_t!(AbbreviatedGrantStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AbbreviatedRevokeStatement<T: AstInfo> {
pub privileges: PrivilegeSpecification,
pub object_type: ObjectType,
pub revokees: Vec<T::RoleName>,
}
impl<T: AstInfo> AstDisplay for AbbreviatedRevokeStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("REVOKE ");
f.write_node(&self.privileges);
f.write_str(" ON ");
f.write_node(&self.object_type);
f.write_str("S FROM ");
f.write_node(&display::comma_separated(&self.revokees));
}
}
impl_display_t!(AbbreviatedRevokeStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum AbbreviatedGrantOrRevokeStatement<T: AstInfo> {
Grant(AbbreviatedGrantStatement<T>),
Revoke(AbbreviatedRevokeStatement<T>),
}
impl<T: AstInfo> AbbreviatedGrantOrRevokeStatement<T> {
pub fn privileges(&self) -> &PrivilegeSpecification {
match self {
AbbreviatedGrantOrRevokeStatement::Grant(grant) => &grant.privileges,
AbbreviatedGrantOrRevokeStatement::Revoke(revoke) => &revoke.privileges,
}
}
pub fn object_type(&self) -> &ObjectType {
match self {
AbbreviatedGrantOrRevokeStatement::Grant(grant) => &grant.object_type,
AbbreviatedGrantOrRevokeStatement::Revoke(revoke) => &revoke.object_type,
}
}
pub fn roles(&self) -> &Vec<T::RoleName> {
match self {
AbbreviatedGrantOrRevokeStatement::Grant(grant) => &grant.grantees,
AbbreviatedGrantOrRevokeStatement::Revoke(revoke) => &revoke.revokees,
}
}
}
impl<T: AstInfo> AstDisplay for AbbreviatedGrantOrRevokeStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
match self {
AbbreviatedGrantOrRevokeStatement::Grant(grant) => f.write_node(grant),
AbbreviatedGrantOrRevokeStatement::Revoke(revoke) => f.write_node(revoke),
}
}
}
impl_display_t!(AbbreviatedGrantOrRevokeStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct AlterDefaultPrivilegesStatement<T: AstInfo> {
pub target_roles: TargetRoleSpecification<T>,
pub target_objects: GrantTargetAllSpecification<T>,
pub grant_or_revoke: AbbreviatedGrantOrRevokeStatement<T>,
}
impl<T: AstInfo> AstDisplay for AlterDefaultPrivilegesStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("ALTER DEFAULT PRIVILEGES");
match &self.target_roles {
TargetRoleSpecification::Roles(_) => {
f.write_str(" FOR ROLE ");
f.write_node(&self.target_roles);
}
TargetRoleSpecification::AllRoles => {
f.write_str(" FOR ");
f.write_node(&self.target_roles);
}
}
match &self.target_objects {
GrantTargetAllSpecification::All => {}
GrantTargetAllSpecification::AllDatabases { databases } => {
f.write_str(" IN DATABASE ");
f.write_node(&display::comma_separated(databases));
}
GrantTargetAllSpecification::AllSchemas { schemas } => {
f.write_str(" IN SCHEMA ");
f.write_node(&display::comma_separated(schemas));
}
}
f.write_str(" ");
f.write_node(&self.grant_or_revoke);
}
}
impl_display_t!(AlterDefaultPrivilegesStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct ReassignOwnedStatement<T: AstInfo> {
pub old_roles: Vec<T::RoleName>,
pub new_role: T::RoleName,
}
impl<T: AstInfo> AstDisplay for ReassignOwnedStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("REASSIGN OWNED BY ");
f.write_node(&display::comma_separated(&self.old_roles));
f.write_str(" TO ");
f.write_node(&self.new_role);
}
}
impl_display_t!(ReassignOwnedStatement);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CommentStatement<T: AstInfo> {
pub object: CommentObjectType<T>,
pub comment: Option<String>,
}
impl<T: AstInfo> AstDisplay for CommentStatement<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_str("COMMENT ON ");
f.write_node(&self.object);
f.write_str(" IS ");
match &self.comment {
Some(s) => {
f.write_str("'");
f.write_node(&display::escape_single_quote_string(s));
f.write_str("'");
}
None => f.write_str("NULL"),
}
}
}
impl_display_t!(CommentStatement);
#[derive(Debug, PartialEq, Eq, Hash, PartialOrd, Ord, Clone)]
pub struct ColumnName<T: AstInfo> {
pub relation: T::ItemName,
pub column: T::ColumnReference,
}
impl<T: AstInfo> AstDisplay for ColumnName<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
f.write_node(&self.relation);
f.write_str(".");
f.write_node(&self.column);
}
}
impl_display_t!(ColumnName);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum CommentObjectType<T: AstInfo> {
Table { name: T::ItemName },
View { name: T::ItemName },
Column { name: ColumnName<T> },
MaterializedView { name: T::ItemName },
Source { name: T::ItemName },
Sink { name: T::ItemName },
Index { name: T::ItemName },
Func { name: T::ItemName },
Connection { name: T::ItemName },
Type { ty: T::DataType },
Secret { name: T::ItemName },
Role { name: T::RoleName },
Database { name: T::DatabaseName },
Schema { name: T::SchemaName },
Cluster { name: T::ClusterName },
ClusterReplica { name: QualifiedReplica },
}
impl<T: AstInfo> AstDisplay for CommentObjectType<T> {
fn fmt<W: fmt::Write>(&self, f: &mut AstFormatter<W>) {
use CommentObjectType::*;
match self {
Table { name } => {
f.write_str("TABLE ");
f.write_node(name);
}
View { name } => {
f.write_str("VIEW ");
f.write_node(name);
}
Column { name } => {
f.write_str("COLUMN ");
f.write_node(name);
}
MaterializedView { name } => {
f.write_str("MATERIALIZED VIEW ");
f.write_node(name);
}
Source { name } => {
f.write_str("SOURCE ");
f.write_node(name);
}
Sink { name } => {
f.write_str("SINK ");
f.write_node(name);
}
Index { name } => {
f.write_str("INDEX ");
f.write_node(name);
}
Func { name } => {
f.write_str("FUNCTION ");
f.write_node(name);
}
Connection { name } => {
f.write_str("CONNECTION ");
f.write_node(name);
}
Type { ty } => {
f.write_str("TYPE ");
f.write_node(ty);
}
Secret { name } => {
f.write_str("SECRET ");
f.write_node(name);
}
Role { name } => {
f.write_str("ROLE ");
f.write_node(name);
}
Database { name } => {
f.write_str("DATABASE ");
f.write_node(name);
}
Schema { name } => {
f.write_str("SCHEMA ");
f.write_node(name);
}
Cluster { name } => {
f.write_str("CLUSTER ");
f.write_node(name);
}
ClusterReplica { name } => {
f.write_str("CLUSTER REPLICA ");
f.write_node(name);
}
}
}
}
impl_display_t!(CommentObjectType);
include!(concat!(env!("OUT_DIR"), "/display.simple_options.rs"));