Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix(security): Use import.meta.hot for communication with Vite 6.0.9+ (and 5.4.14+) dev server #1411

Draft
wants to merge 14 commits into
base: main
Choose a base branch
from
6 changes: 5 additions & 1 deletion packages/wxt/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,10 @@
"./modules": {
"types": "./dist/modules.d.ts",
"default": "./dist/modules.mjs"
},
"./background-client": {
"types": "./dist/background-client.d.ts",
"default": "./dist/background-client.mjs"
}
},
"scripts": {
Expand Down Expand Up @@ -123,7 +127,7 @@
"publish-browser-extension": "^2.2.2",
"scule": "^1.3.0",
"unimport": "^3.13.1",
"vite": "^5.0.0 || ^6.0.0",
"vite": "^5.0.0 || <=6.0.8",
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
"vite": "^5.0.0 || <=6.0.8",
"vite": "^5.0.0 || ^6.0.0",

"vite-node": "^2.1.4",
"web-ext-run": "^0.2.1",
"webextension-polyfill": "^0.12.0"
Expand Down
9 changes: 8 additions & 1 deletion packages/wxt/src/@types/globals.d.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,15 @@
/**
* Replaced by an import to `wxt/background-client`, which changes based on the
* manifest version being targeted.
*/
declare const __WXT_BACKGROUND_CLIENT_IMPORT__: void;

// Dev server globals, replaced during development
declare const __DEV_SERVER_PROTOCOL__: string;
declare const __DEV_SERVER_HOSTNAME__: string;
declare const __DEV_SERVER_PORT__: string;

// Globals defined by the vite-plugins/devServerGlobals.ts and utils/globals.ts
// Global env vars provided by WXT
interface ImportMetaEnv {
readonly COMMAND: WxtCommand;
readonly MANIFEST_VERSION: 2 | 3;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,9 +1,57 @@
/**
* This file contains all the code required for the background to communicate
* with the dev server during development:
*
* - Reload extension or parts of extension when files are saved.
* - Keep MV3 service worker alive indefinitely.
*
* It is not imported directly (`import "wxt/background-client"`), but from
* the dev server (`import "http://localhost:3000/@id/wxt/background-client"`)
* to ensure `import.meta.hot` is defined by Vite.
*/

import { browser } from 'wxt/browser';
import { logger } from '../../sandbox/utils/logger';
import { logger } from './sandbox/utils/logger';
import { MatchPattern } from 'wxt/sandbox';
import type { ReloadContentScriptPayload } from '../../sandbox/dev-server-websocket';

export function reloadContentScript(payload: ReloadContentScriptPayload) {
console.log(import.meta.hot);
if (import.meta.hot) {
import.meta.hot.on('wxt:reload-extension', () => browser.runtime.reload());
import.meta.hot.on('wxt:reload-content-script', (event) =>
reloadContentScript(event),
);

if (import.meta.env.MANIFEST_VERSION === 3) {
let backgroundInitialized = false;
// Tell the server the background script is loaded and ready to receive events
import.meta.hot.on('vite:ws:connect', () => {
if (backgroundInitialized) return;

import.meta.hot?.send('wxt:background-initialized');
backgroundInitialized = true;
});

// Web Socket will disconnect if the service worker is killed. Supposedly,
// just having a web socket connection active should keep the service worker
// alive, but when this was originally implemented on older versions of
// Chrome, that was not true. So this code has stayed around.
// See: https://developer.chrome.com/blog/longer-esw-lifetimes/
setInterval(async () => {
// Calling an async browser API resets the service worker's timeout
await browser.runtime.getPlatformInfo();
}, 5e3);
}

browser.commands.onCommand.addListener((command) => {
if (command === 'wxt:reload-extension') {
browser.runtime.reload();
}
});
} else {
console.error('[wxt] HMR context, import.meta.hot, not found');
}

function reloadContentScript(payload: ReloadContentScriptPayload) {
const manifest = browser.runtime.getManifest();
if (manifest.manifest_version == 2) {
void reloadContentScriptMv2(payload);
Expand All @@ -12,7 +60,7 @@ export function reloadContentScript(payload: ReloadContentScriptPayload) {
}
}

export async function reloadContentScriptMv3({
async function reloadContentScriptMv3({
registration,
contentScript,
}: ReloadContentScriptPayload) {
Expand All @@ -25,9 +73,7 @@ export async function reloadContentScriptMv3({

type ContentScript = ReloadContentScriptPayload['contentScript'];

export async function reloadManifestContentScriptMv3(
contentScript: ContentScript,
) {
async function reloadManifestContentScriptMv3(contentScript: ContentScript) {
const id = `wxt:${contentScript.js![0]}`;
logger.log('Reloading content script:', contentScript);
const registered = await browser.scripting.getRegisteredContentScripts();
Expand All @@ -46,9 +92,7 @@ export async function reloadManifestContentScriptMv3(
await reloadTabsForContentScript(contentScript);
}

export async function reloadRuntimeContentScriptMv3(
contentScript: ContentScript,
) {
async function reloadRuntimeContentScriptMv3(contentScript: ContentScript) {
logger.log('Reloading content script:', contentScript);
const registered = await browser.scripting.getRegisteredContentScripts();
logger.debug('Existing scripts:', registered);
Expand Down Expand Up @@ -92,8 +136,15 @@ async function reloadTabsForContentScript(contentScript: ContentScript) {
);
}

export async function reloadContentScriptMv2(
_payload: ReloadContentScriptPayload,
) {
async function reloadContentScriptMv2(_payload: ReloadContentScriptPayload) {
throw Error('TODO: reloadContentScriptMv2');
}

interface ReloadContentScriptPayload {
registration?: 'manifest' | 'runtime';
contentScript: {
matches: string[];
js?: string[];
css?: string[];
};
}
2 changes: 1 addition & 1 deletion packages/wxt/src/core/builders/vite/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,7 @@ export async function createViteBuilder(
config.plugins.push(
wxtPlugins.download(wxtConfig),
wxtPlugins.devHtmlPrerender(wxtConfig, server),
wxtPlugins.resolveVirtualModules(wxtConfig),
wxtPlugins.resolveVirtualModules(wxtConfig, server),
wxtPlugins.devServerGlobals(wxtConfig, server),
wxtPlugins.tsconfigPaths(wxtConfig),
wxtPlugins.noopBackground(),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ export function devServerGlobals(
return {
name: 'wxt:dev-server-globals',
config() {
if (server == null || config.command == 'build') return;
if (server == null || config.command == 'build') return {};

return {
define: {
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { Plugin } from 'vite';
import { ResolvedConfig } from '../../../../types';
import { ResolvedConfig, WxtDevServer } from '../../../../types';
import { normalizePath } from '../../../utils/paths';
import {
VirtualModuleId,
Expand All @@ -11,10 +11,13 @@ import { resolve } from 'path';
/**
* Resolve all the virtual modules to the `node_modules/wxt/dist/virtual` directory.
*/
export function resolveVirtualModules(config: ResolvedConfig): Plugin[] {
export function resolveVirtualModules(
config: ResolvedConfig,
server: WxtDevServer | undefined,
): Plugin[] {
return virtualModuleNames.map((name) => {
const virtualId: `${VirtualModuleId}?` = `virtual:wxt-${name}?`;
const resolvedVirtualId = '\0' + virtualId;
const resolvedVirtualId = 'resolved-' + virtualId;
return {
name: `wxt:resolve-virtual-${name}`,
resolveId(id) {
Expand All @@ -29,12 +32,26 @@ export function resolveVirtualModules(config: ResolvedConfig): Plugin[] {
async load(id) {
if (!id.startsWith(resolvedVirtualId)) return;

let wxtBackgroundClientImport: string = '';
if (server && config.command === 'serve') {
const wxtBackgroundClientUrl = `http://${server.hostname}:${server.port}/@id/wxt/background-client`;
wxtBackgroundClientImport =
config.manifestVersion === 2
? `import(/* @vite-ignore */ "${wxtBackgroundClientUrl}")`
: `/* @vite-ignore */\nimport "${wxtBackgroundClientUrl}"`;
}

const inputPath = id.replace(resolvedVirtualId, '');
const template = await fs.readFile(
resolve(config.wxtModuleDir, `dist/virtual/${name}.mjs`),
'utf-8',
);
return template.replace(`virtual:user-${name}`, inputPath);
return template
.replace(`virtual:user-${name}`, inputPath)
.replace(
`__WXT_BACKGROUND_CLIENT_IMPORT__`,
wxtBackgroundClientImport,
);
},
};
});
Expand Down
16 changes: 14 additions & 2 deletions packages/wxt/src/core/utils/manifest.ts
Original file line number Diff line number Diff line change
Expand Up @@ -116,8 +116,11 @@ export async function generateManifest(

addEntrypoints(manifest, entrypoints, buildOutput);

if (wxt.config.command === 'serve') addDevModeCsp(manifest);
if (wxt.config.command === 'serve') addDevModePermissions(manifest);
if (wxt.config.command === 'serve') {
addDevModeCsp(manifest);
addDevModePermissions(manifest);
setDevModeBackgroundToEsm(manifest);
}

// TODO: Remove in v1
wxt.config.transformManifest?.(manifest);
Expand Down Expand Up @@ -509,6 +512,15 @@ function addDevModePermissions(manifest: Manifest.WebExtensionManifest) {
if (wxt.config.manifestVersion === 3) addPermission(manifest, 'scripting');
}

/** MV3 service worker must by `type: "module"` to connect to dev server. */
function setDevModeBackgroundToEsm(manifest: Manifest.WebExtensionManifest) {
if (wxt.config.manifestVersion === 3) {
// Background must be ESM for it to be able to connect to the dev server
(manifest as any).background ??= {};
(manifest as any).background.type = 'module';
}
}

/**
* Returns the bundle paths to CSS files associated with a list of content scripts, or undefined if
* there is no associated CSS.
Expand Down
85 changes: 0 additions & 85 deletions packages/wxt/src/sandbox/dev-server-websocket.ts

This file was deleted.

35 changes: 1 addition & 34 deletions packages/wxt/src/virtual/background-entrypoint.ts
Original file line number Diff line number Diff line change
@@ -1,40 +1,7 @@
import definition from 'virtual:user-background-entrypoint';
import { initPlugins } from 'virtual:wxt-plugins';
import { getDevServerWebSocket } from '../sandbox/dev-server-websocket';
import { logger } from '../sandbox/utils/logger';
import { browser } from 'wxt/browser';
import { keepServiceWorkerAlive } from './utils/keep-service-worker-alive';
import { reloadContentScript } from './utils/reload-content-scripts';

if (import.meta.env.COMMAND === 'serve') {
try {
const ws = getDevServerWebSocket();
ws.addWxtEventListener('wxt:reload-extension', () => {
browser.runtime.reload();
});
ws.addWxtEventListener('wxt:reload-content-script', (event) => {
reloadContentScript(event.detail);
});

if (import.meta.env.MANIFEST_VERSION === 3) {
// Tell the server the background script is loaded and ready to go
ws.addEventListener('open', () =>
ws.sendCustom('wxt:background-initialized'),
);

// Web Socket will disconnect if the service worker is killed
keepServiceWorkerAlive();
}
} catch (err) {
logger.error('Failed to setup web socket connection with dev server', err);
}

browser.commands.onCommand.addListener((command) => {
if (command === 'wxt:reload-extension') {
browser.runtime.reload();
}
});
}
__WXT_BACKGROUND_CLIENT_IMPORT__;

let result;

Expand Down
18 changes: 5 additions & 13 deletions packages/wxt/src/virtual/reload-html.ts
Original file line number Diff line number Diff line change
@@ -1,14 +1,6 @@
import { logger } from '../sandbox/utils/logger';
import { getDevServerWebSocket } from '../sandbox/dev-server-websocket';

if (import.meta.env.COMMAND === 'serve') {
try {
const ws = getDevServerWebSocket();
ws.addWxtEventListener('wxt:reload-page', (event) => {
// "popup.html" === "/popup.html".substring(1)
if (event.detail === location.pathname.substring(1)) location.reload();
});
} catch (err) {
logger.error('Failed to setup web socket connection with dev server', err);
}
if (import.meta.hot) {
import.meta.hot.on('wxt:reload-page', (event) => {
// "popup.html" === "/popup.html".substring(1)
if (event.detail === location.pathname.substring(1)) location.reload();
});
}
Comment on lines +1 to 6
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

TODO: Test

Loading