如何在引导的Firefox扩展中实现XPCOM组件(nsIContentPolicy)

时间:2013-08-21 10:11:53

标签: firefox-addon firefox-addon-sdk xpcom

我有一个Firefox的自举扩展程序。 现在我想实现nsIContentPolicy XPCOM组件。 我写了一个组件模块代码。 现在我想注册这个组件。 我想注册组件的原因是我想将我的组件添加到nsICategoryManager.addCategoryEntry并使用“content-policy”类别。

var {Cc, Ci, Cu} = require("chrome");

Cu.import("resource://gre/modules/XPCOMUtils.jsm");

//console.error("Running interceptor");

function Interceptor() 
}

Interceptor.prototype = {

    classDescription: "DeferredTo HTTP requests Interceptor",
    classID: "{B5B3D9A0-08FC-11E3-8253-5EF06188709B}",
    contractID: "@deferredto.com/Interceptor;1",
    QueryInterface: XPCOMUtils.generateQI([Ci.nsIContentPolicy]),

    shouldLoad : function dt_shouldLoad(aContentType, aContentLocation, aRequestOrigin, aContext, aMimeTypeGuess, aExtra) {
        console.log("dt_shouldLoad");

        if (contentLocation.scheme != "http" && contentLocation.scheme != "https")
            return Ci.nsIContentPolicy.ACCEPT;

        let result = Ci.nsIContentPolicy.ACCEPT;

        // we should check for TYPE_SUBDOCUMENT as well if we want frames.
        if ((Ci.nsIContentPolicy.TYPE_DOCUMENT == aContentType) &&
            SOME_REGULAR_EXPRESSION.test(aContentLocation.spec)) {
            // do stuff here, possibly changing result.
        }
        return result;
    },

    shouldProcess: function ILO_shouldProcess() Ci.nsIContentPolicy.ACCEPT,

    _xpcom_categories: [
        { category: "content-policy", service: true }
    ],
    classInfo: XPCOMUtils.generateCI(
    {classID: Components.ID("{B5B3D9A0-08FC-11E3-8253-5EF06188709B}"),
     contractID: "@deferredto.com/Interceptor;1",
     classDescription: "Interceptor implements nsIContentPolicy to block images that are not yet at screen @DeferredTo",
     interfaces: [
                  Ci.nsIContentPolicy,
                  ],
     flags: Ci.nsIClassInfo.SINGLETON})
}

var components = [Interceptor];

var NSGetFactory = XPCOMUtils.generateNSGetFactory([Interceptor]);

问题:

  • 是否可以从bootstrapped extension注册组件?
  • 是否可以从无重启扩展注册组件?
  • 是否可以使用nsICategoryManager.addCategoryEntry“content-policy” 零件?
  • 如何在bootstrapped扩展中注册组件或以某种方式添加 新的“内容政策”类别条目?

我已添加到harness-options.js

"requirements": {
"sdk/page-mod": "sdk/page-mod",
"sdk/self": "sdk/self",
"chrome": "chrome"},

这就是我尝试导入模块的方式:

var {Cc, Ci, Cu} = require("chrome");
Cu.import("resource://deferredto/lib/interceptor.js");

我尝试了许多路径)))但没有一个可行。 chrome.manifest文件中的资源条目不允许用于自举扩展。组件模块文件的路径是:     resources/deferredto/lib/interceptor.js

2 个答案:

答案 0 :(得分:3)

Adblock Plus,无需重启,但在运行时不使用SDK registers an nsIContentPolicy implementation,就像您的SDK一样。可能有一些SDK附加组件在运行时注册组件,但我不知道任何开源的组件,我建议您从头脑中查看。

有关ABP实施的一些要点以及要使其与SDK一起使用的更改内容:

  • 类别管理员可通过Cc["@mozilla.org/categorymanager;1"].getService(Ci.nsICategoryManager)
  • 获取
  • 组件注册商应该可以通过components模块中的chrome,然后components.manager.getService(Ci.nsIComponentRegistrar)来获取。
  • 作为Adblock Plus,您必须自行取消注册您的组件on unload
  • 由于Bug 753687导致您无法同步取消注册您的组件和类别条目,因此卸载部分也很不方便。因此,Adblock Plus使用Util.runAsync进行异步,只需将runnable(事件,如果您愿意)调度到主线程。我不认为你可以在这里使用任何SDK的东西,因为SDK会在任何异步代码有机会运行之前清理,所以你需要自己使用低级别的XPCOM可运行(或计时器)。
  • 您的代码将在运行时注册您的组件。你不会碰harness-options或类似的东西。

(我自己也实现了generic component register功能,但这又不是SDK代码,需要适应在SDK中运行,就像ABP一样。它也非常类似于ABP之一。)

答案 1 :(得分:3)

现在我的nsIContentPolicy基于sdk的组件看起来像这样。档案interceptor.js

'use strict';

var { Class } = require('sdk/core/heritage');
var xpcom = require('sdk/platform/xpcom');
var { Cc, Ci, Cu, Cm } = require('chrome');
var categoryManager = Cc["@mozilla.org/categorymanager;1"]
                      .getService(Ci.nsICategoryManager);


// nsIDOMNode
const TYPE_DOCUMENT_NODE        = Ci.nsIDOMNode.DOCUMENT_NODE;


/// Interceptor


var contractId = "@deferredto.com/Interceptor;1";

var Interceptor = Class({
  extends:  xpcom.Unknown,
  interfaces: [ 'nsIContentPolicy' ],
  get wrappedJSObject() this,

  shouldLoad : function dt_shouldLoad(contentType, contentLocation, requestOrigin, context, mimeTypeGuess, extra) {

        let result = Ci.nsIContentPolicy.ACCEPT;

        return result;
    },

    shouldProcess: function () Ci.nsIContentPolicy.ACCEPT
});

var factory = xpcom.Factory({
  contract: contractId,
  Component: Interceptor,
  unregister: false // see https://bugzilla.mozilla.org/show_bug.cgi?id=753687
});

/// unload 
var unload = require("sdk/system/unload");

unload.when(function() {
  function trueUnregister() {
    categoryManager.deleteCategoryEntry("content-policy", contractId, false);
    try {
      console.log("xpcom.isRegistered(factory)="  + xpcom.isRegistered(factory));
      console.log("trueUnregister");
      xpcom.unregister(factory);
      console.log("xpcom.isRegistered(factory)="  + xpcom.isRegistered(factory));
    } catch (ex) {
        Cu.reportError(ex);
    }      
  }
  if ("dispatch" in Cu) {
    console.log('"dispatch" in Cu');
    Cu.dispatch(trueUnregister, trueUnregister);
  } else {
    console.log('"dispatch" not! in Cu');
    Cu.import("resource://gre/modules/Services.jsm");
    Services.tm.mainThread.dispatch(trueUnregister, 0);
  }
});


//xpcom.register(factory);

var interceptor = Cc[contractId].createInstance(Ci.nsIContentPolicy);

categoryManager.deleteCategoryEntry("content-policy", contractId, false);
categoryManager.addCategoryEntry("content-policy", contractId, contractId, false, true);

你可以从sdk中使用它:

var interceptor = require("./interceptor");