kanidmd_lib/valueset/
datetime.rs
1use 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 false
109 }
110
111 fn len(&self) -> usize {
112 self.set.len()
113 }
114
115 fn generate_idx_eq_keys(&self) -> Vec<String> {
116 self.set
117 .iter()
118 .map(|odt| {
119 debug_assert_eq!(odt.offset(), time::UtcOffset::UTC);
120 #[allow(clippy::expect_used)]
121 odt.format(&Rfc3339)
122 .expect("Failed to format timestamp into RFC3339")
123 })
124 .collect()
125 }
126
127 fn syntax(&self) -> SyntaxType {
128 SyntaxType::DateTime
129 }
130
131 fn validate(&self, _schema_attr: &SchemaAttribute) -> bool {
132 true
133 }
134
135 fn to_proto_string_clone_iter(&self) -> Box<dyn Iterator<Item = String> + '_> {
136 Box::new(self.set.iter().map(|odt| {
137 debug_assert_eq!(odt.offset(), time::UtcOffset::UTC);
138 #[allow(clippy::expect_used)]
139 odt.format(&Rfc3339)
140 .expect("Failed to format timestamp into RFC3339")
141 }))
142 }
143
144 fn to_scim_value(&self) -> Option<ScimResolveStatus> {
145 self.set.iter().next().copied().map(|v| v.into())
146 }
147
148 fn to_db_valueset_v2(&self) -> DbValueSetV2 {
149 DbValueSetV2::DateTime(
150 self.set
151 .iter()
152 .map(|odt| {
153 debug_assert_eq!(odt.offset(), time::UtcOffset::UTC);
154 #[allow(clippy::expect_used)]
155 odt.format(&Rfc3339)
156 .expect("Failed to format timestamp into RFC3339")
157 })
158 .collect(),
159 )
160 }
161
162 fn to_partialvalue_iter(&self) -> Box<dyn Iterator<Item = PartialValue> + '_> {
163 Box::new(self.set.iter().cloned().map(PartialValue::DateTime))
164 }
165
166 fn to_value_iter(&self) -> Box<dyn Iterator<Item = Value> + '_> {
167 Box::new(self.set.iter().cloned().map(Value::DateTime))
168 }
169
170 fn equal(&self, other: &ValueSet) -> bool {
171 if let Some(other) = other.as_datetime_set() {
172 &self.set == other
173 } else {
174 debug_assert!(false);
175 false
176 }
177 }
178
179 fn merge(&mut self, other: &ValueSet) -> Result<(), OperationError> {
180 if let Some(b) = other.as_datetime_set() {
181 mergesets!(self.set, b)
182 } else {
183 debug_assert!(false);
184 Err(OperationError::InvalidValueState)
185 }
186 }
187
188 fn to_datetime_single(&self) -> Option<OffsetDateTime> {
189 if self.set.len() == 1 {
190 self.set.iter().cloned().take(1).next()
191 } else {
192 None
193 }
194 }
195
196 fn as_datetime_set(&self) -> Option<&SmolSet<[OffsetDateTime; 1]>> {
197 Some(&self.set)
198 }
199}
200
201#[cfg(test)]
202mod tests {
203 use super::ValueSetDateTime;
204 use crate::prelude::ValueSet;
205 use std::time::Duration;
206 use time::OffsetDateTime;
207
208 #[test]
209 fn test_scim_datetime() {
210 let odt = OffsetDateTime::UNIX_EPOCH + Duration::from_secs(69_420);
211 let vs: ValueSet = ValueSetDateTime::new(odt);
212
213 crate::valueset::scim_json_reflexive(&vs, r#""1970-01-01T19:17:00Z""#);
214
215 crate::valueset::scim_json_put_reflexive::<ValueSetDateTime>(&vs, &[])
217 }
218}