diff options
Diffstat (limited to 'vendor/unarray/src/map.rs')
-rw-r--r-- | vendor/unarray/src/map.rs | 207 |
1 files changed, 207 insertions, 0 deletions
diff --git a/vendor/unarray/src/map.rs b/vendor/unarray/src/map.rs new file mode 100644 index 000000000..3b7e74574 --- /dev/null +++ b/vendor/unarray/src/map.rs @@ -0,0 +1,207 @@ +use crate::{mark_initialized, uninit_buf}; + +/// An extension trait that adds methods to `[T; N]` +/// +/// This trait provides [`UnarrayArrayExt::map_result`] and [`UnarrayArrayExt::map_option`], +/// which provide functionality similar to the nightly-only [`array::try_map`] +pub trait UnarrayArrayExt<T, const N: usize> { + /// Maps an array, short-circuiting if any element produces an `Err` + /// + /// ``` + /// # use unarray::*; + /// let elements = ["123", "234", "345"]; + /// let mapped = elements.map_result(|s| s.parse()); + /// assert_eq!(mapped, Ok([123, 234, 345])); + /// ``` + /// + /// This function applies `f` to every element. If any element produces an `Err`, the function + /// immediately returns that error. Otherwise, it returns `Ok(result)` where `result` contains + /// the mapped elements in an array. + /// + /// This function does not allocate space on the heap + /// + /// For functions that return an `Option`, consider using [`UnarrayArrayExt::map_option`] + fn map_result<S, E>(self, f: impl FnMut(T) -> Result<S, E>) -> Result<[S; N], E>; + + /// Maps an array, short-circuiting if any element produces a `None` + /// + /// ``` + /// # use unarray::*; + /// fn parse(s: &str) -> Option<bool> { + /// match s { + /// "true" => Some(true), + /// "false" => Some(false), + /// _ => None, + /// } + /// } + /// + /// let elements = ["true", "false", "true"]; + /// let mapped = elements.map_option(parse); + /// assert_eq!(mapped, Some([true, false, true])); + /// ``` + /// + /// This function applies `f` to every element. If any element produces `None`, the function + /// immediately returns `None`. Otherwise, it returns `Some(result)` where `result` contains + /// the mapped elements in an array. + /// + /// This function does not allocate space on the heap + /// + /// For functions that return an `Result`, consider using [`UnarrayArrayExt::map_result`] + fn map_option<S>(self, f: impl FnMut(T) -> Option<S>) -> Option<[S; N]>; +} + +impl<T, const N: usize> UnarrayArrayExt<T, N> for [T; N] { + fn map_result<S, E>(self, mut f: impl FnMut(T) -> Result<S, E>) -> Result<[S; N], E> { + let mut result = uninit_buf(); + + // This is quaranteed to loop over every element (or panic), since both `result` and `self` have N elements + // If a panic occurs, uninitialized data is never dropped, since `MaybeUninit` wraps its + // contained data in `ManuallyDrop` + for (index, (item, slot)) in IntoIterator::into_iter(self).zip(&mut result).enumerate() { + match f(item) { + Ok(s) => slot.write(s), + Err(e) => { + // SAFETY: + // We have failed at `index` which is the `index + 1`th element, so the first + // `index` elements are safe to drop + result + .iter_mut() + .take(index) + .for_each(|slot| unsafe { slot.assume_init_drop() }); + return Err(e); + } + }; + } + + // SAFETY: + // At this point in execution, we have iterated over all elements of `result`. If any + // errors were encountered, we would have already returned. So it's safe to remove the + // MaybeUninit wrapper + Ok(unsafe { mark_initialized(result) }) + } + + fn map_option<S>(self, mut f: impl FnMut(T) -> Option<S>) -> Option<[S; N]> { + // transform to a `Result`-returning function so we can avoid duplicating unsafe code + let actual_f = |t: T| -> Result<S, ()> { f(t).ok_or(()) }; + + let result: Result<[S; N], ()> = UnarrayArrayExt::map_result(self, actual_f); + match result { + Ok(result) => Some(result), + Err(()) => None, + } + } +} + +#[cfg(test)] +mod tests { + use core::{ + convert::TryInto, + sync::atomic::{AtomicUsize, Ordering}, + }; + + use super::UnarrayArrayExt; + use crate::testing::array_strategy; + use proptest::prelude::*; + use test_strategy::proptest; + + #[test] + fn test_map_option() { + let array = [1, 2, 3]; + let result = array.map_option(|i| Some(i * 2)).unwrap(); + assert_eq!(result, [2, 4, 6]); + } + + #[test] + #[should_panic] + fn test_map_option_panic() { + let array = [1, 2, 3]; + array.map_option(|i| { + if i > 2 { + panic!(); + } + + Some(i) + }); + } + + #[test] + fn test_map_result() { + let array = [1, 2, 3]; + let result: Result<_, ()> = array.map_result(|i| Ok(i * 2)); + assert_eq!(result.unwrap(), [2, 4, 6]); + } + + #[test] + #[should_panic] + fn test_map_result_panic() { + let array = [1, 2, 3]; + let _ = array.map_result(|i| -> Result<i32, ()> { + if i > 2 { + panic!(); + } + + Ok(i) + }); + } + + struct IncrementOnDrop<'a>(&'a AtomicUsize); + impl Drop for IncrementOnDrop<'_> { + fn drop(&mut self) { + self.0.fetch_add(1, Ordering::Relaxed); + } + } + + #[test] + fn map_array_result_doesnt_leak() { + let drop_counter = 0.into(); + + // this will successfully create 3 structs, fail on the 4th, we expect 3 drops to be + // called, since the 4th may be in an inconsistent state + let _ = [0, 1, 2, 3, 4].map_result(|i| { + if i == 3 { + Err(()) + } else { + Ok(IncrementOnDrop(&drop_counter)) + } + }); + + assert_eq!(drop_counter.load(Ordering::Relaxed), 3); + } + + #[test] + fn map_array_option_doesnt_leak() { + let drop_counter = 0.into(); + + // this will successfully create 3 structs, fail on the 4th, we expect 3 drops to be + // called, since the 4th may be in an inconsistent state + let _ = [0, 1, 2, 3, 4].map_option(|i| { + if i == 3 { + None + } else { + Some(IncrementOnDrop(&drop_counter)) + } + }); + + assert_eq!(drop_counter.load(Ordering::Relaxed), 3); + } + + const LEN: usize = 100; + + #[proptest] + #[cfg_attr(miri, ignore)] + fn proptest_option_map(#[strategy(array_strategy::<LEN>())] array: [String; LEN]) { + let expected = array.iter().map(|s| s.len()).collect::<Vec<_>>(); + let expected: [usize; LEN] = expected.try_into().unwrap(); + let result = array.map_option(|s| Some(s.len())); + prop_assert_eq!(expected, result.unwrap()); + } + + #[proptest] + #[cfg_attr(miri, ignore)] + fn proptest_result_map(#[strategy(array_strategy::<LEN>())] array: [String; LEN]) { + let expected = array.iter().map(|s| s.len()).collect::<Vec<_>>(); + let expected: [usize; LEN] = expected.try_into().unwrap(); + let result: Result<_, ()> = array.map_result(|s| Ok(s.len())); + prop_assert_eq!(expected, result.unwrap()); + } +} |