Создайте аддон Firefox для просмотра и изменения запросов и ответов XHR - PullRequest
6 голосов
/ 26 января 2010

Обновление: я предполагаю, что субъект дал неверное представление о том, что я ищу существующее дополнение. Это нестандартная проблема, и я НЕ хочу существующего решения.
Я хочу, чтобы написать (или, более уместно, изменить и существующие) Аддон.

Вот мое требование:

  • Я хочу, чтобы мой аддон работал только для определенного сайта
  • Данные на страницах кодируются с использованием 2-х стороннего хэша
  • Большая часть информации загружается запросами XHR, а иногда отображаются в виде анимированных пузырей и т. д.
  • Текущая версия моего аддона анализирует страницу через XPath выражения, декодирует данные и заменяет их

  • Проблема возникает с теми всплывающими окнами, которые отображаются при наведении курсора мыши

  • Таким образом, я понял, что было бы неплохо создать XHR мост, который может слушать все данные и декодировать / кодировать на лету
  • После пары поисков я наткнулся на nsITraceableInterface [1] [2] [3]

Просто хотел узнать, нахожусь ли я на правильном пути. Если "да", то любезно предоставить любые дополнительные указатели и предложения, которые могут быть уместными; и если "нет", тогда .. ну, пожалуйста, помогите с правильными указателями :)

Спасибо
Bipin.

[1]. https://developer.mozilla.org/en/NsITraceableChannel
[2]. http://www.softwareishard.com/blog/firebug/nsitraceablechannel-intercept-http-traffic/
[3]. http://www.ashita.org/howto-xhr-listening-by-a-firefox-addon/

Ответы [ 3 ]

8 голосов
/ 29 января 2010

nsITraceableChannel - действительно путь сюда. сообщения в блоге Яна Одварко (softwareishard.com) и меня (ashita.org) показывают, как это сделать. Вы также можете захотеть увидеть http://www.ashita.org/implementing-an-xpcom-firefox-interface-and-creating-observers/,, однако в действительности нет необходимости делать это в компоненте XPCOM.

Шаги в основном:

  1. Создание прототипа Object, реализующего nsITraceableChannel; и создать наблюдателя для прослушивания http-on-modify-request и http-on-exam-response
  2. регистрация наблюдателя
  3. наблюдатель, прослушивающий два типа запросов, добавляет наш объект nsITraceableChannel в цепочку слушателей и следит за тем, чтобы наш nsITC знал, кто следующий в цепочке
  4. Объект nsITC предоставляет три обратных вызова, каждый из которых будет вызываться на соответствующем этапе: onStartRequest, onDataAvailable и onStopRequest
  5. в каждом из описанных выше обратных вызовов наш объект nsITC должен передавать данные следующему элементу в цепочке

Ниже приведен действительный код написанного мной надстройки для конкретного сайта, который, как я могу сказать, очень похож на ваш.

function TracingListener() {
    //this.receivedData = [];
}

TracingListener.prototype =
{
    originalListener: null,
    receivedData: null,   // array for incoming data.

    onDataAvailable: function(request, context, inputStream, offset, count)
    {
        var binaryInputStream = CCIN("@mozilla.org/binaryinputstream;1", "nsIBinaryInputStream");
        var storageStream = CCIN("@mozilla.org/storagestream;1", "nsIStorageStream");
        binaryInputStream.setInputStream(inputStream);
        storageStream.init(8192, count, null);

        var binaryOutputStream = CCIN("@mozilla.org/binaryoutputstream;1",
                "nsIBinaryOutputStream");

        binaryOutputStream.setOutputStream(storageStream.getOutputStream(0));

        // Copy received data as they come.
        var data = binaryInputStream.readBytes(count);
        //var data = inputStream.readBytes(count);

        this.receivedData.push(data);

        binaryOutputStream.writeBytes(data, count);
        this.originalListener.onDataAvailable(request, context,storageStream.newInputStream(0), offset, count);
    },

    onStartRequest: function(request, context) {
        this.receivedData = [];
        this.originalListener.onStartRequest(request, context);
    },

    onStopRequest: function(request, context, statusCode)
    {
        try 
        {
            request.QueryInterface(Ci.nsIHttpChannel);

            if (request.originalURI && piratequesting.baseURL == request.originalURI.prePath && request.originalURI.path.indexOf("/index.php?ajax=") == 0) 
            {

                var data = null;
                if (request.requestMethod.toLowerCase() == "post") 
                {
                    var postText = this.readPostTextFromRequest(request, context);
                    if (postText) 
                        data = ((String)(postText)).parseQuery();

                }
                var date = Date.parse(request.getResponseHeader("Date"));
                var responseSource = this.receivedData.join('');

                //fix leading spaces bug
                responseSource = responseSource.replace(/^\s+(\S[\s\S]+)/, "$1");

                piratequesting.ProcessRawResponse(request.originalURI.spec, responseSource, date, data);
            }
        } 
        catch (e) 
        {
            dumpError(e);
        }
        this.originalListener.onStopRequest(request, context, statusCode);
    },

    QueryInterface: function (aIID) {
        if (aIID.equals(Ci.nsIStreamListener) ||
            aIID.equals(Ci.nsISupports)) {
            return this;
        }
        throw Components.results.NS_NOINTERFACE;
    },
    readPostTextFromRequest : function(request, context) {
        try
        {
            var is = request.QueryInterface(Ci.nsIUploadChannel).uploadStream;
            if (is)
            {
                var ss = is.QueryInterface(Ci.nsISeekableStream);
                var prevOffset;
                if (ss)
                {
                    prevOffset = ss.tell();
                    ss.seek(Ci.nsISeekableStream.NS_SEEK_SET, 0);
                }

                // Read data from the stream..
                var charset = "UTF-8";
                var text = this.readFromStream(is, charset, true);

                // Seek locks the file so, seek to the beginning only if necko hasn't read it yet,
                // since necko doesn't seek to 0 before reading (at lest not till 459384 is fixed).
                if (ss && prevOffset == 0) 
                    ss.seek(Ci.nsISeekableStream.NS_SEEK_SET, 0);

                return text;
            }
            else {
                dump("Failed to Query Interface for upload stream.\n");
            }
        }
        catch(exc)
        {
            dumpError(exc);
        }

        return null;
    },
    readFromStream : function(stream, charset, noClose) {

        var sis = CCSV("@mozilla.org/binaryinputstream;1", "nsIBinaryInputStream");
        sis.setInputStream(stream);

        var segments = [];
        for (var count = stream.available(); count; count = stream.available())
            segments.push(sis.readBytes(count));

        if (!noClose)
            sis.close();

        var text = segments.join("");
        return text;
    }

}


hRO = {

    observe: function(request, aTopic, aData){
        try {
            if (typeof Cc == "undefined") {
                var Cc = Components.classes;
            }
            if (typeof Ci == "undefined") {
                var Ci = Components.interfaces;
            }
            if (aTopic == "http-on-examine-response") {
                request.QueryInterface(Ci.nsIHttpChannel);

                if (request.originalURI && piratequesting.baseURL == request.originalURI.prePath && request.originalURI.path.indexOf("/index.php?ajax=") == 0) {
                    var newListener = new TracingListener();
                    request.QueryInterface(Ci.nsITraceableChannel);
                    newListener.originalListener = request.setNewListener(newListener);
                }
            } 
        } catch (e) {
            dump("\nhRO error: \n\tMessage: " + e.message + "\n\tFile: " + e.fileName + "  line: " + e.lineNumber + "\n");
        }
    },

    QueryInterface: function(aIID){
        if (typeof Cc == "undefined") {
            var Cc = Components.classes;
        }
        if (typeof Ci == "undefined") {
            var Ci = Components.interfaces;
        }
        if (aIID.equals(Ci.nsIObserver) ||
        aIID.equals(Ci.nsISupports)) {
            return this;
        }

        throw Components.results.NS_NOINTERFACE;

    },
};


var observerService = Cc["@mozilla.org/observer-service;1"]
    .getService(Ci.nsIObserverService);

observerService.addObserver(hRO,
    "http-on-examine-response", false);

В приведенном выше коде originalListener - это слушатель, которого мы вставляем в цепочку. Крайне важно, чтобы вы сохранили эту информацию при создании отслеживающего слушателя и передавали данные во всех трех обратных вызовах. В противном случае ничего не получится (страницы даже не загрузятся. Сам Firefox является последним в цепочке).

Примечание: в приведенном выше коде есть некоторые функции, которые являются частью надстройки для пиратских запросов, например: parseQuery() и dumpError()

0 голосов
/ 27 января 2010

Вы можете попробовать создать скрипт greasemonkey и перезаписать XMLHttpRequest.
Код будет выглядеть примерно так:

function request () {
};
request.prototype.open = function (type, path, block) {
 GM_xmlhttpRequest({
  method: type,
  url: path,
  onload: function (response) {
   // some code here
  }
 });
};
unsafeWindow.XMLHttpRequest = request;

Также обратите внимание, что вы можете превратить скрипт GM в аддон для Firefox

0 голосов
/ 26 января 2010
...