jdk/test/javax/management/remote/mandatory/connection/ConnectionTest.java
changeset 2 90ce3da70b43
child 5506 202f599c92aa
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/jdk/test/javax/management/remote/mandatory/connection/ConnectionTest.java	Sat Dec 01 00:00:00 2007 +0000
@@ -0,0 +1,358 @@
+/*
+ * Copyright 2003 Sun Microsystems, Inc.  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
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation.
+ *
+ * This code is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ * @test
+ * @bug 4865397
+ * @summary Tests remote JMX connections
+ * @author Eamonn McManus
+ * @run clean ConnectionTest
+ * @run build ConnectionTest
+ * @run main ConnectionTest
+ */
+
+import java.io.IOException;
+import java.net.MalformedURLException;
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.Iterator;
+import java.util.LinkedList;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+import java.util.StringTokenizer;
+
+import java.security.Principal;
+import javax.security.auth.Subject;
+
+import javax.management.MBeanServer;
+import javax.management.MBeanServerConnection;
+import javax.management.MBeanServerFactory;
+import javax.management.Notification;
+import javax.management.NotificationListener;
+import javax.management.ObjectName;
+
+import javax.management.remote.JMXAuthenticator;
+import javax.management.remote.JMXConnectionNotification;
+import javax.management.remote.JMXConnector;
+import javax.management.remote.JMXConnectorFactory;
+import javax.management.remote.JMXConnectorServer;
+import javax.management.remote.JMXConnectorServerFactory;
+import javax.management.remote.JMXPrincipal;
+import javax.management.remote.JMXServiceURL;
+
+public class ConnectionTest {
+
+    public static void main(String[] args) {
+//      System.setProperty("java.util.logging.config.file",
+//                         "../../../../logging.properties");
+//      // we are in <workspace>/build/test/JTwork/scratch
+//      java.util.logging.LogManager.getLogManager().readConfiguration();
+        boolean ok = true;
+        String[] protocols = {"rmi", "iiop", "jmxmp"};
+        if (args.length > 0)
+            protocols = args;
+        for (int i = 0; i < protocols.length; i++) {
+            final String proto = protocols[i];
+            System.out.println("Testing for protocol " + proto);
+            try {
+                ok &= test(proto);
+            } catch (Exception e) {
+                System.err.println("Unexpected exception: " + e);
+                e.printStackTrace();
+                ok = false;
+            }
+        }
+
+        if (ok)
+            System.out.println("Test passed");
+        else {
+            System.out.println("TEST FAILED");
+            System.exit(1);
+        }
+    }
+
+    private static boolean test(String proto) throws Exception {
+        ObjectName serverName = ObjectName.getInstance("d:type=server");
+        MBeanServer mbs = MBeanServerFactory.newMBeanServer();
+        JMXAuthenticator authenticator = new BogusAuthenticator();
+        Map env = Collections.singletonMap("jmx.remote.authenticator",
+                                           authenticator);
+        JMXServiceURL url = new JMXServiceURL("service:jmx:" + proto + "://");
+        JMXConnectorServer server;
+        try {
+            server =
+                JMXConnectorServerFactory.newJMXConnectorServer(url, env,
+                                                                null);
+        } catch (MalformedURLException e) {
+            System.out.println("Protocol " + proto +
+                               " not supported, ignoring");
+            return true;
+        }
+        System.out.println("Created connector server");
+        mbs.registerMBean(server, serverName);
+        System.out.println("Registered connector server in MBean server");
+        mbs.addNotificationListener(serverName, logListener, null, null);
+        mbs.invoke(serverName, "start", null, null);
+        System.out.println("Started connector server");
+        JMXServiceURL address =
+            (JMXServiceURL) mbs.getAttribute(serverName, "Address");
+        System.out.println("Retrieved address: " + address);
+
+        if (address.getHost().length() == 0) {
+            System.out.println("Generated address has empty hostname");
+            return false;
+        }
+
+        JMXConnector client = JMXConnectorFactory.connect(address);
+        System.out.println("Client connected");
+
+        String clientConnId = client.getConnectionId();
+        System.out.println("Got connection ID on client: " + clientConnId);
+        boolean ok = checkConnectionId(proto, clientConnId);
+        if (!ok)
+            return false;
+        System.out.println("Connection ID is OK");
+
+        // 4901826: connection ids need some time to be updated using jmxmp
+        // we don't get the notif immediately either
+        // this was originally timeout 1ms, which was not enough
+        Notification notif = waitForNotification(1000);
+        System.out.println("Server got notification: " + notif);
+
+        ok = mustBeConnectionNotification(notif, clientConnId,
+                                         JMXConnectionNotification.OPENED);
+        if (!ok)
+            return false;
+
+        client.close();
+        System.out.println("Closed client");
+
+        notif = waitForNotification(1000);
+        System.out.println("Got notification: " + notif);
+
+        ok = mustBeConnectionNotification(notif, clientConnId,
+                                          JMXConnectionNotification.CLOSED);
+        if (!ok)
+            return false;
+
+        client = JMXConnectorFactory.connect(address);
+        System.out.println("Second client connected");
+
+        String clientConnId2 = client.getConnectionId();
+        if (clientConnId.equals(clientConnId2)) {
+            System.out.println("Same connection ID for two connections: " +
+                               clientConnId2);
+            return false;
+        }
+        System.out.println("Second client connection ID is different");
+
+        notif = waitForNotification(1);
+        ok = mustBeConnectionNotification(notif, clientConnId2,
+                                          JMXConnectionNotification.OPENED);
+        if (!ok)
+            return false;
+
+        MBeanServerConnection mbsc = client.getMBeanServerConnection();
+        Map attrs = (Map) mbsc.getAttribute(serverName, "Attributes");
+        System.out.println("Server attributes received by client: " + attrs);
+
+        server.stop();
+        System.out.println("Server stopped");
+
+        notif = waitForNotification(1000);
+        System.out.println("Server got connection-closed notification: " +
+                           notif);
+
+        ok = mustBeConnectionNotification(notif, clientConnId2,
+                                          JMXConnectionNotification.CLOSED);
+        if (!ok)
+            return false;
+
+        try {
+            mbsc.getDefaultDomain();
+            System.out.println("Connection still working but should not be");
+            return false;
+        } catch (IOException e) {
+            System.out.println("Connection correctly got exception: " + e);
+        }
+
+        try {
+            client = JMXConnectorFactory.connect(address);
+            System.out.println("Connector server still working but should " +
+                               "not be");
+            return false;
+        } catch (IOException e) {
+            System.out.println("New connection correctly got exception: " + e);
+        }
+
+        return true;
+    }
+
+    private static boolean
+        mustBeConnectionNotification(Notification notif,
+                                     String requiredConnId,
+                                     String requiredType) {
+
+        if (!(notif instanceof JMXConnectionNotification)) {
+            System.out.println("Should have been a " +
+                               "JMXConnectionNotification: " +
+                               notif.getClass());
+            return false;
+        }
+
+        JMXConnectionNotification cnotif = (JMXConnectionNotification) notif;
+        if (!cnotif.getType().equals(requiredType)) {
+            System.out.println("Wrong type notif: is \"" + cnotif.getType() +
+                               "\", should be \"" + requiredType + "\"");
+            return false;
+        }
+
+        if (!cnotif.getConnectionId().equals(requiredConnId)) {
+            System.out.println("Wrong connection id: is \"" +
+                               cnotif.getConnectionId() + "\", should be \"" +
+                               requiredConnId);
+            return false;
+        }
+
+        return true;
+    }
+
+    private static boolean checkConnectionId(String proto, String clientConnId)
+            throws Exception {
+        StringTokenizer tok = new StringTokenizer(clientConnId, " ", true);
+        String s;
+        s = tok.nextToken();
+        if (!s.startsWith(proto + ":")) {
+            System.out.println("Expected \"" + proto + ":\", found \"" + s +
+                               "\"");
+            return false;
+        }
+        s = tok.nextToken();
+        if (!s.equals(" ")) {
+            System.out.println("Expected \" \", found \"" + s + "\"");
+            return false;
+        }
+        s = tok.nextToken();
+        StringTokenizer tok2 = new StringTokenizer(s, ";", true);
+        Set principalNames = new HashSet();
+        String s2;
+        s2 = tok2.nextToken();
+        if (s2.equals(";")) {
+            System.out.println("In identity \"" + s +
+                               "\", expected name, found \";\"");
+            return false;
+        }
+        principalNames.add(s2);
+        s2 = tok2.nextToken();
+        if (!s2.equals(";"))
+            throw new Exception("Can't happen");
+        s2 = tok2.nextToken();
+        if (s2.equals(";")) {
+            System.out.println("In identity \"" + s +
+                               "\", expected name, found \";\"");
+            return false;
+        }
+        principalNames.add(s2);
+        if (tok2.hasMoreTokens()) {
+            System.out.println("In identity \"" + s + "\", too many tokens");
+            return false;
+        }
+        if (principalNames.size() != bogusPrincipals.size()) {
+            System.out.println("Wrong number of principal names: " +
+                               principalNames.size() + " != " +
+                               bogusPrincipals.size());
+            return false;
+        }
+        for (Iterator it = bogusPrincipals.iterator(); it.hasNext(); ) {
+            Principal p = (Principal) it.next();
+            if (!principalNames.contains(p.getName())) {
+                System.out.println("Principal names don't contain \"" +
+                                   p.getName() + "\"");
+                return false;
+            }
+        }
+        s = tok.nextToken();
+        if (!s.equals(" ")) {
+            System.out.println("Expected \" \", found \"" + s + "\"");
+            return false;
+        }
+        return true;
+    }
+
+    private static Notification waitForNotification(long timeout)
+            throws InterruptedException {
+        synchronized (log) {
+            if (log.isEmpty()) {
+                long remainingTime = timeout;
+                final long startTime = System.currentTimeMillis();
+
+                while (log.isEmpty() && remainingTime >0) {
+                    log.wait(remainingTime);
+                    remainingTime = timeout - (System.currentTimeMillis() - startTime);
+                }
+
+                if (log.isEmpty()) {
+                    throw new InterruptedException("Timed out waiting for " +
+                                                   "notification!");
+                }
+            }
+            return (Notification) log.remove(0);
+        }
+    }
+
+    private static class LogListener implements NotificationListener {
+        LogListener(List log) {
+            this.log = log;
+        }
+
+        public void handleNotification(Notification n, Object h) {
+            synchronized (log) {
+                log.add(n);
+                log.notifyAll();
+            }
+        }
+
+        private final List log;
+    }
+
+    private static List log = new LinkedList();
+    private static NotificationListener logListener = new LogListener(log);
+
+    private static class BogusAuthenticator implements JMXAuthenticator {
+        public Subject authenticate(Object credentials) {
+            Subject subject =
+                new Subject(true, bogusPrincipals,
+                            Collections.EMPTY_SET, Collections.EMPTY_SET);
+            System.out.println("Authenticator returns: " + subject);
+            return subject;
+        }
+    }
+
+    private static final Set bogusPrincipals = new HashSet();
+    static {
+        bogusPrincipals.add(new JMXPrincipal("foo"));
+        bogusPrincipals.add(new JMXPrincipal("bar"));
+    }
+}