// Copyright 2014 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 #include "extensions/common/error_utils.h" #include "extensions/common/manifest_constants.h" #include "extensions/common/manifest_handlers/externally_connectable.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() {} ~ExternallyConnectableTest() override {} protected: ExternallyConnectableInfo* GetExternallyConnectableInfo( scoped_refptr extension) { return static_cast( extension->GetManifestData(manifest_keys::kExternallyConnectable)); } }; TEST_F(ExternallyConnectableTest, IDsAndMatches) { scoped_refptr extension = LoadAndExpectSuccess("externally_connectable_ids_and_matches.json"); ASSERT_TRUE(extension.get()); EXPECT_TRUE(extension->permissions_data()->HasAPIPermission( APIPermission::kWebConnectable)); 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 = LoadAndExpectSuccess("externally_connectable_ids.json"); ASSERT_TRUE(extension.get()); EXPECT_FALSE(extension->permissions_data()->HasAPIPermission( APIPermission::kWebConnectable)); 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 = LoadAndExpectSuccess("externally_connectable_matches.json"); ASSERT_TRUE(extension.get()); EXPECT_TRUE(extension->permissions_data()->HasAPIPermission( APIPermission::kWebConnectable)); 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 = LoadAndExpectSuccess( "externally_connectable_matches_tls_channel_id.json"); ASSERT_TRUE(extension.get()); EXPECT_TRUE(extension->permissions_data()->HasAPIPermission( APIPermission::kWebConnectable)); 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 = LoadAndExpectSuccess("externally_connectable_all_ids.json"); ASSERT_TRUE(extension.get()); EXPECT_FALSE(extension->permissions_data()->HasAPIPermission( APIPermission::kWebConnectable)); 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. std::string matches_ids_array[] = {"g", "h", "c", "i", "a", "z", "b"}; std::vector matches_ids( matches_ids_array, matches_ids_array + arraysize(matches_ids_array)); std::string nomatches_ids_array[] = {"2", "3", "1"}; // all_ids = false. { ExternallyConnectableInfo info(URLPatternSet(), matches_ids, false, false); for (size_t i = 0; i < matches_ids.size(); ++i) EXPECT_TRUE(info.IdCanConnect(matches_ids[i])); for (size_t i = 0; i < arraysize(nomatches_ids_array); ++i) EXPECT_FALSE(info.IdCanConnect(nomatches_ids_array[i])); } // all_ids = true. { ExternallyConnectableInfo info(URLPatternSet(), matches_ids, true, false); for (size_t i = 0; i < matches_ids.size(); ++i) EXPECT_TRUE(info.IdCanConnect(matches_ids[i])); for (size_t i = 0; i < arraysize(nomatches_ids_array); ++i) EXPECT_TRUE(info.IdCanConnect(nomatches_ids_array[i])); } } 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")); } TEST_F(ExternallyConnectableTest, WarningNoAllURLs) { scoped_refptr extension = LoadAndExpectWarning( "externally_connectable_error_all_urls.json", ErrorUtils::FormatErrorMessage(errors::kErrorWildcardHostsNotAllowed, "")); ExternallyConnectableInfo* info = GetExternallyConnectableInfo(extension); EXPECT_FALSE(info->matches.MatchesAllURLs()); EXPECT_FALSE(info->matches.ContainsPattern( URLPattern(URLPattern::SCHEME_ALL, ""))); EXPECT_TRUE(info->matches.MatchesURL(GURL("https://example.com"))); EXPECT_TRUE(info->matches.MatchesURL(GURL("http://build.chromium.org"))); } TEST_F(ExternallyConnectableTest, AllURLsNotWhitelisted) { scoped_refptr extension = LoadAndExpectSuccess( "externally_connectable_all_urls_not_whitelisted.json"); ExternallyConnectableInfo* info = GetExternallyConnectableInfo(extension); EXPECT_FALSE(info->matches.MatchesAllURLs()); } TEST_F(ExternallyConnectableTest, AllURLsWhitelisted) { scoped_refptr extension = LoadAndExpectSuccess("externally_connectable_all_urls_whitelisted.json"); ExternallyConnectableInfo* info = GetExternallyConnectableInfo(extension); EXPECT_TRUE(info->matches.MatchesAllURLs()); URLPattern pattern(URLPattern::SCHEME_ALL, ""); 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, WarningWildcardHost) { scoped_refptr extension = LoadAndExpectWarning( "externally_connectable_error_wildcard_host.json", ErrorUtils::FormatErrorMessage(errors::kErrorWildcardHostsNotAllowed, "http://*/*")); ExternallyConnectableInfo* info = GetExternallyConnectableInfo(extension); EXPECT_FALSE(info->matches.ContainsPattern( URLPattern(URLPattern::SCHEME_ALL, "http://*/*"))); EXPECT_TRUE(info->matches.MatchesURL(GURL("https://example.com"))); EXPECT_TRUE(info->matches.MatchesURL(GURL("http://build.chromium.org"))); } TEST_F(ExternallyConnectableTest, WarningNoTLD) { scoped_refptr extension = LoadAndExpectWarning( "externally_connectable_error_tld.json", ErrorUtils::FormatErrorMessage(errors::kErrorTopLevelDomainsNotAllowed, "co.uk", "http://*.co.uk/*")); ExternallyConnectableInfo* info = GetExternallyConnectableInfo(extension); EXPECT_FALSE(info->matches.ContainsPattern( URLPattern(URLPattern::SCHEME_ALL, "http://*.co.uk/*"))); EXPECT_TRUE(info->matches.MatchesURL(GURL("https://example.com"))); EXPECT_TRUE(info->matches.MatchesURL(GURL("http://build.chromium.org"))); } TEST_F(ExternallyConnectableTest, WarningNoEffectiveTLD) { scoped_refptr extension = LoadAndExpectWarning( "externally_connectable_error_effective_tld.json", ErrorUtils::FormatErrorMessage(errors::kErrorTopLevelDomainsNotAllowed, "appspot.com", "http://*.appspot.com/*")); ExternallyConnectableInfo* info = GetExternallyConnectableInfo(extension); EXPECT_FALSE(info->matches.ContainsPattern( URLPattern(URLPattern::SCHEME_ALL, "http://*.appspot.com/*"))); EXPECT_TRUE(info->matches.MatchesURL(GURL("https://example.com"))); EXPECT_TRUE(info->matches.MatchesURL(GURL("http://build.chromium.org"))); } TEST_F(ExternallyConnectableTest, WarningUnknownTLD) { scoped_refptr extension = LoadAndExpectWarning( "externally_connectable_error_unknown_tld.json", ErrorUtils::FormatErrorMessage(errors::kErrorTopLevelDomainsNotAllowed, "notatld", "http://*.notatld/*")); ExternallyConnectableInfo* info = GetExternallyConnectableInfo(extension); EXPECT_FALSE(info->matches.ContainsPattern( URLPattern(URLPattern::SCHEME_ALL, "http://*.notatld/*"))); EXPECT_TRUE(info->matches.MatchesURL(GURL("https://example.com"))); EXPECT_TRUE(info->matches.MatchesURL(GURL("http://build.chromium.org"))); } TEST_F(ExternallyConnectableTest, WarningNothingSpecified) { LoadAndExpectWarning("externally_connectable_nothing_specified.json", errors::kErrorNothingSpecified); } } // namespace extensions