1 // Copyright 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
8 #include "base/memory/scoped_ptr.h"
9 #include "base/rand_util.h"
10 #include "base/strings/stringprintf.h"
11 #include "base/strings/utf_string_conversions.h"
12 #include "chrome/browser/sync/glue/synced_session_tracker.h"
13 #include "components/sessions/serialized_navigation_entry_test_helper.h"
14 #include "components/sessions/session_types.h"
15 #include "testing/gtest/include/gtest/gtest.h"
17 namespace browser_sync
{
19 typedef testing::Test SyncedSessionTrackerTest
;
21 TEST_F(SyncedSessionTrackerTest
, GetSession
) {
22 SyncedSessionTracker tracker
;
23 SyncedSession
* session1
= tracker
.GetSession("tag");
24 SyncedSession
* session2
= tracker
.GetSession("tag2");
25 ASSERT_EQ(session1
, tracker
.GetSession("tag"));
26 ASSERT_NE(session1
, session2
);
27 // Should clean up memory on its own.
30 TEST_F(SyncedSessionTrackerTest
, GetTabUnmapped
) {
31 SyncedSessionTracker tracker
;
32 sessions::SessionTab
* tab
= tracker
.GetTab("tag", 0, 0);
33 ASSERT_EQ(tab
, tracker
.GetTab("tag", 0, 0));
34 // Should clean up memory on its own.
37 TEST_F(SyncedSessionTrackerTest
, PutWindowInSession
) {
38 SyncedSessionTracker tracker
;
39 tracker
.PutWindowInSession("tag", 0);
40 SyncedSession
* session
= tracker
.GetSession("tag");
41 ASSERT_EQ(1U, session
->windows
.size());
42 // Should clean up memory on its own.
45 TEST_F(SyncedSessionTrackerTest
, PutTabInWindow
) {
46 SyncedSessionTracker tracker
;
47 tracker
.PutWindowInSession("tag", 10);
48 tracker
.PutTabInWindow("tag", 10, 15, 0); // win id 10, tab id 15, tab ind 0.
49 SyncedSession
* session
= tracker
.GetSession("tag");
50 ASSERT_EQ(1U, session
->windows
.size());
51 ASSERT_EQ(1U, session
->windows
[10]->tabs
.size());
52 ASSERT_EQ(tracker
.GetTab("tag", 15, 1), session
->windows
[10]->tabs
[0]);
53 // Should clean up memory on its own.
56 TEST_F(SyncedSessionTrackerTest
, LookupAllForeignSessions
) {
57 SyncedSessionTracker tracker
;
58 std::vector
<const SyncedSession
*> sessions
;
59 ASSERT_FALSE(tracker
.LookupAllForeignSessions(&sessions
));
60 tracker
.GetSession("tag1");
61 tracker
.GetSession("tag2");
62 tracker
.PutWindowInSession("tag1", 0);
63 tracker
.PutTabInWindow("tag1", 0, 15, 0);
64 sessions::SessionTab
* tab
= tracker
.GetTab("tag1", 15, 1);
66 tab
->navigations
.push_back(
67 sessions::SerializedNavigationEntryTestHelper::CreateNavigation(
68 "bla://valid_url", "title"));
69 ASSERT_TRUE(tracker
.LookupAllForeignSessions(&sessions
));
70 // Only the session with a valid window and tab gets returned.
71 ASSERT_EQ(1U, sessions
.size());
72 ASSERT_EQ("tag1", sessions
[0]->session_tag
);
75 TEST_F(SyncedSessionTrackerTest
, LookupSessionWindows
) {
76 SyncedSessionTracker tracker
;
77 std::vector
<const sessions::SessionWindow
*> windows
;
78 ASSERT_FALSE(tracker
.LookupSessionWindows("tag1", &windows
));
79 tracker
.GetSession("tag1");
80 tracker
.PutWindowInSession("tag1", 0);
81 tracker
.PutWindowInSession("tag1", 2);
82 tracker
.GetSession("tag2");
83 tracker
.PutWindowInSession("tag2", 0);
84 tracker
.PutWindowInSession("tag2", 2);
85 ASSERT_TRUE(tracker
.LookupSessionWindows("tag1", &windows
));
86 ASSERT_EQ(2U, windows
.size()); // Only windows from tag1 session.
87 ASSERT_NE((sessions::SessionWindow
*)nullptr, windows
[0]);
88 ASSERT_NE((sessions::SessionWindow
*)nullptr, windows
[1]);
89 ASSERT_NE(windows
[1], windows
[0]);
92 TEST_F(SyncedSessionTrackerTest
, LookupSessionTab
) {
93 SyncedSessionTracker tracker
;
94 const sessions::SessionTab
* tab
;
95 ASSERT_FALSE(tracker
.LookupSessionTab("tag1", 5, &tab
));
96 tracker
.GetSession("tag1");
97 tracker
.PutWindowInSession("tag1", 0);
98 tracker
.PutTabInWindow("tag1", 0, 5, 0);
99 ASSERT_TRUE(tracker
.LookupSessionTab("tag1", 5, &tab
));
100 ASSERT_NE((sessions::SessionTab
*)nullptr, tab
);
103 TEST_F(SyncedSessionTrackerTest
, Complex
) {
104 const std::string tag1
= "tag";
105 const std::string tag2
= "tag2";
106 const std::string tag3
= "tag3";
107 SyncedSessionTracker tracker
;
108 std::vector
<sessions::SessionTab
*> tabs1
, tabs2
;
109 sessions::SessionTab
* temp_tab
;
110 ASSERT_TRUE(tracker
.Empty());
111 ASSERT_EQ(0U, tracker
.num_synced_sessions());
112 ASSERT_EQ(0U, tracker
.num_synced_tabs(tag1
));
113 tabs1
.push_back(tracker
.GetTab(tag1
, 0, 0));
114 tabs1
.push_back(tracker
.GetTab(tag1
, 1, 1));
115 tabs1
.push_back(tracker
.GetTab(tag1
, 2, 2));
116 ASSERT_EQ(3U, tracker
.num_synced_tabs(tag1
));
117 ASSERT_EQ(0U, tracker
.num_synced_sessions());
118 temp_tab
= tracker
.GetTab(tag1
, 0, 0); // Already created.
119 ASSERT_EQ(3U, tracker
.num_synced_tabs(tag1
));
120 ASSERT_EQ(0U, tracker
.num_synced_sessions());
121 ASSERT_EQ(tabs1
[0], temp_tab
);
122 tabs2
.push_back(tracker
.GetTab(tag2
, 0, 0));
123 ASSERT_EQ(1U, tracker
.num_synced_tabs(tag2
));
124 ASSERT_EQ(0U, tracker
.num_synced_sessions());
125 ASSERT_FALSE(tracker
.DeleteSession(tag3
));
127 SyncedSession
* session
= tracker
.GetSession(tag1
);
128 SyncedSession
* session2
= tracker
.GetSession(tag2
);
129 SyncedSession
* session3
= tracker
.GetSession(tag3
);
130 ASSERT_EQ(3U, tracker
.num_synced_sessions());
132 ASSERT_TRUE(session
);
133 ASSERT_TRUE(session2
);
134 ASSERT_TRUE(session3
);
135 ASSERT_NE(session
, session2
);
136 ASSERT_NE(session2
, session3
);
137 ASSERT_TRUE(tracker
.DeleteSession(tag3
));
138 ASSERT_EQ(2U, tracker
.num_synced_sessions());
140 tracker
.PutWindowInSession(tag1
, 0); // Create a window.
141 tracker
.PutTabInWindow(tag1
, 0, 2, 0); // No longer unmapped.
142 ASSERT_EQ(3U, tracker
.num_synced_tabs(tag1
)); // Has not changed.
144 const sessions::SessionTab
*tab_ptr
;
145 ASSERT_TRUE(tracker
.LookupSessionTab(tag1
, 0, &tab_ptr
));
146 ASSERT_EQ(tab_ptr
, tabs1
[0]);
147 ASSERT_TRUE(tracker
.LookupSessionTab(tag1
, 2, &tab_ptr
));
148 ASSERT_EQ(tab_ptr
, tabs1
[2]);
149 ASSERT_FALSE(tracker
.LookupSessionTab(tag1
, 3, &tab_ptr
));
150 ASSERT_EQ(static_cast<const sessions::SessionTab
*>(NULL
), tab_ptr
);
152 std::vector
<const sessions::SessionWindow
*> windows
;
153 ASSERT_TRUE(tracker
.LookupSessionWindows(tag1
, &windows
));
154 ASSERT_EQ(1U, windows
.size());
155 ASSERT_TRUE(tracker
.LookupSessionWindows(tag2
, &windows
));
156 ASSERT_EQ(0U, windows
.size());
158 // The sessions don't have valid tabs, lookup should not succeed.
159 std::vector
<const SyncedSession
*> sessions
;
160 ASSERT_FALSE(tracker
.LookupAllForeignSessions(&sessions
));
163 ASSERT_EQ(0U, tracker
.num_synced_tabs(tag1
));
164 ASSERT_EQ(0U, tracker
.num_synced_tabs(tag2
));
165 ASSERT_EQ(0U, tracker
.num_synced_sessions());
168 TEST_F(SyncedSessionTrackerTest
, ManyGetTabs
) {
169 SyncedSessionTracker tracker
;
170 ASSERT_TRUE(tracker
.Empty());
171 const int kMaxSessions
= 10;
172 const int kMaxTabs
= 1000;
173 const int kMaxAttempts
= 10000;
174 for (int j
=0; j
<kMaxSessions
; ++j
) {
175 std::string tag
= base::StringPrintf("tag%d", j
);
176 for (int i
=0; i
<kMaxAttempts
; ++i
) {
177 // More attempts than tabs means we'll sometimes get the same tabs,
178 // sometimes have to allocate new tabs.
179 int rand_tab_num
= base::RandInt(0, kMaxTabs
);
180 sessions::SessionTab
* tab
=
181 tracker
.GetTab(tag
, rand_tab_num
, rand_tab_num
+ 1);
187 TEST_F(SyncedSessionTrackerTest
, LookupTabNodeIds
) {
188 SyncedSessionTracker tracker
;
189 std::set
<int> result
;
190 std::string tag1
= "session1";
191 std::string tag2
= "session2";
192 std::string tag3
= "session3";
194 tracker
.GetTab(tag1
, 1, 1);
195 tracker
.GetTab(tag1
, 2, 2);
196 EXPECT_TRUE(tracker
.LookupTabNodeIds(tag1
, &result
));
197 EXPECT_EQ(2U, result
.size());
198 EXPECT_FALSE(result
.end() == result
.find(1));
199 EXPECT_FALSE(result
.end() == result
.find(2));
200 EXPECT_FALSE(tracker
.LookupTabNodeIds(tag2
, &result
));
202 tracker
.PutWindowInSession(tag1
, 0);
203 tracker
.PutTabInWindow(tag1
, 0, 3, 0);
204 EXPECT_TRUE(tracker
.LookupTabNodeIds(tag1
, &result
));
205 EXPECT_EQ(2U, result
.size());
207 tracker
.GetTab(tag1
, 3, 3);
208 EXPECT_TRUE(tracker
.LookupTabNodeIds(tag1
, &result
));
209 EXPECT_EQ(3U, result
.size());
210 EXPECT_FALSE(result
.end() == result
.find(3));
212 tracker
.GetTab(tag2
, 1, 21);
213 tracker
.GetTab(tag2
, 2, 22);
214 EXPECT_TRUE(tracker
.LookupTabNodeIds(tag2
, &result
));
215 EXPECT_EQ(2U, result
.size());
216 EXPECT_FALSE(result
.end() == result
.find(21));
217 EXPECT_FALSE(result
.end() == result
.find(22));
218 EXPECT_TRUE(tracker
.LookupTabNodeIds(tag1
, &result
));
219 EXPECT_EQ(3U, result
.size());
220 EXPECT_FALSE(result
.end() == result
.find(1));
221 EXPECT_FALSE(result
.end() == result
.find(2));
223 EXPECT_FALSE(tracker
.LookupTabNodeIds(tag3
, &result
));
224 tracker
.PutWindowInSession(tag3
, 1);
225 tracker
.PutTabInWindow(tag3
, 1, 5, 0);
226 EXPECT_TRUE(tracker
.LookupTabNodeIds(tag3
, &result
));
227 EXPECT_TRUE(result
.empty());
228 EXPECT_TRUE(tracker
.DeleteSession(tag3
));
229 EXPECT_FALSE(tracker
.LookupTabNodeIds(tag3
, &result
));
231 EXPECT_TRUE(tracker
.DeleteSession(tag1
));
232 EXPECT_FALSE(tracker
.LookupTabNodeIds(tag1
, &result
));
233 EXPECT_TRUE(tracker
.LookupTabNodeIds(tag2
, &result
));
234 EXPECT_EQ(2U, result
.size());
235 EXPECT_FALSE(result
.end() == result
.find(21));
236 EXPECT_FALSE(result
.end() == result
.find(22));
237 EXPECT_TRUE(tracker
.DeleteSession(tag2
));
238 EXPECT_FALSE(tracker
.LookupTabNodeIds(tag2
, &result
));
241 TEST_F(SyncedSessionTrackerTest
, SessionTracking
) {
242 SyncedSessionTracker tracker
;
243 ASSERT_TRUE(tracker
.Empty());
244 std::string tag1
= "tag1";
245 std::string tag2
= "tag2";
247 // Create some session information that is stale.
248 SyncedSession
* session1
= tracker
.GetSession(tag1
);
249 tracker
.PutWindowInSession(tag1
, 0);
250 tracker
.PutTabInWindow(tag1
, 0, 0, 0);
251 tracker
.PutTabInWindow(tag1
, 0, 1, 1);
252 tracker
.GetTab(tag1
, 2, 3U)->window_id
.set_id(0); // Will be unmapped.
253 tracker
.GetTab(tag1
, 3, 4U)->window_id
.set_id(0); // Will be unmapped.
254 tracker
.PutWindowInSession(tag1
, 1);
255 tracker
.PutTabInWindow(tag1
, 1, 4, 0);
256 tracker
.PutTabInWindow(tag1
, 1, 5, 1);
257 ASSERT_EQ(2U, session1
->windows
.size());
258 ASSERT_EQ(2U, session1
->windows
[0]->tabs
.size());
259 ASSERT_EQ(2U, session1
->windows
[1]->tabs
.size());
260 ASSERT_EQ(6U, tracker
.num_synced_tabs(tag1
));
262 // Create a session that should not be affected.
263 SyncedSession
* session2
= tracker
.GetSession(tag2
);
264 tracker
.PutWindowInSession(tag2
, 2);
265 tracker
.PutTabInWindow(tag2
, 2, 1, 0);
266 ASSERT_EQ(1U, session2
->windows
.size());
267 ASSERT_EQ(1U, session2
->windows
[2]->tabs
.size());
268 ASSERT_EQ(1U, tracker
.num_synced_tabs(tag2
));
270 // Reset tracking and get the current windows/tabs.
271 // We simulate moving a tab from one window to another, then closing the
272 // first window (including its one remaining tab), and opening a new tab
273 // on the remaining window.
275 // New tab, arrived before meta node so unmapped.
276 tracker
.GetTab(tag1
, 6, 7U);
277 tracker
.ResetSessionTracking(tag1
);
278 tracker
.PutWindowInSession(tag1
, 0);
279 tracker
.PutTabInWindow(tag1
, 0, 0, 0);
281 tracker
.PutTabInWindow(tag1
, 0, 2, 1); // No longer unmapped.
282 // Tab 3 was unmapped and does not get used.
283 tracker
.PutTabInWindow(tag1
, 0, 4, 2); // Moved from window 1.
284 // Window 1 was closed, along with tab 5.
285 tracker
.PutTabInWindow(tag1
, 0, 6, 3); // No longer unmapped.
286 // Session 2 should not be affected.
287 tracker
.CleanupSession(tag1
);
289 // Verify that only those parts of the session not owned have been removed.
290 ASSERT_EQ(1U, session1
->windows
.size());
291 ASSERT_EQ(4U, session1
->windows
[0]->tabs
.size());
292 ASSERT_EQ(1U, session2
->windows
.size());
293 ASSERT_EQ(1U, session2
->windows
[2]->tabs
.size());
294 ASSERT_EQ(2U, tracker
.num_synced_sessions());
295 ASSERT_EQ(4U, tracker
.num_synced_tabs(tag1
));
296 ASSERT_EQ(1U, tracker
.num_synced_tabs(tag2
));
298 // All memory should be properly deallocated by destructor for the
299 // SyncedSessionTracker.
302 } // namespace browser_sync