| // 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 "chrome/browser/plugins/plugin_info_host_impl.h" |
| |
| #include <map> |
| #include <memory> |
| #include <string> |
| #include <utility> |
| |
| #include "base/files/file_path.h" |
| #include "base/functional/bind.h" |
| #include "base/functional/callback_helpers.h" |
| #include "base/i18n/rtl.h" |
| #include "base/memory/raw_ptr.h" |
| #include "base/run_loop.h" |
| #include "base/test/icu_test_util.h" |
| #include "build/branding_buildflags.h" |
| #include "build/build_config.h" |
| #include "chrome/browser/content_settings/host_content_settings_map_factory.h" |
| #include "chrome/browser/plugins/plugin_metadata.h" |
| #include "chrome/browser/plugins/plugin_utils.h" |
| #include "chrome/common/chrome_content_client.h" |
| #include "chrome/common/plugin.mojom-shared.h" |
| #include "chrome/common/pref_names.h" |
| #include "chrome/test/base/testing_profile.h" |
| #include "components/content_settings/core/browser/host_content_settings_map.h" |
| #include "components/content_settings/core/common/pref_names.h" |
| #include "components/sync_preferences/testing_pref_service_syncable.h" |
| #include "content/public/browser/plugin_service.h" |
| #include "content/public/browser/plugin_service_filter.h" |
| #include "content/public/browser/render_process_host.h" |
| #include "content/public/common/content_constants.h" |
| #include "content/public/common/webplugininfo.h" |
| #include "content/public/test/browser_task_environment.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "url/gurl.h" |
| |
| using content::PluginService; |
| using testing::Eq; |
| |
| namespace { |
| |
| void PluginsLoaded(base::OnceClosure callback, |
| const std::vector<content::WebPluginInfo>& plugins) { |
| std::move(callback).Run(); |
| } |
| |
| class FakePluginServiceFilter : public content::PluginServiceFilter { |
| public: |
| FakePluginServiceFilter() = default; |
| ~FakePluginServiceFilter() override = default; |
| |
| bool IsPluginAvailable(content::BrowserContext* browser_context, |
| const content::WebPluginInfo& plugin) override; |
| |
| bool CanLoadPlugin(int render_process_id, |
| const base::FilePath& path) override; |
| |
| void set_plugin_enabled(const base::FilePath& plugin_path, bool enabled) { |
| plugin_state_[plugin_path] = enabled; |
| } |
| |
| private: |
| std::map<base::FilePath, bool> plugin_state_; |
| }; |
| |
| bool FakePluginServiceFilter::IsPluginAvailable( |
| content::BrowserContext* browser_context, |
| const content::WebPluginInfo& plugin) { |
| auto it = plugin_state_.find(plugin.path); |
| if (it == plugin_state_.end()) { |
| ADD_FAILURE() << "No plugin state for '" << plugin.path.value() << "'"; |
| return false; |
| } |
| return it->second; |
| } |
| |
| bool FakePluginServiceFilter::CanLoadPlugin(int render_process_id, |
| const base::FilePath& path) { |
| return true; |
| } |
| |
| } // namespace |
| |
| class PluginInfoHostImplTest : public ::testing::Test { |
| public: |
| PluginInfoHostImplTest() |
| : foo_plugin_path_(FILE_PATH_LITERAL("/path/to/foo")), |
| bar_plugin_path_(FILE_PATH_LITERAL("/path/to/bar")), |
| context_(0, &profile_), |
| host_content_settings_map_( |
| HostContentSettingsMapFactory::GetForProfile(&profile_)) {} |
| |
| void SetUp() override { |
| PluginService::GetInstance()->Init(); |
| PluginService::GetInstance()->SetFilter(&filter_); |
| |
| content::WebPluginInfo foo_plugin(u"Foo Plugin", foo_plugin_path_, u"1", |
| u"The Foo plugin."); |
| content::WebPluginMimeType mime_type; |
| mime_type.mime_type = "foo/bar"; |
| foo_plugin.mime_types.push_back(mime_type); |
| foo_plugin.type = |
| content::WebPluginInfo::PLUGIN_TYPE_BROWSER_INTERNAL_PLUGIN; |
| PluginService::GetInstance()->RegisterInternalPlugin(foo_plugin, false); |
| |
| content::WebPluginInfo bar_plugin(u"Bar Plugin", bar_plugin_path_, u"1", |
| u"The Bar plugin."); |
| mime_type.mime_type = "foo/bar"; |
| bar_plugin.mime_types.push_back(mime_type); |
| bar_plugin.type = |
| content::WebPluginInfo::PLUGIN_TYPE_BROWSER_INTERNAL_PLUGIN; |
| PluginService::GetInstance()->RegisterInternalPlugin(bar_plugin, false); |
| |
| RefreshPlugins(); |
| } |
| |
| protected: |
| TestingProfile* profile() { return &profile_; } |
| |
| PluginInfoHostImpl::Context* context() { return &context_; } |
| |
| HostContentSettingsMap* host_content_settings_map() { |
| return host_content_settings_map_; |
| } |
| |
| void RefreshPlugins() { |
| PluginService::GetInstance()->RefreshPlugins(); |
| |
| #if !BUILDFLAG(IS_WIN) |
| // Can't go out of process in unit tests. |
| content::RenderProcessHost::SetRunRendererInProcess(true); |
| #endif |
| base::RunLoop run_loop; |
| PluginService::GetInstance()->GetPlugins( |
| base::BindOnce(&PluginsLoaded, run_loop.QuitClosure())); |
| run_loop.Run(); |
| #if !BUILDFLAG(IS_WIN) |
| content::RenderProcessHost::SetRunRendererInProcess(false); |
| #endif |
| } |
| |
| void RegisterAndRefreshPlugin(const content::WebPluginInfo& plugin) { |
| PluginService::GetInstance()->RegisterInternalPlugin( |
| plugin, /*add_at_beginning=*/false); |
| RefreshPlugins(); |
| filter_.set_plugin_enabled(plugin.path, true); |
| } |
| |
| base::FilePath foo_plugin_path_; |
| base::FilePath bar_plugin_path_; |
| FakePluginServiceFilter filter_; |
| |
| private: |
| content::BrowserTaskEnvironment task_environment; |
| TestingProfile profile_; |
| PluginInfoHostImpl::Context context_; |
| raw_ptr<HostContentSettingsMap> host_content_settings_map_; |
| }; |
| |
| TEST_F(PluginInfoHostImplTest, FindEnabledPlugin) { |
| filter_.set_plugin_enabled(foo_plugin_path_, true); |
| filter_.set_plugin_enabled(bar_plugin_path_, true); |
| { |
| chrome::mojom::PluginStatus status; |
| content::WebPluginInfo plugin; |
| std::string actual_mime_type; |
| EXPECT_TRUE(context()->FindEnabledPlugin( |
| GURL(), "foo/bar", &status, &plugin, &actual_mime_type, nullptr)); |
| EXPECT_EQ(chrome::mojom::PluginStatus::kAllowed, status); |
| EXPECT_EQ(foo_plugin_path_.value(), plugin.path.value()); |
| } |
| |
| filter_.set_plugin_enabled(foo_plugin_path_, false); |
| { |
| chrome::mojom::PluginStatus status; |
| content::WebPluginInfo plugin; |
| std::string actual_mime_type; |
| EXPECT_TRUE(context()->FindEnabledPlugin( |
| GURL(), "foo/bar", &status, &plugin, &actual_mime_type, nullptr)); |
| EXPECT_EQ(chrome::mojom::PluginStatus::kAllowed, status); |
| EXPECT_EQ(bar_plugin_path_.value(), plugin.path.value()); |
| } |
| |
| filter_.set_plugin_enabled(bar_plugin_path_, false); |
| { |
| chrome::mojom::PluginStatus status; |
| content::WebPluginInfo plugin; |
| std::string actual_mime_type; |
| std::string identifier; |
| std::u16string plugin_name; |
| EXPECT_FALSE(context()->FindEnabledPlugin( |
| GURL(), "foo/bar", &status, &plugin, &actual_mime_type, nullptr)); |
| EXPECT_EQ(chrome::mojom::PluginStatus::kDisabled, status); |
| EXPECT_EQ(foo_plugin_path_.value(), plugin.path.value()); |
| } |
| { |
| chrome::mojom::PluginStatus status; |
| content::WebPluginInfo plugin; |
| std::string actual_mime_type; |
| EXPECT_FALSE(context()->FindEnabledPlugin( |
| GURL(), "baz/blurp", &status, &plugin, &actual_mime_type, nullptr)); |
| EXPECT_EQ(chrome::mojom::PluginStatus::kNotFound, status); |
| EXPECT_EQ(FILE_PATH_LITERAL(""), plugin.path.value()); |
| } |
| } |
| |
| TEST_F(PluginInfoHostImplTest, FindEnabledPluginWithBidiPdfViewerName) { |
| base::test::ScopedRestoreICUDefaultLocale restore_locale; |
| base::i18n::SetRTLForTesting(true); |
| |
| static constexpr char16_t kPluginName[] = u"Bidi Name"; |
| #if BUILDFLAG(GOOGLE_CHROME_BRANDING) |
| static constexpr char kGroupIdentifier[] = "google-chrome-pdf"; |
| #else |
| static constexpr char kGroupIdentifier[] = "chromium-pdf"; |
| #endif // BUILDFLAG(GOOGLE_CHROME_BRANDING) |
| |
| std::u16string expected_plugin_name = kPluginName; |
| ASSERT_TRUE( |
| base::i18n::AdjustStringForLocaleDirection(&expected_plugin_name)); |
| content::WebPluginInfo bidi_pdf_plugin( |
| expected_plugin_name, |
| base::FilePath(ChromeContentClient::kPDFExtensionPluginPath), u"1", |
| u"The PDF viewer with a bidi plugin name."); |
| |
| content::WebPluginMimeType mime_type; |
| mime_type.mime_type = "fake/pdf-viewer"; |
| bidi_pdf_plugin.mime_types.push_back(mime_type); |
| |
| RegisterAndRefreshPlugin(bidi_pdf_plugin); |
| |
| chrome::mojom::PluginStatus status; |
| content::WebPluginInfo plugin; |
| std::string actual_mime_type; |
| std::unique_ptr<PluginMetadata> metadata; |
| ASSERT_TRUE(context()->FindEnabledPlugin(GURL(), "fake/pdf-viewer", &status, |
| &plugin, &actual_mime_type, |
| &metadata)); |
| |
| EXPECT_EQ(chrome::mojom::PluginStatus::kAllowed, status); |
| EXPECT_EQ(expected_plugin_name, plugin.name); |
| EXPECT_EQ("fake/pdf-viewer", actual_mime_type); |
| |
| EXPECT_EQ(kPluginName, metadata->name()); |
| EXPECT_EQ(kGroupIdentifier, metadata->identifier()); |
| EXPECT_EQ(PluginMetadata::SECURITY_STATUS_FULLY_TRUSTED, |
| metadata->security_status()); |
| } |