use crate::expression::operators::Concat;
use crate::mysql::backend::MysqlOnConflictClause;
use crate::mysql::Mysql;
use crate::query_builder::insert_statement::DefaultValues;
use crate::query_builder::locking_clause::{ForShare, ForUpdate, NoModifier, NoWait, SkipLocked};
use crate::query_builder::nodes::StaticQueryFragment;
use crate::query_builder::upsert::into_conflict_clause::OnConflictSelectWrapper;
use crate::query_builder::upsert::on_conflict_actions::{DoNothing, DoUpdate};
use crate::query_builder::upsert::on_conflict_clause::OnConflictValues;
use crate::query_builder::upsert::on_conflict_target::{ConflictTarget, OnConflictTarget};
use crate::query_builder::where_clause::NoWhereClause;
use crate::query_builder::{AstPass, QueryFragment};
use crate::result::QueryResult;
use crate::{Column, Table};
impl QueryFragment<Mysql> for ForUpdate {
fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, Mysql>) -> QueryResult<()> {
out.push_sql(" FOR UPDATE");
Ok(())
}
}
impl QueryFragment<Mysql> for ForShare {
fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, Mysql>) -> QueryResult<()> {
out.push_sql(" FOR SHARE");
Ok(())
}
}
impl QueryFragment<Mysql> for NoModifier {
fn walk_ast<'b>(&'b self, _out: AstPass<'_, 'b, Mysql>) -> QueryResult<()> {
Ok(())
}
}
impl QueryFragment<Mysql> for SkipLocked {
fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, Mysql>) -> QueryResult<()> {
out.push_sql(" SKIP LOCKED");
Ok(())
}
}
impl QueryFragment<Mysql> for NoWait {
fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, Mysql>) -> QueryResult<()> {
out.push_sql(" NOWAIT");
Ok(())
}
}
impl QueryFragment<Mysql, crate::mysql::backend::MysqlStyleDefaultValueClause> for DefaultValues {
fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, Mysql>) -> QueryResult<()> {
out.push_sql("() VALUES ()");
Ok(())
}
}
impl<L, R> QueryFragment<Mysql, crate::mysql::backend::MysqlConcatClause> for Concat<L, R>
where
L: QueryFragment<Mysql>,
R: QueryFragment<Mysql>,
{
fn walk_ast<'b>(
&'b self,
mut out: crate::query_builder::AstPass<'_, 'b, Mysql>,
) -> crate::result::QueryResult<()> {
out.push_sql("CONCAT(");
self.left.walk_ast(out.reborrow())?;
out.push_sql(",");
self.right.walk_ast(out.reborrow())?;
out.push_sql(")");
Ok(())
}
}
impl<T> QueryFragment<Mysql, crate::mysql::backend::MysqlOnConflictClause> for DoNothing<T>
where
T: Table + StaticQueryFragment,
T::Component: QueryFragment<Mysql>,
T::PrimaryKey: Column,
{
fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, Mysql>) -> QueryResult<()> {
out.push_sql(" UPDATE ");
T::STATIC_COMPONENT.walk_ast(out.reborrow())?;
out.push_sql(".");
out.push_identifier(<T::PrimaryKey as Column>::NAME)?;
out.push_sql(" = ");
T::STATIC_COMPONENT.walk_ast(out.reborrow())?;
out.push_sql(".");
out.push_identifier(<T::PrimaryKey as Column>::NAME)?;
Ok(())
}
}
impl<T, Tab> QueryFragment<Mysql, crate::mysql::backend::MysqlOnConflictClause> for DoUpdate<T, Tab>
where
T: QueryFragment<Mysql>,
Tab: Table + StaticQueryFragment,
Tab::Component: QueryFragment<Mysql>,
Tab::PrimaryKey: Column,
{
fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, Mysql>) -> QueryResult<()> {
out.unsafe_to_cache_prepared();
out.push_sql(" UPDATE ");
if self.changeset.is_noop(out.backend())? {
Tab::STATIC_COMPONENT.walk_ast(out.reborrow())?;
out.push_sql(".");
out.push_identifier(<Tab::PrimaryKey as Column>::NAME)?;
out.push_sql(" = ");
Tab::STATIC_COMPONENT.walk_ast(out.reborrow())?;
out.push_sql(".");
out.push_identifier(<Tab::PrimaryKey as Column>::NAME)?;
} else {
self.changeset.walk_ast(out.reborrow())?;
}
Ok(())
}
}
impl<Values, Target, Action> QueryFragment<Mysql, MysqlOnConflictClause>
for OnConflictValues<Values, Target, Action, NoWhereClause>
where
Values: QueryFragment<Mysql>,
Target: QueryFragment<Mysql>,
Action: QueryFragment<Mysql>,
NoWhereClause: QueryFragment<Mysql>,
{
fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, Mysql>) -> QueryResult<()> {
self.values.walk_ast(out.reborrow())?;
out.push_sql(" ON DUPLICATE KEY");
self.target.walk_ast(out.reborrow())?;
self.action.walk_ast(out.reborrow())?;
self.where_clause.walk_ast(out)?;
Ok(())
}
}
#[derive(Debug, Copy, Clone)]
pub struct DuplicatedKeys;
impl<Tab> OnConflictTarget<Tab> for ConflictTarget<DuplicatedKeys> {}
impl QueryFragment<Mysql, MysqlOnConflictClause> for ConflictTarget<DuplicatedKeys> {
fn walk_ast<'b>(&'b self, _out: AstPass<'_, 'b, Mysql>) -> QueryResult<()> {
Ok(())
}
}
impl<S> QueryFragment<crate::mysql::Mysql> for OnConflictSelectWrapper<S>
where
S: QueryFragment<crate::mysql::Mysql>,
{
fn walk_ast<'b>(&'b self, out: AstPass<'_, 'b, crate::mysql::Mysql>) -> QueryResult<()> {
self.0.walk_ast(out)
}
}