1use crate::{EdifactError, Segment};
9use std::borrow::Cow;
10use std::io::Read;
11use std::str::FromStr;
12
13pub trait EdifactDeserialize: Sized {
20 fn edifact_deserialize(segments: &[Segment<'_>]) -> Result<Self, EdifactError>;
25
26 fn edifact_deserialize_owned(segments: &[crate::OwnedSegment]) -> Result<Self, EdifactError> {
45 let borrowed: Vec<Segment<'_>> = segments.iter().map(|s| s.as_borrowed()).collect();
46 Self::edifact_deserialize(&borrowed)
47 }
48}
49
50pub trait EdifactCompositeDeserialize: Sized {
55 fn edifact_deserialize_composite(composite: CompositeElement<'_>)
57 -> Result<Self, EdifactError>;
58}
59
60impl EdifactCompositeDeserialize for Vec<String> {
61 fn edifact_deserialize_composite(
62 composite: CompositeElement<'_>,
63 ) -> Result<Self, EdifactError> {
64 Ok(composite.iter().map(str::to_owned).collect())
65 }
66}
67
68pub trait EdifactSegmentTag {
73 const SEGMENT_TAG: &'static str;
75
76 const QUALIFIER_PATTERN: Option<&'static str> = None;
82
83 fn matches_qualifier(seg: &Segment<'_>) -> bool {
85 match Self::QUALIFIER_PATTERN {
86 Some(pattern) => seg
87 .element_str(0)
88 .is_some_and(|q| qualifier_matches_pattern(q, pattern)),
89 None => true,
90 }
91 }
92
93 fn matches_segment(seg: &Segment<'_>) -> bool {
98 seg.tag == Self::SEGMENT_TAG && Self::matches_qualifier(seg)
99 }
100
101 fn matches_owned_segment(seg: &crate::OwnedSegment) -> bool {
105 if seg.tag != Self::SEGMENT_TAG {
106 return false;
107 }
108 match Self::QUALIFIER_PATTERN {
109 None => true,
110 Some(pattern) => {
111 let q = seg
112 .elements
113 .first()
114 .and_then(|e| e.components.first())
115 .map(|(c, _)| c.as_str())
116 .unwrap_or("");
117 qualifier_matches_pattern(q, pattern)
118 }
119 }
120 }
121}
122
123impl<T> EdifactDeserialize for Vec<T>
128where
129 T: EdifactDeserialize + EdifactSegmentTag,
130{
131 fn edifact_deserialize(segments: &[Segment<'_>]) -> Result<Self, EdifactError> {
132 segments
133 .iter()
134 .filter(|s| T::matches_segment(s))
135 .map(|seg| T::edifact_deserialize(std::slice::from_ref(seg)))
136 .collect()
137 }
138
139 fn edifact_deserialize_owned(segments: &[crate::OwnedSegment]) -> Result<Self, EdifactError> {
140 segments
141 .iter()
142 .filter(|s| T::matches_owned_segment(s))
143 .map(|seg| T::edifact_deserialize_owned(std::slice::from_ref(seg)))
144 .collect()
145 }
146}
147
148pub fn deserialize<T: EdifactDeserialize>(input: &[u8]) -> Result<T, EdifactError> {
164 let segments: Vec<Segment<'_>> = crate::from_bytes(input).collect::<Result<_, _>>()?;
165 T::edifact_deserialize(&segments)
166}
167
168pub fn deserialize_first_streaming<T>(input: &[u8]) -> Result<T, EdifactError>
173where
174 T: EdifactDeserialize + EdifactSegmentTag,
175{
176 for segment in crate::from_bytes(input) {
177 let segment = segment?;
178 if T::matches_segment(&segment) {
179 return T::edifact_deserialize(std::slice::from_ref(&segment));
180 }
181 }
182
183 Err(EdifactError::MissingSegment {
184 tag: T::SEGMENT_TAG.to_owned(),
185 expected_position: "any position in input".to_owned(),
186 })
187}
188
189pub fn deserialize_all_streaming<T>(input: &[u8]) -> Result<Vec<T>, EdifactError>
193where
194 T: EdifactDeserialize + EdifactSegmentTag,
195{
196 let mut out = Vec::new();
197 for segment in crate::from_bytes(input) {
198 let segment = segment?;
199 if T::matches_segment(&segment) {
200 out.push(T::edifact_deserialize(std::slice::from_ref(&segment))?);
201 }
202 }
203 Ok(out)
204}
205
206pub fn deserialize_first_from_reader<T, R>(reader: R) -> Result<T, EdifactError>
210where
211 T: EdifactDeserialize + EdifactSegmentTag,
212 R: Read,
213{
214 for segment in crate::from_reader_iter(reader) {
215 let segment = segment?;
216 if !T::matches_owned_segment(&segment) {
218 continue;
219 }
220 return T::edifact_deserialize_owned(std::slice::from_ref(&segment));
221 }
222
223 Err(EdifactError::MissingSegment {
224 tag: T::SEGMENT_TAG.to_owned(),
225 expected_position: "any position in input".to_owned(),
226 })
227}
228
229pub fn deserialize_all_from_reader<T, R>(reader: R) -> Result<Vec<T>, EdifactError>
231where
232 T: EdifactDeserialize + EdifactSegmentTag,
233 R: Read,
234{
235 let mut out = Vec::new();
236 for segment in crate::from_reader_iter(reader) {
237 let segment = segment?;
238 if !T::matches_owned_segment(&segment) {
240 continue;
241 }
242 out.push(T::edifact_deserialize_owned(std::slice::from_ref(
243 &segment,
244 ))?);
245 }
246 Ok(out)
247}
248
249pub fn deserialize_str<T: EdifactDeserialize>(input: &str) -> Result<T, EdifactError> {
251 deserialize(input.as_bytes())
252}
253
254pub fn find_segment<'s, 'd>(segments: &'s [Segment<'d>], tag: &str) -> Option<&'s Segment<'d>> {
258 segments.iter().find(|s| s.tag == tag)
259}
260
261pub fn find_segments_iter<'s, 'd: 's>(
263 segments: &'s [Segment<'d>],
264 tag: &'s str,
265) -> impl Iterator<Item = &'s Segment<'d>> {
266 segments.iter().filter(move |s| s.tag == tag)
267}
268
269pub fn find_qualified_segment<'s, 'd>(
271 segments: &'s [Segment<'d>],
272 tag: &str,
273 qualifier: &str,
274) -> Option<&'s Segment<'d>> {
275 segments
276 .iter()
277 .find(|s| s.tag == tag && s.element_str(0).unwrap_or("") == qualifier)
278}
279
280pub fn find_segment_typed<'s, 'd, T>(segments: &'s [Segment<'d>]) -> Option<&'s Segment<'d>>
282where
283 T: EdifactSegmentTag,
284{
285 segments.iter().find(|s| T::matches_segment(s))
286}
287
288pub fn find_segments_typed<'s, 'd: 's, T>(
290 segments: &'s [Segment<'d>],
291) -> impl Iterator<Item = &'s Segment<'d>>
292where
293 T: EdifactSegmentTag,
294{
295 segments.iter().filter(|s| T::matches_segment(s))
296}
297
298pub fn contiguous_groups_by_qualifier<'s, 'd, T>(
303 segments: &'s [Segment<'d>],
304) -> Vec<&'s [Segment<'d>]>
305where
306 T: EdifactSegmentTag,
307{
308 let mut groups = Vec::new();
309 let mut idx = 0;
310 while idx < segments.len() {
311 if T::matches_segment(&segments[idx]) {
312 let start = idx;
313 idx += 1;
314 while idx < segments.len() && T::matches_segment(&segments[idx]) {
315 idx += 1;
316 }
317 groups.push(&segments[start..idx]);
318 } else {
319 idx += 1;
320 }
321 }
322 groups
323}
324
325pub fn contiguous_groups_iter<'s, 'd, T>(
342 segments: &'s [Segment<'d>],
343) -> impl Iterator<Item = &'s [Segment<'d>]> + 's
344where
345 T: EdifactSegmentTag,
346{
347 let mut idx = 0;
348 let len = segments.len();
349 std::iter::from_fn(move || {
350 while idx < len && !T::matches_segment(&segments[idx]) {
352 idx += 1;
353 }
354 if idx >= len {
355 return None;
356 }
357 let start = idx;
358 idx += 1;
359 while idx < len && T::matches_segment(&segments[idx]) {
360 idx += 1;
361 }
362 Some(&segments[start..idx])
363 })
364}
365
366pub fn groups_are_contiguous_by_qualifier<T>(segments: &[Segment<'_>]) -> bool
368where
369 T: EdifactSegmentTag,
370{
371 let mut seen_match = false;
372 let mut seen_gap_after_match = false;
373
374 for seg in segments {
375 if T::matches_segment(seg) {
376 if seen_gap_after_match {
377 return false;
378 }
379 seen_match = true;
380 } else if seen_match {
381 seen_gap_after_match = true;
382 }
383 }
384
385 true
386}
387
388pub fn qualifier_matches_pattern(value: &str, pattern: &str) -> bool {
400 if pattern.is_empty() {
401 return value.is_empty();
402 }
403
404 if !pattern.contains('*') {
405 return value == pattern;
406 }
407
408 if let Some((prefix, suffix)) = pattern.split_once('*') {
410 if !pattern[prefix.len() + 1..].contains('*') {
412 return value.len() >= prefix.len() + suffix.len()
413 && value.starts_with(prefix)
414 && value.ends_with(suffix)
415 && {
416 let mid_start = prefix.len();
418 let mid_end = value.len().saturating_sub(suffix.len());
419 mid_start <= mid_end
420 };
421 }
422 }
423
424 let parts: smallvec::SmallVec<[&str; 4]> = pattern.split('*').collect();
426
427 if parts.len() > 4 {
432 return false;
433 }
434
435 let prefix = parts[0];
436 let suffix = parts[parts.len() - 1];
437
438 if !value.starts_with(prefix) || !value.ends_with(suffix) {
439 return false;
440 }
441
442 let mid_start = prefix.len();
443 let mid_end = value.len().saturating_sub(suffix.len());
444
445 if mid_start > mid_end {
446 return parts[1..parts.len() - 1].iter().all(|p| p.is_empty());
447 }
448
449 let mut remaining = &value[mid_start..mid_end];
450
451 for part in &parts[1..parts.len() - 1] {
452 if part.is_empty() {
453 continue;
454 }
455 match remaining.find(part) {
456 Some(idx) => remaining = &remaining[idx + part.len()..],
457 None => return false,
458 }
459 }
460
461 true
462}
463
464#[inline]
466pub fn element_str<'s>(seg: &'s Segment<'_>, idx: usize) -> &'s str {
467 seg.element_str(idx).unwrap_or("")
468}
469
470pub fn required_element<'a>(seg: &'a Segment<'_>, idx: usize) -> Result<&'a str, EdifactError> {
486 seg.text_element(idx)
487}
488
489pub fn optional_element<'a>(seg: &'a Segment<'_>, idx: usize) -> Option<&'a str> {
495 SegmentAccessor::optional_element(seg, idx)
496}
497
498pub fn required_component<'a>(
516 seg: &'a Segment<'_>,
517 elem_idx: usize,
518 comp_idx: usize,
519) -> Result<&'a str, EdifactError> {
520 seg.required_composite(elem_idx, comp_idx)
521}
522
523pub fn optional_component<'a>(
529 seg: &'a Segment<'_>,
530 elem_idx: usize,
531 comp_idx: usize,
532) -> Option<&'a str> {
533 SegmentAccessor::get_component(seg, elem_idx, comp_idx)
534}
535
536pub fn get_components_iter<'a>(seg: &'a Segment<'_>, idx: usize) -> impl Iterator<Item = &'a str> {
540 seg.elements
541 .get(idx)
542 .into_iter()
543 .flat_map(|elem| elem.components.iter().map(|(c, _)| c.as_ref()))
544}
545
546pub struct CompositeElement<'a> {
556 components: smallvec::SmallVec<[&'a str; 4]>,
557}
558
559impl<'a> CompositeElement<'a> {
560 pub fn from_slice(components: &'a [std::borrow::Cow<'a, str>]) -> Self {
564 Self {
565 components: components.iter().map(|c| c.as_ref()).collect(),
566 }
567 }
568
569 pub(crate) fn from_strs(components: smallvec::SmallVec<[&'a str; 4]>) -> Self {
571 Self { components }
572 }
573
574 pub fn get(&self, i: usize) -> Option<&'a str> {
576 self.components.get(i).copied()
577 }
578
579 pub fn get_or_empty(&self, i: usize) -> &'a str {
581 self.get(i).unwrap_or("")
582 }
583
584 pub fn len(&self) -> usize {
586 self.components.len()
587 }
588
589 pub fn is_empty(&self) -> bool {
591 self.components.is_empty()
592 }
593
594 pub fn iter(&self) -> impl Iterator<Item = &'a str> + '_ {
596 self.components.iter().copied()
597 }
598}
599
600pub fn composite_element<'a, 'd: 'a>(
602 seg: &'a Segment<'d>,
603 idx: usize,
604) -> Option<CompositeElement<'a>> {
605 seg.elements.get(idx).map(|elem| {
608 CompositeElement::from_strs(elem.components.iter().map(|(c, _)| c.as_ref()).collect())
609 })
610}
611
612pub fn find_segment_owned<'s>(
619 segments: &'s [crate::OwnedSegment],
620 tag: &str,
621) -> Option<&'s crate::OwnedSegment> {
622 segments.iter().find(|s| s.tag == tag)
623}
624
625pub fn find_qualified_segment_owned<'s>(
633 segments: &'s [crate::OwnedSegment],
634 tag: &str,
635 qualifier: &str,
636) -> Option<&'s crate::OwnedSegment> {
637 segments
638 .iter()
639 .find(|s| s.tag == tag && s.element_str(0).unwrap_or("") == qualifier)
640}
641
642pub trait SegmentAccessor<'a> {
644 fn get_element(&'a self, idx: usize) -> Option<&'a str>;
646 fn get_component(&'a self, elem: usize, comp: usize) -> Option<&'a str>;
648 fn get_composite(&'a self, idx: usize) -> Option<CompositeElement<'a>>;
650
651 fn text_element(&'a self, idx: usize) -> Result<&'a str, EdifactError>;
653 fn optional_element(&'a self, idx: usize) -> Option<&'a str>;
655 fn code_element<T: FromStr>(&'a self, idx: usize) -> Result<T, EdifactError>;
657 fn required_composite(&'a self, elem: usize, comp: usize) -> Result<&'a str, EdifactError>;
659 fn repeating_components(
665 &'a self,
666 elem: usize,
667 start_idx: usize,
668 count: usize,
669 ) -> Result<Vec<&'a str>, EdifactError> {
670 self.repeating_components_iter(elem, start_idx, count)
673 .collect()
674 }
675
676 fn repeating_components_iter(
681 &'a self,
682 elem: usize,
683 start_idx: usize,
684 count: usize,
685 ) -> impl Iterator<Item = Result<&'a str, EdifactError>> + 'a;
686}
687
688impl<'s, 'd> SegmentAccessor<'s> for Segment<'d>
689where
690 'd: 's,
691{
692 fn get_element(&'s self, idx: usize) -> Option<&'s str> {
693 self.element_str(idx).filter(|s| !s.is_empty())
694 }
695
696 fn get_component(&'s self, elem: usize, comp: usize) -> Option<&'s str> {
697 self.elements
698 .get(elem)
699 .and_then(|e| e.get_component(comp))
700 .filter(|s| !s.is_empty())
701 }
702
703 fn get_composite(&'s self, idx: usize) -> Option<CompositeElement<'s>> {
704 composite_element(self, idx)
705 }
706
707 fn text_element(&'s self, idx: usize) -> Result<&'s str, EdifactError> {
708 <Self as SegmentAccessor>::get_element(self, idx).ok_or_else(|| {
709 EdifactError::MissingRequiredElement {
710 tag: self.tag.to_owned(),
711 element_index: idx,
712 }
713 })
714 }
715
716 fn optional_element(&'s self, idx: usize) -> Option<&'s str> {
717 <Self as SegmentAccessor>::get_element(self, idx)
718 }
719
720 fn code_element<T: FromStr>(&'s self, idx: usize) -> Result<T, EdifactError> {
721 let raw = self.text_element(idx)?;
722 raw.parse::<T>().map_err(|_| EdifactError::InvalidText {
723 offset: self
724 .element_span(idx)
725 .map(|s| s.start)
726 .unwrap_or(self.span.start),
727 })
728 }
729
730 fn required_composite(&'s self, elem: usize, comp: usize) -> Result<&'s str, EdifactError> {
731 match self.elements.get(elem) {
732 None => Err(EdifactError::MissingRequiredElement {
733 tag: self.tag.to_owned(),
734 element_index: elem,
735 }),
736 Some(e) => e
737 .get_component(comp)
738 .filter(|s| !s.is_empty())
739 .ok_or_else(|| EdifactError::MissingRequiredComponent {
740 tag: self.tag.to_owned(),
741 element_index: elem,
742 component_index: comp,
743 }),
744 }
745 }
746
747 fn repeating_components_iter(
748 &'s self,
749 elem: usize,
750 start_idx: usize,
751 count: usize,
752 ) -> impl Iterator<Item = Result<&'s str, EdifactError>> + 's {
753 let tag = self.tag;
754 let element_exists = self.elements.get(elem).is_some();
755 let components = self
756 .elements
757 .get(elem)
758 .map(|e| e.components.as_slice())
759 .unwrap_or(&[]);
760 (start_idx..start_idx + count).map(move |idx| {
761 components
762 .get(idx)
763 .map(|(c, _)| c.as_ref())
764 .filter(|s| !s.is_empty())
765 .ok_or_else(|| {
766 if element_exists {
767 EdifactError::MissingRequiredComponent {
768 tag: tag.to_owned(),
769 element_index: elem,
770 component_index: idx,
771 }
772 } else {
773 EdifactError::MissingRequiredElement {
774 tag: tag.to_owned(),
775 element_index: elem,
776 }
777 }
778 })
779 })
780 }
781}
782
783#[derive(Debug)]
805pub struct MessageWindow<'a> {
806 pub message_type: Option<Cow<'a, str>>,
811 pub association_code: Option<Cow<'a, str>>,
816 pub segments: Vec<crate::Segment<'a>>,
818}
819
820impl<'a> MessageWindow<'a> {
821 fn from_segments(segments: Vec<crate::Segment<'a>>) -> Self {
827 let message_type = segments
828 .first()
829 .filter(|s| s.tag == "UNH")
830 .and_then(|unh| unh_component(unh, 0));
831 let association_code = segments
832 .first()
833 .filter(|s| s.tag == "UNH")
834 .and_then(|unh| unh_component(unh, 4));
835 Self {
836 message_type,
837 association_code,
838 segments,
839 }
840 }
841}
842
843fn unh_component<'a, 'b>(seg: &'b crate::Segment<'a>, comp_idx: usize) -> Option<Cow<'a, str>>
851where
852 'a: 'b,
853{
854 seg.elements
855 .get(1)
856 .and_then(|e| e.components.get(comp_idx))
857 .and_then(|(c, _)| if c.is_empty() { None } else { Some(c.clone()) })
858}
859
860#[derive(Debug, Clone)]
869pub struct OwnedMessageWindow {
870 pub message_type: Option<String>,
872 pub association_code: Option<String>,
874 pub segments: Vec<crate::OwnedSegment>,
876}
877
878impl OwnedMessageWindow {
879 fn from_segments(segments: Vec<crate::OwnedSegment>) -> Self {
880 let unh = segments.first().filter(|s| s.tag == "UNH");
881 let message_type = unh
882 .and_then(|s| s.elements.get(1))
883 .and_then(|e| e.components.first())
884 .map(|(c, _)| c.as_str())
885 .filter(|s| !s.is_empty())
886 .map(str::to_owned);
887 let association_code = unh
888 .and_then(|s| s.elements.get(1))
889 .and_then(|e| e.components.get(4))
890 .map(|(c, _)| c.as_str())
891 .filter(|s| !s.is_empty())
892 .map(str::to_owned);
893 Self {
894 message_type,
895 association_code,
896 segments,
897 }
898 }
899}
900
901pub struct MessageWindowsSliceIter<'a> {
911 inner: crate::FromBytesIter<'a>,
912 buf: Vec<crate::Segment<'a>>,
913 in_message: bool,
914 done: bool,
915}
916
917impl<'a> MessageWindowsSliceIter<'a> {
918 fn new(inner: crate::FromBytesIter<'a>) -> Self {
919 Self {
920 inner,
921 buf: Vec::new(),
922 in_message: false,
923 done: false,
924 }
925 }
926}
927
928impl<'a> Iterator for MessageWindowsSliceIter<'a> {
929 type Item = Result<MessageWindow<'a>, EdifactError>;
930
931 fn next(&mut self) -> Option<Self::Item> {
932 if self.done {
933 return None;
934 }
935 loop {
936 let segment = match self.inner.next() {
937 Some(Ok(s)) => s,
938 Some(Err(e)) => {
939 self.done = true;
940 return Some(Err(e));
941 }
942 None => {
943 self.done = true;
944 if self.in_message && !self.buf.is_empty() {
945 self.in_message = false;
946 let offset = self.buf.last().map(|s| s.span.end).unwrap_or(0);
947 return Some(Err(EdifactError::UnexpectedEof { offset }));
948 }
949 return None;
950 }
951 };
952
953 match segment.tag {
954 "UNH" => {
955 if self.in_message {
956 self.buf.clear();
957 self.in_message = false;
958 self.done = true;
959 let offset = segment.span.start;
960 return Some(Err(EdifactError::InvalidSegmentForMessage {
961 tag: "UNH".to_owned(),
962 message_type: "ENVELOPE".to_owned(),
963 offset,
964 }));
965 }
966 self.buf.clear();
967 self.in_message = true;
968 self.buf.push(segment);
969 }
970 "UNT" if self.in_message => {
971 self.buf.push(segment);
972 self.in_message = false;
973 let segments = std::mem::take(&mut self.buf);
974 return Some(Ok(MessageWindow::from_segments(segments)));
975 }
976 _ if self.in_message => {
977 self.buf.push(segment);
978 }
979 _ => {
980 }
982 }
983 }
984 }
985}
986
987pub struct MessageWindowsIter<I> {
1008 inner: I,
1009 buf: Vec<crate::OwnedSegment>,
1010 in_message: bool,
1011 done: bool,
1014}
1015
1016impl<I: Iterator<Item = Result<crate::OwnedSegment, EdifactError>>> MessageWindowsIter<I> {
1017 pub fn new(inner: I) -> Self {
1019 Self {
1020 inner,
1021 buf: Vec::new(),
1022 in_message: false,
1023 done: false,
1024 }
1025 }
1026}
1027
1028impl<I: Iterator<Item = Result<crate::OwnedSegment, EdifactError>>> Iterator
1029 for MessageWindowsIter<I>
1030{
1031 type Item = Result<OwnedMessageWindow, EdifactError>;
1032
1033 fn next(&mut self) -> Option<Self::Item> {
1034 if self.done {
1035 return None;
1036 }
1037 loop {
1038 let segment = match self.inner.next() {
1039 Some(Ok(s)) => s,
1040 Some(Err(e)) => {
1041 self.done = true;
1042 return Some(Err(e));
1043 }
1044 None => {
1045 self.done = true;
1046 if self.in_message && !self.buf.is_empty() {
1049 self.in_message = false;
1050 let offset = self.buf.last().map(|s| s.span.end).unwrap_or(0);
1051 return Some(Err(EdifactError::UnexpectedEof { offset }));
1052 }
1053 return None;
1054 }
1055 };
1056
1057 match segment.tag.as_str() {
1058 "UNH" => {
1059 if self.in_message {
1060 self.buf.clear();
1062 self.in_message = false;
1063 self.done = true;
1064 let offset = segment.span.start;
1065 return Some(Err(EdifactError::InvalidSegmentForMessage {
1066 tag: "UNH".to_owned(),
1067 message_type: "ENVELOPE".to_owned(),
1068 offset,
1069 }));
1070 }
1071 self.buf.clear();
1072 self.in_message = true;
1073 self.buf.push(segment);
1074 }
1075 "UNT" if self.in_message => {
1076 self.buf.push(segment);
1077 self.in_message = false;
1078 let segments = std::mem::take(&mut self.buf);
1079 return Some(Ok(OwnedMessageWindow::from_segments(segments)));
1080 }
1081 _ if self.in_message => {
1082 self.buf.push(segment);
1083 }
1084 _ => {
1085 }
1087 }
1088 }
1089 }
1090}
1091
1092pub fn message_windows_bytes(input: &[u8]) -> MessageWindowsSliceIter<'_> {
1121 MessageWindowsSliceIter::new(crate::from_bytes(input))
1122}
1123
1124pub fn message_windows_from_reader<R: Read>(
1130 reader: R,
1131) -> MessageWindowsIter<crate::FromReaderIter<R>> {
1132 MessageWindowsIter::new(crate::from_reader_iter(reader))
1133}
1134
1135pub fn deserialize_messages_from_reader<T, R>(
1154 reader: R,
1155) -> impl Iterator<Item = Result<T, EdifactError>>
1156where
1157 T: EdifactDeserialize,
1158 R: Read,
1159{
1160 message_windows_from_reader(reader).map(|window| {
1161 let window = window?;
1162 T::edifact_deserialize_owned(&window.segments)
1163 })
1164}
1165
1166pub fn deserialize_messages_bytes<T>(
1168 input: &[u8],
1169) -> impl Iterator<Item = Result<T, EdifactError>> + '_
1170where
1171 T: EdifactDeserialize,
1172{
1173 message_windows_bytes(input).map(|window| {
1174 let window = window?;
1175 T::edifact_deserialize(&window.segments)
1176 })
1177}
1178
1179pub struct DispatchedMessage {
1183 pub message_type: String,
1185 value: Box<dyn std::any::Any + Send + Sync>,
1186}
1187
1188impl DispatchedMessage {
1189 pub fn downcast<T: std::any::Any + Send + Sync + 'static>(&self) -> Option<&T> {
1193 self.value.downcast_ref::<T>()
1194 }
1195}
1196
1197impl std::fmt::Debug for DispatchedMessage {
1198 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
1199 f.debug_struct("DispatchedMessage")
1200 .field("message_type", &self.message_type)
1201 .finish_non_exhaustive()
1202 }
1203}
1204
1205type DispatchHandlerFn = Box<
1206 dyn for<'a> Fn(&[Segment<'a>]) -> Result<Box<dyn std::any::Any + Send + Sync>, EdifactError>
1207 + Send
1208 + Sync,
1209>;
1210
1211type FallbackHandlerFn = Box<
1212 dyn for<'a> Fn(
1213 &[Segment<'a>],
1214 &str,
1215 ) -> Result<Box<dyn std::any::Any + Send + Sync>, EdifactError>
1216 + Send
1217 + Sync,
1218>;
1219
1220pub struct MessageDispatch {
1245 handlers: Vec<(String, DispatchHandlerFn)>,
1246 fallback: Option<FallbackHandlerFn>,
1247}
1248
1249impl Default for MessageDispatch {
1250 fn default() -> Self {
1251 Self::new()
1252 }
1253}
1254
1255impl MessageDispatch {
1256 pub fn new() -> Self {
1258 Self {
1259 handlers: Vec::new(),
1260 fallback: None,
1261 }
1262 }
1263
1264 pub fn on<T, F>(mut self, message_type: &str, handler: F) -> Self
1269 where
1270 T: std::any::Any + Send + Sync + 'static,
1271 F: for<'a> Fn(&[Segment<'a>]) -> Result<T, EdifactError> + Send + Sync + 'static,
1272 {
1273 let erased: DispatchHandlerFn = Box::new(move |segs| {
1274 let val = handler(segs)?;
1275 Ok(Box::new(val) as Box<dyn std::any::Any + Send + Sync>)
1276 });
1277 self.handlers.push((message_type.to_owned(), erased));
1278 self
1279 }
1280
1281 pub fn fallback<T, F>(mut self, handler: F) -> Self
1286 where
1287 T: std::any::Any + Send + Sync + 'static,
1288 F: for<'a> Fn(&[Segment<'a>], &str) -> Result<T, EdifactError> + Send + Sync + 'static,
1289 {
1290 let erased: FallbackHandlerFn = Box::new(move |segs, mt| {
1291 let val = handler(segs, mt)?;
1292 Ok(Box::new(val) as Box<dyn std::any::Any + Send + Sync>)
1293 });
1294 self.fallback = Some(erased);
1295 self
1296 }
1297
1298 pub fn dispatch(&self, window: &[Segment<'_>]) -> Result<DispatchedMessage, EdifactError> {
1303 let message_type = window
1304 .iter()
1305 .find(|s| s.tag == "UNH")
1306 .and_then(|unh| unh.get_element(1))
1307 .and_then(|e| e.get_component(0))
1308 .map(|s| s.to_owned())
1309 .ok_or_else(|| EdifactError::MissingSegment {
1310 tag: "UNH".to_owned(),
1311 expected_position: "first segment of message window".to_owned(),
1312 })?;
1313
1314 for (mt, handler) in &self.handlers {
1315 if *mt == message_type {
1316 let value = handler(window)?;
1317 return Ok(DispatchedMessage {
1318 message_type,
1319 value,
1320 });
1321 }
1322 }
1323
1324 if let Some(fallback) = &self.fallback {
1325 let value = fallback(window, &message_type)?;
1326 return Ok(DispatchedMessage {
1327 message_type,
1328 value,
1329 });
1330 }
1331
1332 Err(EdifactError::UnexpectedMessageType { message_type })
1333 }
1334
1335 pub fn dispatch_all_from_bytes<'a>(
1340 &'a self,
1341 input: &'a [u8],
1342 ) -> impl Iterator<Item = Result<DispatchedMessage, EdifactError>> + 'a {
1343 message_windows_bytes(input).map(move |window| {
1344 let window = window?;
1345 self.dispatch(&window.segments)
1346 })
1347 }
1348
1349 pub fn dispatch_all_from_reader<R: Read + 'static>(
1356 &self,
1357 reader: R,
1358 ) -> impl Iterator<Item = Result<DispatchedMessage, EdifactError>> + '_ {
1359 message_windows_from_reader(reader).map(|window| {
1360 let window = window?;
1361 let borrowed: Vec<Segment<'_>> =
1362 window.segments.iter().map(|s| s.as_borrowed()).collect();
1363 self.dispatch(&borrowed)
1364 })
1365 }
1366}
1367
1368#[cfg(test)]
1369mod tests {
1370 use super::*;
1371
1372 #[derive(Debug, PartialEq)]
1374 struct BgmSegment {
1375 doc_name_code: String,
1376 pruef_id: String,
1377 msg_function: Option<String>,
1378 }
1379
1380 impl EdifactSegmentTag for BgmSegment {
1381 const SEGMENT_TAG: &'static str = "BGM";
1382 }
1383
1384 struct NadM;
1385
1386 impl EdifactSegmentTag for NadM {
1387 const SEGMENT_TAG: &'static str = "NAD";
1388 const QUALIFIER_PATTERN: Option<&'static str> = Some("M*");
1389 }
1390
1391 struct NadWildcard;
1392
1393 impl EdifactSegmentTag for NadWildcard {
1394 const SEGMENT_TAG: &'static str = "NAD";
1395 const QUALIFIER_PATTERN: Option<&'static str> = Some("M*");
1396 }
1397
1398 impl EdifactDeserialize for BgmSegment {
1399 fn edifact_deserialize(segments: &[Segment<'_>]) -> Result<Self, EdifactError> {
1400 let seg = find_segment(segments, "BGM").ok_or_else(|| {
1401 EdifactError::MissingRequiredElement {
1402 tag: "BGM".to_owned(),
1403 element_index: 0,
1404 }
1405 })?;
1406 Ok(Self {
1407 doc_name_code: element_str(seg, 0).to_owned(),
1408 pruef_id: element_str(seg, 1).to_owned(),
1409 msg_function: seg
1410 .element_str(2)
1411 .filter(|s| !s.is_empty())
1412 .map(str::to_owned),
1413 })
1414 }
1415 }
1416
1417 #[test]
1418 fn deserialize_single_segment() {
1419 let input = b"BGM+E03+11042+9'";
1420 let bgm: BgmSegment = deserialize(input).unwrap();
1421 assert_eq!(bgm.doc_name_code, "E03");
1422 assert_eq!(bgm.pruef_id, "11042");
1423 assert_eq!(bgm.msg_function, Some("9".to_owned()));
1424 }
1425
1426 #[test]
1427 fn streaming_deserialize_first_from_bytes() {
1428 let input = b"UNH+1+ORDERS:D:11A:UN'BGM+E03+11042+9'UNT+3+1'";
1429 let bgm: BgmSegment = deserialize_first_streaming(input).unwrap();
1430 assert_eq!(bgm.pruef_id, "11042");
1431 }
1432
1433 #[test]
1434 fn streaming_deserialize_all_from_bytes() {
1435 let input = b"BGM+E03+11042+9'RFF+AA:1'BGM+E01+11043+9'";
1436 let bgms: Vec<BgmSegment> = deserialize_all_streaming(input).unwrap();
1437 assert_eq!(bgms.len(), 2);
1438 assert_eq!(bgms[0].pruef_id, "11042");
1439 assert_eq!(bgms[1].pruef_id, "11043");
1440 }
1441
1442 #[test]
1443 fn streaming_deserialize_first_from_reader() {
1444 let input =
1445 std::io::Cursor::new(b"UNH+1+ORDERS:D:11A:UN'BGM+E03+11042+9'UNT+3+1'".to_vec());
1446 let bgm: BgmSegment = deserialize_first_from_reader(input).unwrap();
1447 assert_eq!(bgm.pruef_id, "11042");
1448 }
1449
1450 #[test]
1451 fn streaming_deserialize_all_from_reader() {
1452 let input = std::io::Cursor::new(b"BGM+E03+11042+9'BGM+E01+11043+9'".to_vec());
1453 let bgms: Vec<BgmSegment> = deserialize_all_from_reader(input).unwrap();
1454 assert_eq!(bgms.len(), 2);
1455 assert_eq!(bgms[0].pruef_id, "11042");
1456 assert_eq!(bgms[1].pruef_id, "11043");
1457 }
1458
1459 #[test]
1460 fn missing_segment_returns_error() {
1461 let input = b"DTM+137:20230401:102'";
1462 let result: Result<BgmSegment, _> = deserialize(input);
1463 assert!(result.is_err());
1464 }
1465
1466 #[test]
1467 fn vec_collects_all_matching_segments() {
1468 let input = b"DTM+137:20230401:102'BGM+E03+11042+9'BGM+E01+11043+9'";
1469 let bgms: Vec<BgmSegment> = deserialize(input).unwrap();
1470 assert_eq!(bgms.len(), 2);
1471 assert_eq!(bgms[0].pruef_id, "11042");
1472 assert_eq!(bgms[1].pruef_id, "11043");
1473 }
1474
1475 #[test]
1476 fn find_qualified_segment_matches_qualifier() {
1477 let input = b"NAD+MS+9900001+293'NAD+MR+9900002+293'";
1478 let segments: Vec<Segment<'_>> =
1479 crate::from_bytes(input).collect::<Result<_, _>>().unwrap();
1480 let nad_ms = find_qualified_segment(&segments, "NAD", "MS");
1481 let nad_mr = find_qualified_segment(&segments, "NAD", "MR");
1482 assert!(nad_ms.is_some());
1483 assert!(nad_mr.is_some());
1484 assert_eq!(element_str(nad_ms.unwrap(), 0), "MS");
1485 assert_eq!(element_str(nad_mr.unwrap(), 0), "MR");
1486 }
1487
1488 #[test]
1489 fn round_trip_str_api() {
1490 let input = "BGM+E03+11042+9'";
1491 let bgm: BgmSegment = deserialize_str(input).unwrap();
1492 assert_eq!(bgm.pruef_id, "11042");
1493 }
1494
1495 #[test]
1496 fn required_element_extraction() {
1497 let input = b"BGM+E03+11042+9'";
1498 let segments: Vec<Segment<'_>> =
1499 crate::from_bytes(input).collect::<Result<_, _>>().unwrap();
1500 let seg = &segments[0];
1501
1502 assert_eq!(required_element(seg, 0).unwrap(), "E03");
1503 assert_eq!(required_element(seg, 1).unwrap(), "11042");
1504 assert!(required_element(seg, 5).is_err());
1506 }
1507
1508 #[test]
1509 fn optional_element_extraction() {
1510 let input = b"BGM+E03+11042+9'BGM+E01++absent'";
1511 let segments: Vec<Segment<'_>> =
1512 crate::from_bytes(input).collect::<Result<_, _>>().unwrap();
1513
1514 assert_eq!(optional_element(&segments[0], 0), Some("E03"));
1516 assert_eq!(optional_element(&segments[0], 1), Some("11042"));
1517 assert_eq!(optional_element(&segments[0], 5), None);
1518
1519 assert_eq!(optional_element(&segments[1], 1), None);
1521 }
1522
1523 #[test]
1524 fn component_extraction() {
1525 let input = b"UNB+UNOA:1+SENDER+RECEIVER+200101:0900+1'";
1526 let segments: Vec<Segment<'_>> =
1527 crate::from_bytes(input).collect::<Result<_, _>>().unwrap();
1528 let seg = &segments[0];
1529
1530 assert_eq!(required_component(seg, 0, 0).unwrap(), "UNOA");
1531 assert_eq!(required_component(seg, 0, 1).unwrap(), "1");
1532 assert!(required_component(seg, 0, 5).is_err());
1534 }
1535
1536 #[test]
1537 fn composite_element_helper() {
1538 let input = b"UNB+UNOA:1+SENDER+RECEIVER+200101:0900+1'";
1539 let segments: Vec<Segment<'_>> =
1540 crate::from_bytes(input).collect::<Result<_, _>>().unwrap();
1541 let seg = &segments[0];
1542
1543 let comp = composite_element(seg, 0).unwrap();
1544 assert_eq!(comp.len(), 2);
1545 assert_eq!(comp.get(0), Some("UNOA"));
1546 assert_eq!(comp.get(1), Some("1"));
1547 assert_eq!(comp.get(5), None);
1548 assert_eq!(comp.get_or_empty(5), "");
1549 }
1550
1551 #[test]
1552 fn get_all_components() {
1553 let input = b"UNB+UNOA:1+SENDER+RECEIVER+200101:0900+1'";
1555 let segments: Vec<Segment<'_>> =
1556 crate::from_bytes(input).collect::<Result<_, _>>().unwrap();
1557 let seg = &segments[0];
1558
1559 let comps: Vec<&str> = get_components_iter(seg, 0).collect(); assert!(!comps.is_empty(), "Expected components but got empty");
1561 assert_eq!(comps.len(), 2);
1562 assert_eq!(comps[0], "UNOA");
1563 assert_eq!(comps[1], "1");
1564 }
1565
1566 #[test]
1567 fn qualifier_pattern_matching_supports_exact_and_wildcard() {
1568 assert!(qualifier_matches_pattern("MS", "MS"));
1570 assert!(!qualifier_matches_pattern("MS", "M")); assert!(qualifier_matches_pattern("MS", "M*"));
1573 assert!(qualifier_matches_pattern("MRY", "M*Y"));
1574 assert!(!qualifier_matches_pattern("AB", "M*"));
1575 }
1576
1577 #[test]
1579 fn qualifier_matches_pattern_table() {
1580 let cases: &[(&str, &str, bool)] = &[
1582 ("", "", true), ("", "*", true), ("A", "", false), ("", "A", false), ("MS", "MS", true),
1589 ("BY", "BY", true),
1590 ("ms", "MS", false), ("MSX", "MS", false), ("M", "MS", false), ("MS", "M*", true),
1595 ("MULTI", "MUL*", true),
1596 ("AB", "M*", false),
1597 ("", "M*", false), ("MSG", "*G", true),
1600 ("G", "*G", true),
1601 ("MSG", "*X", false),
1602 ("", "*G", false),
1603 ("MRY", "M*Y", true),
1605 ("MAY", "M*Y", true),
1606 ("MY", "M*Y", true), ("MYY", "M*Y", true), ("MAYZ", "M*Y", false), ("AB", "M*Y", false),
1610 ("*", "*", true), ("anything", "*", true),
1613 ("", "*", true),
1614 ("ABCDE", "A*C*E", true),
1616 ("ACE", "A*C*E", true), ("AXCYE", "A*C*E", true),
1618 ("ABCDF", "A*C*E", false),
1619 ("AB", "A**B", true), ("AB", "A*B*C", false),
1623 ("XMS", "MS", false),
1625 ];
1626
1627 for (value, pattern, expected) in cases {
1628 let got = qualifier_matches_pattern(value, pattern);
1629 assert_eq!(
1630 got, *expected,
1631 "qualifier_matches_pattern({value:?}, {pattern:?}) expected {expected} but got {got}"
1632 );
1633 }
1634 }
1635
1636 #[test]
1637 fn typed_qualifier_helpers_work() {
1638 let input = b"NAD+MS+9900001+293'NAD+MR+9900002+293'";
1639 let segments: Vec<Segment<'_>> =
1640 crate::from_bytes(input).collect::<Result<_, _>>().unwrap();
1641
1642 let first = find_segment_typed::<NadM>(&segments).unwrap();
1643 assert_eq!(first.element_str(0), Some("MS"));
1644
1645 let all: Vec<_> = find_segments_typed::<NadWildcard>(&segments).collect();
1646 assert_eq!(all.len(), 2);
1647 }
1648
1649 #[test]
1650 fn segment_accessor_trait_methods_work() {
1651 let input = b"UNB+UNOA:1+SENDER+RECEIVER+200101:0900+1'";
1652 let segments: Vec<Segment<'_>> =
1653 crate::from_bytes(input).collect::<Result<_, _>>().unwrap();
1654 let seg = &segments[0];
1655
1656 assert_eq!(SegmentAccessor::get_element(seg, 1), Some("SENDER"));
1657 assert_eq!(SegmentAccessor::required_composite(seg, 0, 1).unwrap(), "1");
1658 let parsed: i32 = SegmentAccessor::code_element(seg, 4).unwrap();
1659 assert_eq!(parsed, 1);
1660 let reps = SegmentAccessor::repeating_components(seg, 3, 0, 2).unwrap();
1661 assert_eq!(reps, vec!["200101", "0900"]);
1662 }
1663
1664 #[test]
1665 fn group_helpers_detect_contiguity() {
1666 struct NadAny;
1667 impl EdifactSegmentTag for NadAny {
1668 const SEGMENT_TAG: &'static str = "NAD";
1669 }
1670
1671 let contiguous_input = b"NAD+MS+1'NAD+MR+2'RFF+AA:1'";
1672 let contiguous_segments: Vec<Segment<'_>> = crate::from_bytes(contiguous_input)
1673 .collect::<Result<_, _>>()
1674 .unwrap();
1675 assert!(groups_are_contiguous_by_qualifier::<NadAny>(
1676 &contiguous_segments
1677 ));
1678
1679 let non_contiguous_input = b"NAD+MS+1'RFF+AA:1'NAD+MR+2'";
1680 let non_contiguous_segments: Vec<Segment<'_>> = crate::from_bytes(non_contiguous_input)
1681 .collect::<Result<_, _>>()
1682 .unwrap();
1683 assert!(!groups_are_contiguous_by_qualifier::<NadAny>(
1684 &non_contiguous_segments
1685 ));
1686 }
1687
1688 #[test]
1689 fn group_helpers_collect_contiguous_groups() {
1690 struct NadAny;
1691 impl EdifactSegmentTag for NadAny {
1692 const SEGMENT_TAG: &'static str = "NAD";
1693 }
1694
1695 let input = b"NAD+MS+1'NAD+MR+2'RFF+AA:1'NAD+BY+3'";
1696 let segments: Vec<Segment<'_>> =
1697 crate::from_bytes(input).collect::<Result<_, _>>().unwrap();
1698 let groups = contiguous_groups_by_qualifier::<NadAny>(&segments);
1699
1700 assert_eq!(groups.len(), 2);
1701 assert_eq!(groups[0].len(), 2);
1702 assert_eq!(groups[1].len(), 1);
1703 }
1704
1705 #[test]
1708 fn message_windows_bytes_yields_complete_windows() {
1709 let input = b"UNB+UNOA:1+S+R+200101:0900+1'\
1710 UNH+1+ORDERS:D:96A:UN'\
1711 BGM+220+PO-001+9'\
1712 UNT+3+1'\
1713 UNZ+1+1'";
1714 let windows: Vec<_> = message_windows_bytes(input)
1715 .collect::<Result<_, _>>()
1716 .unwrap();
1717 assert_eq!(windows.len(), 1);
1718 assert_eq!(windows[0].segments[0].tag, "UNH");
1719 assert_eq!(windows[0].segments.last().unwrap().tag, "UNT");
1720 assert_eq!(windows[0].message_type.as_deref(), Some("ORDERS"));
1721 assert_eq!(windows[0].association_code.as_deref(), None);
1722 }
1723
1724 #[test]
1725 fn message_windows_bytes_preserves_owned_unh_metadata() {
1726 let input = b"UNB+UNOA:1+S+R+200101:0900+1'\
1727 UNH+1+ORD?ERS:D:96A:UN:5??5??3a'\
1728 BGM+220+PO-001+9'\
1729 UNT+3+1'\
1730 UNZ+1+1'";
1731 let windows: Vec<_> = message_windows_bytes(input)
1732 .collect::<Result<_, _>>()
1733 .unwrap();
1734
1735 assert_eq!(windows.len(), 1);
1736 assert_eq!(windows[0].message_type.as_deref(), Some("ORDERS"));
1737 assert_eq!(windows[0].association_code.as_deref(), Some("5?5?3a"));
1738 }
1739
1740 #[test]
1741 fn message_windows_truncated_stream_returns_error() {
1742 let input = b"UNH+1+ORDERS:D:96A:UN'BGM+220+PO-001+9'";
1744 let results: Vec<_> = message_windows_bytes(input).collect();
1745 assert_eq!(results.len(), 1);
1746 assert!(
1747 matches!(results[0], Err(EdifactError::UnexpectedEof { .. })),
1748 "expected UnexpectedEof for truncated window, got: {:?}",
1749 results[0]
1750 );
1751 }
1752
1753 #[test]
1754 fn message_windows_subsequent_calls_return_none_after_truncation() {
1755 let input = b"UNH+1+ORDERS:D:96A:UN'BGM+220+PO-001+9'";
1756 let mut iter = message_windows_bytes(input);
1757 assert!(matches!(
1758 iter.next(),
1759 Some(Err(EdifactError::UnexpectedEof { .. }))
1760 ));
1761 assert!(iter.next().is_none());
1763 }
1764
1765 #[test]
1766 fn message_windows_unh_without_unt_before_next_unh_returns_error() {
1767 let input = b"UNH+1+ORDERS:D:96A:UN'BGM+220+PO-001+9'\
1768 UNH+2+ORDERS:D:96A:UN'BGM+220+PO-002+9'UNT+3+2'";
1769 let results: Vec<_> = message_windows_bytes(input).collect();
1770 assert!(
1772 matches!(
1773 results[0],
1774 Err(EdifactError::InvalidSegmentForMessage { ref tag, .. }) if tag == "UNH"
1775 ),
1776 "expected InvalidSegmentForMessage(UNH), got: {:?}",
1777 results[0]
1778 );
1779 }
1780
1781 fn parse_one(input: &str) -> crate::OwnedSegment {
1784 crate::from_reader_collect(std::io::Cursor::new(input.as_bytes()))
1785 .expect("parse failed")
1786 .into_iter()
1787 .next()
1788 .expect("at least one segment")
1789 }
1790
1791 #[test]
1792 fn segment_accessor_get_element_returns_value() {
1793 let owned = parse_one("BGM+220+PO-001+9'");
1794 let seg = owned.as_borrowed();
1795 assert_eq!(SegmentAccessor::get_element(&seg, 0), Some("220"));
1796 assert_eq!(SegmentAccessor::get_element(&seg, 1), Some("PO-001"));
1797 assert_eq!(SegmentAccessor::get_element(&seg, 2), Some("9"));
1798 assert_eq!(
1799 SegmentAccessor::get_element(&seg, 9),
1800 None,
1801 "out-of-bounds must return None"
1802 );
1803 }
1804
1805 #[test]
1806 fn segment_accessor_get_element_filters_empty() {
1807 let owned = parse_one("TST+++VALUE'");
1808 let seg = owned.as_borrowed();
1809 assert_eq!(
1811 SegmentAccessor::get_element(&seg, 0),
1812 None,
1813 "empty element must return None"
1814 );
1815 assert_eq!(
1816 SegmentAccessor::get_element(&seg, 1),
1817 None,
1818 "empty element must return None"
1819 );
1820 assert_eq!(SegmentAccessor::get_element(&seg, 2), Some("VALUE"));
1821 }
1822
1823 #[test]
1824 fn segment_accessor_get_component_returns_value() {
1825 let owned = parse_one("UNH+1+ORDERS:D:96A:UN'");
1826 let seg = owned.as_borrowed();
1827 assert_eq!(seg.get_component(1, 0), Some("ORDERS"));
1828 assert_eq!(seg.get_component(1, 1), Some("D"));
1829 assert_eq!(seg.get_component(1, 2), Some("96A"));
1830 assert_eq!(seg.get_component(1, 3), Some("UN"));
1831 assert_eq!(
1832 seg.get_component(1, 9),
1833 None,
1834 "out-of-bounds must return None"
1835 );
1836 }
1837
1838 #[test]
1839 fn segment_accessor_text_element_errors_on_missing() {
1840 let owned = parse_one("BGM+'");
1841 let seg = owned.as_borrowed();
1842 let err = seg.text_element(0);
1844 assert!(
1845 matches!(err, Err(EdifactError::MissingRequiredElement { ref tag, element_index: 0 }) if tag == "BGM"),
1846 "expected MissingRequiredElement, got: {err:?}"
1847 );
1848 }
1849
1850 #[test]
1851 fn segment_accessor_required_composite_errors_on_missing() {
1852 let owned = parse_one("DTM+137'");
1853 let seg = owned.as_borrowed();
1854 let err = seg.required_composite(0, 1);
1856 assert!(
1857 matches!(err, Err(EdifactError::MissingRequiredComponent { ref tag, element_index: 0, component_index: 1 }) if tag == "DTM"),
1858 "expected MissingRequiredComponent, got: {err:?}"
1859 );
1860 }
1861
1862 #[test]
1863 fn segment_accessor_code_element_parses_integer() {
1864 let owned = parse_one("QTY+21:100'");
1865 let seg = owned.as_borrowed();
1866 let qty: u32 = seg.code_element(0).expect("should parse qualifier as u32");
1867 assert_eq!(qty, 21);
1868 }
1869
1870 #[test]
1871 fn segment_accessor_optional_element_absent_returns_none() {
1872 let owned = parse_one("BGM+220'");
1873 let seg = owned.as_borrowed();
1874 assert_eq!(seg.optional_element(5), None);
1875 }
1876}