new.rs 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. use std::{
  2. collections::{HashMap, HashSet},
  3. hash::Hash,
  4. marker::PhantomData,
  5. sync::Mutex,
  6. };
  7. use crate::{
  8. db::{StatementContext, StatementRow},
  9. schema::{
  10. datum::{ConcreteDatum, Datum, DatumList},
  11. entity::{Entity, EntityID, EntityPart, EntityPartList, EntityPartVisitor},
  12. relation::RelationData,
  13. DatabaseItemVisitor, Schema,
  14. },
  15. DBResult,
  16. };
  17. // ----------------------------------------------------------------------
  18. // Entity-related New<> adapter
  19. // ----------------------------------------------------------------------
  20. pub struct NewID<OE: Entity> {
  21. value: OE::ID,
  22. }
  23. impl<OE: Entity> std::fmt::Debug for NewID<OE> {
  24. fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
  25. f.write_fmt(format_args!("NewID({})", self.value.into_raw()))
  26. }
  27. }
  28. impl<OE: Entity> Clone for NewID<OE> {
  29. fn clone(&self) -> Self {
  30. Self {
  31. value: self.value.clone(),
  32. }
  33. }
  34. }
  35. impl<OE: Entity> Copy for NewID<OE> {}
  36. impl<OE: Entity> Hash for NewID<OE> {
  37. fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
  38. self.value.hash(state)
  39. }
  40. }
  41. impl<OE: Entity> PartialEq for NewID<OE> {
  42. fn eq(&self, other: &Self) -> bool {
  43. self.value.eq(&other.value)
  44. }
  45. }
  46. impl<OE: Entity> PartialOrd for NewID<OE> {
  47. fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
  48. self.value.partial_cmp(&other.value)
  49. }
  50. }
  51. impl<OE: Entity> EntityID for NewID<OE> {
  52. type Entity = New<OE>;
  53. fn from_raw(id: i64) -> Self {
  54. Self {
  55. value: OE::ID::from_raw(id),
  56. }
  57. }
  58. fn into_raw(self) -> i64 {
  59. self.value.into_raw()
  60. }
  61. }
  62. impl<OE: Entity> Datum for NewID<OE> {
  63. fn sql_type() -> &'static str {
  64. OE::ID::sql_type()
  65. }
  66. fn bind_to(&self, stmt: &mut StatementContext, index: i32) {
  67. self.value.bind_to(stmt, index)
  68. }
  69. fn build_from(adata: RelationData, stmt: &mut StatementRow, index: &mut i32) -> DBResult<Self>
  70. where
  71. Self: Sized,
  72. {
  73. OE::ID::build_from(adata, stmt, index).map(|value| Self { value })
  74. }
  75. }
  76. impl<OE: Entity> ConcreteDatum for NewID<OE> {}
  77. pub struct NewIDPart<OE: Entity> {
  78. _ghost: std::marker::PhantomData<&'static OE>,
  79. }
  80. impl<OE: Entity> Default for NewIDPart<OE> {
  81. fn default() -> Self {
  82. Self {
  83. _ghost: std::marker::PhantomData,
  84. }
  85. }
  86. }
  87. impl<OE: Entity> Clone for NewIDPart<OE> {
  88. fn clone(&self) -> Self {
  89. Self::default()
  90. }
  91. }
  92. impl<OE: Entity> EntityPart for NewIDPart<OE> {
  93. type Entity = New<OE>;
  94. type Datum = NewID<OE>;
  95. fn desc() -> Option<&'static str> {
  96. None
  97. }
  98. fn unique() -> bool {
  99. false
  100. }
  101. fn part_name() -> &'static str {
  102. "id"
  103. }
  104. fn get_datum(_from: &Self::Entity) -> &Self::Datum {
  105. unreachable!()
  106. }
  107. }
  108. pub struct NewPart<OE: Entity, OP: EntityPart<Entity = OE>> {
  109. _ghost: std::marker::PhantomData<&'static (OE, OP)>,
  110. }
  111. impl<OE: Entity, OP: EntityPart<Entity = OE>> Default for NewPart<OE, OP> {
  112. fn default() -> Self {
  113. Self {
  114. _ghost: std::marker::PhantomData,
  115. }
  116. }
  117. }
  118. impl<OE: Entity, OP: EntityPart<Entity = OE>> Clone for NewPart<OE, OP> {
  119. fn clone(&self) -> Self {
  120. Self::default()
  121. }
  122. }
  123. impl<OE: Entity, OP: EntityPart<Entity = OE>> EntityPart for NewPart<OE, OP> {
  124. type Entity = New<OE>;
  125. type Datum = OP::Datum;
  126. fn desc() -> Option<&'static str> {
  127. OP::desc()
  128. }
  129. fn unique() -> bool {
  130. OP::unique()
  131. }
  132. fn part_name() -> &'static str {
  133. OP::part_name()
  134. }
  135. fn get_datum(from: &Self::Entity) -> &Self::Datum {
  136. OP::get_datum(&from.value)
  137. }
  138. }
  139. pub struct NewPartList<OE: Entity, OPL: EntityPartList> {
  140. _ghost: std::marker::PhantomData<&'static (OE, OPL)>,
  141. }
  142. impl<OE: Entity, OPL: EntityPartList<Entity = OE>> EntityPartList for NewPartList<OE, OPL> {
  143. type Entity = New<OE>;
  144. type ListHead = NewPart<OE, OPL::ListHead>;
  145. type ListTail = NewPartList<OE, OPL::ListTail>;
  146. type DatumList = OPL::DatumList;
  147. const IS_EMPTY: bool = OPL::IS_EMPTY;
  148. fn build_datum_list(stmt: &mut StatementRow) -> DBResult<Self::DatumList> {
  149. OPL::build_datum_list(stmt)
  150. }
  151. fn accept_part_visitor(visitor: &mut impl EntityPartVisitor<Entity = Self::Entity>) {
  152. if Self::IS_EMPTY {
  153. return;
  154. }
  155. Self::ListHead::accept_part_visitor(visitor);
  156. Self::ListTail::accept_part_visitor(visitor);
  157. }
  158. fn accept_part_visitor_ref(
  159. datum_list: &Self::DatumList,
  160. visitor: &mut impl EntityPartVisitor<Entity = Self::Entity>,
  161. ) {
  162. if Self::IS_EMPTY {
  163. return;
  164. }
  165. visitor.visit_datum::<Self::ListHead>(datum_list.list_head());
  166. let tail = datum_list.list_tail();
  167. Self::ListTail::accept_part_visitor_ref(&tail, visitor);
  168. }
  169. }
  170. #[derive(Debug)]
  171. pub struct New<E: Entity> {
  172. value: E,
  173. }
  174. lazy_static::lazy_static! {
  175. static ref NEWNAMES : Mutex<HashMap<std::any::TypeId, &'static str>> = {
  176. Default::default()
  177. };
  178. }
  179. impl<E: Entity> Entity for New<E> {
  180. type ID = NewID<E>;
  181. type Keys = NewPartList<E, E::Keys>;
  182. type Parts = NewPartList<E, E::Parts>;
  183. type IDPart = NewIDPart<E>;
  184. fn entity_name() -> &'static str {
  185. let mut nn = NEWNAMES.lock().unwrap();
  186. let e = nn.entry(std::any::TypeId::of::<Self>());
  187. e.or_insert_with(|| Box::leak(format!("NEW_{}", E::entity_name()).into_boxed_str()))
  188. }
  189. fn build(values: <Self::Parts as EntityPartList>::DatumList) -> Self {
  190. Self {
  191. value: E::build(values),
  192. }
  193. }
  194. fn accept_part_visitor(visitor: &mut impl EntityPartVisitor<Entity = Self>) {
  195. Self::Parts::accept_part_visitor(visitor)
  196. }
  197. fn accept_part_visitor_ref(&self, visitor: &mut impl EntityPartVisitor<Entity = Self>) {
  198. struct PassthroughVisitor<'l, E: Entity, EPV: EntityPartVisitor<Entity = New<E>>>(
  199. &'l mut EPV,
  200. );
  201. impl<'l, E: Entity, EPV: EntityPartVisitor<Entity = New<E>>> EntityPartVisitor
  202. for PassthroughVisitor<'l, E, EPV>
  203. {
  204. type Entity = E;
  205. fn visit_datum<EP: EntityPart<Entity = Self::Entity>>(&mut self, datum: &EP::Datum) {
  206. self.0.visit_datum::<NewPart<E, EP>>(datum);
  207. }
  208. }
  209. let mut pv = PassthroughVisitor(visitor);
  210. self.value.accept_part_visitor_ref(&mut pv);
  211. }
  212. fn accept_part_visitor_mut(&mut self, visitor: &mut impl EntityPartVisitor<Entity = Self>) {
  213. struct PassthroughVisitor<'l, E: Entity, EPV: EntityPartVisitor<Entity = New<E>>>(
  214. &'l mut EPV,
  215. );
  216. impl<'l, E: Entity, EPV: EntityPartVisitor<Entity = New<E>>> EntityPartVisitor
  217. for PassthroughVisitor<'l, E, EPV>
  218. {
  219. type Entity = E;
  220. fn visit_datum_mut<EP: EntityPart<Entity = Self::Entity>>(
  221. &mut self,
  222. datum: &mut EP::Datum,
  223. ) {
  224. self.0.visit_datum_mut::<NewPart<E, EP>>(datum);
  225. }
  226. }
  227. let mut pv = PassthroughVisitor(visitor);
  228. self.value.accept_part_visitor_mut(&mut pv);
  229. }
  230. }
  231. // ----------------------------------------------------------------------
  232. // Schema-level adaptation
  233. // ----------------------------------------------------------------------
  234. pub struct WrapSchema<S: Schema> {
  235. base: S,
  236. towrap: HashSet<String>,
  237. }
  238. impl<S: Schema> WrapSchema<S> {
  239. pub fn new(base: S, towrap: HashSet<String>) -> Self {
  240. Self { base, towrap }
  241. }
  242. }
  243. impl<S: Schema> Schema for WrapSchema<S> {
  244. fn accept_item_visitor(&self, visitor: &mut impl DatabaseItemVisitor)
  245. where
  246. Self: Sized,
  247. {
  248. struct InterceptingVisitor<'a, S: Schema, V: DatabaseItemVisitor>(
  249. &'a WrapSchema<S>,
  250. &'a mut V,
  251. );
  252. impl<'a, S: Schema, V: DatabaseItemVisitor> DatabaseItemVisitor for InterceptingVisitor<'a, S, V> {
  253. fn visit_idmap<T: Entity>(&mut self)
  254. where
  255. Self: Sized,
  256. {
  257. }
  258. fn visit_index<T: Entity, PL: EntityPartList<Entity = T>>(&mut self)
  259. where
  260. Self: Sized,
  261. {
  262. }
  263. }
  264. let mut iv = InterceptingVisitor(self, visitor);
  265. self.base.accept_item_visitor(&mut iv);
  266. }
  267. }