| // 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/reporting/reporting_header_parser.h" |
| |
| #include <string> |
| #include <vector> |
| |
| #include "base/json/json_reader.h" |
| #include "base/test/simple_test_tick_clock.h" |
| #include "base/time/time.h" |
| #include "base/values.h" |
| #include "net/reporting/reporting_cache.h" |
| #include "net/reporting/reporting_client.h" |
| #include "net/reporting/reporting_test_util.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "url/gurl.h" |
| #include "url/origin.h" |
| |
| namespace net { |
| namespace { |
| |
| class ReportingHeaderParserTest : public ReportingTestBase { |
| protected: |
| void ParseHeader(const GURL& url, const std::string& json) { |
| std::unique_ptr<base::Value> value = |
| base::JSONReader::Read("[" + json + "]"); |
| if (value) |
| ReportingHeaderParser::ParseHeader(context(), url, std::move(value)); |
| } |
| |
| const GURL kUrl_ = GURL("https://origin/path"); |
| const url::Origin kOrigin_ = url::Origin::Create(GURL("https://origin/")); |
| const GURL kEndpoint_ = GURL("https://endpoint/"); |
| const std::string kGroup_ = "group"; |
| const std::string kType_ = "type"; |
| }; |
| |
| // TODO(juliatuttle): Ideally these tests should be expecting that JSON parsing |
| // (and therefore header parsing) may happen asynchronously, but the entire |
| // pipeline is also tested by NetworkErrorLoggingEndToEndTest. |
| |
| TEST_F(ReportingHeaderParserTest, Invalid) { |
| static const struct { |
| const char* header_value; |
| const char* description; |
| } kInvalidHeaderTestCases[] = { |
| {"{\"max_age\":1, \"endpoints\": [{}]}", "missing url"}, |
| {"{\"max_age\":1, \"endpoints\": [{\"url\":0}]}", "non-string url"}, |
| {"{\"max_age\":1, \"endpoints\": [{\"url\":\"http://insecure/\"}]}", |
| "insecure url"}, |
| |
| {"{\"endpoints\": [{\"url\":\"https://endpoint/\"}]}", "missing max_age"}, |
| {"{\"max_age\":\"\", \"endpoints\": [{\"url\":\"https://endpoint/\"}]}", |
| "non-integer max_age"}, |
| {"{\"max_age\":-1, \"endpoints\": [{\"url\":\"https://endpoint/\"}]}", |
| "negative max_age"}, |
| {"{\"max_age\":1, \"group\":0, " |
| "\"endpoints\": [{\"url\":\"https://endpoint/\"}]}", |
| "non-string group"}, |
| |
| // Note that a non-boolean include_subdomains field is *not* invalid, per |
| // the spec. |
| |
| {"{\"max_age\":1, " |
| "\"endpoints\": [{\"url\":\"https://endpoint/\",\"priority\":\"\"}]}", |
| "non-integer priority"}, |
| |
| {"{\"max_age\":1, " |
| "\"endpoints\": [{\"url\":\"https://endpoint/\",\"weight\":\"\"}]}", |
| "non-integer weight"}, |
| {"{\"max_age\":1, " |
| "\"endpoints\": [{\"url\":\"https://endpoint/\",\"weight\":-1}]}", |
| "negative weight"}, |
| {"{\"max_age\":1, " |
| "\"endpoints\": [{\"url\":\"https://endpoint/\",\"weight\":0}]}", |
| "zero weight"}, |
| |
| {"[{\"max_age\":1, \"endpoints\": [{\"url\":\"https://a/\"}]}," |
| "{\"max_age\":1, \"endpoints\": [{\"url\":\"https://b/\"}]}]", |
| "wrapped in list"}}; |
| |
| for (size_t i = 0; i < arraysize(kInvalidHeaderTestCases); ++i) { |
| auto& test_case = kInvalidHeaderTestCases[i]; |
| ParseHeader(kUrl_, test_case.header_value); |
| |
| std::vector<const ReportingClient*> clients; |
| cache()->GetClients(&clients); |
| EXPECT_TRUE(clients.empty()) |
| << "Invalid Report-To header (" << test_case.description << ": \"" |
| << test_case.header_value << "\") parsed as valid."; |
| } |
| } |
| |
| TEST_F(ReportingHeaderParserTest, Valid) { |
| ParseHeader(kUrl_, "{\"endpoints\": [{\"url\":\"" + kEndpoint_.spec() + |
| "\"}],\"max_age\":86400}"); |
| |
| const ReportingClient* client = |
| FindClientInCache(cache(), kOrigin_, kEndpoint_); |
| ASSERT_TRUE(client); |
| EXPECT_EQ(kOrigin_, client->origin); |
| EXPECT_EQ(kEndpoint_, client->endpoint); |
| EXPECT_EQ(ReportingClient::Subdomains::EXCLUDE, client->subdomains); |
| EXPECT_EQ(86400, (client->expires - tick_clock()->NowTicks()).InSeconds()); |
| EXPECT_EQ(ReportingClient::kDefaultPriority, client->priority); |
| EXPECT_EQ(ReportingClient::kDefaultWeight, client->weight); |
| } |
| |
| TEST_F(ReportingHeaderParserTest, ZeroMaxAge) { |
| cache()->SetClient( |
| kOrigin_, kEndpoint_, ReportingClient::Subdomains::EXCLUDE, kGroup_, |
| tick_clock()->NowTicks() + base::TimeDelta::FromDays(1), |
| ReportingClient::kDefaultPriority, ReportingClient::kDefaultWeight); |
| |
| ParseHeader(kUrl_, "{\"endpoints\":[{\"url\":\"" + kEndpoint_.spec() + |
| "\"}],\"max_age\":0}"); |
| |
| EXPECT_EQ(nullptr, FindClientInCache(cache(), kOrigin_, kEndpoint_)); |
| } |
| |
| TEST_F(ReportingHeaderParserTest, Subdomains) { |
| ParseHeader(kUrl_, "{\"endpoints\":[{\"url\":\"" + kEndpoint_.spec() + |
| "\"}],\"max_age\":86400," |
| "\"include_subdomains\":true}"); |
| |
| const ReportingClient* client = |
| FindClientInCache(cache(), kOrigin_, kEndpoint_); |
| ASSERT_TRUE(client); |
| EXPECT_EQ(ReportingClient::Subdomains::INCLUDE, client->subdomains); |
| } |
| |
| TEST_F(ReportingHeaderParserTest, PriorityPositive) { |
| ParseHeader(kUrl_, "{\"endpoints\":[{\"url\":\"" + kEndpoint_.spec() + |
| "\",\"priority\":2}],\"max_age\":86400}"); |
| |
| const ReportingClient* client = |
| FindClientInCache(cache(), kOrigin_, kEndpoint_); |
| ASSERT_TRUE(client); |
| EXPECT_EQ(2, client->priority); |
| } |
| |
| TEST_F(ReportingHeaderParserTest, PriorityNegative) { |
| ParseHeader(kUrl_, "{\"endpoints\":[{\"url\":\"" + kEndpoint_.spec() + |
| "\",\"priority\":-2}],\"max_age\":86400}"); |
| |
| const ReportingClient* client = |
| FindClientInCache(cache(), kOrigin_, kEndpoint_); |
| ASSERT_TRUE(client); |
| EXPECT_EQ(-2, client->priority); |
| } |
| |
| TEST_F(ReportingHeaderParserTest, Weight) { |
| ParseHeader(kUrl_, "{\"endpoints\":[{\"url\":\"" + kEndpoint_.spec() + |
| "\",\"weight\":3}],\"max_age\":86400}"); |
| |
| const ReportingClient* client = |
| FindClientInCache(cache(), kOrigin_, kEndpoint_); |
| ASSERT_TRUE(client); |
| EXPECT_EQ(3, client->weight); |
| } |
| |
| TEST_F(ReportingHeaderParserTest, RemoveOld) { |
| static const GURL kDifferentEndpoint_ = GURL("https://endpoint2/"); |
| |
| ParseHeader(kUrl_, "{\"endpoints\":[{\"url\":\"" + kEndpoint_.spec() + |
| "\"}],\"max_age\":86400}"); |
| |
| EXPECT_TRUE(FindClientInCache(cache(), kOrigin_, kEndpoint_)); |
| |
| ParseHeader(kUrl_, "{\"endpoints\":[{\"url\":\"" + |
| kDifferentEndpoint_.spec() + |
| "\"}],\"max_age\":86400}"); |
| |
| EXPECT_FALSE(FindClientInCache(cache(), kOrigin_, kEndpoint_)); |
| EXPECT_TRUE(FindClientInCache(cache(), kOrigin_, kDifferentEndpoint_)); |
| } |
| |
| } // namespace |
| } // namespace net |