| Index: chrome/browser/usb/usb_browsertest.cc
|
| diff --git a/chrome/browser/usb/usb_browsertest.cc b/chrome/browser/usb/usb_browsertest.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..606f852cf1499886d35cd54d828ba4b8c879e062
|
| --- /dev/null
|
| +++ b/chrome/browser/usb/usb_browsertest.cc
|
| @@ -0,0 +1,201 @@
|
| +// Copyright 2016 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 <memory>
|
| +#include <string>
|
| +
|
| +#include "base/command_line.h"
|
| +#include "base/memory/ref_counted.h"
|
| +#include "chrome/browser/ui/browser.h"
|
| +#include "chrome/browser/ui/tabs/tab_strip_model.h"
|
| +#include "chrome/browser/usb/usb_chooser_controller.h"
|
| +#include "chrome/test/base/in_process_browser_test.h"
|
| +#include "chrome/test/base/ui_test_utils.h"
|
| +#include "content/public/browser/render_frame_host.h"
|
| +#include "content/public/common/content_switches.h"
|
| +#include "content/public/test/browser_test_utils.h"
|
| +#include "device/base/mock_device_client.h"
|
| +#include "device/usb/mock_usb_device.h"
|
| +#include "device/usb/mock_usb_service.h"
|
| +#include "device/usb/public/interfaces/chooser_service.mojom.h"
|
| +#include "device/usb/webusb_descriptors.h"
|
| +#include "mojo/public/cpp/bindings/strong_binding.h"
|
| +#include "services/service_manager/public/cpp/interface_registry.h"
|
| +
|
| +using content::RenderFrameHost;
|
| +using device::MockDeviceClient;
|
| +using device::MockUsbDevice;
|
| +
|
| +namespace {
|
| +
|
| +class FakeChooserView : public ChooserController::View {
|
| + public:
|
| + explicit FakeChooserView(std::unique_ptr<ChooserController> controller)
|
| + : controller_(std::move(controller)) {
|
| + controller_->set_view(this);
|
| + }
|
| +
|
| + ~FakeChooserView() override { controller_->set_view(nullptr); }
|
| +
|
| + void OnOptionsInitialized() override {
|
| + if (controller_->NumOptions())
|
| + controller_->Select({0});
|
| + else
|
| + controller_->Cancel();
|
| + delete this;
|
| + }
|
| +
|
| + void OnOptionAdded(size_t index) override { NOTREACHED(); }
|
| + void OnOptionRemoved(size_t index) override { NOTREACHED(); }
|
| + void OnOptionUpdated(size_t index) override { NOTREACHED(); }
|
| + void OnAdapterEnabledChanged(bool enabled) override { NOTREACHED(); }
|
| + void OnRefreshStateChanged(bool refreshing) override { NOTREACHED(); }
|
| +
|
| + private:
|
| + std::unique_ptr<ChooserController> controller_;
|
| +
|
| + DISALLOW_COPY_AND_ASSIGN(FakeChooserView);
|
| +};
|
| +
|
| +class FakeChooserService : public device::usb::ChooserService {
|
| + public:
|
| + static void Create(RenderFrameHost* render_frame_host,
|
| + device::usb::ChooserServiceRequest request) {
|
| + mojo::MakeStrongBinding(
|
| + base::MakeUnique<FakeChooserService>(render_frame_host),
|
| + std::move(request));
|
| + }
|
| +
|
| + explicit FakeChooserService(RenderFrameHost* render_frame_host)
|
| + : render_frame_host_(render_frame_host) {}
|
| +
|
| + ~FakeChooserService() override {}
|
| +
|
| + // device::usb::ChooserService:
|
| + void GetPermission(std::vector<device::usb::DeviceFilterPtr> device_filters,
|
| + const GetPermissionCallback& callback) override {
|
| + auto chooser_controller = base::MakeUnique<UsbChooserController>(
|
| + render_frame_host_, std::move(device_filters), callback);
|
| + new FakeChooserView(std::move(chooser_controller));
|
| + }
|
| +
|
| + private:
|
| + RenderFrameHost* const render_frame_host_;
|
| +
|
| + DISALLOW_COPY_AND_ASSIGN(FakeChooserService);
|
| +};
|
| +
|
| +class WebUsbTest : public InProcessBrowserTest {
|
| + public:
|
| + void SetUpCommandLine(base::CommandLine* command_line) override {
|
| + command_line->AppendSwitch(
|
| + switches::kEnableExperimentalWebPlatformFeatures);
|
| + InProcessBrowserTest::SetUpCommandLine(command_line);
|
| + }
|
| +
|
| + void SetUpOnMainThread() override {
|
| + embedded_test_server()->ServeFilesFromSourceDirectory("content/test/data");
|
| + ASSERT_TRUE(embedded_test_server()->Start());
|
| +
|
| + device_client_.reset(new MockDeviceClient());
|
| + scoped_refptr<MockUsbDevice> mock_device(
|
| + new MockUsbDevice(0, 0, "Test Manufacturer", "Test Device", "123456"));
|
| + device_client_->usb_service()->AddDevice(mock_device);
|
| +
|
| + mock_device =
|
| + new MockUsbDevice(1, 0, "Test Manufacturer", "Test Device", "ABCDEF");
|
| + auto allowed_origins = base::MakeUnique<device::WebUsbAllowedOrigins>();
|
| + allowed_origins->origins.push_back(
|
| + embedded_test_server()->GetURL("localhost", "/").GetOrigin());
|
| + mock_device->set_webusb_allowed_origins(std::move(allowed_origins));
|
| + device_client_->usb_service()->AddDevice(mock_device);
|
| + }
|
| +
|
| + private:
|
| + std::unique_ptr<MockDeviceClient> device_client_;
|
| +};
|
| +
|
| +IN_PROC_BROWSER_TEST_F(WebUsbTest, RequestAndGetDevices) {
|
| + ui_test_utils::NavigateToURL(
|
| + browser(),
|
| + embedded_test_server()->GetURL("localhost", "/simple_page.html"));
|
| + content::WebContents* web_contents =
|
| + browser()->tab_strip_model()->GetActiveWebContents();
|
| + RenderFrameHost* render_frame_host = web_contents->GetMainFrame();
|
| + EXPECT_THAT(render_frame_host->GetLastCommittedOrigin().Serialize(),
|
| + testing::StartsWith("http://localhost:"));
|
| +
|
| + render_frame_host->GetInterfaceRegistry()->AddInterface(
|
| + base::Bind(&FakeChooserService::Create, render_frame_host));
|
| +
|
| + // The mock device with vendorId == 0 has no WebUSB allowed origin descriptor
|
| + // but because this is a top level frame it will be allowed.
|
| + std::string result;
|
| + EXPECT_TRUE(content::ExecuteScriptAndExtractString(
|
| + web_contents,
|
| + "navigator.usb.requestDevice({ filters: [ { vendorId: 0 } ] })"
|
| + " .then(device => {"
|
| + " domAutomationController.send(device.serialNumber);"
|
| + " });",
|
| + &result));
|
| + EXPECT_EQ("123456", result);
|
| +
|
| + EXPECT_TRUE(content::ExecuteScriptAndExtractString(
|
| + web_contents,
|
| + "navigator.usb.getDevices()"
|
| + " .then(devices => {"
|
| + " domAutomationController.send(devices.length.toString());"
|
| + " });",
|
| + &result));
|
| + EXPECT_EQ("1", result);
|
| +}
|
| +
|
| +IN_PROC_BROWSER_TEST_F(WebUsbTest, RequestAndGetDevicesInIframe) {
|
| + ui_test_utils::NavigateToURL(
|
| + browser(),
|
| + embedded_test_server()->GetURL("localhost", "/page_with_iframe.html"));
|
| + content::WebContents* web_contents =
|
| + browser()->tab_strip_model()->GetActiveWebContents();
|
| + RenderFrameHost* main_frame = web_contents->GetMainFrame();
|
| + EXPECT_THAT(main_frame->GetLastCommittedOrigin().Serialize(),
|
| + testing::StartsWith("http://localhost:"));
|
| + RenderFrameHost* embedded_frame = ChildFrameAt(main_frame, 0);
|
| + EXPECT_THAT(embedded_frame->GetLastCommittedOrigin().Serialize(),
|
| + testing::StartsWith("http://localhost:"));
|
| +
|
| + embedded_frame->GetInterfaceRegistry()->AddInterface(
|
| + base::Bind(&FakeChooserService::Create, embedded_frame));
|
| +
|
| + // The mock device with vendorId == 0 has no allowed origin descriptor so an
|
| + // embedded frame will not be able to select it.
|
| + std::string result;
|
| + EXPECT_TRUE(content::ExecuteScriptAndExtractString(
|
| + embedded_frame,
|
| + "navigator.usb.requestDevice({ filters: [ { vendorId: 0 } ] })"
|
| + " .catch(e => { domAutomationController.send(e.toString()); });",
|
| + &result));
|
| + EXPECT_EQ("NotFoundError: No device selected.", result);
|
| +
|
| + // The mock device with vendorId == 1 does however have the embedded test
|
| + // server listed as an allowed origin.
|
| + EXPECT_TRUE(content::ExecuteScriptAndExtractString(
|
| + embedded_frame,
|
| + "navigator.usb.requestDevice({ filters: [ { vendorId: 1 } ] })"
|
| + " .then(device => {"
|
| + " domAutomationController.send(device.serialNumber);"
|
| + " });",
|
| + &result));
|
| + EXPECT_EQ("ABCDEF", result);
|
| +
|
| + EXPECT_TRUE(content::ExecuteScriptAndExtractString(
|
| + embedded_frame,
|
| + "navigator.usb.getDevices()"
|
| + " .then(devices => {"
|
| + " domAutomationController.send(devices.length.toString());"
|
| + " });",
|
| + &result));
|
| + EXPECT_EQ("1", result);
|
| +}
|
| +
|
| +} // namespace
|
|
|