8225499: InetSocketAddress::toString not friendly to IPv6 literal addresses
authorjboes
Thu, 17 Oct 2019 08:56:06 +0100
changeset 58691 eb172a3b1c1c
parent 58690 294fe0fef773
child 58692 04946947ca79
8225499: InetSocketAddress::toString not friendly to IPv6 literal addresses Summary: Enclose IPv6 literal in brackets and adjust string format for unresolved addresses Reviewed-by: dfuchs, chegar
src/java.base/share/classes/java/net/InetSocketAddress.java
test/jdk/java/net/InetSocketAddress/ToString.java
--- a/src/java.base/share/classes/java/net/InetSocketAddress.java	Fri Oct 18 12:20:51 2019 -0400
+++ b/src/java.base/share/classes/java/net/InetSocketAddress.java	Thu Oct 17 08:56:06 2019 +0100
@@ -101,11 +101,20 @@
 
         @Override
         public String toString() {
+
+            String formatted;
+
             if (isUnresolved()) {
-                return hostname + ":" + port;
+                formatted = hostname + "/<unresolved>";
             } 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
@@ -367,7 +376,9 @@
      * Constructs a string representation of this InetSocketAddress.
      * This String is constructed by calling toString() on the InetAddress
      * 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.
      */
--- a/test/jdk/java/net/InetSocketAddress/ToString.java	Fri Oct 18 12:20:51 2019 -0400
+++ b/test/jdk/java/net/InetSocketAddress/ToString.java	Thu Oct 17 08:56:06 2019 +0100
@@ -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.
  *
  * This code is free software; you can redistribute it and/or modify it
@@ -23,19 +23,141 @@
 
 /*
  * @test
- * @bug 4464064
- * @summary InetSocketAddress.toString() throws NPE with unresolved address
+ * @bug 8225499 4464064
+ * @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.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 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));
+    }
+
+    @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"}
+        };
+    }
 
-        System.out.println(new InetSocketAddress("unresolved", 12345));
+    @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);
+        }
     }
 }