Everybody is good! I am a chicken H who likes the front end
demand
- After installing the plug-in, the entire browser can be blocked, and the plug-in can be configured to specify the blocking interface for display, including downloading and exporting the intercepted data content, which can cooperate with the back end to do a lot of things
The problem
- Intercepts all requests to assemble request information and results
- Plug-in and page communication, do corresponding operations
rendering
Json is a configuration file that the Chrome plugin reads to initialize, such as the icon configuration of some plug-ins, the page that the plug-in clicks to display, the permissions required by the plug-in, and so on....
The first plug-in
- Create the manifest.json configuration file
- Create a display icon
- Create a presentation page
//chromePlugin file structure
- icon
- logo.png // Image size cannot be larger than 129... I guess... If not, make it smaller
- background
- index.html
- index.js
- browser_action
- index.html
- index.js
- index.css
- manifest.json
Copy the code
manifest.json
{
// The name of your plugin
"name": "chrome"./ / description
"description": "Chrome plug-ins"./ / version
"version": "1.0".// Fill in 2
"manifest_version": 2.// You can interpret it as a background server for your plug-in injection in the browser
"background": {
"page": "/background/index.html"
},
// The page is displayed after the plugin is clicked
"browser_action": {
"default_icon": "/icon/logo.png"."default_title": "Chrome plug-ins"."default_popup": "/browser_action/index.html"
},
//icons
"icons": {
"16": "/icon/logo.png"."32": "/icon/logo.png"."48": "/icon/logo.png"."128": "/icon/logo.png"}}Copy the code
browser_action/index.html && index.css
//index.html
<! DOCTYPEhtml>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="Width = device - width, initial - scale = 1.0">
<title>Document</title>/* Notice the CSS path rule */ introduced here<link rel="stylesheet" type="text/css" href="/browser_action/index.css">
</head>
<body>
<div class="app">My first Chrome plugin</div>
</body>
</html>//index.css .app{ width: 200px; height: 100px; background: yellow; }Copy the code
A brief description of the configuration file flow, background you can understand that your plugin always resides in the browser code block, can put some data to share with the plugin page, etc... Browser_action is the content of the page when the plugin is clicked. You can try to write some of the content that you want to display. Then open the browser, more tools -> Extensions, and open the development mode in the upper right corner Then drag your project directly into it, it will automatically recognize it, your plugin will be installed without any accidents, then click on the plugin, and congratulations, your first Chrome plugin is complete! 😊
Problem 1: Intercepting all requests assembles request information and results
Idea is: Overrides XMLHttpRequest and FETCH, and then goes to each page via the configuration files provided by Chrome to inject the overwritten code, To block the effect, first recognize a content_scripts configuration, which is a configuration that tells the Chrome plugin I need to load my JS in the current web page to add the following code to manifest.json
// The rule configuration injected into the page JS
"content_scripts": [{// define which pages need to be injected with content script "
" all pages
"matches": ["<all_urls>"].// CSS file address
"css": [].// Address of the injected JS file
"js": ["/contentScript/install.js"].// Control the timing of content script injection. It can be document_start, document_end, or document_idle. The default document_idle.
"run_at":"document_start"}].// Use chrome.extension.getURL to obtain the path of the resources in the package. The access permission web_accessibLE_resources needs to be set in the manifest.json file
"web_accessible_resources": [
"/contentScript/network.js"
]
Copy the code
Ok so I can add into the js script command, and now we need to create good folders and files under the corresponding path/contentScript/the js then under contentScript folder to create a network. The js and install. Js
install.js
setTimeout(() = > {
const script = document.createElement('script');
script.setAttribute('type'.'text/javascript');
// Use chrome.extension.getURL to obtain the path of the resources in the package. The access permission web_accessibLE_resources needs to be set in the manifest.json file
script.setAttribute('src', chrome.extension.getURL('/contentScript/network.js'));
document.head.appendChild(script);
});
Copy the code
Override request interception method network.js
const tool = {
isString(value) {
return Object.prototype.toString.call(value) == '[object String]';
},
isPlainObject(obj) {
let hasOwn = Object.prototype.hasOwnProperty;
// Must be an Object.
if(! obj ||typeofobj ! = ='object' || obj.nodeType || isWindow(obj)) {
return false;
}
try {
if(obj.constructor && ! hasOwn.call(obj,'constructor') && !hasOwn.call(obj.constructor.prototype, 'isPrototypeOf')) {
return false; }}catch (e) {
return false;
}
let key;
for (key in obj) {}
return key === undefined|| hasOwn.call(obj, key); }}/ / this class is based on the tencent open source vconsole (https://github.com/Tencent/vConsole), a class of this plugin
class RewriteNetwork {
constructor() {
this.reqList = {}; // URL as key, request item as value
this._open = undefined; // the origin function
this._send = undefined;
this._setRequestHeader = undefined;
this.status = false;
this.mockAjax();
this.mockFetch();
}
onRemove() {
if (window.XMLHttpRequest) {
window.XMLHttpRequest.prototype.open = this._open;
window.XMLHttpRequest.prototype.send = this._send;
window.XMLHttpRequest.prototype.setRequestHeader = this._setRequestHeader;
this._open = undefined;
this._send = undefined;
this._setRequestHeader = undefined}}/**
* mock ajax request
* @private* /
mockAjax() {
let _XMLHttpRequest = window.XMLHttpRequest;
if(! _XMLHttpRequest) {return; }
const that = this;
// Save the native _XMLHttpRequest method for overriding below
const _open = window.XMLHttpRequest.prototype.open,
_send = window.XMLHttpRequest.prototype.send,
_setRequestHeader = window.XMLHttpRequest.prototype.setRequestHeader;
that._open = _open;
that._send = _send;
that._setRequestHeader = _setRequestHeader;
// Override request header open
window.XMLHttpRequest.prototype.open = function() {
let XMLReq = this;
let args = [].slice.call(arguments),
method = args[0],
url = args[1],
id = that.getUniqueID();
let timer = null;
// may be used by other functions
XMLReq._requestID = id;
XMLReq._method = method;
XMLReq._url = url;
// mock onreadystatechange
let _onreadystatechange = XMLReq.onreadystatechange || function() {};
// Timed polling to see the event handle function called each time the readyState property changes. It may also be called multiple times when readyState is 3.
let onreadystatechange = function() {
let item = that.reqList[id] || {};
// Resume initialization
item.readyState = XMLReq.readyState;
item.status = 0;
// Synchronize XMLReq state
if (XMLReq.readyState > 1) {
item.status = XMLReq.status;
}
item.responseType = XMLReq.responseType;
// Initialization state. The XMLHttpRequest object has been created or reset by the ABORT () method.
if (XMLReq.readyState == 0) {
if(! item.startTime) { item.startTime = (+new Date());
}
// The open() method is called, but the send() method is not. The request has not been sent yet
} else if (XMLReq.readyState == 1) {
if(! item.startTime) { item.startTime = (+new Date());
}
// The Send() method has been called and the HTTP request has been sent to the Web server. No response was received.
} else if (XMLReq.readyState == 2) {
// HEADERS_RECEIVED
item.header = {};
let header = XMLReq.getAllResponseHeaders() || ' ',
headerArr = header.split("\n");
// extract plain text to key-value format
for (let i=0; i<headerArr.length; i++) {
let line = headerArr[i];
if(! line) {continue; }
let arr = line.split(':');
let key = arr[0],
value = arr.slice(1).join(':');
item.header[key] = value;
}
// All response headers have been received. The response body began receiving but did not complete
} else if (XMLReq.readyState == 3) {
// The HTTP response has been fully received.
} else if (XMLReq.readyState == 4) {
clearInterval(timer);
item.endTime = +new Date(),
item.costTime = item.endTime - (item.startTime || item.endTime);
item.response = XMLReq.response;
item.method = XMLReq._method;
item.url = XMLReq._url;
item.req_type = 'xml';
item.getData = XMLReq.getData;
item.postData = XMLReq.postData;
that.filterData(item)
} else {
clearInterval(timer);
}
return _onreadystatechange.apply(XMLReq, arguments);
};
XMLReq.onreadystatechange = onreadystatechange;
// Polls the query status
let preState = -1;
timer = setInterval(function() {
if (preState != XMLReq.readyState) {
preState = XMLReq.readyState;
onreadystatechange.call(XMLReq);
}
}, 10);
return _open.apply(XMLReq, args);
};
// Override request header setRequestHeader
window.XMLHttpRequest.prototype.setRequestHeader = function() {
const XMLReq = this;
const args = [].slice.call(arguments);
const item = that.reqList[XMLReq._requestID];
if (item) {
if(! item.requestHeader) { item.requestHeader = {}; } item.requestHeader[args[0]] = args[1];
}
return _setRequestHeader.apply(XMLReq, args);
};
/ / rewrite the send
window.XMLHttpRequest.prototype.send = function() {
let XMLReq = this;
let args = [].slice.call(arguments),
data = args[0];
let item = that.reqList[XMLReq._requestID] || {};
item.method = XMLReq._method ? XMLReq._method.toUpperCase() : 'GET';
let query = XMLReq._url ? XMLReq._url.split('? ') : []; // a.php? b=c&d=? e => ['a.php', 'b=c&d=', 'e']
item.url = XMLReq._url || ' ';
item.name = query.shift() || ' '; // => ['b=c&d=', 'e']
item.name = item.name.replace(new RegExp('[/] * $'), ' ').split('/').pop() || ' ';
if (query.length > 0) {
item.name += '? ' + query;
item.getData = {};
query = query.join('? '); // => 'b=c&d=? e'
query = query.split('&'); // => ['b=c', 'd=?e']
for (let q of query) {
q = q.split('=');
item.getData[ q[0]] =decodeURIComponent(q[1]); }}if (item.method == 'POST') {
// save POST data
if (tool.isString(data)) {
let arr = data.split('&');
item.postData = {};
for (let q of arr) {
q = q.split('=');
item.postData[ q[0] ] = q[1]; }}else if (tool.isPlainObject(data)) {
item.postData = data;
} else {
item.postData = '[object Object]';
}
}
XMLReq.getData = item.getData || "";
XMLReq.postData = item.postData || "";
return _send.apply(XMLReq, args);
};
};
/**
* mock fetch request
* @private* /
mockFetch() {
const _fetch = window.fetch;
if(! _fetch) {return ""; }
const that = this;
const prevFetch = function(input, init){
let id = that.getUniqueID();
that.reqList[id] = {};
let item = that.reqList[id] || {};
let query = [],
url = ' ',
method = 'GET',
requestHeader = null;
// handle `input` content
if (tool.isString(input)) { // when `input` is a string
method = init.method ? init.method : 'GET';
url = input;
requestHeader = init.headers ? init.headers : null
} else { // when `input` is a `Request` object
method = input.method || 'GET';
url = input.url;
requestHeader = input.headers;
}
query = url.split('? ');
item.id = id;
item.method = method;
item.requestHeader = requestHeader;
item.url = url;
item.name = query.shift() || ' ';
item.name = item.name.replace(new RegExp('[/] * $'), ' ').split('/').pop() || ' ';
if (query.length > 0) {
item.name += '? ' + query;
item.getData = {};
query = query.join('? '); // => 'b=c&d=? e'
query = query.split('&'); // => ['b=c', 'd=?e']
for (let q of query) {
q = q.split('=');
item.getData[ q[0] ] = q[1]; }}if (item.method === "post") {
if (tool.isString(input)) {
if (tool.isString(init.body && init.body)) {
let arr = init.body.split('&');
item.postData = {};
for (let q of arr) {
q = q.split('=');
item.postData[ q[0] ] = q[1]; }}else if (tool.isPlainObject(init.body && init.body)) {
item.postData = init.body && init.body;
} else {
item.postData = '[object Object]'; }}else {
item.postData = '[object Object]'; }}// UNSENT
if(! item.startTime) { item.startTime = (+new Date()); }
return _fetch(url, init).then((response) = > {
response.clone().json().then((json) = > {
item.endTime = +new Date(),
item.costTime = item.endTime - (item.startTime || item.endTime);
item.status = response.status;
item.header = {};
for (let pair of response.headers.entries()) {
item.header[pair[0]] = pair[1];
}
item.response = json;
item.readyState = 4;
const contentType = response.headers.get('content-type');
item.responseType = contentType.includes('application/json')?'json' : contentType.includes('text/html')?'text' : ' ';
item.req_type = 'fetch';
that.filterData(item)
return json;
})
returnresponse; })}window.fetch = prevFetch;
}
filterData({ url,method,req_type,response,getData,postData}){
if(! url)return;
const req_data = {
url,
method,
req_type,
response,
getData, / / query parameters
postData
}
console.log('Result of intercept',req_data)
}
/**
* generate an unique id string (32)
* @private
* @return string* /
getUniqueID() {
let id = 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g.function(c) {
let r = Math.random()*16|0, v = c == 'x' ? r : (r&0x3|0x8);
return v.toString(16);
});
returnid; }}const network = new RewriteNetwork();
Copy the code
Just open a test site, F12 open the console and then there will be the result output interception, so we completed the page interface interception, next we need to complete, plug-in and page communication, do the corresponding operation
Problem 2: Plug-in and page communication
- Inject_js (the JS that is actually inserted into the page) communicates with content_script
Inject_js uses the postMessage method to communicate with the content_script, sending data to the content_script in the intercepting request method
//network.js
const senMes = (data) = >{
window.postMessage(data, The '*'); }...console.log('Result of intercept',req_data)
senMes(req_data)
//install.js
// Receives the Inject page message.window.addEventListener("message".function(e){
const { data } = e;
console.log('Receive networkJS data',data)
}, false);
Copy the code
- Content_script communicates with background (background permanent injection service)
//content_script/install.js
const sendBgMessage = (data) = >{
chrome.runtime.sendMessage({ type:'page_request',data}, function(response) {
console.log('Background reply:' + response);
});
}
// Background (background permanent injection service) receive
chrome.runtime.onMessage.addListener(function(request, sender, sendResponse){
console.log('BACKGROUND receives data',request)
/ / reply
sendResponse('BG background receives message')});Copy the code
- The browser_action page communicates with background.js pretty much the same way
/ / js browser_action page
const sendMes = (data) = >{
return new Promise( resolve= >{
chrome.runtime.sendMessage( data, (res) = >{ resolve(res) }); })}// Background (background permanent injection service) receive
chrome.runtime.onMessage.addListener(function(request, sender, sendResponse){
console.log('BACKGROUND receives data',request)
/ / reply
sendResponse('BG background receives message')});Copy the code
The end
Basically some of the core content is here, the next is to configure according to their own actual business scenarios to complete him, I will post the development document
Chrome Chinese Documentation
Chrome English