2014-06-20 19:34:14 +04:00
|
|
|
/*!
|
2017-08-11 20:18:28 +02:00
|
|
|
* Webogram v0.6.0 - messaging web application for MTProto
|
2014-06-20 19:34:14 +04:00
|
|
|
* https://github.com/zhukov/webogram
|
|
|
|
* Copyright (C) 2014 Igor Zhukov <igor.beatle@gmail.com>
|
|
|
|
* https://github.com/zhukov/webogram/blob/master/LICENSE
|
|
|
|
*/
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function TLSerialization (options) {
|
|
|
|
options = options || {}
|
|
|
|
this.maxLength = options.startMaxLength || 2048 // 2Kb
|
|
|
|
this.offset = 0 // in bytes
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.createBuffer()
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
// this.debug = options.debug !== undefined ? options.debug : Config.Modes.debug
|
|
|
|
this.mtproto = options.mtproto || false
|
|
|
|
return this
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
TLSerialization.prototype.createBuffer = function () {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.buffer = new ArrayBuffer(this.maxLength)
|
|
|
|
this.intView = new Int32Array(this.buffer)
|
|
|
|
this.byteView = new Uint8Array(this.buffer)
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.getArray = function () {
|
2016-06-28 19:40:53 +03:00
|
|
|
var resultBuffer = new ArrayBuffer(this.offset)
|
|
|
|
var resultArray = new Int32Array(resultBuffer)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
resultArray.set(this.intView.subarray(0, this.offset / 4))
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return resultArray
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.getBuffer = function () {
|
2016-06-28 19:40:53 +03:00
|
|
|
return this.getArray().buffer
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2014-10-23 13:33:46 +04:00
|
|
|
TLSerialization.prototype.getBytes = function (typed) {
|
|
|
|
if (typed) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var resultBuffer = new ArrayBuffer(this.offset)
|
|
|
|
var resultArray = new Uint8Array(resultBuffer)
|
2014-10-23 13:33:46 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
resultArray.set(this.byteView.subarray(0, this.offset))
|
2014-10-23 13:33:46 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return resultArray
|
2014-10-23 13:33:46 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var bytes = []
|
2014-06-20 19:34:14 +04:00
|
|
|
for (var i = 0; i < this.offset; i++) {
|
2016-06-28 19:40:53 +03:00
|
|
|
bytes.push(this.byteView[i])
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
return bytes
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.checkLength = function (needBytes) {
|
|
|
|
if (this.offset + needBytes < this.maxLength) {
|
2016-06-28 19:40:53 +03:00
|
|
|
return
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
console.trace('Increase buffer', this.offset, needBytes, this.maxLength)
|
|
|
|
this.maxLength = Math.ceil(Math.max(this.maxLength * 2, this.offset + needBytes + 16) / 4) * 4
|
|
|
|
var previousBuffer = this.buffer
|
|
|
|
var previousArray = new Int32Array(previousBuffer)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.createBuffer()
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
new Int32Array(this.buffer).set(previousArray)
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.writeInt = function (i, field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.debug && console.log('>>>', i.toString(16), i, field)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.checkLength(4)
|
|
|
|
this.intView[this.offset / 4] = i
|
|
|
|
this.offset += 4
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.storeInt = function (i, field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.writeInt(i, (field || '') + ':int')
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.storeBool = function (i, field) {
|
|
|
|
if (i) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.writeInt(0x997275b5, (field || '') + ':bool')
|
2014-06-20 19:34:14 +04:00
|
|
|
} else {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.writeInt(0xbc799737, (field || '') + ':bool')
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.storeLongP = function (iHigh, iLow, field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.writeInt(iLow, (field || '') + ':long[low]')
|
|
|
|
this.writeInt(iHigh, (field || '') + ':long[high]')
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.storeLong = function (sLong, field) {
|
|
|
|
if (angular.isArray(sLong)) {
|
|
|
|
if (sLong.length == 2) {
|
2016-06-28 19:40:53 +03:00
|
|
|
return this.storeLongP(sLong[0], sLong[1], field)
|
2014-06-20 19:34:14 +04:00
|
|
|
} else {
|
2016-06-28 19:40:53 +03:00
|
|
|
return this.storeIntBytes(sLong, 64, field)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-30 00:36:42 +03:00
|
|
|
if (typeof sLong != 'string') {
|
2016-06-28 19:40:53 +03:00
|
|
|
sLong = sLong ? sLong.toString() : '0'
|
2015-06-30 00:36:42 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
var divRem = bigStringInt(sLong).divideAndRemainder(bigint(0x100000000))
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.writeInt(intToUint(divRem[1].intValue()), (field || '') + ':long[low]')
|
|
|
|
this.writeInt(intToUint(divRem[0].intValue()), (field || '') + ':long[high]')
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-04-18 14:17:51 +03:00
|
|
|
TLSerialization.prototype.storeDouble = function (f, field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var buffer = new ArrayBuffer(8)
|
|
|
|
var intView = new Int32Array(buffer)
|
|
|
|
var doubleView = new Float64Array(buffer)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
doubleView[0] = f
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.writeInt(intView[0], (field || '') + ':double[low]')
|
|
|
|
this.writeInt(intView[1], (field || '') + ':double[high]')
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.storeString = function (s, field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.debug && console.log('>>>', s, (field || '') + ':string')
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2015-04-07 23:46:41 +03:00
|
|
|
if (s === undefined) {
|
2016-06-28 19:40:53 +03:00
|
|
|
s = ''
|
2015-04-07 23:46:41 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
var sUTF8 = unescape(encodeURIComponent(s))
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.checkLength(sUTF8.length + 8)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var len = sUTF8.length
|
2014-06-20 19:34:14 +04:00
|
|
|
if (len <= 253) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.byteView[this.offset++] = len
|
2014-06-20 19:34:14 +04:00
|
|
|
} else {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.byteView[this.offset++] = 254
|
|
|
|
this.byteView[this.offset++] = len & 0xFF
|
|
|
|
this.byteView[this.offset++] = (len & 0xFF00) >> 8
|
|
|
|
this.byteView[this.offset++] = (len & 0xFF0000) >> 16
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
for (var i = 0; i < len; i++) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.byteView[this.offset++] = sUTF8.charCodeAt(i)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Padding
|
|
|
|
while (this.offset % 4) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.byteView[this.offset++] = 0
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TLSerialization.prototype.storeBytes = function (bytes, field) {
|
2014-10-30 10:36:41 +03:00
|
|
|
if (bytes instanceof ArrayBuffer) {
|
2016-06-28 19:40:53 +03:00
|
|
|
bytes = new Uint8Array(bytes)
|
2014-10-30 10:36:41 +03:00
|
|
|
}
|
2015-04-07 23:46:41 +03:00
|
|
|
else if (bytes === undefined) {
|
2016-06-28 19:40:53 +03:00
|
|
|
bytes = []
|
2015-04-07 23:46:41 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
this.debug && console.log('>>>', bytesToHex(bytes), (field || '') + ':bytes')
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var len = bytes.byteLength || bytes.length
|
|
|
|
this.checkLength(len + 8)
|
2014-06-20 19:34:14 +04:00
|
|
|
if (len <= 253) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.byteView[this.offset++] = len
|
2014-06-20 19:34:14 +04:00
|
|
|
} else {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.byteView[this.offset++] = 254
|
|
|
|
this.byteView[this.offset++] = len & 0xFF
|
|
|
|
this.byteView[this.offset++] = (len & 0xFF00) >> 8
|
|
|
|
this.byteView[this.offset++] = (len & 0xFF0000) >> 16
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2014-10-30 10:36:41 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.byteView.set(bytes, this.offset)
|
|
|
|
this.offset += len
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
// Padding
|
|
|
|
while (this.offset % 4) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.byteView[this.offset++] = 0
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TLSerialization.prototype.storeIntBytes = function (bytes, bits, field) {
|
2014-10-24 22:37:43 +04:00
|
|
|
if (bytes instanceof ArrayBuffer) {
|
2016-06-28 19:40:53 +03:00
|
|
|
bytes = new Uint8Array(bytes)
|
2014-10-24 22:37:43 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
var len = bytes.length
|
2014-06-20 19:34:14 +04:00
|
|
|
if ((bits % 32) || (len * 8) != bits) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Invalid bits: ' + bits + ', ' + bytes.length)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.debug && console.log('>>>', bytesToHex(bytes), (field || '') + ':int' + bits)
|
|
|
|
this.checkLength(len)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.byteView.set(bytes, this.offset)
|
|
|
|
this.offset += len
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.storeRawBytes = function (bytes, field) {
|
2014-10-24 22:37:43 +04:00
|
|
|
if (bytes instanceof ArrayBuffer) {
|
2016-06-28 19:40:53 +03:00
|
|
|
bytes = new Uint8Array(bytes)
|
2014-10-24 22:37:43 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
var len = bytes.length
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.debug && console.log('>>>', bytesToHex(bytes), (field || ''))
|
|
|
|
this.checkLength(len)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.byteView.set(bytes, this.offset)
|
|
|
|
this.offset += len
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.storeMethod = function (methodName, params) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var schema = this.mtproto ? Config.Schema.MTProto : Config.Schema.API
|
|
|
|
var methodData = false,
|
|
|
|
i
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
for (i = 0; i < schema.methods.length; i++) {
|
|
|
|
if (schema.methods[i].method == methodName) {
|
2016-06-28 19:40:53 +03:00
|
|
|
methodData = schema.methods[i]
|
2014-06-20 19:34:14 +04:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!methodData) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('No method ' + methodName + ' found')
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.storeInt(intToUint(methodData.id), methodName + '[id]')
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var param, type
|
|
|
|
var i, condType
|
|
|
|
var fieldBit
|
|
|
|
var len = methodData.params.length
|
2015-04-14 21:10:17 +03:00
|
|
|
for (i = 0; i < len; i++) {
|
2016-06-28 19:40:53 +03:00
|
|
|
param = methodData.params[i]
|
|
|
|
type = param.type
|
2015-04-06 14:50:38 +03:00
|
|
|
if (type.indexOf('?') !== -1) {
|
2016-06-28 19:40:53 +03:00
|
|
|
condType = type.split('?')
|
|
|
|
fieldBit = condType[0].split('.')
|
2015-04-06 14:50:38 +03:00
|
|
|
if (!(params[fieldBit[0]] & (1 << fieldBit[1]))) {
|
2016-06-28 19:40:53 +03:00
|
|
|
continue
|
2015-04-06 14:50:38 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
type = condType[1]
|
2015-04-06 14:50:38 +03:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.storeObject(params[param.name], type, methodName + '[' + param.name + ']')
|
2015-04-14 21:10:17 +03:00
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return methodData.type
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLSerialization.prototype.storeObject = function (obj, type, field) {
|
|
|
|
switch (type) {
|
2015-04-06 14:50:38 +03:00
|
|
|
case '#':
|
2016-06-28 19:40:53 +03:00
|
|
|
case 'int':
|
|
|
|
return this.storeInt(obj, field)
|
|
|
|
case 'long':
|
|
|
|
return this.storeLong(obj, field)
|
|
|
|
case 'int128':
|
|
|
|
return this.storeIntBytes(obj, 128, field)
|
|
|
|
case 'int256':
|
|
|
|
return this.storeIntBytes(obj, 256, field)
|
|
|
|
case 'int512':
|
|
|
|
return this.storeIntBytes(obj, 512, field)
|
|
|
|
case 'string':
|
|
|
|
return this.storeString(obj, field)
|
|
|
|
case 'bytes':
|
|
|
|
return this.storeBytes(obj, field)
|
|
|
|
case 'double':
|
|
|
|
return this.storeDouble(obj, field)
|
|
|
|
case 'Bool':
|
|
|
|
return this.storeBool(obj, field)
|
|
|
|
case 'true':
|
|
|
|
return
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (angular.isArray(obj)) {
|
|
|
|
if (type.substr(0, 6) == 'Vector') {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.writeInt(0x1cb5c415, field + '[id]')
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
else if (type.substr(0, 6) != 'vector') {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Invalid vector type ' + type)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
var itemType = type.substr(7, type.length - 8); // for "Vector<itemType>"
|
2016-06-28 19:40:53 +03:00
|
|
|
this.writeInt(obj.length, field + '[count]')
|
2014-06-20 19:34:14 +04:00
|
|
|
for (var i = 0; i < obj.length; i++) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.storeObject(obj[i], itemType, field + '[' + i + ']')
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
return true
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
else if (type.substr(0, 6).toLowerCase() == 'vector') {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Invalid vector object')
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!angular.isObject(obj)) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Invalid object for type ' + type)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var schema = this.mtproto ? Config.Schema.MTProto : Config.Schema.API
|
|
|
|
var predicate = obj['_']
|
|
|
|
var isBare = false
|
|
|
|
var constructorData = false,
|
|
|
|
i
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
if (isBare = (type.charAt(0) == '%')) {
|
2016-06-28 19:40:53 +03:00
|
|
|
type = type.substr(1)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < schema.constructors.length; i++) {
|
|
|
|
if (schema.constructors[i].predicate == predicate) {
|
2016-06-28 19:40:53 +03:00
|
|
|
constructorData = schema.constructors[i]
|
2014-06-20 19:34:14 +04:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!constructorData) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('No predicate ' + predicate + ' found')
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (predicate == type) {
|
2016-06-28 19:40:53 +03:00
|
|
|
isBare = true
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!isBare) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.writeInt(intToUint(constructorData.id), field + '[' + predicate + '][id]')
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var param, type
|
|
|
|
var i, condType
|
|
|
|
var fieldBit
|
|
|
|
var len = constructorData.params.length
|
2015-04-14 21:10:17 +03:00
|
|
|
for (i = 0; i < len; i++) {
|
2016-06-28 19:40:53 +03:00
|
|
|
param = constructorData.params[i]
|
|
|
|
type = param.type
|
2015-04-06 14:50:38 +03:00
|
|
|
if (type.indexOf('?') !== -1) {
|
2016-06-28 19:40:53 +03:00
|
|
|
condType = type.split('?')
|
|
|
|
fieldBit = condType[0].split('.')
|
2015-04-06 14:50:38 +03:00
|
|
|
if (!(obj[fieldBit[0]] & (1 << fieldBit[1]))) {
|
2016-06-28 19:40:53 +03:00
|
|
|
continue
|
2015-04-06 14:50:38 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
type = condType[1]
|
2015-04-06 14:50:38 +03:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.storeObject(obj[param.name], type, field + '[' + predicate + '][' + param.name + ']')
|
2015-04-14 21:10:17 +03:00
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return constructorData.type
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
function TLDeserialization (buffer, options) {
|
2016-06-28 19:40:53 +03:00
|
|
|
options = options || {}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.offset = 0 // in bytes
|
|
|
|
this.override = options.override || {}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.buffer = buffer
|
|
|
|
this.intView = new Uint32Array(this.buffer)
|
|
|
|
this.byteView = new Uint8Array(this.buffer)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
// this.debug = options.debug !== undefined ? options.debug : Config.Modes.debug
|
|
|
|
this.mtproto = options.mtproto || false
|
|
|
|
return this
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
TLDeserialization.prototype.readInt = function (field) {
|
|
|
|
if (this.offset >= this.intView.length * 4) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Nothing to fetch: ' + field)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var i = this.intView[this.offset / 4]
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.debug && console.log('<<<', i.toString(16), i, field)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.offset += 4
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return i
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLDeserialization.prototype.fetchInt = function (field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
return this.readInt((field || '') + ':int')
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
TLDeserialization.prototype.fetchDouble = function (field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var buffer = new ArrayBuffer(8)
|
|
|
|
var intView = new Int32Array(buffer)
|
|
|
|
var doubleView = new Float64Array(buffer)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
intView[0] = this.readInt((field || '') + ':double[low]'),
|
2016-06-28 19:40:53 +03:00
|
|
|
intView[1] = this.readInt((field || '') + ':double[high]')
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return doubleView[0]
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLDeserialization.prototype.fetchLong = function (field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var iLow = this.readInt((field || '') + ':long[low]')
|
|
|
|
var iHigh = this.readInt((field || '') + ':long[high]')
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var longDec = bigint(iHigh).shiftLeft(32).add(bigint(iLow)).toString()
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return longDec
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
TLDeserialization.prototype.fetchBool = function (field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var i = this.readInt((field || '') + ':bool')
|
2014-06-20 19:34:14 +04:00
|
|
|
if (i == 0x997275b5) {
|
2016-06-28 19:40:53 +03:00
|
|
|
return true
|
2014-06-20 19:34:14 +04:00
|
|
|
} else if (i == 0xbc799737) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.offset -= 4
|
|
|
|
return this.fetchObject('Object', field)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
TLDeserialization.prototype.fetchString = function (field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var len = this.byteView[this.offset++]
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
if (len == 254) {
|
|
|
|
var len = this.byteView[this.offset++] |
|
2016-06-28 19:40:53 +03:00
|
|
|
(this.byteView[this.offset++] << 8) |
|
|
|
|
(this.byteView[this.offset++] << 16)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var sUTF8 = ''
|
2014-06-20 19:34:14 +04:00
|
|
|
for (var i = 0; i < len; i++) {
|
2016-06-28 19:40:53 +03:00
|
|
|
sUTF8 += String.fromCharCode(this.byteView[this.offset++])
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Padding
|
|
|
|
while (this.offset % 4) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.offset++
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2016-06-28 19:40:53 +03:00
|
|
|
var s = decodeURIComponent(escape(sUTF8))
|
2014-06-20 19:34:14 +04:00
|
|
|
} catch (e) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var s = sUTF8
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.debug && console.log('<<<', s, (field || '') + ':string')
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return s
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
TLDeserialization.prototype.fetchBytes = function (field) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var len = this.byteView[this.offset++]
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
if (len == 254) {
|
2016-06-29 20:22:13 +03:00
|
|
|
len = this.byteView[this.offset++] |
|
2016-06-28 19:40:53 +03:00
|
|
|
(this.byteView[this.offset++] << 8) |
|
|
|
|
(this.byteView[this.offset++] << 16)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var bytes = this.byteView.subarray(this.offset, this.offset + len)
|
|
|
|
this.offset += len
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
// Padding
|
|
|
|
while (this.offset % 4) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.offset++
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.debug && console.log('<<<', bytesToHex(bytes), (field || '') + ':bytes')
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return bytes
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2014-10-24 22:37:43 +04:00
|
|
|
TLDeserialization.prototype.fetchIntBytes = function (bits, typed, field) {
|
2014-06-20 19:34:14 +04:00
|
|
|
if (bits % 32) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Invalid bits: ' + bits)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var len = bits / 8
|
2014-10-24 22:37:43 +04:00
|
|
|
if (typed) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var result = this.byteView.subarray(this.offset, this.offset + len)
|
|
|
|
this.offset += len
|
|
|
|
return result
|
2014-10-24 22:37:43 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var bytes = []
|
2014-06-20 19:34:14 +04:00
|
|
|
for (var i = 0; i < len; i++) {
|
2016-06-28 19:40:53 +03:00
|
|
|
bytes.push(this.byteView[this.offset++])
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.debug && console.log('<<<', bytesToHex(bytes), (field || '') + ':int' + bits)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return bytes
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2014-10-24 22:37:43 +04:00
|
|
|
TLDeserialization.prototype.fetchRawBytes = function (len, typed, field) {
|
2014-06-20 19:34:14 +04:00
|
|
|
if (len === false) {
|
2016-06-28 19:40:53 +03:00
|
|
|
len = this.readInt((field || '') + '_length')
|
2016-12-13 18:35:06 +03:00
|
|
|
if (len > this.byteView.byteLength) {
|
|
|
|
throw new Error('Invalid raw bytes length: ' + len + ', buffer len: ' + this.byteView.byteLength)
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2014-10-24 22:37:43 +04:00
|
|
|
if (typed) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var bytes = new Uint8Array(len)
|
|
|
|
bytes.set(this.byteView.subarray(this.offset, this.offset + len))
|
|
|
|
this.offset += len
|
|
|
|
return bytes
|
2014-10-24 22:37:43 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var bytes = []
|
2014-06-20 19:34:14 +04:00
|
|
|
for (var i = 0; i < len; i++) {
|
2016-06-28 19:40:53 +03:00
|
|
|
bytes.push(this.byteView[this.offset++])
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
this.debug && console.log('<<<', bytesToHex(bytes), (field || ''))
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return bytes
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLDeserialization.prototype.fetchObject = function (type, field) {
|
|
|
|
switch (type) {
|
2015-03-15 23:55:37 +01:00
|
|
|
case '#':
|
2016-06-28 19:40:53 +03:00
|
|
|
case 'int':
|
|
|
|
return this.fetchInt(field)
|
|
|
|
case 'long':
|
|
|
|
return this.fetchLong(field)
|
|
|
|
case 'int128':
|
|
|
|
return this.fetchIntBytes(128, false, field)
|
|
|
|
case 'int256':
|
|
|
|
return this.fetchIntBytes(256, false, field)
|
|
|
|
case 'int512':
|
|
|
|
return this.fetchIntBytes(512, false, field)
|
|
|
|
case 'string':
|
|
|
|
return this.fetchString(field)
|
|
|
|
case 'bytes':
|
|
|
|
return this.fetchBytes(field)
|
|
|
|
case 'double':
|
|
|
|
return this.fetchDouble(field)
|
|
|
|
case 'Bool':
|
|
|
|
return this.fetchBool(field)
|
|
|
|
case 'true':
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
field = field || type || 'Object'
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
if (type.substr(0, 6) == 'Vector' || type.substr(0, 6) == 'vector') {
|
|
|
|
if (type.charAt(0) == 'V') {
|
2016-06-28 19:40:53 +03:00
|
|
|
var constructor = this.readInt(field + '[id]')
|
|
|
|
var constructorCmp = uintToInt(constructor)
|
2015-11-20 16:39:26 +03:00
|
|
|
|
|
|
|
if (constructorCmp == 0x3072cfa1) { // Gzip packed
|
2016-06-28 19:40:53 +03:00
|
|
|
var compressed = this.fetchBytes(field + '[packed_string]')
|
|
|
|
var uncompressed = gzipUncompress(compressed)
|
|
|
|
var buffer = bytesToArrayBuffer(uncompressed)
|
|
|
|
var newDeserializer = (new TLDeserialization(buffer))
|
2015-11-20 16:39:26 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return newDeserializer.fetchObject(type, field)
|
2015-11-20 16:39:26 +03:00
|
|
|
}
|
|
|
|
if (constructorCmp != 0x1cb5c415) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Invalid vector constructor ' + constructor)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
var len = this.readInt(field + '[count]')
|
|
|
|
var result = []
|
2014-06-20 19:34:14 +04:00
|
|
|
if (len > 0) {
|
|
|
|
var itemType = type.substr(7, type.length - 8); // for "Vector<itemType>"
|
|
|
|
for (var i = 0; i < len; i++) {
|
|
|
|
result.push(this.fetchObject(itemType, field + '[' + i + ']'))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return result
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var schema = this.mtproto ? Config.Schema.MTProto : Config.Schema.API
|
|
|
|
var predicate = false
|
|
|
|
var constructorData = false
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
if (type.charAt(0) == '%') {
|
2016-06-28 19:40:53 +03:00
|
|
|
var checkType = type.substr(1)
|
2015-04-14 21:10:17 +03:00
|
|
|
for (var i = 0; i < schema.constructors.length; i++) {
|
2014-06-20 19:34:14 +04:00
|
|
|
if (schema.constructors[i].type == checkType) {
|
2016-06-28 19:40:53 +03:00
|
|
|
constructorData = schema.constructors[i]
|
2014-06-20 19:34:14 +04:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!constructorData) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Constructor not found for type: ' + type)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (type.charAt(0) >= 97 && type.charAt(0) <= 122) {
|
2015-04-14 21:10:17 +03:00
|
|
|
for (var i = 0; i < schema.constructors.length; i++) {
|
2014-06-20 19:34:14 +04:00
|
|
|
if (schema.constructors[i].predicate == type) {
|
2016-06-28 19:40:53 +03:00
|
|
|
constructorData = schema.constructors[i]
|
2014-06-20 19:34:14 +04:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!constructorData) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Constructor not found for predicate: ' + type)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
}else {
|
|
|
|
var constructor = this.readInt(field + '[id]')
|
|
|
|
var constructorCmp = uintToInt(constructor)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
if (constructorCmp == 0x3072cfa1) { // Gzip packed
|
2016-06-28 19:40:53 +03:00
|
|
|
var compressed = this.fetchBytes(field + '[packed_string]')
|
|
|
|
var uncompressed = gzipUncompress(compressed)
|
|
|
|
var buffer = bytesToArrayBuffer(uncompressed)
|
|
|
|
var newDeserializer = (new TLDeserialization(buffer))
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return newDeserializer.fetchObject(type, field)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var index = schema.constructorsIndex
|
2015-04-14 21:10:17 +03:00
|
|
|
if (!index) {
|
2016-06-28 19:40:53 +03:00
|
|
|
schema.constructorsIndex = index = {}
|
2015-04-14 21:10:17 +03:00
|
|
|
for (var i = 0; i < schema.constructors.length; i++) {
|
2016-06-28 19:40:53 +03:00
|
|
|
index[schema.constructors[i].id] = i
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
var i = index[constructorCmp]
|
2015-04-14 21:10:17 +03:00
|
|
|
if (i) {
|
2016-06-28 19:40:53 +03:00
|
|
|
constructorData = schema.constructors[i]
|
2015-04-14 21:10:17 +03:00
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var fallback = false
|
2014-06-20 19:34:14 +04:00
|
|
|
if (!constructorData && this.mtproto) {
|
2016-06-28 19:40:53 +03:00
|
|
|
var schemaFallback = Config.Schema.API
|
2014-06-20 19:34:14 +04:00
|
|
|
for (i = 0; i < schemaFallback.constructors.length; i++) {
|
|
|
|
if (schemaFallback.constructors[i].id == constructorCmp) {
|
2016-06-28 19:40:53 +03:00
|
|
|
constructorData = schemaFallback.constructors[i]
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
delete this.mtproto
|
|
|
|
fallback = true
|
|
|
|
break
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!constructorData) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Constructor not found: ' + constructor + ' ' + this.fetchInt() + ' ' + this.fetchInt())
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
predicate = constructorData.predicate
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var result = {'_': predicate}
|
|
|
|
var overrideKey = (this.mtproto ? 'mt_' : '') + predicate
|
|
|
|
var self = this
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
if (this.override[overrideKey]) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.override[overrideKey].apply(this, [result, field + '[' + predicate + ']'])
|
2014-06-20 19:34:14 +04:00
|
|
|
} else {
|
2016-06-28 19:40:53 +03:00
|
|
|
var i, param
|
|
|
|
var type, isCond
|
|
|
|
var condType, fieldBit
|
|
|
|
var value
|
|
|
|
var len = constructorData.params.length
|
2015-04-14 21:10:17 +03:00
|
|
|
for (i = 0; i < len; i++) {
|
2016-06-28 19:40:53 +03:00
|
|
|
param = constructorData.params[i]
|
|
|
|
type = param.type
|
2015-11-12 23:02:28 +03:00
|
|
|
if (type == '#' && result.pFlags === undefined) {
|
2016-06-28 19:40:53 +03:00
|
|
|
result.pFlags = {}
|
2015-11-12 23:02:28 +03:00
|
|
|
}
|
|
|
|
if (isCond = (type.indexOf('?') !== -1)) {
|
2016-06-28 19:40:53 +03:00
|
|
|
condType = type.split('?')
|
|
|
|
fieldBit = condType[0].split('.')
|
2015-03-15 23:55:37 +01:00
|
|
|
if (!(result[fieldBit[0]] & (1 << fieldBit[1]))) {
|
2016-06-28 19:40:53 +03:00
|
|
|
continue
|
2015-03-15 23:55:37 +01:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
type = condType[1]
|
2015-03-15 23:55:37 +01:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
value = self.fetchObject(type, field + '[' + predicate + '][' + param.name + ']')
|
2015-11-12 23:02:28 +03:00
|
|
|
|
|
|
|
if (isCond && type === 'true') {
|
2016-06-28 19:40:53 +03:00
|
|
|
result.pFlags[param.name] = value
|
2015-11-12 23:02:28 +03:00
|
|
|
} else {
|
2016-06-28 19:40:53 +03:00
|
|
|
result[param.name] = value
|
2015-11-12 23:02:28 +03:00
|
|
|
}
|
2015-04-14 21:10:17 +03:00
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (fallback) {
|
2016-06-28 19:40:53 +03:00
|
|
|
this.mtproto = true
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return result
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLDeserialization.prototype.getOffset = function () {
|
2016-06-28 19:40:53 +03:00
|
|
|
return this.offset
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
TLDeserialization.prototype.fetchEnd = function () {
|
|
|
|
if (this.offset != this.byteView.length) {
|
2016-06-28 19:40:53 +03:00
|
|
|
throw new Error('Fetch end with non-empty buffer')
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
return true
|
|
|
|
}
|