BetterDiscordApp-v2/client/src/builtin/E2EE.js

208 lines
8.6 KiB
JavaScript
Raw Normal View History

2018-08-09 09:56:44 +02:00
/**
* BetterDiscord E2EE Module
* Copyright (c) 2015-present Jiiks/JsSucks - https://github.com/Jiiks / https://github.com/JsSucks
* All rights reserved.
* https://betterdiscord.net
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
2018-08-12 17:10:22 +02:00
import { Settings, Cache } from 'modules';
2018-08-09 09:56:44 +02:00
import BuiltinModule from './BuiltinModule';
import { WebpackModules, ReactComponents, MonkeyPatch, Patcher, DiscordApi, Security } from 'modules';
2018-08-10 14:08:21 +02:00
import { VueInjector, Reflection } from 'ui';
import { ClientLogger as Logger } from 'common';
2018-08-11 14:29:30 +02:00
import { request } from 'vendor';
import { Utils } from 'common';
2018-08-10 14:30:24 +02:00
import E2EEComponent from './E2EEComponent.vue';
import E2EEMessageButton from './E2EEMessageButton.vue';
2018-08-10 14:08:21 +02:00
import aes256 from 'aes256';
2018-08-12 20:43:59 +02:00
import crypto from 'node-crypto';
2018-08-09 09:56:44 +02:00
2018-08-10 16:02:41 +02:00
let seed = Math.random().toString(36).replace(/[^a-z]+/g, '');
2018-08-10 16:01:07 +02:00
2018-08-09 09:56:44 +02:00
export default new class E2EE extends BuiltinModule {
2018-08-10 16:01:07 +02:00
constructor() {
super();
this.master = this.encrypt(seed, 'temporarymasterkey');
this.encryptNewMessages = true;
2018-08-09 09:56:44 +02:00
}
2018-08-10 16:02:41 +02:00
setMaster(key) {
seed = Math.random().toString(36).replace(/[^a-z]+/g, '');
2018-08-10 16:04:09 +02:00
const newMaster = this.encrypt(seed, key);
// TODO re-encrypt everything with new master
return (this.master = newMaster);
2018-08-10 16:02:41 +02:00
}
2018-08-10 16:01:07 +02:00
get settingPath() {
return ['security', 'default', 'e2ee'];
2018-08-10 15:22:30 +02:00
}
get database() {
return Settings.getSetting('security', 'e2eedb', 'e2ekvps').value;
2018-08-10 15:22:30 +02:00
}
2018-08-10 20:17:52 +02:00
encrypt(key, content, prefix = '') {
if (!content) {
// Get key for current channel and encrypt
const haveKey = this.getKey(DiscordApi.currentChannel.id);
if (!haveKey) return 'nokey';
return this.encrypt(this.decrypt(this.decrypt(seed, this.master), haveKey), key);
}
2018-08-10 15:22:30 +02:00
return prefix + aes256.encrypt(key, content);
}
2018-08-10 20:17:52 +02:00
decrypt(key, content, prefix = '') {
return aes256.decrypt(key, content.replace(prefix, ''));
2018-08-10 15:22:30 +02:00
}
2018-08-12 20:43:59 +02:00
async createHmac(data) {
const haveKey = this.getKey(DiscordApi.currentChannel.id);
if (!haveKey) return null;
return Security.createHmac(haveKey, data);
}
2018-08-10 15:22:30 +02:00
getKey(channelId) {
const haveKey = this.database.find(kvp => kvp.value.key === channelId);
if (!haveKey) return null;
return haveKey.value.value;
}
2018-08-10 14:08:21 +02:00
async enabled(e) {
2018-08-10 19:38:02 +02:00
this.patchMessageContent();
const selector = '.' + WebpackModules.getClassName('channelTextArea', 'emojiButton');
2018-08-11 02:39:13 +02:00
const cta = await ReactComponents.getComponent('ChannelTextArea', { selector });
this.patchChannelTextArea(cta);
this.patchChannelTextAreaSubmit(cta);
cta.forceUpdateAll();
}
async patchMessageContent() {
const selector = '.' + WebpackModules.getClassName('container', 'containerCozy', 'containerCompact', 'edited');
const MessageContent = await ReactComponents.getComponent('MessageContent', { selector });
MonkeyPatch('BD:E2EE', MessageContent.component.prototype).before('render', this.beforeRenderMessageContent.bind(this));
MonkeyPatch('BD:E2EE', MessageContent.component.prototype).after('render', this.renderMessageContent.bind(this));
2018-08-11 14:29:30 +02:00
const ImageWrapper = await ReactComponents.getComponent('ImageWrapper', { selector: '.' + WebpackModules.getClassName('imageWrapper') });
MonkeyPatch('BD:E2EE', ImageWrapper.component.prototype).before('render', this.beforeRenderImageWrapper.bind(this));
2018-08-09 09:56:44 +02:00
}
beforeRenderMessageContent(component, args, retVal) {
2018-08-10 19:38:02 +02:00
const key = this.getKey(DiscordApi.currentChannel.id);
2018-08-11 02:39:13 +02:00
if (!key) return; // We don't have a key for this channel
const Message = WebpackModules.getModuleByPrototypes(['isMentioned']);
const MessageParser = WebpackModules.getModuleByName('MessageParser');
const currentChannel = DiscordApi.currentChannel.discordObject;
if (!component.props || !component.props.message) return;
const { content } = component.props.message;
if (typeof content !== 'string') return; // Ignore any non string content
if (!content.startsWith('$:')) return; // Not an encrypted string
let decrypt;
try {
decrypt = this.decrypt(this.decrypt(this.decrypt(seed, this.master), key), component.props.message.content);
} catch (err) { return } // Ignore errors such as non empty
component.props.message.bd_encrypted = true;
2018-08-11 02:39:13 +02:00
// Create a new message to parse it properly
const create = Message.create(MessageParser.createMessage(currentChannel, MessageParser.parse(currentChannel, decrypt).content));
2018-08-11 02:39:13 +02:00
if (!create.content || !create.contentParsed) return;
component.props.message.content = create.content;
component.props.message.contentParsed = create.contentParsed;
2018-08-10 19:38:02 +02:00
}
renderMessageContent(component, args, retVal) {
if (!component.props.message.bd_encrypted) return;
retVal.props.children[0].props.children.props.children.props.children.unshift(VueInjector.createReactElement(E2EEMessageButton));
}
2018-08-11 14:29:30 +02:00
beforeRenderImageWrapper(component, args, retVal) {
if (!component.props || !component.props.src) return;
if (component.props.decrypting) return;
component.props.decrypting = true;
2018-08-11 14:29:30 +02:00
const src = component.props.original || component.props.src.split('?')[0];
2018-08-11 14:29:30 +02:00
if (!src.includes('bde2ee')) return;
component.props.className = 'bd-encryptedImage';
2018-08-11 14:29:30 +02:00
const haveKey = this.getKey(DiscordApi.currentChannel.id);
if (!haveKey) return;
2018-08-12 17:10:22 +02:00
const cached = Cache.find('e2ee:images', item => item.src === src);
if (cached) {
Logger.info('E2EE', 'Returning encrypted image from cache');
try {
const decrypt = this.decrypt(this.decrypt(this.decrypt(seed, this.master), haveKey), cached.image);
component.props.className = 'bd-decryptedImage';
component.props.src = component.props.original = 'data:;base64,' + decrypt;
} catch (err) { return } finally { component.props.readyState = 'READY' }
2018-08-11 14:29:30 +02:00
return;
}
component.props.readyState = 'LOADING';
Logger.info('E2EE', 'Decrypting image: ' + src);
request.get(src, { encoding: 'binary' }).then(res => {
2018-08-12 20:43:59 +02:00
(async () => {
const arr = new Uint8Array(new ArrayBuffer(res.length));
for (let i = 0; i < res.length; i++) arr[i] = res.charCodeAt(i);
const aobindex = Utils.aobscan(arr, [73, 69, 78, 68]) + 8;
const sliced = arr.slice(aobindex);
const image = new TextDecoder().decode(sliced);
const hmac = image.slice(-64);
const data = image.slice(0, -64);
const validateHmac = await this.createHmac(data);
if (hmac !== validateHmac) {
console.log('INVALID HMAC!');
return;
}
Cache.push('e2ee:images', { src, image: data });
if (!component || !component.props) {
Logger.warn('E2EE', 'Component seems to be gone');
return;
}
component.props.decrypting = false;
component.forceUpdate();
})();
2018-08-11 14:29:30 +02:00
}).catch(err => {
console.log('request error', err);
});
}
2018-08-11 02:39:13 +02:00
patchChannelTextArea(cta) {
MonkeyPatch('BD:E2EE', cta.component.prototype).after('render', this.renderChannelTextArea);
}
renderChannelTextArea(component, args, retVal) {
2018-08-10 14:08:21 +02:00
if (!(retVal.props.children instanceof Array)) retVal.props.children = [retVal.props.children];
const inner = retVal.props.children.find(child => child.props.className && child.props.className.includes('inner'));
inner.props.children.splice(0, 0, VueInjector.createReactElement(E2EEComponent));
2018-08-10 14:08:21 +02:00
}
2018-08-09 09:56:44 +02:00
2018-08-11 02:39:13 +02:00
patchChannelTextAreaSubmit(cta) {
MonkeyPatch('BD:E2EE', cta.component.prototype).before('handleSubmit', this.handleChannelTextAreaSubmit.bind(this));
}
handleChannelTextAreaSubmit(component, args, retVal) {
2018-08-10 15:22:30 +02:00
const key = this.getKey(DiscordApi.currentChannel.id);
if (!this.encryptNewMessages || !key) return;
2018-08-10 20:17:52 +02:00
component.props.value = this.encrypt(this.decrypt(this.decrypt(seed, this.master), key), component.props.value, '$:');
2018-08-10 14:08:21 +02:00
}
2018-08-10 19:38:02 +02:00
async disabled(e) {
2018-08-10 14:08:21 +02:00
for (const patch of Patcher.getPatchesByCaller('BD:E2EE')) patch.unpatch();
2018-08-10 19:38:02 +02:00
const ctaComponent = await ReactComponents.getComponent('ChannelTextArea');
ctaComponent.forceUpdateAll();
2018-08-09 09:56:44 +02:00
}
}