blob: c9d8543399f14ba231cec6145c227be9646ac780 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
#include "pagespeed/apache/simple_buffered_apache_fetch.h"
#include <algorithm>
#include "base/logging.h"
#include "pagespeed/kernel/base/abstract_mutex.h"
#include "pagespeed/kernel/base/thread_system.h"
namespace net_instaweb {
SimpleBufferedApacheFetch::SimpleBufferedApacheFetch(
const RequestContextPtr& request_context,
RequestHeaders* request_headers,
ThreadSystem* thread_system,
request_rec* req,
MessageHandler* handler)
: AsyncFetch(request_context),
apache_writer_(new ApacheWriter(req, thread_system)),
message_handler_(handler),
mutex_(thread_system->NewMutex()),
notify_(mutex_->NewCondvar()),
wait_called_(false) {
// We are proxying content, and the caching in the http configuration
// should not apply; we want to use the caching from the proxy.
apache_writer_->set_disable_downstream_header_filters(true);
SetRequestHeadersTakingOwnership(request_headers);
}
SimpleBufferedApacheFetch::~SimpleBufferedApacheFetch() {
CHECK(wait_called_);
}
// Called on the apache request thread. Blocks until the request is retired.
void SimpleBufferedApacheFetch::Wait() {
{
ScopedMutex lock(mutex_.get());
CHECK(!wait_called_);
wait_called_ = true;
}
while (true) {
OpInfo op;
WaitForOp(&op);
switch (op.first) {
case kOpHeadersComplete:
SendOutHeaders();
break;
case kOpWrite:
if (!op.second.empty()) {
apache_writer_->Write(op.second, message_handler_);
}
break;
case kOpFlush:
apache_writer_->Flush(message_handler_);
break;
case kOpDone:
return;
default:
LOG(DFATAL) << "Weird opcode in SimpleBufferedApacheFetch queue:"
<< op.first;
}
}
}
void SimpleBufferedApacheFetch::WaitForOp(OpInfo* out) {
ScopedMutex lock(mutex_.get());
while (queue_.empty()) {
notify_->Wait();
}
*out = std::move(queue_.front());
queue_.pop();
}
// Called by other threads.
void SimpleBufferedApacheFetch::HandleHeadersComplete() {
ScopedMutex lock(mutex_.get());
queue_.emplace(kOpHeadersComplete, GoogleString());
notify_->Signal();
}
// Called on the request thread.
void SimpleBufferedApacheFetch::SendOutHeaders() {
if (content_length_known()) {
apache_writer_->set_content_length(content_length());
}
apache_writer_->OutputHeaders(response_headers());
}
// Called by other threads.
void SimpleBufferedApacheFetch::HandleDone(bool success) {
// TODO(morlovich): ApacheFetch at least warns when success = false,
// and headers were successful.
ScopedMutex lock(mutex_.get());
queue_.emplace(kOpDone, GoogleString());
notify_->Signal();
}
bool SimpleBufferedApacheFetch::HandleWrite(const StringPiece& sp,
MessageHandler* handler) {
ScopedMutex lock(mutex_.get());
if (queue_.empty() || queue_.back().first != kOpWrite) {
queue_.emplace(kOpWrite, GoogleString());
}
sp.AppendToString(&queue_.back().second);
notify_->Signal();
return true;
}
bool SimpleBufferedApacheFetch::HandleFlush(MessageHandler* handler) {
ScopedMutex lock(mutex_.get());
queue_.emplace(kOpFlush, GoogleString());
notify_->Signal();
return true;
}
bool SimpleBufferedApacheFetch::IsCachedResultValid(
const ResponseHeaders& headers) {
LOG(WARNING) << "SimpleBufferedApacheFetch::IsCachedResultValid called; "
"should only get this far in tests";
return true;
}
} // namespace net_instaweb