webogram-i2p/app/js/lib/mtproto.js

1641 lines
56 KiB
JavaScript
Raw Normal View History

2014-01-05 20:07:11 +04:00
/*!
2017-08-11 20:18:28 +02:00
* Webogram v0.6.0 - messaging web application for MTProto
2014-01-05 20:07:11 +04:00
* https://github.com/zhukov/webogram
* Copyright (C) 2014 Igor Zhukov <igor.beatle@gmail.com>
* https://github.com/zhukov/webogram/blob/master/LICENSE
*/
angular.module('izhukov.mtproto', ['izhukov.utils'])
2014-01-05 20:07:11 +04:00
.factory('MtpDcConfigurator', function () {
var sslSubdomains = ['pluto', 'venus', 'aurora', 'vesta', 'flora']
var dcOptions = Config.Modes.test
? [
{id: 1, host: '149.154.175.10', port: 80},
{id: 2, host: '149.154.167.40', port: 80},
{id: 3, host: '149.154.175.117', port: 80}
]
: [
{id: 1, host: '149.154.175.50', port: 80},
{id: 2, host: '149.154.167.51', port: 80},
{id: 3, host: '149.154.175.100', port: 80},
{id: 4, host: '149.154.167.91', port: 80},
{id: 5, host: '149.154.171.5', port: 80}
]
var chosenServers = {}
function chooseServer (dcID, upload) {
if (chosenServers[dcID] === undefined) {
var chosenServer = false,
i, dcOption
if (Config.Modes.ssl || !Config.Modes.http) {
var subdomain = sslSubdomains[dcID - 1] + (upload ? '-1' : '')
var path = Config.Modes.test ? 'apiw_test1' : 'apiw1'
chosenServer = 'https://' + subdomain + '.web.telegram.org/' + path
return chosenServer
}
2014-11-10 17:54:32 +03:00
for (i = 0; i < dcOptions.length; i++) {
dcOption = dcOptions[i]
if (dcOption.id == dcID) {
chosenServer = 'http://' + dcOption.host + (dcOption.port != 80 ? ':' + dcOption.port : '') + '/apiw1'
break
}
2014-01-05 20:07:11 +04:00
}
chosenServers[dcID] = chosenServer
2014-01-05 20:07:11 +04:00
}
return chosenServers[dcID]
2014-01-05 20:07:11 +04:00
}
return {
chooseServer: chooseServer
2014-01-05 20:07:11 +04:00
}
})
.factory('MtpRsaKeysManager', function () {
/**
* Server public key, obtained from here: https://core.telegram.org/api/obtaining_api_id
*
* -----BEGIN RSA PUBLIC KEY-----
* MIIBCgKCAQEAwVACPi9w23mF3tBkdZz+zwrzKOaaQdr01vAbU4E1pvkfj4sqDsm6
* lyDONS789sVoD/xCS9Y0hkkC3gtL1tSfTlgCMOOul9lcixlEKzwKENj1Yz/s7daS
* an9tqw3bfUV/nqgbhGX81v/+7RFAEd+RwFnK7a+XYl9sluzHRyVVaTTveB2GazTw
* Efzk2DWgkBluml8OREmvfraX3bkHZJTKX4EQSjBbbdJ2ZXIsRrYOXfaA+xayEGB+
* 8hdlLmAjbCVfaigxX0CDqWeR1yFL9kwd9P0NsZRPsmoqVwMbMu7mStFai6aIhc3n
* Slv8kg9qv1m6XHVQY3PnEw+QQtqSIXklHwIDAQAB
* -----END RSA PUBLIC KEY-----
*/
var publisKeysHex = [{
modulus: 'c150023e2f70db7985ded064759cfecf0af328e69a41daf4d6f01b538135a6f91f8f8b2a0ec9ba9720ce352efcf6c5680ffc424bd634864902de0b4bd6d49f4e580230e3ae97d95c8b19442b3c0a10d8f5633fecedd6926a7f6dab0ddb7d457f9ea81b8465fcd6fffeed114011df91c059caedaf97625f6c96ecc74725556934ef781d866b34f011fce4d835a090196e9a5f0e4449af7eb697ddb9076494ca5f81104a305b6dd27665722c46b60e5df680fb16b210607ef217652e60236c255f6a28315f4083a96791d7214bf64c1df4fd0db1944fb26a2a57031b32eee64ad15a8ba68885cde74a5bfc920f6abf59ba5c75506373e7130f9042da922179251f',
exponent: '010001'
}]
var publicKeysParsed = {}
var prepared = false
function prepareRsaKeys () {
if (prepared) {
return
}
2014-01-05 20:07:11 +04:00
for (var i = 0; i < publisKeysHex.length; i++) {
var keyParsed = publisKeysHex[i]
2014-01-05 20:07:11 +04:00
var RSAPublicKey = new TLSerialization()
RSAPublicKey.storeBytes(bytesFromHex(keyParsed.modulus), 'n')
RSAPublicKey.storeBytes(bytesFromHex(keyParsed.exponent), 'e')
2014-01-05 20:07:11 +04:00
var buffer = RSAPublicKey.getBuffer()
2014-01-05 20:07:11 +04:00
var fingerprintBytes = sha1BytesSync(buffer).slice(-8)
fingerprintBytes.reverse()
2014-01-05 20:07:11 +04:00
publicKeysParsed[bytesToHex(fingerprintBytes)] = {
modulus: keyParsed.modulus,
exponent: keyParsed.exponent
}
}
2014-01-05 20:07:11 +04:00
prepared = true
}
2014-01-05 20:07:11 +04:00
function selectRsaKeyByFingerPrint (fingerprints) {
prepareRsaKeys()
2014-01-05 20:07:11 +04:00
var fingerprintHex, foundKey, i
for (i = 0; i < fingerprints.length; i++) {
fingerprintHex = bigStringInt(fingerprints[i]).toString(16)
if (foundKey = publicKeysParsed[fingerprintHex]) {
return angular.extend({fingerprint: fingerprints[i]}, foundKey)
}
2014-01-05 20:07:11 +04:00
}
return false
}
2014-01-05 20:07:11 +04:00
return {
prepare: prepareRsaKeys,
select: selectRsaKeyByFingerPrint
}
})
2014-01-05 20:07:11 +04:00
.service('MtpSecureRandom', function ($window) {
$($window).on('click keydown', rng_seed_time)
return new SecureRandom()
})
2014-01-05 20:07:11 +04:00
.factory('MtpTimeManager', function (Storage) {
var lastMessageID = [0, 0]
var timeOffset = 0
2014-01-05 20:07:11 +04:00
Storage.get('server_time_offset').then(function (to) {
if (to) {
timeOffset = to
}
})
2014-01-05 20:07:11 +04:00
function generateMessageID () {
var timeTicks = tsNow(),
timeSec = Math.floor(timeTicks / 1000) + timeOffset,
timeMSec = timeTicks % 1000,
random = nextRandomInt(0xFFFF)
2014-01-05 20:07:11 +04:00
var messageID = [timeSec, (timeMSec << 21) | (random << 3) | 4]
if (lastMessageID[0] > messageID[0] ||
2014-01-05 20:07:11 +04:00
lastMessageID[0] == messageID[0] && lastMessageID[1] >= messageID[1]) {
messageID = [lastMessageID[0], lastMessageID[1] + 4]
}
2014-01-05 20:07:11 +04:00
lastMessageID = messageID
2014-01-05 20:07:11 +04:00
// console.log('generated msg id', messageID, timeOffset)
2014-01-05 20:07:11 +04:00
return longFromInts(messageID[0], messageID[1])
}
2014-01-05 20:07:11 +04:00
function applyServerTime (serverTime, localTime) {
var newTimeOffset = serverTime - Math.floor((localTime || tsNow()) / 1000)
var changed = Math.abs(timeOffset - newTimeOffset) > 10
Storage.set({server_time_offset: newTimeOffset})
2014-01-05 20:07:11 +04:00
lastMessageID = [0, 0]
timeOffset = newTimeOffset
console.log(dT(), 'Apply server time', serverTime, localTime, newTimeOffset, changed)
2014-01-25 20:40:48 +04:00
return changed
}
2014-01-25 20:40:48 +04:00
return {
generateID: generateMessageID,
applyServerTime: applyServerTime
}
})
2014-01-05 20:07:11 +04:00
.factory('MtpAuthorizer', function (MtpDcConfigurator, MtpRsaKeysManager, MtpSecureRandom, MtpTimeManager, CryptoWorker, $http, $q, $timeout) {
var chromeMatches = navigator.userAgent.match(/Chrome\/(\d+(\.\d+)?)/)
var chromeVersion = chromeMatches && parseFloat(chromeMatches[1]) || false
var xhrSendBuffer = !('ArrayBufferView' in window) && (!chromeVersion || chromeVersion < 30)
2014-01-05 20:07:11 +04:00
delete $http.defaults.headers.post['Content-Type']
delete $http.defaults.headers.common['Accept']
2014-01-05 20:07:11 +04:00
function mtpSendPlainRequest (dcID, requestBuffer) {
var requestLength = requestBuffer.byteLength,
requestArray = new Int32Array(requestBuffer)
var header = new TLSerialization()
header.storeLongP(0, 0, 'auth_key_id') // Auth key
header.storeLong(MtpTimeManager.generateID(), 'msg_id') // Msg_id
header.storeInt(requestLength, 'request_length')
2014-07-24 12:16:52 +04:00
var headerBuffer = header.getBuffer(),
headerArray = new Int32Array(headerBuffer)
var headerLength = headerBuffer.byteLength
2014-01-05 20:07:11 +04:00
var resultBuffer = new ArrayBuffer(headerLength + requestLength),
resultArray = new Int32Array(resultBuffer)
2014-01-05 20:07:11 +04:00
resultArray.set(headerArray)
resultArray.set(requestArray, headerArray.length)
2014-01-05 20:07:11 +04:00
var requestData = xhrSendBuffer ? resultBuffer : resultArray,
requestPromise
var url = MtpDcConfigurator.chooseServer(dcID)
var baseError = {code: 406, type: 'NETWORK_BAD_RESPONSE', url: url}
try {
requestPromise = $http.post(url, requestData, {
responseType: 'arraybuffer',
transformRequest: null
})
} catch (e) {
requestPromise = $q.reject(angular.extend(baseError, {originalError: e}))
}
return requestPromise.then(
function (result) {
if (!result.data || !result.data.byteLength) {
return $q.reject(baseError)
}
2014-01-05 20:07:11 +04:00
try {
var deserializer = new TLDeserialization(result.data, {mtproto: true})
var auth_key_id = deserializer.fetchLong('auth_key_id')
var msg_id = deserializer.fetchLong('msg_id')
var msg_len = deserializer.fetchInt('msg_len')
} catch (e) {
return $q.reject(angular.extend(baseError, {originalError: e}))
}
2014-01-05 20:07:11 +04:00
return deserializer
},
function (error) {
if (!error.message && !error.type) {
error = angular.extend(baseError, {originalError: error})
}
return $q.reject(error)
2014-08-02 00:23:44 +01:00
}
)
}
2014-08-02 00:23:44 +01:00
function mtpSendReqPQ (auth) {
var deferred = auth.deferred
var request = new TLSerialization({mtproto: true})
request.storeMethod('req_pq', {nonce: auth.nonce})
console.log(dT(), 'Send req_pq', bytesToHex(auth.nonce))
mtpSendPlainRequest(auth.dcID, request.getBuffer()).then(function (deserializer) {
var response = deserializer.fetchObject('ResPQ')
if (response._ != 'resPQ') {
throw new Error('[MT] resPQ response invalid: ' + response._)
}
2014-01-05 20:07:11 +04:00
if (!bytesCmp(auth.nonce, response.nonce)) {
throw new Error('[MT] resPQ nonce mismatch')
}
2014-01-05 20:07:11 +04:00
auth.serverNonce = response.server_nonce
auth.pq = response.pq
auth.fingerprints = response.server_public_key_fingerprints
console.log(dT(), 'Got ResPQ', bytesToHex(auth.serverNonce), bytesToHex(auth.pq), auth.fingerprints)
2014-01-05 20:07:11 +04:00
auth.publicKey = MtpRsaKeysManager.select(auth.fingerprints)
2014-01-05 20:07:11 +04:00
if (!auth.publicKey) {
throw new Error('[MT] No public key found')
}
2014-01-05 20:07:11 +04:00
console.log(dT(), 'PQ factorization start', auth.pq)
CryptoWorker.factorize(auth.pq).then(function (pAndQ) {
auth.p = pAndQ[0]
auth.q = pAndQ[1]
console.log(dT(), 'PQ factorization done', pAndQ[2])
mtpSendReqDhParams(auth)
}, function (error) {
console.log('Worker error', error, error.stack)
deferred.reject(error)
})
}, function (error) {
console.error(dT(), 'req_pq error', error.message)
deferred.reject(error)
})
2014-01-05 20:07:11 +04:00
$timeout(function () {
MtpRsaKeysManager.prepare()
})
}
2014-01-05 20:07:11 +04:00
function mtpSendReqDhParams (auth) {
var deferred = auth.deferred
2014-01-05 20:07:11 +04:00
auth.newNonce = new Array(32)
MtpSecureRandom.nextBytes(auth.newNonce)
2014-01-05 20:07:11 +04:00
var data = new TLSerialization({mtproto: true})
data.storeObject({
_: 'p_q_inner_data',
pq: auth.pq,
p: auth.p,
q: auth.q,
nonce: auth.nonce,
server_nonce: auth.serverNonce,
new_nonce: auth.newNonce
}, 'P_Q_inner_data', 'DECRYPTED_DATA')
2014-01-05 20:07:11 +04:00
var dataWithHash = sha1BytesSync(data.getBuffer()).concat(data.getBytes())
2014-01-05 20:07:11 +04:00
var request = new TLSerialization({mtproto: true})
request.storeMethod('req_DH_params', {
nonce: auth.nonce,
server_nonce: auth.serverNonce,
p: auth.p,
q: auth.q,
public_key_fingerprint: auth.publicKey.fingerprint,
encrypted_data: rsaEncrypt(auth.publicKey, dataWithHash)
})
2014-01-05 20:07:11 +04:00
console.log(dT(), 'Send req_DH_params')
mtpSendPlainRequest(auth.dcID, request.getBuffer()).then(function (deserializer) {
var response = deserializer.fetchObject('Server_DH_Params', 'RESPONSE')
2014-01-05 20:07:11 +04:00
if (response._ != 'server_DH_params_fail' && response._ != 'server_DH_params_ok') {
deferred.reject(new Error('[MT] Server_DH_Params response invalid: ' + response._))
return false
}
2014-01-05 20:07:11 +04:00
if (!bytesCmp(auth.nonce, response.nonce)) {
deferred.reject(new Error('[MT] Server_DH_Params nonce mismatch'))
return false
}
2014-01-05 20:07:11 +04:00
if (!bytesCmp(auth.serverNonce, response.server_nonce)) {
deferred.reject(new Error('[MT] Server_DH_Params server_nonce mismatch'))
return false
2014-01-05 20:07:11 +04:00
}
if (response._ == 'server_DH_params_fail') {
var newNonceHash = sha1BytesSync(auth.newNonce).slice(-16)
if (!bytesCmp(newNonceHash, response.new_nonce_hash)) {
deferred.reject(new Error('[MT] server_DH_params_fail new_nonce_hash mismatch'))
return false
}
deferred.reject(new Error('[MT] server_DH_params_fail'))
return false
}
2014-01-05 20:07:11 +04:00
try {
mtpDecryptServerDhDataAnswer(auth, response.encrypted_answer)
} catch (e) {
deferred.reject(e)
return false
}
2014-01-05 20:07:11 +04:00
mtpSendSetClientDhParams(auth)
}, function (error) {
deferred.reject(error)
})
}
2014-01-05 20:07:11 +04:00
function mtpDecryptServerDhDataAnswer (auth, encryptedAnswer) {
auth.localTime = tsNow()
2014-01-05 20:07:11 +04:00
auth.tmpAesKey = sha1BytesSync(auth.newNonce.concat(auth.serverNonce)).concat(sha1BytesSync(auth.serverNonce.concat(auth.newNonce)).slice(0, 12))
auth.tmpAesIv = sha1BytesSync(auth.serverNonce.concat(auth.newNonce)).slice(12).concat(sha1BytesSync([].concat(auth.newNonce, auth.newNonce)), auth.newNonce.slice(0, 4))
2014-01-05 20:07:11 +04:00
var answerWithHash = aesDecryptSync(encryptedAnswer, auth.tmpAesKey, auth.tmpAesIv)
2014-01-05 20:07:11 +04:00
var hash = answerWithHash.slice(0, 20)
var answerWithPadding = answerWithHash.slice(20)
var buffer = bytesToArrayBuffer(answerWithPadding)
2014-01-05 20:07:11 +04:00
var deserializer = new TLDeserialization(buffer, {mtproto: true})
var response = deserializer.fetchObject('Server_DH_inner_data')
2014-01-05 20:07:11 +04:00
if (response._ != 'server_DH_inner_data') {
throw new Error('[MT] server_DH_inner_data response invalid: ' + constructor)
}
2014-01-05 20:07:11 +04:00
if (!bytesCmp(auth.nonce, response.nonce)) {
throw new Error('[MT] server_DH_inner_data nonce mismatch')
}
2014-01-05 20:07:11 +04:00
if (!bytesCmp(auth.serverNonce, response.server_nonce)) {
throw new Error('[MT] server_DH_inner_data serverNonce mismatch')
}
2014-01-05 20:07:11 +04:00
console.log(dT(), 'Done decrypting answer')
auth.g = response.g
auth.dhPrime = response.dh_prime
auth.gA = response.g_a
auth.serverTime = response.server_time
auth.retry = 0
2014-01-05 20:07:11 +04:00
mtpVerifyDhParams(auth.g, auth.dhPrime, auth.gA)
var offset = deserializer.getOffset()
2014-01-05 20:07:11 +04:00
if (!bytesCmp(hash, sha1BytesSync(answerWithPadding.slice(0, offset)))) {
throw new Error('[MT] server_DH_inner_data SHA1-hash mismatch')
}
2014-01-05 20:07:11 +04:00
MtpTimeManager.applyServerTime(auth.serverTime, auth.localTime)
}
2014-01-05 20:07:11 +04:00
function mtpVerifyDhParams(g, dhPrime, gA) {
console.log(dT(), 'Verifying DH params')
var dhPrimeHex = bytesToHex(dhPrime);
if (g != 3 ||
dhPrimeHex !== 'c71caeb9c6b1c9048e6c522f70f13f73980d40238e3e21c14934d037563d930f48198a0aa7c14058229493d22530f4dbfa336f6e0ac925139543aed44cce7c3720fd51f69458705ac68cd4fe6b6b13abdc9746512969328454f18faf8c595f642477fe96bb2a941d5bcd1d4ac8cc49880708fa9b378e3c4f3a9060bee67cf9a4a4a695811051907e162753b56b0f6b410dba74d8a84b2a14b3144e0ef1284754fd17ed950d5965b4b9dd46582db1178d169c6bc465b0d6ff9ca3928fef5b9ae4e418fc15e83ebea0f87fa9ff5eed70050ded2849f47bf959d956850ce929851f0d8115f635b105ee2e4e15d04b2454bf6f4fadf034b10403119cd8e3b92fcc5b') {
// The verified value is from https://core.telegram.org/mtproto/security_guidelines
throw new Error('[MT] DH params are not verified: unknown dhPrime')
}
console.log(dT(), 'dhPrime cmp OK')
var gABigInt = new BigInteger(bytesToHex(gA), 16)
var dhPrimeBigInt = new BigInteger(dhPrimeHex, 16)
if (gABigInt.compareTo(BigInteger.ONE) <= 0) {
throw new Error('[MT] DH params are not verified: gA <= 1')
}
if (gABigInt.compareTo(dhPrimeBigInt.subtract(BigInteger.ONE)) >= 0) {
throw new Error('[MT] DH params are not verified: gA >= dhPrime - 1')
}
console.log(dT(), '1 < gA < dhPrime-1 OK')
var two = new BigInteger(null)
two.fromInt(2)
var twoPow = two.pow(2048 - 64)
if (gABigInt.compareTo(twoPow) < 0) {
throw new Error('[MT] DH params are not verified: gA < 2^{2048-64}')
}
if (gABigInt.compareTo(dhPrimeBigInt.subtract(twoPow)) >= 0) {
throw new Error('[MT] DH params are not verified: gA > dhPrime - 2^{2048-64}')
}
console.log(dT(), '2^{2048-64} < gA < dhPrime-2^{2048-64} OK')
return true
}
function mtpSendSetClientDhParams (auth) {
var deferred = auth.deferred
var gBytes = bytesFromHex(auth.g.toString(16))
2014-01-05 20:07:11 +04:00
auth.b = new Array(256)
MtpSecureRandom.nextBytes(auth.b)
2014-01-05 20:07:11 +04:00
CryptoWorker.modPow(gBytes, auth.b, auth.dhPrime).then(function (gB) {
var data = new TLSerialization({mtproto: true})
data.storeObject({
_: 'client_DH_inner_data',
nonce: auth.nonce,
server_nonce: auth.serverNonce,
retry_id: [0, auth.retry++],
g_b: gB
}, 'Client_DH_Inner_Data')
2014-01-05 20:07:11 +04:00
var dataWithHash = sha1BytesSync(data.getBuffer()).concat(data.getBytes())
2014-01-05 20:07:11 +04:00
var encryptedData = aesEncryptSync(dataWithHash, auth.tmpAesKey, auth.tmpAesIv)
2014-01-05 20:07:11 +04:00
var request = new TLSerialization({mtproto: true})
request.storeMethod('set_client_DH_params', {
nonce: auth.nonce,
server_nonce: auth.serverNonce,
encrypted_data: encryptedData
})
2014-01-05 20:07:11 +04:00
console.log(dT(), 'Send set_client_DH_params')
mtpSendPlainRequest(auth.dcID, request.getBuffer()).then(function (deserializer) {
var response = deserializer.fetchObject('Set_client_DH_params_answer')
2014-01-05 20:07:11 +04:00
if (response._ != 'dh_gen_ok' && response._ != 'dh_gen_retry' && response._ != 'dh_gen_fail') {
deferred.reject(new Error('[MT] Set_client_DH_params_answer response invalid: ' + response._))
return false
}
2014-01-05 20:07:11 +04:00
if (!bytesCmp(auth.nonce, response.nonce)) {
deferred.reject(new Error('[MT] Set_client_DH_params_answer nonce mismatch'))
return false
}
2014-01-05 20:07:11 +04:00
if (!bytesCmp(auth.serverNonce, response.server_nonce)) {
deferred.reject(new Error('[MT] Set_client_DH_params_answer server_nonce mismatch'))
return false
}
2014-01-05 20:07:11 +04:00
CryptoWorker.modPow(auth.gA, auth.b, auth.dhPrime).then(function (authKey) {
var authKeyHash = sha1BytesSync(authKey),
authKeyAux = authKeyHash.slice(0, 8),
authKeyID = authKeyHash.slice(-8)
2014-07-04 21:51:39 +04:00
console.log(dT(), 'Got Set_client_DH_params_answer', response._)
switch (response._) {
case 'dh_gen_ok':
var newNonceHash1 = sha1BytesSync(auth.newNonce.concat([1], authKeyAux)).slice(-16)
2014-07-04 21:51:39 +04:00
if (!bytesCmp(newNonceHash1, response.new_nonce_hash1)) {
deferred.reject(new Error('[MT] Set_client_DH_params_answer new_nonce_hash1 mismatch'))
return false
}
2014-07-04 21:51:39 +04:00
var serverSalt = bytesXor(auth.newNonce.slice(0, 8), auth.serverNonce.slice(0, 8))
// console.log('Auth successfull!', authKeyID, authKey, serverSalt)
2014-07-04 21:51:39 +04:00
auth.authKeyID = authKeyID
auth.authKey = authKey
auth.serverSalt = serverSalt
2014-07-04 21:51:39 +04:00
deferred.resolve(auth)
break
2014-07-04 21:51:39 +04:00
case 'dh_gen_retry':
var newNonceHash2 = sha1BytesSync(auth.newNonce.concat([2], authKeyAux)).slice(-16)
if (!bytesCmp(newNonceHash2, response.new_nonce_hash2)) {
deferred.reject(new Error('[MT] Set_client_DH_params_answer new_nonce_hash2 mismatch'))
return false
}
2014-07-04 21:51:39 +04:00
return mtpSendSetClientDhParams(auth)
2014-07-04 21:51:39 +04:00
case 'dh_gen_fail':
var newNonceHash3 = sha1BytesSync(auth.newNonce.concat([3], authKeyAux)).slice(-16)
if (!bytesCmp(newNonceHash3, response.new_nonce_hash3)) {
deferred.reject(new Error('[MT] Set_client_DH_params_answer new_nonce_hash3 mismatch'))
return false
}
2014-07-04 21:51:39 +04:00
deferred.reject(new Error('[MT] Set_client_DH_params_answer fail'))
return false
}
}, function (error) {
deferred.reject(error)
})
2014-07-04 21:51:39 +04:00
}, function (error) {
deferred.reject(error)
2014-07-04 21:51:39 +04:00
})
}, function (error) {
deferred.reject(error)
})
2014-01-05 20:07:11 +04:00
}
var cached = {}
2014-01-05 20:07:11 +04:00
function mtpAuth (dcID) {
if (cached[dcID] !== undefined) {
return cached[dcID]
}
2014-01-05 20:07:11 +04:00
var nonce = []
for (var i = 0; i < 16; i++) {
nonce.push(nextRandomInt(0xFF))
}
2014-07-06 17:53:46 +04:00
if (!MtpDcConfigurator.chooseServer(dcID)) {
return $q.reject(new Error('[MT] No server found for dc ' + dcID))
}
2014-07-06 17:53:46 +04:00
var auth = {
dcID: dcID,
nonce: nonce,
deferred: $q.defer()
}
2014-01-05 20:07:11 +04:00
$timeout(function () {
mtpSendReqPQ(auth)
})
2014-01-05 20:07:11 +04:00
cached[dcID] = auth.deferred.promise
2014-01-05 20:07:11 +04:00
cached[dcID]['catch'](function () {
delete cached[dcID]
})
2014-01-05 20:07:11 +04:00
return cached[dcID]
}
return {
auth: mtpAuth
}
})
.factory('MtpNetworkerFactory', function (MtpDcConfigurator, MtpTimeManager, MtpSecureRandom, Storage, CryptoWorker, AppRuntimeManager, $http, $q, $timeout, $interval, $rootScope) {
var updatesProcessor
var iii = 0,
offline
var offlineInited = false
var akStopped = false
var chromeMatches = navigator.userAgent.match(/Chrome\/(\d+(\.\d+)?)/)
var chromeVersion = chromeMatches && parseFloat(chromeMatches[1]) || false
var xhrSendBuffer = !('ArrayBufferView' in window) && (!chromeVersion || chromeVersion < 30)
delete $http.defaults.headers.post['Content-Type']
delete $http.defaults.headers.common['Accept']
$rootScope.retryOnline = function () {
$(document.body).trigger('online')
}
2014-03-01 18:27:39 +01:00
function MtpNetworker (dcID, authKey, serverSalt, options) {
options = options || {}
2014-02-18 21:44:50 +04:00
this.dcID = dcID
this.iii = iii++
2014-01-05 20:07:11 +04:00
this.authKey = authKey
this.authKeyUint8 = convertToUint8Array(authKey)
this.authKeyBuffer = convertToArrayBuffer(authKey)
this.authKeyID = sha1BytesSync(authKey).slice(-8)
2014-01-05 20:07:11 +04:00
this.serverSalt = serverSalt
2014-01-05 20:07:11 +04:00
this.upload = options.fileUpload || options.fileDownload || false
2014-01-05 20:07:11 +04:00
this.updateSession()
this.lastServerMessages = []
this.currentRequests = 0
this.checkConnectionPeriod = 0
2014-01-05 20:07:11 +04:00
this.sentMessages = {}
this.clientMessages = []
2014-01-05 20:07:11 +04:00
this.pendingMessages = {}
this.pendingAcks = []
this.pendingResends = []
this.connectionInited = false
2014-01-05 20:07:11 +04:00
this.pendingTimeouts = []
2014-01-05 20:07:11 +04:00
this.longPollInt = $interval(this.checkLongPoll.bind(this), 10000)
this.checkLongPoll()
2014-01-05 20:07:11 +04:00
if (!offlineInited) {
offlineInited = true
$rootScope.offline = true
$rootScope.offlineConnecting = true
}
if (Config.Navigator.mobile) {
this.setupMobileSleep()
}
}
MtpNetworker.prototype.updateSession = function () {
this.seqNo = 0
this.prevSessionID = this.sessionID
this.sessionID = new Array(8)
MtpSecureRandom.nextBytes(this.sessionID)
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.setupMobileSleep = function () {
var self = this
$rootScope.$watch('idle.isIDLE', function (isIDLE) {
if (isIDLE) {
self.sleepAfter = tsNow() + 30000
} else {
delete self.sleepAfter
self.checkLongPoll()
}
})
$rootScope.$on('push_received', function () {
// console.log(dT(), 'push recieved', self.sleepAfter)
if (self.sleepAfter) {
self.sleepAfter = tsNow() + 30000
self.checkLongPoll()
}
})
}
2014-01-25 20:40:48 +04:00
MtpNetworker.prototype.updateSentMessage = function (sentMessageID) {
var sentMessage = this.sentMessages[sentMessageID]
if (!sentMessage) {
return false
}
var self = this
if (sentMessage.container) {
var newInner = []
angular.forEach(sentMessage.inner, function (innerSentMessageID) {
var innerSentMessage = self.updateSentMessage(innerSentMessageID)
if (innerSentMessage) {
newInner.push(innerSentMessage.msg_id)
}
})
sentMessage.inner = newInner
}
sentMessage.msg_id = MtpTimeManager.generateID()
sentMessage.seq_no = this.generateSeqNo(
sentMessage.notContentRelated ||
sentMessage.container
)
this.sentMessages[sentMessage.msg_id] = sentMessage
delete self.sentMessages[sentMessageID]
2014-01-25 20:40:48 +04:00
return sentMessage
}
2014-01-25 20:40:48 +04:00
MtpNetworker.prototype.generateSeqNo = function (notContentRelated) {
var seqNo = this.seqNo * 2
2014-01-05 20:07:11 +04:00
if (!notContentRelated) {
seqNo++
this.seqNo++
}
2014-01-05 20:07:11 +04:00
return seqNo
2014-01-05 20:07:11 +04:00
}
MtpNetworker.prototype.wrapMtpCall = function (method, params, options) {
var serializer = new TLSerialization({mtproto: true})
2014-01-05 20:07:11 +04:00
serializer.storeMethod(method, params)
2014-01-05 20:07:11 +04:00
var messageID = MtpTimeManager.generateID()
var seqNo = this.generateSeqNo()
var message = {
msg_id: messageID,
seq_no: seqNo,
body: serializer.getBytes()
}
2014-01-05 20:07:11 +04:00
if (Config.Modes.debug) {
console.log(dT(), 'MT call', method, params, messageID, seqNo)
}
2014-01-05 20:07:11 +04:00
return this.pushMessage(message, options)
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.wrapMtpMessage = function (object, options) {
options = options || {}
2014-01-05 20:07:11 +04:00
var serializer = new TLSerialization({mtproto: true})
serializer.storeObject(object, 'Object')
2014-01-05 20:07:11 +04:00
var messageID = MtpTimeManager.generateID()
var seqNo = this.generateSeqNo(options.notContentRelated)
var message = {
msg_id: messageID,
seq_no: seqNo,
body: serializer.getBytes()
}
2014-01-05 20:07:11 +04:00
if (Config.Modes.debug) {
console.log(dT(), 'MT message', object, messageID, seqNo)
}
2014-01-05 20:07:11 +04:00
return this.pushMessage(message, options)
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.wrapApiCall = function (method, params, options) {
var serializer = new TLSerialization(options)
if (!this.connectionInited) {
serializer.storeInt(0xda9b0d0d, 'invokeWithLayer')
serializer.storeInt(Config.Schema.API.layer, 'layer')
serializer.storeInt(0x69796de9, 'initConnection')
serializer.storeInt(Config.App.id, 'api_id')
serializer.storeString(navigator.userAgent || 'Unknown UserAgent', 'device_model')
serializer.storeString(navigator.platform || 'Unknown Platform', 'system_version')
serializer.storeString(Config.App.version, 'app_version')
serializer.storeString(navigator.language || 'en', 'lang_code')
}
2014-01-05 20:07:11 +04:00
if (options.afterMessageID) {
serializer.storeInt(0xcb9f372d, 'invokeAfterMsg')
serializer.storeLong(options.afterMessageID, 'msg_id')
}
2014-02-20 01:46:47 +01:00
options.resultType = serializer.storeMethod(method, params)
var messageID = MtpTimeManager.generateID()
var seqNo = this.generateSeqNo()
var message = {
msg_id: messageID,
seq_no: seqNo,
body: serializer.getBytes(true),
isAPI: true
}
2014-01-05 20:07:11 +04:00
if (Config.Modes.debug) {
console.log(dT(), 'Api call', method, params, messageID, seqNo, options)
} else {
console.log(dT(), 'Api call', method)
}
2014-01-05 20:07:11 +04:00
return this.pushMessage(message, options)
}
MtpNetworker.prototype.checkLongPoll = function (force) {
var isClean = this.cleanupSent()
// console.log('Check lp', this.longPollPending, tsNow(), this.dcID, isClean)
if (this.longPollPending && tsNow() < this.longPollPending ||
2017-05-11 19:34:26 +03:00
this.offline ||
akStopped) {
return false
}
var self = this
Storage.get('dc').then(function (baseDcID) {
if (isClean && (
baseDcID != self.dcID ||
self.upload ||
self.sleepAfter && tsNow() > self.sleepAfter
)) {
// console.warn(dT(), 'Send long-poll for DC is delayed', self.dcID, self.sleepAfter)
return
}
self.sendLongPoll()
})
2014-01-05 20:07:11 +04:00
}
MtpNetworker.prototype.sendLongPoll = function () {
var maxWait = 25000
var self = this
this.longPollPending = tsNow() + maxWait
// console.log('Set lp', this.longPollPending, tsNow())
this.wrapMtpCall('http_wait', {
max_delay: 500,
wait_after: 150,
max_wait: maxWait
}, {
noResponse: true,
longPoll: true
}).then(function () {
delete self.longPollPending
setZeroTimeout(self.checkLongPoll.bind(self))
}, function () {
console.log('Long-poll failed')
})
2014-02-20 01:46:47 +01:00
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.pushMessage = function (message, options) {
var deferred = $q.defer()
this.sentMessages[message.msg_id] = angular.extend(message, options || {}, {deferred: deferred})
this.pendingMessages[message.msg_id] = 0
2014-01-05 20:07:11 +04:00
if (!options || !options.noShedule) {
this.sheduleRequest()
}
if (angular.isObject(options)) {
options.messageID = message.msg_id
2014-01-05 20:07:11 +04:00
}
return deferred.promise
2014-01-05 20:07:11 +04:00
}
MtpNetworker.prototype.pushResend = function (messageID, delay) {
var value = delay ? tsNow() + delay : 0
var sentMessage = this.sentMessages[messageID]
if (sentMessage.container) {
for (var i = 0; i < sentMessage.inner.length; i++) {
this.pendingMessages[sentMessage.inner[i]] = value
}
} else {
this.pendingMessages[messageID] = value
}
2014-02-18 21:44:50 +04:00
// console.log('Resend due', messageID, this.pendingMessages)
2014-02-18 21:44:50 +04:00
this.sheduleRequest(delay)
}
2014-02-18 21:44:50 +04:00
MtpNetworker.prototype.getMsgKeyIv = function (msgKey, isOut) {
var authKey = this.authKeyUint8
var x = isOut ? 0 : 8
var sha1aText = new Uint8Array(48)
var sha1bText = new Uint8Array(48)
var sha1cText = new Uint8Array(48)
var sha1dText = new Uint8Array(48)
var promises = {}
sha1aText.set(msgKey, 0)
sha1aText.set(authKey.subarray(x, x + 32), 16)
promises.sha1a = CryptoWorker.sha1Hash(sha1aText)
sha1bText.set(authKey.subarray(x + 32, x + 48), 0)
sha1bText.set(msgKey, 16)
sha1bText.set(authKey.subarray(x + 48, x + 64), 32)
promises.sha1b = CryptoWorker.sha1Hash(sha1bText)
sha1cText.set(authKey.subarray(x + 64, x + 96), 0)
sha1cText.set(msgKey, 32)
promises.sha1c = CryptoWorker.sha1Hash(sha1cText)
sha1dText.set(msgKey, 0)
sha1dText.set(authKey.subarray(x + 96, x + 128), 16)
promises.sha1d = CryptoWorker.sha1Hash(sha1dText)
return $q.all(promises).then(function (result) {
var aesKey = new Uint8Array(32)
var aesIv = new Uint8Array(32)
sha1a = new Uint8Array(result.sha1a),
sha1b = new Uint8Array(result.sha1b),
sha1c = new Uint8Array(result.sha1c),
sha1d = new Uint8Array(result.sha1d)
aesKey.set(sha1a.subarray(0, 8))
aesKey.set(sha1b.subarray(8, 20), 8)
aesKey.set(sha1c.subarray(4, 16), 20)
aesIv.set(sha1a.subarray(8, 20))
aesIv.set(sha1b.subarray(0, 8), 12)
aesIv.set(sha1c.subarray(16, 20), 20)
aesIv.set(sha1d.subarray(0, 8), 24)
return [aesKey, aesIv]
})
}
2014-02-18 21:44:50 +04:00
MtpNetworker.prototype.checkConnection = function (event) {
$rootScope.offlineConnecting = true
2014-02-18 21:44:50 +04:00
console.log(dT(), 'Check connection', event)
$timeout.cancel(this.checkConnectionPromise)
2014-02-18 21:44:50 +04:00
var serializer = new TLSerialization({mtproto: true})
var pingID = [nextRandomInt(0xFFFFFFFF), nextRandomInt(0xFFFFFFFF)]
2014-02-18 21:44:50 +04:00
serializer.storeMethod('ping', {ping_id: pingID})
2014-02-18 21:44:50 +04:00
var pingMessage = {
msg_id: MtpTimeManager.generateID(),
seq_no: this.generateSeqNo(true),
body: serializer.getBytes()
2014-02-18 21:44:50 +04:00
}
var self = this
this.sendEncryptedRequest(pingMessage, {timeout: 15000}).then(function (result) {
delete $rootScope.offlineConnecting
self.toggleOffline(false)
}, function () {
console.log(dT(), 'Delay ', self.checkConnectionPeriod * 1000)
self.checkConnectionPromise = $timeout(self.checkConnection.bind(self), parseInt(self.checkConnectionPeriod * 1000))
self.checkConnectionPeriod = Math.min(60, self.checkConnectionPeriod * 1.5)
$timeout(function () {
delete $rootScope.offlineConnecting
}, 1000)
})
2014-02-18 21:44:50 +04:00
}
MtpNetworker.prototype.toggleOffline = function (enabled) {
// console.log('toggle ', enabled, this.dcID, this.iii)
if (this.offline !== undefined && this.offline == enabled) {
return false
}
2014-02-18 21:44:50 +04:00
this.offline = enabled
$rootScope.offline = enabled
$rootScope.offlineConnecting = false
2014-02-18 21:44:50 +04:00
if (this.offline) {
$timeout.cancel(this.nextReqPromise)
delete this.nextReq
2014-01-05 20:07:11 +04:00
if (this.checkConnectionPeriod < 1.5) {
this.checkConnectionPeriod = 0
}
this.checkConnectionPromise = $timeout(this.checkConnection.bind(this), parseInt(this.checkConnectionPeriod * 1000))
this.checkConnectionPeriod = Math.min(30, (1 + this.checkConnectionPeriod) * 1.5)
this.onOnlineCb = this.checkConnection.bind(this)
$(document.body).on('online focus', this.onOnlineCb)
} else {
delete this.longPollPending
this.checkLongPoll()
this.sheduleRequest()
if (this.onOnlineCb) {
$(document.body).off('online focus', this.onOnlineCb)
}
$timeout.cancel(this.checkConnectionPromise)
2014-01-05 20:07:11 +04:00
}
}
MtpNetworker.prototype.performSheduledRequest = function () {
// console.log(dT(), 'sheduled', this.dcID, this.iii)
if (this.offline || akStopped) {
console.log(dT(), 'Cancel sheduled')
return false
}
delete this.nextReq
if (this.pendingAcks.length) {
var ackMsgIDs = []
for (var i = 0; i < this.pendingAcks.length; i++) {
ackMsgIDs.push(this.pendingAcks[i])
}
// console.log('acking messages', ackMsgIDs)
this.wrapMtpMessage({_: 'msgs_ack', msg_ids: ackMsgIDs}, {notContentRelated: true, noShedule: true})
}
if (this.pendingResends.length) {
var resendMsgIDs = []
var resendOpts = {noShedule: true, notContentRelated: true}
for (var i = 0; i < this.pendingResends.length; i++) {
resendMsgIDs.push(this.pendingResends[i])
}
// console.log('resendReq messages', resendMsgIDs)
this.wrapMtpMessage({_: 'msg_resend_req', msg_ids: resendMsgIDs}, resendOpts)
this.lastResendReq = {req_msg_id: resendOpts.messageID, resend_msg_ids: resendMsgIDs}
2014-01-05 20:07:11 +04:00
}
var messages = [],
message
var messagesByteLen = 0
var currentTime = tsNow()
var hasApiCall = false
var hasHttpWait = false
var lengthOverflow = false
var singlesCount = 0
var self = this
angular.forEach(this.pendingMessages, function (value, messageID) {
if (!value || value >= currentTime) {
if (message = self.sentMessages[messageID]) {
var messageByteLength = (message.body.byteLength || message.body.length) + 32
if (!message.notContentRelated &&
lengthOverflow) {
return
}
if (!message.notContentRelated &&
messagesByteLen &&
messagesByteLen + messageByteLength > 655360) { // 640 Kb
lengthOverflow = true
return
}
if (message.singleInRequest) {
singlesCount++
if (singlesCount > 1) {
return
}
}
messages.push(message)
messagesByteLen += messageByteLength
if (message.isAPI) {
hasApiCall = true
}
else if (message.longPoll) {
hasHttpWait = true
}
} else {
// console.log(message, messageID)
}
delete self.pendingMessages[messageID]
2014-01-05 20:07:11 +04:00
}
})
2014-01-05 20:07:11 +04:00
if (hasApiCall && !hasHttpWait) {
var serializer = new TLSerialization({mtproto: true})
serializer.storeMethod('http_wait', {
max_delay: 500,
wait_after: 150,
max_wait: 3000
})
messages.push({
msg_id: MtpTimeManager.generateID(),
seq_no: this.generateSeqNo(),
body: serializer.getBytes()
})
2014-01-05 20:07:11 +04:00
}
if (!messages.length) {
// console.log('no sheduled messages')
return
2014-01-05 20:07:11 +04:00
}
var noResponseMsgs = []
if (messages.length > 1) {
var container = new TLSerialization({mtproto: true, startMaxLength: messagesByteLen + 64})
container.storeInt(0x73f1f8dc, 'CONTAINER[id]')
container.storeInt(messages.length, 'CONTAINER[count]')
var onloads = []
var innerMessages = []
for (var i = 0; i < messages.length; i++) {
container.storeLong(messages[i].msg_id, 'CONTAINER[' + i + '][msg_id]')
innerMessages.push(messages[i].msg_id)
container.storeInt(messages[i].seq_no, 'CONTAINER[' + i + '][seq_no]')
container.storeInt(messages[i].body.length, 'CONTAINER[' + i + '][bytes]')
container.storeRawBytes(messages[i].body, 'CONTAINER[' + i + '][body]')
if (messages[i].noResponse) {
noResponseMsgs.push(messages[i].msg_id)
}
}
2014-01-05 20:07:11 +04:00
var containerSentMessage = {
msg_id: MtpTimeManager.generateID(),
seq_no: this.generateSeqNo(true),
container: true,
inner: innerMessages
}
2014-01-05 20:07:11 +04:00
message = angular.extend({body: container.getBytes(true)}, containerSentMessage)
2014-01-05 20:07:11 +04:00
this.sentMessages[message.msg_id] = containerSentMessage
2014-01-05 20:07:11 +04:00
2014-06-03 15:08:26 +04:00
if (Config.Modes.debug) {
console.log(dT(), 'Container', innerMessages, message.msg_id, message.seq_no)
}
} else {
if (message.noResponse) {
noResponseMsgs.push(message.msg_id)
}
this.sentMessages[message.msg_id] = message
}
this.pendingAcks = []
this.sendEncryptedRequest(message).then(function (result) {
self.toggleOffline(false)
// console.log('parse for', message)
self.parseResponse(result.data).then(function (response) {
if (Config.Modes.debug) {
console.log(dT(), 'Server response', self.dcID, response)
}
self.processMessage(response.response, response.messageID, response.sessionID)
2014-02-18 21:44:50 +04:00
angular.forEach(noResponseMsgs, function (msgID) {
if (self.sentMessages[msgID]) {
var deferred = self.sentMessages[msgID].deferred
delete self.sentMessages[msgID]
deferred.resolve()
}
})
2014-02-18 21:44:50 +04:00
self.checkLongPoll()
2014-02-18 21:44:50 +04:00
this.checkConnectionPeriod = Math.max(1.1, Math.sqrt(this.checkConnectionPeriod))
})
}, function (error) {
console.log('Encrypted request failed', error)
2014-02-18 21:44:50 +04:00
if (message.container) {
angular.forEach(message.inner, function (msgID) {
self.pendingMessages[msgID] = 0
})
delete self.sentMessages[message.msg_id]
} else {
self.pendingMessages[message.msg_id] = 0
2014-02-18 21:44:50 +04:00
}
angular.forEach(noResponseMsgs, function (msgID) {
if (self.sentMessages[msgID]) {
var deferred = self.sentMessages[msgID].deferred
delete self.sentMessages[msgID]
delete self.pendingMessages[msgID]
deferred.reject()
}
})
self.toggleOffline(true)
})
if (lengthOverflow || singlesCount > 1) {
this.sheduleRequest()
}
}
MtpNetworker.prototype.getEncryptedMessage = function (bytes) {
var self = this
// console.log(dT(), 'Start encrypt', bytes.byteLength)
return CryptoWorker.sha1Hash(bytes).then(function (bytesHash) {
// console.log(dT(), 'after hash')
var msgKey = new Uint8Array(bytesHash).subarray(4, 20)
return self.getMsgKeyIv(msgKey, true).then(function (keyIv) {
// console.log(dT(), 'after msg key iv')
return CryptoWorker.aesEncrypt(bytes, keyIv[0], keyIv[1]).then(function (encryptedBytes) {
// console.log(dT(), 'Finish encrypt')
return {
bytes: encryptedBytes,
msgKey: msgKey
}
})
2014-01-05 20:07:11 +04:00
})
})
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.getDecryptedMessage = function (msgKey, encryptedData) {
// console.log(dT(), 'get decrypted start')
return this.getMsgKeyIv(msgKey, false).then(function (keyIv) {
// console.log(dT(), 'after msg key iv')
return CryptoWorker.aesDecrypt(encryptedData, keyIv[0], keyIv[1])
})
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.sendEncryptedRequest = function (message, options) {
var self = this
options = options || {}
// console.log(dT(), 'Send encrypted'/*, message*/)
// console.trace()
var data = new TLSerialization({startMaxLength: message.body.length + 64})
2014-01-05 20:07:11 +04:00
data.storeIntBytes(this.serverSalt, 64, 'salt')
data.storeIntBytes(this.sessionID, 64, 'session_id')
2014-01-05 20:07:11 +04:00
data.storeLong(message.msg_id, 'message_id')
data.storeInt(message.seq_no, 'seq_no')
2014-01-05 20:07:11 +04:00
data.storeInt(message.body.length, 'message_data_length')
data.storeRawBytes(message.body, 'message_data')
2014-01-05 20:07:11 +04:00
return this.getEncryptedMessage(data.getBuffer()).then(function (encryptedResult) {
// console.log(dT(), 'Got encrypted out message'/*, encryptedResult*/)
var request = new TLSerialization({startMaxLength: encryptedResult.bytes.byteLength + 256})
request.storeIntBytes(self.authKeyID, 64, 'auth_key_id')
request.storeIntBytes(encryptedResult.msgKey, 128, 'msg_key')
request.storeRawBytes(encryptedResult.bytes, 'encrypted_data')
2014-01-05 20:07:11 +04:00
var requestData = xhrSendBuffer ? request.getBuffer() : request.getArray()
2014-03-21 14:07:46 +04:00
var requestPromise
var url = MtpDcConfigurator.chooseServer(self.dcID, self.upload)
var baseError = {code: 406, type: 'NETWORK_BAD_RESPONSE', url: url}
try {
options = angular.extend(options || {}, {
responseType: 'arraybuffer',
transformRequest: null
})
requestPromise = $http.post(url, requestData, options)
} catch (e) {
requestPromise = $q.reject(e)
}
return requestPromise.then(
function (result) {
if (!result.data || !result.data.byteLength) {
return $q.reject(baseError)
}
return result
},
function (error) {
if (!error.message && !error.type) {
error = angular.extend(baseError, {type: 'NETWORK_BAD_REQUEST', originalError: error})
}
return $q.reject(error)
2014-08-01 09:37:30 +01:00
}
)
})
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.parseResponse = function (responseBuffer) {
// console.log(dT(), 'Start parsing response')
var self = this
2014-01-05 20:07:11 +04:00
var deserializer = new TLDeserialization(responseBuffer)
2014-01-05 20:07:11 +04:00
var authKeyID = deserializer.fetchIntBytes(64, false, 'auth_key_id')
if (!bytesCmp(authKeyID, this.authKeyID)) {
throw new Error('[MT] Invalid server auth_key_id: ' + bytesToHex(authKeyID))
}
var msgKey = deserializer.fetchIntBytes(128, true, 'msg_key')
var encryptedData = deserializer.fetchRawBytes(responseBuffer.byteLength - deserializer.getOffset(), true, 'encrypted_data')
2014-01-05 20:07:11 +04:00
return this.getDecryptedMessage(msgKey, encryptedData).then(function (dataWithPadding) {
// console.log(dT(), 'after decrypt')
var deserializer = new TLDeserialization(dataWithPadding, {mtproto: true})
2014-01-05 20:07:11 +04:00
var salt = deserializer.fetchIntBytes(64, false, 'salt')
var sessionID = deserializer.fetchIntBytes(64, false, 'session_id')
var messageID = deserializer.fetchLong('message_id')
2014-01-05 20:07:11 +04:00
if (!bytesCmp(sessionID, self.sessionID) &&
(!self.prevSessionID || !bytesCmp(sessionID, self.prevSessionID))) {
console.warn('Sessions', sessionID, self.sessionID, self.prevSessionID)
throw new Error('[MT] Invalid server session_id: ' + bytesToHex(sessionID))
}
var seqNo = deserializer.fetchInt('seq_no')
2014-01-05 20:07:11 +04:00
var offset = deserializer.getOffset()
var totalLength = dataWithPadding.byteLength
2014-01-05 20:07:11 +04:00
var messageBodyLength = deserializer.fetchInt('message_data[length]')
if ((messageBodyLength % 4) ||
messageBodyLength > totalLength - offset) {
throw new Error('[MT] Invalid body length: ' + messageBodyLength)
}
var messageBody = deserializer.fetchRawBytes(messageBodyLength, true, 'message_data')
var offset = deserializer.getOffset()
var paddingLength = totalLength - offset
if (paddingLength < 0 || paddingLength > 15) {
throw new Error('[MT] Invalid padding length: ' + paddingLength)
}
var hashData = convertToUint8Array(dataWithPadding).subarray(0, offset)
2014-01-05 20:07:11 +04:00
return CryptoWorker.sha1Hash(hashData).then(function (dataHash) {
if (!bytesCmp(msgKey, bytesFromArrayBuffer(dataHash).slice(-16))) {
console.warn(msgKey, bytesFromArrayBuffer(dataHash))
throw new Error('[MT] server msgKey mismatch')
}
2014-01-05 20:07:11 +04:00
var buffer = bytesToArrayBuffer(messageBody)
var deserializerOptions = {
mtproto: true,
override: {
mt_message: function (result, field) {
result.msg_id = this.fetchLong(field + '[msg_id]')
result.seqno = this.fetchInt(field + '[seqno]')
result.bytes = this.fetchInt(field + '[bytes]')
var offset = this.getOffset()
try {
result.body = this.fetchObject('Object', field + '[body]')
} catch (e) {
console.error(dT(), 'parse error', e.message, e.stack)
result.body = {_: 'parse_error', error: e}
}
if (this.offset != offset + result.bytes) {
// console.warn(dT(), 'set offset', this.offset, offset, result.bytes)
// console.log(dT(), result)
this.offset = offset + result.bytes
}
// console.log(dT(), 'override message', result)
},
mt_rpc_result: function (result, field) {
result.req_msg_id = this.fetchLong(field + '[req_msg_id]')
var sentMessage = self.sentMessages[result.req_msg_id]
var type = sentMessage && sentMessage.resultType || 'Object'
if (result.req_msg_id && !sentMessage) {
// console.warn(dT(), 'Result for unknown message', result)
return
}
result.result = this.fetchObject(type, field + '[result]')
// console.log(dT(), 'override rpc_result', sentMessage, type, result)
2015-10-23 01:12:44 +02:00
}
}
}
var deserializer = new TLDeserialization(buffer, deserializerOptions)
var response = deserializer.fetchObject('', 'INPUT')
2014-01-05 20:07:11 +04:00
return {
response: response,
messageID: messageID,
sessionID: sessionID,
seqNo: seqNo
}
})
})
2014-01-05 20:07:11 +04:00
}
MtpNetworker.prototype.applyServerSalt = function (newServerSalt) {
var serverSalt = longToBytes(newServerSalt)
var storeObj = {}
storeObj['dc' + this.dcID + '_server_salt'] = bytesToHex(serverSalt)
Storage.set(storeObj)
2014-01-05 20:07:11 +04:00
this.serverSalt = serverSalt
return true
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.sheduleRequest = function (delay) {
if (this.offline) {
this.checkConnection('forced shedule')
}
var nextReq = tsNow() + delay
if (delay && this.nextReq && this.nextReq <= nextReq) {
return false
}
// console.log(dT(), 'shedule req', delay)
// console.trace()
2014-01-05 20:07:11 +04:00
$timeout.cancel(this.nextReqPromise)
if (delay > 0) {
this.nextReqPromise = $timeout(this.performSheduledRequest.bind(this), delay || 0)
} else {
setZeroTimeout(this.performSheduledRequest.bind(this))
}
this.nextReq = nextReq
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.ackMessage = function (msgID) {
// console.log('ack message', msgID)
this.pendingAcks.push(msgID)
this.sheduleRequest(30000)
2014-01-05 20:07:11 +04:00
}
MtpNetworker.prototype.reqResendMessage = function (msgID) {
console.log(dT(), 'Req resend', msgID)
this.pendingResends.push(msgID)
this.sheduleRequest(100)
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.cleanupSent = function () {
var self = this
var notEmpty = false
// console.log('clean start', this.dcID/*, this.sentMessages*/)
angular.forEach(this.sentMessages, function (message, msgID) {
// console.log('clean iter', msgID, message)
if (message.notContentRelated && self.pendingMessages[msgID] === undefined) {
// console.log('clean notContentRelated', msgID)
delete self.sentMessages[msgID]
}
else if (message.container) {
for (var i = 0; i < message.inner.length; i++) {
if (self.sentMessages[message.inner[i]] !== undefined) {
// console.log('clean failed, found', msgID, message.inner[i], self.sentMessages[message.inner[i]].seq_no)
notEmpty = true
return
}
2014-01-25 20:40:48 +04:00
}
// console.log('clean container', msgID)
delete self.sentMessages[msgID]
} else {
notEmpty = true
}
})
return !notEmpty
}
MtpNetworker.prototype.processMessageAck = function (messageID) {
var sentMessage = this.sentMessages[messageID]
if (sentMessage && !sentMessage.acked) {
delete sentMessage.body
sentMessage.acked = true
return true
}
2014-01-05 20:07:11 +04:00
return false
}
2014-01-05 20:07:11 +04:00
MtpNetworker.prototype.processError = function (rawError) {
var matches = (rawError.error_message || '').match(/^([A-Z_0-9]+\b)(: (.+))?/) || []
rawError.error_code = uintToInt(rawError.error_code)
2014-10-30 11:38:07 +03:00
return {
code: !rawError.error_code || rawError.error_code <= 0 ? 500 : rawError.error_code,
type: matches[1] || 'UNKNOWN',
description: matches[3] || ('CODE#' + rawError.error_code + ' ' + rawError.error_message),
originalError: rawError
}
}
MtpNetworker.prototype.processMessage = function (message, messageID, sessionID) {
var msgidInt = parseInt(messageID.toString(10).substr(0, -10), 10)
if (msgidInt % 2) {
console.warn('[MT] Server even message id: ', messageID, message)
return
}
// console.log('process message', message, messageID, sessionID)
switch (message._) {
case 'msg_container':
var len = message.messages.length
for (var i = 0; i < len; i++) {
this.processMessage(message.messages[i], message.messages[i].msg_id, sessionID)
}
break
case 'bad_server_salt':
console.log(dT(), 'Bad server salt', message)
var sentMessage = this.sentMessages[message.bad_msg_id]
if (!sentMessage || sentMessage.seq_no != message.bad_msg_seqno) {
console.log(message.bad_msg_id, message.bad_msg_seqno)
throw new Error('[MT] Bad server salt for invalid message')
2014-10-30 11:38:07 +03:00
}
2014-01-05 20:07:11 +04:00
this.applyServerSalt(message.new_server_salt)
this.pushResend(message.bad_msg_id)
this.ackMessage(messageID)
break
case 'bad_msg_notification':
console.log(dT(), 'Bad msg notification', message)
var sentMessage = this.sentMessages[message.bad_msg_id]
if (!sentMessage || sentMessage.seq_no != message.bad_msg_seqno) {
console.log(message.bad_msg_id, message.bad_msg_seqno)
throw new Error('[MT] Bad msg notification for invalid message')
}
2014-01-05 20:07:11 +04:00
if (message.error_code == 16 || message.error_code == 17) {
if (MtpTimeManager.applyServerTime(
bigStringInt(messageID).shiftRight(32).toString(10)
)) {
console.log(dT(), 'Update session')
this.updateSession()
}
var badMessage = this.updateSentMessage(message.bad_msg_id)
this.pushResend(badMessage.msg_id)
this.ackMessage(messageID)
}
break
case 'message':
if (this.lastServerMessages.indexOf(messageID) != -1) {
// console.warn('[MT] Server same messageID: ', messageID)
this.ackMessage(messageID)
return
}
this.lastServerMessages.push(messageID)
if (this.lastServerMessages.length > 100) {
this.lastServerMessages.shift()
}
this.processMessage(message.body, message.msg_id, sessionID)
break
case 'new_session_created':
this.ackMessage(messageID)
this.processMessageAck(message.first_msg_id)
this.applyServerSalt(message.server_salt)
var self = this
Storage.get('dc').then(function (baseDcID) {
if (baseDcID == self.dcID && !self.upload && updatesProcessor) {
2016-07-06 19:20:39 +03:00
updatesProcessor(message, true)
2014-01-05 20:07:11 +04:00
}
})
break
case 'msgs_ack':
for (var i = 0; i < message.msg_ids.length; i++) {
this.processMessageAck(message.msg_ids[i])
}
break
case 'msg_detailed_info':
if (!this.sentMessages[message.msg_id]) {
this.ackMessage(message.answer_msg_id)
break
}
case 'msg_new_detailed_info':
// this.ackMessage(message.answer_msg_id)
this.reqResendMessage(message.answer_msg_id)
break
case 'msgs_state_info':
this.ackMessage(message.answer_msg_id)
if (this.lastResendReq && this.lastResendReq.req_msg_id == message.req_msg_id && this.pendingResends.length) {
var i, badMsgID, pos
for (i = 0; i < this.lastResendReq.resend_msg_ids.length; i++) {
badMsgID = this.lastResendReq.resend_msg_ids[i]
pos = this.pendingResends.indexOf(badMsgID)
if (pos != -1) {
this.pendingResends.splice(pos, 1)
}
}
}
break
case 'rpc_result':
this.ackMessage(messageID)
var sentMessageID = message.req_msg_id
var sentMessage = this.sentMessages[sentMessageID]
this.processMessageAck(sentMessageID)
if (sentMessage) {
var deferred = sentMessage.deferred
if (message.result._ == 'rpc_error') {
var error = this.processError(message.result)
console.log(dT(), 'Rpc error', error)
if (deferred) {
deferred.reject(error)
}
} else {
if (deferred) {
if (Config.Modes.debug) {
console.log(dT(), 'Rpc response', message.result)
} else {
var dRes = message.result._
if (!dRes) {
if (message.result.length > 5) {
dRes = '[..' + message.result.length + '..]'
} else {
dRes = message.result
}
2014-07-24 12:16:52 +04:00
}
console.log(dT(), 'Rpc response', dRes)
2014-07-24 12:16:52 +04:00
}
sentMessage.deferred.resolve(message.result)
}
if (sentMessage.isAPI) {
this.connectionInited = true
2014-01-27 21:47:04 +04:00
}
2014-01-05 20:07:11 +04:00
}
delete this.sentMessages[sentMessageID]
2014-01-05 20:07:11 +04:00
}
break
2014-01-05 20:07:11 +04:00
default:
this.ackMessage(messageID)
2014-01-05 20:07:11 +04:00
// console.log('Update', message)
if (updatesProcessor) {
2016-07-06 19:20:39 +03:00
updatesProcessor(message, true)
}
break
2014-01-05 20:07:11 +04:00
}
}
2014-01-05 20:07:11 +04:00
function startAll () {
if (akStopped) {
akStopped = false
2016-07-06 19:20:39 +03:00
updatesProcessor({_: 'new_session_created'}, true)
}
}
function stopAll () {
akStopped = true
2014-01-05 20:07:11 +04:00
}
return {
getNetworker: function (dcID, authKey, serverSalt, options) {
return new MtpNetworker(dcID, authKey, serverSalt, options)
},
setUpdatesProcessor: function (callback) {
updatesProcessor = callback
},
stopAll: stopAll,
startAll: startAll
2014-11-07 19:07:54 +03:00
}
})