@tscircuit/eval 0.0.288 → 0.0.290
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/.github/workflows/bun-test.yml +20 -2
- package/dist/blob-url.js +1 -1
- package/dist/eval/index.d.ts +56 -8
- package/dist/eval/index.js +17 -1
- package/dist/lib/index.d.ts +57 -8
- package/dist/lib/index.js +66 -1
- package/dist/webworker/entrypoint.js +352 -349
- package/dist/worker.d.ts +1 -0
- package/dist/worker.js +50 -1
- package/lib/runner/CircuitRunner.ts +19 -0
- package/lib/shared/types.ts +2 -0
- package/lib/worker.ts +61 -0
- package/package.json +9 -7
- package/scripts/validate-test-matrix.js +148 -0
- package/tests/{example1-readme-example.test.tsx → examples/example01-readme-example.test.tsx} +1 -1
- package/tests/{example2-multiple-files.test.tsx → examples/example02-multiple-files.test.tsx} +1 -1
- package/tests/{example3-encoded-url.test.tsx → examples/example03-encoded-url.test.tsx} +1 -1
- package/tests/{example4-root-child-issue.test.tsx → examples/example04-root-child-issue.test.tsx} +1 -1
- package/tests/{example5-event-recording.test.tsx → examples/example05-event-recording.test.tsx} +1 -1
- package/tests/{example7-import-default-and-namespace.test.tsx → examples/example07-import-default-and-namespace.test.tsx} +2 -2
- package/tests/{example8-footprinter-to220.test.tsx → examples/example08-footprinter-to220.test.tsx} +1 -1
- package/tests/{example9-not-defined-component.test.tsx → examples/example09-not-defined-component.test.tsx} +1 -1
- package/tests/{example13-webworker-without-entrypoint.test.tsx → examples/example13-webworker-without-entrypoint.test.tsx} +1 -1
- package/tests/{example16-parts-engine.test.tsx → examples/example16-parts-engine.test.tsx} +1 -1
- package/tests/{example17-parse-tscircuit-config.test.tsx → examples/example17-parse-tscircuit-config.test.tsx} +1 -1
- package/tests/{circuit-event-forwarding.test.tsx → features/circuit-event-forwarding.test.tsx} +1 -1
- package/tests/features/execute-component-runner.test.tsx +47 -0
- package/tests/features/execute-component-worker.test.tsx +25 -0
- package/tests/{fetch-override.test.ts → features/fetch-proxy/fetch-override.test.ts} +3 -3
- package/tests/{fetch-proxy-validation.test.ts → features/fetch-proxy/fetch-proxy-validation.test.ts} +5 -7
- package/tests/{kill.test.ts → features/kill.test.ts} +5 -3
- package/tests/{manual-edits.test.tsx → features/manual-edits.test.tsx} +2 -2
- package/tests/fixtures/resourcePaths.ts +3 -0
- package/webworker/entrypoint.ts +58 -2
- /package/tests/{example6-dynamic-load-blob-url.test.tsx → examples/example06-dynamic-load-blob-url.test.tsx} +0 -0
- /package/tests/{example10-run-tscircuit-code.test.tsx → examples/example10-run-tscircuit-code.test.tsx} +0 -0
- /package/tests/{example11-flexible-import-extensions.test.tsx → examples/example11-flexible-import-extensions.test.tsx} +0 -0
- /package/tests/{example12-import-from-subdirectory.test.tsx → examples/example12-import-from-subdirectory.test.tsx} +0 -0
- /package/tests/{example14-run-tscircuit-module.test.tsx → examples/example14-run-tscircuit-module.test.tsx} +0 -0
- /package/tests/{example15-run-tscircuit-module-with-props.test.tsx → examples/example15-run-tscircuit-module-with-props.test.tsx} +0 -0
- /package/tests/{parent-directory-import.test.tsx → features/parent-directory-import.test.tsx} +0 -0
- /package/tests/{platform-config.test.tsx → features/platform-config.test.tsx} +0 -0
- /package/tests/{prioritize-default-export.test.tsx → features/prioritize-default-export.test.tsx} +0 -0
- /package/tests/{group-wrapper.test.tsx → repros/group-wrapper.test.tsx} +0 -0
- /package/tests/{nine-keyboard-default-export.test.tsx → repros/nine-keyboard-default-export.test.tsx} +0 -0
- /package/tests/{get-imports-from-code.test.tsx → util-fns/get-imports-from-code.test.tsx} +0 -0
- /package/tests/{getPossibleEntrypointComponentPaths.test.ts → util-fns/getPossibleEntrypointComponentPaths.test.ts} +0 -0
package/dist/worker.d.ts
CHANGED
|
@@ -27,6 +27,7 @@ interface WebWorkerConfiguration extends CircuitRunnerConfiguration {
|
|
|
27
27
|
}
|
|
28
28
|
type CircuitWebWorker = {
|
|
29
29
|
execute: (code: string) => Promise<void>;
|
|
30
|
+
executeComponent: (component: any) => Promise<void>;
|
|
30
31
|
executeWithFsMap: (opts: {
|
|
31
32
|
entrypoint?: string;
|
|
32
33
|
mainComponentPath?: string;
|
package/dist/worker.js
CHANGED
|
@@ -107,6 +107,38 @@ var createCircuitWebWorker = async (configuration) => {
|
|
|
107
107
|
}
|
|
108
108
|
const comlinkWorker = Comlink.wrap(rawWorker);
|
|
109
109
|
rawWorker.removeEventListener("message", earlyMessageHandler);
|
|
110
|
+
function serializeReactElement(element) {
|
|
111
|
+
if (!element || typeof element !== "object") {
|
|
112
|
+
return element;
|
|
113
|
+
}
|
|
114
|
+
if (element.type && element.props !== void 0) {
|
|
115
|
+
return {
|
|
116
|
+
__isSerializedReactElement: true,
|
|
117
|
+
type: element.type,
|
|
118
|
+
props: serializeProps(element.props),
|
|
119
|
+
key: element.key
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
return element;
|
|
123
|
+
}
|
|
124
|
+
function serializeProps(props) {
|
|
125
|
+
if (!props || typeof props !== "object") {
|
|
126
|
+
return props;
|
|
127
|
+
}
|
|
128
|
+
const serialized = {};
|
|
129
|
+
for (const [key, value] of Object.entries(props)) {
|
|
130
|
+
if (key === "children") {
|
|
131
|
+
if (Array.isArray(value)) {
|
|
132
|
+
serialized.children = value.map(serializeReactElement);
|
|
133
|
+
} else {
|
|
134
|
+
serialized.children = serializeReactElement(value);
|
|
135
|
+
}
|
|
136
|
+
} else {
|
|
137
|
+
serialized[key] = value;
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
return serialized;
|
|
141
|
+
}
|
|
110
142
|
if (configuration.enableFetchProxy) {
|
|
111
143
|
rawWorker.postMessage({ type: "override_global_fetch" });
|
|
112
144
|
}
|
|
@@ -126,6 +158,23 @@ var createCircuitWebWorker = async (configuration) => {
|
|
|
126
158
|
}
|
|
127
159
|
return comlinkWorker.execute.bind(comlinkWorker)(...args);
|
|
128
160
|
},
|
|
161
|
+
executeComponent: async (component) => {
|
|
162
|
+
if (isTerminated) {
|
|
163
|
+
throw new Error(
|
|
164
|
+
"CircuitWebWorker was terminated, can't executeComponent"
|
|
165
|
+
);
|
|
166
|
+
}
|
|
167
|
+
if (typeof component === "function") {
|
|
168
|
+
return comlinkWorker.executeComponent.bind(comlinkWorker)(component);
|
|
169
|
+
}
|
|
170
|
+
if (component && typeof component === "object" && component.type) {
|
|
171
|
+
const serializedElement = serializeReactElement(component);
|
|
172
|
+
return comlinkWorker.executeComponent.bind(comlinkWorker)(
|
|
173
|
+
serializedElement
|
|
174
|
+
);
|
|
175
|
+
}
|
|
176
|
+
return comlinkWorker.executeComponent.bind(comlinkWorker)(component);
|
|
177
|
+
},
|
|
129
178
|
executeWithFsMap: async (...args) => {
|
|
130
179
|
if (isTerminated) {
|
|
131
180
|
throw new Error(
|
|
@@ -157,4 +206,4 @@ export {
|
|
|
157
206
|
createCircuitWebWorker,
|
|
158
207
|
getImportsFromCode
|
|
159
208
|
};
|
|
160
|
-
//# 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  // 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    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": []
}

|
|
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": []
}

|
|
@@ -102,6 +102,25 @@ export class CircuitRunner implements CircuitRunnerApi {
|
|
|
102
102
|
await importEvalPath("./entrypoint.tsx", this._executionContext)
|
|
103
103
|
}
|
|
104
104
|
|
|
105
|
+
async executeComponent(component: any, opts: { name?: string } = {}) {
|
|
106
|
+
if (this._circuitRunnerConfiguration.verbose) {
|
|
107
|
+
console.log("[CircuitRunner] executeComponent called")
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
this._executionContext = createExecutionContext(
|
|
111
|
+
this._circuitRunnerConfiguration,
|
|
112
|
+
{
|
|
113
|
+
...opts,
|
|
114
|
+
platform: this._circuitRunnerConfiguration.platform,
|
|
115
|
+
},
|
|
116
|
+
)
|
|
117
|
+
this._bindEventListeners(this._executionContext.circuit)
|
|
118
|
+
;(globalThis as any).__tscircuit_circuit = this._executionContext.circuit
|
|
119
|
+
|
|
120
|
+
const element = typeof component === "function" ? component() : component
|
|
121
|
+
this._executionContext.circuit.add(element as any)
|
|
122
|
+
}
|
|
123
|
+
|
|
105
124
|
on(event: string, callback: (...args: any[]) => void) {
|
|
106
125
|
this._eventListeners[event] ??= []
|
|
107
126
|
this._eventListeners[event].push(callback)
|
package/lib/shared/types.ts
CHANGED
|
@@ -36,6 +36,7 @@ export interface CircuitRunnerApi {
|
|
|
36
36
|
name?: string
|
|
37
37
|
},
|
|
38
38
|
) => Promise<void>
|
|
39
|
+
executeComponent: (component: any) => Promise<void>
|
|
39
40
|
executeWithFsMap(opts: {
|
|
40
41
|
entrypoint?: string
|
|
41
42
|
fsMap: Record<string, string>
|
|
@@ -57,6 +58,7 @@ export type InternalWebWorkerApi = CircuitRunnerApi
|
|
|
57
58
|
|
|
58
59
|
export type CircuitWebWorker = {
|
|
59
60
|
execute: (code: string) => Promise<void>
|
|
61
|
+
executeComponent: (component: any) => Promise<void>
|
|
60
62
|
executeWithFsMap: (opts: {
|
|
61
63
|
entrypoint?: string
|
|
62
64
|
mainComponentPath?: string
|
package/lib/worker.ts
CHANGED
|
@@ -122,6 +122,45 @@ export const createCircuitWebWorker = async (
|
|
|
122
122
|
|
|
123
123
|
rawWorker.removeEventListener("message", earlyMessageHandler)
|
|
124
124
|
|
|
125
|
+
// Helper to serialize React elements for cross-worker communication
|
|
126
|
+
function serializeReactElement(element: any): any {
|
|
127
|
+
if (!element || typeof element !== "object") {
|
|
128
|
+
return element
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
if (element.type && element.props !== undefined) {
|
|
132
|
+
// This is a React element
|
|
133
|
+
return {
|
|
134
|
+
__isSerializedReactElement: true,
|
|
135
|
+
type: element.type,
|
|
136
|
+
props: serializeProps(element.props),
|
|
137
|
+
key: element.key,
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
return element
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
function serializeProps(props: any): any {
|
|
145
|
+
if (!props || typeof props !== "object") {
|
|
146
|
+
return props
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
const serialized: any = {}
|
|
150
|
+
for (const [key, value] of Object.entries(props)) {
|
|
151
|
+
if (key === "children") {
|
|
152
|
+
if (Array.isArray(value)) {
|
|
153
|
+
serialized.children = value.map(serializeReactElement)
|
|
154
|
+
} else {
|
|
155
|
+
serialized.children = serializeReactElement(value)
|
|
156
|
+
}
|
|
157
|
+
} else {
|
|
158
|
+
serialized[key] = value
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
return serialized
|
|
162
|
+
}
|
|
163
|
+
|
|
125
164
|
// Conditionally override global fetch inside the worker to route through the parent
|
|
126
165
|
// Only enable when explicitly requested via configuration
|
|
127
166
|
if (configuration.enableFetchProxy) {
|
|
@@ -147,6 +186,28 @@ export const createCircuitWebWorker = async (
|
|
|
147
186
|
}
|
|
148
187
|
return comlinkWorker.execute.bind(comlinkWorker)(...args)
|
|
149
188
|
},
|
|
189
|
+
executeComponent: async (component: any) => {
|
|
190
|
+
if (isTerminated) {
|
|
191
|
+
throw new Error(
|
|
192
|
+
"CircuitWebWorker was terminated, can't executeComponent",
|
|
193
|
+
)
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
// If it's a function, pass it as-is (will be proxied by Comlink)
|
|
197
|
+
if (typeof component === "function") {
|
|
198
|
+
return comlinkWorker.executeComponent.bind(comlinkWorker)(component)
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
// If it's a React element, serialize it to a reconstructable format
|
|
202
|
+
if (component && typeof component === "object" && component.type) {
|
|
203
|
+
const serializedElement = serializeReactElement(component)
|
|
204
|
+
return comlinkWorker.executeComponent.bind(comlinkWorker)(
|
|
205
|
+
serializedElement,
|
|
206
|
+
)
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
return comlinkWorker.executeComponent.bind(comlinkWorker)(component)
|
|
210
|
+
},
|
|
150
211
|
executeWithFsMap: async (...args) => {
|
|
151
212
|
if (isTerminated) {
|
|
152
213
|
throw new Error(
|
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.290",
|
|
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",
|
|
@@ -16,6 +16,7 @@
|
|
|
16
16
|
"format:check": "biome format .",
|
|
17
17
|
"test:playwright": "playwright test",
|
|
18
18
|
"test": "bun test tests",
|
|
19
|
+
"test:validate-matrix": "bun run scripts/validate-test-matrix.js",
|
|
19
20
|
"copy-core-versions": "bun run scripts/copy-core-versions.ts && bun install --ignore-scripts"
|
|
20
21
|
},
|
|
21
22
|
"exports": {
|
|
@@ -53,20 +54,20 @@
|
|
|
53
54
|
"@biomejs/biome": "^1.8.3",
|
|
54
55
|
"@playwright/test": "^1.50.1",
|
|
55
56
|
"@tscircuit/capacity-autorouter": "^0.0.100",
|
|
56
|
-
"@tscircuit/checks": "^0.0.
|
|
57
|
+
"@tscircuit/checks": "^0.0.71",
|
|
57
58
|
"@tscircuit/circuit-json-flex": "^0.0.3",
|
|
58
59
|
"@tscircuit/circuit-json-util": "^0.0.65",
|
|
59
|
-
"@tscircuit/core": "^0.0.
|
|
60
|
+
"@tscircuit/core": "^0.0.641",
|
|
60
61
|
"@tscircuit/footprinter": "^0.0.208",
|
|
61
62
|
"@tscircuit/import-snippet": "^0.0.4",
|
|
62
63
|
"@tscircuit/infgrid-ijump-astar": "^0.0.33",
|
|
63
64
|
"@tscircuit/layout": "^0.0.28",
|
|
64
65
|
"@tscircuit/log-soup": "^1.0.2",
|
|
65
|
-
"@tscircuit/matchpack": "^0.0.
|
|
66
|
+
"@tscircuit/matchpack": "^0.0.9",
|
|
66
67
|
"@tscircuit/math-utils": "^0.0.18",
|
|
67
68
|
"@tscircuit/miniflex": "^0.0.4",
|
|
68
69
|
"@tscircuit/parts-engine": "^0.0.8",
|
|
69
|
-
"@tscircuit/props": "^0.0.
|
|
70
|
+
"@tscircuit/props": "^0.0.287",
|
|
70
71
|
"@tscircuit/schematic-autolayout": "^0.0.6",
|
|
71
72
|
"@tscircuit/schematic-corpus": "^0.0.110",
|
|
72
73
|
"@tscircuit/schematic-match-adapt": "^0.0.16",
|
|
@@ -81,7 +82,7 @@
|
|
|
81
82
|
"bun-match-svg": "0.0.12",
|
|
82
83
|
"calculate-elbow": "^0.0.9",
|
|
83
84
|
"chokidar-cli": "^3.0.0",
|
|
84
|
-
"circuit-json": "^0.0.
|
|
85
|
+
"circuit-json": "^0.0.232",
|
|
85
86
|
"circuit-json-to-bpc": "^0.0.13",
|
|
86
87
|
"circuit-json-to-connectivity-map": "^0.0.22",
|
|
87
88
|
"circuit-json-to-simple-3d": "^0.0.6",
|
|
@@ -99,7 +100,8 @@
|
|
|
99
100
|
"react-dom": "^19.1.0",
|
|
100
101
|
"schematic-symbols": "^0.0.180",
|
|
101
102
|
"ts-expect": "^1.3.0",
|
|
102
|
-
"tsup": "^8.2.4"
|
|
103
|
+
"tsup": "^8.2.4",
|
|
104
|
+
"minicssgrid": "^0.0.8"
|
|
103
105
|
},
|
|
104
106
|
"peerDependencies": {
|
|
105
107
|
"typescript": "^5.0.0",
|
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
import { readdirSync, statSync, readFileSync } from "fs"
|
|
4
|
+
import { join } from "path"
|
|
5
|
+
import { fileURLToPath } from "url"
|
|
6
|
+
import { dirname } from "path"
|
|
7
|
+
|
|
8
|
+
const __filename = fileURLToPath(import.meta.url)
|
|
9
|
+
const __dirname = dirname(__filename)
|
|
10
|
+
|
|
11
|
+
const testsDir = join(__dirname, "..", "tests")
|
|
12
|
+
const workflowFile = join(
|
|
13
|
+
__dirname,
|
|
14
|
+
"..",
|
|
15
|
+
".github",
|
|
16
|
+
"workflows",
|
|
17
|
+
"bun-test.yml",
|
|
18
|
+
)
|
|
19
|
+
|
|
20
|
+
// Get all directories in tests/ that contain test files
|
|
21
|
+
function getTestDirectories() {
|
|
22
|
+
const entries = readdirSync(testsDir)
|
|
23
|
+
const testDirs = []
|
|
24
|
+
|
|
25
|
+
for (const entry of entries) {
|
|
26
|
+
const fullPath = join(testsDir, entry)
|
|
27
|
+
|
|
28
|
+
// Skip non-directories
|
|
29
|
+
if (!statSync(fullPath).isDirectory()) continue
|
|
30
|
+
|
|
31
|
+
// Skip hidden directories like .claude
|
|
32
|
+
if (entry.startsWith(".")) continue
|
|
33
|
+
|
|
34
|
+
// Check if directory contains test files
|
|
35
|
+
if (hasTestFiles(fullPath)) {
|
|
36
|
+
testDirs.push(entry)
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
return testDirs.sort()
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
// Check if a directory contains test files (recursively)
|
|
44
|
+
function hasTestFiles(dirPath) {
|
|
45
|
+
try {
|
|
46
|
+
const entries = readdirSync(dirPath)
|
|
47
|
+
|
|
48
|
+
for (const entry of entries) {
|
|
49
|
+
const fullPath = join(dirPath, entry)
|
|
50
|
+
const stat = statSync(fullPath)
|
|
51
|
+
|
|
52
|
+
if (
|
|
53
|
+
stat.isFile() &&
|
|
54
|
+
(entry.endsWith(".test.ts") || entry.endsWith(".test.tsx"))
|
|
55
|
+
) {
|
|
56
|
+
return true
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
if (stat.isDirectory() && hasTestFiles(fullPath)) {
|
|
60
|
+
return true
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
return false
|
|
65
|
+
} catch (error) {
|
|
66
|
+
console.warn(`Warning: Could not read directory ${dirPath}`)
|
|
67
|
+
return false
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
// Extract test directories from workflow file
|
|
72
|
+
function getWorkflowTestDirectories() {
|
|
73
|
+
const workflowContent = readFileSync(workflowFile, "utf8")
|
|
74
|
+
|
|
75
|
+
// Look for the matrix test-dir configuration
|
|
76
|
+
const matrixMatch = workflowContent.match(/test-dir:\s*\[(.*?)\]/s)
|
|
77
|
+
|
|
78
|
+
if (!matrixMatch) {
|
|
79
|
+
throw new Error(
|
|
80
|
+
"Could not find test-dir matrix configuration in workflow file",
|
|
81
|
+
)
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
const matrixContent = matrixMatch[1]
|
|
85
|
+
const dirs = matrixContent
|
|
86
|
+
.split(",")
|
|
87
|
+
.map((dir) => dir.trim().replace(/['"]/g, ""))
|
|
88
|
+
.filter((dir) => dir.length > 0)
|
|
89
|
+
.sort()
|
|
90
|
+
|
|
91
|
+
return dirs
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
function main() {
|
|
95
|
+
console.log("🔍 Validating test matrix configuration...")
|
|
96
|
+
|
|
97
|
+
const actualTestDirs = getTestDirectories()
|
|
98
|
+
const workflowTestDirs = getWorkflowTestDirectories()
|
|
99
|
+
|
|
100
|
+
console.log(
|
|
101
|
+
`Found ${actualTestDirs.length} test directories:`,
|
|
102
|
+
actualTestDirs,
|
|
103
|
+
)
|
|
104
|
+
console.log(
|
|
105
|
+
`Workflow covers ${workflowTestDirs.length} directories:`,
|
|
106
|
+
workflowTestDirs,
|
|
107
|
+
)
|
|
108
|
+
|
|
109
|
+
const missing = actualTestDirs.filter(
|
|
110
|
+
(dir) => !workflowTestDirs.includes(dir),
|
|
111
|
+
)
|
|
112
|
+
const extra = workflowTestDirs.filter((dir) => !actualTestDirs.includes(dir))
|
|
113
|
+
|
|
114
|
+
if (missing.length > 0) {
|
|
115
|
+
console.error(
|
|
116
|
+
"❌ ERROR: The following test directories are NOT covered by the workflow matrix:",
|
|
117
|
+
)
|
|
118
|
+
for (const dir of missing) {
|
|
119
|
+
console.error(` - ${dir}`)
|
|
120
|
+
}
|
|
121
|
+
console.error(
|
|
122
|
+
"\nPlease add these directories to the test-dir matrix in .github/workflows/bun-test.yml",
|
|
123
|
+
)
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
if (extra.length > 0) {
|
|
127
|
+
console.warn(
|
|
128
|
+
"⚠️ WARNING: The following directories in the workflow matrix do not contain test files:",
|
|
129
|
+
)
|
|
130
|
+
for (const dir of extra) {
|
|
131
|
+
console.warn(` - ${dir}`)
|
|
132
|
+
}
|
|
133
|
+
console.warn(
|
|
134
|
+
"\nConsider removing these from the test-dir matrix in .github/workflows/bun-test.yml",
|
|
135
|
+
)
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
if (missing.length === 0 && extra.length === 0) {
|
|
139
|
+
console.log(
|
|
140
|
+
"✅ All test directories are properly covered by the workflow matrix!",
|
|
141
|
+
)
|
|
142
|
+
process.exit(0)
|
|
143
|
+
} else {
|
|
144
|
+
process.exit(1)
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
main()
|
package/tests/{example1-readme-example.test.tsx → examples/example01-readme-example.test.tsx}
RENAMED
|
@@ -3,7 +3,7 @@ import { expect, test } from "bun:test"
|
|
|
3
3
|
|
|
4
4
|
test("example1-readme-example", async () => {
|
|
5
5
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
6
|
-
webWorkerUrl: new URL("
|
|
6
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
7
7
|
})
|
|
8
8
|
|
|
9
9
|
await circuitWebWorker.executeWithFsMap({
|
package/tests/{example2-multiple-files.test.tsx → examples/example02-multiple-files.test.tsx}
RENAMED
|
@@ -3,7 +3,7 @@ import { expect, test } from "bun:test"
|
|
|
3
3
|
|
|
4
4
|
test("virtual filesystem with components", async () => {
|
|
5
5
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
6
|
-
webWorkerUrl: new URL("
|
|
6
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
7
7
|
})
|
|
8
8
|
|
|
9
9
|
await circuitWebWorker.executeWithFsMap({
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { createCircuitWebWorker } from "lib"
|
|
2
2
|
import { expect, test } from "bun:test"
|
|
3
3
|
// @ts-ignore
|
|
4
|
-
import blobUrl from "
|
|
4
|
+
import blobUrl from "../../dist/blob-url"
|
|
5
5
|
|
|
6
6
|
test("example3-encoded-worker-url", async () => {
|
|
7
7
|
const circuitWebWorker = await createCircuitWebWorker({
|
package/tests/{example4-root-child-issue.test.tsx → examples/example04-root-child-issue.test.tsx}
RENAMED
|
@@ -15,7 +15,7 @@ test(
|
|
|
15
15
|
"example4-root-child-issue",
|
|
16
16
|
async () => {
|
|
17
17
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
18
|
-
webWorkerUrl: new URL("
|
|
18
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
19
19
|
})
|
|
20
20
|
|
|
21
21
|
await circuitWebWorker.executeWithFsMap({
|
package/tests/{example5-event-recording.test.tsx → examples/example05-event-recording.test.tsx}
RENAMED
|
@@ -4,7 +4,7 @@ import { createCircuitWebWorker } from "lib/index"
|
|
|
4
4
|
// Skipped for flakiness, re-enable when flakiness is solved
|
|
5
5
|
test.skip("example5-event-recording", async () => {
|
|
6
6
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
7
|
-
webWorkerUrl: new URL("
|
|
7
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
8
8
|
})
|
|
9
9
|
|
|
10
10
|
let eventCount = 0
|
|
@@ -3,7 +3,7 @@ import { expect, test } from "bun:test"
|
|
|
3
3
|
|
|
4
4
|
test("namespace import syntax", async () => {
|
|
5
5
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
6
|
-
webWorkerUrl: new URL("
|
|
6
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
7
7
|
})
|
|
8
8
|
|
|
9
9
|
await circuitWebWorker.executeWithFsMap({
|
|
@@ -39,7 +39,7 @@ test("namespace import syntax", async () => {
|
|
|
39
39
|
|
|
40
40
|
test("combined default and namespace import with fallback", async () => {
|
|
41
41
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
42
|
-
webWorkerUrl: new URL("
|
|
42
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
43
43
|
})
|
|
44
44
|
|
|
45
45
|
await circuitWebWorker.executeWithFsMap({
|
package/tests/{example8-footprinter-to220.test.tsx → examples/example08-footprinter-to220.test.tsx}
RENAMED
|
@@ -3,7 +3,7 @@ import { expect, test } from "bun:test"
|
|
|
3
3
|
|
|
4
4
|
test("example8-footprinter-to220", async () => {
|
|
5
5
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
6
|
-
webWorkerUrl: new URL("
|
|
6
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
7
7
|
})
|
|
8
8
|
|
|
9
9
|
await circuitWebWorker.execute(`
|
|
@@ -3,7 +3,7 @@ import { expect, test } from "bun:test"
|
|
|
3
3
|
|
|
4
4
|
test("example9-not-defined-component", async () => {
|
|
5
5
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
6
|
-
webWorkerUrl: new URL("
|
|
6
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
7
7
|
})
|
|
8
8
|
|
|
9
9
|
expect(async () => {
|
|
@@ -4,7 +4,7 @@ import type { SourceSimpleResistor } from "circuit-json"
|
|
|
4
4
|
|
|
5
5
|
test("example13-webworker-without-entrypoint", async () => {
|
|
6
6
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
7
|
-
webWorkerUrl: new URL("
|
|
7
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
8
8
|
})
|
|
9
9
|
|
|
10
10
|
try {
|
|
@@ -4,7 +4,7 @@ import type { SourceComponentBase } from "circuit-json"
|
|
|
4
4
|
|
|
5
5
|
test("example16-jlc-parts-engine with entrypoint", async () => {
|
|
6
6
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
7
|
-
webWorkerUrl: new URL("
|
|
7
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
8
8
|
verbose: true,
|
|
9
9
|
})
|
|
10
10
|
|
|
@@ -3,7 +3,7 @@ import { expect, test } from "bun:test"
|
|
|
3
3
|
|
|
4
4
|
test("parse tscircuit.config.js with mainEntrypoint", async () => {
|
|
5
5
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
6
|
-
webWorkerUrl: new URL("
|
|
6
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
7
7
|
})
|
|
8
8
|
|
|
9
9
|
await circuitWebWorker.executeWithFsMap({
|
package/tests/{circuit-event-forwarding.test.tsx → features/circuit-event-forwarding.test.tsx}
RENAMED
|
@@ -6,7 +6,7 @@ test("circuit-web-worker-events", async () => {
|
|
|
6
6
|
const capturedEvents: string[] = []
|
|
7
7
|
|
|
8
8
|
const circuitWebWorker = await createCircuitWebWorker({
|
|
9
|
-
webWorkerUrl: new URL("
|
|
9
|
+
webWorkerUrl: new URL("../../webworker/entrypoint.ts", import.meta.url),
|
|
10
10
|
})
|
|
11
11
|
|
|
12
12
|
await circuitWebWorker.execute(`
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import { expect, test } from "bun:test"
|
|
2
|
+
import * as React from "react"
|
|
3
|
+
import { CircuitRunner } from "lib/runner/CircuitRunner"
|
|
4
|
+
|
|
5
|
+
test("CircuitRunner.executeComponent with React element", async () => {
|
|
6
|
+
const runner = new CircuitRunner()
|
|
7
|
+
const element = React.createElement(
|
|
8
|
+
"board",
|
|
9
|
+
{ width: "10mm", height: "10mm" },
|
|
10
|
+
React.createElement("resistor", {
|
|
11
|
+
name: "R1",
|
|
12
|
+
resistance: "1k",
|
|
13
|
+
footprint: "0402",
|
|
14
|
+
}),
|
|
15
|
+
)
|
|
16
|
+
|
|
17
|
+
await runner.executeComponent(element)
|
|
18
|
+
await runner.renderUntilSettled()
|
|
19
|
+
const circuitJson = await runner.getCircuitJson()
|
|
20
|
+
const r1 = circuitJson.find(
|
|
21
|
+
(el: any) => el.type === "source_component" && el.name === "R1",
|
|
22
|
+
)
|
|
23
|
+
expect(r1).toBeDefined()
|
|
24
|
+
await runner.kill()
|
|
25
|
+
})
|
|
26
|
+
|
|
27
|
+
test("CircuitRunner.executeComponent with factory function", async () => {
|
|
28
|
+
const runner = new CircuitRunner()
|
|
29
|
+
await runner.executeComponent(() =>
|
|
30
|
+
React.createElement(
|
|
31
|
+
"board",
|
|
32
|
+
{ width: "10mm", height: "10mm" },
|
|
33
|
+
React.createElement("resistor", {
|
|
34
|
+
name: "R2",
|
|
35
|
+
resistance: "2k",
|
|
36
|
+
footprint: "0402",
|
|
37
|
+
}),
|
|
38
|
+
),
|
|
39
|
+
)
|
|
40
|
+
await runner.renderUntilSettled()
|
|
41
|
+
const circuitJson = await runner.getCircuitJson()
|
|
42
|
+
const r2 = circuitJson.find(
|
|
43
|
+
(el: any) => el.type === "source_component" && el.name === "R2",
|
|
44
|
+
)
|
|
45
|
+
expect(r2).toBeDefined()
|
|
46
|
+
await runner.kill()
|
|
47
|
+
})
|