Skip to content
Snippets Groups Projects
background.js 8.82 KiB
Newer Older
didierfred's avatar
didierfred committed

/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. 
 *
 * @author didierfred@gmail.com
didierfred's avatar
didierfred committed


"use strict";

didierfred's avatar
didierfred committed
let config;
let started = 'off';
didierfred's avatar
didierfred committed
let debug_mode = false;
didierfred's avatar
didierfred committed

loadFromBrowserStorage(['config','started'],function(result) {
  config = result.config;
 
 // if old storage method
  if (config===undefined)  loadConfigurationFromLocalStorage();
  else started = result.started;
didierfred's avatar
didierfred committed

  if (started==='on') {
    addListener();
    chrome.browserAction.setIcon({ path: 'icons/modify-green-32.png'});
didierfred's avatar
didierfred committed
  }
  else if (started !== 'off') { 
    started = 'off';
    storeInBrowserStorage({started:'off'});    
didierfred's avatar
didierfred committed
  }
  // listen for change in configuration or start/stop
  chrome.runtime.onMessage.addListener(notify);
});


function  loadConfigurationFromLocalStorage() {
  // if configuration exist 
  if (localStorage.getItem('config')) {
    console.log("Load standard config");
    config= JSON.parse(localStorage.getItem('config'));
 
    // If config 1.0 (Simple Modify headers V1.2) , save to format 1.1
    if (config.format_version==="1.0") {
      config.format_version="1.2";
      for (let line of config.headers) line.apply_on="req";
      config.debug_mode=false;
      console.log("save new config"+JSON.stringify(config));
    }
    // If config 1.1 (Simple Modify headers V1.3 to version 1.5) , save to format 1.2	
    if (config.format_version==="1.1") {
      config.format_version="1.2";
      for (let line of config.headers) line.url_contains="";
      config.use_url_contains=false;
      console.log("save new config"+JSON.stringify(config));
didierfred's avatar
didierfred committed
    }
  }
  else {
    // else check if old config exist (Simple Modify headers V1.1)
    if (localStorage.getItem('targetPage')&& localStorage.getItem('modifyTable')) {
      console.log("Load old config");
      let headers = [];
      let modifyTable=JSON.parse(localStorage.getItem("modifyTable"));
      for (const to_modify of modifyTable) {
        headers.push({action:to_modify[0],url_contains:"",header_name:to_modify[1],header_value:to_modify[2],comment:"",apply_on:"req",status:to_modify[3]});
      }
      config = {format_version:"1.1",target_page:localStorage.getItem('targetPage'),headers:headers,debug_mode:false};
    }
    //else no config exists, create a default one
    else {
      console.log("Load default config");
      let headers = [];
      headers.push({url_contains:"",action:"add",header_name:"test-header-name",header_value:"test-header-value",comment:"test",apply_on:"req",status:"on"});
      config = {format_version:"1.1",target_page:"https://httpbin.org/*",headers:headers,debug_mode:false};
    }
didierfred's avatar
didierfred committed
  }
  storeInBrowserStorage({config:JSON.stringify(config)});
  started=localStorage.getItem('started');
  if (started!==undefined) storeInBrowserStorage({started:started});
}	


didierfred's avatar
didierfred committed

function loadFromBrowserStorage(item,callback_function) { 
  chrome.storage.local.get(item, callback_function);
didierfred's avatar
didierfred committed
}
didierfred's avatar
didierfred committed

function storeInBrowserStorage(item,callback_function)  {
  chrome.storage.local.set(item,callback_function);
}
didierfred's avatar
didierfred committed

didierfred's avatar
didierfred committed

didierfred's avatar
didierfred committed
* Standard function to log messages
didierfred's avatar
didierfred committed
function log(message) {
  console.log(new Date() + " SimpleModifyHeader : " + message);
didierfred's avatar
didierfred committed
* Rewrite the request header (add , modify or delete)
didierfred's avatar
didierfred committed
function rewriteRequestHeader(e) {
  if (config.debug_mode) log("Start modify request headers for url " + e.url);
didierfred's avatar
didierfred committed
  for (let to_modify of config.headers) {
didierfred's avatar
didierfred committed
    if ((to_modify.status==="on")&&(to_modify.apply_on==="req")&& (!config.use_url_contains || (config.use_url_contains && e.url.includes(to_modify.url_contains)))) {
didierfred's avatar
didierfred committed
      if (to_modify.action==="add"){
        let new_header = {"name" :to_modify.header_name,"value":to_modify.header_value};
        e.requestHeaders.push(new_header);
didierfred's avatar
didierfred committed
		if (config.debug_mode) log("Add request header : name=" + to_modify.header_name +
		  ",value=" + to_modify.header_value + " for url " + e.url);
didierfred's avatar
didierfred committed
      }
      else if (to_modify.action==="modify") {
	for (let header of e.requestHeaders) {
          if (header.name.toLowerCase() === to_modify.header_name.toLowerCase()) {
didierfred's avatar
didierfred committed
            if (config.debug_mode) log("Modify request header :  name= " + to_modify.header_name +
			  ",old value=" + header.value +  ",new value=" + to_modify.header_value +
			  " for url " + e.url);
didierfred's avatar
didierfred committed
            header.value = to_modify.header_value;
          }
        }
      }
      else if (to_modify.action==="delete") {
        let index = -1;
        for (let i=0; i < e.requestHeaders.length; i++) {
          if (e.requestHeaders[i].name.toLowerCase() === to_modify.header_name.toLowerCase())  index=i;
        }
	if (index!==-1) {
          e.requestHeaders.splice(index,1);
didierfred's avatar
didierfred committed
          if (config.debug_mode) log("Delete request header :  name=" + to_modify.header_name.toLowerCase() +
		    " for url " + e.url);
didierfred's avatar
didierfred committed
        }
      }
    }
  }
  if (config.debug_mode) log("End modify request headers for url " + e.url);
didierfred's avatar
didierfred committed
  return {requestHeaders: e.requestHeaders};
}


didierfred's avatar
didierfred committed
/*
* Rewrite the response header (add , modify or delete)
*
*/
didierfred's avatar
didierfred committed
function rewriteResponseHeader(e) {
  if (config.debug_mode) log("Start modify response headers for url " + e.url);
didierfred's avatar
didierfred committed
  for (let to_modify of config.headers) {
didierfred's avatar
didierfred committed
    if ((to_modify.status==="on")&&(to_modify.apply_on==="res")&& (!config.use_url_contains || (config.use_url_contains && e.url.includes(to_modify.url_contains)))) {
didierfred's avatar
didierfred committed
      if (to_modify.action==="add") {
        let new_header = {"name" :to_modify.header_name,"value":to_modify.header_value};
	e.responseHeaders.push(new_header);
didierfred's avatar
didierfred committed
	if (config.debug_mode) log("Add response header : name=" + to_modify.header_name
							+ ",value=" + to_modify.header_value + " for url " + e.url);
didierfred's avatar
didierfred committed
      }
      else if (to_modify.action==="modify") {
        for (let header of e.responseHeaders) {
          if (header.name.toLowerCase() === to_modify.header_name.toLowerCase()) {
didierfred's avatar
didierfred committed
            if (config.debug_mode) log("Modify response header :  name= " + to_modify.header_name + ",old value="
										+ header.value +  ",new value=" + to_modify.header_value  + " for url " + e.url);
didierfred's avatar
didierfred committed
            header.value = to_modify.header_value;
          }
        }
      }
      else if (to_modify.action==="delete") {
        let index = -1;
        for (let i=0; i < e.responseHeaders.length; i++) {
          if (e.responseHeaders[i].name.toLowerCase() === to_modify.header_name.toLowerCase())  index=i;
didierfred's avatar
didierfred committed
	}
didierfred's avatar
didierfred committed
        if (index!==-1) {
          e.responseHeaders.splice(index,1);
didierfred's avatar
didierfred committed
          if (config.debug_mode) log("Delete response header :  name=" + to_modify.header_name.toLowerCase()
									+ " for url " + e.url);		
didierfred's avatar
didierfred committed
        }
      }
    }
  }
  if (config.debug_mode) log("End modify response headers for url " + e.url);
didierfred's avatar
didierfred committed
  return {responseHeaders: e.responseHeaders};
}

didierfred's avatar
didierfred committed

/*
* Listen for message form config.js
didierfred's avatar
didierfred committed
* if message is reload : reload the configuration
* if message is on : start the modify header
* if message is off : stop the modify header
*
**/
didierfred's avatar
didierfred committed
function notify(message) {
  if (message==="reload") {
    if (config.debug_mode) log("Reload configuration");
    loadFromBrowserStorage(['config'],function (result) {
      config=JSON.parse(result.config);
      if (started==="on") {
        removeListener();
        addListener();
      }
    });
didierfred's avatar
didierfred committed
  }
  else if (message==="off") {
    removeListener();
    chrome.browserAction.setIcon({ path: "icons/modify-32.png"});
didierfred's avatar
didierfred committed
    started="off";
    if (config.debug_mode) log("Stop modifying headers");
  }
  else if (message==="on") {
    addListener();
    chrome.browserAction.setIcon({ path: "icons/modify-green-32.png"});
didierfred's avatar
didierfred committed
    started="on";
    if (config.debug_mode) log("Start modifying headers");
  }
}
didierfred's avatar
didierfred committed

/*
* Add rewriteRequestHeader as a listener to onBeforeSendHeaders, only for the target pages.
* Add rewriteResponseHeader as a listener to onHeadersReceived, only for the target pages.
* Make it "blocking" so we can modify the headers.
didierfred's avatar
didierfred committed
*/
didierfred's avatar
didierfred committed
function addListener() {
  let target = config.target_page;
  if ((target==="*")||(target==="")||(target===" ")) target="<all_urls>";
  chrome.webRequest.onBeforeSendHeaders.addListener(rewriteRequestHeader,
                                          {urls: target.split(";")},
didierfred's avatar
didierfred committed
                                          ["blocking", "requestHeaders"]);
didierfred's avatar
didierfred committed

  chrome.webRequest.onHeadersReceived.addListener(rewriteResponseHeader,
                                          {urls: target.split(";")},
didierfred's avatar
didierfred committed
                                          ["blocking", "responseHeaders"]);
didierfred's avatar
didierfred committed
}
didierfred's avatar
didierfred committed
* Remove the two listener
didierfred's avatar
didierfred committed
*
*/
didierfred's avatar
didierfred committed
function removeListener() {
  chrome.webRequest.onBeforeSendHeaders.removeListener(rewriteRequestHeader);
  chrome.webRequest.onHeadersReceived.removeListener(rewriteResponseHeader);
didierfred's avatar
didierfred committed
}
didierfred's avatar
didierfred committed