diff options
| author | mo khan <mo@mokhan.ca> | 2025-07-10 13:11:11 -0600 |
|---|---|---|
| committer | mo khan <mo@mokhan.ca> | 2025-07-10 13:11:11 -0600 |
| commit | 01959b16a21b22b5df5f16569c2a8e8f92beecef (patch) | |
| tree | 32afa5d747c5466345c59ec52161a7cba3d6d755 /vendor/writeable/src/try_writeable.rs | |
| parent | ff30574117a996df332e23d1fb6f65259b316b5b (diff) | |
chore: vendor dependencies
Diffstat (limited to 'vendor/writeable/src/try_writeable.rs')
| -rw-r--r-- | vendor/writeable/src/try_writeable.rs | 439 |
1 files changed, 439 insertions, 0 deletions
diff --git a/vendor/writeable/src/try_writeable.rs b/vendor/writeable/src/try_writeable.rs new file mode 100644 index 00000000..aa918bbd --- /dev/null +++ b/vendor/writeable/src/try_writeable.rs @@ -0,0 +1,439 @@ +// This file is part of ICU4X. For terms of use, please see the file +// called LICENSE at the top level of the ICU4X source tree +// (online at: https://github.com/unicode-org/icu4x/blob/main/LICENSE ). + +use super::*; +use crate::parts_write_adapter::CoreWriteAsPartsWrite; +use core::convert::Infallible; + +/// A writeable object that can fail while writing. +/// +/// The default [`Writeable`] trait returns a [`fmt::Error`], which originates from the sink. +/// In contrast, this trait allows the _writeable itself_ to trigger an error as well. +/// +/// Implementations are expected to always make a _best attempt_ at writing to the sink +/// and should write replacement values in the error state. Therefore, the returned `Result` +/// can be safely ignored to emulate a "lossy" mode. +/// +/// Any error substrings should be annotated with [`Part::ERROR`]. +/// +/// # Implementer Notes +/// +/// This trait requires that implementers make a _best attempt_ at writing to the sink, +/// _even in the error state_, such as with a placeholder or fallback string. +/// +/// In [`TryWriteable::try_write_to_parts()`], error substrings should be annotated with +/// [`Part::ERROR`]. Because of this, writing to parts is not default-implemented like +/// it is on [`Writeable`]. +/// +/// The trait is implemented on [`Result<T, E>`] where `T` and `E` both implement [`Writeable`]; +/// In the `Ok` case, `T` is written, and in the `Err` case, `E` is written as a fallback value. +/// This impl, which writes [`Part::ERROR`], can be used as a basis for more advanced impls. +/// +/// # Examples +/// +/// Implementing on a custom type: +/// +/// ``` +/// use core::fmt; +/// use writeable::LengthHint; +/// use writeable::PartsWrite; +/// use writeable::TryWriteable; +/// +/// #[derive(Debug, PartialEq, Eq)] +/// enum HelloWorldWriteableError { +/// MissingName, +/// } +/// +/// #[derive(Debug, PartialEq, Eq)] +/// struct HelloWorldWriteable { +/// pub name: Option<&'static str>, +/// } +/// +/// impl TryWriteable for HelloWorldWriteable { +/// type Error = HelloWorldWriteableError; +/// +/// fn try_write_to_parts<S: PartsWrite + ?Sized>( +/// &self, +/// sink: &mut S, +/// ) -> Result<Result<(), Self::Error>, fmt::Error> { +/// sink.write_str("Hello, ")?; +/// // Use `impl TryWriteable for Result` to generate the error part: +/// let err = self.name.ok_or("nobody").try_write_to_parts(sink)?.err(); +/// sink.write_char('!')?; +/// // Return a doubly-wrapped Result. +/// // The outer Result is for fmt::Error, handled by the `?`s above. +/// // The inner Result is for our own Self::Error. +/// if err.is_none() { +/// Ok(Ok(())) +/// } else { +/// Ok(Err(HelloWorldWriteableError::MissingName)) +/// } +/// } +/// +/// fn writeable_length_hint(&self) -> LengthHint { +/// self.name.ok_or("nobody").writeable_length_hint() + 8 +/// } +/// } +/// +/// // Success case: +/// writeable::assert_try_writeable_eq!( +/// HelloWorldWriteable { +/// name: Some("Alice") +/// }, +/// "Hello, Alice!" +/// ); +/// +/// // Failure case, including the ERROR part: +/// writeable::assert_try_writeable_parts_eq!( +/// HelloWorldWriteable { name: None }, +/// "Hello, nobody!", +/// Err(HelloWorldWriteableError::MissingName), +/// [(7, 13, writeable::Part::ERROR)] +/// ); +/// ``` +pub trait TryWriteable { + type Error; + + /// Writes the content of this writeable to a sink. + /// + /// If the sink hits an error, writing immediately ends, + /// `Err(`[`fmt::Error`]`)` is returned, and the sink does not contain valid output. + /// + /// If the writeable hits an error, writing is continued with a replacement value, + /// `Ok(Err(`[`TryWriteable::Error`]`))` is returned, and the caller may continue using the sink. + /// + /// # Lossy Mode + /// + /// The [`fmt::Error`] should always be handled, but the [`TryWriteable::Error`] can be + /// ignored if a fallback string is desired instead of an error. + /// + /// To handle the sink error, but not the writeable error, write: + /// + /// ``` + /// # use writeable::TryWriteable; + /// # let my_writeable: Result<&str, &str> = Ok(""); + /// # let mut sink = String::new(); + /// let _ = my_writeable.try_write_to(&mut sink)?; + /// # Ok::<(), core::fmt::Error>(()) + /// ``` + /// + /// # Examples + /// + /// The following examples use `Result<&str, usize>`, which implements [`TryWriteable`] because both `&str` and `usize` do. + /// + /// Success case: + /// + /// ``` + /// use writeable::TryWriteable; + /// + /// let w: Result<&str, usize> = Ok("success"); + /// let mut sink = String::new(); + /// let result = w.try_write_to(&mut sink); + /// + /// assert_eq!(result, Ok(Ok(()))); + /// assert_eq!(sink, "success"); + /// ``` + /// + /// Failure case: + /// + /// ``` + /// use writeable::TryWriteable; + /// + /// let w: Result<&str, usize> = Err(44); + /// let mut sink = String::new(); + /// let result = w.try_write_to(&mut sink); + /// + /// assert_eq!(result, Ok(Err(44))); + /// assert_eq!(sink, "44"); + /// ``` + fn try_write_to<W: fmt::Write + ?Sized>( + &self, + sink: &mut W, + ) -> Result<Result<(), Self::Error>, fmt::Error> { + self.try_write_to_parts(&mut CoreWriteAsPartsWrite(sink)) + } + + /// Writes the content of this writeable to a sink with parts (annotations). + /// + /// For more information, see: + /// + /// - [`TryWriteable::try_write_to()`] for the general behavior. + /// - [`TryWriteable`] for an example with parts. + /// - [`Part`] for more about parts. + fn try_write_to_parts<S: PartsWrite + ?Sized>( + &self, + sink: &mut S, + ) -> Result<Result<(), Self::Error>, fmt::Error>; + + /// Returns a hint for the number of UTF-8 bytes that will be written to the sink. + /// + /// This function returns the length of the "lossy mode" string; for more information, + /// see [`TryWriteable::try_write_to()`]. + fn writeable_length_hint(&self) -> LengthHint { + LengthHint::undefined() + } + + /// Writes the content of this writeable to a string. + /// + /// In the failure case, this function returns the error and the best-effort string ("lossy mode"). + /// + /// Examples + /// + /// ``` + /// # use std::borrow::Cow; + /// # use writeable::TryWriteable; + /// // use the best-effort string + /// let r1: Cow<str> = Ok::<&str, u8>("ok") + /// .try_write_to_string() + /// .unwrap_or_else(|(_, s)| s); + /// // propagate the error + /// let r2: Result<Cow<str>, u8> = Ok::<&str, u8>("ok") + /// .try_write_to_string() + /// .map_err(|(e, _)| e); + /// ``` + fn try_write_to_string(&self) -> Result<Cow<str>, (Self::Error, Cow<str>)> { + let hint = self.writeable_length_hint(); + if hint.is_zero() { + return Ok(Cow::Borrowed("")); + } + let mut output = String::with_capacity(hint.capacity()); + match self + .try_write_to(&mut output) + .unwrap_or_else(|fmt::Error| Ok(())) + { + Ok(()) => Ok(Cow::Owned(output)), + Err(e) => Err((e, Cow::Owned(output))), + } + } +} + +impl<T, E> TryWriteable for Result<T, E> +where + T: Writeable, + E: Writeable + Clone, +{ + type Error = E; + + #[inline] + fn try_write_to<W: fmt::Write + ?Sized>( + &self, + sink: &mut W, + ) -> Result<Result<(), Self::Error>, fmt::Error> { + match self { + Ok(t) => t.write_to(sink).map(Ok), + Err(e) => e.write_to(sink).map(|()| Err(e.clone())), + } + } + + #[inline] + fn try_write_to_parts<S: PartsWrite + ?Sized>( + &self, + sink: &mut S, + ) -> Result<Result<(), Self::Error>, fmt::Error> { + match self { + Ok(t) => t.write_to_parts(sink).map(Ok), + Err(e) => sink + .with_part(Part::ERROR, |sink| e.write_to_parts(sink)) + .map(|()| Err(e.clone())), + } + } + + #[inline] + fn writeable_length_hint(&self) -> LengthHint { + match self { + Ok(t) => t.writeable_length_hint(), + Err(e) => e.writeable_length_hint(), + } + } + + #[inline] + fn try_write_to_string(&self) -> Result<Cow<str>, (Self::Error, Cow<str>)> { + match self { + Ok(t) => Ok(t.write_to_string()), + Err(e) => Err((e.clone(), e.write_to_string())), + } + } +} + +/// A wrapper around [`TryWriteable`] that implements [`Writeable`] +/// if [`TryWriteable::Error`] is [`Infallible`]. +#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] +#[repr(transparent)] +#[allow(clippy::exhaustive_structs)] // transparent newtype +pub struct TryWriteableInfallibleAsWriteable<T>(pub T); + +impl<T> Writeable for TryWriteableInfallibleAsWriteable<T> +where + T: TryWriteable<Error = Infallible>, +{ + #[inline] + fn write_to<W: fmt::Write + ?Sized>(&self, sink: &mut W) -> fmt::Result { + match self.0.try_write_to(sink) { + Ok(Ok(())) => Ok(()), + Ok(Err(infallible)) => match infallible {}, + Err(e) => Err(e), + } + } + + #[inline] + fn write_to_parts<S: PartsWrite + ?Sized>(&self, sink: &mut S) -> fmt::Result { + match self.0.try_write_to_parts(sink) { + Ok(Ok(())) => Ok(()), + Ok(Err(infallible)) => match infallible {}, + Err(e) => Err(e), + } + } + + #[inline] + fn writeable_length_hint(&self) -> LengthHint { + self.0.writeable_length_hint() + } + + #[inline] + fn write_to_string(&self) -> Cow<str> { + match self.0.try_write_to_string() { + Ok(s) => s, + Err((infallible, _)) => match infallible {}, + } + } +} + +impl<T> fmt::Display for TryWriteableInfallibleAsWriteable<T> +where + T: TryWriteable<Error = Infallible>, +{ + #[inline] + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + self.write_to(f) + } +} + +/// A wrapper around [`Writeable`] that implements [`TryWriteable`] +/// with [`TryWriteable::Error`] set to [`Infallible`]. +#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] +#[repr(transparent)] +#[allow(clippy::exhaustive_structs)] // transparent newtype +pub struct WriteableAsTryWriteableInfallible<T>(pub T); + +impl<T> TryWriteable for WriteableAsTryWriteableInfallible<T> +where + T: Writeable, +{ + type Error = Infallible; + + #[inline] + fn try_write_to<W: fmt::Write + ?Sized>( + &self, + sink: &mut W, + ) -> Result<Result<(), Infallible>, fmt::Error> { + self.0.write_to(sink).map(Ok) + } + + #[inline] + fn try_write_to_parts<S: PartsWrite + ?Sized>( + &self, + sink: &mut S, + ) -> Result<Result<(), Infallible>, fmt::Error> { + self.0.write_to_parts(sink).map(Ok) + } + + #[inline] + fn writeable_length_hint(&self) -> LengthHint { + self.0.writeable_length_hint() + } + + #[inline] + fn try_write_to_string(&self) -> Result<Cow<str>, (Infallible, Cow<str>)> { + Ok(self.0.write_to_string()) + } +} + +/// Testing macros for types implementing [`TryWriteable`]. +/// +/// Arguments, in order: +/// +/// 1. The [`TryWriteable`] under test +/// 2. The expected string value +/// 3. The expected result value, or `Ok(())` if omitted +/// 3. [`*_parts_eq`] only: a list of parts (`[(start, end, Part)]`) +/// +/// Any remaining arguments get passed to `format!` +/// +/// The macros tests the following: +/// +/// - Equality of string content +/// - Equality of parts ([`*_parts_eq`] only) +/// - Validity of size hint +/// +/// For a usage example, see [`TryWriteable`]. +/// +/// [`*_parts_eq`]: assert_try_writeable_parts_eq +#[macro_export] +macro_rules! assert_try_writeable_eq { + ($actual_writeable:expr, $expected_str:expr $(,)?) => { + $crate::assert_try_writeable_eq!($actual_writeable, $expected_str, Ok(())) + }; + ($actual_writeable:expr, $expected_str:expr, $expected_result:expr $(,)?) => { + $crate::assert_try_writeable_eq!($actual_writeable, $expected_str, $expected_result, "") + }; + ($actual_writeable:expr, $expected_str:expr, $expected_result:expr, $($arg:tt)+) => {{ + $crate::assert_try_writeable_eq!(@internal, $actual_writeable, $expected_str, $expected_result, $($arg)*); + }}; + (@internal, $actual_writeable:expr, $expected_str:expr, $expected_result:expr, $($arg:tt)+) => {{ + use $crate::TryWriteable; + let actual_writeable = &$actual_writeable; + let (actual_str, actual_parts, actual_error) = $crate::_internal::try_writeable_to_parts_for_test(actual_writeable); + assert_eq!(actual_str, $expected_str, $($arg)*); + assert_eq!(actual_error, Result::<(), _>::from($expected_result).err(), $($arg)*); + let actual_result = match actual_writeable.try_write_to_string() { + Ok(actual_cow_str) => { + assert_eq!(actual_cow_str, $expected_str, $($arg)+); + Ok(()) + } + Err((e, actual_cow_str)) => { + assert_eq!(actual_cow_str, $expected_str, $($arg)+); + Err(e) + } + }; + assert_eq!(actual_result, Result::<(), _>::from($expected_result), $($arg)*); + let length_hint = actual_writeable.writeable_length_hint(); + assert!( + length_hint.0 <= actual_str.len(), + "hint lower bound {} larger than actual length {}: {}", + length_hint.0, actual_str.len(), format!($($arg)*), + ); + if let Some(upper) = length_hint.1 { + assert!( + actual_str.len() <= upper, + "hint upper bound {} smaller than actual length {}: {}", + length_hint.0, actual_str.len(), format!($($arg)*), + ); + } + actual_parts // return for assert_try_writeable_parts_eq + }}; +} + +/// See [`assert_try_writeable_eq`]. +#[macro_export] +macro_rules! assert_try_writeable_parts_eq { + ($actual_writeable:expr, $expected_str:expr, $expected_parts:expr $(,)?) => { + $crate::assert_try_writeable_parts_eq!($actual_writeable, $expected_str, Ok(()), $expected_parts) + }; + ($actual_writeable:expr, $expected_str:expr, $expected_result:expr, $expected_parts:expr $(,)?) => { + $crate::assert_try_writeable_parts_eq!($actual_writeable, $expected_str, $expected_result, $expected_parts, "") + }; + ($actual_writeable:expr, $expected_str:expr, $expected_result:expr, $expected_parts:expr, $($arg:tt)+) => {{ + let actual_parts = $crate::assert_try_writeable_eq!(@internal, $actual_writeable, $expected_str, $expected_result, $($arg)*); + assert_eq!(actual_parts, $expected_parts, $($arg)+); + }}; +} + +#[test] +fn test_result_try_writeable() { + let mut result: Result<&str, usize> = Ok("success"); + assert_try_writeable_eq!(result, "success"); + result = Err(44); + assert_try_writeable_eq!(result, "44", Err(44)); + assert_try_writeable_parts_eq!(result, "44", Err(44), [(0, 2, Part::ERROR)]) +} |
