| 
									
										
										
										
											2016-09-22 13:11:54 -06:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports.create = function (opts) { | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  |   var machine; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!opts.onMessage && !opts.onmessage) { | 
					
						
							|  |  |  |     machine = new (require('events').EventEmitter)(); | 
					
						
							| 
									
										
										
										
											2016-09-30 00:51:40 -04:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     machine = {}; | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   machine.onMessage = opts.onmessage || opts.onMessage; | 
					
						
							|  |  |  |   machine.onmessage = opts.onmessage || opts.onMessage; | 
					
						
							|  |  |  |   machine.onError = opts.onerror || opts.onError; | 
					
						
							|  |  |  |   machine.onerror = opts.onerror || opts.onError; | 
					
						
							|  |  |  |   machine.onEnd = opts.onend || opts.onEnd; | 
					
						
							|  |  |  |   machine.onend = opts.onend || opts.onEnd; | 
					
						
							| 
									
										
										
										
											2016-09-22 13:11:54 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |   machine._version = 1; | 
					
						
							|  |  |  |   machine.state = 0; | 
					
						
							|  |  |  |   machine.states = { 0: 'version', 1: 'headerLength', 2: 'header', 3: 'data'/*, 4: 'error'*/ }; | 
					
						
							|  |  |  |   machine.states_length = Object.keys(machine.states).length; | 
					
						
							|  |  |  |   machine.chunkIndex = 0; | 
					
						
							|  |  |  |   machine.fns = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   machine.fns.version = function (chunk) { | 
					
						
							|  |  |  |     //console.log('');
 | 
					
						
							|  |  |  |     //console.log('[version]');
 | 
					
						
							|  |  |  |     if ((255 - machine._version) !== chunk[machine.chunkIndex]) { | 
					
						
							|  |  |  |       console.error("not v" + machine._version + " (or data is corrupt)"); | 
					
						
							|  |  |  |       // no idea how to fix this yet
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     machine.chunkIndex += 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   machine.headerLen = 0; | 
					
						
							|  |  |  |   machine.fns.headerLength = function (chunk) { | 
					
						
							|  |  |  |     //console.log('');
 | 
					
						
							|  |  |  |     //console.log('[headerLength]');
 | 
					
						
							|  |  |  |     machine.headerLen = chunk[machine.chunkIndex]; | 
					
						
							|  |  |  |     machine.chunkIndex += 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   machine.buf = null; | 
					
						
							|  |  |  |   machine.bufIndex = 0; | 
					
						
							|  |  |  |   //var buf = Buffer.alloc(4096);
 | 
					
						
							|  |  |  |   machine.fns.header = function (chunk) { | 
					
						
							|  |  |  |     //console.log('');
 | 
					
						
							|  |  |  |     //console.log('[header]');
 | 
					
						
							|  |  |  |     var curSize = machine.bufIndex + (chunk.length - machine.chunkIndex); | 
					
						
							|  |  |  |     var partLen = 0; | 
					
						
							|  |  |  |     var str = ''; | 
					
						
							|  |  |  |     var part; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (curSize < machine.headerLen) { | 
					
						
							|  |  |  |       // I still don't have the whole header,
 | 
					
						
							|  |  |  |       // so just create a large enough buffer,
 | 
					
						
							|  |  |  |       // write these bits, and wait for the next chunk.
 | 
					
						
							|  |  |  |       if (!machine.buf) { | 
					
						
							|  |  |  |         machine.buf = Buffer.alloc(machine.headerLen); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // partLen should be no more than the available size
 | 
					
						
							|  |  |  |       partLen = Math.min(machine.headerLen - machine.bufIndex, chunk.length - machine.chunkIndex); | 
					
						
							|  |  |  |       part = chunk.slice(machine.chunkIndex, machine.chunkIndex + partLen); | 
					
						
							|  |  |  |       chunk.copy(machine.buf, machine.bufIndex, machine.chunkIndex, machine.chunkIndex + partLen); | 
					
						
							|  |  |  |       machine.chunkIndex += partLen; // this MUST be chunk.length
 | 
					
						
							|  |  |  |       machine.bufIndex += partLen; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       // it's now ready to discover the whole header
 | 
					
						
							|  |  |  |       if (machine.buf) { | 
					
						
							|  |  |  |         str += machine.buf.slice(0, machine.bufIndex).toString(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       partLen = machine.headerLen - str.length; | 
					
						
							|  |  |  |       part = chunk.slice(machine.chunkIndex, machine.chunkIndex + partLen); | 
					
						
							|  |  |  |       str += part.toString(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       machine.chunkIndex += partLen; | 
					
						
							|  |  |  |       machine.buf = null; // back to null
 | 
					
						
							|  |  |  |       machine.bufIndex = 0; // back to 0
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       machine._headers = str.split(/,/g); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       machine.family = machine._headers[0]; | 
					
						
							|  |  |  |       machine.address = machine._headers[1]; | 
					
						
							|  |  |  |       machine.port = machine._headers[2]; | 
					
						
							|  |  |  |       machine.bodyLen = parseInt(machine._headers[3], 10) || -1; | 
					
						
							| 
									
										
										
										
											2016-09-27 14:02:57 -04:00
										 |  |  |       machine.service = machine._headers[4]; | 
					
						
							| 
									
										
										
										
											2016-09-30 03:03:26 -04:00
										 |  |  |       //console.log('machine.service', machine.service);
 | 
					
						
							| 
									
										
										
										
											2016-09-22 13:11:54 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   machine.fns.data = function (chunk) { | 
					
						
							|  |  |  |     //console.log('');
 | 
					
						
							|  |  |  |     //console.log('[data]');
 | 
					
						
							|  |  |  |     var curSize = machine.bufIndex + (chunk.length - machine.chunkIndex); | 
					
						
							|  |  |  |     //console.log('curSize:', curSize);
 | 
					
						
							|  |  |  |     //console.log('bodyLen:', machine.bodyLen, typeof machine.bodyLen);
 | 
					
						
							|  |  |  |     var partLen = 0; | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  |     var msg; | 
					
						
							| 
									
										
										
										
											2016-09-30 01:17:38 -04:00
										 |  |  |     var data; | 
					
						
							| 
									
										
										
										
											2016-09-22 13:11:54 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     partLen = Math.min(machine.bodyLen - machine.bufIndex, chunk.length - machine.chunkIndex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (curSize < machine.bodyLen) { | 
					
						
							|  |  |  |       //console.log('curSize < bodyLen');
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // I still don't have the whole header,
 | 
					
						
							|  |  |  |       // so just create a large enough buffer,
 | 
					
						
							|  |  |  |       // write these bits, and wait for the next chunk.
 | 
					
						
							|  |  |  |       if (!machine.buf) { | 
					
						
							|  |  |  |         machine.buf = Buffer.alloc(machine.bodyLen); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       chunk.copy(machine.buf, machine.bufIndex, machine.chunkIndex, machine.chunkIndex + partLen); | 
					
						
							|  |  |  |       machine.chunkIndex += partLen; // this MUST be chunk.length
 | 
					
						
							|  |  |  |       machine.bufIndex += partLen; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (machine.bufIndex > 0) { | 
					
						
							|  |  |  |       // the completing remainder of the body is in the current slice
 | 
					
						
							|  |  |  |       chunk.copy(machine.buf, machine.bufIndex, machine.chunkIndex, machine.chunkIndex + partLen); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       // the whole body is in the current slice
 | 
					
						
							|  |  |  |       machine.buf = chunk.slice(machine.chunkIndex, machine.chunkIndex + partLen); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     machine.bufIndex += partLen; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-30 03:03:26 -04:00
										 |  |  |     machine.service = machine.service; | 
					
						
							| 
									
										
										
										
											2016-09-30 02:44:36 -04:00
										 |  |  |     data = machine.buf.slice(0, machine.bufIndex); | 
					
						
							| 
									
										
										
										
											2016-09-30 03:03:26 -04:00
										 |  |  |     //console.log('machine.service', machine.service);
 | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     //
 | 
					
						
							|  |  |  |     // data, end, error
 | 
					
						
							|  |  |  |     //
 | 
					
						
							| 
									
										
										
										
											2016-09-30 03:03:26 -04:00
										 |  |  |     if ('end' === machine.service) { | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  |       msg = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       msg.family = machine.family; | 
					
						
							|  |  |  |       msg.address = machine.address; | 
					
						
							|  |  |  |       msg.port = machine.port; | 
					
						
							|  |  |  |       msg.service = 'end'; | 
					
						
							| 
									
										
										
										
											2016-09-30 01:17:38 -04:00
										 |  |  |       msg.data = data; | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (machine.emit) { | 
					
						
							|  |  |  |         machine.emit('tunnelEnd', msg); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         (machine.onend||machine.onmessage)(msg); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-30 03:03:26 -04:00
										 |  |  |     else if ('error' === machine.service) { | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  |       try { | 
					
						
							|  |  |  |         msg = JSON.parse(machine.data.toString()); | 
					
						
							|  |  |  |       } catch(e) { | 
					
						
							|  |  |  |         msg = new Error('unknown error'); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       msg.family = machine.family; | 
					
						
							|  |  |  |       msg.address = machine.address; | 
					
						
							|  |  |  |       msg.port = machine.port; | 
					
						
							|  |  |  |       msg.service = 'error'; | 
					
						
							| 
									
										
										
										
											2016-09-30 01:17:38 -04:00
										 |  |  |       msg.data = data; | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (machine.emit) { | 
					
						
							|  |  |  |         machine.emit('tunnelError', msg); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							| 
									
										
										
										
											2016-09-30 01:03:06 -04:00
										 |  |  |         (machine.onerror||machine.onmessage)(msg); | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2016-09-30 01:17:38 -04:00
										 |  |  |       msg = {}; | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |       msg.family = machine.family; | 
					
						
							|  |  |  |       msg.address = machine.address; | 
					
						
							|  |  |  |       msg.port = machine.port; | 
					
						
							|  |  |  |       msg.service = machine.service; | 
					
						
							| 
									
										
										
										
											2016-09-30 01:17:38 -04:00
										 |  |  |       msg.data = data; | 
					
						
							| 
									
										
										
										
											2016-09-30 00:25:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (machine.emit) { | 
					
						
							|  |  |  |         machine.emit('tunnelData', msg); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         machine.onmessage(msg); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-22 13:11:54 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     machine.chunkIndex += partLen;  // === chunk.length
 | 
					
						
							|  |  |  |     machine.buf = null;             // reset to null
 | 
					
						
							|  |  |  |     machine.bufIndex = 0;           // reset to 0
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  |   machine.fns.addChunk = function (chunk) { | 
					
						
							|  |  |  |     //console.log('');
 | 
					
						
							|  |  |  |     //console.log('[addChunk]');
 | 
					
						
							|  |  |  |     machine.chunkIndex = 0; | 
					
						
							|  |  |  |     while (machine.chunkIndex < chunk.length) { | 
					
						
							|  |  |  |       //console.log('chunkIndex:', machine.chunkIndex, 'state:', machine.state);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (true === machine.fns[machine.states[machine.state]](chunk)) { | 
					
						
							|  |  |  |         machine.state += 1; | 
					
						
							|  |  |  |         machine.state %= machine.states_length; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return machine; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-30 03:03:26 -04:00
										 |  |  | module.exports.pack = function (address, data, service) { | 
					
						
							| 
									
										
										
										
											2016-09-30 02:44:36 -04:00
										 |  |  |   data = data || Buffer.alloc(1); | 
					
						
							|  |  |  |   if (!data.byteLength) { | 
					
						
							|  |  |  |     data = Buffer.alloc(1); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-09-30 03:03:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if ('error' === service) { | 
					
						
							|  |  |  |     address.service = 'error'; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if ('end' === service) { | 
					
						
							|  |  |  |     address.service = 'end'; | 
					
						
							| 
									
										
										
										
											2016-09-29 23:41:42 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-22 13:11:54 -06:00
										 |  |  |   var version = 1; | 
					
						
							| 
									
										
										
										
											2016-09-27 14:02:57 -04:00
										 |  |  |   var header = Buffer.from([ | 
					
						
							|  |  |  |     /*servername,*/ address.family, address.address, address.port, data.byteLength | 
					
						
							| 
									
										
										
										
											2016-09-30 03:03:26 -04:00
										 |  |  |   , (address.service || '') | 
					
						
							| 
									
										
										
										
											2016-09-27 14:02:57 -04:00
										 |  |  |   ].join(',')); | 
					
						
							| 
									
										
										
										
											2016-09-22 13:31:31 -06:00
										 |  |  |   var meta = Buffer.from([ 255 - version, header.length ]); | 
					
						
							|  |  |  |   var buf = Buffer.alloc(meta.byteLength + header.byteLength + data.byteLength); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   meta.copy(buf, 0, 0, meta.byteLength); | 
					
						
							|  |  |  |   header.copy(buf, 2, 0, header.byteLength); | 
					
						
							|  |  |  |   data.copy(buf, 2 + header.byteLength, 0, data.byteLength); | 
					
						
							| 
									
										
										
										
											2016-09-22 13:11:54 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return buf; | 
					
						
							|  |  |  | }; |