How to log fetched network resources in JavaScript?

后端 未结 2 544
别跟我提以往
别跟我提以往 2021-02-14 01:00

Is there a way to access the list of resources that the browser requested (the ones found in this Chrome inspector\'s network panel)?

相关标签:
2条回答
  • 2021-02-14 01:37

    Big shoutout to @Elliot B.

    I essentially used what he did but I wanted events to trigger in the content script rather than listeners triggering in the background. For whatever reason, I was unable to connect to the port from the background script so this is what I came up with.

    PS: you need jquery.js in the extension folder to make this work.

    manifest.json

    {
      "manifest_version": 2,
    
      "name": "MNC",
      "version": "0.0.1",
      "description": "Monitor Network Comms",
      "permissions":["webRequest","*://*/"],
      "content_scripts": [{
      "matches": ["<all_urls>"],
      "run_at": "document_start",
      "js": ["content.js",
            "jquery.js"]
      }],
      "background": {
        "scripts": ["background.js"]
      }
    }
    

    background.js

    var aNetworkLog = [];
    
    chrome.webRequest.onResponseStarted.addListener(
      function(oCompleted) {
        var sCompleted = JSON.stringify(oCompleted);
        aNetworkLog.push(sCompleted);
      },{urls: ["https://*/*"]}
    );
    
    chrome.extension.onConnect.addListener(function (port) {
      chrome.webRequest.onResponseStarted.addListener(
        function(){
          port.postMessage({networkLog:JSON.stringify(aNetworkLog)});
        },{urls: ["https://*/*"]}
      );
      port.onMessage.addListener(function (message) {
        if (message.disconnect==true) {
          port.disconnect();
        }
      });
    });
    

    content.js

    div = $('<div id="outputDiv" style="float:left;max-width:fit-content;position:fixed;display:none;"></div>').appendTo(document.body);
    var port = chrome.extension.connect({name:'networkLogging'});
    port.onMessage.addListener(function (message) {
      if (message.networkLog) {
        div[0].innerHTML = message.networkLog;
      }
    });
    observer = new WebKitMutationObserver(function(mutation,observer){
      JSON.parse(mutation[0]['target'].innerHTML).forEach(function(item){
        JSON.parse(item);
      })
    });
    observer.observe(div[0],{childList:true});
    

    This is definitely not the most efficient way of doing things but it works for me. Thought that I would add it in here just in case someone is needing it.

    0 讨论(0)
  • 2021-02-14 01:48

    You can do this, but you will need to use Chrome extensions.

    Chrome extensions have a lot of sandbox-style security. Communication between the Chrome extension and the web page is a multi-step process. Here's the most concise explanation I can offer with a full working example at the end:

    1. A Chrome extension has full access to the chrome.* APIs, but a Chrome extension cannot communicate directly with the web page JS nor can the web page JS communicate directly with the Chrome extension.

    2. To bridge the gap between the Chrome extension and the web page, you need to use a content script . A content script is essentially JavaScript that is injected at the window scope of the targeted web page. The content script cannot invoke functions nor access variables that are created by the web page JS, but they do share access to the same DOM and therefore events as well.

    3. Because directly accessing variables and invoking functions is not allowed, the only way the web page and the content script can communicate is through firing custom events.

    For example, if I wanted to pass a message from the Chrome extension to the page I could do this:

    content_script.js

    document.getElementById("theButton").addEventListener("click", function() {
        window.postMessage({ type: "TO_PAGE", text: "Hello from the extension!" }, "*");
    }, false);
    

    web_page.js

    window.addEventListener("message", function(event) {
        // We only accept messages from ourselves
        if (event.source != window)
          return;
    
        if (event.data.type && (event.data.type == "TO_PAGE")) {
          alert("Received from the content script: " + event.data.text);
        }
    }, false);
    

    `4. Now that you can send a message from the content script to the web page, you now need the Chrome extension gather up all the network info you want. You can accomplish this through a couple different modules, but the most simple option is the webRequest module (see background.js below).

    `5. Use message passing to relay the info on the web requests to the content script and then on to the web page JavaScript.

    Visually, you can think of it like this:

    Chrome extensions and content scripts

    Full working example:

    The first three files comprise your Google Chrome Extension and the last file is the HTML file you should upload to http:// web space somewhere.

    icon.png

    Use any 16x16 PNG file.

    manifest.json

    {
      "name": "webRequest Logging",
      "description": "Displays the network log on the web page",
      "version": "0.1",
      "permissions": [
        "tabs",
        "debugger",
        "webRequest",
        "http://*/*"
      ],
      "background": {
        "scripts": ["background.js"]
      },
      "browser_action": {
        "default_icon": "icon.png",
        "default_title": "webRequest Logging"
      },
       "content_scripts": [
        {
          "matches": ["http://*/*"],
          "js": ["content_script.js"]
        }
      ],
      "manifest_version": 2
    }
    

    background.js

    var aNetworkLog = [];
    
    chrome.webRequest.onCompleted.addListener(function(oCompleted) {
                var sCompleted = JSON.stringify(oCompleted);
                aNetworkLog.push(sCompleted);
            }
            ,{urls: ["http://*/*"]}
     );
    
    chrome.extension.onConnect.addListener(function (port) {
        port.onMessage.addListener(function (message) {
            if (message.action == "getNetworkLog") {
                port.postMessage(aNetworkLog);
            }
        });
    });
    

    content_script.js

    var port = chrome.extension.connect({name:'test'});
    
    document.getElementById("theButton").addEventListener("click", function() {
    
        port.postMessage({action:"getNetworkLog"});
    
    }, false);
    
    port.onMessage.addListener(function(msg) {
      document.getElementById('outputDiv').innerHTML = JSON.stringify(msg);
    });
    

    And use the following for the web page (named whatever you want):

    <!doctype html>
    <html>
    <head>
        <title>webRequest Log</title>
    </head>
    <body>
        <input type="button" value="Retrieve webRequest Log" id="theButton">
        <div id="outputDiv"></div>
    </head>
    </html>
    
    0 讨论(0)
自定义标题
段落格式
字体
字号
代码语言
提交回复
热议问题