UNPKG

pws_gsm

Version:

gsm modem api for sms and gprs

387 lines (304 loc) 11.3 kB
var pduParser = {}; pduParser.parse = function (pdu) { //Cursor points to the last octet we've read. var cursor = 0; var buffer = new Buffer(pdu.slice(0, 4), 'hex'); var smscSize = buffer[0]; var smscType = buffer[1].toString(16); cursor = (smscSize * 2 + 2); var smscNum = pduParser.deSwapNibbles(pdu.slice(4, cursor)); var buffer = new Buffer(pdu.slice(cursor, cursor + 6), 'hex'); cursor += 6; var smsDeliver = buffer[0]; var smsDeliverBits = ("00000000" + parseInt(smsDeliver).toString(2)).slice(-8); var udhi = smsDeliverBits.slice(1, 2) === "1"; var senderSize = buffer[1]; if (senderSize % 2 === 1) senderSize++; var senderType = parseInt(buffer[2]).toString(16) var encodedSender = pdu.slice(cursor, cursor + senderSize); var senderNum; if (senderType === '91') { senderNum = pduParser.deSwapNibbles(encodedSender); } else if (senderType === 'd0') { senderNum = this.decode7Bit(encodedSender).replace(/\0/g, ''); } else { console.error('unsupported sender type.'); } cursor += senderSize; var protocolIdentifier = pdu.slice(cursor, cursor + 2); cursor += 2; var dataCodingScheme = pdu.slice(cursor, cursor + 2); cursor = cursor + 2; var encoding = pduParser.detectEncoding(dataCodingScheme); var timestamp = pduParser.deSwapNibbles(pdu.slice(cursor, cursor + 14)); var time = new Date; time.setUTCFullYear('20' + timestamp.slice(0, 2)); time.setUTCMonth(timestamp.slice(2, 4) - 1); time.setUTCDate(timestamp.slice(4, 6)); time.setUTCHours(timestamp.slice(6, 8)); time.setUTCMinutes(timestamp.slice(8, 10)); time.setUTCSeconds(timestamp.slice(10, 12)); var firstTimezoneOctet = parseInt(timestamp.slice(12, 13)); var binary = ("0000" + firstTimezoneOctet.toString(2)).slice(-4); var factor = binary.slice(0, 1) === '1' ? 1 : -1; var binary = '0' + binary.slice(1, 4); var firstTimezoneOctet = parseInt(binary, 2).toString(10); var timezoneDiff = parseInt(firstTimezoneOctet + timestamp.slice(13, 14)); var time = new Date(time.getTime() + (timezoneDiff * 15 * 1000 * 60 * factor)); cursor += 14; var dataLength = parseInt(pdu.slice(cursor, cursor + 2), 16).toString(10); cursor += 2; if (udhi) { //User-Data-Header-Indicator: means there's some User-Data-Header. var udhLength = pdu.slice(cursor, cursor + 2); var iei = pdu.slice(cursor + 2, cursor + 4); if (iei == "00") { //Concatenated sms. var headerLength = pdu.slice(cursor + 4, cursor + 6); var referenceNumber = pdu.slice(cursor + 6, cursor + 8); var parts = pdu.slice(cursor + 8, cursor + 10); var currentPart = pdu.slice(cursor + 10, cursor + 12); } if (iei == "08") { //Concatenaded sms with a two-bytes reference number var headerLength = pdu.slice(cursor + 4, cursor + 6); var referenceNumber = pdu.slice(cursor + 6, cursor + 10); var parts = pdu.slice(cursor + 10, cursor + 12); var currentPart = pdu.slice(cursor + 12, cursor + 14); } if (encoding === '16bit') if (iei == '00') cursor += (udhLength - 2) * 4; else if (iei == '08') cursor += ((udhLength - 2) * 4) + 2; else cursor += (udhLength - 2) * 2; } if (encoding === '16bit') var text = pduParser.decode16Bit(pdu.slice(cursor), dataLength); else if (encoding === '7bit') var text = pduParser.decode7Bit(pdu.slice(cursor), dataLength); else if (encoding === '8bit') var text = ''; //TODO var data = { 'smsc': smscNum, 'smsc_type': smscType, 'sender': senderNum, 'sender_type': senderType, 'encoding': encoding, 'time': time, 'text': text }; if (udhi) { data['udh'] = { 'length': udhLength, 'iei': iei, }; if (iei == '00' || iei == '08') { data['udh']['reference_number'] = referenceNumber; data['udh']['parts'] = parseInt(parts); data['udh']['current_part'] = parseInt(currentPart); } } return data; } pduParser.detectEncoding = function (dataCodingScheme) { var binary = ('00000000' + (parseInt(dataCodingScheme, 16).toString(2))).slice(-8); if (binary == '00000000') return '7bit'; if (binary.slice(0, 2) === '00') { var compressed = binary.slice(2, 1) === '1'; var bitsHaveMeaning = binary.slice(3, 1) === '1'; if (binary.slice(4, 6) === '00') return '7bit'; if (binary.slice(4, 6) === '01') return '8bit'; if (binary.slice(4, 6) === '10') return '16bit'; } } pduParser.decode16Bit = function (data, length) { //We are getting ucs2 characters. var ucs2 = ''; for (var i = 0; i <= data.length; i = i + 4) { ucs2 += String.fromCharCode("0x" + data[i] + data[i + 1] + data[i + 2] + data[i + 3]); } return ucs2; } pduParser.deSwapNibbles = function (nibbles) { var out = ''; for (var i = 0; i < nibbles.length; i = i + 2) { if (nibbles[i] === 'F') //Dont consider trailing F. out += parseInt(nibbles[i + 1], 16).toString(10); else out += parseInt(nibbles[i + 1], 16).toString(10) + parseInt(nibbles[i], 16).toString(10); } return out; } pduParser.decode7Bit = function (code, count) { //We are getting 'septeps'. We should decode them. var binary = ''; for (var i = 0; i < code.length; i++) binary += ('0000' + parseInt(code.slice(i, i + 1), 16).toString(2)).slice(-4); var bin = Array(); var cursor = 0; var fromPrevious = ''; var i = 0; while (binary[i]) { var remaining = 7 - fromPrevious.length; var toNext = 8 - remaining; bin[i] = binary.slice(cursor + toNext, cursor + toNext + remaining) + fromPrevious; var fromPrevious = binary.slice(cursor, cursor + toNext); if (toNext === 8) fromPrevious = ''; else cursor += 8; i++; } var ascii = ''; for (i in bin) ascii += String.fromCharCode(parseInt(bin[i], 2)); return ascii; } pduParser.encode7Bit = function (ascii) { //We should create septeps now. var octets = new Array(); for (var i = 0; i < ascii.length; i++) octets.push(('0000000' + (ascii.charCodeAt(i).toString(2))).slice(-7)); for (var i in octets) { var i = parseInt(i); var freeSpace = 8 - octets[i].length; if (octets[i + 1] && freeSpace !== 8) { octets[i] = octets[i + 1].slice(7 - freeSpace) + octets[i]; octets[i + 1] = octets[i + 1].slice(0, 7 - freeSpace); } } var hex = ''; for (i in octets) if (octets[i].length > 0) hex += ('00' + (parseInt(octets[i], 2).toString(16))).slice(-2); return hex; } //TODO: TP-Validity-Period (Delivery) pduParser.generate = function (message) { var pdu = '00'; var parts = 1; if (message.encoding === '16bit' && message.text.length > 70) parts = message.text.length / 66; else if (message.encoding === '7bit' && message.text.length > 160) parts = message.text.length / 153; parts = Math.ceil(parts); TPMTI = 1; TPRD = 4; TPVPF = 8; TPSRR = 32; TPUDHI = 64; TPRP = 128; var submit = TPMTI; if (parts > 1) //UDHI submit = submit | TPUDHI; submit = submit | TPSRR; pdu += submit.toString(16); pdu += '00'; //TODO: Reference Number; var receiverSize = ('00' + (parseInt(message.receiver.length, 10).toString(16))).slice(-2); var receiver = pduParser.swapNibbles(message.receiver); var receiverType = 81; //TODO: NOT-Hardcoded PDU generation. Please note that Hamrah1 doesnt work if we set it to 91 (International). pdu += receiverSize.toString(16) + receiverType + receiver; pdu += '00'; //TODO TP-PID if (message.encoding === '16bit') pdu += '08'; else if (message.encoding === '7bit') pdu += '00'; var pdus = new Array(); var csms = randomHexa(2); // CSMS allows to give a reference to a concatenated message for (var i = 0; i < parts; i++) { pdus[i] = pdu; if (message.encoding === '16bit') { /* If there are more than one messages to be sent, we are going to have to put some UDH. Then, we would have space only * for 66 UCS2 characters instead of 70 */ if (parts === 1) var length = 70; else var length = 66; } else if (message.encoding === '7bit') { /* If there are more than one messages to be sent, we are going to have to put some UDH. Then, we would have space only * for 153 ASCII characters instead of 160 */ if (parts === 1) var length = 160; else var length = 153; } var text = message.text.slice(i * length, (i * length) + length); if (message.encoding === '16bit') { user_data = pduParser.encode16Bit(text); var size = (user_data.length / 2); if (parts > 1) size += 6; //6 is the number of data headers we append. } else if (message.encoding === '7bit') { user_data = pduParser.encode7Bit(text); var size = user_data.length / 2; } pdus[i] += ('00' + parseInt(size).toString(16)).slice(-2); if (parts > 1) { pdus[i] += '05'; pdus[i] += '00'; pdus[i] += '03'; pdus[i] += csms; pdus[i] += ('00' + parts.toString(16)).slice(-2); pdus[i] += ('00' + (i + 1).toString(16)).slice(-2); } pdus[i] += user_data; } return pdus; } pduParser.encode16Bit = function (text) { var out = ''; for (var i = 0; i < text.length; i++) { out += ('0000' + (parseInt(text.charCodeAt(i), 10).toString(16))).slice(-4); } return out; } pduParser.swapNibbles = function (nibbles) { var out = ''; for (var i = 0; i < nibbles.length; i = i + 2) { if (typeof (nibbles[i + 1]) === 'undefined') // Add a trailing F. out += 'F' + parseInt(nibbles[i], 16).toString(10); else out += parseInt(nibbles[i + 1], 16).toString(10) + parseInt(nibbles[i], 16).toString(10); } return out; } pduParser.parseStatusReport = function (pdu) { //Cursor points to the last octet we've read. var cursor = 0; var smscSize = parseInt(pdu.slice(0, 2), 16); cursor += 2; var smscType = parseInt(pdu.slice(cursor, cursor + 2), 16); cursor += 2; var smscNum = pduParser.deSwapNibbles(pdu.slice(cursor, (smscSize * 2) + 2)); cursor = (smscSize * 2 + 2); var header = parseInt(pdu.slice(cursor, cursor + 2)); cursor += 2; var reference = parseInt(pdu.slice(cursor, cursor + 2), 16); cursor += 2; var senderSize = parseInt(pdu.slice(cursor, cursor + 2), 16); if (senderSize % 2 === 1) senderSize++; cursor += 2; var senderType = parseInt(pdu.slice(cursor, cursor + 2)); cursor += 2; var sender = pduParser.deSwapNibbles(pdu.slice(cursor, cursor + senderSize)); var status = pdu.slice(-2); return { smsc: smscNum, reference: reference, sender: sender, status: status } } function randomHexa(size) { var text = ""; var possible = "0123456789ABCDEF"; for (var i = 0; i < size; i++) text += possible.charAt(Math.floor(Math.random() * possible.length)); return text; } module.exports = pduParser;