1use std::{marker::PhantomData, rc::Rc};
2
3use crate::{
4 CustomJoin, Expr, IntoExpr, Table, TableRow,
5 joinable::IntoJoinable,
6 lower,
7 private::Joinable,
8 value::{DbTyp, EqTyp},
9};
10
11pub struct Rows<'inner, S> {
19 pub(crate) phantom: PhantomData<fn(&'inner ()) -> &'inner ()>,
21 pub(crate) _p: PhantomData<S>,
22 pub(crate) ast: Rc<lower::Rows>,
23}
24
25impl<'inner, S> Rows<'inner, S> {
26 pub fn join<T: DbTyp>(
37 &mut self,
38 j: impl IntoJoinable<'inner, S, Typ = T>,
39 ) -> Expr<'inner, S, T> {
40 let joinable = j.into_joinable();
41
42 let join = Rc::make_mut(&mut self.ast).join(joinable.table.clone());
43 for (name, val) in joinable.conds {
44 let expr = Rc::new(lower::Expr::RowIndex(
46 lower::RowLike::Join(join.clone()),
47 name,
48 ));
49 self.filter(Expr::adhoc(lower::Expr::Infix(expr, "=", val)));
50 }
51
52 Expr::new_inner(
53 Rc::new(lower::Expr::RowIndex(
54 lower::RowLike::Join(join),
55 joinable.main_column,
56 )),
57 false, )
59 }
60
61 #[doc(hidden)]
62 pub fn join_private<T: Table<Schema = S>>(&mut self) -> Expr<'inner, S, TableRow<T>> {
63 self.join(Joinable::table())
64 }
65
66 pub(crate) fn join_custom<T: CustomJoin<Schema = S>>(
67 &mut self,
68 t: T,
69 ) -> Expr<'inner, S, TableRow<T>> {
70 self.join(Joinable::new(t.name(), t.main_column()))
71 }
72
73 pub(crate) fn join_tmp<T: Table<Schema = S>>(
74 &mut self,
75 tmp: lower::TmpTable,
76 ) -> Expr<'inner, S, TableRow<T>> {
77 self.join(Joinable::new(lower::JoinableTable::Tmp(tmp), T::ID))
78 }
79
80 pub fn filter(&mut self, prop: impl IntoExpr<'inner, S, Typ = bool>) {
82 Rc::make_mut(&mut self.ast).filter(prop.into_expr().inner);
83 }
84
85 pub fn filter_some<Typ: EqTyp>(
89 &mut self,
90 val: impl IntoExpr<'inner, S, Typ = Option<Typ>>,
91 ) -> Expr<'inner, S, Typ> {
92 let val = val.into_expr();
93 Rc::make_mut(&mut self.ast).filter(val.inner.clone());
94
95 Expr::new_inner(val.inner, false)
97 }
98}