openthread/ot/types/
operational_dataset.rs1use crate::prelude_internal::*;
6use std::ops::Deref;
7
8#[derive(Default, Clone)]
10#[repr(transparent)]
11pub struct OperationalDataset(pub otOperationalDataset);
12
13impl_ot_castable!(OperationalDataset, otOperationalDataset);
14
15impl OperationalDataset {
16 pub fn empty() -> OperationalDataset {
18 Self::default()
19 }
20
21 pub fn is_complete(&self) -> bool {
23 self.get_active_timestamp().is_some()
24 && self.get_network_name().is_some()
25 && self.get_network_key().is_some()
26 && self.get_extended_pan_id().is_some()
27 && self.get_mesh_local_prefix().is_some()
28 && self.get_pan_id().is_some()
29 && self.get_channel().is_some()
30 && self.get_pskc().is_some()
31 && self.get_security_policy().is_some()
32 && self.get_channel_mask().is_some()
33 }
34
35 pub fn clear(&mut self) {
37 *self = Self::empty();
38 }
39
40 }
45
46impl std::fmt::Debug for OperationalDataset {
47 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
48 let mut ds = f.debug_struct("OperationalDataset");
49
50 if let Some(x) = self.get_network_name() {
51 ds.field("network_name", &x);
52 }
53 if let Some(x) = self.get_extended_pan_id() {
54 ds.field("xpanid", &x);
55 }
56 if let Some(x) = self.get_network_key() {
57 ds.field("network_key", &x);
58 }
59 if let Some(x) = self.get_mesh_local_prefix() {
60 ds.field("mesh_local_prefix", &x);
61 }
62 if let Some(x) = self.get_pan_id() {
63 ds.field("panid", &x);
64 }
65 if let Some(x) = self.get_channel() {
66 ds.field("channel", &x);
67 }
68 if let Some(x) = self.get_channel_mask() {
69 ds.field("channel_mask", &x);
70 }
71 if let Some(x) = self.get_pskc() {
72 ds.field("pskc", &x);
73 }
74 if let Some(x) = self.get_security_policy() {
75 ds.field("security_policy", &x);
76 }
77 if let Some(x) = self.get_delay() {
78 ds.field("delay", &x);
79 }
80 if let Some(x) = self.get_active_timestamp() {
81 ds.field("active_timestamp", &x);
82 }
83 if let Some(x) = self.get_pending_timestamp() {
84 ds.field("pending_timestamp", &x);
85 }
86 ds.finish()
87 }
88}
89
90impl OperationalDataset {
91 pub fn get_channel(&self) -> Option<ChannelIndex> {
93 self.0.mComponents.mIsChannelPresent.then(|| self.0.mChannel.try_into().unwrap())
94 }
95
96 pub fn get_channel_mask(&self) -> Option<ChannelMask> {
98 self.0.mComponents.mIsChannelMaskPresent.then(|| self.0.mChannelMask.into())
99 }
100
101 pub fn get_delay(&self) -> Option<u32> {
103 self.0.mComponents.mIsDelayPresent.then_some(self.0.mDelay)
104 }
105
106 pub fn get_extended_pan_id(&self) -> Option<&ExtendedPanId> {
108 self.0
109 .mComponents
110 .mIsExtendedPanIdPresent
111 .then(|| ExtendedPanId::ref_from_ot_ref(&self.0.mExtendedPanId))
112 }
113
114 pub fn get_network_key(&self) -> Option<&NetworkKey> {
116 self.0
117 .mComponents
118 .mIsNetworkKeyPresent
119 .then(|| NetworkKey::ref_from_ot_ref(&self.0.mNetworkKey))
120 }
121
122 pub fn get_pskc(&self) -> Option<&Pskc> {
124 self.0.mComponents.mIsPskcPresent.then(|| Pskc::ref_from_ot_ref(&self.0.mPskc))
125 }
126
127 pub fn get_network_name(&self) -> Option<&NetworkName> {
129 self.0
130 .mComponents
131 .mIsNetworkNamePresent
132 .then(|| NetworkName::ref_from_ot_ref(&self.0.mNetworkName))
133 }
134
135 pub fn get_pan_id(&self) -> Option<PanId> {
137 self.0.mComponents.mIsPanIdPresent.then_some(self.0.mPanId)
138 }
139
140 pub fn get_active_timestamp(&self) -> Option<Timestamp> {
142 self.0.mComponents.mIsActiveTimestampPresent.then(|| self.0.mActiveTimestamp.into())
143 }
144
145 pub fn get_pending_timestamp(&self) -> Option<Timestamp> {
147 self.0.mComponents.mIsPendingTimestampPresent.then(|| self.0.mPendingTimestamp.into())
148 }
149
150 pub fn get_security_policy(&self) -> Option<&SecurityPolicy> {
152 self.0
153 .mComponents
154 .mIsSecurityPolicyPresent
155 .then(|| SecurityPolicy::ref_from_ot_ref(&self.0.mSecurityPolicy))
156 }
157
158 pub fn get_mesh_local_prefix(&self) -> Option<&MeshLocalPrefix> {
160 self.0
161 .mComponents
162 .mIsMeshLocalPrefixPresent
163 .then_some(&self.0.mMeshLocalPrefix)
164 .map(Into::into)
165 }
166}
167
168impl OperationalDataset {
169 pub fn set_channel(&mut self, opt: Option<ChannelIndex>) {
171 if let Some(x) = opt {
172 self.0.mChannel = x.into();
173 self.0.mComponents.mIsChannelPresent = true;
174 } else {
175 self.0.mComponents.mIsChannelPresent = false;
176 }
177 }
178
179 pub fn set_channel_mask(&mut self, opt: Option<ChannelMask>) {
181 if let Some(x) = opt {
182 self.0.mChannelMask = x.into();
183 self.0.mComponents.mIsChannelMaskPresent = true;
184 } else {
185 self.0.mComponents.mIsChannelMaskPresent = false;
186 }
187 }
188
189 pub fn set_delay(&mut self, opt: Option<u32>) {
191 if let Some(x) = opt {
192 self.0.mDelay = x;
193 self.0.mComponents.mIsDelayPresent = true;
194 } else {
195 self.0.mComponents.mIsDelayPresent = false;
196 }
197 }
198
199 pub fn set_extended_pan_id(&mut self, opt: Option<&ExtendedPanId>) {
201 if let Some(x) = opt {
202 self.0.mExtendedPanId = *x.as_ot_ref();
203 self.0.mComponents.mIsExtendedPanIdPresent = true;
204 } else {
205 self.0.mComponents.mIsExtendedPanIdPresent = false;
206 }
207 }
208
209 pub fn set_network_key(&mut self, opt: Option<&NetworkKey>) {
211 if let Some(x) = opt {
212 self.0.mNetworkKey = *x.as_ot_ref();
213 self.0.mComponents.mIsNetworkKeyPresent = true;
214 } else {
215 self.0.mComponents.mIsNetworkKeyPresent = false;
216 }
217 }
218
219 pub fn set_network_name(&mut self, opt: Option<&NetworkName>) {
221 if let Some(x) = opt {
222 self.0.mNetworkName = *x.as_ot_ref();
223 self.0.mComponents.mIsNetworkNamePresent = true;
224 } else {
225 self.0.mComponents.mIsNetworkNamePresent = false;
226 }
227 }
228
229 pub fn set_pan_id(&mut self, opt: Option<PanId>) {
231 if let Some(x) = opt {
232 self.0.mPanId = x;
233 self.0.mComponents.mIsPanIdPresent = true;
234 } else {
235 self.0.mComponents.mIsPanIdPresent = false;
236 }
237 }
238
239 pub fn set_active_timestamp(&mut self, opt: Option<Timestamp>) {
241 if let Some(x) = opt {
242 self.0.mActiveTimestamp = x.into();
243 self.0.mComponents.mIsActiveTimestampPresent = true;
244 } else {
245 self.0.mComponents.mIsActiveTimestampPresent = false;
246 }
247 }
248
249 pub fn set_pending_timestamp(&mut self, opt: Option<Timestamp>) {
251 if let Some(x) = opt {
252 self.0.mPendingTimestamp = x.into();
253 self.0.mComponents.mIsPendingTimestampPresent = true;
254 } else {
255 self.0.mComponents.mIsPendingTimestampPresent = false;
256 }
257 }
258
259 pub fn set_security_policy(&mut self, opt: Option<SecurityPolicy>) {
261 if let Some(x) = opt {
262 self.0.mSecurityPolicy = *x.as_ot_ref();
263 self.0.mComponents.mIsSecurityPolicyPresent = true;
264 } else {
265 self.0.mComponents.mIsSecurityPolicyPresent = false;
266 }
267 }
268
269 pub fn set_mesh_local_prefix(&mut self, opt: Option<&MeshLocalPrefix>) {
271 if let Some(x) = opt {
272 self.0.mMeshLocalPrefix = *x.as_ot_ref();
273 self.0.mComponents.mIsMeshLocalPrefixPresent = true;
274 } else {
275 self.0.mComponents.mIsMeshLocalPrefixPresent = false;
276 }
277 }
278}
279
280#[derive(Debug, Default, Clone)]
282#[repr(transparent)]
283pub struct OperationalDatasetTlvs(pub otOperationalDatasetTlvs);
284
285impl_ot_castable!(OperationalDatasetTlvs, otOperationalDatasetTlvs);
286
287impl OperationalDatasetTlvs {
288 pub fn try_to_dataset(&self) -> Result<OperationalDataset> {
291 let mut ret = OperationalDataset::default();
292 Error::from(unsafe { otDatasetParseTlvs(self.as_ot_ptr(), ret.as_ot_mut_ptr()) })
293 .into_result()?;
294 Ok(ret)
295 }
296
297 pub fn try_from_slice(slice: &[u8]) -> Result<Self, ot::WrongSize> {
299 let mut ret = Self::default();
300 let len = slice.len();
301
302 if len > OT_OPERATIONAL_DATASET_MAX_LENGTH as usize {
303 return Err(ot::WrongSize);
304 }
305
306 ret.0.mLength = len.try_into().unwrap();
307
308 ret.0.mTlvs[0..len].clone_from_slice(slice);
309
310 Ok(ret)
311 }
312
313 #[allow(clippy::len_without_is_empty)]
315 pub fn len(&self) -> usize {
316 self.0.mLength as usize
317 }
318
319 pub fn as_slice(&self) -> &[u8] {
321 &self.0.mTlvs[0..self.len()]
322 }
323
324 pub fn to_vec(&self) -> Vec<u8> {
326 self.as_slice().to_vec()
327 }
328}
329
330impl Deref for OperationalDatasetTlvs {
331 type Target = [u8];
332
333 fn deref(&self) -> &Self::Target {
334 self.as_slice()
335 }
336}
337
338impl<'a> TryFrom<&'a [u8]> for OperationalDatasetTlvs {
339 type Error = ot::WrongSize;
340
341 fn try_from(value: &'a [u8]) -> Result<Self, Self::Error> {
342 OperationalDatasetTlvs::try_from_slice(value)
343 }
344}
345
346impl TryFrom<Vec<u8>> for OperationalDatasetTlvs {
347 type Error = ot::WrongSize;
348
349 fn try_from(value: Vec<u8>) -> Result<Self, Self::Error> {
350 OperationalDatasetTlvs::try_from_slice(&value)
351 }
352}
353
354impl From<OperationalDatasetTlvs> for Vec<u8> {
355 fn from(value: OperationalDatasetTlvs) -> Self {
356 value.to_vec()
357 }
358}
359
360impl TryFrom<OperationalDatasetTlvs> for OperationalDataset {
361 type Error = ot::Error;
362
363 fn try_from(value: OperationalDatasetTlvs) -> Result<Self, Self::Error> {
364 value.try_to_dataset()
365 }
366}
367
368#[cfg(test)]
369mod test {
370 use super::*;
371
372 #[test]
373 fn test_operational_dataset() {
374 let dataset_bytes = hex::decode("0e08000062cc8de70000000300001635060004001fffe0020830b02192978a444f0708fd70a9fb17d60000030d4e4553542d50414e2d3043454401020ced0410f73d3809ffd94b329fdab33ba781ba910c0402a0f778").unwrap();
375
376 let dataset_tlvs = OperationalDatasetTlvs::try_from_slice(&dataset_bytes).unwrap();
377
378 let dataset = dataset_tlvs.try_to_dataset().unwrap();
379
380 println!("dataset = {dataset:#?}");
381 }
382}