/* jslint node: true */ 'use strict'; let Config = require('./config.js').config; let Address = require('./ftn_address.js'); let FNV1a = require('./fnv1a.js'); let _ = require('lodash'); let assert = require('assert'); let binary = require('binary'); let fs = require('fs'); let util = require('util'); let iconv = require('iconv-lite'); let moment = require('moment'); let createHash = require('crypto').createHash; let uuid = require('node-uuid'); let os = require('os'); let packageJson = require('../package.json'); // :TODO: Remove "Ftn" from most of these -- it's implied in the module exports.stringToNullPaddedBuffer = stringToNullPaddedBuffer; exports.getMessageSerialNumber = getMessageSerialNumber; exports.createMessageUuid = createMessageUuid; exports.getDateFromFtnDateTime = getDateFromFtnDateTime; exports.getDateTimeString = getDateTimeString; exports.getMessageIdentifier = getMessageIdentifier; exports.getProductIdentifier = getProductIdentifier; exports.getUTCTimeZoneOffset = getUTCTimeZoneOffset; exports.getOrigin = getOrigin; exports.getTearLine = getTearLine; exports.getVia = getVia; exports.getAbbreviatedNetNodeList = getAbbreviatedNetNodeList; exports.parseAbbreviatedNetNodeList = parseAbbreviatedNetNodeList; exports.getUpdatedSeenByEntries = getUpdatedSeenByEntries; exports.getUpdatedPathEntries = getUpdatedPathEntries; exports.getCharacterSetIdentifierByEncoding = getCharacterSetIdentifierByEncoding; exports.getEncodingFromCharacterSetIdentifier = getEncodingFromCharacterSetIdentifier; exports.getQuotePrefix = getQuotePrefix; // // Namespace for RFC-4122 name based UUIDs generated from // FTN kludges MSGID + AREA // const ENIGMA_FTN_MSGID_NAMESPACE = uuid.parse('a5c7ae11-420c-4469-a116-0e9a6d8d2654'); // Up to 5D FTN address RegExp const ENIGMA_FTN_ADDRESS_REGEXP = /^([0-9]+:)?([0-9]+)(\/[0-9]+)?(\.[0-9]+)?(@[a-z0-9\-\.]+)?$/i; // See list here: https://github.com/Mithgol/node-fidonet-jam function stringToNullPaddedBuffer(s, bufLen) { let buffer = new Buffer(bufLen).fill(0x00); let enc = iconv.encode(s, 'CP437').slice(0, bufLen); for(let i = 0; i < enc.length; ++i) { buffer[i] = enc[i]; } return buffer; } // // Convert a FTN style DateTime string to a Date object // // :TODO: Name the next couple methods better - for FTN *packets* function getDateFromFtnDateTime(dateTime) { // // Examples seen in the wild (Working): // "12 Sep 88 18:17:59" // "Tue 01 Jan 80 00:00" // "27 Feb 15 00:00:03" // // :TODO: Use moment.js here return (new Date(Date.parse(dateTime))).toISOString(); } function getDateTimeString(m) { // // From http://ftsc.org/docs/fts-0001.016: // DateTime = (* a character string 20 characters long *) // (* 01 Jan 86 02:34:56 *) // DayOfMonth " " Month " " Year " " // " " HH ":" MM ":" SS // Null // // DayOfMonth = "01" | "02" | "03" | ... | "31" (* Fido 0 fills *) // Month = "Jan" | "Feb" | "Mar" | "Apr" | "May" | "Jun" | // "Jul" | "Aug" | "Sep" | "Oct" | "Nov" | "Dec" // Year = "01" | "02" | .. | "85" | "86" | ... | "99" | "00" // HH = "00" | .. | "23" // MM = "00" | .. | "59" // SS = "00" | .. | "59" // if(!moment.isMoment(m)) { m = moment(m); } return m.format('DD MMM YY HH:mm:ss'); } function createMessageUuid(ftnMsgId, ftnArea) { // // v5 UUID generation code based on the work here: // https://github.com/download13/uuidv5/blob/master/uuid.js // // Note: CrashMail uses MSGID + AREA, so we go with that as well: // https://github.com/larsks/crashmail/blob/master/crashmail/dupe.c // if(!Buffer.isBuffer(ftnMsgId)) { ftnMsgId = iconv.encode(ftnMsgId, 'CP437'); } ftnArea = ftnArea || ''; // AREA is optional if(!Buffer.isBuffer(ftnArea)) { ftnArea = iconv.encode(ftnArea, 'CP437'); } const ns = new Buffer(ENIGMA_FTN_MSGID_NAMESPACE); let digest = createHash('sha1').update( Buffer.concat([ ns, ftnMsgId, ftnArea ])).digest(); let u = new Buffer(16); // bbbb - bb - bb - bb - bbbbbb digest.copy(u, 0, 0, 4); // time_low digest.copy(u, 4, 4, 6); // time_mid digest.copy(u, 6, 6, 8); // time_hi_and_version u[6] = (u[6] & 0x0f) | 0x50; // version, 4 most significant bits are set to version 5 (0101) u[8] = (digest[8] & 0x3f) | 0x80; // clock_seq_hi_and_reserved, 2msb are set to 10 u[9] = digest[9]; digest.copy(u, 10, 10, 16); return uuid.unparse(u); // to string } function getMessageSerialNumber(messageId) { const msSinceEnigmaEpoc = (Date.now() - Date.UTC(2016, 1, 1)); const hash = Math.abs(new FNV1a(msSinceEnigmaEpoc + messageId).value).toString(16); return `00000000${hash}`.substr(-8); // return ('00000000' + ((Math.floor((Date.now() - Date.UTC(2016, 1, 1)) / 1000) + // message.messageId)).toString(16)).substr(-8); } // // Return a FTS-0009.001 compliant MSGID value given a message // See http://ftsc.org/docs/fts-0009.001 // // "A MSGID line consists of the string "^AMSGID:" (where ^A is a // control-A (hex 01) and the double-quotes are not part of the // string), followed by a space, the address of the originating // system, and a serial number unique to that message on the // originating system, i.e.: // // ^AMSGID: origaddr serialno // // The originating address should be specified in a form that // constitutes a valid return address for the originating network. // If the originating address is enclosed in double-quotes, the // entire string between the beginning and ending double-quotes is // considered to be the orginating address. A double-quote character // within a quoted address is represented by by two consecutive // double-quote characters. The serial number may be any eight // character hexadecimal number, as long as it is unique - no two // messages from a given system may have the same serial number // within a three years. The manner in which this serial number is // generated is left to the implementor." // // // Examples & Implementations // // Synchronet: .@ // 2606.agora-agn_tst@46:1/142 19609217 // // Mystic: // 46:3/102 46686263 // // ENiGMA½: .@<5dFtnAddress> // function getMessageIdentifier(message, address) { const addrStr = new Address(address).toString('5D'); return `${message.messageId}.${message.areaTag.toLowerCase()}@${addrStr} ${getMessageSerialNumber(message.messageId)}`; } // // Return a FSC-0046.005 Product Identifier or "PID" // http://ftsc.org/docs/fsc-0046.005 // // Note that we use a variant on the spec for // in which (; ; ) is used instead // function getProductIdentifier() { const version = packageJson.version .replace(/\-/g, '.') .replace(/alpha/,'a') .replace(/beta/,'b'); const nodeVer = process.version.substr(1); // remove 'v' prefix return `ENiGMA1/2 ${version} (${os.platform()}; ${os.arch()}; ${nodeVer})`; } // // Return a FSC-0030.001 compliant (http://ftsc.org/docs/fsc-0030.001) MESSAGE-ID // // // // :TODO: not implemented to spec at all yet :) function getFTNMessageID(messageId, areaId) { return messageId + '.' + areaId + '@' + getFTNAddress() + ' ' + getMessageSerialNumber(messageId) } // // Return a FRL-1004 style time zone offset for a // 'TZUTC' kludge line // // http://ftsc.org/docs/frl-1004.002 // function getUTCTimeZoneOffset() { return moment().format('ZZ').replace(/\+/, ''); } // Get a FSC-0032 style quote prefixes function getQuotePrefix(name) { // :TODO: Add support for real names (e.g. with spaces) -> initials return ' ' + name[0].toUpperCase() + name[1].toLowerCase() + '> '; } // // Return a FTS-0004 Origin line // http://ftsc.org/docs/fts-0004.001 // function getOrigin(address) { const origin = _.has(Config.messageNetworks.originName) ? Config.messageNetworks.originName : Config.general.boardName; const addrStr = new Address(address).toString('5D'); return ` * Origin: ${origin} (${addrStr})`; } function getTearLine() { const nodeVer = process.version.substr(1); // remove 'v' prefix return `--- ENiGMA 1/2 v${packageJson.version} (${os.platform()}; ${os.arch()}; ${nodeVer})`; } // // Return a FRL-1005.001 "Via" line // http://ftsc.org/docs/frl-1005.001 // function getVia(address) { /* FRL-1005.001 states teh following format: ^AVia: @YYYYMMDD.HHMMSS[.Precise][.Time Zone] [Serial Number] */ const addrStr = new Address(address).toString('5D'); const dateTime = moment().utc().format('YYYYMMDD.HHmmSS.SSSS.UTC'); const version = packageJson.version .replace(/\-/g, '.') .replace(/alpha/,'a') .replace(/beta/,'b'); return `${addrStr} @${dateTime} ENiGMA1/2 ${version}`; } function getAbbreviatedNetNodeList(netNodes) { let abbrList = ''; let currNet; netNodes.forEach(netNode => { if(currNet !== netNode.net) { abbrList += `${netNode.net}/`; currNet = netNode.net; } abbrList += `${netNode.node} `; }); return abbrList.trim(); // remove trailing space } function parseAbbreviatedNetNodeList(netNodes) { // // Make sure we have an array of objects. // Allow for a single object or string(s) // if(!_.isArray(netNodes)) { if(_.isString(netNodes)) { netNodes = netNodes.split(' '); } else { netNodes = [ netNodes ]; } } // // Convert any strings to parsed address objects // return netNodes.map(a => { if(_.isObject(a)) { return a; } else { return Address.fromString(a); } }); } // // Return a FTS-0004.001 SEEN-BY entry(s) that include // all pre-existing SEEN-BY entries with the addition // of |additions|. // // See http://ftsc.org/docs/fts-0004.001 // and notes at http://ftsc.org/docs/fsc-0043.002. // // For a great write up, see http://www.skepticfiles.org/aj/basics03.htm // // This method returns an sorted array of values, but // not the "SEEN-BY" prefix itself // function getUpdatedSeenByEntries(existingEntries, additions) { /* From FTS-0004: "There can be many seen-by lines at the end of Conference Mail messages, and they are the real "meat" of the control information. They are used to determine the systems to receive the exported messages. The format of the line is: SEEN-BY: 132/101 113 136/601 1014/1 The net/node numbers correspond to the net/node numbers of the systems having already received the message. In this way a message is never sent to a system twice. In a conference with many participants the number of seen-by lines can be very large. This line is added if it is not already a part of the message, or added to if it already exists, each time a message is exported to other systems. This is a REQUIRED field, and Conference Mail will not function correctly if this field is not put in place by other Echomail compatible programs." */ existingEntries = existingEntries || []; if(!_.isArray(existingEntries)) { existingEntries = [ existingEntries ]; } additions = parseAbbreviatedNetNodeList(additions).sort(Address.getComparator()); // // For now, we'll just append a new SEEN-BY entry // // :TODO: we should at least try and update what is already there in a smart way existingEntries.push(getAbbreviatedNetNodeList(additions)); return existingEntries; } function getUpdatedPathEntries(existingEntries, localAddress) { // :TODO: append to PATH in a smart way! We shoudl try to fit at least the last existing line existingEntries = existingEntries || []; if(!_.isArray(existingEntries)) { existingEntries = [ existingEntries ]; } existingEntries.push(getAbbreviatedNetNodeList( parseAbbreviatedNetNodeList(localAddress))); return existingEntries; } // // Return FTS-5000.001 "CHRS" value // http://ftsc.org/docs/fts-5003.001 // const ENCODING_TO_FTS_5003_001_CHARS = { // level 1 - generally should not be used ascii : [ 'ASCII', 1 ], 'us-ascii' : [ 'ASCII', 1 ], // level 2 - 8 bit, ASCII based cp437 : [ 'CP437', 2 ], cp850 : [ 'CP850', 2 ], // level 3 - reserved // level 4 utf8 : [ 'UTF-8', 4 ], 'utf-8' : [ 'UTF-8', 4 ], }; function getCharacterSetIdentifierByEncoding(encodingName) { const value = ENCODING_TO_FTS_5003_001_CHARS[encodingName.toLowerCase()]; return value ? `${value[0]} ${value[1]}` : encodingName.toUpperCase(); } function getEncodingFromCharacterSetIdentifier(chrs) { const ident = chrs.split(' ')[0].toUpperCase(); // :TODO: fill in the rest!!! return { // level 1 'ASCII' : 'iso-646-1', 'DUTCH' : 'iso-646', 'FINNISH' : 'iso-646-10', 'FRENCH' : 'iso-646', 'CANADIAN' : 'iso-646', 'GERMAN' : 'iso-646', 'ITALIAN' : 'iso-646', 'NORWEIG' : 'iso-646', 'PORTU' : 'iso-646', 'SPANISH' : 'iso-656', 'SWEDISH' : 'iso-646-10', 'SWISS' : 'iso-646', 'UK' : 'iso-646', 'ISO-10' : 'iso-646-10', // level 2 'CP437' : 'cp437', 'CP850' : 'cp850', 'CP852' : 'cp852', 'CP866' : 'cp866', 'CP848' : 'cp848', 'CP1250' : 'cp1250', 'CP1251' : 'cp1251', 'CP1252' : 'cp1252', 'CP10000' : 'macroman', 'LATIN-1' : 'iso-8859-1', 'LATIN-2' : 'iso-8859-2', 'LATIN-5' : 'iso-8859-9', 'LATIN-9' : 'iso-8859-15', // level 4 'UTF-8' : 'utf8', // deprecated stuff 'IBMPC' : 'cp1250', // :TODO: validate '+7_FIDO' : 'cp866', '+7' : 'cp866', 'MAC' : 'macroman', // :TODO: validate }[ident]; }