// Copyright 2020 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/exo/drag_drop_operation.h"
#include <memory>
#include <vector>
#include "ash/drag_drop/drag_drop_controller.h"
#include "ash/shell.h"
#include "ash/test_shell_delegate.h"
#include "base/containers/flat_map.h"
#include "base/files/file_util.h"
#include "base/functional/bind.h"
#include "base/memory/raw_ptr.h"
#include "base/run_loop.h"
#include "base/task/sequenced_task_runner.h"
#include "base/test/bind.h"
#include "base/test/scoped_feature_list.h"
#include "components/exo/buffer.h"
#include "components/exo/data_exchange_delegate.h"
#include "components/exo/data_source.h"
#include "components/exo/data_source_delegate.h"
#include "components/exo/shell_surface.h"
#include "components/exo/surface.h"
#include "components/exo/test/exo_test_base.h"
#include "components/exo/test/exo_test_data_exchange_delegate.h"
#include "components/exo/test/shell_surface_builder.h"
#include "components/exo/test/test_data_source_delegate.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "third_party/abseil-cpp/absl/types/variant.h"
#include "ui/aura/client/drag_drop_client.h"
#include "ui/base/data_transfer_policy/data_transfer_endpoint.h"
#include "ui/base/data_transfer_policy/data_transfer_policy_controller.h"
#include "ui/base/dragdrop/mojom/drag_drop_types.mojom-shared.h"
#include "ui/events/test/event_generator.h"
#include "ui/gfx/geometry/point_f.h"
#include "url/gurl.h"
namespace exo {
namespace {
using test::TestDataSourceDelegate;
using ::testing::_;
using ::testing::NiceMock;
using ::testing::Property;
using ::testing::Return;
constexpr char kTextMimeType[] = "text/plain";
constexpr char kWindowDragMimeType[] = "chromium/x-window-drag";
} // namespace
class DragDropOperationTest : public test::ExoTestBase,
public aura::client::DragDropClientObserver {
public:
DragDropOperationTest() = default;
~DragDropOperationTest() override = default;
DragDropOperationTest(const DragDropOperationTest&) = delete;
DragDropOperationTest& operator=(const DragDropOperationTest&) = delete;
void SetUp() override {
test::ExoTestBase::SetUp();
aura::client::GetDragDropClient(ash::Shell::GetPrimaryRootWindow())
->AddObserver(this);
}
void TearDown() override {
aura::client::GetDragDropClient(ash::Shell::GetPrimaryRootWindow())
->RemoveObserver(this);
test::ExoTestBase::TearDown();
}
// aura::client::DragDropClientObserver:
void OnDragStarted() override {
drag_start_count_++;
if (!drag_blocked_callback_.is_null()) {
base::SequencedTaskRunner::GetCurrentDefault()->PostTask(
FROM_HERE, std::move(drag_blocked_callback_));
}
}
void OnDragCompleted(const ui::DropTargetEvent& event) override {
drag_end_count_++;
}
void OnDragCancelled() override { drag_end_count_++; }
protected:
void set_drag_blocked_callback(base::OnceClosure callback) {
drag_blocked_callback_ = std::move(callback);
}
int GetDragStartCountAndReset() {
int result = drag_start_count_;
drag_start_count_ = 0;
return result;
}
int GetDragEndCountAndReset() {
int result = drag_end_count_;
drag_end_count_ = 0;
return result;
}
private:
// Callback running inside the nested RunLoop in StartDragAndDrop().
base::OnceClosure drag_blocked_callback_;
int drag_start_count_ = 0;
int drag_end_count_ = 0;
};
TEST_F(DragDropOperationTest, DeleteDataSourceDuringDragging) {
TestDataExchangeDelegate data_exchange_delegate;
auto delegate = std::make_unique<TestDataSourceDelegate>();
auto data_source = std::make_unique<DataSource>(delegate.get());
data_source->Offer(kTextMimeType);
auto origin_surface = std::make_unique<Surface>();
ash::Shell::GetPrimaryRootWindow()->AddChild(origin_surface->window());
gfx::Size buffer_size(100, 100);
auto buffer = test::ExoTestHelper::CreateBuffer(buffer_size);
auto icon_surface = std::make_unique<Surface>();
icon_surface->Attach(buffer.get());
auto operation = DragDropOperation::Create(
&data_exchange_delegate, data_source.get(), origin_surface.get(),
icon_surface.get(), gfx::PointF(), ui::mojom::DragEventSource::kMouse);
icon_surface->Commit();
base::RunLoop run_loop;
set_drag_blocked_callback(base::BindOnce(
[](std::unique_ptr<DataSource> data_source,
base::WeakPtr<DragDropOperation> operation,
base::OnceClosure quit_closure) {
// This function runs inside the nested RunLoop in
// ash::DragDropController::StartDragAndDrop().
EXPECT_TRUE(operation);
// Deleting DataSource causes DragDropOperation to be deleted as well.
data_source.reset();
EXPECT_FALSE(operation);
std::move(quit_closure).Run();
},
std::move(data_source), operation, run_loop.QuitClosure()));
run_loop.Run();
EXPECT_FALSE(operation);
}
namespace {
class MockShellDelegate : public ash::TestShellDelegate {
public:
MockShellDelegate() = default;
~MockShellDelegate() override = default;
MOCK_METHOD(bool, IsTabDrag, (const ui::OSExchangeData&), (override));
};
} // namespace
class DragDropOperationTestWithWebUITabStripTest
: public DragDropOperationTest {
public:
DragDropOperationTestWithWebUITabStripTest() {}
// DragDropOperationTest:
void SetUp() override {
auto mock_shell_delegate = std::make_unique<NiceMock<MockShellDelegate>>();
mock_shell_delegate_ = mock_shell_delegate.get();
ExoTestBase::SetUp(std::move(mock_shell_delegate));
aura::client::GetDragDropClient(ash::Shell::GetPrimaryRootWindow())
->AddObserver(this);
}
MockShellDelegate* mock_shell_delegate() { return mock_shell_delegate_; }
private:
raw_ptr<NiceMock<MockShellDelegate>, DanglingUntriaged> mock_shell_delegate_ =
nullptr;
base::test::ScopedFeatureList scoped_feature_list_;
};
TEST_F(DragDropOperationTestWithWebUITabStripTest,
DeleteSurfaceDuringDragging) {
TestDataExchangeDelegate data_exchange_delegate;
auto delegate = std::make_unique<TestDataSourceDelegate>();
auto data_source = std::make_unique<DataSource>(delegate.get());
data_source->Offer(kWindowDragMimeType);
delegate->SetData(kWindowDragMimeType, std::string());
ON_CALL(*mock_shell_delegate(), IsTabDrag(_)).WillByDefault(Return(true));
auto shell_surface =
test::ShellSurfaceBuilder({100, 100}).BuildShellSurface();
auto* origin_surface = shell_surface->surface_for_testing();
gfx::Size buffer_size(100, 100);
auto buffer = test::ExoTestHelper::CreateBuffer(buffer_size);
auto icon_surface = std::make_unique<Surface>();
icon_surface->Attach(buffer.get());
auto operation = DragDropOperation::Create(
&data_exchange_delegate, data_source.get(), origin_surface,
icon_surface.get(), gfx::PointF(), ui::mojom::DragEventSource::kMouse);
icon_surface->Commit();
base::RunLoop run_loop;
set_drag_blocked_callback(base::BindOnce(
[](std::unique_ptr<DataSource> data_source,
std::unique_ptr<ShellSurface> shell_surface,
base::WeakPtr<DragDropOperation> operation,
base::OnceClosure quit_closure) {
// This function runs inside the nested RunLoop in
// ash::DragDropController::StartDragAndDrop().
EXPECT_TRUE(operation);
// Deleting ShellSurface causes DragDropOperation to be deleted as well.
shell_surface.reset();
EXPECT_FALSE(operation);
std::move(quit_closure).Run();
},
std::move(data_source), std::move(shell_surface), operation,
run_loop.QuitClosure()));
run_loop.Run();
EXPECT_FALSE(operation);
}
TEST_F(DragDropOperationTest, DragDropFromPopup) {
static_cast<ash::DragDropController*>(
aura::client::GetDragDropClient(ash::Shell::GetPrimaryRootWindow()))
->set_should_block_during_drag_drop(false);
TestDataExchangeDelegate data_exchange_delegate;
auto delegate = std::make_unique<TestDataSourceDelegate>();
auto data_source = std::make_unique<DataSource>(delegate.get());
data_source->Offer(kTextMimeType);
auto origin_shell_surface = test::ShellSurfaceBuilder(gfx::Size(100, 100))
.SetNoCommit()
.BuildShellSurface();
origin_shell_surface->SetPopup();
origin_shell_surface->Grab();
int closed_count = 0;
origin_shell_surface->set_close_callback(
base::BindLambdaForTesting([&]() { closed_count++; }));
auto* origin_surface = origin_shell_surface->root_surface();
origin_surface->Commit();
gfx::Size buffer_size(32, 32);
auto buffer = test::ExoTestHelper::CreateBuffer(buffer_size);
auto icon_surface = std::make_unique<Surface>();
icon_surface->Attach(buffer.get());
base::RunLoop run_loop;
set_drag_blocked_callback(run_loop.QuitClosure());
ui::test::EventGenerator generator(origin_surface->window()->GetRootWindow(),
origin_surface->window());
generator.PressLeftButton();
gfx::Point location =
generator.current_screen_location() -
origin_surface->window()->GetBoundsInScreen().OffsetFromOrigin();
auto operation = DragDropOperation::Create(
&data_exchange_delegate, data_source.get(),
origin_shell_surface->root_surface(), icon_surface.get(),
gfx::PointF(location), ui::mojom::DragEventSource::kMouse);
icon_surface->Commit();
run_loop.Run();
EXPECT_EQ(1, GetDragStartCountAndReset());
EXPECT_EQ(0, GetDragEndCountAndReset());
EXPECT_EQ(0, closed_count);
generator.MoveMouseBy(150, 150);
EXPECT_EQ(0, GetDragStartCountAndReset());
EXPECT_EQ(0, GetDragEndCountAndReset());
generator.ReleaseLeftButton();
EXPECT_EQ(0, GetDragStartCountAndReset());
EXPECT_EQ(1, GetDragEndCountAndReset());
}
TEST_F(DragDropOperationTest, DragDropFromNestedPopup) {
static_cast<ash::DragDropController*>(
aura::client::GetDragDropClient(ash::Shell::GetPrimaryRootWindow()))
->set_should_block_during_drag_drop(false);
TestDataExchangeDelegate data_exchange_delegate;
auto parent_shell_surface = test::ShellSurfaceBuilder(gfx::Size(100, 100))
.SetNoCommit()
.BuildShellSurface();
parent_shell_surface->SetPopup();
parent_shell_surface->Grab();
parent_shell_surface->root_surface()->Commit();
auto delegate = std::make_unique<TestDataSourceDelegate>();
auto data_source = std::make_unique<DataSource>(delegate.get());
data_source->Offer(kTextMimeType);
auto origin_shell_surface = test::ShellSurfaceBuilder(gfx::Size(100, 100))
.SetNoCommit()
.SetParent(parent_shell_surface.get())
.BuildShellSurface();
origin_shell_surface->SetPopup();
origin_shell_surface->Grab();
auto* origin_surface = origin_shell_surface->root_surface();
origin_surface->Commit();
gfx::Size buffer_size(32, 32);
auto buffer = test::ExoTestHelper::CreateBuffer(buffer_size);
auto icon_surface = std::make_unique<Surface>();
icon_surface->Attach(buffer.get());
base::RunLoop run_loop;
set_drag_blocked_callback(run_loop.QuitClosure());
ui::test::EventGenerator generator(origin_surface->window()->GetRootWindow(),
origin_surface->window());
generator.PressLeftButton();
gfx::Point location =
generator.current_screen_location() -
origin_surface->window()->GetBoundsInScreen().OffsetFromOrigin();
auto operation = DragDropOperation::Create(
&data_exchange_delegate, data_source.get(),
origin_shell_surface->root_surface(), icon_surface.get(),
gfx::PointF(location), ui::mojom::DragEventSource::kMouse);
icon_surface->Commit();
run_loop.Run();
EXPECT_EQ(1, GetDragStartCountAndReset());
EXPECT_EQ(0, GetDragEndCountAndReset());
generator.MoveMouseBy(150, 150);
EXPECT_EQ(0, GetDragStartCountAndReset());
EXPECT_EQ(0, GetDragEndCountAndReset());
// Hide the origin_shell_surface -- that will update the capture but shouldn't
// stop the drag&drop session. Note: do not remove |origin_shell_surface|
// itself, as the testing shell surface will also remove the root surface
// which will stop the drag&drop while root surface remains normally.
origin_shell_surface->GetWidget()->Hide();
EXPECT_EQ(0, GetDragStartCountAndReset());
EXPECT_EQ(0, GetDragEndCountAndReset());
generator.ReleaseLeftButton();
EXPECT_EQ(0, GetDragStartCountAndReset());
EXPECT_EQ(1, GetDragEndCountAndReset());
}
namespace {
class MockDataTransferPolicyController
: public ui::DataTransferPolicyController {
public:
MOCK_METHOD3(IsClipboardReadAllowed,
bool(base::optional_ref<const ui::DataTransferEndpoint> data_src,
base::optional_ref<const ui::DataTransferEndpoint> data_dst,
const std::optional<size_t> size));
MOCK_METHOD5(
PasteIfAllowed,
void(base::optional_ref<const ui::DataTransferEndpoint> data_src,
base::optional_ref<const ui::DataTransferEndpoint> data_dst,
absl::variant<size_t, std::vector<base::FilePath>> pasted_content,
content::RenderFrameHost* rfh,
base::OnceCallback<void(bool)> callback));
MOCK_METHOD4(DropIfAllowed,
void(std::optional<ui::DataTransferEndpoint> data_src,
std::optional<ui::DataTransferEndpoint> data_dst,
std::optional<std::vector<ui::FileInfo>> filenames,
base::OnceClosure drop_cb));
};
} // namespace
// Lacros sends additional metadata about the drag and drop source (e.g. origin
// URL). This synchronizes the source metadata between Lacros to Ash. This is
// used in Data Leak Prevention restrictions where admins can restrict data from
// being copied from restricted locations.
TEST_F(DragDropOperationTest, DragDropCheckSourceFromLacros) {
static_cast<ash::DragDropController*>(
aura::client::GetDragDropClient(ash::Shell::GetPrimaryRootWindow()))
->set_should_block_during_drag_drop(false);
TestDataExchangeDelegate data_exchange_delegate;
data_exchange_delegate.set_endpoint_type(ui::EndpointType::kLacros);
auto delegate = std::make_unique<TestDataSourceDelegate>();
auto data_source = std::make_unique<DataSource>(delegate.get());
auto dlp_controller = std::make_unique<MockDataTransferPolicyController>();
// Encoded source DataTransferEndpoint.
const std::string kEncodedTestDte =
R"({"endpoint_type":"url","url":"https://www.google.com"})";
const std::string kDteMimeType = "chromium/x-data-transfer-endpoint";
data_source->Offer(kDteMimeType);
delegate->SetData(kDteMimeType, kEncodedTestDte);
auto origin_surface = std::make_unique<Surface>();
ash::Shell::GetPrimaryRootWindow()->AddChild(origin_surface->window());
gfx::Size buffer_size(100, 100);
auto buffer = test::ExoTestHelper::CreateBuffer(buffer_size);
auto icon_surface = std::make_unique<Surface>();
icon_surface->Attach(buffer.get());
// Expect the encoded endpoint from Lacros to be correctly parsed.
EXPECT_CALL(*dlp_controller, DropIfAllowed)
.WillOnce([&](std::optional<ui::DataTransferEndpoint> data_src,
std::optional<ui::DataTransferEndpoint> data_dst,
std::optional<std::vector<ui::FileInfo>> filenames,
base::OnceClosure drop_cb) {
ASSERT_TRUE(data_src.has_value());
ASSERT_TRUE(data_src->IsUrlType());
EXPECT_EQ(data_src->GetURL()->spec(), "https://www.google.com/");
std::move(drop_cb).Run();
});
base::RunLoop run_loop;
set_drag_blocked_callback(run_loop.QuitClosure());
ui::test::EventGenerator generator(origin_surface->window()->GetRootWindow(),
origin_surface->window());
generator.PressLeftButton();
gfx::Point location =
generator.current_screen_location() -
origin_surface->window()->GetBoundsInScreen().OffsetFromOrigin();
auto operation = DragDropOperation::Create(
&data_exchange_delegate, data_source.get(), origin_surface.get(),
icon_surface.get(), gfx::PointF(location),
ui::mojom::DragEventSource::kMouse);
icon_surface->Commit();
run_loop.Run();
generator.MoveMouseBy(150, 150);
generator.ReleaseLeftButton();
::testing::Mock::VerifyAndClearExpectations(dlp_controller.get());
}
// Additional source metadata should be ignored from non-Lacros instances.
TEST_F(DragDropOperationTest, DragDropCheckSourceFromNonLacros) {
static_cast<ash::DragDropController*>(
aura::client::GetDragDropClient(ash::Shell::GetPrimaryRootWindow()))
->set_should_block_during_drag_drop(false);
TestDataExchangeDelegate data_exchange_delegate;
data_exchange_delegate.set_endpoint_type(ui::EndpointType::kCrostini);
auto delegate = std::make_unique<TestDataSourceDelegate>();
auto data_source = std::make_unique<DataSource>(delegate.get());
auto dlp_controller = std::make_unique<MockDataTransferPolicyController>();
// Encoded source DataTransferEndpoint.
const std::string kEncodedTestDte =
R"({"endpoint_type":"url","url":"https://www.google.com"})";
const std::string kDteMimeType = "chromium/x-data-transfer-endpoint";
data_source->Offer(kDteMimeType);
delegate->SetData(kDteMimeType, kEncodedTestDte);
auto origin_surface = std::make_unique<Surface>();
ash::Shell::GetPrimaryRootWindow()->AddChild(origin_surface->window());
gfx::Size buffer_size(100, 100);
auto buffer = test::ExoTestHelper::CreateBuffer(buffer_size);
auto icon_surface = std::make_unique<Surface>();
icon_surface->Attach(buffer.get());
// Expect the encoded endpoint from non-Lacros to be ignored.
EXPECT_CALL(*dlp_controller, DropIfAllowed)
.WillOnce([&](std::optional<ui::DataTransferEndpoint> data_src,
std::optional<ui::DataTransferEndpoint> data_dst,
std::optional<std::vector<ui::FileInfo>> filenames,
base::OnceClosure drop_cb) {
ASSERT_TRUE(data_src.has_value());
EXPECT_FALSE(data_src->IsUrlType());
EXPECT_EQ(data_src->type(), ui::EndpointType::kCrostini);
std::move(drop_cb).Run();
});
base::RunLoop run_loop;
set_drag_blocked_callback(run_loop.QuitClosure());
ui::test::EventGenerator generator(origin_surface->window()->GetRootWindow(),
origin_surface->window());
generator.PressLeftButton();
gfx::Point location =
generator.current_screen_location() -
origin_surface->window()->GetBoundsInScreen().OffsetFromOrigin();
auto operation = DragDropOperation::Create(
&data_exchange_delegate, data_source.get(), origin_surface.get(),
icon_surface.get(), gfx::PointF(location),
ui::mojom::DragEventSource::kMouse);
icon_surface->Commit();
run_loop.Run();
generator.MoveMouseBy(150, 150);
generator.ReleaseLeftButton();
::testing::Mock::VerifyAndClearExpectations(dlp_controller.get());
}
} // namespace exo