@tscircuit/eval 0.0.299 → 0.0.301
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/blob-url.js +1 -1
- package/dist/eval/index.d.ts +184 -5
- package/dist/eval/index.js +19 -4
- package/dist/lib/index.d.ts +185 -5
- package/dist/lib/index.js +25 -4
- package/dist/webworker/entrypoint.js +363 -454
- package/dist/worker.d.ts +3 -1
- package/dist/worker.js +7 -1
- package/lib/runner/CircuitRunner.ts +12 -0
- package/lib/shared/types.ts +5 -1
- package/lib/worker.ts +7 -1
- package/package.json +11 -10
- package/tests/features/enable-debug.test.ts +33 -0
- package/webworker/entrypoint.ts +12 -0
- package/webworker/execution-context.ts +6 -0
package/dist/worker.d.ts
CHANGED
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
import { AnyCircuitElement } from 'circuit-json';
|
|
2
|
-
import { RootCircuitEventName } from '@tscircuit/core';
|
|
2
|
+
import { RootCircuitEventName as RootCircuitEventName$1 } from '@tscircuit/core';
|
|
3
3
|
import { PlatformConfig } from '@tscircuit/props';
|
|
4
4
|
|
|
5
5
|
declare const getImportsFromCode: (code: string) => string[];
|
|
6
6
|
|
|
7
|
+
type RootCircuitEventName = RootCircuitEventName$1 | "debug:logOutput";
|
|
7
8
|
interface CircuitRunnerConfiguration {
|
|
8
9
|
snippetsApiBaseUrl: string;
|
|
9
10
|
cjsRegistryUrl: string;
|
|
@@ -37,6 +38,7 @@ type CircuitWebWorker = {
|
|
|
37
38
|
getCircuitJson: () => Promise<AnyCircuitElement[]>;
|
|
38
39
|
on: (event: RootCircuitEventName, callback: (...args: any[]) => void) => void;
|
|
39
40
|
clearEventListeners: () => void;
|
|
41
|
+
enableDebug: (namespace: string) => Promise<void>;
|
|
40
42
|
version: () => Promise<string>;
|
|
41
43
|
kill: () => Promise<void>;
|
|
42
44
|
};
|
package/dist/worker.js
CHANGED
|
@@ -151,6 +151,12 @@ var createCircuitWebWorker = async (configuration) => {
|
|
|
151
151
|
let isTerminated = false;
|
|
152
152
|
const wrapper = {
|
|
153
153
|
clearEventListeners: comlinkWorker.clearEventListeners.bind(comlinkWorker),
|
|
154
|
+
enableDebug: async (...args) => {
|
|
155
|
+
if (isTerminated) {
|
|
156
|
+
throw new Error("CircuitWebWorker was terminated, can't enableDebug");
|
|
157
|
+
}
|
|
158
|
+
return comlinkWorker.enableDebug.bind(comlinkWorker)(...args);
|
|
159
|
+
},
|
|
154
160
|
version: comlinkWorker.version.bind(comlinkWorker),
|
|
155
161
|
execute: async (...args) => {
|
|
156
162
|
if (isTerminated) {
|
|
@@ -206,4 +212,4 @@ export {
|
|
|
206
212
|
createCircuitWebWorker,
|
|
207
213
|
getImportsFromCode
|
|
208
214
|
};
|
|
209
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../lib/worker.ts", "../lib/utils/get-imports-from-code.ts"],
  "sourcesContent": ["import * as Comlink from \"comlink\"\nexport * from \"./utils/index\"\nimport type {\n  InternalWebWorkerApi,\n  WebWorkerConfiguration,\n  CircuitWebWorker,\n} from \"./shared/types\"\nimport type { RootCircuitEventName } from \"@tscircuit/core\"\n\nexport type { CircuitWebWorker, WebWorkerConfiguration }\n\ndeclare global {\n  interface Window {\n    TSCIRCUIT_GLOBAL_CIRCUIT_WORKER: CircuitWebWorker | undefined\n  }\n  var TSCIRCUIT_GLOBAL_CIRCUIT_WORKER: CircuitWebWorker | undefined\n}\n\nexport const createCircuitWebWorker = async (\n  configuration: Partial<WebWorkerConfiguration>,\n): Promise<CircuitWebWorker> => {\n  // Kill existing global worker instance if present\n  const existingWorker = globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER\n  if (existingWorker && typeof existingWorker.kill === \"function\") {\n    if (configuration.verbose) {\n      console.log(\"[Worker] Killing previous global worker instance...\")\n    }\n    try {\n      await existingWorker.kill()\n    } catch (e) {\n      if (configuration.verbose) {\n        console.warn(\n          \"[Worker] Error killing previous global worker instance:\",\n          e,\n        )\n      }\n      // Ensure the key is cleared even if kill throws an error\n      if (globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER === existingWorker) {\n        globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER = undefined\n      }\n    }\n  }\n\n  if (configuration.verbose) {\n    console.log(\n      \"[Worker] Creating circuit web worker with config:\",\n      configuration,\n    )\n  }\n\n  let workerBlobUrl =\n    configuration.webWorkerBlobUrl ?? configuration.webWorkerUrl\n\n  if (!workerBlobUrl) {\n    const cdnUrl = `https://cdn.jsdelivr.net/npm/@tscircuit/eval@${configuration.evalVersion ?? \"latest\"}/dist/webworker/entrypoint.js`\n\n    const workerBlob = await globalThis.fetch(cdnUrl).then((res) => res.blob())\n    workerBlobUrl = URL.createObjectURL(workerBlob)\n  }\n\n  const rawWorker = new Worker(workerBlobUrl, { type: \"module\" })\n  let workerInitError: any\n  rawWorker.addEventListener(\"error\", (event) => {\n    console.error(\"[Worker] Error in worker\", event)\n    workerInitError = event\n  })\n  rawWorker.addEventListener(\"unhandledrejection\", (event) => {\n    console.error(\"[Worker] Unhandled rejection in worker\", event)\n  })\n  rawWorker.addEventListener(\"messageerror\", (event) => {\n    console.error(\"[Worker] Message error in worker\", event)\n  })\n  const earlyMessageHandler = (event: MessageEvent) => {\n    console.log(\"[Worker] Message in worker\", event)\n  }\n  rawWorker.addEventListener(\"message\", earlyMessageHandler)\n\n  // Handle fetch requests from the worker\n  rawWorker.addEventListener(\"message\", async (event: MessageEvent) => {\n    const data = event.data\n    if (data?.type !== \"worker_fetch\") return\n\n    try {\n      const response = await globalThis.fetch(data.input, data.init)\n      const body = await response.text()\n      rawWorker.postMessage({\n        type: \"worker_fetch_result\",\n        requestId: data.requestId,\n        success: true,\n        response: {\n          body,\n          status: response.status,\n          statusText: response.statusText,\n          headers: (() => {\n            const obj: Record<string, string> = {}\n            response.headers.forEach((value, key) => {\n              obj[key] = value\n            })\n            return obj\n          })(),\n        },\n      })\n    } catch (err: any) {\n      rawWorker.postMessage({\n        type: \"worker_fetch_result\",\n        requestId: data.requestId,\n        success: false,\n        error: {\n          name: err.name,\n          message: err.message,\n          stack: err.stack,\n        },\n      })\n    }\n  })\n\n  if (workerInitError) {\n    throw workerInitError\n  }\n\n  const comlinkWorker = Comlink.wrap<InternalWebWorkerApi>(rawWorker)\n\n  rawWorker.removeEventListener(\"message\", earlyMessageHandler)\n\n  // Helper to serialize React elements for cross-worker communication\n  function serializeReactElement(element: any): any {\n    if (!element || typeof element !== \"object\") {\n      return element\n    }\n\n    if (element.type && element.props !== undefined) {\n      // This is a React element\n      return {\n        __isSerializedReactElement: true,\n        type: element.type,\n        props: serializeProps(element.props),\n        key: element.key,\n      }\n    }\n\n    return element\n  }\n\n  function serializeProps(props: any): any {\n    if (!props || typeof props !== \"object\") {\n      return props\n    }\n\n    const serialized: any = {}\n    for (const [key, value] of Object.entries(props)) {\n      if (key === \"children\") {\n        if (Array.isArray(value)) {\n          serialized.children = value.map(serializeReactElement)\n        } else {\n          serialized.children = serializeReactElement(value)\n        }\n      } else {\n        serialized[key] = value\n      }\n    }\n    return serialized\n  }\n\n  // Conditionally override global fetch inside the worker to route through the parent\n  // Only enable when explicitly requested via configuration\n  if (configuration.enableFetchProxy) {\n    rawWorker.postMessage({ type: \"override_global_fetch\" })\n  }\n\n  if (configuration.snippetsApiBaseUrl) {\n    await comlinkWorker.setSnippetsApiBaseUrl(configuration.snippetsApiBaseUrl)\n  }\n  if (configuration.platform) {\n    await comlinkWorker.setPlatformConfig(configuration.platform)\n  }\n\n  let isTerminated = false\n\n  // Create a wrapper that handles events directly through circuit instance\n  const wrapper: CircuitWebWorker = {\n    clearEventListeners: comlinkWorker.clearEventListeners.bind(comlinkWorker),\n    version: comlinkWorker.version.bind(comlinkWorker),\n    execute: async (...args) => {\n      if (isTerminated) {\n        throw new Error(\"CircuitWebWorker was terminated, can't execute\")\n      }\n      return comlinkWorker.execute.bind(comlinkWorker)(...args)\n    },\n    executeComponent: async (component: any) => {\n      if (isTerminated) {\n        throw new Error(\n          \"CircuitWebWorker was terminated, can't executeComponent\",\n        )\n      }\n\n      // If it's a function, pass it as-is (will be proxied by Comlink)\n      if (typeof component === \"function\") {\n        return comlinkWorker.executeComponent.bind(comlinkWorker)(component)\n      }\n\n      // If it's a React element, serialize it to a reconstructable format\n      if (component && typeof component === \"object\" && component.type) {\n        const serializedElement = serializeReactElement(component)\n        return comlinkWorker.executeComponent.bind(comlinkWorker)(\n          serializedElement,\n        )\n      }\n\n      return comlinkWorker.executeComponent.bind(comlinkWorker)(component)\n    },\n    executeWithFsMap: async (...args) => {\n      if (isTerminated) {\n        throw new Error(\n          \"CircuitWebWorker was terminated, can't executeWithFsMap\",\n        )\n      }\n      return comlinkWorker.executeWithFsMap.bind(comlinkWorker)(...args)\n    },\n    renderUntilSettled: comlinkWorker.renderUntilSettled.bind(comlinkWorker),\n    getCircuitJson: comlinkWorker.getCircuitJson.bind(comlinkWorker),\n    on: (event: string, callback: (...args: any[]) => void) => {\n      const proxiedCallback = Comlink.proxy(callback)\n      comlinkWorker.on(event as RootCircuitEventName, proxiedCallback)\n    },\n    kill: async () => {\n      comlinkWorker[Comlink.releaseProxy]()\n      rawWorker.terminate()\n      isTerminated = true\n      if (globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER === wrapper) {\n        globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER = undefined\n      }\n    },\n  }\n  ;(wrapper as any).__rawWorker = rawWorker\n  globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER = wrapper\n  return wrapper\n}\n", "export const getImportsFromCode = (code: string): string[] => {\n  // Match basic import patterns including combined default and namespace imports\n  const importRegex =\n    /^\\s*import\\s+(?:(?:[\\w\\s]+,\\s*)?(?:\\*\\s+as\\s+[\\w\\s]+|\\{[\\s\\w,]+\\}|\\w+)\\s+from\\s+)?['\"](.+?)['\"]/gm\n  const imports: string[] = []\n  let match: RegExpExecArray | null\n\n  // biome-ignore lint/suspicious/noAssignInExpressions: <explanation>\n  while ((match = importRegex.exec(code)) !== null) {\n    imports.push(match[1])\n  }\n\n  // Match re-exports\n  const reExportRegex =\n    /^\\s*export\\s+(?:\\*|(?:\\{[\\s\\w,]+\\}))\\s+from\\s+['\"](.+?)['\"]/gm\n  let reExportMatch: RegExpExecArray | null\n  // biome-ignore lint/suspicious/noAssignInExpressions: <explanation>\n  while ((reExportMatch = reExportRegex.exec(code)) !== null) {\n    imports.push(reExportMatch[1])\n  }\n\n  return imports\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
215
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../lib/worker.ts", "../lib/utils/get-imports-from-code.ts"],
  "sourcesContent": ["import * as Comlink from \"comlink\"\nexport * from \"./utils/index\"\nimport type {\n  InternalWebWorkerApi,\n  WebWorkerConfiguration,\n  CircuitWebWorker,\n} from \"./shared/types\"\nimport type { RootCircuitEventName } from \"./shared/types\"\n\nexport type { CircuitWebWorker, WebWorkerConfiguration }\n\ndeclare global {\n  interface Window {\n    TSCIRCUIT_GLOBAL_CIRCUIT_WORKER: CircuitWebWorker | undefined\n  }\n  var TSCIRCUIT_GLOBAL_CIRCUIT_WORKER: CircuitWebWorker | undefined\n}\n\nexport const createCircuitWebWorker = async (\n  configuration: Partial<WebWorkerConfiguration>,\n): Promise<CircuitWebWorker> => {\n  // Kill existing global worker instance if present\n  const existingWorker = globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER\n  if (existingWorker && typeof existingWorker.kill === \"function\") {\n    if (configuration.verbose) {\n      console.log(\"[Worker] Killing previous global worker instance...\")\n    }\n    try {\n      await existingWorker.kill()\n    } catch (e) {\n      if (configuration.verbose) {\n        console.warn(\n          \"[Worker] Error killing previous global worker instance:\",\n          e,\n        )\n      }\n      // Ensure the key is cleared even if kill throws an error\n      if (globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER === existingWorker) {\n        globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER = undefined\n      }\n    }\n  }\n\n  if (configuration.verbose) {\n    console.log(\n      \"[Worker] Creating circuit web worker with config:\",\n      configuration,\n    )\n  }\n\n  let workerBlobUrl =\n    configuration.webWorkerBlobUrl ?? configuration.webWorkerUrl\n\n  if (!workerBlobUrl) {\n    const cdnUrl = `https://cdn.jsdelivr.net/npm/@tscircuit/eval@${configuration.evalVersion ?? \"latest\"}/dist/webworker/entrypoint.js`\n\n    const workerBlob = await globalThis.fetch(cdnUrl).then((res) => res.blob())\n    workerBlobUrl = URL.createObjectURL(workerBlob)\n  }\n\n  const rawWorker = new Worker(workerBlobUrl, { type: \"module\" })\n  let workerInitError: any\n  rawWorker.addEventListener(\"error\", (event) => {\n    console.error(\"[Worker] Error in worker\", event)\n    workerInitError = event\n  })\n  rawWorker.addEventListener(\"unhandledrejection\", (event) => {\n    console.error(\"[Worker] Unhandled rejection in worker\", event)\n  })\n  rawWorker.addEventListener(\"messageerror\", (event) => {\n    console.error(\"[Worker] Message error in worker\", event)\n  })\n  const earlyMessageHandler = (event: MessageEvent) => {\n    console.log(\"[Worker] Message in worker\", event)\n  }\n  rawWorker.addEventListener(\"message\", earlyMessageHandler)\n\n  // Handle fetch requests from the worker\n  rawWorker.addEventListener(\"message\", async (event: MessageEvent) => {\n    const data = event.data\n    if (data?.type !== \"worker_fetch\") return\n\n    try {\n      const response = await globalThis.fetch(data.input, data.init)\n      const body = await response.text()\n      rawWorker.postMessage({\n        type: \"worker_fetch_result\",\n        requestId: data.requestId,\n        success: true,\n        response: {\n          body,\n          status: response.status,\n          statusText: response.statusText,\n          headers: (() => {\n            const obj: Record<string, string> = {}\n            response.headers.forEach((value, key) => {\n              obj[key] = value\n            })\n            return obj\n          })(),\n        },\n      })\n    } catch (err: any) {\n      rawWorker.postMessage({\n        type: \"worker_fetch_result\",\n        requestId: data.requestId,\n        success: false,\n        error: {\n          name: err.name,\n          message: err.message,\n          stack: err.stack,\n        },\n      })\n    }\n  })\n\n  if (workerInitError) {\n    throw workerInitError\n  }\n\n  const comlinkWorker = Comlink.wrap<InternalWebWorkerApi>(rawWorker)\n\n  rawWorker.removeEventListener(\"message\", earlyMessageHandler)\n\n  // Helper to serialize React elements for cross-worker communication\n  function serializeReactElement(element: any): any {\n    if (!element || typeof element !== \"object\") {\n      return element\n    }\n\n    if (element.type && element.props !== undefined) {\n      // This is a React element\n      return {\n        __isSerializedReactElement: true,\n        type: element.type,\n        props: serializeProps(element.props),\n        key: element.key,\n      }\n    }\n\n    return element\n  }\n\n  function serializeProps(props: any): any {\n    if (!props || typeof props !== \"object\") {\n      return props\n    }\n\n    const serialized: any = {}\n    for (const [key, value] of Object.entries(props)) {\n      if (key === \"children\") {\n        if (Array.isArray(value)) {\n          serialized.children = value.map(serializeReactElement)\n        } else {\n          serialized.children = serializeReactElement(value)\n        }\n      } else {\n        serialized[key] = value\n      }\n    }\n    return serialized\n  }\n\n  // Conditionally override global fetch inside the worker to route through the parent\n  // Only enable when explicitly requested via configuration\n  if (configuration.enableFetchProxy) {\n    rawWorker.postMessage({ type: \"override_global_fetch\" })\n  }\n\n  if (configuration.snippetsApiBaseUrl) {\n    await comlinkWorker.setSnippetsApiBaseUrl(configuration.snippetsApiBaseUrl)\n  }\n  if (configuration.platform) {\n    await comlinkWorker.setPlatformConfig(configuration.platform)\n  }\n\n  let isTerminated = false\n\n  // Create a wrapper that handles events directly through circuit instance\n  const wrapper: CircuitWebWorker = {\n    clearEventListeners: comlinkWorker.clearEventListeners.bind(comlinkWorker),\n    enableDebug: async (...args) => {\n      if (isTerminated) {\n        throw new Error(\"CircuitWebWorker was terminated, can't enableDebug\")\n      }\n      return comlinkWorker.enableDebug.bind(comlinkWorker)(...args)\n    },\n    version: comlinkWorker.version.bind(comlinkWorker),\n    execute: async (...args) => {\n      if (isTerminated) {\n        throw new Error(\"CircuitWebWorker was terminated, can't execute\")\n      }\n      return comlinkWorker.execute.bind(comlinkWorker)(...args)\n    },\n    executeComponent: async (component: any) => {\n      if (isTerminated) {\n        throw new Error(\n          \"CircuitWebWorker was terminated, can't executeComponent\",\n        )\n      }\n\n      // If it's a function, pass it as-is (will be proxied by Comlink)\n      if (typeof component === \"function\") {\n        return comlinkWorker.executeComponent.bind(comlinkWorker)(component)\n      }\n\n      // If it's a React element, serialize it to a reconstructable format\n      if (component && typeof component === \"object\" && component.type) {\n        const serializedElement = serializeReactElement(component)\n        return comlinkWorker.executeComponent.bind(comlinkWorker)(\n          serializedElement,\n        )\n      }\n\n      return comlinkWorker.executeComponent.bind(comlinkWorker)(component)\n    },\n    executeWithFsMap: async (...args) => {\n      if (isTerminated) {\n        throw new Error(\n          \"CircuitWebWorker was terminated, can't executeWithFsMap\",\n        )\n      }\n      return comlinkWorker.executeWithFsMap.bind(comlinkWorker)(...args)\n    },\n    renderUntilSettled: comlinkWorker.renderUntilSettled.bind(comlinkWorker),\n    getCircuitJson: comlinkWorker.getCircuitJson.bind(comlinkWorker),\n    on: (event: string, callback: (...args: any[]) => void) => {\n      const proxiedCallback = Comlink.proxy(callback)\n      comlinkWorker.on(event as RootCircuitEventName, proxiedCallback)\n    },\n    kill: async () => {\n      comlinkWorker[Comlink.releaseProxy]()\n      rawWorker.terminate()\n      isTerminated = true\n      if (globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER === wrapper) {\n        globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER = undefined\n      }\n    },\n  }\n  ;(wrapper as any).__rawWorker = rawWorker\n  globalThis.TSCIRCUIT_GLOBAL_CIRCUIT_WORKER = wrapper\n  return wrapper\n}\n", "export const getImportsFromCode = (code: string): string[] => {\n  // Match basic import patterns including combined default and namespace imports\n  const importRegex =\n    /^\\s*import\\s+(?:(?:[\\w\\s]+,\\s*)?(?:\\*\\s+as\\s+[\\w\\s]+|\\{[\\s\\w,]+\\}|\\w+)\\s+from\\s+)?['\"](.+?)['\"]/gm\n  const imports: string[] = []\n  let match: RegExpExecArray | null\n\n  // biome-ignore lint/suspicious/noAssignInExpressions: <explanation>\n  while ((match = importRegex.exec(code)) !== null) {\n    imports.push(match[1])\n  }\n\n  // Match re-exports\n  const reExportRegex =\n    /^\\s*export\\s+(?:\\*|(?:\\{[\\s\\w,]+\\}))\\s+from\\s+['\"](.+?)['\"]/gm\n  let reExportMatch: RegExpExecArray | null\n  // biome-ignore lint/suspicious/noAssignInExpressions: <explanation>\n  while ((reExportMatch = reExportRegex.exec(code)) !== null) {\n    imports.push(reExportMatch[1])\n  }\n\n  return imports\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
@@ -22,6 +22,7 @@ export class CircuitRunner implements CircuitRunnerApi {
|
|
|
22
22
|
verbose: false,
|
|
23
23
|
}
|
|
24
24
|
_eventListeners: Record<string, ((...args: any[]) => void)[]> = {}
|
|
25
|
+
_debugNamespace: string | undefined
|
|
25
26
|
|
|
26
27
|
constructor(configuration: Partial<CircuitRunnerConfiguration> = {}) {
|
|
27
28
|
Object.assign(this._circuitRunnerConfiguration, configuration)
|
|
@@ -62,6 +63,7 @@ export class CircuitRunner implements CircuitRunnerApi {
|
|
|
62
63
|
{
|
|
63
64
|
name: opts.name,
|
|
64
65
|
platform: this._circuitRunnerConfiguration.platform,
|
|
66
|
+
debugNamespace: this._debugNamespace,
|
|
65
67
|
},
|
|
66
68
|
)
|
|
67
69
|
this._bindEventListeners(this._executionContext.circuit)
|
|
@@ -93,6 +95,7 @@ export class CircuitRunner implements CircuitRunnerApi {
|
|
|
93
95
|
{
|
|
94
96
|
...opts,
|
|
95
97
|
platform: this._circuitRunnerConfiguration.platform,
|
|
98
|
+
debugNamespace: this._debugNamespace,
|
|
96
99
|
},
|
|
97
100
|
)
|
|
98
101
|
this._bindEventListeners(this._executionContext.circuit)
|
|
@@ -112,6 +115,7 @@ export class CircuitRunner implements CircuitRunnerApi {
|
|
|
112
115
|
{
|
|
113
116
|
...opts,
|
|
114
117
|
platform: this._circuitRunnerConfiguration.platform,
|
|
118
|
+
debugNamespace: this._debugNamespace,
|
|
115
119
|
},
|
|
116
120
|
)
|
|
117
121
|
this._bindEventListeners(this._executionContext.circuit)
|
|
@@ -172,6 +176,14 @@ export class CircuitRunner implements CircuitRunnerApi {
|
|
|
172
176
|
this._circuitRunnerConfiguration.platform = platform
|
|
173
177
|
}
|
|
174
178
|
|
|
179
|
+
async enableDebug(namespace: string) {
|
|
180
|
+
this._debugNamespace = namespace
|
|
181
|
+
if (this._executionContext) {
|
|
182
|
+
const circuit = this._executionContext.circuit as any
|
|
183
|
+
circuit.enableDebug?.(namespace)
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
|
|
175
187
|
private _bindEventListeners(circuit: RootCircuit) {
|
|
176
188
|
for (const event in this._eventListeners) {
|
|
177
189
|
for (const listener of this._eventListeners[event]) {
|
package/lib/shared/types.ts
CHANGED
|
@@ -1,7 +1,9 @@
|
|
|
1
1
|
import type { AnyCircuitElement } from "circuit-json"
|
|
2
|
-
import type { RootCircuitEventName } from "@tscircuit/core"
|
|
2
|
+
import type { RootCircuitEventName as CoreRootCircuitEventName } from "@tscircuit/core"
|
|
3
3
|
import type { PlatformConfig } from "@tscircuit/props"
|
|
4
4
|
|
|
5
|
+
export type RootCircuitEventName = CoreRootCircuitEventName | "debug:logOutput"
|
|
6
|
+
|
|
5
7
|
export interface CircuitRunnerConfiguration {
|
|
6
8
|
snippetsApiBaseUrl: string
|
|
7
9
|
cjsRegistryUrl: string
|
|
@@ -46,6 +48,7 @@ export interface CircuitRunnerApi {
|
|
|
46
48
|
getCircuitJson: () => Promise<AnyCircuitElement[]>
|
|
47
49
|
setSnippetsApiBaseUrl: (baseUrl: string) => Promise<void>
|
|
48
50
|
setPlatformConfig: (platform: PlatformConfig) => Promise<void>
|
|
51
|
+
enableDebug: (namespace: string) => Promise<void>
|
|
49
52
|
on: (event: RootCircuitEventName, callback: (...args: any[]) => void) => void
|
|
50
53
|
clearEventListeners: () => void
|
|
51
54
|
kill: () => Promise<void>
|
|
@@ -68,6 +71,7 @@ export type CircuitWebWorker = {
|
|
|
68
71
|
getCircuitJson: () => Promise<AnyCircuitElement[]>
|
|
69
72
|
on: (event: RootCircuitEventName, callback: (...args: any[]) => void) => void
|
|
70
73
|
clearEventListeners: () => void
|
|
74
|
+
enableDebug: (namespace: string) => Promise<void>
|
|
71
75
|
version: () => Promise<string>
|
|
72
76
|
kill: () => Promise<void>
|
|
73
77
|
}
|
package/lib/worker.ts
CHANGED
|
@@ -5,7 +5,7 @@ import type {
|
|
|
5
5
|
WebWorkerConfiguration,
|
|
6
6
|
CircuitWebWorker,
|
|
7
7
|
} from "./shared/types"
|
|
8
|
-
import type { RootCircuitEventName } from "
|
|
8
|
+
import type { RootCircuitEventName } from "./shared/types"
|
|
9
9
|
|
|
10
10
|
export type { CircuitWebWorker, WebWorkerConfiguration }
|
|
11
11
|
|
|
@@ -179,6 +179,12 @@ export const createCircuitWebWorker = async (
|
|
|
179
179
|
// Create a wrapper that handles events directly through circuit instance
|
|
180
180
|
const wrapper: CircuitWebWorker = {
|
|
181
181
|
clearEventListeners: comlinkWorker.clearEventListeners.bind(comlinkWorker),
|
|
182
|
+
enableDebug: async (...args) => {
|
|
183
|
+
if (isTerminated) {
|
|
184
|
+
throw new Error("CircuitWebWorker was terminated, can't enableDebug")
|
|
185
|
+
}
|
|
186
|
+
return comlinkWorker.enableDebug.bind(comlinkWorker)(...args)
|
|
187
|
+
},
|
|
182
188
|
version: comlinkWorker.version.bind(comlinkWorker),
|
|
183
189
|
execute: async (...args) => {
|
|
184
190
|
if (isTerminated) {
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@tscircuit/eval",
|
|
3
3
|
"main": "dist/lib/index.js",
|
|
4
|
-
"version": "0.0.
|
|
4
|
+
"version": "0.0.301",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"scripts": {
|
|
7
7
|
"build": "bun run build:lib && bun run build:webworker && bun run build:blob-url && bun run build:runner && bun run build:worker-wrapper",
|
|
@@ -57,20 +57,20 @@
|
|
|
57
57
|
"@tscircuit/checks": "^0.0.71",
|
|
58
58
|
"@tscircuit/circuit-json-flex": "^0.0.3",
|
|
59
59
|
"@tscircuit/circuit-json-util": "^0.0.67",
|
|
60
|
-
"@tscircuit/core": "^0.0.
|
|
60
|
+
"@tscircuit/core": "^0.0.699",
|
|
61
61
|
"@tscircuit/footprinter": "^0.0.208",
|
|
62
62
|
"@tscircuit/import-snippet": "^0.0.4",
|
|
63
63
|
"@tscircuit/infgrid-ijump-astar": "^0.0.33",
|
|
64
64
|
"@tscircuit/layout": "^0.0.28",
|
|
65
65
|
"@tscircuit/log-soup": "^1.0.2",
|
|
66
|
-
"@tscircuit/matchpack": "^0.0.
|
|
67
|
-
"@tscircuit/math-utils": "^0.0.
|
|
66
|
+
"@tscircuit/matchpack": "^0.0.16",
|
|
67
|
+
"@tscircuit/math-utils": "^0.0.21",
|
|
68
68
|
"@tscircuit/miniflex": "^0.0.4",
|
|
69
69
|
"@tscircuit/parts-engine": "^0.0.8",
|
|
70
|
-
"@tscircuit/props": "0.0.
|
|
70
|
+
"@tscircuit/props": "0.0.298",
|
|
71
71
|
"@tscircuit/schematic-autolayout": "^0.0.6",
|
|
72
72
|
"@tscircuit/schematic-match-adapt": "^0.0.16",
|
|
73
|
-
"@tscircuit/schematic-trace-solver": "^0.0.
|
|
73
|
+
"@tscircuit/schematic-trace-solver": "^0.0.34",
|
|
74
74
|
"@tscircuit/simple-3d-svg": "^0.0.38",
|
|
75
75
|
"@types/babel__standalone": "^7.1.9",
|
|
76
76
|
"@types/bun": "^1.2.16",
|
|
@@ -80,13 +80,13 @@
|
|
|
80
80
|
"@types/react-reconciler": "^0.28.9",
|
|
81
81
|
"bpc-graph": "^0.0.57",
|
|
82
82
|
"bun-match-svg": "0.0.12",
|
|
83
|
-
"calculate-elbow": "^0.0.
|
|
83
|
+
"calculate-elbow": "^0.0.11",
|
|
84
84
|
"chokidar-cli": "^3.0.0",
|
|
85
|
-
"circuit-json": "^0.0.
|
|
85
|
+
"circuit-json": "^0.0.245",
|
|
86
86
|
"circuit-json-to-bpc": "^0.0.13",
|
|
87
87
|
"circuit-json-to-connectivity-map": "^0.0.22",
|
|
88
88
|
"circuit-json-to-simple-3d": "^0.0.6",
|
|
89
|
-
"circuit-to-svg": "^0.0.
|
|
89
|
+
"circuit-to-svg": "^0.0.185",
|
|
90
90
|
"comlink": "^4.4.2",
|
|
91
91
|
"concurrently": "^9.1.2",
|
|
92
92
|
"debug": "^4.3.6",
|
|
@@ -101,7 +101,8 @@
|
|
|
101
101
|
"ts-expect": "^1.3.0",
|
|
102
102
|
"tsup": "^8.2.4",
|
|
103
103
|
"minicssgrid": "^0.0.9",
|
|
104
|
-
"connectivity-map": "^1.0.0"
|
|
104
|
+
"connectivity-map": "^1.0.0",
|
|
105
|
+
"flatbush": "^4.5.0"
|
|
105
106
|
},
|
|
106
107
|
"peerDependencies": {
|
|
107
108
|
"typescript": "^5.0.0",
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { expect, test } from "bun:test"
|
|
2
|
+
import { CircuitRunner } from "lib/runner/CircuitRunner"
|
|
3
|
+
import { createCircuitWebWorker } from "lib"
|
|
4
|
+
import { repoFileUrl } from "tests/fixtures/resourcePaths"
|
|
5
|
+
|
|
6
|
+
// enableDebug should cause the circuit to emit debug:logOutput events
|
|
7
|
+
|
|
8
|
+
test("CircuitRunner emits debug log", async () => {
|
|
9
|
+
const runner = new CircuitRunner()
|
|
10
|
+
const logs: any[] = []
|
|
11
|
+
runner.on("debug:logOutput", (output) => {
|
|
12
|
+
logs.push(output)
|
|
13
|
+
})
|
|
14
|
+
await runner.enableDebug("Group_doInitialPcbTraceRender")
|
|
15
|
+
await runner.execute("circuit.emit('debug:logOutput', 'hi')")
|
|
16
|
+
expect(logs).toContain("hi")
|
|
17
|
+
await runner.kill()
|
|
18
|
+
})
|
|
19
|
+
|
|
20
|
+
test("CircuitWebWorker emits debug log", async () => {
|
|
21
|
+
const worker = await createCircuitWebWorker({
|
|
22
|
+
webWorkerUrl: repoFileUrl("dist/webworker/entrypoint.js").href,
|
|
23
|
+
})
|
|
24
|
+
const logs: any[] = []
|
|
25
|
+
worker.on("debug:logOutput", (output) => {
|
|
26
|
+
logs.push(output)
|
|
27
|
+
})
|
|
28
|
+
await worker.enableDebug("Group_doInitialPcbTraceRender")
|
|
29
|
+
await worker.execute("circuit.emit('debug:logOutput', 'hi')")
|
|
30
|
+
await new Promise((r) => setTimeout(r, 0))
|
|
31
|
+
expect(logs).toContain("hi")
|
|
32
|
+
await worker.kill()
|
|
33
|
+
})
|
package/webworker/entrypoint.ts
CHANGED
|
@@ -20,6 +20,7 @@ globalThis.React = React
|
|
|
20
20
|
setupFetchProxy()
|
|
21
21
|
|
|
22
22
|
let executionContext: ExecutionContext | null = null
|
|
23
|
+
let debugNamespace: string | undefined
|
|
23
24
|
|
|
24
25
|
const circuitRunnerConfiguration: WebWorkerConfiguration = {
|
|
25
26
|
snippetsApiBaseUrl: "https://registry-api.tscircuit.com",
|
|
@@ -80,6 +81,14 @@ const webWorkerApi = {
|
|
|
80
81
|
circuitRunnerConfiguration.platform = platform
|
|
81
82
|
},
|
|
82
83
|
|
|
84
|
+
enableDebug: async (namespace: string) => {
|
|
85
|
+
debugNamespace = namespace
|
|
86
|
+
if (executionContext) {
|
|
87
|
+
const circuit = executionContext.circuit as any
|
|
88
|
+
circuit.enableDebug?.(namespace)
|
|
89
|
+
}
|
|
90
|
+
},
|
|
91
|
+
|
|
83
92
|
version: async () => {
|
|
84
93
|
return "0.0.0"
|
|
85
94
|
},
|
|
@@ -104,6 +113,7 @@ const webWorkerApi = {
|
|
|
104
113
|
executionContext = createExecutionContext(circuitRunnerConfiguration, {
|
|
105
114
|
name: opts.name,
|
|
106
115
|
platform: circuitRunnerConfiguration.platform,
|
|
116
|
+
debugNamespace,
|
|
107
117
|
})
|
|
108
118
|
bindEventListeners(executionContext.circuit)
|
|
109
119
|
executionContext.fsMap = normalizeFsMap(opts.fsMap)
|
|
@@ -126,6 +136,7 @@ const webWorkerApi = {
|
|
|
126
136
|
executionContext = createExecutionContext(circuitRunnerConfiguration, {
|
|
127
137
|
...opts,
|
|
128
138
|
platform: circuitRunnerConfiguration.platform,
|
|
139
|
+
debugNamespace,
|
|
129
140
|
})
|
|
130
141
|
bindEventListeners(executionContext.circuit)
|
|
131
142
|
executionContext.fsMap["entrypoint.tsx"] = code
|
|
@@ -141,6 +152,7 @@ const webWorkerApi = {
|
|
|
141
152
|
executionContext = createExecutionContext(circuitRunnerConfiguration, {
|
|
142
153
|
...opts,
|
|
143
154
|
platform: circuitRunnerConfiguration.platform,
|
|
155
|
+
debugNamespace,
|
|
144
156
|
})
|
|
145
157
|
bindEventListeners(executionContext.circuit)
|
|
146
158
|
;(globalThis as any).__tscircuit_circuit = executionContext.circuit
|
|
@@ -21,6 +21,7 @@ export function createExecutionContext(
|
|
|
21
21
|
opts: {
|
|
22
22
|
name?: string
|
|
23
23
|
platform?: PlatformConfig
|
|
24
|
+
debugNamespace?: string
|
|
24
25
|
} = {},
|
|
25
26
|
): ExecutionContext {
|
|
26
27
|
globalThis.React = React
|
|
@@ -33,6 +34,10 @@ export function createExecutionContext(
|
|
|
33
34
|
circuit.name = opts.name
|
|
34
35
|
}
|
|
35
36
|
|
|
37
|
+
if (opts.debugNamespace) {
|
|
38
|
+
;(circuit as any).enableDebug?.(opts.debugNamespace)
|
|
39
|
+
}
|
|
40
|
+
|
|
36
41
|
return {
|
|
37
42
|
fsMap: {},
|
|
38
43
|
entrypoint: "",
|
|
@@ -41,6 +46,7 @@ export function createExecutionContext(
|
|
|
41
46
|
tscircuit: tscircuitCore,
|
|
42
47
|
"@tscircuit/math-utils": tscircuitMathUtils,
|
|
43
48
|
react: React,
|
|
49
|
+
debug: Debug,
|
|
44
50
|
|
|
45
51
|
// This is usually used as a type import, we can remove the shim when we
|
|
46
52
|
// ignore type imports in getImportsFromCode
|