@tscircuit/cli 0.1.15 → 0.1.17
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/cli/clone/register.ts
CHANGED
|
@@ -86,6 +86,9 @@ export const registerClone = (program: Command) => {
|
|
|
86
86
|
fs.writeFileSync(fullPath, fileContent.package_file.content_text)
|
|
87
87
|
}
|
|
88
88
|
|
|
89
|
+
const npmrcPath = path.join(dirPath, ".npmrc")
|
|
90
|
+
fs.writeFileSync(npmrcPath, "@tsci:registry=https://npm.tscircuit.com")
|
|
91
|
+
|
|
89
92
|
console.log(`Successfully cloned to ./${author}.${snippetName}/`)
|
|
90
93
|
} catch (error) {
|
|
91
94
|
if (error instanceof Error) {
|
package/cli/dev/DevServer.ts
CHANGED
|
@@ -8,6 +8,7 @@ import path from "node:path"
|
|
|
8
8
|
import fs from "node:fs"
|
|
9
9
|
import type { FileUpdatedEvent } from "lib/file-server/FileServerEvent"
|
|
10
10
|
import * as chokidar from "chokidar"
|
|
11
|
+
import { FilesystemTypesHandler } from "lib/dependency-analysis/FilesystemTypesHandler"
|
|
11
12
|
|
|
12
13
|
export class DevServer {
|
|
13
14
|
port: number
|
|
@@ -37,6 +38,8 @@ export class DevServer {
|
|
|
37
38
|
*/
|
|
38
39
|
filesystemWatcher?: chokidar.FSWatcher
|
|
39
40
|
|
|
41
|
+
private typesHandler?: FilesystemTypesHandler
|
|
42
|
+
|
|
40
43
|
constructor({
|
|
41
44
|
port,
|
|
42
45
|
componentFilePath,
|
|
@@ -50,6 +53,7 @@ export class DevServer {
|
|
|
50
53
|
this.fsKy = ky.create({
|
|
51
54
|
prefixUrl: `http://localhost:${port}`,
|
|
52
55
|
}) as any
|
|
56
|
+
this.typesHandler = new FilesystemTypesHandler(this.projectDir)
|
|
53
57
|
}
|
|
54
58
|
|
|
55
59
|
async start() {
|
|
@@ -77,6 +81,8 @@ export class DevServer {
|
|
|
77
81
|
)
|
|
78
82
|
|
|
79
83
|
this.upsertInitialFiles()
|
|
84
|
+
|
|
85
|
+
this.typesHandler?.handleInitialTypeDependencies(this.componentFilePath)
|
|
80
86
|
}
|
|
81
87
|
|
|
82
88
|
async addEntrypoint() {
|
|
@@ -119,6 +125,8 @@ circuit.add(<MyCircuit />)
|
|
|
119
125
|
// because it can be edited by the browser
|
|
120
126
|
if (relativeFilePath.includes("manual-edits.json")) return
|
|
121
127
|
|
|
128
|
+
await this.typesHandler?.handleFileTypeDependencies(absoluteFilePath)
|
|
129
|
+
|
|
122
130
|
console.log(`${relativeFilePath} saved. Applying changes...`)
|
|
123
131
|
await this.fsKy
|
|
124
132
|
.post("api/files/upsert", {
|
package/dist/main.js
CHANGED
|
@@ -54,8 +54,8 @@ export default () => (
|
|
|
54
54
|
};
|
|
55
55
|
|
|
56
56
|
// cli/dev/register.ts
|
|
57
|
-
import * as
|
|
58
|
-
import * as
|
|
57
|
+
import * as path6 from "node:path";
|
|
58
|
+
import * as fs7 from "node:fs";
|
|
59
59
|
|
|
60
60
|
// lib/dependency-analysis/installNodeModuleTypesForSnippet.ts
|
|
61
61
|
import * as fs2 from "node:fs";
|
|
@@ -131,7 +131,7 @@ var package_default = {
|
|
|
131
131
|
name: "@tscircuit/cli",
|
|
132
132
|
main: "dist/main.js",
|
|
133
133
|
type: "module",
|
|
134
|
-
version: "0.1.
|
|
134
|
+
version: "0.1.16",
|
|
135
135
|
bin: {
|
|
136
136
|
tsci: "./dist/main.js"
|
|
137
137
|
},
|
|
@@ -233,10 +233,10 @@ var createHttpServer = async (port = 3e3) => {
|
|
|
233
233
|
res.writeHead(404);
|
|
234
234
|
res.end("Not found");
|
|
235
235
|
});
|
|
236
|
-
return new Promise((
|
|
236
|
+
return new Promise((resolve4) => {
|
|
237
237
|
server.listen(port, () => {
|
|
238
238
|
console.log(`Server running at http://localhost:${port}`);
|
|
239
|
-
|
|
239
|
+
resolve4({ server });
|
|
240
240
|
});
|
|
241
241
|
});
|
|
242
242
|
};
|
|
@@ -293,9 +293,99 @@ var EventsWatcher = class extends EventEmitter {
|
|
|
293
293
|
};
|
|
294
294
|
|
|
295
295
|
// cli/dev/DevServer.ts
|
|
296
|
-
import
|
|
297
|
-
import
|
|
296
|
+
import path5 from "node:path";
|
|
297
|
+
import fs6 from "node:fs";
|
|
298
298
|
import * as chokidar from "chokidar";
|
|
299
|
+
|
|
300
|
+
// lib/dependency-analysis/FilesystemTypesHandler.ts
|
|
301
|
+
import * as fs5 from "node:fs";
|
|
302
|
+
import * as path4 from "node:path";
|
|
303
|
+
|
|
304
|
+
// lib/dependency-analysis/findImportsInSnippet.ts
|
|
305
|
+
import * as fs4 from "node:fs";
|
|
306
|
+
import * as ts2 from "typescript";
|
|
307
|
+
function findImportsInSnippet(snippetPath) {
|
|
308
|
+
const content = fs4.readFileSync(snippetPath, "utf-8");
|
|
309
|
+
const sourceFile = ts2.createSourceFile(
|
|
310
|
+
snippetPath,
|
|
311
|
+
content,
|
|
312
|
+
ts2.ScriptTarget.Latest,
|
|
313
|
+
true
|
|
314
|
+
);
|
|
315
|
+
const imports = [];
|
|
316
|
+
function visit(node) {
|
|
317
|
+
if (ts2.isImportDeclaration(node)) {
|
|
318
|
+
const moduleSpecifier = node.moduleSpecifier;
|
|
319
|
+
if (moduleSpecifier && ts2.isStringLiteral(moduleSpecifier)) {
|
|
320
|
+
const importPath = moduleSpecifier.text;
|
|
321
|
+
if (importPath.startsWith("@tsci/")) {
|
|
322
|
+
imports.push(importPath);
|
|
323
|
+
}
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
ts2.forEachChild(node, visit);
|
|
327
|
+
}
|
|
328
|
+
visit(sourceFile);
|
|
329
|
+
return imports;
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
// lib/dependency-analysis/FilesystemTypesHandler.ts
|
|
333
|
+
var FilesystemTypesHandler = class {
|
|
334
|
+
projectRoot;
|
|
335
|
+
constructor(initialDir) {
|
|
336
|
+
this.projectRoot = this.findProjectRoot(initialDir);
|
|
337
|
+
}
|
|
338
|
+
async handleInitialTypeDependencies(filePath) {
|
|
339
|
+
console.log("Checking initial type dependencies...");
|
|
340
|
+
try {
|
|
341
|
+
if (!this.areTypesInstalled(filePath)) {
|
|
342
|
+
console.log("Installing missing initial types...");
|
|
343
|
+
await installNodeModuleTypesForSnippet(filePath);
|
|
344
|
+
}
|
|
345
|
+
} catch (error) {
|
|
346
|
+
console.warn("Error handling initial type dependencies:", error);
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
async handleFileTypeDependencies(filePath) {
|
|
350
|
+
try {
|
|
351
|
+
if (!this.areTypesInstalled(filePath)) {
|
|
352
|
+
console.log("Installing missing file types...");
|
|
353
|
+
await installNodeModuleTypesForSnippet(filePath);
|
|
354
|
+
}
|
|
355
|
+
} catch (error) {
|
|
356
|
+
console.warn("Failed to verify types:", error);
|
|
357
|
+
}
|
|
358
|
+
}
|
|
359
|
+
areTypesInstalled(filePath) {
|
|
360
|
+
const imports = findImportsInSnippet(filePath);
|
|
361
|
+
return imports.every((imp) => this.checkTypeExists(imp));
|
|
362
|
+
}
|
|
363
|
+
checkTypeExists(importPath) {
|
|
364
|
+
if (!importPath.startsWith("@tsci/")) return true;
|
|
365
|
+
const pathWithoutPrefix = importPath.replace("@tsci/", "");
|
|
366
|
+
const [owner, name] = pathWithoutPrefix.split(".");
|
|
367
|
+
const typePath = path4.join(
|
|
368
|
+
this.projectRoot,
|
|
369
|
+
"node_modules",
|
|
370
|
+
"@tsci",
|
|
371
|
+
`${owner}.${name}`,
|
|
372
|
+
"index.d.ts"
|
|
373
|
+
);
|
|
374
|
+
return fs5.existsSync(typePath);
|
|
375
|
+
}
|
|
376
|
+
findProjectRoot(startDir) {
|
|
377
|
+
let root = path4.resolve(startDir);
|
|
378
|
+
while (root !== path4.parse(root).root) {
|
|
379
|
+
if (fs5.existsSync(path4.join(root, "package.json"))) {
|
|
380
|
+
return root;
|
|
381
|
+
}
|
|
382
|
+
root = path4.dirname(root);
|
|
383
|
+
}
|
|
384
|
+
return startDir;
|
|
385
|
+
}
|
|
386
|
+
};
|
|
387
|
+
|
|
388
|
+
// cli/dev/DevServer.ts
|
|
299
389
|
var DevServer = class {
|
|
300
390
|
port;
|
|
301
391
|
/**
|
|
@@ -321,16 +411,18 @@ var DevServer = class {
|
|
|
321
411
|
* A chokidar instance that watches the project directory for file changes
|
|
322
412
|
*/
|
|
323
413
|
filesystemWatcher;
|
|
414
|
+
typesHandler;
|
|
324
415
|
constructor({
|
|
325
416
|
port,
|
|
326
417
|
componentFilePath
|
|
327
418
|
}) {
|
|
328
419
|
this.port = port;
|
|
329
420
|
this.componentFilePath = componentFilePath;
|
|
330
|
-
this.projectDir =
|
|
421
|
+
this.projectDir = path5.dirname(componentFilePath);
|
|
331
422
|
this.fsKy = ky.create({
|
|
332
423
|
prefixUrl: `http://localhost:${port}`
|
|
333
424
|
});
|
|
425
|
+
this.typesHandler = new FilesystemTypesHandler(this.projectDir);
|
|
334
426
|
}
|
|
335
427
|
async start() {
|
|
336
428
|
const { server } = await createHttpServer(this.port);
|
|
@@ -354,9 +446,10 @@ var DevServer = class {
|
|
|
354
446
|
(filePath) => this.handleFileChangedOnFilesystem(filePath)
|
|
355
447
|
);
|
|
356
448
|
this.upsertInitialFiles();
|
|
449
|
+
this.typesHandler?.handleInitialTypeDependencies(this.componentFilePath);
|
|
357
450
|
}
|
|
358
451
|
async addEntrypoint() {
|
|
359
|
-
const relativeComponentFilePath =
|
|
452
|
+
const relativeComponentFilePath = path5.relative(
|
|
360
453
|
this.projectDir,
|
|
361
454
|
this.componentFilePath
|
|
362
455
|
);
|
|
@@ -378,31 +471,32 @@ circuit.add(<MyCircuit />)
|
|
|
378
471
|
const { file } = await this.fsKy.get("api/files/get", {
|
|
379
472
|
searchParams: { file_path: ev.file_path }
|
|
380
473
|
}).json();
|
|
381
|
-
|
|
382
|
-
|
|
474
|
+
fs6.writeFileSync(
|
|
475
|
+
path5.join(this.projectDir, "manual-edits.json"),
|
|
383
476
|
file.text_content
|
|
384
477
|
);
|
|
385
478
|
}
|
|
386
479
|
}
|
|
387
480
|
async handleFileChangedOnFilesystem(absoluteFilePath) {
|
|
388
|
-
const relativeFilePath =
|
|
481
|
+
const relativeFilePath = path5.relative(this.projectDir, absoluteFilePath);
|
|
389
482
|
if (relativeFilePath.includes("manual-edits.json")) return;
|
|
483
|
+
await this.typesHandler?.handleFileTypeDependencies(absoluteFilePath);
|
|
390
484
|
console.log(`${relativeFilePath} saved. Applying changes...`);
|
|
391
485
|
await this.fsKy.post("api/files/upsert", {
|
|
392
486
|
json: {
|
|
393
487
|
file_path: relativeFilePath,
|
|
394
|
-
text_content:
|
|
488
|
+
text_content: fs6.readFileSync(absoluteFilePath, "utf-8"),
|
|
395
489
|
initiator: "filesystem_change"
|
|
396
490
|
}
|
|
397
491
|
}).json();
|
|
398
492
|
}
|
|
399
493
|
async upsertInitialFiles() {
|
|
400
|
-
const fileNames =
|
|
494
|
+
const fileNames = fs6.readdirSync(this.projectDir);
|
|
401
495
|
for (const fileName of fileNames) {
|
|
402
|
-
if (
|
|
496
|
+
if (fs6.statSync(path5.join(this.projectDir, fileName)).isDirectory())
|
|
403
497
|
continue;
|
|
404
|
-
const fileContent =
|
|
405
|
-
|
|
498
|
+
const fileContent = fs6.readFileSync(
|
|
499
|
+
path5.join(this.projectDir, fileName),
|
|
406
500
|
"utf-8"
|
|
407
501
|
);
|
|
408
502
|
await this.fsKy.post("api/files/upsert", {
|
|
@@ -426,10 +520,10 @@ var registerDev = (program2) => {
|
|
|
426
520
|
const port = parseInt(options.port);
|
|
427
521
|
let absolutePath;
|
|
428
522
|
if (file) {
|
|
429
|
-
absolutePath =
|
|
523
|
+
absolutePath = path6.resolve(file);
|
|
430
524
|
} else {
|
|
431
|
-
const entrypointPath =
|
|
432
|
-
if (
|
|
525
|
+
const entrypointPath = path6.resolve("index.tsx");
|
|
526
|
+
if (fs7.existsSync(entrypointPath)) {
|
|
433
527
|
absolutePath = entrypointPath;
|
|
434
528
|
console.log("No file provided. Using 'index.tsx' as the entrypoint.");
|
|
435
529
|
} else {
|
|
@@ -439,7 +533,7 @@ var registerDev = (program2) => {
|
|
|
439
533
|
return;
|
|
440
534
|
}
|
|
441
535
|
}
|
|
442
|
-
const fileDir =
|
|
536
|
+
const fileDir = path6.dirname(absolutePath);
|
|
443
537
|
try {
|
|
444
538
|
console.log("Installing types for imported snippets...");
|
|
445
539
|
await installNodeModuleTypesForSnippet(absolutePath);
|
|
@@ -568,8 +662,8 @@ var registerConfigPrint = (program2) => {
|
|
|
568
662
|
};
|
|
569
663
|
|
|
570
664
|
// cli/clone/register.ts
|
|
571
|
-
import * as
|
|
572
|
-
import * as
|
|
665
|
+
import * as fs8 from "node:fs";
|
|
666
|
+
import * as path7 from "node:path";
|
|
573
667
|
var registerClone = (program2) => {
|
|
574
668
|
program2.command("clone").description("Clone a snippet from the registry").argument("<snippet>", "Snippet to clone (e.g. author/snippetName)").action(async (snippetPath) => {
|
|
575
669
|
let author;
|
|
@@ -599,8 +693,8 @@ var registerClone = (program2) => {
|
|
|
599
693
|
}
|
|
600
694
|
}).json();
|
|
601
695
|
const dirPath = `./${author}.${snippetName}`;
|
|
602
|
-
if (!
|
|
603
|
-
|
|
696
|
+
if (!fs8.existsSync(dirPath)) {
|
|
697
|
+
fs8.mkdirSync(dirPath);
|
|
604
698
|
}
|
|
605
699
|
for (const fileInfo of packageFileList.package_files) {
|
|
606
700
|
const filePath = fileInfo.file_path.startsWith("/") ? fileInfo.file_path.slice(1) : fileInfo.file_path;
|
|
@@ -611,13 +705,15 @@ var registerClone = (program2) => {
|
|
|
611
705
|
file_path: fileInfo.file_path
|
|
612
706
|
}
|
|
613
707
|
}).json();
|
|
614
|
-
const fullPath =
|
|
615
|
-
const dirName =
|
|
616
|
-
if (!
|
|
617
|
-
|
|
708
|
+
const fullPath = path7.join(dirPath, filePath);
|
|
709
|
+
const dirName = path7.dirname(fullPath);
|
|
710
|
+
if (!fs8.existsSync(dirName)) {
|
|
711
|
+
fs8.mkdirSync(dirName, { recursive: true });
|
|
618
712
|
}
|
|
619
|
-
|
|
713
|
+
fs8.writeFileSync(fullPath, fileContent.package_file.content_text);
|
|
620
714
|
}
|
|
715
|
+
const npmrcPath = path7.join(dirPath, ".npmrc");
|
|
716
|
+
fs8.writeFileSync(npmrcPath, "@tsci:registry=https://npm.tscircuit.com");
|
|
621
717
|
console.log(`Successfully cloned to ./${author}.${snippetName}/`);
|
|
622
718
|
} catch (error) {
|
|
623
719
|
if (error instanceof Error) {
|
|
@@ -638,8 +734,8 @@ import semver from "semver";
|
|
|
638
734
|
import { createCircuitWebWorker } from "@tscircuit/eval-webworker";
|
|
639
735
|
import webWorkerBundleUrl from "@tscircuit/eval-webworker/blob-url";
|
|
640
736
|
import { getVirtualFileSystemFromDirPath } from "make-vfs";
|
|
641
|
-
import
|
|
642
|
-
import
|
|
737
|
+
import path8 from "node:path";
|
|
738
|
+
import fs9 from "node:fs";
|
|
643
739
|
var ALLOWED_FORMATS = [
|
|
644
740
|
"json",
|
|
645
741
|
"circuit-json",
|
|
@@ -671,13 +767,13 @@ Supported formats: ${ALLOWED_FORMATS.join(",")}`
|
|
|
671
767
|
);
|
|
672
768
|
}
|
|
673
769
|
if (!output) {
|
|
674
|
-
output =
|
|
770
|
+
output = path8.basename(file).replace(/\.[^.]+$/, "");
|
|
675
771
|
}
|
|
676
772
|
const worker = await createCircuitWebWorker({
|
|
677
773
|
webWorkerUrl: webWorkerBundleUrl
|
|
678
774
|
});
|
|
679
|
-
const projectDir =
|
|
680
|
-
const relativeComponentPath =
|
|
775
|
+
const projectDir = path8.dirname(file);
|
|
776
|
+
const relativeComponentPath = path8.relative(projectDir, file);
|
|
681
777
|
await worker.executeWithFsMap({
|
|
682
778
|
entrypoint: "entrypoint.tsx",
|
|
683
779
|
fsMap: {
|
|
@@ -694,11 +790,11 @@ circuit.add(<MyCircuit />)
|
|
|
694
790
|
});
|
|
695
791
|
await worker.renderUntilSettled();
|
|
696
792
|
const circuitJson = await worker.getCircuitJson();
|
|
697
|
-
const outputPath =
|
|
793
|
+
const outputPath = path8.join(
|
|
698
794
|
projectDir,
|
|
699
795
|
`${output}${OUTPUT_EXTENSIONS[format]}`
|
|
700
796
|
);
|
|
701
|
-
|
|
797
|
+
fs9.writeFileSync(outputPath, JSON.stringify(circuitJson));
|
|
702
798
|
console.log(`Exported to ${outputPath}`);
|
|
703
799
|
process.exit(0);
|
|
704
800
|
});
|
|
@@ -750,4 +846,4 @@ if (process.argv.length === 2) {
|
|
|
750
846
|
} else {
|
|
751
847
|
program.parse();
|
|
752
848
|
}
|
|
753
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../cli/main.ts", "../cli/init/register.ts", "../cli/dev/register.ts", "../lib/dependency-analysis/installNodeModuleTypesForSnippet.ts", "../cli/dev/DevServer.ts", "../lib/server/createHttpServer.ts", "../package.json", "../lib/site/getIndex.ts", "../lib/server/EventsWatcher.ts", "../lib/cli-config/index.ts", "../cli/auth/login/register.ts", "../lib/registry-api/get-ky.ts", "../cli/auth/logout/register.ts", "../cli/auth/register.ts", "../cli/config/register.ts", "../cli/config/print/register.ts", "../cli/clone/register.ts", "../cli/export/register.ts", "../cli/auth/print-token/register.ts", "../cli/auth/set-token/register.ts"],
  "sourcesContent": ["#!/usr/bin/env node\nimport { Command } from \"commander\"\nimport { registerInit } from \"./init/register\"\nimport { registerDev } from \"./dev/register\"\nimport { registerAuthLogin } from \"./auth/login/register\"\nimport { registerAuthLogout } from \"./auth/logout/register\"\nimport { registerAuth } from \"./auth/register\"\nimport { registerConfig } from \"./config/register\"\nimport { registerConfigPrint } from \"./config/print/register\"\nimport { registerClone } from \"./clone/register\"\nimport { perfectCli } from \"perfect-cli\"\nimport pkg from \"../package.json\"\nimport semver from \"semver\"\nimport { registerExport } from \"./export/register\"\nimport { registerAuthPrintToken } from \"./auth/print-token/register\"\nimport { registerAuthSetToken } from \"./auth/set-token/register\"\n\nconst program = new Command()\n\nprogram\n  .name(\"tsci\")\n  .description(\"CLI for developing tscircuit snippets\")\n  // HACK: at build time the version is old, we need to\n  // fix this at some point...\n  .version(semver.inc(pkg.version, \"patch\") ?? pkg.version)\n\nregisterInit(program)\n\nregisterDev(program)\nregisterClone(program)\n\nregisterAuth(program)\nregisterAuthLogin(program)\nregisterAuthLogout(program)\nregisterAuthPrintToken(program)\nregisterAuthSetToken(program)\n\nregisterConfig(program)\nregisterConfigPrint(program)\n\nregisterExport(program)\n\nif (process.argv.length === 2) {\n  perfectCli(program, process.argv)\n} else {\n  program.parse()\n}\n", "import type { Command } from \"commander\"\nimport * as fs from \"node:fs\"\nimport * as path from \"node:path\"\n\nexport const registerInit = (program: Command) => {\n  program\n    .command(\"init\")\n    .description(\"Initialize a new TSCircuit project in the current directory\")\n    .action(() => {\n      const currentDir = process.cwd()\n\n      const indexFilePath = path.join(currentDir, \"index.tsx\")\n      const npmrcFilePath = path.join(currentDir, \".npmrc\")\n\n      const indexContent = `\nexport default () => (\n  <board width=\"10mm\" height=\"10mm\">\n    <resistor\n      resistance=\"1k\"\n      footprint=\"0402\"\n      name=\"R1\"\n      schX={3}\n      pcbX={3}\n    />\n    <capacitor\n      capacitance=\"1000pF\"\n      footprint=\"0402\"\n      name=\"C1\"\n      schX={-3}\n      pcbX={-3}\n    />\n    <trace from=\".R1 > .pin1\" to=\".C1 > .pin1\" />\n  </board>\n);\n`\n\n      const npmrcContent = `\n@tsci:registry=https://npm.tscircuit.com\n`\n\n      if (!fs.existsSync(indexFilePath)) {\n        fs.writeFileSync(indexFilePath, indexContent.trimStart())\n        console.log(`Created: ${indexFilePath}`)\n      } else {\n        console.log(`Skipped: ${indexFilePath} already exists`)\n      }\n\n      if (!fs.existsSync(npmrcFilePath)) {\n        fs.writeFileSync(npmrcFilePath, npmrcContent.trimStart())\n        console.log(`Created: ${npmrcFilePath}`)\n      } else {\n        console.log(`Skipped: ${npmrcFilePath} already exists`)\n      }\n\n      console.log(\n        `Initialization complete. Run \"tsci dev\" to start developing.`,\n      )\n    })\n}\n", "import type { Command } from \"commander\"\nimport * as path from \"node:path\"\nimport * as chokidar from \"chokidar\"\nimport * as fs from \"node:fs\"\nimport { createHttpServer } from \"lib/server/createHttpServer\"\nimport { getLocalFileDependencies } from \"lib/dependency-analysis/getLocalFileDependencies\"\nimport { installNodeModuleTypesForSnippet } from \"../../lib/dependency-analysis/installNodeModuleTypesForSnippet\"\nimport { EventsWatcher } from \"../../lib/server/EventsWatcher\"\nimport { DevServer } from \"./DevServer\"\n\nexport const registerDev = (program: Command) => {\n  program\n    .command(\"dev\")\n    .description(\"Start development server for a snippet\")\n    .argument(\"[file]\", \"Path to the snippet file\")\n    .option(\"-p, --port <number>\", \"Port to run server on\", \"3000\")\n    .action(async (file: string, options: { port: string }) => {\n      const port = parseInt(options.port)\n      let absolutePath: string\n\n      if (file) {\n        absolutePath = path.resolve(file)\n      } else {\n        const entrypointPath = path.resolve(\"index.tsx\")\n        if (fs.existsSync(entrypointPath)) {\n          absolutePath = entrypointPath\n          console.log(\"No file provided. Using 'index.tsx' as the entrypoint.\")\n        } else {\n          console.log(\n            \"No entrypoint found. Run 'tsci init' to bootstrap a basic project.\",\n          )\n          return\n        }\n      }\n\n      const fileDir = path.dirname(absolutePath)\n\n      try {\n        console.log(\"Installing types for imported snippets...\")\n        await installNodeModuleTypesForSnippet(absolutePath)\n        console.log(\"Types installed successfully\")\n      } catch (error) {\n        console.warn(\"Failed to install types:\", error)\n      }\n\n      const server = new DevServer({\n        port,\n        componentFilePath: absolutePath,\n      })\n\n      await server.start()\n      await server.addEntrypoint()\n    })\n}\n", "import * as fs from \"node:fs\"\nimport * as path from \"node:path\"\nimport * as ts from \"typescript\"\n\ninterface SnippetApiResponse {\n  snippet: {\n    dts: string\n  }\n}\n\nexport async function installNodeModuleTypesForSnippet(snippetPath: string) {\n  const content = fs.readFileSync(snippetPath, \"utf-8\")\n  const sourceFile = ts.createSourceFile(\n    snippetPath,\n    content,\n    ts.ScriptTarget.Latest,\n    true,\n  )\n\n  const imports: string[] = []\n\n  function visit(node: ts.Node) {\n    if (ts.isImportDeclaration(node)) {\n      const moduleSpecifier = node.moduleSpecifier\n      if (moduleSpecifier && ts.isStringLiteral(moduleSpecifier)) {\n        const importPath = moduleSpecifier.text\n        if (importPath.startsWith(\"@tsci/\")) {\n          imports.push(importPath)\n        }\n      }\n    }\n    ts.forEachChild(node, visit)\n  }\n\n  visit(sourceFile)\n\n  let projectRoot = path.dirname(snippetPath)\n  while (projectRoot !== path.parse(projectRoot).root) {\n    if (fs.existsSync(path.join(projectRoot, \"package.json\"))) {\n      break\n    }\n    projectRoot = path.dirname(projectRoot)\n  }\n\n  for (const importPath of imports) {\n    const [owner, name] = importPath.replace(\"@tsci/\", \"\").split(\".\")\n    try {\n      const response = await fetch(\n        `https://registry-api.tscircuit.com/snippets/get?owner_name=${owner}&unscoped_name=${name}`,\n      )\n\n      if (!response.ok) {\n        console.warn(`Failed to fetch types for ${importPath}`)\n        continue\n      }\n\n      const data: SnippetApiResponse = await response.json()\n\n      if (data.snippet.dts) {\n        const packageDir = path.join(\n          projectRoot,\n          \"node_modules\",\n          \"@tsci\",\n          `${owner}.${name}`,\n        )\n        fs.mkdirSync(packageDir, { recursive: true })\n\n        fs.writeFileSync(path.join(packageDir, \"index.d.ts\"), data.snippet.dts)\n      }\n    } catch (error) {\n      console.warn(`Error fetching types for ${importPath}:`, error)\n    }\n  }\n}\n", "import ky from \"ky\"\nimport type { FileServerRoutes } from \"lib/file-server/FileServerRoutes\"\nimport { createHttpServer } from \"lib/server/createHttpServer\"\nimport { EventsWatcher } from \"lib/server/EventsWatcher\"\nimport type http from \"node:http\"\nimport type { TypedKyInstance } from \"typed-ky\"\nimport path from \"node:path\"\nimport fs from \"node:fs\"\nimport type { FileUpdatedEvent } from \"lib/file-server/FileServerEvent\"\nimport * as chokidar from \"chokidar\"\n\nexport class DevServer {\n  port: number\n  /**\n   * The path to a component that exports a <board /> or <group /> component\n   */\n  componentFilePath: string\n\n  projectDir: string\n\n  /**\n   * The HTTP server that hosts the file server and event bus. You can use\n   * fsKy to communicate with the file server/event bus\n   */\n  httpServer?: http.Server\n  /**\n   * Watches for events on the event bus by polling `api/events/list`\n   */\n  eventsWatcher?: EventsWatcher\n  /**\n   * A ky instance that can be used to communicate with the file server and\n   * event bus\n   */\n  fsKy: TypedKyInstance<keyof FileServerRoutes, FileServerRoutes>\n  /**\n   * A chokidar instance that watches the project directory for file changes\n   */\n  filesystemWatcher?: chokidar.FSWatcher\n\n  constructor({\n    port,\n    componentFilePath,\n  }: {\n    port: number\n    componentFilePath: string\n  }) {\n    this.port = port\n    this.componentFilePath = componentFilePath\n    this.projectDir = path.dirname(componentFilePath)\n    this.fsKy = ky.create({\n      prefixUrl: `http://localhost:${port}`,\n    }) as any\n  }\n\n  async start() {\n    const { server } = await createHttpServer(this.port)\n    this.httpServer = server\n\n    this.eventsWatcher = new EventsWatcher(`http://localhost:${this.port}`)\n    this.eventsWatcher.start()\n\n    this.eventsWatcher.on(\n      \"FILE_UPDATED\",\n      this.handleFileUpdatedEventFromServer.bind(this),\n    )\n\n    this.filesystemWatcher = chokidar.watch(this.projectDir, {\n      persistent: true,\n      ignoreInitial: true,\n    })\n\n    this.filesystemWatcher.on(\"change\", (filePath) =>\n      this.handleFileChangedOnFilesystem(filePath),\n    )\n    this.filesystemWatcher.on(\"add\", (filePath) =>\n      this.handleFileChangedOnFilesystem(filePath),\n    )\n\n    this.upsertInitialFiles()\n  }\n\n  async addEntrypoint() {\n    const relativeComponentFilePath = path.relative(\n      this.projectDir,\n      this.componentFilePath,\n    )\n    await this.fsKy.post(\"api/files/upsert\", {\n      json: {\n        file_path: \"entrypoint.tsx\",\n        text_content: `\nimport MyCircuit from \"./${relativeComponentFilePath}\"\n\ncircuit.add(<MyCircuit />)\n`,\n      },\n    })\n  }\n\n  async handleFileUpdatedEventFromServer(ev: FileUpdatedEvent) {\n    if (ev.initiator === \"filesystem_change\") return\n\n    if (ev.file_path === \"manual-edits.json\") {\n      console.log(\"Manual edits updated, updating on filesystem...\")\n      const { file } = await this.fsKy\n        .get(\"api/files/get\", {\n          searchParams: { file_path: ev.file_path },\n        })\n        .json()\n      fs.writeFileSync(\n        path.join(this.projectDir, \"manual-edits.json\"),\n        file.text_content,\n      )\n    }\n  }\n\n  async handleFileChangedOnFilesystem(absoluteFilePath: string) {\n    const relativeFilePath = path.relative(this.projectDir, absoluteFilePath)\n    // We've temporarily disabled upserting manual edits from filesystem changes\n    // because it can be edited by the browser\n    if (relativeFilePath.includes(\"manual-edits.json\")) return\n\n    console.log(`${relativeFilePath} saved. Applying changes...`)\n    await this.fsKy\n      .post(\"api/files/upsert\", {\n        json: {\n          file_path: relativeFilePath,\n          text_content: fs.readFileSync(absoluteFilePath, \"utf-8\"),\n          initiator: \"filesystem_change\",\n        },\n      })\n      .json()\n  }\n\n  async upsertInitialFiles() {\n    // Scan project directory for all files and upsert them\n    const fileNames = fs.readdirSync(this.projectDir)\n    for (const fileName of fileNames) {\n      if (fs.statSync(path.join(this.projectDir, fileName)).isDirectory())\n        continue\n      const fileContent = fs.readFileSync(\n        path.join(this.projectDir, fileName),\n        \"utf-8\",\n      )\n      await this.fsKy.post(\"api/files/upsert\", {\n        json: {\n          file_path: fileName,\n          text_content: fileContent,\n          initiator: \"filesystem_change\",\n        },\n      })\n    }\n  }\n\n  async stop() {\n    this.httpServer?.close()\n    this.eventsWatcher?.stop()\n  }\n}\n", "import * as http from \"node:http\"\nimport * as fs from \"node:fs\"\nimport * as path from \"node:path\"\nimport { getNodeHandler } from \"winterspec/adapters/node\"\nimport pkg from \"../../package.json\"\n\n// @ts-ignore\nimport winterspecBundle from \"@tscircuit/file-server/dist/bundle.js\"\nimport { getIndex } from \"../site/getIndex\"\n\nexport const createHttpServer = async (port = 3000) => {\n  const fileServerHandler = getNodeHandler(winterspecBundle as any, {})\n\n  const server = http.createServer(async (req, res) => {\n    const url = new URL(req.url!, `http://${req.headers.host}`)\n\n    if (url.pathname === \"/standalone.min.js\") {\n      const standaloneFilePath =\n        process.env.RUNFRAME_STANDALONE_FILE_PATH ||\n        path.resolve(\n          process.cwd(),\n          \"node_modules\",\n          \"@tscircuit/runframe/dist/standalone.min.js\",\n        )\n\n      try {\n        const content = fs.readFileSync(standaloneFilePath, \"utf8\")\n        res.writeHead(200, {\n          \"Content-Type\": \"application/javascript; charset=utf-8\",\n        })\n        res.end(content)\n        return\n      } catch (error) {\n        console.error(\"Error serving standalone.min.js:\", error)\n      }\n\n      res.writeHead(302, {\n        Location: `https://cdn.jsdelivr.net/npm/@tscircuit/runframe@${pkg.dependencies[\"@tscircuit/runframe\"].replace(/^[^0-9]+/, \"\")}/dist/standalone.min.js`,\n      })\n      res.end()\n      return\n    }\n\n    if (url.pathname === \"/\") {\n      const html = await getIndex()\n      res.writeHead(200, { \"Content-Type\": \"text/html\" })\n      res.end(html)\n      return\n    }\n\n    if (url.pathname.startsWith(\"/api/\")) {\n      req.url = req.url!.replace(\"/api/\", \"/\")\n      fileServerHandler(req, res)\n      return\n    }\n\n    res.writeHead(404)\n    res.end(\"Not found\")\n  })\n\n  return new Promise<{ server: http.Server }>((resolve) => {\n    server.listen(port, () => {\n      console.log(`Server running at http://localhost:${port}`)\n      resolve({ server })\n    })\n  })\n}\n", "{\n  \"name\": \"@tscircuit/cli\",\n  \"main\": \"dist/main.js\",\n  \"type\": \"module\",\n  \"version\": \"0.1.14\",\n  \"bin\": {\n    \"tsci\": \"./dist/main.js\"\n  },\n  \"scripts\": {\n    \"start\": \"bun run dev\",\n    \"dev\": \"bun --hot ./cli/main.ts dev ./example-dir/snippet1-basic.tsx\",\n    \"build\": \"tsup-node cli/main.ts --format esm --sourcemap inline\",\n    \"format\": \"biome format --write .\",\n    \"format:check\": \"biome format .\",\n    \"cli\": \"bun ./cli/main.ts\"\n  },\n  \"devDependencies\": {\n    \"@biomejs/biome\": \"^1.9.4\",\n    \"@tscircuit/core\": \"^0.0.249\",\n    \"@types/bun\": \"^1.1.15\",\n    \"@types/configstore\": \"^6.0.2\",\n    \"@types/react\": \"^19.0.1\",\n    \"@types/semver\": \"^7.5.8\",\n    \"get-port\": \"^7.1.0\",\n    \"tempy\": \"^3.1.0\",\n    \"tsup\": \"^8.3.5\",\n    \"typed-ky\": \"^0.0.4\"\n  },\n  \"peerDependencies\": {\n    \"typescript\": \"^5.0.0\"\n  },\n  \"dependencies\": {\n    \"@tscircuit/file-server\": \"^0.0.13\",\n    \"@tscircuit/runframe\": \"^0.0.47\",\n    \"chokidar\": \"^4.0.1\",\n    \"commander\": \"^12.1.0\",\n    \"configstore\": \"^7.0.0\",\n    \"cosmiconfig\": \"^9.0.0\",\n    \"delay\": \"^6.0.0\",\n    \"ky\": \"^1.7.4\",\n    \"make-vfs\": \"^1.0.15\",\n    \"perfect-cli\": \"^1.0.20\",\n    \"semver\": \"^7.6.3\"\n  }\n}\n", "import pkg from \"../../package.json\"\n\nexport const getIndex = async () => {\n  return `<html>\n    <head>\n    </head>\n    <body>\n      <script src=\"https://cdn.tailwindcss.com\"></script>\n      <div id=\"root\">loading...</div>\n      <script>\n      globalThis.process = { env: { NODE_ENV: \"production\" } }\n      </script>\n      <script src=\"/standalone.min.js\"></script>\n    </body>\n  </html>`\n}\n\n// <script src=\"https://cdn.jsdelivr.net/npm/@tscircuit/runframe@${pkg.dependencies[\"@tscircuit/runframe\"].replace(/^[^0-9]+/, \"\")}/dist/standalone.min.js\"></script>\n", "import { EventEmitter } from \"events\"\n\ninterface Event {\n  event_id: string\n  created_at: string\n  event_type: string\n  [key: string]: any\n}\n\ninterface EventsResponse {\n  event_list: Event[]\n}\n\nexport class EventsWatcher extends EventEmitter {\n  private lastPollTime: string\n  private pollInterval: number\n  private baseUrl: string\n  private polling = false\n  private timeoutId?: NodeJS.Timeout\n\n  constructor(baseUrl = \"http://localhost:3000\", pollInterval = 1000) {\n    super()\n    this.baseUrl = baseUrl\n    this.pollInterval = pollInterval\n    this.lastPollTime = new Date().toISOString()\n  }\n\n  async start() {\n    if (this.polling) return\n    this.polling = true\n    await this.poll()\n  }\n\n  stop() {\n    this.polling = false\n    if (this.timeoutId) {\n      clearTimeout(this.timeoutId)\n    }\n  }\n\n  private async poll() {\n    if (!this.polling) return\n\n    try {\n      const response = await fetch(\n        `${this.baseUrl}/api/events/list?since=${encodeURIComponent(this.lastPollTime)}`,\n      )\n\n      if (!response.ok) {\n        throw new Error(`HTTP error! status: ${response.status}`)\n      }\n\n      const data: EventsResponse = await response.json()\n\n      // Update last poll time to latest event or current time\n      const latestEvent = data.event_list[data.event_list.length - 1]\n      this.lastPollTime = latestEvent\n        ? latestEvent.created_at\n        : new Date().toISOString()\n\n      // Emit events in chronological order\n      data.event_list.forEach((event) => {\n        this.emit(event.event_type, event)\n        this.emit(\"*\", event)\n      })\n    } catch (error) {\n      this.emit(\"error\", error)\n    }\n    // Schedule next poll\n    this.timeoutId = globalThis.setTimeout(\n      () => this.poll(),\n      this.pollInterval,\n    ) as unknown as NodeJS.Timeout\n  }\n}\n", "import Configstore from \"configstore\"\nimport type { TypedConfigstore } from \"./TypedConfigStore\"\n\nexport interface CliConfig {\n  sessionToken?: string\n  githubUsername?: string\n  registryApiUrl?: string\n}\n\nexport const cliConfig: TypedConfigstore<CliConfig> = new Configstore(\n  \"tscircuit\",\n)\n\nexport const getRegistryApiUrl = (): string => {\n  return cliConfig.get(\"registryApiUrl\") ?? \"https://registry-api.tscircuit.com\"\n}\n", "import type { Command } from \"commander\"\nimport { cliConfig } from \"lib/cli-config\"\nimport delay from \"delay\"\nimport { getKy } from \"lib/registry-api/get-ky\"\nimport type { EndpointResponse } from \"lib/registry-api/endpoint-types\"\n\nexport const registerAuthLogin = (program: Command) => {\n  // Define the login action once to share between both commands\n  const loginAction = async () => {\n    const ky = getKy()\n\n    const { login_page } = await ky\n      .post<EndpointResponse[\"sessions/login_page/create\"]>(\n        \"sessions/login_page/create\",\n        {\n          json: {},\n        },\n      )\n      .json()\n\n    console.log(\"Please visit the following URL to log in:\")\n    console.log(login_page.url)\n\n    // Wait until we receive confirmation\n    while (true) {\n      const { login_page: new_login_page } = await ky\n        .post<EndpointResponse[\"sessions/login_page/get\"]>(\n          \"sessions/login_page/get\",\n          {\n            json: {\n              login_page_id: login_page.login_page_id,\n            },\n            headers: {\n              Authorization: `Bearer ${login_page.login_page_auth_token}`,\n            },\n          },\n        )\n        .json()\n\n      if (new_login_page.was_login_successful) {\n        console.log(\"Logged in! Generating token...\")\n        break\n      }\n\n      if (new_login_page.is_expired) {\n        throw new Error(\"Login page expired\")\n      }\n\n      await delay(1000)\n    }\n\n    const { session } = await ky\n      .post<EndpointResponse[\"sessions/login_page/exchange_for_cli_session\"]>(\n        \"sessions/login_page/exchange_for_cli_session\",\n        {\n          json: {\n            login_page_id: login_page.login_page_id,\n          },\n          headers: {\n            Authorization: `Bearer ${login_page.login_page_auth_token}`,\n          },\n        },\n      )\n      .json()\n\n    cliConfig.set(\"sessionToken\", session.token)\n\n    console.log(\"Ready to use!\")\n  }\n\n  // Register the auth login subcommand\n  program.commands\n    .find((c) => c.name() === \"auth\")!\n    .command(\"login\")\n    .description(\"Authenticate CLI, login to registry\")\n    .action(loginAction)\n\n  // Register the top-level login command as an alias\n  program\n    .command(\"login\")\n    .description(\"Login to tscircuit registry\")\n    .action(loginAction)\n}\n", "import { getRegistryApiUrl } from \"lib/cli-config\"\nimport ky, { type AfterResponseHook } from \"ky\"\n\nconst prettyResponseErrorHook: AfterResponseHook = async (\n  _request,\n  _options,\n  response,\n) => {\n  if (!response.ok) {\n    try {\n      const errorData = await response.json()\n      throw new Error(\n        `FAIL [${response.status}]: ${_request.method} ${\n          new URL(_request.url).pathname\n        } \\n\\n ${JSON.stringify(errorData, null, 2)}`,\n      )\n    } catch (e) {\n      //ignore, allow the error to be thrown\n    }\n  }\n}\n\nexport const getKy = () => {\n  return ky.create({\n    prefixUrl: getRegistryApiUrl(),\n    hooks: {\n      afterResponse: [prettyResponseErrorHook],\n    },\n  })\n}\n", "import type { Command } from \"commander\"\n\nexport const registerAuthLogout = (program: Command) => {\n  program.commands\n    .find((c) => c.name() === \"auth\")!\n    .command(\"logout\")\n    .description(\"Logout from registry\")\n    .action((args) => {\n      console.log(\"logout\")\n    })\n}\n", "import type { Command } from \"commander\"\n\nexport const registerAuth = (program: Command) => {\n  program.command(\"auth\").description(\"Login/logout\")\n}\n", "import type { Command } from \"commander\"\n\nexport const registerConfig = (program: Command) => {\n  program.command(\"config\").description(\"Manage tscircuit CLI configuration\")\n}\n", "import type { Command } from \"commander\"\nimport { cliConfig } from \"lib/cli-config\"\n\nexport const registerConfigPrint = (program: Command) => {\n  program.commands\n    .find((c) => c.name() === \"config\")!\n    .command(\"print\")\n    .description(\"Print the current config\")\n    .action(() => {\n      console.log(JSON.stringify(cliConfig.all, null, 2))\n    })\n}\n", "import type { Command } from \"commander\"\nimport { getKy } from \"lib/registry-api/get-ky\"\nimport * as fs from \"node:fs\"\nimport * as path from \"node:path\"\n\nexport const registerClone = (program: Command) => {\n  program\n    .command(\"clone\")\n    .description(\"Clone a snippet from the registry\")\n    .argument(\"<snippet>\", \"Snippet to clone (e.g. author/snippetName)\")\n    .action(async (snippetPath: string) => {\n      let author: string\n      let snippetName: string\n      if (!snippetPath.startsWith(\"@tsci/\") && snippetPath.includes(\"/\")) {\n        ;[author, snippetName] = snippetPath.split(\"/\")\n      } else {\n        const trimmedPath = snippetPath.replace(\"@tsci/\", \"\")\n        const firstDotIndex = trimmedPath.indexOf(\".\")\n        author = trimmedPath.slice(0, firstDotIndex)\n        snippetName = trimmedPath.slice(firstDotIndex + 1)\n      }\n\n      if (!author || !snippetName) {\n        console.error(\n          \"Invalid snippet path. Use format: author/snippetName, author.snippetName or @tsci/author.snippetName\",\n        )\n        process.exit(1)\n      }\n\n      const ky = getKy()\n\n      try {\n        console.log(`Cloning ${author}/${snippetName}...`)\n\n        const packageFileList = await ky\n          .post<{\n            package_files: Array<{\n              package_file_id: string\n              package_release_id: string\n              file_path: string\n              created_at: string\n            }>\n          }>(\"package_files/list\", {\n            json: {\n              package_name: `${author}/${snippetName}`,\n              use_latest_version: true,\n            },\n          })\n          .json()\n\n        // Create directory if it doesn't exist\n        const dirPath = `./${author}.${snippetName}`\n        if (!fs.existsSync(dirPath)) {\n          fs.mkdirSync(dirPath)\n        }\n\n        // Download each file that doesn't start with dist/\n        for (const fileInfo of packageFileList.package_files) {\n          const filePath = fileInfo.file_path.startsWith(\"/\")\n            ? fileInfo.file_path.slice(1)\n            : fileInfo.file_path\n\n          if (filePath.startsWith(\"dist/\")) continue\n\n          const fileContent = await ky\n            .post<{\n              package_file: {\n                content_text: string\n              }\n            }>(\"package_files/get\", {\n              json: {\n                package_name: `${author}/${snippetName}`,\n                file_path: fileInfo.file_path,\n              },\n            })\n            .json()\n\n          const fullPath = path.join(dirPath, filePath)\n          const dirName = path.dirname(fullPath)\n\n          // Create nested directories if they don't exist\n          if (!fs.existsSync(dirName)) {\n            fs.mkdirSync(dirName, { recursive: true })\n          }\n\n          fs.writeFileSync(fullPath, fileContent.package_file.content_text)\n        }\n\n        console.log(`Successfully cloned to ./${author}.${snippetName}/`)\n      } catch (error) {\n        if (error instanceof Error) {\n          console.error(\"Failed to clone snippet:\", error.message)\n        } else {\n          console.error(\"Failed to clone snippet:\", error)\n        }\n        process.exit(1)\n      }\n    })\n}\n", "import type { Command } from \"commander\"\nimport { createCircuitWebWorker } from \"@tscircuit/eval-webworker\"\nimport webWorkerBundleUrl from \"@tscircuit/eval-webworker/blob-url\"\nimport { getVirtualFileSystemFromDirPath } from \"make-vfs\"\nimport path from \"node:path\"\nimport fs from \"node:fs\"\n\nconst ALLOWED_FORMATS = [\n  \"json\",\n  \"circuit-json\",\n  \"schematic-svg\",\n  \"pcb-svg\",\n  \"gerbers\",\n  \"readable-netlist\",\n  \"gltf\",\n  \"specctra-dsn\",\n] as const\n\ntype Format = (typeof ALLOWED_FORMATS)[number]\n\nconst OUTPUT_EXTENSIONS = {\n  json: \".circuit.json\",\n  \"circuit-json\": \".circuit.json\",\n  \"schematic-svg\": \"-schematic.svg\",\n  \"pcb-svg\": \"-pcb.svg\",\n  gerbers: \"-gerbers.zip\",\n  \"readable-netlist\": \"-readable.netlist\",\n  gltf: \".gltf\",\n  \"specctra-dsn\": \".dsn\",\n}\n\nexport const registerExport = (program: Command) => {\n  program\n    .command(\"export\")\n    .description(\"Export tscircuit code to various formats\")\n    .argument(\"<file>\", \"Path to the snippet file\")\n    .option(\"-f, --format <format>\", \"Output format\")\n    .option(\"-o, --output <path>\", \"Output file path\")\n    .action(async (file, options) => {\n      const { format = \"circuit-json\" } = options\n      let { output } = options\n      if (!ALLOWED_FORMATS.includes(format)) {\n        throw new Error(\n          `Invalid format: ${format}\\nSupported formats: ${ALLOWED_FORMATS.join(\",\")}`,\n        )\n      }\n\n      if (!output) {\n        output = path.basename(file).replace(/\\.[^.]+$/, \"\")\n      }\n\n      const worker = await createCircuitWebWorker({\n        webWorkerUrl: webWorkerBundleUrl,\n      })\n\n      const projectDir = path.dirname(file)\n\n      const relativeComponentPath = path.relative(projectDir, file)\n\n      await worker.executeWithFsMap({\n        entrypoint: \"entrypoint.tsx\",\n        fsMap: {\n          ...((await getVirtualFileSystemFromDirPath({\n            dirPath: projectDir,\n            contentFormat: \"string\",\n          })) as Record<string, string>),\n          \"entrypoint.tsx\": `\nimport MyCircuit from \"./${relativeComponentPath}\"\n\ncircuit.add(<MyCircuit />)\n        `,\n        },\n      })\n\n      await worker.renderUntilSettled()\n\n      const circuitJson = await worker.getCircuitJson()\n\n      const outputPath = path.join(\n        projectDir,\n        `${output}${OUTPUT_EXTENSIONS[format as Format]}`,\n      )\n\n      fs.writeFileSync(outputPath, JSON.stringify(circuitJson))\n\n      console.log(`Exported to ${outputPath}`)\n\n      process.exit(0)\n    })\n}\n", "import type { Command } from \"commander\"\nimport { cliConfig } from \"lib/cli-config\"\n\nexport const registerAuthPrintToken = (program: Command) => {\n  program.commands\n    .find((c) => c.name() === \"auth\")!\n    .command(\"print-token\")\n    .description(\"Prints your auth token\")\n    .action(() => {\n      const token = cliConfig.get(\"sessionToken\")\n      if (!token) return console.log(\"You need to log in to access this.\")\n      console.log(\"Your Token:\\n\", token)\n    })\n}\n", "import type { Command } from \"commander\"\nimport { cliConfig } from \"lib/cli-config\"\n\nfunction validateJWTLength(token: string) {\n  const parts = token.split(\".\")\n\n  if (parts.length === 3 && parts.every((part) => part.length > 0)) {\n    return true\n  } else {\n    return false\n  }\n}\nexport const registerAuthSetToken = (program: Command) => {\n  program.commands\n    .find((c) => c.name() === \"auth\")!\n    .command(\"set-token\")\n    .description(\"Explicitly set your auth token\")\n    .argument(\"<token>\", \"New token to manually configure\")\n    .action((token) => {\n      if (!validateJWTLength(token))\n        return console.log(\"Invalid token provided\")\n      cliConfig.set(\"sessionToken\", token)\n      console.log(\"Token manually updated.\")\n    })\n}\n"],
  "mappings": "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",
  "names": ["program", "path", "fs", "fs", "path", "fs", "path", "resolve", "path", "fs", "program", "ky", "program", "ky", "program", "program", "program", "program", "fs", "path", "program", "ky", "path", "fs", "program", "program", "program"]
}

|
|
849
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../cli/main.ts", "../cli/init/register.ts", "../cli/dev/register.ts", "../lib/dependency-analysis/installNodeModuleTypesForSnippet.ts", "../cli/dev/DevServer.ts", "../lib/server/createHttpServer.ts", "../package.json", "../lib/site/getIndex.ts", "../lib/server/EventsWatcher.ts", "../lib/dependency-analysis/FilesystemTypesHandler.ts", "../lib/dependency-analysis/findImportsInSnippet.ts", "../lib/cli-config/index.ts", "../cli/auth/login/register.ts", "../lib/registry-api/get-ky.ts", "../cli/auth/logout/register.ts", "../cli/auth/register.ts", "../cli/config/register.ts", "../cli/config/print/register.ts", "../cli/clone/register.ts", "../cli/export/register.ts", "../cli/auth/print-token/register.ts", "../cli/auth/set-token/register.ts"],
  "sourcesContent": ["#!/usr/bin/env node\nimport { Command } from \"commander\"\nimport { registerInit } from \"./init/register\"\nimport { registerDev } from \"./dev/register\"\nimport { registerAuthLogin } from \"./auth/login/register\"\nimport { registerAuthLogout } from \"./auth/logout/register\"\nimport { registerAuth } from \"./auth/register\"\nimport { registerConfig } from \"./config/register\"\nimport { registerConfigPrint } from \"./config/print/register\"\nimport { registerClone } from \"./clone/register\"\nimport { perfectCli } from \"perfect-cli\"\nimport pkg from \"../package.json\"\nimport semver from \"semver\"\nimport { registerExport } from \"./export/register\"\nimport { registerAuthPrintToken } from \"./auth/print-token/register\"\nimport { registerAuthSetToken } from \"./auth/set-token/register\"\n\nconst program = new Command()\n\nprogram\n  .name(\"tsci\")\n  .description(\"CLI for developing tscircuit snippets\")\n  // HACK: at build time the version is old, we need to\n  // fix this at some point...\n  .version(semver.inc(pkg.version, \"patch\") ?? pkg.version)\n\nregisterInit(program)\n\nregisterDev(program)\nregisterClone(program)\n\nregisterAuth(program)\nregisterAuthLogin(program)\nregisterAuthLogout(program)\nregisterAuthPrintToken(program)\nregisterAuthSetToken(program)\n\nregisterConfig(program)\nregisterConfigPrint(program)\n\nregisterExport(program)\n\nif (process.argv.length === 2) {\n  perfectCli(program, process.argv)\n} else {\n  program.parse()\n}\n", "import type { Command } from \"commander\"\nimport * as fs from \"node:fs\"\nimport * as path from \"node:path\"\n\nexport const registerInit = (program: Command) => {\n  program\n    .command(\"init\")\n    .description(\"Initialize a new TSCircuit project in the current directory\")\n    .action(() => {\n      const currentDir = process.cwd()\n\n      const indexFilePath = path.join(currentDir, \"index.tsx\")\n      const npmrcFilePath = path.join(currentDir, \".npmrc\")\n\n      const indexContent = `\nexport default () => (\n  <board width=\"10mm\" height=\"10mm\">\n    <resistor\n      resistance=\"1k\"\n      footprint=\"0402\"\n      name=\"R1\"\n      schX={3}\n      pcbX={3}\n    />\n    <capacitor\n      capacitance=\"1000pF\"\n      footprint=\"0402\"\n      name=\"C1\"\n      schX={-3}\n      pcbX={-3}\n    />\n    <trace from=\".R1 > .pin1\" to=\".C1 > .pin1\" />\n  </board>\n);\n`\n\n      const npmrcContent = `\n@tsci:registry=https://npm.tscircuit.com\n`\n\n      if (!fs.existsSync(indexFilePath)) {\n        fs.writeFileSync(indexFilePath, indexContent.trimStart())\n        console.log(`Created: ${indexFilePath}`)\n      } else {\n        console.log(`Skipped: ${indexFilePath} already exists`)\n      }\n\n      if (!fs.existsSync(npmrcFilePath)) {\n        fs.writeFileSync(npmrcFilePath, npmrcContent.trimStart())\n        console.log(`Created: ${npmrcFilePath}`)\n      } else {\n        console.log(`Skipped: ${npmrcFilePath} already exists`)\n      }\n\n      console.log(\n        `Initialization complete. Run \"tsci dev\" to start developing.`,\n      )\n    })\n}\n", "import type { Command } from \"commander\"\nimport * as path from \"node:path\"\nimport * as chokidar from \"chokidar\"\nimport * as fs from \"node:fs\"\nimport { createHttpServer } from \"lib/server/createHttpServer\"\nimport { getLocalFileDependencies } from \"lib/dependency-analysis/getLocalFileDependencies\"\nimport { installNodeModuleTypesForSnippet } from \"../../lib/dependency-analysis/installNodeModuleTypesForSnippet\"\nimport { EventsWatcher } from \"../../lib/server/EventsWatcher\"\nimport { DevServer } from \"./DevServer\"\n\nexport const registerDev = (program: Command) => {\n  program\n    .command(\"dev\")\n    .description(\"Start development server for a snippet\")\n    .argument(\"[file]\", \"Path to the snippet file\")\n    .option(\"-p, --port <number>\", \"Port to run server on\", \"3000\")\n    .action(async (file: string, options: { port: string }) => {\n      const port = parseInt(options.port)\n      let absolutePath: string\n\n      if (file) {\n        absolutePath = path.resolve(file)\n      } else {\n        const entrypointPath = path.resolve(\"index.tsx\")\n        if (fs.existsSync(entrypointPath)) {\n          absolutePath = entrypointPath\n          console.log(\"No file provided. Using 'index.tsx' as the entrypoint.\")\n        } else {\n          console.log(\n            \"No entrypoint found. Run 'tsci init' to bootstrap a basic project.\",\n          )\n          return\n        }\n      }\n\n      const fileDir = path.dirname(absolutePath)\n\n      try {\n        console.log(\"Installing types for imported snippets...\")\n        await installNodeModuleTypesForSnippet(absolutePath)\n        console.log(\"Types installed successfully\")\n      } catch (error) {\n        console.warn(\"Failed to install types:\", error)\n      }\n\n      const server = new DevServer({\n        port,\n        componentFilePath: absolutePath,\n      })\n\n      await server.start()\n      await server.addEntrypoint()\n    })\n}\n", "import * as fs from \"node:fs\"\nimport * as path from \"node:path\"\nimport * as ts from \"typescript\"\n\ninterface SnippetApiResponse {\n  snippet: {\n    dts: string\n  }\n}\n\nexport async function installNodeModuleTypesForSnippet(snippetPath: string) {\n  const content = fs.readFileSync(snippetPath, \"utf-8\")\n  const sourceFile = ts.createSourceFile(\n    snippetPath,\n    content,\n    ts.ScriptTarget.Latest,\n    true,\n  )\n\n  const imports: string[] = []\n\n  function visit(node: ts.Node) {\n    if (ts.isImportDeclaration(node)) {\n      const moduleSpecifier = node.moduleSpecifier\n      if (moduleSpecifier && ts.isStringLiteral(moduleSpecifier)) {\n        const importPath = moduleSpecifier.text\n        if (importPath.startsWith(\"@tsci/\")) {\n          imports.push(importPath)\n        }\n      }\n    }\n    ts.forEachChild(node, visit)\n  }\n\n  visit(sourceFile)\n\n  let projectRoot = path.dirname(snippetPath)\n  while (projectRoot !== path.parse(projectRoot).root) {\n    if (fs.existsSync(path.join(projectRoot, \"package.json\"))) {\n      break\n    }\n    projectRoot = path.dirname(projectRoot)\n  }\n\n  for (const importPath of imports) {\n    const [owner, name] = importPath.replace(\"@tsci/\", \"\").split(\".\")\n    try {\n      const response = await fetch(\n        `https://registry-api.tscircuit.com/snippets/get?owner_name=${owner}&unscoped_name=${name}`,\n      )\n\n      if (!response.ok) {\n        console.warn(`Failed to fetch types for ${importPath}`)\n        continue\n      }\n\n      const data: SnippetApiResponse = await response.json()\n\n      if (data.snippet.dts) {\n        const packageDir = path.join(\n          projectRoot,\n          \"node_modules\",\n          \"@tsci\",\n          `${owner}.${name}`,\n        )\n        fs.mkdirSync(packageDir, { recursive: true })\n\n        fs.writeFileSync(path.join(packageDir, \"index.d.ts\"), data.snippet.dts)\n      }\n    } catch (error) {\n      console.warn(`Error fetching types for ${importPath}:`, error)\n    }\n  }\n}\n", "import ky from \"ky\"\nimport type { FileServerRoutes } from \"lib/file-server/FileServerRoutes\"\nimport { createHttpServer } from \"lib/server/createHttpServer\"\nimport { EventsWatcher } from \"lib/server/EventsWatcher\"\nimport type http from \"node:http\"\nimport type { TypedKyInstance } from \"typed-ky\"\nimport path from \"node:path\"\nimport fs from \"node:fs\"\nimport type { FileUpdatedEvent } from \"lib/file-server/FileServerEvent\"\nimport * as chokidar from \"chokidar\"\nimport { FilesystemTypesHandler } from \"lib/dependency-analysis/FilesystemTypesHandler\"\n\nexport class DevServer {\n  port: number\n  /**\n   * The path to a component that exports a <board /> or <group /> component\n   */\n  componentFilePath: string\n\n  projectDir: string\n\n  /**\n   * The HTTP server that hosts the file server and event bus. You can use\n   * fsKy to communicate with the file server/event bus\n   */\n  httpServer?: http.Server\n  /**\n   * Watches for events on the event bus by polling `api/events/list`\n   */\n  eventsWatcher?: EventsWatcher\n  /**\n   * A ky instance that can be used to communicate with the file server and\n   * event bus\n   */\n  fsKy: TypedKyInstance<keyof FileServerRoutes, FileServerRoutes>\n  /**\n   * A chokidar instance that watches the project directory for file changes\n   */\n  filesystemWatcher?: chokidar.FSWatcher\n\n  private typesHandler?: FilesystemTypesHandler\n\n  constructor({\n    port,\n    componentFilePath,\n  }: {\n    port: number\n    componentFilePath: string\n  }) {\n    this.port = port\n    this.componentFilePath = componentFilePath\n    this.projectDir = path.dirname(componentFilePath)\n    this.fsKy = ky.create({\n      prefixUrl: `http://localhost:${port}`,\n    }) as any\n    this.typesHandler = new FilesystemTypesHandler(this.projectDir)\n  }\n\n  async start() {\n    const { server } = await createHttpServer(this.port)\n    this.httpServer = server\n\n    this.eventsWatcher = new EventsWatcher(`http://localhost:${this.port}`)\n    this.eventsWatcher.start()\n\n    this.eventsWatcher.on(\n      \"FILE_UPDATED\",\n      this.handleFileUpdatedEventFromServer.bind(this),\n    )\n\n    this.filesystemWatcher = chokidar.watch(this.projectDir, {\n      persistent: true,\n      ignoreInitial: true,\n    })\n\n    this.filesystemWatcher.on(\"change\", (filePath) =>\n      this.handleFileChangedOnFilesystem(filePath),\n    )\n    this.filesystemWatcher.on(\"add\", (filePath) =>\n      this.handleFileChangedOnFilesystem(filePath),\n    )\n\n    this.upsertInitialFiles()\n\n    this.typesHandler?.handleInitialTypeDependencies(this.componentFilePath)\n  }\n\n  async addEntrypoint() {\n    const relativeComponentFilePath = path.relative(\n      this.projectDir,\n      this.componentFilePath,\n    )\n    await this.fsKy.post(\"api/files/upsert\", {\n      json: {\n        file_path: \"entrypoint.tsx\",\n        text_content: `\nimport MyCircuit from \"./${relativeComponentFilePath}\"\n\ncircuit.add(<MyCircuit />)\n`,\n      },\n    })\n  }\n\n  async handleFileUpdatedEventFromServer(ev: FileUpdatedEvent) {\n    if (ev.initiator === \"filesystem_change\") return\n\n    if (ev.file_path === \"manual-edits.json\") {\n      console.log(\"Manual edits updated, updating on filesystem...\")\n      const { file } = await this.fsKy\n        .get(\"api/files/get\", {\n          searchParams: { file_path: ev.file_path },\n        })\n        .json()\n      fs.writeFileSync(\n        path.join(this.projectDir, \"manual-edits.json\"),\n        file.text_content,\n      )\n    }\n  }\n\n  async handleFileChangedOnFilesystem(absoluteFilePath: string) {\n    const relativeFilePath = path.relative(this.projectDir, absoluteFilePath)\n    // We've temporarily disabled upserting manual edits from filesystem changes\n    // because it can be edited by the browser\n    if (relativeFilePath.includes(\"manual-edits.json\")) return\n\n    await this.typesHandler?.handleFileTypeDependencies(absoluteFilePath)\n\n    console.log(`${relativeFilePath} saved. Applying changes...`)\n    await this.fsKy\n      .post(\"api/files/upsert\", {\n        json: {\n          file_path: relativeFilePath,\n          text_content: fs.readFileSync(absoluteFilePath, \"utf-8\"),\n          initiator: \"filesystem_change\",\n        },\n      })\n      .json()\n  }\n\n  async upsertInitialFiles() {\n    // Scan project directory for all files and upsert them\n    const fileNames = fs.readdirSync(this.projectDir)\n    for (const fileName of fileNames) {\n      if (fs.statSync(path.join(this.projectDir, fileName)).isDirectory())\n        continue\n      const fileContent = fs.readFileSync(\n        path.join(this.projectDir, fileName),\n        \"utf-8\",\n      )\n      await this.fsKy.post(\"api/files/upsert\", {\n        json: {\n          file_path: fileName,\n          text_content: fileContent,\n          initiator: \"filesystem_change\",\n        },\n      })\n    }\n  }\n\n  async stop() {\n    this.httpServer?.close()\n    this.eventsWatcher?.stop()\n  }\n}\n", "import * as http from \"node:http\"\nimport * as fs from \"node:fs\"\nimport * as path from \"node:path\"\nimport { getNodeHandler } from \"winterspec/adapters/node\"\nimport pkg from \"../../package.json\"\n\n// @ts-ignore\nimport winterspecBundle from \"@tscircuit/file-server/dist/bundle.js\"\nimport { getIndex } from \"../site/getIndex\"\n\nexport const createHttpServer = async (port = 3000) => {\n  const fileServerHandler = getNodeHandler(winterspecBundle as any, {})\n\n  const server = http.createServer(async (req, res) => {\n    const url = new URL(req.url!, `http://${req.headers.host}`)\n\n    if (url.pathname === \"/standalone.min.js\") {\n      const standaloneFilePath =\n        process.env.RUNFRAME_STANDALONE_FILE_PATH ||\n        path.resolve(\n          process.cwd(),\n          \"node_modules\",\n          \"@tscircuit/runframe/dist/standalone.min.js\",\n        )\n\n      try {\n        const content = fs.readFileSync(standaloneFilePath, \"utf8\")\n        res.writeHead(200, {\n          \"Content-Type\": \"application/javascript; charset=utf-8\",\n        })\n        res.end(content)\n        return\n      } catch (error) {\n        console.error(\"Error serving standalone.min.js:\", error)\n      }\n\n      res.writeHead(302, {\n        Location: `https://cdn.jsdelivr.net/npm/@tscircuit/runframe@${pkg.dependencies[\"@tscircuit/runframe\"].replace(/^[^0-9]+/, \"\")}/dist/standalone.min.js`,\n      })\n      res.end()\n      return\n    }\n\n    if (url.pathname === \"/\") {\n      const html = await getIndex()\n      res.writeHead(200, { \"Content-Type\": \"text/html\" })\n      res.end(html)\n      return\n    }\n\n    if (url.pathname.startsWith(\"/api/\")) {\n      req.url = req.url!.replace(\"/api/\", \"/\")\n      fileServerHandler(req, res)\n      return\n    }\n\n    res.writeHead(404)\n    res.end(\"Not found\")\n  })\n\n  return new Promise<{ server: http.Server }>((resolve) => {\n    server.listen(port, () => {\n      console.log(`Server running at http://localhost:${port}`)\n      resolve({ server })\n    })\n  })\n}\n", "{\n  \"name\": \"@tscircuit/cli\",\n  \"main\": \"dist/main.js\",\n  \"type\": \"module\",\n  \"version\": \"0.1.16\",\n  \"bin\": {\n    \"tsci\": \"./dist/main.js\"\n  },\n  \"scripts\": {\n    \"start\": \"bun run dev\",\n    \"dev\": \"bun --hot ./cli/main.ts dev ./example-dir/snippet1-basic.tsx\",\n    \"build\": \"tsup-node cli/main.ts --format esm --sourcemap inline\",\n    \"format\": \"biome format --write .\",\n    \"format:check\": \"biome format .\",\n    \"cli\": \"bun ./cli/main.ts\"\n  },\n  \"devDependencies\": {\n    \"@biomejs/biome\": \"^1.9.4\",\n    \"@tscircuit/core\": \"^0.0.249\",\n    \"@types/bun\": \"^1.1.15\",\n    \"@types/configstore\": \"^6.0.2\",\n    \"@types/react\": \"^19.0.1\",\n    \"@types/semver\": \"^7.5.8\",\n    \"get-port\": \"^7.1.0\",\n    \"tempy\": \"^3.1.0\",\n    \"tsup\": \"^8.3.5\",\n    \"typed-ky\": \"^0.0.4\"\n  },\n  \"peerDependencies\": {\n    \"typescript\": \"^5.0.0\"\n  },\n  \"dependencies\": {\n    \"@tscircuit/file-server\": \"^0.0.13\",\n    \"@tscircuit/runframe\": \"^0.0.47\",\n    \"chokidar\": \"^4.0.1\",\n    \"commander\": \"^12.1.0\",\n    \"configstore\": \"^7.0.0\",\n    \"cosmiconfig\": \"^9.0.0\",\n    \"delay\": \"^6.0.0\",\n    \"ky\": \"^1.7.4\",\n    \"make-vfs\": \"^1.0.15\",\n    \"perfect-cli\": \"^1.0.20\",\n    \"semver\": \"^7.6.3\"\n  }\n}\n", "import pkg from \"../../package.json\"\n\nexport const getIndex = async () => {\n  return `<html>\n    <head>\n    </head>\n    <body>\n      <script src=\"https://cdn.tailwindcss.com\"></script>\n      <div id=\"root\">loading...</div>\n      <script>\n      globalThis.process = { env: { NODE_ENV: \"production\" } }\n      </script>\n      <script src=\"/standalone.min.js\"></script>\n    </body>\n  </html>`\n}\n\n// <script src=\"https://cdn.jsdelivr.net/npm/@tscircuit/runframe@${pkg.dependencies[\"@tscircuit/runframe\"].replace(/^[^0-9]+/, \"\")}/dist/standalone.min.js\"></script>\n", "import { EventEmitter } from \"events\"\n\ninterface Event {\n  event_id: string\n  created_at: string\n  event_type: string\n  [key: string]: any\n}\n\ninterface EventsResponse {\n  event_list: Event[]\n}\n\nexport class EventsWatcher extends EventEmitter {\n  private lastPollTime: string\n  private pollInterval: number\n  private baseUrl: string\n  private polling = false\n  private timeoutId?: NodeJS.Timeout\n\n  constructor(baseUrl = \"http://localhost:3000\", pollInterval = 1000) {\n    super()\n    this.baseUrl = baseUrl\n    this.pollInterval = pollInterval\n    this.lastPollTime = new Date().toISOString()\n  }\n\n  async start() {\n    if (this.polling) return\n    this.polling = true\n    await this.poll()\n  }\n\n  stop() {\n    this.polling = false\n    if (this.timeoutId) {\n      clearTimeout(this.timeoutId)\n    }\n  }\n\n  private async poll() {\n    if (!this.polling) return\n\n    try {\n      const response = await fetch(\n        `${this.baseUrl}/api/events/list?since=${encodeURIComponent(this.lastPollTime)}`,\n      )\n\n      if (!response.ok) {\n        throw new Error(`HTTP error! status: ${response.status}`)\n      }\n\n      const data: EventsResponse = await response.json()\n\n      // Update last poll time to latest event or current time\n      const latestEvent = data.event_list[data.event_list.length - 1]\n      this.lastPollTime = latestEvent\n        ? latestEvent.created_at\n        : new Date().toISOString()\n\n      // Emit events in chronological order\n      data.event_list.forEach((event) => {\n        this.emit(event.event_type, event)\n        this.emit(\"*\", event)\n      })\n    } catch (error) {\n      this.emit(\"error\", error)\n    }\n    // Schedule next poll\n    this.timeoutId = globalThis.setTimeout(\n      () => this.poll(),\n      this.pollInterval,\n    ) as unknown as NodeJS.Timeout\n  }\n}\n", "import * as fs from \"node:fs\"\nimport * as path from \"node:path\"\nimport { findImportsInSnippet } from \"./findImportsInSnippet\"\nimport { installNodeModuleTypesForSnippet } from \"./installNodeModuleTypesForSnippet\"\n\nexport class FilesystemTypesHandler {\n  private projectRoot: string\n\n  constructor(initialDir: string) {\n    this.projectRoot = this.findProjectRoot(initialDir)\n  }\n\n  async handleInitialTypeDependencies(filePath: string) {\n    console.log(\"Checking initial type dependencies...\")\n    try {\n      if (!this.areTypesInstalled(filePath)) {\n        console.log(\"Installing missing initial types...\")\n        await installNodeModuleTypesForSnippet(filePath)\n      }\n    } catch (error) {\n      console.warn(\"Error handling initial type dependencies:\", error)\n    }\n  }\n\n  async handleFileTypeDependencies(filePath: string) {\n    try {\n      if (!this.areTypesInstalled(filePath)) {\n        console.log(\"Installing missing file types...\")\n        await installNodeModuleTypesForSnippet(filePath)\n      }\n    } catch (error) {\n      console.warn(\"Failed to verify types:\", error)\n    }\n  }\n\n  private areTypesInstalled(filePath: string): boolean {\n    const imports = findImportsInSnippet(filePath)\n    return imports.every((imp) => this.checkTypeExists(imp))\n  }\n\n  private checkTypeExists(importPath: string): boolean {\n    if (!importPath.startsWith(\"@tsci/\")) return true\n\n    const pathWithoutPrefix = importPath.replace(\"@tsci/\", \"\")\n    const [owner, name] = pathWithoutPrefix.split(\".\")\n\n    const typePath = path.join(\n      this.projectRoot,\n      \"node_modules\",\n      \"@tsci\",\n      `${owner}.${name}`,\n      \"index.d.ts\",\n    )\n\n    return fs.existsSync(typePath)\n  }\n\n  private findProjectRoot(startDir: string): string {\n    let root = path.resolve(startDir)\n    while (root !== path.parse(root).root) {\n      if (fs.existsSync(path.join(root, \"package.json\"))) {\n        return root\n      }\n      root = path.dirname(root)\n    }\n    return startDir\n  }\n}\n", "import * as fs from \"node:fs\"\nimport * as ts from \"typescript\"\n\nexport function findImportsInSnippet(snippetPath: string): string[] {\n  const content = fs.readFileSync(snippetPath, \"utf-8\")\n  const sourceFile = ts.createSourceFile(\n    snippetPath,\n    content,\n    ts.ScriptTarget.Latest,\n    true,\n  )\n\n  const imports: string[] = []\n\n  function visit(node: ts.Node) {\n    if (ts.isImportDeclaration(node)) {\n      const moduleSpecifier = node.moduleSpecifier\n      if (moduleSpecifier && ts.isStringLiteral(moduleSpecifier)) {\n        const importPath = moduleSpecifier.text\n        if (importPath.startsWith(\"@tsci/\")) {\n          imports.push(importPath)\n        }\n      }\n    }\n    ts.forEachChild(node, visit)\n  }\n\n  visit(sourceFile)\n\n  return imports\n}\n", "import Configstore from \"configstore\"\nimport type { TypedConfigstore } from \"./TypedConfigStore\"\n\nexport interface CliConfig {\n  sessionToken?: string\n  githubUsername?: string\n  registryApiUrl?: string\n}\n\nexport const cliConfig: TypedConfigstore<CliConfig> = new Configstore(\n  \"tscircuit\",\n)\n\nexport const getRegistryApiUrl = (): string => {\n  return cliConfig.get(\"registryApiUrl\") ?? \"https://registry-api.tscircuit.com\"\n}\n", "import type { Command } from \"commander\"\nimport { cliConfig } from \"lib/cli-config\"\nimport delay from \"delay\"\nimport { getKy } from \"lib/registry-api/get-ky\"\nimport type { EndpointResponse } from \"lib/registry-api/endpoint-types\"\n\nexport const registerAuthLogin = (program: Command) => {\n  // Define the login action once to share between both commands\n  const loginAction = async () => {\n    const ky = getKy()\n\n    const { login_page } = await ky\n      .post<EndpointResponse[\"sessions/login_page/create\"]>(\n        \"sessions/login_page/create\",\n        {\n          json: {},\n        },\n      )\n      .json()\n\n    console.log(\"Please visit the following URL to log in:\")\n    console.log(login_page.url)\n\n    // Wait until we receive confirmation\n    while (true) {\n      const { login_page: new_login_page } = await ky\n        .post<EndpointResponse[\"sessions/login_page/get\"]>(\n          \"sessions/login_page/get\",\n          {\n            json: {\n              login_page_id: login_page.login_page_id,\n            },\n            headers: {\n              Authorization: `Bearer ${login_page.login_page_auth_token}`,\n            },\n          },\n        )\n        .json()\n\n      if (new_login_page.was_login_successful) {\n        console.log(\"Logged in! Generating token...\")\n        break\n      }\n\n      if (new_login_page.is_expired) {\n        throw new Error(\"Login page expired\")\n      }\n\n      await delay(1000)\n    }\n\n    const { session } = await ky\n      .post<EndpointResponse[\"sessions/login_page/exchange_for_cli_session\"]>(\n        \"sessions/login_page/exchange_for_cli_session\",\n        {\n          json: {\n            login_page_id: login_page.login_page_id,\n          },\n          headers: {\n            Authorization: `Bearer ${login_page.login_page_auth_token}`,\n          },\n        },\n      )\n      .json()\n\n    cliConfig.set(\"sessionToken\", session.token)\n\n    console.log(\"Ready to use!\")\n  }\n\n  // Register the auth login subcommand\n  program.commands\n    .find((c) => c.name() === \"auth\")!\n    .command(\"login\")\n    .description(\"Authenticate CLI, login to registry\")\n    .action(loginAction)\n\n  // Register the top-level login command as an alias\n  program\n    .command(\"login\")\n    .description(\"Login to tscircuit registry\")\n    .action(loginAction)\n}\n", "import { getRegistryApiUrl } from \"lib/cli-config\"\nimport ky, { type AfterResponseHook } from \"ky\"\n\nconst prettyResponseErrorHook: AfterResponseHook = async (\n  _request,\n  _options,\n  response,\n) => {\n  if (!response.ok) {\n    try {\n      const errorData = await response.json()\n      throw new Error(\n        `FAIL [${response.status}]: ${_request.method} ${\n          new URL(_request.url).pathname\n        } \\n\\n ${JSON.stringify(errorData, null, 2)}`,\n      )\n    } catch (e) {\n      //ignore, allow the error to be thrown\n    }\n  }\n}\n\nexport const getKy = () => {\n  return ky.create({\n    prefixUrl: getRegistryApiUrl(),\n    hooks: {\n      afterResponse: [prettyResponseErrorHook],\n    },\n  })\n}\n", "import type { Command } from \"commander\"\n\nexport const registerAuthLogout = (program: Command) => {\n  program.commands\n    .find((c) => c.name() === \"auth\")!\n    .command(\"logout\")\n    .description(\"Logout from registry\")\n    .action((args) => {\n      console.log(\"logout\")\n    })\n}\n", "import type { Command } from \"commander\"\n\nexport const registerAuth = (program: Command) => {\n  program.command(\"auth\").description(\"Login/logout\")\n}\n", "import type { Command } from \"commander\"\n\nexport const registerConfig = (program: Command) => {\n  program.command(\"config\").description(\"Manage tscircuit CLI configuration\")\n}\n", "import type { Command } from \"commander\"\nimport { cliConfig } from \"lib/cli-config\"\n\nexport const registerConfigPrint = (program: Command) => {\n  program.commands\n    .find((c) => c.name() === \"config\")!\n    .command(\"print\")\n    .description(\"Print the current config\")\n    .action(() => {\n      console.log(JSON.stringify(cliConfig.all, null, 2))\n    })\n}\n", "import type { Command } from \"commander\"\nimport { getKy } from \"lib/registry-api/get-ky\"\nimport * as fs from \"node:fs\"\nimport * as path from \"node:path\"\n\nexport const registerClone = (program: Command) => {\n  program\n    .command(\"clone\")\n    .description(\"Clone a snippet from the registry\")\n    .argument(\"<snippet>\", \"Snippet to clone (e.g. author/snippetName)\")\n    .action(async (snippetPath: string) => {\n      let author: string\n      let snippetName: string\n      if (!snippetPath.startsWith(\"@tsci/\") && snippetPath.includes(\"/\")) {\n        ;[author, snippetName] = snippetPath.split(\"/\")\n      } else {\n        const trimmedPath = snippetPath.replace(\"@tsci/\", \"\")\n        const firstDotIndex = trimmedPath.indexOf(\".\")\n        author = trimmedPath.slice(0, firstDotIndex)\n        snippetName = trimmedPath.slice(firstDotIndex + 1)\n      }\n\n      if (!author || !snippetName) {\n        console.error(\n          \"Invalid snippet path. Use format: author/snippetName, author.snippetName or @tsci/author.snippetName\",\n        )\n        process.exit(1)\n      }\n\n      const ky = getKy()\n\n      try {\n        console.log(`Cloning ${author}/${snippetName}...`)\n\n        const packageFileList = await ky\n          .post<{\n            package_files: Array<{\n              package_file_id: string\n              package_release_id: string\n              file_path: string\n              created_at: string\n            }>\n          }>(\"package_files/list\", {\n            json: {\n              package_name: `${author}/${snippetName}`,\n              use_latest_version: true,\n            },\n          })\n          .json()\n\n        // Create directory if it doesn't exist\n        const dirPath = `./${author}.${snippetName}`\n        if (!fs.existsSync(dirPath)) {\n          fs.mkdirSync(dirPath)\n        }\n\n        // Download each file that doesn't start with dist/\n        for (const fileInfo of packageFileList.package_files) {\n          const filePath = fileInfo.file_path.startsWith(\"/\")\n            ? fileInfo.file_path.slice(1)\n            : fileInfo.file_path\n\n          if (filePath.startsWith(\"dist/\")) continue\n\n          const fileContent = await ky\n            .post<{\n              package_file: {\n                content_text: string\n              }\n            }>(\"package_files/get\", {\n              json: {\n                package_name: `${author}/${snippetName}`,\n                file_path: fileInfo.file_path,\n              },\n            })\n            .json()\n\n          const fullPath = path.join(dirPath, filePath)\n          const dirName = path.dirname(fullPath)\n\n          // Create nested directories if they don't exist\n          if (!fs.existsSync(dirName)) {\n            fs.mkdirSync(dirName, { recursive: true })\n          }\n\n          fs.writeFileSync(fullPath, fileContent.package_file.content_text)\n        }\n\n        const npmrcPath = path.join(dirPath, \".npmrc\")\n        fs.writeFileSync(npmrcPath, \"@tsci:registry=https://npm.tscircuit.com\")\n\n        console.log(`Successfully cloned to ./${author}.${snippetName}/`)\n      } catch (error) {\n        if (error instanceof Error) {\n          console.error(\"Failed to clone snippet:\", error.message)\n        } else {\n          console.error(\"Failed to clone snippet:\", error)\n        }\n        process.exit(1)\n      }\n    })\n}\n", "import type { Command } from \"commander\"\nimport { createCircuitWebWorker } from \"@tscircuit/eval-webworker\"\nimport webWorkerBundleUrl from \"@tscircuit/eval-webworker/blob-url\"\nimport { getVirtualFileSystemFromDirPath } from \"make-vfs\"\nimport path from \"node:path\"\nimport fs from \"node:fs\"\n\nconst ALLOWED_FORMATS = [\n  \"json\",\n  \"circuit-json\",\n  \"schematic-svg\",\n  \"pcb-svg\",\n  \"gerbers\",\n  \"readable-netlist\",\n  \"gltf\",\n  \"specctra-dsn\",\n] as const\n\ntype Format = (typeof ALLOWED_FORMATS)[number]\n\nconst OUTPUT_EXTENSIONS = {\n  json: \".circuit.json\",\n  \"circuit-json\": \".circuit.json\",\n  \"schematic-svg\": \"-schematic.svg\",\n  \"pcb-svg\": \"-pcb.svg\",\n  gerbers: \"-gerbers.zip\",\n  \"readable-netlist\": \"-readable.netlist\",\n  gltf: \".gltf\",\n  \"specctra-dsn\": \".dsn\",\n}\n\nexport const registerExport = (program: Command) => {\n  program\n    .command(\"export\")\n    .description(\"Export tscircuit code to various formats\")\n    .argument(\"<file>\", \"Path to the snippet file\")\n    .option(\"-f, --format <format>\", \"Output format\")\n    .option(\"-o, --output <path>\", \"Output file path\")\n    .action(async (file, options) => {\n      const { format = \"circuit-json\" } = options\n      let { output } = options\n      if (!ALLOWED_FORMATS.includes(format)) {\n        throw new Error(\n          `Invalid format: ${format}\\nSupported formats: ${ALLOWED_FORMATS.join(\",\")}`,\n        )\n      }\n\n      if (!output) {\n        output = path.basename(file).replace(/\\.[^.]+$/, \"\")\n      }\n\n      const worker = await createCircuitWebWorker({\n        webWorkerUrl: webWorkerBundleUrl,\n      })\n\n      const projectDir = path.dirname(file)\n\n      const relativeComponentPath = path.relative(projectDir, file)\n\n      await worker.executeWithFsMap({\n        entrypoint: \"entrypoint.tsx\",\n        fsMap: {\n          ...((await getVirtualFileSystemFromDirPath({\n            dirPath: projectDir,\n            contentFormat: \"string\",\n          })) as Record<string, string>),\n          \"entrypoint.tsx\": `\nimport MyCircuit from \"./${relativeComponentPath}\"\n\ncircuit.add(<MyCircuit />)\n        `,\n        },\n      })\n\n      await worker.renderUntilSettled()\n\n      const circuitJson = await worker.getCircuitJson()\n\n      const outputPath = path.join(\n        projectDir,\n        `${output}${OUTPUT_EXTENSIONS[format as Format]}`,\n      )\n\n      fs.writeFileSync(outputPath, JSON.stringify(circuitJson))\n\n      console.log(`Exported to ${outputPath}`)\n\n      process.exit(0)\n    })\n}\n", "import type { Command } from \"commander\"\nimport { cliConfig } from \"lib/cli-config\"\n\nexport const registerAuthPrintToken = (program: Command) => {\n  program.commands\n    .find((c) => c.name() === \"auth\")!\n    .command(\"print-token\")\n    .description(\"Prints your auth token\")\n    .action(() => {\n      const token = cliConfig.get(\"sessionToken\")\n      if (!token) return console.log(\"You need to log in to access this.\")\n      console.log(\"Your Token:\\n\", token)\n    })\n}\n", "import type { Command } from \"commander\"\nimport { cliConfig } from \"lib/cli-config\"\n\nfunction validateJWTLength(token: string) {\n  const parts = token.split(\".\")\n\n  if (parts.length === 3 && parts.every((part) => part.length > 0)) {\n    return true\n  } else {\n    return false\n  }\n}\nexport const registerAuthSetToken = (program: Command) => {\n  program.commands\n    .find((c) => c.name() === \"auth\")!\n    .command(\"set-token\")\n    .description(\"Explicitly set your auth token\")\n    .argument(\"<token>\", \"New token to manually configure\")\n    .action((token) => {\n      if (!validateJWTLength(token))\n        return console.log(\"Invalid token provided\")\n      cliConfig.set(\"sessionToken\", token)\n      console.log(\"Token manually updated.\")\n    })\n}\n"],
  "mappings": "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",
  "names": ["program", "path", "fs", "fs", "path", "fs", "path", "resolve", "path", "fs", "fs", "path", "fs", "ts", "path", "fs", "program", "ky", "program", "ky", "program", "program", "program", "program", "fs", "path", "program", "ky", "path", "fs", "program", "program", "program"]
}

|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import * as fs from "node:fs"
|
|
2
|
+
import * as path from "node:path"
|
|
3
|
+
import { findImportsInSnippet } from "./findImportsInSnippet"
|
|
4
|
+
import { installNodeModuleTypesForSnippet } from "./installNodeModuleTypesForSnippet"
|
|
5
|
+
|
|
6
|
+
export class FilesystemTypesHandler {
|
|
7
|
+
private projectRoot: string
|
|
8
|
+
|
|
9
|
+
constructor(initialDir: string) {
|
|
10
|
+
this.projectRoot = this.findProjectRoot(initialDir)
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
async handleInitialTypeDependencies(filePath: string) {
|
|
14
|
+
console.log("Checking initial type dependencies...")
|
|
15
|
+
try {
|
|
16
|
+
if (!this.areTypesInstalled(filePath)) {
|
|
17
|
+
console.log("Installing missing initial types...")
|
|
18
|
+
await installNodeModuleTypesForSnippet(filePath)
|
|
19
|
+
}
|
|
20
|
+
} catch (error) {
|
|
21
|
+
console.warn("Error handling initial type dependencies:", error)
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
async handleFileTypeDependencies(filePath: string) {
|
|
26
|
+
try {
|
|
27
|
+
if (!this.areTypesInstalled(filePath)) {
|
|
28
|
+
console.log("Installing missing file types...")
|
|
29
|
+
await installNodeModuleTypesForSnippet(filePath)
|
|
30
|
+
}
|
|
31
|
+
} catch (error) {
|
|
32
|
+
console.warn("Failed to verify types:", error)
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
private areTypesInstalled(filePath: string): boolean {
|
|
37
|
+
const imports = findImportsInSnippet(filePath)
|
|
38
|
+
return imports.every((imp) => this.checkTypeExists(imp))
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
private checkTypeExists(importPath: string): boolean {
|
|
42
|
+
if (!importPath.startsWith("@tsci/")) return true
|
|
43
|
+
|
|
44
|
+
const pathWithoutPrefix = importPath.replace("@tsci/", "")
|
|
45
|
+
const [owner, name] = pathWithoutPrefix.split(".")
|
|
46
|
+
|
|
47
|
+
const typePath = path.join(
|
|
48
|
+
this.projectRoot,
|
|
49
|
+
"node_modules",
|
|
50
|
+
"@tsci",
|
|
51
|
+
`${owner}.${name}`,
|
|
52
|
+
"index.d.ts",
|
|
53
|
+
)
|
|
54
|
+
|
|
55
|
+
return fs.existsSync(typePath)
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
private findProjectRoot(startDir: string): string {
|
|
59
|
+
let root = path.resolve(startDir)
|
|
60
|
+
while (root !== path.parse(root).root) {
|
|
61
|
+
if (fs.existsSync(path.join(root, "package.json"))) {
|
|
62
|
+
return root
|
|
63
|
+
}
|
|
64
|
+
root = path.dirname(root)
|
|
65
|
+
}
|
|
66
|
+
return startDir
|
|
67
|
+
}
|
|
68
|
+
}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import * as fs from "node:fs"
|
|
2
|
+
import * as ts from "typescript"
|
|
3
|
+
|
|
4
|
+
export function findImportsInSnippet(snippetPath: string): string[] {
|
|
5
|
+
const content = fs.readFileSync(snippetPath, "utf-8")
|
|
6
|
+
const sourceFile = ts.createSourceFile(
|
|
7
|
+
snippetPath,
|
|
8
|
+
content,
|
|
9
|
+
ts.ScriptTarget.Latest,
|
|
10
|
+
true,
|
|
11
|
+
)
|
|
12
|
+
|
|
13
|
+
const imports: string[] = []
|
|
14
|
+
|
|
15
|
+
function visit(node: ts.Node) {
|
|
16
|
+
if (ts.isImportDeclaration(node)) {
|
|
17
|
+
const moduleSpecifier = node.moduleSpecifier
|
|
18
|
+
if (moduleSpecifier && ts.isStringLiteral(moduleSpecifier)) {
|
|
19
|
+
const importPath = moduleSpecifier.text
|
|
20
|
+
if (importPath.startsWith("@tsci/")) {
|
|
21
|
+
imports.push(importPath)
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
ts.forEachChild(node, visit)
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
visit(sourceFile)
|
|
29
|
+
|
|
30
|
+
return imports
|
|
31
|
+
}
|
package/package.json
CHANGED
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { test, expect, afterEach } from "bun:test"
|
|
2
|
+
import { DevServer } from "cli/dev/DevServer"
|
|
3
|
+
import { getTestFixture } from "tests/fixtures/get-test-fixture"
|
|
4
|
+
import fs from "node:fs"
|
|
5
|
+
import path from "node:path"
|
|
6
|
+
|
|
7
|
+
async function waitForFile(
|
|
8
|
+
filePath: string,
|
|
9
|
+
timeout: number = 5000,
|
|
10
|
+
interval: number = 500,
|
|
11
|
+
): Promise<boolean> {
|
|
12
|
+
const endTime = Date.now() + timeout
|
|
13
|
+
while (Date.now() < endTime) {
|
|
14
|
+
if (fs.existsSync(filePath)) {
|
|
15
|
+
return true
|
|
16
|
+
}
|
|
17
|
+
await new Promise((resolve) => setTimeout(resolve, interval))
|
|
18
|
+
}
|
|
19
|
+
return false
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
test("types are installed and refreshed when files change", async () => {
|
|
23
|
+
const { tempDirPath, devServerPort } = await getTestFixture({
|
|
24
|
+
vfs: {
|
|
25
|
+
"snippet.tsx": `
|
|
26
|
+
import { useRedLed } from "@tsci/seveibar.red-led"
|
|
27
|
+
export const MyCircuit = () => <></>
|
|
28
|
+
`,
|
|
29
|
+
"package.json": "{}",
|
|
30
|
+
},
|
|
31
|
+
})
|
|
32
|
+
|
|
33
|
+
const devServer = new DevServer({
|
|
34
|
+
port: devServerPort,
|
|
35
|
+
componentFilePath: `${tempDirPath}/snippet.tsx`,
|
|
36
|
+
})
|
|
37
|
+
|
|
38
|
+
await devServer.start()
|
|
39
|
+
|
|
40
|
+
// Wait for the initial type file to be installed
|
|
41
|
+
const typePath = path.join(
|
|
42
|
+
tempDirPath,
|
|
43
|
+
"node_modules/@tsci/seveibar.red-led/index.d.ts",
|
|
44
|
+
)
|
|
45
|
+
const typeFileExists = await waitForFile(typePath)
|
|
46
|
+
expect(typeFileExists).toBe(true)
|
|
47
|
+
|
|
48
|
+
// Simulate file change with new import
|
|
49
|
+
const updatedContent = `
|
|
50
|
+
import { useUsbC } from "@tsci/seveibar.smd-usb-c"
|
|
51
|
+
export const MyCircuit = () => <></>
|
|
52
|
+
`
|
|
53
|
+
fs.writeFileSync(`${tempDirPath}/snippet.tsx`, updatedContent)
|
|
54
|
+
|
|
55
|
+
// Wait for the new type file to be installed after update
|
|
56
|
+
const typePath2 = path.join(
|
|
57
|
+
tempDirPath,
|
|
58
|
+
"node_modules/@tsci/seveibar.smd-usb-c/index.d.ts",
|
|
59
|
+
)
|
|
60
|
+
const typeFileExists2 = await waitForFile(typePath2)
|
|
61
|
+
expect(typeFileExists2).toBe(true)
|
|
62
|
+
|
|
63
|
+
// Stop the dev server after the test
|
|
64
|
+
await devServer.stop()
|
|
65
|
+
}, 10_000)
|