-
Notifications
You must be signed in to change notification settings - Fork 46
Added explanation of the fetch module. #57
base: master
Are you sure you want to change the base?
Changes from 4 commits
4d04169
4b53d40
4ad3a66
55a2d8e
d9ae58d
38c39bd
6a56bf8
9e454be
cb60e6d
43a6755
5f4d130
aa5d513
af9d7be
4aabb63
3419f7c
8c42997
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,147 @@ | ||
| # The Fetch Service | ||
| ## Introduction | ||
| The fetch module can be used to make requests to a server. This allows a Yew application to connect to a backend and provide interactive functionality. | ||
|
|
||
| ## Making requests | ||
| ### Building requests | ||
| Yew has a struct `Request` (which comes from the `http` crate) that is used to 'build' requests before they can be dispatched to a server. The body type of the request must have the trait `Into<Text>`. | ||
| ```rust | ||
| use yew::services::fetch::Request; | ||
| use yew::format::Nothing; | ||
| let get_request = Request::get("https://example.com/api/v1/get/something").body(Nothing).expect("Could not build that request") | ||
| ``` | ||
| ```rust | ||
| use yew::services::fetch::Request; | ||
| use yew::format::Json; | ||
| use serde_json::json; | ||
| let post_request = Request::post("https://example.com/api/v1/post/something").header("Content-Type", "application/json").body(Json(&json!({"key": "value"}))).expect("Could not build that request.") | ||
| ``` | ||
| ### Dispatching requests | ||
| The `FetchService` provides a binding to the browser's `fetch` API. Requests can be sent using either `FetchService::fetch` or `FetchService::fetch_with_options` (`fetch_with_options` should be used where cookies need to be sent in a request). | ||
|
|
||
| `FetchService::fetch` accepts two parameters: a `Request` object and a `Callback`. The closure with which the callback is constructed must take a single parameter of type `Response`. `Response` accepts a type argument – the type must implement `From<Text>`. | ||
|
|
||
| {% hint style="info" %} | ||
| If you keep getting an error saying that "the operation was aborted" or "Error 408" this might be because the [CORS headers](https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS) of the website you are trying to access are not set correctly. | ||
| {% endhint %} | ||
|
|
||
| ```rust | ||
| use yew::prelude::*; | ||
| use yew::services::fetch::{FetchService, FetchTask}; | ||
| use serde::Deserialize; | ||
|
|
||
| #[derive(Debug, Clone)] | ||
| struct FetchServiceExample { | ||
| fetch_service: FetchService, | ||
| ft: Option<FetchTask>, | ||
| iss: Option<ISS>, | ||
| link: ComponentLink<Self> | ||
| } | ||
|
|
||
| #[derive(Deserialize, Debug, Clone)] | ||
| struct ISSPosition { | ||
| latitude: String, | ||
| longitude: String | ||
| } | ||
|
|
||
| #[derive(Deserialize, Debug, Clone)] | ||
| struct ISS { | ||
| message: String, | ||
| timestamp: i32, | ||
| iss_position: ISSPosition, | ||
| fetching: bool | ||
| } | ||
|
|
||
| #[derive(Debug, Clone)] | ||
| enum Msg { | ||
| GetLocation, | ||
| Noop, | ||
| ReceiveLocation(ISS) | ||
| } | ||
|
|
||
| impl Component for FetchServiceExample { | ||
| type Message = Msg; | ||
| type Properties = (); | ||
| fn create(_: Self::Properties, link: ComponentLink<Self>) -> Self { | ||
| Self { | ||
| fetch_service: FetchService::new(), | ||
teymour-aldridge marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
| ft: None, | ||
| iss: None, | ||
| link, | ||
| fetching: false | ||
| } | ||
| } | ||
| fn change(&mut self, _: Self::Properties) -> bool { | ||
| unimplemented!() | ||
teymour-aldridge marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
| } | ||
| fn update(&mut self, msg: Self::Message) -> bool { | ||
| match msg { | ||
| GetLocation => { | ||
| // 1. build the request | ||
| let request = Request::get("http://api.open-notify.org/iss-now.json") | ||
| .body(Nothing) | ||
| .expect("Could not build request."); | ||
| // 2. construct a callback | ||
| let callback = self.link.callback(|response: Response<Json<Result<ISS, anyhow::Error>>>| { | ||
| // split up the response into the HTTP data about the request result and data from the request | ||
| let (meta, Json(data)) = response.into_parts(); | ||
| if meta.status.is_success() { | ||
| match data.message { | ||
| "success" => { | ||
|
||
| Self::Message::ReceiveLocation(data.clone()) | ||
|
||
| } | ||
| _ => { | ||
| Self::Message::Noop | ||
| } | ||
| } | ||
| } else { | ||
| Self::Message::Noop | ||
| } | ||
| }); | ||
| // 3. pass the request and callback to the fetch service | ||
| self.fetch_service.fetch(request, callback); | ||
teymour-aldridge marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
| self.fetching = true; | ||
| // we want to redraw so that the page displays a 'fetching...' message to the user | ||
| true | ||
| } | ||
| Self::Message::ReceiveLocation(location) => { | ||
teymour-aldridge marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
| self.iss = location; | ||
| self.fetching = false; | ||
| // we want to redraw so that the page no longer says 'fetching...' | ||
|
||
| true | ||
| } | ||
| _ => false | ||
| } | ||
| } | ||
| fn view(&self) -> Html { | ||
| html! { | ||
| <> | ||
| {if self.fetching { | ||
|
||
| html! {<p>{"Fetching data..."}</p>} | ||
| } else { | ||
| html! {<p></p>} | ||
| }} | ||
| { | ||
| match self.iss { | ||
| Some(space_station) => html! { | ||
| <p>{"The ISS is at:"}</p> | ||
| <p>{format!("Latitude: {}", space_station.iss_location.latitude)}</p> | ||
| <p>{format!("Longitude: {}", space_station.iss_location.longitude)}</p> | ||
| } | ||
| None => html! { | ||
| <button onclick=self.link.callback(|_| {Self::Message::GetLocation})> | ||
| {"Where is the ISS?"} | ||
| </button> | ||
| } | ||
| } | ||
| } | ||
|
|
||
| </> | ||
| } | ||
| } | ||
| } | ||
| ``` | ||
|
|
||
jstarry marked this conversation as resolved.
Show resolved
Hide resolved
|
||
| ## Further reading | ||
jstarry marked this conversation as resolved.
Show resolved
Hide resolved
|
||
| * [The API documentation](https://docs.rs/yew/0.14.3/yew/services/fetch/index.html) | ||
| * The [dashboard](https://github.com/yewstack/yew/tree/master/examples/dashboard) and [npm_and_rest](https://github.com/yewstack/yew/tree/master/examples/web_sys/npm_and_rest) examples. | ||
Uh oh!
There was an error while loading. Please reload this page.