// Copyright (c) 2009 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "net/tools/flip_server/streamer_interface.h"
#include <string>
#include "net/tools/balsa/balsa_frame.h"
#include "net/tools/flip_server/constants.h"
#include "net/tools/flip_server/flip_config.h"
#include "net/tools/flip_server/sm_connection.h"
namespace net {
std::string StreamerSM::forward_ip_header_;
StreamerSM::StreamerSM(SMConnection* connection,
SMInterface* sm_other_interface,
EpollServer* epoll_server,
FlipAcceptor* acceptor)
: connection_(connection),
sm_other_interface_(sm_other_interface),
epoll_server_(epoll_server),
acceptor_(acceptor),
is_request_(false),
http_framer_(new BalsaFrame) {
VLOG(2) << ACCEPTOR_CLIENT_IDENT << "Creating StreamerSM object";
http_framer_->set_balsa_visitor(this);
http_framer_->set_balsa_headers(&headers_);
http_framer_->set_is_request(false);
}
StreamerSM::~StreamerSM() {
VLOG(1) << ACCEPTOR_CLIENT_IDENT << "Destroying StreamerSM object";
Reset();
delete http_framer_;
}
void StreamerSM::set_is_request() {
is_request_ = true;
http_framer_->set_is_request(true);
}
void StreamerSM::InitSMInterface(SMInterface* sm_other_interface,
int32 server_idx) {
sm_other_interface_ = sm_other_interface;
}
void StreamerSM::InitSMConnection(SMConnectionPoolInterface* connection_pool,
SMInterface* sm_interface,
EpollServer* epoll_server,
int fd,
std::string server_ip,
std::string server_port,
std::string remote_ip,
bool use_ssl) {
VLOG(2) << ACCEPTOR_CLIENT_IDENT << "StreamerSM: Initializing server "
<< "connection.";
connection_->InitSMConnection(connection_pool,
sm_interface,
epoll_server,
fd,
server_ip,
server_port,
remote_ip,
use_ssl);
}
size_t StreamerSM::ProcessReadInput(const char* data, size_t len) {
// For now we only want to parse http requests. Just stream responses
if (is_request_) {
return http_framer_->ProcessInput(data, len);
} else {
return sm_other_interface_->ProcessWriteInput(data, len);
}
}
size_t StreamerSM::ProcessWriteInput(const char* data, size_t len) {
char* dataPtr = new char[len];
memcpy(dataPtr, data, len);
DataFrame* df = new DataFrame;
df->data = (const char*)dataPtr;
df->size = len;
df->delete_when_done = true;
connection_->EnqueueDataFrame(df);
return len;
}
bool StreamerSM::Error() const { return false; }
const char* StreamerSM::ErrorAsString() const { return "(none)"; }
bool StreamerSM::MessageFullyRead() const {
if (is_request_) {
return http_framer_->MessageFullyRead();
} else {
return false;
}
}
void StreamerSM::Reset() {
VLOG(1) << ACCEPTOR_CLIENT_IDENT << "StreamerSM: Reset";
connection_->Cleanup("Server Reset");
http_framer_->Reset();
}
void StreamerSM::ResetForNewConnection() {
http_framer_->Reset();
sm_other_interface_->Reset();
}
void StreamerSM::Cleanup() {
if (is_request_)
http_framer_->Reset();
}
int StreamerSM::PostAcceptHook() {
if (!sm_other_interface_) {
SMConnection* server_connection = SMConnection::NewSMConnection(
epoll_server_, NULL, NULL, acceptor_, "server_conn: ");
if (server_connection == NULL) {
LOG(ERROR) << "StreamerSM: Could not create server conenction.";
return 0;
}
VLOG(2) << ACCEPTOR_CLIENT_IDENT << "StreamerSM: Creating new server "
<< "connection.";
sm_other_interface_ =
new StreamerSM(server_connection, this, epoll_server_, acceptor_);
sm_other_interface_->InitSMInterface(this, 0);
}
// The Streamer interface is used to stream HTTPS connections, so we
// will always use the https_server_ip/port here.
sm_other_interface_->InitSMConnection(NULL,
sm_other_interface_,
epoll_server_,
-1,
acceptor_->https_server_ip_,
acceptor_->https_server_port_,
std::string(),
false);
return 1;
}
size_t StreamerSM::SendSynStream(uint32 stream_id,
const BalsaHeaders& headers) {
return 0;
}
size_t StreamerSM::SendSynReply(uint32 stream_id, const BalsaHeaders& headers) {
return 0;
}
void StreamerSM::ProcessBodyInput(const char* input, size_t size) {
VLOG(2) << ACCEPTOR_CLIENT_IDENT
<< "StreamerHttpSM: Process Body Input Data: "
<< "size " << size;
sm_other_interface_->ProcessWriteInput(input, size);
}
void StreamerSM::MessageDone() {
if (acceptor_->flip_handler_type_ == FLIP_HANDLER_PROXY) {
VLOG(2) << ACCEPTOR_CLIENT_IDENT << "StreamerHttpSM: MessageDone.";
// TODO(kelindsay): anything need to be done ehre?
} else {
VLOG(2) << ACCEPTOR_CLIENT_IDENT << "StraemerHttpSM: MessageDone.";
}
}
void StreamerSM::ProcessHeaders(const BalsaHeaders& headers) {
VLOG(2) << ACCEPTOR_CLIENT_IDENT << "HttpStreamerSM: Process Headers";
BalsaHeaders mod_headers;
mod_headers.CopyFrom(headers);
if (forward_ip_header_.length()) {
LOG(INFO) << "Adding forward header: " << forward_ip_header_;
mod_headers.ReplaceOrAppendHeader(forward_ip_header_,
connection_->client_ip());
} else {
LOG(INFO) << "NOT adding forward header.";
}
SimpleBuffer sb;
char* buffer;
int size;
mod_headers.WriteHeaderAndEndingToBuffer(&sb);
sb.GetReadablePtr(&buffer, &size);
sm_other_interface_->ProcessWriteInput(buffer, size);
}
void StreamerSM::HandleHeaderError(BalsaFrame* framer) { HandleError(); }
void StreamerSM::HandleChunkingError(BalsaFrame* framer) { HandleError(); }
void StreamerSM::HandleBodyError(BalsaFrame* framer) { HandleError(); }
void StreamerSM::HandleError() {
VLOG(1) << ACCEPTOR_CLIENT_IDENT << "Error detected";
}
} // namespace net