entity.rs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. use std::{fmt::Debug, hash::Hash};
  2. use crate::{db::StatementRow, schema::datum::Datum, DBResult};
  3. use super::datum::{ConcreteDatum, ConcreteDatumList, DatumList, QueryEquivalentList};
  4. pub(crate) mod helpers;
  5. /// Integral identifier for an entity.
  6. pub trait EntityID: 'static + PartialEq + Hash + PartialOrd + Debug + Copy + ConcreteDatum {
  7. /// The entity that this ID is representing.
  8. type Entity: Entity<ID = Self>;
  9. /// Construct from a raw integer.
  10. ///
  11. /// Do not use this unless you are _very_ certain it is what you want.
  12. fn from_raw(id: i64) -> Self;
  13. /// Produces a raw integer from an ID. Of dubious use.
  14. fn into_raw(self) -> i64;
  15. }
  16. // ----------------------------------------------------------------------
  17. // EntityPart and related types
  18. // ----------------------------------------------------------------------
  19. /// A single data field in an Entity, automatically declared and derived as part of `#[derive(Entity)]`.
  20. pub trait EntityPart: Default + Clone + 'static {
  21. /// The type of the data field.
  22. type Datum: ConcreteDatum;
  23. /// The type of the struct this field is a member of.
  24. type Entity: Entity;
  25. /// String representing the field name.
  26. fn part_name() -> &'static str;
  27. /// Whether this field has a `#[unique]` attribute or not.
  28. fn unique() -> bool;
  29. /// Documentation comment for this field.
  30. fn desc() -> Option<&'static str>;
  31. /// Accessor function: given an instance of the outer struct, return this field's value in that instance.
  32. fn get_datum(from: &Self::Entity) -> &Self::Datum;
  33. }
  34. /// Visitor for traversing all [`EntityPart`]s in an [`Entity`] or [`EntityPartList`].
  35. pub trait EntityPartVisitor {
  36. /// The [`Entity`] that this visitor will traverse over the parts of.
  37. type Entity: Entity;
  38. /// Visit a part, with no instance information.
  39. fn visit<EP: EntityPart<Entity = Self::Entity>>(&mut self) {}
  40. /// Visit a part, with a datum reference.
  41. fn visit_datum<EP: EntityPart<Entity = Self::Entity>>(&mut self, _datum: &EP::Datum) {}
  42. /// Visit a part, with a mutable datum reference.
  43. fn visit_datum_mut<EP: EntityPart<Entity = Self::Entity>>(&mut self, _datum: &mut EP::Datum) {}
  44. }
  45. /// List of EntityParts.
  46. pub trait EntityPartList: 'static {
  47. /// The [`Entity`] that this list represents the parts of.
  48. type Entity: Entity;
  49. /// A [`ConcreteDatumList`] type that matches this part list, formed as the `EntityPart::Datum`s of
  50. /// each `EntityPart` in this list.
  51. type DatumList: ConcreteDatumList + QueryEquivalentList<Self::DatumList>;
  52. // + DatumList<ListTail = <Self::ListTail as EntityPartList>::DatumList>;
  53. /// The first element of this list. If this list is empty, this will be a generic marker type
  54. /// with poisoned values to be easily visible if processed.
  55. type ListHead: EntityPart<
  56. Entity = Self::Entity,
  57. Datum = <Self::DatumList as DatumList>::ListHead,
  58. >;
  59. /// An [`EntityPartList`] that represents all but the first element of this list.
  60. type ListTail: EntityPartList<
  61. Entity = Self::Entity,
  62. DatumList = <Self::DatumList as DatumList>::ListTail,
  63. >;
  64. /// Whether the current list is empty. Should be used to control recursion if iterating across
  65. /// the list.
  66. const IS_EMPTY: bool = false;
  67. /// Construct an instance of `Self::DatumList` from a table row.
  68. fn build_datum_list(stmt: &mut StatementRow) -> DBResult<Self::DatumList>;
  69. /// Accept a visitor to iterate across each part in the list, with no datum instances.
  70. fn accept_part_visitor(_: &mut impl EntityPartVisitor<Entity = Self::Entity>);
  71. /// Accept a visitor to iterate across each part in the list, with datum instances provided by
  72. /// a `Self::DatumList` instance.
  73. fn accept_part_visitor_ref(
  74. datum_list: &Self::DatumList,
  75. _: &mut impl EntityPartVisitor<Entity = Self::Entity>,
  76. );
  77. }
  78. // trait implementations for EntityPartList
  79. mod part_list;
  80. /// A sentinel type representing a zero-length [`EntityPartList`].
  81. pub use part_list::EmptyList;
  82. // ----------------------------------------------------------------------
  83. // Entity and related types
  84. // ----------------------------------------------------------------------
  85. /// A single database entity, aka an object type that gets its own table.
  86. pub trait Entity: 'static + std::fmt::Debug {
  87. /// A list of all the [`EntityPart`]s that make up the fields of the struct.
  88. type Parts: EntityPartList<Entity = Self>;
  89. /// A list of the [`EntityPart`]s that make up the list of access keys for the entity. Can be an
  90. /// [`EmptyList`] if no keys are defined.
  91. type Keys: EntityPartList<Entity = Self>;
  92. /// The corresponding [`EntityID`] type for this Entity.
  93. type ID: EntityID<Entity = Self>;
  94. /// The [`EntityPart`] type for `Self::ID`.
  95. type IDPart: EntityPart<Datum = Self::ID, Entity = Self>;
  96. /// Construct an instance of this entity from a list of datums that matches each part, in
  97. /// order.
  98. fn build(values: <Self::Parts as EntityPartList>::DatumList) -> Self;
  99. /// A static string used as the database table name for the entity.
  100. fn entity_name() -> &'static str;
  101. /// Accept a visitor to iterate across each part in the entity, with no datum instances.
  102. fn accept_part_visitor(visitor: &mut impl EntityPartVisitor<Entity = Self>);
  103. /// Accept a visitor to iterate across each part in the entity, with datum instances pulled
  104. /// from an instance of `Self`.
  105. fn accept_part_visitor_ref(&self, visitor: &mut impl EntityPartVisitor<Entity = Self>);
  106. /// Accept a visitor to iterate across each part in the entity, with mutable datum instances
  107. /// pulled from an instance of `Self`.
  108. fn accept_part_visitor_mut(&mut self, _: &mut impl EntityPartVisitor<Entity = Self>);
  109. }
  110. /// Visitor for traversing all [`Entity`]s in a container type.
  111. pub trait EntityVisitor {
  112. /// Visit a particular entity.
  113. fn visit<E: Entity>(&mut self);
  114. }
  115. // ----------------------------------------------------------------------
  116. // EntityList and related types
  117. // ----------------------------------------------------------------------
  118. /// A list of [`Entity`] types.
  119. pub trait EntityList {
  120. const LEN: usize;
  121. fn accept_visitor<EV: EntityVisitor>(visitor: &mut EV);
  122. }