The WebRequest module is new in Firefox 41.
The WebRequest module provides an API to add event listeners for the various stages of making an HTTP request. The event listener receives detailed information about the request, and can modify or cancel the request.
You can use this API to implement a content policy in an add-on (for example, an ad or script blocker). Previously you would use nsIContentPolicy
for this, but in multiprocess Firefox you can't use nsIContentPolicy in the chrome process. More generally, you can also use this API instead of HTTP request observers. You can use WebRequest.jsm in either the chrome or content processes.
The WebRequest API is modeled on Chrome's webRequest extension API, which makes it easier to write cross-browser add-on code. There are some differences between this API and Chrome's, though: see Chrome incompatibilities for the details.
To import WebRequest, use code like:
let {WebRequest} = Cu.import("resource://gre/modules/WebRequest.jsm", {});
The WebRequest
object has the following properties, each of which corresponds to a specific stage in executing a web request:
Each of these objects defines two functions:
addListener(callback, filter, opt_extraInfoSpec)
removeListener(callback)
Use addListener
to add a listener to a particular event. It takes one mandatory argument and two optional arguments, as detailed below.
Name | Type | Description |
---|---|---|
callback |
Function |
The It's passed an object whose structure will vary depending on the event you're listening to: see the event-specific documentation for details. |
filter |
Object | The optional filter argument is used to restrict the requests for which the listener callback will be invoked. See Filtering. |
opt_extraInfoSpec |
Array of String |
The optional The instructions you can pass here vary depending on the event you're listening to: see the event-specific documentation for details. In particular, |
Some examples of using addListener()
.
To remove a listener for an event, call removeListener
on the event in question, passing it the listener to remove.
The filter
argument is the second argument to addListener()
. If you don't pass filter
, then the listener is invoked for every web request the browser makes that triggers that particular event.
filter
is an object that may contain up to two properties: urls
and types
.
Name | Type | Description |
---|---|---|
urls |
MatchPattern | Only invoke the listener for URLs that match one of the patterns. |
types |
Array of Strings. | An array of resource types. Only invoke the listener for resource types that match one of the types given. |
To specify match patterns, first import the "MatchPattern.jsm" module, then construct a MatchPattern
. The MatchPattern
constructor accepts:
Cu.import("resource://gre/modules/MatchPattern.jsm");
let pattern = new MatchPattern("https:/developer.mozilla.org/*");
Cu.import("resource://gre/modules/MatchPattern.jsm");
let pattern = new MatchPattern(["https://mozorg.cdn.mozilla.net/*",
"https://mdn.mozillademos.org/*",
"/*"]);
See the match patterns article for details on the syntax of the strings you supply. If you supply an array of strings, then a URL matches if it matches any of the strings.
See some examples of using filter
.
You can use the WebRequest API to modify and cancel network requests. What exactly you're able to do to a network request depends on the stage at which you have intercepted the request: that is, the WebRequest event that your listener is attached to. For example, it obviously doesn't make sense to cancel a network request after it has completed.
The general approach to modifying and canceling requests is this:
"blocking"
as the third argument to your addListener()
call. This tells the browser to wait for your event listener to return before continuing with the request.There are four things you can do with a request:
To perform one of these operations, include a property with the appropriate name in the object returned from the listener, and assign it the desired value, as detailed in the table below:
Operation | Available in | Return property |
---|---|---|
Cancel | onBeforeRequest |
Boolean value set to |
Redirect | onBeforeSendHeaders |
String set to the URL to redirect the request to. See some examples of redirection. |
Modify request headers | onBeforeSendHeaders |
Array of HTTP headers. |
Modify response headers | onHeadersReceived |
Array of HTTP headers. |
See some examples of canceling, redirecting, and modifying request headers.
The events exported by WebRequest.jsm are triggered at the various stages of making a request.
This section describes the stage corresponding to each event. It also documents two data structures that vary from one event to another:
opt_extraInfoSpec
argument to addListener()
This event is triggered when a request is about to be made, and before headers are available. This is a good place to listen if you want to cancel the request.
Name | Description |
---|---|
"blocking" |
Make the browser wait until the listener returns. Use this if you want to cancel the request. See some examples of canceling requests. |
Name | Type | Description |
---|---|---|
url |
String | Target of the request. |
windowId |
Number | The ID of the window making the request, as an outerWindowID . |
parentWindowId |
Number | |
type |
String | The resource type. |
browser |
Object | The XUL browser into which the resource will be loaded, or null if the resource will not be loaded into a XUL browser. |
This event is triggered before sending any HTTP data, but after all HTTP headers are available. This is a good place to listen if you want to modify HTTP headers.
Name | Description |
---|---|
"blocking" |
Make the browser wait until the listener returns. Use this if you want to redirect the request or modify request headers. See examples of redirecting and modifying request headers. |
"requestHeaders" |
Include request headers in the argument passed to the listener. See Modifying headers for an example using this. |
Name | Type | Description |
---|---|---|
url |
String | Target of the request. |
windowId |
Number | The ID of the window making the request, as an outerWindowID . |
parentWindowId |
Number | |
type |
String | The resource type. |
browser |
Object | The XUL browser into which the resource will be loaded, or null if the resource will not be loaded into a XUL browser. |
method |
String | The HTTP method to be used (GET, POST, etc). |
requestHeaders |
Array of HTTP headers. |
The HTTP request headers that will be sent. This is only included if you set |
Triggered just before sending headers. If you modified headers in onBeforeSendHeaders
, you'll see the modified version here. Note that you can't pass "blocking"
for this event, so you can't modify or cancel the request from this event: it's informational only.
Name | Description |
---|---|
"requestHeaders" |
Include request headers in the argument passed to the listener. |
Name | Type | Description |
---|---|---|
url |
String | Target of the request. |
windowId |
Number | The ID of the window making the request, as an outerWindowID . |
parentWindowId |
Number | |
type |
String | The resource type. |
browser |
Object | The XUL browser into which the resource will be loaded, or null if the resource will not be loaded into a XUL browser. |
method |
String | The HTTP method to be used (GET, POST, etc). |
requestHeaders |
Array of HTTP headers. |
The HTTP request headers that will be sent. This is only included if you set |
Triggered once response headers have been received. You can use this event to modify HTTP response headers.
Name | Description |
---|---|
"blocking" |
Make the browser wait until the listener returns. Use this if you want to modify response headers. See Modifying and canceling requests. |
"responseHeaders" |
Include response headers in the argument passed to the listener. |
Name | Type | Description |
---|---|---|
url |
String | Target of the request. |
windowId |
Number | The ID of the window making the request, as an outerWindowID . |
parentWindowId |
Number | |
type |
String | The resource type. |
browser |
Object | The XUL browser into which the resource will be loaded, or null if the resource will not be loaded into a XUL browser. |
method |
String | The HTTP method to be used (GET, POST, etc). |
responseHeaders |
Array of HTTP headers. |
The HTTP response headers that were received. This is only included if you set |
statusCode |
Number | HTTP response code. |
Triggered once the browser has started to receive the response body. Note that you can't pass "blocking"
for this event, so you can't modify or cancel the request from this event: it's informational only.
Name | Description |
---|---|
"responseHeaders" |
Include response headers in the argument passed to the listener. |
Name | Type | Description |
---|---|---|
url |
String | Target of the request. |
windowId |
Number | The ID of the window making the request, as an outerWindowID . |
parentWindowId |
Number | |
type |
String | The resource type. |
browser |
Object | The XUL browser into which the resource will be loaded, or null if the resource will not be loaded into a XUL browser. |
method |
String | The HTTP method used (GET, POST, etc). |
responseHeaders |
Array of HTTP headers. |
The HTTP response headers that were received. This is only included if you set |
statusCode |
Number | HTTP response code. |
Triggered when the response has been received. Note that you can't pass "blocking"
for this event, so you can't modify or cancel the request from this event: it's informational only.
Name | Description |
---|---|
"responseHeaders" |
Include response headers in the argument passed to the listener. |
Name | Type | Description |
---|---|---|
url |
String | Target of the request. |
windowId |
Number | The ID of the window making the request, as an outerWindowID . |
parentWindowId |
Number | |
type |
String | The resource type. |
browser |
Object | The XUL browser into which the resource will be loaded, or null if the resource will not be loaded into a XUL browser. |
method |
String | The HTTP method used (GET, POST, etc). |
responseHeaders |
Array of HTTP headers. |
The HTTP response headers that were received. This is only included if you set |
statusCode |
Number | HTTP response code. |
The resource type string identifies the kind of resource being fetched. It's used in two places: filtering requests by type, and as the type
property of the argument passed to the listener.
The following types are supported:
"main_frame" |
"sub_frame" |
"stylesheet" |
"script" |
"image" |
"object" |
"xmlhttprequest" |
HTTPs headers are represented as objects with two properties, name
and value
:
Name | Type | Description |
---|---|---|
name |
String | Header name, for example "Content-Type" |
value |
String | Header value, for example "image/png" |
Although this API is modeled on Chrome's webRequest extension API, there are some differences. We're working on addressing them, though.
windowId
and tabId
is not supported.onAuthRequired
, onBeforeRedirect
, and onErrorOccurred
events are not supported."requestBody"
instruction in opt_extraInfoSpec
is not supported.onBeforeRequest
or onHeadersReceived
, but is allowed in onBeforeSendHeaders
. See bug 1176092.handlerBehaviorChanged()
is not supported.requestId
is not included in the argument passed to the listener.This example just logs the URL for every request initiated:
let {WebRequest} = Cu.import("resource://gre/modules/WebRequest.jsm", {}); WebRequest.onBeforeRequest.addListener(logURL); function logURL(e) { console.log("Loading: " + e.url); }
This example logs URLs for requests under "/":
let {WebRequest} = Cu.import("resource://gre/modules/WebRequest.jsm", {}); Cu.import("resource://gre/modules/MatchPattern.jsm"); let pattern = new MatchPattern("https:/developer.mozilla.org/*"); WebRequest.onBeforeRequest.addListener(logURL, {urls: pattern}); function logURL(e) { console.log("Loading: " + e.url); }
This listener will be invoked for requests matching any of the three patterns, where the resource type is "stylesheet" or "image":
let {WebRequest} = Cu.import("resource://gre/modules/WebRequest.jsm", {}); Cu.import("resource://gre/modules/MatchPattern.jsm"); let pattern = new MatchPattern(["https://mozorg.cdn.mozilla.net/*", "https://mdn.mozillademos.org/*", "/*"]); WebRequest.onBeforeRequest.addListener(listener, { urls: pattern, types: ["image", "stylesheet"] }); function listener(e) { console.log("Matched: " + e.url); }
This example cancels requests to load content from "http://example.org/":
let {WebRequest} = Cu.import("resource://gre/modules/WebRequest.jsm", {}); Cu.import("resource://gre/modules/MatchPattern.jsm"); let pattern = new MatchPattern("http://example.org/*"); WebRequest.onBeforeRequest.addListener(cancelRequest, {urls: pattern}, ["blocking"]); function cancelRequest(e) { console.log("Canceling: " + e.url); return {cancel: true}; }
This code cancels requests for images that are made to URLs under "https://mdn.mozillademos.org/":
let {WebRequest} = Cu.import("resource://gre/modules/WebRequest.jsm", {}); Cu.import("resource://gre/modules/MatchPattern.jsm"); let pattern = new MatchPattern("https://mdn.mozillademos.org/*"); WebRequest.onBeforeRequest.addListener(cancelImages, { urls: pattern, types: ["image"] }, ["blocking"]); function cancelImages(e) { console.log("Canceling: " + e.url); return {cancel: true}; }
This code replaces, by redirection, all network requests for images that are made to URLs under "https://mdn.mozillademos.org/":
let {WebRequest} = Cu.import("resource://gre/modules/WebRequest.jsm", {}); Cu.import("resource://gre/modules/MatchPattern.jsm"); let pattern = new MatchPattern("https://mdn.mozillademos.org/*"); WebRequest.onBeforeSendHeaders.addListener(redirect, { urls: pattern, types: ["image"] }, ["blocking"]); function redirect(e) { console.log("Redirecting: " + e.url); return {redirectUrl: "https://38.media.tumblr.com/tumblr_ldbj01lZiP1qe0eclo1_500.gif"}; }
This code changes the User Agent header so the browser identifies itself as IE 11, but only when visiting pages under "http://useragentstring.com/":
let {WebRequest} = Cu.import("resource://gre/modules/WebRequest.jsm", {}); Cu.import("resource://gre/modules/MatchPattern.jsm"); let pattern = new MatchPattern("http://useragentstring.com/*"); let ua = "Mozilla/5.0 (Windows NT 6.1; WOW64; Trident/7.0; AS; rv:11.0) like Gecko"; WebRequest.onBeforeSendHeaders.addListener(changeUserAgent, { urls: pattern }, ["blocking", "requestHeaders"]); function changeUserAgent(e) { for (let header of e.requestHeaders) { if (header.name == "User-Agent") { header.value = ua; } } return {requestHeaders: e.requestHeaders}; }