blob: d01575a1f94a439f3960dbf6258ece4ab4d92ce6 [file] [log] [blame]
// Copyright 2016 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 COMPONENTS_VARIATIONS_CHILD_PROCESS_FIELD_TRIAL_SYNCER_H_
#define COMPONENTS_VARIATIONS_CHILD_PROCESS_FIELD_TRIAL_SYNCER_H_
#include <string>
#include "base/callback.h"
#include "base/component_export.h"
#include "base/metrics/field_trial.h"
#include "base/threading/thread_local.h"
namespace variations {
// Syncs the "activated" state of field trials between browser and child
// processes. Specifically, when a field trial is activated in the browser, it
// also activates it in the child process and when a field trial is activated in
// the child process, it notifies the browser process to activate it.
class COMPONENT_EXPORT(VARIATIONS) ChildProcessFieldTrialSyncer
: public base::FieldTrialList::Observer {
public:
using FieldTrialActivatedCallback =
base::RepeatingCallback<void(const std::string& trial_name)>;
// Creates and returns the global ChildProcessFieldTrialSyncer instance for
// this process. Immediately invokes |activated_callback| for each currently
// active field trial. Then, |activated_callback| is invoked as field trial
// are activated. |activated_callback| may be invoked from any sequence and
// must therefore be thread-safe. CreateInstance() must not be called
// concurrently with activating a field trial. The created instance is never
// destroyed because it would be difficult to ensure that no field trial is
// activated concurrently with unregistering it as an observer of
// FieldTrialList (see FieldTrialList::RemoveObserver).
static ChildProcessFieldTrialSyncer* CreateInstance(
FieldTrialActivatedCallback activated_callback);
ChildProcessFieldTrialSyncer(const ChildProcessFieldTrialSyncer&) = delete;
ChildProcessFieldTrialSyncer& operator=(const ChildProcessFieldTrialSyncer&) =
delete;
// Deletes the global ChildProcessFieldTrialSyncer instance.
static void DeleteInstanceForTesting();
// Must be invoked when the browser process notifies this child process that a
// field trial was activated.
void SetFieldTrialGroupFromBrowser(const std::string& trial_name,
const std::string& group_name);
private:
explicit ChildProcessFieldTrialSyncer(
FieldTrialActivatedCallback activated_callback);
~ChildProcessFieldTrialSyncer() override;
// Initializes field trial state change observation and invokes |callback_|
// for any field trials that might have already been activated.
void Init();
// base::FieldTrialList::Observer:
void OnFieldTrialGroupFinalized(const std::string& trial_name,
const std::string& group_name) override;
// Whether SetFieldTrialGroupFromBrowser() is being called on the current
// thread.
base::ThreadLocalBoolean in_set_field_trial_group_from_browser_;
// Callback to invoke when a field trial is activated.
const FieldTrialActivatedCallback activated_callback_;
};
} // namespace variations
#endif // COMPONENTS_VARIATIONS_CHILD_PROCESS_FIELD_TRIAL_SYNCER_H_