This makes src/lib.rs the primary source for crate-level documentation. I've generated README.md with `cargo readme > README.md`. Since links to specific documentation items should be relative when possible, but must be absolute in README.md, I've used the new syntax for intra-rustdoc links (https://github.com/rust-lang/rfcs/blob/master/text/1946-intra-rustdoc-links.md), along with a README.tpl that sets up those links to point at the absolute versions. `cargo readme` uses the README.tpl by default. I've also rewritten the crate level docs, removing some TODO information at the bottom, and moving the license information to CONTRIBUTING.md.
168 lines
6.9 KiB
Markdown
168 lines
6.9 KiB
Markdown
[comment]: # (README.md is autogenerated from src/lib.rs by `cargo readme > README.md`)
|
|
|
|
# ureq
|
|
|
|
A simple, safe HTTP client.
|
|
|
|
Ureq's first priority is being easy for you to use. It's great for
|
|
anyone who wants a low-overhead HTTP client that just gets the job done. Works
|
|
very well with HTTP APIs. Its features include cookies, JSON, HTTP proxies,
|
|
HTTPS, and charset decoding.
|
|
|
|
Ureq is in pure Rust for safety and ease of understanding. It avoids using
|
|
`unsafe` directly. It uses blocking I/O instead of async I/O, because that keeps
|
|
the API simple and and keeps dependencies to a minimum. For TLS, ureq uses
|
|
[rustls].
|
|
|
|
### Usage
|
|
|
|
In its simplest form, ureq looks like this:
|
|
|
|
```rust
|
|
let body: String = ureq::get("http://example.com")
|
|
.set("Accept", "text/html")
|
|
.call()?
|
|
.into_string()?;
|
|
```
|
|
|
|
For more involved tasks, you'll want to create an [Agent]. An Agent
|
|
holds a connection pool for reuse, and a cookie store if you use the
|
|
"cookies" feature. An Agent can be cheaply cloned due to an internal
|
|
[Arc](std::sync::Arc) and all clones of an Agent share state among each other. Creating
|
|
an Agent also allows setting options like the TLS configuration.
|
|
|
|
```rust
|
|
use ureq::{Agent, AgentBuilder};
|
|
use std::time::Duration;
|
|
|
|
let agent: Agent = ureq::AgentBuilder::new()
|
|
.timeout_read(Duration::from_secs(5))
|
|
.timeout_write(Duration::from_secs(5))
|
|
.build();
|
|
let body: String = agent.get("http://example.com/page")
|
|
.call()?
|
|
.into_string()?;
|
|
|
|
// Reuses the connection from previous request.
|
|
let response: String = agent.put("http://example.com/upload")
|
|
.set("Authorization", "example-token")
|
|
.call()?
|
|
.into_string()?;
|
|
```
|
|
|
|
Ureq supports sending and receiving json, if you enable the "json" feature:
|
|
|
|
```rust
|
|
// Requires the `json` feature enabled.
|
|
let resp: String = ureq::post("http://myapi.example.com/ingest")
|
|
.set("X-My-Header", "Secret")
|
|
.send_json(ureq::json!({
|
|
"name": "martin",
|
|
"rust": true
|
|
}))?
|
|
.into_string()?;
|
|
```
|
|
|
|
### Features
|
|
|
|
To enable a minimal dependency tree, some features are off by default.
|
|
You can control them when including ureq as a dependency.
|
|
|
|
`ureq = { version = "*", features = ["json", "charset"] }`
|
|
|
|
* `tls` enables https. This is enabled by default.
|
|
* `cookies` enables cookies.
|
|
* `json` enables [Response::into_json()] and [Request::send_json()] via serde_json.
|
|
* `charset` enables interpreting the charset part of the Content-Type header
|
|
(e.g. `Content-Type: text/plain; charset=iso-8859-1`). Without this, the
|
|
library defaults to Rust's built in `utf-8`.
|
|
|
|
## Plain requests
|
|
|
|
Most standard methods (GET, POST, PUT etc), are supported as functions from the
|
|
top of the library ([get()], [post()], [put()], etc).
|
|
|
|
These top level http method functions create a [Request] instance
|
|
which follows a build pattern. The builders are finished using:
|
|
|
|
* [`.call()`][Request::call()] without a request body.
|
|
* [`.send()`][Request::send()] with a request body as [Read][std::io::Read] (chunked encoding support for non-known sized readers).
|
|
* [`.send_string()`][Request::send_string()] body as string.
|
|
* [`.send_bytes()`][Request::send_bytes()] body as bytes.
|
|
* [`.send_form()`][Request::send_form()] key-value pairs as application/x-www-form-urlencoded.
|
|
|
|
## JSON
|
|
|
|
By enabling the `ureq = { version = "*", features = ["json"] }` feature,
|
|
the library supports serde json.
|
|
|
|
* [`request.send_json()`][Request::send_json()] send body as serde json.
|
|
* [`response.into_json()`][Response::into_json()] transform response to json.
|
|
|
|
## Content-Length and Transfer-Encoding
|
|
|
|
The library will send a Content-Length header on requests with bodies of
|
|
known size, in other words, those sent with
|
|
[`.send_string()`][Request::send_string()],
|
|
[`.send_bytes()`][Request::send_bytes()],
|
|
[`.send_form()`][Request::send_form()], or
|
|
[`.send_json()`][Request::send_json()]. If you send a
|
|
request body with [`.send()`][Request::send()],
|
|
which takes a [Read][std::io::Read] of unknown size, ureq will send Transfer-Encoding:
|
|
chunked, and encode the body accordingly. Bodyless requests
|
|
(GETs and HEADs) are sent with [`.call()`][Request::call()]
|
|
and ureq adds neither a Content-Length nor a Transfer-Encoding header.
|
|
|
|
If you set your own Content-Length or Transfer-Encoding header before
|
|
sending the body, ureq will respect that header by not overriding it,
|
|
and by encoding the body or not, as indicated by the headers you set.
|
|
|
|
```rust
|
|
let resp = ureq::post("http://my-server.com/ingest")
|
|
.set("Transfer-Encoding", "chunked")
|
|
.send_string("Hello world");
|
|
```
|
|
|
|
## Character encoding
|
|
|
|
By enabling the `ureq = { version = "*", features = ["charset"] }` feature,
|
|
the library supports sending/receiving other character sets than `utf-8`.
|
|
|
|
For [`response.into_string()`][Response::into_string()] we read the
|
|
header `Content-Type: text/plain; charset=iso-8859-1` and if it contains a charset
|
|
specification, we try to decode the body using that encoding. In the absence of, or failing
|
|
to interpret the charset, we fall back on `utf-8`.
|
|
|
|
Similarly when using [`request.send_string()`][Request::send_string()],
|
|
we first check if the user has set a `; charset=<whatwg charset>` and attempt
|
|
to encode the request body using that.
|
|
|
|
------------------------------------------------------------------------------
|
|
|
|
Ureq is inspired by other great HTTP clients like
|
|
[superagent](http://visionmedia.github.io/superagent/) and
|
|
[the fetch API](https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API).
|
|
|
|
If ureq is not what you're looking for, check out these other Rust HTTP clients:
|
|
[surf](https://crates.io/crates/surf), [reqwest](https://crates.io/crates/reqwest),
|
|
[isahc](https://crates.io/crates/isahc), [attohttpc](https://crates.io/crates/attohttpc),
|
|
[actix-web](https://crates.io/crates/actix-web), and [hyper](https://crates.io/crates/hyper).
|
|
|
|
|
|
[rustls]: https://docs.rs/rustls/
|
|
[std::sync::Arc]: https://doc.rust-lang.org/stable/alloc/sync/struct.Arc.html
|
|
[std::io::Read]: https://doc.rust-lang.org/stable/std/io/trait.Read.html
|
|
[Agent]: https://docs.rs/ureq/latest/ureq/struct.Agent.html
|
|
[get()]: https://docs.rs/ureq/latest/ureq/fn.get.html
|
|
[post()]: https://docs.rs/ureq/latest/ureq/fn.post.html
|
|
[put()]: https://docs.rs/ureq/latest/ureq/fn.put.html
|
|
[Request]: https://docs.rs/ureq/latest/ureq/struct.Request.html
|
|
[Request::call()]: https://docs.rs/ureq/latest/ureq/struct.Request.html#method.call
|
|
[Request::send()]: https://docs.rs/ureq/latest/ureq/struct.Request.html#method.send
|
|
[Request::send_bytes()]: https://docs.rs/ureq/latest/ureq/struct.Request.html#method.send_bytes
|
|
[Request::send_string()]: https://docs.rs/ureq/latest/ureq/struct.Request.html#method.send_string
|
|
[Request::send_json()]: https://docs.rs/ureq/latest/ureq/struct.Request.html#method.send_json
|
|
[Request::send_form()]: https://docs.rs/ureq/latest/ureq/struct.Request.html#method.send_form
|
|
[Response::into_json()]: https://docs.rs/ureq/latest/ureq/struct.Response.html#method.into_json
|
|
[Response::into_string()]: https://docs.rs/ureq/latest/ureq/struct.Response.html#method.into_string
|