/* * Copyright (c) 2005, 2012, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code 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. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package sun.net.httpserver; import java.nio.*; import java.io.*; import java.nio.channels.*; import com.sun.net.httpserver.*; /** */ class Request { final static int BUF_LEN = 2048; final static byte CR = 13; final static byte LF = 10; private String startLine; private SocketChannel chan; private InputStream is; private OutputStream os; Request (InputStream rawInputStream, OutputStream rawout) throws IOException { is = rawInputStream; os = rawout; do { startLine = readLine(); if (startLine == null) { return; } /* skip blank lines */ } while (startLine == null ? false : startLine.equals ("")); } char[] buf = new char [BUF_LEN]; int pos; StringBuffer lineBuf; public InputStream inputStream () { return is; } public OutputStream outputStream () { return os; } /** * read a line from the stream returning as a String. * Not used for reading headers. */ public String readLine () throws IOException { boolean gotCR = false, gotLF = false; pos = 0; lineBuf = new StringBuffer(); while (!gotLF) { int c = is.read(); if (c == -1) { return null; } if (gotCR) { if (c == LF) { gotLF = true; } else { gotCR = false; consume (CR); consume (c); } } else { if (c == CR) { gotCR = true; } else { consume (c); } } } lineBuf.append (buf, 0, pos); return new String (lineBuf); } private void consume (int c) { if (pos == BUF_LEN) { lineBuf.append (buf); pos = 0; } buf[pos++] = (char)c; } /** * returns the request line (first line of a request) */ public String requestLine () { return startLine; } Headers hdrs = null; @SuppressWarnings("fallthrough") Headers headers () throws IOException { if (hdrs != null) { return hdrs; } hdrs = new Headers(); char s[] = new char[10]; int len = 0; int firstc = is.read(); // check for empty headers if (firstc == CR || firstc == LF) { int c = is.read(); if (c == CR || c == LF) { return hdrs; } s[0] = (char)firstc; len = 1; firstc = c; } while (firstc != LF && firstc != CR && firstc >= 0) { int keyend = -1; int c; boolean inKey = firstc > ' '; s[len++] = (char) firstc; parseloop:{ while ((c = is.read()) >= 0) { switch (c) { /*fallthrough*/ case ':': if (inKey && len > 0) keyend = len; inKey = false; break; case '\t': c = ' '; case ' ': inKey = false; break; case CR: case LF: firstc = is.read(); if (c == CR && firstc == LF) { firstc = is.read(); if (firstc == CR) firstc = is.read(); } if (firstc == LF || firstc == CR || firstc > ' ') break parseloop; /* continuation */ c = ' '; break; } if (len >= s.length) { char ns[] = new char[s.length * 2]; System.arraycopy(s, 0, ns, 0, len); s = ns; } s[len++] = (char) c; } firstc = -1; } while (len > 0 && s[len - 1] <= ' ') len--; String k; if (keyend <= 0) { k = null; keyend = 0; } else { k = String.copyValueOf(s, 0, keyend); if (keyend < len && s[keyend] == ':') keyend++; while (keyend < len && s[keyend] <= ' ') keyend++; } String v; if (keyend >= len) v = new String(); else v = String.copyValueOf(s, keyend, len - keyend); if (hdrs.size() >= ServerConfig.getMaxReqHeaders()) { throw new IOException("Maximum number of request headers (" + "sun.net.httpserver.maxReqHeaders) exceeded, " + ServerConfig.getMaxReqHeaders() + "."); } hdrs.add (k,v); len = 0; } return hdrs; } /** * Implements blocking reading semantics on top of a non-blocking channel */ static class ReadStream extends InputStream { SocketChannel channel; ByteBuffer chanbuf; byte[] one; private boolean closed = false, eof = false; ByteBuffer markBuf; /* reads may be satisifed from this buffer */ boolean marked; boolean reset; int readlimit; static long readTimeout; ServerImpl server; final static int BUFSIZE = 8 * 1024; public ReadStream (ServerImpl server, SocketChannel chan) throws IOException { this.channel = chan; this.server = server; chanbuf = ByteBuffer.allocate (BUFSIZE); chanbuf.clear(); one = new byte[1]; closed = marked = reset = false; } public synchronized int read (byte[] b) throws IOException { return read (b, 0, b.length); } public synchronized int read () throws IOException { int result = read (one, 0, 1); if (result == 1) { return one[0] & 0xFF; } else { return -1; } } public synchronized int read (byte[] b, int off, int srclen) throws IOException { int canreturn, willreturn; if (closed) throw new IOException ("Stream closed"); if (eof) { return -1; } assert channel.isBlocking(); if (off < 0 || srclen < 0|| srclen > (b.length-off)) { throw new IndexOutOfBoundsException (); } if (reset) { /* satisfy from markBuf */ canreturn = markBuf.remaining (); willreturn = canreturn>srclen ? srclen : canreturn; markBuf.get(b, off, willreturn); if (canreturn == willreturn) { reset = false; } } else { /* satisfy from channel */ chanbuf.clear (); if (srclen < BUFSIZE) { chanbuf.limit (srclen); } do { willreturn = channel.read (chanbuf); } while (willreturn == 0); if (willreturn == -1) { eof = true; return -1; } chanbuf.flip (); chanbuf.get(b, off, willreturn); if (marked) { /* copy into markBuf */ try { markBuf.put (b, off, willreturn); } catch (BufferOverflowException e) { marked = false; } } } return willreturn; } public boolean markSupported () { return true; } /* Does not query the OS socket */ public synchronized int available () throws IOException { if (closed) throw new IOException ("Stream is closed"); if (eof) return -1; if (reset) return markBuf.remaining(); return chanbuf.remaining(); } public void close () throws IOException { if (closed) { return; } channel.close (); closed = true; } public synchronized void mark (int readlimit) { if (closed) return; this.readlimit = readlimit; markBuf = ByteBuffer.allocate (readlimit); marked = true; reset = false; } public synchronized void reset () throws IOException { if (closed ) return; if (!marked) throw new IOException ("Stream not marked"); marked = false; reset = true; markBuf.flip (); } } static class WriteStream extends java.io.OutputStream { SocketChannel channel; ByteBuffer buf; SelectionKey key; boolean closed; byte[] one; ServerImpl server; public WriteStream (ServerImpl server, SocketChannel channel) throws IOException { this.channel = channel; this.server = server; assert channel.isBlocking(); closed = false; one = new byte [1]; buf = ByteBuffer.allocate (4096); } public synchronized void write (int b) throws IOException { one[0] = (byte)b; write (one, 0, 1); } public synchronized void write (byte[] b) throws IOException { write (b, 0, b.length); } public synchronized void write (byte[] b, int off, int len) throws IOException { int l = len; if (closed) throw new IOException ("stream is closed"); int cap = buf.capacity(); if (cap < len) { int diff = len - cap; buf = ByteBuffer.allocate (2*(cap+diff)); } buf.clear(); buf.put (b, off, len); buf.flip (); int n; while ((n = channel.write (buf)) < l) { l -= n; if (l == 0) return; } } public void close () throws IOException { if (closed) return; //server.logStackTrace ("Request.OS.close: isOpen="+channel.isOpen()); channel.close (); closed = true; } } }