blob: 8611af6be170abc290785412121bcd571394a01f [file] [log] [blame]
// Copyright 2018 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/media/webrtc/desktop_capture_access_handler.h"
#include <memory>
#include <string>
#include <utility>
#include "base/command_line.h"
#include "base/functional/bind.h"
#include "base/memory/raw_ptr.h"
#include "base/run_loop.h"
#include "base/types/expected.h"
#include "build/build_config.h"
#include "chrome/browser/media/webrtc/fake_desktop_media_picker_factory.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/test/base/chrome_render_view_host_test_harness.h"
#include "content/public/browser/browser_context.h"
#include "content/public/browser/desktop_media_id.h"
#include "content/public/browser/desktop_streams_registry.h"
#include "content/public/browser/media_stream_request.h"
#include "content/public/browser/render_frame_host.h"
#include "content/public/browser/render_process_host.h"
#include "content/public/browser/web_contents.h"
#include "content/public/common/content_switches.h"
#include "extensions/common/extension.h"
#include "extensions/common/extension_builder.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/blink/public/common/mediastream/media_stream_request.h"
#include "third_party/blink/public/mojom/mediastream/media_stream.mojom-shared.h"
#include "third_party/blink/public/mojom/mediastream/media_stream.mojom.h"
#if BUILDFLAG(IS_CHROMEOS)
#include "chrome/browser/chromeos/policy/dlp/test/mock_dlp_content_manager.h"
#include "ui/aura/window.h"
#endif // BUILDFLAG(IS_CHROMEOS)
#if BUILDFLAG(IS_MAC)
#include "chrome/browser/permissions/system/system_media_capture_permissions_mac.h"
#endif
constexpr char kOrigin[] = "https://origin/";
constexpr char kComponentExtension[] = "Component Extension";
class DesktopCaptureAccessHandlerTest : public ChromeRenderViewHostTestHarness {
public:
DesktopCaptureAccessHandlerTest() = default;
~DesktopCaptureAccessHandlerTest() override = default;
void SetUp() override {
ChromeRenderViewHostTestHarness::SetUp();
auto picker_factory = std::make_unique<FakeDesktopMediaPickerFactory>();
picker_factory_ = picker_factory.get();
access_handler_ = std::make_unique<DesktopCaptureAccessHandler>(
std::move(picker_factory));
}
void ProcessGenerateStreamRequest(
const std::vector<std::string>& requested_video_device_ids,
const GURL& origin,
const extensions::Extension* extension,
blink::mojom::MediaStreamRequestResult* request_result,
blink::mojom::StreamDevices* devices_result,
bool expect_result = true) {
#if BUILDFLAG(IS_MAC)
system_permission_settings::SetIsScreenCaptureAllowedForTesting(true);
#endif
content::MediaStreamRequest request(
web_contents()->GetPrimaryMainFrame()->GetProcess()->GetDeprecatedID(),
web_contents()->GetPrimaryMainFrame()->GetRoutingID(),
/*page_request_id=*/0, url::Origin::Create(origin),
/*user_gesture=*/false, blink::MEDIA_GENERATE_STREAM,
/*requested_audio_device_ids=*/{}, requested_video_device_ids,
blink::mojom::MediaStreamType::NO_SERVICE,
blink::mojom::MediaStreamType::GUM_DESKTOP_VIDEO_CAPTURE,
/*disable_local_echo=*/false,
/*request_pan_tilt_zoom_permission=*/false,
/*captured_surface_control_active=*/false);
base::RunLoop wait_loop;
content::MediaResponseCallback callback = base::BindOnce(
[](base::RunLoop* wait_loop, bool expect_result,
blink::mojom::MediaStreamRequestResult* request_result,
blink::mojom::StreamDevices* devices_result,
const blink::mojom::StreamDevicesSet& stream_devices_set,
blink::mojom::MediaStreamRequestResult result,
std::unique_ptr<content::MediaStreamUI> ui) {
*request_result = result;
if (result == blink::mojom::MediaStreamRequestResult::OK) {
ASSERT_EQ(stream_devices_set.stream_devices.size(), 1u);
*devices_result = *stream_devices_set.stream_devices[0];
} else {
ASSERT_TRUE(stream_devices_set.stream_devices.empty());
*devices_result = blink::mojom::StreamDevices();
}
EXPECT_TRUE(expect_result) << "MediaResponseCallback should not be "
"called when expect_result is false.";
wait_loop->Quit();
},
&wait_loop, expect_result, request_result, devices_result);
access_handler_->HandleRequest(web_contents(), request, std::move(callback),
extension);
if (expect_result) {
wait_loop.Run();
} else {
wait_loop.RunUntilIdle();
}
}
void ProcessDeviceUpdateRequest(
const base::expected<content::DesktopMediaID,
blink::mojom::MediaStreamRequestResult>& response,
blink::mojom::MediaStreamRequestResult* request_result,
blink::mojom::StreamDevices* stream_devices_result,
blink::MediaStreamRequestType request_type,
bool request_audio) {
FakeDesktopMediaPickerFactory::TestFlags test_flags[] = {
{.expect_screens = false,
.expect_windows = false,
.expect_tabs = true,
.expect_current_tab = false,
.expect_audio = request_audio,
.picker_result = response}};
picker_factory_->SetTestFlags(test_flags, std::size(test_flags));
blink::mojom::MediaStreamType audio_type =
request_audio ? blink::mojom::MediaStreamType::GUM_DESKTOP_AUDIO_CAPTURE
: blink::mojom::MediaStreamType::NO_SERVICE;
content::MediaStreamRequest request(
0, 0, 0, url::Origin::Create(GURL(kOrigin)), false, request_type,
/*requested_audio_device_ids=*/{},
/*requested_video_device_ids=*/{}, audio_type,
blink::mojom::MediaStreamType::GUM_DESKTOP_VIDEO_CAPTURE,
/*disable_local_echo=*/false,
/*request_pan_tilt_zoom_permission=*/false,
/*captured_surface_control_active=*/false);
base::RunLoop wait_loop;
content::MediaResponseCallback callback = base::BindOnce(
[](base::RunLoop* wait_loop,
blink::mojom::MediaStreamRequestResult* request_result,
blink::mojom::StreamDevices* devices_result,
const blink::mojom::StreamDevicesSet& stream_devices_set,
blink::mojom::MediaStreamRequestResult result,
std::unique_ptr<content::MediaStreamUI> ui) {
*request_result = result;
if (!stream_devices_set.stream_devices.empty()) {
*devices_result = *stream_devices_set.stream_devices[0];
} else {
*devices_result = blink::mojom::StreamDevices();
}
wait_loop->Quit();
},
&wait_loop, request_result, stream_devices_result);
access_handler_->HandleRequest(web_contents(), request, std::move(callback),
nullptr /* extension */);
wait_loop.Run();
EXPECT_TRUE(test_flags[0].picker_created);
access_handler_.reset();
EXPECT_TRUE(test_flags[0].picker_deleted);
}
void NotifyWebContentsDestroyed() {
access_handler_->WebContentsDestroyed(web_contents());
}
const DesktopCaptureAccessHandler::RequestsQueues& GetRequestQueues() {
return access_handler_->pending_requests_;
}
#if BUILDFLAG(IS_CHROMEOS)
void SetPrimaryRootWindow(aura::Window* window) {
access_handler_->primary_root_window_for_testing_ = window;
}
#endif // BUILDFLAG(IS_CHROMEOS)
protected:
raw_ptr<FakeDesktopMediaPickerFactory, DanglingUntriaged> picker_factory_;
std::unique_ptr<DesktopCaptureAccessHandler> access_handler_;
};
TEST_F(DesktopCaptureAccessHandlerTest,
ChangeSourceWithoutAudioRequestPermissionGiven) {
blink::mojom::MediaStreamRequestResult result;
blink::mojom::StreamDevices stream_devices;
ProcessDeviceUpdateRequest(
content::DesktopMediaID(content::DesktopMediaID::TYPE_SCREEN,
content::DesktopMediaID::kFakeId),
&result, &stream_devices, blink::MEDIA_DEVICE_UPDATE,
false /*request_audio*/);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::OK, result);
EXPECT_EQ(1u, blink::CountDevices(stream_devices));
EXPECT_EQ(blink::mojom::MediaStreamType::GUM_DESKTOP_VIDEO_CAPTURE,
stream_devices.video_device.value().type);
}
TEST_F(DesktopCaptureAccessHandlerTest,
ChangeSourceWithAudioRequestPermissionGiven) {
blink::mojom::MediaStreamRequestResult result;
blink::mojom::StreamDevices stream_devices;
ProcessDeviceUpdateRequest(
content::DesktopMediaID(content::DesktopMediaID::TYPE_SCREEN,
content::DesktopMediaID::kFakeId,
true /* audio_share */),
&result, &stream_devices, blink::MEDIA_DEVICE_UPDATE,
true /* request_audio */);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::OK, result);
EXPECT_EQ(2u, blink::CountDevices(stream_devices));
EXPECT_EQ(blink::mojom::MediaStreamType::GUM_DESKTOP_AUDIO_CAPTURE,
stream_devices.audio_device.value().type);
EXPECT_EQ(blink::mojom::MediaStreamType::GUM_DESKTOP_VIDEO_CAPTURE,
stream_devices.video_device.value().type);
}
TEST_F(DesktopCaptureAccessHandlerTest, ChangeSourcePermissionDenied) {
blink::mojom::MediaStreamRequestResult result;
blink::mojom::StreamDevices stream_devices;
ProcessDeviceUpdateRequest(
base::unexpected(
blink::mojom::MediaStreamRequestResult::PERMISSION_DENIED),
&result, &stream_devices, blink::MEDIA_DEVICE_UPDATE,
false /*request audio*/);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::PERMISSION_DENIED, result);
EXPECT_EQ(0u, blink::CountDevices(stream_devices));
}
TEST_F(DesktopCaptureAccessHandlerTest,
ChangeSourceUpdateMediaRequestStateWithClosing) {
const int render_process_id = 0;
const int render_frame_id = 0;
const int page_request_id = 0;
const blink::mojom::MediaStreamType stream_type =
blink::mojom::MediaStreamType::GUM_DESKTOP_VIDEO_CAPTURE;
FakeDesktopMediaPickerFactory::TestFlags test_flags[] = {
{false /* expect_screens */, false /* expect_windows*/,
true /* expect_tabs */, false /* expect_current_tab */,
false /* expect_audio */, content::DesktopMediaID(),
true /* cancelled */}};
picker_factory_->SetTestFlags(test_flags, std::size(test_flags));
content::MediaStreamRequest request(
render_process_id, render_frame_id, page_request_id,
url::Origin::Create(GURL(kOrigin)), false, blink::MEDIA_DEVICE_UPDATE,
/*requested_audio_device_ids=*/{}, /*requested_video_device_ids=*/{},
blink::mojom::MediaStreamType::NO_SERVICE, stream_type,
/*disable_local_echo=*/false, /*request_pan_tilt_zoom_permission=*/false,
/*captured_surface_control_active=*/false);
content::MediaResponseCallback callback;
access_handler_->HandleRequest(web_contents(), request, std::move(callback),
nullptr /* extension */);
EXPECT_TRUE(test_flags[0].picker_created);
EXPECT_EQ(1u, GetRequestQueues().size());
auto queue_it = GetRequestQueues().find(web_contents());
EXPECT_TRUE(queue_it != GetRequestQueues().end());
EXPECT_EQ(1u, queue_it->second.size());
access_handler_->UpdateMediaRequestState(
render_process_id, render_frame_id, page_request_id, stream_type,
content::MEDIA_REQUEST_STATE_CLOSING);
EXPECT_EQ(1u, GetRequestQueues().size());
queue_it = GetRequestQueues().find(web_contents());
EXPECT_TRUE(queue_it != GetRequestQueues().end());
EXPECT_EQ(0u, queue_it->second.size());
EXPECT_TRUE(test_flags[0].picker_deleted);
access_handler_.reset();
}
TEST_F(DesktopCaptureAccessHandlerTest, ChangeSourceWebContentsDestroyed) {
FakeDesktopMediaPickerFactory::TestFlags test_flags[] = {
{false /* expect_screens */, false /* expect_windows*/,
true /* expect_tabs */, false /* expect_current_tab */,
false /* expect_audio */, content::DesktopMediaID(),
true /* cancelled */}};
picker_factory_->SetTestFlags(test_flags, std::size(test_flags));
content::MediaStreamRequest request(
0, 0, 0, url::Origin::Create(GURL(kOrigin)), false,
blink::MEDIA_DEVICE_UPDATE, /*requested_audio_device_ids=*/{},
/*requested_video_device_ids=*/{},
blink::mojom::MediaStreamType::NO_SERVICE,
blink::mojom::MediaStreamType::GUM_DESKTOP_VIDEO_CAPTURE,
/*disable_local_echo=*/false, /*request_pan_tilt_zoom_permission=*/false,
/*captured_surface_control_active=*/false);
content::MediaResponseCallback callback;
access_handler_->HandleRequest(web_contents(), request, std::move(callback),
nullptr /* extension */);
EXPECT_TRUE(test_flags[0].picker_created);
EXPECT_EQ(1u, GetRequestQueues().size());
auto queue_it = GetRequestQueues().find(web_contents());
EXPECT_TRUE(queue_it != GetRequestQueues().end());
EXPECT_EQ(1u, queue_it->second.size());
NotifyWebContentsDestroyed();
EXPECT_EQ(0u, GetRequestQueues().size());
access_handler_.reset();
}
TEST_F(DesktopCaptureAccessHandlerTest, ChangeSourceMultipleRequests) {
FakeDesktopMediaPickerFactory::TestFlags test_flags[] = {
{false /* expect_screens */, false /* expect_windows*/,
true /* expect_tabs */, false /* expect_current_tab */,
false /* expect_audio */,
content::DesktopMediaID(
content::DesktopMediaID::TYPE_SCREEN,
content::DesktopMediaID::kFakeId) /* selected_source */},
{false /* expect_screens */, false /* expect_windows*/,
true /* expect_tabs */, false /* expect_current_tab */,
false /* expect_audio */,
content::DesktopMediaID(
content::DesktopMediaID::TYPE_WINDOW,
content::DesktopMediaID::kNullId) /* selected_source */}};
const size_t kTestFlagCount = 2;
picker_factory_->SetTestFlags(test_flags, kTestFlagCount);
blink::mojom::MediaStreamRequestResult result;
blink::MediaStreamDevices devices;
base::RunLoop wait_loop[kTestFlagCount];
for (base::RunLoop& loop : wait_loop) {
content::MediaStreamRequest request(
0, 0, 0, url::Origin::Create(GURL(kOrigin)), false,
blink::MEDIA_DEVICE_UPDATE, /*requested_audio_device_ids=*/{},
/*requested_video_device_ids=*/{},
blink::mojom::MediaStreamType::NO_SERVICE,
blink::mojom::MediaStreamType::GUM_DESKTOP_VIDEO_CAPTURE,
/*disable_local_echo=*/false,
/*request_pan_tilt_zoom_permission=*/false,
/*captured_surface_control_active=*/false);
content::MediaResponseCallback callback = base::BindOnce(
[](base::RunLoop* wait_loop,
blink::mojom::MediaStreamRequestResult* request_result,
blink::MediaStreamDevices* devices_result,
const blink::mojom::StreamDevicesSet& stream_devices_set,
blink::mojom::MediaStreamRequestResult result,
std::unique_ptr<content::MediaStreamUI> ui) {
ASSERT_EQ(stream_devices_set.stream_devices.size(), 1u);
*request_result = result;
*devices_result = blink::ToMediaStreamDevicesList(stream_devices_set);
wait_loop->Quit();
},
&loop, &result, &devices);
access_handler_->HandleRequest(web_contents(), request, std::move(callback),
nullptr /* extension */);
}
wait_loop[0].Run();
EXPECT_TRUE(test_flags[0].picker_created);
EXPECT_TRUE(test_flags[0].picker_deleted);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::OK, result);
EXPECT_EQ(1u, devices.size());
EXPECT_EQ(blink::mojom::MediaStreamType::GUM_DESKTOP_VIDEO_CAPTURE,
devices[0].type);
blink::MediaStreamDevice first_device = devices[0];
EXPECT_TRUE(test_flags[1].picker_created);
EXPECT_FALSE(test_flags[1].picker_deleted);
wait_loop[1].Run();
EXPECT_TRUE(test_flags[1].picker_deleted);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::OK, result);
EXPECT_EQ(1u, devices.size());
EXPECT_EQ(blink::mojom::MediaStreamType::GUM_DESKTOP_VIDEO_CAPTURE,
devices[0].type);
EXPECT_FALSE(devices[0].IsSameDevice(first_device));
access_handler_.reset();
}
TEST_F(DesktopCaptureAccessHandlerTest, GenerateStreamSuccess) {
blink::mojom::MediaStreamRequestResult result;
blink::mojom::StreamDevices devices;
const GURL origin(kOrigin);
const std::string id =
content::DesktopStreamsRegistry::GetInstance()->RegisterStream(
web_contents()
->GetPrimaryMainFrame()
->GetProcess()
->GetDeprecatedID(),
web_contents()->GetPrimaryMainFrame()->GetRoutingID(),
url::Origin::Create(origin),
content::DesktopMediaID(content::DesktopMediaID::TYPE_SCREEN,
content::DesktopMediaID::kFakeId),
content::DesktopStreamRegistryType::kRegistryStreamTypeDesktop);
ProcessGenerateStreamRequest({id}, origin, /*extension=*/nullptr, &result,
&devices);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::OK, result);
EXPECT_TRUE(devices.video_device.has_value());
}
TEST_F(DesktopCaptureAccessHandlerTest, ScreenCaptureAccessSuccess) {
base::CommandLine::ForCurrentProcess()->AppendSwitch(
switches::kEnableUserMediaScreenCapturing);
base::CommandLine::ForCurrentProcess()->AppendSwitch(
switches::kAllowHttpScreenCapture);
extensions::ExtensionBuilder extensionBuilder(kComponentExtension);
extensionBuilder.SetLocation(extensions::mojom::ManifestLocation::kComponent);
#if BUILDFLAG(IS_CHROMEOS)
std::unique_ptr<aura::Window> primary_root_window =
std::make_unique<aura::Window>(/*delegate=*/nullptr);
primary_root_window->Init(ui::LAYER_NOT_DRAWN);
SetPrimaryRootWindow(primary_root_window.get());
#endif // BUILDFLAG(IS_CHROMEOS)
blink::mojom::MediaStreamRequestResult result;
blink::mojom::StreamDevices devices;
ProcessGenerateStreamRequest(/*requested_video_device_ids=*/{}, GURL(kOrigin),
extensionBuilder.Build().get(), &result,
&devices);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::OK, result);
EXPECT_TRUE(devices.video_device.has_value());
}
#if BUILDFLAG(IS_CHROMEOS)
TEST_F(DesktopCaptureAccessHandlerTest, ScreenCaptureAccessDlpRestricted) {
// Setup Data Leak Prevention restriction.
policy::MockDlpContentManager mock_dlp_content_manager;
policy::ScopedDlpContentObserverForTesting scoped_dlp_content_manager(
&mock_dlp_content_manager);
EXPECT_CALL(mock_dlp_content_manager, CheckScreenShareRestriction)
.WillOnce([&](const content::DesktopMediaID& media_id,
const std::u16string& application_title,
base::OnceCallback<void(bool)> callback) {
std::move(callback).Run(/*should_proceed=*/false);
});
base::CommandLine::ForCurrentProcess()->AppendSwitch(
switches::kEnableUserMediaScreenCapturing);
extensions::ExtensionBuilder extensionBuilder(kComponentExtension);
extensionBuilder.SetLocation(extensions::mojom::ManifestLocation::kComponent);
std::unique_ptr<aura::Window> primary_root_window =
std::make_unique<aura::Window>(/*delegate=*/nullptr);
primary_root_window->Init(ui::LAYER_NOT_DRAWN);
SetPrimaryRootWindow(primary_root_window.get());
blink::mojom::MediaStreamRequestResult result =
blink::mojom::MediaStreamRequestResult::NOT_SUPPORTED;
blink::mojom::StreamDevices devices;
ProcessGenerateStreamRequest(/*requested_video_device_ids=*/{}, GURL(kOrigin),
extensionBuilder.Build().get(), &result,
&devices);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::PERMISSION_DENIED, result);
EXPECT_FALSE(devices.video_device.has_value());
}
TEST_F(DesktopCaptureAccessHandlerTest, ScreenCaptureAccessDlpNotRestricted) {
// Setup Data Leak Prevention restriction.
policy::MockDlpContentManager mock_dlp_content_manager;
policy::ScopedDlpContentObserverForTesting scoped_dlp_content_manager(
&mock_dlp_content_manager);
EXPECT_CALL(mock_dlp_content_manager, CheckScreenShareRestriction)
.WillOnce([&](const content::DesktopMediaID& media_id,
const std::u16string& application_title,
base::OnceCallback<void(bool)> callback) {
std::move(callback).Run(/*should_proceed=*/true);
});
base::CommandLine::ForCurrentProcess()->AppendSwitch(
switches::kEnableUserMediaScreenCapturing);
extensions::ExtensionBuilder extensionBuilder(kComponentExtension);
extensionBuilder.SetLocation(extensions::mojom::ManifestLocation::kComponent);
std::unique_ptr<aura::Window> primary_root_window =
std::make_unique<aura::Window>(/*delegate=*/nullptr);
primary_root_window->Init(ui::LAYER_NOT_DRAWN);
SetPrimaryRootWindow(primary_root_window.get());
blink::mojom::MediaStreamRequestResult result =
blink::mojom::MediaStreamRequestResult::NOT_SUPPORTED;
blink::mojom::StreamDevices devices;
ProcessGenerateStreamRequest(/*requested_video_device_id=*/{}, GURL(kOrigin),
extensionBuilder.Build().get(), &result,
&devices);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::OK, result);
EXPECT_TRUE(devices.video_device.has_value());
}
TEST_F(DesktopCaptureAccessHandlerTest,
ScreenCaptureAccessDlpWebContentsDestroyed) {
// Setup Data Leak Prevention restriction.
policy::MockDlpContentManager mock_dlp_content_manager;
policy::ScopedDlpContentObserverForTesting scoped_dlp_content_manager(
&mock_dlp_content_manager);
EXPECT_CALL(mock_dlp_content_manager, CheckScreenShareRestriction)
.Times(1)
.WillOnce([&](const content::DesktopMediaID& media_id,
const std::u16string& application_title,
base::OnceCallback<void(bool)> callback) {
DeleteContents();
std::move(callback).Run(/*should_proceed=*/false);
});
base::CommandLine::ForCurrentProcess()->AppendSwitch(
switches::kEnableUserMediaScreenCapturing);
extensions::ExtensionBuilder extensionBuilder(kComponentExtension);
extensionBuilder.SetLocation(extensions::mojom::ManifestLocation::kComponent);
std::unique_ptr<aura::Window> primary_root_window =
std::make_unique<aura::Window>(/*delegate=*/nullptr);
primary_root_window->Init(ui::LAYER_NOT_DRAWN);
SetPrimaryRootWindow(primary_root_window.get());
blink::mojom::MediaStreamRequestResult result =
blink::mojom::MediaStreamRequestResult::NOT_SUPPORTED;
blink::mojom::StreamDevices devices;
ProcessGenerateStreamRequest(/*requested_video_device_id=*/{}, GURL(kOrigin),
extensionBuilder.Build().get(), &result,
&devices, /*expect_result=*/false);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::NOT_SUPPORTED, result);
EXPECT_FALSE(devices.video_device.has_value());
}
TEST_F(DesktopCaptureAccessHandlerTest, GenerateStreamDlpRestricted) {
// Setup Data Leak Prevention restriction.
policy::MockDlpContentManager mock_dlp_content_manager;
policy::ScopedDlpContentObserverForTesting scoped_dlp_content_manager(
&mock_dlp_content_manager);
EXPECT_CALL(mock_dlp_content_manager, CheckScreenShareRestriction)
.Times(1)
.WillOnce([](const content::DesktopMediaID& media_id,
const std::u16string& application_title,
base::OnceCallback<void(bool)> callback) {
std::move(callback).Run(/*should_proceed=*/false);
});
const std::string id =
content::DesktopStreamsRegistry::GetInstance()->RegisterStream(
web_contents()
->GetPrimaryMainFrame()
->GetProcess()
->GetDeprecatedID(),
web_contents()->GetPrimaryMainFrame()->GetRoutingID(),
url::Origin::Create(GURL(kOrigin)),
content::DesktopMediaID(content::DesktopMediaID::TYPE_SCREEN,
content::DesktopMediaID::kFakeId),
content::DesktopStreamRegistryType::kRegistryStreamTypeDesktop);
blink::mojom::MediaStreamRequestResult result =
blink::mojom::MediaStreamRequestResult::NOT_SUPPORTED;
blink::mojom::StreamDevices devices;
ProcessGenerateStreamRequest({id}, GURL(kOrigin), /*extension=*/nullptr,
&result, &devices);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::PERMISSION_DENIED, result);
EXPECT_FALSE(devices.video_device.has_value());
}
TEST_F(DesktopCaptureAccessHandlerTest, GenerateStreamDlpNotRestricted) {
// Setup Data Leak Prevention restriction.
policy::MockDlpContentManager mock_dlp_content_manager;
policy::ScopedDlpContentObserverForTesting scoped_dlp_content_manager(
&mock_dlp_content_manager);
EXPECT_CALL(mock_dlp_content_manager, CheckScreenShareRestriction)
.Times(1)
.WillOnce([](const content::DesktopMediaID& media_id,
const std::u16string& application_title,
base::OnceCallback<void(bool)> callback) {
std::move(callback).Run(/*should_proceed=*/true);
});
const std::string id =
content::DesktopStreamsRegistry::GetInstance()->RegisterStream(
web_contents()
->GetPrimaryMainFrame()
->GetProcess()
->GetDeprecatedID(),
web_contents()->GetPrimaryMainFrame()->GetRoutingID(),
url::Origin::Create(GURL(kOrigin)),
content::DesktopMediaID(content::DesktopMediaID::TYPE_SCREEN,
content::DesktopMediaID::kFakeId),
content::DesktopStreamRegistryType::kRegistryStreamTypeDesktop);
blink::mojom::MediaStreamRequestResult result =
blink::mojom::MediaStreamRequestResult::NOT_SUPPORTED;
blink::mojom::StreamDevices devices;
ProcessGenerateStreamRequest({id}, GURL(kOrigin), /*extension=*/nullptr,
&result, &devices);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::OK, result);
EXPECT_TRUE(devices.video_device.has_value());
}
TEST_F(DesktopCaptureAccessHandlerTest, ChangeSourceDlpRestricted) {
// Setup Data Leak Prevention restriction.
policy::MockDlpContentManager mock_dlp_content_manager;
policy::ScopedDlpContentObserverForTesting scoped_dlp_content_manager(
&mock_dlp_content_manager);
EXPECT_CALL(mock_dlp_content_manager, CheckScreenShareRestriction)
.Times(1)
.WillOnce([](const content::DesktopMediaID& media_id,
const std::u16string& application_title,
base::OnceCallback<void(bool)> callback) {
std::move(callback).Run(/*should_proceed=*/false);
});
blink::mojom::MediaStreamRequestResult result =
blink::mojom::MediaStreamRequestResult::NOT_SUPPORTED;
blink::mojom::StreamDevices stream_devices;
ProcessDeviceUpdateRequest(
content::DesktopMediaID(content::DesktopMediaID::TYPE_SCREEN,
content::DesktopMediaID::kFakeId),
&result, &stream_devices, blink::MEDIA_DEVICE_UPDATE,
/*request audio=*/false);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::PERMISSION_DENIED, result);
EXPECT_EQ(0u, blink::CountDevices(stream_devices));
}
TEST_F(DesktopCaptureAccessHandlerTest, ChangeSourceDlpNotRestricted) {
// Setup Data Leak Prevention restriction.
policy::MockDlpContentManager mock_dlp_content_manager;
policy::ScopedDlpContentObserverForTesting scoped_dlp_content_manager(
&mock_dlp_content_manager);
EXPECT_CALL(mock_dlp_content_manager, CheckScreenShareRestriction)
.Times(1)
.WillOnce([](const content::DesktopMediaID& media_id,
const std::u16string& application_title,
base::OnceCallback<void(bool)> callback) {
std::move(callback).Run(/*should_proceed=*/true);
});
blink::mojom::MediaStreamRequestResult result =
blink::mojom::MediaStreamRequestResult::NOT_SUPPORTED;
blink::mojom::StreamDevices stream_devices;
ProcessDeviceUpdateRequest(
content::DesktopMediaID(content::DesktopMediaID::TYPE_SCREEN,
content::DesktopMediaID::kFakeId),
&result, &stream_devices, blink::MEDIA_DEVICE_UPDATE,
/*request audio=*/false);
EXPECT_EQ(blink::mojom::MediaStreamRequestResult::OK, result);
EXPECT_EQ(1u, blink::CountDevices(stream_devices));
}
#endif // BUILDFLAG(IS_CHROMEOS)