spinoso_regexp/error.rs
1use core::fmt;
2use std::borrow::Cow;
3use std::error;
4
5/// Sum type of all errors possibly returned from `Regexp` APIs.
6#[derive(Debug, Clone, Hash, PartialEq, Eq, PartialOrd, Ord)]
7pub enum Error {
8 /// Error that indicates an argument parsing or value logic error occurred.
9 ///
10 /// See [`ArgumentError`].
11 Argument(ArgumentError),
12 /// Error that indicates a `Regexp` was malformed at runtime.
13 ///
14 /// See [`RegexpError`].
15 Regexp(RegexpError),
16 /// Error that indicates a given `Regexp` pattern could not be parsed when
17 /// given as a `/.../` literal in Ruby source code.
18 ///
19 /// See [`SyntaxError`].
20 Syntax(SyntaxError),
21}
22
23impl From<ArgumentError> for Error {
24 #[inline]
25 fn from(err: ArgumentError) -> Self {
26 Self::Argument(err)
27 }
28}
29
30impl From<RegexpError> for Error {
31 #[inline]
32 fn from(err: RegexpError) -> Self {
33 Self::Regexp(err)
34 }
35}
36
37impl From<SyntaxError> for Error {
38 #[inline]
39 fn from(err: SyntaxError) -> Self {
40 Self::Syntax(err)
41 }
42}
43
44impl fmt::Display for Error {
45 #[inline]
46 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
47 f.write_str("Regexp error")
48 }
49}
50
51impl error::Error for Error {
52 #[inline]
53 fn source(&self) -> Option<&(dyn error::Error + 'static)> {
54 match self {
55 Self::Argument(err) => Some(err),
56 Self::Regexp(err) => Some(err),
57 Self::Syntax(err) => Some(err),
58 }
59 }
60}
61
62/// Error that indicates an argument parsing or value logic error occurred.
63///
64/// Argument errors have an associated message.
65///
66/// This error corresponds to the [Ruby `ArgumentError` Exception class].
67///
68/// # Examples
69///
70/// ```
71/// # use spinoso_regexp::ArgumentError;
72/// let err = ArgumentError::new();
73/// assert_eq!(err.message(), "ArgumentError");
74///
75/// let err = ArgumentError::with_message("invalid byte sequence in UTF-8");
76/// assert_eq!(err.message(), "invalid byte sequence in UTF-8");
77/// ```
78///
79/// [Ruby `ArgumentError` Exception class]: https://ruby-doc.org/core-3.1.2/ArgumentError.html
80#[derive(Debug, Clone, Hash, PartialEq, Eq, PartialOrd, Ord)]
81pub struct ArgumentError(Cow<'static, str>);
82
83impl From<&'static str> for ArgumentError {
84 #[inline]
85 fn from(message: &'static str) -> Self {
86 Self::with_message(message)
87 }
88}
89
90impl From<String> for ArgumentError {
91 fn from(message: String) -> Self {
92 Self(Cow::Owned(message))
93 }
94}
95
96impl Default for ArgumentError {
97 #[inline]
98 fn default() -> Self {
99 Self::new()
100 }
101}
102
103impl fmt::Display for ArgumentError {
104 #[inline]
105 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
106 f.write_str(self.message())
107 }
108}
109
110impl error::Error for ArgumentError {}
111
112impl ArgumentError {
113 /// Construct a new, default argument error.
114 ///
115 /// # Examples
116 ///
117 /// ```
118 /// # use spinoso_regexp::ArgumentError;
119 /// const ERR: ArgumentError = ArgumentError::new();
120 /// assert_eq!(ERR.message(), "ArgumentError");
121 /// ```
122 #[inline]
123 #[must_use]
124 pub const fn new() -> Self {
125 Self(Cow::Borrowed("ArgumentError"))
126 }
127
128 /// Construct a new argument error with a message.
129 ///
130 /// # Examples
131 ///
132 /// ```
133 /// # use spinoso_regexp::ArgumentError;
134 /// const ERR: ArgumentError = ArgumentError::with_message("invalid byte sequence in UTF-8");
135 /// assert_eq!(ERR.message(), "invalid byte sequence in UTF-8");
136 /// ```
137 #[inline]
138 #[must_use]
139 pub const fn with_message(message: &'static str) -> Self {
140 Self(Cow::Borrowed(message))
141 }
142
143 #[must_use]
144 pub(crate) const fn unsupported_pattern_encoding() -> Self {
145 Self::with_message("Unsupported pattern encoding")
146 }
147
148 #[must_use]
149 pub(crate) const fn unsupported_haystack_encoding() -> Self {
150 Self::with_message("Unsupported haystack encoding")
151 }
152
153 /// Retrieve the exception message associated with this argument error.
154 ///
155 /// # Examples
156 ///
157 /// ```
158 /// # use spinoso_regexp::ArgumentError;
159 /// let err = ArgumentError::new();
160 /// assert_eq!(err.message(), "ArgumentError");
161 ///
162 /// let err = ArgumentError::with_message("invalid byte sequence in UTF-8");
163 /// assert_eq!(err.message(), "invalid byte sequence in UTF-8");
164 /// ```
165 #[inline]
166 #[must_use]
167 pub fn message(&self) -> &str {
168 self.0.as_ref()
169 }
170}
171
172/// Error that indicates a `Regexp` was malformed at runtime.
173///
174/// This error is typically generated by [`Regexp::compile`].
175///
176/// This error corresponds to the [Ruby `RegexpError` Exception class].
177///
178/// # Examples
179///
180/// ```
181/// # use spinoso_regexp::RegexpError;
182/// let err = RegexpError::new();
183/// assert_eq!(err.message(), "RegexpError");
184///
185/// let err = RegexpError::with_message(r"invalid multibyte character: /\xFF\xFE/");
186/// assert_eq!(err.message(), r"invalid multibyte character: /\xFF\xFE/");
187/// ```
188///
189/// [`Regexp::compile`]: https://ruby-doc.org/core-3.1.2/Regexp.html#method-c-compile
190/// [Ruby `RegexpError` Exception class]: https://ruby-doc.org/core-3.1.2/RegexpError.html
191#[derive(Debug, Clone, Hash, PartialEq, Eq, PartialOrd, Ord)]
192pub struct RegexpError(Cow<'static, str>);
193
194impl From<&'static str> for RegexpError {
195 #[inline]
196 fn from(message: &'static str) -> Self {
197 Self::with_message(message)
198 }
199}
200
201impl From<String> for RegexpError {
202 fn from(message: String) -> Self {
203 Self(Cow::Owned(message))
204 }
205}
206
207impl fmt::Display for RegexpError {
208 #[inline]
209 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
210 f.write_str(self.message())
211 }
212}
213
214impl error::Error for RegexpError {}
215
216impl Default for RegexpError {
217 fn default() -> Self {
218 Self::new()
219 }
220}
221
222impl RegexpError {
223 /// Construct a new, default regexp error.
224 ///
225 /// # Examples
226 ///
227 /// ```
228 /// # use spinoso_regexp::RegexpError;
229 /// const ERR: RegexpError = RegexpError::new();
230 /// assert_eq!(ERR.message(), "RegexpError");
231 /// ```
232 #[inline]
233 #[must_use]
234 pub const fn new() -> Self {
235 Self(Cow::Borrowed("RegexpError"))
236 }
237
238 /// Construct a new regexp error with a message.
239 ///
240 /// # Examples
241 ///
242 /// ```
243 /// # use spinoso_regexp::RegexpError;
244 /// const ERR: RegexpError = RegexpError::with_message(r"invalid multibyte character: /\xFF\xFE/");
245 /// assert_eq!(ERR.message(), r"invalid multibyte character: /\xFF\xFE/");
246 /// ```
247 #[inline]
248 #[must_use]
249 pub const fn with_message(message: &'static str) -> Self {
250 Self(Cow::Borrowed(message))
251 }
252
253 /// Retrieve the exception message associated with this regexp error.
254 ///
255 /// # Examples
256 ///
257 /// ```
258 /// # use spinoso_regexp::RegexpError;
259 /// let err = RegexpError::new();
260 /// assert_eq!(err.message(), "RegexpError");
261 ///
262 /// let err = RegexpError::with_message(r"invalid multibyte character: /\xFF\xFE/");
263 /// assert_eq!(err.message(), r"invalid multibyte character: /\xFF\xFE/");
264 /// ```
265 #[inline]
266 #[must_use]
267 pub fn message(&self) -> &str {
268 &self.0
269 }
270}
271
272/// Error that indicates a given `Regexp` pattern could not be parsed when given
273/// as a `/.../` literal in Ruby source code.
274///
275/// This error is typically generated at parse-time.
276///
277/// This error corresponds to the [Ruby `SyntaxError` Exception class].
278///
279/// # Examples
280///
281/// ```
282/// # use spinoso_regexp::SyntaxError;
283/// let err = SyntaxError::new();
284/// assert_eq!(err.message(), "SyntaxError");
285///
286/// let err = SyntaxError::with_message("premature end of char-class");
287/// assert_eq!(err.message(), "premature end of char-class");
288/// ```
289///
290/// [Ruby `SyntaxError` Exception class]: https://ruby-doc.org/core-3.1.2/SyntaxError.html
291#[derive(Debug, Clone, Hash, PartialEq, Eq, PartialOrd, Ord)]
292pub struct SyntaxError(Cow<'static, str>);
293
294impl From<&'static str> for SyntaxError {
295 #[inline]
296 fn from(message: &'static str) -> Self {
297 Self::with_message(message)
298 }
299}
300
301impl From<String> for SyntaxError {
302 fn from(message: String) -> Self {
303 Self(Cow::Owned(message))
304 }
305}
306
307impl fmt::Display for SyntaxError {
308 #[inline]
309 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
310 f.write_str(self.message())
311 }
312}
313
314impl error::Error for SyntaxError {}
315
316impl Default for SyntaxError {
317 fn default() -> Self {
318 Self::new()
319 }
320}
321
322impl SyntaxError {
323 /// Construct a new, default syntax error.
324 ///
325 /// # Examples
326 ///
327 /// ```
328 /// # use spinoso_regexp::SyntaxError;
329 /// const ERR: SyntaxError = SyntaxError::new();
330 /// assert_eq!(ERR.message(), "SyntaxError");
331 /// ```
332 #[inline]
333 #[must_use]
334 pub const fn new() -> Self {
335 Self(Cow::Borrowed("SyntaxError"))
336 }
337
338 /// Construct a new syntax error with a message.
339 ///
340 /// # Examples
341 ///
342 /// ```
343 /// # use spinoso_regexp::SyntaxError;
344 /// const ERR: SyntaxError = SyntaxError::with_message("premature end of char-class");
345 /// assert_eq!(ERR.message(), "premature end of char-class");
346 /// ```
347 #[inline]
348 #[must_use]
349 pub const fn with_message(message: &'static str) -> Self {
350 Self(Cow::Borrowed(message))
351 }
352
353 /// Retrieve the exception message associated with this syntax error.
354 ///
355 /// # Examples
356 ///
357 /// ```
358 /// # use spinoso_regexp::SyntaxError;
359 /// let err = SyntaxError::new();
360 /// assert_eq!(err.message(), "SyntaxError");
361 ///
362 /// let err = SyntaxError::with_message("premature end of char-class");
363 /// assert_eq!(err.message(), "premature end of char-class");
364 /// ```
365 #[inline]
366 #[must_use]
367 pub fn message(&self) -> &str {
368 &self.0
369 }
370}