blob: a5bad6babb5d26e83446c2ca9e57d70978de01bb [file] [log] [blame]
// Copyright (c) 2012 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 "base/base_switches.h"
#include "base/command_line.h"
#include "base/memory/memory_pressure_listener.h"
#include "base/test/simple_test_tick_clock.h"
#include "build/build_config.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/media/webrtc/media_capture_devices_dispatcher.h"
#include "chrome/browser/media/webrtc/media_stream_capture_indicator.h"
#include "chrome/browser/memory/tab_manager.h"
#include "chrome/browser/memory/tab_manager_web_contents_data.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/browser_commands.h"
#include "chrome/browser/ui/find_bar/find_bar_controller.h"
#include "chrome/browser/ui/tabs/tab_strip_model.h"
#include "chrome/common/chrome_features.h"
#include "chrome/common/url_constants.h"
#include "chrome/test/base/in_process_browser_test.h"
#include "chrome/test/base/interactive_test_utils.h"
#include "content/public/browser/navigation_details.h"
#include "content/public/browser/navigation_entry.h"
#include "content/public/browser/notification_service.h"
#include "content/public/browser/notification_types.h"
#include "content/public/browser/render_process_host.h"
#include "content/public/common/content_switches.h"
#include "content/public/test/test_utils.h"
#include "url/gurl.h"
using content::OpenURLParams;
#if defined(OS_WIN) || defined(OS_MACOSX) || defined(OS_LINUX)
namespace memory {
class TabManagerTest : public InProcessBrowserTest {
};
bool ObserveNavEntryCommitted(const GURL& expected_url,
const content::NotificationSource& source,
const content::NotificationDetails& details) {
return content::Details<content::LoadCommittedDetails>(details)
->entry->GetURL() == expected_url;
}
IN_PROC_BROWSER_TEST_F(TabManagerTest, TabManagerBasics) {
using content::WindowedNotificationObserver;
TabManager* tab_manager = g_browser_process->GetTabManager();
EXPECT_FALSE(tab_manager->recent_tab_discard());
// Disable the protection of recent tabs.
tab_manager->set_minimum_protection_time_for_tests(
base::TimeDelta::FromMinutes(0));
// Get three tabs open.
WindowedNotificationObserver load1(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open1(GURL(chrome::kChromeUIAboutURL), content::Referrer(),
WindowOpenDisposition::CURRENT_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open1);
load1.Wait();
WindowedNotificationObserver load2(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open2(GURL(chrome::kChromeUICreditsURL), content::Referrer(),
WindowOpenDisposition::NEW_FOREGROUND_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open2);
load2.Wait();
WindowedNotificationObserver load3(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open3(GURL(chrome::kChromeUITermsURL), content::Referrer(),
WindowOpenDisposition::NEW_FOREGROUND_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open3);
load3.Wait();
auto* tsm = browser()->tab_strip_model();
EXPECT_EQ(3, tsm->count());
// Navigate the current (third) tab to a different URL, so we can test
// back/forward later.
WindowedNotificationObserver load4(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open4(GURL(chrome::kChromeUIVersionURL), content::Referrer(),
WindowOpenDisposition::CURRENT_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open4);
load4.Wait();
// Navigate the third tab again, such that we have three navigation entries.
WindowedNotificationObserver load5(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open5(GURL("chrome://dns"), content::Referrer(),
WindowOpenDisposition::CURRENT_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open5);
load5.Wait();
EXPECT_EQ(3, tsm->count());
// Discard a tab. It should kill the first tab, since it was the oldest
// and was not selected.
EXPECT_TRUE(tab_manager->DiscardTabImpl());
EXPECT_EQ(3, tsm->count());
EXPECT_TRUE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(0)));
EXPECT_FALSE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(1)));
EXPECT_FALSE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(2)));
EXPECT_TRUE(tab_manager->recent_tab_discard());
// Run discard again, make sure it kills the second tab.
EXPECT_TRUE(tab_manager->DiscardTabImpl());
EXPECT_EQ(3, tsm->count());
EXPECT_TRUE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(0)));
EXPECT_TRUE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(1)));
EXPECT_FALSE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(2)));
// Kill the third tab. It should not kill the last tab, since it is active
// tab.
EXPECT_FALSE(tab_manager->DiscardTabImpl());
EXPECT_TRUE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(0)));
EXPECT_TRUE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(1)));
EXPECT_FALSE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(2)));
// Kill the third tab after making second tab active.
tsm->ActivateTabAt(1, true);
EXPECT_EQ(1, tsm->active_index());
EXPECT_FALSE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(1)));
tab_manager->DiscardWebContentsAt(2, tsm);
EXPECT_TRUE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(2)));
// Force creation of the FindBarController.
browser()->GetFindBarController();
// Select the first tab. It should reload.
WindowedNotificationObserver reload1(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
base::Bind(&ObserveNavEntryCommitted,
GURL(chrome::kChromeUIChromeURLsURL)));
chrome::SelectNumberedTab(browser(), 0);
reload1.Wait();
// Make sure the FindBarController gets the right WebContents.
EXPECT_EQ(browser()->GetFindBarController()->web_contents(),
tsm->GetActiveWebContents());
EXPECT_EQ(0, tsm->active_index());
EXPECT_FALSE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(0)));
EXPECT_FALSE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(1)));
EXPECT_TRUE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(2)));
// Select the third tab. It should reload.
WindowedNotificationObserver reload2(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
base::Bind(&ObserveNavEntryCommitted, GURL("chrome://dns")));
chrome::SelectNumberedTab(browser(), 2);
reload2.Wait();
EXPECT_EQ(2, tsm->active_index());
EXPECT_FALSE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(0)));
EXPECT_FALSE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(1)));
EXPECT_FALSE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(2)));
// Navigate the third tab back twice. We used to crash here due to
// crbug.com/121373.
EXPECT_TRUE(chrome::CanGoBack(browser()));
EXPECT_FALSE(chrome::CanGoForward(browser()));
WindowedNotificationObserver back1(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
base::Bind(&ObserveNavEntryCommitted, GURL(chrome::kChromeUIVersionURL)));
chrome::GoBack(browser(), WindowOpenDisposition::CURRENT_TAB);
back1.Wait();
EXPECT_TRUE(chrome::CanGoBack(browser()));
EXPECT_TRUE(chrome::CanGoForward(browser()));
WindowedNotificationObserver back2(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
base::Bind(&ObserveNavEntryCommitted, GURL(chrome::kChromeUITermsURL)));
chrome::GoBack(browser(), WindowOpenDisposition::CURRENT_TAB);
back2.Wait();
EXPECT_FALSE(chrome::CanGoBack(browser()));
EXPECT_TRUE(chrome::CanGoForward(browser()));
}
// On Linux, memory pressure listener is not implemented yet.
#if defined(OS_WIN) || defined(OS_MACOSX) || defined(OS_CHROMEOS)
// Test that the MemoryPressureListener event is properly triggering a tab
// discard upon |MEMORY_PRESSURE_LEVEL_CRITICAL| event.
IN_PROC_BROWSER_TEST_F(TabManagerTest, OomPressureListener) {
TabManager* tab_manager = g_browser_process->GetTabManager();
// Disable the protection of recent tabs.
tab_manager->set_minimum_protection_time_for_tests(
base::TimeDelta::FromMinutes(0));
// Get three tabs open.
content::WindowedNotificationObserver load1(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open1(GURL(chrome::kChromeUIAboutURL), content::Referrer(),
WindowOpenDisposition::CURRENT_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open1);
load1.Wait();
content::WindowedNotificationObserver load2(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open2(GURL(chrome::kChromeUICreditsURL), content::Referrer(),
WindowOpenDisposition::NEW_FOREGROUND_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open2);
load2.Wait();
EXPECT_FALSE(tab_manager->recent_tab_discard());
// Nothing should happen with a moderate memory pressure event.
base::MemoryPressureListener::NotifyMemoryPressure(
base::MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE);
EXPECT_FALSE(tab_manager->recent_tab_discard());
// A critical memory pressure event should discard a tab.
base::MemoryPressureListener::NotifyMemoryPressure(
base::MemoryPressureListener::MEMORY_PRESSURE_LEVEL_CRITICAL);
// Coming here, an asynchronous operation will collect system stats. Once in,
// a tab should get discarded. As such we need to give it 10s time to discard.
const int kTimeoutTimeInMS = 10000;
const int kIntervalTimeInMS = 5;
int timeout = kTimeoutTimeInMS / kIntervalTimeInMS;
while (--timeout) {
base::PlatformThread::Sleep(
base::TimeDelta::FromMilliseconds(kIntervalTimeInMS));
base::RunLoop().RunUntilIdle();
if (tab_manager->recent_tab_discard())
break;
}
EXPECT_TRUE(tab_manager->recent_tab_discard());
}
#endif
IN_PROC_BROWSER_TEST_F(TabManagerTest, InvalidOrEmptyURL) {
TabManager* tab_manager = g_browser_process->GetTabManager();
// Disable the protection of recent tabs.
tab_manager->set_minimum_protection_time_for_tests(
base::TimeDelta::FromMinutes(0));
// Open two tabs. Wait for the foreground one to load but do not wait for the
// background one.
content::WindowedNotificationObserver load1(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open1(GURL(chrome::kChromeUIAboutURL), content::Referrer(),
WindowOpenDisposition::CURRENT_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open1);
load1.Wait();
content::WindowedNotificationObserver load2(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open2(GURL(chrome::kChromeUICreditsURL), content::Referrer(),
WindowOpenDisposition::NEW_BACKGROUND_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open2);
ASSERT_EQ(2, browser()->tab_strip_model()->count());
// This shouldn't be able to discard a tab as the background tab has not yet
// started loading (its URL is not committed).
EXPECT_FALSE(tab_manager->DiscardTabImpl());
// Wait for the background tab to load which then allows it to be discarded.
load2.Wait();
EXPECT_TRUE(tab_manager->DiscardTabImpl());
}
// Makes sure that PDF pages are protected.
IN_PROC_BROWSER_TEST_F(TabManagerTest, ProtectPDFPages) {
TabManager* tab_manager = g_browser_process->GetTabManager();
// Start the embedded test server so we can get served the required PDF page.
ASSERT_TRUE(embedded_test_server()->InitializeAndListen());
embedded_test_server()->StartAcceptingConnections();
// Get two tabs open, the first one being a PDF page and the second one being
// the foreground tab.
GURL url1 = embedded_test_server()->GetURL("/pdf/test.pdf");
ui_test_utils::NavigateToURL(browser(), url1);
GURL url2(chrome::kChromeUIAboutURL);
ui_test_utils::NavigateToURLWithDisposition(
browser(), url2, WindowOpenDisposition::NEW_FOREGROUND_TAB,
ui_test_utils::BROWSER_TEST_WAIT_FOR_NAVIGATION);
// No discarding should be possible as the only background tab is displaying a
// PDF page, hence protected.
EXPECT_FALSE(tab_manager->DiscardTabImpl());
}
// Makes sure that recently opened or used tabs are protected, depending on the
// value of of |minimum_protection_time_|.
// TODO(georgesak): Move this to a unit test instead (requires change to API).
IN_PROC_BROWSER_TEST_F(TabManagerTest, ProtectRecentlyUsedTabs) {
// TODO(georgesak): Retrieve this value from tab_manager.h once it becomes a
// constant (as of now, it gets set through variations).
const int kProtectionTime = 5;
TabManager* tab_manager = g_browser_process->GetTabManager();
base::SimpleTestTickClock test_clock_;
tab_manager->set_test_tick_clock(&test_clock_);
auto* tsm = browser()->tab_strip_model();
// Set the minimum time of protection.
tab_manager->set_minimum_protection_time_for_tests(
base::TimeDelta::FromMinutes(kProtectionTime));
// Open 2 tabs, the second one being in the background.
ui_test_utils::NavigateToURL(browser(), GURL(chrome::kChromeUIAboutURL));
ui_test_utils::NavigateToURLWithDisposition(
browser(), GURL(chrome::kChromeUIAboutURL),
WindowOpenDisposition::NEW_BACKGROUND_TAB,
ui_test_utils::BROWSER_TEST_WAIT_FOR_NAVIGATION);
EXPECT_EQ(2, tsm->count());
// Advance the clock for less than the protection time.
test_clock_.Advance(base::TimeDelta::FromMinutes(kProtectionTime / 2));
// Should not be able to discard a tab.
ASSERT_FALSE(tab_manager->DiscardTabImpl());
// Advance the clock for more than the protection time.
test_clock_.Advance(base::TimeDelta::FromMinutes(kProtectionTime / 2 + 2));
// Should be able to discard the background tab now.
EXPECT_TRUE(tab_manager->DiscardTabImpl());
// Activate the 2nd tab.
tsm->ActivateTabAt(1, true);
EXPECT_EQ(1, tsm->active_index());
// Advance the clock for less than the protection time.
test_clock_.Advance(base::TimeDelta::FromMinutes(kProtectionTime / 2));
// Should not be able to discard a tab.
ASSERT_FALSE(tab_manager->DiscardTabImpl());
// Advance the clock for more than the protection time.
test_clock_.Advance(base::TimeDelta::FromMinutes(kProtectionTime / 2 + 2));
// Should be able to discard the background tab now.
EXPECT_TRUE(tab_manager->DiscardTabImpl());
// This is necessary otherwise the test crashes in
// WebContentsData::WebContentsDestroyed.
tsm->CloseAllTabs();
}
// Makes sure that tabs using media devices are protected.
IN_PROC_BROWSER_TEST_F(TabManagerTest, ProtectVideoTabs) {
TabManager* tab_manager = g_browser_process->GetTabManager();
// Disable the protection of recent tabs.
tab_manager->set_minimum_protection_time_for_tests(
base::TimeDelta::FromMinutes(0));
// Open 2 tabs, the second one being in the background.
ui_test_utils::NavigateToURL(browser(), GURL(chrome::kChromeUIAboutURL));
ui_test_utils::NavigateToURLWithDisposition(
browser(), GURL(chrome::kChromeUIAboutURL),
WindowOpenDisposition::NEW_BACKGROUND_TAB,
ui_test_utils::BROWSER_TEST_WAIT_FOR_NAVIGATION);
auto* tab = browser()->tab_strip_model()->GetWebContentsAt(1);
// Simulate that a video stream is now being captured.
content::MediaStreamDevice fake_media_device(
content::MEDIA_DEVICE_VIDEO_CAPTURE, "fake_media_device",
"fake_media_device");
content::MediaStreamDevices video_devices(1, fake_media_device);
MediaCaptureDevicesDispatcher* dispatcher =
MediaCaptureDevicesDispatcher::GetInstance();
dispatcher->SetTestVideoCaptureDevices(video_devices);
std::unique_ptr<content::MediaStreamUI> video_stream_ui =
dispatcher->GetMediaStreamCaptureIndicator()->RegisterMediaStream(
tab, video_devices);
video_stream_ui->OnStarted(base::Closure());
// Should not be able to discard a tab.
ASSERT_FALSE(tab_manager->DiscardTabImpl());
// Remove the video stream.
video_stream_ui.reset();
// Should be able to discard the background tab now.
EXPECT_TRUE(tab_manager->DiscardTabImpl());
}
IN_PROC_BROWSER_TEST_F(TabManagerTest, CanSuspendBackgroundedRenderer) {
TabManager* tab_manager = g_browser_process->GetTabManager();
// Open 2 tabs, the second one being in the background.
ui_test_utils::NavigateToURL(browser(), GURL(chrome::kChromeUIAboutURL));
ui_test_utils::NavigateToURLWithDisposition(
browser(), GURL(chrome::kChromeUIAboutURL),
WindowOpenDisposition::NEW_BACKGROUND_TAB,
ui_test_utils::BROWSER_TEST_WAIT_FOR_NAVIGATION);
auto* tab = browser()->tab_strip_model()->GetWebContentsAt(1);
// Simulate that a video stream is now being captured.
content::MediaStreamDevice fake_media_device(
content::MEDIA_DEVICE_VIDEO_CAPTURE, "fake_media_device",
"fake_media_device");
content::MediaStreamDevices video_devices(1, fake_media_device);
MediaCaptureDevicesDispatcher* dispatcher =
MediaCaptureDevicesDispatcher::GetInstance();
dispatcher->SetTestVideoCaptureDevices(video_devices);
std::unique_ptr<content::MediaStreamUI> video_stream_ui =
dispatcher->GetMediaStreamCaptureIndicator()->RegisterMediaStream(
tab, video_devices);
video_stream_ui->OnStarted(base::Closure());
// Should not be able to suspend a tab which plays a video.
int render_process_id = tab->GetRenderProcessHost()->GetID();
ASSERT_FALSE(tab_manager->CanSuspendBackgroundedRenderer(render_process_id));
// Remove the video stream.
video_stream_ui.reset();
// Should be able to suspend the background tab now.
EXPECT_TRUE(tab_manager->CanSuspendBackgroundedRenderer(render_process_id));
}
IN_PROC_BROWSER_TEST_F(TabManagerTest, AutoDiscardable) {
using content::WindowedNotificationObserver;
TabManager* tab_manager = g_browser_process->GetTabManager();
// Disable the protection of recent tabs.
tab_manager->set_minimum_protection_time_for_tests(
base::TimeDelta::FromMinutes(0));
// Get two tabs open.
WindowedNotificationObserver load1(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open1(GURL(chrome::kChromeUIAboutURL), content::Referrer(),
WindowOpenDisposition::CURRENT_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open1);
load1.Wait();
WindowedNotificationObserver load2(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open2(GURL(chrome::kChromeUICreditsURL), content::Referrer(),
WindowOpenDisposition::NEW_FOREGROUND_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open2);
load2.Wait();
// Set the auto-discardable state of the first tab to false.
auto* tsm = browser()->tab_strip_model();
ASSERT_EQ(2, tsm->count());
tab_manager->SetTabAutoDiscardableState(tsm->GetWebContentsAt(0), false);
// Shouldn't discard the tab, since auto-discardable is deactivated.
EXPECT_FALSE(tab_manager->DiscardTabImpl());
// Reset auto-discardable state to true.
tab_manager->SetTabAutoDiscardableState(tsm->GetWebContentsAt(0), true);
// Now it should be able to discard the tab.
EXPECT_TRUE(tab_manager->DiscardTabImpl());
EXPECT_TRUE(tab_manager->IsTabDiscarded(tsm->GetWebContentsAt(0)));
}
IN_PROC_BROWSER_TEST_F(TabManagerTest, PurgeBackgroundRenderer) {
TabManager* tab_manager = g_browser_process->GetTabManager();
base::SimpleTestTickClock test_clock_;
tab_manager->set_test_tick_clock(&test_clock_);
// Get three tabs open.
content::WindowedNotificationObserver load1(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open1(GURL(chrome::kChromeUIAboutURL), content::Referrer(),
WindowOpenDisposition::CURRENT_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open1);
load1.Wait();
content::WindowedNotificationObserver load2(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open2(GURL(chrome::kChromeUICreditsURL), content::Referrer(),
WindowOpenDisposition::NEW_FOREGROUND_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open2);
load2.Wait();
content::WindowedNotificationObserver load3(
content::NOTIFICATION_NAV_ENTRY_COMMITTED,
content::NotificationService::AllSources());
OpenURLParams open3(GURL(chrome::kChromeUITermsURL), content::Referrer(),
WindowOpenDisposition::NEW_FOREGROUND_TAB,
ui::PAGE_TRANSITION_TYPED, false);
browser()->OpenURL(open3);
load3.Wait();
auto* tsm = browser()->tab_strip_model();
TabManager::WebContentsData* tab1_contents_data =
tab_manager->GetWebContentsData(tsm->GetWebContentsAt(0));
TabManager::WebContentsData* tab2_contents_data =
tab_manager->GetWebContentsData(tsm->GetWebContentsAt(1));
TabManager::WebContentsData* tab3_contents_data =
tab_manager->GetWebContentsData(tsm->GetWebContentsAt(2));
// The time-to-purge initialized at ActiveTabChanged should be in the
// right default range.
EXPECT_GE(tab1_contents_data->time_to_purge(),
base::TimeDelta::FromMinutes(30));
EXPECT_LT(tab1_contents_data->time_to_purge(),
base::TimeDelta::FromMinutes(60));
EXPECT_GE(tab2_contents_data->time_to_purge(),
base::TimeDelta::FromMinutes(30));
EXPECT_LT(tab2_contents_data->time_to_purge(),
base::TimeDelta::FromMinutes(60));
EXPECT_GE(tab3_contents_data->time_to_purge(),
base::TimeDelta::FromMinutes(30));
EXPECT_LT(tab3_contents_data->time_to_purge(),
base::TimeDelta::FromMinutes(60));
// To make it easy to test, configure time-to-purge here.
base::TimeDelta time_to_purge1 = base::TimeDelta::FromMinutes(30);
base::TimeDelta time_to_purge2 = base::TimeDelta::FromMinutes(40);
tab1_contents_data->set_time_to_purge(time_to_purge1);
tab2_contents_data->set_time_to_purge(time_to_purge2);
tab3_contents_data->set_time_to_purge(time_to_purge1);
// No tabs are not purged yet.
ASSERT_FALSE(tab1_contents_data->is_purged());
ASSERT_FALSE(tab2_contents_data->is_purged());
ASSERT_FALSE(tab3_contents_data->is_purged());
// Advance the clock for time_to_purge1.
test_clock_.Advance(time_to_purge1);
tab_manager->PurgeBackgroundedTabsIfNeeded();
ASSERT_FALSE(tab1_contents_data->is_purged());
ASSERT_FALSE(tab2_contents_data->is_purged());
ASSERT_FALSE(tab3_contents_data->is_purged());
// Advance the clock for 1 minutes.
test_clock_.Advance(base::TimeDelta::FromMinutes(1));
tab_manager->PurgeBackgroundedTabsIfNeeded();
// Since tab1 is kept inactive and background for more than
// time_to_purge1, tab1 should be purged.
ASSERT_TRUE(tab1_contents_data->is_purged());
ASSERT_FALSE(tab2_contents_data->is_purged());
ASSERT_FALSE(tab3_contents_data->is_purged());
// Advance the clock.
test_clock_.Advance(time_to_purge2 - time_to_purge1);
tab_manager->PurgeBackgroundedTabsIfNeeded();
// Since tab2 is kept inactive and background for more than
// time_to_purge2, tab1 should be purged.
// Since tab3 is active, tab3 should not be purged.
ASSERT_TRUE(tab1_contents_data->is_purged());
ASSERT_TRUE(tab2_contents_data->is_purged());
ASSERT_FALSE(tab3_contents_data->is_purged());
tsm->CloseAllTabs();
}
} // namespace memory
#endif // OS_WIN || OS_MAXOSX || OS_LINUX