aboutsummaryrefslogtreecommitdiff
path: root/src/runtime/hmr.ts
diff options
context:
space:
mode:
Diffstat (limited to 'src/runtime/hmr.ts')
-rw-r--r--src/runtime/hmr.ts192
1 files changed, 152 insertions, 40 deletions
diff --git a/src/runtime/hmr.ts b/src/runtime/hmr.ts
index 5c91683a2..213192c4f 100644
--- a/src/runtime/hmr.ts
+++ b/src/runtime/hmr.ts
@@ -1,7 +1,7 @@
import { ByteBuffer } from "peechy";
import * as API from "../api/schema";
-var __HMRModule, __FastRefreshModule, __HMRClient;
+var __HMRModule, __FastRefreshModule, __HMRClient, __injectFastRefresh;
if (typeof window !== "undefined") {
// We add a scope here to minimize chances of namespace collisions
var runOnce = false;
@@ -60,6 +60,42 @@ if (typeof window !== "undefined") {
},
};
+ const BunError = {
+ module: null,
+ prom: null,
+ cancel: false,
+ lastError: null,
+ render(error, cwd) {
+ BunError.lastError = [error, cwd];
+ BunError.cancel = false;
+
+ if (!BunError.module) {
+ if (BunError.prom) return;
+ BunError.prom = import("/bun:error.js").then((mod) => {
+ BunError.module = mod;
+ !BunError.cancel &&
+ BunError.render(BunError.lastError[0], BunError.lastError[1]);
+ });
+ return;
+ }
+
+ const { renderBuildFailure } = BunError.module;
+ renderBuildFailure(BunError.lastError[0], BunError.lastError[1]);
+ },
+
+ clear() {
+ BunError.lastError = null;
+ BunError.cancel = true;
+
+ if (BunError.module) {
+ const { clearBuildFailure } = BunError.module;
+ clearBuildFailure();
+ } else if ("__BunClearBuildFailure" in globalThis) {
+ globalThis.__BunClearBuildFailure();
+ }
+ },
+ };
+
class CSSLoader {
hmr: HMRClient;
private static cssLoadId: CSSHMRInsertionPoint = {
@@ -284,6 +320,9 @@ if (typeof window !== "undefined") {
}
let filepath = update.file;
+ if (filepath.startsWith(this.hmr.cwd)) {
+ filepath = filepath.substring(this.hmr.cwd.length);
+ }
const _timestamp = timestamp;
const from_timestamp = build.from_timestamp;
function onLoadHandler() {
@@ -635,11 +674,56 @@ if (typeof window !== "undefined") {
}
maybeReportBuildFailure(failure: API.WebsocketMessageBuildFailure) {
- globalThis.renderBuildFailure(failure, this.cwd);
+ BunError.render(failure, this.cwd);
}
+
+ needsConsoleClear = false;
+
reportBuildFailure(failure: API.WebsocketMessageBuildFailure) {
- __hmrlog.error("Build failed", failure.module_path);
- globalThis.renderBuildFailure(failure, this.cwd);
+ BunError.render(failure, this.cwd);
+
+ console.group(
+ `Build failed: ${failure.module_path} (${failure.log.errors} errors)`
+ );
+ this.needsConsoleClear = true;
+ for (let msg of failure.log.msgs) {
+ var logFunction;
+ switch (msg.level) {
+ case API.MessageLevel.err: {
+ logFunction = console.error;
+ break;
+ }
+ case API.MessageLevel.warn: {
+ logFunction = console.warn;
+ break;
+ }
+ default: {
+ logFunction = console.info;
+ break;
+ }
+ }
+
+ const { text, location } = msg.data;
+ var output = "";
+
+ if (location) {
+ if (location.line > -1 && location.column > -1) {
+ output += `${location.file}:${location.line}:${location.column}`;
+ } else if (location.line > -1) {
+ output += `${location.file}:${location.line}`;
+ } else if (location.file.length > 0) {
+ output += `${location.file}`;
+ }
+ }
+ if (location && location.line_text) {
+ output = output.trimRight() + "\n" + location.line_text.trim();
+ }
+
+ output = output.trimRight() + "\n " + text;
+
+ logFunction(output.trim());
+ }
+ console.groupEnd();
}
verbose = false;
@@ -671,6 +755,11 @@ if (typeof window !== "undefined") {
}
if (build.loader === API.Loader.css) {
+ BunError.clear();
+ if (this.needsConsoleClear) {
+ console.clear();
+ this.needsConsoleClear = false;
+ }
return this.loaders.css.handleBuildSuccess(buffer, build, timestamp);
}
@@ -686,10 +775,11 @@ if (typeof window !== "undefined") {
}
if (this.verbose) {
- __hmrlog.debug(
- "Preparing to reload",
- HMRModule.dependencies.modules[index].file_path
- );
+ var filepath = HMRModule.dependencies.modules[index].file_path;
+ if (filepath.startsWith(this.cwd)) {
+ filepath = filepath.substring(this.cwd.length);
+ }
+ __hmrlog.debug("Preparing to reload", filepath);
}
var reload = new HotReload(
@@ -703,11 +793,25 @@ if (typeof window !== "undefined") {
ReloadBehavior.hotReload
);
reload.timings.notify = timestamp - build.from_timestamp;
+
+ BunError.clear();
+
reload.run().then(
([module, timings]) => {
+ var filepath = module.file_path;
+
+ if (filepath.startsWith(this.cwd)) {
+ filepath = filepath.substring(this.cwd.length);
+ }
+
+ if (this.needsConsoleClear) {
+ console.clear();
+ this.needsConsoleClear = false;
+ }
+
__hmrlog.log(
- `Reloaded in ${formatDuration(timings.total)}ms :`,
- module.file_path
+ `[${formatDuration(timings.total)}ms] Reloaded`,
+ filepath
);
},
(err) => {
@@ -1012,6 +1116,11 @@ if (typeof window !== "undefined") {
this.module_index
].additional_updaters.push(oldModule.update.bind(oldModule));
}
+
+ const end = Math.min(
+ this.module_index + 1,
+ HMRModule.dependencies.graph_used
+ );
// -- For generic hot reloading --
// ES Modules delay execution until all imports are parsed
// They execute depth-first
@@ -1029,35 +1138,31 @@ if (typeof window !== "undefined") {
// If we do not find a React Refresh boundary, we must instead perform a full page reload.
for (
let i = 0;
- i <= this.module_index;
+ i <= end;
i++ // let i = HMRModule.dependencies.graph_used - 1; // i > this.module_index; // i--
) {
- let handled =
- !HMRModule.dependencies.modules[i].exports.__hmrDisable;
- if (
- typeof HMRModule.dependencies.modules[i].dispose === "function"
- ) {
- HMRModule.dependencies.modules[i].dispose();
- handled = true;
- }
- if (
- typeof HMRModule.dependencies.modules[i].accept === "function"
- ) {
- HMRModule.dependencies.modules[i].accept();
- handled = true;
- }
-
- // Automatically re-initialize the dependency
- if (!handled) {
- HMRModule.dependencies.modules[i].update();
- }
-
- // If we don't find a boundary, we will need to do a full page load
- if (
- (HMRModule.dependencies.modules[i] as FastRefreshModule)
- .isRefreshBoundary
- ) {
- foundBoundary = true;
+ const mod = HMRModule.dependencies.modules[i];
+ let handled = false;
+
+ if (!mod.exports.__hmrDisable) {
+ if (typeof mod.dispose === "function") {
+ mod.dispose();
+ handled = true;
+ }
+ if (typeof mod.accept === "function") {
+ mod.accept();
+ handled = true;
+ }
+
+ // If we don't find a boundary, we will need to do a full page load
+ if ((mod as FastRefreshModule).isRefreshBoundary) {
+ foundBoundary = true;
+ }
+
+ // Automatically re-initialize the dependency
+ if (!handled) {
+ mod.update();
+ }
}
}
@@ -1204,6 +1309,13 @@ if (typeof window !== "undefined") {
exports = {};
}
+ function injectFastRefresh(RefreshRuntime) {
+ if (!FastRefreshLoader.hasInjectedFastRefresh) {
+ RefreshRuntime.injectIntoGlobalHook(globalThis);
+ FastRefreshLoader.hasInjectedFastRefresh = true;
+ }
+ }
+
class FastRefreshModule extends HMRModule {
constructor(id: number, file_path: string, RefreshRuntime: any) {
super(id, file_path);
@@ -1332,15 +1444,15 @@ if (typeof window !== "undefined") {
__HMRModule = HMRModule;
__FastRefreshModule = FastRefreshModule;
__HMRClient = HMRClient;
-
+ __injectFastRefresh = injectFastRefresh;
if ("document" in globalThis) {
document.addEventListener("onimportcss", HMRClient.onCSSImport, {
passive: true,
});
- window.addEventListener("error", HMRClient.onError, { passive: true });
+ // window.addEventListener("error", HMRClient.onError, { passive: true });
}
globalThis["__BUN"] = HMRClient;
}
-export { __HMRModule, __FastRefreshModule, __HMRClient };
+export { __HMRModule, __FastRefreshModule, __HMRClient, __injectFastRefresh };