/* * Copyright (c) 1998-2011 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 Nam Nguyen */ package com.caucho.quercus.lib.file; import java.io.IOException; import java.io.InputStream; import java.io.Reader; import java.io.UnsupportedEncodingException; import java.util.Iterator; import java.util.Map; import java.util.logging.Level; import java.util.logging.Logger; import com.caucho.quercus.env.ArrayValue; import com.caucho.quercus.env.Env; import com.caucho.quercus.env.EnvCleanup; import com.caucho.quercus.env.StringValue; import com.caucho.quercus.env.Value; import com.caucho.quercus.resources.StreamContextResource; import com.caucho.vfs.Encoding; import com.caucho.vfs.HttpStreamWrapper; import com.caucho.vfs.LockableStream; import com.caucho.vfs.Path; import com.caucho.vfs.ReadStream; import com.caucho.vfs.ReadWritePair; import com.caucho.vfs.TempBuffer; import com.caucho.vfs.WriteStream; public class HttpInputOutput extends AbstractBinaryOutput implements BinaryInput, BinaryOutput, LockableStream, EnvCleanup { private static final Logger log = Logger.getLogger(HttpInputOutput.class.getName()); private Env _env; private Path _path; private StreamContextResource _context; private LineReader _lineReader; private ReadStream _is; private WriteStream _os; private HttpStreamWrapper _httpStream; private Reader _readEncoding; private String _readEncodingName; private byte []_bodyStart; public HttpInputOutput(Env env, Path path, StreamContextResource context) throws IOException { init(env, path, context); } private void init(Env env, Path path, StreamContextResource context) throws IOException { _env = env; _path = path; env.addCleanup(this); _path = path; _lineReader = new LineReader(env); if (context != null) { Value options = context.getOptions().get(env.createString(path.getScheme())); String method = options.get(env.createString("method")).toString(); if (method.equals("POST")) { ReadWritePair pair = path.openReadWrite(); _is = pair.getReadStream(); _os = pair.getWriteStream(); } else _is = path.openRead(); _httpStream = (HttpStreamWrapper) _is.getSource(); setOptions(env, options); if (_os != null && _bodyStart != null && _bodyStart.length > 0) _os.write(_bodyStart, 0, _bodyStart.length); } else { _is = path.openRead(); _httpStream = (HttpStreamWrapper) _is.getSource(); } } private void setOptions(Env env, Value options) throws IOException { Iterator<Map.Entry<Value,Value>> iter = options.getIterator(env); while (iter.hasNext()) { Map.Entry<Value,Value> entry = iter.next(); String optionName = entry.getKey().toString(); Value optionValue = entry.getValue(); if (optionName.equals("method")) _httpStream.setMethod(optionValue.toString()); else if (optionName.equals("header")) { String option = optionValue.toString(); int start = 0; int len = option.length(); while (start < len) { int end = option.indexOf("\r\n", start); if (end < 0) end = len; int i = option.indexOf(':', start); if (i < 0 || i > end) { _httpStream.setAttribute(option.substring(start, end), ""); break; } else { String name = option.substring(start, i); String value = option.substring(i + 1, end).trim(); _httpStream.setAttribute(name, value); } start = end += 2; } } else if (optionName.equals("user_agent")) _httpStream.setAttribute("User-Agent", optionValue.toString()); else if (optionName.equals("content")) _bodyStart = optionValue.toBinaryValue(env).toBytes(); else if (optionName.equals("proxy")) env.stub("StreamContextResource::proxy option"); else if (optionName.equals("request_fulluri")) env.stub("StreamContextResource::request_fulluri option"); else if (optionName.equals("protocol_version")) { double version = optionValue.toDouble(); if (version == 1.1) { } else if (version == 1.0) _httpStream.setHttp10(); else env.stub("StreamContextResource::protocol_version " + version); } else if (optionName.equals("timeout")) { long ms = (long) optionValue.toDouble() * 1000; _httpStream.setSocketTimeout(ms); } else if (optionName.equals("ignore_errors")) env.stub("ignore_errors::ignore_errors option"); else env.stub("ignore_errors::" + optionName + " option"); } } @Override public void write(int ch) throws IOException { if (_os != null) _os.write(ch); } /** * Appends to a string builder. */ public StringValue appendTo(StringValue builder) throws IOException { if (_is != null) return builder.append(_is); else return builder; } /** * Returns the read stream. */ public InputStream getInputStream() { return _is; } /** * Opens a copy. */ public BinaryInput openCopy() throws IOException { return new HttpInputOutput(_env, _path, _context); } /** * Reads a character from a file, returning -1 on EOF. */ public int read() throws IOException { if (_is != null) return _is.read(); else return -1; } /** * Reads a buffer from a file, returning -1 on EOF. */ public int read(byte []buffer, int offset, int length) throws IOException { if (_is != null) return _is.read(buffer, offset, length); else return -1; } /** * Reads a buffer from a file, returning -1 on EOF. */ public int read(char []buffer, int offset, int length) throws IOException { if (_is != null) return _is.read(buffer, offset, length); else return -1; } /** * Reads a Binary string. */ public StringValue read(int length) throws IOException { StringValue bb = _env.createBinaryBuilder(); TempBuffer temp = TempBuffer.allocate(); try { byte []buffer = temp.getBuffer(); while (length > 0) { int sublen = buffer.length; if (length < sublen) sublen = length; sublen = read(buffer, 0, sublen); if (sublen > 0) { bb.append(buffer, 0, sublen); length -= sublen; } else break; } } finally { TempBuffer.free(temp); } return bb; } /** * Reads a line from the buffer. */ @Override public StringValue readLine(long length) throws IOException { return _lineReader.readLine(_env, this, length); } /** * Reads the optional linefeed character from a \r\n */ public boolean readOptionalLinefeed() throws IOException { int ch = read(); if (ch == '\n') { return true; } else { unread(); return false; } } /** * Returns true on the EOF. */ public boolean isEOF() { try { if (_is == null) return true; return _is.available() <= 0; } catch (IOException e) { return true; } } /** * Sets the current read encoding. The encoding can either be a * Java encoding name or a mime encoding. * * @param encoding name of the read encoding */ public void setEncoding(String encoding) throws UnsupportedEncodingException { String mimeName = Encoding.getMimeName(encoding); if (mimeName != null && mimeName.equals(_readEncodingName)) return; _readEncoding = Encoding.getReadEncoding(getInputStream(), encoding); _readEncodingName = mimeName; } /** * Unread a character. */ @Override public void unread() throws IOException { if (_is != null) _is.unread(); } @Override public boolean lock(boolean shared, boolean block) { return false; } @Override public boolean unlock() { return false; } /** * Closes the file for reading. */ public void closeRead() { close(); } /** * Closes the file. */ public void close() { _env.removeCleanup(this); cleanup(); } @Override public void cleanup() { try { ReadStream is = _is; WriteStream os = _os; _is = null; _os = null; if (is != null) is.close(); if (os != null) os.close(); } catch (IOException e) { log.log(Level.FINE, e.toString(), e); } } public String toString() { return "HttpInputOutput[" + _path + "]"; } }