com.sun.enterprise.web.WebContainer Maven / Gradle / Ivy
Show all versions of payara-micro Show documentation
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 1997-2016 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
// Portions Copyright [2016-2020] [Payara Foundation and/or its affiliates]
package com.sun.enterprise.web;
import com.sun.appserv.server.util.Version;
import com.sun.enterprise.config.serverbeans.Applications;
import com.sun.enterprise.config.serverbeans.Config;
import com.sun.enterprise.config.serverbeans.ConfigBeansUtilities;
import com.sun.enterprise.config.serverbeans.DasConfig;
import com.sun.enterprise.config.serverbeans.Domain;
import com.sun.enterprise.config.serverbeans.HttpService;
import com.sun.enterprise.config.serverbeans.SecurityService;
import com.sun.enterprise.config.serverbeans.Server;
import com.sun.enterprise.config.serverbeans.SystemProperty;
import com.sun.enterprise.container.common.spi.JCDIService;
import com.sun.enterprise.container.common.spi.util.ComponentEnvManager;
import com.sun.enterprise.container.common.spi.util.InjectionManager;
import com.sun.enterprise.container.common.spi.util.JavaEEIOUtils;
import com.sun.enterprise.deployment.Application;
import com.sun.enterprise.deployment.WebBundleDescriptor;
import com.sun.enterprise.deployment.WebComponentDescriptor;
import com.sun.enterprise.security.ee.SecurityDeployer;
import com.sun.enterprise.security.integration.RealmInitializer;
import com.sun.enterprise.server.logging.LoggingRuntime;
import com.sun.enterprise.util.Result;
import com.sun.enterprise.util.StringUtils;
import com.sun.enterprise.v3.admin.adapter.AdminConsoleAdapter;
import com.sun.enterprise.v3.services.impl.ContainerMapper;
import com.sun.enterprise.v3.services.impl.GrizzlyService;
import com.sun.enterprise.web.connector.coyote.PECoyoteConnector;
import com.sun.enterprise.web.logger.FileLoggerHandlerFactory;
import com.sun.enterprise.web.logger.IASLogger;
import com.sun.enterprise.web.pluggable.WebContainerFeatureFactory;
import com.sun.enterprise.web.reconfig.WebConfigListener;
import fish.payara.nucleus.hotdeploy.ApplicationState;
import fish.payara.nucleus.hotdeploy.HotDeployService;
import java.io.File;
import java.net.BindException;
import java.net.MalformedURLException;
import java.text.MessageFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.Optional;
import java.util.Properties;
import java.util.ResourceBundle;
import java.util.Set;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.imageio.ImageIO;
import javax.inject.Inject;
import javax.inject.Named;
import javax.inject.Singleton;
import javax.naming.NamingException;
import javax.servlet.Filter;
import javax.servlet.Servlet;
import javax.servlet.http.HttpUpgradeHandler;
import javax.servlet.jsp.JspFactory;
import javax.servlet.jsp.tagext.JspTag;
import org.apache.catalina.Connector;
import org.apache.catalina.Container;
import org.apache.catalina.Context;
import org.apache.catalina.Deployer;
import org.apache.catalina.Engine;
import org.apache.catalina.LifecycleException;
import org.apache.catalina.LifecycleListener;
import org.apache.catalina.Loader;
import org.apache.catalina.Realm;
import org.apache.catalina.connector.Request;
import org.apache.catalina.core.StandardContext;
import org.apache.catalina.core.StandardEngine;
import org.apache.catalina.startup.ContextConfig;
import org.apache.catalina.util.RequestUtil;
import org.apache.catalina.util.ServerInfo;
import org.apache.jasper.runtime.JspFactoryImpl;
import org.apache.jasper.xmlparser.ParserUtils;
import org.glassfish.api.admin.ServerEnvironment;
import org.glassfish.api.deployment.DeploymentContext;
import org.glassfish.api.event.EventListener;
import org.glassfish.api.event.Events;
import org.glassfish.api.invocation.InvocationManager;
import org.glassfish.api.web.TldProvider;
import org.glassfish.grizzly.config.ContextRootInfo;
import org.glassfish.grizzly.config.dom.NetworkConfig;
import org.glassfish.grizzly.config.dom.NetworkListener;
import org.glassfish.grizzly.config.dom.NetworkListeners;
import org.glassfish.grizzly.http.server.util.Mapper;
import org.glassfish.grizzly.http.server.util.MappingData;
import org.glassfish.grizzly.http.util.DataChunk;
import org.glassfish.hk2.api.DynamicConfiguration;
import org.glassfish.hk2.api.DynamicConfigurationService;
import org.glassfish.hk2.api.PostConstruct;
import org.glassfish.hk2.api.PreDestroy;
import org.glassfish.hk2.api.ServiceLocator;
import org.glassfish.internal.api.ClassLoaderHierarchy;
import org.glassfish.internal.api.ServerContext;
import org.glassfish.internal.data.ApplicationInfo;
import org.glassfish.internal.data.ApplicationRegistry;
import org.glassfish.internal.grizzly.ContextMapper;
import org.glassfish.web.LogFacade;
import org.glassfish.web.admin.monitor.HttpServiceStatsProviderBootstrap;
import org.glassfish.web.admin.monitor.JspProbeProvider;
import org.glassfish.web.admin.monitor.RequestProbeProvider;
import org.glassfish.web.admin.monitor.ServletProbeProvider;
import org.glassfish.web.admin.monitor.SessionProbeProvider;
import org.glassfish.web.admin.monitor.WebModuleProbeProvider;
import org.glassfish.web.admin.monitor.WebStatsProviderBootstrap;
import org.glassfish.web.config.serverbeans.SessionProperties;
import org.glassfish.web.deployment.archivist.WebArchivist;
import org.glassfish.web.deployment.runtime.SunWebAppImpl;
import org.glassfish.web.deployment.util.WebValidatorWithoutCL;
import org.glassfish.web.loader.WebappClassLoader;
import org.glassfish.web.valve.GlassFishValve;
import org.jvnet.hk2.annotations.Service;
import org.jvnet.hk2.config.ConfigSupport;
import org.jvnet.hk2.config.ObservableBean;
import org.jvnet.hk2.config.Transactions;
import org.jvnet.hk2.config.types.Property;
import org.xml.sax.EntityResolver;
import static com.sun.enterprise.deployment.WebBundleDescriptor.AFTER_SERVLET_CONTEXT_INITIALIZED_EVENT;
import static com.sun.enterprise.util.StringUtils.parseStringList;
import static com.sun.enterprise.util.io.FileUtils.makeFriendlyFilename;
import static com.sun.enterprise.web.Constants.ACCESS_LOGGING_ENABLED;
import static com.sun.enterprise.web.Constants.ACCESS_LOG_BUFFER_SIZE_PROPERTY;
import static com.sun.enterprise.web.Constants.ACCESS_LOG_PREFIX;
import static com.sun.enterprise.web.Constants.ACCESS_LOG_PROPERTY;
import static com.sun.enterprise.web.Constants.ACCESS_LOG_WRITE_INTERVAL_PROPERTY;
import static com.sun.enterprise.web.Constants.DEFAULT_WEB_MODULE_NAME;
import static com.sun.enterprise.web.Constants.ERROR_REPORT_VALVE;
import static com.sun.enterprise.web.Constants.SSO_ENABLED;
import static java.text.MessageFormat.format;
import static java.util.Arrays.stream;
import static java.util.logging.Level.FINE;
import static java.util.logging.Level.FINEST;
import static java.util.logging.Level.INFO;
import static java.util.logging.Level.SEVERE;
import static java.util.logging.Level.WARNING;
import static java.util.stream.Collectors.toList;
import static org.glassfish.api.admin.ServerEnvironment.DEFAULT_INSTANCE_NAME;
import static org.glassfish.api.event.EventTypes.PREPARE_SHUTDOWN;
import static org.glassfish.api.web.Constants.ADMIN_VS;
import static org.glassfish.internal.deployment.Deployment.ALL_APPLICATIONS_PROCESSED;
import static org.glassfish.internal.deployment.Deployment.DEPLOYMENT_FAILURE;
import static org.glassfish.internal.deployment.Deployment.DISABLE_START;
import static org.glassfish.internal.deployment.Deployment.UNDEPLOYMENT_FAILURE;
import static org.glassfish.web.LogFacade.*;
/**
* Web container service
*
* @author jluehe
* @author amyroh
* @author swchan2
*/
@Service(name = "com.sun.enterprise.web.WebContainer")
@Singleton
public class WebContainer implements org.glassfish.api.container.Container, PostConstruct, PreDestroy, EventListener {
// -------------------------------------------------- Constants
public static final String DISPATCHER_MAX_DEPTH = "dispatcher-max-depth";
public static final String JWS_APPCLIENT_EAR_NAME = "__JWSappclients";
public static final String JWS_APPCLIENT_WAR_NAME = "sys";
private static final String JWS_APPCLIENT_MODULE_NAME = JWS_APPCLIENT_EAR_NAME + ":" + JWS_APPCLIENT_WAR_NAME + ".war";
private static final String DOL_DEPLOYMENT = "com.sun.enterprise.web.deployment.backend";
private static final String MONITORING_NODE_SEPARATOR = "/";
private static final Logger logger = LogFacade.getLogger();
private static final ResourceBundle rb = logger.getResourceBundle();
/**
* Are we using Tomcat deployment backend or DOL?
*/
static boolean useDOLforDeployment = true;
// ----------------------------------------------------- Instance Variables
@Inject
private ApplicationRegistry appRegistry;
@Inject
private ClassLoaderHierarchy classLoaderHierarchy;
@Inject
private ComponentEnvManager componentEnvManager;
@Inject
@org.jvnet.hk2.annotations.Optional
private DasConfig dasConfig;
@Inject
private Domain domain;
@Inject
private Events events;
@Inject
private FileLoggerHandlerFactory fileLoggerHandlerFactory;
@Inject
private GrizzlyService grizzlyService;
@Inject
private ServiceLocator serviceLocator;
@Inject
private JavaEEIOUtils javaEEIOUtils;
@Inject
@org.jvnet.hk2.annotations.Optional
private JCDIService cdiService;
@Inject
@Named(DEFAULT_INSTANCE_NAME)
private Config serverConfig;
@Inject
@Named(DEFAULT_INSTANCE_NAME)
private Server server;
@Inject
private ServerContext _serverContext;
@Inject
private Transactions transactions;
@Inject
private HotDeployService hotDeployService;
@Inject
private LoggingRuntime loggingRuntime;
private final Map connectorMap = new HashMap<>();
private EmbeddedWebContainer _embedded;
private Engine engine;
private String instanceName;
private WebConnector jkConnector;
private String logLevel = "INFO";
/**
* Allow disabling accessLog mechanism
*/
protected boolean globalAccessLoggingEnabled = true;
/**
* AccessLog buffer size for storing logs.
*/
protected String globalAccessLogBufferSize;
/**
* AccessLog interval before the valve flush its buffer to the disk.
*/
protected String globalAccessLogWriteInterval;
/**
* AccessLog prefix
*/
protected String globalAccessLogPrefix;
/**
* The default-redirect port
*/
protected int defaultRedirectPort = -1;
@Inject
protected ServerEnvironment instance = null;
/**
* Controls the verbosity of the web container subsystem's debug messages.
*
* This value is non-zero only when the iAS level is one of FINE, FINER or FINEST.
*/
protected int _debug = 0;
/**
* Absolute path for location where all the deployed standalone modules are stored for this Server Instance.
*/
protected File _modulesRoot;
/**
* Top-level directory for files generated by application web modules.
*/
private String _appsWorkRoot;
/**
* Top-level directory where ejb stubs for applications are stored.
*/
private String appsStubRoot;
/**
* Has this component been started yet?
*/
protected boolean _started;
/**
* The global (at the http-service level) ssoEnabled property.
*/
protected boolean globalSSOEnabled = true;
protected volatile WebContainerFeatureFactory webContainerFeatureFactory;
/**
* The value of the instance-level session property named "enableCookies"
*/
boolean instanceEnableCookies = true;
@Inject
ServerConfigLookup serverConfigLookup;
@Inject
private SecurityDeployer securityDeployer;
protected JspProbeProvider jspProbeProvider;
protected RequestProbeProvider requestProbeProvider;
protected ServletProbeProvider servletProbeProvider;
protected SessionProbeProvider sessionProbeProvider;
protected WebModuleProbeProvider webModuleProbeProvider;
protected WebConfigListener configListener;
// Indicates whether we are being shut down
private boolean isShutdown;
private final Object mapperUpdateSync = new Object();
private SecurityService securityService;
protected HttpServiceStatsProviderBootstrap httpStatsProviderBootstrap;
private WebStatsProviderBootstrap webStatsProviderBootstrap;
private InjectionManager injectionMgr;
private InvocationManager invocationMgr;
private Collection tldProviders;
private String logServiceFile;
/**
* Static initialization
*/
static {
if (System.getProperty(DOL_DEPLOYMENT) != null) {
useDOLforDeployment = Boolean.valueOf(System.getProperty(DOL_DEPLOYMENT));
}
}
private WebConfigListener addAndGetWebConfigListener() {
ServiceLocator locator = serviceLocator;
DynamicConfiguration config = locator.getService(DynamicConfigurationService.class).createDynamicConfiguration();
config.addActiveDescriptor(WebConfigListener.class);
config.commit();
return locator.getService(WebConfigListener.class);
}
@Override
public void postConstruct() {
ReentrantReadWriteLock mapperLock = grizzlyService.obtainMapperLock();
mapperLock.writeLock().lock();
try {
createProbeProviders();
injectionMgr = serviceLocator.getService(InjectionManager.class);
invocationMgr = serviceLocator.getService(InvocationManager.class);
tldProviders = serviceLocator.getAllServices(TldProvider.class);
createStatsProviders();
setJspFactory();
_appsWorkRoot = instance.getApplicationCompileJspPath().getAbsolutePath();
_modulesRoot = instance.getApplicationRepositoryPath();
appsStubRoot = instance.getApplicationStubPath().getAbsolutePath();
// TODO: ParserUtils should become a @Service and it should initialize itself.
// TODO: there should be only one EntityResolver for both DigesterFactory
// and ParserUtils
File root = _serverContext.getInstallRoot();
File libRoot = new File(root, "lib");
File schemas = new File(libRoot, "schemas");
File dtds = new File(libRoot, "dtds");
try {
ParserUtils.setSchemaResourcePrefix(schemas.toURI().toURL().toString());
ParserUtils.setDtdResourcePrefix(dtds.toURI().toURL().toString());
ParserUtils.setEntityResolver(serviceLocator.getService(EntityResolver.class, "web"));
} catch (MalformedURLException e) {
logger.log(SEVERE, EXCEPTION_SET_SCHEMAS_DTDS_LOCATION, e);
}
instanceName = _serverContext.getInstanceName();
webContainerFeatureFactory = getWebContainerFeatureFactory();
configureDynamicReloadingSettings();
setDebugLevel();
String maxDepth = null;
org.glassfish.web.config.serverbeans.WebContainer configWC = serverConfig
.getExtensionByType(org.glassfish.web.config.serverbeans.WebContainer.class);
if (configWC != null) {
maxDepth = configWC.getPropertyValue(DISPATCHER_MAX_DEPTH);
}
if (maxDepth != null) {
int depth = -1;
try {
depth = Integer.parseInt(maxDepth);
} catch (NumberFormatException e) {
}
if (depth > 0) {
Request.setMaxDispatchDepth(depth);
logger.log(FINE, MAX_DISPATCH_DEPTH_SET, maxDepth);
}
}
File currentLogFile = loggingRuntime.getCurrentLogFile();
if (currentLogFile != null) {
logServiceFile = currentLogFile.getAbsolutePath();
}
Level level = Logger.getLogger("org.apache.catalina.level").getLevel();
if (level != null) {
logLevel = level.getName();
}
_embedded = serviceLocator.getService(EmbeddedWebContainer.class);
_embedded.setWebContainer(this);
_embedded.setLogServiceFile(logServiceFile);
_embedded.setLogLevel(logLevel);
_embedded.setFileLoggerHandlerFactory(fileLoggerHandlerFactory);
_embedded.setWebContainerFeatureFactory(webContainerFeatureFactory);
_embedded.setCatalinaHome(instance.getInstanceRoot().getAbsolutePath());
_embedded.setCatalinaBase(instance.getInstanceRoot().getAbsolutePath());
_embedded.setUseNaming(false);
if (_debug > 1) {
_embedded.setDebug(_debug);
}
_embedded.setLogger(new IASLogger(logger));
engine = _embedded.createEngine();
engine.setParentClassLoader(EmbeddedWebContainer.class.getClassLoader());
engine.setService(_embedded);
_embedded.addEngine(engine);
((StandardEngine) engine).setDomain(_serverContext.getDefaultDomainName());
engine.setName(_serverContext.getDefaultDomainName());
/*
* Set the server info. By default, the server info is taken from Version#getVersion. However, customers may override it
* via the product.name system property. Some customers prefer not to disclose the server info for security reasons, in
* which case they would set the value of the product.name system property to the empty string. In this case, the server
* name will not be publicly disclosed via the "Server" HTTP response header (which will be suppressed) or any container
* generated error pages. However, it will still appear in the server logs (see IT 6900).
*/
String serverInfo = System.getProperty("product.name");
if (serverInfo == null) {
ServerInfo.setServerInfo(Version.getVersion());
ServerInfo.setPublicServerInfo(Version.getVersion());
} else if (serverInfo.isEmpty()) {
ServerInfo.setServerInfo(Version.getVersion());
ServerInfo.setPublicServerInfo(serverInfo);
} else {
ServerInfo.setServerInfo(serverInfo);
ServerInfo.setPublicServerInfo(serverInfo);
}
initInstanceSessionProperties();
configListener = addAndGetWebConfigListener();
ObservableBean bean = (ObservableBean) ConfigSupport.getImpl(serverConfig.getHttpService());
bean.addListener(configListener);
bean = (ObservableBean) ConfigSupport.getImpl(serverConfig.getNetworkConfig().getNetworkListeners());
bean.addListener(configListener);
if (serverConfig.getAvailabilityService() != null) {
bean = (ObservableBean) ConfigSupport.getImpl(serverConfig.getAvailabilityService());
bean.addListener(configListener);
}
transactions.addListenerForType(SystemProperty.class, configListener);
configListener.setNetworkConfig(serverConfig.getNetworkConfig());
// embedded mode does not have manager-propertie in domain.xml
if (configListener.managerProperties != null) {
ObservableBean managerBean = (ObservableBean) ConfigSupport.getImpl(configListener.managerProperties);
managerBean.addListener(configListener);
}
if (serverConfig.getJavaConfig() != null) {
((ObservableBean) ConfigSupport.getImpl(serverConfig.getJavaConfig())).addListener(configListener);
}
configListener.setContainer(this);
configListener.setLogger(logger);
events.register(this);
grizzlyService.addMapperUpdateListener(configListener);
HttpService httpService = serverConfig.getHttpService();
NetworkConfig networkConfig = serverConfig.getNetworkConfig();
if (networkConfig != null) {
// continue;
securityService = serverConfig.getSecurityService();
// Configure HTTP listeners
NetworkListeners networkListeners = networkConfig.getNetworkListeners();
if (networkListeners != null) {
List listeners = networkListeners.getNetworkListener();
for (NetworkListener listener : listeners) {
createHttpListener(listener, httpService);
}
}
setDefaultRedirectPort(defaultRedirectPort);
// Configure virtual servers
createHosts(httpService, securityService);
}
loadSystemDefaultWebModules();
// _lifecycle.fireLifecycleEvent(START_EVENT, null);
_started = true;
/*
* Start the embedded container. Make sure to set the thread's context classloader to the classloader of this class (see
* IT 8866 for details)
*/
ClassLoader current = Thread.currentThread().getContextClassLoader();
Thread.currentThread().setContextClassLoader(getClass().getClassLoader());
try {
/*
* Trigger a call to sun.awt.AppContext.getAppContext(). This will pin the classloader of this class in memory and fix a
* memory leak affecting instances of WebappClassLoader that was caused by a JRE implementation change in 1.6.0_15
* onwards. See IT 11110
*/
ImageIO.getCacheDirectory();
_embedded.start();
} catch (LifecycleException le) {
logger.log(SEVERE, UNABLE_TO_START_WEB_CONTAINER, le);
} finally {
// Restore original context classloader
Thread.currentThread().setContextClassLoader(current);
}
} finally {
mapperLock.writeLock().unlock();
}
}
@Override
public void event(Event> event) {
if (event.is(ALL_APPLICATIONS_PROCESSED)) {
// Configure default web modules for virtual servers after all
// applications are processed
loadDefaultWebModulesAfterAllAppsProcessed();
} else if (event.is(PREPARE_SHUTDOWN)) {
isShutdown = true;
} else if (event.is(DEPLOYMENT_FAILURE) || event.is(UNDEPLOYMENT_FAILURE)) {
DeploymentContext deploymentContext = (DeploymentContext) event.hook();
try {
// Fix https://github.com/payara/Payara/issues/315
WebBundleDescriptor webBundleDescriptor = deploymentContext.getModuleMetaData(WebBundleDescriptor.class);
if (webBundleDescriptor != null) {
componentEnvManager.unbindFromComponentNamespace(webBundleDescriptor);
}
} catch (NamingException ex) {
logger.log(SEVERE, EXCEPTION_DURING_DESTROY, ex);
}
} else if (event.is(DISABLE_START)) {
ApplicationInfo applicationInfo = (ApplicationInfo) event.hook();
String appName = applicationInfo.getName();
Applications applications = serviceLocator.getService(Applications.class);
if (applications == null) {
return;
}
com.sun.enterprise.config.serverbeans.Application application = applications.getApplication(appName);
if (application == null) {
return;
}
String contextRoot = application.getContextRoot();
// If no context root, no context to mark as unavailable
if (contextRoot != null) {
StringBuilder stringBuilder = new StringBuilder();
// The final comma gets stripped out in suspendWebModule
getVirtualServers().forEach(virtualServer -> stringBuilder.append(virtualServer.getID()).append(","));
// Just provide all virtual hosts, it'll only suspend the app on those where it's deployed
suspendWebModule(contextRoot, appName, stringBuilder.toString(),
true);
}
}
}
/**
* Notifies any interested listeners that all ServletContextListeners of the web module represented by the given
* WebBundleDescriptor have been invoked at their contextInitialized method
*/
void afterServletContextInitializedEvent(WebBundleDescriptor webBundleDescriptor) {
events.send(new Event<>(AFTER_SERVLET_CONTEXT_INITIALIZED_EVENT, webBundleDescriptor), false);
}
@Override
public void preDestroy() {
try {
for (Connector connector : _embedded.findConnectors()) {
deleteConnector((WebConnector) connector);
}
_embedded.removeEngine(getEngine());
_embedded.destroy();
} catch (LifecycleException le) {
logger.log(SEVERE, UNABLE_TO_STOP_WEB_CONTAINER, le);
}
}
JavaEEIOUtils getJavaEEIOUtils() {
return javaEEIOUtils;
}
/**
* Returns true if the container has been shut down.
*
* @return false if the container has never been started.
*/
public boolean isShutdown() {
return isShutdown;
}
Collection getTldProviders() {
return tldProviders;
}
/**
* Gets the probe provider for servlet related events.
*
* @return
*/
public ServletProbeProvider getServletProbeProvider() {
return servletProbeProvider;
}
/**
* Gets the probe provider for jsp related events.
*
* @return
*/
public JspProbeProvider getJspProbeProvider() {
return jspProbeProvider;
}
/**
* Gets the probe provider for session related events.
*
* @return
*/
public SessionProbeProvider getSessionProbeProvider() {
return sessionProbeProvider;
}
/**
* Gets the probe provider for request/response related events.
*
* @return
*/
public RequestProbeProvider getRequestProbeProvider() {
return requestProbeProvider;
}
/**
* Gets the probe provider for web module related events.
*
* @return
*/
public WebModuleProbeProvider getWebModuleProbeProvider() {
return webModuleProbeProvider;
}
@Override
public String getName() {
return "Web";
}
@Override
public Class extends WebDeployer> getDeployer() {
return WebDeployer.class;
}
InvocationManager getInvocationManager() {
return invocationMgr;
}
public WebConnector getJkConnector() {
return jkConnector;
}
public Map getConnectorMap() {
return connectorMap;
}
/**
* Instantiates and injects the given Servlet class for the given WebModule
*/
T createServletInstance(WebModule module, Class clazz) throws Exception {
validateJSR299Scope(clazz);
WebComponentInvocation webComponentInvocation = new WebComponentInvocation(module);
try {
invocationMgr.preInvoke(webComponentInvocation);
return injectionMgr.createManagedObject(clazz);
} finally {
invocationMgr.postInvoke(webComponentInvocation);
}
}
/**
* Instantiates and injects the given Filter class for the given WebModule
*/
T createFilterInstance(WebModule module, Class clazz) throws Exception {
validateJSR299Scope(clazz);
WebComponentInvocation webComponentInvocation = new WebComponentInvocation(module);
try {
invocationMgr.preInvoke(webComponentInvocation);
return injectionMgr.createManagedObject(clazz);
} finally {
invocationMgr.postInvoke(webComponentInvocation);
}
}
/**
* Instantiates and injects the given EventListener class for the given WebModule
*/
T createListenerInstance(WebModule module, Class clazz) throws Exception {
validateJSR299Scope(clazz);
WebComponentInvocation webComponentInvocation = new WebComponentInvocation(module);
try {
invocationMgr.preInvoke(webComponentInvocation);
return injectionMgr.createManagedObject(clazz);
} finally {
invocationMgr.postInvoke(webComponentInvocation);
}
}
/**
* Instantiates and injects the given HttpUpgradeHandler class for the given WebModule
*/
T createHttpUpgradeHandlerInstance(WebModule module, Class clazz) throws Exception {
validateJSR299Scope(clazz);
WebComponentInvocation webComponentInvocation = new WebComponentInvocation(module);
try {
invocationMgr.preInvoke(webComponentInvocation);
return injectionMgr.createManagedObject(clazz);
} finally {
invocationMgr.postInvoke(webComponentInvocation);
}
}
/**
* Instantiates and injects the given tag handler class for the given WebModule
*
* @param
* @param module
* @param clazz
* @return
* @throws java.lang.Exception
*/
public T createTagHandlerInstance(WebModule module, Class clazz) throws Exception {
WebComponentInvocation webComponentInvocation = new WebComponentInvocation(module);
try {
invocationMgr.preInvoke(webComponentInvocation);
return injectionMgr.createManagedObject(clazz);
} finally {
invocationMgr.postInvoke(webComponentInvocation);
}
}
/**
* Use an network-listener subelements and creates a corresponding Tomcat Connector for each.
*
* @param listener the NetworkListener config object.
* @param httpService the http-service element.
* @return
*/
protected WebConnector createHttpListener(NetworkListener listener, HttpService httpService) {
return createHttpListener(listener, httpService, null);
}
protected WebConnector createHttpListener(NetworkListener listener, HttpService httpService, Mapper mapper) {
if (!Boolean.valueOf(listener.getEnabled())) {
return null;
}
int port = grizzlyService.getRealPort(listener);
WebConnector connector;
checkHostnameUniqueness(listener.getName(), httpService);
if (mapper == null) {
for (Mapper m : serviceLocator.getAllServices(Mapper.class)) {
if (m.getPort() == port && m instanceof ContextMapper) {
ContextMapper contextMapper = (ContextMapper) m;
if (listener.getName().equals(contextMapper.getId())) {
mapper = m;
break;
}
}
}
}
String defaultVirtualServer = listener.findHttpProtocol().getHttp().getDefaultVirtualServer();
if (!defaultVirtualServer.equals(ADMIN_VS)) {
// Before we start a WebConnector, let's makes sure there is
// not another Container already listening on that port
DataChunk host = DataChunk.newInstance();
char[] defaultVirtualServerAsChars = defaultVirtualServer.toCharArray();
host.setChars(defaultVirtualServerAsChars, 0, defaultVirtualServerAsChars.length);
DataChunk dataChunk = DataChunk.newInstance();
dataChunk.setChars(new char[] { '/' }, 0, 1);
MappingData mappingData = new MappingData();
try {
mapper.map(host, dataChunk, mappingData);
} catch (Exception e) {
logger.log(FINE, "", e);
}
if (mappingData.context != null && mappingData.context instanceof ContextRootInfo) {
ContextRootInfo rootInfo = (ContextRootInfo) mappingData.context;
if (!(rootInfo.getHttpHandler() instanceof ContainerMapper)) {
new BindException("Port " + port + " is already used by Container: " + rootInfo.getHttpHandler() + " and will not get started.").printStackTrace();
return null;
}
}
}
/*
* Create Connector. Connector is SSL-enabled if 'security-enabled' attribute in element is set to TRUE.
*/
boolean isSecure = Boolean.valueOf(listener.findHttpProtocol().getSecurityEnabled());
if (isSecure && defaultRedirectPort == -1) {
defaultRedirectPort = port;
}
String address = listener.getAddress();
if ("any".equals(address) || "ANY".equals(address) || "INADDR_ANY".equals(address)) {
address = null;
/*
* Setting 'address' to NULL will cause Tomcat to pass a NULL InetAddress argument to the java.net.ServerSocket
* constructor, meaning that the server socket will accept connections on any/all local addresses.
*/
}
connector = (WebConnector) _embedded.createConnector(address, port, isSecure);
connector.setMapper(mapper);
connector.setJvmRoute(engine.getJvmRoute());
logger.log(INFO, HTTP_LISTENER_CREATED, new Object[] { listener.getName(), listener.getAddress(), Integer.toString(port) });
connector.setDefaultHost(listener.findHttpProtocol().getHttp().getDefaultVirtualServer());
connector.setName(listener.getName());
connector.setInstanceName(instanceName);
connector.configure(listener, isSecure, httpService);
_embedded.addConnector(connector);
connectorMap.put(listener.getName(), connector);
// If we already know the redirect port, then set it now
// This situation will occurs when dynamic reconfiguration occurs
String redirectPort = listener.findHttpProtocol().getHttp().getRedirectPort();
if (redirectPort != null) {
connector.setRedirectPort(Integer.parseInt(redirectPort));
} else if (defaultRedirectPort != -1) {
connector.setRedirectPort(defaultRedirectPort);
}
ObservableBean httpListenerBean = (ObservableBean) ConfigSupport.getImpl(listener);
httpListenerBean.addListener(configListener);
return connector;
}
/**
* Starts the AJP connector that will listen to call from Apache using mod_jk, mod_jk2 or mod_ajp.
*
* @param listener
* @param httpService
* @return
*/
protected WebConnector createJKConnector(NetworkListener listener, HttpService httpService) {
int port = 8009;
boolean isSecure = false;
String address = null;
if (listener == null) {
String portString = System.getProperty("com.sun.enterprise.web.connector.enableJK");
if (portString == null) {
// do not create JK Connector if property is not set
return null;
} else {
try {
port = Integer.parseInt(portString);
} catch (NumberFormatException ex) {
// use default port 8009
port = 8009;
}
}
} else {
port = Integer.parseInt(listener.getPort());
isSecure = Boolean.valueOf(listener.findHttpProtocol().getSecurityEnabled());
address = listener.getAddress();
}
if (isSecure && defaultRedirectPort == -1) {
defaultRedirectPort = port;
}
if ("any".equals(address) || "ANY".equals(address) || "INADDR_ANY".equals(address)) {
address = null;
/*
* Setting 'address' to NULL will cause Tomcat to pass a NULL InetAddress argument to the java.net.ServerSocket
* constructor, meaning that the server socket will accept connections on any/all local addresses.
*/
}
jkConnector = (WebConnector) _embedded.createConnector(address, port, "ajp");
jkConnector.configureJKProperties(listener);
jkConnector.setDomain(_serverContext.getDefaultDomainName());
jkConnector.setInstanceName(instanceName);
String defaultHost = "server";
String jkConnectorName = "jk-connector";
if (listener != null) {
defaultHost = listener.findHttpProtocol().getHttp().getDefaultVirtualServer();
jkConnectorName = listener.getName();
jkConnector.configure(listener, isSecure, httpService);
connectorMap.put(listener.getName(), jkConnector);
if (logger.isLoggable(INFO)) {
logger.log(INFO, JK_LISTENER_CREATED, new Object[] { listener.getName(), listener.getAddress(), listener.getPort() });
}
for (Mapper mapper : serviceLocator.getAllServices(Mapper.class)) {
if (mapper.getPort() == port && mapper instanceof ContextMapper) {
ContextMapper contextMapper = (ContextMapper) mapper;
if (listener.getName().equals(contextMapper.getId())) {
jkConnector.setMapper(mapper);
break;
}
}
}
}
jkConnector.setDefaultHost(defaultHost);
jkConnector.setName(jkConnectorName);
_embedded.addConnector(jkConnector);
return jkConnector;
}
/**
* Assigns the given redirect port to each Connector whose corresponding http-listener element in domain.xml does not
* specify its own redirect-port attribute.
*
* The given defaultRedirectPort corresponds to the port number of the first security-enabled http-listener in
* domain.xml.
*
* This method does nothing if none of the http-listener elements is security-enabled, in which case Tomcat's default
* redirect port (443) will be used.
*
* @param defaultRedirectPort The redirect port to be assigned to any Connector object that doesn't specify its own
*/
private void setDefaultRedirectPort(int defaultRedirectPort) {
if (defaultRedirectPort != -1) {
Connector[] connectors = _embedded.getConnectors();
for (Connector connector : connectors) {
if (connector.getRedirectPort() == -1) {
connector.setRedirectPort(defaultRedirectPort);
}
}
}
}
/**
* Configure http-service properties.
*
* @param httpService
* @param connector
* @deprecated most of these properties are handled elsewhere. validate and remove outdated properties checks
*/
@Deprecated
public void configureHttpServiceProperties(HttpService httpService, PECoyoteConnector connector) {
// Configure Connector with properties
List httpServiceProps = httpService.getProperty();
// Set default ProxyHandler impl, may be overriden by
// proxyHandler property
connector.setProxyHandler(new ProxyHandlerImpl());
globalSSOEnabled = ConfigBeansUtilities.toBoolean(httpService.getSsoEnabled());
globalAccessLoggingEnabled = ConfigBeansUtilities.toBoolean(httpService.getAccessLoggingEnabled());
globalAccessLogWriteInterval = httpService.getAccessLog().getWriteIntervalSeconds();
globalAccessLogBufferSize = httpService.getAccessLog().getBufferSizeBytes();
globalAccessLogPrefix = httpService.getAccessLog().getPropertyValue(Constants.ACCESS_LOG_PREFIX);
if (httpServiceProps != null) {
for (Property httpServiceProp : httpServiceProps) {
String propName = httpServiceProp.getName();
String propValue = httpServiceProp.getValue();
if (connector.configureHttpListenerProperty(propName, propValue)) {
continue;
}
if ("connectionTimeout".equals(propName)) {
connector.setConnectionTimeout(Integer.parseInt(propValue));
} else if ("tcpNoDelay".equals(propName)) {
connector.setTcpNoDelay(ConfigBeansUtilities.toBoolean(propValue));
} else if ("traceEnabled".equals(propName)) {
connector.setAllowTrace(ConfigBeansUtilities.toBoolean(propValue));
} else if ("ssl-session-timeout".equals(propName)) {
connector.setSslSessionTimeout(propValue);
} else if ("ssl3-session-timeout".equals(propName)) {
connector.setSsl3SessionTimeout(propValue);
} else if ("ssl-cache-entries".equals(propName)) {
connector.setSslSessionCacheSize(propValue);
} else if ("proxyHandler".equals(propName)) {
connector.setProxyHandler(propValue);
} else {
String msg = rb.getString(LogFacade.INVALID_HTTP_SERVICE_PROPERTY);
logger.log(Level.WARNING, MessageFormat.format(msg, httpServiceProp.getName()));
}
}
}
}
/*
* Ensures that the host names of all virtual servers associated with the HTTP listener with the given listener id are
* unique.
*
* @param listenerId The id of the HTTP listener whose associated virtual servers are checked for uniqueness of host
* names
*
* @param httpService The http-service element whose virtual servers are checked
*/
private void checkHostnameUniqueness(String listenerId, HttpService httpService) {
List listenerVirtualServers = null;
// Determine all the virtual servers associated with the given listener
for (com.sun.enterprise.config.serverbeans.VirtualServer virtualServer : httpService.getVirtualServer()) {
List networkListeners = parseStringList(virtualServer.getNetworkListeners(), ",");
for (int j = 0; networkListeners != null && j < networkListeners.size(); j++) {
if (listenerId.equals(networkListeners.get(j))) {
if (listenerVirtualServers == null) {
listenerVirtualServers = new ArrayList<>();
}
listenerVirtualServers.add(virtualServer);
break;
}
}
}
if (listenerVirtualServers == null) {
return;
}
for (int i = 0; i < listenerVirtualServers.size(); i++) {
com.sun.enterprise.config.serverbeans.VirtualServer vs = listenerVirtualServers.get(i);
List hosts = parseStringList(vs.getHosts(), ",");
for (int j = 0; hosts != null && j < hosts.size(); j++) {
String host = hosts.get(j);
for (int k = 0; k < listenerVirtualServers.size(); k++) {
if (k <= i) {
continue;
}
com.sun.enterprise.config.serverbeans.VirtualServer otherVs = listenerVirtualServers.get(k);
List otherHosts = parseStringList(otherVs.getHosts(), ",");
for (int l = 0; otherHosts != null && l < otherHosts.size(); l++) {
if (host.equals(otherHosts.get(l))) {
logger.log(SEVERE, DUPLICATE_HOST_NAME, new Object[] { host, vs.getId(), otherVs.getId(), listenerId });
}
}
}
}
}
}
/**
* Enumerates the virtual-server subelements of the given http-service element, and creates a corresponding Host for
* each.
*
* @param httpService The http-service element
* @param securityService The security-service element
*/
protected void createHosts(HttpService httpService, SecurityService securityService) {
for (com.sun.enterprise.config.serverbeans.VirtualServer virtualServer : httpService.getVirtualServer()) {
createHost(virtualServer, httpService, securityService);
if (logger.isLoggable(INFO)) {
logger.log(INFO, VIRTUAL_SERVER_CREATED, virtualServer.getId());
}
}
}
/**
* Creates a Host from a virtual-server config bean.
*
* @param vsBean The virtual-server configuration bean
* @param httpService The http-service element.
* @param securityService The security-service element
* @return
*/
public VirtualServer createHost(com.sun.enterprise.config.serverbeans.VirtualServer vsBean, HttpService httpService, SecurityService securityService) {
String virtualServerId = vsBean.getId();
String docroot = vsBean.getPropertyValue("docroot");
if (docroot == null) {
docroot = vsBean.getDocroot();
}
validateDocroot(docroot, virtualServerId, vsBean.getDefaultWebModule());
VirtualServer virtualServer = createHost(virtualServerId, vsBean, docroot, null);
// cache control
Property cacheProp = vsBean.getProperty("setCacheControl");
if (cacheProp != null) {
virtualServer.configureCacheControl(cacheProp.getValue());
}
PEAccessLogValve accessLogValve = virtualServer.getAccessLogValve();
boolean startAccessLog = accessLogValve.configure(
virtualServerId, vsBean, httpService, domain, serviceLocator, webContainerFeatureFactory, globalAccessLogBufferSize,
globalAccessLogWriteInterval, globalAccessLogPrefix);
if (startAccessLog && virtualServer.isAccessLoggingEnabled(globalAccessLoggingEnabled)) {
virtualServer.addValve((GlassFishValve) accessLogValve);
}
logger.log(FINEST, VIRTUAL_SERVER_CREATED, virtualServerId);
/*
* We must configure the Host with its associated port numbers and alias names before adding it as an engine child and
* thereby starting it, because a MapperListener, which is associated with an HTTP listener and receives notifications
* about Host registrations, relies on these Host properties in order to determine whether a new Host needs to be added
* to the HTTP listener's Mapper.
*/
configureHost(virtualServer, securityService);
virtualServer.setDomain(domain);
virtualServer.setServices(serviceLocator);
virtualServer.setClassLoaderHierarchy(classLoaderHierarchy);
// Add Host to Engine
engine.addChild(virtualServer);
ObservableBean virtualServerBean = (ObservableBean) ConfigSupport.getImpl(vsBean);
virtualServerBean.addListener(configListener);
return virtualServer;
}
/**
* Validate the docroot properties of a virtual-server.
*
* @param docroot
* @param virtualServerId
* @param defaultWebModule
*/
protected void validateDocroot(String docroot, String virtualServerId, String defaultWebModule) {
if (docroot == null) {
return;
}
if (!new File(docroot).exists()) {
throw new IllegalArgumentException(format(rb.getString(VIRTUAL_SERVER_INVALID_DOCROOT), virtualServerId, docroot));
}
}
/**
* Configures the given virtual server.
*
* @param virtualServer The virtual server to be configured
* @param securityService The security-service element
*/
protected void configureHost(VirtualServer virtualServer, SecurityService securityService) {
com.sun.enterprise.config.serverbeans.VirtualServer vsBean = virtualServer.getBean();
virtualServer.configureAliases();
// Set the ports with which this virtual server is associated
List listeners = StringUtils.parseStringList(vsBean.getNetworkListeners(), ",");
if (listeners == null) {
return;
}
HashSet httpListeners = new HashSet<>();
for (String listener : listeners) {
boolean found = false;
for (NetworkListener httpListener : serverConfig.getNetworkConfig().getNetworkListeners().getNetworkListener()) {
if (httpListener.getName().equals(listener)) {
httpListeners.add(httpListener);
found = true;
break;
}
}
if (!found) {
logger.log(SEVERE, format(rb.getString(LISTENER_REFERENCED_BY_HOST_NOT_EXIST), listener, virtualServer.getName()));
}
}
configureHostPortNumbers(virtualServer, httpListeners);
virtualServer.configureCatalinaProperties();
virtualServer.configureAuthRealm(securityService);
virtualServer.addProbes(globalAccessLoggingEnabled);
}
/**
* Configures the given virtual server with the port numbers of its associated http listeners.
*
* @param virtualServer The virtual server to configure
* @param listeners The http listeners with which the given virtual server is associated
*/
protected void configureHostPortNumbers(VirtualServer virtualServer, HashSet listeners) {
boolean addJkListenerName = jkConnector != null && !virtualServer.getName().equalsIgnoreCase(ADMIN_VS);
List listenerNames = new ArrayList<>();
for (NetworkListener listener : listeners) {
if (Boolean.valueOf(listener.getEnabled())) {
listenerNames.add(listener.getName());
logger.fine(VIRTUAL_SERVER_SET_LISTENER_NAME);
} else {
if (virtualServer.getName().equalsIgnoreCase(ADMIN_VS)) {
throw new IllegalArgumentException(format(rb.getString(MUST_NOT_DISABLE), listener.getName(), virtualServer.getName()));
}
}
}
if (addJkListenerName && (!listenerNames.contains(jkConnector.getName()))) {
listenerNames.add(jkConnector.getName());
if (logger.isLoggable(FINE)) {
logger.log(FINE, VIRTUAL_SERVER_SET_JK_LISTENER_NAME, new Object[] { virtualServer.getID(), jkConnector.getName() });
}
}
virtualServer.setNetworkListenerNames(listenerNames.toArray(new String[listenerNames.size()]));
}
// ------------------------------------------------------ Public Methods
/**
* Create a virtual server/host.
*
* @param virtualServerId
* @param virtualServerBean
* @param docroot
* @param mimeMap
* @return
*/
public VirtualServer createHost(String virtualServerId, com.sun.enterprise.config.serverbeans.VirtualServer virtualServerBean, String docroot, MimeMap mimeMap) {
// Initialize the docroot
VirtualServer virtualServer = (VirtualServer) _embedded.createHost(virtualServerId, virtualServerBean, docroot, virtualServerBean.getLogFile(), mimeMap);
virtualServer.configureState();
virtualServer.configureRemoteAddressFilterValve();
virtualServer.configureRemoteHostFilterValve();
virtualServer.configureSingleSignOn(globalSSOEnabled, webContainerFeatureFactory, isSsoFailoverEnabled());
virtualServer.configureRedirect();
virtualServer.configureErrorPage();
virtualServer.configureErrorReportValve();
virtualServer.setServerContext(getServerContext());
virtualServer.setServerConfig(serverConfig);
virtualServer.setGrizzlyService(grizzlyService);
virtualServer.setWebContainer(this);
return virtualServer;
}
/**
* Gracefully terminate the active use of the public methods of this component. This method should be the last one
* called on a given instance of this component.
*
* @throws IllegalStateException if this component has not been started
* @throws LifecycleException if this component detects a fatal error that needs to be reported
*/
public void stop() throws LifecycleException {
// Validate and update our current component state
if (!_started) {
throw new LifecycleException(rb.getString(WEB_CONTAINER_NOT_STARTED));
}
_started = false;
// stop the embedded container
try {
_embedded.stop();
} catch (LifecycleException ex) {
if (!ex.getMessage().contains("has not been started")) {
throw ex;
}
}
}
// ------------------------------------------------------ Private Methods
/**
* Configures a default web module for each virtual server based on the virtual server's docroot if a virtual server
* does not specify any default-web-module, and none of its web modules are loaded at "/"
*
* Needed in postConstruct before Deployment.ALL_APPLICATIONS_PROCESSED for "jsp from docroot before web container
* start" scenario
*
* @see AdminConsoleAdapter -
*/
public void loadSystemDefaultWebModules() {
for (VirtualServer virtualServer : getVirtualServers()) {
/*
* Let AdminConsoleAdapter handle any requests for the root context of the '__asadmin' virtual-server, see
* https://glassfish.dev.java.net/issues/show_bug.cgi?id=5664
*/
if (ADMIN_VS.equals(virtualServer.getName())) {
continue;
}
// Create default web module off of virtual server's docroot if necessary
String defaultPath = null;
WebModuleConfig webModuleConfig = virtualServer.createSystemDefaultWebModuleIfNecessary(serviceLocator.getService(WebArchivist.class));
if (webModuleConfig != null) {
defaultPath = webModuleConfig.getContextPath();
loadStandaloneWebModule(virtualServer, webModuleConfig);
}
if (logger.isLoggable(INFO)) {
logger.log(INFO, VIRTUAL_SERVER_LOADED_DEFAULT_WEB_MODULE, new Object[] { virtualServer.getName(), defaultPath });
}
}
}
private List getVirtualServers() {
return stream(getEngine().findChildren())
.filter(e -> e instanceof VirtualServer)
.map(e -> (VirtualServer)e)
.collect(toList());
}
/**
* Configures a default web module for each virtual server if default-web-module is defined.
*/
public void loadDefaultWebModulesAfterAllAppsProcessed() {
for (Container container : getEngine().findChildren()) {
if (container instanceof VirtualServer) {
VirtualServer virtualServer = (VirtualServer) container;
/*
* Let AdminConsoleAdapter handle any requests for the root context of the '__asadmin' virtual-server, see
* https://glassfish.dev.java.net/issues/show_bug.cgi?id=5664
*/
if (ADMIN_VS.equals(virtualServer.getName())) {
continue;
}
WebModuleConfig webModuleConfig = virtualServer.getDefaultWebModule(domain, serviceLocator.getService(WebArchivist.class), appRegistry);
if (webModuleConfig != null) {
String defaultPath = webModuleConfig.getContextPath();
// Virtual server declares default-web-module
try {
updateDefaultWebModule(virtualServer, virtualServer.getNetworkListenerNames(), webModuleConfig);
} catch (LifecycleException le) {
logger.log(SEVERE, MessageFormat.format(rb.getString(DEFAULT_WEB_MODULE_ERROR), defaultPath, virtualServer.getName()), le);
}
if (logger.isLoggable(INFO)) {
logger.log(INFO, VIRTUAL_SERVER_LOADED_DEFAULT_WEB_MODULE, new Object[] { virtualServer.getName(), defaultPath });
}
}
// No need to create default web module off of virtual
// server's docroot since system web modules are already
// created in WebContainer.postConstruct
}
}
}
/**
* Load a default-web-module on the specified virtual server.
*
* @param vsBean
*/
public void loadDefaultWebModule(com.sun.enterprise.config.serverbeans.VirtualServer vsBean) {
VirtualServer virtualServer = (VirtualServer) getEngine().findChild(vsBean.getId());
if (virtualServer != null) {
loadDefaultWebModule(virtualServer);
}
}
/**
* Load a default-web-module on the specified virtual server.
*
* @param virtualServer
*/
public void loadDefaultWebModule(VirtualServer virtualServer) {
final String defaultPath;
WebModuleConfig wmInfo = virtualServer.getDefaultWebModule(domain, serviceLocator.getService(WebArchivist.class), appRegistry);
if (wmInfo != null) {
defaultPath = wmInfo.getContextPath();
// Virtual server declares default-web-module
try {
updateDefaultWebModule(virtualServer, virtualServer.getNetworkListenerNames(), wmInfo);
} catch (LifecycleException le) {
logger.log(SEVERE, format(rb.getString(DEFAULT_WEB_MODULE_ERROR), defaultPath, virtualServer.getName()), le);
}
} else {
// Create default web module off of virtual
// server's docroot if necessary
wmInfo = virtualServer.createSystemDefaultWebModuleIfNecessary(serviceLocator.getService(WebArchivist.class));
if (wmInfo == null) {
defaultPath = null;
} else {
defaultPath = wmInfo.getContextPath();
loadStandaloneWebModule(virtualServer, wmInfo);
}
}
if (logger.isLoggable(INFO)) {
logger.log(INFO, VIRTUAL_SERVER_LOADED_DEFAULT_WEB_MODULE, new Object[] { virtualServer.getName(), defaultPath });
}
}
/**
* Load the specified web module as a standalone module on the specified virtual server.
*
* @param virtualServer
* @param webModuleConfig
*/
protected void loadStandaloneWebModule(VirtualServer virtualServer, WebModuleConfig webModuleConfig) {
try {
loadWebModule(virtualServer, webModuleConfig, "null", null);
securityDeployer.loadPolicy(webModuleConfig.getDescriptor(), false);
} catch (Throwable t) {
logger.log(SEVERE, format(rb.getString(LOAD_WEB_MODULE_ERROR), webModuleConfig.getName()), t);
}
}
/**
* Whether or not a component (either an application or a module) should be enabled is defined by the "enable" attribute
* on both the application/module element and the application-ref element.
*
* @param moduleName The name of the component (application or module)
* @return boolean
*/
protected boolean isEnabled(String moduleName) {
return true;
}
/**
* Creates and configures a web module for each virtual server that the web module is hosted under.
*
* If no virtual servers have been specified, then the web module will not be loaded.
*
* @param webModuleConfig
* @param j2eeApplication
* @param deploymentProperties
* @return
*/
public List> loadWebModule(WebModuleConfig webModuleConfig, String j2eeApplication, Properties deploymentProperties) {
List> results = new ArrayList<>();
String virtualServerIds = webModuleConfig.getVirtualServers();
List virtualServers = parseStringList(virtualServerIds, " ,");
if (virtualServers == null || virtualServers.isEmpty()) {
if (logger.isLoggable(INFO)) {
logger.log(INFO, WEB_MODULE_NOT_LOADED_NO_VIRTUAL_SERVERS, webModuleConfig.getName());
}
return results;
}
logger.log(FINE, LOADING_WEB_MODULE, virtualServerIds);
List nonProcessedVirtualServers = new ArrayList<>(virtualServers);
Container[] containers = getEngine().findChildren();
List virtualServersToDeploy = new ArrayList<>(containers.length);
for (Container container : containers) {
if (container instanceof VirtualServer) {
VirtualServer virtualServer = (VirtualServer) container;
boolean eqVS = virtualServers.contains(virtualServer.getID());
if (eqVS) {
nonProcessedVirtualServers.remove(virtualServer.getID());
}
Set matchedAliases = matchAlias(virtualServers, virtualServer);
boolean hasMatchedAlias = matchedAliases.size() > 0;
if (hasMatchedAlias) {
nonProcessedVirtualServers.removeAll(matchedAliases);
}
if (eqVS || hasMatchedAlias) {
virtualServersToDeploy.add(virtualServer);
}
}
}
boolean moreThanOneVirtualServer = virtualServersToDeploy.size() > 1;
for (VirtualServer virtualServerToDeploy : virtualServersToDeploy) {
WebModule webModule = null;
ClassLoader appClassLoader = webModuleConfig.getAppClassLoader();
try {
if (moreThanOneVirtualServer) {
WebappClassLoader virtualServerClassLoader = new WebappClassLoader(
appClassLoader,
webModuleConfig.getDeploymentContext().getModuleMetaData(Application.class));
virtualServerClassLoader.start();
// For every virtual server, JSF and other extensions expect a separate class loader
webModuleConfig.setAppClassLoader(virtualServerClassLoader);
}
webModule = loadWebModule(virtualServerToDeploy, webModuleConfig, j2eeApplication, deploymentProperties);
results.add(new Result<>(webModule));
} catch (Throwable t) {
if (webModule != null) {
webModule.setAvailable(false);
}
results.add(new Result<>(t));
} finally {
if (moreThanOneVirtualServer) {
webModuleConfig.setAppClassLoader(appClassLoader);
}
}
}
if (nonProcessedVirtualServers.size() > 0) {
StringBuilder sb = new StringBuilder();
boolean follow = false;
for (String alias : nonProcessedVirtualServers) {
if (follow) {
sb.append(",");
}
sb.append(alias);
follow = true;
}
logger.log(SEVERE, WEB_MODULE_NOT_LOADED_TO_VS, new Object[]{ webModuleConfig.getName(), sb.toString() });
}
return results;
}
/**
* Deploy on aliases as well as host.
*/
private boolean verifyAlias(List vsList, VirtualServer vs) {
for (int i = 0; i < vs.getAliases().length; i++) {
if (vsList.contains(vs.getAliases()[i])) {
return true;
}
}
return false;
}
/**
* Find all matched aliases. This is more expensive than verifyAlias.
*/
private Set matchAlias(List vsList, VirtualServer vs) {
Set matched = new HashSet<>();
for (String alias : vs.getAliases()) {
if (vsList.contains(alias)) {
matched.add(alias);
}
}
return matched;
}
/**
* Creates and configures a web module and adds it to the specified virtual server.
*/
private WebModule loadWebModule(VirtualServer virtualServer, WebModuleConfig webModuleConfig, String j2eeApplication, Properties deploymentProperties) throws Exception {
DeploymentContext dc = webModuleConfig.getDeploymentContext();
String webModuleName = webModuleConfig.getName();
String webModuleContextPath = webModuleConfig.getContextPath();
if (webModuleContextPath.indexOf('%') != -1) {
try {
RequestUtil.urlDecode(webModuleContextPath, "UTF-8");
} catch (Exception e) {
throw new Exception(format(rb.getString(INVALID_ENCODED_CONTEXT_ROOT), webModuleName, webModuleContextPath));
}
}
if (webModuleContextPath.length() == 0 && virtualServer.getDefaultWebModuleID() != null) {
throw new Exception(format(rb.getString(DEFAULT_WEB_MODULE_CONFLICT), new Object[] { webModuleName, virtualServer.getID() }));
}
webModuleConfig.setWorkDirBase(_appsWorkRoot);
webModuleConfig.setStubBaseDir(appsStubRoot);
final String displayContextPath;
if (webModuleContextPath.isEmpty()) {
displayContextPath = "/";
} else {
displayContextPath = webModuleContextPath;
}
final File docBase;
if (JWS_APPCLIENT_MODULE_NAME.equals(webModuleName)) {
docBase = new File(System.getProperty("com.sun.aas.installRoot"));
} else {
docBase = webModuleConfig.getLocation();
}
Map adHocPaths = null;
Map adHocSubtrees = null;
WebModule webModule = (WebModule) virtualServer.findChild(webModuleContextPath);
if (webModule != null) {
Optional appState = hotDeployService.getApplicationState(dc);
if (webModule instanceof AdHocWebModule) {
/*
* Found ad-hoc web module which has been created by web container in order to store mappings for ad-hoc paths and
* subtrees. All these mappings must be propagated to the context that is being deployed.
*/
if (webModule.hasAdHocPaths()) {
adHocPaths = webModule.getAdHocPaths();
}
if (webModule.hasAdHocSubtrees()) {
adHocSubtrees = webModule.getAdHocSubtrees();
}
virtualServer.removeChild(webModule);
} else if (DEFAULT_WEB_MODULE_NAME.equals(webModule.getModuleName())) {
/*
* Dummy context that was created just off of a docroot, (see VirtualServer.createSystemDefaultWebModuleIfNecessary()).
* Unload it so it can be replaced with the web module to be loaded
*/
unloadWebModule(webModuleContextPath, webModule.getWebBundleDescriptor().getApplication().getRegistrationName(), virtualServer.getName(), true, null);
} else if (!webModule.getAvailable()) {
/*
* Context has been marked unavailable by a previous call to disableWebModule. Mark the context as available and return
*/
webModule.setAvailable(true);
return webModule;
} else if (appState.map(ApplicationState::isActive).orElse(false)) {
webModule.stop();
if (webModule.getWebModuleConfig() != webModuleConfig
|| webModule.getWebBundleDescriptor() != webModuleConfig.getDescriptor()) {
_embedded.updateContext(webModule,
virtualServer.getDefaultContextXmlLocation(),
virtualServer.getDefaultWebXmlLocation(),
useDOLforDeployment, webModuleConfig);
processWebBundleDescriptor(virtualServer, webModule, webModuleConfig, displayContextPath);
}
processWebAppClassLoader(webModule, webModuleConfig);
webModule.start();
return webModule;
} else {
throw new Exception(format(rb.getString(DUPLICATE_CONTEXT_ROOT), virtualServer.getID(), webModule.getModuleName(), displayContextPath, webModuleName));
}
}
if (logger.isLoggable(FINEST)) {
logger.log(FINEST, WEB_MODULE_LOADING, new Object[]{ webModuleName, virtualServer.getID(), displayContextPath });
}
webModule = (WebModule)
_embedded.createContext(
webModuleName, webModuleContextPath, docBase, virtualServer.getDefaultContextXmlLocation(), virtualServer.getDefaultWebXmlLocation(),
useDOLforDeployment, webModuleConfig);
// For now disable JNDI
webModule.setUseNaming(false);
// Set JSR 77 object name and attributes
Engine vsContainer = (Engine) virtualServer.getParent();
if (vsContainer != null) {
webModule.setEngineName(vsContainer.getName());
webModule.setJvmRoute(vsContainer.getJvmRoute());
}
webModule.setDomain(_serverContext.getDefaultDomainName());
webModule.setJ2EEServer(_serverContext.getInstanceName());
webModule.setJ2EEApplication(j2eeApplication);
webModule.setCacheControls(virtualServer.getCacheControls());
webModule.setBean(webModuleConfig.getBean());
if (adHocPaths != null) {
webModule.addAdHocPaths(adHocPaths);
}
if (adHocSubtrees != null) {
webModule.addAdHocSubtrees(adHocSubtrees);
}
final WebBundleDescriptor webBundleDescriptor = webModuleConfig.getDescriptor();
if (webBundleDescriptor == null) {
logger.log(WARNING, UNABLE_TO_SET_CONTEXT_ROOT, webModuleConfig);
} else {
webModule.setContextRoot(webBundleDescriptor.getContextRoot());
}
//
// Ensure that the generated directory for JSPs in the document root
// (i.e. those that are serviced by a system default-web-module)
// is different for each virtual server.
final String webModuleWorkDir = webModuleConfig.getWorkDir();
if (webModuleWorkDir != null) {
StringBuilder workDir = new StringBuilder(webModuleConfig.getWorkDir());
if (webModuleName.equals(DEFAULT_WEB_MODULE_NAME)) {
workDir.append("-");
workDir.append(makeFriendlyFilename(virtualServer.getID()));
}
webModule.setWorkDir(workDir.toString());
}
ClassLoader parentLoader = webModuleConfig.getParentLoader();
if (parentLoader == null) {
// Use the shared classloader as the parent for all
// standalone web-modules
parentLoader = _serverContext.getSharedClassLoader();
}
webModule.setParentClassLoader(parentLoader);
processWebBundleDescriptor(virtualServer, webModule, webModuleConfig, displayContextPath);
processWebAppClassLoader(webModule, webModuleConfig);
// set i18n info from locale-charset-info tag in sun-web.xml
webModule.setI18nInfo();
if (webBundleDescriptor != null) {
final boolean isSystem = webModuleConfig.isSystemObjectType();
final Realm realm = serviceLocator.getService(Realm.class);
logger.finest(() -> "Realm provided by the service locator: " + realm);
if (realm instanceof RealmInitializer) {
((RealmInitializer) realm).initializeRealm(webBundleDescriptor, isSystem,
virtualServer.getAuthRealmName());
}
webModule.setRealm(realm);
// post processing DOL object for standalone web module
if (webBundleDescriptor.getApplication() != null && webBundleDescriptor.getApplication().isVirtual()) {
webBundleDescriptor.visit(new WebValidatorWithoutCL());
}
}
// Add virtual server mime mappings, if present
addMimeMappings(webModule, virtualServer.getMimeMap());
String moduleName = Constants.DEFAULT_WEB_MODULE_NAME;
String monitoringNodeName = moduleName;
if (webBundleDescriptor != null && webBundleDescriptor.getApplication() != null) {
// Not a dummy web module
com.sun.enterprise.deployment.Application app = webBundleDescriptor.getApplication();
webModule.setStandalone(app.isVirtual());
if (app.isVirtual()) {
// Standalone web module
moduleName = app.getRegistrationName();
monitoringNodeName = webBundleDescriptor.getModuleID();
} else {
// Nested (inside EAR) web module
moduleName = webBundleDescriptor.getModuleDescriptor().getArchiveUri();
StringBuilder sb = new StringBuilder();
sb.append(app.getRegistrationName()).append(MONITORING_NODE_SEPARATOR).append(moduleName);
monitoringNodeName = sb.toString().replaceAll("\\.", "\\\\.").replaceAll("_war", "\\\\.war");
}
}
webModule.setModuleName(moduleName);
webModule.setMonitoringNodeName(monitoringNodeName);
webStatsProviderBootstrap.registerApplicationStatsProviders(monitoringNodeName, virtualServer.getName(), getServletNames(webBundleDescriptor));
virtualServer.addChild(webModule);
webModule.loadSessions(deploymentProperties);
return webModule;
}
private void processWebBundleDescriptor(VirtualServer virtualServer, WebModule webModule, WebModuleConfig webModuleConfig, String displayContextPath) {
WebBundleDescriptor webBundleDescriptor = webModuleConfig.getDescriptor();
if (webBundleDescriptor != null) {
// Determine if an alternate DD is set for this web-module in the application
webModule.configureAlternateDD(webBundleDescriptor);
webModule.configureWebServices(webBundleDescriptor);
}
// Object containing sun-web.xml information
// The default context is the only case when wbd == null
final SunWebAppImpl iasBean = webBundleDescriptor == null ? null
: (SunWebAppImpl) webBundleDescriptor.getSunDescriptor();
// set the sun-web config bean
webModule.setIasWebAppConfigBean(iasBean);
// Configure SingleThreadedServletPools, work/tmp directory etc
webModule.configureMiscSettings(virtualServer, displayContextPath);
// Configure alternate docroots if dummy web module
if (webModule.getID().startsWith(DEFAULT_WEB_MODULE_NAME)) {
webModule.setAlternateDocBases(virtualServer.getProperties());
}
// Configure the session manager and other related settings
webModule.configureSessionSettings(webBundleDescriptor, webModuleConfig);
}
private void processWebAppClassLoader(WebModule webModule, WebModuleConfig webModuleConfig) {
WebBundleDescriptor webBundleDescriptor = webModuleConfig.getDescriptor();
// Configure the class loader delegation model, classpath etc
Loader loader = webModule.configureLoader(webModule.getIasWebAppConfigBean());
// Set the class loader on the DOL object
if (webBundleDescriptor != null && webBundleDescriptor.hasWebServices()) {
webBundleDescriptor.addExtraAttribute("WEBLOADER", loader);
}
for (LifecycleListener listener : webModule.findLifecycleListeners()) {
if (listener instanceof ContextConfig) {
((ContextConfig) listener).setClassLoader(webModuleConfig.getAppClassLoader());
}
}
}
private List getServletNames(WebBundleDescriptor webBundleDescriptor) {
List servletNames = new ArrayList<>();
if (webBundleDescriptor != null) {
for (WebComponentDescriptor webComponentDescriptor : webBundleDescriptor.getWebComponentDescriptors()) {
if (webComponentDescriptor.isServlet()) {
servletNames.add(webComponentDescriptor.getCanonicalName());
}
}
}
return servletNames;
}
/**
* Updates the given virtual server with the given default path.
*
* The given default path corresponds to the context path of one of the web contexts deployed on the virtual server that
* has been designated as the virtual server's new default-web-module.
*
* @param virtualServer The virtual server to update
*
* @param ports The port numbers of the HTTP listeners with which the given virtual server is associated
*
* @param defaultContextPath The context path of the web module that has been designated as the virtual server's new
* default web module, or null if the virtual server no longer has any default-web-module
*/
protected void updateDefaultWebModule(VirtualServer virtualServer, String[] listenerNames,
WebModuleConfig webModuleConfig) throws LifecycleException {
String defaultContextPath = null;
if (webModuleConfig != null) {
defaultContextPath = webModuleConfig.getContextPath();
if (defaultContextPath != null && !defaultContextPath.startsWith("/")) {
defaultContextPath = "/" + defaultContextPath;
webModuleConfig.getDescriptor().setContextRoot(defaultContextPath);
}
}
Connector[] connectors = _embedded.findConnectors();
for (Connector connector : connectors) {
PECoyoteConnector coyoteConnector = (PECoyoteConnector) connector;
String name = coyoteConnector.getName();
for (String listenerName : listenerNames) {
if (name.equals(listenerName)) {
Mapper mapper = coyoteConnector.getMapper();
try {
mapper.setDefaultContextPath(virtualServer.getName(), defaultContextPath);
for (String alias : virtualServer.findAliases()) {
mapper.setDefaultContextPath(alias, defaultContextPath);
}
virtualServer.setDefaultContextPath(defaultContextPath);
} catch (Exception e) {
throw new LifecycleException(e);
}
}
}
}
}
/**
* Utility Method to access the ServerContext
*
* @return
*/
public ServerContext getServerContext() {
return _serverContext;
}
ServerConfigLookup getServerConfigLookup() {
return serverConfigLookup;
}
/**
* Returns the folder where library files are to be found
*
* @return
*/
File getLibPath() {
return instance.getLibPath();
}
/**
* The application id for this web module HERCULES:add
*
* @param webModule
* @return
*/
public String getApplicationId(WebModule webModule) {
return webModule.getID();
}
/**
* Return the Absolute path for location where all the deployed standalone modules are stored for this Server Instance.
*
* @return
*/
public File getModulesRoot() {
return _modulesRoot;
}
/**
* Undeploy a web application.
*
* @param contextRoot the context's name to undeploy
* @param appName the J2EE appname used at deployment time
* @param virtualServers List of current virtual-server object.
* @param props
*/
public void unloadWebModule(String contextRoot, String appName, String virtualServers, Properties props) {
unloadWebModule(contextRoot, appName, virtualServers, false, props);
}
/**
* Undeploy a web application.
*
* @param contextRoot the context's name to undeploy
* @param appName the J2EE appname used at deployment time
* @param virtualServers List of current virtual-server object.
* @param dummy true if the web module to be undeployed is a dummy web module, that is, a web module created off of a
* virtual server's docroot
* @param props
*/
public void unloadWebModule(String contextRoot, String appName, String virtualServers, boolean dummy, Properties props) {
if (logger.isLoggable(FINEST)) {
logger.log(FINEST, LOADING_WEB_MODULE, new Object[] { contextRoot, virtualServers });
}
// Tomcat contextRoot starts with "/"
if (contextRoot.length() != 0 && !contextRoot.startsWith("/")) {
contextRoot = "/" + contextRoot;
} else if ("/".equals(contextRoot)) {
// Make corresponding change as in WebModuleConfig.getContextPath()
contextRoot = "";
}
List hostList = StringUtils.parseStringList(virtualServers, " ,");
boolean unloadFromAll = hostList == null || hostList.isEmpty();
boolean hasBeenUndeployed = false;
VirtualServer host = null;
WebModule context = null;
Container[] hostArray = getEngine().findChildren();
for (Container aHostArray : hostArray) {
host = (VirtualServer) aHostArray;
if (unloadFromAll || hostList.contains(host.getName()) || verifyAlias(hostList, host)) {
context = (WebModule) host.findChild(contextRoot);
if (context != null && context.getWebBundleDescriptor().getApplication().getRegistrationName().equals(appName)) {
context.saveSessions(props);
host.removeChild(context);
webStatsProviderBootstrap.unregisterApplicationStatsProviders(context.getMonitoringNodeName(), host.getName());
try {
/*
* If the webapp is being undeployed as part of a domain shutdown, we don't want to destroy it, as that would remove any
* sessions persisted to file. Any active sessions need to survive the domain shutdown, so that they may be resumed
* after the domain has been restarted.
*/
if (!isShutdown) {
context.destroy();
}
} catch (Exception ex) {
String msg = rb.getString(LogFacade.EXCEPTION_DURING_DESTROY);
msg = MessageFormat.format(msg, contextRoot, host.getName());
logger.log(Level.WARNING, msg, ex);
}
if (logger.isLoggable(Level.FINEST)) {
logger.log(Level.FINEST, LogFacade.CONTEXT_UNDEPLOYED, new Object[] { contextRoot, host });
}
hasBeenUndeployed = true;
host.fireContainerEvent(Deployer.REMOVE_EVENT, context);
/*
* If the web module that has been unloaded contained any mappings for ad-hoc paths, those mappings must be preserved by
* registering an ad-hoc web module at the same context root
*/
if (context.hasAdHocPaths() || context.hasAdHocSubtrees()) {
WebModule wm = createAdHocWebModule(context.getID(), host, contextRoot, context.getJ2EEApplication());
wm.addAdHocPaths(context.getAdHocPaths());
wm.addAdHocSubtrees(context.getAdHocSubtrees());
}
// START GlassFish 141
if (!dummy && !isShutdown) {
WebModuleConfig wmInfo = host.createSystemDefaultWebModuleIfNecessary(serviceLocator.getService(WebArchivist.class));
if (wmInfo != null) {
loadStandaloneWebModule(host, wmInfo);
}
}
// END GlassFish 141
}
}
}
if (!hasBeenUndeployed) {
logger.log(Level.SEVERE, LogFacade.UNDEPLOY_ERROR, contextRoot);
}
}
/**
* Suspends the web application with the given appName that has been deployed at the given contextRoot on the given
* virtual servers.
*
* @param contextRoot the context root
* @param appName the J2EE appname used at deployment time
* @param hosts the list of virtual servers
*/
public boolean suspendWebModule(String contextRoot, String appName, String hosts) {
return suspendWebModule(contextRoot, appName, hosts, false);
}
/**
* Suspends the web application with the given appName that has been deployed at the given contextRoot on the given
* virtual servers.
*
* @param contextRoot the context root
* @param appName the J2EE appname used at deployment time
* @param hosts the list of virtual servers
* @param fromDisableStartEvent whether or not this invocation came from the disable_start event - used to determine
* a log level about failing to disable the web module due to an "acceptable" race
* condition (unloaded before we managed to reach this point - nothing to do)
*/
public boolean suspendWebModule(String contextRoot, String appName, String hosts, boolean fromDisableStartEvent) {
boolean hasBeenSuspended = false;
List hostList = StringUtils.parseStringList(hosts, " ,");
if (hostList == null || hostList.isEmpty()) {
return hasBeenSuspended;
}
// Tomcat contextRoot starts with "/"
if (contextRoot.length() != 0 && !contextRoot.startsWith("/")) {
contextRoot = "/" + contextRoot;
}
VirtualServer host = null;
Context context = null;
for (Container aHostArray : getEngine().findChildren()) {
host = (VirtualServer) aHostArray;
if (hostList.contains(host.getName()) || verifyAlias(hostList, host)) {
context = (Context) host.findChild(contextRoot);
if (context != null) {
context.setAvailable(false);
if (logger.isLoggable(Level.FINEST)) {
logger.log(Level.FINEST, LogFacade.CONTEXT_DISABLED, new Object[] { contextRoot, host });
}
hasBeenSuspended = true;
}
}
}
if (!hasBeenSuspended) {
if (fromDisableStartEvent) {
// Presumably already unloaded - don't log as warning
logger.log(FINE, "Unable to disable web module at context root {0}, it may have already been unloaded",
contextRoot);
} else {
// Original log message
logger.log(WARNING, DISABLE_WEB_MODULE_ERROR, contextRoot);
}
}
return hasBeenSuspended;
}
/**
* Save the server-wide dynamic reloading settings for use when configuring each web module.
*/
private void configureDynamicReloadingSettings() {
if (dasConfig != null) {
// TODO: dead code removed, but this still can have side effects.
dasConfig.getDynamicReloadPollIntervalInSeconds();
}
}
/**
* Sets the debug level for Catalina's containers based on the logger's log level.
*/
private void setDebugLevel() {
Level logLevel = logger.getLevel() != null ? logger.getLevel() : Level.INFO;
if (logLevel.equals(Level.FINE)) {
_debug = 1;
} else if (logLevel.equals(Level.FINER)) {
_debug = 2;
} else if (logLevel.equals(Level.FINEST)) {
_debug = 5;
} else {
_debug = 0;
}
}
/**
* Get the lifecycle listeners associated with this lifecycle. If this Lifecycle has no listeners registered, a
* zero-length array is returned.
*
* @return
*/
public LifecycleListener[] findLifecycleListeners() {
return new LifecycleListener[0];
}
/**
* Gets all the virtual servers whose http-listeners attribute value contains the given http-listener id.
*/
List getVirtualServersForHttpListenerId(HttpService httpService, String httpListenerId) {
if (httpListenerId == null) {
return null;
}
List result = new ArrayList<>();
for (com.sun.enterprise.config.serverbeans.VirtualServer vs : httpService.getVirtualServer()) {
List listeners = StringUtils.parseStringList(vs.getNetworkListeners(), ",");
if (listeners != null) {
ListIterator iter = listeners.listIterator();
while (iter.hasNext()) {
if (httpListenerId.equals(iter.next())) {
VirtualServer match = (VirtualServer) getEngine().findChild(vs.getId());
if (match != null) {
result.add(match);
}
break;
}
}
}
}
return result;
}
/**
* Adds the given mime mappings to those of the specified context, unless they're already present in the context (that
* is, the mime mappings of the specified context, which correspond to those in default-web.xml, can't be overridden).
*
* @param ctx The StandardContext to whose mime mappings to add
* @param mimeMap The mime mappings to be added
*/
private void addMimeMappings(StandardContext ctx, MimeMap mimeMap) {
if (mimeMap == null) {
return;
}
for (Iterator itr = mimeMap.getExtensions(); itr.hasNext();) {
String extension = itr.next();
if (ctx.findMimeMapping(extension) == null) {
ctx.addMimeMapping(extension, mimeMap.getType(extension));
}
}
}
/**
* Return the parent/top-level container in _embedded for virtual servers.
*
* @return
*/
public Engine getEngine() {
return _embedded.getEngines()[0];
}
public HttpService getHttpService() {
return serverConfig.getHttpService();
}
/**
* Registers the given ad-hoc path at the given context root.
*
* @param path The ad-hoc path to register
* @param ctxtRoot The context root at which to register
* @param appName The name of the application with which the ad-hoc path is associated
* @param servletInfo Info about the ad-hoc servlet that will service requests on the given path
*/
public void registerAdHocPath(String path, String ctxtRoot, String appName, AdHocServletInfo servletInfo) {
registerAdHocPathAndSubtree(path, null, ctxtRoot, appName, servletInfo);
}
/**
* Registers the given ad-hoc path and subtree at the given context root.
*
* @param path The ad-hoc path to register
* @param subtree The ad-hoc subtree path to register
* @param ctxtRoot The context root at which to register
* @param appName The name of the application with which the ad-hoc path and subtree are associated
* @param servletInfo Info about the ad-hoc servlet that will service requests on the given ad-hoc path and subtree
*/
public void registerAdHocPathAndSubtree(String path, String subtree, String ctxtRoot, String appName, AdHocServletInfo servletInfo) {
for (Container container : getEngine().findChildren()) {
VirtualServer virtualServer = (VirtualServer) container;
if (virtualServer.getName().equalsIgnoreCase(ADMIN_VS)) {
// Do not deploy on admin vs
continue;
}
WebModule webModule = (WebModule) virtualServer.findChild(ctxtRoot);
if (webModule == null) {
webModule = createAdHocWebModule(virtualServer, ctxtRoot, appName);
}
webModule.addAdHocPathAndSubtree(path, subtree, servletInfo);
}
}
/**
* Unregisters the given ad-hoc path from the given context root.
*
* @param path The ad-hoc path to unregister
* @param ctxtRoot The context root from which to unregister
*/
public void unregisterAdHocPath(String path, String ctxtRoot) {
unregisterAdHocPathAndSubtree(path, null, ctxtRoot);
}
/**
* Unregisters the given ad-hoc path and subtree from the given context root.
*
* @param path The ad-hoc path to unregister
* @param subtree The ad-hoc subtree to unregister
* @param ctxtRoot The context root from which to unregister
*/
public void unregisterAdHocPathAndSubtree(String path, String subtree, String ctxtRoot) {
for (Container container : getEngine().findChildren()) {
VirtualServer virtualServer = (VirtualServer) container;
if (virtualServer.getName().equalsIgnoreCase(ADMIN_VS)) {
// Do not undeploy from admin vs, because we never deployed onto it
continue;
}
WebModule webModule = (WebModule) virtualServer.findChild(ctxtRoot);
if (webModule == null) {
continue;
}
/*
* If the web module was created by the container for the sole purpose of mapping ad-hoc paths and subtrees, and does no
* longer contain any ad-hoc paths or subtrees, remove the web module.
*/
webModule.removeAdHocPath(path);
webModule.removeAdHocSubtree(subtree);
if (webModule instanceof AdHocWebModule && !webModule.hasAdHocPaths() && !webModule.hasAdHocSubtrees()) {
virtualServer.removeChild(webModule);
try {
webModule.destroy();
} catch (Exception ex) {
logger.log(WARNING, format(rb.getString(EXCEPTION_DURING_DESTROY), webModule.getPath(), virtualServer.getName()), ex);
}
}
}
}
/*
* Creates an ad-hoc web module and registers it on the given virtual server at the given context root.
*
* @param vs The virtual server on which to add the ad-hoc web module
* @param ctxtRoot The context root at which to register the ad-hoc web module
* @param appName The name of the application to which the ad-hoc module being generated belongs
*
* @return The newly created ad-hoc web module
*/
private WebModule createAdHocWebModule(VirtualServer vs, String ctxtRoot, String appName) {
return createAdHocWebModule(appName, vs, ctxtRoot, appName);
}
/*
* Creates an ad-hoc web module and registers it on the given virtual server at the given context root.
*
* @param id the id of the ad-hoc web module
* @param vs The virtual server on which to add the ad-hoc web module
* @param ctxtRoot The context root at which to register the ad-hoc web module
* @param appName The name of the application to which the ad-hoc module being generated belongs
*
* @return The newly created ad-hoc web module
*/
private WebModule createAdHocWebModule(String id, VirtualServer vs, String ctxtRoot, String j2eeApplication) {
AdHocWebModule adHocWebModule = new AdHocWebModule();
adHocWebModule.setID(id);
adHocWebModule.setWebContainer(this);
adHocWebModule.restrictedSetPipeline(new WebPipeline(adHocWebModule));
// The Parent ClassLoader of the AdhocWebModule was null
// [System ClassLoader]. With the new hierarchy, the thread context
// classloader needs to be set.
adHocWebModule.setParentClassLoader(Thread.currentThread().getContextClassLoader());
adHocWebModule.setContextRoot(ctxtRoot);
adHocWebModule.setJ2EEApplication(j2eeApplication);
adHocWebModule.setName(ctxtRoot);
adHocWebModule.setDocBase(vs.getAppBase());
adHocWebModule.setEngineName(vs.getParent().getName());
String domain = _serverContext.getDefaultDomainName();
adHocWebModule.setDomain(domain);
String j2eeServer = _serverContext.getInstanceName();
adHocWebModule.setJ2EEServer(j2eeServer);
adHocWebModule.setCrossContext(true);
vs.addChild(adHocWebModule);
return adHocWebModule;
}
/**
* Removes the dummy module (the module created off of a virtual server's docroot) from the given virtual server if such
* a module exists.
*
* @param vs The virtual server whose dummy module is to be removed
*/
void removeDummyModule(VirtualServer vs) {
WebModule ctx = (WebModule) vs.findChild("");
if (ctx != null && Constants.DEFAULT_WEB_MODULE_NAME.equals(ctx.getModuleName())) {
unloadWebModule("", ctx.getWebBundleDescriptor().getApplication().getRegistrationName(), vs.getName(), true, null);
}
}
/**
* Initializes the instance-level session properties (read from config.web-container.session-config.session-properties
* in domain.xml).
*/
private void initInstanceSessionProperties() {
SessionProperties spBean = serverConfigLookup.getInstanceSessionProperties();
if (spBean == null || spBean.getProperty() == null) {
return;
}
List props = spBean.getProperty();
if (props == null) {
return;
}
for (Property prop : props) {
String propName = prop.getName();
String propValue = prop.getValue();
if (propName == null || propValue == null) {
throw new IllegalArgumentException(rb.getString(LogFacade.NULL_WEB_PROPERTY));
}
if (propName.equalsIgnoreCase("enableCookies")) {
instanceEnableCookies = ConfigBeansUtilities.toBoolean(propValue);
} else if (logger.isLoggable(Level.INFO)) {
logger.log(Level.INFO, LogFacade.PROPERTY_NOT_YET_SUPPORTED, propName);
}
}
}
private static synchronized void setJspFactory() {
if (JspFactory.getDefaultFactory() == null) {
JspFactory.setDefaultFactory(new JspFactoryImpl());
}
}
/**
* Delete virtual-server.
*
* @param httpService element which contains the configuration info.
* @throws org.apache.catalina.LifecycleException
*/
public void deleteHost(HttpService httpService) throws LifecycleException {
VirtualServer virtualServer;
// First we need to find which virtual-server was deleted. In
// reconfig/VirtualServerReconfig, it is impossible to lookup
// the vsBean because the element is removed from domain.xml
// before handleDelete is invoked.
Container[] virtualServers = getEngine().findChildren();
for (int i = 0; i < virtualServers.length; i++) {
for (com.sun.enterprise.config.serverbeans.VirtualServer vse : httpService.getVirtualServer()) {
if (virtualServers[i].getName().equals(vse.getId())) {
virtualServers[i] = null;
break;
}
}
}
for (Container virtualServer1 : virtualServers) {
virtualServer = (VirtualServer) virtualServer1;
if (virtualServer != null) {
if (virtualServer.getID().equals(org.glassfish.api.web.Constants.ADMIN_VS)) {
throw new LifecycleException("Cannot delete admin virtual-server.");
}
Container[] webModules = virtualServer.findChildren();
for (Container webModule : webModules) {
String appName = webModule.getName();
if (webModule instanceof WebModule) {
appName = ((WebModule) webModule).getWebBundleDescriptor().getApplication().getRegistrationName();
}
unloadWebModule(webModule.getName(), appName, virtualServer.getID(), null);
}
try {
virtualServer.destroy();
} catch (Exception e) {
String msg = rb.getString(LogFacade.DESTROY_VS_ERROR);
msg = MessageFormat.format(msg, virtualServer.getID());
logger.log(Level.WARNING, msg, e);
}
}
}
}
/**
* Updates a virtual-server element.
*
* @param vsBean the virtual-server config bean.
*/
public void updateHost(com.sun.enterprise.config.serverbeans.VirtualServer vsBean) throws LifecycleException {
if (ADMIN_VS.equals(vsBean.getId())) {
return;
}
VirtualServer virtualServer = (VirtualServer) getEngine().findChild(vsBean.getId());
if (virtualServer == null) {
logger.log(WARNING, CANNOT_UPDATE_NON_EXISTENCE_VS, vsBean.getId());
return;
}
boolean updateListeners = false;
// Only update connectors if virtual-server.http-listeners is changed dynamically
if (virtualServer.getNetworkListeners() == null) {
if (vsBean.getNetworkListeners() == null) {
updateListeners = false;
} else {
updateListeners = true;
}
} else if (virtualServer.getNetworkListeners().equals(vsBean.getNetworkListeners())) {
updateListeners = false;
} else {
List vsList = StringUtils.parseStringList(virtualServer.getNetworkListeners(), ",");
List vsBeanList = StringUtils.parseStringList(vsBean.getNetworkListeners(), ",");
for (String vsBeanName : vsBeanList) {
if (!vsList.contains(vsBeanName)) {
updateListeners = true;
if (logger.isLoggable(Level.FINE)) {
logger.log(Level.FINE, LogFacade.UPDATE_LISTENER, new Object[] { vsBeanName, virtualServer.getNetworkListeners() });
}
break;
}
}
}
// Must retrieve the old default-web-module before updating the
// virtual server with the new vsBean, because default-web-module is
// read from vsBean
String oldDefaultWebModule = virtualServer.getDefaultWebModuleID();
virtualServer.setBean(vsBean);
String vsLogFile = vsBean.getLogFile();
virtualServer.setLogFile(vsLogFile, logLevel, logServiceFile);
virtualServer.configureState();
virtualServer.clearAliases();
virtualServer.configureAliases();
// support both docroot property and attribute
String docroot = vsBean.getPropertyValue("docroot");
if (docroot == null) {
docroot = vsBean.getDocroot();
}
if (docroot != null) {
// Only update docroot if it is modified
if (!virtualServer.getDocRoot().getAbsolutePath().equals(docroot)) {
updateDocroot(docroot, virtualServer, vsBean);
}
}
List props = virtualServer.getProperties();
for (Property prop : props) {
updateHostProperties(vsBean, prop.getName(), prop.getValue(), securityService, virtualServer);
}
virtualServer.configureSingleSignOn(globalSSOEnabled, webContainerFeatureFactory, isSsoFailoverEnabled());
virtualServer.reconfigureAccessLog(globalAccessLogBufferSize, globalAccessLogWriteInterval, serviceLocator, domain, globalAccessLoggingEnabled, globalAccessLogPrefix);
// old listener names
List oldListenerList = StringUtils.parseStringList(vsBean.getNetworkListeners(), ",");
String[] oldListeners = (oldListenerList != null) ? oldListenerList.toArray(new String[oldListenerList.size()]) : new String[0];
// new listener config
HashSet networkListeners = new HashSet<>();
if (oldListenerList != null) {
for (String listener : oldListeners) {
boolean found = false;
for (NetworkListener httpListener : serverConfig.getNetworkConfig().getNetworkListeners().getNetworkListener()) {
if (httpListener.getName().equals(listener)) {
networkListeners.add(httpListener);
found = true;
break;
}
}
if (!found) {
String msg = rb.getString(LogFacade.LISTENER_REFERENCED_BY_HOST_NOT_EXIST);
msg = MessageFormat.format(msg, listener, virtualServer.getName());
logger.log(Level.SEVERE, msg);
}
}
// Update the port numbers with which the virtual server is
// associated
configureHostPortNumbers(virtualServer, networkListeners);
} else {
// The virtual server is not associated with any http listeners
virtualServer.setNetworkListenerNames(new String[0]);
}
// Disassociate the virtual server from all http listeners that
// have been removed from its http-listeners attribute
for (String oldListener : oldListeners) {
boolean found = false;
for (NetworkListener httpListener : networkListeners) {
if (httpListener.getName().equals(oldListener)) {
found = true;
}
}
if (!found) {
// http listener was removed
Connector[] connectors = _embedded.findConnectors();
for (Connector connector : connectors) {
WebConnector conn = (WebConnector) connector;
if (oldListener.equals(conn.getName())) {
try {
conn.getMapperListener().unregisterHost(virtualServer.getJmxName());
} catch (Exception e) {
throw new LifecycleException(e);
}
}
}
}
}
// Associate the virtual server with all http listeners that
// have been added to its http-listeners attribute
for (NetworkListener httpListener : networkListeners) {
boolean found = false;
for (String oldListener : oldListeners) {
if (httpListener.getName().equals(oldListener)) {
found = true;
}
}
if (!found) {
// http listener was added
Connector[] connectors = _embedded.findConnectors();
for (Connector connector : connectors) {
WebConnector conn = (WebConnector) connector;
if (httpListener.getName().equals(conn.getName())) {
if (!conn.isAvailable()) {
conn.start();
}
try {
conn.getMapperListener().registerHost(virtualServer);
} catch (Exception e) {
throw new LifecycleException(e);
}
}
}
}
}
// Remove the old default web module if one was configured, by
// passing in "null" as the default context path
if (oldDefaultWebModule != null) {
updateDefaultWebModule(virtualServer, oldListeners, null);
}
/*
* Add default web module if one has been configured for the virtual server. If the module declared as the default web
* module has already been deployed at the root context, we don't have to do anything.
*/
WebModuleConfig webModuleConfig = virtualServer.getDefaultWebModule(domain, serviceLocator.getService(WebArchivist.class), appRegistry);
if ((webModuleConfig != null) && (webModuleConfig.getContextPath() != null) && !"".equals(webModuleConfig.getContextPath()) && !"/".equals(webModuleConfig.getContextPath())) {
// Remove dummy context that was created off of docroot, if such
// a context exists
removeDummyModule(virtualServer);
updateDefaultWebModule(virtualServer, virtualServer.getNetworkListenerNames(), webModuleConfig);
} else {
WebModuleConfig wmc = virtualServer.createSystemDefaultWebModuleIfNecessary(serviceLocator.getService(WebArchivist.class));
if (wmc != null) {
loadStandaloneWebModule(virtualServer, wmc);
}
}
if (updateListeners) {
if (logger.isLoggable(FINE)) {
logger.log(FINE, VS_UPDATED_NETWORK_LISTENERS,
new Object[] { virtualServer.getName(), virtualServer.getNetworkListeners(), vsBean.getNetworkListeners() });
}
/*
* Need to update connector and mapper restart is required when virtual-server.http-listeners is changed dynamically
*/
List httpListeners = serverConfig.getNetworkConfig().getNetworkListeners().getNetworkListener();
if (httpListeners != null) {
for (NetworkListener httpListener : httpListeners) {
updateConnector(httpListener, serviceLocator.getService(HttpService.class));
}
}
}
}
/**
* Update virtual-server properties.
*
* @param vsBean
* @param name
* @param value
* @param virtualServer
* @param securityService
*/
public void updateHostProperties(com.sun.enterprise.config.serverbeans.VirtualServer vsBean, String name, String value, SecurityService securityService,
VirtualServer virtualServer) {
if (virtualServer == null) {
return;
}
virtualServer.setBean(vsBean);
if (name == null) {
return;
}
if (name.startsWith("alternatedocroot_")) {
updateAlternateDocroot(virtualServer);
} else if ("setCacheControl".equals(name)) {
virtualServer.configureCacheControl(value);
} else if (ACCESS_LOGGING_ENABLED.equals(name)) {
virtualServer.reconfigureAccessLog(globalAccessLogBufferSize, globalAccessLogWriteInterval, serviceLocator, domain, globalAccessLoggingEnabled, globalAccessLogPrefix);
} else if (ACCESS_LOG_PROPERTY.equals(name)) {
virtualServer.reconfigureAccessLog(globalAccessLogBufferSize, globalAccessLogWriteInterval, serviceLocator, domain, globalAccessLoggingEnabled, globalAccessLogPrefix);
} else if (ACCESS_LOG_WRITE_INTERVAL_PROPERTY.equals(name)) {
virtualServer.reconfigureAccessLog(globalAccessLogBufferSize, globalAccessLogWriteInterval, serviceLocator, domain, globalAccessLoggingEnabled, globalAccessLogPrefix);
} else if (ACCESS_LOG_BUFFER_SIZE_PROPERTY.equals(name)) {
virtualServer.reconfigureAccessLog(globalAccessLogBufferSize, globalAccessLogWriteInterval, serviceLocator, domain, globalAccessLoggingEnabled, globalAccessLogPrefix);
} else if (ACCESS_LOG_PREFIX.equals(name)) {
virtualServer.reconfigureAccessLog(globalAccessLogBufferSize, globalAccessLogWriteInterval, serviceLocator, domain, globalAccessLoggingEnabled, globalAccessLogPrefix);
} else if ("allowRemoteHost".equals(name) || "denyRemoteHost".equals(name)) {
virtualServer.configureRemoteHostFilterValve();
} else if ("allowRemoteAddress".equals(name) || "denyRemoteAddress".equals(name)) {
virtualServer.configureRemoteAddressFilterValve();
} else if (SSO_ENABLED.equals(name)) {
virtualServer.configureSingleSignOn(globalSSOEnabled, webContainerFeatureFactory, isSsoFailoverEnabled());
} else if ("authRealm".equals(name)) {
virtualServer.configureAuthRealm(securityService);
} else if (name.startsWith("send-error")) {
virtualServer.configureErrorPage();
} else if (ERROR_REPORT_VALVE.equals(name)) {
virtualServer.setErrorReportValveClass(value);
} else if (name.startsWith("redirect")) {
virtualServer.configureRedirect();
} else if (name.startsWith("contextXmlDefault")) {
virtualServer.setDefaultContextXmlLocation(value);
}
}
private boolean isSsoFailoverEnabled() {
boolean webContainerAvailabilityEnabled = serverConfigLookup.calculateWebAvailabilityEnabledFromConfig();
boolean isSsoFailoverEnabled = serverConfigLookup.isSsoFailoverEnabledFromConfig();
return isSsoFailoverEnabled && webContainerAvailabilityEnabled;
}
/**
* Processes an update to the http-service element
*
* @param httpService
* @throws org.apache.catalina.LifecycleException
*/
public void updateHttpService(HttpService httpService) throws LifecycleException {
if (httpService == null) {
return;
}
/*
* Update each virtual server with the sso-enabled and access logging related properties of the updated http-service
*/
globalSSOEnabled = ConfigBeansUtilities.toBoolean(httpService.getSsoEnabled());
globalAccessLogWriteInterval = httpService.getAccessLog().getWriteIntervalSeconds();
globalAccessLogBufferSize = httpService.getAccessLog().getBufferSizeBytes();
globalAccessLoggingEnabled = ConfigBeansUtilities.toBoolean(httpService.getAccessLoggingEnabled());
globalAccessLogPrefix = httpService.getAccessLog().getPropertyValue(Constants.ACCESS_LOG_PREFIX);
// for availability-service.web-container-availability
webContainerFeatureFactory = getWebContainerFeatureFactory();
for (com.sun.enterprise.config.serverbeans.VirtualServer virtualServer : httpService.getVirtualServer()) {
VirtualServer vs = (VirtualServer) getEngine().findChild(virtualServer.getId());
if (vs != null) {
vs.configureSingleSignOn(globalSSOEnabled, webContainerFeatureFactory, isSsoFailoverEnabled());
vs.reconfigureAccessLog(globalAccessLogBufferSize, globalAccessLogWriteInterval, serviceLocator, domain, globalAccessLoggingEnabled, globalAccessLogPrefix);
updateHost(virtualServer);
}
}
}
/**
* Update an http-listener property
*
* @param listener the configuration bean.
* @param propName the property name
* @param propValue the property value
* @throws org.apache.catalina.LifecycleException
*/
public void updateConnectorProperty(NetworkListener listener, String propName, String propValue) throws LifecycleException {
WebConnector connector = connectorMap.get(listener.getName());
if (connector != null) {
connector.configHttpProperties(listener.findHttpProtocol().getHttp(), listener.findTransport(), listener.findHttpProtocol().getSsl());
connector.configureHttpListenerProperty(propName, propValue);
}
}
/**
* Update an network-listener
*
* @param networkListener
* @param httpService the configuration bean.
* @throws org.apache.catalina.LifecycleException
*/
public void updateConnector(NetworkListener networkListener, HttpService httpService) throws LifecycleException {
synchronized (mapperUpdateSync) {
// Disable dynamic reconfiguration of the http listener at which
// the admin related webapps (including the admingui) are accessible.
// Notice that in GlassFish v3, we support a domain.xml configuration
// that does not declare any admin-listener, in which case the
// admin-related webapps are accessible on http-listener-1.
if (networkListener.findHttpProtocol().getHttp().getDefaultVirtualServer().equals(org.glassfish.api.web.Constants.ADMIN_VS)
|| "http-listener-1".equals(networkListener.getName()) && connectorMap.get("admin-listener") == null) {
return;
}
WebConnector connector = connectorMap.get(networkListener.getName());
if (connector != null) {
deleteConnector(connector);
}
if (!Boolean.valueOf(networkListener.getEnabled())) {
return;
}
connector = addConnector(networkListener, httpService, false);
// Update the list of listener names of all associated virtual servers with
// the listener's new listener name , so that the associated virtual
// servers will be registered with the listener's request mapper when
// the listener is started
List virtualServers = getVirtualServersForHttpListenerId(httpService, networkListener.getName());
if (virtualServers != null) {
for (VirtualServer vs : virtualServers) {
boolean found = false;
String[] listenerNames = vs.getNetworkListenerNames();
String name = connector.getName();
for (String listenerName : listenerNames) {
if (listenerName.equals(name)) {
found = true;
break;
}
}
if (!found) {
String[] newListenerNames = new String[listenerNames.length + 1];
System.arraycopy(listenerNames, 0, newListenerNames, 0, listenerNames.length);
newListenerNames[listenerNames.length] = connector.getName();
vs.setNetworkListenerNames(newListenerNames);
}
}
}
connector.start();
// GLASSFISH-20932
// Check if virtual server has default-web-module configured,
// and if so, configure the http listener's mapper with this
// information
if (virtualServers != null) {
Mapper mapper = connector.getMapper();
for (VirtualServer vs : virtualServers) {
String defaultWebModulePath = vs.getDefaultContextPath(domain, appRegistry);
if (defaultWebModulePath != null) {
try {
mapper.setDefaultContextPath(vs.getName(), defaultWebModulePath);
vs.setDefaultContextPath(defaultWebModulePath);
} catch (Exception e) {
throw new LifecycleException(e);
}
}
}
}
}
}
/**
* Method gets called, when GrizzlyService changes HTTP Mapper, associated with specific port.
*
* @param httpService {@link HttpService}
* @param httpListener {@link NetworkListener}, which {@link Mapper} was changed
* @param mapper new {@link Mapper} value
*/
public void updateMapper(HttpService httpService, NetworkListener httpListener, Mapper mapper) {
synchronized (mapperUpdateSync) {
WebConnector connector = connectorMap.get(httpListener.getName());
if (connector != null && connector.getMapper() != mapper) {
try {
updateConnector(httpListener, httpService);
} catch (LifecycleException le) {
logger.log(Level.SEVERE, LogFacade.EXCEPTION_CONFIG_HTTP_SERVICE, le);
}
}
}
}
public WebConnector addConnector(NetworkListener httpListener, HttpService httpService, boolean start) throws LifecycleException {
synchronized (mapperUpdateSync) {
int port = grizzlyService.getRealPort(httpListener);
// Add the listener name of the new http-listener to its
// default-virtual-server, so that when the new http-listener
// and its MapperListener are started, they will recognize the
// default-virtual-server as one of their own, and add it to the
// Mapper
String virtualServerName = httpListener.findHttpProtocol().getHttp().getDefaultVirtualServer();
VirtualServer vs = (VirtualServer) getEngine().findChild(virtualServerName);
List list = Arrays.asList(vs.getNetworkListenerNames());
// Avoid adding duplicate network-listener name
if (!list.contains(httpListener.getName())) {
String[] oldListenerNames = vs.getNetworkListenerNames();
String[] newListenerNames = new String[oldListenerNames.length + 1];
System.arraycopy(oldListenerNames, 0, newListenerNames, 0, oldListenerNames.length);
newListenerNames[oldListenerNames.length] = httpListener.getName();
vs.setNetworkListenerNames(newListenerNames);
}
Mapper mapper = null;
for (Mapper m : serviceLocator.getAllServices(Mapper.class)) {
if (m.getPort() == port && m instanceof ContextMapper) {
ContextMapper cm = (ContextMapper) m;
if (httpListener.getName().equals(cm.getId())) {
mapper = m;
break;
}
}
}
WebConnector connector = createHttpListener(httpListener, httpService, mapper);
if (connector.getRedirectPort() == -1) {
connector.setRedirectPort(defaultRedirectPort);
}
if (start) {
connector.start();
}
return connector;
}
}
/**
* Stops and deletes the specified http listener.
*
* @param connector
* @throws org.apache.catalina.LifecycleException
*/
public void deleteConnector(WebConnector connector) throws LifecycleException {
String name = connector.getName();
Connector[] connectors = _embedded.findConnectors();
for (Connector conn : connectors) {
if (name.equals(conn.getName())) {
_embedded.removeConnector(conn);
connectorMap.remove(connector.getName());
}
}
}
/**
* Stops and deletes the specified http listener.
*
* @param httpListener
* @throws org.apache.catalina.LifecycleException
*/
public void deleteConnector(NetworkListener httpListener) throws LifecycleException {
Connector[] connectors = _embedded.findConnectors();
String name = httpListener.getName();
for (Connector conn : connectors) {
if (name.equals(conn.getName())) {
_embedded.removeConnector(conn);
connectorMap.remove(name);
}
}
}
/**
* Reconfigures the access log valve of each virtual server with the updated attributes of the element from
* domain.xml.
*
* @param httpService
*/
public void updateAccessLog(HttpService httpService) {
Container[] virtualServers = getEngine().findChildren();
for (Container virtualServer : virtualServers) {
((VirtualServer) virtualServer).reconfigureAccessLog(httpService, webContainerFeatureFactory);
}
}
/**
* Updates the docroot of the given virtual server
*/
private void updateDocroot(String docroot, VirtualServer vs, com.sun.enterprise.config.serverbeans.VirtualServer vsBean) {
validateDocroot(docroot, vsBean.getId(), vsBean.getDefaultWebModule());
vs.setAppBase(docroot);
removeDummyModule(vs);
WebModuleConfig wmInfo = vs.createSystemDefaultWebModuleIfNecessary(serviceLocator.getService(WebArchivist.class));
if (wmInfo != null) {
loadStandaloneWebModule(vs, wmInfo);
}
}
private void updateAlternateDocroot(VirtualServer vs) {
removeDummyModule(vs);
WebModuleConfig wmInfo = vs.createSystemDefaultWebModuleIfNecessary(serviceLocator.getService(WebArchivist.class));
if (wmInfo != null) {
loadStandaloneWebModule(vs, wmInfo);
}
}
public void updateJvmRoute(HttpService httpService, String jvmOption) {
String jvmRoute = null;
if (jvmOption.contains("{") && jvmOption.contains("}")) {
// Look up system-property
jvmOption = jvmOption.substring(jvmOption.indexOf('{') + 1, jvmOption.indexOf('}'));
jvmRoute = server.getSystemPropertyValue(jvmOption);
if (jvmRoute == null) {
// Try to get it from System property if it exists
jvmRoute = System.getProperty(jvmOption);
}
} else if (jvmOption.contains("=")) {
jvmRoute = jvmOption.substring(jvmOption.indexOf('=') + 1);
}
engine.setJvmRoute(jvmRoute);
for (com.sun.enterprise.config.serverbeans.VirtualServer vsBean : httpService.getVirtualServer()) {
VirtualServer vs = (VirtualServer) engine.findChild(vsBean.getId());
for (Container context : vs.findChildren()) {
if (context instanceof StandardContext) {
((StandardContext) context).setJvmRoute(jvmRoute);
}
}
}
for (Connector connector : _embedded.getConnectors()) {
connector.setJvmRoute(jvmRoute);
}
logger.log(Level.FINE, LogFacade.JVM_ROUTE_UPDATED, jvmRoute);
}
/**
* is Tomcat using default domain name as its domain
*/
protected boolean isTomcatUsingDefaultDomain() {
// need to be careful and make sure tomcat jmx mapping works
// since setting this to true might result in undeployment problems
return true;
}
/**
* Creates probe providers for Servlet, JSP, Session, and Request/Response related events.
*
* While the Servlet, JSP, and Session related probe providers are shared by all web applications (where every web
* application qualifies its probe events with its application name), the Request/Response related probe provider is
* shared by all HTTP listeners.
*/
private void createProbeProviders() {
webModuleProbeProvider = new WebModuleProbeProvider();
servletProbeProvider = new ServletProbeProvider();
jspProbeProvider = new JspProbeProvider();
sessionProbeProvider = new SessionProbeProvider();
requestProbeProvider = new RequestProbeProvider();
}
/**
* Creates statistics providers for Servlet, JSP, Session, and Request/Response related events.
*/
private void createStatsProviders() {
httpStatsProviderBootstrap = serviceLocator.getService(HttpServiceStatsProviderBootstrap.class);
webStatsProviderBootstrap = serviceLocator.getService(WebStatsProviderBootstrap.class);
}
/*
* Loads the class with the given name using the common classloader, which is responsible for loading any classes from
* the domain's lib directory
*
* @param className the name of the class to load
*/
public Class> loadCommonClass(String className) throws Exception {
return classLoaderHierarchy.getCommonClassLoader().loadClass(className);
}
/**
* According to SRV 15.5.15, Servlets, Filters, Listeners can only be without any scope annotation or are annotated with
*
* @Dependent scope. All other scopes are invalid and must be rejected.
*/
private void validateJSR299Scope(Class> clazz) {
if (cdiService != null && cdiService.isCDIScoped(clazz)) {
String msg = rb.getString(LogFacade.INVALID_ANNOTATION_SCOPE);
msg = MessageFormat.format(msg, clazz.getName());
throw new IllegalArgumentException(msg);
}
}
/**
* Return the WebContainerFeatureFactory according to the configuration.
*
* @return WebContainerFeatuerFactory
*/
private WebContainerFeatureFactory getWebContainerFeatureFactory() {
String featureFactoryName = (serverConfigLookup.calculateWebAvailabilityEnabledFromConfig() ? "ha" : "pe");
return webContainerFeatureFactory = serviceLocator.getService(WebContainerFeatureFactory.class, featureFactoryName);
}
}