wlan_hw_sim/event/
convert.rs

1// Copyright 2023 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5//! Fallible conversions.
6//!
7//! This module abstracts over branching (fallible) types like [`Option`] and [`Result`] and is a
8//! stop-gap for traits in the Rust core library that are unstable at time of writing (namely
9//! `core::ops::Try`). The [`Try`] trait allows event handlers to compose such types in their
10//! outputs. For example, [`Handler::expect`] is available when the output of a handler implements
11//! [`Try`].
12//!
13//! See the documentation for `core::ops::Try` etc. for more about how this mechanism works:
14//! https://doc.rust-lang.org/std/ops/trait.Try.html#
15//!
16//! [`Handler::expect`]: crate::event::handler::expect
17//! [`Option`]: core::option::Option
18//! [`Result`]: core::result::Result
19//! [`Try`]: crate::event::Try
20
21// TODO(https://fxbug.dev/42078844): Implement this in terms of the standard `core::ops::Try` and related
22//                         traits when they stabilize. Simplify (and eventually remove) this module
23//                         when those APIs can be used instead. Note that `core::ops::Try` is
24//                         already be implemented for `Option` and `Result`, so `core::ops::Try`
25//                         and `core::ops::FromResidual` must only be implemented for `Handled`.
26
27use std::convert::Infallible;
28use std::fmt::Display;
29use std::ops::ControlFlow;
30
31// This trait provides methods that (at time of writing) are unstable. They must be invoked using
32// fully qualified syntax since they share the same names as their unstable counterparts.
33pub trait ControlFlowExt<B, C> {
34    // TODO(https://fxbug.dev/335283785): Remove or document unused code
35    #[allow(dead_code)]
36    fn map_break<T, F>(self, f: F) -> ControlFlow<T, C>
37    where
38        F: FnOnce(B) -> T;
39
40    // TODO(https://fxbug.dev/335283785): Remove or document unused code
41    #[allow(dead_code)]
42    fn map_continue<T, F>(self, f: F) -> ControlFlow<B, T>
43    where
44        F: FnOnce(C) -> T;
45}
46
47impl<B, C> ControlFlowExt<B, C> for ControlFlow<B, C> {
48    fn map_break<T, F>(self, f: F) -> ControlFlow<T, C>
49    where
50        F: FnOnce(B) -> T,
51    {
52        match self {
53            ControlFlow::Break(inner) => ControlFlow::Break(f(inner)),
54            ControlFlow::Continue(inner) => ControlFlow::Continue(inner),
55        }
56    }
57
58    fn map_continue<T, F>(self, f: F) -> ControlFlow<B, T>
59    where
60        F: FnOnce(C) -> T,
61    {
62        match self {
63            ControlFlow::Break(inner) => ControlFlow::Break(inner),
64            ControlFlow::Continue(inner) => ControlFlow::Continue(f(inner)),
65        }
66    }
67}
68
69// This trait resembles the (at time of writing) unstable `core::ops::Try` and
70// `core::ops::FromResidual` traits. While it does not enable the `?` operator, it provides the
71// same abstraction of branching types and is used to enable APIs like `Handler::expect` and
72// `Handler::try_and`.
73/// Divergent (fallible) types.
74///
75/// This trait abstracts the branching of types that represent independent outputs and non-outputs
76/// (errors). Such types must be constructible from these variants and queried for a corresponding
77/// control flow.
78pub trait Try {
79    type Output;
80    // See the documentation for `core::ops::Try::Residual` for more about this associated type:
81    // https://doc.rust-lang.org/std/ops/trait.Try.html#associatedtype.Residual
82    type Residual;
83
84    fn from_output(output: Self::Output) -> Self;
85
86    fn from_residual(residual: Self::Residual) -> Self;
87
88    fn branch(self) -> ControlFlow<Self::Residual, Self::Output>;
89
90    fn unwrap(self) -> Self::Output
91    where
92        Self: Sized,
93    {
94        match self.branch() {
95            ControlFlow::Continue(output) => output,
96            _ => panic!(),
97        }
98    }
99
100    fn expect(self, message: impl Display) -> Self::Output
101    where
102        Self: Sized,
103    {
104        match self.branch() {
105            ControlFlow::Continue(output) => output,
106            _ => panic!("{}", message),
107        }
108    }
109}
110
111impl<T> Try for Option<T> {
112    type Output = T;
113    type Residual = Option<Infallible>;
114
115    fn from_output(output: Self::Output) -> Self {
116        Some(output)
117    }
118
119    fn from_residual(_: Self::Residual) -> Self {
120        None
121    }
122
123    fn branch(self) -> ControlFlow<Self::Residual, Self::Output> {
124        match self {
125            Some(inner) => ControlFlow::Continue(inner),
126            _ => ControlFlow::Break(None),
127        }
128    }
129}
130
131impl<T, E> Try for Result<T, E> {
132    type Output = T;
133    type Residual = Result<Infallible, E>;
134
135    fn from_output(output: Self::Output) -> Self {
136        Ok(output)
137    }
138
139    fn from_residual(residual: Self::Residual) -> Self {
140        match residual {
141            Err(error) => Err(error),
142            _ => unreachable!(),
143        }
144    }
145
146    fn branch(self) -> ControlFlow<Self::Residual, Self::Output> {
147        match self {
148            Ok(inner) => ControlFlow::Continue(inner),
149            Err(error) => ControlFlow::Break(Err(error)),
150        }
151    }
152}