blob: bbcdcf097df5d6235ec842ba370ad910d2a61372 [file] [log] [blame]
// Copyright 2014 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "extensions/common/manifest_handlers/externally_connectable.h"
#include <stddef.h>
#include <algorithm>
#include <iterator>
#include "extensions/common/error_utils.h"
#include "extensions/common/manifest_constants.h"
#include "extensions/common/manifest_test.h"
#include "extensions/common/permissions/permissions_data.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
using testing::ElementsAre;
namespace extensions {
namespace errors = externally_connectable_errors;
class ExternallyConnectableTest : public ManifestTest {
public:
ExternallyConnectableTest() = default;
~ExternallyConnectableTest() override = default;
protected:
ExternallyConnectableInfo* GetExternallyConnectableInfo(
scoped_refptr<Extension> extension) {
return static_cast<ExternallyConnectableInfo*>(
extension->GetManifestData(manifest_keys::kExternallyConnectable));
}
};
TEST_F(ExternallyConnectableTest, IDsAndMatches) {
scoped_refptr<Extension> extension =
LoadAndExpectSuccess("externally_connectable_ids_and_matches.json");
ASSERT_TRUE(extension.get());
ExternallyConnectableInfo* info =
ExternallyConnectableInfo::Get(extension.get());
ASSERT_TRUE(info);
EXPECT_THAT(info->ids,
ElementsAre("abcdefghijklmnopabcdefghijklmnop",
"ponmlkjihgfedcbaponmlkjihgfedcba"));
EXPECT_FALSE(info->all_ids);
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://example.com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://example.com/")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://example.com/index.html")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://google.com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://google.com/")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://google.com/index.html")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://www.google.com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://www.google.com/")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("https://google.com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("https://google.com/")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://build.chromium.org")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://build.chromium.org/")));
EXPECT_TRUE(
info->matches.MatchesURL(GURL("http://build.chromium.org/index.html")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("https://build.chromium.org")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("https://build.chromium.org/")));
EXPECT_FALSE(
info->matches.MatchesURL(GURL("http://foo.chromium.org/index.html")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://yahoo.com")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://yahoo.com/")));
// TLD-style patterns should match just the TLD.
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://appspot.com/foo.html")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://go/here")));
// TLD-style patterns should *not* match any subdomains of the TLD.
EXPECT_FALSE(
info->matches.MatchesURL(GURL("http://codereview.appspot.com/foo.html")));
EXPECT_FALSE(
info->matches.MatchesURL(GURL("http://chromium.com/index.html")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://here.go/somewhere")));
// Paths that don't have any wildcards should match the exact domain, but
// ignore the trailing slash. This is kind of a corner case, so let's test it.
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://no.wildcard.path")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://no.wildcard.path/")));
EXPECT_FALSE(
info->matches.MatchesURL(GURL("http://no.wildcard.path/index.html")));
}
TEST_F(ExternallyConnectableTest, IDs) {
scoped_refptr<Extension> extension =
LoadAndExpectSuccess("externally_connectable_ids.json");
ASSERT_TRUE(extension.get());
ExternallyConnectableInfo* info =
ExternallyConnectableInfo::Get(extension.get());
ASSERT_TRUE(info);
EXPECT_THAT(info->ids,
ElementsAre("abcdefghijklmnopabcdefghijklmnop",
"ponmlkjihgfedcbaponmlkjihgfedcba"));
EXPECT_FALSE(info->all_ids);
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://google.com/index.html")));
}
TEST_F(ExternallyConnectableTest, Matches) {
scoped_refptr<Extension> extension =
LoadAndExpectSuccess("externally_connectable_matches.json");
ASSERT_TRUE(extension.get());
ExternallyConnectableInfo* info =
ExternallyConnectableInfo::Get(extension.get());
ASSERT_TRUE(info);
EXPECT_THAT(info->ids, ElementsAre());
EXPECT_FALSE(info->all_ids);
EXPECT_FALSE(info->accepts_tls_channel_id);
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://example.com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://example.com/")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://example.com/index.html")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://google.com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://google.com/")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://google.com/index.html")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://www.google.com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://www.google.com/")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("https://google.com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("https://google.com/")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://build.chromium.org")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://build.chromium.org/")));
EXPECT_TRUE(
info->matches.MatchesURL(GURL("http://build.chromium.org/index.html")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("https://build.chromium.org")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("https://build.chromium.org/")));
EXPECT_FALSE(
info->matches.MatchesURL(GURL("http://foo.chromium.org/index.html")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://yahoo.com")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://yahoo.com/")));
}
TEST_F(ExternallyConnectableTest, MatchesWithTlsChannelId) {
scoped_refptr<Extension> extension = LoadAndExpectSuccess(
"externally_connectable_matches_tls_channel_id.json");
ASSERT_TRUE(extension.get());
ExternallyConnectableInfo* info =
ExternallyConnectableInfo::Get(extension.get());
ASSERT_TRUE(info);
EXPECT_THAT(info->ids, ElementsAre());
EXPECT_FALSE(info->all_ids);
EXPECT_TRUE(info->accepts_tls_channel_id);
// The matches portion of the manifest is identical to those in
// externally_connectable_matches, so only a subset of the Matches tests is
// repeated here.
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://example.com")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://example.com/index.html")));
}
TEST_F(ExternallyConnectableTest, AllIDs) {
scoped_refptr<Extension> extension =
LoadAndExpectSuccess("externally_connectable_all_ids.json");
ASSERT_TRUE(extension.get());
ExternallyConnectableInfo* info =
ExternallyConnectableInfo::Get(extension.get());
ASSERT_TRUE(info);
EXPECT_THAT(info->ids,
ElementsAre("abcdefghijklmnopabcdefghijklmnop",
"ponmlkjihgfedcbaponmlkjihgfedcba"));
EXPECT_TRUE(info->all_ids);
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://google.com/index.html")));
}
TEST_F(ExternallyConnectableTest, IdCanConnect) {
// Not in order to test that ExternallyConnectableInfo sorts it.
const std::vector<ExtensionId> matches_ids = {"g", "h", "c", "i",
"a", "z", "b"};
const std::vector<ExtensionId> nomatches_ids = {"2", "3", "1"};
// all_ids = false.
{
ExternallyConnectableInfo info(URLPatternSet(), matches_ids, false, false);
for (const auto& entry : matches_ids) {
EXPECT_TRUE(info.IdCanConnect(entry));
}
for (const auto& entry : nomatches_ids) {
EXPECT_FALSE(info.IdCanConnect(entry));
}
}
// all_ids = true.
{
ExternallyConnectableInfo info(URLPatternSet(), matches_ids, true, false);
for (const auto& entry : matches_ids) {
EXPECT_TRUE(info.IdCanConnect(entry));
}
for (const auto& entry : nomatches_ids) {
EXPECT_TRUE(info.IdCanConnect(entry));
}
}
}
TEST_F(ExternallyConnectableTest, ErrorWrongFormat) {
LoadAndExpectError("externally_connectable_error_wrong_format.json",
"expected dictionary, got string");
}
TEST_F(ExternallyConnectableTest, ErrorBadID) {
LoadAndExpectError(
"externally_connectable_bad_id.json",
ErrorUtils::FormatErrorMessage(errors::kErrorInvalidId, "badid"));
}
TEST_F(ExternallyConnectableTest, ErrorBadMatches) {
LoadAndExpectError("externally_connectable_error_bad_matches.json",
ErrorUtils::FormatErrorMessage(
errors::kErrorInvalidMatchPattern, "www.yahoo.com",
"Missing scheme separator."));
}
TEST_F(ExternallyConnectableTest, AllURLs) {
scoped_refptr<Extension> extension =
LoadAndExpectSuccess("externally_connectable_all_urls.json");
ExternallyConnectableInfo* info = GetExternallyConnectableInfo(extension);
EXPECT_TRUE(info->matches.MatchesAllURLs());
// Sanity check with a pattern and a few URLs.
URLPattern pattern(URLPattern::SCHEME_ALL, "<all_urls>");
EXPECT_TRUE(info->matches.ContainsPattern(pattern));
EXPECT_TRUE(info->matches.MatchesURL(GURL("https://example.com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://build.chromium.org")));
}
TEST_F(ExternallyConnectableTest, WildcardHost) {
scoped_refptr<Extension> extension =
LoadAndExpectSuccess("externally_connectable_wildcard_host.json");
ExternallyConnectableInfo* info = GetExternallyConnectableInfo(extension);
EXPECT_TRUE(info->matches.ContainsPattern(
URLPattern(URLPattern::SCHEME_ALL, "http://*/*")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://example.com")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("https://example.com")));
}
TEST_F(ExternallyConnectableTest, TLD) {
scoped_refptr<Extension> extension =
LoadAndExpectSuccess("externally_connectable_tld.json");
ExternallyConnectableInfo* info = GetExternallyConnectableInfo(extension);
EXPECT_TRUE(info->matches.ContainsPattern(
URLPattern(URLPattern::SCHEME_ALL, "http://*.co.uk/*")));
EXPECT_FALSE(info->matches.MatchesURL(GURL("http://example.com")));
EXPECT_TRUE(info->matches.MatchesURL(GURL("http://example.co.uk")));
}
TEST_F(ExternallyConnectableTest, WarningNothingSpecified) {
LoadAndExpectWarning("externally_connectable_empty_ids.json",
errors::kErrorNothingSpecified);
LoadAndExpectWarning("externally_connectable_empty_matches.json",
errors::kErrorNothingSpecified);
LoadAndExpectWarning("externally_connectable_nothing_specified.json",
errors::kErrorNothingSpecified);
}
TEST_F(ExternallyConnectableTest, WarningUnusedAcceptsTlsChannelId) {
std::vector<std::string> expected_warnings;
expected_warnings.emplace_back(errors::kErrorNothingSpecified);
expected_warnings.emplace_back(errors::kErrorUnusedAcceptsTlsChannelId);
LoadAndExpectWarnings(
"externally_connectable_accepts_tls_channel_id_without_matches.json",
expected_warnings);
}
// Tests that the deprecated externally_connectable.all_urls permission doesn't
// trigger a warning for an extension that requests it.
// TODO(crbug.com/40864987): Remove this test when we remove the
// externally_connectable.all_urls permission.
TEST_F(ExternallyConnectableTest, DeprecatedAllUrlsPermission) {
scoped_refptr<const Extension> extension =
LoadAndExpectSuccess("externally_connectable_all_urls_permission.json");
EXPECT_TRUE(extension->install_warnings().empty());
}
} // namespace extensions