blob: 797b6bbe992934407419296aa1a0aefa510edf4e [file] [log] [blame]
// Copyright 2019 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 "components/paint_preview/browser/paint_preview_base_service.h"
#include "base/files/scoped_temp_dir.h"
#include "base/macros.h"
#include "base/no_destructor.h"
#include "base/strings/strcat.h"
#include "build/build_config.h"
#include "components/paint_preview/browser/paint_preview_base_service_test_factory.h"
#include "components/paint_preview/common/mojom/paint_preview_recorder.mojom.h"
#include "components/paint_preview/common/test_utils.h"
#include "content/public/browser/render_process_host.h"
#include "content/public/test/test_browser_context.h"
#include "content/public/test/test_renderer_host.h"
#include "mojo/public/cpp/bindings/associated_receiver.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/blink/public/common/associated_interfaces/associated_interface_provider.h"
namespace paint_preview {
namespace {
const char kTestFeatureDir[] = "test_feature";
class RejectionPaintPreviewPolicy : public PaintPreviewPolicy {
public:
RejectionPaintPreviewPolicy() = default;
~RejectionPaintPreviewPolicy() override = default;
RejectionPaintPreviewPolicy(const RejectionPaintPreviewPolicy&) = delete;
RejectionPaintPreviewPolicy& operator=(const RejectionPaintPreviewPolicy&) =
delete;
bool SupportedForContents(content::WebContents* web_contents) override {
return false;
}
};
// Builds a PaintPreviewBaseService associated with |key| which will never
// permit paint previews.
std::unique_ptr<KeyedService> BuildServiceWithRejectionPolicy(
SimpleFactoryKey* key) {
return std::make_unique<PaintPreviewBaseService>(
key->GetPath(), kTestFeatureDir,
std::make_unique<RejectionPaintPreviewPolicy>(), key->IsOffTheRecord());
}
} // namespace
class MockPaintPreviewRecorder : public mojom::PaintPreviewRecorder {
public:
MockPaintPreviewRecorder() = default;
~MockPaintPreviewRecorder() override = default;
void CapturePaintPreview(
mojom::PaintPreviewCaptureParamsPtr params,
mojom::PaintPreviewRecorder::CapturePaintPreviewCallback callback)
override {
{
base::ScopedAllowBlockingForTesting scope;
CheckParams(std::move(params));
std::move(callback).Run(status_, std::move(response_));
}
}
void SetExpectedParams(mojom::PaintPreviewCaptureParamsPtr params) {
expected_params_ = std::move(params);
}
void SetResponse(mojom::PaintPreviewStatus status,
mojom::PaintPreviewCaptureResponsePtr response) {
status_ = status;
response_ = std::move(response);
}
void BindRequest(mojo::ScopedInterfaceEndpointHandle handle) {
binding_.Bind(mojo::PendingAssociatedReceiver<mojom::PaintPreviewRecorder>(
std::move(handle)));
}
MockPaintPreviewRecorder(const MockPaintPreviewRecorder&) = delete;
MockPaintPreviewRecorder& operator=(const MockPaintPreviewRecorder&) = delete;
private:
void CheckParams(mojom::PaintPreviewCaptureParamsPtr input_params) {
// Ignore GUID and File as this is internal information not known by the
// Keyed Service API.
EXPECT_EQ(input_params->clip_rect, expected_params_->clip_rect);
EXPECT_EQ(input_params->is_main_frame, expected_params_->is_main_frame);
}
mojom::PaintPreviewCaptureParamsPtr expected_params_;
mojom::PaintPreviewStatus status_;
mojom::PaintPreviewCaptureResponsePtr response_;
mojo::AssociatedReceiver<mojom::PaintPreviewRecorder> binding_{this};
};
class PaintPreviewBaseServiceTest : public content::RenderViewHostTestHarness {
public:
PaintPreviewBaseServiceTest() = default;
~PaintPreviewBaseServiceTest() override = default;
PaintPreviewBaseServiceTest(const PaintPreviewBaseService&) = delete;
PaintPreviewBaseServiceTest& operator=(const PaintPreviewBaseService&) =
delete;
protected:
void SetUp() override {
content::RenderViewHostTestHarness::SetUp();
key_ = std::make_unique<SimpleFactoryKey>(
browser_context()->GetPath(), browser_context()->IsOffTheRecord());
PaintPreviewBaseServiceTestFactory::GetInstance()->SetTestingFactory(
key_.get(),
base::BindRepeating(&PaintPreviewBaseServiceTestFactory::Build));
rejection_policy_key_ = std::make_unique<SimpleFactoryKey>(
browser_context()->GetPath(), browser_context()->IsOffTheRecord());
PaintPreviewBaseServiceTestFactory::GetInstance()->SetTestingFactory(
rejection_policy_key_.get(),
base::BindRepeating(&BuildServiceWithRejectionPolicy));
}
void OverrideInterface(MockPaintPreviewRecorder* service) {
blink::AssociatedInterfaceProvider* remote_interfaces =
main_rfh()->GetRemoteAssociatedInterfaces();
remote_interfaces->OverrideBinderForTesting(
mojom::PaintPreviewRecorder::Name_,
base::BindRepeating(&MockPaintPreviewRecorder::BindRequest,
base::Unretained(service)));
}
PaintPreviewBaseService* GetService() {
return PaintPreviewBaseServiceTestFactory::GetForKey(key_.get());
}
PaintPreviewBaseService* GetServiceWithRejectionPolicy() {
return PaintPreviewBaseServiceTestFactory::GetForKey(
rejection_policy_key_.get());
}
private:
std::unique_ptr<SimpleFactoryKey> key_ = nullptr;
std::unique_ptr<SimpleFactoryKey> rejection_policy_key_ = nullptr;
};
TEST_F(PaintPreviewBaseServiceTest, CaptureMainFrame) {
MockPaintPreviewRecorder recorder;
auto params = mojom::PaintPreviewCaptureParams::New();
params->clip_rect = gfx::Rect(0, 0, 0, 0);
params->is_main_frame = true;
recorder.SetExpectedParams(std::move(params));
auto response = mojom::PaintPreviewCaptureResponse::New();
response->embedding_token = base::nullopt;
recorder.SetResponse(mojom::PaintPreviewStatus::kOk, std::move(response));
OverrideInterface(&recorder);
auto* service = GetService();
EXPECT_FALSE(service->IsOffTheRecord());
base::FilePath path;
ASSERT_TRUE(service->GetFileManager()->CreateOrGetDirectoryFor(
web_contents()->GetLastCommittedURL(), &path));
base::RunLoop loop;
service->CapturePaintPreview(
web_contents(), path, gfx::Rect(0, 0, 0, 0),
base::BindOnce(
[](base::OnceClosure quit_closure,
PaintPreviewBaseService::CaptureStatus expected_status,
const base::FilePath& expected_path,
PaintPreviewBaseService::CaptureStatus status,
std::unique_ptr<PaintPreviewProto> proto) {
EXPECT_EQ(status, expected_status);
EXPECT_TRUE(proto->has_root_frame());
EXPECT_EQ(proto->subframes_size(), 0);
EXPECT_TRUE(proto->root_frame().is_main_frame());
auto token = base::UnguessableToken::Deserialize(
proto->root_frame().embedding_token_high(),
proto->root_frame().embedding_token_low());
#if defined(OS_WIN)
base::FilePath path = base::FilePath(
base::UTF8ToUTF16(proto->root_frame().file_path()));
base::FilePath name(
base::UTF8ToUTF16(base::StrCat({token.ToString(), ".skp"})));
#else
base::FilePath path =
base::FilePath(proto->root_frame().file_path());
base::FilePath name(base::StrCat({token.ToString(), ".skp"}));
#endif
EXPECT_EQ(path.DirName(), expected_path);
EXPECT_EQ(path.BaseName(), name);
std::move(quit_closure).Run();
},
loop.QuitClosure(), PaintPreviewBaseService::CaptureStatus::kOk,
path));
loop.Run();
}
TEST_F(PaintPreviewBaseServiceTest, CaptureFailed) {
MockPaintPreviewRecorder recorder;
auto params = mojom::PaintPreviewCaptureParams::New();
params->clip_rect = gfx::Rect(0, 0, 0, 0);
params->is_main_frame = true;
recorder.SetExpectedParams(std::move(params));
auto response = mojom::PaintPreviewCaptureResponse::New();
response->embedding_token = base::nullopt;
recorder.SetResponse(mojom::PaintPreviewStatus::kFailed, std::move(response));
OverrideInterface(&recorder);
auto* service = GetService();
EXPECT_FALSE(service->IsOffTheRecord());
base::FilePath path;
ASSERT_TRUE(service->GetFileManager()->CreateOrGetDirectoryFor(
web_contents()->GetLastCommittedURL(), &path));
base::RunLoop loop;
service->CapturePaintPreview(
web_contents(), path, gfx::Rect(0, 0, 0, 0),
base::BindOnce(
[](base::OnceClosure quit_closure,
PaintPreviewBaseService::CaptureStatus expected_status,
PaintPreviewBaseService::CaptureStatus status,
std::unique_ptr<PaintPreviewProto> proto) {
EXPECT_EQ(status, expected_status);
EXPECT_EQ(proto, nullptr);
std::move(quit_closure).Run();
},
loop.QuitClosure(),
PaintPreviewBaseService::CaptureStatus::kCaptureFailed));
loop.Run();
}
TEST_F(PaintPreviewBaseServiceTest, CaptureDisallowed) {
MockPaintPreviewRecorder recorder;
auto params = mojom::PaintPreviewCaptureParams::New();
params->clip_rect = gfx::Rect(0, 0, 0, 0);
params->is_main_frame = true;
recorder.SetExpectedParams(std::move(params));
auto response = mojom::PaintPreviewCaptureResponse::New();
response->embedding_token = base::nullopt;
recorder.SetResponse(mojom::PaintPreviewStatus::kFailed, std::move(response));
OverrideInterface(&recorder);
auto* service = GetServiceWithRejectionPolicy();
EXPECT_FALSE(service->IsOffTheRecord());
base::FilePath path;
ASSERT_TRUE(service->GetFileManager()->CreateOrGetDirectoryFor(
web_contents()->GetLastCommittedURL(), &path));
base::RunLoop loop;
service->CapturePaintPreview(
web_contents(), path, gfx::Rect(0, 0, 0, 0),
base::BindOnce(
[](base::OnceClosure quit_closure,
PaintPreviewBaseService::CaptureStatus expected_status,
PaintPreviewBaseService::CaptureStatus status,
std::unique_ptr<PaintPreviewProto> proto) {
EXPECT_EQ(status, expected_status);
EXPECT_EQ(proto, nullptr);
std::move(quit_closure).Run();
},
loop.QuitClosure(),
PaintPreviewBaseService::CaptureStatus::kContentUnsupported));
loop.Run();
}
} // namespace paint_preview