tc_scalar/reference/
after.rs1use std::collections::HashSet;
4use std::fmt;
5
6use async_hash::{Digest, Hash, Output};
7use async_trait::async_trait;
8use destream::{de, en};
9use get_size::GetSize;
10use get_size_derive::*;
11use log::debug;
12use safecast::{Match, TryCastFrom, TryCastInto};
13
14use tc_error::*;
15use tc_transact::public::{Public, StateInstance, ToState};
16use tc_value::Value;
17use tcgeneric::{Id, Instance, Map, PathSegment, TCPathBuf};
18
19use crate::{OpDef, Scalar, Scope};
20
21use super::Refer;
22
23#[derive(Clone, Eq, PartialEq, GetSize)]
25pub struct After {
26 when: Scalar,
27 then: Scalar,
28}
29
30#[async_trait]
31impl<State> Refer<State> for After
32where
33 State: StateInstance + Refer<State> + From<Scalar>,
34 State::Closure: From<(Map<State>, OpDef)> + TryCastFrom<State>,
35 Map<State>: TryFrom<State, Error = TCError>,
36 Value: TryFrom<State, Error = TCError> + TryCastFrom<State>,
37 bool: TryCastFrom<State>,
38{
39 fn dereference_self(self, path: &TCPathBuf) -> Self {
40 Self {
41 when: self.when.dereference_self(path),
42 then: self.then.dereference_self(path),
43 }
44 }
45
46 fn is_conditional(&self) -> bool {
47 self.then.is_conditional()
48 }
49
50 fn is_inter_service_write(&self, cluster_path: &[PathSegment]) -> bool {
51 self.when.is_inter_service_write(cluster_path)
52 || self.then.is_inter_service_write(cluster_path)
53 }
54
55 fn is_ref(&self) -> bool {
56 true
57 }
58
59 fn reference_self(self, path: &TCPathBuf) -> Self {
60 Self {
61 when: self.when.reference_self(path),
62 then: self.then.reference_self(path),
63 }
64 }
65
66 fn requires(&self, deps: &mut HashSet<Id>) {
67 self.when.requires(deps);
68 self.then.requires(deps);
69 }
70
71 async fn resolve<'a, T: ToState<State> + Public<State> + Instance>(
72 self,
73 context: &'a Scope<'a, State, T>,
74 txn: &'a State::Txn,
75 ) -> TCResult<State> {
76 debug!("After::resolve {:?} from context ()", self);
77 if self.when.is_conditional() {
78 return Err(bad_request!(
79 "After does not allow a conditional clause {:?}",
80 self.when,
81 ));
82 }
83
84 self.when.resolve(context, txn).await?;
85 self.then.resolve(context, txn).await
86 }
87}
88
89impl<'a, D: Digest> Hash<D> for &'a After {
90 fn hash(self) -> Output<D> {
91 Hash::<D>::hash((&self.when, &self.then))
92 }
93}
94
95impl TryCastFrom<Scalar> for After {
96 fn can_cast_from(scalar: &Scalar) -> bool {
97 scalar.matches::<(Scalar, Scalar)>()
98 }
99
100 fn opt_cast_from(scalar: Scalar) -> Option<Self> {
101 scalar
102 .opt_cast_into()
103 .map(|(when, then)| Self { when, then })
104 }
105}
106
107#[async_trait]
108impl de::FromStream for After {
109 type Context = ();
110
111 async fn from_stream<D: de::Decoder>(context: (), decoder: &mut D) -> Result<Self, D::Error> {
112 let (when, then) =
113 <(Scalar, Scalar) as de::FromStream>::from_stream(context, decoder).await?;
114
115 Ok(Self { when, then })
116 }
117}
118
119impl<'en> en::IntoStream<'en> for After {
120 fn into_stream<E: en::Encoder<'en>>(self, encoder: E) -> Result<E::Ok, E::Error> {
121 (self.when, self.then).into_stream(encoder)
122 }
123}
124
125impl<'en> en::ToStream<'en> for After {
126 fn to_stream<E: en::Encoder<'en>>(&'en self, encoder: E) -> Result<E::Ok, E::Error> {
127 en::IntoStream::into_stream((&self.when, &self.then), encoder)
128 }
129}
130
131impl fmt::Debug for After {
132 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
133 write!(f, "after {:?} then {:?}", self.when, self.then)
134 }
135}