/** * This file Copyright (c) 2005-2008 Aptana, Inc. This program is * dual-licensed under both the Aptana Public License and the GNU General * Public license. You may elect to use one or the other of these licenses. * * This program is distributed in the hope that it will be useful, but * AS-IS and WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, TITLE, or * NONINFRINGEMENT. Redistribution, except as permitted by whichever of * the GPL or APL you select, is prohibited. * * 1. For the GPL license (GPL), you can redistribute and/or modify this * program under the terms of the GNU General Public License, * Version 3, as published by the Free Software Foundation. You should * have received a copy of the GNU General Public License, Version 3 along * with this program; if not, write to the Free Software Foundation, Inc., 51 * Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Aptana provides a special exception to allow redistribution of this file * with certain other free and open source software ("FOSS") code and certain additional terms * pursuant to Section 7 of the GPL. You may view the exception and these * terms on the web at http://www.aptana.com/legal/gpl/. * * 2. For the Aptana Public License (APL), this program and the * accompanying materials are made available under the terms of the APL * v1.0 which accompanies this distribution, and is available at * http://www.aptana.com/legal/apl/. * * You may view the GPL, Aptana's exception and additional terms, and the * APL in the file titled license.html at the root of the corresponding * plugin containing this source file. * * Any modifications to this file must keep this entire header intact. */ package com.aptana.ide.server.jetty; import java.net.URI; import org.eclipse.core.resources.IResource; import org.eclipse.core.resources.ResourcesPlugin; import org.eclipse.core.runtime.IPath; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.IStatus; import org.eclipse.core.runtime.Path; import org.eclipse.core.runtime.Status; import org.eclipse.debug.core.model.IProcess; import org.mortbay.jetty.NCSARequestLog; import org.mortbay.jetty.Server; import org.mortbay.jetty.handler.RequestLogHandler; import org.mortbay.jetty.servlet.Context; import org.mortbay.jetty.servlet.ServletHolder; import com.aptana.ide.core.FileUtils; import com.aptana.ide.core.IdeLog; import com.aptana.ide.core.StringUtils; import com.aptana.ide.server.ServerCore; import com.aptana.ide.server.core.IAbstractConfiguration; import com.aptana.ide.server.core.ILog; import com.aptana.ide.server.core.IModule; import com.aptana.ide.server.core.IServer; import com.aptana.ide.server.core.IServerType; import com.aptana.ide.server.core.impl.servers.AbstractServer; import com.aptana.ide.server.core.impl.servers.ServerManager; import com.aptana.ide.server.http.HttpServer; import com.aptana.ide.server.jetty.builder.JettyServerBuilder; /** * @author Pavel Petrochenko */ public class JettyServer extends AbstractServer { private static final IProcess[] NO_PROCESS = new IProcess[0]; private int port; private String serverAddress; private Server server; private String boundName; private String documentRoot; private IPath log; /** * @see com.aptana.ide.server.core.impl.servers.AbstractServer#installConfig(com.aptana.ide.server.core.IAbstractConfiguration) */ @Override protected void installConfig(IAbstractConfiguration configuration) { this.port = configuration.getIntAttribute(IServer.KEY_PORT); this.documentRoot = configuration.getStringAttribute(JettyServerTypeDelegate.KEY_SERVERID); this.boundName = configuration.getStringAttribute(IServer.KEY_ASSOCIATION_SERVER_ID); String id = configuration.getStringAttribute(IServer.KEY_ID); this.log = new Path(FileUtils.systemTempDir).append("jetty_server_" + id + ".log"); //$NON-NLS-1$ //$NON-NLS-2$ super.installConfig(configuration); } /** * @see com.aptana.ide.server.core.impl.servers.AbstractServer#storeConfiguration(com.aptana.ide.server.core.IAbstractConfiguration) */ public void storeConfiguration(IAbstractConfiguration config) { config.setIntAttribute(IServer.KEY_PORT, port); config.setStringAttribute(JettyServerTypeDelegate.KEY_SERVERID, documentRoot); config.setStringAttribute(IServer.KEY_ASSOCIATION_SERVER_ID, boundName); super.storeConfiguration(config); } /** * @param type * @param configuration */ public JettyServer(IServerType type, IAbstractConfiguration configuration) { super(type, configuration); } /** * @see com.aptana.ide.server.core.impl.servers.AbstractServer#restart(java.lang.String, * org.eclipse.core.runtime.IProgressMonitor) */ @Override protected IStatus restart(String mode, IProgressMonitor monitor) { try { stop(true, monitor); serverChanged(); // wait a little try { Thread.sleep(100); } catch (InterruptedException e) { IdeLog.log(JettyPlugin.getDefault(), IStatus.WARNING, "interrupted while sleeping", e); //$NON-NLS-1$ } start(mode, monitor); return Status.OK_STATUS; } catch (Exception e) { return new Status(IStatus.ERROR, JettyPlugin.PLUGIN_ID, IStatus.ERROR, StringUtils.format( Messages.JettyServer_START_EXCEPTION, getName()), e); } } /** * @see com.aptana.ide.server.core.impl.servers.AbstractServer#start(java.lang.String, * org.eclipse.core.runtime.IProgressMonitor) */ @Override protected IStatus start(String mode, IProgressMonitor monitor) { IResource findMember = ResourcesPlugin.getWorkspace().getRoot().getContainerForLocation(new Path(documentRoot)); server = new Server(port); RequestLogHandler logger = new RequestLogHandler(); NCSARequestLog log = new NCSARequestLog(this.log.toFile().getAbsolutePath()); log.setLogCookies(true); log.setLogLatency(true); log.setRetainDays(90); log.setAppend(true); log.setExtended(true); log.setLogTimeZone("GMT"); //$NON-NLS-1$ logger.setRequestLog(log); server.addHandler(logger); server.getConnectors()[0].setHost(HttpServer.getServerAddress()); try { Context context = new Context(server, "/", Context.SESSIONS); //$NON-NLS-1$ ResourceBaseServlet servlet = new ResourceBaseServlet(findMember.getLocation().toFile().getAbsolutePath()); servlet.setNoCache(true); context.addServlet(new ServletHolder(servlet), "/"); //$NON-NLS-1$ if (this.boundName.length() > 0) { IServer server = ServerManager.getInstance().findServer(this.boundName); if (server != null) { JettyServerBuilder.getInstance().buildServer(context, JettyServerTypeDelegate.ID, this.boundName, server.getHostname(), server.getPort(), new JettyDocumentRootResolver(documentRoot)); } } server.setStopAtShutdown(true); server.start(); setServerState(IServer.STATE_STARTED); return Status.OK_STATUS; } catch (Exception e) { return new Status(IStatus.ERROR, JettyPlugin.PLUGIN_ID, IStatus.ERROR, StringUtils.format( Messages.JettyServer_Status_Exception, getName()), e); } } /** * @see com.aptana.ide.server.core.impl.servers.AbstractServer#stop(boolean, * org.eclipse.core.runtime.IProgressMonitor) */ @Override protected IStatus stop(boolean force, IProgressMonitor monitor) { try { server.stop(); server.destroy(); server = null; setServerState(IServer.STATE_STOPPED); return Status.OK_STATUS; } catch (Exception e) { return new Status(IStatus.ERROR, JettyPlugin.PLUGIN_ID, IStatus.ERROR, StringUtils.format( Messages.JettyServer_STOP_EXCEPTION, getName()), e); } } /** * @see com.aptana.ide.server.core.IServer#canHaveModule(com.aptana.ide.server.core.IModule) */ public IStatus canHaveModule(IModule module) { return new Status(IStatus.ERROR, JettyPlugin.PLUGIN_ID, IStatus.ERROR, StringUtils.format( Messages.JettyServer_Status_Exception, getName()), null); } /** * @see com.aptana.ide.server.core.impl.servers.AbstractServer#getConfigurationDescription() */ public String getConfigurationDescription() { return StringUtils.format(Messages.JettyServer_DESCRIPTION, new Object[] { port, documentRoot }); } /** * @see com.aptana.ide.server.core.IServer#getLog() */ public ILog getLog() { return new ILog() { public URI getURI() { return log.toFile().toURI(); } public boolean exists() { return log.toFile().exists(); } }; } /** * @return no processes */ public IProcess[] getProcesses() { return NO_PROCESS; } /** * @see com.aptana.ide.server.core.IServer#getHost() */ public String getHost() { return getHostname()+":" + this.port; //$NON-NLS-1$ } /** * @return Can we serve web content? */ public boolean isWebServer() { return true; } /** * @see IServer#getAssociatedServers() */ public IServer[] getAssociatedServers() { if (this.boundName.length() > 0) { IServer server = ServerCore.getServerManager().findServer(this.boundName); if (server != null) { return new IServer[] { server }; } } return new IServer[0]; } /** * @see com.aptana.ide.server.core.IServer#getHostname() */ public String getHostname() { return serverAddress == null ? "127.0.0.1" : serverAddress; //$NON-NLS-1$ } /** * @see com.aptana.ide.server.core.IServer#getPort() */ public int getPort() { return this.port; } /** * @see com.aptana.ide.server.core.IServer#getDocumentRoot() */ public IPath getDocumentRoot() { return new Path(documentRoot); } /** * @see com.aptana.ide.server.core.IServer#fetchStatistics() */ public String fetchStatistics() { return null; } }