2014-06-20 19:34:14 +04:00
|
|
|
/*!
|
2016-04-22 20:10:16 +03:00
|
|
|
* Webogram v0.5.4 - 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
angular.module('izhukov.mtproto.wrapper', ['izhukov.utils', 'izhukov.mtproto'])
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
.factory('MtpApiManager', function (Storage, MtpAuthorizer, MtpNetworkerFactory, MtpSingleInstanceService, AppRuntimeManager, ErrorService, qSync, $rootScope, $q, TelegramMeWebService) {
|
|
|
|
var cachedNetworkers = {}
|
|
|
|
var cachedUploadNetworkers = {}
|
|
|
|
var cachedExportPromise = {}
|
|
|
|
var baseDcID = false
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var telegramMeNotified
|
2014-12-15 18:05:38 -08:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
MtpSingleInstanceService.start()
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
Storage.get('dc').then(function (dcID) {
|
|
|
|
if (dcID) {
|
|
|
|
baseDcID = dcID
|
|
|
|
}
|
|
|
|
})
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function telegramMeNotify (newValue) {
|
|
|
|
if (telegramMeNotified !== newValue) {
|
|
|
|
telegramMeNotified = newValue
|
|
|
|
TelegramMeWebService.setAuthorized(telegramMeNotified)
|
|
|
|
}
|
2014-12-15 18:05:38 -08:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
|
|
|
|
function mtpSetUserAuth (dcID, userAuth) {
|
|
|
|
var fullUserAuth = angular.extend({dcID: dcID}, userAuth)
|
|
|
|
Storage.set({
|
|
|
|
dc: dcID,
|
|
|
|
user_auth: fullUserAuth
|
|
|
|
})
|
|
|
|
telegramMeNotify(true)
|
|
|
|
$rootScope.$broadcast('user_auth', fullUserAuth)
|
|
|
|
|
|
|
|
baseDcID = dcID
|
2014-11-12 20:07:34 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
|
|
|
|
function mtpLogOut () {
|
|
|
|
var storageKeys = []
|
|
|
|
for (var dcID = 1; dcID <= 5; dcID++) {
|
|
|
|
storageKeys.push('dc' + dcID + '_auth_key')
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
return Storage.get(storageKeys).then(function (storageResult) {
|
|
|
|
var logoutPromises = []
|
|
|
|
for (var i = 0; i < storageResult.length; i++) {
|
|
|
|
if (storageResult[i]) {
|
|
|
|
logoutPromises.push(mtpInvokeApi('auth.logOut', {}, {dcID: i + 1}))
|
2016-06-07 21:32:27 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
}
|
|
|
|
return $q.all(logoutPromises).then(function () {
|
|
|
|
Storage.remove('dc', 'user_auth')
|
|
|
|
baseDcID = false
|
|
|
|
telegramMeNotify(false)
|
|
|
|
return mtpClearStorage()
|
|
|
|
}, function (error) {
|
|
|
|
storageKeys.push('dc', 'user_auth')
|
|
|
|
Storage.remove(storageKeys)
|
|
|
|
baseDcID = false
|
|
|
|
error.handled = true
|
|
|
|
telegramMeNotify(false)
|
|
|
|
return mtpClearStorage()
|
|
|
|
})
|
|
|
|
})
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function mtpClearStorage () {
|
|
|
|
var saveKeys = []
|
|
|
|
for (var dcID = 1; dcID <= 5; dcID++) {
|
|
|
|
saveKeys.push('dc' + dcID + '_auth_key')
|
|
|
|
saveKeys.push('t_dc' + dcID + '_auth_key')
|
|
|
|
}
|
|
|
|
Storage.noPrefix()
|
|
|
|
Storage.get(saveKeys).then(function (values) {
|
|
|
|
Storage.clear().then(function () {
|
|
|
|
var restoreObj = {}
|
|
|
|
angular.forEach(saveKeys, function (key, i) {
|
|
|
|
var value = values[i]
|
|
|
|
if (value !== false && value !== undefined) {
|
|
|
|
restoreObj[key] = value
|
|
|
|
}
|
|
|
|
})
|
|
|
|
Storage.noPrefix()
|
|
|
|
return Storage.set(restoreObj)
|
|
|
|
})
|
|
|
|
})
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function mtpGetNetworker (dcID, options) {
|
|
|
|
options = options || {}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var cache = (options.fileUpload || options.fileDownload)
|
|
|
|
? cachedUploadNetworkers
|
|
|
|
: cachedNetworkers
|
|
|
|
if (!dcID) {
|
|
|
|
throw new Exception('get Networker without dcID')
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
if (cache[dcID] !== undefined) {
|
2016-06-28 19:40:53 +03:00
|
|
|
return qSync.when(cache[dcID])
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var akk = 'dc' + dcID + '_auth_key'
|
|
|
|
var ssk = 'dc' + dcID + '_server_salt'
|
|
|
|
|
|
|
|
return Storage.get(akk, ssk).then(function (result) {
|
|
|
|
if (cache[dcID] !== undefined) {
|
|
|
|
return cache[dcID]
|
2016-06-07 21:32:27 +03:00
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var authKeyHex = result[0]
|
|
|
|
var serverSaltHex = result[1]
|
|
|
|
// console.log('ass', dcID, authKeyHex, serverSaltHex)
|
|
|
|
if (authKeyHex && authKeyHex.length == 512) {
|
|
|
|
if (!serverSaltHex || serverSaltHex.length != 16) {
|
|
|
|
serverSaltHex = 'AAAAAAAAAAAAAAAA'
|
|
|
|
}
|
|
|
|
var authKey = bytesFromHex(authKeyHex)
|
|
|
|
var serverSalt = bytesFromHex(serverSaltHex)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return cache[dcID] = MtpNetworkerFactory.getNetworker(dcID, authKey, serverSalt, options)
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
if (!options.createNetworker) {
|
|
|
|
return $q.reject({type: 'AUTH_KEY_EMPTY', code: 401})
|
|
|
|
}
|
|
|
|
|
|
|
|
return MtpAuthorizer.auth(dcID).then(function (auth) {
|
|
|
|
var storeObj = {}
|
|
|
|
storeObj[akk] = bytesToHex(auth.authKey)
|
|
|
|
storeObj[ssk] = bytesToHex(auth.serverSalt)
|
|
|
|
Storage.set(storeObj)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return cache[dcID] = MtpNetworkerFactory.getNetworker(dcID, auth.authKey, auth.serverSalt, options)
|
|
|
|
}, function (error) {
|
|
|
|
console.log('Get networker error', error, error.stack)
|
|
|
|
return $q.reject(error)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function mtpInvokeApi (method, params, options) {
|
|
|
|
options = options || {}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var deferred = $q.defer()
|
|
|
|
var rejectPromise = function (error) {
|
2014-06-20 19:34:14 +04:00
|
|
|
if (!error) {
|
2016-06-28 19:40:53 +03:00
|
|
|
error = {type: 'ERROR_EMPTY'}
|
2014-06-20 19:34:14 +04:00
|
|
|
} else if (!angular.isObject(error)) {
|
2016-06-28 19:40:53 +03:00
|
|
|
error = {message: error}
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
deferred.reject(error)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
|
|
|
if (!options.noErrorBox) {
|
2016-06-28 19:40:53 +03:00
|
|
|
error.input = method
|
|
|
|
error.stack = stack || (error.originalError && error.originalError.stack) || error.stack || (new Error()).stack
|
2014-06-20 19:34:14 +04:00
|
|
|
setTimeout(function () {
|
|
|
|
if (!error.handled) {
|
2015-04-08 21:25:39 +03:00
|
|
|
if (error.code == 401) {
|
|
|
|
mtpLogOut()['finally'](function () {
|
|
|
|
if (location.protocol == 'http:' &&
|
2016-06-28 19:40:53 +03:00
|
|
|
!Config.Modes.http &&
|
|
|
|
Config.App.domains.indexOf(location.hostname) != -1) {
|
|
|
|
location.href = location.href.replace(/^http:/, 'https:')
|
2015-04-08 21:25:39 +03:00
|
|
|
} else {
|
2016-06-28 19:40:53 +03:00
|
|
|
location.hash = '/login'
|
|
|
|
AppRuntimeManager.reload()
|
2015-04-08 21:25:39 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
})
|
2015-04-08 21:25:39 +03:00
|
|
|
} else {
|
2016-06-28 19:40:53 +03:00
|
|
|
ErrorService.show({error: error})
|
2015-04-08 21:25:39 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
error.handled = true
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
}, 100)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
dcID,
|
2016-06-28 19:40:53 +03:00
|
|
|
networkerPromise
|
|
|
|
|
|
|
|
var cachedNetworker
|
|
|
|
var stack = (new Error()).stack || 'empty stack'
|
|
|
|
var performRequest = function (networker) {
|
|
|
|
return (cachedNetworker = networker).wrapApiCall(method, params, options).then(
|
|
|
|
function (result) {
|
|
|
|
deferred.resolve(result)
|
|
|
|
},
|
|
|
|
function (error) {
|
|
|
|
console.error(dT(), 'Error', error.code, error.type, baseDcID, dcID)
|
|
|
|
if (error.code == 401 && baseDcID == dcID) {
|
|
|
|
Storage.remove('dc', 'user_auth')
|
|
|
|
telegramMeNotify(false)
|
|
|
|
rejectPromise(error)
|
|
|
|
}
|
|
|
|
else if (error.code == 401 && baseDcID && dcID != baseDcID) {
|
|
|
|
if (cachedExportPromise[dcID] === undefined) {
|
|
|
|
var exportDeferred = $q.defer()
|
|
|
|
|
|
|
|
mtpInvokeApi('auth.exportAuthorization', {dc_id: dcID}, {noErrorBox: true}).then(function (exportedAuth) {
|
|
|
|
mtpInvokeApi('auth.importAuthorization', {
|
|
|
|
id: exportedAuth.id,
|
|
|
|
bytes: exportedAuth.bytes
|
|
|
|
}, {dcID: dcID, noErrorBox: true}).then(function () {
|
|
|
|
exportDeferred.resolve()
|
|
|
|
}, function (e) {
|
|
|
|
exportDeferred.reject(e)
|
|
|
|
})
|
2014-06-20 19:34:14 +04:00
|
|
|
}, function (e) {
|
2016-06-28 19:40:53 +03:00
|
|
|
exportDeferred.reject(e)
|
2014-06-20 19:34:14 +04:00
|
|
|
})
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
cachedExportPromise[dcID] = exportDeferred.promise
|
|
|
|
}
|
|
|
|
|
|
|
|
cachedExportPromise[dcID].then(function () {
|
|
|
|
(cachedNetworker = networker).wrapApiCall(method, params, options).then(function (result) {
|
|
|
|
deferred.resolve(result)
|
|
|
|
}, rejectPromise)
|
|
|
|
}, rejectPromise)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
else if (error.code == 303) {
|
|
|
|
var newDcID = error.type.match(/^(PHONE_MIGRATE_|NETWORK_MIGRATE_|USER_MIGRATE_)(\d+)/)[2]
|
|
|
|
if (newDcID != dcID) {
|
|
|
|
if (options.dcID) {
|
|
|
|
options.dcID = newDcID
|
|
|
|
} else {
|
|
|
|
Storage.set({dc: baseDcID = newDcID})
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
mtpGetNetworker(newDcID, options).then(function (networker) {
|
|
|
|
networker.wrapApiCall(method, params, options).then(function (result) {
|
|
|
|
deferred.resolve(result)
|
|
|
|
}, rejectPromise)
|
|
|
|
}, rejectPromise)
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
else if (!options.rawError && error.code == 420) {
|
|
|
|
var waitTime = error.type.match(/^FLOOD_WAIT_(\d+)/)[1] || 10
|
|
|
|
if (waitTime > (options.timeout || 60)) {
|
|
|
|
return rejectPromise(error)
|
|
|
|
}
|
|
|
|
setTimeout(function () {
|
|
|
|
performRequest(cachedNetworker)
|
|
|
|
}, waitTime * 1000)
|
2015-11-02 13:44:05 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
else if (!options.rawError && (error.code == 500 || error.type == 'MSG_WAIT_FAILED')) {
|
|
|
|
var now = tsNow()
|
|
|
|
if (options.stopTime) {
|
|
|
|
if (now >= options.stopTime) {
|
|
|
|
return rejectPromise(error)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
options.stopTime = now + (options.timeout !== undefined ? options.timeout : 10) * 1000
|
2015-11-02 13:44:05 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
options.waitTime = options.waitTime ? Math.min(60, options.waitTime * 1.5) : 1
|
|
|
|
setTimeout(function () {
|
|
|
|
performRequest(cachedNetworker)
|
|
|
|
}, options.waitTime * 1000)
|
|
|
|
}else {
|
|
|
|
rejectPromise(error)
|
2015-11-02 13:44:05 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
})
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
if (dcID = (options.dcID || baseDcID)) {
|
|
|
|
mtpGetNetworker(dcID, options).then(performRequest, rejectPromise)
|
|
|
|
} else {
|
|
|
|
Storage.get('dc').then(function (baseDcID) {
|
|
|
|
mtpGetNetworker(dcID = baseDcID || 2, options).then(performRequest, rejectPromise)
|
|
|
|
})
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return deferred.promise
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function mtpGetUserID () {
|
|
|
|
return Storage.get('user_auth').then(function (auth) {
|
|
|
|
telegramMeNotify(auth && auth.id > 0 || false)
|
|
|
|
return auth.id || 0
|
|
|
|
})
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function getBaseDcID () {
|
|
|
|
return baseDcID || false
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return {
|
|
|
|
getBaseDcID: getBaseDcID,
|
|
|
|
getUserID: mtpGetUserID,
|
|
|
|
invokeApi: mtpInvokeApi,
|
|
|
|
getNetworker: mtpGetNetworker,
|
|
|
|
setUserAuth: mtpSetUserAuth,
|
|
|
|
logOut: mtpLogOut
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
})
|
|
|
|
|
|
|
|
.factory('MtpApiFileManager', function (MtpApiManager, $q, qSync, FileManager, IdbFileStorage, TmpfsFileStorage, MemoryFileStorage, WebpManager) {
|
|
|
|
var cachedFs = false
|
|
|
|
var cachedFsPromise = false
|
|
|
|
var cachedSavePromises = {}
|
|
|
|
var cachedDownloadPromises = {}
|
|
|
|
var cachedDownloads = {}
|
|
|
|
|
|
|
|
var downloadPulls = {}
|
|
|
|
var downloadActives = {}
|
|
|
|
|
|
|
|
function downloadRequest (dcID, cb, activeDelta) {
|
|
|
|
if (downloadPulls[dcID] === undefined) {
|
|
|
|
downloadPulls[dcID] = []
|
|
|
|
downloadActives[dcID] = 0
|
|
|
|
}
|
|
|
|
var downloadPull = downloadPulls[dcID]
|
|
|
|
var deferred = $q.defer()
|
|
|
|
downloadPull.push({cb: cb, deferred: deferred, activeDelta: activeDelta})
|
|
|
|
setZeroTimeout(function () {
|
|
|
|
downloadCheck(dcID)
|
|
|
|
})
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return deferred.promise
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var index = 0
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function downloadCheck (dcID) {
|
|
|
|
var downloadPull = downloadPulls[dcID]
|
|
|
|
var downloadLimit = dcID == 'upload' ? 11 : 5
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
if (downloadActives[dcID] >= downloadLimit || !downloadPull || !downloadPull.length) {
|
|
|
|
return false
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var data = downloadPull.shift()
|
|
|
|
var activeDelta = data.activeDelta || 1
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
downloadActives[dcID] += activeDelta
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var a = index++
|
|
|
|
data.cb()
|
|
|
|
.then(function (result) {
|
|
|
|
downloadActives[dcID] -= activeDelta
|
|
|
|
downloadCheck(dcID)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
data.deferred.resolve(result)
|
|
|
|
}, function (error) {
|
|
|
|
downloadActives[dcID] -= activeDelta
|
|
|
|
downloadCheck(dcID)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
data.deferred.reject(error)
|
|
|
|
})
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function getFileName (location) {
|
|
|
|
switch (location._) {
|
|
|
|
case 'inputDocumentFileLocation':
|
|
|
|
var fileName = (location.file_name || '').split('.', 2)
|
|
|
|
var ext = fileName[1] || ''
|
|
|
|
if (location.sticker && !WebpManager.isWebpSupported()) {
|
|
|
|
ext += '.png'
|
|
|
|
}
|
|
|
|
return fileName[0] + '_' + location.id + '.' + ext
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
default:
|
|
|
|
if (!location.volume_id) {
|
|
|
|
console.trace('Empty location', location)
|
|
|
|
}
|
|
|
|
var ext = 'jpg'
|
|
|
|
if (location.sticker) {
|
|
|
|
ext = WebpManager.isWebpSupported() ? 'webp' : 'png'
|
|
|
|
}
|
|
|
|
return location.volume_id + '_' + location.local_id + '_' + location.secret + '.' + ext
|
2016-01-18 15:43:22 +03:00
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function getTempFileName (file) {
|
|
|
|
var size = file.size || -1
|
|
|
|
var random = nextRandomInt(0xFFFFFFFF)
|
|
|
|
return '_temp' + random + '_' + size
|
2014-07-21 19:39:28 +04:00
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function getCachedFile (location) {
|
|
|
|
if (!location) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
var fileName = getFileName(location)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return cachedDownloads[fileName] || false
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function getFileStorage () {
|
|
|
|
if (!Config.Modes.memory_only) {
|
|
|
|
if (TmpfsFileStorage.isAvailable()) {
|
|
|
|
return TmpfsFileStorage
|
2015-01-02 13:31:29 +01:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
if (IdbFileStorage.isAvailable()) {
|
|
|
|
return IdbFileStorage
|
2015-06-30 00:37:01 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
}
|
|
|
|
return MemoryFileStorage
|
|
|
|
}
|
2014-10-29 21:54:17 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function saveSmallFile (location, bytes) {
|
|
|
|
var fileName = getFileName(location)
|
|
|
|
var mimeType = 'image/jpeg'
|
2014-10-29 21:54:17 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
if (!cachedSavePromises[fileName]) {
|
|
|
|
cachedSavePromises[fileName] = getFileStorage().saveFile(fileName, bytes).then(function (blob) {
|
|
|
|
return cachedDownloads[fileName] = blob
|
|
|
|
}, function (error) {
|
|
|
|
delete cachedSavePromises[fileName]
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return cachedSavePromises[fileName]
|
2014-07-21 19:39:28 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function downloadSmallFile (location) {
|
|
|
|
if (!FileManager.isAvailable()) {
|
|
|
|
return $q.reject({type: 'BROWSER_BLOB_NOT_SUPPORTED'})
|
|
|
|
}
|
|
|
|
var fileName = getFileName(location)
|
|
|
|
var mimeType = location.sticker ? 'image/webp' : 'image/jpeg'
|
|
|
|
var cachedPromise = cachedSavePromises[fileName] || cachedDownloadPromises[fileName]
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
if (cachedPromise) {
|
|
|
|
return cachedPromise
|
2015-06-30 14:17:32 +03:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var fileStorage = getFileStorage()
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return cachedDownloadPromises[fileName] = fileStorage.getFile(fileName).then(function (blob) {
|
|
|
|
return cachedDownloads[fileName] = blob
|
|
|
|
}, function () {
|
|
|
|
var downloadPromise = downloadRequest(location.dc_id, function () {
|
|
|
|
var inputLocation = location
|
|
|
|
if (!inputLocation._ || inputLocation._ == 'fileLocation') {
|
|
|
|
inputLocation = angular.extend({}, location, {_: 'inputFileLocation'})
|
|
|
|
}
|
|
|
|
// console.log('next small promise')
|
|
|
|
return MtpApiManager.invokeApi('upload.getFile', {
|
|
|
|
location: inputLocation,
|
|
|
|
offset: 0,
|
|
|
|
limit: 1024 * 1024
|
|
|
|
}, {
|
|
|
|
dcID: location.dc_id,
|
|
|
|
fileDownload: true,
|
|
|
|
createNetworker: true,
|
|
|
|
noErrorBox: true
|
|
|
|
})
|
|
|
|
})
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var processDownloaded = function (bytes) {
|
|
|
|
if (!location.sticker || WebpManager.isWebpSupported()) {
|
|
|
|
return qSync.when(bytes)
|
|
|
|
}
|
|
|
|
return WebpManager.getPngBlobFromWebp(bytes)
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return fileStorage.getFileWriter(fileName, mimeType).then(function (fileWriter) {
|
|
|
|
return downloadPromise.then(function (result) {
|
|
|
|
return processDownloaded(result.bytes).then(function (proccessedResult) {
|
|
|
|
return FileManager.write(fileWriter, proccessedResult).then(function () {
|
|
|
|
return cachedDownloads[fileName] = fileWriter.finalize()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
2014-06-20 19:34:14 +04:00
|
|
|
})
|
2016-06-28 19:40:53 +03:00
|
|
|
})
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function getDownloadedFile (location, size) {
|
|
|
|
var fileStorage = getFileStorage()
|
|
|
|
var fileName = getFileName(location)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return fileStorage.getFile(fileName, size)
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function downloadFile (dcID, location, size, options) {
|
|
|
|
if (!FileManager.isAvailable()) {
|
|
|
|
return $q.reject({type: 'BROWSER_BLOB_NOT_SUPPORTED'})
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
options = options || {}
|
|
|
|
|
|
|
|
var processSticker = false
|
|
|
|
if (location.sticker && !WebpManager.isWebpSupported()) {
|
|
|
|
if (options.toFileEntry || size > 524288) {
|
|
|
|
delete location.sticker
|
|
|
|
} else {
|
|
|
|
processSticker = true
|
|
|
|
options.mime = 'image/png'
|
2015-06-30 14:17:32 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// console.log(dT(), 'Dload file', dcID, location, size)
|
|
|
|
var fileName = getFileName(location)
|
|
|
|
var toFileEntry = options.toFileEntry || null
|
|
|
|
var cachedPromise = cachedSavePromises[fileName] || cachedDownloadPromises[fileName]
|
|
|
|
|
|
|
|
var fileStorage = getFileStorage()
|
|
|
|
|
|
|
|
// console.log(dT(), 'fs', fileStorage.name, fileName, cachedPromise)
|
|
|
|
|
|
|
|
if (cachedPromise) {
|
|
|
|
if (toFileEntry) {
|
|
|
|
return cachedPromise.then(function (blob) {
|
|
|
|
return FileManager.copy(blob, toFileEntry)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return cachedPromise
|
|
|
|
}
|
|
|
|
|
|
|
|
var deferred = $q.defer()
|
|
|
|
var canceled = false
|
|
|
|
var resolved = false
|
|
|
|
var mimeType = options.mime || 'image/jpeg',
|
|
|
|
cacheFileWriter
|
|
|
|
var errorHandler = function (error) {
|
|
|
|
deferred.reject(error)
|
|
|
|
errorHandler = angular.noop
|
|
|
|
if (cacheFileWriter &&
|
|
|
|
(!error || error.type != 'DOWNLOAD_CANCELED')) {
|
|
|
|
cacheFileWriter.truncate(0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fileStorage.getFile(fileName, size).then(function (blob) {
|
|
|
|
if (toFileEntry) {
|
|
|
|
FileManager.copy(blob, toFileEntry).then(function () {
|
|
|
|
deferred.resolve()
|
|
|
|
}, errorHandler)
|
|
|
|
} else {
|
|
|
|
deferred.resolve(cachedDownloads[fileName] = blob)
|
|
|
|
}
|
|
|
|
}, function () {
|
|
|
|
var fileWriterPromise = toFileEntry ? FileManager.getFileWriter(toFileEntry) : fileStorage.getFileWriter(fileName, mimeType)
|
|
|
|
|
|
|
|
var processDownloaded = function (bytes) {
|
|
|
|
if (!processSticker) {
|
|
|
|
return qSync.when(bytes)
|
2015-02-04 15:30:55 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
return WebpManager.getPngBlobFromWebp(bytes)
|
2014-10-18 21:58:54 +02:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
|
|
|
|
fileWriterPromise.then(function (fileWriter) {
|
|
|
|
cacheFileWriter = fileWriter
|
|
|
|
var limit = 524288,
|
|
|
|
offset
|
|
|
|
var startOffset = 0
|
|
|
|
var writeFilePromise = $q.when(),
|
|
|
|
writeFileDeferred
|
|
|
|
if (fileWriter.length) {
|
|
|
|
startOffset = fileWriter.length
|
|
|
|
if (startOffset >= size) {
|
|
|
|
if (toFileEntry) {
|
|
|
|
deferred.resolve()
|
|
|
|
} else {
|
|
|
|
deferred.resolve(cachedDownloads[fileName] = fileWriter.finalize())
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
fileWriter.seek(startOffset)
|
|
|
|
deferred.notify({done: startOffset, total: size})
|
|
|
|
}
|
|
|
|
for (offset = startOffset; offset < size; offset += limit) {
|
|
|
|
writeFileDeferred = $q.defer()
|
|
|
|
;(function (isFinal, offset, writeFileDeferred, writeFilePromise) {
|
|
|
|
return downloadRequest(dcID, function () {
|
2014-06-20 19:34:14 +04:00
|
|
|
if (canceled) {
|
2016-06-28 19:40:53 +03:00
|
|
|
return $q.when()
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
return MtpApiManager.invokeApi('upload.getFile', {
|
|
|
|
location: location,
|
|
|
|
offset: offset,
|
|
|
|
limit: limit
|
|
|
|
}, {
|
|
|
|
dcID: dcID,
|
|
|
|
fileDownload: true,
|
|
|
|
singleInRequest: window.safari !== undefined,
|
|
|
|
createNetworker: true
|
|
|
|
})
|
|
|
|
}, 2).then(function (result) {
|
|
|
|
writeFilePromise.then(function () {
|
|
|
|
if (canceled) {
|
|
|
|
return $q.when()
|
|
|
|
}
|
|
|
|
return processDownloaded(result.bytes).then(function (processedResult) {
|
|
|
|
return FileManager.write(fileWriter, processedResult).then(function () {
|
|
|
|
writeFileDeferred.resolve()
|
|
|
|
}, errorHandler).then(function () {
|
|
|
|
if (isFinal) {
|
|
|
|
resolved = true
|
|
|
|
if (toFileEntry) {
|
|
|
|
deferred.resolve()
|
|
|
|
} else {
|
|
|
|
deferred.resolve(cachedDownloads[fileName] = fileWriter.finalize())
|
|
|
|
}
|
2015-06-30 14:17:32 +03:00
|
|
|
} else {
|
2016-06-28 19:40:53 +03:00
|
|
|
deferred.notify({done: offset + limit, total: size})
|
2015-06-30 14:17:32 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
})
|
|
|
|
})
|
2015-06-30 14:17:32 +03:00
|
|
|
})
|
2016-06-28 19:40:53 +03:00
|
|
|
})
|
|
|
|
})(offset + limit >= size, offset, writeFileDeferred, writeFilePromise)
|
|
|
|
writeFilePromise = writeFileDeferred.promise
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
deferred.promise.cancel = function () {
|
|
|
|
if (!canceled && !resolved) {
|
|
|
|
canceled = true
|
|
|
|
delete cachedDownloadPromises[fileName]
|
|
|
|
errorHandler({type: 'DOWNLOAD_CANCELED'})
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
if (!toFileEntry) {
|
|
|
|
cachedDownloadPromises[fileName] = deferred.promise
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return deferred.promise
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function uploadFile (file) {
|
|
|
|
var fileSize = file.size,
|
|
|
|
isBigFile = fileSize >= 10485760,
|
|
|
|
canceled = false,
|
|
|
|
resolved = false,
|
|
|
|
doneParts = 0,
|
|
|
|
partSize = 262144, // 256 Kb
|
|
|
|
activeDelta = 2
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
if (fileSize > 67108864) {
|
|
|
|
partSize = 524288
|
|
|
|
activeDelta = 4
|
|
|
|
}
|
|
|
|
else if (fileSize < 102400) {
|
|
|
|
partSize = 32768
|
|
|
|
activeDelta = 1
|
|
|
|
}
|
|
|
|
var totalParts = Math.ceil(fileSize / partSize)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
if (totalParts > 3000) {
|
|
|
|
return $q.reject({type: 'FILE_TOO_BIG'})
|
|
|
|
}
|
|
|
|
|
|
|
|
var fileID = [nextRandomInt(0xFFFFFFFF), nextRandomInt(0xFFFFFFFF)]
|
|
|
|
var deferred = $q.defer()
|
|
|
|
var errorHandler = function (error) {
|
|
|
|
// console.error('Up Error', error)
|
|
|
|
deferred.reject(error)
|
|
|
|
canceled = true
|
|
|
|
errorHandler = angular.noop
|
2014-06-20 19:34:14 +04:00
|
|
|
},
|
|
|
|
part = 0,
|
|
|
|
offset,
|
|
|
|
resultInputFile = {
|
|
|
|
_: isBigFile ? 'inputFileBig' : 'inputFile',
|
|
|
|
id: fileID,
|
|
|
|
parts: totalParts,
|
|
|
|
name: file.name,
|
|
|
|
md5_checksum: ''
|
2016-06-28 19:40:53 +03:00
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
for (offset = 0; offset < fileSize; offset += partSize) {
|
|
|
|
(function (offset, part) {
|
|
|
|
downloadRequest('upload', function () {
|
|
|
|
var uploadDeferred = $q.defer()
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var reader = new FileReader()
|
|
|
|
var blob = file.slice(offset, offset + partSize)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
reader.onloadend = function (e) {
|
|
|
|
if (canceled) {
|
|
|
|
uploadDeferred.reject()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if (e.target.readyState != FileReader.DONE) {
|
|
|
|
return
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
MtpApiManager.invokeApi(isBigFile ? 'upload.saveBigFilePart' : 'upload.saveFilePart', {
|
|
|
|
file_id: fileID,
|
|
|
|
file_part: part,
|
|
|
|
file_total_parts: totalParts,
|
|
|
|
bytes: e.target.result
|
|
|
|
}, {
|
|
|
|
startMaxLength: partSize + 256,
|
|
|
|
fileUpload: true,
|
|
|
|
singleInRequest: true
|
|
|
|
}).then(function (result) {
|
|
|
|
doneParts++
|
|
|
|
uploadDeferred.resolve()
|
|
|
|
if (doneParts >= totalParts) {
|
|
|
|
deferred.resolve(resultInputFile)
|
|
|
|
resolved = true
|
|
|
|
} else {
|
|
|
|
console.log(dT(), 'Progress', doneParts * partSize / fileSize)
|
|
|
|
deferred.notify({done: doneParts * partSize, total: fileSize})
|
|
|
|
}
|
|
|
|
}, errorHandler)
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
reader.readAsArrayBuffer(blob)
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return uploadDeferred.promise
|
|
|
|
}, activeDelta)
|
|
|
|
})(offset, part++)
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
deferred.promise.cancel = function () {
|
|
|
|
console.log('cancel upload', canceled, resolved)
|
|
|
|
if (!canceled && !resolved) {
|
|
|
|
canceled = true
|
|
|
|
errorHandler({type: 'UPLOAD_CANCELED'})
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return deferred.promise
|
|
|
|
}
|
2014-06-20 19:34:14 +04:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return {
|
|
|
|
getCachedFile: getCachedFile,
|
|
|
|
getDownloadedFile: getDownloadedFile,
|
|
|
|
downloadFile: downloadFile,
|
|
|
|
downloadSmallFile: downloadSmallFile,
|
|
|
|
saveSmallFile: saveSmallFile,
|
|
|
|
uploadFile: uploadFile
|
|
|
|
}
|
|
|
|
})
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
.service('MtpSingleInstanceService', function (_, $rootScope, $compile, $timeout, $interval, $modalStack, Storage, AppRuntimeManager, IdleManager, ErrorService, MtpNetworkerFactory) {
|
|
|
|
var instanceID = nextRandomInt(0xFFFFFFFF)
|
|
|
|
var started = false
|
|
|
|
var masterInstance = false
|
|
|
|
var deactivatePromise = false
|
|
|
|
var deactivated = false
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function start () {
|
|
|
|
if (!started && !Config.Navigator.mobile && !Config.Modes.packed) {
|
|
|
|
started = true
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
IdleManager.start()
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
$rootScope.$watch('idle.isIDLE', checkInstance)
|
|
|
|
$interval(checkInstance, 5000)
|
|
|
|
checkInstance()
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
try {
|
|
|
|
$($window).on('beforeunload', clearInstance)
|
|
|
|
} catch (e) {}
|
|
|
|
}
|
|
|
|
}
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function clearInstance () {
|
|
|
|
Storage.remove(masterInstance ? 'xt_instance' : 'xt_idle_instance')
|
2014-11-07 19:07:54 +03:00
|
|
|
}
|
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function deactivateInstance () {
|
|
|
|
if (masterInstance || deactivated) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
console.log(dT(), 'deactivate')
|
|
|
|
deactivatePromise = false
|
|
|
|
deactivated = true
|
|
|
|
clearInstance()
|
|
|
|
$modalStack.dismissAll()
|
|
|
|
|
|
|
|
document.title = _('inactive_tab_title_raw')
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
var inactivePageCompiled = $compile('<ng-include src="\'partials/desktop/inactive.html\'"></ng-include>')
|
|
|
|
|
|
|
|
var scope = $rootScope.$new(true)
|
|
|
|
scope.close = function () {
|
|
|
|
AppRuntimeManager.close()
|
|
|
|
}
|
|
|
|
scope.reload = function () {
|
|
|
|
AppRuntimeManager.reload()
|
|
|
|
}
|
|
|
|
inactivePageCompiled(scope, function (clonedElement) {
|
|
|
|
$('.page_wrap').hide()
|
|
|
|
$(clonedElement).appendTo($('body'))
|
|
|
|
})
|
|
|
|
$rootScope.idle.deactivated = true
|
2015-10-21 15:59:34 +02:00
|
|
|
}
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
function checkInstance () {
|
|
|
|
if (deactivated) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
var time = tsNow()
|
|
|
|
var idle = $rootScope.idle && $rootScope.idle.isIDLE
|
|
|
|
var newInstance = {id: instanceID, idle: idle, time: time}
|
|
|
|
|
|
|
|
Storage.get('xt_instance', 'xt_idle_instance').then(function (result) {
|
|
|
|
var curInstance = result[0]
|
|
|
|
var idleInstance = result[1]
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
// console.log(dT(), 'check instance', newInstance, curInstance, idleInstance)
|
|
|
|
if (!idle ||
|
2015-10-21 15:59:34 +02:00
|
|
|
!curInstance ||
|
2014-11-07 19:07:54 +03:00
|
|
|
curInstance.id == instanceID ||
|
2015-10-21 15:59:34 +02:00
|
|
|
curInstance.time < time - 60000) {
|
2016-06-28 19:40:53 +03:00
|
|
|
if (idleInstance &&
|
2015-10-21 15:59:34 +02:00
|
|
|
idleInstance.id == instanceID) {
|
2016-06-28 19:40:53 +03:00
|
|
|
Storage.remove('xt_idle_instance')
|
|
|
|
}
|
|
|
|
Storage.set({xt_instance: newInstance})
|
|
|
|
if (!masterInstance) {
|
|
|
|
MtpNetworkerFactory.startAll()
|
|
|
|
console.warn(dT(), 'now master instance', newInstance)
|
|
|
|
}
|
|
|
|
masterInstance = true
|
|
|
|
if (deactivatePromise) {
|
|
|
|
$timeout.cancel(deactivatePromise)
|
|
|
|
deactivatePromise = false
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Storage.set({xt_idle_instance: newInstance})
|
|
|
|
if (masterInstance) {
|
|
|
|
MtpNetworkerFactory.stopAll()
|
|
|
|
console.warn(dT(), 'now idle instance', newInstance)
|
|
|
|
if (!deactivatePromise) {
|
|
|
|
deactivatePromise = $timeout(deactivateInstance, 30000)
|
|
|
|
}
|
2015-10-21 15:59:34 +02:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
masterInstance = false
|
2014-11-07 19:07:54 +03:00
|
|
|
}
|
2016-06-28 19:40:53 +03:00
|
|
|
})
|
|
|
|
}
|
2014-11-07 19:07:54 +03:00
|
|
|
|
2016-06-28 19:40:53 +03:00
|
|
|
return {
|
|
|
|
start: start
|
|
|
|
}
|
|
|
|
})
|