/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. */ package org.apache.tuscany.sca.binding.rest.provider; import java.io.IOException; import java.util.ArrayList; import java.util.Arrays; import java.util.Calendar; import java.util.Collections; import java.util.Date; import java.util.Enumeration; import java.util.GregorianCalendar; import java.util.List; import java.util.Properties; import java.util.logging.Level; import java.util.logging.Logger; import javax.servlet.FilterConfig; import javax.servlet.ServletContext; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.ws.rs.HttpMethod; import javax.ws.rs.core.Application; import javax.ws.rs.core.MediaType; import org.apache.tuscany.sca.assembly.Binding; import org.apache.tuscany.sca.binding.rest.RESTBinding; import org.apache.tuscany.sca.common.http.HTTPCacheContext; import org.apache.tuscany.sca.common.http.HTTPHeader; import org.apache.tuscany.sca.core.ExtensionPointRegistry; import org.apache.tuscany.sca.extensibility.ClassLoaderContext; import org.apache.wink.common.internal.registry.ProvidersRegistry; import org.apache.wink.common.internal.registry.metadata.MethodMetadata; import org.apache.wink.server.handlers.HandlersChain; import org.apache.wink.server.handlers.HandlersFactory; import org.apache.wink.server.handlers.MessageContext; import org.apache.wink.server.handlers.RequestHandler; import org.apache.wink.server.handlers.ResponseHandler; import org.apache.wink.server.internal.DeploymentConfiguration; import org.apache.wink.server.internal.RequestProcessor; import org.apache.wink.server.internal.handlers.CheckLocationHeaderHandler; import org.apache.wink.server.internal.registry.ResourceRecord; import org.apache.wink.server.internal.servlet.RestServlet; /** * */ public class TuscanyRESTServlet extends RestServlet { private static final Logger logger = Logger.getLogger(TuscanyRESTServlet.class.getName()); private static final long serialVersionUID = 89997233133964915L; private ExtensionPointRegistry registry; private RESTBinding binding; private Class<?> resourceClass; private boolean fixed; public TuscanyRESTServlet(ExtensionPointRegistry registry, Binding binding, Class<?> resourceClass) { super(); this.registry = registry; this.binding = (RESTBinding) binding; this.resourceClass = resourceClass; } public void init() throws ServletException { ClassLoader cl = ClassLoaderContext.setContextClassLoader(Thread.currentThread().getContextClassLoader(), registry.getServiceDiscovery(), "/META-INF/server/wink-providers"); try { super.init(); } finally { if (cl != null) { // return previous classLoader Thread.currentThread().setContextClassLoader(cl); } } } /** * Create Tuscany own DeploymentConfiguration in order to be able to * add ResponseHandler to the Wink HandlerChain */ public DeploymentConfiguration createDeploymentConfiguration() throws ClassNotFoundException, InstantiationException, IllegalAccessException { return new TuscanyDeploymentConfiguration(); } @SuppressWarnings({"unchecked", "rawtypes"}) @Override public DeploymentConfiguration getDeploymentConfiguration() throws ClassNotFoundException, InstantiationException, IllegalAccessException, IOException { // setup proper classLoader to work on OSGi environment ClassLoader cl = ClassLoaderContext.setContextClassLoader(Thread.currentThread().getContextClassLoader(), registry.getServiceDiscovery(), "javax.ws.rs.ext.RuntimeDelegate", "/META-INF/wink-alternate-shortcuts.properties", "/META-INF/server/wink-providers"); DeploymentConfiguration config = null; try { config = super.getDeploymentConfiguration(); } finally { if (cl != null) { // return previous classLoader Thread.currentThread().setContextClassLoader(cl); } } // [rfeng] FIXME: This is a hack to fool Apache wink to not remove the servlet path config.setFilterConfig(new FilterConfig() { public ServletContext getServletContext() { return getServletContext(); } public Enumeration getInitParameterNames() { return getInitParameterNames(); } public String getInitParameter(String arg0) { return getInitParameter(arg0); } public String getFilterName() { return getServletName(); } }); ProvidersRegistry providers = config.getProvidersRegistry(); providers.addProvider(new DataBindingJAXRSReader(registry), 0.001, true); providers.addProvider(new DataBindingJAXRSWriter(registry), 0.001, true); return config; } private synchronized void fixMediaTypes(DeploymentConfiguration config) { if (fixed) { return; } // FIXME: A hacky workaround for https://issues.apache.org/jira/browse/TUSCANY-3572 ResourceRecord record = config.getResourceRegistry().getRecord(resourceClass); for (MethodMetadata methodMetadata : record.getMetadata().getResourceMethods()) { String method = methodMetadata.getHttpMethod(); if (HttpMethod.GET.equals(method) || HttpMethod.HEAD.equals(method) || HttpMethod.DELETE.equals(method)) { methodMetadata.addConsumes(MediaType.APPLICATION_OCTET_STREAM_TYPE); methodMetadata.addConsumes(MediaType.WILDCARD_TYPE); } if (HttpMethod.HEAD.equals(method) || HttpMethod.DELETE.equals(method)) { methodMetadata.addProduces(MediaType.APPLICATION_OCTET_STREAM_TYPE); methodMetadata.addConsumes(MediaType.WILDCARD_TYPE); } } for (MethodMetadata methodMetadata : record.getMetadata().getSubResourceMethods()) { String method = methodMetadata.getHttpMethod(); if (HttpMethod.GET.equals(method) || HttpMethod.HEAD.equals(method) || HttpMethod.DELETE.equals(method)) { methodMetadata.addConsumes(MediaType.APPLICATION_OCTET_STREAM_TYPE); methodMetadata.addConsumes(MediaType.WILDCARD_TYPE); } if (HttpMethod.HEAD.equals(method) || HttpMethod.DELETE.equals(method)) { methodMetadata.addProduces(MediaType.APPLICATION_OCTET_STREAM_TYPE); methodMetadata.addConsumes(MediaType.WILDCARD_TYPE); } } fixed = true; } @Override public RequestProcessor getRequestProcessor() { RequestProcessor processor = super.getRequestProcessor(); // The 1st call returns null if (processor != null) { fixMediaTypes(processor.getConfiguration()); } return processor; } /** * TuscanyDeploymentConfiguration * * Required to inject TuscanyResponseHandler to the HandlerChain */ class TuscanyDeploymentConfiguration extends DeploymentConfiguration { @Override protected List<ResponseHandler> initResponseUserHandlers() { List<ResponseHandler> list = super.initResponseUserHandlers(); list.add(new TuscanyResponseHandler()); return list; } } /** * TuscanyResponseHandler * * Required to support declartive HTTP Headers */ class TuscanyResponseHandler implements ResponseHandler { public void handleResponse(MessageContext context, HandlersChain chain) throws Throwable { // assert response is not committed final HttpServletResponse httpResponse = context.getAttribute(HttpServletResponse.class); if (httpResponse.isCommitted()) { logger.log(Level.FINE, "The response is already committed. Nothing to do."); return; } //process declarative headers for(HTTPHeader header : binding.getHttpHeaders()) { //treat special headers that need to be calculated if(header.getName().equalsIgnoreCase("Expires")) { GregorianCalendar calendar = new GregorianCalendar(); calendar.setTime(new Date()); calendar.add(Calendar.HOUR, Integer.parseInt(header.getValue())); httpResponse.setHeader("Expires", HTTPCacheContext.RFC822DateFormat.format( calendar.getTime() )); } else { //default behaviour to pass the header value to HTTP response httpResponse.setHeader(header.getName(), header.getValue()); } } chain.doChain(context); } public void init(Properties props) { } } }