| // Copyright 2012 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef COMPONENTS_VARIATIONS_SERVICE_VARIATIONS_SERVICE_H_ |
| #define COMPONENTS_VARIATIONS_SERVICE_VARIATIONS_SERVICE_H_ |
| |
| #include <memory> |
| #include <string> |
| #include <vector> |
| |
| #include "base/compiler_specific.h" |
| #include "base/gtest_prod_util.h" |
| #include "base/memory/raw_ptr.h" |
| #include "base/memory/weak_ptr.h" |
| #include "base/metrics/field_trial.h" |
| #include "base/observer_list.h" |
| #include "base/time/time.h" |
| #include "components/variations/client_filterable_state.h" |
| #include "components/variations/entropy_provider.h" |
| #include "components/variations/service/safe_seed_manager.h" |
| #include "components/variations/service/ui_string_overrider.h" |
| #include "components/variations/service/variations_field_trial_creator.h" |
| #include "components/variations/service/variations_service_client.h" |
| #include "components/variations/variations_request_scheduler.h" |
| #include "components/variations/variations_seed_simulator.h" |
| #include "components/variations/variations_seed_store.h" |
| #include "components/web_resource/resource_request_allowed_notifier.h" |
| #include "url/gurl.h" |
| |
| class PrefService; |
| class PrefRegistrySimple; |
| |
| namespace base { |
| class FeatureList; |
| class Version; |
| } // namespace base |
| |
| namespace metrics { |
| class MetricsStateManager; |
| } |
| |
| namespace network { |
| class SimpleURLLoader; |
| } |
| |
| namespace user_prefs { |
| class PrefRegistrySyncable; |
| } |
| |
| namespace variations { |
| struct StudyGroupNames; |
| class VariationsSeed; |
| } // namespace variations |
| |
| namespace variations { |
| |
| #if BUILDFLAG(IS_CHROMEOS) |
| class DeviceVariationsRestrictionByPolicyApplicator; |
| #endif |
| |
| // Used to (a) set up field trials based on stored variations seed data and (b) |
| // fetch new seed data from the variations server. |
| class VariationsService |
| : public web_resource::ResourceRequestAllowedNotifier::Observer { |
| public: |
| class Observer { |
| public: |
| // How critical a detected experiment change is. Whether it should be |
| // handled on a "best-effort" basis or, for a more critical change, if it |
| // should be given higher priority. |
| enum Severity { |
| BEST_EFFORT, |
| CRITICAL, |
| }; |
| |
| // Called when the VariationsService detects that there will be significant |
| // experiment changes on a restart. This notification can then be used to |
| // update UI (i.e. badging an icon). |
| virtual void OnExperimentChangesDetected(Severity severity) = 0; |
| |
| protected: |
| virtual ~Observer() = default; |
| }; |
| |
| VariationsService(const VariationsService&) = delete; |
| VariationsService& operator=(const VariationsService&) = delete; |
| |
| ~VariationsService() override; |
| |
| // Enum used to choose whether GetVariationsServerURL will return an HTTPS |
| // URL or an HTTP one. The HTTP URL is used as a fallback for seed retrieval |
| // in cases where an HTTPS connection fails. |
| enum HttpOptions { USE_HTTP, USE_HTTPS }; |
| |
| // Should be called before startup of the main message loop. |
| void PerformPreMainMessageLoopStartup(); |
| |
| // Adds an observer to listen for detected experiment changes. |
| void AddObserver(Observer* observer); |
| |
| // Removes a previously-added observer. |
| void RemoveObserver(Observer* observer); |
| |
| // Called when the application enters foreground. This may trigger a |
| // FetchVariationsSeed call. |
| // TODO(rkaplow): Handle this and the similar event in metrics_service by |
| // observing an 'OnAppEnterForeground' event instead of requiring the frontend |
| // code to notify each service individually. |
| void OnAppEnterForeground(); |
| |
| // Sets the value of the "restrict" URL param to the variations service that |
| // should be used for variation seed requests. This takes precedence over any |
| // value coming from policy prefs. This should be called prior to any calls |
| // to |StartRepeatedVariationsSeedFetch|. |
| void SetRestrictMode(const std::string& restrict_mode); |
| |
| // Returns true if the restrict mode is likely that of a dogfood client, false |
| // otherwise. Note that that this might be a bit over-broad, returning true |
| // for clients that are not actually dogfooders. |
| bool IsLikelyDogfoodClient() const; |
| |
| // Sets the return value for subsequent calls to `IsLikelyDogfoodClient()`. |
| // This is a convenience function only for testing, because the two approaches |
| // that follow production code paths are cumbersome to get right: |
| // * `SetRestrictMode` also configures this behavior, but must be called early |
| // during the `VariationsService` initialization flow. |
| // * Enterprise policies also configure this behavior, but the logic is |
| // different per-platform. In particular, Ash ChromeOS and Lacros each have |
| // distinct flows vs. other platforms. |
| // |
| // Warning: Depending on exactly when this is called, this might also change |
| // the constructed variations server URL's params. In other cases, it will |
| // cause the server URL to be out of sync with the `restrict_mode_`. In tests |
| // that require these to be in sync, prefer to call `SetRestrictMode()` at the |
| // appropriate time. |
| void SetIsLikelyDogfoodClientForTesting(bool is_dogfood_client); |
| |
| // Returns the variations server URL. |http_options| determines whether to |
| // use the http or https URL. This function will return an empty GURL when |
| // the restrict param exists for USE_HTTP, to indicate that no HTTP fallback |
| // should happen in that case. |
| GURL GetVariationsServerURL(HttpOptions http_options); |
| |
| // Returns the permanent overridden country code stored for this client. This |
| // value will not be updated on Chrome updates. |
| // Country code is in the format of lowercase ISO 3166-1 alpha-2. Example: us, |
| // br, in. |
| std::string GetOverriddenPermanentCountry() const; |
| |
| // Returns the permanent country code stored for this client. |
| // Country code is in the format of lowercase ISO 3166-1 alpha-2. Example: us, |
| // br, in. This can only be called after field trials have been initialized or |
| // if OverrideStoredPermanentCountry() has been called. |
| std::string GetStoredPermanentCountry() const; |
| |
| // Forces an override of the stored permanent country. Returns true |
| // if the variable has been updated. Return false if the override country is |
| // the same as the stored variable, or if the update failed for any other |
| // reason. |
| bool OverrideStoredPermanentCountry(const std::string& override_country); |
| |
| // Returns what variations will consider to be the latest country. Returns |
| // empty if it is not available. |
| // Country code is in the format of lowercase ISO 3166-1 alpha-2. Example: us, |
| // br, in. |
| std::string GetLatestCountry() const; |
| |
| // Ensures the locale that was used for evaluating variations matches the |
| // passed |locale|. This is used to ensure that the locale determined after |
| // loading the resource bundle (which is passed here) corresponds to what |
| // was used for variations during an earlier stage of start up. |
| void EnsureLocaleEquals(const std::string& locale); |
| |
| // Exposed for testing. |
| static std::string GetDefaultVariationsServerURLForTesting(); |
| |
| // Register Variations related prefs in Local State. |
| static void RegisterPrefs(PrefRegistrySimple* registry); |
| |
| // Register Variations related prefs in the Profile prefs. |
| static void RegisterProfilePrefs(user_prefs::PrefRegistrySyncable* registry); |
| |
| // Creates a VariationsService instance. Does not take ownership of |
| // |state_manager|, so callers should ensure that |state_manager| is valid for |
| // the lifetime of this class. |
| // |
| // |client| provides some platform-specific operations for variations. Must |
| // not be null. |
| // |local_state| provides access to Local State prefs. Must not be null. |
| // |state_manager| provides access to metrics state info. Must not be null. |
| // |disable_network_switch| is a command-line switch that can be used to |
| // disable network communication. |
| // |ui_string_overrider| provides overrides for UI strings. |
| // |network_connection_tracker_getter| allows the VariationsService to |
| // observe network state changes. |
| static std::unique_ptr<VariationsService> Create( |
| std::unique_ptr<VariationsServiceClient> client, |
| PrefService* local_state, |
| metrics::MetricsStateManager* state_manager, |
| const char* disable_network_switch, |
| const UIStringOverrider& ui_string_overrider, |
| web_resource::ResourceRequestAllowedNotifier:: |
| NetworkConnectionTrackerGetter network_connection_tracker_getter); |
| |
| // Enables fetching the seed for testing, even for unofficial builds. This |
| // should be used along with overriding |DoActualFetch| or using |
| // |net::TestURLLoaderFactory|. |
| static void EnableFetchForTesting(); |
| |
| // Set the PrefService responsible for getting policy-related preferences, |
| // such as the restrict parameter. |
| void set_policy_pref_service(PrefService* service) { |
| DCHECK(service); |
| policy_pref_service_ = service; |
| } |
| |
| // Returns the ClientFilterableState, i.e., the state used to do trial |
| // filtering. Should only be used for testing and debugging purposes. |
| std::unique_ptr<ClientFilterableState> GetClientFilterableStateForVersion(); |
| |
| web_resource::ResourceRequestAllowedNotifier* |
| GetResourceRequestAllowedNotifierForTesting() { |
| return resource_request_allowed_notifier_.get(); |
| } |
| |
| // Wrapper around VariationsFieldTrialCreator::SetUpFieldTrials(). |
| bool SetUpFieldTrials( |
| const std::vector<std::string>& variation_ids, |
| const std::string& command_line_variation_ids, |
| const std::vector<base::FeatureList::FeatureOverrideInfo>& |
| extra_overrides, |
| std::unique_ptr<base::FeatureList> feature_list, |
| PlatformFieldTrials* platform_field_trials); |
| |
| // Returns the names of studies and their groups which could possibly be |
| // forced. |
| std::vector<StudyGroupNames> GetStudiesAvailableToForce(); |
| |
| // The seed type used. |
| SeedType GetSeedType() const; |
| |
| // Overrides cached UI strings on the resource bundle once it is initialized. |
| void OverrideCachedUIStrings(); |
| |
| int request_count() const { return request_count_; } |
| |
| // Cancels the currently pending fetch request. |
| void CancelCurrentRequestForTesting(); |
| |
| // Exposes StartRepeatedVariationsSeedFetch for testing. |
| void StartRepeatedVariationsSeedFetchForTesting(); |
| |
| // Allows the embedder to override the platform and override the OS name in |
| // the variations server url. This is useful for android webview and weblayer |
| // which are distinct from regular android chrome. |
| void OverridePlatform(Study::Platform platform, |
| const std::string& osname_server_param_override); |
| |
| // Returns the seed store. Exposed for testing. |
| VariationsSeedStore* GetSeedStoreForTesting(); |
| |
| // Returns the fetch time of the latest seed. Returns base::Time() if there is |
| // no seed. |
| base::Time GetLatestSeedFetchTime(); |
| |
| protected: |
| // Gets the serial number of the most recent Finch seed. Virtual for testing. |
| virtual const std::string& GetLatestSerialNumber(); |
| |
| // Starts the fetching process once, where |OnURLFetchComplete| is called with |
| // the response. This calls DoFetchToURL with the set url. |
| virtual void DoActualFetch(); |
| |
| // Attempts a seed fetch from the set |url|. Returns true if the fetch was |
| // started successfully, false otherwise. |is_http_retry| should be true if |
| // this is a retry over HTTP, false otherwise. |
| virtual bool DoFetchFromURL(const GURL& url, bool is_http_retry); |
| |
| // Stores the seed to prefs. Set as virtual and protected so that it can be |
| // overridden by tests. |
| // Note: Strings are passed by value to support std::move() semantics. |
| virtual void StoreSeed(std::string seed_data, |
| std::string seed_signature, |
| std::string country_code, |
| base::Time date_fetched, |
| bool is_delta_compressed, |
| bool is_gzip_compressed); |
| |
| // Processes the result of StoreSeed(). |
| void OnSeedStoreResult(bool is_delta_compressed, |
| bool store_success, |
| VariationsSeed seed); |
| |
| // Use the |Create| factory method to create a VariationsService. See |Create| |
| // for more details. |
| VariationsService( |
| std::unique_ptr<VariationsServiceClient> client, |
| std::unique_ptr<web_resource::ResourceRequestAllowedNotifier> notifier, |
| PrefService* local_state, |
| metrics::MetricsStateManager* state_manager, |
| const UIStringOverrider& ui_string_overrider); |
| |
| // Sets the URL for querying the variations server. Used for testing. |
| void set_variations_server_url(const GURL& url) { |
| variations_server_url_ = url; |
| } |
| |
| // Sets the URL for querying the variations server when doing HTTP retries. |
| // Used for testing. |
| void set_insecure_variations_server_url(const GURL& url) { |
| insecure_variations_server_url_ = url; |
| } |
| |
| // Sets the |last_request_was_http_retry_| flag. Used for testing. |
| void set_last_request_was_http_retry(bool was_http_retry) { |
| last_request_was_http_retry_ = was_http_retry; |
| } |
| |
| // The client that provides access to the embedder's environment. |
| // Protected so testing subclasses can access it. |
| VariationsServiceClient* client() { return client_.get(); } |
| |
| // Exposes MaybeRetryOverHTTP for testing. |
| bool CallMaybeRetryOverHTTPForTesting(); |
| |
| // Records that a new seed has been stored. Writes the currently active seed |
| // to the |seed_store| as a safe seed, if appropriate. Also, clears failure |
| // streaks. |
| void RecordSuccessfulFetchNewSeed(); |
| |
| // Like VariationsService::RecordSuccessfulFetchNewSeed(), but intended to be |
| // called for 304 responses from the variations server. Also, updates the seed |
| // date and client fetch time. |
| void RecordSuccessfulFetchSeedNotModified(base::Time response_date); |
| |
| private: |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, Observer); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, SeedStoredWhenOKStatus); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, SeedNotStoredWhenNonOKStatus); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, InstanceManipulations); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, CountryHeader); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, GetVariationsServerURL); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, VariationsURLHasParams); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, RequestsInitiallyAllowed); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, RequestsInitiallyNotAllowed); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, |
| SafeMode_SuccessfulFetchClearsFailureStreaks); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, |
| SafeMode_NotModifiedFetchClearsFailureStreaks); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, InsecurelyFetchedSetWhenHTTP); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, |
| InsecurelyFetchedNotSetWhenHTTPS); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, DoNotRetryAfterARetry); |
| FRIEND_TEST_ALL_PREFIXES(VariationsServiceTest, |
| DoNotRetryIfInsecureURLIsHTTPS); |
| |
| void InitResourceRequestedAllowedNotifier(); |
| |
| // Calls FetchVariationsSeed once and repeats this periodically. See |
| // implementation for details on the period. |
| void StartRepeatedVariationsSeedFetch(); |
| |
| // Checks if prerequisites for fetching the Variations seed are met, and if |
| // so, performs the actual fetch using |DoActualFetch|. |
| void FetchVariationsSeed(); |
| |
| // Notify any observers of this service based on the simulation |result|. |
| void NotifyObservers(const SeedSimulationResult& result); |
| |
| // Called by SimpleURLLoader when |pending_seed_request_| load completes. |
| void OnSimpleLoaderComplete(std::optional<std::string> response_body); |
| |
| // Retry the fetch over HTTP, called by OnSimpleLoaderComplete when a request |
| // fails. Returns true is the fetch was successfully started, this does not |
| // imply the actual fetch was successful. |
| bool MaybeRetryOverHTTP(); |
| |
| // ResourceRequestAllowedNotifier::Observer implementation: |
| void OnResourceRequestsAllowed() override; |
| |
| // Performs a variations seed simulation with the given |seed| and |version| |
| // and logs the simulation results as histograms. |
| void PerformSimulationWithVersion(const VariationsSeed& seed, |
| const base::Version& version); |
| |
| // Encrypts a string using the encrypted_messages component, input is passed |
| // in as |plaintext|, outputs a serialized EncryptedMessage protobuf as |
| // |encrypted|. Returns true on success, false on failure. The encryption can |
| // be done in-place. |
| bool EncryptString(const std::string& plaintext, std::string* encrypted); |
| |
| std::unique_ptr<VariationsServiceClient> client_; |
| |
| // The pref service used to store persist the variations seed. |
| raw_ptr<PrefService> local_state_; |
| |
| // Used for instantiating entropy providers for variations seed simulation. |
| // Weak pointer. |
| raw_ptr<metrics::MetricsStateManager> state_manager_; |
| |
| // Used to obtain policy-related preferences. Depending on the platform, will |
| // either be Local State or Profile prefs. |
| raw_ptr<PrefService> policy_pref_service_; |
| |
| // Contains the scheduler instance that handles timing for requests to the |
| // server. Initially NULL and instantiated when the initial fetch is |
| // requested. |
| std::unique_ptr<VariationsRequestScheduler> request_scheduler_; |
| |
| // Contains the current seed request. Will only have a value while a request |
| // is pending, and will be reset by |OnURLFetchComplete|. |
| std::unique_ptr<network::SimpleURLLoader> pending_seed_request_; |
| |
| // The value of the "restrict" URL param to the variations server that has |
| // been specified via |SetRestrictMode|. If empty, the URL param will be set |
| // based on policy prefs. |
| std::string restrict_mode_; |
| |
| // The URL to use for querying the variations server. |
| GURL variations_server_url_; |
| |
| // HTTP URL used as a fallback if HTTPS fetches fail. If not set, retries |
| // over HTTP will not be attempted. |
| GURL insecure_variations_server_url_; |
| |
| // Tracks whether the initial request to the variations server had completed. |
| bool initial_request_completed_ = false; |
| |
| // Tracks whether any errors resolving delta compression were encountered |
| // since the last time a seed was fetched successfully. |
| bool delta_error_since_last_success_ = false; |
| |
| // Helper class used to tell this service if it's allowed to make network |
| // resource requests. |
| std::unique_ptr<web_resource::ResourceRequestAllowedNotifier> |
| resource_request_allowed_notifier_; |
| |
| // The start time of the last seed request. This is used to measure the |
| // latency of seed requests. Initially zero. |
| base::TimeTicks last_request_started_time_; |
| |
| // The number of requests to the variations server that have been performed. |
| int request_count_ = 0; |
| |
| // List of observers of the VariationsService. |
| base::ObserverList<Observer>::Unchecked observer_list_; |
| |
| // The main entry point for managing safe mode state. |
| SafeSeedManager safe_seed_manager_; |
| |
| // Used to provide entropy to field trials. |
| std::unique_ptr<const EntropyProviders> entropy_providers_; |
| |
| // Member responsible for creating trials from a variations seed. |
| VariationsFieldTrialCreator field_trial_creator_; |
| |
| // True if the last request was a retry over http. |
| bool last_request_was_http_retry_ = false; |
| |
| // When not empty, contains an override for the os name in the variations |
| // server url. |
| std::string osname_server_param_override_; |
| |
| #if BUILDFLAG(IS_CHROMEOS) |
| std::unique_ptr<DeviceVariationsRestrictionByPolicyApplicator> |
| device_variations_restrictions_by_policy_applicator_; |
| #endif |
| |
| SEQUENCE_CHECKER(sequence_checker_); |
| |
| base::WeakPtrFactory<VariationsService> weak_ptr_factory_{this}; |
| }; |
| |
| } // namespace variations |
| |
| #endif // COMPONENTS_VARIATIONS_SERVICE_VARIATIONS_SERVICE_H_ |