-
Notifications
You must be signed in to change notification settings - Fork 206
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
Introduce generic fluent client generator to non-sdk codegen #463
Conversation
Hmm, the documentation test failure is a little awkward. |
The error about "documentation test in private item" is a known compiler error: rust-lang/rust#72081 I'll ignore that warning for now. |
Since the function "doesn't exist", we can't link to it. Arguably, the docs should only be tested with all features enabled, but for now just don't try to link to `native_tls`.
The standard syntax doesn't work: rust-lang/rust#86120
...c/main/kotlin/software/amazon/smithy/rust/codegen/smithy/generators/HttpProtocolGenerator.kt
Outdated
Show resolved
Hide resolved
This is a pretty awesome PR and I'm excited to see more smithy work happening. I do think the tower components would be a good fit here, especially for a smithy client. I found reviewing the PR challenging but not because of the code itself, which is clean and well documented. The generics and bounds make it pretty hard to jump in and understand the code. Going "full tower" has pros / cons. Pros: being able to use off the shelf tower components + enable users to customize the middleware stack. Cons: very intense generics. In the end, the trade off calculus depends on who will be interacting with smithy-client. Will it be common for services engineers to have to use smithy-client directly or is smithy-client more for sharing code between projects that use smithy-client to expose specialized client libs (e.g. the AWS Rust SDK). As long as engineers that are new to Rust and want to build something with smithy don't have to see these generics as part of the getting started experience, then the trade off might make sense. |
Yeah, it's definitely a step up in complexity. Though in some sense it's fundamental — if we want the non-sdk "fluent" client to have swappable middleware (for authn/z), connectors (for hyper vs testing), and retry policies, those will all need to have some bound to represent the behavior those swappable pieces must exhibit. The most onerous bound in here, I think, is the fact that the middleware is a But at the same time, I think developers will only actually face those bounds if they are looking to implement their own middleware, in which case implementing Layer in the right way isn't all that onerous or error-prone (see Honestly, I think the biggest ergonomics pain with the current implementation is a) all the repeated Back to your original query:
This, I think, is exactly the right question. I think it is very unlikely that service engineers will work with Concretely, when someone generates a client for some Smithy service let mw = AwsMiddleware;
let mw = mw.layer(TracingMiddleware);
let client = Builder::new().https().middleware(mw).build(); That's not trivial, but it's also not too onerous, and we can provide good docs for that (and already do provide some in this PR). Specifically, what helps is that I'm expecting most consumers will want to take some existing middleware that provides authn/z, and then just add some custom bits for their particular use-case. Which guards them from most of the complexity. Nevertheless, writing this code does take a bit of understanding of tower (which hopefully the coming guides will help with), so a complete Rust newcomer will probably struggle a bit with it, as it does require understanding what But, once someone has written that code to construct the client, and you have a An aside about trait objects — I don't think trait objects help simplify the API here. They remove the generic type parameter, true, but the same bounds would still need to be there on the |
With the extra "helpful" bound, we also end up enforcing that C implements Service, but since we're in a builder, C may not have been set yet, and may be (), which in turn means that it isn't Service. So users would end up with an error if they write: Builder::new().middleware_fn(|r| r).https().build() but it would work with Builder::new().https().middleware_fn(|r| r).build() which is silly.
LGTM! I'm very excited for this! |
Replaced by #496 |
This change adds a
FluentClientGenerator
tocodegen
. The generated client uses a new crate,smithy_client
, which now contains most of what used to be inaws_hyper
.aws_hyper
, in turn, is now a thin wrapper aroundsmithy_client
that mainly exposes type aliases with generic parameters already filled in. It's public interface is mostly unchanged (notice thataws-hyper/tests/e2e_test.rs
has not changed.smithy_client
makes the client generic over three type parameters:Connector
, which encapsulates the behavior exposed by theconn
submodule previously. This is essentially aMiddleware
, which encapsulates the layers of the service stack that will vary between different Smithy deployments, such as request signing, adding a user agent, etc. This type is essentially aLayer
type is there so that theMiddleware
can be injected in the existing request stack (alongsideConnector
,DispatchLayer
, and friends). It is instantiated to a newAwsMiddleware
type inaws_hyper
that encapsulates the behavior that used to be hard-coded in https://github.com/awslabs/smithy-rs/blob/5ef157dc49f1f36611b365f8eb34407f9ff942e1/aws/rust-runtime/aws-hyper/src/lib.rs#L130-L137RetryPolicy
, which encapsulates the retry policy for the client. It defaults tosmithy_client::retry::Standard
, which is exactly equivalent to the oldaws_hyper::RetryHandleFactory
, but can be swapped out for anything that can instantiate new values of a type that implementstower::Policy
.The exact requirements of each of these generic types are represented in the new
smithy_client::bounds
module, which is used to make the trait bounds on methods easier to read (rather than listing out fifteen different bounds). Consumers should never implemented the traits inbounds::
, but should instead implementtower::Service
,tower::Layer
, and others as appropriate.So, given
smithy_client::Client<Connector, Middleware, RetryPolicy>
,aws_hyper::Client
now becomes:which takes care of hiding most of the generic bounds. The remaining one,
C
, is useful to allow consumers to specify their own connection backend in a zero-overhead way. To continue to support the convenient type-erased API that existed previously,smithy_client
also exposesDynConnector
(andDynMiddleware
) which represents a dynamically-dispatched version of aConnector
. This is used as the default type forClient
'sC
parameter, so users will not need to specify it unless they specifically want to use a custom type there.Client::https
now returns aClient<DynConnector>
(which is whatStandardClient
now holds) instead of aClient<conn::Standard>
, but this is unlikely to make a difference to consumers in practice.smithy_client
also introduces a builder-style constructor with a twist. Rather than take values, the builder takes types (and values of those types).Builder
is generic over<C, M, R>
just likeClient
, and each method consumesself
and returns a newBuilder
type with the appropriate type parameter changed depending on the method (e.g.,fn connector<C2>(self, C2) -> Builder<C2, M, R>
).Builder::build
gives aClient
.Builder
also comes with convenience methods likeBuilder::rustls
for setting the connector (C
) to a Rustls hyper connector,Builder::hyper
for using a customhyper::Client
, andBuilder::connector_fn
for using aasync fn(http::Request) -> http::Response
as the connector (handy for testing). It also exposesmap_connector
andmap_middleware
for wrapping an already-set connector or middleware.aws_hyper
also exposes aBuilder
through a simple type alias tosmithy_client::Builder
withM
andR
set andC
defaulting toDynConnector
.The fluent generator for
codegen
generates a fluent client that is itself generic overC
,M
, andR
. It implementsFrom<smithy_client::Client<C, M, R>>
so thatsmithy_client::Builder
can be used to full effect, as well as methods similar to the existing ones (from_conf_and_conn
) except that they takesmithy_client::Client
instead. The fluent generator for the sdk now generates aClient
that is generic overC
, but with a default ofDynConnector
. Its constructors remain the same, except that the type forconn
in the relevant methods is nowC
instead ofconn::Standard
.Oh, yeah, and
smithy_client
makeshyper
an optional dependency, since all it's used for is if consumers specifically want to use a hyper connector forC
.With this all in place,
aws_hyper
is now trivial enough that it could (and should) probably be inlined into the generated client directly rather than be its own crate. Furthermore, the fluent client generator for the sdk could likely just be a slight customization to the one generated bycodegen
rather than a re-implementation of all the generation bits. But let's leave that for follow-on work.By submitting this pull request, I confirm that you can use, modify, copy, and redistribute this contribution, under the terms of your choice.