(function() { 'use strict'; var {Cc, Ci} = require('chrome'), main = require('wappalyzer'), w = main.wappalyzer, tabCache = {}, headersCache = {}, categoryNames = {}, initTab, Panel, panel, Widget, widget, UrlBar, urlBar; exports.main = function(options, callbacks) { w.log('main: ' + options.loadReason); w.init(); }; exports.onUnload = function(reason) { w.log('unload: ' + reason); if ( urlBar ) { urlBar.destroy(); } if ( widget ) { widget.destroy(); } if ( panel ) { panel.destroy(); } }; initTab = function(tab) { w.log('initTab'); tabCache[tab.id] = { count: 0, appsDetected: [] }; tab.on('ready', function(tab) { var worker = tab.attach({ contentScriptFile: require('sdk/self').data.url('js/tab.js') }); worker.port.on('analyze', function(message) { var url = message.url.replace(/#.*$/, ''); if ( typeof headersCache[url] !== 'undefined' ) { message.analyze.headers = headersCache[url]; } w.analyze(message.hostname, url, message.analyze); }); worker.port.on('log', function(message) { w.log('[ tab.js ] ' + message); }); }); }; require('sdk/tabs') .on('open', initTab) .on('close', function(tab) { tabCache[tab.id] = null; }) .on('activate', function(tab) { w.driver.displayApps(); }); Panel = function() { var self = this; this.panel = require('sdk/panel').Panel({ width: 250, height: 50, contentURL: require('sdk/self').data.url('panel.html'), contentScriptFile: require('sdk/self').data.url('js/panel.js'), position: { right: 30, top: 30 } }); this.panel.port.on('resize', function(height) { self.panel.height = height; }); this.panel.port.on('goToUrl', function(url) { self.panel.hide(); w.driver.goToURL({ url: w.config.websiteURL + url, medium: 'panel' }); }); }; Panel.prototype.get = function() { return this.panel; }; Panel.prototype.destroy = function() { this.panel.destroy(); }; Widget = function(panel) { this.widget = require('sdk/widget').Widget({ id: 'wappalyzer', label: 'Wappalyzer', contentURL: require('sdk/self').data.url('images/icon32.png'), panel: panel.get() }); }; Widget.prototype.get = function() { return this.widget; }; Widget.prototype.destroy = function() { this.widget.destroy(); }; UrlBar = function(panel) { var self = this; this.panel = panel; this.onClick = function() { self.panel.get().show(); } this.document = Cc['@mozilla.org/appshell/window-mediator;1'].getService(Ci.nsIWindowMediator) .getMostRecentWindow('navigator:browser').document; this.urlBar = this.document.createElement('hbox'); this.urlBar.setAttribute('id', 'wappalyzer-urlbar'); this.urlBar.setAttribute('style', 'cursor: pointer; margin: 0 2px;'); this.urlBar.setAttribute('tooltiptext', require('sdk/l10n').get('name')); this.urlBar.addEventListener('click', this.onClick); this.document.getElementById('urlbar-icons').appendChild(this.urlBar); }; UrlBar.prototype.get = function() { return this.urlBar; }; UrlBar.prototype.addIcon = function(appName) { var icon = this.document.createElement('image'), url = typeof appName !== 'undefined' ? 'images/icons/' + appName + '.png' : 'images/icon32.png', tooltipText = ( typeof appName !== 'undefined' ? appName + ' - ' + require('sdk/l10n').get('clickForDetails') + ' - ' : '' ) + require('sdk/l10n').get('name'); icon.setAttribute('src', require('sdk/self').data.url(url)); icon.setAttribute('class', 'wappalyzer-icon'); icon.setAttribute('width', '16'); icon.setAttribute('height', '16'); icon.setAttribute('style', 'margin: 0 1px;'); icon.setAttribute('tooltiptext', tooltipText); this.get().appendChild(icon); return this; }; UrlBar.prototype.clear = function() { var icons; do { icons = this.get().getElementsByClassName('wappalyzer-icon'); if ( icons.length ) { urlBar.get().removeChild(icons[0]); } } while ( icons.length ); return this; }; UrlBar.prototype.destroy = function() { this.urlBar.removeEventListener('click', this.onClick); this.urlBar.remove(); return this; } w.driver = { /** * Log messages to console */ log: function(args) { console.log(args.message); }, /** * Initialize */ init: function(callback) { var id, tab, version, httpRequestObserver, json = JSON.parse(require('sdk/self').data.load('apps.json')); w.log('driver.init'); panel = new Panel(); if ( require('sdk/simple-prefs').prefs.urlbar ) { urlBar = new UrlBar(panel); } else { widget = new Widget(panel); } try { var version = require('sdk/self').version; if ( !require('sdk/simple-storage').storage.version ) { w.driver.goToURL({ url: w.config.websiteURL + 'installed', medium: 'install' }); } else if ( version !== require('sdk/simple-storage').storage.version ) { w.driver.goToURL({ url: w.config.websiteURL + 'upgraded', medium: 'upgrade', background: true }); } require('sdk/simple-storage').storage.version = version; } catch(e) { } w.apps = json.apps; w.categories = json.categories; for ( id in w.categories ) { categoryNames[id] = require('sdk/l10n').get('cat' + id); } for each ( tab in require('sdk/tabs') ) { initTab(tab); } require('sdk/simple-prefs').on('urlbar', function() { panel = new Panel(); if ( !require('sdk/simple-prefs').prefs.urlbar ) { urlBar.destroy(); widget = new Widget(panel); } else { urlBar = new UrlBar(panel); widget.get().destroy(); } w.driver.displayApps(); }); httpRequestObserver = { init: function() { var observerService = Cc['@mozilla.org/observer-service;1'].getService(Ci.nsIObserverService); observerService.addObserver(this, 'http-on-examine-response', false); }, observe: function(subject, topic, data) { if ( topic == 'http-on-examine-response' ) { subject.QueryInterface(Ci.nsIHttpChannel); this.onExamineResponse(subject); } }, onExamineResponse: function (subject) { var uri = subject.URI.spec.replace(/#.*$/, ''); // Remove hash if ( headersCache.length > 50 ) { headersCache = {}; } if ( subject.contentType === 'text/html' ) { if ( typeof headersCache[uri] === 'undefined' ) { headersCache[uri] = {}; } subject.visitResponseHeaders(function(header, value) { headersCache[uri][header.toLowerCase()] = value; }); } } }; httpRequestObserver.init(); w.driver.displayApps(); }, goToURL: function(args) { var url = args.url + ( typeof args.medium === 'undefined' ? '' : '?pk_campaign=firefox&pk_kwd=' + args.medium); require('sdk/tabs').open({ url: url, inBackground: typeof args.background !== 'undefined' && args.background }); }, displayApps: function() { var url, count; w.log('display apps'); if ( !require('sdk/tabs').activeTab ) { return; } url = require('sdk/tabs').activeTab.url.replace(/#.*$/, ''); count = w.detected[url] ? Object.keys(w.detected[url]).length : 0; if ( !panel.get() ) { panel = new Panel(); if ( require('sdk/simple-prefs').prefs.urlbar ) { urlBar = new UrlBar(panel); } else { widget = new Widget(panel); } } if ( typeof tabCache[require('sdk/tabs').activeTab.id] === 'undefined' ) { initTab(require('sdk/tabs').activeTab); } tabCache[require('sdk/tabs').activeTab.id].count = count; tabCache[require('sdk/tabs').activeTab.id].appsDetected = w.detected[url]; if ( require('sdk/simple-prefs').prefs.urlbar ) { urlBar.clear(); // Add icons if ( count ) { for ( appName in tabCache[require('sdk/tabs').activeTab.id].appsDetected ) { urlBar.addIcon(appName); } } else { urlBar.addIcon(); } } else { widget.get().contentURL = require('sdk/self').data.url('images/icon32_hot.png'); if ( count ) { // Find the main application to display var appName, found = false; w.driver.categoryOrder.forEach(function(match) { for ( appName in w.detected[url] ) { w.apps[appName].cats.forEach(function(cat) { if ( cat == match && !found ) { widget.get().contentURL = require('sdk/self').data.url('images/icons/' + appName + '.png'), found = true; } }); } }); } } panel.get().port.emit('displayApps', { tabCache: tabCache[require('sdk/tabs').activeTab.id], apps: w.apps, categories: w.categories, categoryNames: categoryNames }); }, ping: function() { var Request = require('sdk/request').Request; if ( Object.keys(w.ping.hostnames).length && require('sdk/simple-prefs').prefs.tracking ) { Request({ url: w.config.websiteURL + 'ping/v2/', content: { json: encodeURIComponent(JSON.stringify(w.ping)) }, onComplete: function (response) { w.log('w.driver.ping: status ' + response.status); } }).post(); w.log('w.driver.ping: ' + JSON.stringify(w.ping)); w.ping = { hostnames: {} }; } }, categoryOrder: [ // Used to pick the main application 1, // CMS 11, // Blog 6, // Web Shop 2, // Message Board 8, // Wiki 13, // Issue Tracker 30, // Web Mail 18, // Web Framework 21, // LMS 7, // Photo Gallery 38, // Media Server 3, // Database Manager 34, // Database 4, // Documentation Tool 9, // Hosting Panel 29, // Search Engine 12, // Javascript Framework 26, // Mobile Framework 25, // Javascript Graphics 22, // Web Server 27, // Programming Language 28, // Operating System 15, // Comment System 20, // Editor 41, // Payment Processor 10, // Analytics 32, // Marketing Automation 31, // CDN 23, // Cache Tool 17, // Font Script 24, // Rich Text Editor 35, // Map 5, // Widget 14, // Video Player 16, // Captcha 33, // Web Server Extension 37, // Network Device 39, // Webcam 40, // Printer 36, // Advertising Network 19 // Miscellaneous ] }; }());