mirror of
https://github.com/bitwarden/browser.git
synced 2024-12-25 16:59:17 +01:00
[PM-3756] Disable node integration and enable context isolation in desktop (#6975)
* Disable node integration and enable context isolation * Review comments * Log in renderer through IPC * Missed imports * Mock electron API * resourcesPath is undefined in the preload, but process.windowsStore works correctly * Replace fromBufferToUtf8 conditional implementation for the `buffer` package The current non-node implementation is different than the node implementation, as the non-node would break when the contents can't be parsed as a URI component. Replacing the impl by the `buffer` package makes the result match in both environments. * Fix lint * Add some more tests * Remove buffer from devDependencies
This commit is contained in:
parent
070d8556cf
commit
4be25e3df3
@ -584,7 +584,10 @@ export class SettingsComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async saveBrowserIntegration() {
|
async saveBrowserIntegration() {
|
||||||
if (process.platform === "darwin" && !this.platformUtilsService.isMacAppStore()) {
|
if (
|
||||||
|
ipc.platform.deviceType === DeviceType.MacOsDesktop &&
|
||||||
|
!this.platformUtilsService.isMacAppStore()
|
||||||
|
) {
|
||||||
await this.dialogService.openSimpleDialog({
|
await this.dialogService.openSimpleDialog({
|
||||||
title: { key: "browserIntegrationUnsupportedTitle" },
|
title: { key: "browserIntegrationUnsupportedTitle" },
|
||||||
content: { key: "browserIntegrationMasOnlyDesc" },
|
content: { key: "browserIntegrationMasOnlyDesc" },
|
||||||
@ -606,7 +609,7 @@ export class SettingsComponent implements OnInit {
|
|||||||
|
|
||||||
this.form.controls.enableBrowserIntegration.setValue(false);
|
this.form.controls.enableBrowserIntegration.setValue(false);
|
||||||
return;
|
return;
|
||||||
} else if (process.platform == "linux") {
|
} else if (ipc.platform.deviceType === DeviceType.LinuxDesktop) {
|
||||||
await this.dialogService.openSimpleDialog({
|
await this.dialogService.openSimpleDialog({
|
||||||
title: { key: "browserIntegrationUnsupportedTitle" },
|
title: { key: "browserIntegrationUnsupportedTitle" },
|
||||||
content: { key: "browserIntegrationLinuxDesc" },
|
content: { key: "browserIntegrationLinuxDesc" },
|
||||||
|
@ -1,18 +1,12 @@
|
|||||||
import { enableProdMode } from "@angular/core";
|
import { enableProdMode } from "@angular/core";
|
||||||
import { platformBrowserDynamic } from "@angular/platform-browser-dynamic";
|
import { platformBrowserDynamic } from "@angular/platform-browser-dynamic";
|
||||||
|
|
||||||
import { ipc } from "../preload";
|
|
||||||
import { isDev } from "../utils";
|
|
||||||
|
|
||||||
// Temporary polyfill for preload script
|
|
||||||
(window as any).ipc = ipc;
|
|
||||||
|
|
||||||
require("../scss/styles.scss");
|
require("../scss/styles.scss");
|
||||||
require("../scss/tailwind.css");
|
require("../scss/tailwind.css");
|
||||||
|
|
||||||
import { AppModule } from "./app.module";
|
import { AppModule } from "./app.module";
|
||||||
|
|
||||||
if (!isDev()) {
|
if (!ipc.platform.isDev) {
|
||||||
enableProdMode();
|
enableProdMode();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -49,7 +49,7 @@ import { DialogService } from "@bitwarden/components";
|
|||||||
import { LoginGuard } from "../../auth/guards/login.guard";
|
import { LoginGuard } from "../../auth/guards/login.guard";
|
||||||
import { Account } from "../../models/account";
|
import { Account } from "../../models/account";
|
||||||
import { ElectronCryptoService } from "../../platform/services/electron-crypto.service";
|
import { ElectronCryptoService } from "../../platform/services/electron-crypto.service";
|
||||||
import { ElectronLogService } from "../../platform/services/electron-log.service";
|
import { ElectronLogRendererService } from "../../platform/services/electron-log.renderer.service";
|
||||||
import { ElectronPlatformUtilsService } from "../../platform/services/electron-platform-utils.service";
|
import { ElectronPlatformUtilsService } from "../../platform/services/electron-platform-utils.service";
|
||||||
import { ElectronRendererMessagingService } from "../../platform/services/electron-renderer-messaging.service";
|
import { ElectronRendererMessagingService } from "../../platform/services/electron-renderer-messaging.service";
|
||||||
import { ElectronRendererSecureStorageService } from "../../platform/services/electron-renderer-secure-storage.service";
|
import { ElectronRendererSecureStorageService } from "../../platform/services/electron-renderer-secure-storage.service";
|
||||||
@ -91,7 +91,7 @@ const RELOAD_CALLBACK = new InjectionToken<() => any>("RELOAD_CALLBACK");
|
|||||||
provide: RELOAD_CALLBACK,
|
provide: RELOAD_CALLBACK,
|
||||||
useValue: null,
|
useValue: null,
|
||||||
},
|
},
|
||||||
{ provide: LogServiceAbstraction, useClass: ElectronLogService, deps: [] },
|
{ provide: LogServiceAbstraction, useClass: ElectronLogRendererService, deps: [] },
|
||||||
{
|
{
|
||||||
provide: PlatformUtilsServiceAbstraction,
|
provide: PlatformUtilsServiceAbstraction,
|
||||||
useClass: ElectronPlatformUtilsService,
|
useClass: ElectronPlatformUtilsService,
|
||||||
|
@ -30,14 +30,14 @@ import { Account } from "./models/account";
|
|||||||
import { BiometricsService, BiometricsServiceAbstraction } from "./platform/main/biometric/index";
|
import { BiometricsService, BiometricsServiceAbstraction } from "./platform/main/biometric/index";
|
||||||
import { ClipboardMain } from "./platform/main/clipboard.main";
|
import { ClipboardMain } from "./platform/main/clipboard.main";
|
||||||
import { DesktopCredentialStorageListener } from "./platform/main/desktop-credential-storage-listener";
|
import { DesktopCredentialStorageListener } from "./platform/main/desktop-credential-storage-listener";
|
||||||
import { ElectronLogService } from "./platform/services/electron-log.service";
|
import { ElectronLogMainService } from "./platform/services/electron-log.main.service";
|
||||||
import { ElectronStateService } from "./platform/services/electron-state.service";
|
import { ElectronStateService } from "./platform/services/electron-state.service";
|
||||||
import { ElectronStorageService } from "./platform/services/electron-storage.service";
|
import { ElectronStorageService } from "./platform/services/electron-storage.service";
|
||||||
import { I18nMainService } from "./platform/services/i18n.main.service";
|
import { I18nMainService } from "./platform/services/i18n.main.service";
|
||||||
import { ElectronMainMessagingService } from "./services/electron-main-messaging.service";
|
import { ElectronMainMessagingService } from "./services/electron-main-messaging.service";
|
||||||
|
|
||||||
export class Main {
|
export class Main {
|
||||||
logService: ElectronLogService;
|
logService: ElectronLogMainService;
|
||||||
i18nService: I18nMainService;
|
i18nService: I18nMainService;
|
||||||
storageService: ElectronStorageService;
|
storageService: ElectronStorageService;
|
||||||
memoryStorageService: MemoryStorageService;
|
memoryStorageService: MemoryStorageService;
|
||||||
@ -89,8 +89,7 @@ export class Main {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
this.logService = new ElectronLogService(null, app.getPath("userData"));
|
this.logService = new ElectronLogMainService(null, app.getPath("userData"));
|
||||||
this.logService.init();
|
|
||||||
this.i18nService = new I18nMainService("en", "./locales/");
|
this.i18nService = new I18nMainService("en", "./locales/");
|
||||||
|
|
||||||
const storageDefaults: any = {};
|
const storageDefaults: any = {};
|
||||||
|
@ -160,11 +160,11 @@ export class WindowMain {
|
|||||||
backgroundColor: await this.getBackgroundColor(),
|
backgroundColor: await this.getBackgroundColor(),
|
||||||
alwaysOnTop: this.enableAlwaysOnTop,
|
alwaysOnTop: this.enableAlwaysOnTop,
|
||||||
webPreferences: {
|
webPreferences: {
|
||||||
// preload: path.join(__dirname, "preload.js"),
|
preload: path.join(__dirname, "preload.js"),
|
||||||
spellcheck: false,
|
spellcheck: false,
|
||||||
nodeIntegration: true,
|
nodeIntegration: false,
|
||||||
backgroundThrottling: false,
|
backgroundThrottling: false,
|
||||||
contextIsolation: false,
|
contextIsolation: true,
|
||||||
session: this.session,
|
session: this.session,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { ipcRenderer } from "electron";
|
import { ipcRenderer } from "electron";
|
||||||
|
|
||||||
import { DeviceType } from "@bitwarden/common/enums";
|
import { DeviceType } from "@bitwarden/common/enums";
|
||||||
import { ThemeType, KeySuffixOptions } from "@bitwarden/common/platform/enums";
|
import { ThemeType, KeySuffixOptions, LogLevelType } from "@bitwarden/common/platform/enums";
|
||||||
import { EncString } from "@bitwarden/common/platform/models/domain/enc-string";
|
import { EncString } from "@bitwarden/common/platform/models/domain/enc-string";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
@ -11,7 +11,7 @@ import {
|
|||||||
UnencryptedMessageResponse,
|
UnencryptedMessageResponse,
|
||||||
} from "../models/native-messaging";
|
} from "../models/native-messaging";
|
||||||
import { BiometricMessage, BiometricAction } from "../types/biometric-message";
|
import { BiometricMessage, BiometricAction } from "../types/biometric-message";
|
||||||
import { isDev, isWindowsStore } from "../utils";
|
import { isDev, isMacAppStore, isWindowsStore } from "../utils";
|
||||||
|
|
||||||
import { ClipboardWriteMessage } from "./types/clipboard";
|
import { ClipboardWriteMessage } from "./types/clipboard";
|
||||||
|
|
||||||
@ -82,8 +82,10 @@ export default {
|
|||||||
},
|
},
|
||||||
deviceType: deviceType(),
|
deviceType: deviceType(),
|
||||||
isDev: isDev(),
|
isDev: isDev(),
|
||||||
|
isMacAppStore: isMacAppStore(),
|
||||||
isWindowsStore: isWindowsStore(),
|
isWindowsStore: isWindowsStore(),
|
||||||
reloadProcess: () => ipcRenderer.send("reload-process"),
|
reloadProcess: () => ipcRenderer.send("reload-process"),
|
||||||
|
log: (level: LogLevelType, message: string) => ipcRenderer.invoke("ipc.log", { level, message }),
|
||||||
|
|
||||||
openContextMenu: (
|
openContextMenu: (
|
||||||
menu: {
|
menu: {
|
||||||
|
@ -1,22 +1,20 @@
|
|||||||
import * as path from "path";
|
import * as path from "path";
|
||||||
|
|
||||||
import log from "electron-log";
|
import { ipcMain } from "electron";
|
||||||
|
import log from "electron-log/main";
|
||||||
|
|
||||||
import { LogLevelType } from "@bitwarden/common/platform/enums/log-level-type.enum";
|
import { LogLevelType } from "@bitwarden/common/platform/enums/log-level-type.enum";
|
||||||
import { ConsoleLogService as BaseLogService } from "@bitwarden/common/platform/services/console-log.service";
|
import { ConsoleLogService as BaseLogService } from "@bitwarden/common/platform/services/console-log.service";
|
||||||
|
|
||||||
import { isDev } from "../../utils";
|
import { isDev } from "../../utils";
|
||||||
|
|
||||||
export class ElectronLogService extends BaseLogService {
|
export class ElectronLogMainService extends BaseLogService {
|
||||||
constructor(
|
constructor(
|
||||||
protected filter: (level: LogLevelType) => boolean = null,
|
protected filter: (level: LogLevelType) => boolean = null,
|
||||||
private logDir: string = null,
|
private logDir: string = null,
|
||||||
) {
|
) {
|
||||||
super(isDev(), filter);
|
super(isDev(), filter);
|
||||||
}
|
|
||||||
|
|
||||||
// Initialize the log file transport. Only needs to be done once in the main process.
|
|
||||||
init() {
|
|
||||||
if (log.transports == null) {
|
if (log.transports == null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -26,6 +24,10 @@ export class ElectronLogService extends BaseLogService {
|
|||||||
log.transports.file.resolvePathFn = () => path.join(this.logDir, "app.log");
|
log.transports.file.resolvePathFn = () => path.join(this.logDir, "app.log");
|
||||||
}
|
}
|
||||||
log.initialize();
|
log.initialize();
|
||||||
|
|
||||||
|
ipcMain.handle("ipc.log", (_event, { level, message }) => {
|
||||||
|
this.write(level, message);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
write(level: LogLevelType, message: string) {
|
write(level: LogLevelType, message: string) {
|
@ -0,0 +1,35 @@
|
|||||||
|
import { LogLevelType } from "@bitwarden/common/platform/enums/log-level-type.enum";
|
||||||
|
import { ConsoleLogService as BaseLogService } from "@bitwarden/common/platform/services/console-log.service";
|
||||||
|
|
||||||
|
export class ElectronLogRendererService extends BaseLogService {
|
||||||
|
constructor(protected filter: (level: LogLevelType) => boolean = null) {
|
||||||
|
super(ipc.platform.isDev, filter);
|
||||||
|
}
|
||||||
|
|
||||||
|
write(level: LogLevelType, message: string) {
|
||||||
|
if (this.filter != null && this.filter(level)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* eslint-disable no-console */
|
||||||
|
ipc.platform.log(level, message).catch((e) => console.log("Error logging", e));
|
||||||
|
|
||||||
|
/* eslint-disable no-console */
|
||||||
|
switch (level) {
|
||||||
|
case LogLevelType.Debug:
|
||||||
|
console.debug(message);
|
||||||
|
break;
|
||||||
|
case LogLevelType.Info:
|
||||||
|
console.info(message);
|
||||||
|
break;
|
||||||
|
case LogLevelType.Warning:
|
||||||
|
console.warn(message);
|
||||||
|
break;
|
||||||
|
case LogLevelType.Error:
|
||||||
|
console.error(message);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,9 +1,14 @@
|
|||||||
import { ElectronLogService } from "./electron-log.service";
|
import { ElectronLogMainService } from "./electron-log.main.service";
|
||||||
|
|
||||||
describe("ElectronLogService", () => {
|
// Mock the use of the electron API to avoid errors
|
||||||
|
jest.mock("electron", () => ({
|
||||||
|
ipcMain: { handle: jest.fn() },
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe("ElectronLogMainService", () => {
|
||||||
it("sets dev based on electron method", () => {
|
it("sets dev based on electron method", () => {
|
||||||
process.env.ELECTRON_IS_DEV = "1";
|
process.env.ELECTRON_IS_DEV = "1";
|
||||||
const logService = new ElectronLogService();
|
const logService = new ElectronLogMainService();
|
||||||
expect(logService).toEqual(expect.objectContaining({ isDev: true }) as any);
|
expect(logService).toEqual(expect.objectContaining({ isDev: true }) as any);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -6,7 +6,6 @@ import {
|
|||||||
PlatformUtilsService,
|
PlatformUtilsService,
|
||||||
} from "@bitwarden/common/platform/abstractions/platform-utils.service";
|
} from "@bitwarden/common/platform/abstractions/platform-utils.service";
|
||||||
|
|
||||||
import { isMacAppStore } from "../../utils";
|
|
||||||
import { ClipboardWriteMessage } from "../types/clipboard";
|
import { ClipboardWriteMessage } from "../types/clipboard";
|
||||||
|
|
||||||
export class ElectronPlatformUtilsService implements PlatformUtilsService {
|
export class ElectronPlatformUtilsService implements PlatformUtilsService {
|
||||||
@ -53,7 +52,7 @@ export class ElectronPlatformUtilsService implements PlatformUtilsService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
isMacAppStore(): boolean {
|
isMacAppStore(): boolean {
|
||||||
return isMacAppStore();
|
return ipc.platform.isMacAppStore;
|
||||||
}
|
}
|
||||||
|
|
||||||
isViewOpen(): Promise<boolean> {
|
isViewOpen(): Promise<boolean> {
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// import { contextBridge } from "electron";
|
import { contextBridge } from "electron";
|
||||||
|
|
||||||
import auth from "./auth/preload";
|
import auth from "./auth/preload";
|
||||||
import platform from "./platform/preload";
|
import platform from "./platform/preload";
|
||||||
|
|
||||||
@ -18,4 +19,4 @@ export const ipc = {
|
|||||||
platform,
|
platform,
|
||||||
};
|
};
|
||||||
|
|
||||||
// contextBridge.exposeInMainWorld("ipc", ipc);
|
contextBridge.exposeInMainWorld("ipc", ipc);
|
||||||
|
@ -55,7 +55,7 @@ export function isWindowsStore() {
|
|||||||
if (
|
if (
|
||||||
windows &&
|
windows &&
|
||||||
!windowsStore &&
|
!windowsStore &&
|
||||||
process.resourcesPath.indexOf("8bitSolutionsLLC.bitwardendesktop_") > -1
|
process.resourcesPath?.indexOf("8bitSolutionsLLC.bitwardendesktop_") > -1
|
||||||
) {
|
) {
|
||||||
windowsStore = true;
|
windowsStore = true;
|
||||||
}
|
}
|
||||||
|
@ -54,6 +54,10 @@ const common = {
|
|||||||
extensions: [".tsx", ".ts", ".js"],
|
extensions: [".tsx", ".ts", ".js"],
|
||||||
symlinks: false,
|
symlinks: false,
|
||||||
modules: [path.resolve("../../node_modules")],
|
modules: [path.resolve("../../node_modules")],
|
||||||
|
fallback: {
|
||||||
|
path: require.resolve("path-browserify"),
|
||||||
|
fs: false,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
output: {
|
output: {
|
||||||
filename: "[name].js",
|
filename: "[name].js",
|
||||||
@ -64,9 +68,7 @@ const common = {
|
|||||||
const renderer = {
|
const renderer = {
|
||||||
mode: NODE_ENV,
|
mode: NODE_ENV,
|
||||||
devtool: "source-map",
|
devtool: "source-map",
|
||||||
// TODO: Replace this with web.
|
target: "web",
|
||||||
// target: "web",
|
|
||||||
target: "electron-renderer",
|
|
||||||
node: {
|
node: {
|
||||||
__dirname: false,
|
__dirname: false,
|
||||||
},
|
},
|
||||||
|
@ -258,6 +258,7 @@ describe("Utils Service", () => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const asciiHelloWorld = "hello world";
|
||||||
const asciiHelloWorldArray = [104, 101, 108, 108, 111, 32, 119, 111, 114, 108, 100];
|
const asciiHelloWorldArray = [104, 101, 108, 108, 111, 32, 119, 111, 114, 108, 100];
|
||||||
const b64HelloWorldString = "aGVsbG8gd29ybGQ=";
|
const b64HelloWorldString = "aGVsbG8gd29ybGQ=";
|
||||||
|
|
||||||
@ -623,4 +624,59 @@ describe("Utils Service", () => {
|
|||||||
expect(Utils.daysRemaining(new Date(2024, 9, 2, 10))).toBe(366);
|
expect(Utils.daysRemaining(new Date(2024, 9, 2, 10))).toBe(366);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("fromBufferToUtf8(...)", () => {
|
||||||
|
const originalIsNode = Utils.isNode;
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
Utils.isNode = originalIsNode;
|
||||||
|
});
|
||||||
|
|
||||||
|
runInBothEnvironments("should convert an ArrayBuffer to a utf8 string", () => {
|
||||||
|
const buffer = new Uint8Array(asciiHelloWorldArray).buffer;
|
||||||
|
const str = Utils.fromBufferToUtf8(buffer);
|
||||||
|
expect(str).toBe(asciiHelloWorld);
|
||||||
|
});
|
||||||
|
|
||||||
|
runInBothEnvironments("should handle an empty buffer", () => {
|
||||||
|
const buffer = new ArrayBuffer(0);
|
||||||
|
const str = Utils.fromBufferToUtf8(buffer);
|
||||||
|
expect(str).toBe("");
|
||||||
|
});
|
||||||
|
|
||||||
|
runInBothEnvironments("should convert a binary ArrayBuffer to a binary string", () => {
|
||||||
|
const cases = [
|
||||||
|
{
|
||||||
|
input: [
|
||||||
|
174, 21, 17, 79, 39, 130, 132, 173, 49, 180, 113, 118, 160, 15, 47, 99, 57, 208, 141,
|
||||||
|
187, 54, 194, 153, 12, 37, 130, 155, 213, 125, 196, 241, 101,
|
||||||
|
],
|
||||||
|
output: "<22>O'<27><><EFBFBD>1<EFBFBD>qv<71>/c9Ѝ<39>6%<25><><EFBFBD>}<7D><>e",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: [
|
||||||
|
88, 17, 69, 41, 75, 69, 128, 225, 252, 219, 146, 72, 162, 14, 139, 120, 30, 239, 105,
|
||||||
|
229, 14, 131, 174, 119, 61, 88, 108, 135, 60, 88, 120, 145,
|
||||||
|
],
|
||||||
|
output: "XE)KE<4B><45><EFBFBD>ےH<DB92><0E>x<1E>i<EFBFBD><0E><>w=Xl<58><Xx<58>",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: [
|
||||||
|
121, 110, 81, 148, 48, 67, 209, 43, 3, 39, 143, 184, 237, 184, 213, 183, 84, 157, 47, 6,
|
||||||
|
31, 183, 99, 142, 155, 156, 192, 107, 118, 64, 176, 36,
|
||||||
|
],
|
||||||
|
output: "ynQ<6E>0C<30>+'<27><><EFBFBD><EFBFBD>շT<D5B7>/<1F>c<EFBFBD><63><EFBFBD><EFBFBD>kv@<40>$",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
cases.forEach((c) => {
|
||||||
|
const buffer = new Uint8Array(c.input).buffer;
|
||||||
|
const str = Utils.fromBufferToUtf8(buffer);
|
||||||
|
// Match the expected output
|
||||||
|
expect(str).toBe(c.output);
|
||||||
|
// Make sure it matches with the Node.js Buffer output
|
||||||
|
expect(str).toBe(Buffer.from(buffer).toString("utf8"));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
/* eslint-disable no-useless-escape */
|
/* eslint-disable no-useless-escape */
|
||||||
import * as path from "path";
|
import * as path from "path";
|
||||||
|
|
||||||
|
import { Buffer as BufferLib } from "buffer/";
|
||||||
import { Observable, of, switchMap } from "rxjs";
|
import { Observable, of, switchMap } from "rxjs";
|
||||||
import { getHostname, parse } from "tldts";
|
import { getHostname, parse } from "tldts";
|
||||||
import { Merge } from "type-fest";
|
import { Merge } from "type-fest";
|
||||||
@ -145,13 +146,7 @@ export class Utils {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static fromBufferToUtf8(buffer: ArrayBuffer): string {
|
static fromBufferToUtf8(buffer: ArrayBuffer): string {
|
||||||
if (Utils.isNode) {
|
return BufferLib.from(buffer).toString("utf8");
|
||||||
return Buffer.from(buffer).toString("utf8");
|
|
||||||
} else {
|
|
||||||
const bytes = new Uint8Array(buffer);
|
|
||||||
const encodedString = String.fromCharCode.apply(null, bytes);
|
|
||||||
return decodeURIComponent(escape(encodedString));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static fromBufferToByteString(buffer: ArrayBuffer): string {
|
static fromBufferToByteString(buffer: ArrayBuffer): string {
|
||||||
|
@ -83,7 +83,6 @@
|
|||||||
"@webcomponents/custom-elements": "1.6.0",
|
"@webcomponents/custom-elements": "1.6.0",
|
||||||
"autoprefixer": "10.4.16",
|
"autoprefixer": "10.4.16",
|
||||||
"base64-loader": "1.0.0",
|
"base64-loader": "1.0.0",
|
||||||
"buffer": "6.0.3",
|
|
||||||
"chromatic": "10.0.0",
|
"chromatic": "10.0.0",
|
||||||
"clean-webpack-plugin": "4.0.0",
|
"clean-webpack-plugin": "4.0.0",
|
||||||
"concurrently": "8.2.2",
|
"concurrently": "8.2.2",
|
||||||
@ -169,6 +168,7 @@
|
|||||||
"big-integer": "1.6.51",
|
"big-integer": "1.6.51",
|
||||||
"bootstrap": "4.6.0",
|
"bootstrap": "4.6.0",
|
||||||
"braintree-web-drop-in": "1.42.0",
|
"braintree-web-drop-in": "1.42.0",
|
||||||
|
"buffer": "6.0.3",
|
||||||
"bufferutil": "4.0.8",
|
"bufferutil": "4.0.8",
|
||||||
"chalk": "4.1.2",
|
"chalk": "4.1.2",
|
||||||
"commander": "11.1.0",
|
"commander": "11.1.0",
|
||||||
|
Loading…
Reference in New Issue
Block a user