X-Chrome-Variations refactoring.
Caveats:
(1) I had to add a lock since both AC and RDH run on different threads
(2) Because the cache is lazily initialized, it's possible that a few AC requests will run without headers
(3) SP doesn't have access to ResourceContext and in turn we can't get ProfileIOData, so we can't transmit "UMA is enabled" header
(4) I used a singleton
BUG=163999
TEST=hand tested by adding LOG(INFO) and checking headers sent on autocomplete/search requests
Also, ran tcpdump and observed transmitted headers.
Review URL: https://chromiumcodereview.appspot.com/11522009
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@173740 0039d316-1c4b-4281-b951-d872f2087c98
diff --git a/chrome/browser/chrome_metrics_helper.h b/chrome/browser/chrome_metrics_helper.h
new file mode 100644
index 0000000..268b55cc
--- /dev/null
+++ b/chrome/browser/chrome_metrics_helper.h
@@ -0,0 +1,83 @@
+// 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.
+
+#ifndef CHROME_BROWSER_CHROME_METRICS_HELPER_H_
+#define CHROME_BROWSER_CHROME_METRICS_HELPER_H_
+
+#include <set>
+#include <string>
+
+#include "base/basictypes.h"
+#include "base/metrics/field_trial.h"
+#include "base/synchronization/lock.h"
+#include "chrome/common/metrics/variations/variation_ids.h"
+
+namespace content {
+class ResourceContext;
+}
+
+namespace net {
+class HttpRequestHeaders;
+}
+
+class GURL;
+class Profile;
+class ProfileIOData;
+
+template <typename T> struct DefaultSingletonTraits;
+
+// A helper class for maintaining Chrome experiments and metrics state
+// transmitted in custom HTTP request headers.
+// This class is a thread-safe singleton.
+class ChromeMetricsHelper : base::FieldTrialList::Observer {
+ public:
+ static ChromeMetricsHelper* GetInstance();
+
+ // Adds Chrome experiment and metrics state as custom headers to |headers|.
+ // Some headers may not be set given the |incognito| mode or whether
+ // the user has |uma_enabled|. Also, we never transmit headers to non-Google
+ // sites, which is checked based on the destination |url|.
+ void AppendHeaders(const GURL& url,
+ bool incognito,
+ bool uma_enabled,
+ net::HttpRequestHeaders* headers);
+
+ private:
+ friend struct DefaultSingletonTraits<ChromeMetricsHelper>;
+
+ ChromeMetricsHelper();
+ virtual ~ChromeMetricsHelper();
+
+ // base::FieldTrialList::Observer implementation.
+ // This will add the variation ID associated with |trial_name| and
+ // |group_name| to the variation ID cache.
+ virtual void OnFieldTrialGroupFinalized(
+ const std::string& trial_name,
+ const std::string& group_name) OVERRIDE;
+
+ // Prepares the variation IDs cache with initial values if not already done.
+ // This method also registers the caller with the FieldTrialList to receive
+ // new variation IDs.
+ void InitVariationIDsCacheIfNeeded();
+
+ // Takes whatever is currently in |variation_ids_set_| and recreates
+ // |variation_ids_header_| with it.
+ void UpdateVariationIDsHeaderValue();
+
+ // Guards |variation_ids_cache_initialized_|, |variation_ids_set_| and
+ // |variation_ids_header_|.
+ base::Lock lock_;
+
+ // Whether or not we've initialized the cache.
+ bool variation_ids_cache_initialized_;
+
+ // Keep a cache of variation IDs that are transmitted in headers to Google.
+ // This consists of a list of valid IDs, and the actual transmitted header.
+ std::set<chrome_variations::VariationID> variation_ids_set_;
+ std::string variation_ids_header_;
+
+ DISALLOW_COPY_AND_ASSIGN(ChromeMetricsHelper);
+};
+
+#endif // CHROME_BROWSER_CHROME_METRICS_HELPER_H_