kanidmd_lib/valueset/
datetime.rs1use crate::prelude::*;
2use crate::schema::SchemaAttribute;
3use crate::valueset::{
4 DbValueSetV2, ScimResolveStatus, ValueSet, ValueSetResolveStatus, ValueSetScimPut,
5};
6use kanidm_proto::scim_v1::{client::ScimDateTime, JsonValue};
7use smolset::SmolSet;
8use time::OffsetDateTime;
9
10#[derive(Debug, Clone)]
11pub struct ValueSetDateTime {
12 set: SmolSet<[OffsetDateTime; 1]>,
13}
14
15impl ValueSetDateTime {
16 pub fn new(b: OffsetDateTime) -> Box<Self> {
17 let mut set = SmolSet::new();
18 set.insert(b);
19 Box::new(ValueSetDateTime { set })
20 }
21
22 pub fn push(&mut self, b: OffsetDateTime) -> bool {
23 self.set.insert(b)
24 }
25
26 pub fn from_dbvs2(data: Vec<String>) -> Result<ValueSet, OperationError> {
27 let set = data
28 .into_iter()
29 .map(|s| {
30 OffsetDateTime::parse(&s, &Rfc3339)
31 .map(|odt| odt.to_offset(time::UtcOffset::UTC))
32 .map_err(|_| OperationError::InvalidValueState)
33 })
34 .collect::<Result<_, _>>()?;
35 Ok(Box::new(ValueSetDateTime { set }))
36 }
37
38 #[allow(clippy::should_implement_trait)]
41 pub fn from_iter<T>(iter: T) -> Option<Box<Self>>
42 where
43 T: IntoIterator<Item = OffsetDateTime>,
44 {
45 let set = iter.into_iter().collect();
46 Some(Box::new(ValueSetDateTime { set }))
47 }
48}
49
50impl ValueSetScimPut for ValueSetDateTime {
51 fn from_scim_json_put(value: JsonValue) -> Result<ValueSetResolveStatus, OperationError> {
52 let ScimDateTime { date_time } = serde_json::from_value(value).map_err(|err| {
53 error!(?err, "SCIM DateTime syntax invalid");
54 OperationError::SC0010DateTimeSyntaxInvalid
55 })?;
56
57 let mut set = SmolSet::new();
58 set.insert(date_time);
59
60 Ok(ValueSetResolveStatus::Resolved(Box::new(
61 ValueSetDateTime { set },
62 )))
63 }
64}
65
66impl ValueSetT for ValueSetDateTime {
67 fn insert_checked(&mut self, value: Value) -> Result<bool, OperationError> {
68 match value {
69 Value::DateTime(u) => Ok(self.set.insert(u)),
70 _ => {
71 debug_assert!(false);
72 Err(OperationError::InvalidValueState)
73 }
74 }
75 }
76
77 fn clear(&mut self) {
78 self.set.clear();
79 }
80
81 fn remove(&mut self, pv: &PartialValue, _cid: &Cid) -> bool {
82 match pv {
83 PartialValue::DateTime(u) => self.set.remove(u),
84 _ => false,
85 }
86 }
87
88 fn contains(&self, pv: &PartialValue) -> bool {
89 match pv {
90 PartialValue::DateTime(u) => self.set.contains(u),
91 _ => false,
92 }
93 }
94
95 fn substring(&self, _pv: &PartialValue) -> bool {
96 false
97 }
98
99 fn startswith(&self, _pv: &PartialValue) -> bool {
100 false
101 }
102
103 fn endswith(&self, _pv: &PartialValue) -> bool {
104 false
105 }
106
107 fn lessthan(&self, pv: &PartialValue) -> bool {
108 match pv {
109 PartialValue::DateTime(u) => self.set.iter().all(|set_value| set_value < u),
110 _ => false,
111 }
112 }
113
114 fn len(&self) -> usize {
115 self.set.len()
116 }
117
118 fn generate_idx_eq_keys(&self) -> Vec<String> {
119 self.set
120 .iter()
121 .map(|odt| {
122 debug_assert_eq!(odt.offset(), time::UtcOffset::UTC);
123 #[allow(clippy::expect_used)]
124 odt.format(&Rfc3339)
125 .expect("Failed to format timestamp into RFC3339")
126 })
127 .collect()
128 }
129
130 fn syntax(&self) -> SyntaxType {
131 SyntaxType::DateTime
132 }
133
134 fn validate(&self, _schema_attr: &SchemaAttribute) -> bool {
135 true
136 }
137
138 fn to_proto_string_clone_iter(&self) -> Box<dyn Iterator<Item = String> + '_> {
139 Box::new(self.set.iter().map(|odt| {
140 debug_assert_eq!(odt.offset(), time::UtcOffset::UTC);
141 #[allow(clippy::expect_used)]
142 odt.format(&Rfc3339)
143 .expect("Failed to format timestamp into RFC3339")
144 }))
145 }
146
147 fn to_scim_value(&self) -> Option<ScimResolveStatus> {
148 self.set.iter().next().copied().map(|v| v.into())
149 }
150
151 fn to_db_valueset_v2(&self) -> DbValueSetV2 {
152 DbValueSetV2::DateTime(
153 self.set
154 .iter()
155 .map(|odt| {
156 debug_assert_eq!(odt.offset(), time::UtcOffset::UTC);
157 #[allow(clippy::expect_used)]
158 odt.format(&Rfc3339)
159 .expect("Failed to format timestamp into RFC3339")
160 })
161 .collect(),
162 )
163 }
164
165 fn to_partialvalue_iter(&self) -> Box<dyn Iterator<Item = PartialValue> + '_> {
166 Box::new(self.set.iter().cloned().map(PartialValue::DateTime))
167 }
168
169 fn to_value_iter(&self) -> Box<dyn Iterator<Item = Value> + '_> {
170 Box::new(self.set.iter().cloned().map(Value::DateTime))
171 }
172
173 fn equal(&self, other: &ValueSet) -> bool {
174 if let Some(other) = other.as_datetime_set() {
175 &self.set == other
176 } else {
177 debug_assert!(false);
178 false
179 }
180 }
181
182 fn merge(&mut self, other: &ValueSet) -> Result<(), OperationError> {
183 if let Some(b) = other.as_datetime_set() {
184 mergesets!(self.set, b)
185 } else {
186 debug_assert!(false);
187 Err(OperationError::InvalidValueState)
188 }
189 }
190
191 fn to_datetime_single(&self) -> Option<OffsetDateTime> {
192 if self.set.len() == 1 {
193 self.set.iter().cloned().take(1).next()
194 } else {
195 None
196 }
197 }
198
199 fn as_datetime_set(&self) -> Option<&SmolSet<[OffsetDateTime; 1]>> {
200 Some(&self.set)
201 }
202}
203
204#[cfg(test)]
205mod tests {
206 use super::ValueSetDateTime;
207 use crate::prelude::ValueSet;
208 use std::time::Duration;
209 use time::OffsetDateTime;
210
211 #[test]
212 fn test_scim_datetime() {
213 let odt = OffsetDateTime::UNIX_EPOCH + Duration::from_secs(69_420);
214 let vs: ValueSet = ValueSetDateTime::new(odt);
215
216 crate::valueset::scim_json_reflexive(&vs, r#""1970-01-01T19:17:00Z""#);
217
218 crate::valueset::scim_json_put_reflexive::<ValueSetDateTime>(&vs, &[])
220 }
221}