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.
		
		
		
		
		
			
		
			
				
	
	
		
			142 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			JavaScript
		
	
			
		
		
	
	
			142 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			JavaScript
		
	
/* global libloki, Multibase, libsignal, StringView, dcodeIO */
 | 
						|
 | 
						|
'use strict';
 | 
						|
 | 
						|
async function generateSnodeKeysAndAddress() {
 | 
						|
  // snode identitys is a ed25519 keypair
 | 
						|
  const sodium = await window.getSodium();
 | 
						|
  const ed25519KeyPair = sodium.crypto_sign_keypair();
 | 
						|
  const keyPair = {
 | 
						|
    pubKey: ed25519KeyPair.publicKey,
 | 
						|
    privKey: ed25519KeyPair.privateKey,
 | 
						|
  };
 | 
						|
  // snode address is the pubkey in base32z
 | 
						|
  let address = Multibase.encode(
 | 
						|
    'base32z',
 | 
						|
    Multibase.Buffer.from(keyPair.pubKey)
 | 
						|
  ).toString();
 | 
						|
  // remove first letter, which is the encoding code
 | 
						|
  address = address.substring(1);
 | 
						|
  return { keyPair, address };
 | 
						|
}
 | 
						|
 | 
						|
describe('Snode Channel', () => {
 | 
						|
  describe('snodeCipher singleton', () => {
 | 
						|
    it('should be defined at libloki.crypto', () => {
 | 
						|
      assert.isDefined(libloki.crypto.snodeCipher);
 | 
						|
      assert.isTrue(
 | 
						|
        libloki.crypto.snodeCipher instanceof libloki.crypto._LokiSnodeChannel
 | 
						|
      );
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  describe('#decodeSnodeAddressToPubKey', () => {
 | 
						|
    it('should decode a base32z encoded .snode address', async () => {
 | 
						|
      const { keyPair, address } = await generateSnodeKeysAndAddress();
 | 
						|
 | 
						|
      const buffer = libloki.crypto._decodeSnodeAddressToPubKey(
 | 
						|
        `http://${address}.snode`
 | 
						|
      );
 | 
						|
 | 
						|
      const expected = new Uint8Array(keyPair.pubKey);
 | 
						|
      assert.strictEqual(expected.length, 32);
 | 
						|
      assert.strictEqual(buffer.length, 32);
 | 
						|
      for (let i = 0; i < buffer.length; i += 1) {
 | 
						|
        assert.strictEqual(buffer[i], expected[i]);
 | 
						|
      }
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  describe('#LokiSnodeChannel', () => {
 | 
						|
    it('should generate an ephemeral key pair', () => {
 | 
						|
      const channel = new libloki.crypto._LokiSnodeChannel();
 | 
						|
 | 
						|
      assert.isDefined(channel._ephemeralKeyPair);
 | 
						|
      assert.isTrue(channel._ephemeralKeyPair.privKey instanceof ArrayBuffer);
 | 
						|
      assert.isTrue(channel._ephemeralKeyPair.pubKey instanceof ArrayBuffer);
 | 
						|
      const pubKeyHex = StringView.arrayBufferToHex(
 | 
						|
        channel._ephemeralKeyPair.pubKey
 | 
						|
      );
 | 
						|
      assert.strictEqual(channel.getChannelPublicKeyHex(), pubKeyHex);
 | 
						|
    });
 | 
						|
 | 
						|
    it('should cache something by snode address', async () => {
 | 
						|
      const { address } = await generateSnodeKeysAndAddress();
 | 
						|
 | 
						|
      const channel = new libloki.crypto._LokiSnodeChannel();
 | 
						|
      // cache should be empty
 | 
						|
      assert.strictEqual(Object.keys(channel._cache).length, 0);
 | 
						|
 | 
						|
      // push to cache
 | 
						|
      await channel._getSymmetricKey(address);
 | 
						|
 | 
						|
      assert.strictEqual(Object.keys(channel._cache).length, 1);
 | 
						|
      assert.strictEqual(Object.keys(channel._cache)[0], address);
 | 
						|
    });
 | 
						|
 | 
						|
    it('should encrypt data correctly', async () => {
 | 
						|
      // message sent by Loki Messenger
 | 
						|
      const snode = await generateSnodeKeysAndAddress();
 | 
						|
      const messageSent = 'I am Groot';
 | 
						|
      const textEncoder = new TextEncoder();
 | 
						|
      const data = textEncoder.encode(messageSent);
 | 
						|
 | 
						|
      const channel = new libloki.crypto._LokiSnodeChannel();
 | 
						|
      const encrypted = await channel.encrypt(snode.address, data);
 | 
						|
 | 
						|
      assert.strictEqual(typeof encrypted, 'string');
 | 
						|
 | 
						|
      // message received by storage server
 | 
						|
      const senderPubKey = StringView.hexToArrayBuffer(
 | 
						|
        channel.getChannelPublicKeyHex()
 | 
						|
      );
 | 
						|
      const sodium = await window.getSodium();
 | 
						|
      const snodePrivKey = sodium.crypto_sign_ed25519_sk_to_curve25519(
 | 
						|
        snode.keyPair.privKey
 | 
						|
      ).buffer;
 | 
						|
      const symmetricKey = libsignal.Curve.calculateAgreement(
 | 
						|
        senderPubKey,
 | 
						|
        snodePrivKey
 | 
						|
      );
 | 
						|
      const encryptedArrayBuffer = dcodeIO.ByteBuffer.wrap(
 | 
						|
        encrypted,
 | 
						|
        'base64'
 | 
						|
      ).toArrayBuffer();
 | 
						|
      const decrypted = await libloki.crypto.DHDecrypt(
 | 
						|
        symmetricKey,
 | 
						|
        encryptedArrayBuffer
 | 
						|
      );
 | 
						|
      const textDecoder = new TextDecoder();
 | 
						|
      const messageReceived = textDecoder.decode(decrypted);
 | 
						|
      assert.strictEqual(messageSent, messageReceived);
 | 
						|
    });
 | 
						|
 | 
						|
    it('should decrypt data correctly', async () => {
 | 
						|
      const channel = new libloki.crypto._LokiSnodeChannel();
 | 
						|
      // message sent by storage server
 | 
						|
      const snode = await generateSnodeKeysAndAddress();
 | 
						|
      const messageSent = 'You are Groot';
 | 
						|
      const textEncoder = new TextEncoder();
 | 
						|
      const data = textEncoder.encode(messageSent);
 | 
						|
      const senderPubKey = StringView.hexToArrayBuffer(
 | 
						|
        channel.getChannelPublicKeyHex()
 | 
						|
      );
 | 
						|
      const sodium = await window.getSodium();
 | 
						|
      const snodePrivKey = sodium.crypto_sign_ed25519_sk_to_curve25519(
 | 
						|
        snode.keyPair.privKey
 | 
						|
      ).buffer;
 | 
						|
      const symmetricKey = libsignal.Curve.calculateAgreement(
 | 
						|
        senderPubKey,
 | 
						|
        snodePrivKey
 | 
						|
      );
 | 
						|
      const encrypted = await libloki.crypto.DHEncrypt(symmetricKey, data);
 | 
						|
      const encryptedBase64 = dcodeIO.ByteBuffer.wrap(encrypted).toString(
 | 
						|
        'base64'
 | 
						|
      );
 | 
						|
      // message received by Loki Messenger
 | 
						|
      const decrypted = await channel.decrypt(snode.address, encryptedBase64);
 | 
						|
      assert.strictEqual(messageSent, decrypted);
 | 
						|
    });
 | 
						|
  });
 | 
						|
});
 |