2020-10-05 20:05:48 +02:00
|
|
|
import { promises as fs, existsSync } from 'fs';
|
2020-10-05 15:11:37 +02:00
|
|
|
import * as ipc from 'node-ipc';
|
2020-10-05 19:48:51 +02:00
|
|
|
import * as path from 'path';
|
|
|
|
import * as util from 'util';
|
2020-10-05 15:11:37 +02:00
|
|
|
|
2020-10-05 20:05:48 +02:00
|
|
|
import { LogService } from 'jslib/abstractions/log.service';
|
2020-10-11 20:41:10 +02:00
|
|
|
import { BiometricMain } from 'jslib/abstractions/biometric.main';
|
2020-10-05 20:05:48 +02:00
|
|
|
|
2020-10-05 15:11:37 +02:00
|
|
|
export class NativeMessagingService {
|
|
|
|
private connected = false;
|
|
|
|
|
2020-10-11 20:41:10 +02:00
|
|
|
constructor(private logService: LogService, private biometricMain: BiometricMain, private userPath: string, private appPath: string) {}
|
2020-10-05 19:48:51 +02:00
|
|
|
|
2020-10-05 15:11:37 +02:00
|
|
|
listen() {
|
|
|
|
ipc.config.id = 'bitwarden';
|
|
|
|
ipc.config.retry = 1500;
|
|
|
|
|
|
|
|
ipc.serve(() => {
|
|
|
|
ipc.server.on('message', (data: any, socket: any) => {
|
2020-10-11 20:41:10 +02:00
|
|
|
this.messageHandler(data, socket);
|
2020-10-05 15:11:37 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
ipc.server.on('connect', () => {
|
|
|
|
this.connected = true;
|
|
|
|
})
|
|
|
|
|
|
|
|
ipc.server.on(
|
|
|
|
'socket.disconnected',
|
|
|
|
(socket: any, destroyedSocketID: any) => {
|
|
|
|
this.connected = false;
|
|
|
|
ipc.log(
|
|
|
|
'client ' + destroyedSocketID + ' has disconnected!'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
ipc.server.start();
|
|
|
|
}
|
2020-10-05 19:48:51 +02:00
|
|
|
|
2020-10-07 15:11:01 +02:00
|
|
|
stop() {
|
|
|
|
ipc.server.stop();
|
|
|
|
}
|
|
|
|
|
2020-10-11 20:41:10 +02:00
|
|
|
send(message: object, socket: any) {
|
|
|
|
ipc.server.emit(socket, 'message', message);
|
|
|
|
}
|
|
|
|
|
2020-10-05 19:48:51 +02:00
|
|
|
generateManifests() {
|
|
|
|
const baseJson = {
|
|
|
|
'name': 'com.8bit.bitwarden',
|
|
|
|
'description': 'Bitwarden desktop <-> browser bridge',
|
|
|
|
'path': path.join(this.appPath, 'proxys', this.binaryName()),
|
|
|
|
'type': 'stdio',
|
|
|
|
}
|
|
|
|
|
|
|
|
const firefoxJson = {...baseJson, ...{ 'allowed_origins': ['446900e4-71c2-419f-a6a7-df9c091e268b']}}
|
|
|
|
const chromeJson = {...baseJson, ...{ 'allowed_origins': ['chrome-extension://ijeheppnniijonkinoakkofcdhdfojda/']}}
|
|
|
|
|
2020-10-05 20:05:48 +02:00
|
|
|
if (!existsSync(path.join(this.userPath, 'browsers'))) {
|
|
|
|
fs.mkdir(path.join(this.userPath, 'browsers'))
|
|
|
|
.catch(this.logService.error)
|
|
|
|
}
|
2020-10-05 19:48:51 +02:00
|
|
|
|
2020-10-05 20:05:48 +02:00
|
|
|
switch (process.platform) {
|
|
|
|
case 'win32':
|
2020-10-05 20:28:00 +02:00
|
|
|
const destination = path.join(this.userPath, 'browsers')
|
|
|
|
this.writeManifest(path.join(destination, 'firefox.json'), firefoxJson);
|
|
|
|
this.writeManifest(path.join(destination, 'chrome.json'), chromeJson);
|
|
|
|
|
|
|
|
this.createWindowsRegistry('HKLM\\SOFTWARE\\Mozilla\\Firefox', 'HKCU\\SOFTWARE\\Mozilla\\NativeMessagingHosts\\com.8bit.bitwarden', path.join(destination, 'firefox.json'))
|
|
|
|
this.createWindowsRegistry('HKCU\\SOFTWARE\\Google\\Chrome', 'HKCU\\SOFTWARE\\Google\\Chrome\\NativeMessagingHosts\\com.8bit.bitwarden', path.join(destination, 'chrome.json'))
|
2020-10-05 20:05:48 +02:00
|
|
|
break;
|
|
|
|
case 'darwin':
|
2020-10-05 20:28:00 +02:00
|
|
|
if (existsSync('~/Library/Application Support/Mozilla/')) {
|
|
|
|
fs.mkdir('~/Library/Application Support/Mozilla/NativeMessagingHosts/');
|
|
|
|
this.writeManifest('~/Library/Application Support/Mozilla/NativeMessagingHosts/com.8bit.bitwarden.json', firefoxJson);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (existsSync('~/Library/Application Support/Google/Chrome/')) {
|
|
|
|
fs.mkdir('~/Library/Application Support/Google/Chrome/NativeMessagingHosts/');
|
|
|
|
this.writeManifest('~/Library/Application Support/Google/Chrome/NativeMessagingHosts/com.8bit.bitwarden.json', chromeJson);
|
|
|
|
}
|
2020-10-05 20:05:48 +02:00
|
|
|
break;
|
|
|
|
case 'linux':
|
2020-10-05 20:28:00 +02:00
|
|
|
if (existsSync('~/.mozilla/')) {
|
|
|
|
fs.mkdir('~/.mozilla/native-messaging-hosts');
|
|
|
|
this.writeManifest('~/.mozilla/native-messaging-hosts/com.8bit.bitwarden.json', firefoxJson);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (existsSync('~/.config/google-chrome/')) {
|
|
|
|
fs.mkdir('~/.config/google-chrome/NativeMessagingHosts/');
|
|
|
|
this.writeManifest('~/.config/google-chrome/NativeMessagingHosts/com.8bit.bitwarden.json', chromeJson);
|
|
|
|
}
|
|
|
|
break;
|
2020-10-05 20:05:48 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-07 15:11:01 +02:00
|
|
|
removeManifests() {
|
|
|
|
switch (process.platform) {
|
|
|
|
case 'win32':
|
|
|
|
this.deleteWindowsRegistry('HKCU\\SOFTWARE\\Mozilla\\NativeMessagingHosts\\com.8bit.bitwarden');
|
|
|
|
this.deleteWindowsRegistry('HKCU\\SOFTWARE\\Google\\Chrome\\NativeMessagingHosts\\com.8bit.bitwarden');
|
|
|
|
break;
|
|
|
|
case 'darwin':
|
|
|
|
if (existsSync('~/Library/Application Support/Mozilla/NativeMessagingHosts/com.8bit.bitwarden.json')) {
|
|
|
|
fs.unlink('~/Library/Application Support/Mozilla/NativeMessagingHosts/com.8bit.bitwarden.json')
|
|
|
|
}
|
|
|
|
|
|
|
|
if (existsSync('~/Library/Application Support/Google/Chrome/NativeMessagingHosts/com.8bit.bitwarden.json')) {
|
|
|
|
fs.unlink('~/Library/Application Support/Mozilla/NativeMessagingHosts/com.8bit.bitwarden.json')
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'linux':
|
|
|
|
if (existsSync('~/.mozilla/native-messaging-hosts/com.8bit.bitwarden.json')) {
|
|
|
|
fs.unlink('~/.mozilla/native-messaging-hosts/com.8bit.bitwarden.json')
|
|
|
|
}
|
|
|
|
|
|
|
|
if (existsSync('~/.config/google-chrome/NativeMessagingHosts/com.8bit.bitwarden.json')) {
|
|
|
|
fs.unlink('~/.config/google-chrome/NativeMessagingHosts/com.8bit.bitwarden.json')
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-05 20:28:00 +02:00
|
|
|
private writeManifest(destination: string, manifest: object) {
|
|
|
|
fs.writeFile(destination, JSON.stringify(manifest, null, 2)).catch(this.logService.error);
|
2020-10-05 19:48:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private binaryName() {
|
|
|
|
switch (process.platform) {
|
|
|
|
case 'win32':
|
|
|
|
return 'app-win.exe'
|
|
|
|
case 'darwin':
|
|
|
|
return 'app-linux'
|
|
|
|
case 'linux':
|
|
|
|
default:
|
|
|
|
return 'app-macos'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-07 15:11:01 +02:00
|
|
|
private async createWindowsRegistry(check: string, location: string, jsonFile: string) {
|
2020-10-05 19:48:51 +02:00
|
|
|
const regedit = require('regedit');
|
|
|
|
regedit.setExternalVBSLocation('resources/regedit/vbs');
|
|
|
|
|
|
|
|
const list = util.promisify(regedit.list);
|
|
|
|
const createKey = util.promisify(regedit.createKey);
|
|
|
|
const putValue = util.promisify(regedit.putValue);
|
|
|
|
|
2020-10-05 20:05:48 +02:00
|
|
|
this.logService.debug(`Adding registry: ${location}`)
|
|
|
|
|
2020-10-05 19:48:51 +02:00
|
|
|
// Check installed
|
2020-10-07 15:11:01 +02:00
|
|
|
try {
|
|
|
|
await list(check)
|
|
|
|
} catch {
|
|
|
|
return;
|
|
|
|
}
|
2020-10-05 19:48:51 +02:00
|
|
|
|
2020-10-07 15:11:01 +02:00
|
|
|
try {
|
|
|
|
await createKey(location);
|
|
|
|
|
|
|
|
// Insert path to manifest
|
|
|
|
const obj: any = {};
|
|
|
|
obj[location] = {
|
|
|
|
'default': {
|
2020-10-11 20:41:10 +02:00
|
|
|
value: jsonFile,
|
2020-10-07 15:11:01 +02:00
|
|
|
type: 'REG_DEFAULT',
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
return putValue(obj);
|
|
|
|
} catch (error) {
|
|
|
|
this.logService.error(error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private async deleteWindowsRegistry(key: string) {
|
2020-10-11 20:41:10 +02:00
|
|
|
const regedit = require('regedit');
|
2020-10-07 15:11:01 +02:00
|
|
|
|
|
|
|
const list = util.promisify(regedit.list);
|
|
|
|
const deleteKey = util.promisify(regedit.deleteKey);
|
|
|
|
|
2020-10-11 20:41:10 +02:00
|
|
|
this.logService.debug(`Removing registry: ${key}`)
|
2020-10-07 15:11:01 +02:00
|
|
|
|
|
|
|
try {
|
|
|
|
await list(key);
|
2020-10-11 20:41:10 +02:00
|
|
|
await deleteKey(key);
|
2020-10-07 15:11:01 +02:00
|
|
|
} catch {
|
|
|
|
// Do nothing
|
|
|
|
}
|
2020-10-05 19:48:51 +02:00
|
|
|
}
|
2020-10-11 20:41:10 +02:00
|
|
|
|
|
|
|
private async messageHandler(message: any, socket: any) {
|
|
|
|
switch (message.command) {
|
|
|
|
case 'biometricUnlock':
|
|
|
|
if (! this.biometricMain) {
|
|
|
|
return this.send({command: 'biometricUnlock', response: 'not supported'}, socket)
|
|
|
|
}
|
|
|
|
|
|
|
|
const response = await this.biometricMain.requestCreate();
|
|
|
|
if (response) {
|
|
|
|
this.send({command: 'biometricUnlock', response: 'unlocked'}, socket);
|
|
|
|
} else {
|
|
|
|
this.send({command: 'biometricUnlock', response: 'canceled'}, socket);
|
|
|
|
}
|
2020-10-12 18:03:16 +02:00
|
|
|
|
2020-10-11 20:41:10 +02:00
|
|
|
break;
|
|
|
|
default:
|
2020-10-12 18:03:16 +02:00
|
|
|
console.error('UNKNOWN COMMAND')
|
2020-10-11 20:41:10 +02:00
|
|
|
}
|
|
|
|
}
|
2020-10-05 15:11:37 +02:00
|
|
|
}
|