You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			245 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
			
		
		
	
	
			245 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
/*
 | 
						|
 * vim: ts=4:sw=4:expandtab
 | 
						|
 */
 | 
						|
function OutgoingMessage(server, timestamp, numbers, message, silent, callback) {
 | 
						|
    if (message instanceof textsecure.protobuf.DataMessage) {
 | 
						|
        var content = new textsecure.protobuf.Content();
 | 
						|
        content.dataMessage = message;
 | 
						|
        message = content;
 | 
						|
    }
 | 
						|
    this.server = server;
 | 
						|
    this.timestamp = timestamp;
 | 
						|
    this.numbers = numbers;
 | 
						|
    this.message = message; // ContentMessage proto
 | 
						|
    this.callback = callback;
 | 
						|
    this.silent = silent;
 | 
						|
 | 
						|
    this.numbersCompleted = 0;
 | 
						|
    this.errors = [];
 | 
						|
    this.successfulNumbers = [];
 | 
						|
}
 | 
						|
 | 
						|
OutgoingMessage.prototype = {
 | 
						|
    constructor: OutgoingMessage,
 | 
						|
    numberCompleted: function() {
 | 
						|
        this.numbersCompleted++;
 | 
						|
        if (this.numbersCompleted >= this.numbers.length) {
 | 
						|
            this.callback({successfulNumbers: this.successfulNumbers, errors: this.errors});
 | 
						|
        }
 | 
						|
    },
 | 
						|
    registerError: function(number, reason, error) {
 | 
						|
        if (!error || error.name === 'HTTPError' && error.code !== 404) {
 | 
						|
            error = new textsecure.OutgoingMessageError(number, this.message.toArrayBuffer(), this.timestamp, error);
 | 
						|
        }
 | 
						|
 | 
						|
        error.number = number;
 | 
						|
        error.reason = reason;
 | 
						|
        this.errors[this.errors.length] = error;
 | 
						|
        this.numberCompleted();
 | 
						|
    },
 | 
						|
    reloadDevicesAndSend: function(number, recurse) {
 | 
						|
        return function() {
 | 
						|
            return textsecure.storage.protocol.getDeviceIds(number).then(function(deviceIds) {
 | 
						|
                if (deviceIds.length == 0) {
 | 
						|
                    return this.registerError(number, "Got empty device list when loading device keys", null);
 | 
						|
                }
 | 
						|
                return this.doSendMessage(number, deviceIds, recurse);
 | 
						|
            }.bind(this));
 | 
						|
        }.bind(this);
 | 
						|
    },
 | 
						|
 | 
						|
    getKeysForNumber: function(number, updateDevices) {
 | 
						|
        var handleResult = function(response) {
 | 
						|
            return Promise.all(response.devices.map(function(device) {
 | 
						|
                device.identityKey = response.identityKey;
 | 
						|
                if (updateDevices === undefined || updateDevices.indexOf(device.deviceId) > -1) {
 | 
						|
                    var address = new libsignal.SignalProtocolAddress(number, device.deviceId);
 | 
						|
                    var builder = new libsignal.SessionBuilder(textsecure.storage.protocol, address);
 | 
						|
                    if (device.registrationId === 0) {
 | 
						|
                        console.log("device registrationId 0!");
 | 
						|
                    }
 | 
						|
                    return builder.processPreKey(device).catch(function(error) {
 | 
						|
                        if (error.message === "Identity key changed") {
 | 
						|
                            error.timestamp = this.timestamp;
 | 
						|
                            error.originalMessage = this.message.toArrayBuffer();
 | 
						|
                            error.identityKey = device.identityKey;
 | 
						|
                        }
 | 
						|
                        throw error;
 | 
						|
                    }.bind(this));
 | 
						|
                }
 | 
						|
            }.bind(this)));
 | 
						|
        }.bind(this);
 | 
						|
 | 
						|
        if (updateDevices === undefined) {
 | 
						|
            return this.server.getKeysForNumber(number).then(handleResult);
 | 
						|
        } else {
 | 
						|
            var promise = Promise.resolve();
 | 
						|
            updateDevices.forEach(function(device) {
 | 
						|
                promise = promise.then(function() {
 | 
						|
                    return this.server.getKeysForNumber(number, device).then(handleResult).catch(function(e) {
 | 
						|
                        if (e.name === 'HTTPError' && e.code === 404) {
 | 
						|
                            if (device !== 1) {
 | 
						|
                                return this.removeDeviceIdsForNumber(number, [device]);
 | 
						|
                            } else {
 | 
						|
                                throw new textsecure.UnregisteredUserError(number, e);
 | 
						|
                            }
 | 
						|
                        } else {
 | 
						|
                            throw e;
 | 
						|
                        }
 | 
						|
                    }.bind(this));
 | 
						|
                }.bind(this));
 | 
						|
            }.bind(this));
 | 
						|
 | 
						|
            return promise;
 | 
						|
        }
 | 
						|
    },
 | 
						|
 | 
						|
    transmitMessage: function(number, jsonData, timestamp) {
 | 
						|
        return this.server.sendMessages(number, jsonData, timestamp, this.silent).catch(function(e) {
 | 
						|
            if (e.name === 'HTTPError' && (e.code !== 409 && e.code !== 410)) {
 | 
						|
                // 409 and 410 should bubble and be handled by doSendMessage
 | 
						|
                // 404 should throw UnregisteredUserError
 | 
						|
                // all other network errors can be retried later.
 | 
						|
                if (e.code === 404) {
 | 
						|
                    throw new textsecure.UnregisteredUserError(number, e);
 | 
						|
                }
 | 
						|
                throw new textsecure.SendMessageNetworkError(number, jsonData, e, timestamp);
 | 
						|
            }
 | 
						|
            throw e;
 | 
						|
        });
 | 
						|
    },
 | 
						|
 | 
						|
    getPaddedMessageLength: function(messageLength) {
 | 
						|
        var messageLengthWithTerminator = messageLength + 1;
 | 
						|
        var messagePartCount            = Math.floor(messageLengthWithTerminator / 160);
 | 
						|
 | 
						|
        if (messageLengthWithTerminator % 160 !== 0) {
 | 
						|
            messagePartCount++;
 | 
						|
        }
 | 
						|
 | 
						|
        return messagePartCount * 160;
 | 
						|
    },
 | 
						|
 | 
						|
    getPlaintext: function() {
 | 
						|
        if (!this.plaintext) {
 | 
						|
            var messageBuffer = this.message.toArrayBuffer();
 | 
						|
            this.plaintext = new Uint8Array(
 | 
						|
                this.getPaddedMessageLength(messageBuffer.byteLength + 1) - 1
 | 
						|
            );
 | 
						|
            this.plaintext.set(new Uint8Array(messageBuffer));
 | 
						|
            this.plaintext[messageBuffer.byteLength] = 0x80;
 | 
						|
        }
 | 
						|
        return this.plaintext;
 | 
						|
    },
 | 
						|
 | 
						|
    doSendMessage: function(number, deviceIds, recurse) {
 | 
						|
        var ciphers = {};
 | 
						|
        var plaintext = this.getPlaintext();
 | 
						|
 | 
						|
        return Promise.all(deviceIds.map(function(deviceId) {
 | 
						|
            var address = new libsignal.SignalProtocolAddress(number, deviceId);
 | 
						|
 | 
						|
            var ourNumber = textsecure.storage.user.getNumber();
 | 
						|
            var options = {};
 | 
						|
 | 
						|
            // No limit on message keys if we're communicating with our other devices
 | 
						|
            if (ourNumber === number) {
 | 
						|
                options.messageKeysLimit = false;
 | 
						|
            }
 | 
						|
 | 
						|
            var sessionCipher = new libsignal.SessionCipher(textsecure.storage.protocol, address, options);
 | 
						|
            ciphers[address.getDeviceId()] = sessionCipher;
 | 
						|
            return sessionCipher.encrypt(plaintext).then(function(ciphertext) {
 | 
						|
                return {
 | 
						|
                    type                      : ciphertext.type,
 | 
						|
                    destinationDeviceId       : address.getDeviceId(),
 | 
						|
                    destinationRegistrationId : ciphertext.registrationId,
 | 
						|
                    content                   : btoa(ciphertext.body)
 | 
						|
                };
 | 
						|
            });
 | 
						|
        }.bind(this))).then(function(jsonData) {
 | 
						|
            return this.transmitMessage(number, jsonData, this.timestamp).then(function() {
 | 
						|
                this.successfulNumbers[this.successfulNumbers.length] = number;
 | 
						|
                this.numberCompleted();
 | 
						|
            }.bind(this));
 | 
						|
        }.bind(this)).catch(function(error) {
 | 
						|
            if (error instanceof Error && error.name == "HTTPError" && (error.code == 410 || error.code == 409)) {
 | 
						|
                if (!recurse)
 | 
						|
                    return this.registerError(number, "Hit retry limit attempting to reload device list", error);
 | 
						|
 | 
						|
                var p;
 | 
						|
                if (error.code == 409) {
 | 
						|
                    p = this.removeDeviceIdsForNumber(number, error.response.extraDevices);
 | 
						|
                } else {
 | 
						|
                    p = Promise.all(error.response.staleDevices.map(function(deviceId) {
 | 
						|
                        return ciphers[deviceId].closeOpenSessionForDevice();
 | 
						|
                    }));
 | 
						|
                }
 | 
						|
 | 
						|
                return p.then(function() {
 | 
						|
                    var resetDevices = ((error.code == 410) ? error.response.staleDevices : error.response.missingDevices);
 | 
						|
                    return this.getKeysForNumber(number, resetDevices)
 | 
						|
                        .then(this.reloadDevicesAndSend(number, error.code == 409));
 | 
						|
                }.bind(this));
 | 
						|
            } else if (error.message === "Identity key changed") {
 | 
						|
                error.timestamp = this.timestamp;
 | 
						|
                error.originalMessage = this.message.toArrayBuffer();
 | 
						|
                console.log('Got "key changed" error from encrypt - no identityKey for application layer', number, deviceIds)
 | 
						|
                throw error;
 | 
						|
            } else {
 | 
						|
                this.registerError(number, "Failed to create or send message", error);
 | 
						|
            }
 | 
						|
        }.bind(this));
 | 
						|
    },
 | 
						|
 | 
						|
    getStaleDeviceIdsForNumber: function(number) {
 | 
						|
        return textsecure.storage.protocol.getDeviceIds(number).then(function(deviceIds) {
 | 
						|
            if (deviceIds.length === 0) {
 | 
						|
                return [1];
 | 
						|
            }
 | 
						|
            var updateDevices = [];
 | 
						|
            return Promise.all(deviceIds.map(function(deviceId) {
 | 
						|
                var address = new libsignal.SignalProtocolAddress(number, deviceId);
 | 
						|
                var sessionCipher = new libsignal.SessionCipher(textsecure.storage.protocol, address);
 | 
						|
                return sessionCipher.hasOpenSession().then(function(hasSession) {
 | 
						|
                    if (!hasSession) {
 | 
						|
                        updateDevices.push(deviceId);
 | 
						|
                    }
 | 
						|
                });
 | 
						|
            })).then(function() {
 | 
						|
                return updateDevices;
 | 
						|
            });
 | 
						|
        });
 | 
						|
    },
 | 
						|
 | 
						|
    removeDeviceIdsForNumber: function(number, deviceIdsToRemove) {
 | 
						|
        var promise = Promise.resolve();
 | 
						|
        for (var j in deviceIdsToRemove) {
 | 
						|
            promise = promise.then(function() {
 | 
						|
                var encodedNumber = number + "." + deviceIdsToRemove[j];
 | 
						|
                return textsecure.storage.protocol.removeSession(encodedNumber);
 | 
						|
            });
 | 
						|
        }
 | 
						|
        return promise;
 | 
						|
    },
 | 
						|
 | 
						|
    sendToNumber: function(number) {
 | 
						|
        return this.getStaleDeviceIdsForNumber(number).then(function(updateDevices) {
 | 
						|
            return this.getKeysForNumber(number, updateDevices)
 | 
						|
                .then(this.reloadDevicesAndSend(number, true))
 | 
						|
                .catch(function(error) {
 | 
						|
                    if (error.message === "Identity key changed") {
 | 
						|
                        error = new textsecure.OutgoingIdentityKeyError(
 | 
						|
                            number, error.originalMessage, error.timestamp, error.identityKey
 | 
						|
                        );
 | 
						|
                        this.registerError(number, "Identity key changed", error);
 | 
						|
                    } else {
 | 
						|
                        this.registerError(
 | 
						|
                            number, "Failed to retrieve new device keys for number " + number, error
 | 
						|
                        );
 | 
						|
                    }
 | 
						|
                }.bind(this));
 | 
						|
        }.bind(this));
 | 
						|
    }
 | 
						|
};
 |