Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add type safe state extractor #1155

Merged
merged 50 commits into from
Aug 17, 2022
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
Show all changes
50 commits
Select commit Hold shift + click to select a range
2eb4c14
begin threading the state through
davidpdrsn Jul 9, 2022
d9cc822
Pass state to extractors
davidpdrsn Jul 9, 2022
1f95766
make state extractor work
davidpdrsn Jul 10, 2022
18bb70b
make sure nesting with different states work
davidpdrsn Jul 11, 2022
61bb214
impl Service for MethodRouter<()>
davidpdrsn Jul 11, 2022
2117c97
Fix some of axum-macro's tests
davidpdrsn Jul 11, 2022
09e1898
Implement more traits for `State`
davidpdrsn Jul 14, 2022
f5cd34f
Update examples to use `State`
davidpdrsn Jul 14, 2022
0081c0d
consistent naming of request body param
davidpdrsn Jul 26, 2022
e9b0b6f
swap type params
davidpdrsn Jul 26, 2022
2c2187d
Default the state param to ()
davidpdrsn Jul 26, 2022
8c2027e
fix docs references
davidpdrsn Jul 26, 2022
c1fe325
Docs and handler state refactoring
davidpdrsn Jul 26, 2022
96a41f4
docs clean ups
davidpdrsn Jul 26, 2022
838201c
more consistent naming
davidpdrsn Jul 26, 2022
ea8160c
when does MethodRouter implement Service?
davidpdrsn Jul 26, 2022
0f80479
add missing docs
davidpdrsn Jul 26, 2022
7de141c
use `Router`'s default state type param
davidpdrsn Jul 26, 2022
9fd0c5e
changelog
davidpdrsn Jul 26, 2022
619f267
don't use default type param for FromRequest and RequestParts
davidpdrsn Jul 27, 2022
8896190
Merge branch 'main' into router-state-extractor
davidpdrsn Aug 11, 2022
32ace95
fix examples
davidpdrsn Aug 11, 2022
dec3e46
minor docs tweaks
davidpdrsn Aug 11, 2022
703d738
clarify how to convert handlers into services
davidpdrsn Aug 11, 2022
28df3c5
group methods in one impl block
davidpdrsn Aug 11, 2022
b1c5f65
make sure merged `MethodRouter`s can access state
davidpdrsn Aug 11, 2022
3f94c1c
fix docs link
davidpdrsn Aug 11, 2022
ec1cfba
test merge with same state type
davidpdrsn Aug 11, 2022
babb073
Document how to access state from middleware
davidpdrsn Aug 12, 2022
c055d0b
Port cookie extractors to use state to extract keys (#1250)
davidpdrsn Aug 12, 2022
47d8ef5
Updates ECOSYSTEM with a new sample project (#1252)
danipardo Aug 12, 2022
e4b2075
Avoid unhelpful compiler suggestion (#1251)
jplatte Aug 12, 2022
c021616
Merge branch 'main' into router-state-extractor
davidpdrsn Aug 12, 2022
a489cf7
fix docs typo
davidpdrsn Aug 12, 2022
b71dc5e
Merge branch 'main' into router-state-extractor
davidpdrsn Aug 15, 2022
1ed3e06
document how library authors should access state
davidpdrsn Aug 15, 2022
94f050f
Add `RequestParts::with_state`
davidpdrsn Aug 16, 2022
a7d558a
Merge branch 'main' into router-state-extractor
davidpdrsn Aug 17, 2022
aeab0c4
fix example
davidpdrsn Aug 17, 2022
719192f
Merge branch 'main' into router-state-extractor
davidpdrsn Aug 17, 2022
02bc5ba
apply suggestions from review
davidpdrsn Aug 17, 2022
9f63017
add relevant changes to axum-extra and axum-core changelogs
davidpdrsn Aug 17, 2022
c1074b7
Add `route_service_with_tsr`
davidpdrsn Aug 17, 2022
0668fd4
fix trybuild expectations
davidpdrsn Aug 17, 2022
b6a2f39
make sure `SpaRouter` works with routers that have state
davidpdrsn Aug 17, 2022
02a1f81
Change order of type params on FromRequest and RequestParts
davidpdrsn Aug 17, 2022
e211b15
reverse order of `RequestParts::with_state` args to match type params
davidpdrsn Aug 17, 2022
96531b7
Add `FromRef` trait (#1268)
davidpdrsn Aug 17, 2022
e23d81c
Avoid unnecessary `MethodRouter`
davidpdrsn Aug 17, 2022
34d24d2
apply suggestions from review
davidpdrsn Aug 17, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion axum-core/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0

# Unreleased

- None.
- **breaking:** `FromRequest` and `RequestParts` has a new `S` type param which
represents the state ([#1155])

[#1155]: https://github.com/tokio-rs/axum/pull/1155

# 0.2.6 (18. June, 2022)

Expand Down
4 changes: 4 additions & 0 deletions axum-extra/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -25,9 +25,13 @@ and this project adheres to [Semantic Versioning].
- **added:** `WithRejection` extractor for customizing other extractors' rejections ([#1262])
- **added:** Add sync constructors to `CookieJar`, `PrivateCookieJar`, and
`SignedCookieJar` so they're easier to use in custom middleware
- **breaking:** `Resource` has a new `S` type param which represents the state ([#1155])
- **breaking:** `RouterExt::route_with_tsr` now only accepts `MethodRouter`s ([#1155])
- **added:** `RouterExt::route_service_with_tsr` for routing to any `Service` ([#1155])
davidpdrsn marked this conversation as resolved.
Show resolved Hide resolved

[#1086]: https://github.com/tokio-rs/axum/pull/1086
[#1119]: https://github.com/tokio-rs/axum/pull/1119
[#1155]: https://github.com/tokio-rs/axum/pull/1155
[#1170]: https://github.com/tokio-rs/axum/pull/1170
[#1214]: https://github.com/tokio-rs/axum/pull/1214
[#1239]: https://github.com/tokio-rs/axum/pull/1239
Expand Down
9 changes: 5 additions & 4 deletions axum-extra/src/either.rs
Original file line number Diff line number Diff line change
Expand Up @@ -190,15 +190,16 @@ macro_rules! impl_traits_for_either {
$last:ident $(,)?
) => {
#[async_trait]
impl<B, $($ident),*, $last> FromRequest<B> for $either<$($ident),*, $last>
impl<B, S, $($ident),*, $last> FromRequest<B, S> for $either<$($ident),*, $last>
where
$($ident: FromRequest<B>),*,
$last: FromRequest<B>,
$($ident: FromRequest<B, S>),*,
$last: FromRequest<B, S>,
B: Send,
S: Send,
{
type Rejection = $last::Rejection;

async fn from_request(req: &mut RequestParts<B>) -> Result<Self, Self::Rejection> {
async fn from_request(req: &mut RequestParts<B, S>) -> Result<Self, Self::Rejection> {
$(
if let Ok(value) = req.extract().await {
return Ok(Self::$ident(value));
Expand Down
2 changes: 1 addition & 1 deletion axum-extra/src/extract/cached.rs
Original file line number Diff line number Diff line change
Expand Up @@ -155,7 +155,7 @@ mod tests {
}
}

let mut req = RequestParts::with_state(Request::new(()), ());
let mut req = RequestParts::new(Request::new(()));

let first = Cached::<Extractor>::from_request(&mut req).await.unwrap().0;
assert_eq!(COUNTER.load(Ordering::SeqCst), 1);
Expand Down
15 changes: 10 additions & 5 deletions axum-extra/src/extract/with_rejection.rs
Original file line number Diff line number Diff line change
Expand Up @@ -107,15 +107,16 @@ impl<E, R> DerefMut for WithRejection<E, R> {
}

#[async_trait]
impl<B, E, R> FromRequest<B> for WithRejection<E, R>
impl<B, E, R, S> FromRequest<B, S> for WithRejection<E, R>
where
B: Send,
E: FromRequest<B>,
S: Send,
E: FromRequest<B, S>,
R: From<E::Rejection> + IntoResponse,
{
type Rejection = R;

async fn from_request(req: &mut RequestParts<B>) -> Result<Self, Self::Rejection> {
async fn from_request(req: &mut RequestParts<B, S>) -> Result<Self, Self::Rejection> {
let extractor = req.extract::<E>().await?;
Ok(WithRejection(extractor, PhantomData))
}
Expand All @@ -134,10 +135,14 @@ mod tests {
struct TestRejection;

#[async_trait]
impl<B: Send> FromRequest<B> for TestExtractor {
impl<B, S> FromRequest<B, S> for TestExtractor
where
B: Send,
S: Send,
{
type Rejection = ();

async fn from_request(_: &mut RequestParts<B>) -> Result<Self, Self::Rejection> {
async fn from_request(_: &mut RequestParts<B, S>) -> Result<Self, Self::Rejection> {
Err(())
}
}
Expand Down
41 changes: 0 additions & 41 deletions axum-extra/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -76,47 +76,6 @@ pub mod json_lines;
#[cfg(feature = "protobuf")]
pub mod protobuf;

/// Combines two extractors or responses into a single type.
#[derive(Debug, Copy, Clone)]
pub enum Either<L, R> {
/// A value of type L.
Left(L),
/// A value of type R.
Right(R),
}

#[async_trait]
impl<L, R, B, S> FromRequest<B, S> for Either<L, R>
where
L: FromRequest<B, S>,
R: FromRequest<B, S>,
B: Send,
S: Send,
{
type Rejection = R::Rejection;

async fn from_request(req: &mut RequestParts<B, S>) -> Result<Self, Self::Rejection> {
if let Ok(l) = req.extract().await {
return Ok(Either::Left(l));
}

Ok(Either::Right(req.extract().await?))
}
}

impl<L, R> IntoResponse for Either<L, R>
where
L: IntoResponse,
R: IntoResponse,
{
fn into_response(self) -> axum::response::Response {
match self {
Self::Left(inner) => inner.into_response(),
Self::Right(inner) => inner.into_response(),
}
}
}

#[cfg(feature = "typed-routing")]
#[doc(hidden)]
pub mod __private {
Expand Down
37 changes: 35 additions & 2 deletions axum-extra/src/routing/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,13 @@

use axum::{
handler::Handler,
response::Redirect,
http::Request,
response::{IntoResponse, Redirect},
routing::{any, MethodRouter},
Router,
};
use std::future::ready;
use std::{convert::Infallible, future::ready};
use tower_service::Service;

mod resource;

Expand Down Expand Up @@ -161,6 +163,16 @@ pub trait RouterExt<S, B>: sealed::Sealed {
fn route_with_tsr(self, path: &str, method_router: MethodRouter<S, B>) -> Self
davidpdrsn marked this conversation as resolved.
Show resolved Hide resolved
where
Self: Sized;

/// Add another route to the router with an additional "trailing slash redirect" route.
///
/// This works like [`RouterExt::route_with_tsr`] but accepts any [`Service`].
fn route_service_with_tsr<T>(self, path: &str, service: T) -> Self
where
T: Service<Request<B>, Error = Infallible> + Clone + Send + 'static,
T::Response: IntoResponse,
T::Future: Send + 'static,
Self: Sized;
}

impl<S, B> RouterExt<S, B> for Router<S, B>
Expand Down Expand Up @@ -265,6 +277,27 @@ where
self.route(&format!("{}/", path), any(move || ready(redirect.clone())))
}
}

fn route_service_with_tsr<T>(mut self, path: &str, service: T) -> Self
where
T: Service<Request<B>, Error = Infallible> + Clone + Send + 'static,
T::Response: IntoResponse,
T::Future: Send + 'static,
Self: Sized,
{
self = self.route_service(path, service);

let redirect = Redirect::permanent(path);

if let Some(path_without_trailing_slash) = path.strip_suffix('/') {
self.route(
path_without_trailing_slash,
any(move || ready(redirect.clone())),
)
} else {
self.route(&format!("{}/", path), any(move || ready(redirect.clone())))
davidpdrsn marked this conversation as resolved.
Show resolved Hide resolved
}
}
}

mod sealed {
Expand Down
3 changes: 1 addition & 2 deletions axum-macros/tests/from_request/fail/double_via_attr.rs
Original file line number Diff line number Diff line change
@@ -1,8 +1,7 @@
use axum_macros::FromRequest;
use axum::extract::Extension;

#[derive(FromRequest)]
struct Extractor(#[from_request(via(Extension), via(Extension))] State);
struct Extractor(#[from_request(via(axum::Extension), via(axum::Extension))] State);

#[derive(Clone)]
struct State;
Expand Down
14 changes: 3 additions & 11 deletions axum-macros/tests/from_request/fail/double_via_attr.stderr
Original file line number Diff line number Diff line change
@@ -1,13 +1,5 @@
error: `via` specified more than once
--> tests/from_request/fail/double_via_attr.rs:5:49
--> tests/from_request/fail/double_via_attr.rs:4:55
|
5 | struct Extractor(#[from_request(via(Extension), via(Extension))] State);
| ^^^

warning: unused import: `axum::extract::Extension`
--> tests/from_request/fail/double_via_attr.rs:2:5
|
2 | use axum::extract::Extension;
| ^^^^^^^^^^^^^^^^^^^^^^^^
|
= note: `#[warn(unused_imports)]` on by default
4 | struct Extractor(#[from_request(via(axum::Extension), via(axum::Extension))] State);
| ^^^
4 changes: 2 additions & 2 deletions axum-macros/tests/from_request/fail/generic_without_via.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use axum::{body::Body, routing::get, Extension, Router};
use axum::{body::Body, routing::get, Router};
use axum_macros::FromRequest;

#[derive(FromRequest, Clone)]
Expand All @@ -7,5 +7,5 @@ struct Extractor<T>(T);
async fn foo(_: Extractor<()>) {}

fn main() {
Router::<Body>::new().route("/", get(foo));
Router::<(), Body>::new().route("/", get(foo));
}
27 changes: 5 additions & 22 deletions axum-macros/tests/from_request/fail/generic_without_via.stderr
Original file line number Diff line number Diff line change
Expand Up @@ -4,30 +4,13 @@ error: #[derive(FromRequest)] only supports generics when used with #[from_reque
5 | struct Extractor<T>(T);
| ^

warning: unused import: `Extension`
--> tests/from_request/fail/generic_without_via.rs:1:38
|
1 | use axum::{body::Body, routing::get, Extension, Router};
| ^^^^^^^^^
|
= note: `#[warn(unused_imports)]` on by default

error[E0599]: no function or associated item named `new` found for struct `Router<Body>` in the current scope
--> tests/from_request/fail/generic_without_via.rs:10:21
|
10 | Router::<Body>::new().route("/", get(foo));
| ^^^ function or associated item not found in `Router<Body>`
|
= note: the function or associated item was found for
- `Router<(), B>`

error[E0277]: the trait bound `fn(Extractor<()>) -> impl Future<Output = ()> {foo}: Handler<_, _, _>` is not satisfied
--> tests/from_request/fail/generic_without_via.rs:10:42
--> tests/from_request/fail/generic_without_via.rs:10:46
|
10 | Router::<Body>::new().route("/", get(foo));
| --- ^^^ the trait `Handler<_, _, _>` is not implemented for `fn(Extractor<()>) -> impl Future<Output = ()> {foo}`
| |
| required by a bound introduced by this call
10 | Router::<(), Body>::new().route("/", get(foo));
| --- ^^^ the trait `Handler<_, _, _>` is not implemented for `fn(Extractor<()>) -> impl Future<Output = ()> {foo}`
| |
| required by a bound introduced by this call
|
= help: the trait `Handler<T, S, B>` is implemented for `Layered<L, H, T, S, B>`
note: required by a bound in `axum::routing::get`
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use axum::{body::Body, routing::get, Extension, Router};
use axum::{body::Body, routing::get, Router};
use axum_macros::FromRequest;

#[derive(FromRequest, Clone)]
Expand All @@ -8,5 +8,5 @@ struct Extractor<T>(T);
async fn foo(_: Extractor<()>) {}

fn main() {
Router::<Body>::new().route("/", get(foo));
Router::<(), Body>::new().route("/", get(foo));
}
Original file line number Diff line number Diff line change
Expand Up @@ -4,30 +4,13 @@ error: #[derive(FromRequest)] only supports generics when used with #[from_reque
6 | struct Extractor<T>(T);
| ^

warning: unused import: `Extension`
--> tests/from_request/fail/generic_without_via_rejection.rs:1:38
|
1 | use axum::{body::Body, routing::get, Extension, Router};
| ^^^^^^^^^
|
= note: `#[warn(unused_imports)]` on by default

error[E0599]: no function or associated item named `new` found for struct `Router<Body>` in the current scope
--> tests/from_request/fail/generic_without_via_rejection.rs:11:21
|
11 | Router::<Body>::new().route("/", get(foo));
| ^^^ function or associated item not found in `Router<Body>`
|
= note: the function or associated item was found for
- `Router<(), B>`

error[E0277]: the trait bound `fn(Extractor<()>) -> impl Future<Output = ()> {foo}: Handler<_, _, _>` is not satisfied
--> tests/from_request/fail/generic_without_via_rejection.rs:11:42
--> tests/from_request/fail/generic_without_via_rejection.rs:11:46
|
11 | Router::<Body>::new().route("/", get(foo));
| --- ^^^ the trait `Handler<_, _, _>` is not implemented for `fn(Extractor<()>) -> impl Future<Output = ()> {foo}`
| |
| required by a bound introduced by this call
11 | Router::<(), Body>::new().route("/", get(foo));
| --- ^^^ the trait `Handler<_, _, _>` is not implemented for `fn(Extractor<()>) -> impl Future<Output = ()> {foo}`
| |
| required by a bound introduced by this call
|
= help: the trait `Handler<T, S, B>` is implemented for `Layered<L, H, T, S, B>`
note: required by a bound in `axum::routing::get`
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use axum::{body::Body, routing::get, Extension, Router};
use axum::{body::Body, routing::get, Router};
use axum_macros::FromRequest;

#[derive(FromRequest, Clone)]
Expand All @@ -8,5 +8,5 @@ struct Extractor<T>(T);
async fn foo(_: Extractor<()>) {}

fn main() {
Router::<Body>::new().route("/", get(foo));
Router::<(), Body>::new().route("/", get(foo));
}
Original file line number Diff line number Diff line change
Expand Up @@ -4,30 +4,13 @@ error: #[derive(FromRequest)] only supports generics when used with #[from_reque
6 | struct Extractor<T>(T);
| ^

warning: unused import: `Extension`
--> tests/from_request/fail/generic_without_via_rejection_derive.rs:1:38
|
1 | use axum::{body::Body, routing::get, Extension, Router};
| ^^^^^^^^^
|
= note: `#[warn(unused_imports)]` on by default

error[E0599]: no function or associated item named `new` found for struct `Router<Body>` in the current scope
--> tests/from_request/fail/generic_without_via_rejection_derive.rs:11:21
|
11 | Router::<Body>::new().route("/", get(foo));
| ^^^ function or associated item not found in `Router<Body>`
|
= note: the function or associated item was found for
- `Router<(), B>`

error[E0277]: the trait bound `fn(Extractor<()>) -> impl Future<Output = ()> {foo}: Handler<_, _, _>` is not satisfied
--> tests/from_request/fail/generic_without_via_rejection_derive.rs:11:42
--> tests/from_request/fail/generic_without_via_rejection_derive.rs:11:46
|
11 | Router::<Body>::new().route("/", get(foo));
| --- ^^^ the trait `Handler<_, _, _>` is not implemented for `fn(Extractor<()>) -> impl Future<Output = ()> {foo}`
| |
| required by a bound introduced by this call
11 | Router::<(), Body>::new().route("/", get(foo));
| --- ^^^ the trait `Handler<_, _, _>` is not implemented for `fn(Extractor<()>) -> impl Future<Output = ()> {foo}`
| |
| required by a bound introduced by this call
|
= help: the trait `Handler<T, S, B>` is implemented for `Layered<L, H, T, S, B>`
note: required by a bound in `axum::routing::get`
Expand Down
Original file line number Diff line number Diff line change
@@ -1,8 +1,7 @@
use axum_macros::FromRequest;
use axum::extract::Extension;

#[derive(FromRequest, Clone)]
#[from_request(rejection_derive(!Error), via(Extension))]
#[from_request(rejection_derive(!Error), via(axum::Extension))]
struct Extractor {
config: String,
}
Expand Down
Loading