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

Make ClientRequestContext.authority() and host() return non-null #5969

Open
wants to merge 5 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@

package com.linecorp.armeria.client.brave;

import static com.google.common.base.MoreObjects.firstNonNull;
import static com.linecorp.armeria.client.ClientRequestContextUtil.firstNonNullException;

import java.net.InetSocketAddress;

Expand Down Expand Up @@ -72,7 +72,7 @@ public void parse(brave.http.HttpRequest request, TraceContext context, SpanCust
return;
}

span.tag(SpanTags.TAG_HTTP_HOST, firstNonNull(ctx.authority(), "UNKNOWN"))
span.tag(SpanTags.TAG_HTTP_HOST, firstNonNullException(ctx::authority, "UNKNOWN"))
Copy link
Contributor

Choose a reason for hiding this comment

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

Q) Did you change this because some were tests broken?

Copy link
Contributor Author

@chickenchickenlove chickenchickenlove Nov 12, 2024

Choose a reason for hiding this comment

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

Yes, a test case testEmptyEndpointTags() in BraveClientTest.class failed.

// BraveClientTest.class
void testEmptyEndpointTags() {
    ...
    // expected: "UNKNOWN"
    // but was: null
    assertThat(span.tag("http.host")).isEqualTo("UNKNOWN"); <--- Assertion Error
}

.tag(SpanTags.TAG_HTTP_URL, ctx.uri().toString());
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -290,14 +290,12 @@ ClientRequestContext newDerivedContext(RequestId id, @Nullable HttpRequest req,
* <li>{@link Endpoint#authority()}.</li>
* </ol>
*/
@Nullable
@UnstableApi
String authority();

/**
* Returns the host part of {@link #authority()}, without a port number.
*/
@Nullable
@UnstableApi
String host();

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
/*
* Copyright 2024 LINE Corporation
*
* LINE Corporation licenses this file to you under the Apache License,
* version 2.0 (the "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at:
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
package com.linecorp.armeria.client;

import java.util.function.Supplier;

import javax.annotation.CheckForNull;

/**
* The utility class for ClientRequestContext.
*/
public final class ClientRequestContextUtil {
Copy link
Contributor

Choose a reason for hiding this comment

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

The implementations of this class are not related to ClientRequestContext.
Would rename it to ObjectUtil?


/**
* Returns a non-null value. If an unexpected exception occurs while retrieving the first value,
* or the first value is null, this function will return the second value.
* Otherwise, it returns the first value.
*/
public static <T> T firstNonNullException(@CheckForNull Supplier<T> firstSupplier, @CheckForNull T second) {
Copy link
Contributor

Choose a reason for hiding this comment

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

Armeria uses @NonNullByDefault annotation so CheckForNull is unnecessary.

Suggested change
public static <T> T firstNonNullException(@CheckForNull Supplier<T> firstSupplier, @CheckForNull T second) {
public static <T> T firstNonNullException(Supplier<T> firstSupplier, T second) {

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks for your comments!
I didn't know that at all 🙇‍♂️

try {
if (firstSupplier != null) {
final T first = firstSupplier.get();
if (first != null) {
return first;
}
}
} catch (Exception e) {
// Ignore
}

if (second != null) {
return second;
}

throw new NullPointerException("Both parameters are null");
}

private ClientRequestContextUtil() {
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -70,13 +70,11 @@ public String fragment() {
return unwrap().fragment();
}

@Nullable
@Override
public String authority() {
return unwrap().authority();
}

@Nullable
@Override
public String host() {
return unwrap().host();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@

package com.linecorp.armeria.client.observation;

import static com.google.common.base.MoreObjects.firstNonNull;
import static com.linecorp.armeria.client.ClientRequestContextUtil.firstNonNullException;

import java.net.InetSocketAddress;

Expand Down Expand Up @@ -106,7 +106,7 @@ public KeyValues getHighCardinalityKeyValues(ClientObservationContext context) {
}
final ImmutableList.Builder<KeyValue> builder = ImmutableList.builderWithExpectedSize(expectedSize);
builder.add(HighCardinalityKeys.HTTP_PATH.withValue(ctx.path()),
HighCardinalityKeys.HTTP_HOST.withValue(firstNonNull(ctx.authority(), "UNKNOWN")),
HighCardinalityKeys.HTTP_HOST.withValue(firstNonNullException(ctx::authority, "UNKNOWN")),
HighCardinalityKeys.HTTP_URL.withValue(ctx.uri().toString()));
addIfNotNull(addressRemote, builder);
addIfNotNull(addressLocal, builder);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -483,15 +483,20 @@ private void failEarly(Throwable cause) {

// TODO(ikhoon): Consider moving the logic for filling authority to `HttpClientDelegate.exceute()`.
private void autoFillSchemeAuthorityAndOrigin() {
final String authority = authority();
if (authority != null && endpoint != null && endpoint.isIpAddrOnly()) {
// The connection will be established with the IP address but `host` set to the `Endpoint`
// could be used for SNI. It would make users send HTTPS requests with CSLB or configure a reverse
// proxy based on an authority.
final String host = SchemeAndAuthority.of(null, authority).host();
if (!NetUtil.isValidIpV4Address(host) && !NetUtil.isValidIpV6Address(host)) {
endpoint = endpoint.withHost(host);

try {
final String authority = authority();
if (endpoint != null && endpoint.isIpAddrOnly()) {
// The connection will be established with the IP address but `host` set to the `Endpoint`
// could be used for SNI. It would make users send HTTPS requests
// with CSLB or configure a reverse proxy based on an authority.
final String host = SchemeAndAuthority.of(null, authority).host();
if (!NetUtil.isValidIpV4Address(host) && !NetUtil.isValidIpV6Address(host)) {
endpoint = endpoint.withHost(host);
}
}
} catch (IllegalStateException e) {
// Just pass, because it's normal condition.
Copy link
Contributor

Choose a reason for hiding this comment

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

What do you think of creating a private method to a nullable value instead of catching the exception?

@Nullable
private String authorityOrNull();

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Wow, It is better way!
@ikhoon nim, I have a question.
Should the methods authority() and host() throw a CheckedException so that the user is forced to handle the error?

IMHO, I think UncheckedException is better than CheckedException in case. because it's easy to read.

}

final HttpHeadersBuilder headersBuilder = internalRequestHeaders.toBuilder();
Expand Down Expand Up @@ -750,7 +755,6 @@ public String fragment() {
return requestTarget().fragment();
}

@Nullable
@Override
public String authority() {
final HttpHeaders additionalRequestHeaders = this.additionalRequestHeaders;
Expand All @@ -774,6 +778,11 @@ public String authority() {
if (authority == null) {
authority = internalRequestHeaders.get(HttpHeaderNames.HOST);
}
if (authority == null) {
throw new IllegalStateException(
"ClientRequestContext may be in the process of initialization." +
"In this case, host() or authority() could be null");
}
return authority;
}

Expand All @@ -794,29 +803,25 @@ private String origin() {
return origin;
}

@Nullable
@Override
public String host() {
final String authority = authority();
if (authority == null) {
return null;
}
return SchemeAndAuthority.of(null, authority).host();
}

@Override
public URI uri() {
final String scheme = getScheme(sessionProtocol());
final String authority = authority();
final String path = path();
final String query = query();
final String fragment = fragment();
try (TemporaryThreadLocals tmp = TemporaryThreadLocals.acquire()) {
final StringBuilder buf = tmp.stringBuilder();
buf.append(scheme);
if (authority != null) {
try {
final String authority = authority();
buf.append("://").append(authority);
} else {
} catch (IllegalStateException e) {
buf.append(':');
}
buf.append(path);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,8 @@
import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.api.Assertions.assertThatThrownBy;

import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CountDownLatch;
import java.util.function.Function;
import java.util.stream.Stream;

Expand All @@ -29,6 +31,7 @@
import org.junit.jupiter.params.provider.ArgumentsSource;
import org.junit.jupiter.params.provider.ValueSource;

import com.linecorp.armeria.common.AggregatedHttpResponse;
import com.linecorp.armeria.common.HttpMethod;
import com.linecorp.armeria.common.HttpRequest;
import com.linecorp.armeria.common.RequestContext;
Expand Down Expand Up @@ -269,6 +272,31 @@ void hasOwnAttr() {
}
}

@Test
void callAuthorityShouldBeThrownDuringPartiallyInit() {
final CountDownLatch countDownLatch = new CountDownLatch(1);
final WebClient client = WebClient
.builder("http://localhost:8080")
.contextCustomizer(ctx -> {
countDownLatch.countDown();
assertThatThrownBy(ctx::host)
.isInstanceOf(IllegalStateException.class)
.hasMessage(
"ClientRequestContext may be in the process of initialization." +
"In this case, host() or authority() could be null");
assertThatThrownBy(ctx::authority)
.isInstanceOf(IllegalStateException.class)
.hasMessage(
"ClientRequestContext may be in the process of initialization." +
"In this case, host() or authority() could be null"
);
}).build();

final CompletableFuture<AggregatedHttpResponse> aggregate = client.get("/").aggregate();
assertThat(countDownLatch.getCount()).isEqualTo(0);
aggregate.cancel(true);
}

@ParameterizedTest
@ValueSource(strings = {"%", "http:///", "http://foo.com/bar"})
void updateRequestWithInvalidPath(String path) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -221,7 +221,8 @@ void testAuthorityOverridden() {
HttpMethod.POST, "/foo",
HttpHeaderNames.SCHEME, "http"));
final DefaultClientRequestContext ctx = newContext(ClientOptions.of(), request1);
assertThat(ctx.authority()).isNull();

assertThatThrownBy(() -> ctx.authority()).isInstanceOf(IllegalStateException.class);
assertThat(ctx.uri().toString()).isEqualTo("http:/foo");
assertThat(ctx.uri()).hasScheme("http").hasAuthority(null).hasPath("/foo");

Expand Down
Loading