diff --git a/examples/connector.js b/examples/connector.js index 84211cd..096f2df 100644 --- a/examples/connector.js +++ b/examples/connector.js @@ -3,6 +3,7 @@ var net = require('net'); var mtrude = require('mtrude'); var rtmp = mtrude.rtmp; +var AMF = rtmp.AMF; var ChunkStream = rtmp.ChunkStream; var MessageStream = rtmp.MessageStream; var Application = rtmp.Application; @@ -21,11 +22,13 @@ function workWithChunkStream(chunkStream) { function main() { var optimist = require('optimist') - .usage('Usage: $0 [--debug] [in [out]]') - .boolean('debug') + .usage('Usage: $0 [--debugamf] [--debugmessage] [in [out]]') + .boolean('debugamf') + .boolean('debugmessage') .boolean('help') .alias('h', 'help') - .describe('debug', 'Set MessageStream.DBG = true') + .describe('debugamf', 'Set AMF.DBG = true') + .describe('debugmessage', 'Set MessageStream.DBG = true') .describe('nocolor', 'Don\'t use colors') ; var argv = optimist.argv; @@ -35,7 +38,8 @@ function main() { return; } - if (argv.debug) MessageStream.DBG = true; + MessageStream.DBG = !!argv.debugmessage; + AMF.DBG = !!argv.debugamf; if (argv._.length == 0) { var server = net.createServer(); diff --git a/examples/dumpAmf.js b/examples/dumpAmf.js index 1ec0a9d..4ab1f3c 100644 --- a/examples/dumpAmf.js +++ b/examples/dumpAmf.js @@ -35,7 +35,7 @@ function main() { return; } - if (argv.nocolor) dumpTools.dontColor(); + if (argv.nocolor) utils.dontColor(); if (argv.debug) MessageStream.DBG = true; diff --git a/examples/dumpMessages.js b/examples/dumpMessages.js index 0b9aca2..5f22708 100644 --- a/examples/dumpMessages.js +++ b/examples/dumpMessages.js @@ -28,7 +28,10 @@ function main() { return; } - if (argv.nocolor) dumpTools.dontColor(); + if (argv.nocolor) utils.dontColor(); + + // todo PING shoulg also dump ti (it's always the same but it is nice to show + // that). console.log('Dump format:\n' + 'MSG : %s %s %s %s %s:%s %s\n' @@ -66,10 +69,10 @@ function main() { dumpMessageStream(messageStream); } -var dump8 = dumpTools.dump8; -var hex2 = dumpTools.hex2; -var hex6 = dumpTools.hex6; -var ascii8 = dumpTools.ascii8; +var dump8 = utils.dump8; +var hex2 = utils.hex2; +var hex6 = utils.hex6; +var ascii8 = utils.ascii8; function dumpMessageStream(messageStream) { messageStream.on('error', function(errorMessage) { diff --git a/lib/rtmp/AMF.js b/lib/rtmp/AMF.js index a8c9ea3..77780f5 100644 --- a/lib/rtmp/AMF.js +++ b/lib/rtmp/AMF.js @@ -2,10 +2,13 @@ var util = require('util'); var assert = require('assert'); +var utils = require('../utils'); var BufferChain = require('../BufferChain'); var AMF = module.exports = {}; +AMF.DBG = false; + AMF.deserializeZ = function(data, emitter) { if (data.ptr == null) data.ptr = 0; @@ -22,7 +25,7 @@ AMF.deserializeZ = function(data, emitter) { return values; } -AMF.amfZMarkers = { +var a0 = AMF.amfZMarkers = { NUMBER: 0x00, BOOLEAN: 0x01, STRING: 0x02, @@ -41,9 +44,7 @@ AMF.amfZMarkers = { XML_DOCUMENT: 0x0f, TYPED_OBJECT: 0x10, AVMPLUS: 0x11, // switch to AMF3 -} - -var a0 = AMF.amfZMarkers; +}; function amfZAny(data) { var marker = data.readUInt8(data.ptr++); @@ -77,7 +78,7 @@ function amfZObject(data) { var key = amfZString(data); if (key == '') { var marker = data.readUInt8(data.ptr++); - if (marker != AMF.amfZMarkers.OBJECT_END) throw new Error( + if (marker != a0.OBJECT_END) throw new Error( 'Bad end-of-object marker 0x%s != 0x9', marker.toString(16)); break; } @@ -111,12 +112,16 @@ AMF.serializeZ = function() { var serBuffer = { buffer: new Buffer(200), ptr: 0 }; for (var i = 0; i < arguments.length; i++) serZAny(serBuffer, arguments[i]); + var buffer = serBuffer.buffer.slice(0, serBuffer.ptr); - dbg(hexdump(serBuffer.buffer)); + dbg(utils.hexDump(buffer)); - return serBuffer.buffer; + return buffer; } +function dbg() { if (AMF.DBG) console.log.apply(console, arguments) } + + function serZAny(serBuffer, value) { dbgSerBuffer('Any', serBuffer); if (value === null) return serZNull(serBuffer); @@ -124,7 +129,7 @@ function serZAny(serBuffer, value) { if (value === true) return serZTrue(serBuffer); if (value === false) return serZFalse(serBuffer); if (typeof value == 'number') return serZNumber(serBuffer, value); - if (typeof value == 'string') return serZString(serBuffer, value); + if (typeof value == 'string') return serZString(serBuffer, value, true); if (Array.isArray(value)) return serZArray(serBuffer, value); if (typeof value == 'object') return serZObject(serBuffer, value); throw new Error('value type not supported'); @@ -138,7 +143,7 @@ function dbgSerBuffer(text, serBuffer, value) { return s; } var at = ' @ ' + serBuffer.ptr + ' / ' + serBuffer.buffer.length; - console.log('serZ' + text + spaces(20 - text.length) + dbg('serZ' + text + spaces(20 - text.length) + at + spaces(16 - at.length) + (value == null ? '' : util.inspect(value))); } @@ -173,8 +178,8 @@ function serZNumber(serBuffer, value) { dbgSerBuffer('Number', serBuffer, value); maybeExpand(serBuffer, 9); - serBuffer.buffer.writeUInt8(a0.NUMBER, serBuffer.ptr); - serBuffer.buffer.writeDoubleBE(value, serBuffer.ptr + 1); + serBuffer.buffer.writeUInt8(a0.NUMBER, serBuffer.ptr++); + serBuffer.buffer.writeDoubleBE(value, serBuffer.ptr); serBuffer.ptr += 8; } @@ -204,6 +209,9 @@ function serZArray(serBuffer, value) { function serZObject(serBuffer, value) { dbgSerBuffer('Object', serBuffer, {keys: Object.keys(value).join(' ')}); + maybeExpand(serBuffer, 1); + serBuffer.buffer.writeUInt8(a0.OBJECT, serBuffer.ptr++); + for (var key in value) { serZString(serBuffer, key, false); serZAny(serBuffer, value[key]); @@ -240,7 +248,7 @@ AMF.deserialize3 = function(data, emitter) { } -AMF.amf3Markers = { +var a3 = AMF.amf3Markers = { UNDEFINED: 0x00, NULL: 0x01, FALSE: 0x02, @@ -257,7 +265,6 @@ AMF.amf3Markers = { }; function amf3Any(data) { - var a3 = AMF.amf3Markers; var marker = data.readUInt8(data.ptr++); var ptr = data.ptr; @@ -345,6 +352,8 @@ function amf3Object(data) { } } +Object.seal(AMF); + diff --git a/lib/rtmp/Application.js b/lib/rtmp/Application.js index 5ddb19a..20eb75c 100644 --- a/lib/rtmp/Application.js +++ b/lib/rtmp/Application.js @@ -31,12 +31,10 @@ p.handleMessage = function(message) { if (command[0] == 'connect') { var ok = this.connect ? this.connect(message) : true; if (ok) this.messageStream.send(types.INVOKE, message.msid, message.csid, - new Date().getTime(), AMF.serializeZ(['_result', 1, + new Date().getTime(), AMF.serializeZ('_result', 1, { fmsVer: 'FMS/3,0,1,123', capabilities: 31 }, { level: 'status', code: 'NetConnection.Connect.Success', - description: 'Connection succeeded', objectEncoding: 3 } - ]) - ); + description: 'Connection succeeded', objectEncoding: 3 })); return; } } diff --git a/lib/utils.js b/lib/utils.js index 8296be4..55502b2 100644 --- a/lib/utils.js +++ b/lib/utils.js @@ -1,5 +1,6 @@ "use strict"; +require('colors'); var fs = require('fs'); var asSocket = exports.asSocket = function(incoming, outgoing, icb, ocb) { @@ -40,21 +41,52 @@ var dump8 = exports.dump8 = function(data) { return b(0) + b(1) + b(2) + b(3) + ' ' + b(4) + b(5) + b(6) + b(7); } -var hex2 = exports.hex2 = function(byte) { - function hex1(nybble) { return "0123456789abcdef"[nybble & 0xf]; } - return hex1(byte >> 4) + hex1(byte); +var hex1 = exports.hex1 = function(nybble) { + return "0123456789abcdef"[nybble & 0xf]; +} + +var hex2 = exports.hex2 = function(int8) { + return hex1(int8 >> 4) + hex1(int8); +} + +var hex3 = exports.hex3 = function(int12) { + return hex2(int12 >> 4) + hex1(int12); } var hex6 = exports.hex6 = function(int24) { return hex2(int24 >> 16) + hex2(int24 >> 8) + hex2(int24); } -var ascii8 = exports.ascii8 = function(data) { +var ascii8 = exports.ascii8 = function(data, ptr) { + if (ptr == null) ptr = 0; function asChar(b) { return String.fromCharCode(b); } - function ascii(b) { return b > 31 && b < 128 ? asChar(b).yellow : '·'.black; } - function b(i) { return i < data.length ? ascii(data.readUInt8(i)) : '·'.white; } + function ascii(b) { + return b > 31 && b < 128 ? asChar(b).yellow : '·'.white; + } + function b(i) { + return i + ptr < data.length ? ascii(data.readUInt8(i + ptr)) : ' '; + } - return b(0) + b(1) + b(2) + b(3) + ' ' + b(4) + b(5) + b(6) + b(7); + return b(0) + b(1) + b(2) + b(3) + b(4) + b(5) + b(6) + b(7); +} + +var hexDump = exports.hexDump = function(data, indent, max) { + function b(i) { return i < data.length ? hex2(data.readUInt8(i)) : ' '; } + + indent = indent || ''; + max = max || 4096; + var length = Math.min(max, data.length); + var s = ''; + for (var i = 0; i < length; i += 16) { + s += indent + hex6(i) + ' '; + for (var j = 0; j < 16; j += 4) { + s += (b(i + j) + ' ' + b(i + j + 1) + ' ' + + b(i + j + 2) + ' ' + b(i + j + 3)).blue + ' '; + } + s += '|' + ascii8(data, i) + ascii8(data, i + 8) + '|\n'; + } + + return s; } var dontColor = exports.dontColor = function() {