blob: f18fd296c445c79e4a28061addd20d15cd55cf73 [file] [log] [blame]
// Copyright 2014 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_SUPERVISED_USER_SUPERVISED_USER_SERVICE_H_
#define CHROME_BROWSER_SUPERVISED_USER_SUPERVISED_USER_SERVICE_H_
#include <stddef.h>
#include <map>
#include <string>
#include <vector>
#include "base/callback.h"
#include "base/gtest_prod_util.h"
#include "base/macros.h"
#include "base/memory/scoped_ptr.h"
#include "base/memory/scoped_vector.h"
#include "base/observer_list.h"
#include "base/prefs/pref_change_registrar.h"
#include "base/scoped_observer.h"
#include "base/strings/string16.h"
#include "build/build_config.h"
#include "chrome/browser/supervised_user/experimental/supervised_user_blacklist.h"
#include "chrome/browser/supervised_user/supervised_user_url_filter.h"
#include "chrome/browser/supervised_user/supervised_users.h"
#include "chrome/browser/ui/browser_list_observer.h"
#include "components/keyed_service/core/keyed_service.h"
#include "components/sync_driver/sync_service_observer.h"
#include "components/sync_driver/sync_type_preference_provider.h"
#include "net/url_request/url_request_context_getter.h"
#if defined(ENABLE_EXTENSIONS)
#include "extensions/browser/management_policy.h"
#endif
class Browser;
class FileDownloader;
class GoogleServiceAuthError;
class PermissionRequestCreator;
class Profile;
class SupervisedUserRegistrationUtility;
class SupervisedUserServiceObserver;
class SupervisedUserSettingsService;
class SupervisedUserSiteList;
class SupervisedUserURLFilter;
class SupervisedUserWhitelistService;
namespace base {
class FilePath;
class Version;
}
namespace content {
class WebContents;
}
namespace extensions {
class ExtensionRegistry;
}
namespace user_prefs {
class PrefRegistrySyncable;
}
// This class handles all the information related to a given supervised profile
// (e.g. the installed content packs, the default URL filtering behavior, or
// manual whitelist/blacklist overrides).
class SupervisedUserService : public KeyedService,
#if defined(ENABLE_EXTENSIONS)
public extensions::ManagementPolicy::Provider,
#endif
public SyncTypePreferenceProvider,
#if !defined(OS_ANDROID) && !defined(OS_IOS)
public sync_driver::SyncServiceObserver,
public chrome::BrowserListObserver,
#endif
public SupervisedUserURLFilter::Observer {
public:
using NavigationBlockedCallback = base::Callback<void(content::WebContents*)>;
using AuthErrorCallback = base::Callback<void(const GoogleServiceAuthError&)>;
using SuccessCallback = base::Callback<void(bool)>;
class Delegate {
public:
virtual ~Delegate() {}
// Returns true to indicate that the delegate handled the (de)activation, or
// false to indicate that the SupervisedUserService itself should handle it.
virtual bool SetActive(bool active) = 0;
};
~SupervisedUserService() override;
static void RegisterProfilePrefs(user_prefs::PrefRegistrySyncable* registry);
// Initializes this object.
void Init();
void SetDelegate(Delegate* delegate);
// Returns the URL filter for the IO thread, for filtering network requests
// (in SupervisedUserResourceThrottle).
scoped_refptr<const SupervisedUserURLFilter> GetURLFilterForIOThread();
// Returns the URL filter for the UI thread, for filtering navigations and
// classifying sites in the history view.
SupervisedUserURLFilter* GetURLFilterForUIThread();
// Returns the whitelist service.
SupervisedUserWhitelistService* GetWhitelistService();
// Whether the user can request to get access to blocked URLs or to new
// extensions.
bool AccessRequestsEnabled();
// Adds an access request for the given URL.
void AddURLAccessRequest(const GURL& url, const SuccessCallback& callback);
// Adds an update request for the given WebStore item (App/Extension).
void AddExtensionUpdateRequest(const std::string& extension_id,
const base::Version& version,
const SuccessCallback& callback);
// Same as above, but without a callback, just logging errors on failure.
void AddExtensionUpdateRequest(const std::string& extension_id,
const base::Version& version);
// Get the string used to identify an extension update request. Public for
// testing.
static std::string GetExtensionUpdateRequestId(
const std::string& extension_id,
const base::Version& version);
// Returns the email address of the custodian.
std::string GetCustodianEmailAddress() const;
// Returns the name of the custodian, or the email address if the name is
// empty.
std::string GetCustodianName() const;
// Returns the email address of the second custodian, or the empty string
// if there is no second custodian.
std::string GetSecondCustodianEmailAddress() const;
// Returns the name of the second custodian, or the email address if the name
// is empty, or the empty string is there is no second custodian.
std::string GetSecondCustodianName() const;
// Returns a message saying that extensions can only be modified by the
// custodian.
base::string16 GetExtensionsLockedMessage() const;
#if !defined(OS_ANDROID) && !defined(OS_IOS)
// Initializes this profile for syncing, using the provided |refresh_token| to
// mint access tokens for Sync.
void InitSync(const std::string& refresh_token);
// Convenience method that registers this supervised user using
// |registration_utility| and initializes sync with the returned token.
// The |callback| will be called when registration is complete,
// whether it succeeded or not -- unless registration was cancelled manually,
// in which case the callback will be ignored.
void RegisterAndInitSync(
SupervisedUserRegistrationUtility* registration_utility,
Profile* custodian_profile,
const std::string& supervised_user_id,
const AuthErrorCallback& callback);
#endif
void AddNavigationBlockedCallback(const NavigationBlockedCallback& callback);
void DidBlockNavigation(content::WebContents* web_contents);
void AddObserver(SupervisedUserServiceObserver* observer);
void RemoveObserver(SupervisedUserServiceObserver* observer);
void AddPermissionRequestCreator(
scoped_ptr<PermissionRequestCreator> creator);
// ProfileKeyedService override:
void Shutdown() override;
// SyncTypePreferenceProvider implementation:
syncer::ModelTypeSet GetPreferredDataTypes() const override;
#if !defined(OS_ANDROID) && !defined(OS_IOS)
// sync_driver::SyncServiceObserver implementation:
void OnStateChanged() override;
// chrome::BrowserListObserver implementation:
void OnBrowserSetLastActive(Browser* browser) override;
#endif // !defined(OS_ANDROID) && !defined(OS_IOS)
// SupervisedUserURLFilter::Observer implementation:
void OnSiteListUpdated() override;
private:
friend class SupervisedUserServiceExtensionTestBase;
friend class SupervisedUserServiceFactory;
FRIEND_TEST_ALL_PREFIXES(SingleClientSupervisedUserSettingsSyncTest, Sanity);
FRIEND_TEST_ALL_PREFIXES(SupervisedUserServiceTest, ClearOmitOnRegistration);
FRIEND_TEST_ALL_PREFIXES(SupervisedUserServiceTest,
ChangesIncludedSessionOnChangedSettings);
FRIEND_TEST_ALL_PREFIXES(SupervisedUserServiceTest,
ChangesSyncSessionStateOnChangedSettings);
FRIEND_TEST_ALL_PREFIXES(SupervisedUserServiceExtensionTest,
ExtensionManagementPolicyProvider);
using CreatePermissionRequestCallback =
base::Callback<void(PermissionRequestCreator*, const SuccessCallback&)>;
// A bridge from the UI thread to the SupervisedUserURLFilters, one of which
// lives on the IO thread. This class mediates access to them and makes sure
// they are kept in sync.
class URLFilterContext {
public:
URLFilterContext();
~URLFilterContext();
SupervisedUserURLFilter* ui_url_filter() const;
SupervisedUserURLFilter* io_url_filter() const;
void SetDefaultFilteringBehavior(
SupervisedUserURLFilter::FilteringBehavior behavior);
void LoadWhitelists(
const std::vector<scoped_refptr<SupervisedUserSiteList>>& site_lists);
// TODO(treib): Make SupervisedUserBlacklist refcounted, so the IO thread
// will retain a reference to the blacklist.
void SetBlacklist(const SupervisedUserBlacklist* blacklist);
bool HasBlacklist() const;
void SetManualHosts(scoped_ptr<std::map<std::string, bool>> host_map);
void SetManualURLs(scoped_ptr<std::map<GURL, bool>> url_map);
void InitAsyncURLChecker(
const scoped_refptr<net::URLRequestContextGetter>& context);
bool HasAsyncURLChecker() const;
void ClearAsyncURLChecker();
void Clear();
private:
void OnBlacklistLoaded(const base::Closure& callback);
// SupervisedUserURLFilter is refcounted because the IO thread filter is
// used both by ProfileImplIOData and OffTheRecordProfileIOData (to filter
// network requests), so they both keep a reference to it.
// Clients should not keep references to the UI thread filter, however
// (the filter will live as long as the profile lives, and afterwards it
// should not be used anymore either).
scoped_refptr<SupervisedUserURLFilter> ui_url_filter_;
scoped_refptr<SupervisedUserURLFilter> io_url_filter_;
DISALLOW_COPY_AND_ASSIGN(URLFilterContext);
};
// Use |SupervisedUserServiceFactory::GetForProfile(..)| to get
// an instance of this service.
explicit SupervisedUserService(Profile* profile);
void SetActive(bool active);
#if !defined(OS_ANDROID) && !defined(OS_IOS)
void OnCustodianProfileDownloaded(const base::string16& full_name);
void OnSupervisedUserRegistered(const AuthErrorCallback& callback,
Profile* custodian_profile,
const GoogleServiceAuthError& auth_error,
const std::string& token);
void SetupSync();
void StartSetupSync();
void FinishSetupSyncWhenReady();
void FinishSetupSync();
#endif
bool ProfileIsSupervised() const;
void OnCustodianInfoChanged();
#if defined(ENABLE_EXTENSIONS)
// extensions::ManagementPolicy::Provider implementation:
std::string GetDebugPolicyProviderName() const override;
bool UserMayLoad(const extensions::Extension* extension,
base::string16* error) const override;
bool UserMayModifySettings(const extensions::Extension* extension,
base::string16* error) const override;
bool MustRemainInstalled(const extensions::Extension* extension,
base::string16* error) const override;
// Extensions helper to SetActive().
void SetExtensionsActive();
#endif
SupervisedUserSettingsService* GetSettingsService();
size_t FindEnabledPermissionRequestCreator(size_t start);
void AddPermissionRequestInternal(
const CreatePermissionRequestCallback& create_request,
const SuccessCallback& callback,
size_t index);
void OnPermissionRequestIssued(
const CreatePermissionRequestCallback& create_request,
const SuccessCallback& callback,
size_t index,
bool success);
void OnSupervisedUserIdChanged();
void OnDefaultFilteringBehaviorChanged();
void OnSafeSitesSettingChanged();
void OnSiteListsChanged(
const std::vector<scoped_refptr<SupervisedUserSiteList>>& site_lists);
// Asynchronously loads a blacklist from a binary file at |path| and applies
// it to the URL filters. If no file exists at |path| yet, downloads a file
// from |url| and stores it at |path| first.
void LoadBlacklist(const base::FilePath& path, const GURL& url);
void OnBlacklistFileChecked(const base::FilePath& path,
const GURL& url,
bool file_exists);
// Asynchronously loads a blacklist from a binary file at |path| and applies
// it to the URL filters.
void LoadBlacklistFromFile(const base::FilePath& path);
void OnBlacklistDownloadDone(const base::FilePath& path, bool success);
void OnBlacklistLoaded();
void UpdateBlacklist();
// Updates the manual overrides for hosts in the URL filters when the
// corresponding preference is changed.
void UpdateManualHosts();
// Updates the manual overrides for URLs in the URL filters when the
// corresponding preference is changed.
void UpdateManualURLs();
// Returns the human readable name of the supervised user.
std::string GetSupervisedUserName() const;
// Subscribes to the SupervisedUserPrefStore, refreshes
// |includes_sync_sessions_type_| and triggers reconfiguring the
// ProfileSyncService.
void OnHistoryRecordingStateChanged();
// Returns true if the syncer::SESSIONS type should be included in Sync.
bool IncludesSyncSessionsType() const;
// The option a custodian sets to either record or prevent recording the
// supervised user's history. Set by |FetchNewSessionSyncState()| and
// defaults to true.
bool includes_sync_sessions_type_;
// Owns us via the KeyedService mechanism.
Profile* profile_;
bool active_;
Delegate* delegate_;
PrefChangeRegistrar pref_change_registrar_;
// True iff we're waiting for the Sync service to be initialized.
bool waiting_for_sync_initialization_;
bool is_profile_active_;
std::vector<NavigationBlockedCallback> navigation_blocked_callbacks_;
// True only when |Init()| method has been called.
bool did_init_;
// True only when |Shutdown()| method has been called.
bool did_shutdown_;
URLFilterContext url_filter_context_;
enum class BlacklistLoadState {
NOT_LOADED,
LOAD_STARTED,
LOADED
} blacklist_state_;
SupervisedUserBlacklist blacklist_;
scoped_ptr<FileDownloader> blacklist_downloader_;
scoped_ptr<SupervisedUserWhitelistService> whitelist_service_;
// Used to create permission requests.
ScopedVector<PermissionRequestCreator> permissions_creators_;
base::ObserverList<SupervisedUserServiceObserver> observer_list_;
base::WeakPtrFactory<SupervisedUserService> weak_ptr_factory_;
};
#endif // CHROME_BROWSER_SUPERVISED_USER_SUPERVISED_USER_SERVICE_H_