-
Notifications
You must be signed in to change notification settings - Fork 940
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
Remove EndpointGroupRegistry
, allow specifying EndpointGroup
#2381
Conversation
EndpointGroupRegistry
and allow specifying EndpointGroup
w…EndpointGroupRegistry
and allow specifying EndpointGroup
w…
EndpointGroupRegistry
and allow specifying EndpointGroup
w…EndpointGroupRegistry
and allow specifying EndpointGroup
…
Meant to create a draft PR but that big green button was so tempting. 🤣 Added |
…hen building a client. Motivation: See line#1084 Modifications: - Remove `EndpointGroupRegistry` - Make `Endpoint` always refer to a specific host - Change the input parameter `Endpoint` to `EndpointGroup` wherever possible, to allow a user to specify an `EndpointGroup` when building a client. - `EndpointGroup` now always has its own `EndpointGroupSelector`. - `ClientRequestContext.endpointSelector()` has been replaced with `endpointGroup()`. Result: - Much cleaner API - No group name clashes - Lots of breaking changes if a user was using `EndpointGroupRegistry` extensively. To-dos: - `ClientFactory` internally requires a URI even if an `EndpointGroup` is specified. We need to clean this up. - Retrofit `@Url` annotation support - Overall Javadoc clean-up, Checkstyle, ...
f5b3ab0
to
91ca122
Compare
a258f8b
to
cc9b8c5
Compare
cc9b8c5
to
e49fb07
Compare
@kojilin Because this change removed
What do you think? Is there any use case I did not cover yet? Do you need any convenience methods in |
Build now passes with |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How does it look?
I love this style that injects EndpointGroup
to WebClient.builder(SessionProtocol, EndpointGroup)
than using EndpointGroupRegistry
.
Because it is clear and explicit. 👍
core/src/main/java/com/linecorp/armeria/client/ClientBuilder.java
Outdated
Show resolved
Hide resolved
Oops... I tapped Approved. 😅 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks! This generally looks much better. Going to take a bit more to grok the retrofit part
retrofit2/src/main/java/com/linecorp/armeria/client/retrofit2/ArmeriaRetrofitBuilder.java
Outdated
Show resolved
Hide resolved
Codecov Report
@@ Coverage Diff @@
## master #2381 +/- ##
============================================
- Coverage 73.49% 73.49% -0.01%
- Complexity 10682 10685 +3
============================================
Files 939 939
Lines 40930 40968 +38
Branches 5088 5093 +5
============================================
+ Hits 30082 30108 +26
- Misses 8246 8258 +12
Partials 2602 2602
Continue to review full report at Codecov.
|
EndpointGroupRegistry
and allow specifying EndpointGroup
…EndpointGroupRegistry
and allow specifying EndpointGroup
…
This pull request is now ready for full reviews. |
core/src/main/java/com/linecorp/armeria/client/ClientBuilder.java
Outdated
Show resolved
Hide resolved
} | ||
|
||
private Scheme scheme() { | ||
return scheme == null ? Scheme.of(format, protocol) : scheme; | ||
} | ||
|
||
private void ensureEndpoint() { | ||
if (endpoint == null) { | ||
private void ensureEndpointGroup() { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe overkill but had a thought there is probably a compile-time way of ensuring this (UriClientBuilder
vs EndpointClientBuilder
interfaces)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good point. We will also need the same for WebClientBuilder
. Maybe it's too much? 🤔
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So are we doing this or not? 😄
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not? 😅
assertThat(res.status()).isEqualTo(HttpStatus.OK); | ||
assertThat(res.contentUtf8()).isEqualTo("success"); | ||
} finally { | ||
EndpointGroupRegistry.unregister(groupName); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice cleanup :)
public ArmeriaRetrofitBuilder withClientOptions( | ||
BiFunction<String, ? super ClientOptionsBuilder, ClientOptionsBuilder> configurator) { | ||
this.configurator = requireNonNull(configurator, "configurator"); | ||
public ArmeriaRetrofitBuilder nonBaseClientFactory( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Since normal usage doesn't need to worry about this, I guess it's ok. But doesn't it seem like too advanced functionality? The asymmetry vs WebClient
(the one I pointed to above) seems unnatural too - if we really want this functionality, we could move it to a ClientOption
so it applies to both?
And while it doesn't actually solve the asymmetry problem by itself, I'm wondering whether we could just let the EndpointSelector
layer take care of it, it seems simpler to reason about and don't even need to worry about a client cache
class HostAwareEndpointGroup implements EndpointGroup {
public Endpoint select(ClientRequestContext ctx) {
switch (ctx.request().authority()) {
case "group-bar":
return groupBarEndpointGroup.select(ctx);
...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think ENDPOINT_REMAPPER
option handles this better. What do you think?
retrofit2/src/main/java/com/linecorp/armeria/client/retrofit2/ArmeriaRetrofitBuilder.java
Show resolved
Hide resolved
core/src/main/java/com/linecorp/armeria/client/DecoratingClientFactory.java
Outdated
Show resolved
Hide resolved
core/src/main/java/com/linecorp/armeria/client/DecoratingClientFactory.java
Outdated
Show resolved
Hide resolved
core/src/main/java/com/linecorp/armeria/client/DefaultClientBuilderParams.java
Show resolved
Hide resolved
private final HostType hostType; | ||
@Nullable | ||
private String authority; | ||
|
||
private Endpoint(String groupName) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
👍
public final class WebClientOptions { | ||
|
||
/** | ||
* A {@link Function} that remaps an {@link Endpoint} or an absolute URL's authority into |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks nice
core/src/main/java/com/linecorp/armeria/client/endpoint/EndpointSelectionStrategy.java
Outdated
Show resolved
Hide resolved
} | ||
|
||
private Scheme scheme() { | ||
return scheme == null ? Scheme.of(format, protocol) : scheme; | ||
} | ||
|
||
private void ensureEndpoint() { | ||
if (endpoint == null) { | ||
private void ensureEndpointGroup() { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So are we doing this or not? 😄
core/src/main/java/com/linecorp/armeria/client/ClientBuilderParams.java
Outdated
Show resolved
Hide resolved
core/src/main/java/com/linecorp/armeria/client/DefaultClientBuilderParams.java
Outdated
Show resolved
Hide resolved
core/src/main/java/com/linecorp/armeria/client/DefaultClientFactory.java
Outdated
Show resolved
Hide resolved
*/ | ||
<T> T newClient(Scheme scheme, Endpoint endpoint, @Nullable String path, Class<T> clientType, | ||
ClientOptions options); | ||
Object newClient(ClientBuilderParams params); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can ClientBuilderParams
have a type parameter for the client type so that wrong ClientBuilderParams
cannot be passed as a parameter? 🤔
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I tried but it seems to make the overall type signature more complex without much gain, like adding another type parameter to UserClient
.
Perhaps it's not worth it given a user is not supposed to create a client using a ClientBuilderParams
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah, thanks! It's fine as it is. 😄
core/src/main/java/com/linecorp/armeria/client/DefaultWebClient.java
Outdated
Show resolved
Hide resolved
core/src/main/java/com/linecorp/armeria/client/endpoint/EmptyEndpointGroupException.java
Outdated
Show resolved
Hide resolved
core/src/main/java/com/linecorp/armeria/client/endpoint/EmptyEndpointGroupException.java
Outdated
Show resolved
Hide resolved
Moved
|
/** | ||
* Returns the {@link String} that consists of path, query string and fragment. | ||
*/ | ||
String absolutePathRef(); // Name inspired by https://stackoverflow.com/a/47545070/55808 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
👍
core/src/main/java/com/linecorp/armeria/client/DefaultClientRequestContext.java
Outdated
Show resolved
Hide resolved
core/src/main/java/com/linecorp/armeria/client/endpoint/EmptyEndpointGroupException.java
Outdated
Show resolved
Hide resolved
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Great job @trustin!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Awesome work! Thanks, @trustin
EndpointGroupRegistry
and allow specifying EndpointGroup
…EndpointGroupRegistry
, allow specifying EndpointGroup
…e#2381) …when building a client. Related: line#1084 Motivation: Using a 'string' as a way to specify a target endpoint group makes it hard to build a clear dependency between an `EndpointGroup` and a `Client`. Modifications: - (Breaking) Remove `EndpointGroupRegistry` - Change the input parameter from `Endpoint` to `EndpointGroup` wherever applicable, to allow a user to specify an `EndpointGroup` when building a client, e.g. `Clients.newClient(SessionProtocol, EndpointGroup)`. - `ClientBuilderParams` now provides all properties to required for creating a derived `Client`. - (Breaking) `ClientFactory.newClient(ClientBuilderParams)` replaced all other `newClient()` methods. - A user was never supposed to use these methods but `Clients.newClient()` or `Clients.builder()` anyway. - Add various public validation methods to `ClientFactory`, which is used by other `ClientFactory` implementations and `DefaultClientBuilderParams`. - (Breaking) `Endpoint` now always refers to a single host. - 'group:groupName' notation is not used anymore. - Remove `Endpoint.ofGroup()` - Remove `Endpoint.isGroup()` - Remove `Endpoint.groupName()` - Remove `Endpoint.resolve()` - `EndpointGroup` is now an `EndpointGroupSelector`. - (Breaking) `ClientRequestContext.endpointSelector()` has been replaced with `endpointGroup()`. - When creating most `EndpointGroup`s, a user can specify an `EndpointSelectionStrategy`, which is weighted round-robin by default. - Add `EndpointGroup.selectionStrategy()` - Add `EndpointGroup.of()` which requires an `EndpointSelectionStrategy` - (Breaking) Overhaul `armeria-retrofit` - Add `ArmeriaRetrofit` which provides the factory methods for `Retrofit` and `ArmeriaRetrofitBuilder` - A user can now specify a `WebClient` instead of specifying `ClientOption`s. - A user now has much more control over how a `WebClient` for a non-base URL is created. - Miscellaneous: - Add `Clients.isUndefinedUri()` - Add `EmptyEndpointGroupException` - (Deprecation) `EndpointSelectionStrategy.ROUND_ROBIN` and `WEIGHT_ROUND_ROBIN` in favor or `unweightedRoundRobin()` and `weightRoundRobin()` - (Deprecation) Some `PropertiesEndpointGroup.of()` in favor of `PropertiesEndpointGroupBuilder`. - (Deprecation) `ZooKeeperEndpointGroup` constructors in favor of `ZooKeeperEndpointGroup.of()` and `ZooKeeperEndpointGroupBuilder`. - (Breaking) `StaticEndpointGroup` is not public anymore. Result: - Much cleaner API - No group name clashes - Lots of breaking changes if a user was using `EndpointGroupRegistry` extensively.
…when building a client.
Related: #1084
Motivation:
Using a 'string' as a way to specify a target endpoint group makes it hard to build a clear dependency between an
EndpointGroup
and aClient
.Modifications:
EndpointGroupRegistry
Endpoint
toEndpointGroup
wherever applicable, to allow a user to specify anEndpointGroup
when building a client, e.g.Clients.newClient(SessionProtocol, EndpointGroup)
.ClientBuilderParams
now provides all properties to required for creating a derivedClient
.ClientFactory.newClient(ClientBuilderParams)
replaced all othernewClient()
methods.Clients.newClient()
orClients.builder()
anyway.ClientFactory
, which is used by otherClientFactory
implementations andDefaultClientBuilderParams
.Endpoint
now always refers to a single host.Endpoint.ofGroup()
Endpoint.isGroup()
Endpoint.groupName()
Endpoint.resolve()
EndpointGroup
is now anEndpointGroupSelector
.ClientRequestContext.endpointSelector()
has been replaced withendpointGroup()
.EndpointGroup
s, a user can specify anEndpointSelectionStrategy
, which is weighted round-robin by default.EndpointGroup.selectionStrategy()
EndpointGroup.of()
which requires anEndpointSelectionStrategy
armeria-retrofit
ArmeriaRetrofit
which provides the factory methods forRetrofit
andArmeriaRetrofitBuilder
WebClient
instead of specifyingClientOption
s.WebClient
for a non-base URL is created.Clients.isUndefinedUri()
EmptyEndpointGroupException
EndpointSelectionStrategy.ROUND_ROBIN
andWEIGHT_ROUND_ROBIN
in favor orunweightedRoundRobin()
andweightRoundRobin()
PropertiesEndpointGroup.of()
in favor ofPropertiesEndpointGroupBuilder
.ZooKeeperEndpointGroup
constructors in favor ofZooKeeperEndpointGroup.of()
andZooKeeperEndpointGroupBuilder
.StaticEndpointGroup
is not public anymore.Result:
EndpointGroupRegistry
extensively.