1use super::accepted_save_contract_for_descriptor;
8use crate::{
9 db::{
10 DbSession, PersistedRow, WriteBatchResponse,
11 data::{FieldSlot, StructuralPatch},
12 executor::MutationMode,
13 schema::{AcceptedFieldAbsencePolicy, AcceptedRowLayoutRuntimeContract},
14 },
15 error::InternalError,
16 traits::{CanisterKind, EntityCreateInput, EntityValue},
17 value::Value,
18};
19
20fn append_accepted_structural_patch_field(
25 entity_path: &'static str,
26 descriptor: &AcceptedRowLayoutRuntimeContract<'_>,
27 patch: StructuralPatch,
28 field_name: &str,
29 value: Value,
30) -> Result<StructuralPatch, InternalError> {
31 let slot = descriptor
32 .field_slot_index_by_name(field_name)
33 .ok_or_else(|| InternalError::mutation_structural_field_unknown(entity_path, field_name))?;
34
35 Ok(patch.set(FieldSlot::from_validated_index(slot), value))
36}
37
38fn validate_structural_patch_schema_policy<E>(
44 descriptor: &AcceptedRowLayoutRuntimeContract<'_>,
45 patch: &StructuralPatch,
46 mode: MutationMode,
47) -> Result<(), InternalError>
48where
49 E: PersistedRow + EntityValue,
50{
51 reject_explicit_generated_fields_from_accepted_patch::<E>(descriptor, patch)?;
52
53 if matches!(mode, MutationMode::Update) {
54 return Ok(());
55 }
56
57 let mut provided_slots = vec![false; descriptor.required_slot_count()];
58 for entry in patch.entries() {
59 let slot = entry.slot().index();
60 if slot < provided_slots.len() {
61 provided_slots[slot] = true;
62 }
63 }
64
65 for field in descriptor.fields() {
69 let slot = usize::from(field.slot().get());
70 if provided_slots.get(slot).copied().unwrap_or(false) {
71 continue;
72 }
73
74 if matches!(field.absence_policy(), AcceptedFieldAbsencePolicy::Required) {
75 return Err(
76 InternalError::mutation_structural_patch_required_field_missing(
77 E::PATH,
78 field.name(),
79 ),
80 );
81 }
82 }
83
84 Ok(())
85}
86
87fn reject_explicit_generated_fields_from_accepted_patch<E>(
93 descriptor: &AcceptedRowLayoutRuntimeContract<'_>,
94 patch: &StructuralPatch,
95) -> Result<(), InternalError>
96where
97 E: PersistedRow + EntityValue,
98{
99 for entry in patch.entries() {
100 let slot = entry.slot().index();
101 let Some(accepted_field) = descriptor.field_for_slot_index(slot) else {
102 continue;
103 };
104 let write_policy = accepted_field.write_policy();
105
106 if write_policy.insert_generation().is_some()
107 && !descriptor.is_primary_key_field_name(accepted_field.name())
108 {
109 return Err(InternalError::mutation_generated_field_explicit(
110 E::PATH,
111 accepted_field.name(),
112 ));
113 }
114 }
115
116 Ok(())
117}
118
119impl<C: CanisterKind> DbSession<C> {
120 pub fn insert<E>(&self, entity: E) -> Result<E, InternalError>
122 where
123 E: PersistedRow<Canister = C> + EntityValue,
124 {
125 self.execute_save_entity(|save| save.insert(entity))
126 }
127
128 pub fn create<I>(&self, input: I) -> Result<I::Entity, InternalError>
130 where
131 I: EntityCreateInput,
132 I::Entity: PersistedRow<Canister = C> + EntityValue,
133 {
134 self.execute_save_entity(|save| save.create(input))
135 }
136
137 pub fn insert_many_atomic<E>(
143 &self,
144 entities: impl IntoIterator<Item = E>,
145 ) -> Result<WriteBatchResponse<E>, InternalError>
146 where
147 E: PersistedRow<Canister = C> + EntityValue,
148 {
149 self.execute_save_batch(|save| save.insert_many_atomic(entities))
150 }
151
152 pub fn insert_many_non_atomic<E>(
156 &self,
157 entities: impl IntoIterator<Item = E>,
158 ) -> Result<WriteBatchResponse<E>, InternalError>
159 where
160 E: PersistedRow<Canister = C> + EntityValue,
161 {
162 self.execute_save_batch(|save| save.insert_many_non_atomic(entities))
163 }
164
165 pub fn replace<E>(&self, entity: E) -> Result<E, InternalError>
167 where
168 E: PersistedRow<Canister = C> + EntityValue,
169 {
170 self.execute_save_entity(|save| save.replace(entity))
171 }
172
173 pub fn mutate_structural<E>(
179 &self,
180 key: E::Key,
181 patch: StructuralPatch,
182 mode: MutationMode,
183 ) -> Result<E, InternalError>
184 where
185 E: PersistedRow<Canister = C> + EntityValue,
186 {
187 let accepted_schema = self.ensure_accepted_schema_snapshot::<E>()?;
188 let (descriptor, _) = AcceptedRowLayoutRuntimeContract::from_generated_compatible_schema(
189 &accepted_schema,
190 E::MODEL,
191 )?;
192 validate_structural_patch_schema_policy::<E>(&descriptor, &patch, mode)?;
193 let (
194 row_decode_contract,
195 mutation_row_decode_contract,
196 accepted_schema_info,
197 accepted_schema_fingerprint,
198 ) = accepted_save_contract_for_descriptor::<E>(&accepted_schema, &descriptor)?;
199
200 self.execute_save_with_checked_accepted_row_contract(
201 row_decode_contract,
202 accepted_schema_info,
203 accepted_schema_fingerprint,
204 |save| save.apply_structural_mutation(mode, key, patch, mutation_row_decode_contract),
205 std::convert::identity,
206 )
207 }
208
209 pub fn structural_patch<E, I, S>(&self, fields: I) -> Result<StructuralPatch, InternalError>
216 where
217 E: PersistedRow<Canister = C> + EntityValue,
218 I: IntoIterator<Item = (S, Value)>,
219 S: AsRef<str>,
220 {
221 let accepted_schema = self.ensure_accepted_schema_snapshot::<E>()?;
222 let (descriptor, _) = AcceptedRowLayoutRuntimeContract::from_generated_compatible_schema(
223 &accepted_schema,
224 E::MODEL,
225 )?;
226 let mut patch = StructuralPatch::new();
227
228 for (field_name, value) in fields {
232 let field_name = field_name.as_ref();
233 patch = append_accepted_structural_patch_field(
234 E::PATH,
235 &descriptor,
236 patch,
237 field_name,
238 value,
239 )?;
240 }
241
242 Ok(patch)
243 }
244
245 #[cfg(test)]
251 pub(in crate::db) fn replace_structural<E>(
252 &self,
253 key: E::Key,
254 patch: StructuralPatch,
255 ) -> Result<E, InternalError>
256 where
257 E: PersistedRow<Canister = C> + EntityValue,
258 {
259 self.mutate_structural(key, patch, MutationMode::Replace)
260 }
261
262 pub fn replace_many_atomic<E>(
268 &self,
269 entities: impl IntoIterator<Item = E>,
270 ) -> Result<WriteBatchResponse<E>, InternalError>
271 where
272 E: PersistedRow<Canister = C> + EntityValue,
273 {
274 self.execute_save_batch(|save| save.replace_many_atomic(entities))
275 }
276
277 pub fn replace_many_non_atomic<E>(
281 &self,
282 entities: impl IntoIterator<Item = E>,
283 ) -> Result<WriteBatchResponse<E>, InternalError>
284 where
285 E: PersistedRow<Canister = C> + EntityValue,
286 {
287 self.execute_save_batch(|save| save.replace_many_non_atomic(entities))
288 }
289
290 pub fn update<E>(&self, entity: E) -> Result<E, InternalError>
292 where
293 E: PersistedRow<Canister = C> + EntityValue,
294 {
295 self.execute_save_entity(|save| save.update(entity))
296 }
297
298 #[cfg(test)]
304 pub(in crate::db) fn insert_structural<E>(
305 &self,
306 key: E::Key,
307 patch: StructuralPatch,
308 ) -> Result<E, InternalError>
309 where
310 E: PersistedRow<Canister = C> + EntityValue,
311 {
312 self.mutate_structural(key, patch, MutationMode::Insert)
313 }
314
315 #[cfg(test)]
321 pub(in crate::db) fn update_structural<E>(
322 &self,
323 key: E::Key,
324 patch: StructuralPatch,
325 ) -> Result<E, InternalError>
326 where
327 E: PersistedRow<Canister = C> + EntityValue,
328 {
329 self.mutate_structural(key, patch, MutationMode::Update)
330 }
331
332 pub fn update_many_atomic<E>(
338 &self,
339 entities: impl IntoIterator<Item = E>,
340 ) -> Result<WriteBatchResponse<E>, InternalError>
341 where
342 E: PersistedRow<Canister = C> + EntityValue,
343 {
344 self.execute_save_batch(|save| save.update_many_atomic(entities))
345 }
346
347 pub fn update_many_non_atomic<E>(
351 &self,
352 entities: impl IntoIterator<Item = E>,
353 ) -> Result<WriteBatchResponse<E>, InternalError>
354 where
355 E: PersistedRow<Canister = C> + EntityValue,
356 {
357 self.execute_save_batch(|save| save.update_many_non_atomic(entities))
358 }
359}