blob: 58306a090fbadd2b6277b94b922e3ba3ac02cae1 [file] [log] [blame]
// Copyright 2017 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 EXTENSIONS_BROWSER_API_DECLARATIVE_NET_REQUEST_RULESET_MANAGER_H_
#define EXTENSIONS_BROWSER_API_DECLARATIVE_NET_REQUEST_RULESET_MANAGER_H_
#include <stddef.h>
#include <memory>
#include <set>
#include <vector>
#include "base/containers/flat_set.h"
#include "base/macros.h"
#include "base/optional.h"
#include "base/sequence_checker.h"
#include "base/time/time.h"
#include "extensions/browser/api/declarative_net_request/utils.h"
#include "extensions/common/extension_id.h"
#include "extensions/common/permissions/permissions_data.h"
namespace content {
class BrowserContext;
class RenderFrameHost;
}
namespace extensions {
class ExtensionPrefs;
class PermissionHelper;
struct WebRequestInfo;
namespace declarative_net_request {
class CompositeMatcher;
struct RequestAction;
struct RequestParams;
// Manages the set of active rulesets for the Declarative Net Request API. Can
// be constructed on any sequence but must be accessed and destroyed from the
// same sequence.
class RulesetManager {
public:
explicit RulesetManager(content::BrowserContext* browser_context);
~RulesetManager();
// An observer used for testing purposes.
class TestObserver {
public:
virtual void OnEvaluateRequest(const WebRequestInfo& request,
bool is_incognito_context) {}
// Called whenever a ruleset is added or removed.
virtual void OnRulesetCountChanged(size_t new_count) {}
protected:
virtual ~TestObserver() {}
};
// Adds the ruleset for the given |extension_id|. Should not be called twice
// in succession for an extension.
void AddRuleset(const ExtensionId& extension_id,
std::unique_ptr<CompositeMatcher> matcher);
// Removes the ruleset for |extension_id|. Should be called only after a
// corresponding AddRuleset.
void RemoveRuleset(const ExtensionId& extension_id);
// Returns the set of extensions which have active rulesets.
std::set<ExtensionId> GetExtensionsWithRulesets() const;
// Returns the CompositeMatcher corresponding to the |extension_id| or null
// if no matcher is present for the extension.
CompositeMatcher* GetMatcherForExtension(const ExtensionId& extension_id);
// Returns the action to take for the given request; does not return an
// |ALLOW| action. Note: the returned action is owned by |request|.
// Precedence order: Allow > Blocking > Redirect rules.
// For redirect rules, most recently installed extensions are given
// preference.
const std::vector<RequestAction>& EvaluateRequest(
const WebRequestInfo& request,
bool is_incognito_context) const;
// Returns true if there is an active matcher which modifies "extraHeaders".
bool HasAnyExtraHeadersMatcher() const;
// Returns true if there is a matcher which modifies "extraHeaders" for the
// given |request|.
bool HasExtraHeadersMatcherForRequest(const WebRequestInfo& request,
bool is_incognito_context) const;
void OnRenderFrameCreated(content::RenderFrameHost* host);
void OnRenderFrameDeleted(content::RenderFrameHost* host);
void OnDidFinishNavigation(content::RenderFrameHost* host);
// Returns the number of CompositeMatchers currently being managed.
size_t GetMatcherCountForTest() const { return rulesets_.size(); }
// Sets the TestObserver. Client maintains ownership of |observer|.
void SetObserverForTest(TestObserver* observer);
private:
struct ExtensionRulesetData {
ExtensionRulesetData(const ExtensionId& extension_id,
const base::Time& extension_install_time,
std::unique_ptr<CompositeMatcher> matcher);
~ExtensionRulesetData();
ExtensionRulesetData(ExtensionRulesetData&& other);
ExtensionRulesetData& operator=(ExtensionRulesetData&& other);
ExtensionId extension_id;
base::Time extension_install_time;
std::unique_ptr<CompositeMatcher> matcher;
bool operator<(const ExtensionRulesetData& other) const;
DISALLOW_COPY_AND_ASSIGN(ExtensionRulesetData);
};
base::Optional<RequestAction> GetBeforeRequestAction(
const std::vector<const ExtensionRulesetData*>& rulesets,
const WebRequestInfo& request,
const int tab_id,
const bool crosses_incognito,
const RequestParams& params) const;
std::vector<RequestAction> GetRemoveHeadersActions(
const std::vector<const ExtensionRulesetData*>& rulesets,
const RequestParams& params) const;
// Helper for EvaluateRequest.
std::vector<RequestAction> EvaluateRequestInternal(
const WebRequestInfo& request,
bool is_incognito_context) const;
// Returns true if the given |request| should be evaluated for
// blocking/redirection.
bool ShouldEvaluateRequest(const WebRequestInfo& request) const;
// Returns whether |ruleset| should be evaluated for the given |request|.
// Note: this does not take the extension's host permissions into account.
bool ShouldEvaluateRulesetForRequest(const ExtensionRulesetData& ruleset,
const WebRequestInfo& request,
bool is_incognito_context) const;
// Sorted in decreasing order of |extension_install_time|.
// Use a flat_set instead of std::set/map. This makes [Add/Remove]Ruleset
// O(n), but it's fine since the no. of rulesets are expected to be quite
// small.
base::flat_set<ExtensionRulesetData> rulesets_;
// Non-owning pointer to BrowserContext.
content::BrowserContext* const browser_context_;
// Guaranteed to be valid through-out the lifetime of this instance.
ExtensionPrefs* const prefs_;
PermissionHelper* const permission_helper_;
// Non-owning pointer to TestObserver.
TestObserver* test_observer_ = nullptr;
SEQUENCE_CHECKER(sequence_checker_);
DISALLOW_COPY_AND_ASSIGN(RulesetManager);
};
} // namespace declarative_net_request
} // namespace extensions
#endif // EXTENSIONS_BROWSER_API_DECLARATIVE_NET_REQUEST_RULESET_MANAGER_H_