From 778539f452e6391d30aa48f72f07b9cfad3681af Mon Sep 17 00:00:00 2001 From: Harry Barber <106155934+hlbarber@users.noreply.github.com> Date: Fri, 26 Aug 2022 17:53:34 +0100 Subject: [PATCH] Add protocol specific routers (#1666) * Add protocol specific routers * Replace internals of `Router` with protocol specific routers --- .../aws-smithy-http-server/src/protocols.rs | 12 + .../aws-smithy-http-server/src/response.rs | 6 + .../src/routing/future.rs | 22 +- .../aws-smithy-http-server/src/routing/mod.rs | 213 +++++------------- .../src/routing/routers/aws_json.rs | 154 +++++++++++++ .../src/routing/routers/mod.rs | 163 ++++++++++++++ .../src/routing/routers/rest.rs | 149 ++++++++++++ .../src/runtime_error.rs | 4 - 8 files changed, 544 insertions(+), 179 deletions(-) create mode 100644 rust-runtime/aws-smithy-http-server/src/routing/routers/aws_json.rs create mode 100644 rust-runtime/aws-smithy-http-server/src/routing/routers/mod.rs create mode 100644 rust-runtime/aws-smithy-http-server/src/routing/routers/rest.rs diff --git a/rust-runtime/aws-smithy-http-server/src/protocols.rs b/rust-runtime/aws-smithy-http-server/src/protocols.rs index 5f099f8eb4..d9bae8258d 100644 --- a/rust-runtime/aws-smithy-http-server/src/protocols.rs +++ b/rust-runtime/aws-smithy-http-server/src/protocols.rs @@ -7,6 +7,18 @@ use crate::rejection::MissingContentTypeReason; use crate::request::RequestParts; +/// [AWS REST JSON 1.0 Protocol](https://awslabs.github.io/smithy/2.0/aws/protocols/aws-restjson1-protocol.html). +pub struct AwsRestJson1; + +/// [AWS REST XML Protocol](https://awslabs.github.io/smithy/2.0/aws/protocols/aws-restxml-protocol.html). +pub struct AwsRestXml; + +/// [AWS JSON 1.0 Protocol](https://awslabs.github.io/smithy/2.0/aws/protocols/aws-json-1_0-protocol.html). +pub struct AwsJson10; + +/// [AWS JSON 1.1 Protocol](https://awslabs.github.io/smithy/2.0/aws/protocols/aws-json-1_1-protocol.html). +pub struct AwsJson11; + /// Supported protocols. #[derive(Debug, Clone, Copy)] pub enum Protocol { diff --git a/rust-runtime/aws-smithy-http-server/src/response.rs b/rust-runtime/aws-smithy-http-server/src/response.rs index fab5f5e0f2..4c7bab848d 100644 --- a/rust-runtime/aws-smithy-http-server/src/response.rs +++ b/rust-runtime/aws-smithy-http-server/src/response.rs @@ -36,3 +36,9 @@ use crate::body::BoxBody; #[doc(hidden)] pub type Response = http::Response; + +/// A protocol aware function taking `self` to [`http::Response`]. +pub trait IntoResponse { + /// Performs a conversion into a [`http::Response`]. + fn into_response(self) -> http::Response; +} diff --git a/rust-runtime/aws-smithy-http-server/src/routing/future.rs b/rust-runtime/aws-smithy-http-server/src/routing/future.rs index a0e6345042..95d4cbcf5f 100644 --- a/rust-runtime/aws-smithy-http-server/src/routing/future.rs +++ b/rust-runtime/aws-smithy-http-server/src/routing/future.rs @@ -33,29 +33,11 @@ */ //! Future types. -use crate::body::BoxBody; -use futures_util::future::Either; -use http::{Request, Response}; -use std::{convert::Infallible, future::ready}; -use tower::util::Oneshot; +use super::Route; pub use super::{into_make_service::IntoMakeService, route::RouteFuture}; -type OneshotRoute = Oneshot, Request>; -type ReadyResponse = std::future::Ready, Infallible>>; - opaque_future! { /// Response future for [`Router`](super::Router). - pub type RouterFuture = - futures_util::future::Either, ReadyResponse>; -} - -impl RouterFuture { - pub(super) fn from_oneshot(future: Oneshot, Request>) -> Self { - Self::new(Either::Left(future)) - } - - pub(super) fn from_response(response: Response) -> Self { - Self::new(Either::Right(ready(Ok(response)))) - } + pub type RouterFuture = super::routers::RoutingFuture, B>; } diff --git a/rust-runtime/aws-smithy-http-server/src/routing/mod.rs b/rust-runtime/aws-smithy-http-server/src/routing/mod.rs index 31c2d94f66..0c65e4aff9 100644 --- a/rust-runtime/aws-smithy-http-server/src/routing/mod.rs +++ b/rust-runtime/aws-smithy-http-server/src/routing/mod.rs @@ -8,18 +8,17 @@ //! [Smithy specification]: https://awslabs.github.io/smithy/1.0/spec/core/http-traits.html use self::request_spec::RequestSpec; -use self::tiny_map::TinyMap; +use self::routers::{aws_json::AwsJsonRouter, rest::RestRouter, RoutingService}; use crate::body::{boxed, Body, BoxBody, HttpBody}; use crate::error::BoxError; -use crate::protocols::Protocol; -use crate::runtime_error::{RuntimeError, RuntimeErrorKind}; -use http::{Request, Response, StatusCode}; +use crate::protocols::{AwsJson10, AwsJson11, AwsRestJson1, AwsRestXml}; + +use http::{Request, Response}; use std::{ convert::Infallible, task::{Context, Poll}, }; use tower::layer::Layer; -use tower::util::ServiceExt; use tower::{Service, ServiceBuilder}; use tower_http::map_response_body::MapResponseBodyLayer; @@ -31,6 +30,7 @@ mod lambda_handler; pub mod request_spec; mod route; +mod routers; mod tiny_map; pub use self::lambda_handler::LambdaHandler; @@ -61,11 +61,6 @@ pub struct Router { routes: Routes, } -// This constant determines when the `TinyMap` implementation switches from being a `Vec` to a -// `HashMap`. This is chosen to be 15 as a result of the discussion around -// https://github.com/awslabs/smithy-rs/pull/1429#issuecomment-1147516546 -const ROUTE_CUTOFF: usize = 15; - /// Protocol-aware routes types. /// /// RestJson1 and RestXml routes are stored in a `Vec` because there can be multiple matches on the @@ -75,10 +70,10 @@ const ROUTE_CUTOFF: usize = 15; /// directly found in the `X-Amz-Target` HTTP header. #[derive(Debug)] enum Routes { - RestXml(Vec<(Route, RequestSpec)>), - RestJson1(Vec<(Route, RequestSpec)>), - AwsJson10(TinyMap, ROUTE_CUTOFF>), - AwsJson11(TinyMap, ROUTE_CUTOFF>), + RestXml(RoutingService>, AwsRestXml>), + RestJson1(RoutingService>, AwsRestJson1>), + AwsJson10(RoutingService>, AwsJson10>), + AwsJson11(RoutingService>, AwsJson11>), } impl Clone for Router { @@ -104,29 +99,6 @@ impl Router where B: Send + 'static, { - /// Return the correct, protocol-specific "Not Found" response for an unknown operation. - fn unknown_operation(&self) -> RouterFuture { - let protocol = match &self.routes { - Routes::RestJson1(_) => Protocol::RestJson1, - Routes::RestXml(_) => Protocol::RestXml, - Routes::AwsJson10(_) => Protocol::AwsJson10, - Routes::AwsJson11(_) => Protocol::AwsJson11, - }; - let error = RuntimeError { - protocol, - kind: RuntimeErrorKind::UnknownOperation, - }; - RouterFuture::from_response(error.into_response()) - } - - /// Return the HTTP error response for non allowed method. - fn method_not_allowed(&self) -> RouterFuture { - RouterFuture::from_response({ - let mut res = Response::new(crate::body::empty()); - *res.status_mut() = StatusCode::METHOD_NOT_ALLOWED; - res - }) - } /// Convert this router into a [`MakeService`], that is a [`Service`] whose /// response is another service. /// @@ -155,46 +127,21 @@ where NewResBody::Error: Into, { let layer = ServiceBuilder::new() - .layer_fn(Route::new) .layer(MapResponseBodyLayer::new(boxed)) .layer(layer); match self.routes { - Routes::RestJson1(routes) => { - let routes = routes - .into_iter() - .map(|(route, request_spec)| (Layer::layer(&layer, route), request_spec)) - .collect(); - Router { - routes: Routes::RestJson1(routes), - } - } - Routes::RestXml(routes) => { - let routes = routes - .into_iter() - .map(|(route, request_spec)| (Layer::layer(&layer, route), request_spec)) - .collect(); - Router { - routes: Routes::RestXml(routes), - } - } - Routes::AwsJson10(routes) => { - let routes = routes - .into_iter() - .map(|(operation, route)| (operation, Layer::layer(&layer, route))) - .collect(); - Router { - routes: Routes::AwsJson10(routes), - } - } - Routes::AwsJson11(routes) => { - let routes = routes - .into_iter() - .map(|(operation, route)| (operation, Layer::layer(&layer, route))) - .collect(); - Router { - routes: Routes::AwsJson11(routes), - } - } + Routes::RestJson1(routes) => Router { + routes: Routes::RestJson1(routes.map(|router| router.layer(layer).boxed())), + }, + Routes::RestXml(routes) => Router { + routes: Routes::RestXml(routes.map(|router| router.layer(layer).boxed())), + }, + Routes::AwsJson10(routes) => Router { + routes: Routes::AwsJson10(routes.map(|router| router.layer(layer).boxed())), + }, + Routes::AwsJson11(routes) => Router { + routes: Routes::AwsJson11(routes.map(|router| router.layer(layer).boxed())), + }, } } @@ -211,18 +158,14 @@ where ), >, { - let mut routes: Vec<(Route, RequestSpec)> = routes - .into_iter() - .map(|(svc, request_spec)| (Route::from_box_clone_service(svc), request_spec)) - .collect(); - - // Sort them once by specifity, with the more specific routes sorted before the less - // specific ones, so that when routing a request we can simply iterate through the routes - // and pick the first one that matches. - routes.sort_by_key(|(_route, request_spec)| std::cmp::Reverse(request_spec.rank())); - + let svc = RoutingService::new( + routes + .into_iter() + .map(|(svc, request_spec)| (request_spec, Route::from_box_clone_service(svc))) + .collect(), + ); Self { - routes: Routes::RestJson1(routes), + routes: Routes::RestJson1(svc), } } @@ -239,18 +182,14 @@ where ), >, { - let mut routes: Vec<(Route, RequestSpec)> = routes - .into_iter() - .map(|(svc, request_spec)| (Route::from_box_clone_service(svc), request_spec)) - .collect(); - - // Sort them once by specifity, with the more specific routes sorted before the less - // specific ones, so that when routing a request we can simply iterate through the routes - // and pick the first one that matches. - routes.sort_by_key(|(_route, request_spec)| std::cmp::Reverse(request_spec.rank())); - + let svc = RoutingService::new( + routes + .into_iter() + .map(|(svc, request_spec)| (request_spec, Route::from_box_clone_service(svc))) + .collect(), + ); Self { - routes: Routes::RestXml(routes), + routes: Routes::RestXml(svc), } } @@ -267,13 +206,15 @@ where ), >, { - let routes = routes - .into_iter() - .map(|(svc, operation)| (operation, Route::from_box_clone_service(svc))) - .collect(); + let svc = RoutingService::new( + routes + .into_iter() + .map(|(svc, operation)| (operation, Route::from_box_clone_service(svc))) + .collect(), + ); Self { - routes: Routes::AwsJson10(routes), + routes: Routes::AwsJson10(svc), } } @@ -290,13 +231,15 @@ where ), >, { - let routes = routes - .into_iter() - .map(|(svc, operation)| (operation, Route::from_box_clone_service(svc))) - .collect(); + let svc = RoutingService::new( + routes + .into_iter() + .map(|(svc, operation)| (operation, Route::from_box_clone_service(svc))) + .collect(), + ); Self { - routes: Routes::AwsJson11(routes), + routes: Routes::AwsJson11(svc), } } } @@ -316,55 +259,15 @@ where #[inline] fn call(&mut self, req: Request) -> Self::Future { - match &self.routes { + let fut = match &mut self.routes { // REST routes. - Routes::RestJson1(routes) | Routes::RestXml(routes) => { - let mut method_not_allowed = false; - - // Loop through all the routes and validate if any of them matches. Routes are already ranked. - for (route, request_spec) in routes { - match request_spec.matches(&req) { - request_spec::Match::Yes => { - return RouterFuture::from_oneshot(route.clone().oneshot(req)); - } - request_spec::Match::MethodNotAllowed => method_not_allowed = true, - // Continue looping to see if another route matches. - request_spec::Match::No => continue, - } - } - - if method_not_allowed { - // The HTTP method is not correct. - self.method_not_allowed() - } else { - // In any other case return the `RuntimeError::UnknownOperation`. - self.unknown_operation() - } - } + Routes::RestJson1(routes) => routes.call(req), + Routes::RestXml(routes) => routes.call(req), // AwsJson routes. - Routes::AwsJson10(routes) | Routes::AwsJson11(routes) => { - if req.uri() == "/" { - // Check the request method for POST. - if req.method() == http::Method::POST { - // Find the `x-amz-target` header. - if let Some(target) = req.headers().get("x-amz-target") { - if let Ok(target) = target.to_str() { - // Lookup in the `TinyMap` for a route for the target. - let route = routes.get(target); - if let Some(route) = route { - return RouterFuture::from_oneshot(route.clone().oneshot(req)); - } - } - } - } else { - // The HTTP method is not POST. - return self.method_not_allowed(); - } - } - // In any other case return the `RuntimeError::UnknownOperation`. - self.unknown_operation() - } - } + Routes::AwsJson10(routes) => routes.call(req), + Routes::AwsJson11(routes) => routes.call(req), + }; + RouterFuture::new(fut) } } @@ -376,7 +279,7 @@ mod rest_tests { routing::request_spec::*, }; use futures_util::Future; - use http::{HeaderMap, Method}; + use http::{HeaderMap, Method, StatusCode}; use std::pin::Pin; /// Helper function to build a `Request`. Used in other test modules. @@ -601,7 +504,7 @@ mod awsjson_tests { use super::*; use crate::body::boxed; use futures_util::Future; - use http::{HeaderMap, HeaderValue, Method}; + use http::{HeaderMap, HeaderValue, Method, StatusCode}; use pretty_assertions::assert_eq; use std::pin::Pin; diff --git a/rust-runtime/aws-smithy-http-server/src/routing/routers/aws_json.rs b/rust-runtime/aws-smithy-http-server/src/routing/routers/aws_json.rs new file mode 100644 index 0000000000..eb83d0c178 --- /dev/null +++ b/rust-runtime/aws-smithy-http-server/src/routing/routers/aws_json.rs @@ -0,0 +1,154 @@ +/* + * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. + * SPDX-License-Identifier: Apache-2.0 + */ + +use std::convert::Infallible; + +use http::header::ToStrError; +use thiserror::Error; +use tower::{Layer, Service}; + +use crate::{ + body::{empty, BoxBody}, + extension::RuntimeErrorExtension, + protocols::{AwsJson10, AwsJson11}, + response::IntoResponse, + routing::{tiny_map::TinyMap, Route}, +}; + +use super::Router; + +/// An AWS JSON routing error. +#[derive(Debug, Error)] +pub enum Error { + /// Relative URI was not "/". + #[error("relative URI is not \"/\"")] + NotRootUrl, + /// Method was not `POST`. + #[error("method not POST")] + MethodNotAllowed, + /// Missing the `x-amz-target` header. + #[error("missing the \"x-amz-target\" header")] + MissingHeader, + /// Unable to parse header into UTF-8. + #[error("failed to parse header: {0}")] + InvalidHeader(ToStrError), + /// Operation not found. + #[error("operation not found")] + NotFound, +} + +impl IntoResponse for Error { + fn into_response(self) -> http::Response { + match self { + Error::MethodNotAllowed => super::method_disallowed(), + _ => http::Response::builder() + .status(http::StatusCode::NOT_FOUND) + .header(http::header::CONTENT_TYPE, "application/x-amz-json-1.0") + .extension(RuntimeErrorExtension::new( + super::UNKNOWN_OPERATION_EXCEPTION.to_string(), + )) + .body(empty()) + .expect("invalid HTTP response for AWS JSON routing error; please file a bug report under https://github.com/awslabs/smithy-rs/issues"), + } + } +} + +impl IntoResponse for Error { + fn into_response(self) -> http::Response { + match self { + Error::MethodNotAllowed => super::method_disallowed(), + _ => http::Response::builder() + .status(http::StatusCode::NOT_FOUND) + .header(http::header::CONTENT_TYPE, "application/x-amz-json-1.1") + .extension(RuntimeErrorExtension::new( + super::UNKNOWN_OPERATION_EXCEPTION.to_string(), + )) + .body(empty()) + .expect("invalid HTTP response for AWS JSON routing error; please file a bug report under https://github.com/awslabs/smithy-rs/issues"), + } + } +} + +// This constant determines when the `TinyMap` implementation switches from being a `Vec` to a +// `HashMap`. This is chosen to be 15 as a result of the discussion around +// https://github.com/awslabs/smithy-rs/pull/1429#issuecomment-1147516546 +const ROUTE_CUTOFF: usize = 15; + +/// A [`Router`] supporting [`AWS JSON 1.0`] and [`AWS JSON 1.1`] protocols. +/// +/// [AWS JSON 1.0]: https://awslabs.github.io/smithy/2.0/aws/protocols/aws-json-1_0-protocol.html +/// [AWS JSON 1.1]: https://awslabs.github.io/smithy/2.0/aws/protocols/aws-json-1_1-protocol.html +#[derive(Debug, Clone)] +pub struct AwsJsonRouter { + routes: TinyMap, +} + +impl AwsJsonRouter { + /// Applies a [`Layer`] uniformly to all routes. + pub fn layer(self, layer: L) -> AwsJsonRouter + where + L: Layer, + { + AwsJsonRouter { + routes: self + .routes + .into_iter() + .map(|(key, route)| (key, layer.layer(route))) + .collect(), + } + } + + /// Applies type erasure to the inner route using [`Route::new`]. + pub fn boxed(self) -> AwsJsonRouter> + where + S: Service, Response = http::Response, Error = Infallible>, + S: Send + Clone + 'static, + S::Future: Send + 'static, + { + AwsJsonRouter { + routes: self.routes.into_iter().map(|(key, s)| (key, Route::new(s))).collect(), + } + } +} + +impl Router for AwsJsonRouter +where + S: Clone, +{ + type Service = S; + type Error = Error; + + fn match_route(&self, request: &http::Request) -> Result { + // The URI must be root, + if request.uri() != "/" { + return Err(Error::NotRootUrl); + } + + // Only `Method::POST` is allowed. + if request.method() != http::Method::POST { + return Err(Error::MethodNotAllowed); + } + + // Find the `x-amz-target` header. + let target = request.headers().get("x-amz-target").ok_or(Error::MissingHeader)?; + let target = target.to_str().map_err(Error::InvalidHeader)?; + + // Lookup in the `TinyMap` for a route for the target. + let route = self.routes.get(target).ok_or(Error::NotFound)?; + Ok(route.clone()) + } +} + +impl FromIterator<(String, S)> for AwsJsonRouter { + #[inline] + fn from_iter>(iter: T) -> Self { + Self { + routes: iter + .into_iter() + .map(|(svc, request_spec)| (svc, request_spec)) + .collect(), + } + } +} diff --git a/rust-runtime/aws-smithy-http-server/src/routing/routers/mod.rs b/rust-runtime/aws-smithy-http-server/src/routing/routers/mod.rs new file mode 100644 index 0000000000..a6d0d2fc07 --- /dev/null +++ b/rust-runtime/aws-smithy-http-server/src/routing/routers/mod.rs @@ -0,0 +1,163 @@ +/* + * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. + * SPDX-License-Identifier: Apache-2.0 + */ + +use std::{ + error::Error, + fmt, + future::{ready, Future, Ready}, + marker::PhantomData, + pin::Pin, + task::{Context, Poll}, +}; + +use futures_util::future::Either; +use http::Response; +use tower::{util::Oneshot, Service, ServiceExt}; +use tracing::debug; + +use crate::{body::BoxBody, response::IntoResponse}; + +pub mod aws_json; +pub mod rest; + +const UNKNOWN_OPERATION_EXCEPTION: &str = "UnknownOperationException"; + +/// Constructs common response to method disallowed. +fn method_disallowed() -> http::Response { + let mut responses = http::Response::default(); + *responses.status_mut() = http::StatusCode::METHOD_NOT_ALLOWED; + responses +} + +/// An interface for retrieving an inner [`Service`] given a [`http::Request`]. +pub trait Router { + type Service; + type Error; + + /// Matches a [`http::Request`] to a target [`Service`]. + fn match_route(&self, request: &http::Request) -> Result; +} + +/// A [`Service`] using the a [`Router`] `R` to redirect messages to specific routes. +/// +/// The `Protocol` parameter is used to determine +pub struct RoutingService { + router: R, + _protocol: PhantomData, +} + +impl fmt::Debug for RoutingService +where + R: fmt::Debug, +{ + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + f.debug_struct("RoutingService") + .field("router", &self.router) + .field("_protocol", &self._protocol) + .finish() + } +} + +impl Clone for RoutingService +where + R: Clone, +{ + fn clone(&self) -> Self { + Self { + router: self.router.clone(), + _protocol: PhantomData, + } + } +} + +impl RoutingService { + /// Creates a [`RoutingService`] from a [`Router`]. + pub fn new(router: R) -> Self { + Self { + router, + _protocol: PhantomData, + } + } + + /// Maps a [`Router`] using a closure. + pub fn map(self, f: F) -> RoutingService + where + F: FnOnce(R) -> RNew, + { + RoutingService { + router: f(self.router), + _protocol: PhantomData, + } + } +} + +type EitherOneshotReady = Either< + Oneshot>, + Ready>>::Response, >>::Error>>, +>; + +pin_project_lite::pin_project! { + pub struct RoutingFuture where S: Service> { + #[pin] + inner: EitherOneshotReady + } +} + +impl RoutingFuture +where + S: Service>, +{ + /// Creates a [`RoutingFuture`] from [`ServiceExt::oneshot`]. + pub(super) fn from_oneshot(future: Oneshot>) -> Self { + Self { + inner: Either::Left(future), + } + } + + /// Creates a [`RoutingFuture`] from [`Service::Response`]. + pub(super) fn from_response(response: S::Response) -> Self { + Self { + inner: Either::Right(ready(Ok(response))), + } + } +} + +impl Future for RoutingFuture +where + S: Service>, +{ + type Output = ::Output; + + fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { + self.project().inner.poll(cx) + } +} + +impl Service> for RoutingService +where + R: Router, + R::Service: Service, Response = http::Response> + Clone, + R::Error: IntoResponse

+ Error, +{ + type Response = Response; + type Error = >>::Error; + type Future = RoutingFuture; + + fn poll_ready(&mut self, _cx: &mut Context<'_>) -> Poll> { + Poll::Ready(Ok(())) + } + + fn call(&mut self, req: http::Request) -> Self::Future { + match self.router.match_route(&req) { + // Successfully routed, use the routes `Service::call`. + Ok(ok) => RoutingFuture::from_oneshot(ok.oneshot(req)), + // Failed to route, use the `R::Error`s `IntoResponse

`. + Err(error) => { + debug!(%error, "failed to route"); + RoutingFuture::from_response(error.into_response()) + } + } + } +} diff --git a/rust-runtime/aws-smithy-http-server/src/routing/routers/rest.rs b/rust-runtime/aws-smithy-http-server/src/routing/routers/rest.rs new file mode 100644 index 0000000000..53d16de55f --- /dev/null +++ b/rust-runtime/aws-smithy-http-server/src/routing/routers/rest.rs @@ -0,0 +1,149 @@ +/* + * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. + * SPDX-License-Identifier: Apache-2.0 + */ + +use std::convert::Infallible; + +use thiserror::Error; +use tower::{Layer, Service}; + +use crate::{ + body::{empty, BoxBody}, + extension::RuntimeErrorExtension, + protocols::{AwsRestJson1, AwsRestXml}, + response::IntoResponse, + routing::{ + request_spec::{Match, RequestSpec}, + Route, + }, +}; + +use super::Router; + +/// An AWS REST routing error. +#[derive(Debug, Error)] +pub enum Error { + /// Operation not found. + #[error("operation not found")] + NotFound, + /// Method was not allowed. + #[error("method was not allowed")] + MethodNotAllowed, +} + +impl IntoResponse for Error { + fn into_response(self) -> http::Response { + match self { + Error::NotFound => http::Response::builder() + .status(http::StatusCode::NOT_FOUND) + .header(http::header::CONTENT_TYPE, "application/json") + .header("X-Amzn-Errortype", super::UNKNOWN_OPERATION_EXCEPTION) + .extension(RuntimeErrorExtension::new( + super::UNKNOWN_OPERATION_EXCEPTION.to_string(), + )) + .body(crate::body::to_boxed("{}")) + .expect("invalid HTTP response for REST JSON routing error; please file a bug report under https://github.com/awslabs/smithy-rs/issues"), + Error::MethodNotAllowed => super::method_disallowed(), + } + } +} + +impl IntoResponse for Error { + fn into_response(self) -> http::Response { + match self { + Error::NotFound => http::Response::builder() + .status(http::StatusCode::NOT_FOUND) + .header(http::header::CONTENT_TYPE, "application/xml") + .extension(RuntimeErrorExtension::new( + super::UNKNOWN_OPERATION_EXCEPTION.to_string(), + )) + .body(empty()) + .expect("invalid HTTP response for REST JSON routing error; please file a bug report under https://github.com/awslabs/smithy-rs/issues"), + Error::MethodNotAllowed => super::method_disallowed(), + } + } +} + +/// A [`Router`] supporting [`AWS REST JSON 1.0`] and [`AWS REST XML`] protocols. +/// +/// [AWS REST JSON 1.0]: https://awslabs.github.io/smithy/2.0/aws/protocols/aws-restjson1-protocol.html +/// [AWS REST XML]: https://awslabs.github.io/smithy/2.0/aws/protocols/aws-restxml-protocol.html +#[derive(Debug, Clone)] +pub struct RestRouter { + routes: Vec<(RequestSpec, S)>, +} + +impl RestRouter { + /// Applies a [`Layer`] uniformly to all routes. + pub fn layer(self, layer: L) -> RestRouter + where + L: Layer, + { + RestRouter { + routes: self + .routes + .into_iter() + .map(|(request_spec, route)| (request_spec, layer.layer(route))) + .collect(), + } + } + + /// Applies type erasure to the inner route using [`Route::new`]. + pub fn boxed(self) -> RestRouter> + where + S: Service, Response = http::Response, Error = Infallible>, + S: Send + Clone + 'static, + S::Future: Send + 'static, + { + RestRouter { + routes: self.routes.into_iter().map(|(spec, s)| (spec, Route::new(s))).collect(), + } + } +} + +impl Router for RestRouter +where + S: Clone, +{ + type Service = S; + type Error = Error; + + fn match_route(&self, request: &http::Request) -> Result { + let mut method_allowed = true; + + for (request_spec, route) in &self.routes { + match request_spec.matches(request) { + // Match found. + Match::Yes => return Ok(route.clone()), + // Match found, but method disallowed. + Match::MethodNotAllowed => method_allowed = false, + // Continue looping to see if another route matches. + Match::No => continue, + } + } + + if method_allowed { + Err(Error::NotFound) + } else { + Err(Error::MethodNotAllowed) + } + } +} + +impl FromIterator<(RequestSpec, S)> for RestRouter { + #[inline] + fn from_iter>(iter: T) -> Self { + let mut routes: Vec<(RequestSpec, S)> = iter + .into_iter() + .map(|(request_spec, svc)| (request_spec, svc)) + .collect(); + + // Sort them once by specificity, with the more specific routes sorted before the less + // specific ones, so that when routing a request we can simply iterate through the routes + // and pick the first one that matches. + routes.sort_by_key(|(request_spec, _route)| std::cmp::Reverse(request_spec.rank())); + + Self { routes } + } +} diff --git a/rust-runtime/aws-smithy-http-server/src/runtime_error.rs b/rust-runtime/aws-smithy-http-server/src/runtime_error.rs index 5ecf8d0df3..78e94bc9fb 100644 --- a/rust-runtime/aws-smithy-http-server/src/runtime_error.rs +++ b/rust-runtime/aws-smithy-http-server/src/runtime_error.rs @@ -25,8 +25,6 @@ use crate::{protocols::Protocol, response::Response}; #[derive(Debug)] pub enum RuntimeErrorKind { - /// The requested operation does not exist. - UnknownOperation, /// Request failed to deserialize or response failed to serialize. Serialization(crate::Error), /// As of writing, this variant can only occur upon failure to extract an @@ -45,7 +43,6 @@ impl RuntimeErrorKind { match self { RuntimeErrorKind::Serialization(_) => "SerializationException", RuntimeErrorKind::InternalFailure(_) => "InternalFailureException", - RuntimeErrorKind::UnknownOperation => "UnknownOperationException", RuntimeErrorKind::NotAcceptable => "NotAcceptableException", } } @@ -62,7 +59,6 @@ impl RuntimeError { let status_code = match self.kind { RuntimeErrorKind::Serialization(_) => http::StatusCode::BAD_REQUEST, RuntimeErrorKind::InternalFailure(_) => http::StatusCode::INTERNAL_SERVER_ERROR, - RuntimeErrorKind::UnknownOperation => http::StatusCode::NOT_FOUND, RuntimeErrorKind::NotAcceptable => http::StatusCode::NOT_ACCEPTABLE, };