Skip to content

Commit 3f36dd8

Browse files
committedApr 13, 2023
8305529: DefaultProxySelector.select(URI) in certain cases returns a List with null element
Reviewed-by: dfuchs, djelinski, michaelm
1 parent 425ef06 commit 3f36dd8

File tree

3 files changed

+131
-2
lines changed

3 files changed

+131
-2
lines changed
 

‎src/java.base/unix/native/libnet/DefaultProxySelector.c

+16-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2004, 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2004, 2023, 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
@@ -97,13 +97,15 @@ typedef GSocketConnectable* g_network_address_parse_uri_func();
9797
typedef const char* g_network_address_get_hostname_func();
9898
typedef unsigned short g_network_address_get_port_func();
9999
typedef void g_strfreev_func();
100+
typedef void g_clear_error_func();
100101

101102
static g_proxy_resolver_get_default_func* g_proxy_resolver_get_default = NULL;
102103
static g_proxy_resolver_lookup_func* g_proxy_resolver_lookup = NULL;
103104
static g_network_address_parse_uri_func* g_network_address_parse_uri = NULL;
104105
static g_network_address_get_hostname_func* g_network_address_get_hostname = NULL;
105106
static g_network_address_get_port_func* g_network_address_get_port = NULL;
106107
static g_strfreev_func* g_strfreev = NULL;
108+
static g_clear_error_func* g_clear_error = NULL;
107109

108110
static void* gconf_client = NULL;
109111
static int use_gproxyResolver = 0;
@@ -317,13 +319,16 @@ static int initGProxyResolver() {
317319

318320
g_strfreev = (g_strfreev_func*)dlsym(gio_handle, "g_strfreev");
319321

322+
g_clear_error = (g_clear_error_func*)dlsym(gio_handle, "g_clear_error");
323+
320324
if (!my_g_type_init_func ||
321325
!g_proxy_resolver_get_default ||
322326
!g_proxy_resolver_lookup ||
323327
!g_network_address_parse_uri ||
324328
!g_network_address_get_hostname ||
325329
!g_network_address_get_port ||
326-
!g_strfreev)
330+
!g_strfreev ||
331+
!g_clear_error)
327332
{
328333
dlclose(gio_handle);
329334
return 0;
@@ -412,7 +417,13 @@ static jobjectArray getProxyByGProxyResolver(JNIEnv *env, const char *cproto,
412417
proxy_array = NULL;
413418
break;
414419
}
420+
} else {
421+
proxy_array = NULL;
422+
break;
415423
}
424+
} else {
425+
proxy_array = NULL;
426+
break;
416427
}
417428
} else {
418429
/* direct connection - no proxy */
@@ -432,6 +443,9 @@ static jobjectArray getProxyByGProxyResolver(JNIEnv *env, const char *cproto,
432443
}
433444
}
434445
(*g_strfreev)(proxies);
446+
// as per API doc, g_clear_error doesn't complain if error is NULL, so it's safe to
447+
// call without null checks
448+
(*g_clear_error)(&error);
435449
}
436450

437451
return proxy_array;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
/*
2+
* Copyright (c) 2023, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
import jdk.test.lib.process.ProcessTools;
25+
26+
/*
27+
* @test
28+
* @bug 8305529
29+
* @summary Verifies that the sun.net.spi.DefaultProxySelector#select(URI) doesn't return a List
30+
* with null elements in it
31+
* @modules java.base/sun.net.spi:+open
32+
* @library /test/lib
33+
* @build jdk.test.lib.process.ProcessTools SystemProxyTest
34+
* @run driver SystemProxyDriver
35+
*/
36+
public class SystemProxyDriver {
37+
// launches the SystemProxyTest as a separate process and verifies that the test passes
38+
public static void main(final String[] args) throws Exception {
39+
final String[] commandArgs = new String[]{
40+
"--add-opens",
41+
"java.base/sun.net.spi=ALL-UNNAMED",
42+
// trigger use of the http_proxy environment variable that we pass when launching
43+
// this Java program
44+
"-Djava.net.useSystemProxies=true",
45+
"SystemProxyTest"
46+
};
47+
final ProcessBuilder pb = ProcessTools.createTestJvm(commandArgs);
48+
pb.inheritIO();
49+
pb.environment().put("http_proxy", "foo://"); // intentionally use a value without host/port
50+
final Process p = pb.start();
51+
final int exitCode = p.waitFor();
52+
if (exitCode != 0) {
53+
throw new RuntimeException("Test failed, exitCode: " + exitCode);
54+
}
55+
}
56+
}
+59
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
/*
2+
* Copyright (c) 2023, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
import java.net.Proxy;
25+
import java.net.ProxySelector;
26+
import java.net.URI;
27+
import java.util.List;
28+
29+
import sun.net.spi.DefaultProxySelector;
30+
31+
// this test is launched from SystemProxyDriver
32+
public class SystemProxyTest {
33+
34+
// calls the DefaultProxySelector.select(URI) and verifies that the returned List is
35+
// not null, not empty and doesn't contain null elements.
36+
public static void main(final String[] args) throws Exception {
37+
final ProxySelector ps = new DefaultProxySelector();
38+
final URI uri = new URI("http://example.com"); // the target URL doesn't matter
39+
final List<Proxy> proxies = ps.select(uri);
40+
if (proxies == null) {
41+
// null isn't expected to be returned by the select() API
42+
throw new AssertionError("DefaultProxySelector.select(URI) returned null for uri: "
43+
+ uri);
44+
}
45+
if (proxies.isEmpty()) {
46+
// empty list isn't expected to be returned by the select() API, instead when
47+
// no proxy is configured, the returned list is expected to contain one entry with
48+
// a Proxy instance representing direct connection
49+
throw new AssertionError("DefaultProxySelector.select(URI) returned empty list" +
50+
" for uri: " + uri);
51+
}
52+
System.out.println("returned proxies list: " + proxies);
53+
for (final Proxy p : proxies) {
54+
if (p == null) {
55+
throw new AssertionError("null proxy in proxies list for uri: " + uri);
56+
}
57+
}
58+
}
59+
}

0 commit comments

Comments
 (0)
Please sign in to comment.