# 1.7.0
## Features
### All AUTH plugin improvements!!
#### `:auth`
The `:auth` plugin can now be used with a dynamic callable object (methods, procs...) to generate the token.
```ruby
# static token, pre 1.7.0
HTTPX.plugin(:auth).authorization("API-TOKEN")
# dynamically generate token!
HTTPX.plugin(:auth).authorization { generate_new_ephemeral_token }
```
The `.authorization` method is now syntactic sugar for a new option, `:auth_header_value`, which can be used directly, alongside a `:auth_header_type`:
```ruby
HTTPX.plugin(:auth).authorization("API-TOKEN")
HTTPX.plugin(:auth).authorization { generate_new_ephemeral_token }
HTTPX.plugin(:auth).authorization("Bearer API-TOKEN")
# same as
HTTPX.plugin(:auth, auth_header_value: "API-TOKEN")
HTTPX.plugin(:auth, auth_header_value: -> { generate_new_ephemeral_token })
HTTPX.plugin(:auth, auth_header_type: "Bearer", auth_header_value: "API-TOKEN")
```
A new option `:generate_auth_value_on_retry` (which can be passed a callable receiving a response object) is now available; when used alongside the `:retries` plugin, it'll use the callable passed to the `.authorization` method to generate a new token before retrying the request:
```ruby
authed = HTTPX.plugin(:retries).plugin(:auth, generate_auth_value_on_retry: ->(res) {
res.status == 401
}).authorization { generate_new_ephemeral_token }
authed.get("
https://example.com")
```
Read more about it in the [auth plugin wiki](
https://honeyryderchuck.gitlab.io/httpx/wiki/Auth).
#### `:oauth`
The `:oauth` plugin implementation was revamped to make use of the `:auth` plugin new functionality, in order to make managing an oauth session more seamless.
Take the following example:
```ruby
session = HTTPX.plugin(:oauth).with_oauth_options(
issuer: server.origin,
client_id: "CLIENT_ID",
client_secret: "SECRET",
)
session.get("
https://example.com") #=> will load server metadata, request an access token, and perform the request with the access token.
# 2 hours later...
session.get("
https://example.com")
# it'll reuse the same acces token, and if the request fails with 401, it'll request a new
# access token using the refresh token grant (when supported by the token issuer), and
# reperform the original request with the new access token.
```
A new option, `:oauth_options`, is now available. The same parameters previously supported by the `:oauth_session` options are supported.
The following components are therefore deprecated and scheduled for removal in a future major version:
* `:oauth_session` option
* `.oauth_auth` session method
* `.with_access_token` session method
#### `:bearer_auth`, `:digest_auth`; `:ntlm_auth`
The `:auth` plugin is now the foundation of each of these plugins, which haven't suffered major API changes.
Read more about it in the [auth plugin wiki](
https://honeyryderchuck.gitlab.io/httpx/wiki/OAuth).
### `:retries` plugin: `:retry_after` backoff algorithms
The `:retries` plugins supports two new possible values for the `:retry_after` option: `:exponential_backoff` and `:polynomial_backoff`. They'll implement the respective calculation per each retry of a given request.
```ruby
# will wait 1, 2, 4, 8, 16 seconds... depending of how many retries it can wait for
session = HTTPX.plugin(:retries, retry_after: :exponential_backoff)
```
Read more about it in the [retries plugin wiki](
https://honeyryderchuck.gitlab.io/httpx/wiki/Retries).
### Ractor compatibility
`httpx` can be used within a ractor:
```ruby
# ruby 4.0 syntax
response = Ractor.new(uri) do |uri|
HTTPX.get(uri)
end.value
```
Bear in mind that, if you're connection via HTTPS, you'll need make sure you're using version 4.0 or higher of the `openssl` gem.
The test suite isn't exhaustive for ractors yet, but most plugins should also be ractor-compatible. If they don't work, that's a bug, and you're recommended to report it.
## Improvements
* When encoding the `:json` param to send it as `application/json` payload, (example: `HTTPX.post("
https://example.com", json: { foo: "bar })`), and the method uses the `json` standard library, it'll use `JSON.generate` (instead of `JSON.dump`) to encode the JSON payload. The reason is that, unlike `JSON.dump`, it doesn't rely on access to a global mutable hash, and is therefore ractor-safe.
* `:stream` plugin: the stream response class (the object that is returned in request calls is a stream response) can be extended now. You can add a `StreamResponseMethods` method to your plugin. Read more about it in the documentation.
* The resolver name cache (used by the native and https resolvers) was remade into a LRU cache, and will therefore not keep on growing when `httpx` is used to connect to a huge number of hostnames in a process.
* the native and https DNS resolvers will ignore answers with SERVFAIL code while there are retries left (some resolvers use such error code for rate limiting).
* `:timeout` option values are now validated, and an error is raised when passing an unrecognized timeout option (which is a good layer of protection for typos).
* pool: try passing the scheduler to a thread waiting on a connection, to avoid the current case where a connection may be checked-in-then-immediately-out-after when doing multiple requests in a loop, never giving a chance to others and potentially making the pool time out.
* headers deep-freeze and dup.
## Bugfixes
* recover and close connection when an `IOError` is raised while waiting for IO readiness (could cause busy loops during HTTP/2 termination handshake).
* `:stream_bidi` plugin: improve thread-safety of buffer operations when the session is used from multiple threads.
* `:stream_bidi` plugin: added missing methods to signal in order to comply with the Selectable API (it was reported as raising `NoMethodError` under certain conditions).
* `:stream_bidi` plugin: can support non-bidirectional stream requests using the same session.
* `:stream` plugin: is now compatible with fiber scheduler engines (via the `:fiber_concurrency` plugin).
* `:stream` plugin: make sure that stream long-running requests do not share the same connection as regular threads.
* `:digest_auth` plugin: can now support qop values wrapped inside parentheses in the `www-authenticate` header (i.e. `qop="('auth',)"`).
* https resolver: handle 3XX redirect responses in HTTP DNS queries.
* https resolver: do not close HTTP connections whhich are shared across AAAA and A resolution paths when its in use by one of them.
* fix access to private method from `http-2` which was made public in more recent versions, but not in older still-supported versions.
* fixed resolver log message using a "connection" label.
* `HTTPX::Response.copy_to` will explicitly close the response at the end; given that the body file can be moved as a result, there is no guarantee that the response is still usable, so might as well just close it altogether.
* selector: avoid skipping persistent connections in the selector to deactivate due to iterate-and-modify.
## Breaking Changes
### `:digest_auth` error
The main error class for the `:digest_auth` plugin has been moved to a different location. If you were rescuing the `HTTPX::Plugins::DigestAuth::DigestError` error, you should now point to the `HTTPX::Authentication::Digest::Error`.
### `:stream` plugin: `build_request` should receive `stream: true` for stream requests
In case you're building request objects before passing them to the session, you're now forced to create them with the `:stream` option on:
```ruby
session = HTTPX.plugin(:stream)
# before
req = session.build_request("GET", "
https://example.com/stream")
session.request(req, stream: true)
# after
req = session.build_request("GET", "
https://example.com/stream", stream: true)
session.request(req)
```
Previous code may still work in a few cases, but it is not guaranteed to work on all cases.