Add MessageSender

textsecure.MessageSender takes server url and credentials and returns
a message sending interface configured for that server.

Used a wrapper function to insert a TextSecureServer instance into
sendmessage.js code at runtime. This will result in function duplication
between different MessageSender objects, pending further refactoring to
use prototypal inheritence.

// FREEBIE
pull/749/head
lilia 10 years ago
parent 98aa5156b0
commit 9e9d767a30

@ -102,6 +102,7 @@
messageReceiver.addEventListener('contactsync', onContactSyncComplete); messageReceiver.addEventListener('contactsync', onContactSyncComplete);
window.textsecure.messaging = new textsecure.MessageSender(SERVER_URL, USERNAME, PASSWORD);
if (firstRun === true && textsecure.storage.user.getDeviceId() != '1') { if (firstRun === true && textsecure.storage.user.getDeviceId() != '1') {
textsecure.messaging.sendRequestContactSyncMessage().then(function() { textsecure.messaging.sendRequestContactSyncMessage().then(function() {
textsecure.messaging.sendRequestGroupSyncMessage(); textsecure.messaging.sendRequestGroupSyncMessage();

@ -39586,441 +39586,445 @@ var TextSecureServer = (function() {
* vim: ts=4:sw=4:expandtab * vim: ts=4:sw=4:expandtab
*/ */
// sendMessage(numbers = [], message = PushMessageContentProto, callback(success/failure map)) // sendMessage(numbers = [], message = PushMessageContentProto, callback(success/failure map))
window.textsecure.messaging = function() { window.textsecure.MessageSender = function(url, username, password) {
'use strict'; 'use strict';
var server = new TextSecureServer(url, username, password);
var self = {}; return (function(TextSecureServer) {
'use strict';
// message == DataMessage or ContentMessage proto
function sendMessageToDevices(timestamp, number, deviceObjectList, message) { var self = {};
var relay = deviceObjectList[0].relay;
for (var i=1; i < deviceObjectList.length; ++i) { // message == DataMessage or ContentMessage proto
if (deviceObjectList[i].relay !== relay) { function sendMessageToDevices(timestamp, number, deviceObjectList, message) {
throw new Error("Mismatched relays for number " + number); var relay = deviceObjectList[0].relay;
for (var i=1; i < deviceObjectList.length; ++i) {
if (deviceObjectList[i].relay !== relay) {
throw new Error("Mismatched relays for number " + number);
}
} }
} return Promise.all(deviceObjectList.map(function(device) {
return Promise.all(deviceObjectList.map(function(device) { return textsecure.protocol_wrapper.encryptMessageFor(device, message).then(function(encryptedMsg) {
return textsecure.protocol_wrapper.encryptMessageFor(device, message).then(function(encryptedMsg) { return textsecure.protocol_wrapper.getRegistrationId(device.encodedNumber).then(function(registrationId) {
return textsecure.protocol_wrapper.getRegistrationId(device.encodedNumber).then(function(registrationId) { return textsecure.storage.devices.removeTempKeysFromDevice(device.encodedNumber).then(function() {
return textsecure.storage.devices.removeTempKeysFromDevice(device.encodedNumber).then(function() { var json = {
var json = { type: encryptedMsg.type,
type: encryptedMsg.type, destinationDeviceId: textsecure.utils.unencodeNumber(device.encodedNumber)[1],
destinationDeviceId: textsecure.utils.unencodeNumber(device.encodedNumber)[1], destinationRegistrationId: registrationId,
destinationRegistrationId: registrationId, content: encryptedMsg.body,
content: encryptedMsg.body, timestamp: timestamp
timestamp: timestamp };
};
if (device.relay !== undefined) { if (device.relay !== undefined) {
json.relay = device.relay; json.relay = device.relay;
} }
return json; return json;
});
}); });
}); });
})).then(function(jsonData) {
var legacy = (message instanceof textsecure.protobuf.DataMessage);
return TextSecureServer.sendMessages(number, jsonData, legacy);
}); });
})).then(function(jsonData) { }
var legacy = (message instanceof textsecure.protobuf.DataMessage);
return TextSecureServer.sendMessages(number, jsonData, legacy);
});
}
function makeAttachmentPointer(attachment) { function makeAttachmentPointer(attachment) {
var proto = new textsecure.protobuf.AttachmentPointer(); var proto = new textsecure.protobuf.AttachmentPointer();
proto.key = textsecure.crypto.getRandomBytes(64); proto.key = textsecure.crypto.getRandomBytes(64);
var iv = textsecure.crypto.getRandomBytes(16); var iv = textsecure.crypto.getRandomBytes(16);
return textsecure.crypto.encryptAttachment(attachment.data, proto.key, iv).then(function(encryptedBin) { return textsecure.crypto.encryptAttachment(attachment.data, proto.key, iv).then(function(encryptedBin) {
return TextSecureServer.putAttachment(encryptedBin).then(function(id) { return TextSecureServer.putAttachment(encryptedBin).then(function(id) {
proto.id = id; proto.id = id;
proto.contentType = attachment.contentType; proto.contentType = attachment.contentType;
return proto; return proto;
});
}); });
}); }
}
var tryMessageAgain = function(number, encodedMessage, timestamp) { var tryMessageAgain = function(number, encodedMessage, timestamp) {
var proto = textsecure.protobuf.DataMessage.decode(encodedMessage); var proto = textsecure.protobuf.DataMessage.decode(encodedMessage);
return new Promise(function(resolve, reject) { return new Promise(function(resolve, reject) {
sendMessageProto(timestamp, [number], proto, function(res) { sendMessageProto(timestamp, [number], proto, function(res) {
if (res.failure.length > 0) if (res.failure.length > 0)
reject(res.failure); reject(res.failure);
else else
resolve(); resolve();
});
}); });
});
};
textsecure.replay.registerFunction(tryMessageAgain, textsecure.replay.Type.SEND_MESSAGE);
var sendMessageProto = function(timestamp, numbers, message, callback) {
var numbersCompleted = 0;
var errors = [];
var successfulNumbers = [];
var numberCompleted = function() {
numbersCompleted++;
if (numbersCompleted >= numbers.length)
callback({success: successfulNumbers, failure: errors});
};
var registerError = function(number, reason, error) {
if (!error) {
error = new Error(reason);
}
error.number = number;
error.reason = reason;
errors[errors.length] = error;
numberCompleted();
}; };
textsecure.replay.registerFunction(tryMessageAgain, textsecure.replay.Type.SEND_MESSAGE);
var reloadDevicesAndSend = function(number, recurse) { var sendMessageProto = function(timestamp, numbers, message, callback) {
return function() { var numbersCompleted = 0;
return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) { var errors = [];
if (devicesForNumber.length == 0) var successfulNumbers = [];
return registerError(number, "Got empty device list when loading device keys", null);
doSendMessage(number, devicesForNumber, recurse);
});
}
};
function getKeysForNumber(number, updateDevices) { var numberCompleted = function() {
var handleResult = function(response) { numbersCompleted++;
return Promise.all(response.devices.map(function(device) { if (numbersCompleted >= numbers.length)
if (updateDevices === undefined || updateDevices.indexOf(device.deviceId) > -1) callback({success: successfulNumbers, failure: errors});
return textsecure.storage.devices.saveKeysToDeviceObject({
encodedNumber: number + "." + device.deviceId,
identityKey: response.identityKey,
preKey: device.preKey.publicKey,
preKeyId: device.preKey.keyId,
signedKey: device.signedPreKey.publicKey,
signedKeyId: device.signedPreKey.keyId,
signedKeySignature: device.signedPreKey.signature,
registrationId: device.registrationId
}).catch(function(error) {
if (error.message === "Identity key changed") {
error = new textsecure.OutgoingIdentityKeyError(number, message.toArrayBuffer(), timestamp, error.identityKey);
registerError(number, "Identity key changed", error);
}
throw error;
});
}));
}; };
if (updateDevices === undefined) { var registerError = function(number, reason, error) {
return TextSecureServer.getKeysForNumber(number).then(handleResult); if (!error) {
} else { error = new Error(reason);
var promises = []; }
for (var i in updateDevices) error.number = number;
promises[promises.length] = TextSecureServer.getKeysForNumber(number, updateDevices[i]).then(handleResult); error.reason = reason;
errors[errors.length] = error;
return Promise.all(promises);
}
}
var doSendMessage = function(number, devicesForNumber, recurse) {
return sendMessageToDevices(timestamp, number, devicesForNumber, message).then(function(result) {
successfulNumbers[successfulNumbers.length] = number;
numberCompleted(); numberCompleted();
}).catch(function(error) { };
if (error instanceof Error && error.name == "HTTPError" && (error.code == 410 || error.code == 409)) {
if (!recurse)
return registerError(number, "Hit retry limit attempting to reload device list", error);
var p;
if (error.code == 409) {
p = textsecure.storage.devices.removeDeviceIdsForNumber(number, error.response.extraDevices);
} else {
p = Promise.all(error.response.staleDevices.map(function(deviceId) {
return textsecure.protocol_wrapper.closeOpenSessionForDevice(number + '.' + deviceId);
}));
}
p.then(function() { var reloadDevicesAndSend = function(number, recurse) {
var resetDevices = ((error.code == 410) ? error.response.staleDevices : error.response.missingDevices); return function() {
getKeysForNumber(number, resetDevices) return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) {
.then(reloadDevicesAndSend(number, (error.code == 409))) if (devicesForNumber.length == 0)
.catch(function(error) { return registerError(number, "Got empty device list when loading device keys", null);
registerError(number, "Failed to reload device keys", error); doSendMessage(number, devicesForNumber, recurse);
});
}); });
}
};
function getKeysForNumber(number, updateDevices) {
var handleResult = function(response) {
return Promise.all(response.devices.map(function(device) {
if (updateDevices === undefined || updateDevices.indexOf(device.deviceId) > -1)
return textsecure.storage.devices.saveKeysToDeviceObject({
encodedNumber: number + "." + device.deviceId,
identityKey: response.identityKey,
preKey: device.preKey.publicKey,
preKeyId: device.preKey.keyId,
signedKey: device.signedPreKey.publicKey,
signedKeyId: device.signedPreKey.keyId,
signedKeySignature: device.signedPreKey.signature,
registrationId: device.registrationId
}).catch(function(error) {
if (error.message === "Identity key changed") {
error = new textsecure.OutgoingIdentityKeyError(number, message.toArrayBuffer(), timestamp, error.identityKey);
registerError(number, "Identity key changed", error);
}
throw error;
});
}));
};
if (updateDevices === undefined) {
return TextSecureServer.getKeysForNumber(number).then(handleResult);
} else { } else {
registerError(number, "Failed to create or send message", error); var promises = [];
for (var i in updateDevices)
promises[promises.length] = TextSecureServer.getKeysForNumber(number, updateDevices[i]).then(handleResult);
return Promise.all(promises);
} }
}); }
};
numbers.forEach(function(number) { var doSendMessage = function(number, devicesForNumber, recurse) {
textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) { return sendMessageToDevices(timestamp, number, devicesForNumber, message).then(function(result) {
return Promise.all(devicesForNumber.map(function(device) { successfulNumbers[successfulNumbers.length] = number;
return textsecure.protocol_wrapper.hasOpenSession(device.encodedNumber).then(function(result) { numberCompleted();
if (!result) }).catch(function(error) {
return getKeysForNumber(number, [parseInt(textsecure.utils.unencodeNumber(device.encodedNumber)[1])]); if (error instanceof Error && error.name == "HTTPError" && (error.code == 410 || error.code == 409)) {
}); if (!recurse)
})).then(function() { return registerError(number, "Hit retry limit attempting to reload device list", error);
return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) {
if (devicesForNumber.length == 0) { var p;
getKeysForNumber(number, [1]) if (error.code == 409) {
.then(reloadDevicesAndSend(number, true)) p = textsecure.storage.devices.removeDeviceIdsForNumber(number, error.response.extraDevices);
} else {
p = Promise.all(error.response.staleDevices.map(function(deviceId) {
return textsecure.protocol_wrapper.closeOpenSessionForDevice(number + '.' + deviceId);
}));
}
p.then(function() {
var resetDevices = ((error.code == 410) ? error.response.staleDevices : error.response.missingDevices);
getKeysForNumber(number, resetDevices)
.then(reloadDevicesAndSend(number, (error.code == 409)))
.catch(function(error) { .catch(function(error) {
registerError(number, "Failed to retreive new device keys for number " + number, error); registerError(number, "Failed to reload device keys", error);
}); });
} else });
doSendMessage(number, devicesForNumber, true); } else {
registerError(number, "Failed to create or send message", error);
}
});
};
numbers.forEach(function(number) {
textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) {
return Promise.all(devicesForNumber.map(function(device) {
return textsecure.protocol_wrapper.hasOpenSession(device.encodedNumber).then(function(result) {
if (!result)
return getKeysForNumber(number, [parseInt(textsecure.utils.unencodeNumber(device.encodedNumber)[1])]);
});
})).then(function() {
return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) {
if (devicesForNumber.length == 0) {
getKeysForNumber(number, [1])
.then(reloadDevicesAndSend(number, true))
.catch(function(error) {
registerError(number, "Failed to retreive new device keys for number " + number, error);
});
} else
doSendMessage(number, devicesForNumber, true);
});
}); });
}); });
}); });
}); }
}
var sendIndividualProto = function(number, proto, timestamp) { var sendIndividualProto = function(number, proto, timestamp) {
return new Promise(function(resolve, reject) { return new Promise(function(resolve, reject) {
sendMessageProto(timestamp, [number], proto, function(res) { sendMessageProto(timestamp, [number], proto, function(res) {
if (res.failure.length > 0) if (res.failure.length > 0)
reject(res.failure); reject(res.failure);
else else
resolve(); resolve();
});
}); });
});
}
var sendSyncMessage = function(message, timestamp, destination) {
var myNumber = textsecure.storage.user.getNumber();
var myDevice = textsecure.storage.user.getDeviceId();
if (myDevice != 1) {
var sentMessage = new textsecure.protobuf.SyncMessage.Sent();
sentMessage.timestamp = timestamp;
sentMessage.message = message;
if (destination) {
sentMessage.destination = destination;
}
var syncMessage = new textsecure.protobuf.SyncMessage();
syncMessage.sent = sentMessage;
var contentMessage = new textsecure.protobuf.Content();
contentMessage.syncMessage = syncMessage;
return sendIndividualProto(myNumber, contentMessage, Date.now());
} }
}
self.sendRequestGroupSyncMessage = function() { var sendSyncMessage = function(message, timestamp, destination) {
var myNumber = textsecure.storage.user.getNumber(); var myNumber = textsecure.storage.user.getNumber();
var myDevice = textsecure.storage.user.getDeviceId(); var myDevice = textsecure.storage.user.getDeviceId();
if (myDevice != 1) { if (myDevice != 1) {
var request = new textsecure.protobuf.SyncMessage.Request(); var sentMessage = new textsecure.protobuf.SyncMessage.Sent();
request.type = textsecure.protobuf.SyncMessage.Request.Type.GROUPS; sentMessage.timestamp = timestamp;
var syncMessage = new textsecure.protobuf.SyncMessage(); sentMessage.message = message;
syncMessage.request = request; if (destination) {
var contentMessage = new textsecure.protobuf.Content(); sentMessage.destination = destination;
contentMessage.syncMessage = syncMessage; }
var syncMessage = new textsecure.protobuf.SyncMessage();
syncMessage.sent = sentMessage;
var contentMessage = new textsecure.protobuf.Content();
contentMessage.syncMessage = syncMessage;
return sendIndividualProto(myNumber, contentMessage, Date.now()); return sendIndividualProto(myNumber, contentMessage, Date.now());
} }
};
self.sendRequestContactSyncMessage = function() {
var myNumber = textsecure.storage.user.getNumber();
var myDevice = textsecure.storage.user.getDeviceId();
if (myDevice != 1) {
var request = new textsecure.protobuf.SyncMessage.Request();
request.type = textsecure.protobuf.SyncMessage.Request.Type.CONTACTS;
var syncMessage = new textsecure.protobuf.SyncMessage();
syncMessage.request = request;
var contentMessage = new textsecure.protobuf.Content();
contentMessage.syncMessage = syncMessage;
return sendIndividualProto(myNumber, contentMessage, Date.now());
} }
};
var sendGroupProto = function(numbers, proto, timestamp) { self.sendRequestGroupSyncMessage = function() {
timestamp = timestamp || Date.now(); var myNumber = textsecure.storage.user.getNumber();
var me = textsecure.storage.user.getNumber(); var myDevice = textsecure.storage.user.getDeviceId();
numbers = numbers.filter(function(number) { return number != me; }); if (myDevice != 1) {
var request = new textsecure.protobuf.SyncMessage.Request();
request.type = textsecure.protobuf.SyncMessage.Request.Type.GROUPS;
var syncMessage = new textsecure.protobuf.SyncMessage();
syncMessage.request = request;
var contentMessage = new textsecure.protobuf.Content();
contentMessage.syncMessage = syncMessage;
return new Promise(function(resolve, reject) { return sendIndividualProto(myNumber, contentMessage, Date.now());
sendMessageProto(timestamp, numbers, proto, function(res) { }
if (res.failure.length > 0) };
reject(res.failure); self.sendRequestContactSyncMessage = function() {
else var myNumber = textsecure.storage.user.getNumber();
resolve(); var myDevice = textsecure.storage.user.getDeviceId();
}); if (myDevice != 1) {
}).then(function() { var request = new textsecure.protobuf.SyncMessage.Request();
return sendSyncMessage(proto, timestamp); request.type = textsecure.protobuf.SyncMessage.Request.Type.CONTACTS;
}); var syncMessage = new textsecure.protobuf.SyncMessage();
} syncMessage.request = request;
var contentMessage = new textsecure.protobuf.Content();
contentMessage.syncMessage = syncMessage;
return sendIndividualProto(myNumber, contentMessage, Date.now());
}
};
self.sendMessageToNumber = function(number, messageText, attachments, timestamp) { var sendGroupProto = function(numbers, proto, timestamp) {
var proto = new textsecure.protobuf.DataMessage(); timestamp = timestamp || Date.now();
proto.body = messageText; var me = textsecure.storage.user.getNumber();
numbers = numbers.filter(function(number) { return number != me; });
var promises = [];
for (var i in attachments)
promises.push(makeAttachmentPointer(attachments[i]));
return Promise.all(promises).then(function(attachmentsArray) {
proto.attachments = attachmentsArray;
return sendIndividualProto(number, proto, timestamp).then(function() {
return sendSyncMessage(proto, timestamp, number);
});
});
}
self.closeSession = function(number) { return new Promise(function(resolve, reject) {
var proto = new textsecure.protobuf.DataMessage(); sendMessageProto(timestamp, numbers, proto, function(res) {
proto.body = "TERMINATE"; if (res.failure.length > 0)
proto.flags = textsecure.protobuf.DataMessage.Flags.END_SESSION; reject(res.failure);
return sendIndividualProto(number, proto, Date.now()).then(function(res) { else
return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devices) { resolve();
return Promise.all(devices.map(function(device) {
return textsecure.protocol_wrapper.closeOpenSessionForDevice(device.encodedNumber);
})).then(function() {
return res;
}); });
}).then(function() {
return sendSyncMessage(proto, timestamp);
}); });
}); }
}
self.sendMessageToGroup = function(groupId, messageText, attachments, timestamp) {
var proto = new textsecure.protobuf.DataMessage();
proto.body = messageText;
proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.DELIVER;
return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) { self.sendMessageToNumber = function(number, messageText, attachments, timestamp) {
if (numbers === undefined) var proto = new textsecure.protobuf.DataMessage();
return Promise.reject(new Error("Unknown Group")); proto.body = messageText;
var promises = []; var promises = [];
for (var i in attachments) for (var i in attachments)
promises.push(makeAttachmentPointer(attachments[i])); promises.push(makeAttachmentPointer(attachments[i]));
return Promise.all(promises).then(function(attachmentsArray) { return Promise.all(promises).then(function(attachmentsArray) {
proto.attachments = attachmentsArray; proto.attachments = attachmentsArray;
return sendGroupProto(numbers, proto, timestamp); return sendIndividualProto(number, proto, timestamp).then(function() {
return sendSyncMessage(proto, timestamp, number);
});
}); });
}); }
}
self.createGroup = function(numbers, name, avatar) { self.closeSession = function(number) {
var proto = new textsecure.protobuf.DataMessage(); var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext(); proto.body = "TERMINATE";
proto.flags = textsecure.protobuf.DataMessage.Flags.END_SESSION;
return sendIndividualProto(number, proto, Date.now()).then(function(res) {
return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devices) {
return Promise.all(devices.map(function(device) {
return textsecure.protocol_wrapper.closeOpenSessionForDevice(device.encodedNumber);
})).then(function() {
return res;
});
});
});
}
return textsecure.storage.groups.createNewGroup(numbers).then(function(group) { self.sendMessageToGroup = function(groupId, messageText, attachments, timestamp) {
proto.group.id = toArrayBuffer(group.id); var proto = new textsecure.protobuf.DataMessage();
var numbers = group.numbers; proto.body = messageText;
proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.DELIVER;
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE; return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) {
proto.group.members = numbers; if (numbers === undefined)
proto.group.name = name; return Promise.reject(new Error("Unknown Group"));
if (avatar !== undefined) { var promises = [];
return makeAttachmentPointer(avatar).then(function(attachment) { for (var i in attachments)
proto.group.avatar = attachment; promises.push(makeAttachmentPointer(attachments[i]));
return Promise.all(promises).then(function(attachmentsArray) {
proto.attachments = attachmentsArray;
return sendGroupProto(numbers, proto, timestamp);
});
});
}
self.createGroup = function(numbers, name, avatar) {
var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext();
return textsecure.storage.groups.createNewGroup(numbers).then(function(group) {
proto.group.id = toArrayBuffer(group.id);
var numbers = group.numbers;
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
proto.group.members = numbers;
proto.group.name = name;
if (avatar !== undefined) {
return makeAttachmentPointer(avatar).then(function(attachment) {
proto.group.avatar = attachment;
return sendGroupProto(numbers, proto).then(function() {
return proto.group.id;
});
});
} else {
return sendGroupProto(numbers, proto).then(function() { return sendGroupProto(numbers, proto).then(function() {
return proto.group.id; return proto.group.id;
}); });
}); }
} else { });
return sendGroupProto(numbers, proto).then(function() { }
return proto.group.id;
});
}
});
}
self.updateGroup = function(groupId, name, avatar, numbers) { self.updateGroup = function(groupId, name, avatar, numbers) {
var proto = new textsecure.protobuf.DataMessage(); var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext(); proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId); proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE; proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
proto.group.name = name; proto.group.name = name;
return textsecure.storage.groups.addNumbers(groupId, numbers).then(function(numbers) { return textsecure.storage.groups.addNumbers(groupId, numbers).then(function(numbers) {
if (numbers === undefined) { if (numbers === undefined) {
return Promise.reject(new Error("Unknown Group")); return Promise.reject(new Error("Unknown Group"));
} }
proto.group.members = numbers; proto.group.members = numbers;
if (avatar !== undefined && avatar !== null) { if (avatar !== undefined && avatar !== null) {
return makeAttachmentPointer(avatar).then(function(attachment) { return makeAttachmentPointer(avatar).then(function(attachment) {
proto.group.avatar = attachment; proto.group.avatar = attachment;
return sendGroupProto(numbers, proto).then(function() {
return proto.group.id;
});
});
} else {
return sendGroupProto(numbers, proto).then(function() { return sendGroupProto(numbers, proto).then(function() {
return proto.group.id; return proto.group.id;
}); });
}); }
} else { });
return sendGroupProto(numbers, proto).then(function() { }
return proto.group.id;
});
}
});
}
self.addNumberToGroup = function(groupId, number) { self.addNumberToGroup = function(groupId, number) {
var proto = new textsecure.protobuf.DataMessage(); var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext(); proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId); proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE; proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
return textsecure.storage.groups.addNumbers(groupId, [number]).then(function(numbers) { return textsecure.storage.groups.addNumbers(groupId, [number]).then(function(numbers) {
if (numbers === undefined) if (numbers === undefined)
return Promise.reject(new Error("Unknown Group")); return Promise.reject(new Error("Unknown Group"));
proto.group.members = numbers; proto.group.members = numbers;
return sendGroupProto(numbers, proto); return sendGroupProto(numbers, proto);
}); });
} }
self.setGroupName = function(groupId, name) { self.setGroupName = function(groupId, name) {
var proto = new textsecure.protobuf.DataMessage(); var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext(); proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId); proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE; proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
proto.group.name = name; proto.group.name = name;
return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) { return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) {
if (numbers === undefined) if (numbers === undefined)
return Promise.reject(new Error("Unknown Group")); return Promise.reject(new Error("Unknown Group"));
proto.group.members = numbers; proto.group.members = numbers;
return sendGroupProto(numbers, proto); return sendGroupProto(numbers, proto);
}); });
} }
self.setGroupAvatar = function(groupId, avatar) { self.setGroupAvatar = function(groupId, avatar) {
var proto = new textsecure.protobuf.DataMessage(); var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext(); proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId); proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE; proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) { return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) {
if (numbers === undefined) if (numbers === undefined)
return Promise.reject(new Error("Unknown Group")); return Promise.reject(new Error("Unknown Group"));
proto.group.members = numbers; proto.group.members = numbers;
return makeAttachmentPointer(avatar).then(function(attachment) { return makeAttachmentPointer(avatar).then(function(attachment) {
proto.group.avatar = attachment; proto.group.avatar = attachment;
return sendGroupProto(numbers, proto); return sendGroupProto(numbers, proto);
});
}); });
}); }
}
self.leaveGroup = function(groupId) { self.leaveGroup = function(groupId) {
var proto = new textsecure.protobuf.DataMessage(); var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext(); proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId); proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.QUIT; proto.group.type = textsecure.protobuf.GroupContext.Type.QUIT;
return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) { return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) {
if (numbers === undefined) if (numbers === undefined)
return Promise.reject(new Error("Unknown Group")); return Promise.reject(new Error("Unknown Group"));
return textsecure.storage.groups.deleteGroup(groupId).then(function() { return textsecure.storage.groups.deleteGroup(groupId).then(function() {
return sendGroupProto(numbers, proto); return sendGroupProto(numbers, proto);
});
}); });
}); }
}
return self; return self;
}(); })(server);
};
/* /*
* vim: ts=4:sw=4:expandtab * vim: ts=4:sw=4:expandtab

@ -2,438 +2,442 @@
* vim: ts=4:sw=4:expandtab * vim: ts=4:sw=4:expandtab
*/ */
// sendMessage(numbers = [], message = PushMessageContentProto, callback(success/failure map)) // sendMessage(numbers = [], message = PushMessageContentProto, callback(success/failure map))
window.textsecure.messaging = function() { window.textsecure.MessageSender = function(url, username, password) {
'use strict'; 'use strict';
var server = new TextSecureServer(url, username, password);
var self = {}; return (function(TextSecureServer) {
'use strict';
// message == DataMessage or ContentMessage proto
function sendMessageToDevices(timestamp, number, deviceObjectList, message) { var self = {};
var relay = deviceObjectList[0].relay;
for (var i=1; i < deviceObjectList.length; ++i) { // message == DataMessage or ContentMessage proto
if (deviceObjectList[i].relay !== relay) { function sendMessageToDevices(timestamp, number, deviceObjectList, message) {
throw new Error("Mismatched relays for number " + number); var relay = deviceObjectList[0].relay;
for (var i=1; i < deviceObjectList.length; ++i) {
if (deviceObjectList[i].relay !== relay) {
throw new Error("Mismatched relays for number " + number);
}
} }
} return Promise.all(deviceObjectList.map(function(device) {
return Promise.all(deviceObjectList.map(function(device) { return textsecure.protocol_wrapper.encryptMessageFor(device, message).then(function(encryptedMsg) {
return textsecure.protocol_wrapper.encryptMessageFor(device, message).then(function(encryptedMsg) { return textsecure.protocol_wrapper.getRegistrationId(device.encodedNumber).then(function(registrationId) {
return textsecure.protocol_wrapper.getRegistrationId(device.encodedNumber).then(function(registrationId) { return textsecure.storage.devices.removeTempKeysFromDevice(device.encodedNumber).then(function() {
return textsecure.storage.devices.removeTempKeysFromDevice(device.encodedNumber).then(function() { var json = {
var json = { type: encryptedMsg.type,
type: encryptedMsg.type, destinationDeviceId: textsecure.utils.unencodeNumber(device.encodedNumber)[1],
destinationDeviceId: textsecure.utils.unencodeNumber(device.encodedNumber)[1], destinationRegistrationId: registrationId,
destinationRegistrationId: registrationId, content: encryptedMsg.body,
content: encryptedMsg.body, timestamp: timestamp
timestamp: timestamp };
};
if (device.relay !== undefined) {
if (device.relay !== undefined) { json.relay = device.relay;
json.relay = device.relay; }
}
return json; return json;
});
}); });
}); });
})).then(function(jsonData) {
var legacy = (message instanceof textsecure.protobuf.DataMessage);
return TextSecureServer.sendMessages(number, jsonData, legacy);
}); });
})).then(function(jsonData) { }
var legacy = (message instanceof textsecure.protobuf.DataMessage);
return TextSecureServer.sendMessages(number, jsonData, legacy);
});
}
function makeAttachmentPointer(attachment) {
var proto = new textsecure.protobuf.AttachmentPointer();
proto.key = textsecure.crypto.getRandomBytes(64);
var iv = textsecure.crypto.getRandomBytes(16);
return textsecure.crypto.encryptAttachment(attachment.data, proto.key, iv).then(function(encryptedBin) {
return TextSecureServer.putAttachment(encryptedBin).then(function(id) {
proto.id = id;
proto.contentType = attachment.contentType;
return proto;
});
});
}
var tryMessageAgain = function(number, encodedMessage, timestamp) {
var proto = textsecure.protobuf.DataMessage.decode(encodedMessage);
return new Promise(function(resolve, reject) {
sendMessageProto(timestamp, [number], proto, function(res) {
if (res.failure.length > 0)
reject(res.failure);
else
resolve();
});
});
};
textsecure.replay.registerFunction(tryMessageAgain, textsecure.replay.Type.SEND_MESSAGE);
var sendMessageProto = function(timestamp, numbers, message, callback) {
var numbersCompleted = 0;
var errors = [];
var successfulNumbers = [];
var numberCompleted = function() {
numbersCompleted++;
if (numbersCompleted >= numbers.length)
callback({success: successfulNumbers, failure: errors});
};
var registerError = function(number, reason, error) { function makeAttachmentPointer(attachment) {
if (!error) { var proto = new textsecure.protobuf.AttachmentPointer();
error = new Error(reason); proto.key = textsecure.crypto.getRandomBytes(64);
}
error.number = number;
error.reason = reason;
errors[errors.length] = error;
numberCompleted();
};
var reloadDevicesAndSend = function(number, recurse) { var iv = textsecure.crypto.getRandomBytes(16);
return function() { return textsecure.crypto.encryptAttachment(attachment.data, proto.key, iv).then(function(encryptedBin) {
return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) { return TextSecureServer.putAttachment(encryptedBin).then(function(id) {
if (devicesForNumber.length == 0) proto.id = id;
return registerError(number, "Got empty device list when loading device keys", null); proto.contentType = attachment.contentType;
doSendMessage(number, devicesForNumber, recurse); return proto;
}); });
} });
}; }
function getKeysForNumber(number, updateDevices) { var tryMessageAgain = function(number, encodedMessage, timestamp) {
var handleResult = function(response) { var proto = textsecure.protobuf.DataMessage.decode(encodedMessage);
return Promise.all(response.devices.map(function(device) { return new Promise(function(resolve, reject) {
if (updateDevices === undefined || updateDevices.indexOf(device.deviceId) > -1) sendMessageProto(timestamp, [number], proto, function(res) {
return textsecure.storage.devices.saveKeysToDeviceObject({ if (res.failure.length > 0)
encodedNumber: number + "." + device.deviceId, reject(res.failure);
identityKey: response.identityKey, else
preKey: device.preKey.publicKey, resolve();
preKeyId: device.preKey.keyId, });
signedKey: device.signedPreKey.publicKey, });
signedKeyId: device.signedPreKey.keyId, };
signedKeySignature: device.signedPreKey.signature, textsecure.replay.registerFunction(tryMessageAgain, textsecure.replay.Type.SEND_MESSAGE);
registrationId: device.registrationId
}).catch(function(error) {
if (error.message === "Identity key changed") {
error = new textsecure.OutgoingIdentityKeyError(number, message.toArrayBuffer(), timestamp, error.identityKey);
registerError(number, "Identity key changed", error);
}
throw error;
});
}));
};
if (updateDevices === undefined) { var sendMessageProto = function(timestamp, numbers, message, callback) {
return TextSecureServer.getKeysForNumber(number).then(handleResult); var numbersCompleted = 0;
} else { var errors = [];
var promises = []; var successfulNumbers = [];
for (var i in updateDevices)
promises[promises.length] = TextSecureServer.getKeysForNumber(number, updateDevices[i]).then(handleResult);
return Promise.all(promises); var numberCompleted = function() {
} numbersCompleted++;
} if (numbersCompleted >= numbers.length)
callback({success: successfulNumbers, failure: errors});
};
var doSendMessage = function(number, devicesForNumber, recurse) { var registerError = function(number, reason, error) {
return sendMessageToDevices(timestamp, number, devicesForNumber, message).then(function(result) { if (!error) {
successfulNumbers[successfulNumbers.length] = number; error = new Error(reason);
}
error.number = number;
error.reason = reason;
errors[errors.length] = error;
numberCompleted(); numberCompleted();
}).catch(function(error) { };
if (error instanceof Error && error.name == "HTTPError" && (error.code == 410 || error.code == 409)) {
if (!recurse)
return registerError(number, "Hit retry limit attempting to reload device list", error);
var p;
if (error.code == 409) {
p = textsecure.storage.devices.removeDeviceIdsForNumber(number, error.response.extraDevices);
} else {
p = Promise.all(error.response.staleDevices.map(function(deviceId) {
return textsecure.protocol_wrapper.closeOpenSessionForDevice(number + '.' + deviceId);
}));
}
p.then(function() { var reloadDevicesAndSend = function(number, recurse) {
var resetDevices = ((error.code == 410) ? error.response.staleDevices : error.response.missingDevices); return function() {
getKeysForNumber(number, resetDevices) return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) {
.then(reloadDevicesAndSend(number, (error.code == 409))) if (devicesForNumber.length == 0)
.catch(function(error) { return registerError(number, "Got empty device list when loading device keys", null);
registerError(number, "Failed to reload device keys", error); doSendMessage(number, devicesForNumber, recurse);
});
}); });
}
};
function getKeysForNumber(number, updateDevices) {
var handleResult = function(response) {
return Promise.all(response.devices.map(function(device) {
if (updateDevices === undefined || updateDevices.indexOf(device.deviceId) > -1)
return textsecure.storage.devices.saveKeysToDeviceObject({
encodedNumber: number + "." + device.deviceId,
identityKey: response.identityKey,
preKey: device.preKey.publicKey,
preKeyId: device.preKey.keyId,
signedKey: device.signedPreKey.publicKey,
signedKeyId: device.signedPreKey.keyId,
signedKeySignature: device.signedPreKey.signature,
registrationId: device.registrationId
}).catch(function(error) {
if (error.message === "Identity key changed") {
error = new textsecure.OutgoingIdentityKeyError(number, message.toArrayBuffer(), timestamp, error.identityKey);
registerError(number, "Identity key changed", error);
}
throw error;
});
}));
};
if (updateDevices === undefined) {
return TextSecureServer.getKeysForNumber(number).then(handleResult);
} else { } else {
registerError(number, "Failed to create or send message", error); var promises = [];
for (var i in updateDevices)
promises[promises.length] = TextSecureServer.getKeysForNumber(number, updateDevices[i]).then(handleResult);
return Promise.all(promises);
} }
}); }
};
numbers.forEach(function(number) { var doSendMessage = function(number, devicesForNumber, recurse) {
textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) { return sendMessageToDevices(timestamp, number, devicesForNumber, message).then(function(result) {
return Promise.all(devicesForNumber.map(function(device) { successfulNumbers[successfulNumbers.length] = number;
return textsecure.protocol_wrapper.hasOpenSession(device.encodedNumber).then(function(result) { numberCompleted();
if (!result) }).catch(function(error) {
return getKeysForNumber(number, [parseInt(textsecure.utils.unencodeNumber(device.encodedNumber)[1])]); if (error instanceof Error && error.name == "HTTPError" && (error.code == 410 || error.code == 409)) {
}); if (!recurse)
})).then(function() { return registerError(number, "Hit retry limit attempting to reload device list", error);
return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) {
if (devicesForNumber.length == 0) { var p;
getKeysForNumber(number, [1]) if (error.code == 409) {
.then(reloadDevicesAndSend(number, true)) p = textsecure.storage.devices.removeDeviceIdsForNumber(number, error.response.extraDevices);
} else {
p = Promise.all(error.response.staleDevices.map(function(deviceId) {
return textsecure.protocol_wrapper.closeOpenSessionForDevice(number + '.' + deviceId);
}));
}
p.then(function() {
var resetDevices = ((error.code == 410) ? error.response.staleDevices : error.response.missingDevices);
getKeysForNumber(number, resetDevices)
.then(reloadDevicesAndSend(number, (error.code == 409)))
.catch(function(error) { .catch(function(error) {
registerError(number, "Failed to retreive new device keys for number " + number, error); registerError(number, "Failed to reload device keys", error);
}); });
} else });
doSendMessage(number, devicesForNumber, true); } else {
registerError(number, "Failed to create or send message", error);
}
});
};
numbers.forEach(function(number) {
textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) {
return Promise.all(devicesForNumber.map(function(device) {
return textsecure.protocol_wrapper.hasOpenSession(device.encodedNumber).then(function(result) {
if (!result)
return getKeysForNumber(number, [parseInt(textsecure.utils.unencodeNumber(device.encodedNumber)[1])]);
});
})).then(function() {
return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devicesForNumber) {
if (devicesForNumber.length == 0) {
getKeysForNumber(number, [1])
.then(reloadDevicesAndSend(number, true))
.catch(function(error) {
registerError(number, "Failed to retreive new device keys for number " + number, error);
});
} else
doSendMessage(number, devicesForNumber, true);
});
}); });
}); });
}); });
});
}
var sendIndividualProto = function(number, proto, timestamp) {
return new Promise(function(resolve, reject) {
sendMessageProto(timestamp, [number], proto, function(res) {
if (res.failure.length > 0)
reject(res.failure);
else
resolve();
});
});
}
var sendSyncMessage = function(message, timestamp, destination) {
var myNumber = textsecure.storage.user.getNumber();
var myDevice = textsecure.storage.user.getDeviceId();
if (myDevice != 1) {
var sentMessage = new textsecure.protobuf.SyncMessage.Sent();
sentMessage.timestamp = timestamp;
sentMessage.message = message;
if (destination) {
sentMessage.destination = destination;
}
var syncMessage = new textsecure.protobuf.SyncMessage();
syncMessage.sent = sentMessage;
var contentMessage = new textsecure.protobuf.Content();
contentMessage.syncMessage = syncMessage;
return sendIndividualProto(myNumber, contentMessage, Date.now());
} }
}
var sendIndividualProto = function(number, proto, timestamp) {
self.sendRequestGroupSyncMessage = function() { return new Promise(function(resolve, reject) {
var myNumber = textsecure.storage.user.getNumber(); sendMessageProto(timestamp, [number], proto, function(res) {
var myDevice = textsecure.storage.user.getDeviceId(); if (res.failure.length > 0)
if (myDevice != 1) { reject(res.failure);
var request = new textsecure.protobuf.SyncMessage.Request(); else
request.type = textsecure.protobuf.SyncMessage.Request.Type.GROUPS; resolve();
var syncMessage = new textsecure.protobuf.SyncMessage(); });
syncMessage.request = request; });
var contentMessage = new textsecure.protobuf.Content();
contentMessage.syncMessage = syncMessage;
return sendIndividualProto(myNumber, contentMessage, Date.now());
} }
};
self.sendRequestContactSyncMessage = function() { var sendSyncMessage = function(message, timestamp, destination) {
var myNumber = textsecure.storage.user.getNumber(); var myNumber = textsecure.storage.user.getNumber();
var myDevice = textsecure.storage.user.getDeviceId(); var myDevice = textsecure.storage.user.getDeviceId();
if (myDevice != 1) { if (myDevice != 1) {
var request = new textsecure.protobuf.SyncMessage.Request(); var sentMessage = new textsecure.protobuf.SyncMessage.Sent();
request.type = textsecure.protobuf.SyncMessage.Request.Type.CONTACTS; sentMessage.timestamp = timestamp;
var syncMessage = new textsecure.protobuf.SyncMessage(); sentMessage.message = message;
syncMessage.request = request; if (destination) {
var contentMessage = new textsecure.protobuf.Content(); sentMessage.destination = destination;
contentMessage.syncMessage = syncMessage; }
var syncMessage = new textsecure.protobuf.SyncMessage();
return sendIndividualProto(myNumber, contentMessage, Date.now()); syncMessage.sent = sentMessage;
var contentMessage = new textsecure.protobuf.Content();
contentMessage.syncMessage = syncMessage;
return sendIndividualProto(myNumber, contentMessage, Date.now());
}
} }
};
self.sendRequestGroupSyncMessage = function() {
var sendGroupProto = function(numbers, proto, timestamp) { var myNumber = textsecure.storage.user.getNumber();
timestamp = timestamp || Date.now(); var myDevice = textsecure.storage.user.getDeviceId();
var me = textsecure.storage.user.getNumber(); if (myDevice != 1) {
numbers = numbers.filter(function(number) { return number != me; }); var request = new textsecure.protobuf.SyncMessage.Request();
request.type = textsecure.protobuf.SyncMessage.Request.Type.GROUPS;
return new Promise(function(resolve, reject) { var syncMessage = new textsecure.protobuf.SyncMessage();
sendMessageProto(timestamp, numbers, proto, function(res) { syncMessage.request = request;
if (res.failure.length > 0) var contentMessage = new textsecure.protobuf.Content();
reject(res.failure); contentMessage.syncMessage = syncMessage;
else
resolve(); return sendIndividualProto(myNumber, contentMessage, Date.now());
}); }
}).then(function() { };
return sendSyncMessage(proto, timestamp); self.sendRequestContactSyncMessage = function() {
}); var myNumber = textsecure.storage.user.getNumber();
} var myDevice = textsecure.storage.user.getDeviceId();
if (myDevice != 1) {
self.sendMessageToNumber = function(number, messageText, attachments, timestamp) { var request = new textsecure.protobuf.SyncMessage.Request();
var proto = new textsecure.protobuf.DataMessage(); request.type = textsecure.protobuf.SyncMessage.Request.Type.CONTACTS;
proto.body = messageText; var syncMessage = new textsecure.protobuf.SyncMessage();
syncMessage.request = request;
var promises = []; var contentMessage = new textsecure.protobuf.Content();
for (var i in attachments) contentMessage.syncMessage = syncMessage;
promises.push(makeAttachmentPointer(attachments[i]));
return Promise.all(promises).then(function(attachmentsArray) { return sendIndividualProto(myNumber, contentMessage, Date.now());
proto.attachments = attachmentsArray; }
return sendIndividualProto(number, proto, timestamp).then(function() { };
return sendSyncMessage(proto, timestamp, number);
}); var sendGroupProto = function(numbers, proto, timestamp) {
}); timestamp = timestamp || Date.now();
} var me = textsecure.storage.user.getNumber();
numbers = numbers.filter(function(number) { return number != me; });
self.closeSession = function(number) {
var proto = new textsecure.protobuf.DataMessage(); return new Promise(function(resolve, reject) {
proto.body = "TERMINATE"; sendMessageProto(timestamp, numbers, proto, function(res) {
proto.flags = textsecure.protobuf.DataMessage.Flags.END_SESSION; if (res.failure.length > 0)
return sendIndividualProto(number, proto, Date.now()).then(function(res) { reject(res.failure);
return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devices) { else
return Promise.all(devices.map(function(device) { resolve();
return textsecure.protocol_wrapper.closeOpenSessionForDevice(device.encodedNumber);
})).then(function() {
return res;
}); });
}).then(function() {
return sendSyncMessage(proto, timestamp);
}); });
}); }
}
self.sendMessageToGroup = function(groupId, messageText, attachments, timestamp) {
var proto = new textsecure.protobuf.DataMessage();
proto.body = messageText;
proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.DELIVER;
return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) { self.sendMessageToNumber = function(number, messageText, attachments, timestamp) {
if (numbers === undefined) var proto = new textsecure.protobuf.DataMessage();
return Promise.reject(new Error("Unknown Group")); proto.body = messageText;
var promises = []; var promises = [];
for (var i in attachments) for (var i in attachments)
promises.push(makeAttachmentPointer(attachments[i])); promises.push(makeAttachmentPointer(attachments[i]));
return Promise.all(promises).then(function(attachmentsArray) { return Promise.all(promises).then(function(attachmentsArray) {
proto.attachments = attachmentsArray; proto.attachments = attachmentsArray;
return sendGroupProto(numbers, proto, timestamp); return sendIndividualProto(number, proto, timestamp).then(function() {
return sendSyncMessage(proto, timestamp, number);
});
}); });
}); }
}
self.createGroup = function(numbers, name, avatar) { self.closeSession = function(number) {
var proto = new textsecure.protobuf.DataMessage(); var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext(); proto.body = "TERMINATE";
proto.flags = textsecure.protobuf.DataMessage.Flags.END_SESSION;
return sendIndividualProto(number, proto, Date.now()).then(function(res) {
return textsecure.storage.devices.getDeviceObjectsForNumber(number).then(function(devices) {
return Promise.all(devices.map(function(device) {
return textsecure.protocol_wrapper.closeOpenSessionForDevice(device.encodedNumber);
})).then(function() {
return res;
});
});
});
}
return textsecure.storage.groups.createNewGroup(numbers).then(function(group) { self.sendMessageToGroup = function(groupId, messageText, attachments, timestamp) {
proto.group.id = toArrayBuffer(group.id); var proto = new textsecure.protobuf.DataMessage();
var numbers = group.numbers; proto.body = messageText;
proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.DELIVER;
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE; return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) {
proto.group.members = numbers; if (numbers === undefined)
proto.group.name = name; return Promise.reject(new Error("Unknown Group"));
if (avatar !== undefined) { var promises = [];
return makeAttachmentPointer(avatar).then(function(attachment) { for (var i in attachments)
proto.group.avatar = attachment; promises.push(makeAttachmentPointer(attachments[i]));
return Promise.all(promises).then(function(attachmentsArray) {
proto.attachments = attachmentsArray;
return sendGroupProto(numbers, proto, timestamp);
});
});
}
self.createGroup = function(numbers, name, avatar) {
var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext();
return textsecure.storage.groups.createNewGroup(numbers).then(function(group) {
proto.group.id = toArrayBuffer(group.id);
var numbers = group.numbers;
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
proto.group.members = numbers;
proto.group.name = name;
if (avatar !== undefined) {
return makeAttachmentPointer(avatar).then(function(attachment) {
proto.group.avatar = attachment;
return sendGroupProto(numbers, proto).then(function() {
return proto.group.id;
});
});
} else {
return sendGroupProto(numbers, proto).then(function() { return sendGroupProto(numbers, proto).then(function() {
return proto.group.id; return proto.group.id;
}); });
}); }
} else { });
return sendGroupProto(numbers, proto).then(function() { }
return proto.group.id;
});
}
});
}
self.updateGroup = function(groupId, name, avatar, numbers) { self.updateGroup = function(groupId, name, avatar, numbers) {
var proto = new textsecure.protobuf.DataMessage(); var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext(); proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId); proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE; proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
proto.group.name = name; proto.group.name = name;
return textsecure.storage.groups.addNumbers(groupId, numbers).then(function(numbers) { return textsecure.storage.groups.addNumbers(groupId, numbers).then(function(numbers) {
if (numbers === undefined) { if (numbers === undefined) {
return Promise.reject(new Error("Unknown Group")); return Promise.reject(new Error("Unknown Group"));
} }
proto.group.members = numbers; proto.group.members = numbers;
if (avatar !== undefined && avatar !== null) { if (avatar !== undefined && avatar !== null) {
return makeAttachmentPointer(avatar).then(function(attachment) { return makeAttachmentPointer(avatar).then(function(attachment) {
proto.group.avatar = attachment; proto.group.avatar = attachment;
return sendGroupProto(numbers, proto).then(function() {
return proto.group.id;
});
});
} else {
return sendGroupProto(numbers, proto).then(function() { return sendGroupProto(numbers, proto).then(function() {
return proto.group.id; return proto.group.id;
}); });
}); }
} else { });
return sendGroupProto(numbers, proto).then(function() { }
return proto.group.id;
}); self.addNumberToGroup = function(groupId, number) {
} var proto = new textsecure.protobuf.DataMessage();
}); proto.group = new textsecure.protobuf.GroupContext();
} proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
self.addNumberToGroup = function(groupId, number) {
var proto = new textsecure.protobuf.DataMessage(); return textsecure.storage.groups.addNumbers(groupId, [number]).then(function(numbers) {
proto.group = new textsecure.protobuf.GroupContext(); if (numbers === undefined)
proto.group.id = toArrayBuffer(groupId); return Promise.reject(new Error("Unknown Group"));
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE; proto.group.members = numbers;
return textsecure.storage.groups.addNumbers(groupId, [number]).then(function(numbers) {
if (numbers === undefined)
return Promise.reject(new Error("Unknown Group"));
proto.group.members = numbers;
return sendGroupProto(numbers, proto);
});
}
self.setGroupName = function(groupId, name) {
var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
proto.group.name = name;
return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) {
if (numbers === undefined)
return Promise.reject(new Error("Unknown Group"));
proto.group.members = numbers;
return sendGroupProto(numbers, proto);
});
}
self.setGroupAvatar = function(groupId, avatar) {
var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) {
if (numbers === undefined)
return Promise.reject(new Error("Unknown Group"));
proto.group.members = numbers;
return makeAttachmentPointer(avatar).then(function(attachment) {
proto.group.avatar = attachment;
return sendGroupProto(numbers, proto); return sendGroupProto(numbers, proto);
}); });
}); }
}
self.setGroupName = function(groupId, name) {
self.leaveGroup = function(groupId) { var proto = new textsecure.protobuf.DataMessage();
var proto = new textsecure.protobuf.DataMessage(); proto.group = new textsecure.protobuf.GroupContext();
proto.group = new textsecure.protobuf.GroupContext(); proto.group.id = toArrayBuffer(groupId);
proto.group.id = toArrayBuffer(groupId); proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
proto.group.type = textsecure.protobuf.GroupContext.Type.QUIT; proto.group.name = name;
return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) { return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) {
if (numbers === undefined) if (numbers === undefined)
return Promise.reject(new Error("Unknown Group")); return Promise.reject(new Error("Unknown Group"));
return textsecure.storage.groups.deleteGroup(groupId).then(function() { proto.group.members = numbers;
return sendGroupProto(numbers, proto); return sendGroupProto(numbers, proto);
}); });
}); }
}
self.setGroupAvatar = function(groupId, avatar) {
var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.UPDATE;
return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) {
if (numbers === undefined)
return Promise.reject(new Error("Unknown Group"));
proto.group.members = numbers;
return makeAttachmentPointer(avatar).then(function(attachment) {
proto.group.avatar = attachment;
return sendGroupProto(numbers, proto);
});
});
}
self.leaveGroup = function(groupId) {
var proto = new textsecure.protobuf.DataMessage();
proto.group = new textsecure.protobuf.GroupContext();
proto.group.id = toArrayBuffer(groupId);
proto.group.type = textsecure.protobuf.GroupContext.Type.QUIT;
return textsecure.storage.groups.getNumbers(groupId).then(function(numbers) {
if (numbers === undefined)
return Promise.reject(new Error("Unknown Group"));
return textsecure.storage.groups.deleteGroup(groupId).then(function() {
return sendGroupProto(numbers, proto);
});
});
}
return self; return self;
}(); })(server);
};

Loading…
Cancel
Save