* URL handler for the given path.
*/
public void service(ServletRequest req1, ServletResponse res1)
throws ServletException {
final HttpServletRequest req = (HttpServletRequest) req1;
final HttpServletResponse res = (HttpServletResponse) res1;
final HTTPRequestHeader http_header = new HTTPRequestHeader();
if (req.getServletPath().equals("/admin")) try {
log.debug("Forwarding /admin request back to self");
req.getRequestDispatcher("WebMail/admin").forward(req1, res1);
return;
} catch (IOException ioe) {
log.fatal("Forward from '/admin' failed", ioe);
throw new ServletException(ioe.getMessage());
}
final Enumeration en = req.getHeaderNames();
while (en.hasMoreElements()) {
final String s = (String) en.nextElement();
http_header.setHeader(s, req.getHeader(s));
}
http_header.setPath(
req.getPathInfo() == null ? "/" : req.getPathInfo());
InetAddress addr;
try {
addr = InetAddress.getByName(req.getRemoteHost());
} catch (final UnknownHostException e) {
try {
addr = InetAddress.getByName(req.getRemoteAddr());
} catch (final Exception ex) {
throw new ServletException("Remote host must identify!");
}
}
HTMLDocument content = null;
final int err_code = 400;
HTTPSession sess = null;
/*
* Here we try to parse the MIME content that the Client sent in his
* POST since the JServ doesn't do that for us:-( At least we can use
* the functionality provided by the standalone server where we need to
* parse the content ourself anyway.
*/
try {
final BufferedOutputStream out =
new BufferedOutputStream(res.getOutputStream());
/*
* First we try to use the Servlet API's methods to parse the
* parameters. Unfortunately, it doesn't know how to handle MIME
* multipart POSTs, so we will have to handle that ourselves
*/
/*
* First get all the parameters and set their values into
* http_header
*/
Enumeration enum2 = req.getParameterNames();
while (enum2.hasMoreElements()) {
final String s = (String) enum2.nextElement();
http_header.setContent(s, req.getParameter(s));
// log.info("Parameter "+s);
}
/* Then we set all the headers in http_header */
enum2 = req.getHeaderNames();
while (enum2.hasMoreElements()) {
final String s = (String) enum2.nextElement();
http_header.setHeader(s, req.getHeader(s));
}
/*
* In Servlet API 2.2 we might want to fetch the attributes also,
* but this doesn't work in API 2.0, so we leave it commented out
*/
// enum2=req.getAttributeNames();
// while(enum2.hasMoreElements()) {
// String s=(String)enum2.nextElement();
// log.info("Attribute "+s);
// }
/* Now let's try to handle multipart/form-data posts */
if (req.getContentType() != null
&& req.getContentType().toUpperCase().
startsWith("MULTIPART/FORM-DATA")) {
final int size = Integer.parseInt(WebMailServer.
getStorage().getConfig("max attach size"));
final MultipartParser mparser = new MultipartParser(req, size);
Part p;
while ((p = mparser.readNextPart()) != null) {
if (p.isFile()) {
final ByteStore bs = ByteStore.getBinaryFromIS(
((FilePart) p).getInputStream(), size);
bs.setName(((FilePart) p).getFileName());
bs.setContentType(getStorage().getMimeType(
((FilePart) p).getFileName()));
http_header.setContent(p.getName(), bs);
log.info("File name " + bs.getName());
log.info("Type " + bs.getContentType());
} else if (p.isParam()) {
http_header.setContent(p.getName(),
((ParamPart) p).getStringValue());
}
// log.info("Parameter "+p.getName());
}
}
try {
final String url = http_header.getPath();
try {
/* Find out about the session id */
sess = req.getSession(false) == null
? null
: (HTTPSession) req.getSession(false).
getAttribute("webmail.session");
/*
* If the user was logging on, he doesn't have a session id,
* so generate one. If he already had one, all the better,
* we will take the old one
*/
if (sess == null && url.startsWith("/login")) {
sess = newSession(req, http_header);
} else if (sess == null && url.startsWith("/admin/login")) {
http_header.setHeader("LOGIN", "Administrator");
sess = newAdminSession(req, http_header);
}
if (sess == null && !url.equals("/")
&& !url.startsWith("/passthrough")
&& !url.startsWith("/admin")) {
content = getURLHandler().handleURL(
"/logout", sess, http_header);
} else {
/* Ensure that the session state is up-to-date */
if (sess != null) {
sess.setEnv();
}
/* Let the URLHandler determine the result of the query */
content = getURLHandler().
handleURL(url, sess, http_header);
}
} catch (final InvalidPasswordException e) {
log.error("Connection to " + addr.toString()
+ ": Authentication failed!");
if (url.startsWith("/admin/login")) {
content = getURLHandler().
handleURL("/admin", null, http_header);
} else if (url.startsWith("/login")) {
content = getURLHandler().
handleURL("/", null, http_header);
} else
// content=new
// HTMLErrorMessage(getStorage(),e.getMessage());
throw new ServletException("Invalid URL called!");
} catch (final Exception ex) {
content = getURLHandler().
handleException(ex, sess, http_header);
log.debug("Some strange error while handling request", ex);
}
/*
* Set some HTTP headers: Date is now, the document should
* expire in 5 minutes, proxies and clients shouldn't cache it
* and all WebMail documents must be revalidated when they think
* they don't have to follow the "no-cache".
*/
res.setDateHeader("Date:", System.currentTimeMillis());
res.setDateHeader(
"Expires", System.currentTimeMillis() + 300000);
res.setHeader("Pragma", "no-cache");
res.setHeader("Cache-Control", "must-revalidate");
synchronized (out) {
res.setStatus(content.getReturnCode());
if (content.hasHTTPHeader()) {
final Enumeration enumVar = content.getHTTPHeaderKeys();
while (enumVar.hasMoreElements()) {
final String s = (String) enumVar.nextElement();
res.setHeader(s, content.getHTTPHeader(s));
}
}
/*
* What we will send is an image or some other sort of
* binary
*/
if (content instanceof HTMLImage) {
final HTMLImage img = (HTMLImage) content;
/*
* the HTMLImage class provides us with most of the
* necessary information that we want to send
*/
res.setHeader("Content-Type", img.getContentType());
res.setHeader("Content-Transfer-Encoding",
img.getContentEncoding());
res.setHeader("Content-Length", "" + img.size());
res.setHeader("Connection", "Keep-Alive");
/* Send 8k junks */
int offset = 0;
while (offset + chunk_size < img.size()) {
out.write(img.toBinary(), offset, chunk_size);
offset += chunk_size;
}
out.write(img.toBinary(), offset, img.size() - offset);
out.flush();
out.close();
} else {
final byte[] encoded_content =
content.toString().getBytes("UTF-8");
/*
* We are sending HTML text. Set the encoding to UTF-8
* for Unicode messages
*/
res.setHeader("Content-Length",
"" + (encoded_content.length + 2));
res.setHeader("Connection", "Keep-Alive");
res.setHeader(
"Content-Type", "text/html; charset=\"UTF-8\"");
out.write(encoded_content);
out.write("\r\n".getBytes());