8225499: InetSocketAddress::toString not friendly to IPv6 literal addresses

Enclose IPv6 literal in brackets and adjust string format for unresolved addresses

Reviewed-by: dfuchs, chegar
This commit is contained in:
Julia Boes 2019-10-17 08:56:06 +01:00
parent 2c573582ab
commit f8440c33d2
2 changed files with 141 additions and 8 deletions

View File

@ -101,11 +101,20 @@ public class InetSocketAddress
@Override @Override
public String toString() { public String toString() {
String formatted;
if (isUnresolved()) { if (isUnresolved()) {
return hostname + ":" + port; formatted = hostname + "/<unresolved>";
} else { } else {
return addr.toString() + ":" + port; formatted = addr.toString();
if (addr instanceof Inet6Address) {
int i = formatted.lastIndexOf("/");
formatted = formatted.substring(0, i + 1)
+ "[" + formatted.substring(i + 1) + "]";
}
} }
return formatted + ":" + port;
} }
@Override @Override
@ -367,7 +376,9 @@ public class InetSocketAddress
* Constructs a string representation of this InetSocketAddress. * Constructs a string representation of this InetSocketAddress.
* This String is constructed by calling toString() on the InetAddress * This String is constructed by calling toString() on the InetAddress
* and concatenating the port number (with a colon). If the address * and concatenating the port number (with a colon). If the address
* is unresolved then the part before the colon will only contain the hostname. * is an IPv6 address, the IPv6 literal is enclosed in square brackets.
* If the address is {@linkplain #isUnresolved() unresolved},
* {@code <unresolved>} is displayed in place of the address literal.
* *
* @return a string representation of this object. * @return a string representation of this object.
*/ */

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2019, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -23,19 +23,141 @@
/* /*
* @test * @test
* @bug 4464064 * @bug 8225499 4464064
* @summary InetSocketAddress.toString() throws NPE with unresolved address * @library /test/lib
* @summary InetSocketAddress::toString not friendly to IPv6 literal addresses
* @run testng/othervm ToString
* @run testng/othervm -Djava.net.preferIPv4Stack=true ToString
* @run testng/othervm -Djava.net.preferIPv6Addresses=true ToString
*/ */
import java.net.*; import java.net.*;
import java.io.*;
import jdk.test.lib.net.IPSupport;
import org.testng.annotations.BeforeTest;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Test;
public class ToString { public class ToString {
public static void main (String args[]){ private static final String loopbackAddr;
private static final String wildcardAddr;
private static final String localAddr;
static {
try {
InetAddress loopback = InetAddress.getLoopbackAddress();
String addr = loopback.getHostAddress();
if (loopback instanceof Inet6Address) {
addr = "[" + addr + "]";
}
loopbackAddr = addr;
InetSocketAddress isa = new InetSocketAddress((InetAddress) null, 80);
addr = isa.getAddress().toString();
if (isa.getAddress() instanceof Inet6Address) {
addr = "::/[0:0:0:0:0:0:0:0]";
}
wildcardAddr = addr;
InetAddress ia = InetAddress.getLocalHost();
addr = ia.toString();
if (ia instanceof Inet6Address) {
addr = ia.getHostName() + "/[" + ia.getHostAddress() + "]";
}
localAddr = addr;
} catch (UnknownHostException uhe) {
throw new RuntimeException(uhe);
}
}
@BeforeTest
public void setup() {
IPSupport.throwSkippedExceptionIfNonOperational();
}
@Test
// InetSocketAddress.toString() throws NPE with unresolved address
public static void NPETest() {
System.out.println(new InetSocketAddress("unresolved", 12345)); System.out.println(new InetSocketAddress("unresolved", 12345));
}
@DataProvider(name = "hostPortArgs")
public Object[][] createArgs1() {
return new Object[][]{
// hostname, port number, expected string in format
// <hostname>/<IP literal>:<port> or
// <hostname>/<unresolved>:<port> if address is unresolved
{"::1", 80, "/[0:0:0:0:0:0:0:1]:80"},
{"fedc:ba98:7654:3210:fedc:ba98:7654:3210", 80, "/[fedc:ba98:7654:3210:fedc:ba98:7654:3210]:80"},
{"::192.9.5.5", 80, "/[0:0:0:0:0:0:c009:505]:80"},
{"127.0.0.1", 80, "/127.0.0.1:80"},
{"::ffff:192.0.2.128", 80, "/192.0.2.128:80"},
{"0", 80, "/0.0.0.0:80"},
{":", 80, ":/<unresolved>:80"},
{":1", 80, ":1/<unresolved>:80"}
};
}
@Test(dataProvider = "hostPortArgs")
public static void testConstructor(String host, int port, String string) {
String received = new InetSocketAddress(host, port).toString();
if (!string.equals(received)) {
throw new RuntimeException("Expected: " + string + " Received: " + received);
}
}
@DataProvider(name = "addrPortArgs")
public Object[][] createArgs2() {
InetAddress nullAddr = null;
try {
return new Object[][]{
// InetAddress, port number, expected string
{InetAddress.getLoopbackAddress(), 80, "localhost/" + loopbackAddr + ":80"},
{InetAddress.getLocalHost(), 80, localAddr + ":80"},
{InetAddress.getByAddress(new byte[]{1, 1, 1, 1}), 80, "/1.1.1.1:80"},
{InetAddress.getByAddress(new byte[]{1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1}), 80, "/[101:101:101:101:101:101:101:101]:80"},
{InetAddress.getByName("225.225.225.0"), 80, "/225.225.225.0:80"},
{nullAddr, 80, wildcardAddr + ":80"}
};
} catch (UnknownHostException uhe) {
throw new RuntimeException("Data provider creation failed: " + uhe, uhe);
}
}
@Test(dataProvider = "addrPortArgs")
public static void testConstructor(InetAddress addr, int port, String string) {
String received = new InetSocketAddress(addr, port).toString();
if (!string.equals(received)) {
throw new RuntimeException("Expected: " + string + " Received: " + received);
}
}
@DataProvider(name = "unresolved")
public Object[][] createArgs3() {
return new Object[][]{
// hostname, port number, expected string
{"::1", 80, "::1/<unresolved>:80"},
{"fedc:ba98:7654:3210:fedc:ba98:7654:3210", 80, "fedc:ba98:7654:3210:fedc:ba98:7654:3210/<unresolved>:80"},
{"::192.9.5.5", 80, "::192.9.5.5/<unresolved>:80"},
{"127.0.0.1", 80, "127.0.0.1/<unresolved>:80"},
{"::ffff:192.0.2.128", 80, "::ffff:192.0.2.128/<unresolved>:80"},
{"0", 80, "0/<unresolved>:80"},
{"foo", 80, "foo/<unresolved>:80"},
{":", 80, ":/<unresolved>:80"},
{":1", 80, ":1/<unresolved>:80"}
};
}
@Test(dataProvider = "unresolved")
public static void testCreateUnresolved(String host, int port, String string) {
String received = InetSocketAddress.createUnresolved(host, port).toString();
if (!string.equals(received)) {
throw new RuntimeException("Expected: " + string + " Received: " + received);
}
} }
} }