use crate::ast::{ColumnName, Function, Operator, Value};
use serde::{Deserialize, Serialize};
use sqlparser::ast as sql;
use std::fmt;
#[derive(Debug, PartialEq, Clone, Serialize, Deserialize)]
pub enum Expr {
Column(ColumnName),
Function(Function),
Value(Value),
MultiValue(Vec<Value>),
BinaryOperation(Box<BinaryOperation>),
Nested(Box<Expr>),
}
#[derive(Debug, PartialEq, Clone, Serialize, Deserialize)]
pub struct ExprRename {
pub expr: Expr,
pub rename: Option<String>,
}
#[derive(Debug, PartialEq, Clone, Serialize, Deserialize)]
pub struct BinaryOperation {
pub left: Expr,
pub operator: Operator,
pub right: Expr,
}
impl Into<sql::Expr> for &Expr {
fn into(self) -> sql::Expr {
match self {
Expr::Column(column) => {
sql::Expr::Identifier(sql::Ident::new(&column.name))
}
Expr::Function(function) => {
sql::Expr::Function(Into::into(function))
}
Expr::Value(value) => sql::Expr::Value(Into::into(value)),
Expr::MultiValue(values) => sql::Expr::Tuple(
values
.into_iter()
.map(|v| sql::Expr::Value(Into::into(v)))
.collect(),
),
Expr::BinaryOperation(binop) => sql::Expr::BinaryOp {
left: Box::new(Into::into(&binop.left)),
op: Into::into(&binop.operator),
right: Box::new(Into::into(&binop.right)),
},
Expr::Nested(expr) => {
sql::Expr::Nested(Box::new(Into::into(expr.as_ref())))
}
}
}
}
impl fmt::Display for Expr {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self {
Expr::Column(column) => column.fmt(f),
Expr::Function(function) => function.fmt(f),
Expr::Value(value) => value.fmt(f),
Expr::MultiValue(values) => {
write!(f, "[")?;
for (i, value) in values.iter().enumerate() {
if i > 0 {
write!(f, ",")?;
}
value.fmt(f)?;
}
write!(f, "]")
}
Expr::BinaryOperation(bop) => bop.fmt(f),
Expr::Nested(expr) => write!(f, "({})", expr),
}
}
}
impl fmt::Display for BinaryOperation {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
if self.operator.needs_separator() {
write!(f, "{}={}.{}", self.left, self.operator, self.right)
} else {
write!(f, "{}{}{}", self.left, self.operator, self.right)
}
}
}
impl fmt::Display for ExprRename {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
self.expr.fmt(f)?;
if let Some(rename) = &self.rename {
write!(f, "=>{}", rename)?;
}
Ok(())
}
}