com.sun.grizzly.tcp.http11.GrizzlyAdapterChain Maven / Gradle / Ivy
/*
*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright 2007-2008 Sun Microsystems, Inc. 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.html
* or glassfish/bootstrap/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 glassfish/bootstrap/legal/LICENSE.txt.
* Sun designates this particular file as subject to the "Classpath" exception
* as provided by Sun in the GPL Version 2 section of the License file that
* accompanied this code. If applicable, add the following below the License
* Header, with the fields enclosed by brackets [] replaced by your own
* identifying information: "Portions Copyrighted [year]
* [mapping 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.
*
*/
package com.sun.grizzly.tcp.http11;
import com.sun.grizzly.tcp.Request;
import com.sun.grizzly.util.buf.ByteChunk;
import com.sun.grizzly.util.buf.MessageBytes;
import com.sun.grizzly.util.buf.UDecoder;
import com.sun.grizzly.util.http.HttpRequestURIDecoder;
import com.sun.grizzly.util.http.mapper.Mapper;
import com.sun.grizzly.util.http.mapper.MappingData;
import java.util.Map.Entry;
import java.util.concurrent.ConcurrentHashMap;
import java.util.logging.Level;
/**
* The GrizzlyAdapterChain class allow the invokation of multiple {@link GrizzlyAdapter}
* every time a new HTTP requests is ready to be handled. Requests are mapped
* to their associated {@link GrizzlyAdapter} at runtime using the mapping
* information configured when invoking the {@link GrizzlyAdapterChain#addGrizzlyAdapter
* (com.sun.grizzly.tcp.http11.GrizzlyAdapter, java.lang.String[])
*
* Below is a simple example using two {@link Servlet}
*
*
GrizzlyWebServer ws = new GrizzlyWebServer(path);
ServletAdapter sa = new ServletAdapter();
sa.setRootFolder(".");
sa.setServletInstance(new ServletTest("Adapter-1"));
ws.addGrizzlyAdapter(sa, new String[]{"/Adapter-1"});
ServletAdapter sa2 = new ServletAdapter();
sa2.setRootFolder("/tmp");
sa2.setServletInstance(new ServletTest("Adapter-2"));
ws.addGrizzlyAdapter(sa2, new String[]{"/Adapter-2"});
System.out.println("Grizzly WebServer listening on port 8080");
ws.start();
*
*
*
* Note: This class is NOT thread-safe, so make sure you synchronize
* when dynamically adding and removing {@link GrizzlyAdapter}
*
* @author Jeanfrancois Arcand
*/
public class GrizzlyAdapterChain extends GrizzlyAdapter{
private UDecoder urlDecoder = new UDecoder();
protected final static int MAPPING_DATA = 12;
protected final static int MAPPED_ADAPTER = 13;
/**
* The list of {@link GrizzlyAdapter} instance.
*/
private ConcurrentHashMap adapters =
new ConcurrentHashMap();
/**
* Internal {@link Mapper} used to Map request to their associated {@link GrizzlyAdapter}
*/
private Mapper mapper = new Mapper();
/**
* The default host.
*/
private final static String LOCAL_HOST="localhost";
/**
* Use the deprecated mechanism.
*/
private boolean oldMappingAlgorithm = false;
public GrizzlyAdapterChain(){
mapper.setDefaultHostName(LOCAL_HOST);
setHandleStaticResources(false);
// We will decode it
setDecodeUrl(false);
}
@Override
public void start(){
for (Entry entry: adapters.entrySet()){
entry.getKey().start();
}
}
/**
* Map the {@link GrizzlyRequest} to the proper {@link GrizzlyAdapter}
* @param request The {@link GrizzlyRequest}
* @param response The {@link GrizzlyResponse}
*/
@Override
public void service(GrizzlyRequest request, GrizzlyResponse response) {
// For backward compatibility.
if (oldMappingAlgorithm){
int i = 0;
int size = adapters.size();
for (Entry entry: adapters.entrySet()){
entry.getKey().service(request, response);
if (response.getStatus() == 404 && i != size -1){
// Reset the
response.setStatus(200, "OK");
} else {
return;
}
}
} else {
Request req = request.getRequest();
MappingData mappingData = null;
try{
MessageBytes decodedURI = req.decodedURI();
decodedURI.duplicate(req.requestURI());
mappingData = (MappingData)req.getNote(MAPPING_DATA);
if (mappingData == null){
mappingData = new MappingData();
req.setNote(MAPPING_DATA, mappingData);
} else {
mappingData.recycle();
}
// Map the request without any trailling.
ByteChunk uriBB = decodedURI.getByteChunk();
int semicolon = uriBB.indexOf(';',0);
if (semicolon > 0){
decodedURI.setBytes(uriBB.getBuffer(), uriBB.getStart(), semicolon);
}
HttpRequestURIDecoder.decode(decodedURI,urlDecoder,null,null);
if (mappingData==null) {
mappingData = (MappingData)req.getNote(MAPPING_DATA);
}
// Map the request to its Adapter
mapper.map(req.serverName(), decodedURI, mappingData);
GrizzlyAdapter adapter = null;
if (mappingData.context != null && mappingData.context instanceof GrizzlyAdapter) {
if (mappingData.wrapper != null){
adapter = (GrizzlyAdapter)mappingData.wrapper;
} else {
adapter = (GrizzlyAdapter)mappingData.context;
}
// We already decoded the URL.
adapter.setDecodeUrl(false);
adapter.service(request.getRequest(), response.getResponse());
} else {
response.getResponse().setStatus(404);
customizedErrorPage(req,response.getResponse());
}
} catch (Throwable t){
try{
response.setStatus(404);
if (logger.isLoggable(Level.FINE)) {
logger.log(Level.FINE, "Invalid URL: " + req.decodedURI(), t);
}
customizedErrorPage(req,response.getResponse());
} catch (Exception ex2){
if (logger.isLoggable(Level.WARNING)) {
logger.log(Level.WARNING, "Unable to error page", ex2);
}
}
}
}
}
/**
* Add a {@link GrizzlyAdapter} to the chain.
* @param {@link GrizzlyAdapter} to the chain.
* @deprecated - uses {@link GrizzlyAdapterChain#addGrizzlyAdapter(com.sun.grizzly.tcp.http11.GrizzlyAdapter, java.lang.String[])}
*/
public void addGrizzlyAdapter(GrizzlyAdapter adapter){
oldMappingAlgorithm = true;
adapters.put(adapter,new String[]{""});
}
/**
* Add a {@link GrizzlyAdapter} and its assciated array of mapping. The mapping
* data will be used to map incoming request to its associated {@link GrizzlyAdapter}.
* @param adapter {@link GrizzlyAdapter} instance
* @param mappings an array of mapping.
*/
public void addGrizzlyAdapter(GrizzlyAdapter adapter, String[] mappings){
if (oldMappingAlgorithm){
throw new IllegalStateException("Cannot mix addGrizzlyAdapter(GrizzlyAdapter) " +
"and addGrizzlyAdapter(GrizzlyAdapter,String[]");
}
if (mappings.length == 0){
addGrizzlyAdapter(adapter);
} else {
adapters.put(adapter,mappings);
for(String mapping: mappings){
String ctx = getContextPath(mapping);
mapper.addContext(LOCAL_HOST, ctx, adapter,
new String[] {"index.html", "index.htm"}, null);
mapper.addWrapper(LOCAL_HOST, ctx,mapping.substring(ctx.length()), adapter);
}
}
}
private String getContextPath(String mapping){
String ctx = "";
int slash = mapping.indexOf("/",1);
if (slash != -1) {
ctx = mapping.substring(0,slash);
} else {
ctx = mapping;
}
if (ctx.startsWith("/*")){
ctx= "";
}
// Special case for the root context
if (ctx.equals("/")) {
ctx = "";
}
return ctx;
}
@Override
public void destroy(){
for (Entry adapter: adapters.entrySet()){
adapter.getKey().destroy();
}
}
/**
* Remove a {@link GrizzlyAdapter}
* @return true if removed
*/
public boolean removeAdapter(GrizzlyAdapter adapter){
if (adapter == null) throw new IllegalStateException();
String[] mappings = adapters.remove(adapter);
if (mappings != null){
for (String mapping : mappings){
String ctx = getContextPath(mapping);
mapper.removeContext(LOCAL_HOST,ctx);
}
adapter.destroy();
}
return (mappings != null);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy