1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
use js_sys::{Error as JsError, Reflect};
use serde::{de, ser};
use std::error;
use std::fmt::{Display, Formatter};
use wasm_bindgen::JsValue;
pub type Result<R> = std::result::Result<R, Error>;
#[derive(Debug, Clone)]
pub enum Error {
Custom(String),
InvalidType(String),
JsError {
msg: Option<String>,
stack: Option<String>,
},
}
impl From<JsValue> for Error {
fn from(src: JsValue) -> Self {
let msg_key: JsValue = "message".into();
let msg = if Reflect::has(&src, &msg_key).unwrap_or(false) {
Reflect::get(&src, &msg_key)
.map(|v| v.as_string())
.unwrap_or(None)
} else {
None
};
let stack_key: JsValue = "stack".into();
let stack = if Reflect::has(&src, &stack_key).unwrap_or(false) {
Reflect::get(&src, &stack_key)
.map(|v| v.as_string())
.unwrap_or(None)
} else {
None
};
Error::JsError { msg, stack }
}
}
impl From<Error> for JsValue {
fn from(src: Error) -> Self {
JsError::new(&src.to_string()).into()
}
}
impl Display for Error {
fn fmt(&self, f: &mut Formatter) -> std::fmt::Result {
write!(f, "jsvalue_serde error: {:?}", self)
}
}
impl de::StdError for Error {
fn source(&self) -> Option<&(dyn error::Error + 'static)> {
None
}
}
impl ser::Error for Error {
fn custom<T: Display>(msg: T) -> Error {
Error::Custom(format!("{}", msg))
}
}
impl de::Error for Error {
fn custom<T: Display>(msg: T) -> Error {
Error::Custom(format!("{}", msg))
}
fn invalid_type(unexp: de::Unexpected, exp: &dyn de::Expected) -> Self {
Error::InvalidType(format!("received {}, expected {}", unexp, exp))
}
}
pub trait ExpectValue<T> {
fn expect_value(self, msg: &str) -> Result<T>;
}
impl<T> ExpectValue<T> for Option<T> {
fn expect_value(self, msg: &str) -> Result<T> {
if let Some(val) = self {
Ok(val)
} else {
Err(Error::InvalidType(format!("expected: {}", msg)))
}
}
}