1 // Copyright (c) 2011 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.
4
5 #include "chrome/browser/profiles/profile_metrics.h"
6
7 #include "base/files/file_path.h"
8 #include "base/logging.h"
9 #include "base/metrics/histogram.h"
10 #include "chrome/browser/browser_process.h"
11 #include "chrome/browser/profiles/profile.h"
12 #include "chrome/browser/profiles/profile_info_cache.h"
13 #include "chrome/browser/profiles/profile_manager.h"
14 #include "chrome/browser/signin/signin_header_helper.h"
15 #include "chrome/common/chrome_constants.h"
16 #include "chrome/installer/util/google_update_settings.h"
17 #include "content/public/browser/browser_thread.h"
18 #include "content/public/browser/user_metrics.h"
19
20 namespace {
21
22 const int kMaximumReportedProfileCount = 5;
23 const int kMaximumDaysOfDisuse = 4 * 7; // Should be integral number of weeks.
24
GetProfileType(const base::FilePath & profile_path)25 ProfileMetrics::ProfileType GetProfileType(
26 const base::FilePath& profile_path) {
27 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI));
28 ProfileMetrics::ProfileType metric = ProfileMetrics::SECONDARY;
29 ProfileManager* manager = g_browser_process->profile_manager();
30 base::FilePath user_data_dir;
31 // In unittests, we do not always have a profile_manager so check.
32 if (manager) {
33 user_data_dir = manager->user_data_dir();
34 }
35 if (profile_path == user_data_dir.AppendASCII(chrome::kInitialProfile)) {
36 metric = ProfileMetrics::ORIGINAL;
37 }
38 return metric;
39 }
40
UpdateReportedOSProfileStatistics(int active,int signedin)41 void UpdateReportedOSProfileStatistics(int active, int signedin) {
42 #if defined(OS_WIN)
43 GoogleUpdateSettings::UpdateProfileCounts(active, signedin);
44 #endif
45 }
46
LogLockedProfileInformation(ProfileManager * manager)47 void LogLockedProfileInformation(ProfileManager* manager) {
48 const ProfileInfoCache& info_cache = manager->GetProfileInfoCache();
49 size_t number_of_profiles = info_cache.GetNumberOfProfiles();
50
51 base::Time now = base::Time::Now();
52 const int kMinutesInProfileValidDuration =
53 base::TimeDelta::FromDays(28).InMinutes();
54 for (size_t i = 0; i < number_of_profiles; ++i) {
55 // Find when locked profiles were locked
56 if (info_cache.ProfileIsSigninRequiredAtIndex(i)) {
57 base::TimeDelta time_since_lock = now -
58 info_cache.GetProfileActiveTimeAtIndex(i);
59 // Specifying 100 buckets for the histogram to get a higher level of
60 // granularity in the reported data, given the large number of possible
61 // values (kMinutesInProfileValidDuration > 40,000).
62 UMA_HISTOGRAM_CUSTOM_COUNTS("Profile.LockedProfilesDuration",
63 time_since_lock.InMinutes(),
64 1,
65 kMinutesInProfileValidDuration,
66 100);
67 }
68 }
69 }
70
HasProfileAtIndexBeenActiveSince(const ProfileInfoCache & info_cache,int index,const base::Time & active_limit)71 bool HasProfileAtIndexBeenActiveSince(const ProfileInfoCache& info_cache,
72 int index,
73 const base::Time& active_limit) {
74 #if !defined(OS_ANDROID) && !defined(OS_IOS)
75 // TODO(mlerman): iOS and Android should set an ActiveTime in the
76 // ProfileInfoCache. (see ProfileManager::OnBrowserSetLastActive)
77 if (info_cache.GetProfileActiveTimeAtIndex(index) < active_limit)
78 return false;
79 #endif
80 return true;
81 }
82
83 } // namespace
84
85 enum ProfileAvatar {
86 AVATAR_GENERIC = 0, // The names for avatar icons
87 AVATAR_GENERIC_AQUA,
88 AVATAR_GENERIC_BLUE,
89 AVATAR_GENERIC_GREEN,
90 AVATAR_GENERIC_ORANGE,
91 AVATAR_GENERIC_PURPLE,
92 AVATAR_GENERIC_RED,
93 AVATAR_GENERIC_YELLOW,
94 AVATAR_SECRET_AGENT,
95 AVATAR_SUPERHERO,
96 AVATAR_VOLLEYBALL, // 10
97 AVATAR_BUSINESSMAN,
98 AVATAR_NINJA,
99 AVATAR_ALIEN,
100 AVATAR_AWESOME,
101 AVATAR_FLOWER,
102 AVATAR_PIZZA,
103 AVATAR_SOCCER,
104 AVATAR_BURGER,
105 AVATAR_CAT,
106 AVATAR_CUPCAKE, // 20
107 AVATAR_DOG,
108 AVATAR_HORSE,
109 AVATAR_MARGARITA,
110 AVATAR_NOTE,
111 AVATAR_SUN_CLOUD,
112 AVATAR_PLACEHOLDER,
113 AVATAR_UNKNOWN, // 27
114 AVATAR_GAIA, // 28
115 NUM_PROFILE_AVATAR_METRICS
116 };
117
CountProfileInformation(ProfileManager * manager,ProfileCounts * counts)118 bool ProfileMetrics::CountProfileInformation(ProfileManager* manager,
119 ProfileCounts* counts) {
120 const ProfileInfoCache& info_cache = manager->GetProfileInfoCache();
121 size_t number_of_profiles = info_cache.GetNumberOfProfiles();
122 counts->total = number_of_profiles;
123
124 // Ignore other metrics if we have no profiles, e.g. in Chrome Frame tests.
125 if (!number_of_profiles)
126 return false;
127
128 // Maximum age for "active" profile is 4 weeks.
129 base::Time oldest = base::Time::Now() -
130 base::TimeDelta::FromDays(kMaximumDaysOfDisuse);
131
132 for (size_t i = 0; i < number_of_profiles; ++i) {
133 if (!HasProfileAtIndexBeenActiveSince(info_cache, i, oldest)) {
134 counts->unused++;
135 } else {
136 if (info_cache.ProfileIsSupervisedAtIndex(i))
137 counts->supervised++;
138 if (!info_cache.GetUserNameOfProfileAtIndex(i).empty()) {
139 counts->signedin++;
140 if (info_cache.IsUsingGAIAPictureOfProfileAtIndex(i))
141 counts->gaia_icon++;
142 }
143 }
144 }
145 return true;
146 }
147
148
UpdateReportedProfilesStatistics(ProfileManager * manager)149 void ProfileMetrics::UpdateReportedProfilesStatistics(ProfileManager* manager) {
150 ProfileCounts counts;
151 if (CountProfileInformation(manager, &counts)) {
152 int limited_total = counts.total;
153 int limited_signedin = counts.signedin;
154 if (limited_total > kMaximumReportedProfileCount) {
155 limited_total = kMaximumReportedProfileCount + 1;
156 limited_signedin =
157 (int)((float)(counts.signedin * limited_total)
158 / counts.total + 0.5);
159 }
160 UpdateReportedOSProfileStatistics(limited_total, limited_signedin);
161 }
162 }
163
LogNumberOfProfiles(ProfileManager * manager)164 void ProfileMetrics::LogNumberOfProfiles(ProfileManager* manager) {
165 ProfileCounts counts;
166 bool success = CountProfileInformation(manager, &counts);
167 UMA_HISTOGRAM_COUNTS_100("Profile.NumberOfProfiles", counts.total);
168
169 // Ignore other metrics if we have no profiles, e.g. in Chrome Frame tests.
170 if (success) {
171 UMA_HISTOGRAM_COUNTS_100("Profile.NumberOfManagedProfiles",
172 counts.supervised);
173 UMA_HISTOGRAM_COUNTS_100("Profile.PercentageOfManagedProfiles",
174 100 * counts.supervised / counts.total);
175 UMA_HISTOGRAM_COUNTS_100("Profile.NumberOfSignedInProfiles",
176 counts.signedin);
177 UMA_HISTOGRAM_COUNTS_100("Profile.NumberOfUnusedProfiles",
178 counts.unused);
179 UMA_HISTOGRAM_COUNTS_100("Profile.NumberOfSignedInProfilesWithGAIAIcons",
180 counts.gaia_icon);
181
182 LogLockedProfileInformation(manager);
183 UpdateReportedOSProfileStatistics(counts.total, counts.signedin);
184 }
185 }
186
LogProfileAddNewUser(ProfileAdd metric)187 void ProfileMetrics::LogProfileAddNewUser(ProfileAdd metric) {
188 DCHECK(metric < NUM_PROFILE_ADD_METRICS);
189 UMA_HISTOGRAM_ENUMERATION("Profile.AddNewUser", metric,
190 NUM_PROFILE_ADD_METRICS);
191 UMA_HISTOGRAM_ENUMERATION("Profile.NetUserCount", ADD_NEW_USER,
192 NUM_PROFILE_NET_METRICS);
193 }
194
LogProfileAvatarSelection(size_t icon_index)195 void ProfileMetrics::LogProfileAvatarSelection(size_t icon_index) {
196 DCHECK(icon_index < NUM_PROFILE_AVATAR_METRICS);
197 ProfileAvatar icon_name = AVATAR_UNKNOWN;
198 switch (icon_index) {
199 case 0:
200 icon_name = AVATAR_GENERIC;
201 break;
202 case 1:
203 icon_name = AVATAR_GENERIC_AQUA;
204 break;
205 case 2:
206 icon_name = AVATAR_GENERIC_BLUE;
207 break;
208 case 3:
209 icon_name = AVATAR_GENERIC_GREEN;
210 break;
211 case 4:
212 icon_name = AVATAR_GENERIC_ORANGE;
213 break;
214 case 5:
215 icon_name = AVATAR_GENERIC_PURPLE;
216 break;
217 case 6:
218 icon_name = AVATAR_GENERIC_RED;
219 break;
220 case 7:
221 icon_name = AVATAR_GENERIC_YELLOW;
222 break;
223 case 8:
224 icon_name = AVATAR_SECRET_AGENT;
225 break;
226 case 9:
227 icon_name = AVATAR_SUPERHERO;
228 break;
229 case 10:
230 icon_name = AVATAR_VOLLEYBALL;
231 break;
232 case 11:
233 icon_name = AVATAR_BUSINESSMAN;
234 break;
235 case 12:
236 icon_name = AVATAR_NINJA;
237 break;
238 case 13:
239 icon_name = AVATAR_ALIEN;
240 break;
241 case 14:
242 icon_name = AVATAR_AWESOME;
243 break;
244 case 15:
245 icon_name = AVATAR_FLOWER;
246 break;
247 case 16:
248 icon_name = AVATAR_PIZZA;
249 break;
250 case 17:
251 icon_name = AVATAR_SOCCER;
252 break;
253 case 18:
254 icon_name = AVATAR_BURGER;
255 break;
256 case 19:
257 icon_name = AVATAR_CAT;
258 break;
259 case 20:
260 icon_name = AVATAR_CUPCAKE;
261 break;
262 case 21:
263 icon_name = AVATAR_DOG;
264 break;
265 case 22:
266 icon_name = AVATAR_HORSE;
267 break;
268 case 23:
269 icon_name = AVATAR_MARGARITA;
270 break;
271 case 24:
272 icon_name = AVATAR_NOTE;
273 break;
274 case 25:
275 icon_name = AVATAR_SUN_CLOUD;
276 break;
277 case 26:
278 icon_name = AVATAR_PLACEHOLDER;
279 break;
280 case 28:
281 icon_name = AVATAR_GAIA;
282 break;
283 default: // We should never actually get here.
284 NOTREACHED();
285 break;
286 }
287 UMA_HISTOGRAM_ENUMERATION("Profile.Avatar", icon_name,
288 NUM_PROFILE_AVATAR_METRICS);
289 }
290
LogProfileDeleteUser(ProfileDelete metric)291 void ProfileMetrics::LogProfileDeleteUser(ProfileDelete metric) {
292 DCHECK(metric < NUM_DELETE_PROFILE_METRICS);
293 UMA_HISTOGRAM_ENUMERATION("Profile.DeleteProfileAction", metric,
294 NUM_DELETE_PROFILE_METRICS);
295 UMA_HISTOGRAM_ENUMERATION("Profile.NetUserCount", PROFILE_DELETED,
296 NUM_PROFILE_NET_METRICS);
297 }
298
LogProfileOpenMethod(ProfileOpen metric)299 void ProfileMetrics::LogProfileOpenMethod(ProfileOpen metric) {
300 DCHECK(metric < NUM_PROFILE_OPEN_METRICS);
301 UMA_HISTOGRAM_ENUMERATION("Profile.OpenMethod", metric,
302 NUM_PROFILE_OPEN_METRICS);
303 }
304
LogProfileSwitchGaia(ProfileGaia metric)305 void ProfileMetrics::LogProfileSwitchGaia(ProfileGaia metric) {
306 if (metric == GAIA_OPT_IN)
307 LogProfileAvatarSelection(AVATAR_GAIA);
308 UMA_HISTOGRAM_ENUMERATION("Profile.SwitchGaiaPhotoSettings",
309 metric,
310 NUM_PROFILE_GAIA_METRICS);
311 }
312
LogProfileSwitchUser(ProfileOpen metric)313 void ProfileMetrics::LogProfileSwitchUser(ProfileOpen metric) {
314 DCHECK(metric < NUM_PROFILE_OPEN_METRICS);
315 UMA_HISTOGRAM_ENUMERATION("Profile.OpenMethod", metric,
316 NUM_PROFILE_OPEN_METRICS);
317 }
318
LogProfileSyncInfo(ProfileSync metric)319 void ProfileMetrics::LogProfileSyncInfo(ProfileSync metric) {
320 DCHECK(metric < NUM_PROFILE_SYNC_METRICS);
321 UMA_HISTOGRAM_ENUMERATION("Profile.SyncCustomize", metric,
322 NUM_PROFILE_SYNC_METRICS);
323 }
324
LogProfileAuthResult(ProfileAuth metric)325 void ProfileMetrics::LogProfileAuthResult(ProfileAuth metric) {
326 UMA_HISTOGRAM_ENUMERATION("Profile.AuthResult", metric,
327 NUM_PROFILE_AUTH_METRICS);
328 }
329
LogProfileDesktopMenu(ProfileDesktopMenu metric,signin::GAIAServiceType gaia_service)330 void ProfileMetrics::LogProfileDesktopMenu(
331 ProfileDesktopMenu metric,
332 signin::GAIAServiceType gaia_service) {
333 // The first parameter to the histogram needs to be literal, because of the
334 // optimized implementation of |UMA_HISTOGRAM_ENUMERATION|. Do not attempt
335 // to refactor.
336 switch (gaia_service) {
337 case signin::GAIA_SERVICE_TYPE_NONE:
338 UMA_HISTOGRAM_ENUMERATION("Profile.DesktopMenu.NonGAIA", metric,
339 NUM_PROFILE_DESKTOP_MENU_METRICS);
340 break;
341 case signin::GAIA_SERVICE_TYPE_SIGNOUT:
342 UMA_HISTOGRAM_ENUMERATION("Profile.DesktopMenu.GAIASignout", metric,
343 NUM_PROFILE_DESKTOP_MENU_METRICS);
344 break;
345 case signin::GAIA_SERVICE_TYPE_INCOGNITO:
346 UMA_HISTOGRAM_ENUMERATION("Profile.DesktopMenu.GAIAIncognito",
347 metric, NUM_PROFILE_DESKTOP_MENU_METRICS);
348 break;
349 case signin::GAIA_SERVICE_TYPE_ADDSESSION:
350 UMA_HISTOGRAM_ENUMERATION("Profile.DesktopMenu.GAIAAddSession", metric,
351 NUM_PROFILE_DESKTOP_MENU_METRICS);
352 break;
353 case signin::GAIA_SERVICE_TYPE_REAUTH:
354 UMA_HISTOGRAM_ENUMERATION("Profile.DesktopMenu.GAIAReAuth", metric,
355 NUM_PROFILE_DESKTOP_MENU_METRICS);
356 break;
357 case signin::GAIA_SERVICE_TYPE_SIGNUP:
358 UMA_HISTOGRAM_ENUMERATION("Profile.DesktopMenu.GAIASignup", metric,
359 NUM_PROFILE_DESKTOP_MENU_METRICS);
360 break;
361 case signin::GAIA_SERVICE_TYPE_DEFAULT:
362 UMA_HISTOGRAM_ENUMERATION("Profile.DesktopMenu.GAIADefault", metric,
363 NUM_PROFILE_DESKTOP_MENU_METRICS);
364 break;
365 }
366 }
367
LogProfileDelete(bool profile_was_signed_in)368 void ProfileMetrics::LogProfileDelete(bool profile_was_signed_in) {
369 UMA_HISTOGRAM_BOOLEAN("Profile.Delete", profile_was_signed_in);
370 }
371
LogProfileNewAvatarMenuNotYou(ProfileNewAvatarMenuNotYou metric)372 void ProfileMetrics::LogProfileNewAvatarMenuNotYou(
373 ProfileNewAvatarMenuNotYou metric) {
374 DCHECK_LT(metric, NUM_PROFILE_AVATAR_MENU_NOT_YOU_METRICS);
375 UMA_HISTOGRAM_ENUMERATION("Profile.NewAvatarMenu.NotYou", metric,
376 NUM_PROFILE_AVATAR_MENU_NOT_YOU_METRICS);
377 }
378
LogProfileNewAvatarMenuSignin(ProfileNewAvatarMenuSignin metric)379 void ProfileMetrics::LogProfileNewAvatarMenuSignin(
380 ProfileNewAvatarMenuSignin metric) {
381 DCHECK_LT(metric, NUM_PROFILE_AVATAR_MENU_SIGNIN_METRICS);
382 UMA_HISTOGRAM_ENUMERATION("Profile.NewAvatarMenu.Signin", metric,
383 NUM_PROFILE_AVATAR_MENU_SIGNIN_METRICS);
384 }
385
LogProfileNewAvatarMenuUpgrade(ProfileNewAvatarMenuUpgrade metric)386 void ProfileMetrics::LogProfileNewAvatarMenuUpgrade(
387 ProfileNewAvatarMenuUpgrade metric) {
388 DCHECK_LT(metric, NUM_PROFILE_AVATAR_MENU_UPGRADE_METRICS);
389 UMA_HISTOGRAM_ENUMERATION("Profile.NewAvatarMenu.Upgrade", metric,
390 NUM_PROFILE_AVATAR_MENU_UPGRADE_METRICS);
391 }
392
393 #if defined(OS_ANDROID)
LogProfileAndroidAccountManagementMenu(ProfileAndroidAccountManagementMenu metric,signin::GAIAServiceType gaia_service)394 void ProfileMetrics::LogProfileAndroidAccountManagementMenu(
395 ProfileAndroidAccountManagementMenu metric,
396 signin::GAIAServiceType gaia_service) {
397 // The first parameter to the histogram needs to be literal, because of the
398 // optimized implementation of |UMA_HISTOGRAM_ENUMERATION|. Do not attempt
399 // to refactor.
400 switch (gaia_service) {
401 case signin::GAIA_SERVICE_TYPE_NONE:
402 UMA_HISTOGRAM_ENUMERATION(
403 "Profile.AndroidAccountManagementMenu.NonGAIA",
404 metric,
405 NUM_PROFILE_ANDROID_ACCOUNT_MANAGEMENT_MENU_METRICS);
406 break;
407 case signin::GAIA_SERVICE_TYPE_SIGNOUT:
408 UMA_HISTOGRAM_ENUMERATION(
409 "Profile.AndroidAccountManagementMenu.GAIASignout",
410 metric,
411 NUM_PROFILE_ANDROID_ACCOUNT_MANAGEMENT_MENU_METRICS);
412 break;
413 case signin::GAIA_SERVICE_TYPE_INCOGNITO:
414 UMA_HISTOGRAM_ENUMERATION(
415 "Profile.AndroidAccountManagementMenu.GAIASignoutIncognito",
416 metric,
417 NUM_PROFILE_ANDROID_ACCOUNT_MANAGEMENT_MENU_METRICS);
418 break;
419 case signin::GAIA_SERVICE_TYPE_ADDSESSION:
420 UMA_HISTOGRAM_ENUMERATION(
421 "Profile.AndroidAccountManagementMenu.GAIAAddSession",
422 metric,
423 NUM_PROFILE_ANDROID_ACCOUNT_MANAGEMENT_MENU_METRICS);
424 break;
425 case signin::GAIA_SERVICE_TYPE_REAUTH:
426 UMA_HISTOGRAM_ENUMERATION(
427 "Profile.AndroidAccountManagementMenu.GAIAReAuth",
428 metric,
429 NUM_PROFILE_ANDROID_ACCOUNT_MANAGEMENT_MENU_METRICS);
430 break;
431 case signin::GAIA_SERVICE_TYPE_SIGNUP:
432 UMA_HISTOGRAM_ENUMERATION(
433 "Profile.AndroidAccountManagementMenu.GAIASignup",
434 metric,
435 NUM_PROFILE_ANDROID_ACCOUNT_MANAGEMENT_MENU_METRICS);
436 break;
437 case signin::GAIA_SERVICE_TYPE_DEFAULT:
438 UMA_HISTOGRAM_ENUMERATION(
439 "Profile.AndroidAccountManagementMenu.GAIADefault",
440 metric,
441 NUM_PROFILE_ANDROID_ACCOUNT_MANAGEMENT_MENU_METRICS);
442 break;
443 }
444 }
445 #endif // defined(OS_ANDROID)
446
LogProfileLaunch(Profile * profile)447 void ProfileMetrics::LogProfileLaunch(Profile* profile) {
448 base::FilePath profile_path = profile->GetPath();
449 UMA_HISTOGRAM_ENUMERATION("Profile.LaunchBrowser",
450 GetProfileType(profile_path),
451 NUM_PROFILE_TYPE_METRICS);
452
453 if (profile->IsSupervised()) {
454 content::RecordAction(
455 base::UserMetricsAction("ManagedMode_NewManagedUserWindow"));
456 }
457 }
458
LogProfileSyncSignIn(const base::FilePath & profile_path)459 void ProfileMetrics::LogProfileSyncSignIn(const base::FilePath& profile_path) {
460 UMA_HISTOGRAM_ENUMERATION("Profile.SyncSignIn",
461 GetProfileType(profile_path),
462 NUM_PROFILE_TYPE_METRICS);
463 }
464
LogProfileUpdate(const base::FilePath & profile_path)465 void ProfileMetrics::LogProfileUpdate(const base::FilePath& profile_path) {
466 UMA_HISTOGRAM_ENUMERATION("Profile.Update",
467 GetProfileType(profile_path),
468 NUM_PROFILE_TYPE_METRICS);
469 }
470