javax.servlet.http.HttpServlet Maven / Gradle / Ivy
/*
* Copyright (c) 1998-2018 Caucho Technology -- all rights reserved
*
* This file is part of Resin(R) Open Source
*
* Each copy or derived work must preserve the copyright notice and this
* notice unmodified.
*
* Resin Open Source is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* Resin Open Source is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
* of NON-INFRINGEMENT. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with Resin Open Source; if not, write to the
*
* Free Software Foundation, Inc.
* 59 Temple Place, Suite 330
* Boston, MA 02111-1307 USA
*
* @author Scott Ferguson
*/
package javax.servlet.http;
import javax.servlet.GenericServlet;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.io.Serializable;
import java.io.UnsupportedEncodingException;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.Set;
/**
* HttpServlet is a convenient abstract class for creating servlets.
* Normally, servlet writers will only need to override
* doGet
or doPost
.
*
* Caching
*
* HttpServlet makes caching simple. Just override
* getLastModified
. As long as the page hasn't changed,
* it can avoid the overhead of any heavy processing or database queries.
* You cannot use getLastModified
if the response depends
* on sessions, cookies, or any headers in the servlet request.
*
* Hello, world
*
* The Hello.java belongs in myapp/WEB-INF/classes/test/Hello.java under the
* application's root. Normally, it will be called as
* http://myhost.com/myapp/servlet/test.Hello. If the server doesn't use
* applications, then use /servlet/test.Hello.
*
*
* package test;
*
* import java.io.*;
* import javax.servlet.*;
* import javax.servlet.http.*;
*
* public class Hello extends HttpServlet {
* public void doGet(HttpServletRequest request,
* HttpServletResponse response)
* throws ServletException, IOException
* {
* response.setContentType("text/html");
* PrintWriter out = response.getWriter();
* out.println("Hello, World");
* out.close();
* }
* }
*
*/
public abstract class HttpServlet extends GenericServlet
implements Serializable {
/**
* Service a request. Normally not overridden. If you need to override
* this, use GenericServlet instead.
*/
public void service(ServletRequest request, ServletResponse response)
throws ServletException, IOException
{
HttpServletRequest req = (HttpServletRequest) request;
HttpServletResponse res = (HttpServletResponse) response;
service(req, res);
}
/**
* Services a HTTP request. Automatically dispatches based on the
* request method and handles "If-Modified-Since" headers. Normally
* not overridden.
*
* @param req request information
* @param res response object for returning data to the client.
*/
protected void service(HttpServletRequest req, HttpServletResponse res)
throws ServletException, IOException
{
String method = req.getMethod();
boolean isHead = false;
if (method.equals("GET") || (isHead = method.equals("HEAD"))) {
long lastModified = getLastModified(req);
if (lastModified <= 0) {
if (isHead)
doHead(req, res);
else
doGet(req, res);
return;
}
char []newETag = null;
String etag = req.getHeader("If-None-Match");
if (etag != null) {
newETag = generateETag(lastModified);
int len = etag.length();
if (len == newETag.length) {
for (len--; len >= 0; len--) {
if (etag.charAt(len) != newETag[len])
break;
}
}
if (len < 0) {
res.sendError(res.SC_NOT_MODIFIED);
return;
}
}
long requestLastModified = req.getDateHeader("If-Modified-Since");
if ((lastModified / 1000) == (requestLastModified / 1000)) {
res.sendError(res.SC_NOT_MODIFIED);
return;
}
if (newETag == null)
newETag = generateETag(lastModified);
res.setHeader("ETag", new String(newETag));
res.setDateHeader("Last-Modified", lastModified);
if (isHead)
doHead(req, res);
else
doGet(req, res);
}
else if (method.equals("POST")) {
doPost(req, res);
}
else if (method.equals("PUT")) {
doPut(req, res);
}
else if (method.equals("DELETE")) {
doDelete(req, res);
}
else if (method.equals("OPTIONS")) {
doOptions(req, res);
}
else if (method.equals("TRACE")) {
doTrace(req, res);
}
else {
res.sendError(res.SC_NOT_IMPLEMENTED, "Method not implemented");
}
}
private static char []generateETag(long data)
{
char []buf = new char[13];
buf[0] = '"';
buf[1] = encodeBase64(data >> 60);
buf[2] = encodeBase64(data >> 54);
buf[3] = encodeBase64(data >> 48);
buf[4] = encodeBase64(data >> 42);
buf[5] = encodeBase64(data >> 36);
buf[6] = encodeBase64(data >> 30);
buf[7] = encodeBase64(data >> 24);
buf[8] = encodeBase64(data >> 18);
buf[9] = encodeBase64(data >> 12);
buf[10] = encodeBase64(data >> 6);
buf[11] = encodeBase64(data);
buf[12] = '"';
return buf;
}
private static char encodeBase64(long d)
{
d &= 0x3f;
if (d < 26)
return (char) (d + 'A');
else if (d < 52)
return (char) (d + 'a' - 26);
else if (d < 62)
return (char) (d + '0' - 52);
else if (d == 62)
return '+';
else
return '/';
}
/**
* Returns the last-modified time for the page for caching.
* If at all possible, pages should override getLastModified
* to improve performance. Servlet engines like Resin can
* cache the results of the page, resulting in near-static performance.
*
* @param req the request
* @return the last-modified time of the page.
*/
protected long getLastModified(HttpServletRequest req)
{
return -1;
}
/**
* Process a HEAD request. By default, uses doGet.
*
* @param req the client request
* @param res response to the client
*/
protected void doHead(HttpServletRequest req, HttpServletResponse res)
throws ServletException, IOException
{
doGet(req, res);
}
/**
* Process a GET or HEAD request
*
* @param req the client request
* @param res response to the client
*/
protected void doGet(HttpServletRequest req, HttpServletResponse res)
throws ServletException, IOException
{
String protocol = req.getProtocol();
String msg = req.getMethod() + " not supported";
if (protocol.endsWith("1.1")) {
res.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, msg);
} else {
res.sendError(HttpServletResponse.SC_BAD_REQUEST, msg);
}
}
/**
* Process a POST request
*
* @param req the client request
* @param res response to the client
*/
protected void doPost(HttpServletRequest req, HttpServletResponse res)
throws ServletException, IOException
{
String protocol = req.getProtocol();
String msg = req.getMethod() + " not supported";
if (protocol.endsWith("1.1")) {
res.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, msg);
} else {
res.sendError(HttpServletResponse.SC_BAD_REQUEST, msg);
}
}
/**
* Process a PUT request
*
* @param req the client request
* @param res response to the client
*/
protected void doPut(HttpServletRequest req, HttpServletResponse res)
throws ServletException, IOException
{
String protocol = req.getProtocol();
String msg = req.getMethod() + " not supported";
if (protocol.endsWith("1.1")) {
res.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, msg);
} else {
res.sendError(HttpServletResponse.SC_BAD_REQUEST, msg);
}
}
/**
* Process a DELETE request
*
* @param req the client request
* @param res response to the client
*/
protected void doDelete(HttpServletRequest req, HttpServletResponse res)
throws ServletException, IOException
{
String protocol = req.getProtocol();
String msg = req.getMethod() + " not supported";
if (protocol.endsWith("1.1")) {
res.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, msg);
} else {
res.sendError(HttpServletResponse.SC_BAD_REQUEST, msg);
}
}
/**
* Process an OPTIONS request
*
* @param req the client request
* @param res response to the client
*/
protected void doOptions(HttpServletRequest req, HttpServletResponse res)
throws ServletException, IOException
{
Set methods = new HashSet();
Class []parameters
= new Class[]{HttpServletRequest.class, HttpServletResponse.class};
final String []options
= new String[]{"doGet", "doPost", "doDelete", "doPut"};
Class servletClass = this.getClass();
do {
for (String option : options) {
if (methods.contains(option))
continue;
try {
servletClass.getDeclaredMethod(option, parameters);
methods.add(option);
} catch (NoSuchMethodException e) {
}
}
servletClass = servletClass.getSuperclass();
} while (!HttpServlet.class.equals(servletClass));
StringBuilder builder = new StringBuilder();
if (methods.contains("doGet"))
builder.append("GET,HEAD,");
if (methods.contains("doPost"))
builder.append("POST,");
if (methods.contains("doPut"))
builder.append("PUT,");
if (methods.contains("doDelete"))
builder.append("DELETE,");
builder.append("TRACE,OPTIONS");
res.setHeader("Allow", builder.toString());
}
/**
* Process a TRACE request
*
* @param req the client request
* @param res response to the client
*/
protected void doTrace(HttpServletRequest req, HttpServletResponse res)
throws ServletException, IOException
{
res.setContentType("message/http");
PrintWriter out = res.getWriter();
Enumeration headers = req.getHeaderNames();
while (headers.hasMoreElements()) {
String header = headers.nextElement();
out.print(header);
out.print(" : ");
out.println(req.getHeader(header));
}
}
}
/**
* @since 3.0
*/
class NoBodyResponse
extends HttpServletResponseWrapper
{
NoBodyResponse(HttpServletResponse r)
{
super(r);
}
void setContentLength()
{
}
public void setContentLength(int len)
{
throw new UnsupportedOperationException("unimplemented");
}
public ServletOutputStream getOutputStream()
throws IOException
{
throw new UnsupportedOperationException("unimplemented");
}
public PrintWriter getWriter()
throws UnsupportedEncodingException
{
throw new UnsupportedOperationException("unimplemented");
}
}
/**
* @since 3.0
*/
class NoBodyOutputStream
extends ServletOutputStream
{
NoBodyOutputStream()
{
}
int getContentLength()
{
throw new UnsupportedOperationException("unimplemented");
}
public void write(int b)
{
throw new UnsupportedOperationException("unimplemented");
}
public void write(byte buf[], int offset, int len)
throws IOException
{
if (true) throw new UnsupportedOperationException("unimplemented");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy