chromium/ios/chrome/browser/device_sharing/model/device_sharing_browser_agent_unittest.mm

// 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.

#import "ios/chrome/browser/device_sharing/model/device_sharing_browser_agent.h"

#import <memory>

#import "components/handoff/handoff_manager.h"
#import "ios/chrome/browser/device_sharing/model/device_sharing_manager.h"
#import "ios/chrome/browser/device_sharing/model/device_sharing_manager_factory.h"
#import "ios/chrome/browser/device_sharing/model/device_sharing_manager_impl.h"
#import "ios/chrome/browser/shared/model/browser/test/test_browser.h"
#import "ios/chrome/browser/shared/model/profile/test/test_profile_ios.h"
#import "ios/chrome/browser/shared/model/web_state_list/web_state_list.h"
#import "ios/chrome/browser/shared/model/web_state_list/web_state_opener.h"
#import "ios/web/public/test/fakes/fake_web_state.h"
#import "ios/web/public/test/web_task_environment.h"
#import "net/base/apple/url_conversions.h"
#import "testing/gtest_mac.h"
#import "testing/platform_test.h"
#import "url/gurl.h"

class DeviceSharingBrowserAgentTest : public PlatformTest {
 public:
  DeviceSharingBrowserAgentTest()
      : url_1_("http://www.test.com/1.html"),
        url_2_("http://www.test.com/2.html"),
        url_3_("http://www.test.com/3.html"),
        url_4_("http://www.test.com/4.html") {
    TestChromeBrowserState::Builder test_browser_state_builder;
    test_browser_state_builder.AddTestingFactory(
        DeviceSharingManagerFactory::GetInstance(),
        DeviceSharingManagerFactory::GetDefaultFactory());

    chrome_browser_state_ = std::move(test_browser_state_builder).Build();
    browser_ = std::make_unique<TestBrowser>(chrome_browser_state_.get());

    other_browser_ = std::make_unique<TestBrowser>(chrome_browser_state_.get());
    incognito_browser_ = std::make_unique<TestBrowser>(
        chrome_browser_state_->GetOffTheRecordChromeBrowserState());
  }

  NSURL* ActiveHandoffUrl() {
    DeviceSharingManagerImpl* sharing_manager =
        static_cast<DeviceSharingManagerImpl*>(
            DeviceSharingManagerFactory::GetForBrowserState(
                chrome_browser_state_.get()));
    return [sharing_manager->handoff_manager_ userActivityWebpageURL];
  }

  web::FakeWebState* AppendNewWebState(Browser* browser,
                                       const GURL url,
                                       bool activate = true) {
    auto fake_web_state = std::make_unique<web::FakeWebState>();
    fake_web_state->SetCurrentURL(url);
    web::FakeWebState* inserted_web_state = fake_web_state.get();
    browser->GetWebStateList()->InsertWebState(
        std::move(fake_web_state),
        WebStateList::InsertionParams::Automatic().Activate(activate));
    return inserted_web_state;
  }

  const GURL url_1_;
  const GURL url_2_;
  const GURL url_3_;
  const GURL url_4_;

  web::WebTaskEnvironment task_environment_;
  std::unique_ptr<TestChromeBrowserState> chrome_browser_state_;
  std::unique_ptr<Browser> browser_;
  std::unique_ptr<Browser> other_browser_;
  std::unique_ptr<Browser> incognito_browser_;
};

TEST_F(DeviceSharingBrowserAgentTest, UpdateEmptyBrowser) {
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
}

TEST_F(DeviceSharingBrowserAgentTest, UpdatePopulatedBrowser) {
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
  AppendNewWebState(browser_.get(), url_1_);
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  // `browser_` isn't the active browser in the device manager yet, so expect
  // the active URL hasn't yet changed.
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));
}

TEST_F(DeviceSharingBrowserAgentTest, ActivateInBrowser) {
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  // As the active browser, newly actiated web states will change the active
  // URL.
  AppendNewWebState(browser_.get(), url_1_);
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));
  // Appending a web state without activating will not change the active URL.
  AppendNewWebState(browser_.get(), url_2_, /*activate=*/false);
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));
}

TEST_F(DeviceSharingBrowserAgentTest, NavigateInBrowser) {
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  web::FakeWebState* web_state = AppendNewWebState(browser_.get(), url_1_);
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));

  web_state->SetVisibleURL(url_2_);
  web_state->OnNavigationFinished(nullptr);
  // Navigating the active web state should update the active URL.
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_2_));
}

TEST_F(DeviceSharingBrowserAgentTest, NavigateInactiveInBrowser) {
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  web::FakeWebState* web_state =
      AppendNewWebState(browser_.get(), url_1_, /*activate=*/false);
  AppendNewWebState(browser_.get(), url_2_);

  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_2_));

  web_state->SetVisibleURL(url_3_);
  web_state->OnNavigationFinished(nullptr);
  // Navigating the non-active web state should not update the active URL.
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_2_));
}

TEST_F(DeviceSharingBrowserAgentTest, DestroyBrowser) {
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  AppendNewWebState(browser_.get(), url_1_);
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));
  browser_.reset();
  // Destroying the active browser should clear the active URL.
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
}

TEST_F(DeviceSharingBrowserAgentTest, UpdatePopulatedIncognitoBrowser) {
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
  AppendNewWebState(incognito_browser_.get(), url_1_);
  DeviceSharingBrowserAgent::CreateForBrowser(incognito_browser_.get());
  DeviceSharingBrowserAgent::FromBrowser(incognito_browser_.get())
      ->UpdateForActiveBrowser();
  // The incognito browser, when active, should never update the active URL.
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
}

TEST_F(DeviceSharingBrowserAgentTest, ActivateInIncognitoBrowser) {
  DeviceSharingBrowserAgent::CreateForBrowser(incognito_browser_.get());
  DeviceSharingBrowserAgent::FromBrowser(incognito_browser_.get())
      ->UpdateForActiveBrowser();
  AppendNewWebState(incognito_browser_.get(), url_1_);
  // The incognito browser, when active, should never update the active URL.
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
}

TEST_F(DeviceSharingBrowserAgentTest, NavigateInIncognitoBrowser) {
  DeviceSharingBrowserAgent::CreateForBrowser(incognito_browser_.get());
  web::FakeWebState* incognito_web_state =
      AppendNewWebState(incognito_browser_.get(), url_1_);

  incognito_web_state->SetVisibleURL(url_2_);
  incognito_web_state->OnNavigationFinished(nullptr);
  // Navigating the non-active web state should not update the active URL.
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
}

TEST_F(DeviceSharingBrowserAgentTest, UpdateTwoBrowsersOneEmpty) {
  AppendNewWebState(browser_.get(), url_1_);
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));
  // Activate an empty browser.
  DeviceSharingBrowserAgent::CreateForBrowser(other_browser_.get());
  DeviceSharingBrowserAgent::FromBrowser(other_browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
  // Switch back.
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));
}

TEST_F(DeviceSharingBrowserAgentTest, UpdateTwoPopulatedBrowsers) {
  AppendNewWebState(browser_.get(), url_1_);
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  AppendNewWebState(other_browser_.get(), url_2_);
  DeviceSharingBrowserAgent::CreateForBrowser(other_browser_.get());
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);

  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));

  DeviceSharingBrowserAgent::FromBrowser(other_browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_2_));

  // Append to `browser_`, but expect no change in active URL as `browser_`
  // isn't active.
  AppendNewWebState(browser_.get(), url_3_);
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_2_));

  // Now make `browser_` active and its URL should be the active one.
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_3_));

  // Destroy `browser_` and the ative browser should be cleared.
  browser_.reset();
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);

  // Activate `other_browser_` and its URL should become active again.
  DeviceSharingBrowserAgent::FromBrowser(other_browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_2_));
}

TEST_F(DeviceSharingBrowserAgentTest, UpdateAndNavigateTwoBrowsers) {
  web::FakeWebState* web_state = AppendNewWebState(browser_.get(), url_1_);
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  web::FakeWebState* other_web_state =
      AppendNewWebState(other_browser_.get(), url_2_);
  DeviceSharingBrowserAgent::CreateForBrowser(other_browser_.get());
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);

  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));

  DeviceSharingBrowserAgent::FromBrowser(other_browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_2_));

  // Navigate in `browser_`, but expect no change in active URL as `browser_`
  // isn't active.
  web_state->SetVisibleURL(url_3_);
  web_state->OnNavigationFinished(nullptr);
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_2_));

  // Now make `browser_` active and its URL should be the active one.
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_3_));

  // Navigate in `other_browser_`, but again expect no change because it isn't
  // active.
  other_web_state->SetVisibleURL(url_4_);
  other_web_state->OnNavigationFinished(nullptr);
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_3_));

  // Activate `other_browser` and observe the navigated URL
  DeviceSharingBrowserAgent::FromBrowser(other_browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_4_));
}

TEST_F(DeviceSharingBrowserAgentTest, UpdateRegularAndIncognitoBrowsers) {
  AppendNewWebState(browser_.get(), url_1_);
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  AppendNewWebState(other_browser_.get(), url_2_);
  DeviceSharingBrowserAgent::CreateForBrowser(incognito_browser_.get());
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);

  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));

  DeviceSharingBrowserAgent::FromBrowser(incognito_browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);

  // Append to `browser_`, but expect no change in active URL as `browser_`
  // isn't active.
  AppendNewWebState(browser_.get(), url_3_);
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);

  // Now make `browser_` active and its URL should be the active one.
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_3_));
}

TEST_F(DeviceSharingBrowserAgentTest, NavigateInRegularAndIncognitoBrowsers) {
  web::FakeWebState* web_state = AppendNewWebState(browser_.get(), url_1_);
  DeviceSharingBrowserAgent::CreateForBrowser(browser_.get());
  web::FakeWebState* incognito_web_state =
      AppendNewWebState(incognito_browser_.get(), url_2_);
  DeviceSharingBrowserAgent::CreateForBrowser(incognito_browser_.get());
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);

  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_1_));

  DeviceSharingBrowserAgent::FromBrowser(incognito_browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);

  // Navigate in `browser_`, but expect no change in active URL as `browser_`
  // isn't active.
  web_state->SetVisibleURL(url_3_);
  web_state->OnNavigationFinished(nullptr);
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);

  // Now make `browser_` active and its URL should be the active one.
  DeviceSharingBrowserAgent::FromBrowser(browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_3_));

  // Navigate in `incognito_browser_`, but again expect no change because it
  // isn't active.
  incognito_web_state->SetVisibleURL(url_4_);
  incognito_web_state->OnNavigationFinished(nullptr);
  EXPECT_NSEQ(ActiveHandoffUrl(), net::NSURLWithGURL(url_3_));

  // Activate `incognito_browser_` and observe no URL.
  DeviceSharingBrowserAgent::FromBrowser(incognito_browser_.get())
      ->UpdateForActiveBrowser();
  EXPECT_NSEQ(ActiveHandoffUrl(), nil);
}