use wasm_bindgen::prelude::*; pub use epi::http::{Request, Response}; /// NOTE: Ok(..) is returned on network error. /// Err is only for failure to use the fetch api. pub async fn fetch_async(request: &Request) -> Result { fetch_jsvalue(request) .await .map_err(|err| err.as_string().unwrap_or_default()) } /// NOTE: Ok(..) is returned on network error. /// Err is only for failure to use the fetch api. async fn fetch_jsvalue(request: &Request) -> Result { let Request { method, url, body } = request; // https://rustwasm.github.io/wasm-bindgen/examples/fetch.html use wasm_bindgen::JsCast; use wasm_bindgen_futures::JsFuture; let mut opts = web_sys::RequestInit::new(); opts.method(method); opts.mode(web_sys::RequestMode::Cors); if !body.is_empty() { opts.body(Some(&JsValue::from_str(body))); } let request = web_sys::Request::new_with_str_and_init(&url, &opts)?; request.headers().set("Accept", "*/*")?; let window = web_sys::window().unwrap(); let response = JsFuture::from(window.fetch_with_request(&request)).await?; assert!(response.is_instance_of::()); let response: web_sys::Response = response.dyn_into().unwrap(); // // TODO: support binary get // let body = JsFuture::from(response.text()?).await?; // let body = body.as_string().unwrap_or_default(); let array_buffer = JsFuture::from(response.array_buffer()?).await?; let uint8_array = js_sys::Uint8Array::new(&array_buffer); let bytes = uint8_array.to_vec(); let header_content_type = response .headers() .get("Content-Type") .ok() .flatten() .unwrap_or_default(); let text = if header_content_type.starts_with("text") || header_content_type == "application/javascript" { String::from_utf8(bytes.clone()).ok() } else { None }; Ok(Response { status_text: response.status_text(), url: response.url(), ok: response.ok(), status: response.status(), header_content_type, bytes, text, }) } // ---------------------------------------------------------------------------- pub(crate) struct WebHttp {} impl epi::backend::Http for WebHttp { fn fetch_dyn( &self, request: Request, on_done: Box) + Send>, ) { crate::spawn_future(async move { let result = crate::http::fetch_async(&request).await; on_done(result) }); } }