View Source Req.Steps (req v0.4.14)
The collection of built-in steps.
Req is composed of three main pieces:
Req
- the high-level APIReq.Request
- the low-level API and the request structReq.Steps
- the collection of built-in steps (you're here!)
Summary
Request Steps
Sets request authentication.
Performs HTTP caching using if-modified-since
header.
Sets expected response body checksum.
Compresses the request body.
Asks the server to return compressed response.
Encodes the request body.
Signs request with AWS Signature Version 4.
Sets base URL for all requests.
Adds params to request query string.
Uses a templated request path.
Runs the request against a plug instead of over the network.
Sets the "Range" request header.
Sets the user-agent header.
Runs the request using Finch
.
Response Steps
Decodes response body based on the detected format.
Decompresses the response body based on the content-encoding
header.
Handles HTTP 4xx/5xx error responses.
Follows redirects.
Verifies the response body checksum.
Error Steps
Retries a request in face of errors.
Request Steps
Sets request authentication.
Request Options
:auth
- sets theauthorization
header:string
- sets to this value;{:basic, userinfo}
- uses Basic HTTP authentication;{:bearer, token}
- uses Bearer HTTP authentication;:netrc
- load credentials from.netrc
at path specified inNETRC
environment variable. IfNETRC
is not set, load.netrc
in user's home directory;{:netrc, path}
- load credentials frompath
Examples
iex> Req.get!("https://httpbin.org/basic-auth/foo/bar", auth: {:basic, "foo:foo"}).status
401
iex> Req.get!("https://httpbin.org/basic-auth/foo/bar", auth: {:basic, "foo:bar"}).status
200
iex> Req.get!("https://httpbin.org/bearer", auth: {:bearer, ""}).status
401
iex> Req.get!("https://httpbin.org/bearer", auth: {:bearer, "foo"}).status
200
iex> System.put_env("NETRC", "./test/my_netrc")
iex> Req.get!("https://httpbin.org/basic-auth/foo/bar", auth: :netrc).status
200
iex> Req.get!("https://httpbin.org/basic-auth/foo/bar", auth: {:netrc, "./test/my_netrc"}).status
200
Performs HTTP caching using if-modified-since
header.
Only successful (200 OK) responses are cached.
This step also prepends a response step that loads and writes the cache. Be careful when prepending other response steps, make sure the cache is loaded/written as soon as possible.
Options
:cache
- iftrue
, performs simple caching usingif-modified-since
header. Defaults tofalse
.:cache_dir
- the directory to store the cache, defaults to<user_cache_dir>/req
(see::filename.basedir/3
)
Examples
iex> url = "https://elixir-lang.org"
iex> response1 = Req.get!(url, cache: true)
iex> response2 = Req.get!(url, cache: true)
iex> response1 == response2
true
Sets expected response body checksum.
Request Options
:checksum
- if set, this is the expected response body checksum.Can be one of:
"md5:(...)"
"sha1:(...)"
"sha256:(...)"
Examples
iex> resp = Req.get!("https://httpbin.org/json", checksum: "sha1:9274ffd9cf273d4a008750f44540c4c5d4c8227c")
iex> resp.status
200
iex> Req.get!("https://httpbin.org/json", checksum: "sha1:bad")
** (Req.ChecksumMismatchError) checksum mismatch
expected: sha1:bad
actual: sha1:9274ffd9cf273d4a008750f44540c4c5d4c8227c
Compresses the request body.
Request Options
:compress_body
- if set totrue
, compresses the request body using gzip. Defaults tofalse
.
Asks the server to return compressed response.
Supported formats:
Request Options
:compressed
- if set totrue
, sets theaccept-encoding
header with compression algorithms that Req supports. Defaults totrue
.When streaming response body (
into: fun | collectable
),compressed
defaults tofalse
.
Examples
Req automatically decompresses response body (decompress_body/1
step) so let's disable that by
passing raw: true
.
By default, we ask the server to send compressed response. Let's look at the headers and the raw
body. Notice the body starts with <<31, 139>>
(<<0x1F, 0x8B>>
), the "magic bytes" for gzip:
iex> response = Req.get!("https://elixir-lang.org", raw: true)
iex> Req.Response.get_header(response, "content-encoding")
["gzip"]
iex> response.body |> binary_part(0, 2)
<<31, 139>>
Now, let's pass compressed: false
and notice the raw body was not compressed:
iex> response = Req.get!("https://elixir-lang.org", raw: true, compressed: false)
iex> response.body |> binary_part(0, 15)
"<!DOCTYPE html>"
The Brotli and Zstandard compression algorithms are also supported if the optional packages are installed:
Mix.install([
:req,
{:brotli, "~> 0.3.0"},
{:ezstd, "~> 1.0"}
])
response = Req.get!("https://httpbin.org/anything")
response.body["headers"]["Accept-Encoding"]
#=> "zstd, br, gzip"
Encodes the request body.
Request Options
:form
- if set, encodes the request body as form data (usingURI.encode_query/1
).:json
- if set, encodes the request body as JSON (usingJason.encode_to_iodata!/1
), sets theaccept
header toapplication/json
, and thecontent-type
header toapplication/json
.
Examples
iex> Req.post!("https://httpbin.org/anything", form: [x: 1]).body["form"]
%{"x" => "1"}
iex> Req.post!("https://httpbin.org/post", json: %{x: 2}).body["json"]
%{"x" => 2}
Signs request with AWS Signature Version 4.
This step requires :aws_signature
dependency:
{:aws_signature, "~> 0.3.0"}
Request Options
:aws_sigv4
- if set, the AWS options to sign request::access_key_id
- the AWS access key id.:secret_access_key
- the AWS secret access key.:service
- the AWS service.:region
- if set, AWS region. Defaults to"us-east-1"
.:datetime
- the request datetime, defaults toDateTime.utc_now(:second)
.
Examples
iex> req =
...> Req.new(
...> base_url: "https://s3.amazonaws.com",
...> aws_sigv4: [
...> access_key_id: System.get_env("AWS_ACCESS_KEY_ID"),
...> secret_access_key: System.get_env("AWS_SECRET_ACCESS_KEY"),
...> service: :s3
...> ]
...> )
iex>
iex> %{status: 200} = Req.put!(req, "/bucket1/key1", body: "Hello, World!")
iex> resp = Req.get!(req, "/bucket1/key1").body
"Hello, World!"
Request body streaming also works though content-length
header must be explicitly set:
iex> path = "a.txt"
iex> File.write!(path, String.duplicate("a", 100_000))
iex> size = File.stat!(path).size
iex> chunk_size = 10 * 1024
iex> stream = File.stream!(path, chunk_size)
iex> %{status: 200} = Req.put!(req, url: "/key1", headers: [content_length: size], body: stream)
iex> byte_size(Req.get!(req, "/bucket1/key1").body)
100_000
Sets base URL for all requests.
Request Options
:base_url
- if set, the request URL is merged with this base URL.The base url can be a string, a
%URI{}
struct, a 0-arity function, or a{mod, fun, args}
tuple describing a function to call.
Examples
iex> req = Req.new(base_url: "https://httpbin.org")
iex> Req.get!(req, url: "/status/200").status
200
iex> Req.get!(req, url: "/status/201").status
201
Adds params to request query string.
Request Options
:params
- params to add to the request query string. Defaults to[]
.
Examples
iex> Req.get!("https://httpbin.org/anything/query", params: [x: 1, y: 2]).body["args"]
%{"x" => "1", "y" => "2"}
Uses a templated request path.
Request Options
:path_params
- params to add to the templated path. Defaults to[]
.
Examples
iex> Req.get!("https://httpbin.org/status/:code", path_params: [code: 200]).status
200
Runs the request against a plug instead of over the network.
Request Options
:plug
- if set, the plug to run the request against.
Examples
This step is particularly useful to test plugs:
defmodule Echo do
def call(conn, _) do
"/" <> path = conn.request_path
Plug.Conn.send_resp(conn, 200, path)
end
end
test "echo" do
assert Req.get!("http:///hello", plug: Echo).body == "hello"
end
You can define plugs as functions too:
test "echo" do
echo = fn conn ->
"/" <> path = conn.request_path
Plug.Conn.send_resp(conn, 200, path)
end
assert Req.get!("http:///hello", plug: echo).body == "hello"
end
which is particularly useful to create HTTP service mocks, similar to tools like Bypass.
Response streaming is also supported however at the moment the entire response body is emitted as one chunk:
test "echo" do
plug = fn conn ->
conn = Plug.Conn.send_chunked(conn, 200)
{:ok, conn} = Plug.Conn.chunk(conn, "echo")
{:ok, conn} = Plug.Conn.chunk(conn, "echo")
conn
end
assert Req.get!(plug: plug, into: []).body == ["echoecho"]
end
Sets the "Range" request header.
Request Options
:range
- can be one of the following:a string - returned as is
a
first..last
range - converted to"bytes=<first>-<last>"
Examples
iex> response = Req.get!("https://httpbin.org/range/100", range: 0..3)
iex> response.status
206
iex> response.body
"abcd"
iex> Req.Response.get_header(response, "content-range")
["bytes 0-3/100"]
Sets the user-agent header.
Request Options
:user_agent
- sets theuser-agent
header. Defaults to"req/0.4.14"
.
Examples
iex> Req.get!("https://httpbin.org/user-agent").body
%{"user-agent" => "req/0.4.14"}
iex> Req.get!("https://httpbin.org/user-agent", user_agent: "foo").body
%{"user-agent" => "foo"}
Runs the request using Finch
.
This is the default Req adapter. See
"Adapter" section in the Req.Request
module documentation
for more information on adapters.
HTTP/1 Pools
On HTTP/1 connections, Finch creates a pool per {scheme, host, port}
tuple. These pools
are kept around to re-use connections as much as possible, however they are not automatically
terminated. To do so, you can configure custom Finch pool:
{:ok, _} =
Finch.start_link(
name: MyFinch,
pools: %{
default: [
# terminate idle {scheme, host, port} pool after 60s
pool_max_idle_time: 60_000
]
}
)
Req.get!("https://httpbin.org/json", finch: MyFinch)
Request Options
:finch
- the name of the Finch pool. Defaults to a pool automatically started by Req.:connect_options
- dynamically starts (or re-uses already started) Finch pool with the given connection options::timeout
- socket connect timeout in milliseconds, defaults to30_000
.:protocols
- the HTTP protocols to use, defaults to[:http1]
.:hostname
- Mint explicit hostname, seeMint.HTTP.connect/4
for more information.:transport_opts
- Mint transport options, seeMint.HTTP.connect/4
for more information.:proxy_headers
- Mint proxy headers, seeMint.HTTP.connect/4
for more information.:proxy
- Mint HTTP/1 proxy settings, a{schema, address, port, options}
tuple. SeeMint.HTTP.connect/4
for more information.:client_settings
- Mint HTTP/2 client settings, seeMint.HTTP.connect/4
for more information.
:inet6
- if set to true, uses IPv6. Defaults tofalse
. This is a shortcut for settingconnect_options: [transport_opts: [inet6: true]]
.:pool_timeout
- pool checkout timeout in milliseconds, defaults to5000
.:receive_timeout
- socket receive timeout in milliseconds, defaults to15_000
.:unix_socket
- if set, connect through the given UNIX domain socket.:finch_private
- a map or keyword list of private metadata to add to the Finch request. May be useful for adding custom data when handling telemetry withFinch.Telemetry
.:finch_request
- a function that executes the Finch request, defaults to usingFinch.request/3
.The function should accept 4 arguments:
request
- the%Req.Request{}
structfinch_request
- the Finch requestfinch_name
- the Finch namefinch_options
- the Finch options
And it should return either
{request, response}
or{request, exception}
.
Examples
Custom :receive_timeout
:
iex> Req.get!(url, receive_timeout: 1000)
Connecting through UNIX socket:
iex> Req.get!("http:///v1.41/_ping", unix_socket: "/var/run/docker.sock").body
"OK"
Connecting with custom connection options:
iex> Req.get!(url, connect_options: [timeout: 5000])
iex> Req.get!(url, connect_options: [protocols: [:http2]])
Connecting with built-in CA store (requires OTP 25+):
iex> Req.get!(url, connect_options: [transport_opts: [cacerts: :public_key.cacerts_get()]])
Stream response body using Finch.stream/5
:
fun = fn request, finch_request, finch_name, finch_options ->
fun = fn
{:status, status}, response ->
%{response | status: status}
{:headers, headers}, response ->
%{response | headers: headers}
{:data, data}, response ->
IO.puts(data)
response
end
case Finch.stream(finch_request, finch_name, Req.Response.new(), fun, finch_options) do
{:ok, response} -> {request, response}
{:error, exception} -> {request, exception}
end
end
Req.get!("https://httpbin.org/stream/10", finch_request: fun)
Response Steps
Decodes response body based on the detected format.
Supported formats:
Format | Decoder |
---|---|
json | Jason.decode!/2 |
gzip | :zlib.gunzip/1 |
tar , tgz | :erl_tar.extract/2 |
zip | :zip.unzip/2 |
csv | NimbleCSV.RFC4180.parse_string/2 (if nimble_csv is installed) |
The format is determined based on the content-type
header of the response. For example,
if the content-type
is application/json
, the response body is decoded as JSON. The built-in
decoders also understand format extensions, such as decoding as JSON for a content-type of
application/vnd.api+json
. To do this, Req falls back to MIME.extensions/1
; check the
documentation for that function for more information.
This step is disabled on response body streaming. If response body is not a binary, in other words it has been transformed by another step, it is left as is.
Request Options
:decode_body
- if set tofalse
, disables automatic response body decoding. Defaults totrue
.:decode_json
- options to pass toJason.decode!/2
, defaults to[]
.:raw
- if set totrue
, disables response body decoding. Defaults tofalse
.
Examples
Decode JSON:
iex> response = Req.get!("https://httpbin.org/json")
...> response.body["slideshow"]["title"]
"Sample Slide Show"
Decode gzip:
iex> response = Req.get!("https://httpbin.org/gzip")
...> response.body["gzipped"]
true
Decompresses the response body based on the content-encoding
header.
This step is disabled on response body streaming. If response body is not a binary, in other words it has been transformed by another step, it is left as is.
Supported formats:
Format | Decoder |
---|---|
gzip, x-gzip | :zlib.gunzip/1 |
br | :brotli.decode/1 (if brotli is installed) |
zstd | :ezstd.decompress/1 (if ezstd is installed) |
other | Returns data as is |
This step updates the following headers to reflect the changes:
content-encoding
is removedcontent-length
is removed
Options
:raw
- if set totrue
, disables response body decompression. Defaults tofalse
.
Examples
iex> response = Req.get!("https://httpbin.org/gzip")
iex> response.body["gzipped"]
true
If the brotli package is installed, Brotli is also supported:
Mix.install([
:req,
{:brotli, "~> 0.3.0"}
])
response = Req.get!("https://httpbin.org/brotli")
Req.Response.get_header(response, "content-encoding")
#=> ["br"]
response.body["brotli"]
#=> true
Handles HTTP 4xx/5xx error responses.
Request Options
:http_errors
- how to handle HTTP 4xx/5xx error responses. Can be one of the following::return
(default) - return the response:raise
- raise an error
Examples
iex> Req.get!("https://httpbin.org/status/404").status
404
iex> Req.get!("https://httpbin.org/status/404", http_errors: :raise)
** (RuntimeError) The requested URL returned error: 404
Response body: ""
Follows redirects.
The original request method may be changed to GET depending on the status code:
Code | Method handling |
---|---|
301, 302, 303 | Changed to GET |
307, 308 | Method not changed |
Request Options
:redirect
- if set tofalse
, disables automatic response redirects. Defaults totrue
.:redirect_trusted
- by default, authorization credentials are only sent on redirects with the same host, scheme and port. If:redirect_trusted
is set totrue
, credentials will be sent to any host.:redirect_log_level
- the log level to emit redirect logs at. Can also be set tofalse
to disable logging these messages. Defaults to:debug
.:max_redirects
- the maximum number of redirects, defaults to10
. If the limit is reached, the pipeline is halted and aReq.TooManyRedirectsError
exception is returned.
Examples
iex> Req.get!("http://api.github.com").status
# 23:24:11.670 [debug] redirecting to https://api.github.com/
200
iex> Req.get!("https://httpbin.org/redirect/4", max_redirects: 3)
# 23:07:59.570 [debug] redirecting to /relative-redirect/3
# 23:08:00.068 [debug] redirecting to /relative-redirect/2
# 23:08:00.206 [debug] redirecting to /relative-redirect/1
** (RuntimeError) too many redirects (3)
iex> Req.get!("http://api.github.com", redirect_log_level: false)
200
iex> Req.get!("http://api.github.com", redirect_log_level: :error)
# 23:24:11.670 [error] redirecting to https://api.github.com/
200
Verifies the response body checksum.
See checksum/1
for more information.
Error Steps
Retries a request in face of errors.
This function can be used as either or both response and error step.
Request Options
:retry
- can be one of the following::safe_transient
(default) - retry safe (GET/HEAD) requests on HTTP 408/429/500/502/503/504 responses or exceptions withreason
field set to:timeout
/:econnrefused
/:closed
.:transient
- same as:safe_transient
except retries all HTTP methods (POST, DELETE, etc.)fun
- a 2-arity function that accepts aReq.Request
and either aReq.Response
or an exception struct and returns one of the following:true
- retry with the default delay controller by default delay option described below.{:delay, milliseconds}
- retry with the given delay.false/nil
- don't retry.
false
- don't retry.
:retry_delay
- if not set, which is the default, the retry delay is determined by the value ofretry-delay
header on HTTP 429/503 responses. If the header is not set, the default delay follows a simple exponential backoff: 1s, 2s, 4s, 8s, ...:retry_delay
can be set to a function that receives the retry count (starting at 0) and returns the delay, the number of milliseconds to sleep before making another attempt.:retry_log_level
- the log level to emit retry logs at. Can also be set tofalse
to disable logging these messages. Defaults to:error
.:max_retries
- maximum number of retry attempts, defaults to3
(for a total of4
requests to the server, including the initial one.)
Examples
With default options:
iex> Req.get!("https://httpbin.org/status/500,200").status
# 19:02:08.463 [error] retry: got response with status 500, will retry in 2000ms, 2 attempts left
# 19:02:10.710 [error] retry: got response with status 500, will retry in 4000ms, 1 attempt left
200
Delay with jitter:
iex> delay = fn n -> trunc(Integer.pow(2, n) * 1000 * (1 - 0.1 * :rand.uniform())) end
iex> Req.get!("https://httpbin.org/status/500,200", retry_delay: delay).status
# 08:43:19.101 [error] retry: got response with status 500, will retry in 941ms, 2 attempts left
# 08:43:22.958 [error] retry: got response with status 500, will retry in 1877s, 1 attempt left
200