blob: 7001f6c6b2998cc7e7f284dcef018de8784be4d3 [file] [log] [blame]
// Copyright 2017 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/extensions/external_install_error.h"
#include <memory>
#include "base/test/scoped_feature_list.h"
#include "chrome/browser/extensions/extension_browsertest.h"
#include "chrome/browser/extensions/extension_install_prompt.h"
#include "chrome/browser/extensions/extension_service.h"
#include "chrome/browser/extensions/external_install_manager.h"
#include "chrome/browser/extensions/webstore_data_fetcher.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/ui/global_error/global_error_waiter.h"
#include "content/public/test/browser_test.h"
#include "content/public/test/test_utils.h"
#include "extensions/browser/extension_prefs.h"
#include "extensions/browser/extension_registry.h"
#include "extensions/browser/mock_external_provider.h"
#include "extensions/browser/test_extension_registry_observer.h"
#include "extensions/common/extension_features.h"
namespace extensions {
namespace {
constexpr char kMockUserCountString[] = "1000";
constexpr char kMockRatingCountString[] = "100";
constexpr int kMockRatingCount = 100;
constexpr double kMockAverageRating = 4.4;
std::unique_ptr<FetchItemSnippetResponse> CreateMockResponse(
const ExtensionId& id) {
auto mock_response = std::make_unique<FetchItemSnippetResponse>();
mock_response->set_item_id(id);
mock_response->set_user_count_string(kMockUserCountString);
mock_response->set_rating_count_string(kMockRatingCountString);
mock_response->set_rating_count(kMockRatingCount);
mock_response->set_average_rating(kMockAverageRating);
return mock_response;
}
} // namespace
using ExternalInstallErrorTest = ExtensionBrowserTest;
// Test that global errors don't crash on shutdown. See crbug.com/720081.
// TODO(crbug.com/325314721): Remove this test once we stop using the old item
// JSON API to fetch webstore data.
IN_PROC_BROWSER_TEST_F(ExternalInstallErrorTest, TestShutdown) {
// This relies on prompting for external extensions.
FeatureSwitch::ScopedOverride feature_override(
FeatureSwitch::prompt_for_external_extensions(), true);
ExtensionRegistry* registry = ExtensionRegistry::Get(profile());
const char kId[] = "ldnnhddmnhbkjipkidpdiheffobcpfmf";
{
// Wait for an external extension to be installed and a global error about
// it added.
test::GlobalErrorWaiter waiter(profile());
TestExtensionRegistryObserver observer(registry);
auto provider = std::make_unique<MockExternalProvider>(
extension_service(), mojom::ManifestLocation::kExternalPref);
provider->UpdateOrAddExtension(kId, "1.0.0.0",
test_data_dir_.AppendASCII("good.crx"));
extension_service()->AddProviderForTesting(std::move(provider));
extension_service()->CheckForExternalUpdates();
auto extension = observer.WaitForExtensionInstalled();
EXPECT_EQ(extension->id(), kId);
waiter.Wait();
}
// Verify the extension is in the expected state (disabled for being
// unacknowledged).
EXPECT_FALSE(registry->enabled_extensions().Contains(kId));
ExtensionPrefs* prefs = ExtensionPrefs::Get(profile());
EXPECT_FALSE(prefs->IsExternalExtensionAcknowledged(kId));
EXPECT_EQ(disable_reason::DISABLE_EXTERNAL_EXTENSION,
prefs->GetDisableReasons(kId));
// Verify the external error.
ExternalInstallManager* manager =
extension_service()->external_install_manager();
std::vector<ExternalInstallError*> errors = manager->GetErrorsForTesting();
ASSERT_EQ(1u, errors.size());
EXPECT_EQ(kId, errors[0]->extension_id());
// End the test and shutdown without removing the global error. This should
// not crash.
}
class ExternalInstallErrorItemSnippetsTest : public ExternalInstallErrorTest {
public:
ExternalInstallErrorItemSnippetsTest() {
scoped_feature_list_.InitAndEnableFeature(
extensions_features::kUseItemSnippetsAPI);
}
private:
base::test::ScopedFeatureList scoped_feature_list_;
};
// Similar to the ExternalInstallErrorTest version of this test except
// webstore data is fetched from the item snippets API (API response is mocked)
// for this test.
IN_PROC_BROWSER_TEST_F(ExternalInstallErrorItemSnippetsTest, TestShutdown) {
// This relies on prompting for external extensions.
FeatureSwitch::ScopedOverride feature_override(
FeatureSwitch::prompt_for_external_extensions(), true);
const char kId[] = "ldnnhddmnhbkjipkidpdiheffobcpfmf";
auto mock_response = CreateMockResponse(kId);
WebstoreDataFetcher::SetMockItemSnippetReponseForTesting(mock_response.get());
ExtensionRegistry* registry = ExtensionRegistry::Get(profile());
{
// Wait for an external extension to be installed and a global error about
// it added.
test::GlobalErrorWaiter waiter(profile());
TestExtensionRegistryObserver observer(registry);
auto provider = std::make_unique<MockExternalProvider>(
extension_service(), mojom::ManifestLocation::kExternalPref);
provider->UpdateOrAddExtension(kId, "1.0.0.0",
test_data_dir_.AppendASCII("good.crx"));
extension_service()->AddProviderForTesting(std::move(provider));
extension_service()->CheckForExternalUpdates();
auto extension = observer.WaitForExtensionInstalled();
EXPECT_EQ(extension->id(), kId);
waiter.Wait();
}
// Verify the extension is in the expected state (disabled for being
// unacknowledged).
EXPECT_FALSE(registry->enabled_extensions().Contains(kId));
ExtensionPrefs* prefs = ExtensionPrefs::Get(profile());
EXPECT_FALSE(prefs->IsExternalExtensionAcknowledged(kId));
EXPECT_EQ(disable_reason::DISABLE_EXTERNAL_EXTENSION,
prefs->GetDisableReasons(kId));
// Verify the external error.
ExternalInstallManager* manager =
extension_service()->external_install_manager();
std::vector<ExternalInstallError*> errors = manager->GetErrorsForTesting();
ASSERT_EQ(1u, errors.size());
EXPECT_EQ(kId, errors[0]->extension_id());
ExtensionInstallPrompt::Prompt* alert_prompt =
errors[0]->GetPromptForTesting();
ASSERT_TRUE(alert_prompt);
// Validate that the alert prompt's data corresponds to what is returned from
// the item snippets API.
EXPECT_TRUE(alert_prompt->has_webstore_data());
EXPECT_EQ(kMockRatingCountString, alert_prompt->localized_rating_count());
EXPECT_EQ(kMockRatingCount, alert_prompt->rating_count());
EXPECT_EQ(kMockAverageRating, alert_prompt->average_rating());
}
} // namespace extensions