All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.felix.webconsole.servlet.AbstractServlet Maven / Gradle / Ivy

/*
 * 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.felix.webconsole.servlet;


import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URL;
import java.net.URLConnection;

import org.apache.felix.webconsole.internal.Util;

import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;


/**
 * This class can be used as a base class for a web console plugin.
 * The plugin (servlet) needs to be registered as a servlet service
 * with at least the label property.
 * 

* Extending this servlet is one way of providing a plugin. The other * is to simply direclty implement the servlet interface. In both cases * requests to resources with a path of "LABEL/res/*" are automatically * handled. *

* For html (content) requests, the web console automatically renders the header, * footer and navigation. *

* Support for Jakarta servlets requires that the Jakarta Servlet API and the * Apache Felix Http Wrappers are available in the runtime. *

* If you are upgrading from {@link org.apache.felix.webconsole.AbstractWebConsolePlugin} * there are some changes to be aware of: *

    *
  • Resources are detected via the {@link #getResource(String)} method.
  • *
  • Regardless of the http method, if the corresponding doXXX method does not create * a response, the {@link #renderContent(HttpServletRequest, HttpServletResponse)} * method is invoked.
  • *
  • If a doXXX method other than doGet wants to create a response, it must not call * doGet or renderContent directly. It rather just does not create any response and * returns. In this case the web console will invoke the renderContent method * afterwards.
  • *
* * @see ServletConstants#PLUGIN_LABEL * @see ServletConstants#PLUGIN_TITLE * @see ServletConstants#PLUGIN_CATEGORY * @see ServletConstants#PLUGIN_CSS_REFERENCES */ public abstract class AbstractServlet extends HttpServlet { /** * Called to identify resources. * By default, if the path starts with "/res/" this is treated as a resource * and the URL to the resource is tried to be loaded via the class loader. * @param path the path * @return the URL of the resource or null if not found. */ protected URL getResource( final String path ) { final int index = path.indexOf( '/', 1 ); if (index != -1) { if (path.substring(index).startsWith("/res/") ) { return getClass().getResource( path.substring(index) ); } } return null; } /** * Handle get requests. This method can be used to return resources, like JSON responses etc. * If the plugin is serving a resource, this method call {@link HttpServletResponse#setStatus(int)}. * This method is also allowed to send a redirect or an error. * If none of the three applies, the webconsole will call {@link #renderContent(HttpServletRequest, HttpServletResponse)} */ @Override protected void doGet( final HttpServletRequest request, final HttpServletResponse response ) throws ServletException, IOException { this.spoolResource( request, response ); } /** * This method is used to render the main contents of the plugin * @param request The request * @param response The response * @throws ServletException If an error occurs * @throws IOException If writing the response fails */ public void renderContent( final HttpServletRequest request, final HttpServletResponse response) throws ServletException, IOException { throw new ServletException("Render method not implemented"); } /** * If the request addresses a resource , this method serves it * and returns true. Otherwise false is returned. *

* If true is returned, the request is considered complete and * request processing terminates. Otherwise request processing continues * with normal plugin rendering. * * @param request The request object * @param response The response object * * @throws IOException If an error occurs accessing or spooling the resource. */ protected final void spoolResource(final HttpServletRequest request, final HttpServletResponse response) throws IOException { // check for a resource, fail if none final URL url = this.getResource(request.getPathInfo()); if ( url == null ) { return; } // open the connection and the stream (we use the stream to be able // to at least hint to close the connection because there is no // method to explicitly close the conneciton, unfortunately) final URLConnection connection = url.openConnection(); try ( final InputStream ins = connection.getInputStream()) { // FELIX-2017 Equinox may return an URL for a non-existing // resource but then (instead of throwing) return null on // getInputStream. We should account for this situation and // just assume a non-existing resource in this case. if (ins == null) { return; } // check whether we may return 304/UNMODIFIED long lastModified = connection.getLastModified(); if ( lastModified > 0 ) { long ifModifiedSince = request.getDateHeader( "If-Modified-Since" ); if ( ifModifiedSince >= ( lastModified / 1000 * 1000 ) ) { // Round down to the nearest second for a proper compare // A ifModifiedSince of -1 will always be less response.setStatus( HttpServletResponse.SC_NOT_MODIFIED ); return; } // have to send, so set the last modified header now response.setDateHeader( "Last-Modified", lastModified ); } // describe the contents response.setContentType( getServletContext().getMimeType( request.getPathInfo() ) ); if (connection.getContentLength() != -1) { response.setContentLength( connection.getContentLength() ); } response.setStatus( HttpServletResponse.SC_OK); // spool the actual contents final OutputStream out = response.getOutputStream(); final byte[] buf = new byte[2048]; int rd; while ( ( rd = ins.read( buf ) ) >= 0 ) { out.write( buf, 0, rd ); } } } /** * Reads the templateFile as a resource through the class * loader of this class converting the binary data into a string using * UTF-8 encoding. * * @param templateFile The absolute path to the template file to read. * @return The contents of the template file as a string * * @throws NullPointerException if templateFile is * null * @throws FileNotFoundException If template file cannot be found * @throws IOException On any other error reading the template file */ protected final String readTemplateFile( final String templateFile ) throws IOException { return Util.readTemplateFile( getClass(), templateFile ); } protected RequestVariableResolver getVariableResolver(final HttpServletRequest request) { return (RequestVariableResolver) request.getAttribute(RequestVariableResolver.REQUEST_ATTRIBUTE); } /** * Sets response headers to force the client to not cache the response * sent back. This method must be called before the response is committed * otherwise it will have no effect. *

* This method sets the Cache-Control, Expires, * and Pragma headers. * * @param response The response for which to set the cache prevention */ public static final void setNoCache(final HttpServletResponse response) { response.setHeader("Cache-Control", "no-cache"); response.addHeader("Cache-Control", "no-store"); response.addHeader("Cache-Control", "must-revalidate"); response.addHeader("Cache-Control", "max-age=0"); response.setHeader("Expires", "Thu, 01 Jan 1970 01:00:00 GMT"); response.setHeader("Pragma", "no-cache"); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy