2020-11-17 23:43:58 +00:00
|
|
|
use wasm_bindgen::prelude::*;
|
|
|
|
|
2020-12-30 19:56:50 +00:00
|
|
|
pub use epi::http::{Request, Response};
|
2020-11-17 23:43:58 +00:00
|
|
|
|
|
|
|
/// NOTE: Ok(..) is returned on network error.
|
|
|
|
/// Err is only for failure to use the fetch api.
|
2020-12-30 19:56:50 +00:00
|
|
|
pub async fn fetch_async(request: &Request) -> Result<Response, String> {
|
|
|
|
fetch_jsvalue(request)
|
2020-11-17 23:43:58 +00:00
|
|
|
.await
|
2021-08-15 14:56:46 +00:00
|
|
|
.map_err(|err| err.as_string().unwrap_or(format!("{:#?}", err)))
|
2020-11-17 23:43:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// NOTE: Ok(..) is returned on network error.
|
|
|
|
/// Err is only for failure to use the fetch api.
|
2020-12-30 19:56:50 +00:00
|
|
|
async fn fetch_jsvalue(request: &Request) -> Result<Response, JsValue> {
|
2020-11-17 23:43:58 +00:00
|
|
|
// https://rustwasm.github.io/wasm-bindgen/examples/fetch.html
|
2021-08-15 14:56:46 +00:00
|
|
|
// https://github.com/seanmonstar/reqwest/blob/master/src/wasm/client.rs
|
2020-11-17 23:43:58 +00:00
|
|
|
|
|
|
|
use wasm_bindgen::JsCast;
|
|
|
|
use wasm_bindgen_futures::JsFuture;
|
|
|
|
|
|
|
|
let mut opts = web_sys::RequestInit::new();
|
2021-08-15 14:56:46 +00:00
|
|
|
opts.method(&request.method);
|
2020-11-17 23:43:58 +00:00
|
|
|
opts.mode(web_sys::RequestMode::Cors);
|
|
|
|
|
2021-08-15 14:56:46 +00:00
|
|
|
if !request.body.is_empty() {
|
|
|
|
let body_bytes: &[u8] = &request.body;
|
|
|
|
let body_array: js_sys::Uint8Array = body_bytes.into();
|
|
|
|
let js_value: &JsValue = body_array.as_ref();
|
|
|
|
opts.body(Some(js_value));
|
2021-01-26 20:32:16 +00:00
|
|
|
}
|
|
|
|
|
2021-08-15 14:56:46 +00:00
|
|
|
let js_request = web_sys::Request::new_with_str_and_init(&request.url, &opts)?;
|
2020-11-18 20:38:29 +00:00
|
|
|
|
2021-08-15 14:56:46 +00:00
|
|
|
for h in &request.headers {
|
|
|
|
js_request.headers().set(h.0, h.1)?;
|
|
|
|
}
|
2020-11-18 20:38:29 +00:00
|
|
|
|
2021-08-15 14:56:46 +00:00
|
|
|
let window = web_sys::window().unwrap();
|
|
|
|
let response = JsFuture::from(window.fetch_with_request(&js_request)).await?;
|
|
|
|
let response: web_sys::Response = response.dyn_into()?;
|
2020-11-18 20:38:29 +00:00
|
|
|
|
|
|
|
let array_buffer = JsFuture::from(response.array_buffer()?).await?;
|
|
|
|
let uint8_array = js_sys::Uint8Array::new(&array_buffer);
|
|
|
|
let bytes = uint8_array.to_vec();
|
2020-11-17 23:43:58 +00:00
|
|
|
|
2021-08-15 14:56:46 +00:00
|
|
|
// https://developer.mozilla.org/en-US/docs/Web/API/Headers
|
|
|
|
// "Note: When Header values are iterated over, [...] values from duplicate header names are combined."
|
|
|
|
let mut headers = std::collections::BTreeMap::<String, String>::new();
|
|
|
|
let js_headers: web_sys::Headers = response.headers();
|
|
|
|
let js_iter = js_sys::try_iter(&js_headers)
|
|
|
|
.expect("headers try_iter")
|
|
|
|
.expect("headers have an iterator");
|
|
|
|
|
|
|
|
for item in js_iter {
|
|
|
|
let item = item.expect("headers iterator");
|
|
|
|
let array: js_sys::Array = item.into();
|
|
|
|
let v: Vec<JsValue> = array.to_vec();
|
|
|
|
|
|
|
|
let mut key = v[0]
|
|
|
|
.as_string()
|
|
|
|
.ok_or_else(|| JsValue::from_str("headers name"))?;
|
|
|
|
let value = v[1]
|
|
|
|
.as_string()
|
|
|
|
.ok_or_else(|| JsValue::from_str("headers value"))?;
|
|
|
|
|
|
|
|
// for easy lookup
|
|
|
|
key.make_ascii_lowercase();
|
|
|
|
headers.insert(key, value);
|
|
|
|
}
|
2020-11-17 23:43:58 +00:00
|
|
|
|
|
|
|
Ok(Response {
|
2020-11-18 20:38:29 +00:00
|
|
|
url: response.url(),
|
|
|
|
ok: response.ok(),
|
|
|
|
status: response.status(),
|
2021-08-15 14:56:46 +00:00
|
|
|
status_text: response.status_text(),
|
2020-11-18 20:38:29 +00:00
|
|
|
bytes,
|
2021-08-15 14:56:46 +00:00
|
|
|
headers,
|
2020-11-17 23:43:58 +00:00
|
|
|
})
|
|
|
|
}
|
2020-12-31 13:31:11 +00:00
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
pub(crate) struct WebHttp {}
|
|
|
|
|
|
|
|
impl epi::backend::Http for WebHttp {
|
|
|
|
fn fetch_dyn(
|
|
|
|
&self,
|
|
|
|
request: Request,
|
|
|
|
on_done: Box<dyn FnOnce(Result<Response, String>) + Send>,
|
|
|
|
) {
|
|
|
|
crate::spawn_future(async move {
|
|
|
|
let result = crate::http::fetch_async(&request).await;
|
|
|
|
on_done(result)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|