mirror of
https://github.com/bitwarden/browser
synced 2025-12-13 06:43:35 +00:00
* [PM-5742] Rework Usage of Extension APIs that Cannot be Called with the Background Service Worker * [PM-5742] Implementing jest tests for the updated BrowserApi methods * [PM-5742] Implementing jest tests to validate logic within added API calls * [PM-5742] Implementing jest tests to validate logic within added API calls * [PM-5742] Fixing broken Jest tests * [PM-5742] Fixing linter error
359 lines
11 KiB
TypeScript
359 lines
11 KiB
TypeScript
import { ClientType, DeviceType } from "@bitwarden/common/enums";
|
|
import { MessagingService } from "@bitwarden/common/platform/abstractions/messaging.service";
|
|
import { PlatformUtilsService } from "@bitwarden/common/platform/abstractions/platform-utils.service";
|
|
|
|
import { SafariApp } from "../../browser/safariApp";
|
|
import { BrowserApi } from "../browser/browser-api";
|
|
|
|
export default class BrowserPlatformUtilsService implements PlatformUtilsService {
|
|
private static deviceCache: DeviceType = null;
|
|
|
|
constructor(
|
|
private messagingService: MessagingService,
|
|
private clipboardWriteCallback: (clipboardValue: string, clearMs: number) => void,
|
|
private biometricCallback: () => Promise<boolean>,
|
|
private win: Window & typeof globalThis,
|
|
) {}
|
|
|
|
static getDevice(win: Window & typeof globalThis): DeviceType {
|
|
if (this.deviceCache) {
|
|
return this.deviceCache;
|
|
}
|
|
|
|
if (BrowserPlatformUtilsService.isFirefox()) {
|
|
this.deviceCache = DeviceType.FirefoxExtension;
|
|
} else if (BrowserPlatformUtilsService.isOpera(win)) {
|
|
this.deviceCache = DeviceType.OperaExtension;
|
|
} else if (BrowserPlatformUtilsService.isEdge()) {
|
|
this.deviceCache = DeviceType.EdgeExtension;
|
|
} else if (BrowserPlatformUtilsService.isVivaldi()) {
|
|
this.deviceCache = DeviceType.VivaldiExtension;
|
|
} else if (BrowserPlatformUtilsService.isChrome(win)) {
|
|
this.deviceCache = DeviceType.ChromeExtension;
|
|
} else if (BrowserPlatformUtilsService.isSafari(win)) {
|
|
this.deviceCache = DeviceType.SafariExtension;
|
|
}
|
|
|
|
return this.deviceCache;
|
|
}
|
|
|
|
getDevice(): DeviceType {
|
|
return BrowserPlatformUtilsService.getDevice(this.win);
|
|
}
|
|
|
|
getDeviceString(): string {
|
|
const device = DeviceType[this.getDevice()].toLowerCase();
|
|
return device.replace("extension", "");
|
|
}
|
|
|
|
getClientType(): ClientType {
|
|
return ClientType.Browser;
|
|
}
|
|
|
|
/**
|
|
* @deprecated Do not call this directly, use getDevice() instead
|
|
*/
|
|
static isFirefox(): boolean {
|
|
return (
|
|
navigator.userAgent.indexOf(" Firefox/") !== -1 ||
|
|
navigator.userAgent.indexOf(" Gecko/") !== -1
|
|
);
|
|
}
|
|
|
|
isFirefox(): boolean {
|
|
return this.getDevice() === DeviceType.FirefoxExtension;
|
|
}
|
|
|
|
/**
|
|
* @deprecated Do not call this directly, use getDevice() instead
|
|
*/
|
|
private static isChrome(win: Window & typeof globalThis): boolean {
|
|
return win.chrome && navigator.userAgent.indexOf(" Chrome/") !== -1;
|
|
}
|
|
|
|
isChrome(): boolean {
|
|
return this.getDevice() === DeviceType.ChromeExtension;
|
|
}
|
|
|
|
/**
|
|
* @deprecated Do not call this directly, use getDevice() instead
|
|
*/
|
|
private static isEdge(): boolean {
|
|
return navigator.userAgent.indexOf(" Edg/") !== -1;
|
|
}
|
|
|
|
isEdge(): boolean {
|
|
return this.getDevice() === DeviceType.EdgeExtension;
|
|
}
|
|
|
|
/**
|
|
* @deprecated Do not call this directly, use getDevice() instead
|
|
*/
|
|
private static isOpera(win: Window & typeof globalThis): boolean {
|
|
return (
|
|
(!!win.opr && !!win.opr.addons) || !!win.opera || navigator.userAgent.indexOf(" OPR/") >= 0
|
|
);
|
|
}
|
|
|
|
isOpera(): boolean {
|
|
return this.getDevice() === DeviceType.OperaExtension;
|
|
}
|
|
|
|
/**
|
|
* @deprecated Do not call this directly, use getDevice() instead
|
|
*/
|
|
private static isVivaldi(): boolean {
|
|
return navigator.userAgent.indexOf(" Vivaldi/") !== -1;
|
|
}
|
|
|
|
isVivaldi(): boolean {
|
|
return this.getDevice() === DeviceType.VivaldiExtension;
|
|
}
|
|
|
|
/**
|
|
* @deprecated Do not call this directly, use getDevice() instead
|
|
*/
|
|
static isSafari(win: Window & typeof globalThis): boolean {
|
|
// Opera masquerades as Safari, so make sure we're not there first
|
|
return (
|
|
!BrowserPlatformUtilsService.isOpera(win) && navigator.userAgent.indexOf(" Safari/") !== -1
|
|
);
|
|
}
|
|
|
|
private static safariVersion(): string {
|
|
return navigator.userAgent.match("Version/([0-9.]*)")?.[1];
|
|
}
|
|
|
|
/**
|
|
* Safari previous to version 16.1 had a bug which caused artifacts on hover in large extension popups.
|
|
* https://bugs.webkit.org/show_bug.cgi?id=218704
|
|
*/
|
|
static shouldApplySafariHeightFix(win: Window & typeof globalThis): boolean {
|
|
if (BrowserPlatformUtilsService.getDevice(win) !== DeviceType.SafariExtension) {
|
|
return false;
|
|
}
|
|
|
|
const version = BrowserPlatformUtilsService.safariVersion();
|
|
const parts = version?.split(".")?.map((v) => Number(v));
|
|
return parts?.[0] < 16 || (parts?.[0] === 16 && parts?.[1] === 0);
|
|
}
|
|
|
|
isSafari(): boolean {
|
|
return this.getDevice() === DeviceType.SafariExtension;
|
|
}
|
|
|
|
isIE(): boolean {
|
|
return false;
|
|
}
|
|
|
|
isMacAppStore(): boolean {
|
|
return false;
|
|
}
|
|
|
|
async isViewOpen(): Promise<boolean> {
|
|
if (await BrowserApi.isPopupOpen()) {
|
|
return true;
|
|
}
|
|
|
|
if (this.isSafari()) {
|
|
return false;
|
|
}
|
|
|
|
// Opera has "sidebar_panel" as a ViewType but doesn't currently work
|
|
if (this.isFirefox() && BrowserApi.getExtensionViews({ type: "sidebar" }).length > 0) {
|
|
return true;
|
|
}
|
|
|
|
// Opera sidebar has type of "tab" (will stick around for a while after closing sidebar)
|
|
const tabOpen = BrowserApi.getExtensionViews({ type: "tab" }).length > 0;
|
|
return tabOpen;
|
|
}
|
|
|
|
lockTimeout(): number {
|
|
return null;
|
|
}
|
|
|
|
launchUri(uri: string, options?: any): void {
|
|
// FIXME: Verify that this floating promise is intentional. If it is, add an explanatory comment and ensure there is proper error handling.
|
|
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
BrowserApi.createNewTab(uri, options && options.extensionPage === true);
|
|
}
|
|
|
|
getApplicationVersion(): Promise<string> {
|
|
return Promise.resolve(BrowserApi.getApplicationVersion());
|
|
}
|
|
|
|
async getApplicationVersionNumber(): Promise<string> {
|
|
return (await this.getApplicationVersion()).split(RegExp("[+|-]"))[0].trim();
|
|
}
|
|
|
|
supportsWebAuthn(win: Window): boolean {
|
|
return typeof PublicKeyCredential !== "undefined";
|
|
}
|
|
|
|
supportsDuo(): boolean {
|
|
return true;
|
|
}
|
|
|
|
showToast(
|
|
type: "error" | "success" | "warning" | "info",
|
|
title: string,
|
|
text: string | string[],
|
|
options?: any,
|
|
): void {
|
|
this.messagingService.send("showToast", {
|
|
text: text,
|
|
title: title,
|
|
type: type,
|
|
options: options,
|
|
});
|
|
}
|
|
|
|
isDev(): boolean {
|
|
return process.env.ENV === "development";
|
|
}
|
|
|
|
isSelfHost(): boolean {
|
|
return false;
|
|
}
|
|
|
|
copyToClipboard(text: string, options?: any): void {
|
|
let win = this.win;
|
|
let doc = this.win.document;
|
|
if (options && (options.window || options.win)) {
|
|
win = options.window || options.win;
|
|
doc = win.document;
|
|
} else if (options && options.doc) {
|
|
doc = options.doc;
|
|
}
|
|
const clearing = options ? !!options.clearing : false;
|
|
const clearMs: number = options && options.clearMs ? options.clearMs : null;
|
|
|
|
if (this.isSafari()) {
|
|
// FIXME: Verify that this floating promise is intentional. If it is, add an explanatory comment and ensure there is proper error handling.
|
|
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
SafariApp.sendMessageToApp("copyToClipboard", text).then(() => {
|
|
if (!clearing && this.clipboardWriteCallback != null) {
|
|
this.clipboardWriteCallback(text, clearMs);
|
|
}
|
|
});
|
|
} else if (
|
|
this.isFirefox() &&
|
|
(win as any).navigator.clipboard &&
|
|
(win as any).navigator.clipboard.writeText
|
|
) {
|
|
(win as any).navigator.clipboard.writeText(text).then(() => {
|
|
if (!clearing && this.clipboardWriteCallback != null) {
|
|
this.clipboardWriteCallback(text, clearMs);
|
|
}
|
|
});
|
|
} else if (doc.queryCommandSupported && doc.queryCommandSupported("copy")) {
|
|
if (this.isChrome() && text === "") {
|
|
text = "\u0000";
|
|
}
|
|
|
|
const textarea = doc.createElement("textarea");
|
|
textarea.textContent = text == null || text === "" ? " " : text;
|
|
// Prevent scrolling to bottom of page in MS Edge.
|
|
textarea.style.position = "fixed";
|
|
doc.body.appendChild(textarea);
|
|
textarea.select();
|
|
|
|
try {
|
|
// Security exception may be thrown by some browsers.
|
|
if (doc.execCommand("copy") && !clearing && this.clipboardWriteCallback != null) {
|
|
this.clipboardWriteCallback(text, clearMs);
|
|
}
|
|
} catch (e) {
|
|
// eslint-disable-next-line
|
|
console.warn("Copy to clipboard failed.", e);
|
|
} finally {
|
|
doc.body.removeChild(textarea);
|
|
}
|
|
}
|
|
}
|
|
|
|
async readFromClipboard(options?: any): Promise<string> {
|
|
let win = this.win;
|
|
let doc = this.win.document;
|
|
if (options && (options.window || options.win)) {
|
|
win = options.window || options.win;
|
|
doc = win.document;
|
|
} else if (options && options.doc) {
|
|
doc = options.doc;
|
|
}
|
|
|
|
if (this.isSafari()) {
|
|
return await SafariApp.sendMessageToApp("readFromClipboard");
|
|
} else if (
|
|
this.isFirefox() &&
|
|
(win as any).navigator.clipboard &&
|
|
(win as any).navigator.clipboard.readText
|
|
) {
|
|
return await (win as any).navigator.clipboard.readText();
|
|
} else if (doc.queryCommandSupported && doc.queryCommandSupported("paste")) {
|
|
const textarea = doc.createElement("textarea");
|
|
// Prevent scrolling to bottom of page in MS Edge.
|
|
textarea.style.position = "fixed";
|
|
doc.body.appendChild(textarea);
|
|
textarea.focus();
|
|
try {
|
|
// Security exception may be thrown by some browsers.
|
|
if (doc.execCommand("paste")) {
|
|
return textarea.value;
|
|
}
|
|
} catch (e) {
|
|
// eslint-disable-next-line
|
|
console.warn("Read from clipboard failed.", e);
|
|
} finally {
|
|
doc.body.removeChild(textarea);
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
async supportsBiometric() {
|
|
const platformInfo = await BrowserApi.getPlatformInfo();
|
|
if (platformInfo.os === "android") {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
authenticateBiometric() {
|
|
return this.biometricCallback();
|
|
}
|
|
|
|
supportsSecureStorage(): boolean {
|
|
return false;
|
|
}
|
|
|
|
async getAutofillKeyboardShortcut(): Promise<string> {
|
|
let autofillCommand: string;
|
|
// You can not change the command in Safari or obtain it programmatically
|
|
if (this.isSafari()) {
|
|
autofillCommand = "Cmd+Shift+L";
|
|
} else if (this.isFirefox()) {
|
|
autofillCommand = (await browser.commands.getAll()).find(
|
|
(c) => c.name === "autofill_login",
|
|
).shortcut;
|
|
// Firefox is returning Ctrl instead of Cmd for the modifier key on macOS if
|
|
// the command is the default one set on installation.
|
|
if (
|
|
(await browser.runtime.getPlatformInfo()).os === "mac" &&
|
|
autofillCommand === "Ctrl+Shift+L"
|
|
) {
|
|
autofillCommand = "Cmd+Shift+L";
|
|
}
|
|
} else {
|
|
await new Promise((resolve) =>
|
|
chrome.commands.getAll((c) =>
|
|
resolve((autofillCommand = c.find((c) => c.name === "autofill_login").shortcut)),
|
|
),
|
|
);
|
|
}
|
|
return autofillCommand;
|
|
}
|
|
}
|