我有一个使用Tomcat 6编写的Web应用程序,我正在尝试使其与Tomcat 7一起使用。在启动期间,应用程序除了其他内容之外,还将其Web服务组件注册到某个远程目录。为此,它需要提供自己的URL。以下(有点天真)方法应该返回webservice URL:
import org.apache.catalina.ServerFactory;
import org.apache.catalina.connector.Connector;
.
.
.
private String getWsUrl(ServletContext context)
throws UnknownHostException, MalformedURLException {
String host = java.net.InetAddress.getLocalHost().getCanonicalHostName();
int port = -1;
for (Connector c : ServerFactory.getServer().findServices()[0].findConnectors()) {
if (c.getProtocol().contains("HTTP")) {
port = c.getPort();
break;
}
}
URL wsURL = new URL("http", host, port, context.getContextPath()
+ C.WEB_SERVICE_PATH /* this is just a constant string */ );
return wsURL.toString();
}
ServerFactory.getServer()
部分证明是有问题的:Tomcat 7中没有org.apache.catalina.ServerFactory
类。有关如何为Tomcat 7重写此内容的任何建议吗?我也很乐意拥有更多可移植的非tomcat特定代码。
答案 0 :(得分:3)
我遇到了同样的问题:我需要知道端口号来为特定的Tomcat实例构建一个URL,端口号可能会有所不同(因为我运行多个实例进行测试),而且从Tomcat 7开始,ServerFactory已经消失了
我编写了以下代码来查找和解析Tomcat server.xml文件。它没有解析太多,只是获取HTTP“port”和“redirectPort”值。它取决于“catalina.home”或“catalina.base”系统属性,它应存在于任何正在运行的Tomcat实例中。美妙之处在于它不依赖于任何Tomcat类,并使用JVM的XML解析器。
我希望这会有所帮助。
public final class TomcatConfigUtil {
private static final String CONFIG_FILE_PATH = "conf/server.xml";
private static Map<String, String> properties = null;
// No instances, please.
private TomcatConfigUtil() { }
/**
* Get the configuration as a map of name/value pairs, or throw an exception if it wasn't found.
* All values are returned as Strings.
* <ul>
* <li> httpPort - the HTTP port</li>
* <li> httpRedirectPort - the HTTP redirect port (which seems to be the SSL port) </li>
* </ul>
* @exception FileNotFoundException if the configuration file wasn't found
* @exception IOException if there was a problem reading the configuration file
* @exception SAXException if there was a problem parsing the configuration file
*/
public static synchronized Map<String, String> getConfig() throws FileNotFoundException, IOException, SAXException {
if (properties != null) {
return properties;
}
final File serverConfigFile = findServerConfigFile();
if (serverConfigFile == null) {
throw new FileNotFoundException("Couldn't find the configuration file.");
}
final Map<String, String> tmpProperties = new HashMap<String, String>();
// Content-handler does the actual parsing.
final ServerConfigContentHandler contentHandler = new ServerConfigContentHandler(tmpProperties);
final XMLReader xmlReader = XMLReaderFactory.createXMLReader();
xmlReader.setContentHandler(contentHandler);
// Pass the config file as the input source for parsing.
final FileReader fileReader = new FileReader(serverConfigFile);
xmlReader.parse(new InputSource(fileReader));
fileReader.close();
return (properties = Collections.unmodifiableMap(tmpProperties));
}
private static File findServerConfigFile() {
if (System.getProperty("catalina.home") != null) {
final File file = new File(System.getProperty("catalina.home"), CONFIG_FILE_PATH);
if (file.isFile()) {
return file;
}
}
if (System.getProperty("catalina.base") != null) {
final File file = new File(System.getProperty("catalina.base"), CONFIG_FILE_PATH);
if (file.isFile()) {
return file;
}
}
return null;
}
/**
* ContentHandler implementation for the XML parser.
*/
private static class ServerConfigContentHandler implements ContentHandler {
private final Map<String, String> map;
private boolean inServerElement;
private boolean inCatalinaServiceElement;
private ServerConfigContentHandler(final Map<String, String> map) {
this.map = map;
}
@Override
public void startDocument() throws SAXException {
this.inServerElement = false;
this.inCatalinaServiceElement = false;
}
@Override
public void startElement(final String uri, final String localName, final String qName, final Attributes atts) throws SAXException {
if (!this.inServerElement && "Server".equals(localName)) {
this.inServerElement = true;
}
else if (this.inServerElement && "Service".equals(localName) && "Catalina".equals(atts.getValue("name"))) {
this.inCatalinaServiceElement = true;
}
else if (this.inCatalinaServiceElement && "Connector".equals(localName) && "HTTP/1.1".equals(atts.getValue("protocol"))) {
if ((atts.getValue("SSLEnabled") == null || "false".equals(atts.getValue("SSLEnabled"))) &&
(atts.getValue("secure") == null || "false".equals(atts.getValue("secure"))) &&
(atts.getValue("scheme") == null || "http".equals(atts.getValue("scheme")))) {
final String portStr = atts.getValue("port");
if (portStr != null) {
this.map.put("httpPort", portStr);
}
final String redirectPortStr = atts.getValue("redirectPort");
if (redirectPortStr != null) {
this.map.put("httpRedirectPort", redirectPortStr);
}
}
}
}
@Override
public void endElement(final String uri, final String localName, final String qName) throws SAXException {
if (this.inCatalinaServiceElement && "Service".equals(localName)) {
this.inCatalinaServiceElement = false;
}
else if (this.inServerElement && "Server".equals(localName)) {
this.inServerElement = false;
}
}
@Override
public void endDocument() throws SAXException {
this.inServerElement = false;
this.inCatalinaServiceElement = false;
}
@Override
public void characters(final char[] ch, final int start, final int length) throws SAXException { }
@Override
public void endPrefixMapping(final String prefix) throws SAXException { }
@Override
public void ignorableWhitespace(final char[] ch, final int start, final int length) throws SAXException { }
@Override
public void processingInstruction(final String target, final String data) throws SAXException { }
@Override
public void setDocumentLocator(final Locator locator) { }
@Override
public void skippedEntity(final String name) throws SAXException { }
@Override
public void startPrefixMapping(final String prefix, final String uri) throws SAXException { }
}
}
答案 1 :(得分:0)
我从来没有使用它,它可能不会返回带有主机名和地址的URL,但是ServletContext.getResource("/")
有没有机会做你想做的事情?我知道它的目的是通过servlet在内部访问资源,但你永远不知道。