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.
		
		
		
		
		
			
		
			
				
	
	
		
			223 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			JavaScript
		
	
			
		
		
	
	
			223 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			JavaScript
		
	
/* global window, libsignal, textsecure */
 | 
						|
 | 
						|
// eslint-disable-next-line func-names
 | 
						|
(function() {
 | 
						|
  window.libloki = window.libloki || {};
 | 
						|
 | 
						|
  async function getPreKeyBundleForContact(pubKey) {
 | 
						|
    const myKeyPair = await textsecure.storage.protocol.getIdentityKeyPair();
 | 
						|
    const identityKey = myKeyPair.pubKey;
 | 
						|
 | 
						|
    // Retrieve ids. The ids stored are always the latest generated + 1
 | 
						|
    const signedKeyId = textsecure.storage.get('signedKeyId', 2) - 1;
 | 
						|
 | 
						|
    const [signedKey, preKey] = await Promise.all([
 | 
						|
      textsecure.storage.protocol.loadSignedPreKey(signedKeyId),
 | 
						|
      new Promise(async resolve => {
 | 
						|
        // retrieve existing prekey if we already generated one for that recipient
 | 
						|
        const storedPreKey = await textsecure.storage.protocol.loadPreKeyForContact(
 | 
						|
          pubKey
 | 
						|
        );
 | 
						|
        if (storedPreKey) {
 | 
						|
          resolve({ pubKey: storedPreKey.pubKey, keyId: storedPreKey.keyId });
 | 
						|
        } else {
 | 
						|
          // generate and store new prekey
 | 
						|
          const preKeyId = textsecure.storage.get('maxPreKeyId', 1);
 | 
						|
          textsecure.storage.put('maxPreKeyId', preKeyId + 1);
 | 
						|
          const newPreKey = await libsignal.KeyHelper.generatePreKey(preKeyId);
 | 
						|
          await textsecure.storage.protocol.storePreKey(
 | 
						|
            newPreKey.keyId,
 | 
						|
            newPreKey.keyPair,
 | 
						|
            pubKey
 | 
						|
          );
 | 
						|
          resolve({ pubKey: newPreKey.keyPair.pubKey, keyId: preKeyId });
 | 
						|
        }
 | 
						|
      }),
 | 
						|
    ]);
 | 
						|
 | 
						|
    return {
 | 
						|
      identityKey: new Uint8Array(identityKey),
 | 
						|
      deviceId: 1, // TODO: fetch from somewhere
 | 
						|
      preKeyId: preKey.keyId,
 | 
						|
      signedKeyId,
 | 
						|
      preKey: new Uint8Array(preKey.pubKey),
 | 
						|
      signedKey: new Uint8Array(signedKey.pubKey),
 | 
						|
      signature: new Uint8Array(signedKey.signature),
 | 
						|
    };
 | 
						|
  }
 | 
						|
  async function removeContactPreKeyBundle(pubKey) {
 | 
						|
    await Promise.all([
 | 
						|
      textsecure.storage.protocol.removeContactPreKey(pubKey),
 | 
						|
      textsecure.storage.protocol.removeContactSignedPreKey(pubKey),
 | 
						|
    ]);
 | 
						|
  }
 | 
						|
 | 
						|
  async function verifyFriendRequestAcceptPreKey(pubKey, buffer) {
 | 
						|
    const storedPreKey = await textsecure.storage.protocol.loadPreKeyForContact(
 | 
						|
      pubKey
 | 
						|
    );
 | 
						|
    if (!storedPreKey) {
 | 
						|
      throw new Error(
 | 
						|
        'Received a friend request from a pubkey for which no prekey bundle was created'
 | 
						|
      );
 | 
						|
    }
 | 
						|
    // need to pop the version
 | 
						|
    // eslint-disable-next-line no-unused-vars
 | 
						|
    const version = buffer.readUint8();
 | 
						|
    const preKeyProto = window.textsecure.protobuf.PreKeyWhisperMessage.decode(
 | 
						|
      buffer
 | 
						|
    );
 | 
						|
    if (!preKeyProto) {
 | 
						|
      throw new Error(
 | 
						|
        'Could not decode PreKeyWhisperMessage while attempting to match the preKeyId'
 | 
						|
      );
 | 
						|
    }
 | 
						|
    const { preKeyId } = preKeyProto;
 | 
						|
    if (storedPreKey.keyId !== preKeyId) {
 | 
						|
      throw new Error(
 | 
						|
        'Received a preKeyWhisperMessage (friend request accept) from an unknown source'
 | 
						|
      );
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  function getGuardNodes() {
 | 
						|
    return window.Signal.Data.getGuardNodes();
 | 
						|
  }
 | 
						|
 | 
						|
  function updateGuardNodes(nodes) {
 | 
						|
    return window.Signal.Data.updateGuardNodes(nodes);
 | 
						|
  }
 | 
						|
 | 
						|
  window.libloki.storage = {
 | 
						|
    getPreKeyBundleForContact,
 | 
						|
    removeContactPreKeyBundle,
 | 
						|
    verifyFriendRequestAcceptPreKey,
 | 
						|
    getGuardNodes,
 | 
						|
    updateGuardNodes,
 | 
						|
  };
 | 
						|
 | 
						|
  // Libloki protocol store
 | 
						|
 | 
						|
  const store = window.SignalProtocolStore.prototype;
 | 
						|
 | 
						|
  store.storeContactPreKey = async (pubKey, preKey) => {
 | 
						|
    const key = {
 | 
						|
      // id: (autoincrement)
 | 
						|
      identityKeyString: pubKey,
 | 
						|
      publicKey: preKey.publicKey,
 | 
						|
      keyId: preKey.keyId,
 | 
						|
    };
 | 
						|
 | 
						|
    await window.Signal.Data.createOrUpdateContactPreKey(key);
 | 
						|
  };
 | 
						|
 | 
						|
  store.loadContactPreKey = async pubKey => {
 | 
						|
    const preKey = await window.Signal.Data.getContactPreKeyByIdentityKey(
 | 
						|
      pubKey
 | 
						|
    );
 | 
						|
    if (preKey) {
 | 
						|
      return {
 | 
						|
        id: preKey.id,
 | 
						|
        keyId: preKey.keyId,
 | 
						|
        publicKey: preKey.publicKey,
 | 
						|
        identityKeyString: preKey.identityKeyString,
 | 
						|
      };
 | 
						|
    }
 | 
						|
 | 
						|
    window.log.warn('Failed to fetch contact prekey:', pubKey);
 | 
						|
    return undefined;
 | 
						|
  };
 | 
						|
 | 
						|
  store.loadContactPreKeys = async filters => {
 | 
						|
    const { keyId, identityKeyString } = filters;
 | 
						|
    const keys = await window.Signal.Data.getContactPreKeys(
 | 
						|
      keyId,
 | 
						|
      identityKeyString
 | 
						|
    );
 | 
						|
    if (keys) {
 | 
						|
      return keys.map(preKey => ({
 | 
						|
        id: preKey.id,
 | 
						|
        keyId: preKey.keyId,
 | 
						|
        publicKey: preKey.publicKey,
 | 
						|
        identityKeyString: preKey.identityKeyString,
 | 
						|
      }));
 | 
						|
    }
 | 
						|
 | 
						|
    window.log.warn('Failed to fetch signed prekey with filters', filters);
 | 
						|
    return undefined;
 | 
						|
  };
 | 
						|
 | 
						|
  store.removeContactPreKey = async pubKey => {
 | 
						|
    await window.Signal.Data.removeContactPreKeyByIdentityKey(pubKey);
 | 
						|
  };
 | 
						|
 | 
						|
  store.clearContactPreKeysStore = async () => {
 | 
						|
    await window.Signal.Data.removeAllContactPreKeys();
 | 
						|
  };
 | 
						|
 | 
						|
  store.storeContactSignedPreKey = async (pubKey, signedPreKey) => {
 | 
						|
    const key = {
 | 
						|
      // id: (autoincrement)
 | 
						|
      identityKeyString: pubKey,
 | 
						|
      keyId: signedPreKey.keyId,
 | 
						|
      publicKey: signedPreKey.publicKey,
 | 
						|
      signature: signedPreKey.signature,
 | 
						|
      created_at: Date.now(),
 | 
						|
      confirmed: false,
 | 
						|
    };
 | 
						|
    await window.Signal.Data.createOrUpdateContactSignedPreKey(key);
 | 
						|
  };
 | 
						|
 | 
						|
  store.loadContactSignedPreKey = async pubKey => {
 | 
						|
    const preKey = await window.Signal.Data.getContactSignedPreKeyByIdentityKey(
 | 
						|
      pubKey
 | 
						|
    );
 | 
						|
    if (preKey) {
 | 
						|
      return {
 | 
						|
        id: preKey.id,
 | 
						|
        identityKeyString: preKey.identityKeyString,
 | 
						|
        publicKey: preKey.publicKey,
 | 
						|
        signature: preKey.signature,
 | 
						|
        created_at: preKey.created_at,
 | 
						|
        keyId: preKey.keyId,
 | 
						|
        confirmed: preKey.confirmed,
 | 
						|
      };
 | 
						|
    }
 | 
						|
    window.log.warn('Failed to fetch contact signed prekey:', pubKey);
 | 
						|
    return undefined;
 | 
						|
  };
 | 
						|
 | 
						|
  store.loadContactSignedPreKeys = async filters => {
 | 
						|
    const { keyId, identityKeyString } = filters;
 | 
						|
    const keys = await window.Signal.Data.getContactSignedPreKeys(
 | 
						|
      keyId,
 | 
						|
      identityKeyString
 | 
						|
    );
 | 
						|
    if (keys) {
 | 
						|
      return keys.map(preKey => ({
 | 
						|
        id: preKey.id,
 | 
						|
        identityKeyString: preKey.identityKeyString,
 | 
						|
        publicKey: preKey.publicKey,
 | 
						|
        signature: preKey.signature,
 | 
						|
        created_at: preKey.created_at,
 | 
						|
        keyId: preKey.keyId,
 | 
						|
        confirmed: preKey.confirmed,
 | 
						|
      }));
 | 
						|
    }
 | 
						|
 | 
						|
    window.log.warn(
 | 
						|
      'Failed to fetch contact signed prekey with filters',
 | 
						|
      filters
 | 
						|
    );
 | 
						|
    return undefined;
 | 
						|
  };
 | 
						|
 | 
						|
  store.removeContactSignedPreKey = async pubKey => {
 | 
						|
    await window.Signal.Data.removeContactSignedPreKeyByIdentityKey(pubKey);
 | 
						|
  };
 | 
						|
 | 
						|
  store.clearContactSignedPreKeysStore = async () => {
 | 
						|
    await window.Signal.Data.removeAllContactSignedPreKeys();
 | 
						|
  };
 | 
						|
})();
 |