diff options
author | petersont@google.com <petersont@google.com@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-10-16 02:18:07 +0000 |
---|---|---|
committer | petersont@google.com <petersont@google.com@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-10-16 02:18:07 +0000 |
commit | d01b873dae981f54da8b3d0a818cb0c487d56ec6 (patch) | |
tree | e82f49c73a0ae789c96c5752bc239e45861c3dea | |
parent | 41aa0e41f6b14d4d6c847a2498994ef382d51f38 (diff) | |
download | chromium_src-d01b873dae981f54da8b3d0a818cb0c487d56ec6.zip chromium_src-d01b873dae981f54da8b3d0a818cb0c487d56ec6.tar.gz chromium_src-d01b873dae981f54da8b3d0a818cb0c487d56ec6.tar.bz2 |
This is the same change as 2419 minus a line of debug code I accidentally left in browser_main.cc which ignored the user preference.
Review URL: http://codereview.chromium.org/4229
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@3448 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r-- | chrome/browser/browser_main.cc | 21 | ||||
-rw-r--r-- | chrome/browser/metrics_log.cc | 1 | ||||
-rw-r--r-- | chrome/browser/metrics_service.cc | 497 | ||||
-rw-r--r-- | chrome/browser/metrics_service.h | 168 | ||||
-rw-r--r-- | chrome/browser/views/options/advanced_contents_view.cc | 63 | ||||
-rw-r--r-- | chrome/common/chrome_switches.cc | 12 | ||||
-rw-r--r-- | chrome/common/chrome_switches.h | 2 | ||||
-rw-r--r-- | chrome/test/ui/ui_test.cc | 2 |
8 files changed, 575 insertions, 191 deletions
diff --git a/chrome/browser/browser_main.cc b/chrome/browser/browser_main.cc index 049a402..6e819b2 100644 --- a/chrome/browser/browser_main.cc +++ b/chrome/browser/browser_main.cc @@ -531,14 +531,27 @@ int BrowserMain(CommandLine &parsed_command_line, int show_command, MetricsService* metrics = NULL; if (!parsed_command_line.HasSwitch(switches::kDisableMetrics)) { - if (parsed_command_line.HasSwitch(switches::kDisableMetricsReporting)) { + if (parsed_command_line.HasSwitch(switches::kMetricsRecordingOnly)) { local_state->transient()->SetBoolean(prefs::kMetricsReportingEnabled, false); } metrics = browser_process->metrics_service(); DCHECK(metrics); - // Start user experience metrics recording, if enabled. - metrics->SetRecording(local_state->GetBoolean(prefs::kMetricsIsRecording)); + + // If we're testing then we don't care what the user + // preference is, we turn on recording, but not reporting, otherwise tests + // fail. + if (parsed_command_line.HasSwitch(switches::kMetricsRecordingOnly)) { + metrics->StartRecordingOnly(); + } else { + // If the user permits metrics reporting with the checkbox in the + // prefs, we turn on recording. + bool enabled = local_state->GetBoolean(prefs::kMetricsReportingEnabled); + + metrics->SetUserPermitsUpload(enabled); + if (enabled) + metrics->Start(); + } } InstallJankometer(parsed_command_line); @@ -559,7 +572,7 @@ int BrowserMain(CommandLine &parsed_command_line, int show_command, } if (metrics) - metrics->SetRecording(false); // Force persistent save. + metrics->Stop(); // browser_shutdown takes care of deleting browser_process, so we need to // release it. diff --git a/chrome/browser/metrics_log.cc b/chrome/browser/metrics_log.cc index a1d8f67..a68ca2c5 100644 --- a/chrome/browser/metrics_log.cc +++ b/chrome/browser/metrics_log.cc @@ -28,7 +28,6 @@ inline const unsigned char* UnsignedChar(const char* input) { // static void MetricsLog::RegisterPrefs(PrefService* local_state) { local_state->RegisterListPref(prefs::kStabilityPluginStats); - local_state->RegisterBooleanPref(prefs::kMetricsIsRecording, true); } MetricsLog::MetricsLog(const std::string& client_id, int session_id) diff --git a/chrome/browser/metrics_service.cc b/chrome/browser/metrics_service.cc index d40f078..cefe611 100644 --- a/chrome/browser/metrics_service.cc +++ b/chrome/browser/metrics_service.cc @@ -195,7 +195,6 @@ static const char kMetricsType[] = "application/vnd.mozilla.metrics.bz2"; static const int kInitialInterlogDuration = 60; // one minute // The default maximum number of events in a log uploaded to the UMA server. -// TODO(petersont): Honor the limit when the log is actually sent. static const int kInitialEventLimit = 600; // If an upload fails, and the transmission was over this byte count, then we @@ -327,19 +326,22 @@ void MetricsService::RegisterPrefs(PrefService* local_state) { } MetricsService::MetricsService() - : recording_(false), - reporting_(true), + : recording_active_(false), + reporting_active_(false), + user_permits_upload_(false), + server_permits_upload_(true), + state_(INITIALIZED), pending_log_(NULL), pending_log_text_(""), current_fetch_(NULL), current_log_(NULL), - state_(INITIALIZED), + idle_since_last_transmission_(false), next_window_id_(0), log_sender_factory_(this), state_saver_factory_(this), logged_samples_(), interlog_duration_(TimeDelta::FromSeconds(kInitialInterlogDuration)), - event_limit_(kInitialEventLimit), + log_event_limit_(kInitialEventLimit), timer_pending_(false) { DCHECK(IsSingleThreaded()); InitializeMetricsState(); @@ -349,10 +351,30 @@ MetricsService::~MetricsService() { SetRecording(false); } +void MetricsService::SetUserPermitsUpload(bool enabled) { + HandleIdleSinceLastTransmission(false); + user_permits_upload_ = enabled; +} + +void MetricsService::Start() { + SetRecording(true); + SetReporting(true); +} + +void MetricsService::StartRecordingOnly() { + SetRecording(true); + SetReporting(false); +} + +void MetricsService::Stop() { + SetReporting(false); + SetRecording(false); +} + void MetricsService::SetRecording(bool enabled) { DCHECK(IsSingleThreaded()); - if (enabled == recording_) + if (enabled == recording_active_) return; if (enabled) { @@ -366,29 +388,25 @@ void MetricsService::SetRecording(bool enabled) { if (state_ > INITIAL_LOG_READY && unsent_logs()) state_ = SEND_OLD_INITIAL_LOGS; } - recording_ = enabled; + recording_active_ = enabled; } -bool MetricsService::IsRecording() const { +bool MetricsService::recording_active() const { DCHECK(IsSingleThreaded()); - return recording_; + return recording_active_; } -bool MetricsService::EnableReporting(bool enable) { - bool done = GoogleUpdateSettings::SetCollectStatsConsent(enable); - if (!done) { - bool update_pref = GoogleUpdateSettings::GetCollectStatsConsent(); - if (enable != update_pref) { - DLOG(INFO) << "METRICS: Unable to set crash report status to " << enable; - return false; - } - } - if (reporting_ != enable) { - reporting_ = enable; - if (reporting_) +void MetricsService::SetReporting(bool enable) { + if (reporting_active_ != enable) { + reporting_active_ = enable; + if (reporting_active_) StartLogTransmissionTimer(); } - return true; +} + +bool MetricsService::reporting_active() const { + DCHECK(IsSingleThreaded()); + return reporting_active_; } void MetricsService::Observe(NotificationType type, @@ -459,7 +477,25 @@ void MetricsService::Observe(NotificationType type, NOTREACHED(); break; } - StartLogTransmissionTimer(); + + HandleIdleSinceLastTransmission(false); + + if (current_log_) + DLOG(INFO) << "METRICS: NUMBER OF LOGS = " << current_log_->num_events(); +} + +void MetricsService::HandleIdleSinceLastTransmission(bool in_idle) { + // If there wasn't a lot of action, maybe the computer was asleep, in which + // case, the log transmissions should have stopped. Here we start them up + // again. + if (in_idle) { + idle_since_last_transmission_ = true; + } else { + if (idle_since_last_transmission_) { + idle_since_last_transmission_ = false; + StartLogTransmissionTimer(); + } + } } void MetricsService::RecordCleanShutdown() { @@ -515,9 +551,6 @@ void MetricsService::InitializeMetricsState() { ++session_id_; pref->SetInteger(prefs::kMetricsSessionID, session_id_); - bool done = EnableReporting(GoogleUpdateSettings::GetCollectStatsConsent()); - DCHECK(done); - // Stability bookkeeping IncrementPrefValue(prefs::kStabilityLaunchCount); @@ -643,7 +676,7 @@ void MetricsService::StopRecording(MetricsLog** log) { // TODO(jar): Integrate bounds on log recording more consistently, so that we // can stop recording logs that are too big much sooner. - if (current_log_->num_events() > kInitialEventLimit) { + if (current_log_->num_events() > log_event_limit_) { UMA_HISTOGRAM_COUNTS(L"UMA.Discarded Log Events", current_log_->num_events()); current_log_->CloseLog(); @@ -709,7 +742,7 @@ void MetricsService::PushPendingLogsToUnsentLists() { if (state_ == INITIAL_LOG_READY) { // We may race here, and send second copy of initial log later. unsent_initial_logs_.push_back(pending_log_text_); - state_ = SENDING_CURRENT_LOGS; + state_ = SEND_OLD_INITIAL_LOGS; } else { PushPendingLogTextToUnsentOngoingLogs(); } @@ -724,6 +757,11 @@ void MetricsService::PushPendingLogsToUnsentLists() { } void MetricsService::PushPendingLogTextToUnsentOngoingLogs() { + // If UMA response told us not to upload, there's no need to save the pending + // log. It wasn't supposed to be uploaded anyway. + if (!server_permits_upload_) + return; + if (pending_log_text_.length() > kUploadLogAvoidRetransmitSize) { UMA_HISTOGRAM_COUNTS(L"UMA.Large Accumulated Log Not Persisted", static_cast<int>(pending_log_text_.length())); @@ -736,14 +774,32 @@ void MetricsService::PushPendingLogTextToUnsentOngoingLogs() { // Transmission of logs methods void MetricsService::StartLogTransmissionTimer() { + // If we're not reporting, there's no point in starting a log transmission + // timer. + if (!reporting_active()) + return; + if (!current_log_) return; // Recorder is shutdown. - if (timer_pending_ || !reporting_) + + // If there is already a timer running, we leave it running. + // If timer_pending is true because the fetch is waiting for a response, + // we return for now and let the response handler start the timer. + if (timer_pending_) return; - // If there is no work to do, don't set a timer yet. - if (!current_log_->num_events() && !pending_log() && !unsent_logs()) + + // Finally, if somehow we got here and the program is still idle since the + // last transmission, we shouldn't wake everybody up by starting a timer. + if (idle_since_last_transmission_) return; + + // Before starting the timer, set timer_pending_ to true. timer_pending_ = true; + + // Right before the UMA transmission gets started, there's one more thing we'd + // like to record: the histogram of memory usage, so we spawn a task to + // collect the memory details and when that task is finished, we arrange for + // TryToStartTransmission to take over. MessageLoop::current()->PostDelayedTask(FROM_HERE, log_sender_factory_. NewRunnableMethod(&MetricsService::CollectMemoryDetails), @@ -753,75 +809,152 @@ void MetricsService::StartLogTransmissionTimer() { void MetricsService::TryToStartTransmission() { DCHECK(IsSingleThreaded()); - DCHECK(timer_pending_); // ONLY call via timer. + // This function should only be called via timer, so timer_pending_ + // should be true. + DCHECK(timer_pending_); + timer_pending_ = false; DCHECK(!current_fetch_.get()); - if (current_fetch_.get()) - return; // Redundant defensive coding. - timer_pending_ = false; + // If we're getting no notifications, then the log won't have much in it, and + // it's possible the computer is about to go to sleep, so don't upload and + // don't restart the transmission timer. + if (idle_since_last_transmission_) + return; + + // If somehow there is a fetch in progress, we return setting timer_pending_ + // to true and hope things work out. + if (current_fetch_.get()) { + timer_pending_ = true; + return; + } + + // If uploads are forbidden by UMA response, there's no point in keeping + // the current_log_, and the more often we delete it, the less likely it is + // to expand forever. + if (!server_permits_upload_ && current_log_) { + StopRecording(NULL); + StartRecording(); + } if (!current_log_) return; // Logging was disabled. - if (!reporting_ ) + if (!reporting_active()) return; // Don't do work if we're not going to send anything now. - if (!pending_log()) - switch (state_) { - case INITIALIZED: // We must be further along by now. - DCHECK(false); - return; - - case PLUGIN_LIST_REQUESTED: - StartLogTransmissionTimer(); - return; - - case PLUGIN_LIST_ARRIVED: - // We need to wait for the initial log to be ready before sending - // anything, because the server will tell us whether it wants to hear - // from us. - PrepareInitialLog(); - DCHECK(state_ == PLUGIN_LIST_ARRIVED); - RecallUnsentLogs(); - state_ = INITIAL_LOG_READY; - break; + MakePendingLog(); - case SEND_OLD_INITIAL_LOGS: + // MakePendingLog should have put something in the pending log, if it didn't, + // we start the timer again, return and hope things work out. + if (!pending_log()) { + StartLogTransmissionTimer(); + return; + } + + // If we're not supposed to upload any UMA data because the response or the + // user said so, cancel the upload at this point, but start the timer. + if (!TransmissionPermitted()) { + DiscardPendingLog(); + StartLogTransmissionTimer(); + return; + } + + PrepareFetchWithPendingLog(); + + if (!current_fetch_.get()) { + // Compression failed, and log discarded :-/. + DiscardPendingLog(); + StartLogTransmissionTimer(); // Maybe we'll do better next time + // TODO(jar): If compression failed, we should have created a tiny log and + // compressed that, so that we can signal that we're losing logs. + return; + } + + DCHECK(!timer_pending_); + + // The URL fetch is a like timer in that after a while we get called back + // so we set timer_pending_ true just as we start the url fetch. + timer_pending_ = true; + current_fetch_->Start(); + + HandleIdleSinceLastTransmission(true); +} + + +void MetricsService::MakePendingLog() { + if (pending_log()) + return; + + switch (state_) { + case INITIALIZED: + case PLUGIN_LIST_REQUESTED: // We should be further along by now. + DCHECK(false); + return; + + case PLUGIN_LIST_ARRIVED: + // We need to wait for the initial log to be ready before sending + // anything, because the server will tell us whether it wants to hear + // from us. + PrepareInitialLog(); + DCHECK(state_ == PLUGIN_LIST_ARRIVED); + RecallUnsentLogs(); + state_ = INITIAL_LOG_READY; + break; + + case SEND_OLD_INITIAL_LOGS: if (!unsent_initial_logs_.empty()) { pending_log_text_ = unsent_initial_logs_.back(); break; } - state_ = SENDING_OLD_LOGS; - // Fall through. - - case SENDING_OLD_LOGS: - if (!unsent_ongoing_logs_.empty()) { - pending_log_text_ = unsent_ongoing_logs_.back(); - break; - } - state_ = SENDING_CURRENT_LOGS; - // Fall through. + state_ = SENDING_OLD_LOGS; + // Fall through. - case SENDING_CURRENT_LOGS: - if (!current_log_->num_events()) - return; // Nothing to send. - StopRecording(&pending_log_); - StartRecording(); + case SENDING_OLD_LOGS: + if (!unsent_ongoing_logs_.empty()) { + pending_log_text_ = unsent_ongoing_logs_.back(); break; + } + state_ = SENDING_CURRENT_LOGS; + // Fall through. - default: - DCHECK(false); - return; + case SENDING_CURRENT_LOGS: + StopRecording(&pending_log_); + StartRecording(); + break; + + default: + DCHECK(false); + return; } + DCHECK(pending_log()); +} - PreparePendingLogForTransmission(); - if (!current_fetch_.get()) - return; // Compression failed, and log discarded :-/. +bool MetricsService::TransmissionPermitted() const { + // If the user forbids uploading that's they're business, and we don't upload + // anything. If the server forbids uploading, that's our business, so we take + // that to mean it forbids current logs, but we still send up the inital logs + // and any old logs. - DCHECK(!timer_pending_); - timer_pending_ = true; // The URL fetch is a pseudo timer. - current_fetch_->Start(); + if (!user_permits_upload_) + return false; + + if (server_permits_upload_) { + return true; + } else { + switch (state_) { + case INITIAL_LOG_READY: + case SEND_OLD_INITIAL_LOGS: + case SENDING_OLD_LOGS: + return true; + + case SENDING_CURRENT_LOGS: + default: + return false; + } + } + + return false; } void MetricsService::CollectMemoryDetails() { @@ -919,7 +1052,7 @@ void MetricsService::PreparePendingLogText() { original_size); } -void MetricsService::PreparePendingLogForTransmission() { +void MetricsService::PrepareFetchWithPendingLog() { DCHECK(pending_log()); DCHECK(!current_fetch_.get()); PreparePendingLogText(); @@ -1042,6 +1175,7 @@ void MetricsService::OnURLFetchComplete(const URLFetcher* source, if (response_code != 200) { HandleBadResponseCode(); } else { // Success. + DLOG(INFO) << "METRICS RESPONSE DATA: " << data; switch (state_) { case INITIAL_LOG_READY: state_ = SEND_OLD_INITIAL_LOGS; @@ -1066,7 +1200,7 @@ void MetricsService::OnURLFetchComplete(const URLFetcher* source, DCHECK(false); break; } - DLOG(INFO) << "METRICS RESPONSE DATA: " << data; + DiscardPendingLog(); // Since we sent a log, make sure our in-memory state is recorded to disk. PrefService* local_state = g_browser_process->local_state(); @@ -1115,9 +1249,10 @@ void MetricsService::HandleBadResponseCode() { void MetricsService::GetSettingsFromResponseData(const std::string& data) { // We assume that the file is structured as a block opened by <response> - // and that inside response, there is a block opened by tag <config> - // other tags are ignored for now except the content of <config>. - DLOG(INFO) << data; + // and that inside response, there is a block opened by tag <chrome_config> + // other tags are ignored for now except the content of <chrome_config>. + DLOG(INFO) << "METRICS: getting settings from response data: " << data; + int data_size = static_cast<int>(data.size()); if (data_size < 0) { DLOG(INFO) << "METRICS: server response data bad size " << @@ -1127,60 +1262,171 @@ void MetricsService::GetSettingsFromResponseData(const std::string& data) { xmlDocPtr doc = xmlReadMemory(data.c_str(), data_size, "", NULL, 0); DCHECK(doc); - // if the document is malformed, we just use the settings that were there - if (!doc) + // If the document is malformed, we just use the settings that were there. + if (!doc) { + DLOG(INFO) << "METRICS: reading xml from server response data failed"; return; + } - xmlNodePtr top_node = xmlDocGetRootElement(doc), config_node = NULL; - // Here, we find the config node by name. + xmlNodePtr top_node = xmlDocGetRootElement(doc), chrome_config_node = NULL; + // Here, we find the chrome_config node by name. for (xmlNodePtr p = top_node->children; p; p = p->next) { - if (xmlStrEqual(p->name, BAD_CAST "config")) { - config_node = p; + if (xmlStrEqual(p->name, BAD_CAST "chrome_config")) { + chrome_config_node = p; break; } } // If the server data is formatted wrong and there is no // config node where we expect, we just drop out. - if (config_node != NULL) - GetSettingsFromConfigNode(config_node); + if (chrome_config_node != NULL) + GetSettingsFromChromeConfigNode(chrome_config_node); xmlFreeDoc(doc); } -void MetricsService::GetSettingsFromConfigNode(xmlNodePtr config_node) { - for (xmlNodePtr current_node = config_node->children; - current_node; - current_node = current_node->next) { - // If the node is collectors list, we iterate through the children - // to get the types of collectors. - if (xmlStrEqual(current_node->name, BAD_CAST "collectors")) { - collectors_.clear(); - // Iterate through children and get the property "type". - for (xmlNodePtr sub_node = current_node->children; - sub_node; - sub_node = sub_node->next) { - if (xmlStrEqual(sub_node->name, BAD_CAST "collector")) { - xmlChar* type_value = xmlGetProp(sub_node, BAD_CAST "type"); - collectors_.insert(reinterpret_cast<char*>(type_value)); - } - } +void MetricsService::GetSettingsFromChromeConfigNode( + xmlNodePtr chrome_config_node) { + // Iterate through all children of the config node. + for (xmlNodePtr current_node = chrome_config_node->children; + current_node; + current_node = current_node->next) { + // If we find the upload tag, we appeal to another function + // GetSettingsFromUploadNode to read all the data in it. + if (xmlStrEqual(current_node->name, BAD_CAST "upload")) { + GetSettingsFromUploadNode(current_node); continue; } - // Search for other tags, limit and upload. Again if the server data - // does not contain those tags, the settings remain unchanged. - if (xmlStrEqual(current_node->name, BAD_CAST "limit")) { - xmlChar* event_limit_value = xmlGetProp(current_node, BAD_CAST "events"); - event_limit_ = atoi(reinterpret_cast<char*>(event_limit_value)); - continue; + } +} + +void MetricsService::InheritedProperties::OverwriteWhereNeeded( + xmlNodePtr node) { + xmlChar* salt_value = xmlGetProp(node, BAD_CAST "salt"); + if (salt_value) // If the property isn't there, xmlGetProp returns NULL. + salt = atoi(reinterpret_cast<char*>(salt_value)); + // If the property isn't there, we keep the value the property had before + + xmlChar* denominator_value = xmlGetProp(node, BAD_CAST "denominator"); + if (denominator_value) + denominator = atoi(reinterpret_cast<char*>(denominator_value)); +} + +void MetricsService::GetSettingsFromUploadNode(xmlNodePtr upload_node) { + InheritedProperties props; + GetSettingsFromUploadNodeRecursive(upload_node, props, "", true); +} + +void MetricsService::GetSettingsFromUploadNodeRecursive(xmlNodePtr node, + InheritedProperties props, std::string path_prefix, bool uploadOn) { + props.OverwriteWhereNeeded(node); + + // The bool uploadOn is set to true if the data represented by current + // node should be uploaded. This gets inherited in the tree; the children + // of a node that has already been rejected for upload get rejected for + // upload. + uploadOn = uploadOn && NodeProbabilityTest(node, props); + + // The path is a / separated list of the node names ancestral to the current + // one. So, if you want to check if the current node has a certain name, + // compare to name. If you want to check if it is a certan tag at a certain + // place in the tree, compare to the whole path. + std::string name = std::string(reinterpret_cast<const char*>(node->name)); + std::string path = path_prefix + "/" + name; + + if (path == "/upload") { + xmlChar* upload_interval_val = xmlGetProp(node, BAD_CAST "interval"); + if (upload_interval_val) { + interlog_duration_ = TimeDelta::FromSeconds( + atoi(reinterpret_cast<char*>(upload_interval_val))); } - if (xmlStrEqual(current_node->name, BAD_CAST "upload")) { - xmlChar* upload_interval_val = xmlGetProp(current_node, - BAD_CAST "interval"); - int upload_interval_sec = - atoi(reinterpret_cast<char*>(upload_interval_val)); - interlog_duration_ = TimeDelta::FromSeconds(upload_interval_sec); - continue; + + server_permits_upload_ = uploadOn; + } + if (path == "/upload/logs") { + xmlChar* log_event_limit_val = xmlGetProp(node, BAD_CAST "event_limit"); + if (log_event_limit_val) + log_event_limit_ = atoi(reinterpret_cast<char*>(log_event_limit_val)); + } + if (name == "histogram") { + xmlChar* type_value = xmlGetProp(node, BAD_CAST "type"); + if (type_value) { + std::string type = (reinterpret_cast<char*>(type_value)); + if (uploadOn) + histograms_to_upload_.insert(type); + else + histograms_to_omit_.insert(type); } } + if (name == "log") { + xmlChar* type_value = xmlGetProp(node, BAD_CAST "type"); + if (type_value) { + std::string type = (reinterpret_cast<char*>(type_value)); + if (uploadOn) + logs_to_upload_.insert(type); + else + logs_to_omit_.insert(type); + } + } + + // Recursive call. If the node is a leaf i.e. if it ends in a "/>", then it + // doesn't have children, so node->children is NULL, and this loop doesn't + // call (that's how the recursion ends). + for (xmlNodePtr child_node = node->children; + child_node; + child_node = child_node->next) { + GetSettingsFromUploadNodeRecursive(child_node, props, path, uploadOn); + } +} + +bool MetricsService::NodeProbabilityTest(xmlNodePtr node, + InheritedProperties props) const { + // Default value of probability on any node is 1, but recall that + // its parents can already have been rejected for upload. + double probability = 1; + + // If a probability is specified in the node, we use it instead. + xmlChar* probability_value = xmlGetProp(node, BAD_CAST "probability"); + if (probability_value) + probability = atoi(reinterpret_cast<char*>(probability_value)); + + return ProbabilityTest(probability, props.salt, props.denominator); +} + +bool MetricsService::ProbabilityTest(double probability, + int salt, + int denominator) const { + // Okay, first we figure out how many of the digits of the + // client_id_ we need in order to make a nice pseudorandomish + // number in the range [0,denominator). Too many digits is + // fine. + int relevant_digits = static_cast<int>( + ::log10(static_cast<double>(denominator))+1.0); + + // n is the length of the client_id_ string + size_t n = client_id_.size(); + + // idnumber is a positive integer generated from the client_id_. + // It plus salt is going to give us our pseudorandom number. + int idnumber = 0; + const char* client_id_c_str = client_id_.c_str(); + + // Here we hash the relevant digits of the client_id_ + // string somehow to get a big integer idnumber (could be negative + // from wraparound) + int big = 1; + for (size_t j = n-1; j >= 0; --j) { + idnumber += static_cast<int>(client_id_c_str[j])*big; + big *= 10; + } + + // Mod id number by denominator making sure to get a non-negative + // answer. + idnumber = ((idnumber%denominator)+denominator)%denominator; + + // ((idnumber+salt)%denominator)/denominator is in the range [0,1] + // if it's less than probability we call that an affirmative coin + // toss. + return static_cast<double>((idnumber+salt)%denominator) < + probability*denominator; } void MetricsService::LogWindowChange(NotificationType type, @@ -1448,6 +1694,10 @@ void MetricsService::RecordCurrentHistograms() { histograms.end() != it; it++) { if ((*it)->flags() & kUmaTargetedHistogramFlag) + // TODO(petersont): + // Bug http://code.google.com/p/chromium/issues/detail?id=2739. + // Only record historgrams if they are not + // precluded by the UMA response data. RecordHistogram(**it); } } @@ -1512,4 +1762,3 @@ static bool IsSingleThreaded() { thread_id = GetCurrentThreadId(); return GetCurrentThreadId() == thread_id; } - diff --git a/chrome/browser/metrics_service.h b/chrome/browser/metrics_service.h index a4d8caf..5eec5e7 100644 --- a/chrome/browser/metrics_service.h +++ b/chrome/browser/metrics_service.h @@ -54,26 +54,22 @@ class MetricsService : public NotificationObserver, MetricsService(); virtual ~MetricsService(); + // Sets whether the user permits uploading. The argument of this function + // should match the checkbox in Options. + void SetUserPermitsUpload(bool enabled); + + // Start/stop the metrics recording and uploading machine. These should be + // used on startup and when the user clicks the checkbox in the prefs. + // StartRecordingOnly starts the metrics recording but not reporting, for use + // in tests only. + void Start(); + void StartRecordingOnly(); + void Stop(); + // At startup, prefs needs to be called with a list of all the pref names and // types we'll be using. static void RegisterPrefs(PrefService* local_state); - // Sets and gets whether metrics recording is active. - // SetRecording(false) also forces a persistent save of logging state (if - // anything has been recorded, or transmitted). - void SetRecording(bool enabled); - bool IsRecording() const; - - // Enable/disable transmission of accumulated logs and crash reports (dumps). - // Return value "true" indicates setting was definitively set as requested). - // Return value of "false" indicates that the enable state is effectively - // stuck in the other logical setting. - // Google Update maintains the authoritative preference in the registry, so - // the caller *might* not be able to actually change the setting. - // It is always possible to set this to at least one value, which matches the - // current value reported by querying Google Update. - bool EnableReporting(bool enable); - // Implementation of NotificationObserver virtual void Observe(NotificationType type, const NotificationSource& source, @@ -125,25 +121,55 @@ class MetricsService : public NotificationObserver, class GetPluginListTask; class GetPluginListTaskComplete; + // Sets and gets whether metrics recording is active. + // SetRecording(false) also forces a persistent save of logging state (if + // anything has been recorded, or transmitted). + void SetRecording(bool enabled); + bool recording_active() const; + + // Enable/disable transmission of accumulated logs and crash reports (dumps). + // Return value "true" indicates setting was definitively set as requested). + // Return value of "false" indicates that the enable state is effectively + // stuck in the other logical setting. + // Google Update maintains the authoritative preference in the registry, so + // the caller *might* not be able to actually change the setting. + // It is always possible to set this to at least one value, which matches the + // current value reported by querying Google Update. + void SetReporting(bool enabled); + bool reporting_active() const; + + // If in_idle is true, sets idle_since_last_transmission to true. + // If in_idle is false and idle_since_last_transmission_ is true, sets + // idle_since_last_transmission to false and starts the timer (provided + // starting the timer is permitted). + void HandleIdleSinceLastTransmission(bool in_idle); + // Set up client ID, session ID, etc. void InitializeMetricsState(); + // Generates a new client ID to use to identify self to metrics server. static std::string GenerateClientID(); + // Schedule the next save of LocalState information. This is called // automatically by the task that performs each save to schedule the next one. void ScheduleNextStateSave(); + // Save the LocalState information immediately. This should not be called by // anybody other than the scheduler to avoid doing too many writes. When you // make a change, call ScheduleNextStateSave() instead. void SaveLocalState(); // Called to start recording user experience metrics. + // Constructs a new, empty current_log_. void StartRecording(); + // Called to stop recording user experience metrics. The caller takes // ownership of the resulting MetricsLog object via the log parameter, // or passes in NULL to indicate that the log should simply be deleted. void StopRecording(MetricsLog** log); + void ListenerRegistration(bool start_listening); + // Adds or Removes (depending on the value of is_add) the given observer // to the given notification type for all sources. static void AddOrRemoveObserver(NotificationObserver* observer, @@ -151,7 +177,7 @@ class MetricsService : public NotificationObserver, bool is_add); // Deletes pending_log_ and current_log_, and pushes their text into the - // appropriate unsent_log vectors. + // appropriate unsent_log vectors. Called when Chrome shuts down. void PushPendingLogsToUnsentLists(); // Save the pending_log_text_ persistently in a pref for transmission when we @@ -163,8 +189,20 @@ class MetricsService : public NotificationObserver, // Do not call TryToStartTransmission() directly. // Use StartLogTransmissionTimer() to schedule a call. void TryToStartTransmission(); + + // Takes whatever log should be uploaded next (according to the state_) + // and makes it the pending log. If pending_log_ is not NULL, + // MakePendingLog does nothing and returns. + void MakePendingLog(); + + // Determines from state_ and permissions set out by the server and by + // the user whether the pending_log_ should be sent or discarded. Called by + // TryToStartTransmission. + bool TransmissionPermitted() const; + // Internal function to collect process memory information. void CollectMemoryDetails(); + // Check to see if there is a log that needs to be, or is being, transmitted. bool pending_log() const { return pending_log_ || !pending_log_text_.empty(); @@ -179,8 +217,12 @@ class MetricsService : public NotificationObserver, void RecallUnsentLogs(); // Convert pending_log_ to XML in pending_log_text_ for transmission. void PreparePendingLogText(); + // Convert pending_log_ to XML, compress it, and prepare to pass to server. - void PreparePendingLogForTransmission(); + // Upon return, current_fetch_ should be reset with its upload data set to + // a compressed copy of the pending log. + void PrepareFetchWithPendingLog(); + // Discard pending_log_, and clear pending_log_text_. Called after processing // of this log is complete. void DiscardPendingLog(); @@ -199,14 +241,50 @@ class MetricsService : public NotificationObserver, // a response code not equal to 200. void HandleBadResponseCode(); + // Class to hold all attributes that gets inherited by children in the UMA + // response data xml tree. This is to make it convenient in the + // recursive function that does the tree traversal to pass all such + // data in the recursive call. If you want to add more such attributes, + // add them to this class. + class InheritedProperties { + public: + InheritedProperties() : salt(123123), denominator(1000000) {} + int salt, denominator; + // Notice salt and denominator are inherited from parent nodes, but + // not probability; the default value of probability is 1. + + // When a new node is reached it might have fields which overwrite inherited + // properties for that node (and its children). Call this method to + // overwrite those settings. + void OverwriteWhereNeeded(xmlNodePtr node); + }; + // Called by OnURLFetchComplete with data as the argument // parses the xml returned by the server in the call to OnURLFetchComplete // and extracts settings for subsequent frequency and content of log posts. void GetSettingsFromResponseData(const std::string& data); // This is a helper function for GetSettingsFromResponseData which iterates - // through the xml tree at the level of the <config> node. - void GetSettingsFromConfigNode(xmlNodePtr config_node); + // through the xml tree at the level of the <chrome_config> node. + void GetSettingsFromChromeConfigNode(xmlNodePtr chrome_config_node); + + // GetSettingsFromUploadNode handles iteration over the children of the + // <upload> child of the <chrome_config> node. It calls the recursive + // function GetSettingsFromUploadNodeRecursive which does the actual + // tree traversal. + void GetSettingsFromUploadNode(xmlNodePtr upload_node); + void GetSettingsFromUploadNodeRecursive(xmlNodePtr node, + InheritedProperties props, + std::string path_prefix, + bool uploadOn); + + // NodeProbabilityTest gets called at every node in the tree traversal + // performed by GetSettingsFromUploadNodeRecursive. It determines from + // the inherited attributes (salt, denominator) and the probability + // assiciated with the node whether that node and its contents should + // contribute to the upload. + bool NodeProbabilityTest(xmlNodePtr node, InheritedProperties props) const; + bool ProbabilityTest(double probability, int salt, int denominator) const; // Records a window-related notification. void LogWindowChange(NotificationType type, @@ -255,9 +333,10 @@ class MetricsService : public NotificationObserver, // buffered plugin stability statistics. void RecordCurrentState(PrefService* pref); - // Record complete list of histograms. Called when we close a log. + // Record complete list of histograms into the current log. + // Called when we close a log. void RecordCurrentHistograms(); - // Record a specific histogram. + // Record a specific histogram . void RecordHistogram(const Histogram& histogram); // Logs the initiation of a page load @@ -280,8 +359,21 @@ class MetricsService : public NotificationObserver, // Sets the value of the specified path in prefs and schedules a save. void RecordBooleanPrefValue(const wchar_t* path, bool value); - bool recording_; - bool reporting_; // if false, metrics logs are discarded rather than sent + // Indicate whether recording and reporting are currently happening. + // These should not be set directly, but by calling SetRecording and + // SetReporting. + bool recording_active_; + bool reporting_active_; + + // Coincides with the check box in options window that lets the user control + // whether to upload. + bool user_permits_upload_; + + // The variable server_permits_upload_ is set true when the response + // data forbids uploading. This should coinside with the "die roll" + // with probability in the upload tag of the response data came out + // affirmative. + bool server_permits_upload_; // The progession of states made by the browser are recorded in the following // state. @@ -289,16 +381,25 @@ class MetricsService : public NotificationObserver, // A log that we are currently transmiting, or about to try to transmit. MetricsLog* pending_log_; + // An alternate form of pending_log_. We persistently save this text version // into prefs if we can't transmit it. As a result, sometimes all we have is // the text version (recalled from a previous session). std::string pending_log_text_; + // The outstanding transmission appears as a URL Fetch operation. scoped_ptr<URLFetcher> current_fetch_; + // The log that we are still appending to. MetricsLog* current_log_; + // The identifier that's sent to the server with the log reports. std::string client_id_; + + // Whether the MetricsService object has received any notifications since + // the last time a transmission was sent. + bool idle_since_last_transmission_; + // A number that identifies the how many times the app has been launched. int session_id_; @@ -340,16 +441,19 @@ class MetricsService : public NotificationObserver, // quickly transmit those unsent logs while we continue to build a log. TimeDelta interlog_duration_; - // The maximum number of events which get transmitted in the log. This is - // provided by the UMA server in the server response data. - int event_limit_; + // The maximum number of events which get transmitted in a log. This defaults + // to a constant and otherwise is provided by the UMA server in the server + // response data. + int log_event_limit_; - // The types of data that are to be included in the log. These are called - // "collectors" in the server response data. - std::set<std::string> collectors_; + // The types of data that are to be included in the logs and histograms + // according to the UMA response data. + std::set<std::string> logs_to_upload_; + std::set<std::string> logs_to_omit_; + std::set<std::string> histograms_to_upload_; + std::set<std::string> histograms_to_omit_; - // Indicate that a timer for sending the next log has already been queued, - // or that a URLFetch (i.e., log transmission) is in progress. + // Indicate that a timer for sending the next log has already been queued. bool timer_pending_; DISALLOW_EVIL_CONSTRUCTORS(MetricsService); diff --git a/chrome/browser/views/options/advanced_contents_view.cc b/chrome/browser/views/options/advanced_contents_view.cc index 7df768c..a6fbec6 100644 --- a/chrome/browser/views/options/advanced_contents_view.cc +++ b/chrome/browser/views/options/advanced_contents_view.cc @@ -34,6 +34,7 @@ #include "chrome/common/pref_member.h" #include "chrome/common/pref_names.h" #include "chrome/common/resource_bundle.h" +#include "chrome/installer/util/google_update_settings.h" #include "chrome/views/background.h" #include "chrome/views/checkbox.h" #include "chrome/views/combo_box.h" @@ -323,6 +324,8 @@ class GeneralSection : public AdvancedSection, StringPrefMember auto_open_files_; BooleanPrefMember enable_metrics_recording_; + void ResolveMetricsReportingEnabled(); + DISALLOW_EVIL_CONSTRUCTORS(GeneralSection); }; @@ -342,23 +345,15 @@ void GeneralSection::ButtonPressed(ChromeViews::NativeButton* sender) { profile()->GetPrefs()); } else if (sender == reporting_enabled_checkbox_) { bool enabled = reporting_enabled_checkbox_->IsSelected(); - // Do what we can, but we might not be able to get what was asked for. - bool done = g_browser_process->metrics_service()->EnableReporting(enabled); - if (!done) { - enabled = !enabled; - done = g_browser_process->metrics_service()->EnableReporting(enabled); - DCHECK(done); - reporting_enabled_checkbox_->SetIsSelected(enabled); - } else { - if (enabled) { - UserMetricsRecordAction(L"Options_MetricsReportingCheckbox_Enable", + if (enabled) + UserMetricsRecordAction(L"Options_MetricsReportingCheckbox_Enable", profile()->GetPrefs()); - } else { - UserMetricsRecordAction(L"Options_MetricsReportingCheckbox_Disable", + else + UserMetricsRecordAction(L"Options_MetricsReportingCheckbox_Disable", profile()->GetPrefs()); - } + ResolveMetricsReportingEnabled(); + if (enabled == reporting_enabled_checkbox_->IsSelected()) RestartMessageBox::ShowMessageBox(GetRootWindow()); - } enable_metrics_recording_.SetValue(enabled); } } @@ -444,15 +439,39 @@ void GeneralSection::NotifyPrefChanged(const std::wstring* pref_name) { reset_file_handlers_button_->SetEnabled(enabled); } if (!pref_name || *pref_name == prefs::kMetricsReportingEnabled) { - bool enabled = enable_metrics_recording_.GetValue(); - bool done = g_browser_process->metrics_service()->EnableReporting(enabled); - if (!done) { - enabled = !enabled; - done = g_browser_process->metrics_service()->EnableReporting(enabled); - DCHECK(done); - } - reporting_enabled_checkbox_->SetIsSelected(enabled); + reporting_enabled_checkbox_->SetIsSelected( + enable_metrics_recording_.GetValue()); + ResolveMetricsReportingEnabled(); + } +} + +void GeneralSection::ResolveMetricsReportingEnabled() { + bool enabled = reporting_enabled_checkbox_->IsSelected(); + + GoogleUpdateSettings::SetCollectStatsConsent(enabled); + bool update_pref = GoogleUpdateSettings::GetCollectStatsConsent(); + + if (enabled != update_pref) { + DLOG(INFO) << + "GENERAL SECTION: Unable to set crash report status to " << + enabled; } + + // Only change the pref if GoogleUpdateSettings::GetCollectStatsConsent + // succeeds. + enabled = update_pref; + + MetricsService* metrics = g_browser_process->metrics_service(); + DCHECK(metrics); + if (metrics) { + metrics->SetUserPermitsUpload(enabled); + if (enabled) + metrics->Start(); + else + metrics->Stop(); + } + + reporting_enabled_checkbox_->SetIsSelected(enabled); } //////////////////////////////////////////////////////////////////////////////// diff --git a/chrome/common/chrome_switches.cc b/chrome/common/chrome_switches.cc index 2224e39..1cfc40e7 100644 --- a/chrome/common/chrome_switches.cc +++ b/chrome/common/chrome_switches.cc @@ -17,12 +17,12 @@ const wchar_t kDisableHangMonitor[] = L"disable-hang-monitor"; // Completely disables UMA metrics system. const wchar_t kDisableMetrics[] = L"disable-metrics"; -// Disables only the sending of metrics reports. In contrast to -// kDisableMetrics, this executes all the code that a normal client would use -// for reporting, except the report is dropped rather than sent to the server. -// This is useful for finding issues in the metrics code during UI and -// performance tests. -const wchar_t kDisableMetricsReporting[] = L"disable-metrics-reporting"; +// Enables the recording of metrics reports but disables reporting. +// In contrast to kDisableMetrics, this executes all the code that a normal +// client would use for reporting, except the report is dropped rather than sent +// to the server. This is useful for finding issues in the metrics code during +// UI and performance tests. +const wchar_t kMetricsRecordingOnly[] = L"metrics-recording-only"; // Causes the browser process to throw an assertion on startup. const wchar_t kBrowserAssertTest[] = L"assert-test"; diff --git a/chrome/common/chrome_switches.h b/chrome/common/chrome_switches.h index 79c1d05..4c76cdc 100644 --- a/chrome/common/chrome_switches.h +++ b/chrome/common/chrome_switches.h @@ -13,7 +13,7 @@ namespace switches { extern const wchar_t kDisableHangMonitor[]; extern const wchar_t kDisableMetrics[]; -extern const wchar_t kDisableMetricsReporting[]; +extern const wchar_t kMetricsRecordingOnly[]; extern const wchar_t kBrowserAssertTest[]; extern const wchar_t kRendererAssertTest[]; extern const wchar_t kBrowserCrashTest[]; diff --git a/chrome/test/ui/ui_test.cc b/chrome/test/ui/ui_test.cc index 1b47e7d..263b7b7 100644 --- a/chrome/test/ui/ui_test.cc +++ b/chrome/test/ui/ui_test.cc @@ -203,7 +203,7 @@ void UITest::LaunchBrowser(const std::wstring& arguments, bool clear_profile) { switches::kJavaScriptFlags, js_flags_); - CommandLine::AppendSwitch(&command_line, switches::kDisableMetricsReporting); + CommandLine::AppendSwitch(&command_line, switches::kMetricsRecordingOnly); // We always want to enable chrome logging CommandLine::AppendSwitch(&command_line, switches::kEnableLogging); |