Skip to content

Commit 01b681c

Browse files
committedOct 22, 2024
8326949: Authorization header is removed when a proxy Authenticator is set on HttpClient
Reviewed-by: dfuchs, jpai, djelinski
1 parent d10eecf commit 01b681c

File tree

9 files changed

+598
-212
lines changed

9 files changed

+598
-212
lines changed
 

‎src/java.net.http/share/classes/java/net/http/HttpClient.java

+8
Original file line numberDiff line numberDiff line change
@@ -410,6 +410,14 @@ public interface Builder {
410410
/**
411411
* Sets an authenticator to use for HTTP authentication.
412412
*
413+
* @implNote
414+
* In the JDK built-in implementation of the {@code HttpClient},
415+
* if a {@link HttpRequest} has an {@code Authorization} or {@code
416+
* Proxy-Authorization} header set then its value is used and
417+
* the {@link Authenticator} is not invoked for the corresponding
418+
* authentication. In this case, any authentication errors are returned
419+
* to the user and requests are not automatically retried.
420+
*
413421
* @param authenticator the Authenticator
414422
* @return this builder
415423
*/

‎src/java.net.http/share/classes/jdk/internal/net/http/AuthenticationFilter.java

+8
Original file line numberDiff line numberDiff line change
@@ -244,6 +244,14 @@ public HttpRequestImpl response(Response r) throws IOException {
244244
HttpHeaders hdrs = r.headers();
245245
HttpRequestImpl req = r.request();
246246

247+
if (req.getUserSetAuthFlag(SERVER) && status == UNAUTHORIZED) {
248+
// return the response. We don't handle it.
249+
return null;
250+
} else if (req.getUserSetAuthFlag(PROXY) && status == PROXY_UNAUTHORIZED) {
251+
// same
252+
return null;
253+
}
254+
247255
if (status != PROXY_UNAUTHORIZED) {
248256
if (exchange.proxyauth != null && !exchange.proxyauth.fromcache) {
249257
AuthInfo au = exchange.proxyauth;

‎src/java.net.http/share/classes/jdk/internal/net/http/Http1Request.java

+6-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2024, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -45,6 +45,8 @@
4545
import jdk.internal.net.http.common.Utils;
4646

4747
import static java.lang.String.format;
48+
import static java.net.Authenticator.RequestorType.PROXY;
49+
import static java.net.Authenticator.RequestorType.SERVER;
4850
import static java.nio.charset.StandardCharsets.US_ASCII;
4951

5052
/**
@@ -91,7 +93,6 @@ private void logHeaders(String completeHeaders) {
9193
}
9294
}
9395

94-
9596
public void collectHeaders0(StringBuilder sb) {
9697
BiPredicate<String,String> filter =
9798
connection.headerFilter(request);
@@ -104,7 +105,9 @@ public void collectHeaders0(StringBuilder sb) {
104105

105106
// Filter overridable headers from userHeaders
106107
userHeaders = HttpHeaders.of(userHeaders.map(),
107-
connection.contextRestricted(request, client));
108+
connection.contextRestricted(request));
109+
110+
Utils.setUserAuthFlags(request, userHeaders);
108111

109112
final HttpHeaders uh = userHeaders;
110113

‎src/java.net.http/share/classes/jdk/internal/net/http/HttpConnection.java

+3-3
Original file line numberDiff line numberDiff line change
@@ -356,13 +356,13 @@ BiPredicate<String,String> headerFilter(HttpRequestImpl request) {
356356
}
357357
}
358358

359-
BiPredicate<String,String> contextRestricted(HttpRequestImpl request, HttpClient client) {
359+
BiPredicate<String,String> contextRestricted(HttpRequestImpl request) {
360360
if (!isTunnel() && request.isConnect()) {
361361
// establishing a proxy tunnel
362362
assert request.proxy() == null;
363-
return Utils.PROXY_TUNNEL_RESTRICTED(client);
363+
return Utils.PROXY_TUNNEL_RESTRICTED();
364364
} else {
365-
return Utils.CONTEXT_RESTRICTED(client);
365+
return Utils.ACCEPT_ALL;
366366
}
367367
}
368368

‎src/java.net.http/share/classes/jdk/internal/net/http/HttpRequestImpl.java

+30-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2023, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2024, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -26,6 +26,7 @@
2626
package jdk.internal.net.http;
2727

2828
import java.io.IOException;
29+
import java.net.Authenticator;
2930
import java.net.InetSocketAddress;
3031
import java.net.Proxy;
3132
import java.net.ProxySelector;
@@ -47,6 +48,8 @@
4748
import jdk.internal.net.http.common.Utils;
4849
import jdk.internal.net.http.websocket.WebSocketRequest;
4950

51+
import static java.net.Authenticator.RequestorType.PROXY;
52+
import static java.net.Authenticator.RequestorType.SERVER;
5053
import static jdk.internal.net.http.common.Utils.ALLOWED_HEADERS;
5154
import static jdk.internal.net.http.common.Utils.ProxyHeaders;
5255

@@ -66,6 +69,8 @@ public class HttpRequestImpl extends HttpRequest implements WebSocketRequest {
6669
private volatile AccessControlContext acc;
6770
private final Duration timeout; // may be null
6871
private final Optional<HttpClient.Version> version;
72+
private volatile boolean userSetAuthorization;
73+
private volatile boolean userSetProxyAuthorization;
6974

7075
private static String userAgent() {
7176
PrivilegedAction<String> pa = () -> System.getProperty("java.version");
@@ -333,6 +338,30 @@ boolean isWebSocket() {
333338
return isWebSocket;
334339
}
335340

341+
/**
342+
* These flags are set if the user set an Authorization or Proxy-Authorization header
343+
* overriding headers produced by an Authenticator that was also set
344+
*
345+
* The values are checked in the AuthenticationFilter which tells the library
346+
* to return whatever response received to the user instead of causing request
347+
* to be resent, in case of error.
348+
*/
349+
public void setUserSetAuthFlag(Authenticator.RequestorType type, boolean value) {
350+
if (type == SERVER) {
351+
userSetAuthorization = value;
352+
} else {
353+
userSetProxyAuthorization = value;
354+
}
355+
}
356+
357+
public boolean getUserSetAuthFlag(Authenticator.RequestorType type) {
358+
if (type == SERVER) {
359+
return userSetAuthorization;
360+
} else {
361+
return userSetProxyAuthorization;
362+
}
363+
}
364+
336365
@Override
337366
public Optional<BodyPublisher> bodyPublisher() {
338367
return requestPublisher == null ? Optional.empty()

‎src/java.net.http/share/classes/jdk/internal/net/http/Stream.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -825,7 +825,8 @@ private OutgoingHeaders<Stream<T>> headerFrame(long contentLength) {
825825
HttpHeaders sysh = filterHeaders(h.build());
826826
HttpHeaders userh = filterHeaders(request.getUserHeaders());
827827
// Filter context restricted from userHeaders
828-
userh = HttpHeaders.of(userh.map(), Utils.CONTEXT_RESTRICTED(client()));
828+
userh = HttpHeaders.of(userh.map(), Utils.ACCEPT_ALL);
829+
Utils.setUserAuthFlags(request, userh);
829830

830831
// Don't override Cookie values that have been set by the CookieHandler.
831832
final HttpHeaders uh = userh;

‎src/java.net.http/share/classes/jdk/internal/net/http/common/Utils.java

+17-16
Original file line numberDiff line numberDiff line change
@@ -82,6 +82,8 @@
8282
import static java.lang.String.format;
8383
import static java.nio.charset.StandardCharsets.US_ASCII;
8484
import static java.util.stream.Collectors.joining;
85+
import static java.net.Authenticator.RequestorType.PROXY;
86+
import static java.net.Authenticator.RequestorType.SERVER;
8587

8688
/**
8789
* Miscellaneous utilities
@@ -210,24 +212,10 @@ private static Set<String> getDisallowedHeaders() {
210212
return true;
211213
};
212214

213-
// Headers that are not generally restricted, and can therefore be set by users,
214-
// but can in some contexts be overridden by the implementation.
215-
// Currently, only contains "Authorization" which will
216-
// be overridden, when an Authenticator is set on the HttpClient.
217-
// Needs to be BiPred<String,String> to fit with general form of predicates
218-
// used by caller.
219-
220-
public static final BiPredicate<String, String> CONTEXT_RESTRICTED(HttpClient client) {
221-
return (k, v) -> client.authenticator().isEmpty() ||
222-
(!k.equalsIgnoreCase("Authorization")
223-
&& !k.equalsIgnoreCase("Proxy-Authorization"));
224-
}
225-
226215
public record ProxyHeaders(HttpHeaders userHeaders, HttpHeaders systemHeaders) {}
227216

228-
private static final BiPredicate<String, String> HOST_RESTRICTED = (k,v) -> !"host".equalsIgnoreCase(k);
229-
public static final BiPredicate<String, String> PROXY_TUNNEL_RESTRICTED(HttpClient client) {
230-
return CONTEXT_RESTRICTED(client).and(HOST_RESTRICTED);
217+
public static final BiPredicate<String, String> PROXY_TUNNEL_RESTRICTED() {
218+
return (k,v) -> !"host".equalsIgnoreCase(k);
231219
}
232220

233221
private static final Predicate<String> IS_HOST = "host"::equalsIgnoreCase;
@@ -310,6 +298,19 @@ private static final boolean isAllowedForProxy(String name,
310298
public static final BiPredicate<String, String> NO_PROXY_HEADERS_FILTER =
311299
(n,v) -> Utils.NO_PROXY_HEADER.test(n);
312300

301+
/**
302+
* Check the user headers to see if the Authorization or ProxyAuthorization
303+
* were set. We need to set special flags in the request if so. Otherwise
304+
* we can't distinguish user set from Authenticator set headers
305+
*/
306+
public static void setUserAuthFlags(HttpRequestImpl request, HttpHeaders userHeaders) {
307+
if (userHeaders.firstValue("Authorization").isPresent()) {
308+
request.setUserSetAuthFlag(SERVER, true);
309+
}
310+
if (userHeaders.firstValue("Proxy-Authorization").isPresent()) {
311+
request.setUserSetAuthFlag(PROXY, true);
312+
}
313+
}
313314

314315
public static boolean proxyHasDisabledSchemes(boolean tunnel) {
315316
return tunnel ? ! PROXY_AUTH_TUNNEL_DISABLED_SCHEMES.isEmpty()

‎test/jdk/java/net/httpclient/AuthFilter.java

-188
This file was deleted.

‎test/jdk/java/net/httpclient/UserAuthWithAuthenticator.java

+524
Large diffs are not rendered by default.

0 commit comments

Comments
 (0)
Please sign in to comment.