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

Remove EndpointGroupRegistry, allow specifying EndpointGroup #2381

Merged
merged 28 commits into from
Jan 13, 2020

Conversation

trustin
Copy link
Member

@trustin trustin commented Jan 6, 2020

…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 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 EndpointGroups, 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 ClientOptions.
    • 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.

@trustin trustin added this to the 0.98.0 milestone Jan 6, 2020
@trustin trustin requested review from anuraaga, ikhoon and minwoox January 6, 2020 15:22
@trustin trustin changed the title Remove EndpointGroupRegistry and allow specifying EndpointGroup w… [WIP] Remove EndpointGroupRegistry and allow specifying EndpointGroup w… Jan 6, 2020
@trustin trustin changed the title [WIP] Remove EndpointGroupRegistry and allow specifying EndpointGroup w… [WIP] Remove EndpointGroupRegistry and allow specifying EndpointGroup Jan 6, 2020
@trustin
Copy link
Member Author

trustin commented Jan 6, 2020

Meant to create a draft PR but that big green button was so tempting. 🤣 Added [WIP] header.

…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, ...
@trustin trustin force-pushed the endpoint_group_meets_strategy branch from f5b3ab0 to 91ca122 Compare January 6, 2020 15:28
@trustin trustin force-pushed the endpoint_group_meets_strategy branch from a258f8b to cc9b8c5 Compare January 7, 2020 10:50
@trustin
Copy link
Member Author

trustin commented Jan 7, 2020

@anuraaga @minwoox @ikhoon How does it look? I think most stuff works, except armeria-retrofit which requires a base URL.

@trustin trustin force-pushed the endpoint_group_meets_strategy branch from cc9b8c5 to e49fb07 Compare January 7, 2020 13:43
@trustin
Copy link
Member Author

trustin commented Jan 7, 2020

@kojilin Because this change removed EndpointGroupRegistry completely, armeria-retrofit required quite a bit of breaking changes as well. Here's the test case that shows the new API that allows:

What do you think? Is there any use case I did not cover yet? Do you need any convenience methods in ArmeriaRetrofitBuilder?

@trustin
Copy link
Member Author

trustin commented Jan 7, 2020

Build now passes with -PnoCheckstyle option. Will remove WIP after fixing Checkstyle violations and documentation.

Copy link
Contributor

@ikhoon ikhoon left a 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. 👍

@ikhoon ikhoon self-requested a review January 8, 2020 05:01
@ikhoon
Copy link
Contributor

ikhoon commented Jan 8, 2020

Oops... I tapped Approved. 😅

Copy link
Collaborator

@anuraaga anuraaga left a 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

@codecov
Copy link

codecov bot commented Jan 8, 2020

Codecov Report

Merging #2381 into master will decrease coverage by <.01%.
The diff coverage is 67.82%.

Impacted file tree graph

@@             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
Impacted Files Coverage Δ Complexity Δ
...orp/armeria/client/ClientRequestContextCaptor.java 0% <ø> (ø) 0 <0> (ø) ⬇️
...java/com/linecorp/armeria/server/file/HttpVfs.java 28.57% <ø> (ø) 2 <0> (ø) ⬇️
.../linecorp/armeria/client/DefaultClientFactory.java 75.86% <ø> (ø) 18 <0> (ø) ⬇️
...n/java/com/linecorp/armeria/client/UserClient.java 83.33% <ø> (ø) 13 <0> (ø) ⬇️
...inecorp/armeria/client/endpoint/EndpointGroup.java 75.67% <ø> (ø) 17 <0> (ø) ⬇️
...m/linecorp/armeria/client/ClientBuilderParams.java 100% <ø> (ø) 2 <0> (ø) ⬇️
...java/com/linecorp/armeria/server/RouteBuilder.java 79.66% <ø> (ø) 51 <0> (ø) ⬇️
...a/client/endpoint/EmptyEndpointGroupException.java 83.33% <ø> (ø) 4 <0> (ø) ⬇️
...ria/client/endpoint/EndpointSelectionStrategy.java 100% <ø> (ø) 3 <0> (ø) ⬇️
.../com/linecorp/armeria/client/WebClientBuilder.java 65.45% <0%> (-12.81%) 6 <0> (ø)
... and 31 more

Continue to review full report at Codecov.

Legend - Click here to learn more
Δ = absolute <relative> (impact), ø = not affected, ? = missing data
Powered by Codecov. Last update 8d0f543...919f008. Read the comment docs.

@trustin trustin changed the title [WIP] Remove EndpointGroupRegistry and allow specifying EndpointGroup Remove EndpointGroupRegistry and allow specifying EndpointGroup Jan 9, 2020
@trustin
Copy link
Member Author

trustin commented Jan 9, 2020

This pull request is now ready for full reviews.

}

private Scheme scheme() {
return scheme == null ? Scheme.of(format, protocol) : scheme;
}

private void ensureEndpoint() {
if (endpoint == null) {
private void ensureEndpointGroup() {
Copy link
Collaborator

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)

Copy link
Member Author

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? 🤔

Copy link
Contributor

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? 😄

Copy link
Member Author

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);
Copy link
Collaborator

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(
Copy link
Collaborator

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);
      ...

Copy link
Member Author

@trustin trustin Jan 9, 2020

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?

private final HostType hostType;
@Nullable
private String authority;

private Endpoint(String groupName) {
Copy link
Contributor

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
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks nice

}

private Scheme scheme() {
return scheme == null ? Scheme.of(format, protocol) : scheme;
}

private void ensureEndpoint() {
if (endpoint == null) {
private void ensureEndpointGroup() {
Copy link
Contributor

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? 😄

*/
<T> T newClient(Scheme scheme, Endpoint endpoint, @Nullable String path, Class<T> clientType,
ClientOptions options);
Object newClient(ClientBuilderParams params);
Copy link
Contributor

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? 🤔

Copy link
Member Author

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?

Copy link
Contributor

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. 😄

@trustin
Copy link
Member Author

trustin commented Jan 11, 2020

Moved ENDPOINT_MAPPER to ClientOptions and it looks better. This will be useful to non-WebClients when:

  • A user is given with a client that talks to a single Endpoint and wants to override that.
  • We provide a way to send a request to an arbitrary Endpoint by using a single client instance in the future, e.g. THttpClient.execute("tbinary+http://foo.com/api", ...);

/**
* Returns the {@link String} that consists of path, query string and fragment.
*/
String absolutePathRef(); // Name inspired by https://stackoverflow.com/a/47545070/55808
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👍

Copy link
Contributor

@minwoox minwoox left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great job @trustin!

Copy link
Contributor

@ikhoon ikhoon left a 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

@trustin trustin changed the title Remove EndpointGroupRegistry and allow specifying EndpointGroup Remove EndpointGroupRegistry, allow specifying EndpointGroup Jan 13, 2020
@trustin trustin merged commit c714ec1 into line:master Jan 13, 2020
@trustin trustin deleted the endpoint_group_meets_strategy branch January 13, 2020 09:21
fmguerreiro pushed a commit to fmguerreiro/armeria that referenced this pull request Sep 19, 2020
…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.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants