/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.catalina.mbeans;
import java.io.File;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Set;
import javax.management.MBeanServer;
import javax.management.MalformedObjectNameException;
import javax.management.ObjectName;
import org.junit.
Assert;
import org.junit.Test;
import org.apache.catalina.Context;
import org.apache.catalina.Realm;
import org.apache.catalina.core.StandardHost;
import org.apache.catalina.realm.CombinedRealm;
import org.apache.catalina.realm.NullRealm;
import org.apache.catalina.startup.Tomcat;
import org.apache.catalina.startup.TomcatBaseTest;
import org.apache.tomcat.util.modeler.Registry;
/**
* General tests around the process of registration and de-registration that
* don't necessarily apply to one specific Tomcat class.
*
*/
public class TestRegistration
extends TomcatBaseTest {
private static final String contextName =
"/foo";
private static final String ADDRESS;
static {
String address;
try {
address = InetAddress.getByName(
"localhost").getHostAddress();
}
catch (UnknownHostException e) {
address =
"INIT_FAILED";
}
ADDRESS = address;
}
private static String[] basicMBeanNames() {
return new String[] {
"Tomcat:type=Engine",
"Tomcat:type=Realm,realmPath=/realm0",
"Tomcat:type=Mapper",
"Tomcat:type=MBeanFactory",
"Tomcat:type=NamingResources",
"Tomcat:type=Server",
"Tomcat:type=Service",
"Tomcat:type=StringCache",
"Tomcat:type=UtilityExecutor",
"Tomcat:type=Valve,name=StandardEngineValve",
};
}
private static String[] hostMBeanNames(String host) {
return new String[] {
"Tomcat:type=Host,host=" + host,
"Tomcat:type=Valve,host=" + host +
",name=ErrorReportValve",
"Tomcat:type=Valve,host=" + host +
",name=StandardHostValve",
};
}
private String[] optionalMBeanNames(String host) {
if (isAccessLogEnabled()) {
return new String[] {
"Tomcat:type=Valve,host=" + host +
",name=AccessLogValve",
};
}
else {
return new String[] { };
}
}
private static String[] requestMBeanNames(String port, String type) {
return new String[] {
"Tomcat:type=RequestProcessor,worker=" +
ObjectName.quote(
"http-" + type +
"-" + ADDRESS +
"-" + port) +
",name=HttpRequest1",
};
}
private static String[] contextMBeanNames(String host, String context) {
return new String[] {
"Tomcat:j2eeType=WebModule,name=//" + host + context +
",J2EEApplication=none,J2EEServer=none",
"Tomcat:type=Loader,host=" + host +
",context=" + context,
"Tomcat:type=Manager,host=" + host +
",context=" + context,
"Tomcat:type=NamingResources,host=" + host +
",context=" + context,
"Tomcat:type=Valve,host=" + host +
",context=" + context +
",name=NonLoginAuthenticator",
"Tomcat:type=Valve,host=" + host +
",context=" + context +
",name=StandardContextValve",
"Tomcat:type=ParallelWebappClassLoader,host=" + host +
",context=" + context,
"Tomcat:type=WebResourceRoot,host=" + host +
",context=" + context,
"Tomcat:type=WebResourceRoot,host=" + host +
",context=" + context +
",name=Cache",
"Tomcat:type=Realm,realmPath=/realm0,host=" + host +
",context=" + context,
"Tomcat:type=Realm,realmPath=/realm0/realm0,host=" + host +
",context=" + context
};
}
private static String[] connectorMBeanNames(String port, String type) {
return new String[] {
"Tomcat:type=Connector,port=" + port +
",address="
+ ObjectName.quote(ADDRESS),
"Tomcat:type=GlobalRequestProcessor,name="
+ ObjectName.quote(
"http-" + type +
"-" + ADDRESS +
"-" + port),
"Tomcat:type=ProtocolHandler,port=" + port +
",address="
+ ObjectName.quote(ADDRESS),
"Tomcat:type=ThreadPool,name="
+ ObjectName.quote(
"http-" + type +
"-" + ADDRESS +
"-" + port),
"Tomcat:type=SocketProperties,name="
+ ObjectName.quote(
"http-" + type +
"-" + ADDRESS +
"-" + port),
};
}
/*
* Test verifying that Tomcat correctly de-registers the MBeans it has
* registered.
* @author Marc Guillemot
*/
@Test
public void testMBeanDeregistration()
throws Exception {
final MBeanServer mbeanServer = Registry.getRegistry(
null,
null).getMBeanServer();
// Verify there are no Catalina or Tomcat MBeans
Set<ObjectName> onames = mbeanServer.queryNames(
new ObjectName(
"Catalina:*"),
null);
log.info(MBeanDumper.dumpBeans(mbeanServer, onames));
Assert.assertEquals(
"Unexpected: " + onames, 0, onames.size());
onames = mbeanServer.queryNames(
new ObjectName(
"Tomcat:*"),
null);
log.info(MBeanDumper.dumpBeans(mbeanServer, onames));
Assert.assertEquals(
"Unexpected: " + onames, 0, onames.size());
final Tomcat tomcat = getTomcatInstance();
final File contextDir =
new File(getTemporaryDirectory(),
"webappFoo");
addDeleteOnTearDown(contextDir);
if (!contextDir.mkdirs() && !contextDir.isDirectory()) {
Assert.fail(
"Failed to create: [" + contextDir.toString() +
"]");
}
Context ctx = tomcat.addContext(contextName, contextDir.getAbsolutePath());
CombinedRealm combinedRealm =
new CombinedRealm();
Realm nullRealm =
new NullRealm();
combinedRealm.addRealm(nullRealm);
ctx.setRealm(combinedRealm);
tomcat.start();
getUrl(
"http://localhost:" + getPort());
// Verify there are no Catalina MBeans
onames = mbeanServer.queryNames(
new ObjectName(
"Catalina:*"),
null);
log.info(MBeanDumper.dumpBeans(mbeanServer, onames));
Assert.assertEquals(
"Found: " + onames, 0, onames.size());
// Verify there are the correct Tomcat MBeans
onames = mbeanServer.queryNames(
new ObjectName(
"Tomcat:*"),
null);
ArrayList<String> found =
new ArrayList<>(onames.size());
for (ObjectName on: onames) {
found.add(on.toString());
}
// Create the list of expected MBean names
String protocol = tomcat.getConnector().getProtocolHandlerClassName();
if (protocol.indexOf(
"Nio2") > 0) {
protocol =
"nio2";
}
else {
protocol =
"nio";
}
String index = tomcat.getConnector().getProperty(
"nameIndex").toString();
ArrayList<String> expected =
new ArrayList<>(Arrays.asList(basicMBeanNames()));
expected.addAll(Arrays.asList(hostMBeanNames(
"localhost")));
expected.addAll(Arrays.asList(contextMBeanNames(
"localhost", contextName)));
expected.addAll(Arrays.asList(connectorMBeanNames(
"auto-" + index, protocol)));
expected.addAll(Arrays.asList(optionalMBeanNames(
"localhost")));
expected.addAll(Arrays.asList(requestMBeanNames(
"auto-" + index +
"-" + getPort(), protocol)));
// Did we find all expected MBeans?
ArrayList<String> missing =
new ArrayList<>(expected);
missing.removeAll(found);
Assert.assertTrue(
"Missing Tomcat MBeans: " + missing, missing.isEmpty());
// Did we find any unexpected MBeans?
List<String> additional = found;
additional.removeAll(expected);
Assert.assertTrue(
"Unexpected Tomcat MBeans: " + additional, additional.isEmpty());
// Check a known attribute
String connectorName = Arrays.asList(connectorMBeanNames(
"auto-" + index, protocol)).g
et(0);
// This should normally return "http", but any non null non exception is good enough
Assert.assertNotNull(mbeanServer.getAttribute(new ObjectName(connectorName), "scheme"));
tomcat.stop();
// There should still be some Tomcat MBeans
onames = mbeanServer.queryNames(new ObjectName("Tomcat:*"), null);
Assert.assertTrue("No Tomcat MBeans", onames.size() > 0);
// add a new host
StandardHost host = new StandardHost();
host.setName("otherhost");
tomcat.getEngine().addChild(host);
final File contextDir2 = new File(getTemporaryDirectory(), "webappFoo2");
addDeleteOnTearDown(contextDir2);
if (!contextDir2.mkdirs() && !contextDir2.isDirectory()) {
Assert.fail("Failed to create: [" + contextDir2.toString() + "]");
}
tomcat.addContext(host, contextName + "2", contextDir2.getAbsolutePath());
tomcat.start();
tomcat.stop();
tomcat.destroy();
// There should be no Catalina MBeans and no Tomcat MBeans
onames = mbeanServer.queryNames(new ObjectName("Catalina:*"), null);
log.info(MBeanDumper.dumpBeans(mbeanServer, onames));
Assert.assertEquals("Remaining: " + onames, 0, onames.size());
onames = mbeanServer.queryNames(new ObjectName("Tomcat:*"), null);
log.info(MBeanDumper.dumpBeans(mbeanServer, onames));
Assert.assertEquals("Remaining: " + onames, 0, onames.size());
}
/*
* Confirm that, as far as ObjectName is concerned, the order of the key
* properties is not significant.
*/
@Test
public void testNames() throws MalformedObjectNameException {
ObjectName on1 = new ObjectName("test:foo=a,bar=b");
ObjectName on2 = new ObjectName("test:bar=b,foo=a");
Assert.assertTrue(on1.equals(on2));
}
}