throws IOException
{
boolean isLoggable = log.isLoggable(Level.FINE);
ReadStream is = getRawRead();
WriteStream os = getRawWrite();
CharBuffer cb = getCharBuffer();
int code;
int len;
while (true) {
_rawWrite.flush();
code = is.read();
Server server = getServer();
if (server == null || server.isDestroyed()) {
log.fine(dbgId() + " request after server close");
killKeepalive("after server close");
return false;
}
switch (code) {
case -1:
if (isLoggable)
log.fine(dbgId() + "r: end of file");
_filter.setClientClosed(true);
killKeepalive("hmux end of file");
return false;
case HMUX_CHANNEL:
int channel = (is.read() << 8) + is.read();
if (isLoggable)
log.fine(dbgId() + "channel-r " + channel);
break;
case HMUX_YIELD:
if (log.isLoggable(Level.FINER))
log.finer(dbgId() + (char) code + "-r: yield");
os.write(HMUX_ACK);
os.write(0);
os.write(0);
os.flush();
break;
case HMUX_QUIT:
if (isLoggable)
log.fine(dbgId() + (char) code + "-r: end of request");
return true;
case HMUX_EXIT:
if (isLoggable)
log.fine(dbgId() + (char) code + "-r: end of socket");
killKeepalive("hmux exit");
return true;
case HMUX_PROTOCOL:
len = (is.read() << 8) + is.read();
if (len != 4) {
log.fine(dbgId() + (char) code + "-r: protocol length (" + len + ") must be 4.");
killKeepalive("hmux protocol");
return false;
}
int value = ((is.read() << 24)
+ (is.read() << 16)
+ (is.read() << 8)
+ (is.read()));
dispatchProtocol(is, code, value);
return true;
case HMUX_URI:
len = (is.read() << 8) + is.read();
_uri.setLength(len);
is.readAll(_uri.getBuffer(), 0, len);
if (isLoggable)
log.fine(dbgId() + (char) code + ":uri " + _uri);
_hasRequest = true;
break;
case HMUX_METHOD:
len = (is.read() << 8) + is.read();
is.readAll(_method, len);
if (isLoggable)
log.fine(dbgId() +
(char) code + ":method " + _method);
break;
case CSE_REAL_PATH:
len = (is.read() << 8) + is.read();
_cb1.clear();
is.readAll(_cb1, len);
code = is.read();
if (code != HMUX_STRING)
throw new IOException("protocol expected HMUX_STRING");
_cb2.clear();
is.readAll(_cb2, readLength(is));
//http.setRealPath(cb1.toString(), cb2.toString());
if (isLoggable)
log.fine(dbgId() + (char) code + " " +
_cb1.toString() + "->" + _cb2.toString());
//throw new RuntimeException();
break;
case CSE_REMOTE_HOST:
len = (is.read() << 8) + is.read();
is.readAll(_remoteHost, len);
if (isLoggable)
log.fine(dbgId() + (char) code + " " + _remoteHost);
break;
case CSE_REMOTE_ADDR:
len = (is.read() << 8) + is.read();
is.readAll(_remoteAddr, len);
if (isLoggable)
log.fine(dbgId() + (char) code + " " + _remoteAddr);
break;
case HMUX_SERVER_NAME:
len = (is.read() << 8) + is.read();
is.readAll(_serverName, len);
if (isLoggable)
log.fine(dbgId() + (char) code + " server-host: " + _serverName);
break;
case CSE_REMOTE_PORT:
len = (is.read() << 8) + is.read();
is.readAll(_remotePort, len);
if (isLoggable)
log.fine(dbgId() + (char) code +
" remote-port: " + _remotePort);
break;
case CSE_SERVER_PORT:
len = (is.read() << 8) + is.read();
is.readAll(_serverPort, len);
if (isLoggable)
log.fine(dbgId() + (char) code +
" server-port: " + _serverPort);
break;
case CSE_QUERY_STRING:
len = (is.read() << 8) + is.read();
if (len > 0) {
_uri.add('?');
_uri.ensureCapacity(_uri.getLength() + len);
is.readAll(_uri.getBuffer(), _uri.getLength(), len);
_uri.setLength(_uri.getLength() + len);
}
break;
case CSE_PROTOCOL:
len = (is.read() << 8) + is.read();
is.readAll(_protocol, len);
if (isLoggable)
log.fine(dbgId() + (char) code + " protocol: " + _protocol);
for (int i = 0; i < len; i++) {
char ch = _protocol.charAt(i);
if ('0' <= ch && ch <= '9')
_version = 16 * _version + ch - '0';
else if (ch == '.')
_version = 16 * _version;
}
break;
case HMUX_HEADER:
len = (is.read() << 8) + is.read();
int headerSize = _headerSize;
CharBuffer key = _headerKeys[headerSize];
key.clear();
CharBuffer valueCb = _headerValues[headerSize];
valueCb.clear();
is.readAll(key, len);
code = is.read();
if (code != HMUX_STRING)
throw new IOException("protocol expected HMUX_STRING at " + (char) code);