blob: cc5e99b826ad48d5314742bdc287853fb5dd23a1 [file] [log] [blame]
// Copyright (c) 2012 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/test/embedded_test_server/http_response.h"
#include "base/bind.h"
#include "base/format_macros.h"
#include "base/logging.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "base/threading/sequenced_task_runner_handle.h"
#include "net/http/http_status_code.h"
namespace net {
namespace test_server {
HttpResponse::~HttpResponse() = default;
RawHttpResponse::RawHttpResponse(const std::string& headers,
const std::string& contents)
: headers_(headers), contents_(contents) {}
RawHttpResponse::~RawHttpResponse() = default;
void RawHttpResponse::SendResponse(const SendBytesCallback& send,
const SendCompleteCallback& done) {
std::string response;
if (!headers_.empty()) {
response = headers_;
// LocateEndOfHeadersHelper() searches for the first "\n\n" and "\n\r\n" as
// the end of the header.
std::size_t index = response.find_last_not_of("\r\n");
if (index != std::string::npos)
response.erase(index + 1);
response += "\n\n";
response += contents_;
} else {
response = contents_;
}
send.Run(response, done);
}
void RawHttpResponse::AddHeader(const std::string& key_value_pair) {
headers_.append(base::StringPrintf("%s\r\n", key_value_pair.c_str()));
}
BasicHttpResponse::BasicHttpResponse() : code_(HTTP_OK) {
}
BasicHttpResponse::~BasicHttpResponse() = default;
std::string BasicHttpResponse::ToResponseString() const {
// Response line with headers.
std::string response_builder;
std::string http_reason_phrase(GetHttpReasonPhrase(code_));
// TODO(mtomasz): For http/1.0 requests, send http/1.0.
base::StringAppendF(&response_builder,
"HTTP/1.1 %d %s\r\n",
code_,
http_reason_phrase.c_str());
base::StringAppendF(&response_builder, "Connection: close\r\n");
base::StringAppendF(&response_builder, "Content-Length: %" PRIuS "\r\n",
content_.size());
base::StringAppendF(&response_builder, "Content-Type: %s\r\n",
content_type_.c_str());
for (size_t i = 0; i < custom_headers_.size(); ++i) {
const std::string& header_name = custom_headers_[i].first;
const std::string& header_value = custom_headers_[i].second;
DCHECK(header_value.find_first_of("\n\r") == std::string::npos) <<
"Malformed header value.";
base::StringAppendF(&response_builder,
"%s: %s\r\n",
header_name.c_str(),
header_value.c_str());
}
base::StringAppendF(&response_builder, "\r\n");
return response_builder + content_;
}
void BasicHttpResponse::SendResponse(const SendBytesCallback& send,
const SendCompleteCallback& done) {
send.Run(ToResponseString(), done);
}
DelayedHttpResponse::DelayedHttpResponse(const base::TimeDelta delay)
: delay_(delay) {}
DelayedHttpResponse::~DelayedHttpResponse() = default;
void DelayedHttpResponse::SendResponse(const SendBytesCallback& send,
const SendCompleteCallback& done) {
base::SequencedTaskRunnerHandle::Get()->PostDelayedTask(
FROM_HERE, base::BindOnce(send, ToResponseString(), done), delay_);
}
void HungResponse::SendResponse(const SendBytesCallback& send,
const SendCompleteCallback& done) {}
} // namespace test_server
} // namespace net