-
Notifications
You must be signed in to change notification settings - Fork 38.3k
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
Document custom HttpServiceArgumentResolver usage #34227
Comments
If you want something in between like a dedicated |
Yes, I did not find anything else used for HTTP interfaces.
But it does not work that way though, at least from what I can see. There does not exist any methods on
This is what this issue is requesting from Spring Framework. To provide the ability to provide an object ( |
Sorry, my bad. It doesn't make sense to pass RequestEntity into RestClient as the two provide a similar API for building the request. RequestEntity is useful to provide a similar experience with the RestTemplate. We could support RequestEntity as an argument, but the main value of an HTTP interface is to indicate the inputs required for the endpoint. If you generalize that how would does user has to know what to pass? If you generalize it all the way into RequestEntity then what value do you get for an HTTP interface? |
I understand and agree with that. However, I think this falls apart by allowing/accepting a generic Line 86 in d5da602
That does not really indicate the required input since a map is just that, a map with no meaning, but I digress. What I'm trying to get at here, how does one express the required inputs without having excessively long method arguments? Here's a concrete example: https://developer.usps.com/addressesv3 interface AddressesService {
@GetMapping(path = "/addresses/v3/address", consumes = MediaType.APPLICATION_JSON_VALUE)
Address standardizeAddress(@RequestParam String firm,
@RequestParam String streetAddress,
@RequestParam String secondaryAddress,
@RequestParam String city,
@RequestParam String state,
@RequestParam String urbanization,
@RequestParam String ZIPCode,
@RequestParam String ZIPPlus4);
} That's just not nice to look at which led me to creating wrapper objects that I mentioned originally. This then led me to thinking that maybe Spring Framework could provide support for interface AddressesService {
@GetMapping(path = "/addresses/v3/address", consumes = MediaType.APPLICATION_JSON_VALUE)
Address standardizeAddress(HttpRequestValues values);
}
record AddressRequest(...) {
public HttpRequestValues toHttpRequestValues() {
return HttpRequestValues.builder()
.addRequestParameter(...)
.build();
}
}
AddressesService addressService = ....
addressService.standardizeAddress(new AddressRequest(...).toHttpRequestValues()) This just avoids what I originally had above where a Given all that though, I think the best approach is what was said initially to create a custom argument resolver to add values to However, consider repurpsing this issue or a new one to at minimum document that |
We're supporting map as a way to easily multiple values for
I agree, this
I think this is a sane approach, especially if you can provide to API users a sensible type (with a builder?). You can then model a complex HTTP request and still design an HTTP interface that has clear semantics. I'm repurposing this issue. |
When an API requires multiple values, the number of arguments for HTTP service method can be a bit painful. For example:
The service method can be simplified to:
Which is better, but I think it would be easier or nicer even to allow a single argument that encapsulates all of the request details.
That argument could also override the predefined ones such as the
consumes
orpath
from the exchange annotation. This is more or less whatUriBuilderFactory
does today for the URL.If I'm not mistaken,
RestTemplate
provided something similar throughRequestEntity
.Note I know that you can drop down to the 'lower level'
RestClient
to fully customize the request as indicated in this table in the documentation. But providing that control at the HTTP interface level I think will be a welcome addition.Looking at Framework's code,
HttpServiceArgumentResolver
is responsible for consuming the various annotations and applying those values toHttpRequestValues
. So aHttpServiceArgumentResolver
that accepts aHttpRequestValues
or similar may work I think.My company has variety of modern and legacy APIs and Web services. While we try to conform to some standard when modernizing legacy services or APIs, there still exists APIs that require a lot of information in their request.
To work around this, I started wrapping the request details into an intermediary object and then delegate to the service method. This leads to cleaner code (IMO). For example:
The text was updated successfully, but these errors were encountered: