mirror of
https://github.com/bitwarden/desktop.git
synced 2025-01-21 21:01:52 +01:00
Split native messaging into renderer and background service. Encrypt messages and verify timestamp
This commit is contained in:
parent
44cd222626
commit
51b749b1dc
@ -17,7 +17,6 @@ export default class IPC {
|
|||||||
'## connected to bitwarden desktop ##',
|
'## connected to bitwarden desktop ##',
|
||||||
ipc.config.delay
|
ipc.config.delay
|
||||||
);
|
);
|
||||||
ipc.of.bitwarden.emit('message', 'hello');
|
|
||||||
});
|
});
|
||||||
|
|
||||||
ipc.of.bitwarden.on('disconnect', () => {
|
ipc.of.bitwarden.on('disconnect', () => {
|
||||||
|
@ -52,6 +52,7 @@ import { UserService } from 'jslib/abstractions/user.service';
|
|||||||
import { VaultTimeoutService } from 'jslib/abstractions/vaultTimeout.service';
|
import { VaultTimeoutService } from 'jslib/abstractions/vaultTimeout.service';
|
||||||
|
|
||||||
import { ConstantsService } from 'jslib/services/constants.service';
|
import { ConstantsService } from 'jslib/services/constants.service';
|
||||||
|
import { NativeMessagingService } from '../services/nativeMessaging.service';
|
||||||
|
|
||||||
const BroadcasterSubscriptionId = 'AppComponent';
|
const BroadcasterSubscriptionId = 'AppComponent';
|
||||||
const IdleTimeout = 60000 * 10; // 10 minutes
|
const IdleTimeout = 60000 * 10; // 10 minutes
|
||||||
@ -97,7 +98,7 @@ export class AppComponent implements OnInit {
|
|||||||
private searchService: SearchService, private notificationsService: NotificationsService,
|
private searchService: SearchService, private notificationsService: NotificationsService,
|
||||||
private platformUtilsService: PlatformUtilsService, private systemService: SystemService,
|
private platformUtilsService: PlatformUtilsService, private systemService: SystemService,
|
||||||
private stateService: StateService, private eventService: EventService,
|
private stateService: StateService, private eventService: EventService,
|
||||||
private policyService: PolicyService) { }
|
private policyService: PolicyService, private nativeMessagingService: NativeMessagingService) { }
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.ngZone.runOutsideAngular(() => {
|
this.ngZone.runOutsideAngular(() => {
|
||||||
|
@ -18,6 +18,7 @@ import { isDev } from 'jslib/electron/utils';
|
|||||||
import { DeviceType } from 'jslib/enums/deviceType';
|
import { DeviceType } from 'jslib/enums/deviceType';
|
||||||
|
|
||||||
import { I18nService } from '../services/i18n.service';
|
import { I18nService } from '../services/i18n.service';
|
||||||
|
import { NativeMessagingService } from '../services/nativeMessaging.service';
|
||||||
|
|
||||||
import { AuthGuardService } from 'jslib/angular/services/auth-guard.service';
|
import { AuthGuardService } from 'jslib/angular/services/auth-guard.service';
|
||||||
import { BroadcasterService } from 'jslib/angular/services/broadcaster.service';
|
import { BroadcasterService } from 'jslib/angular/services/broadcaster.service';
|
||||||
@ -128,6 +129,7 @@ const environmentService = new EnvironmentService(apiService, storageService, no
|
|||||||
const eventService = new EventService(storageService, apiService, userService, cipherService);
|
const eventService = new EventService(storageService, apiService, userService, cipherService);
|
||||||
const systemService = new SystemService(storageService, vaultTimeoutService, messagingService, platformUtilsService,
|
const systemService = new SystemService(storageService, vaultTimeoutService, messagingService, platformUtilsService,
|
||||||
null);
|
null);
|
||||||
|
const nativeMessagingService = new NativeMessagingService(cryptoService, platformUtilsService)
|
||||||
|
|
||||||
const analytics = new Analytics(window, () => isDev(), platformUtilsService, storageService, appIdService);
|
const analytics = new Analytics(window, () => isDev(), platformUtilsService, storageService, appIdService);
|
||||||
containerService.attachToGlobal(window);
|
containerService.attachToGlobal(window);
|
||||||
@ -211,6 +213,7 @@ export function initFactory(): Function {
|
|||||||
{ provide: EventServiceAbstraction, useValue: eventService },
|
{ provide: EventServiceAbstraction, useValue: eventService },
|
||||||
{ provide: PolicyServiceAbstraction, useValue: policyService },
|
{ provide: PolicyServiceAbstraction, useValue: policyService },
|
||||||
{ provide: CryptoFunctionServiceAbstraction, useValue: cryptoFunctionService },
|
{ provide: CryptoFunctionServiceAbstraction, useValue: cryptoFunctionService },
|
||||||
|
{ provide: NativeMessagingService, useValue: nativeMessagingService },
|
||||||
{
|
{
|
||||||
provide: APP_INITIALIZER,
|
provide: APP_INITIALIZER,
|
||||||
useFactory: initFactory,
|
useFactory: initFactory,
|
||||||
|
@ -18,7 +18,7 @@ import { ElectronStorageService } from 'jslib/electron/services/electronStorage.
|
|||||||
import { TrayMain } from 'jslib/electron/tray.main';
|
import { TrayMain } from 'jslib/electron/tray.main';
|
||||||
import { UpdaterMain } from 'jslib/electron/updater.main';
|
import { UpdaterMain } from 'jslib/electron/updater.main';
|
||||||
import { WindowMain } from 'jslib/electron/window.main';
|
import { WindowMain } from 'jslib/electron/window.main';
|
||||||
import { NativeMessagingService } from './services/nativeMessaging.service';
|
import { NativeMessagingMain } from './main/nativeMessaging.main';
|
||||||
|
|
||||||
export class Main {
|
export class Main {
|
||||||
logService: ElectronLogService;
|
logService: ElectronLogService;
|
||||||
@ -34,7 +34,7 @@ export class Main {
|
|||||||
powerMonitorMain: PowerMonitorMain;
|
powerMonitorMain: PowerMonitorMain;
|
||||||
trayMain: TrayMain;
|
trayMain: TrayMain;
|
||||||
biometricMain: BiometricMain;
|
biometricMain: BiometricMain;
|
||||||
nativeMessagingService: NativeMessagingService;
|
nativeMessagingMain: NativeMessagingMain;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
// Set paths for portable builds
|
// Set paths for portable builds
|
||||||
@ -119,7 +119,7 @@ export class Main {
|
|||||||
this.biometricMain = new BiometricDarwinMain(this.storageService, this.i18nService);
|
this.biometricMain = new BiometricDarwinMain(this.storageService, this.i18nService);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.nativeMessagingService = new NativeMessagingService(this.logService, this.biometricMain, app.getPath('userData'), app.getAppPath());
|
this.nativeMessagingMain = new NativeMessagingMain(this.logService, this.windowMain, app.getPath('userData'), app.getAppPath());
|
||||||
}
|
}
|
||||||
|
|
||||||
bootstrap() {
|
bootstrap() {
|
||||||
@ -145,7 +145,7 @@ export class Main {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (await this.storageService.get<boolean>(ElectronConstants.enableBrowserIntegration)) {
|
if (await this.storageService.get<boolean>(ElectronConstants.enableBrowserIntegration)) {
|
||||||
this.nativeMessagingService.listen();
|
this.nativeMessagingMain.listen();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!app.isDefaultProtocolClient('bitwarden')) {
|
if (!app.isDefaultProtocolClient('bitwarden')) {
|
||||||
|
@ -45,12 +45,12 @@ export class MessagingMain {
|
|||||||
this.main.trayMain.hideToTray();
|
this.main.trayMain.hideToTray();
|
||||||
break;
|
break;
|
||||||
case 'enableBrowserIntegration':
|
case 'enableBrowserIntegration':
|
||||||
this.main.nativeMessagingService.generateManifests();
|
this.main.nativeMessagingMain.generateManifests();
|
||||||
this.main.nativeMessagingService.listen();
|
this.main.nativeMessagingMain.listen();
|
||||||
break;
|
break;
|
||||||
case 'disableBrowserIntegration':
|
case 'disableBrowserIntegration':
|
||||||
this.main.nativeMessagingService.removeManifests();
|
this.main.nativeMessagingMain.removeManifests();
|
||||||
this.main.nativeMessagingService.stop();
|
this.main.nativeMessagingMain.stop();
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
|
204
src/main/nativeMessaging.main.ts
Normal file
204
src/main/nativeMessaging.main.ts
Normal file
@ -0,0 +1,204 @@
|
|||||||
|
import { promises as fs, existsSync } from 'fs';
|
||||||
|
import * as ipc from 'node-ipc';
|
||||||
|
import * as path from 'path';
|
||||||
|
import * as util from 'util';
|
||||||
|
|
||||||
|
import { LogService } from 'jslib/abstractions/log.service';
|
||||||
|
import { MessagingService } from 'jslib/abstractions/messaging.service';
|
||||||
|
import { ipcMain, ipcRenderer } from 'electron';
|
||||||
|
import { WindowMain } from 'jslib/electron/window.main';
|
||||||
|
|
||||||
|
export class NativeMessagingMain {
|
||||||
|
private connected = false;
|
||||||
|
|
||||||
|
constructor(private logService: LogService, private windowMain: WindowMain, private userPath: string, private appPath: string) {}
|
||||||
|
|
||||||
|
listen() {
|
||||||
|
ipc.config.id = 'bitwarden';
|
||||||
|
ipc.config.retry = 1500;
|
||||||
|
|
||||||
|
ipc.serve(() => {
|
||||||
|
ipc.server.on('message', (data: any, socket: any) => {
|
||||||
|
// This is a ugly hack until electron is updated 7.0.0 which supports ipcMain.invoke
|
||||||
|
this.windowMain.win.webContents.send('nativeMessaging', data);
|
||||||
|
ipcMain.once('nativeMessagingReply', (event, msg) => {
|
||||||
|
if (msg != null) {
|
||||||
|
this.send(msg, socket);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
|
||||||
|
stop() {
|
||||||
|
ipc.server.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
send(message: object, socket: any) {
|
||||||
|
ipc.server.emit(socket, 'message', message);
|
||||||
|
}
|
||||||
|
|
||||||
|
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/']}}
|
||||||
|
|
||||||
|
if (!existsSync(path.join(this.userPath, 'browsers'))) {
|
||||||
|
fs.mkdir(path.join(this.userPath, 'browsers'))
|
||||||
|
.catch(this.logService.error)
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (process.platform) {
|
||||||
|
case 'win32':
|
||||||
|
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'))
|
||||||
|
break;
|
||||||
|
case 'darwin':
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'linux':
|
||||||
|
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;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private writeManifest(destination: string, manifest: object) {
|
||||||
|
fs.writeFile(destination, JSON.stringify(manifest, null, 2)).catch(this.logService.error);
|
||||||
|
}
|
||||||
|
|
||||||
|
private binaryName() {
|
||||||
|
switch (process.platform) {
|
||||||
|
case 'win32':
|
||||||
|
return 'app-win.exe'
|
||||||
|
case 'darwin':
|
||||||
|
return 'app-linux'
|
||||||
|
case 'linux':
|
||||||
|
default:
|
||||||
|
return 'app-macos'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async createWindowsRegistry(check: string, location: string, jsonFile: string) {
|
||||||
|
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);
|
||||||
|
|
||||||
|
this.logService.debug(`Adding registry: ${location}`)
|
||||||
|
|
||||||
|
// Check installed
|
||||||
|
try {
|
||||||
|
await list(check)
|
||||||
|
} catch {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await createKey(location);
|
||||||
|
|
||||||
|
// Insert path to manifest
|
||||||
|
const obj: any = {};
|
||||||
|
obj[location] = {
|
||||||
|
'default': {
|
||||||
|
value: jsonFile,
|
||||||
|
type: 'REG_DEFAULT',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
return putValue(obj);
|
||||||
|
} catch (error) {
|
||||||
|
this.logService.error(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async deleteWindowsRegistry(key: string) {
|
||||||
|
const regedit = require('regedit');
|
||||||
|
|
||||||
|
const list = util.promisify(regedit.list);
|
||||||
|
const deleteKey = util.promisify(regedit.deleteKey);
|
||||||
|
|
||||||
|
this.logService.debug(`Removing registry: ${key}`)
|
||||||
|
|
||||||
|
try {
|
||||||
|
await list(key);
|
||||||
|
await deleteKey(key);
|
||||||
|
} catch {
|
||||||
|
// Do nothing
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,211 +1,34 @@
|
|||||||
import { promises as fs, existsSync } from 'fs';
|
import { CryptoService } from 'jslib/abstractions/crypto.service';
|
||||||
import * as ipc from 'node-ipc';
|
import { PlatformUtilsService } from 'jslib/abstractions';
|
||||||
import * as path from 'path';
|
import { ipcRenderer } from 'electron';
|
||||||
import * as util from 'util';
|
|
||||||
|
|
||||||
import { LogService } from 'jslib/abstractions/log.service';
|
|
||||||
import { BiometricMain } from 'jslib/abstractions/biometric.main';
|
|
||||||
|
|
||||||
export class NativeMessagingService {
|
export class NativeMessagingService {
|
||||||
private connected = false;
|
constructor(private cryptoService: CryptoService, private platformUtilService: PlatformUtilsService) {
|
||||||
|
ipcRenderer.on('nativeMessaging', async (event: any, message: any) => {
|
||||||
constructor(private logService: LogService, private biometricMain: BiometricMain, private userPath: string, private appPath: string) {}
|
this.messageHandler(message);
|
||||||
|
|
||||||
listen() {
|
|
||||||
ipc.config.id = 'bitwarden';
|
|
||||||
ipc.config.retry = 1500;
|
|
||||||
|
|
||||||
ipc.serve(() => {
|
|
||||||
ipc.server.on('message', (data: any, socket: any) => {
|
|
||||||
this.messageHandler(data, socket);
|
|
||||||
});
|
|
||||||
|
|
||||||
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();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
stop() {
|
private async messageHandler(rawMessage: any) {
|
||||||
ipc.server.stop();
|
const message = JSON.parse(await this.cryptoService.decryptToUtf8(rawMessage));
|
||||||
}
|
|
||||||
|
|
||||||
send(message: object, socket: any) {
|
if (Math.abs(message.timestamp - Date.now()) > 10*1000) {
|
||||||
ipc.server.emit(socket, 'message', message);
|
console.error("MESSAGE IS TO OLD");
|
||||||
}
|
|
||||||
|
|
||||||
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/']}}
|
|
||||||
|
|
||||||
if (!existsSync(path.join(this.userPath, 'browsers'))) {
|
|
||||||
fs.mkdir(path.join(this.userPath, 'browsers'))
|
|
||||||
.catch(this.logService.error)
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (process.platform) {
|
|
||||||
case 'win32':
|
|
||||||
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'))
|
|
||||||
break;
|
|
||||||
case 'darwin':
|
|
||||||
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);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'linux':
|
|
||||||
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;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private writeManifest(destination: string, manifest: object) {
|
|
||||||
fs.writeFile(destination, JSON.stringify(manifest, null, 2)).catch(this.logService.error);
|
|
||||||
}
|
|
||||||
|
|
||||||
private binaryName() {
|
|
||||||
switch (process.platform) {
|
|
||||||
case 'win32':
|
|
||||||
return 'app-win.exe'
|
|
||||||
case 'darwin':
|
|
||||||
return 'app-linux'
|
|
||||||
case 'linux':
|
|
||||||
default:
|
|
||||||
return 'app-macos'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private async createWindowsRegistry(check: string, location: string, jsonFile: string) {
|
|
||||||
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);
|
|
||||||
|
|
||||||
this.logService.debug(`Adding registry: ${location}`)
|
|
||||||
|
|
||||||
// Check installed
|
|
||||||
try {
|
|
||||||
await list(check)
|
|
||||||
} catch {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
|
||||||
await createKey(location);
|
|
||||||
|
|
||||||
// Insert path to manifest
|
|
||||||
const obj: any = {};
|
|
||||||
obj[location] = {
|
|
||||||
'default': {
|
|
||||||
value: jsonFile,
|
|
||||||
type: 'REG_DEFAULT',
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
return putValue(obj);
|
|
||||||
} catch (error) {
|
|
||||||
this.logService.error(error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private async deleteWindowsRegistry(key: string) {
|
|
||||||
const regedit = require('regedit');
|
|
||||||
|
|
||||||
const list = util.promisify(regedit.list);
|
|
||||||
const deleteKey = util.promisify(regedit.deleteKey);
|
|
||||||
|
|
||||||
this.logService.debug(`Removing registry: ${key}`)
|
|
||||||
|
|
||||||
try {
|
|
||||||
await list(key);
|
|
||||||
await deleteKey(key);
|
|
||||||
} catch {
|
|
||||||
// Do nothing
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private async messageHandler(message: any, socket: any) {
|
|
||||||
switch (message.command) {
|
switch (message.command) {
|
||||||
case 'biometricUnlock':
|
case 'biometricUnlock':
|
||||||
if (! this.biometricMain) {
|
if (! this.platformUtilService.supportsBiometric()) {
|
||||||
return this.send({command: 'biometricUnlock', response: 'not supported'}, socket)
|
ipcRenderer.send('nativeMessagingSync', )
|
||||||
|
return this.send({command: 'biometricUnlock', response: 'not supported'})
|
||||||
}
|
}
|
||||||
|
|
||||||
const response = await this.biometricMain.requestCreate();
|
const response = await this.platformUtilService.authenticateBiometric();
|
||||||
if (response) {
|
if (response) {
|
||||||
this.send({command: 'biometricUnlock', response: 'unlocked'}, socket);
|
this.send({command: 'biometricUnlock', response: 'unlocked'});
|
||||||
} else {
|
} else {
|
||||||
this.send({command: 'biometricUnlock', response: 'canceled'}, socket);
|
this.send({command: 'biometricUnlock', response: 'canceled'});
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
@ -213,4 +36,11 @@ export class NativeMessagingService {
|
|||||||
console.error('UNKNOWN COMMAND')
|
console.error('UNKNOWN COMMAND')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async send(message: any) {
|
||||||
|
message.timestamp = Date.now();
|
||||||
|
const encrypted = await this.cryptoService.encrypt(JSON.stringify(message));
|
||||||
|
|
||||||
|
ipcRenderer.send('nativeMessagingReply', encrypted);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user