2015-01-15 00:42:01 +01:00
|
|
|
//TODO: Remove almost everything here...
|
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
;(function() {
|
|
|
|
window.axolotl = window.axolotl || {};
|
|
|
|
window.axolotl.api = {
|
2015-01-16 05:29:32 +01:00
|
|
|
getMyRegistrationId: function() {
|
|
|
|
return textsecure.storage.getUnencrypted("registrationId");
|
|
|
|
},
|
2015-01-15 00:42:01 +01:00
|
|
|
storage: {
|
|
|
|
put: function(key, value) {
|
|
|
|
return textsecure.storage.putEncrypted(key, value);
|
|
|
|
},
|
|
|
|
get: function(key, defaultValue) {
|
|
|
|
return textsecure.storage.getEncrypted(key, defaultValue);
|
|
|
|
},
|
|
|
|
remove: function(key) {
|
|
|
|
return textsecure.storage.removeEncrypted(key);
|
|
|
|
},
|
2015-01-21 06:46:41 +01:00
|
|
|
|
|
|
|
sessions: {
|
|
|
|
get: function(identifier) {
|
2015-01-19 20:56:48 +01:00
|
|
|
var device = textsecure.storage.devices.getDeviceObject(identifier, true);
|
|
|
|
if (device === undefined)
|
|
|
|
return undefined;
|
|
|
|
var record = new axolotl.sessions.RecipientRecord();
|
|
|
|
if (device.sessions !== undefined)
|
|
|
|
record.deserialize(device.sessions);
|
|
|
|
else
|
|
|
|
// TODO: (even for numbers, not devices) We MUST return an identityKey if we have one available
|
|
|
|
record.identityKey = device.identityKey;
|
|
|
|
if (getString(device.identityKey) !== getString(record.identityKey))
|
|
|
|
throw new Error("Got mismatched identity key on sessions load");
|
|
|
|
return record;
|
2015-01-21 06:46:41 +01:00
|
|
|
},
|
2015-01-19 20:56:48 +01:00
|
|
|
put: function(identifier, record) {
|
|
|
|
var device = textsecure.storage.devices.getDeviceObject(identifier);
|
|
|
|
if (device === undefined) {
|
|
|
|
device = { encodedNumber: identifier,
|
|
|
|
//TODO: Remove this duplication (esp registrationId?)
|
|
|
|
identityKey: record.identityKey,
|
|
|
|
registrationId: record.registrationId
|
|
|
|
};
|
|
|
|
}
|
|
|
|
if (getString(device.identityKey) !== getString(record.identityKey))
|
|
|
|
throw new Error("Tried to put session for device with changed identity key");
|
|
|
|
device.sessions = record.serialize();
|
|
|
|
return textsecure.storage.devices.saveDeviceObject(device);
|
2015-01-21 06:46:41 +01:00
|
|
|
}
|
|
|
|
}
|
2015-01-15 00:42:01 +01:00
|
|
|
},
|
2015-01-16 01:30:55 +01:00
|
|
|
updateKeys: function(keys) {
|
|
|
|
return textsecure.api.registerKeys(keys).catch(function(e) {
|
|
|
|
//TODO: Notify the user somehow?
|
|
|
|
console.error(e);
|
|
|
|
});
|
|
|
|
},
|
2015-01-15 00:42:01 +01:00
|
|
|
};
|
2015-01-15 21:42:32 +01:00
|
|
|
|
2015-01-16 01:30:55 +01:00
|
|
|
var decodeMessageContents = function(res) {
|
|
|
|
var finalMessage = textsecure.protobuf.PushMessageContent.decode(res[0]);
|
|
|
|
|
2015-01-18 22:10:39 +01:00
|
|
|
if ((finalMessage.flags & textsecure.protobuf.PushMessageContent.Flags.END_SESSION)
|
|
|
|
== textsecure.protobuf.PushMessageContent.Flags.END_SESSION)
|
|
|
|
res[1]();
|
|
|
|
|
|
|
|
return finalMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
var decodeDeviceContents = function(res) {
|
|
|
|
var finalMessage = textsecure.protobuf.DeviceControl.decode(res[0]);
|
|
|
|
|
|
|
|
//TODO: Add END_SESSION flag for device control messages
|
2015-01-16 01:30:55 +01:00
|
|
|
/*if ((finalMessage.flags & textsecure.protobuf.PushMessageContent.Flags.END_SESSION)
|
|
|
|
== textsecure.protobuf.PushMessageContent.Flags.END_SESSION)
|
2015-01-18 22:10:39 +01:00
|
|
|
res[1]();*/
|
2015-01-16 01:30:55 +01:00
|
|
|
|
|
|
|
return finalMessage;
|
|
|
|
}
|
|
|
|
|
2015-01-15 21:42:32 +01:00
|
|
|
window.textsecure = window.textsecure || {};
|
|
|
|
window.textsecure.protocol_wrapper = {
|
|
|
|
handleIncomingPushMessageProto: function(proto) {
|
|
|
|
switch(proto.type) {
|
|
|
|
case textsecure.protobuf.IncomingPushMessageSignal.Type.PLAINTEXT:
|
|
|
|
return Promise.resolve(textsecure.protobuf.PushMessageContent.decode(proto.message));
|
|
|
|
case textsecure.protobuf.IncomingPushMessageSignal.Type.CIPHERTEXT:
|
|
|
|
var from = proto.source + "." + (proto.sourceDevice == null ? 0 : proto.sourceDevice);
|
2015-01-16 06:18:31 +01:00
|
|
|
return axolotl.protocol.decryptWhisperMessage(from, getString(proto.message)).then(decodeMessageContents);
|
2015-01-15 21:42:32 +01:00
|
|
|
case textsecure.protobuf.IncomingPushMessageSignal.Type.PREKEY_BUNDLE:
|
|
|
|
if (proto.message.readUint8() != ((3 << 4) | 3))
|
|
|
|
throw new Error("Bad version byte");
|
|
|
|
var from = proto.source + "." + (proto.sourceDevice == null ? 0 : proto.sourceDevice);
|
2015-01-16 06:18:31 +01:00
|
|
|
return axolotl.protocol.handlePreKeyWhisperMessage(from, getString(proto.message)).then(decodeMessageContents);
|
2015-01-15 21:42:32 +01:00
|
|
|
case textsecure.protobuf.IncomingPushMessageSignal.Type.RECEIPT:
|
|
|
|
return Promise.resolve(null);
|
|
|
|
case textsecure.protobuf.IncomingPushMessageSignal.Type.PREKEY_BUNDLE_DEVICE_CONTROL:
|
|
|
|
if (proto.message.readUint8() != ((3 << 4) | 3))
|
|
|
|
throw new Error("Bad version byte");
|
|
|
|
var from = proto.source + "." + (proto.sourceDevice == null ? 0 : proto.sourceDevice);
|
2015-01-18 22:10:39 +01:00
|
|
|
return axolotl.protocol.handlePreKeyWhisperMessage(from, getString(proto.message)).then(decodeDeviceContents);
|
2015-01-15 21:42:32 +01:00
|
|
|
case textsecure.protobuf.IncomingPushMessageSignal.Type.DEVICE_CONTROL:
|
|
|
|
var from = proto.source + "." + (proto.sourceDevice == null ? 0 : proto.sourceDevice);
|
2015-01-18 22:10:39 +01:00
|
|
|
return axolotl.protocol.decryptWhisperMessage(from, getString(proto.message)).then(decodeDeviceContents);
|
2015-01-15 21:42:32 +01:00
|
|
|
default:
|
|
|
|
return new Promise(function(resolve, reject) { reject(new Error("Unknown message type")); });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2015-01-16 01:30:55 +01:00
|
|
|
|
|
|
|
var wipeIdentityAndTryMessageAgain = function(from, encodedMessage, message_id) {
|
|
|
|
// Wipe identity key!
|
|
|
|
//TODO: Encapsuate with the rest of textsecure.storage.devices
|
|
|
|
textsecure.storage.removeEncrypted("devices" + from.split('.')[0]);
|
|
|
|
//TODO: Probably breaks with a devicecontrol message
|
2015-01-16 06:18:31 +01:00
|
|
|
return axolotl.protocol.handlePreKeyWhisperMessage(from, encodedMessage).then(decodeMessageContents).then(
|
2015-01-16 01:30:55 +01:00
|
|
|
function(pushMessageContent) {
|
|
|
|
extension.trigger('message:decrypted', {
|
|
|
|
message_id : message_id,
|
|
|
|
data : pushMessageContent
|
|
|
|
});
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
textsecure.replay.registerFunction(wipeIdentityAndTryMessageAgain, textsecure.replay.Type.INIT_SESSION);
|
2015-01-15 00:42:01 +01:00
|
|
|
})();
|