Skip to content
Snippets Groups Projects
background.js 9.63 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
const isChrome = (navigator.userAgent.indexOf("chrome") !== -1);
didierfred's avatar
didierfred committed

didierfred's avatar
didierfred committed
loadFromBrowserStorage(['config', 'started'], function (result) {
didierfred's avatar
didierfred committed

  // if old storage method
didierfred's avatar
didierfred committed
  if (result.config === undefined) loadConfigurationFromLocalStorage();
  else {
didierfred's avatar
didierfred committed
    started = result.started;
didierfred's avatar
didierfred committed
    config = JSON.parse(result.config);
  }
didierfred's avatar
didierfred committed

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


didierfred's avatar
didierfred committed
function loadConfigurationFromLocalStorage() {
  // if configuration exist 
  if (localStorage.getItem('config')) {
    console.log("Load standard config");
didierfred's avatar
didierfred committed
    config = JSON.parse(localStorage.getItem('config'));

    // If config 1.0 (Simple Modify headers V1.2) , save to format 1.1
didierfred's avatar
didierfred committed
    if (config.format_version === "1.0") {
      config.format_version = "1.2";
      for (let line of config.headers) {
didierfred's avatar
didierfred committed
        line.apply_on = "req";
        line.url_contains = "";
didierfred's avatar
didierfred committed
      config.debug_mode = false;
      config.use_url_contains = 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	
didierfred's avatar
didierfred committed
    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)
didierfred's avatar
didierfred committed
    if (localStorage.getItem('targetPage') && localStorage.getItem('modifyTable')) {
      console.log("Load old config");
      let headers = [];
didierfred's avatar
didierfred committed
      let modifyTable = JSON.parse(localStorage.getItem("modifyTable"));
      for (const to_modify of modifyTable) {
didierfred's avatar
didierfred committed
        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] });
didierfred's avatar
didierfred committed
      config = { format_version: "1.1", target_page: localStorage.getItem('targetPage'), headers: headers, debug_mode: false, use_url_contains: false };
    }
    //else no config exists, create a default one
    else {
      console.log("Load default config");
      let headers = [];
didierfred's avatar
didierfred committed
      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, use_url_contains: false };
didierfred's avatar
didierfred committed
  }
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

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

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)))) {
      if (to_modify.action === "add") {
        let new_header = { "name": to_modify.header_name, "value": to_modify.header_value };
didierfred's avatar
didierfred committed
        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
      }
didierfred's avatar
didierfred committed
      else if (to_modify.action === "modify") {
        for (let header of e.requestHeaders) {
didierfred's avatar
didierfred committed
          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 +
didierfred's avatar
didierfred committed
              ",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;
          }
        }
      }
didierfred's avatar
didierfred committed
      else if (to_modify.action === "delete") {
didierfred's avatar
didierfred committed
        let index = -1;
didierfred's avatar
didierfred committed
        for (let i = 0; i < e.requestHeaders.length; i++) {
          if (e.requestHeaders[i].name.toLowerCase() === to_modify.header_name.toLowerCase()) index = i;
didierfred's avatar
didierfred committed
        }
didierfred's avatar
didierfred committed
        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() +
didierfred's avatar
didierfred committed
            " 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
}


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)))) {
      if (to_modify.action === "add") {
        let new_header = { "name": to_modify.header_name, "value": to_modify.header_value };
        e.responseHeaders.push(new_header);
        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
      }
didierfred's avatar
didierfred committed
      else if (to_modify.action === "modify") {
didierfred's avatar
didierfred committed
        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="
didierfred's avatar
didierfred committed
              + header.value + ",new value=" + to_modify.header_value + " for url " + e.url);
didierfred's avatar
didierfred committed
            header.value = to_modify.header_value;
          }
        }
      }
didierfred's avatar
didierfred committed
      else if (to_modify.action === "delete") {
didierfred's avatar
didierfred committed
        let index = -1;
didierfred's avatar
didierfred committed
        for (let i = 0; i < e.responseHeaders.length; i++) {
          if (e.responseHeaders[i].name.toLowerCase() === to_modify.header_name.toLowerCase()) index = i;
        }
        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()
didierfred's avatar
didierfred committed
            + " 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) {
didierfred's avatar
didierfred committed
  if (message === "reload") {
didierfred's avatar
didierfred committed
    if (config.debug_mode) log("Reload configuration");
didierfred's avatar
didierfred committed
    loadFromBrowserStorage(['config'], function (result) {
      config = JSON.parse(result.config);
      if (started === "on") {
        removeListener();
        addListener();
      }
    });
didierfred's avatar
didierfred committed
  }
didierfred's avatar
didierfred committed
  else if (message === "off") {
didierfred's avatar
didierfred committed
    removeListener();
didierfred's avatar
didierfred committed
    chrome.browserAction.setIcon({ path: "icons/modify-32.png" });
    started = "off";
didierfred's avatar
didierfred committed
    if (config.debug_mode) log("Stop modifying headers");
  }
didierfred's avatar
didierfred committed
  else if (message === "on") {
didierfred's avatar
didierfred committed
    addListener();
didierfred's avatar
didierfred committed
    chrome.browserAction.setIcon({ path: "icons/modify-green-32.png" });
    started = "on";
didierfred's avatar
didierfred committed
    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;
didierfred's avatar
didierfred committed
  if ((target === "*") || (target === "") || (target === " ")) target = "<all_urls>";

  // need to had "extraHeaders" option for chrome https://developer.chrome.com/extensions/webRequest
didierfred's avatar
didierfred committed
  if (isChrome) {
    chrome.webRequest.onBeforeSendHeaders.addListener(rewriteRequestHeader,
      { urls: target.split(";") },
      ["blocking", "requestHeaders", "extraHeaders"]);

    chrome.webRequest.onHeadersReceived.addListener(rewriteResponseHeader,
      { urls: target.split(";") },
      ["blocking", "responseHeaders", "extraHeaders"]);
didierfred's avatar
didierfred committed

  else {
    chrome.webRequest.onBeforeSendHeaders.addListener(rewriteRequestHeader,
      { urls: target.split(";") },
      ["blocking", "requestHeaders"]);
    chrome.webRequest.onHeadersReceived.addListener(rewriteResponseHeader,
      { urls: target.split(";") },
      ["blocking", "responseHeaders"]);
didierfred's avatar
didierfred committed

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