blob: fb48d68642a6c28f8578568d7ee8d8ef577d415a [file] [log] [blame]
// Copyright 2017 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/url_request/redirect_util.h"
#include <string>
#include "net/http/http_request_headers.h"
#include "net/url_request/redirect_info.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "url/gurl.h"
namespace net {
namespace {
TEST(RedirectUtilTest, UpdateHttpRequest) {
const GURL original_url("https://www.example.com/test.php");
const char* kOriginalMethod = "POST";
const char* kCustomHeader = "Custom-Header-For-Test";
const char* kCustomHeaderValue = "custom header value";
struct TestCase {
const char* new_method;
const char* new_url;
const char* modified_headers;
bool expected_should_clear_upload;
// nullptr if the origin header should not exist
const char* expected_origin_header;
};
const TestCase kTests[] = {
{
"POST" /* new_method */,
"https://www.example.com/redirected.php" /* new_url */,
"Header1: Value1\r\nHeader2: Value2" /* modified_headers */,
false /* expected_should_clear_upload */,
"https://origin.example.com" /* expected_origin_header */
},
{
"GET" /* new_method */,
"https://www.example.com/redirected.php" /* new_url */,
"Header1: Value1\r\nHeader2: Value2" /* modified_headers */,
true /* expected_should_clear_upload */,
nullptr /* expected_origin_header */
},
{
"POST" /* new_method */,
"https://other.example.com/redirected.php" /* new_url */,
"Header1: Value1\r\nHeader2: Value2" /* modified_headers */,
false /* expected_should_clear_upload */,
"null" /* expected_origin_header */
},
{
"GET" /* new_method */,
"https://other.example.com/redirected.php" /* new_url */,
"Header1: Value1\r\nHeader2: Value2" /* modified_headers */,
true /* expected_should_clear_upload */,
nullptr /* expected_origin_header */
}};
for (const auto& test : kTests) {
SCOPED_TRACE(::testing::Message() << "new_method: " << test.new_method
<< " new_url: " << test.new_url);
RedirectInfo redirect_info;
redirect_info.new_method = test.new_method;
redirect_info.new_url = GURL(test.new_url);
net::HttpRequestHeaders modified_headers;
modified_headers.AddHeadersFromString(test.modified_headers);
std::string expected_modified_header1, expected_modified_header2;
modified_headers.GetHeader("Header1", &expected_modified_header1);
modified_headers.GetHeader("Header2", &expected_modified_header2);
HttpRequestHeaders request_headers;
request_headers.SetHeader(HttpRequestHeaders::kOrigin,
"https://origin.example.com");
request_headers.SetHeader(HttpRequestHeaders::kContentLength, "100");
request_headers.SetHeader(HttpRequestHeaders::kContentType,
"text/plain; charset=utf-8");
request_headers.SetHeader(kCustomHeader, kCustomHeaderValue);
request_headers.SetHeader("Header1", "Initial-Value1");
bool should_clear_upload = !test.expected_should_clear_upload;
RedirectUtil::UpdateHttpRequest(
original_url, kOriginalMethod, redirect_info,
base::nullopt /* removed_headers */, modified_headers, &request_headers,
&should_clear_upload);
EXPECT_EQ(test.expected_should_clear_upload, should_clear_upload);
EXPECT_EQ(!test.expected_should_clear_upload,
request_headers.HasHeader(HttpRequestHeaders::kContentLength));
EXPECT_EQ(!test.expected_should_clear_upload,
request_headers.HasHeader(HttpRequestHeaders::kContentType));
EXPECT_TRUE(request_headers.HasHeader(kCustomHeader));
std::string origin_header_value;
EXPECT_EQ(test.expected_origin_header != nullptr,
request_headers.GetHeader(HttpRequestHeaders::kOrigin,
&origin_header_value));
if (test.expected_origin_header) {
EXPECT_EQ(test.expected_origin_header, origin_header_value);
}
std::string modified_header1, modified_header2;
EXPECT_TRUE(request_headers.GetHeader("Header1", &modified_header1));
EXPECT_EQ(expected_modified_header1, modified_header1);
EXPECT_TRUE(request_headers.GetHeader("Header2", &modified_header2));
EXPECT_EQ(expected_modified_header2, modified_header2);
}
}
TEST(RedirectUtilTest, RemovedHeaders) {
struct TestCase {
std::vector<const char*> initial_headers;
std::vector<const char*> modified_headers;
std::vector<const char*> removed_headers;
std::vector<const char*> final_headers;
};
const TestCase kTests[] = {
// Remove no headers (empty vector).
{
{}, // Initial headers
{}, // Modified headers
{}, // Removed headers
{}, // Final headers
},
// Remove an existing header.
{
{"A:0"}, // Initial headers
{}, // Modified headers
{"A"}, // Removed headers
{}, // Final headers
},
// Remove a missing header.
{
{}, // Initial headers
{}, // Modified headers
{"A"}, // Removed headers
{}, // Final headers
},
// Remove two different headers.
{
{"A:0", "B:0"}, // Initial headers
{}, // Modified headers
{"A", "B"}, // Removed headers
{}, // Final headers
},
// Remove two times the same headers.
{
{"A:0"}, // Initial headers
{}, // Modified headers
{"A", "A"}, // Removed headers
{}, // Final headers
},
// Remove an existing header that is also modified.
{
{"A:0"}, // Initial headers
{"A:1"}, // Modified headers
{"A"}, // Removed headers
{"A:1"}, // Final headers
},
// Some headers are removed, some aren't.
{
{"A:0", "B:0"}, // Initial headers
{}, // Modified headers
{"A"}, // Removed headers
{"B:0"}, // Final headers
},
};
for (const auto& test : kTests) {
HttpRequestHeaders initial_headers, modified_headers, final_headers;
std::vector<std::string> removed_headers;
for (const char* header : test.initial_headers)
initial_headers.AddHeadersFromString(header);
for (const char* header : test.modified_headers)
modified_headers.AddHeadersFromString(header);
for (const char* header : test.removed_headers)
removed_headers.push_back(header);
for (const char* header : test.final_headers)
final_headers.AddHeadersFromString(header);
bool should_clear_upload(false); // unused.
RedirectUtil::UpdateHttpRequest(GURL(), // original_url
std::string(), // original_method
RedirectInfo(), removed_headers,
modified_headers, &initial_headers,
&should_clear_upload);
// The initial_headers have been updated and should match the expected final
// headers.
EXPECT_EQ(initial_headers.ToString(), final_headers.ToString());
}
}
// Test with removed_headers = base::nullopt.
TEST(RedirectUtilTest, RemovedHeadersNullOpt) {
HttpRequestHeaders initial_headers, final_headers;
initial_headers.AddHeadersFromString("A:0");
final_headers.AddHeadersFromString("A:0");
base::Optional<std::vector<std::string>> removed_headers(base::nullopt);
base::Optional<HttpRequestHeaders> modified_headers(base::in_place);
bool should_clear_upload(false); // unused.
RedirectUtil::UpdateHttpRequest(GURL(), // original_url
std::string(), // original_method
RedirectInfo(), removed_headers,
modified_headers, &initial_headers,
&should_clear_upload);
// The initial_headers have been updated and should match the expected final
// headers.
EXPECT_EQ(initial_headers.ToString(), final_headers.ToString());
}
// Test with modified_headers = base::nullopt.
TEST(RedirectUtilTest, ModifyHeadersNullopt) {
HttpRequestHeaders initial_headers, final_headers;
initial_headers.AddHeadersFromString("A:0");
final_headers.AddHeadersFromString("A:0");
base::Optional<std::vector<std::string>> removed_headers(base::in_place);
base::Optional<HttpRequestHeaders> modified_headers(base::nullopt);
bool should_clear_upload(false); // unused.
RedirectUtil::UpdateHttpRequest(GURL(), // original_url
std::string(), // original_method
RedirectInfo(), removed_headers,
modified_headers, &initial_headers,
&should_clear_upload);
// The initial_headers have been updated and should match the expected final
// headers.
EXPECT_EQ(initial_headers.ToString(), final_headers.ToString());
}
} // namespace
} // namespace net