blob: e1c09ec581c63ca379e94e311b65703b84e6ca69 [file] [log] [blame]
// 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.
#include "extensions/browser/api/declarative_webrequest/webrequest_action.h"
#include <limits>
#include <optional>
#include <string_view>
#include <utility>
#include "base/check_op.h"
#include "base/no_destructor.h"
#include "base/notreached.h"
#include "base/strings/string_util.h"
#include "base/values.h"
#include "content/public/common/url_constants.h"
#include "extensions/browser/api/declarative/deduping_factory.h"
#include "extensions/browser/api/declarative_webrequest/request_stage.h"
#include "extensions/browser/api/declarative_webrequest/webrequest_condition.h"
#include "extensions/browser/api/declarative_webrequest/webrequest_constants.h"
#include "extensions/browser/api/web_request/web_request_api_constants.h"
#include "extensions/browser/api/web_request/web_request_api_helpers.h"
#include "extensions/browser/api/web_request/web_request_info.h"
#include "extensions/browser/api/web_request/web_request_permissions.h"
#include "extensions/browser/extension_navigation_ui_data.h"
#include "extensions/browser/guest_view/web_view/web_view_renderer_state.h"
#include "extensions/common/error_utils.h"
#include "extensions/common/extension.h"
#include "extensions/common/extension_id.h"
#include "net/base/registry_controlled_domains/registry_controlled_domain.h"
#include "net/http/http_util.h"
#include "third_party/abseil-cpp/absl/strings/ascii.h"
#include "third_party/re2/src/re2/re2.h"
using extension_web_request_api_helpers::EventResponseDelta;
namespace extensions {
namespace helpers = extension_web_request_api_helpers;
namespace keys = declarative_webrequest_constants;
namespace {
// Error messages.
const char kIgnoreRulesRequiresParameterError[] =
"IgnoreRules requires at least one parameter.";
const char kTransparentImageUrl[] = ""
"EUgAAAAEAAAABCAYAAAAfFcSJAAAACklEQVR4nGMAAQAABQABDQottAAAAABJRU5ErkJggg==";
const char kEmptyDocumentUrl[] = "data:text/html,";
#define INPUT_FORMAT_VALIDATE(test) \
do { \
if (!(test)) { \
*bad_message = true; \
return scoped_refptr<const WebRequestAction>(nullptr); \
} \
} while (0)
helpers::RequestCookie ParseRequestCookie(const base::Value::Dict& dict) {
helpers::RequestCookie result;
if (const std::string* name = dict.FindString(keys::kNameKey))
result.name = *name;
if (const std::string* value = dict.FindString(keys::kValueKey))
result.value = *value;
return result;
}
void ParseResponseCookieImpl(const base::Value::Dict& dict,
helpers::ResponseCookie* cookie) {
if (const std::string* v = dict.FindString(keys::kNameKey))
cookie->name = *v;
if (const std::string* v = dict.FindString(keys::kValueKey))
cookie->value = *v;
if (const std::string* v = dict.FindString(keys::kExpiresKey))
cookie->expires = *v;
if (std::optional<int> v = dict.FindInt(keys::kMaxAgeKey)) {
cookie->max_age = *v;
}
if (const std::string* v = dict.FindString(keys::kDomainKey))
cookie->domain = *v;
if (const std::string* v = dict.FindString(keys::kPathKey))
cookie->path = *v;
if (std::optional<bool> v = dict.FindBool(keys::kSecureKey)) {
cookie->secure = *v;
}
if (std::optional<bool> v = dict.FindBool(keys::kHttpOnlyKey)) {
cookie->http_only = *v;
}
}
helpers::ResponseCookie ParseResponseCookie(const base::Value::Dict& dict) {
helpers::ResponseCookie result;
ParseResponseCookieImpl(dict, &result);
return result;
}
helpers::FilterResponseCookie ParseFilterResponseCookie(
const base::Value::Dict& dict) {
helpers::FilterResponseCookie result;
ParseResponseCookieImpl(dict, &result);
if (std::optional<int> v = dict.FindInt(keys::kAgeUpperBoundKey)) {
result.age_upper_bound = *v;
}
if (std::optional<int> v = dict.FindInt(keys::kAgeLowerBoundKey)) {
result.age_lower_bound = *v;
}
if (std::optional<bool> v = dict.FindBool(keys::kSessionCookieKey)) {
result.session_cookie = *v;
}
return result;
}
// Helper function for WebRequestActions that can be instantiated by just
// calling the constructor.
template <class T>
scoped_refptr<const WebRequestAction> CallConstructorFactoryMethod(
const std::string& instance_type,
const base::Value::Dict& value,
std::string* error,
bool* bad_message) {
return base::MakeRefCounted<T>();
}
scoped_refptr<const WebRequestAction> CreateRedirectRequestAction(
const std::string& instance_type,
const base::Value::Dict& value,
std::string* error,
bool* bad_message) {
const std::string* redirect_url_string =
value.FindString(keys::kRedirectUrlKey);
INPUT_FORMAT_VALIDATE(redirect_url_string);
GURL redirect_url(*redirect_url_string);
return base::MakeRefCounted<WebRequestRedirectAction>(redirect_url);
}
scoped_refptr<const WebRequestAction> CreateRedirectRequestByRegExAction(
const std::string& instance_type,
const base::Value::Dict& value,
std::string* error,
bool* bad_message) {
const std::string* from = value.FindString(keys::kFromKey);
const std::string* to = value.FindString(keys::kToKey);
INPUT_FORMAT_VALIDATE(from);
INPUT_FORMAT_VALIDATE(to);
const std::string re2style_to =
WebRequestRedirectByRegExAction::PerlToRe2Style(*to);
RE2::Options options;
options.set_case_sensitive(false);
std::unique_ptr<RE2> from_pattern = std::make_unique<RE2>(*from, options);
if (!from_pattern->ok()) {
*error = "Invalid pattern '" + *from + "' -> '" + re2style_to + "'";
return scoped_refptr<const WebRequestAction>(nullptr);
}
return base::MakeRefCounted<WebRequestRedirectByRegExAction>(
std::move(from_pattern), re2style_to);
}
scoped_refptr<const WebRequestAction> CreateSetRequestHeaderAction(
const std::string& instance_type,
const base::Value::Dict& dict,
std::string* error,
bool* bad_message) {
const std::string* name = dict.FindString(keys::kNameKey);
const std::string* value = dict.FindString(keys::kValueKey);
INPUT_FORMAT_VALIDATE(name);
INPUT_FORMAT_VALIDATE(value);
if (!net::HttpUtil::IsValidHeaderName(*name)) {
*error = extension_web_request_api_constants::kInvalidHeaderName;
return scoped_refptr<const WebRequestAction>(nullptr);
}
if (!net::HttpUtil::IsValidHeaderValue(*value)) {
*error = ErrorUtils::FormatErrorMessage(
extension_web_request_api_constants::kInvalidHeaderValue, *name);
return scoped_refptr<const WebRequestAction>(nullptr);
}
return base::MakeRefCounted<WebRequestSetRequestHeaderAction>(*name, *value);
}
scoped_refptr<const WebRequestAction> CreateRemoveRequestHeaderAction(
const std::string& instance_type,
const base::Value::Dict& value,
std::string* error,
bool* bad_message) {
const std::string* name = value.FindString(keys::kNameKey);
INPUT_FORMAT_VALIDATE(name);
if (!net::HttpUtil::IsValidHeaderName(*name)) {
*error = extension_web_request_api_constants::kInvalidHeaderName;
return scoped_refptr<const WebRequestAction>(nullptr);
}
return base::MakeRefCounted<WebRequestRemoveRequestHeaderAction>(*name);
}
scoped_refptr<const WebRequestAction> CreateAddResponseHeaderAction(
const std::string& instance_type,
const base::Value::Dict& dict,
std::string* error,
bool* bad_message) {
const std::string* name = dict.FindString(keys::kNameKey);
const std::string* value = dict.FindString(keys::kValueKey);
INPUT_FORMAT_VALIDATE(name);
INPUT_FORMAT_VALIDATE(value);
if (!net::HttpUtil::IsValidHeaderName(*name)) {
*error = extension_web_request_api_constants::kInvalidHeaderName;
return scoped_refptr<const WebRequestAction>(nullptr);
}
if (!net::HttpUtil::IsValidHeaderValue(*value)) {
*error = ErrorUtils::FormatErrorMessage(
extension_web_request_api_constants::kInvalidHeaderValue, *name);
return scoped_refptr<const WebRequestAction>(nullptr);
}
return base::MakeRefCounted<WebRequestAddResponseHeaderAction>(*name, *value);
}
scoped_refptr<const WebRequestAction> CreateRemoveResponseHeaderAction(
const std::string& instance_type,
const base::Value::Dict& dict,
std::string* error,
bool* bad_message) {
const std::string* name = dict.FindString(keys::kNameKey);
INPUT_FORMAT_VALIDATE(name);
const std::string* value = dict.FindString(keys::kValueKey);
if (!net::HttpUtil::IsValidHeaderName(*name)) {
*error = extension_web_request_api_constants::kInvalidHeaderName;
return scoped_refptr<const WebRequestAction>(nullptr);
}
if (value && !net::HttpUtil::IsValidHeaderValue(*value)) {
*error = ErrorUtils::FormatErrorMessage(
extension_web_request_api_constants::kInvalidHeaderValue, *name);
return scoped_refptr<const WebRequestAction>(nullptr);
}
const std::string& value_string = value ? *value : std::string();
return base::MakeRefCounted<WebRequestRemoveResponseHeaderAction>(
*name, value_string, value != nullptr);
}
scoped_refptr<const WebRequestAction> CreateIgnoreRulesAction(
const std::string& instance_type,
const base::Value::Dict& value,
std::string* error,
bool* bad_message) {
bool has_parameter = false;
int minimum_priority = std::numeric_limits<int>::min();
std::string ignore_tag;
if (value.Find(keys::kLowerPriorityThanKey)) {
std::optional<int> minimum_priority_value =
value.FindInt(keys::kLowerPriorityThanKey);
INPUT_FORMAT_VALIDATE(minimum_priority_value);
minimum_priority = *minimum_priority_value;
has_parameter = true;
}
if (value.Find(keys::kHasTagKey)) {
const std::string* tag = value.FindString(keys::kHasTagKey);
INPUT_FORMAT_VALIDATE(tag);
ignore_tag = *tag;
has_parameter = true;
}
if (!has_parameter) {
*error = kIgnoreRulesRequiresParameterError;
return scoped_refptr<const WebRequestAction>(nullptr);
}
return base::MakeRefCounted<WebRequestIgnoreRulesAction>(minimum_priority,
ignore_tag);
}
scoped_refptr<const WebRequestAction> CreateRequestCookieAction(
const std::string& instance_type,
const base::Value::Dict& value,
std::string* error,
bool* bad_message) {
using extension_web_request_api_helpers::RequestCookieModification;
RequestCookieModification modification;
// Get modification type.
if (instance_type == keys::kAddRequestCookieType)
modification.type = helpers::ADD;
else if (instance_type == keys::kEditRequestCookieType)
modification.type = helpers::EDIT;
else if (instance_type == keys::kRemoveRequestCookieType)
modification.type = helpers::REMOVE;
else
INPUT_FORMAT_VALIDATE(false);
// Get filter.
if (modification.type == helpers::EDIT ||
modification.type == helpers::REMOVE) {
const base::Value::Dict* filter = value.FindDict(keys::kFilterKey);
INPUT_FORMAT_VALIDATE(filter);
modification.filter = ParseRequestCookie(*filter);
}
// Get new value.
if (modification.type == helpers::ADD) {
const base::Value::Dict* cookie_dict = value.FindDict(keys::kCookieKey);
INPUT_FORMAT_VALIDATE(cookie_dict);
modification.modification = ParseRequestCookie(*cookie_dict);
} else if (modification.type == helpers::EDIT) {
const base::Value::Dict* modification_dict =
value.FindDict(keys::kModificationKey);
INPUT_FORMAT_VALIDATE(modification_dict);
modification.modification = ParseRequestCookie(*modification_dict);
}
return base::MakeRefCounted<WebRequestRequestCookieAction>(
std::move(modification));
}
scoped_refptr<const WebRequestAction> CreateResponseCookieAction(
const std::string& instance_type,
const base::Value::Dict& value,
std::string* error,
bool* bad_message) {
using extension_web_request_api_helpers::ResponseCookieModification;
ResponseCookieModification modification;
// Get modification type.
if (instance_type == keys::kAddResponseCookieType)
modification.type = helpers::ADD;
else if (instance_type == keys::kEditResponseCookieType)
modification.type = helpers::EDIT;
else if (instance_type == keys::kRemoveResponseCookieType)
modification.type = helpers::REMOVE;
else
INPUT_FORMAT_VALIDATE(false);
// Get filter.
if (modification.type == helpers::EDIT ||
modification.type == helpers::REMOVE) {
const base::Value::Dict* filter = value.FindDict(keys::kFilterKey);
INPUT_FORMAT_VALIDATE(filter);
modification.filter = ParseFilterResponseCookie(*filter);
}
// Get new value.
if (modification.type == helpers::ADD) {
const base::Value::Dict* dict_value = value.FindDict(keys::kCookieKey);
INPUT_FORMAT_VALIDATE(dict_value);
modification.modification = ParseResponseCookie(*dict_value);
} else if (modification.type == helpers::EDIT) {
const base::Value::Dict* dict_value =
value.FindDict(keys::kModificationKey);
INPUT_FORMAT_VALIDATE(dict_value);
modification.modification = ParseResponseCookie(*dict_value);
}
return base::MakeRefCounted<WebRequestResponseCookieAction>(
std::move(modification));
}
scoped_refptr<const WebRequestAction> CreateSendMessageToExtensionAction(
const std::string& name,
const base::Value::Dict& value,
std::string* error,
bool* bad_message) {
const std::string* message = value.FindString(keys::kMessageKey);
INPUT_FORMAT_VALIDATE(message);
return base::MakeRefCounted<WebRequestSendMessageToExtensionAction>(*message);
}
struct WebRequestActionFactory {
using FactoryT = DedupingFactory<WebRequestAction, const base::Value::Dict&>;
FactoryT factory;
WebRequestActionFactory() : factory(5) {
factory.RegisterFactoryMethod(keys::kAddRequestCookieType,
FactoryT::IS_PARAMETERIZED,
&CreateRequestCookieAction);
factory.RegisterFactoryMethod(keys::kAddResponseCookieType,
FactoryT::IS_PARAMETERIZED,
&CreateResponseCookieAction);
factory.RegisterFactoryMethod(keys::kAddResponseHeaderType,
FactoryT::IS_PARAMETERIZED,
&CreateAddResponseHeaderAction);
factory.RegisterFactoryMethod(
keys::kCancelRequestType, FactoryT::IS_NOT_PARAMETERIZED,
&CallConstructorFactoryMethod<WebRequestCancelAction>);
factory.RegisterFactoryMethod(keys::kEditRequestCookieType,
FactoryT::IS_PARAMETERIZED,
&CreateRequestCookieAction);
factory.RegisterFactoryMethod(keys::kEditResponseCookieType,
FactoryT::IS_PARAMETERIZED,
&CreateResponseCookieAction);
factory.RegisterFactoryMethod(keys::kRedirectByRegExType,
FactoryT::IS_PARAMETERIZED,
&CreateRedirectRequestByRegExAction);
factory.RegisterFactoryMethod(keys::kRedirectRequestType,
FactoryT::IS_PARAMETERIZED,
&CreateRedirectRequestAction);
factory.RegisterFactoryMethod(
keys::kRedirectToTransparentImageType, FactoryT::IS_NOT_PARAMETERIZED,
&CallConstructorFactoryMethod<
WebRequestRedirectToTransparentImageAction>);
factory.RegisterFactoryMethod(
keys::kRedirectToEmptyDocumentType, FactoryT::IS_NOT_PARAMETERIZED,
&CallConstructorFactoryMethod<WebRequestRedirectToEmptyDocumentAction>);
factory.RegisterFactoryMethod(keys::kRemoveRequestCookieType,
FactoryT::IS_PARAMETERIZED,
&CreateRequestCookieAction);
factory.RegisterFactoryMethod(keys::kRemoveResponseCookieType,
FactoryT::IS_PARAMETERIZED,
&CreateResponseCookieAction);
factory.RegisterFactoryMethod(keys::kSetRequestHeaderType,
FactoryT::IS_PARAMETERIZED,
&CreateSetRequestHeaderAction);
factory.RegisterFactoryMethod(keys::kRemoveRequestHeaderType,
FactoryT::IS_PARAMETERIZED,
&CreateRemoveRequestHeaderAction);
factory.RegisterFactoryMethod(keys::kRemoveResponseHeaderType,
FactoryT::IS_PARAMETERIZED,
&CreateRemoveResponseHeaderAction);
factory.RegisterFactoryMethod(keys::kIgnoreRulesType,
FactoryT::IS_PARAMETERIZED,
&CreateIgnoreRulesAction);
factory.RegisterFactoryMethod(keys::kSendMessageToExtensionType,
FactoryT::IS_PARAMETERIZED,
&CreateSendMessageToExtensionAction);
}
};
WebRequestActionFactory& GetWebRequestActionFactory() {
static base::NoDestructor<WebRequestActionFactory> instance;
return *instance;
}
} // namespace
//
// WebRequestAction
//
WebRequestAction::~WebRequestAction() = default;
bool WebRequestAction::Equals(const WebRequestAction* other) const {
return type() == other->type();
}
bool WebRequestAction::HasPermission(ApplyInfo* apply_info,
const ExtensionId& extension_id) const {
PermissionHelper* permission_helper = apply_info->permission_helper;
const WebRequestInfo* request = apply_info->request_data->request;
if (WebRequestPermissions::HideRequest(permission_helper, *request))
return false;
// In unit tests we don't have a permission_helper object here and skip host
// permission checks.
if (!permission_helper)
return true;
// The embedder can always access all hosts from within a <webview>.
// The same is not true of extensions.
if (request->is_web_view)
return true;
WebRequestPermissions::HostPermissionsCheck permission_check =
WebRequestPermissions::REQUIRE_ALL_URLS;
switch (host_permissions_strategy()) {
case STRATEGY_DEFAULT: // Default value is already set.
break;
case STRATEGY_NONE:
permission_check = WebRequestPermissions::DO_NOT_CHECK_HOST;
break;
case STRATEGY_HOST:
permission_check = WebRequestPermissions::REQUIRE_HOST_PERMISSION_FOR_URL;
break;
}
// TODO(devlin): Pass in the real tab id here.
return WebRequestPermissions::CanExtensionAccessURL(
permission_helper, extension_id, request->url, -1,
apply_info->crosses_incognito, permission_check,
request->initiator, request->web_request_type) ==
PermissionsData::PageAccess::kAllowed;
}
// static
scoped_refptr<const WebRequestAction> WebRequestAction::Create(
content::BrowserContext* browser_context,
const Extension* extension,
const base::Value::Dict& json_action,
std::string* error,
bool* bad_message) {
*error = "";
*bad_message = false;
const std::string* instance_type =
json_action.FindString(keys::kInstanceTypeKey);
INPUT_FORMAT_VALIDATE(instance_type);
WebRequestActionFactory& factory = GetWebRequestActionFactory();
return factory.factory.Instantiate(*instance_type, json_action, error,
bad_message);
}
void WebRequestAction::Apply(const ExtensionId& extension_id,
base::Time extension_install_time,
ApplyInfo* apply_info) const {
if (!HasPermission(apply_info, extension_id))
return;
if (stages() & apply_info->request_data->stage) {
std::optional<EventResponseDelta> delta = CreateDelta(
*apply_info->request_data, extension_id, extension_install_time);
if (delta.has_value())
apply_info->deltas->push_back(std::move(delta.value()));
if (type() == WebRequestAction::ACTION_IGNORE_RULES) {
const WebRequestIgnoreRulesAction* ignore_action =
static_cast<const WebRequestIgnoreRulesAction*>(this);
if (!ignore_action->ignore_tag().empty())
apply_info->ignored_tags->insert(ignore_action->ignore_tag());
}
}
}
WebRequestAction::WebRequestAction(int stages,
Type type,
int minimum_priority,
HostPermissionsStrategy strategy)
: stages_(stages),
type_(type),
minimum_priority_(minimum_priority),
host_permissions_strategy_(strategy) {}
//
// WebRequestCancelAction
//
WebRequestCancelAction::WebRequestCancelAction()
: WebRequestAction(ON_BEFORE_REQUEST | ON_BEFORE_SEND_HEADERS |
ON_HEADERS_RECEIVED | ON_AUTH_REQUIRED,
ACTION_CANCEL_REQUEST,
std::numeric_limits<int>::min(),
STRATEGY_NONE) {}
WebRequestCancelAction::~WebRequestCancelAction() = default;
std::string WebRequestCancelAction::GetName() const {
return keys::kCancelRequestType;
}
std::optional<EventResponseDelta> WebRequestCancelAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
EventResponseDelta result(extension_id, extension_install_time);
result.cancel = true;
return result;
}
//
// WebRequestRedirectAction
//
WebRequestRedirectAction::WebRequestRedirectAction(const GURL& redirect_url)
: WebRequestAction(ON_BEFORE_REQUEST | ON_HEADERS_RECEIVED,
ACTION_REDIRECT_REQUEST,
std::numeric_limits<int>::min(),
STRATEGY_DEFAULT),
redirect_url_(redirect_url) {}
WebRequestRedirectAction::~WebRequestRedirectAction() = default;
bool WebRequestRedirectAction::Equals(const WebRequestAction* other) const {
return WebRequestAction::Equals(other) &&
redirect_url_ ==
static_cast<const WebRequestRedirectAction*>(other)->redirect_url_;
}
std::string WebRequestRedirectAction::GetName() const {
return keys::kRedirectRequestType;
}
std::optional<EventResponseDelta> WebRequestRedirectAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
if (request_data.request->url == redirect_url_)
return std::nullopt;
EventResponseDelta result(extension_id, extension_install_time);
result.new_url = redirect_url_;
return result;
}
//
// WebRequestRedirectToTransparentImageAction
//
WebRequestRedirectToTransparentImageAction::
WebRequestRedirectToTransparentImageAction()
: WebRequestAction(ON_BEFORE_REQUEST | ON_HEADERS_RECEIVED,
ACTION_REDIRECT_TO_TRANSPARENT_IMAGE,
std::numeric_limits<int>::min(),
STRATEGY_NONE) {}
WebRequestRedirectToTransparentImageAction::
~WebRequestRedirectToTransparentImageAction() {}
std::string WebRequestRedirectToTransparentImageAction::GetName() const {
return keys::kRedirectToTransparentImageType;
}
std::optional<EventResponseDelta>
WebRequestRedirectToTransparentImageAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
EventResponseDelta result(extension_id, extension_install_time);
result.new_url = GURL(kTransparentImageUrl);
return result;
}
//
// WebRequestRedirectToEmptyDocumentAction
//
WebRequestRedirectToEmptyDocumentAction::
WebRequestRedirectToEmptyDocumentAction()
: WebRequestAction(ON_BEFORE_REQUEST | ON_HEADERS_RECEIVED,
ACTION_REDIRECT_TO_EMPTY_DOCUMENT,
std::numeric_limits<int>::min(),
STRATEGY_NONE) {}
WebRequestRedirectToEmptyDocumentAction::
~WebRequestRedirectToEmptyDocumentAction() {}
std::string WebRequestRedirectToEmptyDocumentAction::GetName() const {
return keys::kRedirectToEmptyDocumentType;
}
std::optional<EventResponseDelta>
WebRequestRedirectToEmptyDocumentAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
EventResponseDelta result(extension_id, extension_install_time);
result.new_url = GURL(kEmptyDocumentUrl);
return result;
}
//
// WebRequestRedirectByRegExAction
//
WebRequestRedirectByRegExAction::WebRequestRedirectByRegExAction(
std::unique_ptr<RE2> from_pattern,
const std::string& to_pattern)
: WebRequestAction(ON_BEFORE_REQUEST | ON_HEADERS_RECEIVED,
ACTION_REDIRECT_BY_REGEX_DOCUMENT,
std::numeric_limits<int>::min(),
STRATEGY_DEFAULT),
from_pattern_(std::move(from_pattern)),
to_pattern_(to_pattern.data(), to_pattern.size()) {}
WebRequestRedirectByRegExAction::~WebRequestRedirectByRegExAction() = default;
// About the syntax of the two languages:
//
// ICU (Perl) states:
// $n The text of capture group n will be substituted for $n. n must be >= 0
// and not greater than the number of capture groups. A $ not followed by a
// digit has no special meaning, and will appear in the substitution text
// as itself, a $.
// \ Treat the following character as a literal, suppressing any special
// meaning. Backslash escaping in substitution text is only required for
// '$' and '\', but may be used on any other character without bad effects.
//
// RE2, derived from RE2::Rewrite()
// \ May only be followed by a digit or another \. If followed by a single
// digit, both characters represent the respective capture group. If followed
// by another \, it is used as an escape sequence.
// static
std::string WebRequestRedirectByRegExAction::PerlToRe2Style(
const std::string& perl) {
std::string::const_iterator i = perl.begin();
std::string result;
while (i != perl.end()) {
if (*i == '$') {
++i;
if (i == perl.end()) {
result += '$';
return result;
} else if (absl::ascii_isdigit(static_cast<unsigned char>(*i))) {
result += '\\';
result += *i;
} else {
result += '$';
result += *i;
}
} else if (*i == '\\') {
++i;
if (i == perl.end()) {
result += '\\';
} else if (*i == '$') {
result += '$';
} else if (*i == '\\') {
result += "\\\\";
} else {
result += *i;
}
} else {
result += *i;
}
++i;
}
return result;
}
bool WebRequestRedirectByRegExAction::Equals(
const WebRequestAction* other) const {
if (!WebRequestAction::Equals(other))
return false;
const WebRequestRedirectByRegExAction* casted_other =
static_cast<const WebRequestRedirectByRegExAction*>(other);
return from_pattern_->pattern() == casted_other->from_pattern_->pattern() &&
to_pattern_ == casted_other->to_pattern_;
}
std::string WebRequestRedirectByRegExAction::GetName() const {
return keys::kRedirectByRegExType;
}
std::optional<EventResponseDelta> WebRequestRedirectByRegExAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
CHECK(from_pattern_.get());
const std::string& old_url = request_data.request->url.spec();
std::string new_url = old_url;
if (!RE2::Replace(&new_url, *from_pattern_, to_pattern_) ||
new_url == old_url) {
return std::nullopt;
}
EventResponseDelta result(extension_id, extension_install_time);
result.new_url = GURL(new_url);
return result;
}
//
// WebRequestSetRequestHeaderAction
//
WebRequestSetRequestHeaderAction::WebRequestSetRequestHeaderAction(
const std::string& name,
const std::string& value)
: WebRequestAction(ON_BEFORE_SEND_HEADERS,
ACTION_SET_REQUEST_HEADER,
std::numeric_limits<int>::min(),
STRATEGY_DEFAULT),
name_(name),
value_(value) {}
WebRequestSetRequestHeaderAction::~WebRequestSetRequestHeaderAction() = default;
bool WebRequestSetRequestHeaderAction::Equals(
const WebRequestAction* other) const {
if (!WebRequestAction::Equals(other))
return false;
const WebRequestSetRequestHeaderAction* casted_other =
static_cast<const WebRequestSetRequestHeaderAction*>(other);
return name_ == casted_other->name_ && value_ == casted_other->value_;
}
std::string WebRequestSetRequestHeaderAction::GetName() const {
return keys::kSetRequestHeaderType;
}
std::optional<EventResponseDelta> WebRequestSetRequestHeaderAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
EventResponseDelta result(extension_id, extension_install_time);
result.modified_request_headers.SetHeader(name_, value_);
return result;
}
//
// WebRequestRemoveRequestHeaderAction
//
WebRequestRemoveRequestHeaderAction::WebRequestRemoveRequestHeaderAction(
const std::string& name)
: WebRequestAction(ON_BEFORE_SEND_HEADERS,
ACTION_REMOVE_REQUEST_HEADER,
std::numeric_limits<int>::min(),
STRATEGY_DEFAULT),
name_(name) {}
WebRequestRemoveRequestHeaderAction::~WebRequestRemoveRequestHeaderAction() =
default;
bool WebRequestRemoveRequestHeaderAction::Equals(
const WebRequestAction* other) const {
if (!WebRequestAction::Equals(other))
return false;
const WebRequestRemoveRequestHeaderAction* casted_other =
static_cast<const WebRequestRemoveRequestHeaderAction*>(other);
return name_ == casted_other->name_;
}
std::string WebRequestRemoveRequestHeaderAction::GetName() const {
return keys::kRemoveRequestHeaderType;
}
std::optional<EventResponseDelta>
WebRequestRemoveRequestHeaderAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
EventResponseDelta result(extension_id, extension_install_time);
result.deleted_request_headers.push_back(name_);
return result;
}
//
// WebRequestAddResponseHeaderAction
//
WebRequestAddResponseHeaderAction::WebRequestAddResponseHeaderAction(
const std::string& name,
const std::string& value)
: WebRequestAction(ON_HEADERS_RECEIVED,
ACTION_ADD_RESPONSE_HEADER,
std::numeric_limits<int>::min(),
STRATEGY_DEFAULT),
name_(name),
value_(value) {}
WebRequestAddResponseHeaderAction::~WebRequestAddResponseHeaderAction() =
default;
bool WebRequestAddResponseHeaderAction::Equals(
const WebRequestAction* other) const {
if (!WebRequestAction::Equals(other))
return false;
const WebRequestAddResponseHeaderAction* casted_other =
static_cast<const WebRequestAddResponseHeaderAction*>(other);
return name_ == casted_other->name_ && value_ == casted_other->value_;
}
std::string WebRequestAddResponseHeaderAction::GetName() const {
return keys::kAddResponseHeaderType;
}
std::optional<EventResponseDelta>
WebRequestAddResponseHeaderAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
const net::HttpResponseHeaders* headers =
request_data.original_response_headers;
if (!headers)
return std::nullopt;
// Don't generate the header if it exists already.
if (headers->HasHeaderValue(name_, value_))
return std::nullopt;
EventResponseDelta result(extension_id, extension_install_time);
result.added_response_headers.push_back(make_pair(name_, value_));
return result;
}
//
// WebRequestRemoveResponseHeaderAction
//
WebRequestRemoveResponseHeaderAction::WebRequestRemoveResponseHeaderAction(
const std::string& name,
const std::string& value,
bool has_value)
: WebRequestAction(ON_HEADERS_RECEIVED,
ACTION_REMOVE_RESPONSE_HEADER,
std::numeric_limits<int>::min(),
STRATEGY_DEFAULT),
name_(name),
value_(value),
has_value_(has_value) {}
WebRequestRemoveResponseHeaderAction::~WebRequestRemoveResponseHeaderAction() =
default;
bool WebRequestRemoveResponseHeaderAction::Equals(
const WebRequestAction* other) const {
if (!WebRequestAction::Equals(other))
return false;
const WebRequestRemoveResponseHeaderAction* casted_other =
static_cast<const WebRequestRemoveResponseHeaderAction*>(other);
return name_ == casted_other->name_ && value_ == casted_other->value_ &&
has_value_ == casted_other->has_value_;
}
std::string WebRequestRemoveResponseHeaderAction::GetName() const {
return keys::kRemoveResponseHeaderType;
}
std::optional<EventResponseDelta>
WebRequestRemoveResponseHeaderAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
const net::HttpResponseHeaders* headers =
request_data.original_response_headers;
if (!headers)
return std::nullopt;
EventResponseDelta result(extension_id, extension_install_time);
size_t iter = 0;
std::optional<std::string_view> current_value;
while ((current_value = headers->EnumerateHeader(&iter, name_))) {
if (has_value_ &&
!base::EqualsCaseInsensitiveASCII(*current_value, value_)) {
continue;
}
result.deleted_response_headers.emplace_back(name_, *current_value);
}
return result;
}
//
// WebRequestIgnoreRulesAction
//
WebRequestIgnoreRulesAction::WebRequestIgnoreRulesAction(
int minimum_priority,
const std::string& ignore_tag)
: WebRequestAction(ON_BEFORE_REQUEST | ON_BEFORE_SEND_HEADERS |
ON_HEADERS_RECEIVED | ON_AUTH_REQUIRED,
ACTION_IGNORE_RULES,
minimum_priority,
STRATEGY_NONE),
ignore_tag_(ignore_tag) {}
WebRequestIgnoreRulesAction::~WebRequestIgnoreRulesAction() = default;
bool WebRequestIgnoreRulesAction::Equals(const WebRequestAction* other) const {
if (!WebRequestAction::Equals(other))
return false;
const WebRequestIgnoreRulesAction* casted_other =
static_cast<const WebRequestIgnoreRulesAction*>(other);
return minimum_priority() == casted_other->minimum_priority() &&
ignore_tag_ == casted_other->ignore_tag_;
}
std::string WebRequestIgnoreRulesAction::GetName() const {
return keys::kIgnoreRulesType;
}
std::optional<EventResponseDelta> WebRequestIgnoreRulesAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
return std::nullopt;
}
//
// WebRequestRequestCookieAction
//
WebRequestRequestCookieAction::WebRequestRequestCookieAction(
RequestCookieModification request_cookie_modification)
: WebRequestAction(ON_BEFORE_SEND_HEADERS,
ACTION_MODIFY_REQUEST_COOKIE,
std::numeric_limits<int>::min(),
STRATEGY_DEFAULT),
request_cookie_modification_(std::move(request_cookie_modification)) {}
WebRequestRequestCookieAction::~WebRequestRequestCookieAction() = default;
bool WebRequestRequestCookieAction::Equals(
const WebRequestAction* other) const {
if (!WebRequestAction::Equals(other))
return false;
const WebRequestRequestCookieAction* casted_other =
static_cast<const WebRequestRequestCookieAction*>(other);
return request_cookie_modification_ ==
casted_other->request_cookie_modification_;
}
std::string WebRequestRequestCookieAction::GetName() const {
switch (request_cookie_modification_.type) {
case helpers::ADD:
return keys::kAddRequestCookieType;
case helpers::EDIT:
return keys::kEditRequestCookieType;
case helpers::REMOVE:
return keys::kRemoveRequestCookieType;
}
NOTREACHED();
}
std::optional<EventResponseDelta> WebRequestRequestCookieAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
EventResponseDelta result(extension_id, extension_install_time);
result.request_cookie_modifications.push_back(
request_cookie_modification_.Clone());
return result;
}
//
// WebRequestResponseCookieAction
//
WebRequestResponseCookieAction::WebRequestResponseCookieAction(
ResponseCookieModification response_cookie_modification)
: WebRequestAction(ON_HEADERS_RECEIVED,
ACTION_MODIFY_RESPONSE_COOKIE,
std::numeric_limits<int>::min(),
STRATEGY_DEFAULT),
response_cookie_modification_(std::move(response_cookie_modification)) {}
WebRequestResponseCookieAction::~WebRequestResponseCookieAction() = default;
bool WebRequestResponseCookieAction::Equals(
const WebRequestAction* other) const {
if (!WebRequestAction::Equals(other))
return false;
const WebRequestResponseCookieAction* casted_other =
static_cast<const WebRequestResponseCookieAction*>(other);
return response_cookie_modification_ ==
casted_other->response_cookie_modification_;
}
std::string WebRequestResponseCookieAction::GetName() const {
switch (response_cookie_modification_.type) {
case helpers::ADD:
return keys::kAddResponseCookieType;
case helpers::EDIT:
return keys::kEditResponseCookieType;
case helpers::REMOVE:
return keys::kRemoveResponseCookieType;
}
NOTREACHED();
}
std::optional<EventResponseDelta> WebRequestResponseCookieAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
EventResponseDelta result(extension_id, extension_install_time);
result.response_cookie_modifications.push_back(
response_cookie_modification_.Clone());
return result;
}
//
// WebRequestSendMessageToExtensionAction
//
WebRequestSendMessageToExtensionAction::WebRequestSendMessageToExtensionAction(
const std::string& message)
: WebRequestAction(ON_BEFORE_REQUEST | ON_BEFORE_SEND_HEADERS |
ON_HEADERS_RECEIVED | ON_AUTH_REQUIRED,
ACTION_SEND_MESSAGE_TO_EXTENSION,
std::numeric_limits<int>::min(),
STRATEGY_HOST),
message_(message) {}
WebRequestSendMessageToExtensionAction::
~WebRequestSendMessageToExtensionAction() {}
bool WebRequestSendMessageToExtensionAction::Equals(
const WebRequestAction* other) const {
if (!WebRequestAction::Equals(other))
return false;
const WebRequestSendMessageToExtensionAction* casted_other =
static_cast<const WebRequestSendMessageToExtensionAction*>(other);
return message_ == casted_other->message_;
}
std::string WebRequestSendMessageToExtensionAction::GetName() const {
return keys::kSendMessageToExtensionType;
}
std::optional<EventResponseDelta>
WebRequestSendMessageToExtensionAction::CreateDelta(
const WebRequestData& request_data,
const ExtensionId& extension_id,
const base::Time& extension_install_time) const {
CHECK(request_data.stage & stages());
EventResponseDelta result(extension_id, extension_install_time);
result.messages_to_extension.insert(message_);
return result;
}
} // namespace extensions