diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 17:32:43 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 17:32:43 +0000 |
commit | 6bf0a5cb5034a7e684dcc3500e841785237ce2dd (patch) | |
tree | a68f146d7fa01f0134297619fbe7e33db084e0aa /comm/suite/chatzilla/js/lib | |
parent | Initial commit. (diff) | |
download | thunderbird-upstream.tar.xz thunderbird-upstream.zip |
Adding upstream version 1:115.7.0.upstream/1%115.7.0upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to '')
21 files changed, 13139 insertions, 0 deletions
diff --git a/comm/suite/chatzilla/js/lib/chatzilla-protocol-script.js b/comm/suite/chatzilla/js/lib/chatzilla-protocol-script.js new file mode 100644 index 0000000000..e8cb72d259 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/chatzilla-protocol-script.js @@ -0,0 +1,10 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * 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/. */ + +let { ChatZillaProtocols } = ChromeUtils.import( + "chrome://chatzilla/content/lib/js/protocol-handlers.jsm" +); + +ChatZillaProtocols.init(); diff --git a/comm/suite/chatzilla/js/lib/chatzilla-service.js b/comm/suite/chatzilla/js/lib/chatzilla-service.js new file mode 100644 index 0000000000..8772f3a782 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/chatzilla-service.js @@ -0,0 +1,311 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * 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/. */ + +const { classes: Cc, interfaces: Ci, utils: Cu, results: Cr } = Components; + +const MEDIATOR_CONTRACTID = + "@mozilla.org/appshell/window-mediator;1"; +const ASS_CONTRACTID = + "@mozilla.org/appshell/appShellService;1"; +const RDFS_CONTRACTID = + "@mozilla.org/rdf/rdf-service;1"; +const CATMAN_CONTRACTID = + "@mozilla.org/categorymanager;1"; +const PPMM_CONTRACTID = + "@mozilla.org/parentprocessmessagemanager;1"; + +const CLINE_SERVICE_CONTRACTID = + "@mozilla.org/commandlinehandler/general-startup;1?type=chat"; +const CLINE_SERVICE_CID = + Components.ID("{38a95514-1dd2-11b2-97e7-9da958640f2c}"); +const STARTUP_CID = + Components.ID("{ae6ad015-433b-42ab-9afc-1636af5a7fc4}"); + + +var { + ChatZillaProtocols, + IRCProtocolHandlerFactory, + IRCSProtocolHandlerFactory, + IRCPROT_HANDLER_CID, + IRCSPROT_HANDLER_CID, +} = ChromeUtils.import( + "chrome://chatzilla/content/lib/js/protocol-handlers.jsm" +); + +function spawnChatZilla(uri, count) +{ + const wm = Cc[MEDIATOR_CONTRACTID].getService(Ci.nsIWindowMediator); + const ass = Cc[ASS_CONTRACTID].getService(Ci.nsIAppShellService); + const hiddenWin = ass.hiddenDOMWindow; + + // Ok, not starting currently, so check if we've got existing windows. + const w = wm.getMostRecentWindow("irc:chatzilla"); + + // Claiming that a ChatZilla window is loading. + if ("ChatZillaStarting" in hiddenWin) + { + dump("cz-service: ChatZilla claiming to be starting.\n"); + if (w && ("client" in w) && ("initialized" in w.client) && + w.client.initialized) + { + dump("cz-service: It lied. It's finished starting.\n"); + // It's actually loaded ok. + delete hiddenWin.ChatZillaStarting; + } + } + + if ("ChatZillaStarting" in hiddenWin) + { + count = count || 0; + + if ((new Date() - hiddenWin.ChatZillaStarting) > 10000) + { + dump("cz-service: Continuing to be unable to talk to existing window!\n"); + } + else + { + // We have a ChatZilla window, but we're still loading. + hiddenWin.setTimeout(function wrapper(count) { + spawnChatZilla(uri, count + 1); + }, 250, count); + return true; + } + } + + // We have a window. + if (w) + { + dump("cz-service: Existing, fully loaded window. Using.\n"); + // Window is working and initialized ok. Use it. + w.focus(); + if (uri) + w.gotoIRCURL(uri); + return true; + } + + dump("cz-service: No windows, starting new one.\n"); + // Ok, no available window, loading or otherwise, so start ChatZilla. + const args = new Object(); + if (uri) + args.url = uri; + + hiddenWin.ChatZillaStarting = new Date(); + hiddenWin.openDialog("chrome://chatzilla/content/chatzilla.xul", "_blank", + "chrome,menubar,toolbar,status,resizable,dialog=no", + args); + + return true; +} + + +function CommandLineService() +{ +} + +CommandLineService.prototype = +{ + /* nsISupports */ + QueryInterface(iid) + { + if (iid.equals(Ci.nsISupports)) + return this; + + if (Ci.nsICommandLineHandler && iid.equals(Ci.nsICommandLineHandler)) + return this; + + throw Cr.NS_ERROR_NO_INTERFACE; + }, + + /* nsICommandLineHandler */ + handle(cmdLine) + { + var uri; + try + { + uri = cmdLine.handleFlagWithParam("chat", false); + } + catch (e) + { + } + + if (uri || cmdLine.handleFlag("chat", false)) + { + spawnChatZilla(uri || null) + cmdLine.preventDefault = true; + } + }, + + helpInfo: "-chat [<ircurl>] Start with an IRC chat client.\n", +}; + + +/* factory for command line handler service (CommandLineService) */ +const CommandLineFactory = +{ + createInstance(outer, iid) + { + if (outer != null) + throw Cr.NS_ERROR_NO_AGGREGATION; + + return new CommandLineService().QueryInterface(iid); + }, +}; + + +function ProcessHandler() +{ +} + +ProcessHandler.prototype = +{ + /* nsISupports */ + QueryInterface(iid) + { + if (iid.equals(Ci.nsISupports) || + iid.equals(Ci.nsIObserver) || + iid.equals(Ci.nsIMessageListener)) + { + return this; + } + + throw Cr.NS_ERROR_NO_INTERFACE; + }, + + /* nsIObserver */ + observe(subject, topic, data) + { + if (topic !== "profile-after-change") + return; + + var ppmm; + // Ci.nsIMessageBroadcaster went in Gecko 61. + if (Ci.nsIMessageBroadcaster) + { + ppmm = Cc[PPMM_CONTRACTID].getService(Ci.nsIMessageBroadcaster); + } + else + { + ppmm = Cc[PPMM_CONTRACTID].getService(); + } + ppmm.loadProcessScript("chrome://chatzilla/content/lib/js/chatzilla-protocol-script.js", true); + ppmm.addMessageListener("ChatZilla:SpawnChatZilla", this); + }, + + /* nsIMessageListener */ + receiveMessage(msg) + { + if (msg.name !== "ChatZilla:SpawnChatZilla") + return; + + spawnChatZilla(msg.data.uri); + }, +}; + + +const StartupFactory = +{ + createInstance(outer, iid) + { + if (outer) + throw Cr.NS_ERROR_NO_AGGREGATION; + + if (!iid.equals(Ci.nsISupports)) + throw Cr.NS_ERROR_NO_INTERFACE; + + // startup: + return new ProcessHandler(); + }, +}; + + +const ChatZillaModule = +{ + registerSelf(compMgr, fileSpec, location, type) + { + compMgr = compMgr.QueryInterface(Ci.nsIComponentRegistrar); + const catman = Cc[CATMAN_CONTRACTID].getService(Ci.nsICategoryManager); + + debug("*** Registering -chat handler.\n"); + compMgr.registerFactoryLocation(CLINE_SERVICE_CID, + "ChatZilla CommandLine Service", + CLINE_SERVICE_CONTRACTID, + fileSpec, location, type); + catman.addCategoryEntry("command-line-argument-handlers", + "chatzilla command line handler", + CLINE_SERVICE_CONTRACTID, true, true); + catman.addCategoryEntry("command-line-handler", + "m-irc", + CLINE_SERVICE_CONTRACTID, true, true); + + debug("*** Registering irc protocol handler.\n"); + ChatZillaProtocols.initObsolete(compMgr, fileSpec, location, type); + + debug("*** Registering done.\n"); + }, + + unregisterSelf(compMgr, fileSpec, location) + { + compMgr = compMgr.QueryInterface(Ci.nsIComponentRegistrar); + + const catman = Cc[CATMAN_CONTRACTID].getService(Ci.nsICategoryManager); + catman.deleteCategoryEntry("command-line-argument-handlers", + "chatzilla command line handler", true); + catman.deleteCategoryEntry("command-line-handler", + "m-irc", true); + }, + + getClassObject(compMgr, cid, iid) + { + // Checking if we're disabled in the Chrome Registry. + var rv; + try + { + const rdfSvc = Cc[RDFS_CONTRACTID].getService(Ci.nsIRDFService); + const rdfDS = rdfSvc.GetDataSource("rdf:chrome"); + const resSelf = rdfSvc.GetResource("urn:mozilla:package:chatzilla"); + const resDisabled = rdfSvc.GetResource("http://www.mozilla.org/rdf/chrome#disabled"); + rv = rdfDS.GetTarget(resSelf, resDisabled, true); + } + catch (e) + { + } + if (rv) + throw Cr.NS_ERROR_NO_INTERFACE; + + if (cid.equals(CLINE_SERVICE_CID)) + return CommandLineFactory; + + if (cid.equals(IRCPROT_HANDLER_CID)) + return IRCProtocolHandlerFactory; + + if (cid.equals(IRCSPROT_HANDLER_CID)) + return IRCSProtocolHandlerFactory; + + if (cid.equals(STARTUP_CID)) + return StartupFactory; + + if (!iid.equals(Ci.nsIFactory)) + throw Cr.NS_ERROR_NOT_IMPLEMENTED; + + throw Cr.NS_ERROR_NO_INTERFACE; + }, + + canUnload(compMgr) + { + return true; + }, +}; + + +/* entrypoint */ +function NSGetModule(compMgr, fileSpec) +{ + return ChatZillaModule; +} + +function NSGetFactory(cid) +{ + return ChatZillaModule.getClassObject(null, cid, null); +} diff --git a/comm/suite/chatzilla/js/lib/chatzilla-service.manifest b/comm/suite/chatzilla/js/lib/chatzilla-service.manifest new file mode 100644 index 0000000000..28a0d84d8f --- /dev/null +++ b/comm/suite/chatzilla/js/lib/chatzilla-service.manifest @@ -0,0 +1,6 @@ +component {38a95514-1dd2-11b2-97e7-9da958640f2c} chatzilla-service.js +contract @mozilla.org/commandlinehandler/general-startup;1?type=chat {38a95514-1dd2-11b2-97e7-9da958640f2c} +category command-line-handler m-irc @mozilla.org/commandlinehandler/general-startup;1?type=chat +component {ae6ad015-433b-42ab-9afc-1636af5a7fc4} chatzilla-service.js +contract @chatzilla.mozilla.org/startup;1 {ae6ad015-433b-42ab-9afc-1636af5a7fc4} +category profile-after-change chatzilla-startup @chatzilla.mozilla.org/startup;1 diff --git a/comm/suite/chatzilla/js/lib/command-manager.js b/comm/suite/chatzilla/js/lib/command-manager.js new file mode 100644 index 0000000000..76d2818c3c --- /dev/null +++ b/comm/suite/chatzilla/js/lib/command-manager.js @@ -0,0 +1,952 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +// @internal +function getAccessKey(str) +{ + var i = str.indexOf("&"); + if (i == -1) + return ""; + return str[i + 1]; +} + +// @internal +function CommandRecord(name, func, usage, help, label, accesskey, flags, + keystr, tip, format, helpUsage) +{ + this.name = name; + this.func = func; + this._usage = usage; + this.scanUsage(); + this.help = help; + this.label = label ? label : name; + this.accesskey = accesskey ? accesskey : ""; + this.format = format; + this.helpUsage = helpUsage; + this.labelstr = label.replace ("&", ""); + this.tip = tip; + this.flags = flags; + this._enabled = true; + this.keyNodes = new Array(); + this.keystr = keystr; + this.uiElements = new Array(); +} + +CommandRecord.prototype.__defineGetter__ ("enabled", cr_getenable); +function cr_getenable () +{ + return this._enabled; +} + +CommandRecord.prototype.__defineSetter__ ("enabled", cr_setenable); +function cr_setenable (state) +{ + for (var i = 0; i < this.uiElements.length; ++i) + { + if (state) + this.uiElements[i].removeAttribute ("disabled"); + else + this.uiElements[i].setAttribute ("disabled", "true"); + } + return (this._enabled = state); +} + +CommandRecord.prototype.__defineSetter__ ("usage", cr_setusage); +function cr_setusage (usage) +{ + this._usage = usage; + this.scanUsage(); +} + +CommandRecord.prototype.__defineGetter__ ("usage", cr_getusage); +function cr_getusage() +{ + return this._usage; +} + +/** + * @internal + * + * Scans the argument spec, in the format "<a1> <a2> [<o1> <o2>]", into an + * array of strings. + */ +CommandRecord.prototype.scanUsage = +function cr_scanusage() +{ + var spec = this._usage; + var currentName = ""; + var inName = false; + var len = spec.length; + var capNext = false; + + this._usage = spec; + this.argNames = new Array(); + + for (var i = 0; i < len; ++i) + { + switch (spec[i]) + { + case '[': + this.argNames.push (":"); + break; + + case '<': + inName = true; + break; + + case '-': + capNext = true; + break; + + case '>': + inName = false; + this.argNames.push (currentName); + currentName = ""; + capNext = false; + break; + + default: + if (inName) + currentName += capNext ? spec[i].toUpperCase() : spec[i]; + capNext = false; + break; + } + } +} + +/** + * Manages commands, with accelerator keys, help text and argument processing. + * + * You should never need to create an instance of this prototype; access the + * command manager through |client.commandManager|. + * + * @param defaultBundle An |nsIStringBundle| object to load command parameters, + * labels a help text from. + */ +function CommandManager(defaultBundle) +{ + this.commands = new Object(); + this.commandHistory = new Object(); + this.defaultBundle = defaultBundle; + this.currentDispatchDepth = 0; + this.maxDispatchDepth = 10; + this.dispatchUnwinding = false; +} + +// @undocumented +CommandManager.prototype.defaultFlags = 0; + +/** + * Adds multiple commands in a single call. + * + * @param cmdary |Array| containing commands to define; each item in the |Array| + * is also an |Array|, with either 3 or 4 items - corresponding to + * the first three or four arguments of |defineCommand|. An extra + * property, |stringBundle|, may be set on the |cmdary| |Array| + * to override the |defaultBundle| for all the commands. + */ +CommandManager.prototype.defineCommands = +function cmgr_defcmds(cmdary) +{ + var len = cmdary.length; + var commands = new Object(); + var bundle = "stringBundle" in cmdary ? cmdary.stringBundle : null; + + for (var i = 0; i < len; ++i) + { + let name = cmdary[i][0]; + let func = cmdary[i][1]; + let flags = cmdary[i][2]; + let usage = (3 in cmdary[i]) ? cmdary[i][3] : ""; + commands[name] = this.defineCommand(name, func, flags, usage, bundle); + } + + return commands; +} + +/** + * Adds a single command. + * + * @param name The |String| name of the command to define. + * @param func A |Function| to call to handle dispatch of the new command. + * @param flags Optional. A |Number| indicating any special requirements for the + * command. + * @param usage Optional. A |String| specifying the arguments to the command. If + * not specified, then it is assumed there are none. + * @param bundle Optional. An |nsIStringBundle| to fetch parameters, labels, + * accelerator keys and help from. If not specified, the + * |defaultBundle| is used. + */ +CommandManager.prototype.defineCommand = +function cmdmgr_defcmd(name, func, flags, usage, bundle) +{ + if (!bundle) + bundle = this.defaultBundle; + + var helpDefault = MSG_NO_HELP; + var labelDefault = name; + var aliasFor; + + if (typeof flags != "number") + flags = this.defaultFlags; + + if (typeof usage != "string") + usage = ""; + + if (typeof func == "string") + { + var ary = func.match(/(\S+)/); + if (ary) + aliasFor = ary[1]; + else + aliasFor = null; + helpDefault = getMsg (MSG_DEFAULT_ALIAS_HELP, func); + if (aliasFor) + labelDefault = getMsgFrom (bundle, "cmd." + aliasFor + ".label", + null, name); + } + + var label = getMsgFrom(bundle, "cmd." + name + ".label", null, + labelDefault); + var accesskey = getMsgFrom(bundle, "cmd." + name + ".accesskey", null, + getAccessKey(label)); + var help = helpDefault; + var helpUsage = ""; + // Help is only shown for commands that available from the console. + if (flags & CMD_CONSOLE) + { + help = getMsgFrom(bundle, "cmd." + name + ".help", null, helpDefault); + // Only need to lookup localized helpUsage for commands that have them. + if (usage) + { + helpUsage = getMsgFrom(bundle, "cmd." + name + ".helpUsage", null, + ""); + } + } + var keystr = getMsgFrom (bundle, "cmd." + name + ".key", null, ""); + var format = getMsgFrom (bundle, "cmd." + name + ".format", null, null); + var tip = getMsgFrom (bundle, "cmd." + name + ".tip", null, ""); + var command = new CommandRecord(name, func, usage, help, label, accesskey, + flags, keystr, tip, format, helpUsage); + this.addCommand(command); + if (aliasFor) + command.aliasFor = aliasFor; + + return command; +} + +/** + * Installs accelerator keys for commands into an existing document. + * + * @internal + * @param document An |XULDocument| within which to install the accelerator + * keys. Each command's key is installed by |installKey|. + * @param commands Optional. An |Array| or |Object| continaing |CommandRecord| + * objects. If not specified, all commands in the + * |CommandManager| are installed. + */ +CommandManager.prototype.installKeys = +function cmgr_instkeys(document, commands) +{ + var parentElem = document.getElementById("dynamic-keys"); + if (!parentElem) + { + parentElem = document.createElement("keyset"); + parentElem.setAttribute("id", "dynamic-keys"); + document.documentElement.appendChild(parentElem); + } + + if (!commands) + commands = this.commands; + + for (var c in commands) + this.installKey (parentElem, commands[c]); +} + +/** + * Installs the accelerator key for a single command. + * + * This creates a <key> XUL element inside |parentElem|. It should usually be + * called once per command, per document, so that accelerator keys work in all + * application windows. + * + * @internal + * @param parentElem An |XULElement| to add the <key> too. + * @param command The |CommandRecord| to install. + */ +CommandManager.prototype.installKey = +function cmgr_instkey(parentElem, command) +{ + if (!command.keystr) + return; + + var ary = command.keystr.match (/(.*\s)?([\S]+)$/); + if (!ASSERT(ary, "couldn't parse key string ``" + command.keystr + + "'' for command ``" + command.name + "''")) + { + return; + } + + var key = document.createElement ("key"); + key.setAttribute ("id", "key:" + command.name); + key.setAttribute ("oncommand", "dispatch('" + command.name + + "', {isInteractive: true, source: 'keyboard'});"); + + if (ary[1]) + key.setAttribute ("modifiers", ary[1]); + + if (ary[2].indexOf("VK_") == 0) + key.setAttribute ("keycode", ary[2]); + else + key.setAttribute ("key", ary[2]); + + parentElem.appendChild(key); + command.keyNodes.push(key); +} + +/** + * Uninstalls accelerator keys for commands from a document. + * + * @internal + * @param commands Optional. An |Array| or |Object| continaing |CommandRecord| + * objects. If not specified, all commands in the + * |CommandManager| are uninstalled. + */ +CommandManager.prototype.uninstallKeys = +function cmgr_uninstkeys(commands) +{ + if (!commands) + commands = this.commands; + + for (var c in commands) + this.uninstallKey (commands[c]); +} + +/** + * Uninstalls the accelerator key for a single command. + * + * @internal + * @param command The |CommandRecord| to uninstall. + */ +CommandManager.prototype.uninstallKey = +function cmgr_uninstkey(command) +{ + for (var i in command.keyNodes) + { + try + { + /* document may no longer exist in a useful state. */ + command.keyNodes[i].parentNode.removeChild(command.keyNodes[i]); + } + catch (ex) + { + dd ("*** caught exception uninstalling key node: " + ex); + } + } +} + +/** + * Use |defineCommand|. + * + * @internal + * @param command The |CommandRecord| to add to the |CommandManager|. + */ +CommandManager.prototype.addCommand = +function cmgr_add(command) +{ + if (objectContains(this.commands, command.name)) + { + /* We've already got a command with this name - invoke the history + * storage so that we can undo this back to its original state. + */ + if (!objectContains(this.commandHistory, command.name)) + this.commandHistory[command.name] = new Array(); + this.commandHistory[command.name].push(this.commands[command.name]); + } + this.commands[command.name] = command; +} + +/** + * Removes multiple commands in a single call. + * + * @param cmdary An |Array| or |Object| containing |CommandRecord| objects. + * Ideally use the value returned from |defineCommands|. + */ +CommandManager.prototype.removeCommands = +function cmgr_removes(cmdary) +{ + for (var i in cmdary) + { + var command = isinstance(cmdary[i], Array) ? + {name: cmdary[i][0]} : cmdary[i]; + this.removeCommand(command); + } +} + +/** + * Removes a single command. + * + * @param command The |CommandRecord| to remove from the |CommandManager|. + * Ideally use the value returned from |defineCommand|. + */ +CommandManager.prototype.removeCommand = +function cmgr_remove(command) +{ + delete this.commands[command.name]; + if (objectContains(this.commandHistory, command.name)) + { + /* There was a previous command with this name - restore the most + * recent from the history, returning the command to its former glory. + */ + this.commands[command.name] = this.commandHistory[command.name].pop(); + if (this.commandHistory[command.name].length == 0) + delete this.commandHistory[command.name]; + } +} + +/** + * Registers a hook for a particular command. + * + * A command hook is uniquely identified by the pair |id|, |before|; only a + * single hook may exist for a given pair of |id| and |before| values. It is + * wise to use a unique |id|; plugins should construct an |id| using + * |plugin.id|, e.g. |plugin.id + "-my-hook-1"|. + * + * @param commandName A |String| command name to hook. The command named must + * already exist in the |CommandManager|; if it does not, no + * hook is added. + * @param func A |Function| to handle the hook. + * @param id A |String| identifier for the hook. + * @param before A |Boolean| indicating whether the hook wishes to be + * called before or after the command executes. + */ +CommandManager.prototype.addHook = +function cmgr_hook (commandName, func, id, before) +{ + if (!ASSERT(objectContains(this.commands, commandName), + "Unknown command '" + commandName + "'")) + { + return; + } + + var command = this.commands[commandName]; + + if (before) + { + if (!("beforeHooks" in command)) + command.beforeHooks = new Object(); + command.beforeHooks[id] = func; + } + else + { + if (!("afterHooks" in command)) + command.afterHooks = new Object(); + command.afterHooks[id] = func; + } +} + +/** + * Registers multiple hooks for commands. + * + * @param hooks An |Object| containing |Function| objects to call for each + * hook; the key of each item is the name of the command it + * wishes to hook. Optionally, the |_before| property can be + * added to a |function| to override the default |before| value + * of |false|. + * @param prefix Optional. A |String| prefix to apply to each hook's command + * name to compute an |id| for it. + */ +CommandManager.prototype.addHooks = +function cmgr_hooks (hooks, prefix) +{ + if (!prefix) + prefix = ""; + + for (var h in hooks) + { + this.addHook(h, hooks[h], prefix + ":" + h, + ("_before" in hooks[h]) ? hooks[h]._before : false); + } +} + +/** + * Unregisters multiple hooks for commands. + * + * @param hooks An |Object| identical to the one passed to |addHooks|. + * @param prefix Optional. A |String| identical to the one passed to |addHooks|. + */ +CommandManager.prototype.removeHooks = +function cmgr_remhooks (hooks, prefix) +{ + if (!prefix) + prefix = ""; + + for (var h in hooks) + { + this.removeHook(h, prefix + ":" + h, + ("before" in hooks[h]) ? hooks[h].before : false); + } +} + +/** + * Unregisters a hook for a particular command. + * + * The arguments to |removeHook| are the same as |addHook|, but without the + * hook function itself. + * + * @param commandName The |String| command name to unhook. + * @param id The |String| identifier for the hook. + * @param before A |Boolean| indicating whether the hook was to be + * called before or after the command executed. + */ +CommandManager.prototype.removeHook = +function cmgr_unhook (commandName, id, before) +{ + var command = this.commands[commandName]; + + if (before) + delete command.beforeHooks[id]; + else + delete command.afterHooks[id]; +} + +/** + * Gets a sorted |Array| of |CommandRecord| objects which match. + * + * After filtering by |flags| (if specified), if an exact match for + * |partialName| is found, only that is returned; otherwise, all commands + * starting with |partialName| are returned in alphabetical order by |label|. + * + * @param partialName Optional. A |String| prefix to search for. + * @param flags Optional. Flags to logically AND with commands. + */ +CommandManager.prototype.list = +function cmgr_list(partialName, flags, exact) +{ + /* returns array of command objects which look like |partialName|, or + * all commands if |partialName| is not specified */ + function compare (a, b) + { + a = a.labelstr.toLowerCase(); + b = b.labelstr.toLowerCase(); + + if (a == b) + return 0; + + if (a > b) + return 1; + + return -1; + } + + var ary = new Array(); + var commandNames = keys(this.commands); + + for (var name of commandNames) + { + let command = this.commands[name]; + if ((!flags || (command.flags & flags)) && + (!partialName || command.name.startsWith(partialName))) + { + if (exact && partialName && + partialName.length == command.name.length) + { + /* exact match */ + return [command]; + } + ary.push(command); + } + } + + ary.sort(compare); + return ary; +} + +/** + * Gets a sorted |Array| of command names which match. + * + * |listNames| operates identically to |list|, except that only command names + * are returned, not |CommandRecord| objects. + */ +CommandManager.prototype.listNames = +function cmgr_listnames (partialName, flags) +{ + var cmds = this.list(partialName, flags, false); + var cmdNames = new Array(); + + for (var c in cmds) + cmdNames.push (cmds[c].name); + + cmdNames.sort(); + return cmdNames; +} + +/** + * Internal use only. + * + * Called to parse the arguments stored in |e.inputData|, as properties of |e|, + * for the CommandRecord stored on |e.command|. + * + * @params e Event object to be processed. + */ +// @undocumented +CommandManager.prototype.parseArguments = +function cmgr_parseargs (e) +{ + var rv = this.parseArgumentsRaw(e); + //dd("parseArguments '" + e.command.usage + "' " + + // (rv ? "passed" : "failed") + "\n" + dumpObjectTree(e)); + delete e.currentArgIndex; + return rv; +} + +/** + * Internal use only. + * + * Don't call parseArgumentsRaw directly, use parseArguments instead. + * + * Parses the arguments stored in the |inputData| property of the event object, + * according to the format specified by the |command| property. + * + * On success this method returns true, and propery names corresponding to the + * argument names used in the format spec will be created on the event object. + * All optional parameters will be initialized to |null| if not already present + * on the event. + * + * On failure this method returns false and a description of the problem + * will be stored in the |parseError| property of the event. + * + * For example... + * Given the argument spec "<int> <word> [ <word2> <word3> ]", and given the + * input string "411 foo", stored as |e.command.usage| and |e.inputData| + * respectively, this method would add the following propertys to the event + * object... + * -name---value--notes- + * e.int 411 Parsed as an integer + * e.word foo Parsed as a string + * e.word2 null Optional parameters not specified will be set to null. + * e.word3 null If word2 had been provided, word3 would be required too. + * + * Each parameter is parsed by calling the function with the same name, located + * in this.argTypes. The first parameter is parsed by calling the function + * this.argTypes["int"], for example. This function is expected to act on + * e.unparsedData, taking it's chunk, and leaving the rest of the string. + * The default parse functions are... + * <word> parses contiguous non-space characters. + * <int> parses as an int. + * <rest> parses to the end of input data. + * <state> parses yes, on, true, 1, 0, false, off, no as a boolean. + * <toggle> parses like a <state>, except allows "toggle" as well. + * <...> parses according to the parameter type before it, until the end + * of the input data. Results are stored in an array named + * paramnameList, where paramname is the name of the parameter + * before <...>. The value of the parameter before this will be + * paramnameList[0]. + * + * If there is no parse function for an argument type, "word" will be used by + * default. You can alias argument types with code like... + * commandManager.argTypes["my-integer-name"] = commandManager.argTypes["int"]; + */ +// @undocumented +CommandManager.prototype.parseArgumentsRaw = +function parse_parseargsraw (e) +{ + var argc = e.command.argNames.length; + + function initOptionals() + { + for (var i = 0; i < argc; ++i) + { + if (e.command.argNames[i] != ":" && + e.command.argNames[i] != "..." && + !(e.command.argNames[i] in e)) + { + e[e.command.argNames[i]] = null; + } + + if (e.command.argNames[i] == "...") + { + var paramName = e.command.argNames[i - 1]; + if (paramName == ":") + paramName = e.command.argNames[i - 2]; + var listName = paramName + "List"; + if (!(listName in e)) + e[listName] = [ e[paramName] ]; + } + } + } + + if ("inputData" in e && e.inputData) + { + /* if data has been provided, parse it */ + e.unparsedData = e.inputData; + var parseResult; + var currentArg; + e.currentArgIndex = 0; + + if (argc) + { + currentArg = e.command.argNames[e.currentArgIndex]; + + while (e.unparsedData) + { + if (currentArg != ":") + { + if (!this.parseArgument (e, currentArg)) + return false; + } + if (++e.currentArgIndex < argc) + currentArg = e.command.argNames[e.currentArgIndex]; + else + break; + } + + if (e.currentArgIndex < argc && currentArg != ":") + { + /* parse loop completed because it ran out of data. We haven't + * parsed all of the declared arguments, and we're not stopped + * at an optional marker, so we must be missing something + * required... */ + e.parseError = getMsg(MSG_ERR_REQUIRED_PARAM, + e.command.argNames[e.currentArgIndex]); + return false; + } + } + + if (e.unparsedData) + { + /* parse loop completed with unparsed data, which means we've + * successfully parsed all arguments declared. Whine about the + * extra data... */ + display (getMsg(MSG_EXTRA_PARAMS, e.unparsedData), MT_WARN); + } + } + + var rv = this.isCommandSatisfied(e); + if (rv) + initOptionals(); + return rv; +} + +/** + * Returns true if |e| has the properties required to call the command + * |command|. + * + * If |command| is not provided, |e.command| is used instead. + * + * @param e Event object to test against the command. + * @param command Command to test. + */ +// @undocumented +CommandManager.prototype.isCommandSatisfied = +function cmgr_isok (e, command) +{ + if (typeof command == "undefined") + command = e.command; + else if (typeof command == "string") + command = this.commands[command]; + + if (!command.enabled) + return false; + + for (var i = 0; i < command.argNames.length; ++i) + { + if (command.argNames[i] == ":") + return true; + + if (!(command.argNames[i] in e)) + { + e.parseError = getMsg(MSG_ERR_REQUIRED_PARAM, command.argNames[i]); + //dd("command '" + command.name + "' unsatisfied: " + e.parseError); + return false; + } + } + + //dd ("command '" + command.name + "' satisfied."); + return true; +} + +/** + * Internal use only. + * See parseArguments above and the |argTypes| object below. + * + * Parses the next argument by calling an appropriate parser function, or the + * generic "word" parser if none other is found. + * + * @param e event object. + * @param name property name to use for the parse result. + */ +// @undocumented +CommandManager.prototype.parseArgument = +function cmgr_parsearg (e, name) +{ + var parseResult; + + if (name in this.argTypes) + parseResult = this.argTypes[name](e, name, this); + else + parseResult = this.argTypes["word"](e, name, this); + + if (!parseResult) + e.parseError = getMsg(MSG_ERR_INVALID_PARAM, + [name, e.unparsedData]); + + return parseResult; +} + +// @undocumented +CommandManager.prototype.argTypes = new Object(); + +/** + * Convenience function used to map a list of new types to an existing parse + * function. + */ +// @undocumented +CommandManager.prototype.argTypes.__aliasTypes__ = +function at_alias (list, type) +{ + for (var i in list) + { + this[list[i]] = this[type]; + } +} + +/** + * Internal use only. + * + * Parses an integer, stores result in |e[name]|. + */ +// @undocumented +CommandManager.prototype.argTypes["int"] = +function parse_int (e, name) +{ + var ary = e.unparsedData.match (/(\d+)(?:\s+(.*))?$/); + if (!ary) + return false; + e[name] = Number(ary[1]); + e.unparsedData = arrayHasElementAt(ary, 2) ? ary[2] : ""; + return true; +} + +/** + * Internal use only. + * + * Parses a word, which is defined as a list of nonspace characters. + * + * Stores result in |e[name]|. + */ +// @undocumented +CommandManager.prototype.argTypes["word"] = +function parse_word (e, name) +{ + var ary = e.unparsedData.match (/(\S+)(?:\s+(.*))?$/); + if (!ary) + return false; + e[name] = ary[1]; + e.unparsedData = arrayHasElementAt(ary, 2) ? ary[2] : ""; + return true; +} + +/** + * Internal use only. + * + * Parses a "state" which can be "true", "on", "yes", or 1 to indicate |true|, + * or "false", "off", "no", or 0 to indicate |false|. + * + * Stores result in |e[name]|. + */ +// @undocumented +CommandManager.prototype.argTypes["state"] = +function parse_state (e, name) +{ + var ary = + e.unparsedData.match (/(true|on|yes|1|false|off|no|0)(?:\s+(.*))?$/i); + if (!ary) + return false; + if (ary[1].search(/true|on|yes|1/i) != -1) + e[name] = true; + else + e[name] = false; + e.unparsedData = arrayHasElementAt(ary, 2) ? ary[2] : ""; + return true; +} + +/** + * Internal use only. + * + * Parses a "toggle" which can be "true", "on", "yes", or 1 to indicate |true|, + * or "false", "off", "no", or 0 to indicate |false|. In addition, the string + * "toggle" is accepted, in which case |e[name]| will be the string "toggle". + * + * Stores result in |e[name]|. + */ +// @undocumented +CommandManager.prototype.argTypes["toggle"] = +function parse_toggle (e, name) +{ + var ary = e.unparsedData.match + (/(toggle|true|on|yes|1|false|off|no|0)(?:\s+(.*))?$/i); + + if (!ary) + return false; + if (ary[1].search(/toggle/i) != -1) + e[name] = "toggle"; + else if (ary[1].search(/true|on|yes|1/i) != -1) + e[name] = true; + else + e[name] = false; + e.unparsedData = arrayHasElementAt(ary, 2) ? ary[2] : ""; + return true; +} + +/** + * Internal use only. + * + * Returns all unparsed data to the end of the line. + * + * Stores result in |e[name]|. + */ +// @undocumented +CommandManager.prototype.argTypes["rest"] = +function parse_rest (e, name) +{ + e[name] = e.unparsedData; + e.unparsedData = ""; + return true; +} + +/** + * Internal use only. + * + * Parses the rest of the unparsed data the same way the previous argument was + * parsed. Can't be used as the first parameter. if |name| is "..." then the + * name of the previous argument, plus the suffix "List" will be used instead. + * + * Stores result in |e[name]| or |e[lastName + "List"]|. + */ +// @undocumented +CommandManager.prototype.argTypes["..."] = +function parse_repeat (e, name, cm) +{ + ASSERT (e.currentArgIndex > 0, "<...> can't be the first argument."); + + var lastArg = e.command.argNames[e.currentArgIndex - 1]; + if (lastArg == ":") + lastArg = e.command.argNames[e.currentArgIndex - 2]; + + var listName = lastArg + "List"; + e[listName] = [ e[lastArg] ]; + + while (e.unparsedData) + { + if (!cm.parseArgument(e, lastArg)) + return false; + e[listName].push(e[lastArg]); + } + + e[lastArg] = e[listName][0]; + return true; +} diff --git a/comm/suite/chatzilla/js/lib/connection-xpcom.js b/comm/suite/chatzilla/js/lib/connection-xpcom.js new file mode 100644 index 0000000000..031689d74d --- /dev/null +++ b/comm/suite/chatzilla/js/lib/connection-xpcom.js @@ -0,0 +1,703 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +const NS_ERROR_MODULE_NETWORK = 2152398848; + +const NS_ERROR_UNKNOWN_HOST = NS_ERROR_MODULE_NETWORK + 30; +const NS_ERROR_CONNECTION_REFUSED = NS_ERROR_MODULE_NETWORK + 13; +const NS_ERROR_NET_TIMEOUT = NS_ERROR_MODULE_NETWORK + 14; +const NS_ERROR_OFFLINE = NS_ERROR_MODULE_NETWORK + 16; +const NS_ERROR_NET_RESET = NS_ERROR_MODULE_NETWORK + 20; +const NS_ERROR_UNKNOWN_PROXY_HOST = NS_ERROR_MODULE_NETWORK + 42; +const NS_ERROR_NET_INTERRUPT = NS_ERROR_MODULE_NETWORK + 71; +const NS_ERROR_PROXY_CONNECTION_REFUSED = NS_ERROR_MODULE_NETWORK + 72; + +// Offline error constants: +const NS_ERROR_BINDING_ABORTED = NS_ERROR_MODULE_NETWORK + 2; +const NS_ERROR_ABORT = 0x80004004; + +const NS_NET_STATUS_RESOLVING_HOST = NS_ERROR_MODULE_NETWORK + 3; +const NS_NET_STATUS_CONNECTED_TO = NS_ERROR_MODULE_NETWORK + 4; +const NS_NET_STATUS_SENDING_TO = NS_ERROR_MODULE_NETWORK + 5; +const NS_NET_STATUS_RECEIVING_FROM = NS_ERROR_MODULE_NETWORK + 6; +const NS_NET_STATUS_CONNECTING_TO = NS_ERROR_MODULE_NETWORK + 7; + +// Security error class constants: +const ERROR_CLASS_SSL_PROTOCOL = 1; +const ERROR_CLASS_BAD_CERT = 2; + +// Security Constants. +const STATE_IS_BROKEN = 1; +const STATE_IS_SECURE = 2; +const STATE_IS_INSECURE = 3; + +const nsIScriptableInputStream = Components.interfaces.nsIScriptableInputStream; + +const nsIBinaryInputStream = Components.interfaces.nsIBinaryInputStream; +const nsIBinaryOutputStream = Components.interfaces.nsIBinaryOutputStream; + +function toSInputStream(stream, binary) +{ + var sstream; + + if (binary) + { + sstream = Components.classes["@mozilla.org/binaryinputstream;1"]; + sstream = sstream.createInstance(nsIBinaryInputStream); + sstream.setInputStream(stream); + } + else + { + sstream = Components.classes["@mozilla.org/scriptableinputstream;1"]; + sstream = sstream.createInstance(nsIScriptableInputStream); + sstream.init(stream); + } + + return sstream; +} + +function toSOutputStream(stream, binary) +{ + var sstream; + + if (binary) + { + sstream = Components.classes["@mozilla.org/binaryoutputstream;1"]; + sstream = sstream.createInstance(Components.interfaces.nsIBinaryOutputStream); + sstream.setOutputStream(stream); + } + else + { + sstream = stream; + } + + return sstream; +} + +/* This object implements nsIBadCertListener2 + * The idea is to suppress the default UI's alert box + * and allow the exception to propagate normally + */ +function BadCertHandler() +{ +} + +BadCertHandler.prototype.getInterface = +function badcert_getinterface(aIID) +{ + return this.QueryInterface(aIID); +} + +BadCertHandler.prototype.QueryInterface = +function badcert_queryinterface(aIID) +{ + if (aIID.equals(Components.interfaces.nsIBadCertListener2) || + aIID.equals(Components.interfaces.nsIInterfaceRequestor) || + aIID.equals(Components.interfaces.nsISupports)) + { + return this; + } + + throw Components.results.NS_ERROR_NO_INTERFACE; +} + +/* Returning true in the following two callbacks + * means suppress default the error UI (modal alert). + */ +BadCertHandler.prototype.notifyCertProblem = +function badcert_notifyCertProblem(socketInfo, sslStatus, targetHost) +{ + return true; +} + +/** + * Wraps up various mechanics of sockets for easy consumption by other code. + * + * @param binary Provide |true| or |false| here to override the automatic + * selection of binary or text streams. This should only ever be + * specified as |true| or omitted, otherwise you will be shooting + * yourself in the foot on some versions - let the code handle + * the choice unless you know you need binary. + */ +function CBSConnection (binary) +{ + /* Since 2003-01-17 18:14, Mozilla has had this contract ID for the STS. + * Prior to that it didn't have one, so we also include the CID for the + * STS back then - DO NOT UPDATE THE ID if it changes in Mozilla. + */ + const sockClassByName = + Components.classes["@mozilla.org/network/socket-transport-service;1"]; + const sockClassByID = + Components.classesByID["{c07e81e0-ef12-11d2-92b6-00105a1b0d64}"]; + + var sockServiceClass = (sockClassByName || sockClassByID); + + if (!sockServiceClass) + throw ("Couldn't get socket service class."); + + var sockService = sockServiceClass.getService(); + if (!sockService) + throw ("Couldn't get socket service."); + + this._sockService = sockService.QueryInterface + (Components.interfaces.nsISocketTransportService); + + /* Note: as part of the mess from bug 315288 and bug 316178, ChatZilla now + * uses the *binary* stream interfaces for all network + * communications. + * + * However, these interfaces do not exist prior to 1999-11-05. To + * make matters worse, an incompatible change to the "readBytes" + * method of this interface was made on 2003-03-13; luckly, this + * change also added a "readByteArray" method, which we will check + * for below, to determine if we can use the binary streams. + */ + + // We want to check for working binary streams only the first time. + if (CBSConnection.prototype.workingBinaryStreams == -1) + { + CBSConnection.prototype.workingBinaryStreams = false; + + if (typeof nsIBinaryInputStream != "undefined") + { + var isCls = Components.classes["@mozilla.org/binaryinputstream;1"]; + var inputStream = isCls.createInstance(nsIBinaryInputStream); + if ("readByteArray" in inputStream) + CBSConnection.prototype.workingBinaryStreams = true; + } + } + + /* + * As part of the changes in Gecko 1.9, invalid SSL certificates now + * produce a horrible error message. We must look up the toolkit version + * to see if we need to catch these errors cleanly - see bug 454966. + */ + if (!("strictSSL" in CBSConnection.prototype)) + { + CBSConnection.prototype.strictSSL = false; + var app = getService("@mozilla.org/xre/app-info;1", "nsIXULAppInfo"); + if (app && ("platformVersion" in app) && + compareVersions("1.9", app.platformVersion) >= 0) + { + CBSConnection.prototype.strictSSL = true; + } + } + + this.wrappedJSObject = this; + if (typeof binary != "undefined") + this.binaryMode = binary; + else + this.binaryMode = this.workingBinaryStreams; + + if (!ASSERT(!this.binaryMode || this.workingBinaryStreams, + "Unable to use binary streams in this build.")) + { + throw ("Unable to use binary streams in this build."); + } +} + +CBSConnection.prototype.workingBinaryStreams = -1; + +CBSConnection.prototype.connect = +function bc_connect(host, port, config, observer) +{ + this.host = host.toLowerCase(); + this.port = port; + + /* The APIs below want host:port. Later on, we also reformat the host to + * strip IPv6 literal brackets. + */ + var hostPort = host + ":" + port; + + if (!config) + config = {}; + + if (!("proxyInfo" in config)) + { + // Lets get a transportInfo for this + var pps = getService("@mozilla.org/network/protocol-proxy-service;1", + "nsIProtocolProxyService"); + + /* Force Necko to supply the HTTP proxy info if desired. For none, + * force no proxy. Other values will get default treatment. + */ + var uri = "irc://" + hostPort; + if ("proxy" in config) + { + if (config.proxy == "http") + uri = "http://" + hostPort; + else if (config.proxy == "none") + uri = ""; + } + + var self = this; + function continueWithProxy(proxyInfo) + { + config.proxyInfo = proxyInfo; + try + { + self.connect(host, port, config, observer); + } + catch (ex) + { + if ("onSocketConnection" in observer) + observer.onSocketConnection(host, port, config, ex); + return; + } + if ("onSocketConnection" in observer) + observer.onSocketConnection(host, port, config); + } + + if (uri) + { + uri = Services.io.newURI(uri); + if ("asyncResolve" in pps) + { + pps.asyncResolve(uri, 0, { + onProxyAvailable: function(request, uri, proxyInfo, status) { + continueWithProxy(proxyInfo); + } + }); + } + else if ("resolve" in pps) + { + continueWithProxy(pps.resolve(uri, 0)); + } + else if ("examineForProxy" in pps) + { + continueWithProxy(pps.examineForProxy(uri)); + } + else + { + throw "Unable to find method to resolve proxies"; + } + } + else + { + continueWithProxy(null); + } + return true; + } + + // Strip the IPv6 literal brackets; all the APIs below don't want them. + if (host[0] == '[' && host[host.length - 1] == ']') + host = host.substr(1, host.length - 2); + + /* Since the proxy info is opaque, we need to check that we got + * something for our HTTP proxy - we can't just check proxyInfo.type. + */ + var proxyInfo = config.proxyInfo || null; + var usingHTTPCONNECT = ("proxy" in config) && (config.proxy == "http") + && proxyInfo; + + if (proxyInfo && ("type" in proxyInfo) && (proxyInfo.type == "unknown")) + throw JSIRC_ERR_PAC_LOADING; + + /* use new necko interfaces */ + if (("isSecure" in config) && config.isSecure) + { + this._transport = this._sockService. + createTransport(["ssl"], 1, host, port, + proxyInfo); + + if (this.strictSSL) + this._transport.securityCallbacks = new BadCertHandler(); + } + else + { + this._transport = this._sockService. + createTransport(["starttls"], 1, host, port, proxyInfo); + } + if (!this._transport) + throw ("Error creating transport."); + + var openFlags = 0; + + /* no limit on the output stream buffer */ + this._outputStream = + this._transport.openOutputStream(openFlags, 4096, -1); + if (!this._outputStream) + throw "Error getting output stream."; + this._sOutputStream = toSOutputStream(this._outputStream, + this.binaryMode); + + this._inputStream = this._transport.openInputStream(openFlags, 0, 0); + if (!this._inputStream) + throw "Error getting input stream."; + this._sInputStream = toSInputStream(this._inputStream, + this.binaryMode); + + this.connectDate = new Date(); + this.isConnected = true; + + // Bootstrap the connection if we're proxying via an HTTP proxy. + if (usingHTTPCONNECT) + this.sendData("CONNECT " + hostPort + " HTTP/1.1\r\n\r\n"); + + return true; + +} + +CBSConnection.prototype.startTLS = +function bc_starttls() +{ + if (!this.isConnected || !this._transport.securityInfo) + return; + + var secInfo = this._transport.securityInfo; + var sockControl = secInfo.QueryInterface(Ci.nsITLSSocketControl); + sockControl.StartTLS(); +} + +CBSConnection.prototype.listen = +function bc_listen(port, observer) +{ + var serverSockClass = + Components.classes["@mozilla.org/network/server-socket;1"]; + + if (!serverSockClass) + throw ("Couldn't get server socket class."); + + var serverSock = serverSockClass.createInstance(); + if (!serverSock) + throw ("Couldn't get server socket."); + + this._serverSock = serverSock.QueryInterface + (Components.interfaces.nsIServerSocket); + + this._serverSock.init(port, false, -1); + + this._serverSockListener = new SocketListener(this, observer); + + this._serverSock.asyncListen(this._serverSockListener); + + this.port = this._serverSock.port; + + return true; +} + +CBSConnection.prototype.accept = +function bc_accept(transport, observer) +{ + this._transport = transport; + this.host = this._transport.host.toLowerCase(); + this.port = this._transport.port; + + var openFlags = 0; + + /* no limit on the output stream buffer */ + this._outputStream = + this._transport.openOutputStream(openFlags, 4096, -1); + if (!this._outputStream) + throw "Error getting output stream."; + this._sOutputStream = toSOutputStream(this._outputStream, + this.binaryMode); + + this._inputStream = this._transport.openInputStream(openFlags, 0, 0); + if (!this._inputStream) + throw "Error getting input stream."; + this._sInputStream = toSInputStream(this._inputStream, + this.binaryMode); + + this.connectDate = new Date(); + this.isConnected = true; + + // Clean up listening socket. + this.close(); + + return this.isConnected; +} + +CBSConnection.prototype.close = +function bc_close() +{ + if ("_serverSock" in this && this._serverSock) + this._serverSock.close(); +} + +CBSConnection.prototype.disconnect = +function bc_disconnect() +{ + if ("_inputStream" in this && this._inputStream) + this._inputStream.close(); + if ("_outputStream" in this && this._outputStream) + this._outputStream.close(); + this.isConnected = false; + /* + this._streamProvider.close(); + if (this._streamProvider.isBlocked) + this._write_req.resume(); + */ +} + +CBSConnection.prototype.sendData = +function bc_senddata(str) +{ + if (!this.isConnected) + throw "Not Connected."; + + this.sendDataNow(str); +} + +CBSConnection.prototype.readData = +function bc_readdata(timeout, count) +{ + if (!this.isConnected) + throw "Not Connected."; + + var rv; + + if (!("_sInputStream" in this)) { + this._sInputStream = toSInputStream(this._inputStream); + dump("OMG, setting up _sInputStream!\n"); + } + + try + { + // XPCshell h4x + if (typeof count == "undefined") + count = this._sInputStream.available(); + if (this.binaryMode) + rv = this._sInputStream.readBytes(count); + else + rv = this._sInputStream.read(count); + } + catch (ex) + { + dd ("*** Caught " + ex + " while reading."); + this.disconnect(); + throw (ex); + } + + return rv; +} + +CBSConnection.prototype.startAsyncRead = +function bc_saread (observer) +{ + var cls = Components.classes["@mozilla.org/network/input-stream-pump;1"]; + var pump = cls.createInstance(Components.interfaces.nsIInputStreamPump); + // Account for Bug 1402888 which removed the startOffset and readLimit + // parameters from init. + if (pump.init.length > 5) + { + pump.init(this._inputStream, -1, -1, 0, 0, false); + } else + { + pump.init(this._inputStream, 0, 0, false); + } + pump.asyncRead(new StreamListener(observer), this); +} + +CBSConnection.prototype.asyncWrite = +function bc_awrite (str) +{ + this._streamProvider.pendingData += str; + if (this._streamProvider.isBlocked) + { + this._write_req.resume(); + this._streamProvider.isBlocked = false; + } +} + +CBSConnection.prototype.hasPendingWrite = +function bc_haspwrite () +{ + return false; /* data already pushed to necko */ +} + +CBSConnection.prototype.sendDataNow = +function bc_senddatanow(str) +{ + var rv = false; + + try + { + if (this.binaryMode) + this._sOutputStream.writeBytes(str, str.length); + else + this._sOutputStream.write(str, str.length); + rv = true; + } + catch (ex) + { + dd ("*** Caught " + ex + " while sending."); + this.disconnect(); + throw (ex); + } + + return rv; +} + +/** + * Gets information about the security of the connection. + * + * |STATE_IS_BROKEN| is returned if any errors occur and |STATE_IS_INSECURE| is + * returned for disconnected sockets. + * + * @returns A value from the |STATE_IS_*| enumeration at the top of this file. + */ +CBSConnection.prototype.getSecurityState = +function bc_getsecuritystate() +{ + if (!this.isConnected || !this._transport.securityInfo) + return STATE_IS_INSECURE; + + try + { + // Get the actual SSL Status + let sslSp = this._transport.securityInfo + .QueryInterface(Ci.nsISSLStatusProvider); + if (!sslSp.SSLStatus) + return STATE_IS_BROKEN; + let sslStatus = sslSp.SSLStatus.QueryInterface(Ci.nsISSLStatus); + // Store appropriate status + if (!("keyLength" in sslStatus) || !sslStatus.keyLength) + return STATE_IS_BROKEN; + + return STATE_IS_SECURE; + } + catch (ex) + { + // Something goes wrong -> broken security icon + dd("Exception getting certificate for connection: " + ex.message); + return STATE_IS_BROKEN; + } +} + +CBSConnection.prototype.getCertificate = +function bc_getcertificate() +{ + if (!this.isConnected || !this._transport.securityInfo) + return null; + + // Get the actual SSL Status + let sslSp = this._transport.securityInfo + .QueryInterface(Ci.nsISSLStatusProvider); + if (!sslSp.SSLStatus) + return null; + let sslStatus = sslSp.SSLStatus.QueryInterface(Ci.nsISSLStatus); + + // return the certificate + return sslStatus.serverCert; +} + +CBSConnection.prototype.asyncWrite = +function bc_asyncwrite() +{ + throw "Not Implemented."; +} + +function StreamProvider(observer) +{ + this._observer = observer; +} + +StreamProvider.prototype.pendingData = ""; +StreamProvider.prototype.isBlocked = true; + +StreamProvider.prototype.close = +function sp_close () +{ + this.isClosed = true; +} + +StreamProvider.prototype.onDataWritable = +function sp_datawrite (request, ctxt, ostream, offset, count) +{ + //dd ("StreamProvider.prototype.onDataWritable"); + + if ("isClosed" in this && this.isClosed) + throw Components.results.NS_BASE_STREAM_CLOSED; + + if (!this.pendingData) + { + this.isBlocked = true; + + /* this is here to support pre-XPCDOM builds (0.9.0 era), which + * don't have this result code mapped. */ + if (!Components.results.NS_BASE_STREAM_WOULD_BLOCK) + throw 2152136711; + + throw Components.results.NS_BASE_STREAM_WOULD_BLOCK; + } + + var len = ostream.write (this.pendingData, this.pendingData.length); + this.pendingData = this.pendingData.substr (len); +} + +StreamProvider.prototype.onStartRequest = +function sp_startreq (request, ctxt) +{ + //dd ("StreamProvider::onStartRequest: " + request + ", " + ctxt); +} + + +StreamProvider.prototype.onStopRequest = +function sp_stopreq (request, ctxt, status) +{ + //dd ("StreamProvider::onStopRequest: " + request + ", " + ctxt + ", " + + // status); + if (this._observer) + this._observer.onStreamClose(status); +} + +function StreamListener(observer) +{ + this._observer = observer; +} + +StreamListener.prototype.onStartRequest = +function sl_startreq (request, ctxt) +{ + //dd ("StreamListener::onStartRequest: " + request + ", " + ctxt); +} + +StreamListener.prototype.onStopRequest = +function sl_stopreq (request, ctxt, status) +{ + //dd ("StreamListener::onStopRequest: " + request + ", " + ctxt + ", " + + //status); + if (this._observer) + this._observer.onStreamClose(status); +} + +StreamListener.prototype.onDataAvailable = +function sl_dataavail (request, ctxt, inStr, sourceOffset, count) +{ + ctxt = ctxt.wrappedJSObject; + if (!ctxt) + { + dd ("*** Can't get wrappedJSObject from ctxt in " + + "StreamListener.onDataAvailable ***"); + return; + } + + if (!("_sInputStream" in ctxt)) + ctxt._sInputStream = toSInputStream(inStr, false); + + if (this._observer) + this._observer.onStreamDataAvailable(request, inStr, sourceOffset, + count); +} + +function SocketListener(connection, observer) +{ + this._connection = connection; + this._observer = observer; +} + +SocketListener.prototype.onSocketAccepted = +function sl_onSocketAccepted(socket, transport) +{ + this._observer.onSocketAccepted(socket, transport); +} +SocketListener.prototype.onStopListening = +function sl_onStopListening(socket, status) +{ + delete this._connection._serverSockListener; + delete this._connection._serverSock; +} diff --git a/comm/suite/chatzilla/js/lib/dcc.js b/comm/suite/chatzilla/js/lib/dcc.js new file mode 100644 index 0000000000..d12b8f1a52 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/dcc.js @@ -0,0 +1,1198 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +/* We pick a random start ID, from 0 to DCC_ID_MAX inclusive, then go through + * the IDs one at a time, in sequence. We wrap when we get to DCC_ID_MAX. No + * uniqueness checking is done, but it takes DCC_ID_MAX connections before we + * hit the start ID again. 65,536 IDs ought to be enough for now. :) + */ +const DCC_ID_MAX = 0xFFFF; + +function CIRCDCC(parent) +{ + this.parent = parent; + + this.users = new Object(); + this.chats = new Array(); + this.files = new Array(); + this.last = null; + this.lastTime = null; + + this.sendChunk = 4096; + this.maxUnAcked = 32; // 4096 * 32 == 128KiB 'in transit'. + + this.requestTimeout = 3 * 60 * 1000; // 3 minutes. + + // Can't do anything 'til this is set! + this.localIPlist = new Array(); + this.localIP = null; + this._lastPort = null; + + try { + var dnsComp = Components.classes["@mozilla.org/network/dns-service;1"]; + this._dnsSvc = dnsComp.getService(Components.interfaces.nsIDNSService); + + // Get local hostname. + if ("myHostName" in this._dnsSvc) { + // Using newer (1.7a+) version with DNS re-write. + this.addHost(this._dnsSvc.myHostName); + } + if ("myIPAddress" in this._dnsSvc) { + // Older Mozilla, have to use this method. + this.addIP(this._dnsSvc.myIPAddress); + } + this.addHost("localhost"); + } catch(ex) { + // what to do? + dd("Error getting local IPs: " + ex); + } + + this._lastID = Math.round(Math.random() * DCC_ID_MAX); + + return this; +} + +CIRCDCC.prototype.TYPE = "IRCDCC"; +CIRCDCC.prototype.listenPorts = []; + +CIRCDCC.prototype.addUser = +function dcc_adduser(user, remoteIP) +{ + // user == CIRCUser object. + // remoteIP == remoteIP as specified in CTCP DCC message. + return new CIRCDCCUser(this, user, remoteIP); +} + +CIRCDCC.prototype.addChat = +function dcc_addchat(user, port) +{ + // user == CIRCDCCUser object. + // port == port as specified in CTCP DCC message. + return new CIRCDCCChat(this, user, port); +} + +CIRCDCC.prototype.addFileTransfer = +function dcc_addfile(user, port, file, size) +{ + return new CIRCDCCFileTransfer(this, user, port, file, size); +} + +CIRCDCC.prototype.addHost = +function dcc_addhost(host, auth) +{ + var me = this; + var listener = { + onLookupComplete: function _onLookupComplete(request, record, status) { + // record == null if it failed. We can't do anything with a failure. + if (record) + { + while (record.hasMore()) + me.addIP(record.getNextAddrAsString(), auth); + } + } + }; + + try { + var th = getService("@mozilla.org/thread-manager;1").currentThread; + var dnsRecord = this._dnsSvc.asyncResolve(host, false, listener, th); + } catch (ex) { + dd("Error resolving host to IP: " + ex); + } +} + +CIRCDCC.prototype.addIP = +function dcc_addip(ip, auth) +{ + if (auth) + this.localIPlist.unshift(ip); + else + this.localIPlist.push(ip); + + if (this.localIPlist.length > 0) + this.localIP = this.localIPlist[0]; +} + +CIRCDCC.prototype.getMatches = +function dcc_getmatches(nickname, filename, types, dirs, states) +{ + function matchNames(name, otherName) + { + return ((name.match(new RegExp(otherName, "i"))) || + (name.toLowerCase().indexOf(otherName.toLowerCase()) != -1)); + }; + + var k; + var list = new Array(); + if (!types) + types = ["chat", "file"]; + + var n = nickname; + var f = filename; + + if (arrayIndexOf(types, "chat") >= 0) + { + for (k = 0; k < this.chats.length; k++) + { + if ((!nickname || matchNames(this.chats[k].user.unicodeName, n)) && + (!dirs || arrayIndexOf(dirs, this.chats[k].state.dir) >= 0) && + (!states || arrayIndexOf(states, this.chats[k].state.state) >= 0)) + { + list.push(this.chats[k]); + } + } + } + if (arrayIndexOf(types, "file") >= 0) + { + for (k = 0; k < this.files.length; k++) + { + if ((!nickname || matchNames(this.files[k].user.unicodeName, n)) && + (!filename || matchNames(this.files[k].filename, f)) && + (!dirs || arrayIndexOf(dirs, this.files[k].state.dir) >= 0) && + (!states || arrayIndexOf(states, this.files[k].state.state) >= 0)) + { + list.push(this.files[k]); + } + } + } + + return list; +} + +CIRCDCC.prototype.getNextPort = +function dcc_getnextport() +{ + var portList = this.listenPorts; + + var newPort = this._lastPort; + + for (var i = 0; i < portList.length; i++) + { + var m = portList[i].match(/^(\d+)(?:-(\d+))?$/); + if (m) + { + if (!newPort) + { + // We dodn't have any previous port, so just take the first we + // find. + return this._lastPort = Number(m[1]); + } + else if (arrayHasElementAt(m, 2)) + { + // Port range. Anything before range, or in [exl. last value] + // is ok. Make sure first value is lowest value returned. + if (newPort < m[2]) + return this._lastPort = Math.max(newPort + 1, Number(m[1])); + } + else + { + // Single port. + if (newPort < m[1]) + return this._lastPort = Number(m[1]); + } + } + } + + // No ports found, and no last port --> use OS. + if (newPort == null) + return -1; + + // Didn't find anything... d'oh. Need to start from the begining again. + this._lastPort = null; + return this.getNextPort(); +} + +CIRCDCC.prototype.getNextID = +function dcc_getnextid() +{ + this._lastID++; + if (this._lastID > DCC_ID_MAX) + this._lastID = 0; + + // Format to DCC_ID_MAX's number of digits. + var id = this._lastID.toString(16); + while (id.length < DCC_ID_MAX.toString(16).length) + id = "0" + id; + return id; +} + +CIRCDCC.prototype.findByID = +function dcc_findbyid(id) +{ + if (typeof id != "string") + return null; + + var i; + for (i = 0; i < this.chats.length; i++) + { + if (this.chats[i].id == id) + return this.chats[i]; + } + for (i = 0; i < this.files.length; i++) + { + if (this.files[i].id == id) + return this.files[i]; + } + return null; +} + + +// JavaScript won't let you delete things declared with "var", workaround: +window.val = -1; + +const DCC_STATE_FAILED = val++; // try connect (accept), but it failed. +const DCC_STATE_INIT = val++; // not "doing" anything +const DCC_STATE_REQUESTED = val++; // waiting +const DCC_STATE_ACCEPTED = val++; // accepted. +const DCC_STATE_DECLINED = val++; // declined. +const DCC_STATE_CONNECTED = val++; // all going ok. +const DCC_STATE_DONE = val++; // finished ok. +const DCC_STATE_ABORTED = val++; // send wasn't accepted in time. + +delete window.val; + +const DCC_DIR_UNKNOWN = 0; +const DCC_DIR_SENDING = 1; +const DCC_DIR_GETTING = 2; + + +function CIRCDCCUser(parent, user, remoteIP) +{ + // user == CIRCUser object. + // remoteIP == remoteIP as specified in CTCP DCC message. + + if ("dccUser" in user) + { + if (remoteIP) + user.dccUser.remoteIP = remoteIP; + + return user.dccUser; + } + + this.parent = parent; + this.netUser = user; + this.id = parent.getNextID(); + this.unicodeName = user.unicodeName; + this.viewName = user.unicodeName; + this.canonicalName = user.collectionKey.substr(1); + this.remoteIP = remoteIP; + + this.key = escape(user.collectionKey) + ":" + remoteIP; + user.dccUser = this; + this.parent.users[this.key] = this; + + if ("onInit" in this) + this.onInit(); + + return this; +} + +CIRCDCCUser.prototype.TYPE = "IRCDCCUser"; + + +// Keeps track of the state of a DCC connection. +function CIRCDCCState(parent, owner, eventType) +{ + // parent == central CIRCDCC object. + // owner == DCC Chat or File object. + // eventType == "dcc-chat" or "dcc-file". + + this.parent = parent; + this.owner = owner; + this.eventType = eventType; + + this.eventPump = owner.eventPump; + + this.state = DCC_STATE_INIT; + this.dir = DCC_DIR_UNKNOWN; + + return this; +} + +CIRCDCCState.prototype.TYPE = "IRCDCCState"; + +CIRCDCCState.prototype.sendRequest = +function dccstate_sendRequest() +{ + if (!this.parent.localIP || (this.state != DCC_STATE_INIT)) + throw "Must have a local IP and be in INIT state."; + + this.state = DCC_STATE_REQUESTED; + this.dir = DCC_DIR_SENDING; + this.requested = new Date(); + + this.requestTimeout = setTimeout(function (o){ o.abort(); }, + this.parent.requestTimeout, this.owner); + + this.eventPump.addEvent(new CEvent(this.eventType, "request", + this.owner, "onRequest")); +} + +CIRCDCCState.prototype.getRequest = +function dccstate_getRequest() +{ + if (this.state != DCC_STATE_INIT) + throw "Must be in INIT state."; + + this.state = DCC_STATE_REQUESTED; + this.dir = DCC_DIR_GETTING; + this.requested = new Date(); + + this.parent.last = this.owner; + this.parent.lastTime = new Date(); + + this.requestTimeout = setTimeout(function (o){ o.abort(); }, + this.parent.requestTimeout, this.owner); + + this.eventPump.addEvent(new CEvent(this.eventType, "request", + this.owner, "onRequest")); +} + +CIRCDCCState.prototype.sendAccept = +function dccstate_sendAccept() +{ + if ((this.state != DCC_STATE_REQUESTED) || (this.dir != DCC_DIR_GETTING)) + throw "Must be in REQUESTED state and direction GET."; + + // Clear out "last" incoming request if that's us. + if (this.parent.last == this.owner) + { + this.parent.last = null; + this.parent.lastTime = null; + } + + clearTimeout(this.requestTimeout); + delete this.requestTimeout; + + this.state = DCC_STATE_ACCEPTED; + this.accepted = new Date(); + + this.eventPump.addEvent(new CEvent(this.eventType, "accept", + this.owner, "onAccept")); +} + +CIRCDCCState.prototype.getAccept = +function dccstate_getAccept() +{ + if ((this.state != DCC_STATE_REQUESTED) || (this.dir != DCC_DIR_SENDING)) + throw "Must be in REQUESTED state and direction SEND."; + + clearTimeout(this.requestTimeout); + delete this.requestTimeout; + + this.state = DCC_STATE_ACCEPTED; + this.accepted = new Date(); + + this.eventPump.addEvent(new CEvent(this.eventType, "accept", + this.owner, "onAccept")); +} + +CIRCDCCState.prototype.sendDecline = +function dccstate_sendDecline() +{ + if ((this.state != DCC_STATE_REQUESTED) || (this.dir != DCC_DIR_GETTING)) + throw "Must be in REQUESTED state and direction GET."; + + // Clear out "last" incoming request if that's us. + if (this.parent.last == this.owner) + { + this.parent.last = null; + this.parent.lastTime = null; + } + + clearTimeout(this.requestTimeout); + delete this.requestTimeout; + + this.state = DCC_STATE_DECLINED; + this.declined = new Date(); + + this.eventPump.addEvent(new CEvent(this.eventType, "decline", + this.owner, "onDecline")); +} + +CIRCDCCState.prototype.getDecline = +function dccstate_getDecline() +{ + if ((this.state != DCC_STATE_REQUESTED) || (this.dir != DCC_DIR_SENDING)) + throw "Must be in REQUESTED state and direction SEND."; + + clearTimeout(this.requestTimeout); + delete this.requestTimeout; + + this.state = DCC_STATE_DECLINED; + this.declined = new Date(); + + this.eventPump.addEvent(new CEvent(this.eventType, "decline", + this.owner, "onDecline")); +} + +// The sockets connected. +CIRCDCCState.prototype.socketConnected = +function dccstate_socketConnected() +{ + if (this.state != DCC_STATE_ACCEPTED) + throw "Not in ACCEPTED state."; + + this.state = DCC_STATE_CONNECTED; + + this.eventPump.addEvent(new CEvent(this.eventType, "connect", + this.owner, "onConnect")); +} + +// Someone disconnected something. +CIRCDCCState.prototype.socketDisconnected = +function dccstate_socketDisconnected() +{ + if (this.state != DCC_STATE_CONNECTED) + throw "Not CONNECTED!"; + + this.state = DCC_STATE_DONE; + + this.eventPump.addEvent(new CEvent(this.eventType, "disconnect", + this.owner, "onDisconnect")); +} + +CIRCDCCState.prototype.sendAbort = +function dccstate_sendAbort() +{ + if ((this.state != DCC_STATE_REQUESTED) && + (this.state != DCC_STATE_ACCEPTED) && + (this.state != DCC_STATE_CONNECTED)) + { + throw "Can't abort at this point."; + } + + this.state = DCC_STATE_ABORTED; + + this.eventPump.addEvent(new CEvent(this.eventType, "abort", + this.owner, "onAbort")); +} + +CIRCDCCState.prototype.getAbort = +function dccstate_getAbort() +{ + if ((this.state != DCC_STATE_REQUESTED) && + (this.state != DCC_STATE_ACCEPTED) && + (this.state != DCC_STATE_CONNECTED)) + { + throw "Can't abort at this point."; + } + + this.state = DCC_STATE_ABORTED; + + this.eventPump.addEvent(new CEvent(this.eventType, "abort", + this.owner, "onAbort")); +} + +CIRCDCCState.prototype.failed = +function dccstate_failed() +{ + if ((this.state != DCC_STATE_REQUESTED) && + (this.state != DCC_STATE_ACCEPTED) && + (this.state != DCC_STATE_CONNECTED)) + { + throw "Can't fail at this point."; + } + + this.state = DCC_STATE_FAILED; + + this.eventPump.addEvent(new CEvent(this.eventType, "fail", + this.owner, "onFail")); +} + + + + +function CIRCDCCChat(parent, user, port) +{ + // user == CIRCDCCUser object. + // port == port as specified in CTCP DCC message. + + this.READ_TIMEOUT = 50; + + // Link up all our data. + this.parent = parent; + this.id = parent.getNextID(); + this.eventPump = parent.parent.eventPump; + this.user = user; + this.localIP = this.parent.localIP; + this.remoteIP = user.remoteIP; + this.port = port; + this.unicodeName = user.unicodeName; + this.viewName = "DCC: " + user.unicodeName; + + // Set up the initial state. + this.requested = null; + this.connection = null; + this.savedLine = ""; + this.state = new CIRCDCCState(parent, this, "dcc-chat"); + this.parent.chats.push(this); + + // Give ourselves a "me" object for the purposes of displaying stuff. + this.me = this.parent.addUser(this.user.netUser.parent.me, "0.0.0.0"); + + if ("onInit" in this) + this.onInit(); + + return this; +} + +CIRCDCCChat.prototype.TYPE = "IRCDCCChat"; + +CIRCDCCChat.prototype.getURL = +function dchat_geturl() +{ + return "x-irc-dcc-chat:" + this.id; +} + +CIRCDCCChat.prototype.isActive = +function dchat_isactive() +{ + return (this.state.state == DCC_STATE_REQUESTED) || + (this.state.state == DCC_STATE_ACCEPTED) || + (this.state.state == DCC_STATE_CONNECTED); +} + +// Call to make this end request DCC Chat with targeted user. +CIRCDCCChat.prototype.request = +function dchat_request() +{ + this.state.sendRequest(); + + this.localIP = this.parent.localIP; + + this.connection = new CBSConnection(); + if (!this.connection.listen(this.port, this)) + { + this.state.failed(); + return false; + } + + this.port = this.connection.port; + + // Send the CTCP DCC request via the net user (CIRCUser object). + var ipNumber; + var ipParts = this.localIP.match(/(\d+)\.(\d+)\.(\d+)\.(\d+)/); + if (ipParts) + ipNumber = Number(ipParts[1]) * 256 * 256 * 256 + + Number(ipParts[2]) * 256 * 256 + + Number(ipParts[3]) * 256 + + Number(ipParts[4]); + else + return false; + // What should we do here? Panic? + + this.user.netUser.ctcp("DCC", "CHAT chat " + ipNumber + " " + this.port); + + return true; +} + +// Call to make this end accept DCC Chat with target user. +CIRCDCCChat.prototype.accept = +function dchat_accept() +{ + this.state.sendAccept(); + + this.connection = new CBSConnection(); + this.connection.connect(this.remoteIP, this.port, null, this); + + return true; +} + +// This may be called synchronously or asynchronously by CBSConnection.connect. +CIRCDCCChat.prototype.onSocketConnection = +function dchat_onsocketconnection(host, port, config, exception) +{ + if (!exception) + { + this.state.socketConnected(); + + this.connection.startAsyncRead(this); + } + else + { + this.state.failed(); + } +} + +// Call to make this end decline DCC Chat with target user. +CIRCDCCChat.prototype.decline = +function dchat_decline() +{ + this.state.sendDecline(); + + // Tell the other end, if they care, that we refused. + this.user.netUser.ctcp("DCC", "REJECT CHAT chat"); + + return true; +} + +// Call to close the connection. +CIRCDCCChat.prototype.disconnect = +function dchat_disconnect() +{ + this.connection.disconnect(); + + return true; +} + +// Aborts the connection. +CIRCDCCChat.prototype.abort = +function dchat_abort() +{ + if (this.state.state == DCC_STATE_CONNECTED) + { + this.disconnect(); + return; + } + + this.state.sendAbort(); + + if (this.connection) + this.connection.close(); +} + +// Event to handle a request from the target user. +// CIRCUser points the event here. +CIRCDCCChat.prototype.onGotRequest = +function dchat_onGotRequest(e) +{ + this.state.getRequest(); + + // Pass over to the base user. + e.destObject = this.user.netUser; +} + +// Event to handle a client connecting to the listening socket. +// CBSConnection points the event here. +CIRCDCCChat.prototype.onSocketAccepted = +function dchat_onSocketAccepted(socket, transport) +{ + this.state.getAccept(); + + this.connection.accept(transport, null); + + this.state.socketConnected(); + + this.remoteIP = transport.host; + + // Start the reading! + this.connection.startAsyncRead(this); +} + +CIRCDCCChat.prototype.onStreamDataAvailable = +function dchat_sda(request, inStream, sourceOffset, count) +{ + var ev = new CEvent("dcc-chat", "data-available", this, "onDataAvailable"); + ev.line = this.connection.readData(0, count); + this.eventPump.routeEvent(ev); +} + +CIRCDCCChat.prototype.onStreamClose = +function dchat_sockdiscon(status) +{ + this.state.socketDisconnected(); + + //var ev = new CEvent("dcc-chat", "disconnect", this, "onDisconnect"); + //ev.server = this; + //ev.disconnectStatus = status; + //this.eventPump.addEvent(ev); +} + +CIRCDCCChat.prototype.onDataAvailable = +function dchat_dataavailable(e) +{ + var line = e.line; + + var incomplete = (line[line.length] != '\n'); + var lines = line.split("\n"); + + if (this.savedLine) + { + lines[0] = this.savedLine + lines[0]; + this.savedLine = ""; + } + + if (incomplete) + this.savedLine = lines.pop(); + + for (var i in lines) + { + var ev = new CEvent("dcc-chat", "rawdata", this, "onRawData"); + ev.data = lines[i]; + ev.replyTo = this; + this.eventPump.addEvent (ev); + } + + return true; +} + +// Raw data from DCC Chat stream. +CIRCDCCChat.prototype.onRawData = +function dchat_rawdata(e) +{ + e.code = "PRIVMSG"; + e.line = e.data; + e.user = this.user; + e.type = "parseddata"; + e.destObject = this; + e.destMethod = "onParsedData"; + + return true; +} + +CIRCDCCChat.prototype.onParsedData = +function dchat_onParsedData(e) +{ + e.type = e.code.toLowerCase(); + if (!e.code[0]) + { + dd (dumpObjectTree (e)); + return false; + } + + if (e.line.search(/\x01.*\x01/i) != -1) { + e.type = "ctcp"; + e.destMethod = "onCTCP"; + e.set = "dcc-chat"; + e.destObject = this; + } + else + { + e.type = "privmsg"; + e.destMethod = "onPrivmsg"; + e.set = "dcc-chat"; + e.destObject = this; + } + + // Allow DCC Chat to handle it before "falling back" to DCC User. + if (typeof this[e.destMethod] == "function") + e.destObject = this; + else if (typeof this.user[e.destMethod] == "function") + e.destObject = this.user; + else if (typeof this["onUnknown"] == "function") + e.destMethod = "onUnknown"; + else if (typeof this.parent[e.destMethod] == "function") + { + e.set = "dcc"; + e.destObject = this.parent; + } + else + { + e.set = "dcc"; + e.destObject = this.parent; + e.destMethod = "onUnknown"; + } + + return true; +} + +CIRCDCCChat.prototype.onCTCP = +function serv_ctcp(e) +{ + // The \x0D? is a BIG HACK to make this work with X-Chat. + var ary = e.line.match(/^\x01([^ ]+) ?(.*)\x01\x0D?$/i); + if (ary == null) + return false; + + e.CTCPData = ary[2] ? ary[2] : ""; + e.CTCPCode = ary[1].toLowerCase(); + if (e.CTCPCode.search(/^reply/i) == 0) + { + dd("dropping spoofed reply."); + return false; + } + + e.CTCPCode = toUnicode(e.CTCPCode, e.replyTo); + e.CTCPData = toUnicode(e.CTCPData, e.replyTo); + + e.type = "ctcp-" + e.CTCPCode; + e.destMethod = "onCTCP" + ary[1][0].toUpperCase() + + ary[1].substr(1, ary[1].length).toLowerCase(); + + if (typeof this[e.destMethod] != "function") + { + e.destObject = e.user; + e.set = "dcc-user"; + if (typeof e.user[e.destMethod] != "function") + { + e.type = "unk-ctcp"; + e.destMethod = "onUnknownCTCP"; + } + } + else + { + e.destObject = this; + } + return true; +} + +CIRCDCCChat.prototype.ctcp = +function dchat_ctcp(code, msg) +{ + msg = msg || ""; + + this.connection.sendData("\x01" + fromUnicode(code, this) + " " + + fromUnicode(msg, this) + "\x01\n"); +} + +CIRCDCCChat.prototype.say = +function dchat_say (msg) +{ + this.connection.sendData(fromUnicode(msg, this) + "\n"); +} + +CIRCDCCChat.prototype.act = +function dchat_act(msg) +{ + this.ctcp("ACTION", msg); +} + + +function CIRCDCCFileTransfer(parent, user, port, file, size) +{ + // user == CIRCDCCUser object. + // port == port as specified in CTCP DCC message. + // file == name of file being sent/got. + // size == size of said file. + + this.READ_TIMEOUT = 50; + + // Link up all our data. + this.parent = parent; + this.id = parent.getNextID(); + this.eventPump = parent.parent.eventPump; + this.user = user; + this.localIP = this.parent.localIP; + this.remoteIP = user.remoteIP; + this.port = port; + this.filename = file; + this.size = size; + this.unicodeName = user.unicodeName; + this.viewName = "File: " + this.filename; + + // Set up the initial state. + this.requested = null; + this.connection = null; + this.state = new CIRCDCCState(parent, this, "dcc-file"); + this.parent.files.push(this); + + // Give ourselves a "me" object for the purposes of displaying stuff. + this.me = this.parent.addUser(this.user.netUser.parent.me, "0.0.0.0"); + + if ("onInit" in this) + this.onInit(); + + return this; +} + +CIRCDCCFileTransfer.prototype.TYPE = "IRCDCCFileTransfer"; + +CIRCDCCFileTransfer.prototype.getURL = +function dfile_geturl() +{ + return "x-irc-dcc-file:" + this.id; +} + +CIRCDCCFileTransfer.prototype.isActive = +function dfile_isactive() +{ + return (this.state.state == DCC_STATE_REQUESTED) || + (this.state.state == DCC_STATE_ACCEPTED) || + (this.state.state == DCC_STATE_CONNECTED); +} + +CIRCDCCFileTransfer.prototype.dispose = +function dfile_dispose() +{ + if (this.connection) + { + // close is for the server socket, disconnect for the client socket. + this.connection.close(); + this.connection.disconnect(); + } + + if (this.localFile) + this.localFile.close(); + + this.connection = null; + this.localFile = null; + this.filestream = null; +} + +// Call to make this end offer DCC File to targeted user. +CIRCDCCFileTransfer.prototype.request = +function dfile_request(localFile) +{ + this.state.sendRequest(); + + this.localFile = new LocalFile(localFile, "<"); + this.filename = localFile.leafName; + this.size = localFile.fileSize; + + // Update view name. + this.viewName = "File: " + this.filename; + + // Double-quote file names with spaces. + // FIXME: Do we need any better checking? + if (this.filename.match(/ /)) + this.filename = '"' + this.filename + '"'; + + this.localIP = this.parent.localIP; + + this.connection = new CBSConnection(true); + if (!this.connection.listen(this.port, this)) + { + this.state.failed(); + this.dispose(); + return false; + } + + this.port = this.connection.port; + + // Send the CTCP DCC request via the net user (CIRCUser object). + var ipNumber; + var ipParts = this.localIP.match(/(\d+)\.(\d+)\.(\d+)\.(\d+)/); + if (ipParts) + ipNumber = Number(ipParts[1]) * 256 * 256 * 256 + + Number(ipParts[2]) * 256 * 256 + + Number(ipParts[3]) * 256 + + Number(ipParts[4]); + else + return false; + // What should we do here? Panic? + + this.user.netUser.ctcp("DCC", "SEND " + this.filename + " " + + ipNumber + " " + this.port + " " + this.size); + + return true; +} + +// Call to make this end accept DCC File from target user. +CIRCDCCFileTransfer.prototype.accept = +function dfile_accept(localFile) +{ + const nsIBinaryOutputStream = Components.interfaces.nsIBinaryOutputStream; + + this.state.sendAccept(); + + this.localFile = new LocalFile(localFile, ">"); + this.localPath = localFile.path; + + this.filestream = Components.classes["@mozilla.org/binaryoutputstream;1"]; + this.filestream = this.filestream.createInstance(nsIBinaryOutputStream); + this.filestream.setOutputStream(this.localFile.outputStream); + + this.position = 0; + this.connection = new CBSConnection(true); + this.connection.connect(this.remoteIP, this.port, null, this); + + return true; +} + +// This may be called synchronously or asynchronously by CBSConnection.connect. +CIRCDCCFileTransfer.prototype.onSocketConnection = +function dfile_onsocketconnection(host, port, config, exception) +{ + if (!exception) + { + this.state.socketConnected(); + + this.connection.startAsyncRead(this); + } + else + { + this.state.failed(); + this.dispose(); + } +} + +// Call to make this end decline DCC File from target user. +CIRCDCCFileTransfer.prototype.decline = +function dfile_decline() +{ + this.state.sendDecline(); + + // Tell the other end, if they care, that we refused. + this.user.netUser.ctcp("DCC", "REJECT FILE " + this.filename); + + return true; +} + +// Call to close the connection. +CIRCDCCFileTransfer.prototype.disconnect = +function dfile_disconnect() +{ + this.dispose(); + + return true; +} + +// Aborts the connection. +CIRCDCCFileTransfer.prototype.abort = +function dfile_abort() +{ + if (this.state.state == DCC_STATE_CONNECTED) + { + this.disconnect(); + return; + } + + this.state.sendAbort(); + this.dispose(); +} + +// Event to handle a request from the target user. +// CIRCUser points the event here. +CIRCDCCFileTransfer.prototype.onGotRequest = +function dfile_onGotRequest(e) +{ + this.state.getRequest(); + + // Pass over to the base user. + e.destObject = this.user.netUser; +} + +// Event to handle a client connecting to the listening socket. +// CBSConnection points the event here. +CIRCDCCFileTransfer.prototype.onSocketAccepted = +function dfile_onSocketAccepted(socket, transport) +{ + this.state.getAccept(); + + this.connection.accept(transport, null); + + this.state.socketConnected(); + + this.position = 0; + this.ackPosition = 0; + this.remoteIP = transport.host; + + this.eventPump.addEvent(new CEvent("dcc-file", "connect", + this, "onConnect")); + + try { + this.filestream = Components.classes["@mozilla.org/binaryinputstream;1"]; + this.filestream = this.filestream.createInstance(nsIBinaryInputStream); + this.filestream.setInputStream(this.localFile.baseInputStream); + + // Start the reading! + var d; + if (this.parent.sendChunk > this.size) + d = this.filestream.readBytes(this.size); + else + d = this.filestream.readBytes(this.parent.sendChunk); + this.position += d.length; + this.connection.sendData(d); + } + catch(ex) + { + dd(ex); + } + + // Start the reading! + this.connection.startAsyncRead(this); +} + +CIRCDCCFileTransfer.prototype.onStreamDataAvailable = +function dfile_sda(request, inStream, sourceOffset, count) +{ + var ev = new CEvent("dcc-file", "data-available", this, "onDataAvailable"); + ev.data = this.connection.readData(0, count); + this.eventPump.routeEvent(ev); +} + +CIRCDCCFileTransfer.prototype.onStreamClose = +function dfile_sockdiscon(status) +{ + if (this.position != this.size) + this.state.failed(); + else + this.state.socketDisconnected(); + this.dispose(); +} + +CIRCDCCFileTransfer.prototype.onDataAvailable = +function dfile_dataavailable(e) +{ + e.type = "rawdata"; + e.destMethod = "onRawData"; + + try + { + if (this.state.dir == DCC_DIR_SENDING) + { + while (e.data.length >= 4) + { + var word = e.data.substr(0, 4); + e.data = e.data.substr(4, e.data.length - 4); + var pos = word.charCodeAt(0) * 0x01000000 + + word.charCodeAt(1) * 0x00010000 + + word.charCodeAt(2) * 0x00000100 + + word.charCodeAt(3) * 0x00000001; + this.ackPosition = pos; + } + + while (this.position <= this.ackPosition + this.parent.maxUnAcked) + { + var d; + if (this.position + this.parent.sendChunk > this.size) + d = this.filestream.readBytes(this.size - this.position); + else + d = this.filestream.readBytes(this.parent.sendChunk); + + this.position += d.length; + this.connection.sendData(d); + + if (this.position >= this.size) + { + this.dispose(); + break; + } + } + } + else if (this.state.dir == DCC_DIR_GETTING) + { + this.filestream.writeBytes(e.data, e.data.length); + + // Send back ack data. + this.position += e.data.length; + var bytes = new Array(); + for (var i = 0; i < 4; i++) + bytes.push(Math.floor(this.position / Math.pow(256, i)) & 255); + + this.connection.sendData(String.fromCharCode(bytes[3], bytes[2], + bytes[1], bytes[0])); + + if (this.size && (this.position >= this.size)) + this.disconnect(); + } + this.eventPump.addEvent(new CEvent("dcc-file", "progress", + this, "onProgress")); + } + catch(ex) + { + this.disconnect(); + } + + return true; +} + +CIRCDCCFileTransfer.prototype.sendData = +function dfile_say(data) +{ + this.connection.sendData(data); +} + +CIRCDCCFileTransfer.prototype.size = 0; +CIRCDCCFileTransfer.prototype.position = 0; +CIRCDCCFileTransfer.prototype.__defineGetter__("progress", + function dfile_get_progress() + { + if (this.size > 0) + return Math.floor(100 * this.position / this.size); + return 0; + }); + diff --git a/comm/suite/chatzilla/js/lib/events.js b/comm/suite/chatzilla/js/lib/events.js new file mode 100644 index 0000000000..b48de11ede --- /dev/null +++ b/comm/suite/chatzilla/js/lib/events.js @@ -0,0 +1,365 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +/** + * Event class for |CEventPump|. + */ +function CEvent (set, type, destObject, destMethod) +{ + this.set = set; + this.type = type; + this.destObject = destObject; + this.destMethod = destMethod; + this.hooks = new Array(); + +} + +/** + * The event pump keeps a queue of pending events, processing them on-demand. + * + * You should never need to create an instance of this prototype; access the + * event pump through |client.eventPump|. Most code should only need to use the + * |addHook|, |getHook| and |removeHookByName| methods. + */ +function CEventPump (eventsPerStep) +{ + /* event routing stops after this many levels, safety valve */ + this.MAX_EVENT_DEPTH = 50; + /* When there are this many 'used' items in a queue, always clean up. At + * this point it is MUCH more effecient to remove a block than a single + * item (i.e. removing 1000 is much much faster than removing 1 item 1000 + * times [1]). + */ + this.FORCE_CLEANUP_PTR = 1000; + /* If there are less than this many items in a queue, clean up. This keeps + * the queue empty normally, and is not that ineffecient [1]. + */ + this.MAX_AUTO_CLEANUP_LEN = 100; + this.eventsPerStep = eventsPerStep; + this.queue = new Array(); + this.queuePointer = 0; + this.bulkQueue = new Array(); + this.bulkQueuePointer = 0; + this.hooks = new Array(); + + /* [1] The delay when removing items from an array (with unshift or splice, + * and probably most operations) is NOT perportional to the number of items + * being removed, instead it is proportional to the number of items LEFT. + * Because of this, it is better to only remove small numbers of items when + * the queue is small (MAX_AUTO_CLEANUP_LEN), and when it is large remove + * only large chunks at a time (FORCE_CLEANUP_PTR), reducing the number of + * resizes being done. + */ +} + +CEventPump.prototype.onHook = +function ep_hook(e, hooks) +{ + var h; + + if (typeof hooks == "undefined") + hooks = this.hooks; + + hook_loop: + for (h = hooks.length - 1; h >= 0; h--) + { + if (!hooks[h].enabled || + !matchObject (e, hooks[h].pattern, hooks[h].neg)) + continue hook_loop; + + e.hooks.push(hooks[h]); + try + { + var rv = hooks[h].f(e); + } + catch(ex) + { + dd("hook #" + h + " '" + + ((typeof hooks[h].name != "undefined") ? hooks[h].name : + "") + "' had an error!"); + dd(formatException(ex)); + } + if ((typeof rv == "boolean") && + (rv == false)) + { + dd("hook #" + h + " '" + + ((typeof hooks[h].name != "undefined") ? hooks[h].name : + "") + "' stopped hook processing."); + return true; + } + } + + return false; +} + +/** + * Adds an event hook to be called when matching events are processed. + * + * All hooks should be given a meaningful name, to aid removal and debugging. + * For plugins, an ideal technique for the name is to use |plugin.id| as a + * prefix (e.g. <tt>plugin.id + "-my-super-hook"</tt>). + * + * @param f The function to call when an event matches |pattern|. + * @param name A unique name for the hook. Used for removing the hook and + * debugging. + * @param neg Optional. If specified with a |true| value, the hook will be + * called for events *not* matching |pattern|. Otherwise, the hook + * will be called for events matching |pattern|. + * @param enabled Optional. If specified, sets the initial enabled/disabled + * state of the hook. By default, hooks are enabled. See + * |getHook|. + * @param hooks Internal. Do not use. + */ +CEventPump.prototype.addHook = +function ep_addhook(pattern, f, name, neg, enabled, hooks) +{ + if (typeof hooks == "undefined") + hooks = this.hooks; + + if (typeof f != "function") + return false; + + if (typeof enabled == "undefined") + enabled = true; + else + enabled = Boolean(enabled); + + neg = Boolean(neg); + + var hook = { + pattern: pattern, + f: f, + name: name, + neg: neg, + enabled: enabled + }; + + hooks.push(hook); + + return hook; + +} + +/** + * Finds and returns data about a named event hook. + * + * You can use |getHook| to change the enabled state of an existing event hook: + * <tt>client.eventPump.getHook(myHookName).enabled = false;</tt> + * <tt>client.eventPump.getHook(myHookName).enabled = true;</tt> + * + * @param name The unique hook name to find and return data about. + * @param hooks Internal. Do not use. + * @returns If a match is found, an |Object| with properties matching the + * arguments to |addHook| is returned. Otherwise, |null| is returned. + */ +CEventPump.prototype.getHook = +function ep_gethook(name, hooks) +{ + if (typeof hooks == "undefined") + hooks = this.hooks; + + for (var h in hooks) + if (hooks[h].name.toLowerCase() == name.toLowerCase()) + return hooks[h]; + + return null; + +} + +/** + * Removes an existing event hook by its name. + * + * @param name The unique hook name to find and remove. + * @param hooks Internal. Do not use. + * @returns |true| if the hook was found and removed, |false| otherwise. + */ +CEventPump.prototype.removeHookByName = +function ep_remhookname(name, hooks) +{ + if (typeof hooks == "undefined") + hooks = this.hooks; + + for (var h in hooks) + if (hooks[h].name.toLowerCase() == name.toLowerCase()) + { + arrayRemoveAt (hooks, h); + return true; + } + + return false; + +} + +CEventPump.prototype.removeHookByIndex = +function ep_remhooki(idx, hooks) +{ + if (typeof hooks == "undefined") + hooks = this.hooks; + + return arrayRemoveAt (hooks, idx); + +} + +CEventPump.prototype.addEvent = +function ep_addevent (e) +{ + e.queuedAt = new Date(); + this.queue.push(e); + return true; +} + +CEventPump.prototype.addBulkEvent = +function ep_addevent (e) +{ + e.queuedAt = new Date(); + this.bulkQueue.push(e); + return true; +} + +CEventPump.prototype.routeEvent = +function ep_routeevent (e) +{ + var count = 0; + + this.currentEvent = e; + + e.level = 0; + while (e.destObject) + { + e.level++; + this.onHook (e); + var destObject = e.destObject; + e.currentObject = destObject; + e.destObject = (void 0); + + switch (typeof destObject[e.destMethod]) + { + case "function": + if (1) + try + { + destObject[e.destMethod] (e); + } + catch (ex) + { + if (typeof ex == "string") + { + dd ("Error routing event " + e.set + "." + + e.type + ": " + ex); + } + else + { + dd ("Error routing event " + e.set + "." + + e.type + ": " + dumpObjectTree(ex) + + " in " + e.destMethod + "\n" + ex); + if ("stack" in ex) + dd(ex.stack); + } + } + else + destObject[e.destMethod] (e); + + if (count++ > this.MAX_EVENT_DEPTH) + throw "Too many events in chain"; + break; + + case "undefined": + //dd ("** " + e.destMethod + " does not exist."); + break; + + default: + dd ("** " + e.destMethod + " is not a function."); + } + + if ((e.type != "event-end") && (!e.destObject)) + { + e.lastSet = e.set; + e.set = "eventpump"; + e.lastType = e.type; + e.type = "event-end"; + e.destMethod = "onEventEnd"; + e.destObject = this; + } + + } + + delete this.currentEvent; + + return true; + +} + +CEventPump.prototype.stepEvents = +function ep_stepevents() +{ + var i = 0; + var st, en, e; + + st = new Date(); + while (i < this.eventsPerStep) + { + if (this.queuePointer >= this.queue.length) + break; + + e = this.queue[this.queuePointer++]; + + if (e.type == "yield") + break; + + this.routeEvent(e); + i++; + } + while (i < this.eventsPerStep) + { + if (this.bulkQueuePointer >= this.bulkQueue.length) + break; + + e = this.bulkQueue[this.bulkQueuePointer++]; + + if (e.type == "yield") + break; + + this.routeEvent(e); + i++; + } + en = new Date(); + + // i == number of items handled this time. + // We only want to do this if we handled at least 25% of our step-limit + // and if we have a sane interval between st and en (not zero). + if ((i * 4 >= this.eventsPerStep) && (en - st > 0)) + { + // Calculate the number of events that can be processed in 400ms. + var newVal = (400 * i) / (en - st); + + // If anything skews it majorly, limit it to a minimum value. + if (newVal < 10) + newVal = 10; + + // Adjust the step-limit based on this "target" limit, but only do a + // 25% change (underflow filter). + this.eventsPerStep += Math.round((newVal - this.eventsPerStep) / 4); + } + + // Clean up if we've handled a lot, or the queue is small. + if ((this.queuePointer >= this.FORCE_CLEANUP_PTR) || + (this.queue.length <= this.MAX_AUTO_CLEANUP_LEN)) + { + this.queue.splice(0, this.queuePointer); + this.queuePointer = 0; + } + + // Clean up if we've handled a lot, or the queue is small. + if ((this.bulkQueuePointer >= this.FORCE_CLEANUP_PTR) || + (this.bulkQueue.length <= this.MAX_AUTO_CLEANUP_LEN)) + { + this.bulkQueue.splice(0, this.bulkQueuePointer); + this.bulkQueuePointer = 0; + } + + return i; + +} diff --git a/comm/suite/chatzilla/js/lib/file-utils.js b/comm/suite/chatzilla/js/lib/file-utils.js new file mode 100644 index 0000000000..d85edac8b6 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/file-utils.js @@ -0,0 +1,430 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +/* notice that these valuse are octal. */ +const PERM_IRWXU = 0o700; /* read, write, execute/search by owner */ +const PERM_IRUSR = 0o400; /* read permission, owner */ +const PERM_IWUSR = 0o200; /* write permission, owner */ +const PERM_IXUSR = 0o100; /* execute/search permission, owner */ +const PERM_IRWXG = 0o070; /* read, write, execute/search by group */ +const PERM_IRGRP = 0o040; /* read permission, group */ +const PERM_IWGRP = 0o020; /* write permission, group */ +const PERM_IXGRP = 0o010; /* execute/search permission, group */ +const PERM_IRWXO = 0o007; /* read, write, execute/search by others */ +const PERM_IROTH = 0o004; /* read permission, others */ +const PERM_IWOTH = 0o002; /* write permission, others */ +const PERM_IXOTH = 0o001; /* execute/search permission, others */ + +const MODE_RDONLY = 0x01; +const MODE_WRONLY = 0x02; +const MODE_RDWR = 0x04; +const MODE_CREATE = 0x08; +const MODE_APPEND = 0x10; +const MODE_TRUNCATE = 0x20; +const MODE_SYNC = 0x40; +const MODE_EXCL = 0x80; + +const PICK_OK = Components.interfaces.nsIFilePicker.returnOK; +const PICK_CANCEL = Components.interfaces.nsIFilePicker.returnCancel; +const PICK_REPLACE = Components.interfaces.nsIFilePicker.returnReplace; + +const FILTER_ALL = Components.interfaces.nsIFilePicker.filterAll; +const FILTER_HTML = Components.interfaces.nsIFilePicker.filterHTML; +const FILTER_TEXT = Components.interfaces.nsIFilePicker.filterText; +const FILTER_IMAGES = Components.interfaces.nsIFilePicker.filterImages; +const FILTER_XML = Components.interfaces.nsIFilePicker.filterXML; +const FILTER_XUL = Components.interfaces.nsIFilePicker.filterXUL; + +const FTYPE_DIR = Components.interfaces.nsIFile.DIRECTORY_TYPE; +const FTYPE_FILE = Components.interfaces.nsIFile.NORMAL_FILE_TYPE; + +// evald f = fopen("/home/rginda/foo.txt", MODE_WRONLY | MODE_CREATE) +// evald f = fopen("/home/rginda/vnk.txt", MODE_RDONLY) + +var futils = new Object(); + +futils.umask = PERM_IWOTH | PERM_IWGRP; +futils.MSG_SAVE_AS = "Save As"; +futils.MSG_OPEN = "Open"; + +/** + * Internal function used by |pickSaveAs|, |pickOpen| and |pickGetFolder|. + * + * @param initialPath (*defaultDir* in |pick| functions) Sets the + * initial directory for the dialog. The user may browse + * to any other directory - it does not restrict anything. + * @param typeList Optional. An |Array| or space-separated string of allowed + * file types for the dialog. An item in the array may be a + * string (used as title and filter) or a two-element array + * (title and filter, respectively); when using a string, + * the following standard filters may be used: |$all|, |$html|, + * |$text|, |$images|, |$xml|, |$xul|, |$noAll| (prevents "All + * Files" filter being included). + * @param attribs Optional. Takes an object with either or both of the + * properties: |defaultString| (*defaultFile* in |pick| + * functions) sets the initial/default filename, and + * |defaultExtension| XXX FIXME (this seems wrong?) XXX. + * @returns An |Object| with |ok| (Boolean), |file| (|nsIFile|) and + * |picker| (|nsIFilePicker|) properties. + */ +futils.getPicker = +function futils_nosepicker(initialPath, typeList, attribs) +{ + const classes = Components.classes; + const interfaces = Components.interfaces; + + const PICKER_CTRID = "@mozilla.org/filepicker;1"; + const LOCALFILE_CTRID = "@mozilla.org/file/local;1"; + + const nsIFilePicker = interfaces.nsIFilePicker; + const nsIFile = interfaces.nsIFile; + + var picker = classes[PICKER_CTRID].createInstance(nsIFilePicker); + if (attribs) + { + if (typeof attribs == "object") + { + for (var a in attribs) + picker[a] = attribs[a]; + } + else + { + throw "bad type for param |attribs|"; + } + } + + if (initialPath) + { + var localFile; + + if (typeof initialPath == "string") + { + localFile = + classes[LOCALFILE_CTRID].createInstance(nsIFile); + localFile.initWithPath(initialPath); + } + else + { + if (!isinstance(initialPath, nsIFile)) + throw "bad type for argument |initialPath|"; + + localFile = initialPath; + } + + picker.displayDirectory = localFile + } + + var allIncluded = false; + + if (typeof typeList == "string") + typeList = typeList.split(" "); + + if (isinstance(typeList, Array)) + { + for (var i in typeList) + { + switch (typeList[i]) + { + case "$all": + allIncluded = true; + picker.appendFilters(FILTER_ALL); + break; + + case "$html": + picker.appendFilters(FILTER_HTML); + break; + + case "$text": + picker.appendFilters(FILTER_TEXT); + break; + + case "$images": + picker.appendFilters(FILTER_IMAGES); + break; + + case "$xml": + picker.appendFilters(FILTER_XML); + break; + + case "$xul": + picker.appendFilters(FILTER_XUL); + break; + + case "$noAll": + // This prevents the automatic addition of "All Files" + // as a file type option by pretending it is already there. + allIncluded = true; + break; + + default: + if ((typeof typeList[i] == "object") && isinstance(typeList[i], Array)) + picker.appendFilter(typeList[i][0], typeList[i][1]); + else + picker.appendFilter(typeList[i], typeList[i]); + break; + } + } + } + + if (!allIncluded) + picker.appendFilters(FILTER_ALL); + + return picker; +} + +function getPickerChoice(picker) +{ + var obj = new Object(); + obj.picker = picker; + obj.ok = false; + obj.file = null; + + try + { + obj.reason = picker.show(); + } + catch (ex) + { + dd ("caught exception from file picker: " + ex); + return obj; + } + + if (obj.reason != PICK_CANCEL) + { + obj.file = picker.file; + obj.ok = true; + } + + return obj; +} + +/** + * Displays a standard file save dialog. + * + * @param title Optional. The title for the dialog. + * @param typeList Optional. See |futils.getPicker| for details. + * @param defaultFile Optional. See |futils.getPicker| for details. + * @param defaultDir Optional. See |futils.getPicker| for details. + * @param defaultExt Optional. See |futils.getPicker| for details. + * @returns An |Object| with "ok" (Boolean), "file" (|nsIFile|) and + * "picker" (|nsIFilePicker|) properties. + */ +function pickSaveAs (title, typeList, defaultFile, defaultDir, defaultExt) +{ + if (!defaultDir && "lastSaveAsDir" in futils) + defaultDir = futils.lastSaveAsDir; + + var picker = futils.getPicker (defaultDir, typeList, + {defaultString: defaultFile, + defaultExtension: defaultExt}); + picker.init (window, title ? title : futils.MSG_SAVE_AS, + Components.interfaces.nsIFilePicker.modeSave); + + var rv = getPickerChoice(picker); + if (rv.ok) + futils.lastSaveAsDir = picker.file.parent; + + return rv; +} + +/** + * Displays a standard file open dialog. + * + * @param title Optional. The title for the dialog. + * @param typeList Optional. See |futils.getPicker| for details. + * @param defaultFile Optional. See |futils.getPicker| for details. + * @param defaultDir Optional. See |futils.getPicker| for details. + * @returns An |Object| with "ok" (Boolean), "file" (|nsIFile|) and + * "picker" (|nsIFilePicker|) properties. + */ +function pickOpen (title, typeList, defaultFile, defaultDir) +{ + if (!defaultDir && "lastOpenDir" in futils) + defaultDir = futils.lastOpenDir; + + var picker = futils.getPicker (defaultDir, typeList, + {defaultString: defaultFile}); + picker.init (window, title ? title : futils.MSG_OPEN, + Components.interfaces.nsIFilePicker.modeOpen); + + var rv = getPickerChoice(picker); + if (rv.ok) + futils.lastOpenDir = picker.file.parent; + + return rv; +} + +/** + * Displays a standard directory selection dialog. + * + * @param title Optional. The title for the dialog. + * @param defaultDir Optional. See |futils.getPicker| for details. + * @returns An |Object| with "ok" (Boolean), "file" (|nsIFile|) and + * "picker" (|nsIFilePicker|) properties. + */ +function pickGetFolder(title, defaultDir) +{ + if (!defaultDir && "lastOpenDir" in futils) + defaultDir = futils.lastOpenDir; + + var picker = futils.getPicker(defaultDir); + picker.init(window, title ? title : futils.MSG_OPEN, + Components.interfaces.nsIFilePicker.modeGetFolder); + + var rv = getPickerChoice(picker); + if (rv.ok) + futils.lastOpenDir = picker.file; + + return rv; +} + +function mkdir (localFile, perms) +{ + if (typeof perms == "undefined") + perms = 0o766 & ~futils.umask; + + localFile.create(FTYPE_DIR, perms); +} + +function getTempFile(path, name) +{ + var tempFile = new nsLocalFile(path); + tempFile.append(name); + tempFile.createUnique(0, 0o600); + return tempFile; +} + +function nsLocalFile(path) +{ + const LOCALFILE_CTRID = "@mozilla.org/file/local;1"; + const nsIFile = Components.interfaces.nsIFile; + + var localFile = + Components.classes[LOCALFILE_CTRID].createInstance(nsIFile); + localFile.initWithPath(path); + return localFile; +} + +function fopen (path, mode, perms, tmp) +{ + return new LocalFile(path, mode, perms, tmp); +} + +function LocalFile(file, mode, perms, tmp) +{ + const classes = Components.classes; + const interfaces = Components.interfaces; + + const LOCALFILE_CTRID = "@mozilla.org/file/local;1"; + const FILEIN_CTRID = "@mozilla.org/network/file-input-stream;1"; + const FILEOUT_CTRID = "@mozilla.org/network/file-output-stream;1"; + const SCRIPTSTREAM_CTRID = "@mozilla.org/scriptableinputstream;1"; + + const nsIFile = interfaces.nsIFile; + const nsIFileOutputStream = interfaces.nsIFileOutputStream; + const nsIFileInputStream = interfaces.nsIFileInputStream; + const nsIScriptableInputStream = interfaces.nsIScriptableInputStream; + + if (typeof perms == "undefined") + perms = 0o666 & ~futils.umask; + + if (typeof mode == "string") + { + switch (mode) + { + case ">": + mode = MODE_WRONLY | MODE_CREATE | MODE_TRUNCATE; + break; + case ">>": + mode = MODE_WRONLY | MODE_CREATE | MODE_APPEND; + break; + case "<": + mode = MODE_RDONLY; + break; + default: + throw "Invalid mode ``" + mode + "''"; + } + } + + if (typeof file == "string") + { + this.localFile = new nsLocalFile(file); + } + else if (isinstance(file, nsIFile)) + { + this.localFile = file; + } + else + { + throw "bad type for argument |file|."; + } + + this.path = this.localFile.path; + + if (mode & (MODE_WRONLY | MODE_RDWR)) + { + this.outputStream = + classes[FILEOUT_CTRID].createInstance(nsIFileOutputStream); + this.outputStream.init(this.localFile, mode, perms, 0); + } + + if (mode & (MODE_RDONLY | MODE_RDWR)) + { + this.baseInputStream = + classes[FILEIN_CTRID].createInstance(nsIFileInputStream); + this.baseInputStream.init(this.localFile, mode, perms, tmp); + this.inputStream = + classes[SCRIPTSTREAM_CTRID].createInstance(nsIScriptableInputStream); + this.inputStream.init(this.baseInputStream); + } +} + +LocalFile.prototype.write = +function fo_write(buf) +{ + if (!("outputStream" in this)) + throw "file not open for writing."; + + return this.outputStream.write(buf, buf.length); +} + +// Will return null if there is no more data in the file. +// Will block until it has some data to return. +// Will return an empty string if there is data, but it couldn't be read. +LocalFile.prototype.read = +function fo_read(max) +{ + if (!("inputStream" in this)) + throw "file not open for reading."; + + if (typeof max == "undefined") + max = this.inputStream.available(); + + try + { + var rv = this.inputStream.read(max); + return (rv != "") ? rv : null; + } + catch (ex) + { + return ""; + } +} + +LocalFile.prototype.close = +function fo_close() +{ + if ("outputStream" in this) + this.outputStream.close(); + if ("inputStream" in this) + this.inputStream.close(); +} + +LocalFile.prototype.flush = +function fo_close() +{ + return this.outputStream.flush(); +} diff --git a/comm/suite/chatzilla/js/lib/http.js b/comm/suite/chatzilla/js/lib/http.js new file mode 100644 index 0000000000..45ec3ddbbc --- /dev/null +++ b/comm/suite/chatzilla/js/lib/http.js @@ -0,0 +1,177 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +function CHTTPDoc (server, path) +{ + + this.GET_TIMEOUT = 2 * 60; /* 2 minute timeout on gets */ + this.state = "new"; + this.server = server; + this.path = path; + +} + +CHTTPDoc.prototype.get = +function http_get (ep) +{ + + this.connection = new CBSConnection(); + if (!this.connection.connect (this.server, 80, (void 0), true)) + { + this.state = "connect-error"; + var e = new CEvent ("httpdoc", "complete", this, "onComplete"); + this.eventPump.addEvent (e); + return false; + } + + this.eventPump = ep; + this.state = "opened"; + this.data = ""; + this.duration = 0; + + var e = new CEvent ("httpdoc", "poll", this, "onPoll"); + this.eventPump.addEvent (e); + + this.connection.sendData ("GET " + this.path + "\n"); + this.startDate = new Date(); + + return true; + +} + +CHTTPDoc.prototype.onPoll = +function http_poll (e) +{ + var line = ""; + var ex, c; + var need_more = false; + + if (this.duration < this.GET_TIMEOUT) + try + { + line = this.connection.readData (50); + need_more = true; + } + catch (ex) + { + if (typeof (ex) == "undefined") + { + line = ""; + need_more = true; + } + else + { + dd ("** Caught exception: '" + ex + "' while receiving " + + this.server + this.path); + this.state = "read-error"; + } + } + else + this.state = "receive-timeout"; + + switch (this.state) + { + case "opened": + case "receive-header": + if (this.state == "opened") + this.headers = ""; + + c = line.search(/\<html\>/i); + if (c != -1) + { + this.data = line.substr (c, line.length); + this.state = "receive-data"; + line = line.substr (0, c); + + c = this.data.search(/\<\/html\>/i); + if (c != -1) + { + this.docType = stringTrim(this.docType); + this.state = "complete"; + need_more = false; + } + + } + + this.headers += line; + c = this.headers.search (/\<\!doctype/i); + if (c != -1) + { + this.docType = this.headers.substr (c, line.length); + if (this.state == "opened") + this.state = "receive-doctype"; + this.headers = this.headers.substr (0, c); + } + + if (this.state == "opened") + this.state = "receive-header"; + break; + + case "receive-doctype": + var c = line.search (/\<html\>/i); + if (c != -1) + { + this.docType = stringTrim(this.docType); + this.data = line.substr (c, line.length); + this.state = "receive-data"; + line = line.substr (0, c); + } + + this.docType += line; + break; + + case "receive-data": + this.data += line; + var c = this.data.search(/\<\/html\>/i); + if (c != -1) + this.state = "complete"; + break; + + case "read-error": + case "receive-timeout": + break; + + default: + dd ("** INVALID STATE in HTTPDoc object (" + this.state + ") **"); + need_more = false; + this.state = "error"; + break; + + } + + if ((this.state != "complete") && (need_more)) + var e = new CEvent ("httpdoc", "poll", this, "onPoll"); + else + { + this.connection.disconnect(); + if (this.data) + { + var m = this.data.match(/\<title\>(.*)\<\/title\>/i); + if (m != null) + this.title = m[1]; + else + this.title = ""; + } + var e = new CEvent ("httpdoc", "complete", this, "onComplete"); + } + + this.eventPump.addEvent (e); + this.duration = (new Date() - this.startDate) / 1000; + + return true; + +} + +CHTTPDoc.prototype.onComplete = +function http_complete(e) +{ + + return true; + +} + + + diff --git a/comm/suite/chatzilla/js/lib/ident.js b/comm/suite/chatzilla/js/lib/ident.js new file mode 100644 index 0000000000..5110d70165 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/ident.js @@ -0,0 +1,203 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +// RFC1413-ish identification server for ChatZilla +// One Ident Server is used for all networks. When multiple networks are in the +// process of connecting, it won't stop listening until they're all done. + +function IdentServer(parent) +{ + this.responses = new Array(); + this.listening = false; + this.dns = getService("@mozilla.org/network/dns-service;1","nsIDNSService"); + + this.parent = parent; + this.eventPump = parent.eventPump; +} + +IdentServer.prototype.start = +function ident_start() +{ + if (this.listening) + return true; + + if (!this.socket) + this.socket = new CBSConnection(); + + if (!this.socket.listen(113, this)) + return false; + + this.listening = true; + return true; +} + +IdentServer.prototype.stop = +function ident_stop() +{ + if (!this.socket || !this.listening) + return; + + // No need to destroy socket, listen() will work again. + this.socket.close(); + this.listening = false; +} + +IdentServer.prototype.addNetwork = +function ident_add(net, serv) +{ + var addr, dnsRecord = this.dns.resolve(serv.hostname, 0); + + while (dnsRecord.hasMore()) + { + addr = dnsRecord.getNextAddrAsString(); + this.responses.push({net: net, ip: addr, port: serv.port, + username: net.INITIAL_NAME || net.INITIAL_NICK}); + } + + if (this.responses.length == 0) + return false; + + // Start the ident server if necessary. + if (!this.listening) + return this.start(); + + return true; +} + +IdentServer.prototype.removeNetwork = +function ident_remove(net) +{ + var newResponses = new Array(); + + for (var i = 0; i < this.responses.length; ++i) + { + if (this.responses[i].net != net) + newResponses.push(this.responses[i]); + } + this.responses = newResponses; + + // Only stop listening if responses is empty - no networks need us anymore. + if (this.responses.length == 0) + this.stop(); +} + +IdentServer.prototype.onSocketAccepted = +function ident_gotconn(serv, transport) +{ + // Using the listening CBSConnection to accept would stop it listening. + // A new CBSConnection does exactly what we want. + var connection = new CBSConnection(); + connection.accept(transport); + + connection.startAsyncRead(new IdentListener(this, connection)); +} + +function IdentListener(server, connection) +{ + this.server = server; + this.connection = connection; +} + +IdentListener.prototype.onStreamDataAvailable = +function ident_listener_sda(request, inStream, sourceOffset, count) +{ + var ev = new CEvent("ident-listener", "data-available", this, + "onDataAvailable"); + ev.line = this.connection.readData(0, count); + this.server.eventPump.routeEvent(ev); +} + +IdentListener.prototype.onStreamClose = +function ident_listener_sclose(status) +{ +} + +IdentListener.prototype.onDataAvailable = +function ident_listener_dataavailable(e) +{ + var incomplete = (e.line.substr(-2) != "\r\n"); + var lines = e.line.split(/\r\n/); + + if (this.savedLine) + { + lines[0] = this.savedLine + lines[0]; + this.savedLine = ""; + } + + if (incomplete) + this.savedLine = lines.pop() + + for (var i in lines) + { + var ev = new CEvent("ident-listener", "rawdata", this, "onRawData"); + ev.line = lines[i]; + this.server.eventPump.routeEvent(ev); + } +} + +IdentListener.prototype.onRawData = +function ident_listener_rawdata(e) +{ + var ports = e.line.match(/(\d+) *, *(\d+)/); + // <port-on-server> , <port-on-client> + // (where "server" is the ident server) + + if (!ports) + { + this.connection.disconnect(); // same meaning as "ERROR : UNKNOWN-ERROR" + return; + } + + e.type = "parsedrequest"; + e.destObject = this; + e.destMethod = "onParsedRequest"; + e.localPort = ports[1]; + e.remotePort = ports[2]; +} + +IdentListener.prototype.onParsedRequest = +function ident_listener_request(e) +{ + function response(str) + { + return e.localPort + " , " + e.remotePort + " : " + str + "\r\n"; + }; + + function validPort(p) + { + return (p >= 1) && (p <= 65535); + }; + + if (!validPort(e.localPort) || !validPort(e.remotePort)) + { + this.connection.sendData(response("ERROR : INVALID-PORT")); + this.connection.disconnect(); + return; + } + + var found, responses = this.server.responses; + for (var i = 0; i < responses.length; ++i) + { + if ((e.remotePort == responses[i].port) && + (this.connection._transport.host == responses[i].ip)) + { + // charset defaults to US-ASCII + // anything except an OS username should use OTHER + // however, ircu sucks, so we can't do that. + this.connection.sendData(response("USERID : CHATZILLA :" + + responses[i].username)); + found = true; + break; + } + } + + if (!found) + this.connection.sendData(response("ERROR : NO-USER")); + + // Spec gives us a choice: drop the connection, or listen for more queries. + // Since IRC servers will only ever want one name, we disconnect. + this.connection.disconnect(); +} diff --git a/comm/suite/chatzilla/js/lib/irc-debug.js b/comm/suite/chatzilla/js/lib/irc-debug.js new file mode 100644 index 0000000000..b0e95fee2e --- /dev/null +++ b/comm/suite/chatzilla/js/lib/irc-debug.js @@ -0,0 +1,84 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +/* + * Hook used to trace events. + */ +function event_tracer (e) +{ + var name = ""; + var data = ("debug" in e) ? e.debug : ""; + + switch (e.set) + { + case "server": + name = e.destObject.connection.host; + if (e.type == "rawdata") + data = "'" + e.data + "'"; + if (e.type == "senddata") + { + var nextLine = + e.destObject.sendQueue[e.destObject.sendQueue.length - 1]; + if ("logged" in nextLine) + return true; /* don't print again */ + + if (nextLine) { + data = "'" + nextLine.replace ("\n", "\\n") + "'"; + nextLine.logged = true; + } + else + data = "!!! Nothing to send !!!"; + } + break; + + case "network": + case "channel": + case "user": + name = e.destObject.unicodeName; + break; + + case "httpdoc": + name = e.destObject.server + e.destObject.path; + if (e.destObject.state != "complete") + data = "state: '" + e.destObject.state + "', received " + + e.destObject.data.length; + else + dd ("document done:\n" + dumpObjectTree (this)); + break; + + case "dcc-chat": + case "dcc-file": + name = e.destObject.localIP + ":" + e.destObject.port; + if (e.type == "rawdata") + data = "'" + e.data + "'"; + break; + + case "client": + if (e.type == "do-connect") + data = "attempt: " + e.attempt + "/" + + e.destObject.MAX_CONNECT_ATTEMPTS; + break; + + default: + break; + } + + if (name) + name = "[" + name + "]"; + + if (e.type == "info") + data = "'" + e.msg + "'"; + + var str = "Level " + e.level + ": '" + e.type + "', " + + e.set + name + "." + e.destMethod; + if (data) + str += "\ndata : " + data; + + dd (str); + + return true; + +} diff --git a/comm/suite/chatzilla/js/lib/irc.js b/comm/suite/chatzilla/js/lib/irc.js new file mode 100644 index 0000000000..5e7d210c44 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/irc.js @@ -0,0 +1,4518 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +const JSIRC_ERR_NO_SOCKET = "JSIRCE:NS"; +const JSIRC_ERR_EXHAUSTED = "JSIRCE:E"; +const JSIRC_ERR_CANCELLED = "JSIRCE:C"; +const JSIRC_ERR_NO_SECURE = "JSIRCE:NO_SECURE"; +const JSIRC_ERR_OFFLINE = "JSIRCE:OFFLINE"; +const JSIRC_ERR_PAC_LOADING = "JSIRCE:PAC_LOADING"; + +const JSIRCV3_SUPPORTED_CAPS = [ + "account-notify", + "account-tag", + "away-notify", + "batch", + "cap-notify", + "chghost", + "echo-message", + "extended-join", + "invite-notify", + //"labeled-response", + "message-tags", + //"metadata", + "multi-prefix", + "sasl", + "server-time", + "tls", + "userhost-in-names", +]; + +function userIsMe (user) +{ + + switch (user.TYPE) + { + case "IRCUser": + return (user == user.parent.me); + break; + + case "IRCChanUser": + return (user.__proto__ == user.parent.parent.me); + break; + + default: + return false; + + } + + return false; +} + +/* + * Attached to event objects in onRawData + */ +function decodeParam(number, charsetOrObject) +{ + if (!charsetOrObject) + charsetOrObject = this.currentObject; + + var rv = toUnicode(this.params[number], charsetOrObject); + + return rv; +} + +// JavaScript won't let you delete things declared with "var", workaround: +window.i = 1; + +const NET_OFFLINE = i++; // Initial, disconected. +const NET_WAITING = i++; // Waiting before trying. +const NET_CONNECTING = i++; // Trying a connect... +const NET_CANCELLING = i++; // Cancelling connect. +const NET_ONLINE = i++; // Connected ok. +const NET_DISCONNECTING = i++; // Disconnecting. + +delete window.i; + +function CIRCNetwork (name, serverList, eventPump, temporary) +{ + this.unicodeName = name; + this.viewName = name; + this.canonicalName = name; + this.collectionKey = ":" + name; + this.encodedName = name; + this.servers = new Object(); + this.serverList = new Array(); + this.ignoreList = new Object(); + this.ignoreMaskCache = new Object(); + this.state = NET_OFFLINE; + this.temporary = Boolean(temporary); + + for (var i = 0; i < serverList.length; ++i) + { + var server = serverList[i]; + var password = ("password" in server) ? server.password : null; + var isSecure = ("isSecure" in server) ? server.isSecure : false; + this.serverList.push(new CIRCServer(this, server.name, server.port, isSecure, + password)); + } + + this.eventPump = eventPump; + if ("onInit" in this) + this.onInit(); +} + +/** Clients should override this stuff themselves **/ +CIRCNetwork.prototype.INITIAL_NICK = "js-irc"; +CIRCNetwork.prototype.INITIAL_NAME = "INITIAL_NAME"; +CIRCNetwork.prototype.INITIAL_DESC = "INITIAL_DESC"; +CIRCNetwork.prototype.USE_SASL = false; +CIRCNetwork.prototype.UPGRADE_INSECURE = false; +CIRCNetwork.prototype.STS_MODULE = null; +/* set INITIAL_CHANNEL to "" if you don't want a primary channel */ +CIRCNetwork.prototype.INITIAL_CHANNEL = "#jsbot"; +CIRCNetwork.prototype.INITIAL_UMODE = "+iw"; + +CIRCNetwork.prototype.MAX_CONNECT_ATTEMPTS = 5; +CIRCNetwork.prototype.PAC_RECONNECT_DELAY = 5 * 1000; +CIRCNetwork.prototype.getReconnectDelayMs = function() { return 15000; } +CIRCNetwork.prototype.stayingPower = false; + +// "http" = use HTTP proxy, "none" = none, anything else = auto. +CIRCNetwork.prototype.PROXY_TYPE_OVERRIDE = ""; + +CIRCNetwork.prototype.TYPE = "IRCNetwork"; + +/** + * Returns the IRC URL representation of this network. + * + * @param target A network-specific object to target the URL at. Instead of + * passing it in here, call the target's |getURL| method. + * @param flags An |Object| with flags (as properties) to be applied to the URL. + */ +CIRCNetwork.prototype.getURL = +function net_geturl(target, flags) +{ + if (this.temporary) + return this.serverList[0].getURL(target, flags); + + /* Determine whether to use the irc:// or ircs:// scheme */ + var scheme = "irc"; + if ((("primServ" in this) && this.primServ.isConnected && + this.primServ.isSecure) || + this.hasOnlySecureServers()) + { + scheme = "ircs" + } + + var obj = {host: this.unicodeName, scheme: scheme}; + + if (target) + obj.target = target; + + if (flags) + { + for (var i = 0; i < flags.length; i++) + obj[flags[i]] = true; + } + + return constructIRCURL(obj); +} + +CIRCNetwork.prototype.getUser = +function net_getuser (nick) +{ + if ("primServ" in this && this.primServ) + return this.primServ.getUser(nick); + + return null; +} + +CIRCNetwork.prototype.addServer = +function net_addsrv(host, port, isSecure, password) +{ + this.serverList.push(new CIRCServer(this, host, port, isSecure, password)); +} + +/** + * Returns |true| iif a network has a secure server in its list. + */ +CIRCNetwork.prototype.hasSecureServer = +function net_hasSecure() +{ + for (var i = 0; i < this.serverList.length; i++) + { + if (this.serverList[i].isSecure) + return true; + } + + return false; +} + +/** + * Returns |true| iif a network only has secure servers in its list. + */ +CIRCNetwork.prototype.hasOnlySecureServers = +function net_hasOnlySecure() +{ + for (var i = 0; i < this.serverList.length; i++) + { + if (!this.serverList[i].isSecure) + return false; + } + + return true; +} + +CIRCNetwork.prototype.clearServerList = +function net_clearserverlist() +{ + /* Note: we don't have to worry about being connected, since primServ + * keeps the currently connected server alive if we still need it. + */ + this.servers = new Object(); + this.serverList = new Array(); +} + +/** + * Trigger an |onDoConnect| event after a delay. + */ +CIRCNetwork.prototype.delayedConnect = +function net_delayedConnect(eventProperties) +{ + function reconnectFn(network, eventProperties) + { + network.immediateConnect(eventProperties); + }; + + if ((-1 != this.MAX_CONNECT_ATTEMPTS) && + (this.connectAttempt >= this.MAX_CONNECT_ATTEMPTS)) + { + this.state = NET_OFFLINE; + + var ev = new CEvent("network", "error", this, "onError"); + ev.debug = "Connection attempts exhausted, giving up."; + ev.errorCode = JSIRC_ERR_EXHAUSTED; + this.eventPump.addEvent(ev); + + return; + } + + this.state = NET_WAITING; + this.reconnectTimer = setTimeout(reconnectFn, + this.getReconnectDelayMs(), + this, + eventProperties); +} + +/** + * Immediately trigger an |onDoConnect| event. Use |delayedConnect| for automatic + * repeat attempts, instead, to throttle the attempts to a reasonable pace. + */ +CIRCNetwork.prototype.immediateConnect = +function net_immediateConnect(eventProperties) +{ + var ev = new CEvent("network", "do-connect", this, "onDoConnect"); + + if (typeof eventProperties != "undefined") + for (var key in eventProperties) + ev[key] = eventProperties[key]; + + this.eventPump.addEvent(ev); +} + +CIRCNetwork.prototype.connect = +function net_connect(requireSecurity) +{ + if ("primServ" in this && this.primServ.isConnected) + return true; + + // We need to test for secure servers in the network object here, + // because without them all connection attempts will fail anyway. + if (requireSecurity && !this.hasSecureServer()) + { + // No secure server, cope. + ev = new CEvent ("network", "error", this, "onError"); + ev.server = this; + ev.debug = "No connection attempted: no secure servers in list"; + ev.errorCode = JSIRC_ERR_NO_SECURE; + this.eventPump.addEvent(ev); + + return false; + } + + this.state = NET_CONNECTING; + this.connectAttempt = 0; // actual connection attempts + this.connectCandidate = 0; // incl. requireSecurity non-attempts + this.nextHost = 0; + this.requireSecurity = requireSecurity || false; + this.immediateConnect({"password": null}); + return true; +} + +/** + * Disconnects the network with a given reason. + */ +CIRCNetwork.prototype.quit = +function net_quit (reason) +{ + if (this.isConnected()) + this.primServ.logout(reason); +} + +/** + * Cancels the network's connection (whatever its current state). + */ +CIRCNetwork.prototype.cancel = +function net_cancel() +{ + // We're online, pull the plug on the current connection, or... + if (this.state == NET_ONLINE) + { + this.quit(); + } + // We're waiting for the 001, too late to throw a reconnect, or... + else if (this.state == NET_CONNECTING) + { + this.state = NET_CANCELLING; + if ("primServ" in this && this.primServ.isConnected) + { + this.primServ.connection.disconnect(); + + var ev = new CEvent("network", "error", this, "onError"); + ev.server = this.primServ; + ev.debug = "Connect sequence was canceled."; + ev.errorCode = JSIRC_ERR_CANCELLED; + this.eventPump.addEvent(ev); + } + } + // We're waiting for onDoConnect, so try a reconnect (which will fail us) + else if (this.state == NET_WAITING) + { + this.state = NET_CANCELLING; + // onDoConnect will throw the error events for us, as it will fail + this.immediateConnect(); + } + else + { + dd("Network cancel in odd state: " + this.state); + } +} + +CIRCNetwork.prototype.onDoConnect = +function net_doconnect(e) +{ + const NS_ERROR_OFFLINE = 0x804b0010; + var c; + + // Clear the timer, if there is one. + if ("reconnectTimer" in this) + { + clearTimeout(this.reconnectTimer); + delete this.reconnectTimer; + } + + var ev; + + if (this.state == NET_CANCELLING) + { + if ("primServ" in this && this.primServ.isConnected) + this.primServ.connection.disconnect(); + else + this.state = NET_OFFLINE; + + ev = new CEvent("network", "error", this, "onError"); + ev.server = this.primServ; + ev.debug = "Connect sequence was canceled."; + ev.errorCode = JSIRC_ERR_CANCELLED; + this.eventPump.addEvent(ev); + + return false; + } + + if ("primServ" in this && this.primServ.isConnected) + return true; + + this.connectAttempt++; + this.connectCandidate++; + + this.state = NET_CONNECTING; /* connection is considered "made" when server + * sends a 001 message (see server.on001) */ + + var host = this.nextHost++; + if (host >= this.serverList.length) + { + this.nextHost = 1; + host = 0; + } + + // If STS is enabled, check the cache for a secure port to connect to. + if (this.STS_MODULE.ENABLED && !this.serverList[host].isSecure) + { + var newPort = this.STS_MODULE.getUpgradePolicy(this.serverList[host].hostname); + if (newPort) + { + // If we're a temporary network, just change the server prior to connecting. + if (this.temporary) + { + this.serverList[host].port = newPort; + this.serverList[host].isSecure = true; + } + // Otherwise, find or create a server with the specified host and port. + else + { + var hostname = this.serverList[host].hostname; + var matches = this.serverList.filter(function(s) { + return s.hostname == hostname && s.port == newPort; + }); + if (matches.length > 0) + { + host = arrayIndexOf(this.serverList, matches[0]); + } + else + { + this.addServer(hostname, newPort, true, + this.serverList[host].password); + host = this.serverList.length - 1; + } + } + } + } + + if (this.serverList[host].isSecure || !this.requireSecurity) + { + ev = new CEvent ("network", "startconnect", this, "onStartConnect"); + ev.debug = "Connecting to " + this.serverList[host].unicodeName + ":" + + this.serverList[host].port + ", attempt " + this.connectAttempt + + " of " + this.MAX_CONNECT_ATTEMPTS + "..."; + ev.host = this.serverList[host].hostname; + ev.port = this.serverList[host].port; + ev.server = this.serverList[host]; + ev.connectAttempt = this.connectAttempt; + ev.reconnectDelayMs = this.getReconnectDelayMs(); + this.eventPump.addEvent (ev); + + try + { + this.serverList[host].connect(); + } + catch(ex) + { + this.state = NET_OFFLINE; + + ev = new CEvent("network", "error", this, "onError"); + ev.server = this; + ev.debug = "Exception opening socket: " + ex; + ev.errorCode = JSIRC_ERR_NO_SOCKET; + if ((typeof ex == "object") && (ex.result == NS_ERROR_OFFLINE)) + ev.errorCode = JSIRC_ERR_OFFLINE; + if ((typeof ex == "string") && (ex == JSIRC_ERR_PAC_LOADING)) + { + ev.errorCode = JSIRC_ERR_PAC_LOADING; + ev.retryDelay = CIRCNetwork.prototype.PAC_RECONNECT_DELAY; + /* PAC loading is not a problem with any specific server. We'll + * retry the connection in 5 seconds. + */ + this.nextHost--; + this.state = NET_WAITING; + setTimeout(function(n) { n.immediateConnect() }, + ev.retryDelay, this); + } + this.eventPump.addEvent(ev); + } + } + else + { + /* Server doesn't use SSL as requested, try next one. + * In the meantime, correct the connection attempt counter */ + this.connectAttempt--; + this.immediateConnect(); + } + + return true; +} + +/** + * Returns |true| iff this network has a socket-level connection. + */ +CIRCNetwork.prototype.isConnected = +function net_connected (e) +{ + return ("primServ" in this && this.primServ.isConnected); +} + + +CIRCNetwork.prototype.ignore = +function net_ignore (hostmask) +{ + var input = getHostmaskParts(hostmask); + + if (input.mask in this.ignoreList) + return false; + + this.ignoreList[input.mask] = input; + this.ignoreMaskCache = new Object(); + return true; +} + +CIRCNetwork.prototype.unignore = +function net_ignore (hostmask) +{ + var input = getHostmaskParts(hostmask); + + if (!(input.mask in this.ignoreList)) + return false; + + delete this.ignoreList[input.mask]; + this.ignoreMaskCache = new Object(); + return true; +} + +function CIRCServer (parent, hostname, port, isSecure, password) +{ + var serverName = hostname + ":" + port; + + var s; + if (serverName in parent.servers) + { + s = parent.servers[serverName]; + } + else + { + s = this; + s.channels = new Object(); + s.users = new Object(); + } + + s.unicodeName = serverName; + s.viewName = serverName; + s.canonicalName = serverName; + s.collectionKey = ":" + serverName; + s.encodedName = serverName; + s.hostname = hostname; + s.port = port; + s.parent = parent; + s.isSecure = isSecure; + s.password = password; + s.connection = null; + s.isConnected = false; + s.sendQueue = new Array(); + s.lastSend = new Date("1/1/1980"); + s.lastPingSent = null; + s.lastPing = null; + s.savedLine = ""; + s.lag = -1; + s.usersStable = true; + s.supports = null; + s.channelTypes = null; + s.channelModes = null; + s.channelCount = -1; + s.userModes = null; + s.maxLineLength = 400; + s.caps = new Object(); + s.capvals = new Object(); + + parent.servers[s.collectionKey] = s; + if ("onInit" in s) + s.onInit(); + return s; +} + +CIRCServer.prototype.MS_BETWEEN_SENDS = 1500; +CIRCServer.prototype.READ_TIMEOUT = 100; +CIRCServer.prototype.VERSION_RPLY = "JS-IRC Library v0.01, " + + "Copyright (C) 1999 Robert Ginda; rginda@ndcico.com"; +CIRCServer.prototype.OS_RPLY = "Unknown"; +CIRCServer.prototype.HOST_RPLY = "Unknown"; +CIRCServer.prototype.DEFAULT_REASON = "no reason"; +/* true means WHO command doesn't collect hostmask, username, etc. */ +CIRCServer.prototype.LIGHTWEIGHT_WHO = false; +/* Unique identifier for WHOX commands. */ +CIRCServer.prototype.WHOX_TYPE = "314"; +/* -1 == never, 0 == prune onQuit, >0 == prune when >X ms old */ +CIRCServer.prototype.PRUNE_OLD_USERS = -1; + +CIRCServer.prototype.TYPE = "IRCServer"; + +// Define functions to set modes so they're easily readable. +// name is the name used on the CIRCChanMode object +// getValue is a function returning the value the canonicalmode should be set to +// given a certain modifier and appropriate data. +CIRCServer.prototype.canonicalChanModes = { + i: { + name: "invite", + getValue: function (modifier) { return (modifier == "+"); } + }, + m: { + name: "moderated", + getValue: function (modifier) { return (modifier == "+"); } + }, + n: { + name: "publicMessages", + getValue: function (modifier) { return (modifier == "-"); } + }, + t: { + name: "publicTopic", + getValue: function (modifier) { return (modifier == "-"); } + }, + s: { + name: "secret", + getValue: function (modifier) { return (modifier == "+"); } + }, + p: { + name: "pvt", + getValue: function (modifier) { return (modifier == "+"); } + }, + k: { + name: "key", + getValue: function (modifier, data) + { + if (modifier == "+") + return data; + else + return ""; + } + }, + l: { + name: "limit", + getValue: function (modifier, data) + { + // limit is special - we return -1 if there is no limit. + if (modifier == "-") + return -1; + else + return data; + } + } +}; + +CIRCServer.prototype.toLowerCase = +function serv_tolowercase(str) +{ + /* This is an implementation that lower-cases strings according to the + * prevailing CASEMAPPING setting for the server. Values for this are: + * + * o "ascii": The ASCII characters 97 to 122 (decimal) are defined as + * the lower-case characters of ASCII 65 to 90 (decimal). No other + * character equivalency is defined. + * o "strict-rfc1459": The ASCII characters 97 to 125 (decimal) are + * defined as the lower-case characters of ASCII 65 to 93 (decimal). + * No other character equivalency is defined. + * o "rfc1459": The ASCII characters 97 to 126 (decimal) are defined as + * the lower-case characters of ASCII 65 to 94 (decimal). No other + * character equivalency is defined. + * + */ + + function replaceFunction(chr) + { + return String.fromCharCode(chr.charCodeAt(0) + 32); + } + + var mapping = "rfc1459"; + if (this.supports) + mapping = this.supports.casemapping; + + /* NOTE: There are NO breaks in this switch. This is CORRECT. + * Each mapping listed is a super-set of those below, thus we only + * transform the extra characters, and then fall through. + */ + switch (mapping) + { + case "rfc1459": + str = str.replace(/\^/g, replaceFunction); + case "strict-rfc1459": + str = str.replace(/[\[\\\]]/g, replaceFunction); + case "ascii": + str = str.replace(/[A-Z]/g, replaceFunction); + } + return str; +} + +// Iterates through the keys in an object and, if specified, the keys of +// child objects. +CIRCServer.prototype.renameProperties = +function serv_renameproperties(obj, child) +{ + for (let key in obj) + { + let item = obj[key]; + item.canonicalName = this.toLowerCase(item.encodedName); + item.collectionKey = ":" + item.canonicalName; + renameProperty(obj, key, item.collectionKey); + if (child && (child in item)) + this.renameProperties(item[child], null); + } +} + +// Encodes tag data to send. +CIRCServer.prototype.encodeTagData = +function serv_encodetagdata(obj) +{ + var dict = new Object(); + dict[";"] = ":"; + dict[" "] = "s"; + dict["\\"] = "\\"; + dict["\r"] = "r"; + dict["\n"] = "n"; + + // Function for escaping key values. + function escapeTagValue(data) + { + var rv = ""; + for (var i = 0; i < data.length; i++) + { + var ci = data[i]; + var co = dict[data[i]]; + if (co) + rv += "\\" + co; + else + rv += ci; + } + + return rv; + } + + var str = ""; + + for(var key in obj) + { + var val = obj[key]; + str += key; + if (val) + { + str += "="; + str += escapeTagValue(val); + } + str += ";"; + } + + // Remove any trailing semicolons. + if (str[str.length - 1] == ";") + str = str.substring(0, str.length - 1); + + return str; +} + +// Decodes received tag data. +CIRCServer.prototype.decodeTagData = +function serv_decodetagdata(str) +{ + // Remove the leading '@' if we have one. + if (str[0] == "@") + str = str.substring(1); + + var dict = new Object(); + dict[":"] = ";"; + dict["s"] = " "; + dict["\\"] = "\\"; + dict["r"] = "\r"; + dict["n"] = "\n"; + + // Function for unescaping key values. + function unescapeTagValue(data) + { + var rv = ""; + for (let j = 0; j < data.length; j++) + { + let currentItem = data[j]; + if (currentItem == "\\" && j < data.length - 1) + { + let nextItem = data[j + 1]; + if (nextItem in dict) + rv += dict[nextItem]; + else + rv += nextItem; + j++ + } + else if (currentItem != "\\") + rv += currentItem; + } + + return rv; + } + + var obj = Object(); + + var tags = str.split(";"); + for (var i = 0; i < tags.length; i++) + { + var [key, val] = tags[i].split("="); + val = unescapeTagValue(val); + obj[key] = val; + } + + return obj; +} + +// Returns the IRC URL representation of this server. +CIRCServer.prototype.getURL = +function serv_geturl(target, flags) +{ + var scheme = (this.isSecure ? "ircs" : "irc"); + var obj = {host: this.hostname, scheme: scheme, isserver: true, + port: this.port, needpass: Boolean(this.password)}; + + if (target) + obj.target = target; + + if (flags) + { + for (var i = 0; i < flags.length; i++) + obj[flags[i]] = true; + } + + return constructIRCURL(obj); +} + +CIRCServer.prototype.getUser = +function chan_getuser(nick) +{ + var tnick = ":" + this.toLowerCase(nick); + + if (tnick in this.users) + return this.users[tnick]; + + tnick = ":" + this.toLowerCase(fromUnicode(nick, this)); + + if (tnick in this.users) + return this.users[tnick]; + + return null; +} + +CIRCServer.prototype.getChannel = +function chan_getchannel(name) +{ + var tname = ":" + this.toLowerCase(name); + + if (tname in this.channels) + return this.channels[tname]; + + tname = ":" + this.toLowerCase(fromUnicode(name, this)); + + if (tname in this.channels) + return this.channels[tname]; + + return null; +} + +CIRCServer.prototype.connect = +function serv_connect() +{ + if (this.connection != null) + throw "Server already has a connection pending or established"; + + var config = { isSecure: this.isSecure }; + if (this.parent.PROXY_TYPE_OVERRIDE) + config.proxy = this.parent.PROXY_TYPE_OVERRIDE; + + this.connection = new CBSConnection(); + this.connection.connect(this.hostname, this.port, config, this); +} + +// This may be called synchronously or asynchronously by CBSConnection.connect. +CIRCServer.prototype.onSocketConnection = +function serv_onsocketconnection(host, port, config, exception) +{ + if (this.parent.state == NET_CANCELLING) + { + this.connection.disconnect(); + this.connection = null; + this.parent.state = NET_OFFLINE; + + var ev = new CEvent("network", "error", this.parent, "onError"); + ev.server = this; + ev.debug = "Connect sequence was canceled."; + ev.errorCode = JSIRC_ERR_CANCELLED; + this.parent.eventPump.addEvent(ev); + } + else if (!exception) + { + var ev = new CEvent("server", "connect", this, "onConnect"); + ev.server = this; + this.parent.eventPump.addEvent(ev); + this.isConnected = true; + + this.connection.startAsyncRead(this); + } + else + { + var ev = new CEvent("server", "disconnect", this, "onDisconnect"); + ev.server = this; + ev.reason = "error"; + ev.exception = exception; + ev.disconnectStatus = NS_ERROR_ABORT; + this.parent.eventPump.addEvent(ev); + } +} + +/* + * What to do when the client connects to it's primary server + */ +CIRCServer.prototype.onConnect = +function serv_onconnect (e) +{ + this.parent.primServ = e.server; + + this.sendData("CAP LS 302\n"); + this.pendingCapNegotiation = true; + + this.caps = new Object(); + this.capvals = new Object(); + + this.login(this.parent.INITIAL_NICK, this.parent.INITIAL_NAME, + this.parent.INITIAL_DESC); + return true; +} + +CIRCServer.prototype.onStreamDataAvailable = +function serv_sda (request, inStream, sourceOffset, count) +{ + var ev = new CEvent ("server", "data-available", this, + "onDataAvailable"); + + ev.line = this.connection.readData(0, count); + + /* route data-available as we get it. the data-available handler does + * not do much, so we can probably get away with this without starving + * the UI even under heavy input traffic. + */ + this.parent.eventPump.routeEvent(ev); +} + +CIRCServer.prototype.onStreamClose = +function serv_sockdiscon(status) +{ + var ev = new CEvent ("server", "disconnect", this, "onDisconnect"); + ev.server = this; + ev.disconnectStatus = status; + if (ev.disconnectStatus == NS_ERROR_BINDING_ABORTED) + ev.disconnectStatus = NS_ERROR_ABORT; + + this.parent.eventPump.addEvent (ev); +} + + +CIRCServer.prototype.flushSendQueue = +function serv_flush() +{ + this.sendQueue.length = 0; + dd("sendQueue flushed."); + + return true; +} + +CIRCServer.prototype.login = +function serv_login(nick, name, desc) +{ + nick = nick.replace(/ /g, "_"); + name = name.replace(/ /g, "_"); + + if (!nick) + nick = "nick"; + + if (!name) + name = nick; + + if (!desc) + desc = nick; + + this.me = new CIRCUser(this, nick, null, name); + if (this.password) + this.sendData("PASS " + this.password + "\n"); + this.changeNick(this.me.unicodeName); + this.sendData("USER " + name + " * * :" + + fromUnicode(desc, this) + "\n"); +} + +CIRCServer.prototype.logout = +function serv_logout(reason) +{ + if (reason == null || typeof reason == "undefined") + reason = this.DEFAULT_REASON; + + this.quitting = true; + + this.connection.sendData("QUIT :" + + fromUnicode(reason, this.parent) + "\n"); + this.connection.disconnect(); +} + +CIRCServer.prototype.sendAuthResponse = +function serv_authresponse(resp) +{ + // Encode the response and break into 400-byte parts. + var resp = btoa(resp); + var part = null; + var n = 0; + do + { + part = resp.substring(0, 400); + n = part.length; + resp = resp.substring(400); + + this.sendData("AUTHENTICATE " + part + '\n'); + } + while (resp.length > 0); + + // Send empty auth response if last part was exactly 400 bytes long. + if (n == 400) + { + this.sendData("AUTHENTICATE +\n"); + } +} + +CIRCServer.prototype.sendAuthAbort = +function serv_authabort() +{ + // Abort an in-progress SASL authentication. + this.sendData("AUTHENTICATE *\n"); +} + +CIRCServer.prototype.sendMonitorList = +function serv_monitorlist(nicks, isAdd) +{ + if (!nicks.length) + return; + + var prefix; + if (isAdd) + prefix = "MONITOR + "; + else + prefix = "MONITOR - "; + + /* Send monitor list updates in chunks less than + maxLineLength in size. */ + var nicks_string = nicks.join(","); + while (nicks_string.length > this.maxLineLength) + { + var nicks_part = nicks_string.substring(0, this.maxLineLength); + var i = nicks_part.lastIndexOf(","); + nicks_part = nicks_string.substring(0, i); + nicks_string = nicks_string.substring(i + 1); + this.sendData(prefix + nicks_part + "\n"); + } + this.sendData(prefix + nicks_string + "\n"); +} + +CIRCServer.prototype.addTarget = +function serv_addtarget(name) +{ + if (arrayIndexOf(this.channelTypes, name[0]) != -1) { + return this.addChannel(name); + } else { + return this.addUser(name); + } +} + +CIRCServer.prototype.addChannel = +function serv_addchan(unicodeName, charset) +{ + return new CIRCChannel(this, unicodeName, fromUnicode(unicodeName, charset)); +} + +CIRCServer.prototype.addUser = +function serv_addusr(unicodeName, name, host) +{ + return new CIRCUser(this, unicodeName, null, name, host); +} + +CIRCServer.prototype.getChannelsLength = +function serv_chanlen() +{ + var i = 0; + + for (var p in this.channels) + i++; + + return i; +} + +CIRCServer.prototype.getUsersLength = +function serv_chanlen() +{ + var i = 0; + + for (var p in this.users) + i++; + + return i; +} + +CIRCServer.prototype.sendData = +function serv_senddata (msg) +{ + this.queuedSendData (msg); +} + +CIRCServer.prototype.queuedSendData = +function serv_senddata (msg) +{ + if (this.sendQueue.length == 0) + this.parent.eventPump.addEvent (new CEvent ("server", "senddata", + this, "onSendData")); + arrayInsertAt (this.sendQueue, 0, new String(msg)); +} + +// Utility method for splitting large lines prior to sending. +CIRCServer.prototype.splitLinesForSending = +function serv_splitlines(line, prettyWrap) +{ + let lines = String(line).split("\n"); + let realLines = []; + for (let i = 0; i < lines.length; i++) + { + if (lines[i]) + { + while (lines[i].length > this.maxLineLength) + { + var extraLine = lines[i].substr(0, this.maxLineLength - 5); + var pos = extraLine.lastIndexOf(" "); + + if ((pos >= 0) && (pos >= this.maxLineLength - 15)) + { + // Smart-split. + extraLine = lines[i].substr(0, pos); + lines[i] = lines[i].substr(extraLine.length + 1); + if (prettyWrap) + { + extraLine += "..."; + lines[i] = "..." + lines[i]; + } + } + else + { + // Dumb-split. + extraLine = lines[i].substr(0, this.maxLineLength); + lines[i] = lines[i].substr(extraLine.length); + } + realLines.push(extraLine); + } + realLines.push(lines[i]); + } + } + return realLines; +} + +CIRCServer.prototype.messageTo = +function serv_messto(code, target, msg, ctcpCode) +{ + let lines = this.splitLinesForSending(msg, true); + + let i = 0; + let pfx = ""; + let sfx = ""; + + if (ctcpCode) + { + pfx = "\01" + ctcpCode; + sfx = "\01"; + } + + // We may have no message at all with CTCP commands. + if (!lines.length && ctcpCode) + lines.push(""); + + for (i in lines) + { + if ((lines[i] != "") || ctcpCode) + { + var line = code + " " + target + " :" + pfx; + if (lines[i] != "") + { + if (ctcpCode) + line += " "; + line += lines[i] + sfx; + } + else + line += sfx; + //dd ("-*- irc sending '" + line + "'"); + this.sendData(line + "\n"); + } + } +} + +CIRCServer.prototype.sayTo = +function serv_sayto (target, msg) +{ + this.messageTo("PRIVMSG", target, msg); +} + +CIRCServer.prototype.noticeTo = +function serv_noticeto (target, msg) +{ + this.messageTo("NOTICE", target, msg); +} + +CIRCServer.prototype.actTo = +function serv_actto (target, msg) +{ + this.messageTo("PRIVMSG", target, msg, "ACTION"); +} + +CIRCServer.prototype.ctcpTo = +function serv_ctcpto (target, code, msg, method) +{ + msg = msg || ""; + method = method || "PRIVMSG"; + + code = code.toUpperCase(); + if (code == "PING" && !msg) + msg = Number(new Date()); + this.messageTo(method, target, msg, code); +} + +CIRCServer.prototype.changeNick = +function serv_changenick(newNick) +{ + this.sendData("NICK " + fromUnicode(newNick, this) + "\n"); +} + +CIRCServer.prototype.updateLagTimer = +function serv_uptimer() +{ + this.connection.sendData("PING :LAGTIMER\n"); + this.lastPing = this.lastPingSent = new Date(); +} + +CIRCServer.prototype.userhost = +function serv_userhost(target) +{ + this.sendData("USERHOST " + fromUnicode(target, this) + "\n"); +} + +CIRCServer.prototype.userip = +function serv_userip(target) +{ + this.sendData("USERIP " + fromUnicode(target, this) + "\n"); +} + +CIRCServer.prototype.who = +function serv_who(target) +{ + this.sendData("WHO " + fromUnicode(target, this) + "\n"); +} + +/** + * Abstracts the whois command. + * + * @param target intended user(s). + */ +CIRCServer.prototype.whois = +function serv_whois (target) +{ + this.sendData("WHOIS " + fromUnicode(target, this) + "\n"); +} + +CIRCServer.prototype.whowas = +function serv_whowas(target, limit) +{ + if (typeof limit == "undefined") + limit = 1; + else if (limit == 0) + limit = ""; + + this.sendData("WHOWAS " + fromUnicode(target, this) + " " + limit + "\n"); +} + +CIRCServer.prototype.onDisconnect = +function serv_disconnect(e) +{ + function stateChangeFn(network, state) { + network.state = state; + }; + + function delayedConnectFn(network) { + network.delayedConnect(); + }; + + /* If we're not connected and get this, it means we have almost certainly + * encountered a read or write error on the socket post-disconnect. There's + * no point propagating this any further, as we've already notified the + * user of the disconnect (with the right error). + */ + if (!this.isConnected) + return; + + // Don't reconnect from a certificate error. + var certError = (getNSSErrorClass(e.disconnectStatus) == ERROR_CLASS_BAD_CERT); + + // Don't reconnect if our connection was aborted. + var wasAborted = (e.disconnectStatus == NS_ERROR_ABORT); + var dontReconnect = certError || wasAborted; + + if (((this.parent.state == NET_CONNECTING) && !dontReconnect) || + /* fell off while connecting, try again */ + (this.parent.primServ == this) && (this.parent.state == NET_ONLINE) && + (!("quitting" in this) && this.parent.stayingPower && !dontReconnect)) + { /* fell off primary server, reconnect to any host in the serverList */ + setTimeout(delayedConnectFn, 0, this.parent); + } + else + { + setTimeout(stateChangeFn, 0, this.parent, NET_OFFLINE); + } + + e.server = this; + e.set = "network"; + e.destObject = this.parent; + + e.quitting = this.quitting; + + for (var c in this.channels) + { + this.channels[c].users = new Object(); + this.channels[c].active = false; + } + + if (this.isStartTLS) + { + this.isSecure = false; + delete this.isStartTLS; + } + + delete this.batches; + + this.connection = null; + this.isConnected = false; + + delete this.quitting; +} + +CIRCServer.prototype.onSendData = +function serv_onsenddata (e) +{ + if (!this.isConnected || (this.parent.state == NET_CANCELLING)) + { + dd ("Can't send to disconnected socket"); + this.flushSendQueue(); + return false; + } + + var d = new Date(); + + // Wheee, some sanity checking! (there's been at least one case of lastSend + // ending up in the *future* at this point, which kinda busts things) + if (this.lastSend > d) + this.lastSend = 0; + + if (((d - this.lastSend) >= this.MS_BETWEEN_SENDS) && + this.sendQueue.length > 0) + { + var s = this.sendQueue.pop(); + + if (s) + { + try + { + this.connection.sendData(s); + } + catch(ex) + { + dd("Exception in queued send: " + ex); + this.flushSendQueue(); + + var ev = new CEvent("server", "disconnect", + this, "onDisconnect"); + ev.server = this; + ev.reason = "error"; + ev.exception = ex; + ev.disconnectStatus = NS_ERROR_ABORT; + this.parent.eventPump.addEvent(ev); + + return false; + } + this.lastSend = d; + } + + } + else + { + this.parent.eventPump.addEvent(new CEvent("event-pump", "yield", + null, "")); + } + + if (this.sendQueue.length > 0) + this.parent.eventPump.addEvent(new CEvent("server", "senddata", + this, "onSendData")); + return true; +} + +CIRCServer.prototype.onPoll = +function serv_poll(e) +{ + var lines; + var ex; + var ev; + + try + { + if (this.parent.state != NET_CANCELLING) + line = this.connection.readData(this.READ_TIMEOUT); + } + catch (ex) + { + dd ("*** Caught exception " + ex + " reading from server " + + this.hostname); + ev = new CEvent ("server", "disconnect", this, "onDisconnect"); + ev.server = this; + ev.reason = "error"; + ev.exception = ex; + ev.disconnectStatus = NS_ERROR_ABORT; + this.parent.eventPump.addEvent (ev); + return false; + } + + this.parent.eventPump.addEvent (new CEvent ("server", "poll", this, + "onPoll")); + + if (line) + { + ev = new CEvent ("server", "data-available", this, "onDataAvailable"); + ev.line = line; + this.parent.eventPump.routeEvent(ev); + } + + return true; +} + +CIRCServer.prototype.onDataAvailable = +function serv_ppline(e) +{ + var line = e.line; + + if (line == "") + return false; + + var incomplete = (line[line.length - 1] != '\n'); + var lines = line.split("\n"); + + if (this.savedLine) + { + lines[0] = this.savedLine + lines[0]; + this.savedLine = ""; + } + + if (incomplete) + this.savedLine = lines.pop(); + + for (var i in lines) + { + var ev = new CEvent("server", "rawdata", this, "onRawData"); + ev.data = lines[i].replace(/\r/g, ""); + if (ev.data) + { + if (ev.data.match(/^(?::[^ ]+ )?(?:32[123]|352|354|315) /i)) + this.parent.eventPump.addBulkEvent(ev); + else + this.parent.eventPump.addEvent(ev); + } + } + + return true; +} + +/* + * onRawData begins shaping the event by parsing the IRC message at it's + * simplest level. After onRawData, the event will have the following + * properties: + * name value + * + * set............"server" + * type..........."parsedata" + * destMethod....."onParsedData" + * destObject.....server (this) + * server.........server (this) + * connection.....CBSConnection (this.connection) + * source.........the <prefix> of the message (if it exists) + * user...........user object initialized with data from the message <prefix> + * params.........array containing the parameters of the message + * code...........the first parameter (most messages have this) + * + * See Section 2.3.1 of RFC 1459 for details on <prefix>, <middle> and + * <trailing> tokens. + */ +CIRCServer.prototype.onRawData = +function serv_onRawData(e) +{ + var ary; + var l = e.data; + + if (l.length == 0) + { + dd ("empty line on onRawData?"); + return false; + } + + if (l[0] == "@") + { + e.tagdata = l.substring(0, l.indexOf(" ")); + e.tags = this.decodeTagData(e.tagdata); + l = l.substring(l.indexOf(" ") + 1); + } + else + { + e.tagdata = new Object(); + e.tags = new Object(); + } + + if (l[0] == ":") + { + // Must split only on REAL spaces here, not just any old whitespace. + ary = l.match(/:([^ ]+) +(.*)/); + e.source = ary[1]; + l = ary[2]; + ary = e.source.match(/([^ ]+)!([^ ]+)@(.*)/); + if (ary) + { + e.user = new CIRCUser(this, null, ary[1], ary[2], ary[3]); + } + else + { + ary = e.source.match(/([^ ]+)@(.*)/); + if (ary) + { + e.user = new CIRCUser(this, null, ary[1], null, ary[2]); + } + else + { + ary = e.source.match(/([^ ]+)!(.*)/); + if (ary) + e.user = new CIRCUser(this, null, ary[1], ary[2], null); + } + } + } + + if (("user" in e) && e.user && e.tags.account) + { + e.user.account = e.tags.account; + } + + e.ignored = false; + if (("user" in e) && e.user && ("ignoreList" in this.parent)) + { + // Assumption: if "ignoreList" is in this.parent, we assume that: + // a) it's an array. + // b) ignoreMaskCache also exists, and + // c) it's an array too. + + if (!(e.source in this.parent.ignoreMaskCache)) + { + for (var m in this.parent.ignoreList) + { + if (hostmaskMatches(e.user, this.parent.ignoreList[m])) + { + e.ignored = true; + break; + } + } + /* Save this exact source in the cache, with results of tests. */ + this.parent.ignoreMaskCache[e.source] = e.ignored; + } + else + { + e.ignored = this.parent.ignoreMaskCache[e.source]; + } + } + + e.server = this; + + var sep = l.indexOf(" :"); + + if (sep != -1) /* <trailing> param, if there is one */ + { + var trail = l.substr (sep + 2, l.length); + e.params = l.substr(0, sep).split(/ +/); + e.params[e.params.length] = trail; + } + else + { + e.params = l.split(/ +/); + } + + e.decodeParam = decodeParam; + e.code = e.params[0].toUpperCase(); + + // Ignore all private (inc. channel) messages, notices and invites here. + if (e.ignored && ((e.code == "PRIVMSG") || (e.code == "NOTICE") || + (e.code == "INVITE") || (e.code == "TAGMSG"))) + return true; + + // If the message is part of a batch, store it for later. + if (this.batches && e.tags["batch"] && e.code != "BATCH") + { + var reftag = e.tags["batch"]; + // Check if the batch is already open. + // If not, ignore the incoming message. + if (this.batches[reftag]) + this.batches[reftag].messages.push(e); + return false; + } + + e.type = "parseddata"; + e.destObject = this; + e.destMethod = "onParsedData"; + + return true; +} + +/* + * onParsedData forwards to next event, based on |e.code| + */ +CIRCServer.prototype.onParsedData = +function serv_onParsedData(e) +{ + e.type = this.toLowerCase(e.code); + if (!e.code[0]) + { + dd (dumpObjectTree (e)); + return false; + } + + e.destMethod = "on" + e.code[0].toUpperCase() + + e.code.substr (1, e.code.length).toLowerCase(); + + if (typeof this[e.destMethod] == "function") + e.destObject = this; + else if (typeof this["onUnknown"] == "function") + e.destMethod = "onUnknown"; + else if (typeof this.parent[e.destMethod] == "function") + { + e.set = "network"; + e.destObject = this.parent; + } + else + { + e.set = "network"; + e.destObject = this.parent; + e.destMethod = "onUnknown"; + } + + return true; +} + +/* User changed topic */ +CIRCServer.prototype.onTopic = +function serv_topic (e) +{ + e.channel = new CIRCChannel(this, null, e.params[1]); + e.channel.topicBy = e.user.unicodeName; + e.channel.topicDate = new Date(); + e.channel.topic = toUnicode(e.params[2], e.channel); + e.destObject = e.channel; + e.set = "channel"; + + return true; +} + +/* Successful login */ +CIRCServer.prototype.on001 = +function serv_001 (e) +{ + this.parent.connectAttempt = 0; + this.parent.connectCandidate = 0; + //Mark capability negotiation as finished, if we haven't already. + delete this.parent.pendingCapNegotiation; + this.parent.state = NET_ONLINE; + // nextHost is incremented after picking a server. Push it back here. + this.parent.nextHost--; + + /* servers won't send a nick change notification if user was forced + * to change nick while logging in (eg. nick already in use.) We need + * to verify here that what the server thinks our name is, matches what + * we think it is. If not, the server wins. + */ + if (e.params[1] != e.server.me.encodedName) + { + renameProperty(e.server.users, e.server.me.collectionKey, + ":" + this.toLowerCase(e.params[1])); + e.server.me.changeNick(toUnicode(e.params[1], this)); + } + + /* Set up supports defaults here. + * This is so that we don't waste /huge/ amounts of RAM for the network's + * servers just because we know about them. Until we connect, that is. + * These defaults are taken from the draft 005 RPL_ISUPPORTS here: + * http://www.ietf.org/internet-drafts/draft-brocklesby-irc-isupport-02.txt + */ + this.supports = new Object(); + this.supports.modes = 3; + this.supports.maxchannels = 10; + this.supports.nicklen = 9; + this.supports.casemapping = "rfc1459"; + this.supports.channellen = 200; + this.supports.chidlen = 5; + /* Make sure it's possible to tell if we've actually got a 005 message. */ + this.supports.rpl_isupport = false; + this.channelTypes = [ '#', '&' ]; + /* This next one isn't in the isupport draft, but instead is defaulting to + * the codes we understand. It should be noted, some servers include the + * mode characters (o, h, v) in the 'a' list, although the draft spec says + * they should be treated as type 'b'. Luckly, in practise this doesn't + * matter, since both 'a' and 'b' types always take a parameter in the + * MODE message, and parsing is not affected. */ + this.channelModes = { + a: ['b'], + b: ['k'], + c: ['l'], + d: ['i', 'm', 'n', 'p', 's', 't'] + }; + // Default to support of v/+ and o/@ only. + this.userModes = [ + { mode: 'o', symbol: '@' }, + { mode: 'v', symbol: '+' } + ]; + // Assume the server supports no extra interesting commands. + this.servCmds = {}; + + if (this.parent.INITIAL_UMODE) + { + e.server.sendData("MODE " + e.server.me.encodedName + " :" + + this.parent.INITIAL_UMODE + "\n"); + } + + this.parent.users = this.users; + e.destObject = this.parent; + e.set = "network"; +} + +/* server features */ +CIRCServer.prototype.on005 = +function serv_005 (e) +{ + var oldCaseMapping = this.supports["casemapping"]; + /* Drop params 0 and 1. */ + for (var i = 2; i < e.params.length; i++) { + var itemStr = e.params[i]; + /* Items may be of the forms: + * NAME + * -NAME + * NAME=value + * Value may be empty on occasion. + * No value allowed for -NAME items. + */ + var item = itemStr.match(/^(-?)([A-Z]+)(=(.*))?$/i); + if (! item) + continue; + + var name = item[2].toLowerCase(); + if (("3" in item) && item[3]) + { + // And other items are stored as-is, though numeric items + // get special treatment to make our life easier later. + if (("4" in item) && item[4].match(/^\d+$/)) + this.supports[name] = Number(item[4]); + else + this.supports[name] = item[4]; + } + else + { + // Boolean-type items stored as 'true'. + this.supports[name] = !(("1" in item) && item[1] == "-"); + } + } + // Update all users and channels if the casemapping changed. + if (this.supports["casemapping"] != oldCaseMapping) + { + this.renameProperties(this.users, null); + this.renameProperties(this.channels, "users"); + } + + // Supported 'special' items: + // CHANTYPES (--> channelTypes{}), + // PREFIX (--> userModes[{mode,symbol}]), + // CHANMODES (--> channelModes{a:[], b:[], c:[], d:[]}). + + var m; + if ("chantypes" in this.supports) + { + this.channelTypes = []; + for (m = 0; m < this.supports.chantypes.length; m++) + this.channelTypes.push( this.supports.chantypes[m] ); + } + + if ("prefix" in this.supports) + { + var mlist = this.supports.prefix.match(/^\((.*)\)(.*)$/i); + if ((! mlist) || (mlist[1].length != mlist[2].length)) + { + dd ("** Malformed PREFIX entry in 005 SUPPORTS message **"); + } + else + { + this.userModes = []; + for (m = 0; m < mlist[1].length; m++) + this.userModes.push( { mode: mlist[1][m], + symbol: mlist[2][m] } ); + } + } + + if ("chanmodes" in this.supports) + { + var cmlist = this.supports.chanmodes.split(/,/); + if ((!cmlist) || (cmlist.length < 4)) + { + dd ("** Malformed CHANMODES entry in 005 SUPPORTS message **"); + } + else + { + // 4 types - list, set-unset-param, set-only-param, flag. + this.channelModes = { + a: cmlist[0].split(''), + b: cmlist[1].split(''), + c: cmlist[2].split(''), + d: cmlist[3].split('') + }; + } + } + + if ("cmds" in this.supports) + { + // Map this.supports.cmds [comma-list] into this.servCmds [props]. + var cmdlist = this.supports.cmds.split(/,/); + for (var i = 0; i < cmdlist.length; i++) + this.servCmds[cmdlist[i].toLowerCase()] = true; + } + + this.supports.rpl_isupport = true; + + e.destObject = this.parent; + e.set = "network"; + + return true; +} + +/* users */ +CIRCServer.prototype.on251 = +function serv_251(e) +{ + // 251 is the first message we get after 005, so it's now safe to do + // things that might depend upon server features. + + if (("namesx" in this.supports) && this.supports.namesx) + { + // "multi-prefix" is the same as "namesx" but PROTOCTL doesn't reply. + this.caps["multi-prefix"] = true; + this.sendData("PROTOCTL NAMESX\n"); + } + + if (this.parent.INITIAL_CHANNEL) + { + this.parent.primChan = this.addChannel(this.parent.INITIAL_CHANNEL); + this.parent.primChan.join(); + } + + e.destObject = this.parent; + e.set = "network"; +} + +/* channels */ +CIRCServer.prototype.on254 = +function serv_254(e) +{ + this.channelCount = e.params[2]; + e.destObject = this.parent; + e.set = "network"; +} + +/* user away message */ +CIRCServer.prototype.on301 = +function serv_301(e) +{ + e.user = new CIRCUser(this, null, e.params[2]); + e.user.awayMessage = e.decodeParam(3, e.user); + e.destObject = this.parent; + e.set = "network"; +} + +/* whois name */ +CIRCServer.prototype.on311 = +function serv_311 (e) +{ + e.user = new CIRCUser(this, null, e.params[2], e.params[3], e.params[4]); + e.user.desc = e.decodeParam(6, e.user); + e.destObject = this.parent; + e.set = "network"; + + this.pendingWhoisLines = e.user; +} + +/* whois server */ +CIRCServer.prototype.on312 = +function serv_312 (e) +{ + e.user = new CIRCUser(this, null, e.params[2]); + e.user.connectionHost = e.params[3]; + + e.destObject = this.parent; + e.set = "network"; +} + +/* whois idle time */ +CIRCServer.prototype.on317 = +function serv_317 (e) +{ + e.user = new CIRCUser(this, null, e.params[2]); + e.user.idleSeconds = e.params[3]; + + e.destObject = this.parent; + e.set = "network"; +} + +/* whois channel list */ +CIRCServer.prototype.on319 = +function serv_319(e) +{ + e.user = new CIRCUser(this, null, e.params[2]); + + e.destObject = this.parent; + e.set = "network"; +} + +/* end of whois */ +CIRCServer.prototype.on318 = +function serv_318(e) +{ + e.user = new CIRCUser(this, null, e.params[2]); + + if ("pendingWhoisLines" in this) + delete this.pendingWhoisLines; + + e.destObject = this.parent; + e.set = "network"; +} + +/* ircu's 330 numeric ("X is logged in as Y") */ +CIRCServer.prototype.on330 = +function serv_330(e) +{ + e.user = new CIRCUser(this, null, e.params[2]); + var account = (e.params[3] == "*" ? null : e.params[3]); + this.users[e.user.collectionKey].account = account; + + e.destObject = this.parent; + e.set = "network"; +} + +/* TOPIC reply - no topic set */ +CIRCServer.prototype.on331 = +function serv_331 (e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.channel.topic = ""; + e.destObject = e.channel; + e.set = "channel"; + + return true; +} + +/* TOPIC reply - topic set */ +CIRCServer.prototype.on332 = +function serv_332 (e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.channel.topic = toUnicode(e.params[3], e.channel); + e.destObject = e.channel; + e.set = "channel"; + + return true; +} + +/* topic information */ +CIRCServer.prototype.on333 = +function serv_333 (e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.channel.topicBy = toUnicode(e.params[3], this); + e.channel.topicDate = new Date(Number(e.params[4]) * 1000); + e.destObject = e.channel; + e.set = "channel"; + + return true; +} + +/* who reply */ +CIRCServer.prototype.on352 = +function serv_352 (e) +{ + e.userHasChanges = false; + if (this.LIGHTWEIGHT_WHO) + { + e.user = new CIRCUser(this, null, e.params[6]); + } + else + { + e.user = new CIRCUser(this, null, e.params[6], e.params[3], e.params[4]); + e.user.connectionHost = e.params[5]; + if (8 in e.params) + { + var ary = e.params[8].match(/(?:(\d+)\s)?(.*)/); + e.user.hops = ary[1]; + var desc = fromUnicode(ary[2], e.user); + if (e.user.desc != desc) + { + e.userHasChanges = true; + e.user.desc = desc; + } + } + } + var away = (e.params[7][0] == "G"); + if (e.user.isAway != away) + { + e.userHasChanges = true; + e.user.isAway = away; + } + + e.destObject = this.parent; + e.set = "network"; + + return true; +} + +/* extended who reply */ +CIRCServer.prototype.on354 = +function serv_354(e) +{ + // Discard if the type is not ours. + if (e.params[2] != this.WHOX_TYPE) + return; + + e.userHasChanges = false; + if (this.LIGHTWEIGHT_WHO) + { + e.user = new CIRCUser(this, null, e.params[7]); + } + else + { + e.user = new CIRCUser(this, null, e.params[7], e.params[4], e.params[5]); + e.user.connectionHost = e.params[6]; + // Hops is a separate parameter in WHOX. + e.user.hops = e.params[9]; + var account = (e.params[10] == "0" ? null : e.params[10]); + e.user.account = account; + if (11 in e.params) + { + var desc = e.decodeParam(11, e.user); + if (e.user.desc != desc) + { + e.userHasChanges = true; + e.user.desc = desc; + } + } + } + var away = (e.params[8][0] == "G"); + if (e.user.isAway != away) + { + e.userHasChanges = true; + e.user.isAway = away; + } + + e.destObject = this.parent; + e.set = "network"; + + return true; +} + +/* end of who */ +CIRCServer.prototype.on315 = +function serv_315 (e) +{ + e.user = new CIRCUser(this, null, e.params[1]); + e.destObject = this.parent; + e.set = "network"; + + return true; +} + +/* names reply */ +CIRCServer.prototype.on353 = +function serv_353 (e) +{ + e.channel = new CIRCChannel(this, null, e.params[3]); + if (e.channel.usersStable) + { + e.channel.users = new Object(); + e.channel.usersStable = false; + } + + e.destObject = e.channel; + e.set = "channel"; + + var nicks = e.params[4].split (" "); + var mList = this.userModes; + + for (var n in nicks) + { + var nick = nicks[n]; + if (nick == "") + break; + + var modes = new Array(); + var multiPrefix = (("namesx" in this.supports) && this.supports.namesx) + || (("multi-prefix" in this.caps) + && this.caps["multi-prefix"]); + do + { + var found = false; + for (var m in mList) + { + if (nick[0] == mList[m].symbol) + { + nick = nick.substr(1); + modes.push(mList[m].mode); + found = true; + break; + } + } + } while (found && multiPrefix); + + var ary = nick.match(/([^ ]+)!([^ ]+)@(.*)/); + var user = null; + var host = null; + + if (this.caps["userhost-in-names"] && ary) + { + nick = ary[1]; + user = ary[2]; + host = ary[3]; + } + + new CIRCChanUser(e.channel, null, nick, modes, true, user, host); + } + + return true; +} + +/* end of names */ +CIRCServer.prototype.on366 = +function serv_366 (e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.destObject = e.channel; + e.set = "channel"; + e.channel.usersStable = true; + + return true; +} + +/* channel time stamp? */ +CIRCServer.prototype.on329 = +function serv_329 (e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.destObject = e.channel; + e.set = "channel"; + e.channel.timeStamp = new Date (Number(e.params[3]) * 1000); + + return true; +} + +/* channel mode reply */ +CIRCServer.prototype.on324 = +function serv_324 (e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.destObject = this; + e.type = "chanmode"; + e.destMethod = "onChanMode"; + + return true; +} + +/* channel ban entry */ +CIRCServer.prototype.on367 = +function serv_367(e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.destObject = e.channel; + e.set = "channel"; + e.ban = e.params[3]; + e.user = new CIRCUser(this, null, e.params[4]); + e.banTime = new Date (Number(e.params[5]) * 1000); + + if (typeof e.channel.bans[e.ban] == "undefined") + { + e.channel.bans[e.ban] = {host: e.ban, user: e.user, time: e.banTime }; + var ban_evt = new CEvent("channel", "ban", e.channel, "onBan"); + ban_evt.tags = e.tags; + ban_evt.channel = e.channel; + ban_evt.ban = e.ban; + ban_evt.source = e.user; + this.parent.eventPump.addEvent(ban_evt); + } + + return true; +} + +/* channel ban list end */ +CIRCServer.prototype.on368 = +function serv_368(e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.destObject = e.channel; + e.set = "channel"; + + /* This flag is cleared in a timeout (which occurs right after the current + * message has been processed) so that the new event target (the channel) + * will still have the flag set when it executes. + */ + if ("pendingBanList" in e.channel) + setTimeout(function() { delete e.channel.pendingBanList; }, 0); + + return true; +} + +/* channel except entry */ +CIRCServer.prototype.on348 = +function serv_348(e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.destObject = e.channel; + e.set = "channel"; + e.except = e.params[3]; + e.user = new CIRCUser(this, null, e.params[4]); + e.exceptTime = new Date (Number(e.params[5]) * 1000); + + if (typeof e.channel.excepts[e.except] == "undefined") + { + e.channel.excepts[e.except] = {host: e.except, user: e.user, + time: e.exceptTime }; + } + + return true; +} + +/* channel except list end */ +CIRCServer.prototype.on349 = +function serv_349(e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.destObject = e.channel; + e.set = "channel"; + + if ("pendingExceptList" in e.channel) + setTimeout(function (){ delete e.channel.pendingExceptList; }, 0); + + return true; +} + +/* don't have operator perms */ +CIRCServer.prototype.on482 = +function serv_482(e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + e.destObject = e.channel; + e.set = "channel"; + + /* Some servers (e.g. Hybrid) don't let you get the except list without ops, + * so we might be waiting for this list forever otherwise. + */ + if ("pendingExceptList" in e.channel) + setTimeout(function (){ delete e.channel.pendingExceptList; }, 0); + + return true; +} + +/* userhost reply */ +CIRCServer.prototype.on302 = +function serv_302(e) +{ + var list = e.params[2].split(/\s+/); + + for (var i = 0; i < list.length; i++) + { + // <reply> ::= <nick>['*'] '=' <'+'|'-'><hostname> + // '*' == IRCop. '+' == here, '-' == away. + var data = list[i].match(/^(.*)(\*?)=([-+])(.*)@(.*)$/); + if (data) + this.addUser(data[1], data[4], data[5]); + } + + e.destObject = this.parent; + e.set = "network"; + + return true; +} + +/* CAP response */ +CIRCServer.prototype.onCap = +function my_cap (e) +{ + // We expect some sort of identifier. + if (e.params.length < 2) + return; + + if (e.params[2] == "LS") + { + /* We're getting a list of all server capabilities. Set them all to + * null (if they don't exist) to indicate we don't know if they're + * enabled or not (but this will evaluate to false which matches that + * capabilities are only enabled on request). + */ + var caps = e.params[3].split(/\s+/); + var multiline = (e.params[3] == "*"); + if (multiline) + caps = e.params[4].split(/\s+/); + + for (var i = 0; i < caps.length; i++) + { + var [cap, value] = caps[i].split(/=(.+)/); + cap = cap.replace(/^-/, "").trim(); + if (!(cap in this.caps)) + this.caps[cap] = null; + if (value) + this.capvals[cap] = value; + } + + // Don't do anything until the end of the response. + if (multiline) + return true; + + //Only request capabilities we support if we are connecting. + if (this.pendingCapNegotiation) + { + // If we have an STS upgrade policy, immediately disconnect + // and reconnect on the secure port. + if (this.parent.STS_MODULE.ENABLED && ("sts" in this.caps) && !this.isSecure) + { + var policy = this.parent.STS_MODULE.parseParameters(this.capvals["sts"]); + if (policy && policy.port) + { + e.stsUpgradePort = policy.port; + e.destObject = this.parent; + e.set = "network"; + return false; + } + } + + // Request STARTTLS if we are configured to do so. + if (!this.isSecure && ("tls" in this.caps) && this.parent.UPGRADE_INSECURE) + this.sendData("STARTTLS\n"); + + var caps_req = JSIRCV3_SUPPORTED_CAPS.filter(i => (i in this.caps)); + + // Don't send requests for these caps. + let caps_noreq = ["tls", "sts", "echo-message"]; + + if (!this.parent.USE_SASL) + caps_noreq.push("sasl"); + + caps_req = caps_req.filter(i => caps_noreq.indexOf(i) === -1); + + if (caps_req.length > 0) + { + caps_req = caps_req.join(" "); + e.server.sendData("CAP REQ :" + caps_req + "\n"); + } + else + { + e.server.sendData("CAP END\n"); + delete this.pendingCapNegotiation; + } + } + } + else if (e.params[2] == "LIST") + { + /* Received list of enabled capabilities. Just use this as a sanity + * check. */ + var caps = e.params[3].trim().split(/\s+/); + var multiline = (e.params[3] == "*"); + if (multiline) + caps = e.params[4].trim().split(/\s+/); + + for (var i = 0; i < caps.length; i++) + { + this.caps[caps[i]] = true; + } + + // Don't do anything until the end of the response. + if (multiline) + return true; + } + else if (e.params[2] == "ACK") + { + /* One or more capability changes have been successfully applied. An enabled + * capability is just "cap" whilst a disabled capability is "-cap". + */ + var caps = e.params[3].trim().split(/\s+/); + e.capsOn = new Array(); + e.capsOff = new Array(); + for (var i = 0; i < caps.length; i++) + { + var cap = caps[i].replace(/^-/,"").trim(); + var enabled = caps[i][0] != "-"; + if (enabled) + e.capsOn.push(cap); + else + e.capsOff.push(cap); + this.caps[cap] = enabled; + } + + // Try SASL authentication if we are configured to do so. + if (caps.indexOf("sasl") != -1) + { + var ev = new CEvent("server", "sasl-start", this, "onSASLStart"); + ev.server = this; + if (this.capvals["sasl"]) + ev.mechs = this.capvals["sasl"].toLowerCase().split(/,/); + ev.destObject = this.parent; + this.parent.eventPump.routeEvent(ev); + + if (this.pendingCapNegotiation) + return true; + } + + if (this.pendingCapNegotiation) + { + e.server.sendData("CAP END\n"); + delete this.pendingCapNegotiation; + + //Don't show the raw message while connecting. + return true; + } + } + else if (e.params[2] == "NAK") + { + // A capability change has failed. + var caps = e.params[3].trim().split(/\s+/); + e.caps = new Array(); + for (var i = 0; i < caps.length; i++) + { + var cap = caps[i].replace(/^-/, "").trim(); + e.caps.push(cap); + } + + if (this.pendingCapNegotiation) + { + e.server.sendData("CAP END\n"); + delete this.pendingCapNegotiation; + + //Don't show the raw message while connecting. + return true; + } + } + else if (e.params[2] == "NEW") + { + // A capability is now available, so request it if we can. + var caps = e.params[3].split(/\s+/); + e.newcaps = []; + for (var i = 0; i < caps.length; i++) + { + var [cap, value] = caps[i].split(/=(.+)/); + cap = cap.trim(); + this.caps[cap] = null; + e.newcaps.push(cap); + if (value) + this.capvals[cap] = value; + } + + var caps_req = JSIRCV3_SUPPORTED_CAPS.filter(i => (i in e.newcaps)); + + // Don't send requests for these caps. + caps_noreq = ["tls", "sts", "sasl", "echo-message"]; + caps_req = caps_req.filter(i => caps_noreq.indexOf(i) === -1); + + if (caps_req.length > 0) + { + caps_req = caps_req.join(" "); + e.server.sendData("CAP REQ :" + caps_req + "\n"); + } + } + else if (e.params[2] == "DEL") + { + // A capability is no longer available. + var caps = e.params[3].split(/\s+/); + var caps_nodel = ["sts"]; + for (var i = 0; i < caps.length; i++) + { + var cap = caps[i].split(/=(.+)/)[0]; + cap = cap.trim(); + + if (arrayContains(caps_nodel, cap)) + continue; + + this.caps[cap] = null; + } + } + else + { + dd("Unknown CAP reply " + e.params[2]); + } + + e.destObject = this.parent; + e.set = "network"; +} + +/* BATCH start or end */ +CIRCServer.prototype.onBatch = +function serv_batch(e) +{ + // We should at least get a ref tag. + if (e.params.length < 2) + return false; + + e.reftag = e.params[1].substring(1); + switch (e.params[1][0]) + { + case "+": + e.starting = true; + break; + case "-": + e.starting = false; + break; + default: + // Invalid reference tag. + return false; + } + var isPlayback = (this.batches && this.batches[e.reftag] && + this.batches[e.reftag].playback); + + if (!isPlayback) + { + if (e.starting) + { + // We're starting a batch, so we also need a type. + if (e.params.length < 3) + return false; + + if (!this.batches) + this.batches = new Object(); + // The batch object holds the messages queued up as part + // of this batch, and a boolean value indicating whether + // it is being played back. + var newBatch = new Object(); + newBatch.messages = [e]; + newBatch.type = e.params[2].toUpperCase(); + if (e.params[3] && (e.params[3] in this.channels)) + { + newBatch.destObject = this.channels[e.params[3]]; + } + else if (e.params[3] && (e.params[3] in this.users)) + { + newBatch.destObject = this.users[e.params[3]]; + } + else + { + newBatch.destObject = this.parent; + } + newBatch.playback = false; + this.batches[e.reftag] = newBatch; + } + else + { + if (!this.batches[e.reftag]) + { + // Got a close tag without an open tag, so ignore it. + return false; + } + + var batch = this.batches[e.reftag]; + + // Closing the batch, prepare for playback. + batch.messages.push(e); + batch.playback = true; + if (e.tags["batch"]) + { + // We are an inner batch. Append the message queue + // to the outer batch's message queue. + var parentRef = e.tags["batch"]; + var parentMsgs = this.batches[parentRef].messages; + parentMsgs = parentMsgs.concat(batch.messages); + } + else + { + // We are an outer batch. Playback! + for (var i = 0; i < batch.messages.length; i++) + { + var ev = batch.messages[i]; + ev.type = "parseddata"; + ev.destObject = this; + ev.destMethod = "onParsedData"; + this.parent.eventPump.routeEvent(ev); + } + } + } + return false; + } + else + { + // Batch command is ready for handling. + e.batchtype = this.batches[e.reftag].type; + e.destObject = this.batches[e.reftag].destObject; + if (e.destObject.TYPE == "CIRCChannel") + { + e.set = "channel"; + } + else + { + e.set = "network"; + } + + if (!e.starting) + { + // If we've reached the end of a batch in playback, + // do some cleanup. + delete this.batches[e.reftag]; + if (Object.entries(this.batches).length == 0) + delete this.batches; + } + + // Massage the batchtype into a method name for handlers: + // netsplit - onNetsplitBatch + // some-batch-type - onSomeBatchTypeBatch + // example.com/example - onExampleComExampleBatch + var batchCode = e.batchtype.split(/[\.\/-]/).map(function(s) + { + return s[0].toUpperCase() + s.substr(1).toLowerCase(); + }).join(""); + e.destMethod = "on" + batchCode + "Batch"; + + if (!e.destObject[e.destMethod]) + e.destMethod = "onUnknownBatch"; + } +} + +/* SASL authentication responses */ +CIRCServer.prototype.on902 = /* Nick locked */ +CIRCServer.prototype.on903 = /* Auth success */ +CIRCServer.prototype.on904 = /* Auth failed */ +CIRCServer.prototype.on905 = /* Command too long */ +CIRCServer.prototype.on906 = /* Aborted */ +CIRCServer.prototype.on907 = /* Already authenticated */ +CIRCServer.prototype.on908 = /* Mechanisms */ +function cap_on900(e) +{ + if (this.pendingCapNegotiation) + { + delete this.pendingCapNegotiation; + this.sendData("CAP END\n"); + } + + if (e.code == "908") + { + // Update our list of SASL mechanics. + this.capvals["sasl"] = e.params[2]; + } + + e.destObject = this.parent; + e.set = "network"; +} + +/* STARTTLS responses */ +CIRCServer.prototype.on670 = /* Success */ +function cap_on670(e) +{ + this.caps["tls"] = true; + e.server.connection.startTLS(); + e.server.isSecure = true; + e.server.isStartTLS = true; + + e.destObject = this.parent; + e.set = "network"; +} + +CIRCServer.prototype.on691 = /* Failure */ +function cap_on691(e) +{ + this.caps["tls"] = false; + + e.destObject = this.parent; + e.set = "network"; +} + +/* User away status changed */ +CIRCServer.prototype.onAway = +function serv_away(e) +{ + e.user.isAway = e.params[1] ? true : false; + e.destObject = this.parent; + e.set = "network"; +} + +/* User host changed */ +CIRCServer.prototype.onChghost = +function serv_chghost(e) +{ + this.users[e.user.collectionKey].name = e.params[1]; + this.users[e.user.collectionKey].host = e.params[2]; + e.destObject = this.parent; + e.set = "network"; +} + +/* user changed the mode */ +CIRCServer.prototype.onMode = +function serv_mode (e) +{ + e.destObject = this; + /* modes are not allowed in +channels -> no need to test that here.. */ + if (arrayIndexOf(this.channelTypes, e.params[1][0]) != -1) + { + e.channel = new CIRCChannel(this, null, e.params[1]); + if ("user" in e && e.user) + e.user = new CIRCChanUser(e.channel, e.user.unicodeName); + e.type = "chanmode"; + e.destMethod = "onChanMode"; + } + else + { + e.type = "usermode"; + e.destMethod = "onUserMode"; + } + + return true; +} + +CIRCServer.prototype.onUserMode = +function serv_usermode (e) +{ + e.user = new CIRCUser(this, null, e.params[1]) + e.user.modestr = e.params[2]; + e.destObject = this.parent; + e.set = "network"; + + // usermode usually happens on connect, after the MOTD, so it's a good + // place to kick off the lag timer. + this.updateLagTimer(); + + return true; +} + +CIRCServer.prototype.onChanMode = +function serv_chanmode (e) +{ + var modifier = ""; + var params_eaten = 0; + var BASE_PARAM; + + if (e.code.toUpperCase() == "MODE") + BASE_PARAM = 2; + else + if (e.code == "324") + BASE_PARAM = 3; + else + { + dd ("** INVALID CODE in ChanMode event **"); + return false; + } + + var mode_str = e.params[BASE_PARAM]; + params_eaten++; + + e.modeStr = mode_str; + e.usersAffected = new Array(); + + var nick; + var user; + var umList = this.userModes; + var cmList = this.channelModes; + var modeMap = this.canonicalChanModes; + var canonicalModeValue; + + for (var i = 0; i < mode_str.length ; i++) + { + /* Take care of modifier first. */ + if ((mode_str[i] == '+') || (mode_str[i] == '-')) + { + modifier = mode_str[i]; + continue; + } + + var done = false; + for (var m in umList) + { + if ((mode_str[i] == umList[m].mode) && (modifier != "")) + { + nick = e.params[BASE_PARAM + params_eaten]; + user = new CIRCChanUser(e.channel, null, nick, + [ modifier + umList[m].mode ]); + params_eaten++; + e.usersAffected.push (user); + done = true; + break; + } + } + if (done) + continue; + + // Update legacy canonical modes if necessary. + if (mode_str[i] in modeMap) + { + // Get the data in case we need it, but don't increment the counter. + var datacounter = BASE_PARAM + params_eaten; + var data = (datacounter in e.params) ? e.params[datacounter] : null; + canonicalModeValue = modeMap[mode_str[i]].getValue(modifier, data); + e.channel.mode[modeMap[mode_str[i]].name] = canonicalModeValue; + } + + if (arrayContains(cmList.a, mode_str[i])) + { + var data = e.params[BASE_PARAM + params_eaten++]; + if (modifier == "+") + { + e.channel.mode.modeA[data] = true; + } + else + { + if (data in e.channel.mode.modeA) + { + delete e.channel.mode.modeA[data]; + } + else + { + dd("** Trying to remove channel mode '" + mode_str[i] + + "'/'" + data + "' which does not exist in list."); + } + } + } + else if (arrayContains(cmList.b, mode_str[i])) + { + var data = e.params[BASE_PARAM + params_eaten++]; + if (modifier == "+") + { + e.channel.mode.modeB[mode_str[i]] = data; + } + else + { + // Save 'null' even though we have some data. + e.channel.mode.modeB[mode_str[i]] = null; + } + } + else if (arrayContains(cmList.c, mode_str[i])) + { + if (modifier == "+") + { + var data = e.params[BASE_PARAM + params_eaten++]; + e.channel.mode.modeC[mode_str[i]] = data; + } + else + { + e.channel.mode.modeC[mode_str[i]] = null; + } + } + else if (arrayContains(cmList.d, mode_str[i])) + { + e.channel.mode.modeD[mode_str[i]] = (modifier == "+"); + } + else + { + dd("** UNKNOWN mode symbol '" + mode_str[i] + "' in ChanMode event **"); + } + } + + e.destObject = e.channel; + e.set = "channel"; + return true; +} + +CIRCServer.prototype.onNick = +function serv_nick (e) +{ + var newNick = e.params[1]; + var newKey = ":" + this.toLowerCase(newNick); + var oldKey = e.user.collectionKey; + var ev; + + renameProperty (this.users, oldKey, newKey); + e.oldNick = e.user.unicodeName; + e.user.changeNick(toUnicode(newNick, this)); + + for (var c in this.channels) + { + if (this.channels[c].active && + ((oldKey in this.channels[c].users) || e.user == this.me)) + { + var cuser = this.channels[c].users[oldKey]; + renameProperty (this.channels[c].users, oldKey, newKey); + + // User must be a channel user, update sort name for userlist, + // before we route the event further: + cuser.updateSortName(); + + ev = new CEvent ("channel", "nick", this.channels[c], "onNick"); + ev.tags = e.tags; + ev.channel = this.channels[c]; + ev.user = cuser; + ev.server = this; + ev.oldNick = e.oldNick; + this.parent.eventPump.routeEvent(ev); + } + } + + if (e.user == this.me) + { + /* if it was me, tell the network about the nick change as well */ + ev = new CEvent ("network", "nick", this.parent, "onNick"); + ev.tags = e.tags; + ev.user = e.user; + ev.server = this; + ev.oldNick = e.oldNick; + this.parent.eventPump.routeEvent(ev); + } + + e.destObject = e.user; + e.set = "user"; + + return true; +} + +CIRCServer.prototype.onQuit = +function serv_quit (e) +{ + var reason = e.decodeParam(1); + + for (var c in e.server.channels) + { + if (e.server.channels[c].active && + e.user.collectionKey in e.server.channels[c].users) + { + var ev = new CEvent ("channel", "quit", e.server.channels[c], + "onQuit"); + ev.tags = e.tags; + ev.user = e.server.channels[c].users[e.user.collectionKey]; + ev.channel = e.server.channels[c]; + ev.server = ev.channel.parent; + ev.reason = reason; + this.parent.eventPump.routeEvent(ev); + delete e.server.channels[c].users[e.user.collectionKey]; + } + } + + this.users[e.user.collectionKey].lastQuitMessage = reason; + this.users[e.user.collectionKey].lastQuitDate = new Date(); + + // 0 == prune onQuit. + if (this.PRUNE_OLD_USERS == 0) + delete this.users[e.user.collectionKey]; + + e.reason = reason; + e.destObject = e.user; + e.set = "user"; + + return true; +} + +CIRCServer.prototype.onPart = +function serv_part (e) +{ + e.channel = new CIRCChannel(this, null, e.params[1]); + e.reason = (e.params.length > 2) ? e.decodeParam(2, e.channel) : ""; + e.user = new CIRCChanUser(e.channel, e.user.unicodeName); + if (userIsMe(e.user)) + { + e.channel.active = false; + e.channel.joined = false; + } + e.channel.removeUser(e.user.encodedName); + e.destObject = e.channel; + e.set = "channel"; + + return true; +} + +CIRCServer.prototype.onKick = +function serv_kick (e) +{ + e.channel = new CIRCChannel(this, null, e.params[1]); + e.lamer = new CIRCChanUser(e.channel, null, e.params[2]); + delete e.channel.users[e.lamer.collectionKey]; + if (userIsMe(e.lamer)) + { + e.channel.active = false; + e.channel.joined = false; + } + e.reason = e.decodeParam(3, e.channel); + e.destObject = e.channel; + e.set = "channel"; + + return true; +} + +CIRCServer.prototype.onJoin = +function serv_join(e) +{ + e.channel = new CIRCChannel(this, null, e.params[1]); + // Passing undefined here because CIRCChanUser doesn't like "null" + e.user = new CIRCChanUser(e.channel, e.user.unicodeName, null, + undefined, true); + + if (e.params[2] && e.params[3]) + { + var account = (e.params[2] == "*" ? null : e.params[2]); + var desc = e.decodeParam([3], e.user); + this.users[e.user.collectionKey].account = account; + this.users[e.user.collectionKey].desc = desc; + } + + if (userIsMe(e.user)) + { + var delayFn1 = function(t) { + if (!e.channel.active) + return; + + // Give us the channel mode! + e.server.sendData("MODE " + e.channel.encodedName + "\n"); + }; + // Between 1s - 3s. + setTimeout(delayFn1, 1000 + 2000 * Math.random(), this); + + var delayFn2 = function(t) { + if (!e.channel.active) + return; + + // Get a full list of bans and exceptions, if supported. + if (arrayContains(t.channelModes.a, "b")) + { + e.server.sendData("MODE " + e.channel.encodedName + " +b\n"); + e.channel.pendingBanList = true; + } + if (arrayContains(t.channelModes.a, "e")) + { + e.server.sendData("MODE " + e.channel.encodedName + " +e\n"); + e.channel.pendingExceptList = true; + } + + //If away-notify is active, query the list of users for away status. + if (e.server.caps["away-notify"]) + { + // If the server supports extended who, use it. + // This lets us initialize the account property. + if (e.server.supports["whox"]) + e.server.who(e.channel.unicodeName + " %acdfhnrstu," + e.server.WHOX_TYPE); + else + e.server.who(e.channel.unicodeName); + } + }; + // Between 10s - 20s. + setTimeout(delayFn2, 10000 + 10000 * Math.random(), this); + + /* Clean up the topic, since servers don't always send RPL_NOTOPIC + * (no topic set) when joining a channel without a topic. In fact, + * the RFC even fails to mention sending a RPL_NOTOPIC after a join! + */ + e.channel.topic = ""; + e.channel.topicBy = null; + e.channel.topicDate = null; + + // And we're in! + e.channel.active = true; + e.channel.joined = true; + } + + e.destObject = e.channel; + e.set = "channel"; + + return true; +} + +CIRCServer.prototype.onAccount = +function serv_acct(e) +{ + var account = (e.params[1] == "*" ? null : e.params[1]); + this.users[e.user.collectionKey].account = account; + + return true; +} + +CIRCServer.prototype.onPing = +function serv_ping (e) +{ + /* non-queued send, so we can calcualte lag */ + this.connection.sendData("PONG :" + e.params[1] + "\n"); + this.updateLagTimer(); + e.destObject = this.parent; + e.set = "network"; + + return true; +} + +CIRCServer.prototype.onPong = +function serv_pong (e) +{ + if (e.params[2] != "LAGTIMER") + return true; + + if (this.lastPingSent) + this.lag = (new Date() - this.lastPingSent) / 1000; + + this.lastPingSent = null; + + e.destObject = this.parent; + e.set = "network"; + + return true; +} + +CIRCServer.prototype.onInvite = +function serv_invite(e) +{ + e.channel = new CIRCChannel(this, null, e.params[2]); + + e.destObject = this.parent; + e.set = "network"; +} + +CIRCServer.prototype.onNotice = +CIRCServer.prototype.onPrivmsg = +CIRCServer.prototype.onTagmsg = +function serv_notice_privmsg (e) +{ + var targetName = e.params[1]; + + if (this.userModes) + { + // Strip off one (and only one) user mode prefix. + for (var i = 0; i < this.userModes.length; i++) + { + if (targetName[0] == this.userModes[i].symbol) + { + e.msgPrefix = this.userModes[i]; + targetName = targetName.substr(1); + break; + } + } + } + + /* setting replyTo provides a standard place to find the target for */ + /* replies associated with this event. */ + if (arrayIndexOf(this.channelTypes, targetName[0]) != -1) + { + e.channel = new CIRCChannel(this, null, targetName); + if ("user" in e) + e.user = new CIRCChanUser(e.channel, e.user.unicodeName); + e.replyTo = e.channel; + e.set = "channel"; + } + else if (!("user" in e)) + { + e.set = "network"; + e.destObject = this.parent; + return true; + } + else + { + e.set = "user"; + e.replyTo = e.user; /* send replies to the user who sent the message */ + } + + /* The capability identify-msg adds a + or - in front the message to + * indicate their network registration status. + */ + if (("identify-msg" in this.caps) && this.caps["identify-msg"]) + { + e.identifyMsg = false; + var flag = e.params[2].substring(0,1); + if (flag == "+") + { + e.identifyMsg = true; + e.params[2] = e.params[2].substring(1); + } + else if (flag == "-") + { + e.params[2] = e.params[2].substring(1); + } + else + { + // Just print to console on failure - or we'd spam the user + dd("Warning: IDENTIFY-MSG is on, but there's no message flags"); + } + } + + // TAGMSG doesn't have a message parameter, so just pass it on. + if (e.code == "TAGMSG") + { + e.destObject = e.replyTo; + return true; + } + + if (e.params[2].search (/^\x01[^ ]+.*\x01$/) != -1) + { + if (e.code == "NOTICE") + { + e.type = "ctcp-reply"; + e.destMethod = "onCTCPReply"; + } + else // e.code == "PRIVMSG" + { + e.type = "ctcp"; + e.destMethod = "onCTCP"; + } + e.set = "server"; + e.destObject = this; + } + else + { + e.msg = e.decodeParam(2, e.replyTo); + e.destObject = e.replyTo; + } + + return true; +} + +CIRCServer.prototype.onWallops = +function serv_wallops(e) +{ + if (("user" in e) && e.user) + { + e.msg = e.decodeParam(1, e.user); + e.replyTo = e.user; + } + else + { + e.msg = e.decodeParam(1); + e.replyTo = this; + } + + e.destObject = this.parent; + e.set = "network"; + + return true; +} + +CIRCServer.prototype.onCTCPReply = +function serv_ctcpr (e) +{ + var ary = e.params[2].match (/^\x01([^ ]+) ?(.*)\x01$/i); + + if (ary == null) + return false; + + e.CTCPData = ary[2] ? ary[2] : ""; + + e.CTCPCode = ary[1].toLowerCase(); + e.type = "ctcp-reply-" + e.CTCPCode; + e.destMethod = "onCTCPReply" + ary[1][0].toUpperCase() + + ary[1].substr (1, ary[1].length).toLowerCase(); + + if (typeof this[e.destMethod] != "function") + { /* if there's no place to land the event here, try to forward it */ + e.destObject = this.parent; + e.set = "network"; + + if (typeof e.destObject[e.destMethod] != "function") + { /* if there's no place to forward it, send it to unknownCTCP */ + e.type = "unk-ctcp-reply"; + e.destMethod = "onUnknownCTCPReply"; + if (e.destMethod in this) + { + e.set = "server"; + e.destObject = this; + } + else + { + e.set = "network"; + e.destObject = this.parent; + } + } + } + else + e.destObject = this; + + return true; +} + +CIRCServer.prototype.onCTCP = +function serv_ctcp (e) +{ + var ary = e.params[2].match (/^\x01([^ ]+) ?(.*)\x01$/i); + + if (ary == null) + return false; + + e.CTCPData = ary[2] ? ary[2] : ""; + + e.CTCPCode = ary[1].toLowerCase(); + if (e.CTCPCode.search (/^reply/i) == 0) + { + dd ("dropping spoofed reply."); + return false; + } + + e.CTCPCode = toUnicode(e.CTCPCode, e.replyTo); + e.CTCPData = toUnicode(e.CTCPData, e.replyTo); + + e.type = "ctcp-" + e.CTCPCode; + e.destMethod = "onCTCP" + ary[1][0].toUpperCase() + + ary[1].substr (1, ary[1].length).toLowerCase(); + + if (typeof this[e.destMethod] != "function") + { /* if there's no place to land the event here, try to forward it */ + e.destObject = e.replyTo; + e.set = (e.replyTo == e.user) ? "user" : "channel"; + + if (typeof e.replyTo[e.destMethod] != "function") + { /* if there's no place to forward it, send it to unknownCTCP */ + e.type = "unk-ctcp"; + e.destMethod = "onUnknownCTCP"; + } + } + else + e.destObject = this; + + var ev = new CEvent("server", "ctcp-receive", this, "onReceiveCTCP"); + ev.tags = e.tags; + ev.server = this; + ev.CTCPCode = e.CTCPCode; + ev.CTCPData = e.CTCPData; + ev.type = e.type; + ev.user = e.user; + ev.destObject = this.parent; + this.parent.eventPump.addEvent(ev); + + return true; +} + +CIRCServer.prototype.onCTCPClientinfo = +function serv_ccinfo (e) +{ + var clientinfo = new Array(); + + if (e.CTCPData) + { + var cmdName = "onCTCP" + e.CTCPData[0].toUpperCase() + + e.CTCPData.substr (1, e.CTCPData.length).toLowerCase(); + var helpName = cmdName.replace(/^onCTCP/, "CTCPHelp"); + + // Check we support the command. + if (cmdName in this) + { + // Do we have help for it? + if (helpName in this) + { + var msg; + if (typeof this[helpName] == "function") + msg = this[helpName](); + else + msg = this[helpName]; + + e.user.ctcp("CLIENTINFO", msg, "NOTICE"); + } + else + { + e.user.ctcp("CLIENTINFO", + getMsg(MSG_ERR_NO_CTCP_HELP, e.CTCPData), "NOTICE"); + } + } + else + { + e.user.ctcp("CLIENTINFO", + getMsg(MSG_ERR_NO_CTCP_CMD, e.CTCPData), "NOTICE"); + } + return true; + } + + for (var fname in this) + { + var ary = fname.match(/^onCTCP(.+)/); + if (ary && ary[1].search(/^Reply/) == -1) + clientinfo.push (ary[1].toUpperCase()); + } + + e.user.ctcp("CLIENTINFO", clientinfo.join(" "), "NOTICE"); + + return true; +} + +CIRCServer.prototype.onCTCPAction = +function serv_cact (e) +{ + e.destObject = e.replyTo; + e.set = (e.replyTo == e.user) ? "user" : "channel"; +} + +CIRCServer.prototype.onCTCPFinger = +function serv_cfinger (e) +{ + e.user.ctcp("FINGER", this.parent.INITIAL_DESC, "NOTICE"); + return true; +} + +CIRCServer.prototype.onCTCPTime = +function serv_cping (e) +{ + e.user.ctcp("TIME", new Date(), "NOTICE"); + + return true; +} + +CIRCServer.prototype.onCTCPVersion = +function serv_cver (e) +{ + var lines = e.server.VERSION_RPLY.split ("\n"); + + for (var i in lines) + e.user.ctcp("VERSION", lines[i], "NOTICE"); + + e.destObject = e.replyTo; + e.set = (e.replyTo == e.user) ? "user" : "channel"; + + return true; +} + +CIRCServer.prototype.onCTCPSource = +function serv_csrc (e) +{ + e.user.ctcp("SOURCE", this.SOURCE_RPLY, "NOTICE"); + + return true; +} + +CIRCServer.prototype.onCTCPOs = +function serv_os(e) +{ + e.user.ctcp("OS", this.OS_RPLY, "NOTICE"); + + return true; +} + +CIRCServer.prototype.onCTCPHost = +function serv_host(e) +{ + e.user.ctcp("HOST", this.HOST_RPLY, "NOTICE"); + + return true; +} + +CIRCServer.prototype.onCTCPPing = +function serv_cping (e) +{ + /* non-queued send */ + this.connection.sendData("NOTICE " + e.user.encodedName + " :\01PING " + + e.CTCPData + "\01\n"); + e.destObject = e.replyTo; + e.set = (e.replyTo == e.user) ? "user" : "channel"; + + return true; +} + +CIRCServer.prototype.onCTCPDcc = +function serv_dcc (e) +{ + var ary = e.CTCPData.match (/([^ ]+)? ?(.*)/); + + e.DCCData = ary[2]; + e.type = "dcc-" + ary[1].toLowerCase(); + e.destMethod = "onDCC" + ary[1][0].toUpperCase() + + ary[1].substr (1, ary[1].length).toLowerCase(); + + if (typeof this[e.destMethod] != "function") + { /* if there's no place to land the event here, try to forward it */ + e.destObject = e.replyTo; + e.set = (e.replyTo == e.user) ? "user" : "channel"; + } + else + e.destObject = this; + + return true; +} + +CIRCServer.prototype.onDCCChat = +function serv_dccchat (e) +{ + var ary = e.DCCData.match (/(chat) (\d+) (\d+)/i); + + if (ary == null) + return false; + + e.id = ary[2]; + // Longword --> dotted IP conversion. + var host = Number(e.id); + e.host = ((host >> 24) & 0xFF) + "." + + ((host >> 16) & 0xFF) + "." + + ((host >> 8) & 0xFF) + "." + + (host & 0xFF); + e.port = Number(ary[3]); + e.destObject = e.replyTo; + e.set = (e.replyTo == e.user) ? "user" : "channel"; + + return true; +} + +CIRCServer.prototype.onDCCSend = +function serv_dccsend (e) +{ + var ary = e.DCCData.match(/([^ ]+) (\d+) (\d+) (\d+)/); + + /* Just for mIRC: filenames with spaces may be enclosed in double-quotes. + * (though by default it replaces spaces with underscores, but we might as + * well cope). */ + if ((ary[1][0] == '"') || (ary[1][ary[1].length - 1] == '"')) + ary = e.DCCData.match(/"(.+)" (\d+) (\d+) (\d+)/); + + if (ary == null) + return false; + + e.file = ary[1]; + e.id = ary[2]; + // Longword --> dotted IP conversion. + var host = Number(e.id); + e.host = ((host >> 24) & 0xFF) + "." + + ((host >> 16) & 0xFF) + "." + + ((host >> 8) & 0xFF) + "." + + (host & 0xFF); + e.port = Number(ary[3]); + e.size = Number(ary[4]); + e.destObject = e.replyTo; + e.set = (e.replyTo == e.user) ? "user" : "channel"; + + return true; +} + +function CIRCChannel(parent, unicodeName, encodedName) +{ + // Both unicodeName and encodedName are optional, but at least one must be + // present. + + if (!encodedName && !unicodeName) + throw "Hey! Come on, I need either an encoded or a Unicode name."; + if (!encodedName) + encodedName = fromUnicode(unicodeName, parent); + + let collectionKey = ":" + parent.toLowerCase(encodedName); + if (collectionKey in parent.channels) + return parent.channels[collectionKey]; + + this.parent = parent; + this.encodedName = encodedName; + this.canonicalName = collectionKey.substr(1); + this.collectionKey = collectionKey; + this.unicodeName = unicodeName || toUnicode(encodedName, this); + this.viewName = this.unicodeName; + + this.users = new Object(); + this.bans = new Object(); + this.excepts = new Object(); + this.mode = new CIRCChanMode(this); + this.usersStable = true; + /* These next two flags represent a subtle difference in state: + * active - in the channel, from the server's point of view. + * joined - in the channel, from the user's point of view. + * e.g. parting the channel clears both, but being disconnected only + * clears |active| - the user still wants to be in the channel, even + * though they aren't physically able to until we've reconnected. + */ + this.active = false; + this.joined = false; + + this.parent.channels[this.collectionKey] = this; + if ("onInit" in this) + this.onInit(); + + return this; +} + +CIRCChannel.prototype.TYPE = "IRCChannel"; +CIRCChannel.prototype.topic = ""; + +// Returns the IRC URL representation of this channel. +CIRCChannel.prototype.getURL = +function chan_geturl() +{ + var target = this.encodedName; + var flags = this.mode.key ? ["needkey"] : []; + + if ((target[0] == "#") && (target.length > 1) && + arrayIndexOf(this.parent.channelTypes, target[1]) == -1) + { + /* First character is "#" (which we're allowed to omit), and the + * following character is NOT a valid prefix, so it's safe to remove. + */ + target = target.substr(1); + } + return this.parent.parent.getURL(target, flags); +} + +CIRCChannel.prototype.rehome = +function chan_rehome(newParent) +{ + delete this.parent.channels[this.collectionKey]; + this.parent = newParent; + this.parent.channels[this.collectionKey] = this; +} + +CIRCChannel.prototype.addUser = +function chan_adduser (unicodeName, modes) +{ + return new CIRCChanUser(this, unicodeName, null, modes); +} + +CIRCChannel.prototype.getUser = +function chan_getuser(nick) +{ + // Try assuming it's an encodedName first. + let tnick = ":" + this.parent.toLowerCase(nick); + if (tnick in this.users) + return this.users[tnick]; + + // Ok, failed, so try assuming it's a unicodeName. + tnick = ":" + this.parent.toLowerCase(fromUnicode(nick, this.parent)); + if (tnick in this.users) + return this.users[tnick]; + + return null; +} + +CIRCChannel.prototype.removeUser = +function chan_removeuser(nick) +{ + // Try assuming it's an encodedName first. + let key = ":" + this.parent.toLowerCase(nick); + if (key in this.users) + delete this.users[key]; // see ya + + // Ok, failed, so try assuming it's a unicodeName. + key = ":" + this.parent.toLowerCase(fromUnicode(nick, this.parent)); + if (key in this.users) + delete this.users[key]; +} + +CIRCChannel.prototype.getUsersLength = +function chan_userslen (mode) +{ + var i = 0; + var p; + this.opCount = 0; + this.halfopCount = 0; + this.voiceCount = 0; + + if (typeof mode == "undefined") + { + for (p in this.users) + { + if (this.users[p].isOp) + this.opCount++; + if (this.users[p].isHalfOp) + this.halfopCount++; + if (this.users[p].isVoice) + this.voiceCount++; + i++; + } + } + else + { + for (p in this.users) + if (arrayContains(this.users[p].modes, mode)) + i++; + } + + return i; +} + +CIRCChannel.prototype.iAmOp = +function chan_amop() +{ + return this.active && this.users[this.parent.me.collectionKey].isOp; +} + +CIRCChannel.prototype.iAmHalfOp = +function chan_amhalfop() +{ + return this.active && this.users[this.parent.me.collectionKey].isHalfOp; +} + +CIRCChannel.prototype.iAmVoice = +function chan_amvoice() +{ + return this.active && this.users[this.parent.me.collectionKey].isVoice; +} + +CIRCChannel.prototype.setTopic = +function chan_topic (str) +{ + this.parent.sendData ("TOPIC " + this.encodedName + " :" + + fromUnicode(str, this) + "\n"); +} + +CIRCChannel.prototype.say = +function chan_say (msg) +{ + this.parent.sayTo(this.encodedName, fromUnicode(msg, this)); +} + +CIRCChannel.prototype.act = +function chan_say (msg) +{ + this.parent.actTo(this.encodedName, fromUnicode(msg, this)); +} + +CIRCChannel.prototype.notice = +function chan_notice (msg) +{ + this.parent.noticeTo(this.encodedName, fromUnicode(msg, this)); +} + +CIRCChannel.prototype.ctcp = +function chan_ctcpto (code, msg, type) +{ + msg = msg || ""; + type = type || "PRIVMSG"; + + this.parent.ctcpTo(this.encodedName, fromUnicode(code, this), + fromUnicode(msg, this), type); +} + +CIRCChannel.prototype.join = +function chan_join (key) +{ + if (!key) + key = ""; + + this.parent.sendData ("JOIN " + this.encodedName + " " + key + "\n"); + return true; +} + +CIRCChannel.prototype.part = +function chan_part (reason) +{ + if (!reason) + reason = ""; + this.parent.sendData ("PART " + this.encodedName + " :" + + fromUnicode(reason, this) + "\n"); + this.users = new Object(); + return true; +} + +/** + * Invites a user to a channel. + * + * @param nick the user name to invite. + */ +CIRCChannel.prototype.invite = +function chan_inviteuser (nick) +{ + var rawNick = fromUnicode(nick, this.parent); + this.parent.sendData("INVITE " + rawNick + " " + this.encodedName + "\n"); + return true; +} + +CIRCChannel.prototype.findUsers = +function chan_findUsers(mask) +{ + var ary = []; + var unchecked = 0; + mask = getHostmaskParts(mask); + for (var nick in this.users) + { + var user = this.users[nick]; + if (!user.host || !user.name) + unchecked++; + else if (hostmaskMatches(user, mask)) + ary.push(user); + } + return { users: ary, unchecked: unchecked }; +} + +/** + * Stores a channel's current mode settings. + * + * You should never need to create an instance of this prototype; access the + * channel mode information through the |CIRCChannel.mode| property. + * + * @param parent The |CIRCChannel| to which this mode belongs. + */ +function CIRCChanMode (parent) +{ + this.parent = parent; + + this.modeA = new Object(); + this.modeB = new Object(); + this.modeC = new Object(); + this.modeD = new Object(); + + this.invite = false; + this.moderated = false; + this.publicMessages = true; + this.publicTopic = true; + this.secret = false; + this.pvt = false; + this.key = ""; + this.limit = -1; +} + +CIRCChanMode.prototype.TYPE = "IRCChanMode"; + +// Returns the complete mode string, as constructed from its component parts. +CIRCChanMode.prototype.getModeStr = +function chan_modestr (f) +{ + var str = ""; + var modeCparams = ""; + + /* modeA are 'list' ones, and so should not be shown. + * modeB are 'param' ones, like +k key, so we wont show them either. + * modeC are 'on-param' ones, like +l limit, which we will show. + * modeD are 'boolean' ones, which we will definitely show. + */ + + // Add modeD: + for (var m in this.modeD) + { + if (this.modeD[m]) + str += m; + } + + // Add modeC, save parameters for adding all the way at the end: + for (var m in this.modeC) + { + if (this.modeC[m]) + { + str += m; + modeCparams += " " + this.modeC[m]; + } + } + + // Add parameters: + if (str) + str = "+" + str + modeCparams; + + return str; +} + +// Sends the given mode string to the server with the channel pre-filled. +CIRCChanMode.prototype.setMode = +function chanm_mode (modestr) +{ + this.parent.parent.sendData ("MODE " + this.parent.encodedName + " " + + modestr + "\n"); + + return true; +} + +// Sets (|n| > 0) or clears (|n| <= 0) the user count limit. +CIRCChanMode.prototype.setLimit = +function chanm_limit (n) +{ + if ((typeof n == "undefined") || (n <= 0)) + { + this.parent.parent.sendData("MODE " + this.parent.encodedName + + " -l\n"); + } + else + { + this.parent.parent.sendData("MODE " + this.parent.encodedName + " +l " + + Number(n) + "\n"); + } + + return true; +} + +// Locks the channel with a given key. +CIRCChanMode.prototype.lock = +function chanm_lock (k) +{ + this.parent.parent.sendData("MODE " + this.parent.encodedName + " +k " + + k + "\n"); + return true; +} + +// Unlocks the channel with a given key. +CIRCChanMode.prototype.unlock = +function chan_unlock (k) +{ + this.parent.parent.sendData("MODE " + this.parent.encodedName + " -k " + + k + "\n"); + return true; +} + +// Sets or clears the moderation mode. +CIRCChanMode.prototype.setModerated = +function chan_moderate (f) +{ + var modifier = (f) ? "+" : "-"; + + this.parent.parent.sendData("MODE " + this.parent.encodedName + " " + + modifier + "m\n"); + return true; +} + +// Sets or clears the allow public messages mode. +CIRCChanMode.prototype.setPublicMessages = +function chan_pmessages (f) +{ + var modifier = (f) ? "-" : "+"; + + this.parent.parent.sendData("MODE " + this.parent.encodedName + " " + + modifier + "n\n"); + return true; +} + +// Sets or clears the public topic mode. +CIRCChanMode.prototype.setPublicTopic = +function chan_ptopic (f) +{ + var modifier = (f) ? "-" : "+"; + + this.parent.parent.sendData("MODE " + this.parent.encodedName + " " + + modifier + "t\n"); + return true; +} + +// Sets or clears the invite-only mode. +CIRCChanMode.prototype.setInvite = +function chan_invite (f) +{ + var modifier = (f) ? "+" : "-"; + + this.parent.parent.sendData("MODE " + this.parent.encodedName + " " + + modifier + "i\n"); + return true; +} + +// Sets or clears the private channel mode. +CIRCChanMode.prototype.setPvt = +function chan_pvt (f) +{ + var modifier = (f) ? "+" : "-"; + + this.parent.parent.sendData("MODE " + this.parent.encodedName + " " + + modifier + "p\n"); + return true; +} + +// Sets or clears the secret channel mode. +CIRCChanMode.prototype.setSecret = +function chan_secret (f) +{ + var modifier = (f) ? "+" : "-"; + + this.parent.parent.sendData("MODE " + this.parent.encodedName + " " + + modifier + "s\n"); + return true; +} + +function CIRCUser(parent, unicodeName, encodedName, name, host) +{ + // Both unicodeName and encodedName are optional, but at least one must be + // present. + + if (!encodedName && !unicodeName) + throw "Hey! Come on, I need either an encoded or a Unicode name."; + if (!encodedName) + encodedName = fromUnicode(unicodeName, parent); + + let collectionKey = ":" + parent.toLowerCase(encodedName); + if (collectionKey in parent.users) + { + let existingUser = parent.users[collectionKey]; + if (name) + existingUser.name = name; + if (host) + existingUser.host = host; + return existingUser; + } + + this.parent = parent; + this.encodedName = encodedName; + this.canonicalName = collectionKey.substr(1); + this.collectionKey = collectionKey; + this.unicodeName = unicodeName || toUnicode(encodedName, this.parent); + this.viewName = this.unicodeName; + + this.name = name; + this.host = host; + this.desc = ""; + this.account = null; + this.connectionHost = null; + this.isAway = false; + this.modestr = this.parent.parent.INITIAL_UMODE; + + this.parent.users[this.collectionKey] = this; + if ("onInit" in this) + this.onInit(); + + return this; +} + +CIRCUser.prototype.TYPE = "IRCUser"; + +// Returns the IRC URL representation of this user. +CIRCUser.prototype.getURL = +function usr_geturl() +{ + return this.parent.parent.getURL(this.encodedName, ["isnick"]); +} + +CIRCUser.prototype.rehome = +function usr_rehome(newParent) +{ + delete this.parent.users[this.collectionKey]; + this.parent = newParent; + this.parent.users[this.collectionKey] = this; +} + +CIRCUser.prototype.changeNick = +function usr_changenick(unicodeName) +{ + this.unicodeName = unicodeName; + this.viewName = this.unicodeName; + this.encodedName = fromUnicode(this.unicodeName, this.parent); + this.canonicalName = this.parent.toLowerCase(this.encodedName); + this.collectionKey = ":" + this.canonicalName; +} + +CIRCUser.prototype.getHostMask = +function usr_hostmask (pfx) +{ + pfx = (typeof pfx != "undefined") ? pfx : "*!" + this.name + "@*."; + var idx = this.host.indexOf("."); + if (idx == -1) + return pfx + this.host; + + return (pfx + this.host.substr(idx + 1, this.host.length)); +} + +CIRCUser.prototype.getBanMask = +function usr_banmask() +{ + if (!this.host) + return this.unicodeName + "!*@*"; + + return "*!*@" + this.host; +} + +CIRCUser.prototype.say = +function usr_say (msg) +{ + this.parent.sayTo(this.encodedName, fromUnicode(msg, this)); +} + +CIRCUser.prototype.notice = +function usr_notice (msg) +{ + this.parent.noticeTo(this.encodedName, fromUnicode(msg, this)); +} + +CIRCUser.prototype.act = +function usr_act (msg) +{ + this.parent.actTo(this.encodedName, fromUnicode(msg, this)); +} + +CIRCUser.prototype.ctcp = +function usr_ctcp (code, msg, type) +{ + msg = msg || ""; + type = type || "PRIVMSG"; + + this.parent.ctcpTo(this.encodedName, fromUnicode(code, this), + fromUnicode(msg, this), type); +} + +CIRCUser.prototype.whois = +function usr_whois () +{ + this.parent.whois(this.unicodeName); +} + +/* + * channel user + */ +function CIRCChanUser(parent, unicodeName, encodedName, modes, userInChannel, name, host) +{ + // Both unicodeName and encodedName are optional, but at least one must be + // present. + + if (!encodedName && !unicodeName) + throw "Hey! Come on, I need either an encoded or a Unicode name."; + else if (encodedName && !unicodeName) + unicodeName = toUnicode(encodedName, parent); + else if (!encodedName && unicodeName) + encodedName = fromUnicode(unicodeName, parent); + + // We should have both unicode and encoded names by now. + let collectionKey = ":" + parent.parent.toLowerCase(encodedName); + + if (collectionKey in parent.users) + { + let existingUser = parent.users[collectionKey]; + if (modes) + { + // If we start with a single character mode, assume we're replacing + // the list. (i.e. the list is either all +/- modes, or all normal) + if ((modes.length >= 1) && (modes[0].search(/^[-+]/) == -1)) + { + // Modes, but no +/- prefixes, so *replace* mode list. + existingUser.modes = modes; + } + else + { + // We have a +/- mode list, so carefully update the mode list. + for (var m in modes) + { + // This will remove '-' modes, and all other modes will be + // added. + var mode = modes[m][1]; + if (modes[m][0] == "-") + { + if (arrayContains(existingUser.modes, mode)) + { + var i = arrayIndexOf(existingUser.modes, mode); + arrayRemoveAt(existingUser.modes, i); + } + } + else + { + if (!arrayContains(existingUser.modes, mode)) + existingUser.modes.push(mode); + } + } + } + } + existingUser.isFounder = (arrayContains(existingUser.modes, "q")) ? + true : false; + existingUser.isAdmin = (arrayContains(existingUser.modes, "a")) ? + true : false; + existingUser.isOp = (arrayContains(existingUser.modes, "o")) ? + true : false; + existingUser.isHalfOp = (arrayContains(existingUser.modes, "h")) ? + true : false; + existingUser.isVoice = (arrayContains(existingUser.modes, "v")) ? + true : false; + existingUser.updateSortName(); + return existingUser; + } + + var protoUser = new CIRCUser(parent.parent, unicodeName, encodedName, name, host); + + this.__proto__ = protoUser; + this.getURL = cusr_geturl; + this.setOp = cusr_setop; + this.setHalfOp = cusr_sethalfop; + this.setVoice = cusr_setvoice; + this.setBan = cusr_setban; + this.kick = cusr_kick; + this.kickBan = cusr_kban; + this.say = cusr_say; + this.notice = cusr_notice; + this.act = cusr_act; + this.whois = cusr_whois; + this.updateSortName = cusr_updatesortname; + this.parent = parent; + this.TYPE = "IRCChanUser"; + + this.modes = new Array(); + if (typeof modes != "undefined") + this.modes = modes; + this.isFounder = (arrayContains(this.modes, "q")) ? true : false; + this.isAdmin = (arrayContains(this.modes, "a")) ? true : false; + this.isOp = (arrayContains(this.modes, "o")) ? true : false; + this.isHalfOp = (arrayContains(this.modes, "h")) ? true : false; + this.isVoice = (arrayContains(this.modes, "v")) ? true : false; + this.updateSortName(); + + if (userInChannel) + parent.users[this.collectionKey] = this; + + return this; +} + +function cusr_updatesortname() +{ + // Check for the highest mode the user has (for sorting the userlist) + const userModes = this.parent.parent.userModes; + var modeLevel = 0; + var mode; + for (var i = 0; i < this.modes.length; i++) + { + for (var j = 0; j < userModes.length; j++) + { + if (userModes[j].mode == this.modes[i]) + { + if (userModes.length - j > modeLevel) + { + modeLevel = userModes.length - j; + mode = userModes[j]; + } + break; + } + } + } + // Counts numerically down from 9. + this.sortName = (9 - modeLevel) + "-" + this.unicodeName; +} + +function cusr_geturl() +{ + // Don't ask. + return this.parent.parent.parent.getURL(this.encodedName, ["isnick"]); +} + +function cusr_setop(f) +{ + var server = this.parent.parent; + var me = server.me; + + var modifier = (f) ? " +o " : " -o "; + server.sendData("MODE " + this.parent.encodedName + modifier + this.encodedName + "\n"); + + return true; +} + +function cusr_sethalfop (f) +{ + var server = this.parent.parent; + var me = server.me; + + var modifier = (f) ? " +h " : " -h "; + server.sendData("MODE " + this.parent.encodedName + modifier + this.encodedName + "\n"); + + return true; +} + +function cusr_setvoice (f) +{ + var server = this.parent.parent; + var me = server.me; + + var modifier = (f) ? " +v " : " -v "; + server.sendData("MODE " + this.parent.encodedName + modifier + this.encodedName + "\n"); + + return true; +} + +function cusr_kick (reason) +{ + var server = this.parent.parent; + var me = server.me; + + reason = typeof reason == "string" ? reason : ""; + + server.sendData("KICK " + this.parent.encodedName + " " + this.encodedName + " :" + + fromUnicode(reason, this) + "\n"); + + return true; +} + +function cusr_setban (f) +{ + var server = this.parent.parent; + var me = server.me; + + if (!this.host) + return false; + + var modifier = (f) ? " +b " : " -b "; + modifier += fromUnicode(this.getBanMask(), server) + " "; + + server.sendData("MODE " + this.parent.encodedName + modifier + "\n"); + + return true; +} + +function cusr_kban (reason) +{ + var server = this.parent.parent; + var me = server.me; + + if (!this.host) + return false; + + reason = (typeof reason != "undefined") ? reason : this.encodedName; + var modifier = " -o+b " + this.encodedName + " " + + fromUnicode(this.getBanMask(), server) + " "; + + server.sendData("MODE " + this.parent.encodedName + modifier + "\n" + + "KICK " + this.parent.encodedName + " " + + this.encodedName + " :" + reason + "\n"); + + return true; +} + +function cusr_say (msg) +{ + this.__proto__.say (msg); +} + +function cusr_notice (msg) +{ + this.__proto__.notice (msg); +} + +function cusr_act (msg) +{ + this.__proto__.act (msg); +} + +function cusr_whois () +{ + this.__proto__.whois (); +} + + +// IRC URL parsing and generating + +function parseIRCURL(url) +{ + var specifiedHost = ""; + + var rv = new Object(); + rv.spec = url; + rv.scheme = url.split(":")[0]; + rv.host = null; + rv.target = ""; + rv.port = (rv.scheme == "ircs" ? 6697 : 6667); + rv.msg = ""; + rv.pass = null; + rv.key = null; + rv.charset = null; + rv.needpass = false; + rv.needkey = false; + rv.isnick = false; + rv.isserver = false; + + if (url.search(/^(ircs?:\/?\/?)$/i) != -1) + return rv; + + /* split url into <host>/<everything-else> pieces */ + var ary = url.match(/^ircs?:\/\/([^\/\s]+)?(\/[^\s]*)?$/i); + if (!ary || !ary[1]) + { + dd("parseIRCURL: initial split failed"); + return null; + } + var host = ary[1]; + var rest = arrayHasElementAt(ary, 2) ? ary[2] : ""; + + /* split <host> into server (or network) / port */ + ary = host.match(/^([^\:]+|\[[^\]]+\])(\:\d+)?$/i); + if (!ary) + { + dd("parseIRCURL: host/port split failed"); + return null; + } + + // 1 = hostname or IPv4 address, 2 = port. + specifiedHost = rv.host = ary[1].toLowerCase(); + rv.isserver = arrayHasElementAt(ary, 2) || /\.|:/.test(specifiedHost); + if (arrayHasElementAt(ary, 2)) + rv.port = parseInt(ary[2].substr(1)); + + if (rest) + { + ary = rest.match(/^\/([^\?\s\/,]*)?\/?(,[^\?]*)?(\?.*)?$/); + if (!ary) + { + dd("parseIRCURL: rest split failed ``" + rest + "''"); + return null; + } + + rv.target = arrayHasElementAt(ary, 1) ? ecmaUnescape(ary[1]) : ""; + + if (rv.target.search(/[\x07,\s]/) != -1) + { + dd("parseIRCURL: invalid characters in channel name"); + return null; + } + + var params = arrayHasElementAt(ary, 2) ? ary[2].toLowerCase() : ""; + var query = arrayHasElementAt(ary, 3) ? ary[3] : ""; + + if (params) + { + params = params.split(","); + while (params.length) + { + var param = params.pop(); + // split doesn't take out empty bits: + if (param == "") + continue; + switch (param) + { + case "isnick": + rv.isnick = true; + if (!rv.target) + { + dd("parseIRCURL: isnick w/o target"); + /* isnick w/o a target is bogus */ + return null; + } + break; + + case "isserver": + rv.isserver = true; + if (!specifiedHost) + { + dd("parseIRCURL: isserver w/o host"); + /* isserver w/o a host is bogus */ + return null; + } + break; + + case "needpass": + case "needkey": + rv[param] = true; + break; + + default: + /* If we didn't understand it, ignore but warn: */ + dd("parseIRCURL: Unrecognized param '" + param + + "' in URL!"); + } + } + } + + if (query) + { + ary = query.substr(1).split("&"); + while (ary.length) + { + var arg = ary.pop().split("="); + /* + * we don't want to accept *any* query, or folks could + * say things like "target=foo", and overwrite what we've + * already parsed, so we only use query args we know about. + */ + switch (arg[0].toLowerCase()) + { + case "msg": + rv.msg = ecmaUnescape(arg[1]).replace("\n", "\\n"); + break; + + case "pass": + rv.needpass = true; + rv.pass = ecmaUnescape(arg[1]).replace("\n", "\\n"); + break; + + case "key": + rv.needkey = true; + rv.key = ecmaUnescape(arg[1]).replace("\n", "\\n"); + break; + + case "charset": + rv.charset = ecmaUnescape(arg[1]).replace("\n", "\\n"); + break; + } + } + } + } + + return rv; +} + +function constructIRCURL(obj) +{ + function parseQuery(obj) + { + var rv = new Array(); + if ("msg" in obj) + rv.push("msg=" + ecmaEscape(obj.msg.replace("\\n", "\n"))); + if ("pass" in obj) + rv.push("pass=" + ecmaEscape(obj.pass.replace("\\n", "\n"))); + if ("key" in obj) + rv.push("key=" + ecmaEscape(obj.key.replace("\\n", "\n"))); + if ("charset" in obj) + rv.push("charset=" + ecmaEscape(obj.charset.replace("\\n", "\n"))); + + return rv.length ? "?" + rv.join("&") : ""; + }; + function parseFlags(obj) + { + var rv = new Array(); + var haveTarget = ("target" in obj) && obj.target; + if (("needpass" in obj) && obj.needpass) + rv.push(",needpass"); + if (("needkey" in obj) && obj.needkey && haveTarget) + rv.push(",needkey"); + if (("isnick" in obj) && obj.isnick && haveTarget) + rv.push(",isnick"); + + return rv.join(""); + }; + + var flags = ""; + var scheme = ("scheme" in obj) ? obj.scheme : "irc"; + if (!("host" in obj) || !obj.host) + return scheme + "://"; + + var url = scheme + "://" + obj.host; + + // Add port if non-standard: + if (("port" in obj) && (((scheme == "ircs") && (obj.port != 6697)) || + ((scheme == "irc") && (obj.port != 6667)))) + { + url += ":" + obj.port; + } + // Need to add ",isserver" if there's no port and no dots in the hostname: + else if (("isserver" in obj) && obj.isserver && + (obj.host.indexOf(".") == -1)) + { + flags += ",isserver"; + } + url += "/"; + + if (("target" in obj) && obj.target) + { + if (obj.target.search(/[\x07,\s]/) != -1) + { + dd("parseIRCObject: invalid characters in channel/nick name"); + return null; + } + url += ecmaEscape(obj.target).replace(/\//g, "%2f"); + } + + return url + flags + parseFlags(obj) + parseQuery(obj); +} + +/* Canonicalizing an IRC URL removes all items which aren't necessary to + * identify the target. For example, an IRC URL with ?pass=password and one + * without (but otherwise identical) are refering to the same target, so + * ?pass= is removed. + */ +function makeCanonicalIRCURL(url) +{ + var canonicalProps = { scheme: true, host: true, port: true, + target: true, isserver: true, isnick: true }; + + var urlObject = parseIRCURL(url); + if (!urlObject) + return ""; // Input wasn't a valid IRC URL. + for (var prop in urlObject) + { + if (!(prop in canonicalProps)) + delete urlObject[prop]; + } + return constructIRCURL(urlObject); +} diff --git a/comm/suite/chatzilla/js/lib/json-serializer.js b/comm/suite/chatzilla/js/lib/json-serializer.js new file mode 100644 index 0000000000..93333f2bb1 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/json-serializer.js @@ -0,0 +1,103 @@ +/* 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/. */ + +/* This is a simple set of functions for serializing and parsing JS objects + * to and from files. + */ + +function JSONSerializer(file) { + if (typeof file == "string") + this._file = new nsLocalFile(file); + else + this._file = file; + this._open = false; +} + +JSONSerializer.prototype = { + /* + * Opens the serializer on the file specified when created, in either the read + * ("<") or write (">") directions. When the file is open, only the + * appropriate direction of serialization/deserialization may be performed. + * + * Note: serialize and deserialize automatically open the file if it is not + * open. + * + * @param dir The string representing the direction of serialization. + * @returns Value indicating whether the file was opened successfully. + */ + open: function(dir) { + if (!ASSERT((dir == ">") || (dir == "<"), "Bad serialization direction!")) { + return false; + } + if (this._open) { + return false; + } + + this._fileStream = new LocalFile(this._file, dir); + if ((typeof this._fileStream == "object") && this._fileStream) { + this._open = true; + } + + return this._open; + }, + + /* + * Closes the file stream and ends reading or writing. + * + * @returns Value indicating whether the file was closed successfully. + */ + close: function() { + if (this._open) { + this._fileStream.close(); + delete this._fileStream; + this._open = false; + } + return true; + }, + + /* + * Serializes a single object into the file stream. All properties of the + * object are stored in the stream, including properties that contain other + * objects. + * + * @param obj JS object to serialize to the file. + */ + serialize: function(obj) { + if (!this._open) { + this.open(">"); + } + if (!ASSERT(this._open, "Unable to open the file for writing!")) { + return; + } + + this._fileStream.write(JSON.stringify(obj, null, 2)); + }, + + /* + * Reads in enough of the file to deserialize (realize) a single object. The + * object deserialized is returned; all sub-properties of the object are + * deserialized with it. + * + * @returns JS object parsed from the file. + */ + deserialize: function() { + if (!this._open) { + this.open("<"); + } + if (!ASSERT(this._open, "Unable to open the file for reading!")) + return false; + + let rv = null; + try { + rv = JSON.parse(this._fileStream.read()); + } + catch(ex) { + dd("Syntax error while deserializing file!"); + dd(ex.message); + dd(ex.stack); + } + + return rv; + }, +}; diff --git a/comm/suite/chatzilla/js/lib/menu-manager.js b/comm/suite/chatzilla/js/lib/menu-manager.js new file mode 100644 index 0000000000..6fd0686833 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/menu-manager.js @@ -0,0 +1,848 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +// +function MenuManager(commandManager, menuSpecs, contextFunction, commandStr) +{ + var menuManager = this; + + this.commandManager = commandManager; + this.menuSpecs = menuSpecs; + this.contextFunction = contextFunction; + this.commandStr = commandStr; + this.repeatId = 0; + this.cxStore = new Object(); + + this.onPopupShowing = + function mmgr_onshow(event) { return menuManager.showPopup(event); }; + this.onPopupHiding = + function mmgr_onhide(event) { return menuManager.hidePopup(event); }; + this.onMenuCommand = + function mmgr_oncmd(event) { return menuManager.menuCommand(event); }; + + /* The code using us may override these with functions which will be called + * after all our internal processing is done. Both are called with the + * arguments 'event' (DOM), 'cx' (JS), 'popup' (DOM). + */ + this.onCallbackPopupShowing = null; + this.onCallbackPopupHiding = null; +} + +MenuManager.prototype.appendMenuItems = +function mmgr_append(menuId, items) +{ + for (var i = 0; i < items.length; ++i) + this.menuSpecs[menuId].items.push(items[i]); +} + +MenuManager.prototype.createContextMenus = +function mmgr_initcxs (document) +{ + for (var id in this.menuSpecs) + { + if (id.indexOf("context:") == 0) + this.createContextMenu(document, id); + } +} + +MenuManager.prototype.createContextMenu = +function mmgr_initcx (document, id) +{ + if (!document.getElementById(id)) + { + if (!ASSERT(id in this.menuSpecs, "unknown context menu " + id)) + return; + + var dp = document.getElementById("dynamic-popups"); + var popup = this.appendPopupMenu (dp, null, id, id); + var items = this.menuSpecs[id].items; + this.createMenuItems (popup, null, items); + + if (!("uiElements" in this.menuSpecs[id])) + this.menuSpecs[id].uiElements = [popup]; + else if (!arrayContains(this.menuSpecs[id].uiElements, popup)) + this.menuSpecs[id].uiElements.push(popup); + } +} + + +MenuManager.prototype.createMenus = +function mmgr_createtb(document, menuid) +{ + var menu = document.getElementById(menuid); + for (var id in this.menuSpecs) + { + var domID; + if ("domID" in this.menuSpecs[id]) + domID = this.menuSpecs[id].domID; + else + domID = id; + + if (id.indexOf(menuid + ":") == 0) + this.createMenu(menu, null, id, domID); + } +} + +MenuManager.prototype.createMainToolbar = +function mmgr_createtb(document, id) +{ + var toolbar = document.getElementById(id); + var spec = this.menuSpecs[id]; + for (var i in spec.items) + { + this.appendToolbarItem (toolbar, null, spec.items[i]); + } + + toolbar.className = "toolbar-primary chromeclass-toolbar"; +} + +MenuManager.prototype.updateMenus = +function mmgr_updatemenus(document, menus) +{ + // Cope with one string (update just the one menu)... + if (isinstance(menus, String)) + { + menus = [menus]; + } + // Or nothing/nonsense (update everything). + else if ((typeof menus != "object") || !isinstance(menus, Array)) + { + menus = []; + for (var k in this.menuSpecs) + { + if ((/^(mainmenu|context)/).test(k)) + menus.push(k); + } + } + + var menuBar = document.getElementById("mainmenu"); + + // Loop through this array and update everything we need to. + for (var i = 0; i < menus.length; i++) + { + var id = menus[i]; + if (!(id in this.menuSpecs)) + continue; + var menu = this.menuSpecs[id]; + var domID; + if ("domID" in this.menuSpecs[id]) + domID = this.menuSpecs[id].domID; + else + domID = id; + + // Context menus need to be deleted in order to be regenerated... + if ((/^context/).test(id)) + { + var cxMenuNode; + if ((cxMenuNode = document.getElementById(id))) + cxMenuNode.parentNode.removeChild(cxMenuNode); + this.createContextMenu(document, id); + } + else if ((/^mainmenu/).test(id) && + !("uiElements" in this.menuSpecs[id])) + { + this.createMenu(menuBar, null, id, domID); + continue; + } + else if ((/^(mainmenu|popup)/).test(id) && + ("uiElements" in this.menuSpecs[id])) + { + for (var j = 0; j < menu.uiElements.length; j++) + { + var node = menu.uiElements[j]; + domID = node.parentNode.id; + // Clear the menu node. + while (node.lastChild) + node.removeChild(node.lastChild); + + this.createMenu(node.parentNode.parentNode, + node.parentNode.nextSibling, + id, domID); + } + } + + + } +} + + +/** + * Internal use only. + * + * Registers event handlers on a given menu. + */ +MenuManager.prototype.hookPopup = +function mmgr_hookpop (node) +{ + node.addEventListener ("popupshowing", this.onPopupShowing, false); + node.addEventListener ("popuphiding", this.onPopupHiding, false); +} + +/** + * Internal use only. + * + * |showPopup| is called from the "onpopupshowing" event of menus managed + * by the CommandManager. If a command is disabled, represents a command + * that cannot be "satisfied" by the current command context |cx|, or has an + * "enabledif" attribute that eval()s to false, then the menuitem is disabled. + * In addition "checkedif" and "visibleif" attributes are eval()d and + * acted upon accordingly. + */ +MenuManager.prototype.showPopup = +function mmgr_showpop (event) +{ + /* returns true if the command context has the properties required to + * execute the command associated with |menuitem|. + */ + function satisfied() + { + if (menuitem.hasAttribute("isSeparator") || + !menuitem.hasAttribute("commandname")) + { + return true; + } + + if (menuitem.hasAttribute("repeatfor")) + return false; + + if (!("menuManager" in cx)) + { + dd ("no menuManager in cx"); + return false; + } + + var name = menuitem.getAttribute("commandname"); + var commandManager = cx.menuManager.commandManager; + var commands = commandManager.commands; + + if (!ASSERT (name in commands, + "menu contains unknown command '" + name + "'")) + { + return false; + } + + var rv = commandManager.isCommandSatisfied(cx, commands[name]); + delete cx.parseError; + return rv; + }; + + /* Convenience function for "enabledif", etc, attributes. */ + function has (prop) + { + return (prop in cx); + }; + + /* evals the attribute named |attr| on the node |node|. */ + function evalIfAttribute (node, attr) + { + var ex; + var expr = node.getAttribute(attr); + if (!expr) + return true; + + expr = expr.replace (/\Wand\W/gi, " && "); + expr = expr.replace (/\Wor\W/gi, " || "); + + try + { + return eval("(" + expr + ")"); + } + catch (ex) + { + dd ("caught exception evaling '" + node.getAttribute("id") + "'.'" + + attr + "': '" + expr + "'\n" + ex); + } + return true; + }; + + /* evals the attribute named |attr| on the node |node|. */ + function evalAttribute(node, attr) + { + var ex; + var expr = node.getAttribute(attr); + if (!expr) + return null; + + try + { + return eval(expr); + } + catch (ex) + { + dd ("caught exception evaling '" + node.getAttribute("id") + "'.'" + + attr + "': '" + expr + "'\n" + ex); + } + return null; + }; + + var cx; + var popup = event.originalTarget; + var menuName = popup.getAttribute("menuName"); + + /* If the host provided a |contextFunction|, use it now. Remember the + * return result as this.cx for use if something from this menu is actually + * dispatched. */ + if (typeof this.contextFunction == "function") + { + cx = this.cx = this.contextFunction(menuName, event); + } + else + { + cx = this.cx = { menuManager: this, originalEvent: event }; + } + + // Keep the context around by menu name. Removed in hidePopup. + this.cxStore[menuName] = cx; + + var menuitem = popup.firstChild; + do + { + if (!menuitem.hasAttribute("repeatfor")) + continue; + + // Remove auto-generated items (located prior to real item). + while (menuitem.previousSibling && + menuitem.previousSibling.hasAttribute("repeatgenerated")) + { + menuitem.parentNode.removeChild(menuitem.previousSibling); + } + + if (!("repeatList" in cx)) + cx.repeatList = new Object(); + + /* Get the array of new items to add by evaluating "repeatfor" with + * "cx" in scope. Usually will return an already-calculated Array + * either from "cx" or somewhere in the object model. + */ + var ary = evalAttribute(menuitem, "repeatfor"); + + if ((typeof ary != "object") || !isinstance(ary, Array)) + ary = []; + + /* The item itself should only be shown if there's no items in the + * array - this base item is always disabled. + */ + if (ary.length > 0) + menuitem.setAttribute("hidden", "true"); + else + menuitem.removeAttribute("hidden"); + + // Save the array in the context object. + cx.repeatList[menuitem.getAttribute("repeatid")] = ary; + + /* Get the maximum number of items we're allowed to show from |ary| by + * evaluating "repeatlimit" with "cx" in scope. This could be a fixed + * limit or dynamically calculated (e.g. from prefs). + */ + var limit = evalAttribute(menuitem, "repeatlimit"); + // Make sure we've got a number at all... + if (typeof limit != "number") + limit = ary.length; + // ...and make sure it's no higher than |ary.length|. + limit = Math.min(ary.length, limit); + + var cmd = menuitem.getAttribute("commandname"); + var props = { repeatgenerated: true, repeatindex: -1, + repeatid: menuitem.getAttribute("repeatid"), + repeatmap: menuitem.getAttribute("repeatmap") }; + + /* Clone non-repeat attributes. All attributes except those starting + * with 'repeat', and those matching 'hidden' or 'disabled' are saved + * to |props|, which is then supplied to |appendMenuItem| later. + */ + for (var i = 0; i < menuitem.attributes.length; i++) + { + var name = menuitem.attributes[i].nodeName; + if (!name.match(/^(repeat|(hidden|disabled)$)/)) + props[name] = menuitem.getAttribute(name); + } + + var lastGroup = ""; + for (i = 0; i < limit; i++) + { + /* Check for groupings. For each item we add, if "repeatgroup" gives + * a different value, we insert a separator. + */ + if (menuitem.getAttribute("repeatgroup")) + { + cx.index = i; + ary = cx.repeatList[menuitem.getAttribute("repeatid")]; + var item = ary[i]; + /* Apply any updates to "cx" for this item by evaluating + * "repeatmap" with "cx" and "item" in scope. This may just + * copy some attributes from "item" to "cx" or it may do more. + */ + evalAttribute(menuitem, "repeatmap"); + /* Get the item's group by evaluating "repeatgroup" with "cx" + * and "item" in scope. Usually will return an appropriate + * property from "item". + */ + var group = evalAttribute(menuitem, "repeatgroup"); + + if ((i > 0) && (lastGroup != group)) + this.appendMenuSeparator(popup, menuitem, props); + + lastGroup = group; + } + + props.repeatindex = i; + this.appendMenuItem(popup, menuitem, cmd, props); + } + } while ((menuitem = menuitem.nextSibling)); + + menuitem = popup.firstChild; + do + { + if (menuitem.hasAttribute("repeatgenerated") && + menuitem.hasAttribute("repeatmap")) + { + cx.index = menuitem.getAttribute("repeatindex"); + ary = cx.repeatList[menuitem.getAttribute("repeatid")]; + var item = ary[cx.index]; + /* Apply any updates to "cx" for this item by evaluating + * "repeatmap" with "cx" and "item" in scope. This may just + * copy some attributes from "item" to "cx" or it may do more. + */ + evalAttribute(menuitem, "repeatmap"); + } + + /* should it be visible? */ + if (menuitem.hasAttribute("visibleif")) + { + if (evalIfAttribute(menuitem, "visibleif")) + menuitem.removeAttribute ("hidden"); + else + { + menuitem.setAttribute ("hidden", "true"); + continue; + } + } + + /* it's visible, maybe it has a dynamic label? */ + if (menuitem.hasAttribute("format")) + { + var label = replaceVars(menuitem.getAttribute("format"), cx); + if (label.indexOf("\$") != -1) + label = menuitem.getAttribute("backupLabel"); + menuitem.setAttribute("label", label); + } + + /* ok, it's visible, maybe it should be disabled? */ + if (satisfied()) + { + if (menuitem.hasAttribute("enabledif")) + { + if (evalIfAttribute(menuitem, "enabledif")) + menuitem.removeAttribute ("disabled"); + else + menuitem.setAttribute ("disabled", "true"); + } + else + menuitem.removeAttribute ("disabled"); + } + else + { + menuitem.setAttribute ("disabled", "true"); + } + + /* should it have a check? */ + if (menuitem.hasAttribute("checkedif")) + { + if (evalIfAttribute(menuitem, "checkedif")) + menuitem.setAttribute ("checked", "true"); + else + menuitem.removeAttribute ("checked"); + } + } while ((menuitem = menuitem.nextSibling)); + + if (typeof this.onCallbackPopupShowing == "function") + this.onCallbackPopupShowing(event, cx, popup); + + return true; +} + +/** + * Internal use only. + * + * |hidePopup| is called from the "onpopuphiding" event of menus + * managed by the CommandManager. Clean up this.cxStore, but + * not this.cx because that messes up nested menus. + */ +MenuManager.prototype.hidePopup = +function mmgr_hidepop(event) +{ + var popup = event.originalTarget; + var menuName = popup.getAttribute("menuName"); + + if (typeof this.onCallbackPopupHiding == "function") + this.onCallbackPopupHiding(event, this.cxStore[menuName], popup); + + delete this.cxStore[menuName]; + + return true; +} + +MenuManager.prototype.menuCommand = +function mmgr_menucmd(event) +{ + /* evals the attribute named |attr| on the node |node|. */ + function evalAttribute(node, attr) + { + var ex; + var expr = node.getAttribute(attr); + if (!expr) + return null; + + try + { + return eval(expr); + } + catch (ex) + { + dd ("caught exception evaling '" + node.getAttribute("id") + "'.'" + + attr + "': '" + expr + "'\n" + ex); + } + return null; + }; + + var menuitem = event.originalTarget; + var cx = this.cx; + /* We need to re-run the repeat-map if the user has selected a special + * repeat-generated menu item, so that the context object is correct. + */ + if (menuitem.hasAttribute("repeatgenerated") && + menuitem.hasAttribute("repeatmap")) + { + cx.index = menuitem.getAttribute("repeatindex"); + var ary = cx.repeatList[menuitem.getAttribute("repeatid")]; + var item = ary[cx.index]; + /* Apply any updates to "cx" for this item by evaluating + * "repeatmap" with "cx" and "item" in scope. This may just + * copy some attributes from "item" to "cx" or it may do more. + */ + evalAttribute(menuitem, "repeatmap"); + } + + eval(this.commandStr); +}; + + +/** + * Appends a sub-menu to an existing menu. + * @param parentNode DOM Node to insert into + * @param beforeNode DOM Node already contained by parentNode, to insert before + * @param domId ID of the sub-menu to add. + * @param label Text to use for this sub-menu. + * @param accesskey Accesskey to use for the sub-menu. + * @param attribs Object containing CSS attributes to set on the element. + */ +MenuManager.prototype.appendSubMenu = +function mmgr_addsmenu(parentNode, beforeNode, menuName, domId, label, + accesskey, attribs) +{ + var document = parentNode.ownerDocument; + + /* sometimes the menu is already there, for overlay purposes. */ + var menu = document.getElementById(domId); + + if (!menu) + { + menu = document.createElement ("menu"); + menu.setAttribute ("id", domId); + } + + var menupopup = menu.firstChild; + + if (!menupopup) + { + menupopup = document.createElement ("menupopup"); + menupopup.setAttribute ("id", domId + "-popup"); + menu.appendChild(menupopup); + menupopup = menu.firstChild; + } + + menupopup.setAttribute ("menuName", menuName); + + menu.setAttribute("accesskey", accesskey); + label = label.replace("&", ""); + menu.setAttribute ("label", label); + menu.setAttribute ("isSeparator", true); + + // Only attach the menu if it's not there already. This can't be in the + // if (!menu) block because the updateMenus code clears toplevel menus, + // orphaning the submenus, to (parts of?) which we keep handles in the + // uiElements array. See the updateMenus code. + if (!menu.parentNode) + parentNode.insertBefore(menu, beforeNode); + + if (typeof attribs == "object") + { + for (var p in attribs) + menu.setAttribute (p, attribs[p]); + } + + this.hookPopup (menupopup); + + return menupopup; +} + +/** + * Appends a popup to an existing popupset. + * @param parentNode DOM Node to insert into + * @param beforeNode DOM Node already contained by parentNode, to insert before + * @param id ID of the popup to add. + * @param label Text to use for this popup. Popup menus don't normally have + * labels, but we set a "label" attribute anyway, in case + * the host wants it for some reason. Any "&" characters will + * be stripped. + * @param attribs Object containing CSS attributes to set on the element. + */ +MenuManager.prototype.appendPopupMenu = +function mmgr_addpmenu (parentNode, beforeNode, menuName, id, label, attribs) +{ + var document = parentNode.ownerDocument; + var popup = document.createElement ("menupopup"); + popup.setAttribute ("id", id); + if (label) + popup.setAttribute ("label", label.replace("&", "")); + if (typeof attribs == "object") + { + for (var p in attribs) + popup.setAttribute (p, attribs[p]); + } + + popup.setAttribute ("menuName", menuName); + + parentNode.insertBefore(popup, beforeNode); + this.hookPopup (popup); + + return popup; +} + +/** + * Appends a menuitem to an existing menu or popup. + * @param parentNode DOM Node to insert into + * @param beforeNode DOM Node already contained by parentNode, to insert before + * @param command A reference to the CommandRecord this menu item will represent. + * @param attribs Object containing CSS attributes to set on the element. + */ +MenuManager.prototype.appendMenuItem = +function mmgr_addmenu (parentNode, beforeNode, commandName, attribs) +{ + var menuManager = this; + + var document = parentNode.ownerDocument; + if (commandName == "-") + return this.appendMenuSeparator(parentNode, beforeNode, attribs); + + var parentId = parentNode.getAttribute("id"); + + if (!ASSERT(commandName in this.commandManager.commands, + "unknown command " + commandName + " targeted for " + + parentId)) + { + return null; + } + + var command = this.commandManager.commands[commandName]; + var menuitem = document.createElement ("menuitem"); + menuitem.setAttribute ("id", parentId + ":" + commandName); + menuitem.setAttribute ("commandname", command.name); + // Add keys if this isn't a context menu: + if (parentId.indexOf("context") != 0) + menuitem.setAttribute("key", "key:" + command.name); + menuitem.setAttribute("accesskey", command.accesskey); + var label = command.label.replace("&", ""); + menuitem.setAttribute ("label", label); + if (command.format) + { + menuitem.setAttribute("format", command.format); + menuitem.setAttribute("backupLabel", label); + } + + if ((typeof attribs == "object") && attribs) + { + for (var p in attribs) + menuitem.setAttribute (p, attribs[p]); + if ("repeatfor" in attribs) + menuitem.setAttribute("repeatid", this.repeatId++); + } + + command.uiElements.push(menuitem); + parentNode.insertBefore (menuitem, beforeNode); + /* It seems, bob only knows why, that this must be done AFTER the node is + * added to the document. + */ + menuitem.addEventListener("command", this.onMenuCommand, false); + + return menuitem; +} + +/** + * Appends a menuseparator to an existing menu or popup. + * @param parentNode DOM Node to insert into + * @param beforeNode DOM Node already contained by parentNode, to insert before + * @param attribs Object containing CSS attributes to set on the element. + */ +MenuManager.prototype.appendMenuSeparator = +function mmgr_addsep (parentNode, beforeNode, attribs) +{ + var document = parentNode.ownerDocument; + var menuitem = document.createElement ("menuseparator"); + menuitem.setAttribute ("isSeparator", true); + if (typeof attribs == "object") + { + for (var p in attribs) + menuitem.setAttribute (p, attribs[p]); + } + parentNode.insertBefore (menuitem, beforeNode); + + return menuitem; +} + +/** + * Appends a toolbaritem to an existing box element. + * @param parentNode DOM Node to insert into + * @param beforeNode DOM Node already contained by parentNode, to insert before + * @param command A reference to the CommandRecord this toolbaritem will + * represent. + * @param attribs Object containing CSS attributes to set on the element. + */ +MenuManager.prototype.appendToolbarItem = +function mmgr_addtb (parentNode, beforeNode, commandName, attribs) +{ + if (commandName == "-") + return this.appendToolbarSeparator(parentNode, beforeNode, attribs); + + var parentId = parentNode.getAttribute("id"); + + if (!ASSERT(commandName in this.commandManager.commands, + "unknown command " + commandName + " targeted for " + + parentId)) + { + return null; + } + + var command = this.commandManager.commands[commandName]; + var document = parentNode.ownerDocument; + var tbitem = document.createElement ("toolbarbutton"); + + var id = parentNode.getAttribute("id") + ":" + commandName; + tbitem.setAttribute ("id", id); + tbitem.setAttribute ("class", "toolbarbutton-1"); + if (command.tip) + tbitem.setAttribute ("tooltiptext", command.tip); + tbitem.setAttribute ("label", command.label.replace("&", "")); + tbitem.setAttribute ("oncommand", + "dispatch('" + commandName + "');"); + if (typeof attribs == "object") + { + for (var p in attribs) + tbitem.setAttribute (p, attribs[p]); + } + + command.uiElements.push(tbitem); + parentNode.insertBefore (tbitem, beforeNode); + + return tbitem; +} + +/** + * Appends a toolbarseparator to an existing box. + * @param parentNode DOM Node to insert into + * @param beforeNode DOM Node already contained by parentNode, to insert before + * @param attribs Object containing CSS attributes to set on the element. + */ +MenuManager.prototype.appendToolbarSeparator = +function mmgr_addmenu (parentNode, beforeNode, attribs) +{ + var document = parentNode.ownerDocument; + var tbitem = document.createElement ("toolbarseparator"); + tbitem.setAttribute ("isSeparator", true); + if (typeof attribs == "object") + { + for (var p in attribs) + tbitem.setAttribute (p, attribs[p]); + } + parentNode.appendChild (tbitem); + + return tbitem; +} + +/** + * Creates menu DOM nodes from a menu specification. + * @param parentNode DOM Node to insert into + * @param beforeNode DOM Node already contained by parentNode, to insert before + * @param menuSpec array of menu items + */ +MenuManager.prototype.createMenu = +function mmgr_newmenu (parentNode, beforeNode, menuName, domId, attribs) +{ + if (typeof domId == "undefined") + domId = menuName; + + if (!ASSERT(menuName in this.menuSpecs, "unknown menu name " + menuName)) + return null; + + var menuSpec = this.menuSpecs[menuName]; + if (!("accesskey" in menuSpec)) + menuSpec.accesskey = getAccessKey(menuSpec.label); + + var subMenu = this.appendSubMenu(parentNode, beforeNode, menuName, domId, + menuSpec.label, menuSpec.accesskey, + attribs); + + // Keep track where we're adding popup nodes derived from some menuSpec + if (!("uiElements" in this.menuSpecs[menuName])) + this.menuSpecs[menuName].uiElements = [subMenu]; + else if (!arrayContains(this.menuSpecs[menuName].uiElements, subMenu)) + this.menuSpecs[menuName].uiElements.push(subMenu); + + this.createMenuItems (subMenu, null, menuSpec.items); + return subMenu; +} + +MenuManager.prototype.createMenuItems = +function mmgr_newitems (parentNode, beforeNode, menuItems) +{ + function itemAttribs() + { + return (1 in menuItems[i]) ? menuItems[i][1] : null; + }; + + var parentId = parentNode.getAttribute("id"); + + for (var i in menuItems) + { + var itemName = menuItems[i][0]; + if (itemName[0] == ">") + { + itemName = itemName.substr(1); + if (!ASSERT(itemName in this.menuSpecs, + "unknown submenu " + itemName + " referenced in " + + parentId)) + { + continue; + } + this.createMenu (parentNode, beforeNode, itemName, + parentId + ":" + itemName, itemAttribs()); + } + else if (itemName in this.commandManager.commands) + { + this.appendMenuItem (parentNode, beforeNode, itemName, + itemAttribs()); + } + else if (itemName == "-") + { + this.appendMenuSeparator (parentNode, beforeNode, itemAttribs()); + } + else + { + dd ("unknown command " + itemName + " referenced in " + parentId); + } + } +} + diff --git a/comm/suite/chatzilla/js/lib/message-manager.js b/comm/suite/chatzilla/js/lib/message-manager.js new file mode 100644 index 0000000000..56020d48f6 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/message-manager.js @@ -0,0 +1,356 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +// +function MessageManager(entities) +{ + const UC_CTRID = "@mozilla.org/intl/scriptableunicodeconverter"; + const nsIUnicodeConverter = + Components.interfaces.nsIScriptableUnicodeConverter; + + this.ucConverter = + Components.classes[UC_CTRID].getService(nsIUnicodeConverter); + this.defaultBundle = null; + this.bundleList = new Array(); + // Provide a fallback so we don't break getMsg and related constants later. + this.entities = entities || {}; +} + +// ISO-2022-JP (often used for Japanese on IRC) doesn't contain any support +// for hankaku kana (half-width katakana), so we support the option to convert +// it to zenkaku kana (full-width katakana). This does not affect any other +// encoding at this time. +MessageManager.prototype.enableHankakuToZenkaku = false; + +MessageManager.prototype.loadBrands = +function mm_loadbrands() +{ + var entities = this.entities; + var app = getService("@mozilla.org/xre/app-info;1", "nsIXULAppInfo"); + if (app) + { + // Use App info if possible + entities.brandShortName = app.name; + entities.brandFullName = app.name + " " + app.version; + entities.brandVendorName = app.vendor; + return; + } + + var brandBundle; + var path = "chrome://branding/locale/brand.properties"; + try + { + brandBundle = this.addBundle(path); + } + catch (exception) + { + // May be an older mozilla version, try another location. + path = "chrome://global/locale/brand.properties"; + brandBundle = this.addBundle(path); + } + + entities.brandShortName = brandBundle.GetStringFromName("brandShortName"); + entities.brandVendorName = brandBundle.GetStringFromName("vendorShortName"); + // Not all versions of Suite / Fx have this defined; Cope: + try + { + entities.brandFullName = brandBundle.GetStringFromName("brandFullName"); + } + catch(exception) + { + entities.brandFullName = entities.brandShortName; + } + + // Remove all of this junk, or it will be the default bundle for getMsg... + this.bundleList.pop(); +} + +MessageManager.prototype.addBundle = +function mm_addbundle(bundlePath, targetWindow) +{ + var bundle = srGetStrBundle(bundlePath); + this.bundleList.push(bundle); + + // The bundle will load if the file doesn't exist. This will fail though. + // We want to be clean and remove the bundle again. + try + { + this.importBundle(bundle, targetWindow, this.bundleList.length - 1); + } + catch (exception) + { + // Clean up and return the exception. + this.bundleList.pop(); + throw exception; + } + return bundle; +} + +MessageManager.prototype.importBundle = +function mm_importbundle(bundle, targetWindow, index) +{ + var me = this; + function replaceEntities(matched, entity) + { + if (entity in me.entities) + return me.entities[entity]; + + return matched; + }; + const nsIPropertyElement = Components.interfaces.nsIPropertyElement; + + if (!targetWindow) + targetWindow = window; + + if (typeof index == "undefined") + index = arrayIndexOf(this.bundleList, bundle); + + var pfx; + if (index == 0) + pfx = ""; + else + pfx = index + ":"; + + var enumer = bundle.getSimpleEnumeration(); + + while (enumer.hasMoreElements()) + { + var prop = enumer.getNext().QueryInterface(nsIPropertyElement); + var ary = prop.key.match (/^(msg|msn)/); + if (ary) + { + var constValue; + var constName = prop.key.toUpperCase().replace (/\./g, "_"); + if (ary[1] == "msn" || prop.value.search(/%(\d+\$)?s/i) != -1) + constValue = pfx + prop.key; + else + constValue = prop.value.replace (/^\"/, "").replace (/\"$/, ""); + + constValue = constValue.replace(/\&(\w+)\;/g, replaceEntities); + targetWindow[constName] = constValue; + } + } + + if (this.bundleList.length == 1) + this.defaultBundle = bundle; +} + +MessageManager.prototype.convertHankakuToZenkaku = +function mm_converthankakutozenkaku(msg) +{ + const basicMapping = [ + /* 0xFF60 */ 0xFF60,0x3002,0x300C,0x300D,0x3001,0x30FB,0x30F2,0x30A1, + /* 0xFF68 */ 0x30A3,0x30A5,0x30A7,0x30A9,0x30E3,0x30E5,0x30E7,0x30C3, + /* 0xFF70 */ 0x30FC,0x30A2,0x30A4,0x30A6,0x30A8,0x30AA,0x30AB,0x30AD, + /* 0xFF78 */ 0x30AF,0x30B1,0x30B3,0x30B5,0x30B7,0x30B9,0x30BB,0x30BD, + /* 0xFF80 */ 0x30BF,0x30C1,0x30C4,0x30C6,0x30C8,0x30CA,0x30CB,0x30CC, + /* 0xFF88 */ 0x30CD,0x30CE,0x30CF,0x30D2,0x30D5,0x30D8,0x30DB,0x30DE, + /* 0xFF90 */ 0x30DF,0x30E0,0x30E1,0x30E2,0x30E4,0x30E6,0x30E8,0x30E9, + /* 0xFF98 */ 0x30EA,0x30EB,0x30EC,0x30ED,0x30EF,0x30F3,0x309B,0x309C + ]; + + const HANKAKU_BASE1 = 0xFF60; + const HANKAKU_BASE2 = 0xFF80; + const HANKAKU_MASK = 0xFFE0; + + const MOD_NIGORI = 0xFF9E; + const NIGORI_MIN1 = 0xFF76; + const NIGORI_MAX1 = 0xFF84; + const NIGORI_MIN2 = 0xFF8A; + const NIGORI_MAX2 = 0xFF8E; + const NIGORI_MODIFIER = 1; + + const MOD_MARU = 0xFF9F; + const MARU_MIN = 0xFF8A; + const MARU_MAX = 0xFF8E; + const MARU_MODIFIER = 2; + + var i, src, srcMod, dest; + var rv = ""; + + for (i = 0; i < msg.length; i++) + { + // Get both this character and the next one, which could be a modifier. + src = msg.charCodeAt(i); + if (i < msg.length - 1) + srcMod = msg.charCodeAt(i + 1); + + // Is the source characher hankaku? + if ((HANKAKU_BASE1 == (src & HANKAKU_MASK)) || + (HANKAKU_BASE2 == (src & HANKAKU_MASK))) + { + // Do the basic character mapping first. + dest = basicMapping[src - HANKAKU_BASE1]; + + // If the source character is in the nigori or maru ranges and + // the following character is the associated modifier, we apply + // the modification and skip over the modifier. + if (i < msg.length - 1) + { + if ((MOD_NIGORI == srcMod) && + (((src >= NIGORI_MIN1) && (src <= NIGORI_MAX1)) || + ((src >= NIGORI_MIN2) && (src <= NIGORI_MAX2)))) + { + dest += NIGORI_MODIFIER; + i++; + } + else if ((MOD_MARU == srcMod) && + (src >= MARU_MIN) && (src <= MARU_MAX)) + { + dest += MARU_MODIFIER; + i++; + } + } + + rv += String.fromCharCode(dest); + } + else + { + rv += msg[i]; + } + } + + return rv; +} + +MessageManager.prototype.checkCharset = +function mm_checkset(charset) +{ + try + { + this.ucConverter.charset = charset; + } + catch (ex) + { + return false; + } + + return true; +} + +MessageManager.prototype.toUnicode = +function mm_tounicode(msg, charset) +{ + if (!charset) + return msg; + + try + { + this.ucConverter.charset = charset; + msg = this.ucConverter.ConvertToUnicode(msg); + } + catch (ex) + { + //dd ("caught exception " + ex + " converting " + msg + " to charset " + + // charset); + } + + return msg; +} + +MessageManager.prototype.fromUnicode = +function mm_fromunicode(msg, charset) +{ + if (!charset) + return msg; + + if (this.enableHankakuToZenkaku && (charset.toLowerCase() == "iso-2022-jp")) + msg = this.convertHankakuToZenkaku(msg); + + try + { + // This can actually fail in bizare cases. Cope. + if (charset != this.ucConverter.charset) + this.ucConverter.charset = charset; + + if ("Finish" in this.ucConverter) + { + msg = this.ucConverter.ConvertFromUnicode(msg) + + this.ucConverter.Finish(); + } + else + { + msg = this.ucConverter.ConvertFromUnicode(msg + " "); + msg = msg.substr(0, msg.length - 1); + } + } + catch (ex) + { + //dd ("caught exception " + ex + " converting " + msg + " to charset " + + // charset); + } + + return msg; +} + +MessageManager.prototype.getMsg = +function mm_getmsg (msgName, params, deflt) +{ + try + { + var bundle; + var ary = msgName.match (/(\d+):(.+)/); + if (ary) + { + return (this.getMsgFrom(this.bundleList[ary[1]], ary[2], params, + deflt)); + } + + return this.getMsgFrom(this.bundleList[0], msgName, params, deflt); + } + catch (ex) + { + ASSERT (0, "Caught exception getting message: " + msgName + "/" + + params); + return deflt ? deflt : msgName; + } +} + +MessageManager.prototype.getMsgFrom = +function mm_getfrom (bundle, msgName, params, deflt) +{ + var me = this; + function replaceEntities(matched, entity) + { + if (entity in me.entities) + return me.entities[entity]; + + return matched; + }; + + try + { + var rv; + + if (params && isinstance(params, Array)) + rv = bundle.formatStringFromName (msgName, params, params.length); + else if (params || params == 0) + rv = bundle.formatStringFromName (msgName, [params], 1); + else + rv = bundle.GetStringFromName (msgName); + + /* strip leading and trailing quote characters, see comment at the + * top of venkman.properties. + */ + rv = rv.replace(/^\"/, ""); + rv = rv.replace(/\"$/, ""); + rv = rv.replace(/\&(\w+)\;/g, replaceEntities); + + return rv; + } + catch (ex) + { + if (typeof deflt == "undefined") + { + ASSERT (0, "caught exception getting value for ``" + msgName + + "''\n" + ex + "\n"); + return msgName; + } + return deflt; + } + + return null; +} diff --git a/comm/suite/chatzilla/js/lib/pref-manager.js b/comm/suite/chatzilla/js/lib/pref-manager.js new file mode 100644 index 0000000000..3d0b09ff86 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/pref-manager.js @@ -0,0 +1,443 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +const PREF_RELOAD = true; +const PREF_WRITETHROUGH = true; +const PREF_CHARSET = "utf-8"; // string prefs stored in this charset + +function PrefRecord(name, defaultValue, label, help, group) +{ + this.name = name; + this.defaultValue = defaultValue; + this.help = help; + this.label = label ? label : name; + this.group = group ? group : ""; + // Prepend the group 'general' if there isn't one already. + if (this.group.match(/^(\.|$)/)) + this.group = "general" + this.group; + this.realValue = null; +} + +function PrefManager (branchName, defaultBundle) +{ + var prefManager = this; + + function pm_observe (prefService, topic, prefName) + { + prefManager.onPrefChanged(prefName); + }; + + const PREF_CTRID = "@mozilla.org/preferences-service;1"; + const nsIPrefService = Components.interfaces.nsIPrefService; + const nsIPrefBranch = Components.interfaces.nsIPrefBranch; + + this.prefService = + Components.classes[PREF_CTRID].getService(nsIPrefService); + this.prefBranch = this.prefService.getBranch(branchName); + this.prefSaveTime = 0; + this.prefSaveTimer = 0; + this.branchName = branchName; + this.defaultValues = new Object(); + this.prefs = new Object(); + this.prefNames = new Array(); + this.prefRecords = new Object(); + this.observer = { observe: pm_observe, branch: branchName }; + this.observers = new Array(); + + this.nsIPrefBranch = + this.prefBranch.QueryInterface(nsIPrefBranch); + this.nsIPrefBranch.addObserver("", this.observer, false); + + this.defaultBundle = defaultBundle; + + this.valid = true; +} + +// Delay between change and save. +PrefManager.prototype.PREF_SAVE_DELAY = 5000; // 5 seconds. +/* The timer is reset for each change. Only reset if it hasn't been delayed by + * this much already, or we could put off a save indefinitely. + */ +PrefManager.prototype.PREF_MAX_DELAY = 15000; // 15 seconds. + +// +PrefManager.prototype.destroy = +function pm_destroy() +{ + if (this.valid) + { + this.nsIPrefBranch.removeObserver("", this.observer); + this.valid = false; + } +} + +PrefManager.prototype.getBranch = +function pm_getbranch(suffix) +{ + return this.prefService.getBranch(this.prefBranch.root + suffix); +} + +PrefManager.prototype.getBranchManager = +function pm_getbranchmgr(suffix) +{ + return new PrefManager(this.prefBranch.root + suffix); +} + +PrefManager.prototype.addObserver = +function pm_addobserver(observer) +{ + if (!("onPrefChanged" in observer)) + throw "Bad observer!"; + + this.observers.push(observer); +} + +PrefManager.prototype.removeObserver = +function pm_removeobserver(observer) +{ + for (var i = 0; i < this.observers.length; i++) + { + if (this.observers[i] == observer) + { + arrayRemoveAt(this.observers, i); + break; + } + } +} + +PrefManager.prototype.delayedSave = +function pm_delayedsave() +{ + // this.prefSaveTimer + var now = Number(new Date()); + + /* If the time == 0, there is no delayed save in progress, and we should + * start one. If it isn't 0, check the delayed save was started within the + * allowed time - this means that if we keep putting off a save, it will + * go through eventually, as we will stop resetting it. + */ + if ((this.prefSaveTime == 0) || + (now - this.prefSaveTime < this.PREF_MAX_DELAY)) + { + if (this.prefSaveTime == 0) + this.prefSaveTime = now; + if (this.prefSaveTimer != 0) + clearTimeout(this.prefSaveTimer); + this.prefSaveTimer = setTimeout(function(o) { o.forceSave() }, + this.PREF_SAVE_DELAY, this); + } +} + +PrefManager.prototype.forceSave = +function pm_forcesave() +{ + this.prefSaveTime = 0; + this.prefSaveTimer = 0; + try { + this.prefService.savePrefFile(null); + } catch(ex) { + dd("Exception saving preferences: " + formatException(ex)); + } +} + +PrefManager.prototype.onPrefChanged = +function pm_prefchanged(prefName, realValue, oldValue) +{ + var r, oldValue; + // We're only interested in prefs we actually know about. + if (!(prefName in this.prefRecords) || !(r = this.prefRecords[prefName])) + return; + + if (r.realValue != null) + oldValue = r.realValue; + else if (typeof r.defaultValue == "function") + oldValue = r.defaultValue(prefName); + else + oldValue = r.defaultValue; + + var realValue = this.getPref(prefName, PREF_RELOAD); + + for (var i = 0; i < this.observers.length; i++) + this.observers[i].onPrefChanged(prefName, realValue, oldValue); +} + +PrefManager.prototype.listPrefs = +function pm_listprefs (prefix) +{ + var list = new Array(); + var names = this.prefNames; + for (var i = 0; i < names.length; ++i) + { + if (!prefix || names[i].indexOf(prefix) == 0) + list.push (names[i]); + } + + return list; +} + +PrefManager.prototype.readPrefs = +function pm_readprefs () +{ + const nsIPrefBranch = Components.interfaces.nsIPrefBranch; + + var list = this.prefBranch.getChildList("", {}); + for (var i = 0; i < list.length; ++i) + { + if (!(list[i] in this)) + { + var type = this.prefBranch.getPrefType (list[i]); + var defaultValue; + + switch (type) + { + case nsIPrefBranch.PREF_INT: + defaultValue = 0; + break; + + case nsIPrefBranch.PREF_BOOL: + defaultValue = false; + break; + + default: + defaultValue = ""; + } + + this.addPref(list[i], defaultValue); + } + } +} + +PrefManager.prototype.isKnownPref = +function pm_ispref(prefName) +{ + return (prefName in this.prefRecords); +} + +PrefManager.prototype.addPrefs = +function pm_addprefs(prefSpecs) +{ + var bundle = "stringBundle" in prefSpecs ? prefSpecs.stringBundle : null; + for (var i = 0; i < prefSpecs.length; ++i) + { + this.addPref(prefSpecs[i][0], prefSpecs[i][1], + 3 in prefSpecs[i] ? prefSpecs[i][3] : null, bundle, + 2 in prefSpecs[i] ? prefSpecs[i][2] : null); + } +} + +PrefManager.prototype.updateArrayPref = +function pm_arrayupdate(prefName) +{ + var record = this.prefRecords[prefName]; + if (!ASSERT(record, "Unknown pref: " + prefName)) + return; + + if (record.realValue == null) + record.realValue = record.defaultValue; + + if (!ASSERT(isinstance(record.realValue, Array), "Pref is not an array")) + return; + + this.prefBranch.setCharPref(prefName, this.arrayToString(record.realValue)); + this.delayedSave(); +} + +PrefManager.prototype.stringToArray = +function pm_s2a(string) +{ + if (string.search(/\S/) == -1) + return []; + + var ary = string.split(/\s*;\s*/); + for (var i = 0; i < ary.length; ++i) + ary[i] = toUnicode(unescape(ary[i]), PREF_CHARSET); + + return ary; +} + +PrefManager.prototype.arrayToString = +function pm_a2s(ary) +{ + var escapedAry = new Array() + for (var i = 0; i < ary.length; ++i) + escapedAry[i] = escape(fromUnicode(ary[i], PREF_CHARSET)); + + return escapedAry.join("; "); +} + +PrefManager.prototype.getPref = +function pm_getpref(prefName, reload) +{ + var prefManager = this; + + function updateArrayPref() { prefManager.updateArrayPref(prefName); }; + + var record = this.prefRecords[prefName]; + if (!ASSERT(record, "Unknown pref: " + prefName)) + return null; + + var defaultValue; + + if (typeof record.defaultValue == "function") + { + // deferred pref, call the getter, and don't cache the result. + defaultValue = record.defaultValue(prefName); + } + else + { + if (!reload && record.realValue != null) + return record.realValue; + + defaultValue = record.defaultValue; + } + + var realValue = defaultValue; + + try + { + if (typeof defaultValue == "boolean") + { + realValue = this.prefBranch.getBoolPref(prefName); + } + else if (typeof defaultValue == "number") + { + realValue = this.prefBranch.getIntPref(prefName); + } + else if (isinstance(defaultValue, Array)) + { + realValue = this.prefBranch.getCharPref(prefName); + realValue = this.stringToArray(realValue); + realValue.update = updateArrayPref; + } + else if (typeof defaultValue == "string" || + defaultValue == null) + { + realValue = toUnicode(this.prefBranch.getCharPref(prefName), + PREF_CHARSET); + } + } + catch (ex) + { + // if the pref doesn't exist, ignore the exception. + } + + record.realValue = realValue; + return realValue; +} + +PrefManager.prototype.setPref = +function pm_setpref(prefName, value) +{ + var prefManager = this; + + function updateArrayPref() { prefManager.updateArrayPref(prefName); }; + + var record = this.prefRecords[prefName]; + if (!ASSERT(record, "Unknown pref: " + prefName)) + return null; + + var defaultValue = record.defaultValue; + + if (typeof defaultValue == "function") + defaultValue = defaultValue(prefName); + + if ((record.realValue == null && value == defaultValue) || + record.realValue == value) + { + // no realvalue, and value is the same as default value ... OR ... + // no change at all. just bail. + return record.realValue; + } + + if (value == defaultValue) + { + this.clearPref(prefName); + return value; + } + + if (typeof defaultValue == "boolean") + { + this.prefBranch.setBoolPref(prefName, value); + } + else if (typeof defaultValue == "number") + { + this.prefBranch.setIntPref(prefName, value); + } + else if (isinstance(defaultValue, Array)) + { + var str = this.arrayToString(value); + this.prefBranch.setCharPref(prefName, str); + value.update = updateArrayPref; + } + else + { + this.prefBranch.setCharPref(prefName, fromUnicode(value, PREF_CHARSET)); + } + this.delayedSave(); + + // Always update this after changing the preference. + record.realValue = value; + + return value; +} + +PrefManager.prototype.clearPref = +function pm_reset(prefName) +{ + try { + this.prefBranch.clearUserPref(prefName); + } catch(ex) { + // Do nothing, the pref didn't exist. + } + this.delayedSave(); + + // Always update this after changing the preference. + this.prefRecords[prefName].realValue = null; +} + +PrefManager.prototype.addPref = +function pm_addpref(prefName, defaultValue, setter, bundle, group) +{ + var prefManager = this; + if (!bundle) + bundle = this.defaultBundle; + + function updateArrayPref() { prefManager.updateArrayPref(prefName); }; + function prefGetter() { return prefManager.getPref(prefName); }; + function prefSetter(value) { return prefManager.setPref(prefName, value); }; + + if (!ASSERT(!(prefName in this.defaultValues), + "Preference already exists: " + prefName)) + { + return; + } + + if (!setter) + setter = prefSetter; + + if (isinstance(defaultValue, Array)) + defaultValue.update = updateArrayPref; + + var label = getMsgFrom(bundle, "pref." + prefName + ".label", null, prefName); + var help = getMsgFrom(bundle, "pref." + prefName + ".help", null, + MSG_NO_HELP); + if (group != "hidden") + { + if (label == prefName) + dd("WARNING: !!! Preference without label: " + prefName); + if (help == MSG_NO_HELP) + dd("WARNING: Preference without help text: " + prefName); + } + + this.prefRecords[prefName] = new PrefRecord (prefName, defaultValue, + label, help, group); + + this.prefNames.push(prefName); + this.prefNames.sort(); + + this.prefs.__defineGetter__(prefName, prefGetter); + this.prefs.__defineSetter__(prefName, setter); +} diff --git a/comm/suite/chatzilla/js/lib/protocol-handlers.jsm b/comm/suite/chatzilla/js/lib/protocol-handlers.jsm new file mode 100644 index 0000000000..a24ca1a0de --- /dev/null +++ b/comm/suite/chatzilla/js/lib/protocol-handlers.jsm @@ -0,0 +1,250 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * 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/. */ + +const EXPORTED_SYMBOLS = [ + "ChatZillaProtocols", + "IRCProtocolHandlerFactory", + "IRCSProtocolHandlerFactory", + "IRCPROT_HANDLER_CID", + "IRCSPROT_HANDLER_CID" +]; + +const { classes: Cc, interfaces: Ci, results: Cr } = Components; + +const STANDARDURL_CONTRACTID = + "@mozilla.org/network/standard-url;1"; +const IOSERVICE_CONTRACTID = + "@mozilla.org/network/io-service;1"; + +const IRCPROT_HANDLER_CONTRACTID = + "@mozilla.org/network/protocol;1?name=irc"; +const IRCSPROT_HANDLER_CONTRACTID = + "@mozilla.org/network/protocol;1?name=ircs"; +this.IRCPROT_HANDLER_CID = + Components.ID("{f21c35f4-1dd1-11b2-a503-9bf8a539ea39}"); +this.IRCSPROT_HANDLER_CID = + Components.ID("{f21c35f4-1dd1-11b2-a503-9bf8a539ea3a}"); + +const IRC_MIMETYPE = "application/x-irc"; +const IRCS_MIMETYPE = "application/x-ircs"; + +//XXXgijs: Because necko is annoying and doesn't expose this error flag, we +// define our own constant for it. Throwing something else will show +// ugly errors instead of seeminly doing nothing. +const NS_ERROR_MODULE_NETWORK_BASE = 0x804b0000; +const NS_ERROR_NO_CONTENT = NS_ERROR_MODULE_NETWORK_BASE + 17; + + +function spawnChatZilla(uri) { + var cpmm; + // Ci.nsISyncMessageSender went in Gecko 61. + if (Ci.nsISyncMessageSender) { + cpmm = Cc["@mozilla.org/childprocessmessagemanager;1"] + .getService(Ci.nsISyncMessageSender); + } else { + cpmm = Cc["@mozilla.org/childprocessmessagemanager;1"].getService(); + } + cpmm.sendAsyncMessage("ChatZilla:SpawnChatZilla", { uri }); +} + + +function IRCProtocolHandler(isSecure) +{ + this.isSecure = isSecure; +} + +var protocolFlags = Ci.nsIProtocolHandler.URI_NORELATIVE | + Ci.nsIProtocolHandler.ALLOWS_PROXY; +if ("URI_DANGEROUS_TO_LOAD" in Ci.nsIProtocolHandler) { + protocolFlags |= Ci.nsIProtocolHandler.URI_LOADABLE_BY_ANYONE; +} +if ("URI_NON_PERSISTABLE" in Ci.nsIProtocolHandler) { + protocolFlags |= Ci.nsIProtocolHandler.URI_NON_PERSISTABLE; +} +if ("URI_DOES_NOT_RETURN_DATA" in Ci.nsIProtocolHandler) { + protocolFlags |= Ci.nsIProtocolHandler.URI_DOES_NOT_RETURN_DATA; +} + +IRCProtocolHandler.prototype = +{ + protocolFlags: protocolFlags, + + allowPort(port, scheme) + { + // Allow all ports to connect, so long as they are irc: or ircs: + return (scheme === 'irc' || scheme === 'ircs'); + }, + + newURI(spec, charset, baseURI) + { + const port = this.isSecure ? 6697 : 6667; + + if (!Cc.hasOwnProperty("@mozilla.org/network/standard-url-mutator;1")) { + const cls = Cc[STANDARDURL_CONTRACTID]; + const url = cls.createInstance(Ci.nsIStandardURL); + + url.init(Ci.nsIStandardURL.URLTYPE_STANDARD, port, spec, charset, baseURI); + + return url.QueryInterface(Ci.nsIURI); + } + return Cc["@mozilla.org/network/standard-url-mutator;1"] + .createInstance(Ci.nsIStandardURLMutator) + .init(Ci.nsIStandardURL.URLTYPE_STANDARD, port, spec, charset, baseURI) + .finalize(); + }, + + newChannel(URI) + { + const ios = Cc[IOSERVICE_CONTRACTID].getService(Ci.nsIIOService); + if (!ios.allowPort(URI.port, URI.scheme)) + throw Cr.NS_ERROR_FAILURE; + + return new BogusChannel(URI, this.isSecure); + }, +}; + + +this.IRCProtocolHandlerFactory = +{ + createInstance(outer, iid) + { + if (outer != null) + throw Cr.NS_ERROR_NO_AGGREGATION; + + if (!iid.equals(Ci.nsIProtocolHandler) && !iid.equals(Ci.nsISupports)) + throw Cr.NS_ERROR_INVALID_ARG; + + const protHandler = new IRCProtocolHandler(false); + protHandler.scheme = "irc"; + protHandler.defaultPort = 6667; + return protHandler; + }, +}; + + +this.IRCSProtocolHandlerFactory = +{ + createInstance(outer, iid) + { + if (outer != null) + throw Cr.NS_ERROR_NO_AGGREGATION; + + if (!iid.equals(Ci.nsIProtocolHandler) && !iid.equals(Ci.nsISupports)) + throw Cr.NS_ERROR_INVALID_ARG; + + const protHandler = new IRCProtocolHandler(true); + protHandler.scheme = "ircs"; + protHandler.defaultPort = 6697; + return protHandler; + }, +}; + + +/* Bogus IRC channel used by the IRCProtocolHandler */ +function BogusChannel(URI, isSecure) +{ + this.URI = URI; + this.originalURI = URI; + this.isSecure = isSecure; + this.contentType = this.isSecure ? IRCS_MIMETYPE : IRC_MIMETYPE; +} + +BogusChannel.prototype = +{ + /* nsISupports */ + QueryInterface(iid) + { + if (iid.equals(Ci.nsISupports) || + iid.equals(Ci.nsIChannel) || + iid.equals(Ci.nsIRequest)) + { + return this; + } + + throw Cr.NS_ERROR_NO_INTERFACE; + }, + + /* nsIChannel */ + loadAttributes: null, + contentLength: 0, + owner: null, + loadGroup: null, + notificationCallbacks: null, + securityInfo: null, + + open(observer, context) + { + spawnChatZilla(this.URI.spec); + // We don't throw this (a number, not a real 'resultcode') because it + // upsets xpconnect if we do (error in the js console). + Components.returnCode = NS_ERROR_NO_CONTENT; + }, + + asyncOpen(observer, context) + { + spawnChatZilla(this.URI.spec); + // We don't throw this (a number, not a real 'resultcode') because it + // upsets xpconnect if we do (error in the js console). + Components.returnCode = NS_ERROR_NO_CONTENT; + }, + + asyncRead(listener, context) + { + throw Cr.NS_ERROR_NOT_IMPLEMENTED; + }, + + /* nsIRequest */ + isPending() + { + return true; + }, + + status: Cr.NS_OK, + + cancel(status) + { + this.status = status; + }, + + suspend() + { + throw Cr.NS_ERROR_NOT_IMPLEMENTED; + }, + + resume() + { + throw Cr.NS_ERROR_NOT_IMPLEMENTED; + }, +}; + + +this.ChatZillaProtocols = +{ + init() + { + const compMgr = Components.manager.QueryInterface(Ci.nsIComponentRegistrar); + compMgr.registerFactory(IRCPROT_HANDLER_CID, + "IRC protocol handler", + IRCPROT_HANDLER_CONTRACTID, + IRCProtocolHandlerFactory); + compMgr.registerFactory(IRCSPROT_HANDLER_CID, + "IRC protocol handler", + IRCSPROT_HANDLER_CONTRACTID, + IRCSProtocolHandlerFactory); + }, + + initObsolete(compMgr, fileSpec, location, type) + { + compMgr = compMgr.QueryInterface(Ci.nsIComponentRegistrar); + compMgr.registerFactoryLocation(IRCPROT_HANDLER_CID, + "IRC protocol handler", + IRCPROT_HANDLER_CONTRACTID, + fileSpec, location, type); + compMgr.registerFactoryLocation(IRCSPROT_HANDLER_CID, + "IRCS protocol handler", + IRCSPROT_HANDLER_CONTRACTID, + fileSpec, location, type); + }, +}; diff --git a/comm/suite/chatzilla/js/lib/sts.js b/comm/suite/chatzilla/js/lib/sts.js new file mode 100644 index 0000000000..25052b0985 --- /dev/null +++ b/comm/suite/chatzilla/js/lib/sts.js @@ -0,0 +1,210 @@ +/* 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/. */ + +/** + * The base CIRCSTS object. + */ +function CIRCSTS() +{ + this.policyList = new Object(); + this.preloadList = new Object(); +} + +CIRCSTS.prototype.ENABLED = false; +CIRCSTS.prototype.USE_PRELOAD_LIST = true; + +/** + * Initializes the STS module. + * @param policyFile |nsIFile| for storing the STS cache. + */ +CIRCSTS.prototype.init = function(policyFile) +{ + this.policyFile = policyFile; + this.readCacheFromFile(); +} + +/** + * Reads the policy cache from disk. + */ +CIRCSTS.prototype.readCacheFromFile = function() +{ + if (!this.policyFile.exists()) + return; + + cacheReader = new JSONSerializer(this.policyFile); + this.policyList = cacheReader.deserialize(); + cacheReader.close(); +} + +/** + * Writes the policy cache to disk. + */ +CIRCSTS.prototype.writeCacheToFile = function() +{ + cacheWriter = new JSONSerializer(this.policyFile); + cacheWriter.serialize(this.policyList); + cacheWriter.close(); +} + +/** + * Utility method for determining if an expiration time has passed. + * An expire time of zero is never considered to be expired (as is + * the case for knockout entries). + * + * @param expireTime the time to evaluate, in seconds since the UNIX + * epoch. + * @return boolean value indicating whether the expiration + * time has passed. + */ +CIRCSTS.prototype.isPolicyExpired = function(expireTime) +{ + if (!expireTime) + { + return false; + } + + if (Date.now() > expireTime) + { + return true; + } + + return false; +} + +/** + * Utility method for parsing the raw CAP value for an STS policy, typically + * received via CAP LS or CAP NEW. + * + * @param params the comma-separated list of parameters in + * key[=value] format. + * @return the received parameters in JSON. + * + */ +CIRCSTS.prototype.parseParameters = function(params) +{ + var rv = new Object(); + var keys = params.toLowerCase().split(","); + for (var i = 0; i < keys.length; i++) + { + var [key, value] = keys[i].split("="); + rv[key] = value; + } + + return rv; +} + +/** + * Remove a policy from the cache. + * + * @param host the host to remove a policy for. + */ +CIRCSTS.prototype.removePolicy = function(host) +{ + // If this host is in the preload list, we have to store a knockout entry. + // To do this, we set the port and expiration time to zero. + if (this.getPreloadPolicy(host)) + { + this.policyList[host] = { + port: 0, + expireTime: 0 + }; + } + else + { + delete this.policyList[host]; + } + + this.writeCacheToFile(); +} + +/** + * Retrieves a policy from the preload list for the specified host. + * + * @param host the host to retrieve a policy for. + * @return the policy from the preload list, if any. + */ +CIRCSTS.prototype.getPreloadPolicy = function(host) +{ + if (this.USE_PRELOAD_LIST && (host in this.preloadList)) + { + return this.preloadList[host]; + } + + return null; +} + +/** + * Checks whether there is an upgrade policy in place for a given host + * and, if so, returns the port to connect to. + * + * @param host the host to query an upgrade policy for. + * @return the secure port to connect to, if any. + */ +CIRCSTS.prototype.getUpgradePolicy = function(host) +{ + if (!this.ENABLED) + return null; + + var cdata = this.policyList[host]; + var pdata = this.getPreloadPolicy(host); + + if (cdata) + { + // We have a cached policy. + if (!this.isPolicyExpired(cdata.expireTime)) + { + // Return null if it's a knockout entry. + return cdata.port || null; + } + else if (!pdata) + { + // Remove the policy if it is expired and not in the preload list. + this.removePolicy(host); + } + } + + if (pdata) + { + return pdata.port; + } + + return null; +} + +/** + * Processes a given policy and caches it. This may also be used to renew a + * persistance policy. This should ONLY be called if we are using a secure + * connection. Insecure connections MUST switch to a secure port first. + * + * @param host the host to store an STS policy for. + * @param port the currently connected secure port. + * @param duration the duration of the new policy. + * @param duration the duration of the new policy, in seconds. This may be + * zero if the policy needs to be removed. + */ +CIRCSTS.prototype.setPolicy = function(host, port, duration) +{ + if (!this.ENABLED) + return; + + port = Number(port); + duration = Number(duration); + + // If duration is zero, that's an indication to immediately remove the + // policy, so here's a shortcut. + if (!duration) + { + this.removePolicy(host); + return; + } + + var expireTime = Date.now() + duration*1000; + + this.policyList[host] = { + port: port, + expireTime: expireTime + }; + + this.writeCacheToFile(); +} diff --git a/comm/suite/chatzilla/js/lib/text-logger.js b/comm/suite/chatzilla/js/lib/text-logger.js new file mode 100644 index 0000000000..731dbd2eaf --- /dev/null +++ b/comm/suite/chatzilla/js/lib/text-logger.js @@ -0,0 +1,134 @@ +/* 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/. */ + + +/** + * Serializer for lists of data that can be printed line-by-line. + * If you pass an autoLimit, it will automatically call limit() once the number + * of appended items exceeds the limit (so the number of items will never + * exceed limit*2). + */ + +function TextLogger(path, autoLimit) +{ + // Check if we can open the path. This will throw if it doesn't work + var f = fopen(path, ">>"); + f.close(); + this.path = path; + + this.appended = 0; + if (typeof autoLimit == "number") + this.autoLimit = autoLimit; + else + this.autoLimit = -1; + + // Limit the amount of data in the file when constructing, when asked to. + if (this.autoLimit != -1) + this.limit(); +} + +/** + * Append data (an array or single item) to the file + */ +TextLogger.prototype.append = +function tl_append(data) +{ + if (!isinstance(data, Array)) + data = [data]; + + // If we go over the limit, don't write everything twice: + if ((this.autoLimit != -1) && + (data.length + this.appended > this.autoLimit)) + { + // Collect complete set of data instead: + var dataInFile = this.read(); + var newData = dataInFile.concat(data); + // Get the last |autoLimit| items: yay JS negative indexing! + newData = newData.slice(-this.autoLimit); + this.limit(newData); + return true; + } + + var file = fopen(this.path, ">>"); + for (var i = 0; i < data.length; i++) + file.write(ecmaEscape(data[i]) + "\n"); + file.close(); + this.appended += data.length; + + return true; +} + +/** + * Limit the data already in the file to the data provided, or the count given. + */ +TextLogger.prototype.limit = +function tl_limit(dataOrCount) +{ + // Find data and count: + var data = null, count = -1; + if (isinstance(dataOrCount, Array)) + { + data = dataOrCount; + count = data.length; + } + else if (typeof dataOrCount == "number") + { + count = dataOrCount; + data = this.read(); + } + else if (this.autoLimit != -1) + { + count = this.autoLimit; + data = this.read(); + } + else + { + throw "Can't limit the length of the file without a limit..." + } + + // Write the right data out. Note that we use the back of the array, not + // the front (start from data.length - count), without dropping below 0: + var start = Math.max(data.length - count, 0); + var file = fopen(this.path, ">"); + for (var i = start; i < data.length; i++) + file.write(ecmaEscape(data[i]) + "\n"); + file.close(); + this.appended = 0; + + return true; +} + +/** + * Reads out the data currently in the file, and returns an array. + */ +TextLogger.prototype.read = +function tl_read() +{ + var rv = new Array(), parsedLines = new Array(), buffer = ""; + var file = fopen(this.path, "<"); + while (true) + { + var newData = file.read(); + if (newData) + buffer += newData; + else if (buffer.length == 0) + break; + + // Got more data in the buffer, so split into lines. Unless we're + // done, the last one might not be complete yet, so save that one. + // We split rather strictly on line ends, because empty lines should + // be preserved. + var lines = buffer.split(/\r?\n/); + if (!newData) + buffer = ""; + else + buffer = lines.pop(); + + rv = rv.concat(lines); + } + // Unescape here... + for (var i = 0; i < rv.length; i++) + rv[i] = ecmaUnescape(rv[i]); + return rv; +} diff --git a/comm/suite/chatzilla/js/lib/text-serializer.js b/comm/suite/chatzilla/js/lib/text-serializer.js new file mode 100644 index 0000000000..d9d2a3c7af --- /dev/null +++ b/comm/suite/chatzilla/js/lib/text-serializer.js @@ -0,0 +1,348 @@ +/* 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/. */ + +/* The serialized file format is pretty generic... each line (using any line + * separator, so we don't mind being moved between platforms) consists of + * a command name, and some parameters (optionally). The commands 'start' + * and 'end' mark the chunks of properties for each object - in this case + * motifs. Every command inside a start/end block is considered a property + * for the object. There are some rules, but we are generally pretty flexible. + * + * Example file: + * START <Array> + * START 0 + * "message" "Food%3a%20Mmm...%20food..." + * END + * START 1 + * "message" "Busy%3a%20Working." + * END + * START 2 + * "message" "Not%20here." + * END + * END + * + * The whitespace at the start of the inner lines is generated by the + * serialisation process, but is ignored when parsing - it is only to make + * the file more readable. + * + * The START command may be followed by one or both of a class name (enclosed + * in angle brackets, as above) and a property name (the first non-<>-enclosed + * word). Top-level START commands must not have a property name, although a + * class name is fine. Only the following class names are supported: + * - Object (the default) + * - Array + * + * For arrays, there are some limitations; saving an array cannot save any + * properties that are not numerics, due to limitations in JS' for...in + * enumeration. Thus, for loading, only items with numeric property names are + * allowed. If an item is STARTed inside an array, and specifies no property + * name, it will be push()ed into the array instead. + */ + +function TextSerializer(file) +{ + this._initialized = false; + if (typeof file == "string") + this._file = new nsLocalFile(file); + else + this._file = file; + this._open = false; + this._buffer = ""; + this._lines = []; + this.lineEnd = "\n"; + this._initialized = true; +} + +/* open(direction) + * + * Opens the serializer on the file specified when created, in either the read + * ("<") or write (">") directions. When the file is open, only the appropriate + * direction of serialization/deserialization may be performed. + * + * Note: serialize and deserialize automatically open the file if it is not + * open. + */ +TextSerializer.prototype.open = +function ts_open(dir) +{ + if (!ASSERT((dir == ">") || (dir == "<"), "Bad serialization direction!")) + return false; + if (this._open) + return false; + + this._fileStream = new LocalFile(this._file, dir); + if ((typeof this._fileStream == "object") && this._fileStream) + this._open = true; + + return this._open; +} + +/* close() + * + * Closes the file stream and ends reading or writing. + */ +TextSerializer.prototype.close = +function ts_close() +{ + if (this._open) + { + this._fileStream.close(); + delete this._fileStream; + this._open = false; + } + return true; +} + +/* serialize(object) + * + * Serializes a single object into the file stream. All properties of the object + * are stored in the stream, including properties that contain other objects. + */ +TextSerializer.prototype.serialize = +function ts_serialize(obj) +{ + if (!this._open) + this.open(">"); + if (!ASSERT(this._open, "Unable to open the file for writing!")) + return; + + var me = this; + + function writeObjProps(o, indent) + { + function writeProp(name, val) + { + me._fileStream.write(indent + "\"" + ecmaEscape(name) + "\" " + val + + me.lineEnd); + }; + + for (var p in o) + { + switch (typeof o[p]) + { + case "string": + writeProp(p, '"' + ecmaEscape(o[p]) + '"'); + break; + + case "number": + case "boolean": + case "null": // (just in case) + case "undefined": + // These all serialise to what we want. + writeProp(p, o[p]); + break; + + case "function": + if (isinstance(o[p], RegExp)) + writeProp(p, ecmaEscape("" + o[p])); + // Can't serialize non-RegExp functions (yet). + break; + + case "object": + if (o[p] == null) + { + // typeof null == "object", just to catch us out. + writeProp(p, "null"); + } + else + { + var className = ""; + if (isinstance(o[p], Array)) + className = "<Array> "; + + me._fileStream.write(indent + "START " + className + + ecmaEscape(p) + me.lineEnd); + writeObjProps(o[p], indent + " "); + me._fileStream.write(indent + "END" + me.lineEnd); + } + break; + + default: + // Can't handle anything else! + } + } + }; + + if (isinstance(obj, Array)) + this._fileStream.write("START <Array>" + this.lineEnd); + else + this._fileStream.write("START" + this.lineEnd); + writeObjProps(obj, " "); + this._fileStream.write("END" + this.lineEnd); +} + +/* deserialize() + * + * Reads in enough of the file to deserialize (realize) a single object. The + * object deserialized is returned; all sub-properties of the object are + * deserialized with it. + */ +TextSerializer.prototype.deserialize = +function ts_deserialize() +{ + if (!this._open) + this.open("<"); + if (!ASSERT(this._open, "Unable to open the file for reading!")) + return false; + + var obj = null; + var rv = null; + var objs = new Array(); + + while (true) + { + if (this._lines.length == 0) + { + var newData = this._fileStream.read(); + if (newData) + this._buffer += newData; + else if (this._buffer.length == 0) + break; + + // Got more data in the buffer, so split into lines. Unless we're + // done, the last one might not be complete yet, so save that one. + var lines = this._buffer.split(/[\r\n]+/); + if (!newData) + this._buffer = ""; + else + this._buffer = lines.pop(); + + this._lines = this._lines.concat(lines); + if (this._lines.length == 0) + break; + } + + // Split each line into "command params...". + var parts = this._lines[0].match(/^\s*(\S+)(?:\s+(.*))?$/); + var command = parts[1]; + var params = parts[2]; + + // 'start' and 'end' commands are special. + switch (command.toLowerCase()) + { + case "start": + var paramList = new Array(); + if (params) + paramList = params.split(/\s+/g); + + var className = ""; + if ((paramList.length > 0) && /^<\w+>$/i.test(paramList[0])) + { + className = paramList[0].substr(1, paramList[0].length - 2); + paramList.shift(); + } + + if (!rv) + { + /* The top-level objects are not allowed a property name + * in their START command (it is meaningless). + */ + ASSERT(paramList.length == 0, "Base object with name!"); + // Construct the top-level object. + if (className) + rv = obj = new window[className](); + else + rv = obj = new Object(); + } + else + { + var n; + if (paramList.length == 0) + { + /* Create a new object level, but with no name. This is + * only valid if the parent level is an array. + */ + if (!ASSERT(isinstance(obj, Array), "Parent not Array!")) + return null; + if (className) + n = new window[className](); + else + n = new Object(); + objs.push(obj); + obj.push(n); + obj = n; + } + else + { + /* Create a new object level, store the reference on the + * parent, and set the new object as the current. + */ + if (className) + n = new window[className](); + else + n = new Object(); + objs.push(obj); + obj[ecmaUnescape(paramList[0])] = n; + obj = n; + } + } + + this._lines.shift(); + break; + + case "end": + this._lines.shift(); + if (rv && (objs.length == 0)) + { + // We're done for the day. + return rv; + } + // Return to the previous object level. + obj = objs.pop(); + if (!ASSERT(obj, "Waaa! no object level to return to!")) + return rv; + break; + + default: + this._lines.shift(); + // The property name may be enclosed in quotes. + if (command[0] == '"') + command = command.substr(1, command.length - 2); + // But it is always escaped. + command = ecmaUnescape(command); + + if (!obj) + { + /* If we find a line that is NOT starting a new object, and + * we don't have a current object, we just assume the START + * command was missed. + */ + rv = obj = new Object(); + } + if (params[0] == '"') // String + { + // Remove quotes, then unescape. + params = params.substr(1, params.length - 2); + obj[command] = ecmaUnescape(params); + } + else if (params[0] == "/") // RegExp + { + var p = params.match(/^\/(.*)\/(\w*)$/); + if (ASSERT(p, "RepExp entry malformed, ignored!")) + { + var re = new RegExp(ecmaUnescape(p[1]), p[2]); + obj[command] = re; + } + } + else if (params == "null") // null + { + obj[command] = null; + } + else if (params == "undefined") // undefined + { + obj[command] = undefined; + } + else if ((params == "true") || (params == "false")) // boolean + { + obj[command] = (params == "true"); + } + else // Number + { + obj[command] = Number(params); + } + break; + } + } + return null; +} diff --git a/comm/suite/chatzilla/js/lib/utils.js b/comm/suite/chatzilla/js/lib/utils.js new file mode 100644 index 0000000000..d8aa88baef --- /dev/null +++ b/comm/suite/chatzilla/js/lib/utils.js @@ -0,0 +1,1490 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * + * 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/. */ + +// Namespaces we happen to need: +const XHTML_NS = "http://www.w3.org/1999/xhtml"; + +var utils = new Object(); + +var DEBUG = true; +var dd, warn, TEST, ASSERT; + +if (DEBUG) { + var _dd_pfx = ""; + var _dd_singleIndent = " "; + var _dd_indentLength = _dd_singleIndent.length; + var _dd_currentIndent = ""; + var _dd_lastDumpWasOpen = false; + var _dd_timeStack = new Array(); + var _dd_disableDepth = Number.MAX_VALUE; + var _dd_currentDepth = 0; + dd = function _dd(str) { + if (typeof str != "string") { + dump(str + "\n"); + } else if (str == "") { + dump("<empty-string>\n"); + } else if (str[str.length - 1] == "{") { + ++_dd_currentDepth; + if (_dd_currentDepth >= _dd_disableDepth) + return; + if (str.indexOf("OFF") == 0) + _dd_disableDepth = _dd_currentDepth; + _dd_timeStack.push (new Date()); + if (_dd_lastDumpWasOpen) + dump("\n"); + dump (_dd_pfx + _dd_currentIndent + str); + _dd_currentIndent += _dd_singleIndent; + _dd_lastDumpWasOpen = true; + } else if (str[0] == "}") { + if (--_dd_currentDepth >= _dd_disableDepth) + return; + _dd_disableDepth = Number.MAX_VALUE; + var sufx = (new Date() - _dd_timeStack.pop()) / 1000 + " sec"; + _dd_currentIndent = + _dd_currentIndent.substr(0, _dd_currentIndent.length - + _dd_indentLength); + if (_dd_lastDumpWasOpen) + dump(str + " " + sufx + "\n"); + else + dump(_dd_pfx + _dd_currentIndent + str + " " + + sufx + "\n"); + _dd_lastDumpWasOpen = false; + } else { + if (_dd_currentDepth >= _dd_disableDepth) + return; + if (_dd_lastDumpWasOpen) + dump("\n"); + dump(_dd_pfx + _dd_currentIndent + str + "\n"); + _dd_lastDumpWasOpen = false; + } + } + warn = function (msg) { dd("** WARNING " + msg + " **"); } + TEST = ASSERT = function _assert(expr, msg) { + if (!expr) { + var m = "** ASSERTION FAILED: " + msg + " **\n" + + getStackTrace(); + try { + alert(m); + } catch(ex) {} + dd(m); + return false; + } else { + return true; + } + } +} else { + dd = warn = TEST = ASSERT = function (){}; +} + +function dumpObject (o, pfx, sep) +{ + var p; + var s = ""; + + sep = (typeof sep == "undefined") ? " = " : sep; + pfx = (typeof pfx == "undefined") ? "" : pfx; + + for (p in o) + { + if (typeof (o[p]) != "function") + s += pfx + p + sep + o[p] + "\n"; + else + s += pfx + p + sep + "function\n"; + } + + return s; + +} + +/* Dumps an object in tree format, recurse specifiec the the number of objects + * to recurse, compress is a boolean that can uncompress (true) the output + * format, and level is the number of levels to intitialy indent (only useful + * internally.) A sample dumpObjectTree (o, 1) is shown below. + * + * + parent (object) + * + users (object) + * | + jsbot (object) + * | + mrjs (object) + * | + nakkezzzz (object) + * | * + * + bans (object) + * | * + * + topic (string) 'ircclient.js:59: nothing is not defined' + * + getUsersLength (function) 9 lines + * * + */ +function dumpObjectTree (o, recurse, compress, level) +{ + var s = ""; + var pfx = ""; + + if (typeof recurse == "undefined") + recurse = 0; + if (typeof level == "undefined") + level = 0; + if (typeof compress == "undefined") + compress = true; + + for (var i = 0; i < level; i++) + pfx += (compress) ? "| " : "| "; + + var tee = (compress) ? "+ " : "+- "; + + for (i in o) + { + var t, ex; + + try + { + t = typeof o[i]; + } + catch (ex) + { + t = "ERROR"; + } + + switch (t) + { + case "function": + var sfunc = String(o[i]).split("\n"); + if (sfunc[2] == " [native code]") + sfunc = "[native code]"; + else + if (sfunc.length == 1) + sfunc = String(sfunc); + else + sfunc = sfunc.length + " lines"; + s += pfx + tee + i + " (function) " + sfunc + "\n"; + break; + + case "object": + s += pfx + tee + i + " (object)"; + if (o[i] == null) + { + s += " null\n"; + break; + } + + s += "\n"; + + if (!compress) + s += pfx + "|\n"; + if ((i != "parent") && (recurse)) + s += dumpObjectTree (o[i], recurse - 1, + compress, level + 1); + break; + + case "string": + if (o[i].length > 200) + s += pfx + tee + i + " (" + t + ") " + + o[i].length + " chars\n"; + else + s += pfx + tee + i + " (" + t + ") '" + o[i] + "'\n"; + break; + + case "ERROR": + s += pfx + tee + i + " (" + t + ") ?\n"; + break; + + default: + s += pfx + tee + i + " (" + t + ") " + o[i] + "\n"; + + } + + if (!compress) + s += pfx + "|\n"; + + } + + s += pfx + "*\n"; + + return s; + +} + +function ecmaEscape(str) +{ + function replaceNonPrintables(ch) + { + var rv = ch.charCodeAt().toString(16); + if (rv.length == 1) + rv = "0" + rv; + else if (rv.length == 3) + rv = "u0" + rv; + else if (rv.length == 4) + rv = "u" + rv; + + return "%" + rv; + }; + + // Replace any character that is not in the 69 character set + // [ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789@*_+-./] + // with an escape sequence. Two digit sequences in the form %XX are used + // for characters whose codepoint is less than 255, %uXXXX for all others. + // See section B.2.1 of ECMA-262 rev3 for more information. + return str.replace(/[^A-Za-z0-9@*_+.\-\/]/g, replaceNonPrintables); +} + +function ecmaUnescape(str) +{ + function replaceEscapes(seq) + { + var ary = seq.match(/([\da-f]{1,2})(.*)|u([\da-f]{1,4})/i); + if (!ary) + return "<ERROR>"; + + var rv; + if (ary[1]) + { + // two digit escape, possibly with cruft after + rv = String.fromCharCode(parseInt(ary[1], 16)) + ary[2]; + } + else + { + // four digits, no cruft + rv = String.fromCharCode(parseInt(ary[3], 16)); + } + + return rv; + }; + + // Replace the escape sequences %X, %XX, %uX, %uXX, %uXXX, and %uXXXX with + // the characters they represent, where X is a hexadecimal digit. + // See section B.2.2 of ECMA-262 rev3 for more information. + return str.replace(/%u?([\da-f]{1,4})/ig, replaceEscapes); +} + +function encodeForXMLAttribute(value) { + return value.replace(/&/g, "&").replace(/</g, "<") + .replace(/>/g, ">").replace(/"/g, """) + .replace(/'/g, "'"); +} + +function replaceVars(str, vars) +{ + // replace "string $with a $variable", with + // "string " + vars["with"] + " with a " + vars["variable"] + + function doReplace(symbol) + { + var name = symbol.substr(1); + if (name in vars) + return vars[name]; + + return "$" + name; + }; + + return str.replace(/(\$\w[\w\d\-]+)/g, doReplace); +} + +function formatException(ex) +{ + if (isinstance(ex, Error)) + { + return getMsg(MSG_FMT_JSEXCEPTION, [ex.name, ex.message, ex.fileName, + ex.lineNumber]); + } + if ((typeof ex == "object") && ("filename" in ex)) + { + return getMsg(MSG_FMT_JSEXCEPTION, [ex.name, ex.message, ex.filename, + ex.lineNumber]); + } + + return String(ex); +} + +/* + * Clones an existing object (Only the enumerable properties + * of course.) use as a function.. + * var c = Clone (obj); + * or a constructor... + * var c = new Clone (obj); + */ +function Clone (obj) +{ + var robj = new Object(); + + if ("__proto__" in obj) + { + // Special clone for Spidermonkey. + for (var p in obj) + { + if (obj.hasOwnProperty(p)) + robj[p] = obj[p]; + } + robj.__proto__ = obj.__proto__; + } + else + { + for (var p in obj) + robj[p] = obj[p]; + } + + return robj; + +} + +function Copy(source, dest, overwrite) +{ + if (!dest) + dest = new Object(); + + for (var p in source) + { + if (overwrite || !(p in dest)) + dest[p] = source[p]; + } + + return dest; +} + +/* + * matches a real object against one or more pattern objects. + * if you pass an array of pattern objects, |negate| controls whether to check + * if the object matches ANY of the patterns, or NONE of the patterns. + */ +function matchObject (o, pattern, negate) +{ + negate = Boolean(negate); + + function _match (o, pattern) + { + if (isinstance(pattern, Function)) + return pattern(o); + + for (var p in pattern) + { + var val; + /* nice to have, but slow as molases, allows you to match + * properties of objects with obj$prop: "foo" syntax */ + /* + if (p[0] == "$") + val = eval ("o." + + p.substr(1,p.length).replace (/\$/g, ".")); + else + */ + val = o[p]; + + if (isinstance(pattern[p], Function)) + { + if (!pattern[p](val)) + return false; + } + else + { + var ary = (new String(val)).match(pattern[p]); + if (ary == null) + return false; + else + o.matchresult = ary; + } + } + + return true; + + } + + if (!isinstance(pattern, Array)) + return Boolean (negate ^ _match(o, pattern)); + + for (var i in pattern) + if (_match (o, pattern[i])) + return !negate; + + return negate; + +} + +function equalsObject(o1, o2) +{ + for (var p in o1) + { + if (!(p in o2) || (o1[p] != o2[p])) + return false; + } + for (p in o2) + { + // If the property did exist in o1, the previous loop tested it: + if (!(p in o1)) + return false; + } + return true; +} + +function utils_lcfn(text) +{ + return text.toLowerCase(); +} + +function matchEntry (partialName, list, lcFn) +{ + + if ((typeof partialName == "undefined") || + (String(partialName) == "")) + { + var ary = new Array(); + for (var i in list) + ary.push(i); + return ary; + } + + if (typeof lcFn != "function") + lcFn = utils_lcfn; + + ary = new Array(); + + for (i in list) + { + if (lcFn(list[i]).indexOf(lcFn(partialName)) == 0) + ary.push(i); + } + + return ary; + +} + +function encodeChar(ch) +{ + return "%" + ch.charCodeAt(0).toString(16); +} + +function escapeFileName(fileName) +{ + // Escape / \ : * ? " < > | so they don't cause trouble. + return fileName.replace(/[\/\\\:\*\?"<>\|]/g, encodeChar); +} + +function getCommonPfx (list, lcFn) +{ + var pfx = list[0]; + var l = list.length; + + if (typeof lcFn != "function") + lcFn = utils_lcfn; + + for (var i = 0; i < l; i++) + { + for (var c = 0; c < pfx.length; ++c) + { + if (c >= list[i].length) + { + pfx = pfx.substr(0, c); + break; + } + else + { + if (lcFn(pfx[c]) != lcFn(list[i][c])) + pfx = pfx.substr(0, c); + } + } + } + + return pfx; + +} + +function openTopWin (url) +{ + return openDialog (getBrowserURL(), "_blank", "chrome,all,dialog=no", url); +} + +function getWindowByType (windowType) +{ + const MEDIATOR_CONTRACTID = + "@mozilla.org/appshell/window-mediator;1"; + const nsIWindowMediator = Components.interfaces.nsIWindowMediator; + + var windowManager = + Components.classes[MEDIATOR_CONTRACTID].getService(nsIWindowMediator); + + return windowManager.getMostRecentWindow(windowType); +} + +function toOpenWindowByType(inType, url, features) +{ + var topWindow = getWindowByType(inType); + + if (typeof features == "undefined") + features = "chrome,extrachrome,menubar,resizable," + + "scrollbars,status,toolbar"; + + if (topWindow) + topWindow.focus(); + else + window.open(url, "_blank", features); +} + +function renameProperty (obj, oldname, newname) +{ + + if (oldname == newname) + return; + + obj[newname] = obj[oldname]; + delete obj[oldname]; + +} + +function newObject(contractID, iface) +{ + var rv; + var cls = Components.classes[contractID]; + + if (!cls) + return null; + + switch (typeof iface) + { + case "undefined": + rv = cls.createInstance(); + break; + + case "string": + rv = cls.createInstance(Components.interfaces[iface]); + break; + + case "object": + rv = cls.createInstance(iface); + break; + + default: + rv = null; + break; + } + + return rv; + +} + +function getService(contractID, iface) +{ + var rv; + var cls = Components.classes[contractID]; + + if (!cls) + return null; + + switch (typeof iface) + { + case "undefined": + rv = cls.getService(); + break; + + case "string": + rv = cls.getService(Components.interfaces[iface]); + break; + + case "object": + rv = cls.getService(iface); + break; + + default: + rv = null; + break; + } + + return rv; + +} + +function getNSSErrorClass(errorCode) +{ + var nssErrSvc = getService("@mozilla.org/nss_errors_service;1", "nsINSSErrorsService"); + + try + { + return nssErrSvc.getErrorClass(errorCode); + } + catch + { + return 0; + } +} + +function getContentWindow(frame) +{ + try + { + if (!frame || !("contentWindow" in frame)) + return false; + + // The "in" operator does not detect wrappedJSObject, so don't bother. + if (frame.contentWindow.wrappedJSObject) + return frame.contentWindow.wrappedJSObject; + return frame.contentWindow; + } + catch (ex) + { + // throws exception is contentWindow is gone + return null; + } +} + +function getContentDocument(frame) +{ + try + { + if (!frame || !("contentDocument" in frame)) + return false; + + // The "in" operator does not detect wrappedJSObject, so don't bother. + if (frame.contentDocument.wrappedJSObject) + return frame.contentDocument.wrappedJSObject; + return frame.contentDocument; + } + catch (ex) + { + // throws exception is contentDocument is gone + return null; + } +} + +function getPriv (priv) +{ + var rv = true; + + try + { + netscape.security.PrivilegeManager.enablePrivilege(priv); + } + catch (e) + { + dd ("getPriv: unable to get privlege '" + priv + "': " + e); + rv = false; + } + + return rv; + +} + +function len(o) +{ + var l = 0; + + for (var p in o) + ++l; + + return l; +} + +function keys (o) +{ + var rv = new Array(); + + for (var p in o) + rv.push(p); + + return rv; + +} + +function stringTrim (s) +{ + if (!s) + return ""; + s = s.replace (/^\s+/, ""); + return s.replace (/\s+$/, ""); + +} + +/* the offset should be in seconds, it will be rounded to 2 decimal places */ +function formatDateOffset (offset, format) +{ + var seconds = roundTo(offset % 60, 2); + var minutes = Math.floor(offset / 60); + var hours = Math.floor(minutes / 60); + minutes = minutes % 60; + var days = Math.floor(hours / 24); + hours = hours % 24; + + if (!format) + { + var ary = new Array(); + + if (days == 1) + ary.push(MSG_DAY); + else if (days > 0) + ary.push(getMsg(MSG_DAYS, days)); + + if (hours == 1) + ary.push(MSG_HOUR); + else if (hours > 0) + ary.push(getMsg(MSG_HOURS, hours)); + + if (minutes == 1) + ary.push(MSG_MINUTE); + else if (minutes > 0) + ary.push(getMsg(MSG_MINUTES, minutes)); + + if (seconds == 1) + ary.push(MSG_SECOND); + else if (seconds > 0 || offset == 0) + ary.push(getMsg(MSG_SECONDS, seconds)); + + format = ary.join(", "); + } + else + { + format = format.replace ("%d", days); + format = format.replace ("%h", hours); + format = format.replace ("%m", minutes); + format = format.replace ("%s", seconds); + } + + return format; +} + +function arrayHasElementAt(ary, i) +{ + return typeof ary[i] != "undefined"; +} + +function arrayContains (ary, elem) +{ + return (arrayIndexOf (ary, elem) != -1); +} + +function arrayIndexOf (ary, elem) +{ + for (var i in ary) + if (ary[i] == elem) + return i; + + return -1; +} + +function arrayInsertAt (ary, i, o) +{ + ary.splice (i, 0, o); +} + +function arrayRemoveAt (ary, i) +{ + ary.splice (i, 1); +} + +function objectContains(o, p) +{ + return Object.hasOwnProperty.call(o, p); +} + +/* length should be an even number >= 6 */ +function abbreviateWord (str, length) +{ + if (str.length <= length || length < 6) + return str; + + var left = str.substr (0, (length / 2) - 1); + var right = str.substr (str.length - (length / 2) + 1); + + return left + "..." + right; +} + +/* + * Inserts the string |hyphen| into string |str| every |pos| characters. + * If there are any wordbreaking characters in |str| within -/+5 characters of + * of a |pos| then the hyphen is inserted there instead, in order to produce a + * "cleaner" break. + */ +function hyphenateWord (str, pos, hyphen) +{ + if (str.length <= pos) + return str; + if (typeof hyphen == "undefined") + hyphen = " "; + + /* search for a nice place to break the word, fuzzfactor of +/-5, centered + * around |pos| */ + var splitPos = + str.substring(pos - 5, pos + 5).search(/[^A-Za-z0-9]/); + + splitPos = (splitPos != -1) ? pos - 4 + splitPos : pos; + var left = str.substr (0, splitPos); + var right = hyphenateWord(str.substr (splitPos), pos, hyphen); + + return left + hyphen + right; +} + +/* + * Like hyphenateWord, except individual chunks of the word are returned as + * elements of an array. + */ +function splitLongWord (str, pos) +{ + if (str.length <= pos) + return [str]; + + var ary = new Array(); + var right = str; + + while (right.length > pos) + { + /* search for a nice place to break the word, fuzzfactor of +/-5, + * centered around |pos| */ + var splitPos = + right.substring(pos - 5, pos + 5).search(/[^A-Za-z0-9]/); + + splitPos = (splitPos != -1) ? pos - 4 + splitPos : pos; + ary.push(right.substr (0, splitPos)); + right = right.substr (splitPos); + } + + ary.push (right); + + return ary; +} + +function getRandomElement (ary) +{ + + return ary[Math.floor(Math.random() * ary.length)]; + +} + +function roundTo (num, prec) +{ + + return Math.round(num * Math.pow (10, prec)) / Math.pow (10, prec); + +} + +function randomRange (min, max) +{ + + if (typeof min == "undefined") + min = 0; + + if (typeof max == "undefined") + max = 1; + + return Math.floor(Math.random() * (max - min + 1)) + min; + +} + +// Creates a random string of |len| characters from a-z, A-Z, 0-9. +function randomString(len) { + var chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"; + var rv = ""; + + for (var i = 0; i < len; i++) + rv += chars.substr(Math.floor(Math.random() * chars.length), 1); + + return rv; +} + +function getStackTrace () +{ + var frame = Components.stack.caller; + var str = "<top>"; + + while (frame) + { + var name = frame.name ? frame.name : "[anonymous]"; + str += "\n" + name + "@" + frame.lineNumber; + frame = frame.caller; + } + + return str; + +} + +function getInterfaces (cls) +{ + var rv = new Object(); + var e; + + for (var i in Components.interfaces) + { + try + { + var ifc = Components.interfaces[i]; + cls.QueryInterface(ifc); + rv[i] = ifc; + } + catch (e) + { + /* nada */ + } + } + + return rv; + +} + +/** + * Calls a named function for each element in an array, sending + * the same parameter each call. + * + * @param ary an array of objects + * @param func_name string name of function to call. + * @param data data object to pass to each object. + */ +function mapObjFunc(ary, func_name, data) +{ + /* + * WARNING: Caller assumes resonsibility to verify ary + * and func_name + */ + + for (var i in ary) + ary[i][func_name](data); +} + +/** + * Passes each element of an array to a given function object. + * + * @param func a function object. + * @param ary an array of values. + */ +function map(func, ary) { + + /* + * WARNING: Caller assumnes responsibility to verify + * func and ary. + */ + + for (var i in ary) + func(ary[i]); + +} + +function getSpecialDirectory(name) +{ + if (!("directoryService" in utils)) + { + const DS_CTR = "@mozilla.org/file/directory_service;1"; + const nsIProperties = Components.interfaces.nsIProperties; + + utils.directoryService = + Components.classes[DS_CTR].getService(nsIProperties); + } + + return utils.directoryService.get(name, Components.interfaces.nsIFile); +} + +function getFileFromURLSpec(url) +{ + var handler = Services.io.getProtocolHandler("file") + .QueryInterface(Ci.nsIFileProtocolHandler); + return handler.getFileFromURLSpec(url); +} + +function getURLSpecFromFile(file) +{ + if (!file) + return null; + + if (typeof file == "string") + { + let fileObj = Cc["@mozilla.org/file/local;1"] + .createInstance(Ci.nsIFile); + fileObj.initWithPath(file); + file = fileObj; + } + + var fileHandler = Services.io.getProtocolHandler("file") + .QueryInterface(Ci.nsIFileProtocolHandler); + return fileHandler.getURLSpecFromFile(file); +} + +function alert(msg, parent, title) +{ + var PROMPT_CTRID = "@mozilla.org/embedcomp/prompt-service;1"; + var nsIPromptService = Components.interfaces.nsIPromptService; + var ps = Components.classes[PROMPT_CTRID].getService(nsIPromptService); + if (!parent) + parent = window; + if (!title) + title = MSG_ALERT; + ps.alert (parent, title, msg); +} + +function confirm(msg, parent, title) +{ + var PROMPT_CTRID = "@mozilla.org/embedcomp/prompt-service;1"; + var nsIPromptService = Components.interfaces.nsIPromptService; + var ps = Components.classes[PROMPT_CTRID].getService(nsIPromptService); + if (!parent) + parent = window; + if (!title) + title = MSG_CONFIRM; + return ps.confirm (parent, title, msg); +} + +function confirmEx(msg, buttons, defaultButton, checkText, + checkVal, parent, title) +{ + /* Note that on versions before Mozilla 0.9, using 3 buttons, + * the revert or dontsave button, or custom button titles will NOT work. + * + * The buttons should be listed in the 'accept', 'cancel' and 'extra' order, + * and the exact button order is host app- and platform-dependant. + * For example, on Windows this is usually [button 1] [button 3] [button 2], + * and on Linux [button 3] [button 2] [button 1]. + */ + var PROMPT_CTRID = "@mozilla.org/embedcomp/prompt-service;1"; + var nsIPromptService = Components.interfaces.nsIPromptService; + var ps = Components.classes[PROMPT_CTRID].getService(nsIPromptService); + + var buttonConstants = { + ok: ps.BUTTON_TITLE_OK, + cancel: ps.BUTTON_TITLE_CANCEL, + yes: ps.BUTTON_TITLE_YES, + no: ps.BUTTON_TITLE_NO, + save: ps.BUTTON_TITLE_SAVE, + revert: ps.BUTTON_TITLE_REVERT, + dontsave: ps.BUTTON_TITLE_DONT_SAVE + }; + var buttonFlags = 0; + var buttonText = [null, null, null]; + + if (!isinstance(buttons, Array)) + throw "buttons parameter must be an Array"; + if ((buttons.length < 1) || (buttons.length > 3)) + throw "the buttons array must have 1, 2 or 3 elements"; + + for (var i = 0; i < buttons.length; i++) + { + var buttonFlag = ps.BUTTON_TITLE_IS_STRING; + if ((buttons[i][0] == "!") && (buttons[i].substr(1) in buttonConstants)) + buttonFlag = buttonConstants[buttons[i].substr(1)]; + else + buttonText[i] = buttons[i]; + + buttonFlags += ps["BUTTON_POS_" + i] * buttonFlag; + } + + // ignore anything but a proper number + var defaultIsNumber = (typeof defaultButton == "number"); + if (defaultIsNumber && arrayHasElementAt(buttons, defaultButton)) + buttonFlags += ps["BUTTON_POS_" + defaultButton + "_DEFAULT"]; + + if (!parent) + parent = window; + if (!title) + title = MSG_CONFIRM; + if (!checkVal) + checkVal = new Object(); + + var rv = ps.confirmEx(parent, title, msg, buttonFlags, buttonText[0], + buttonText[1], buttonText[2], checkText, checkVal); + return rv; +} + +function prompt(msg, initial, parent, title) +{ + var PROMPT_CTRID = "@mozilla.org/embedcomp/prompt-service;1"; + var nsIPromptService = Components.interfaces.nsIPromptService; + var ps = Components.classes[PROMPT_CTRID].getService(nsIPromptService); + if (!parent) + parent = window; + if (!title) + title = MSG_PROMPT; + var rv = { value: initial }; + + if (!ps.prompt (parent, title, msg, rv, null, {value: null})) + return null; + + return rv.value; +} + +function promptPassword(msg, initial, parent, title) +{ + var PROMPT_CTRID = "@mozilla.org/embedcomp/prompt-service;1"; + var nsIPromptService = Components.interfaces.nsIPromptService; + var ps = Components.classes[PROMPT_CTRID].getService(nsIPromptService); + if (!parent) + parent = window; + if (!title) + title = MSG_PROMPT; + var rv = { value: initial }; + + if (!ps.promptPassword (parent, title, msg, rv, null, {value: null})) + return null; + + return rv.value; +} + +function viewCert(cert, parent) +{ + var cd = getService("@mozilla.org/nsCertificateDialogs;1", + "nsICertificateDialogs"); + if (!parent) + parent = window; + cd.viewCert(parent, cert); +} + +function addOrUpdateLogin(url, type, username, password) +{ + username = username.toLowerCase(); + var newinfo = newObject("@mozilla.org/login-manager/loginInfo;1", + "nsILoginInfo"); + newinfo.init(url, null, type, username, password, "", ""); + var oldinfo = getLogin(url, type, username); + + if (oldinfo) { + Services.logins.modifyLogin(oldinfo, newinfo); + return true; //updated + } + + Services.logins.addLogin(newinfo); + return false; //added +} + +function getLogin(url, realm, username) +{ + username = username.toLowerCase(); + + let logins = Services.logins.findLogins({}, url, null, realm); + for (let login of logins) { + if (login.username == username) { + return login; + } + } + + return null; +} + +function getHostmaskParts(hostmask) +{ + var rv; + // A bit cheeky this, we try the matches here, and then branch + // according to the ones we like. + var ary1 = hostmask.match(/([^ ]*)!([^ ]*)@(.*)/); + var ary2 = hostmask.match(/([^ ]*)@(.*)/); + var ary3 = hostmask.match(/([^ ]*)!(.*)/); + if (ary1) + rv = { nick: ary1[1], user: ary1[2], host: ary1[3] }; + else if (ary2) + rv = { nick: "*", user: ary2[1], host: ary2[2] }; + else if (ary3) + rv = { nick: ary3[1], user: ary3[2], host: "*" }; + else + rv = { nick: hostmask, user: "*", host: "*" }; + // Make sure we got something for all fields. + if (!rv.nick) + rv.nick = "*"; + if (!rv.user) + rv.user = "*"; + if (!rv.host) + rv.host = "*"; + // And re-construct the 'parsed' hostmask. + rv.mask = rv.nick + "!" + rv.user + "@" + rv.host; + return rv; +} + +function makeMaskRegExp(text) +{ + function escapeChars(c) + { + if (c == "*") + return ".*"; + if (c == "?") + return "."; + return "\\" + c; + } + // Anything that's not alpha-numeric gets escaped. + // "*" and "?" are 'escaped' to ".*" and ".". + // Optimisation; * translates as 'match all'. + return new RegExp("^" + text.replace(/[^\w\d]/g, escapeChars) + "$", "i"); +} + +function hostmaskMatches(user, mask) +{ + // Need to match .nick, .user, and .host. + if (!("nickRE" in mask)) + { + // We cache all the regexp objects, but use null if the term is + // just "*", so we can skip having the object *and* the .match + // later on. + if (mask.nick == "*") + mask.nickRE = null; + else + mask.nickRE = makeMaskRegExp(mask.nick); + + if (mask.user == "*") + mask.userRE = null; + else + mask.userRE = makeMaskRegExp(mask.user); + + if (mask.host == "*") + mask.hostRE = null; + else + mask.hostRE = makeMaskRegExp(mask.host); + } + + var lowerNick; + if (user.TYPE == "IRCChanUser") + lowerNick = user.parent.parent.toLowerCase(user.unicodeName); + else + lowerNick = user.parent.toLowerCase(user.unicodeName); + + if ((!mask.nickRE || lowerNick.match(mask.nickRE)) && + (!mask.userRE || user.name.match(mask.userRE)) && + (!mask.hostRE || user.host.match(mask.hostRE))) + return true; + return false; +} + +function isinstance(inst, base) +{ + /* Returns |true| if |inst| was constructed by |base|. Not 100% accurate, + * but plenty good enough for us. This is to work around the fix for bug + * 254067 which makes instanceof fail if the two sides are 'from' + * different windows (something we don't care about). + */ + return (inst && base && + ((inst instanceof base) || + (inst.constructor && (inst.constructor.name == base.name)))); +} + +function isDefaultPrevented(ev) +{ + if ("defaultPrevented" in ev) + return ev.defaultPrevented; + return ev.getPreventDefault(); +} + +function scaleNumberBy1024(number) +{ + var scale = 0; + while ((number >= 1000) && (scale < 6)) + { + scale++; + number /= 1024; + } + + return [scale, number]; +} + +function getSISize(size) +{ + var data = scaleNumberBy1024(size); + + if (data[1] < 10) + data[1] = data[1].toFixed(2); + else if (data[1] < 100) + data[1] = data[1].toFixed(1); + else + data[1] = data[1].toFixed(0); + + return getMsg(MSG_SI_SIZE, [data[1], getMsg("msg.si.size." + data[0])]); +} + +function getSISpeed(speed) +{ + var data = scaleNumberBy1024(speed); + + if (data[1] < 10) + data[1] = data[1].toFixed(2); + else if (data[1] < 100) + data[1] = data[1].toFixed(1); + else + data[1] = data[1].toFixed(0); + + return getMsg(MSG_SI_SPEED, [data[1], getMsg("msg.si.speed." + data[0])]); +} + +// Returns -1 if version 1 is newer, +1 if version 2 is newer, and 0 for same. +function compareVersions(ver1, ver2) +{ + var ver1parts = ver1.split("."); + var ver2parts = ver2.split("."); + + while ((ver1parts.length > 0) && (ver2parts.length > 0)) + { + if (ver1parts[0] < ver2parts[0]) + return 1; + if (ver1parts[0] > ver2parts[0]) + return -1; + ver1parts.shift(); + ver2parts.shift(); + } + if (ver1parts.length > 0) + return -1; + if (ver2parts.length > 0) + return 1; + return 0; +} + +// Zero-pad Numbers (or pad with something else if you wish) +function padNumber(num, digits, pad) +{ + pad = pad || "0"; + var rv = num.toString(); + while (rv.length < digits) + rv = pad + rv; + return rv; +} + +const timestr = { + A: { method: "getDay" }, + a: { method: "getDay" }, + B: { method: "getMonth" }, + b: { method: "getMonth" }, + c: { replace: null }, + D: { replace: "%m/%d/%y" }, + d: { method: "getDate", pad: 2 }, + e: { method: "getDate", pad: 2, padwith: " " }, + F: { replace: "%Y-%m-%d" }, + h: { replace: "%b" }, + H: { method: "getHours", pad: 2 }, + k: { method: "getHours", pad: 2, padwith: " " }, + M: { method: "getMinutes", pad: 2 }, + p: { AM: null, PM: null }, + P: { AM: null, PM: null }, + r: { replace: null }, + R: { replace: "%H:%M" }, + S: { method: "getSeconds", pad: 2 }, + T: { replace: "%H:%M:%S" }, + w: { method: "getDay" }, + x: { replace: null }, + X: { replace: null }, + Y: { method: "getFullYear" }, + initialized: false +} + +function strftime(format, time) +{ + /* Javascript implementation of standard C strftime */ + + if (!timestr.initialized) + { + timestr.A.values = getMsg("datetime.day.long").split("^"); + timestr.a.values = getMsg("datetime.day.short").split("^"); + timestr.B.values = getMsg("datetime.month.long").split("^"); + timestr.b.values = getMsg("datetime.month.short").split("^"); + // Just make sure the locale isn't playing silly with us. + ASSERT(timestr.A.values.length == 7, "datetime.day.long bad!"); + ASSERT(timestr.a.values.length == 7, "datetime.day.short bad!"); + ASSERT(timestr.B.values.length == 12, "datetime.month.long bad!"); + ASSERT(timestr.b.values.length == 12, "datetime.month.short bad!"); + + timestr.p.AM = getMsg("datetime.uam"); + timestr.p.PM = getMsg("datetime.upm"); + timestr.P.AM = getMsg("datetime.lam"); + timestr.P.PM = getMsg("datetime.lpm"); + + timestr.c.replace = getMsg("datetime.presets.lc"); + timestr.r.replace = getMsg("datetime.presets.lr"); + timestr.x.replace = getMsg("datetime.presets.lx"); + timestr.X.replace = getMsg("datetime.presets.ux"); + + timestr.initialized = true; + } + + + function getDayOfYear(dateobj) + { + var yearobj = new Date(dateobj.getFullYear(), 0, 1, 0, 0, 0, 0); + return Math.floor((dateobj - yearobj) / 86400000) + 1; + }; + + time = time || new Date(); + if (!isinstance(time, Date)) + throw "Expected date object"; + + var ary; + while ((ary = format.match(/(^|[^%])%(\w)/))) + { + var start = ary[1] ? (ary.index + 1) : ary.index; + var rpl = ""; + if (ary[2] in timestr) + { + var tbranch = timestr[ary[2]]; + if (("method" in tbranch) && ("values" in tbranch)) + rpl = tbranch.values[time[tbranch.method]()]; + else if ("method" in tbranch) + rpl = time[tbranch.method]().toString(); + else if ("replace" in tbranch) + rpl = tbranch.replace; + + if ("pad" in tbranch) + { + var padwith = ("padwith" in tbranch) ? tbranch.padwith : "0"; + rpl = padNumber(rpl, tbranch.pad, padwith); + } + } + if (!rpl) + { + switch (ary[2]) + { + case "C": + var century = Math.floor(time.getFullYear() / 100); + rpl = padNumber(century, 2); + break; + case "I": + case "l": + var hour = (time.getHours() + 11) % 12 + 1; + var padwith = (ary[2] == "I") ? "0" : " "; + rpl = padNumber(hour, 2, padwith); + break; + case "j": + rpl = padNumber(getDayOfYear(time), 3); + break; + case "m": + rpl = padNumber(time.getMonth() + 1, 2); + break; + case "p": + case "P": + var bit = (time.getHours() < 12) ? "AM" : "PM"; + rpl = timestr[ary[2]][bit]; + break; + case "s": + rpl = Math.round(time.getTime() / 1000); + break; + case "u": + rpl = (time.getDay() + 6) % 7 + 1; + break; + case "y": + rpl = time.getFullYear().toString().substr(2); + break; + case "z": + var mins = time.getTimezoneOffset(); + rpl = (mins > 0) ? "-" : "+"; + mins = Math.abs(mins); + var hours = Math.floor(mins / 60); + rpl += padNumber(hours, 2) + padNumber(mins - (hours * 60), 2); + break; + } + } + if (!rpl) + rpl = "%%" + ary[2]; + format = format.substr(0, start) + rpl + format.substr(start + 2); + } + return format.replace(/%%/, "%"); +} + +// This used to be strres.js, copied here to help remove that... +var strBundleService = null; +function srGetStrBundle(path) +{ + const STRBSCID = "@mozilla.org/intl/stringbundle;1"; + const STRBSIF = "nsIStringBundleService"; + var strBundle = null; + if (!strBundleService) + { + try + { + strBundleService = getService(STRBSCID, STRBSIF); + } + catch (ex) + { + dump("\n--** strBundleService failed: " + ex + "\n"); + return null; + } + } + + strBundle = strBundleService.createBundle(path); + if (!strBundle) + dump("\n--** strBundle createInstance failed **--\n"); + + return strBundle; +} + +// No-op window.getAttention if it's not found, this is for in-a-tab mode. +if (typeof getAttention == "undefined") + getAttention = function() {}; |