@tscircuit/cli 0.1.3 → 0.1.5

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/biome.json CHANGED
@@ -32,6 +32,7 @@
32
32
  "noForEach": "off"
33
33
  },
34
34
  "style": {
35
+ "noUnusedTemplateLiteral": "off",
35
36
  "noUselessElse": "off",
36
37
  "noNonNullAssertion": "off",
37
38
  "useNumberNamespace": "off",
package/bun.lockb CHANGED
Binary file
@@ -0,0 +1,155 @@
1
+ import ky from "ky"
2
+ import type { FileServerRoutes } from "lib/file-server/FileServerRoutes"
3
+ import { createHttpServer } from "lib/server/createHttpServer"
4
+ import { EventsWatcher } from "lib/server/EventsWatcher"
5
+ import type http from "node:http"
6
+ import type { TypedKyInstance } from "typed-ky"
7
+ import path from "node:path"
8
+ import fs from "node:fs"
9
+ import type { FileUpdatedEvent } from "lib/file-server/FileServerEvent"
10
+ import * as chokidar from "chokidar"
11
+
12
+ export class DevServer {
13
+ port: number
14
+ /**
15
+ * The path to a component that exports a <board /> or <group /> component
16
+ */
17
+ componentFilePath: string
18
+
19
+ projectDir: string
20
+
21
+ /**
22
+ * The HTTP server that hosts the file server and event bus. You can use
23
+ * fsKy to communicate with the file server/event bus
24
+ */
25
+ httpServer?: http.Server
26
+ /**
27
+ * Watches for events on the event bus by polling `api/events/list`
28
+ */
29
+ eventsWatcher?: EventsWatcher
30
+ /**
31
+ * A ky instance that can be used to communicate with the file server and
32
+ * event bus
33
+ */
34
+ fsKy: TypedKyInstance<keyof FileServerRoutes, FileServerRoutes>
35
+ /**
36
+ * A chokidar instance that watches the project directory for file changes
37
+ */
38
+ filesystemWatcher?: chokidar.FSWatcher
39
+
40
+ constructor({
41
+ port,
42
+ componentFilePath,
43
+ }: {
44
+ port: number
45
+ componentFilePath: string
46
+ }) {
47
+ this.port = port
48
+ this.componentFilePath = componentFilePath
49
+ this.projectDir = path.dirname(componentFilePath)
50
+ this.fsKy = ky.create({
51
+ prefixUrl: `http://localhost:${port}`,
52
+ }) as any
53
+ }
54
+
55
+ async start() {
56
+ const { server } = await createHttpServer(this.port)
57
+ this.httpServer = server
58
+
59
+ this.eventsWatcher = new EventsWatcher(`http://localhost:${this.port}`)
60
+ this.eventsWatcher.start()
61
+
62
+ this.eventsWatcher.on(
63
+ "FILE_UPDATED",
64
+ this.handleFileUpdatedEventFromServer.bind(this),
65
+ )
66
+
67
+ this.filesystemWatcher = chokidar.watch(this.projectDir, {
68
+ persistent: true,
69
+ ignoreInitial: true,
70
+ })
71
+
72
+ this.filesystemWatcher.on("change", (filePath) =>
73
+ this.handleFileChangedOnFilesystem(filePath),
74
+ )
75
+ this.filesystemWatcher.on("add", (filePath) =>
76
+ this.handleFileChangedOnFilesystem(filePath),
77
+ )
78
+
79
+ this.upsertInitialFiles()
80
+ }
81
+
82
+ async addEntrypoint() {
83
+ const relativeComponentFilePath = path.relative(
84
+ this.projectDir,
85
+ this.componentFilePath,
86
+ )
87
+ await this.fsKy.post("api/files/upsert", {
88
+ json: {
89
+ file_path: "entrypoint.tsx",
90
+ text_content: `
91
+ import MyCircuit from "./${relativeComponentFilePath}"
92
+
93
+ circuit.add(<MyCircuit />)
94
+ `,
95
+ },
96
+ })
97
+ }
98
+
99
+ async handleFileUpdatedEventFromServer(ev: FileUpdatedEvent) {
100
+ if (ev.initiator === "filesystem_change") return
101
+
102
+ if (ev.file_path === "manual-edits.json") {
103
+ console.log("Manual edits updated, updating on filesystem...")
104
+ const { file } = await this.fsKy
105
+ .get("api/files/get", {
106
+ searchParams: { file_path: ev.file_path },
107
+ })
108
+ .json()
109
+ fs.writeFileSync(
110
+ path.join(this.projectDir, "manual-edits.json"),
111
+ file.text_content,
112
+ )
113
+ }
114
+ }
115
+
116
+ async handleFileChangedOnFilesystem(absoluteFilePath: string) {
117
+ const relativeFilePath = path.relative(this.projectDir, absoluteFilePath)
118
+
119
+ // We've temporarily disabled upserting manual edits from filesystem changes
120
+ // because it can be edited by the browser
121
+ if (relativeFilePath.includes("manual-edits.json")) return
122
+
123
+ await this.fsKy
124
+ .post("api/files/upsert", {
125
+ json: {
126
+ file_path: relativeFilePath,
127
+ text_content: fs.readFileSync(absoluteFilePath, "utf-8"),
128
+ initiator: "filesystem_change",
129
+ },
130
+ })
131
+ .json()
132
+ }
133
+
134
+ async upsertInitialFiles() {
135
+ // Scan project directory for all files and upsert them
136
+ const fileNames = fs.readdirSync(this.projectDir)
137
+ for (const fileName of fileNames) {
138
+ await this.fsKy.post("api/files/upsert", {
139
+ json: {
140
+ file_path: fileName,
141
+ text_content: fs.readFileSync(
142
+ path.join(this.projectDir, fileName),
143
+ "utf-8",
144
+ ),
145
+ initiator: "filesystem_change",
146
+ },
147
+ })
148
+ }
149
+ }
150
+
151
+ async stop() {
152
+ this.httpServer?.close()
153
+ this.eventsWatcher?.stop()
154
+ }
155
+ }
@@ -2,10 +2,11 @@ import type { Command } from "commander"
2
2
  import * as path from "node:path"
3
3
  import * as chokidar from "chokidar"
4
4
  import * as fs from "node:fs"
5
- import { createServer } from "lib/server/createServer"
5
+ import { createHttpServer } from "lib/server/createHttpServer"
6
6
  import { getLocalFileDependencies } from "lib/dependency-analysis/getLocalFileDependencies"
7
- import { installTypes } from "../../lib/dependency-analysis/installNodeModuleTypes"
7
+ import { installNodeModuleTypesForSnippet } from "../../lib/dependency-analysis/installNodeModuleTypesForSnippet"
8
8
  import { EventsWatcher } from "../../lib/server/EventsWatcher"
9
+ import { DevServer } from "./DevServer"
9
10
 
10
11
  export const registerDev = (program: Command) => {
11
12
  program
@@ -20,92 +21,18 @@ export const registerDev = (program: Command) => {
20
21
 
21
22
  try {
22
23
  console.log("Installing types for imported snippets...")
23
- await installTypes(absolutePath)
24
+ await installNodeModuleTypesForSnippet(absolutePath)
24
25
  console.log("Types installed successfully")
25
26
  } catch (error) {
26
27
  console.warn("Failed to install types:", error)
27
28
  }
28
29
 
29
- // Start the server
30
- await createServer(port)
31
-
32
- const eventsWatcher = new EventsWatcher(`http://localhost:${port}`)
33
- eventsWatcher.start()
34
-
35
- await fetch(`http://localhost:${port}/api/files/upsert`, {
36
- method: "POST",
37
- headers: { "Content-Type": "application/json" },
38
- body: JSON.stringify({
39
- file_path: "entrypoint.tsx",
40
- text_content: `
41
- import MyCircuit from "./snippet.tsx"
42
-
43
- circuit.add(<MyCircuit />)
44
- `,
45
- }),
46
- })
47
-
48
- // Function to update file content
49
- const updateFile = async (filePath: string) => {
50
- try {
51
- const content = await fs.promises.readFile(filePath, "utf-8")
52
- const response = await fetch(
53
- `http://localhost:${port}/api/files/upsert`,
54
- {
55
- method: "POST",
56
- headers: { "Content-Type": "application/json" },
57
- body: JSON.stringify({
58
- file_path: path.relative(fileDir, filePath),
59
- text_content: content,
60
- }),
61
- },
62
- )
63
- if (!response.ok) {
64
- console.error(`Failed to update ${filePath}`)
65
- }
66
- } catch (error) {
67
- console.error(`Error updating ${filePath}:`, error)
68
- }
69
- }
70
-
71
- // Get initial dependencies
72
- const dependencies = new Set([absolutePath])
73
- try {
74
- const deps = getLocalFileDependencies(absolutePath)
75
- deps.forEach((dep) => dependencies.add(dep))
76
- } catch (error) {
77
- console.warn("Failed to analyze dependencies:", error)
78
- }
79
-
80
- // Watch the main file and its dependencies
81
- const filesystemWatcher = chokidar.watch(Array.from(dependencies), {
82
- persistent: true,
83
- ignoreInitial: false,
84
- })
85
-
86
- filesystemWatcher.on("change", async (filePath) => {
87
- console.log(`File ${filePath} changed`)
88
- await updateFile(filePath)
89
- })
90
-
91
- filesystemWatcher.on("add", async (filePath) => {
92
- console.log(`File ${filePath} added`)
93
- await updateFile(filePath)
94
- })
95
-
96
- eventsWatcher.on("FILE_UPDATED", async (ev) => {
97
- if (ev.file_path === "manual-edits.json") {
98
- console.log("Manual edits updated, updating on filesystem...")
99
- const { file } = await fetch(
100
- `http://localhost:${port}/api/files/get?file_path=manual-edits.json`,
101
- ).then((r) => r.json())
102
- fs.writeFileSync(
103
- path.join(fileDir, "manual-edits.json"),
104
- file.text_content,
105
- )
106
- }
30
+ const server = new DevServer({
31
+ port,
32
+ componentFilePath: absolutePath,
107
33
  })
108
34
 
109
- console.log(`Watching ${file} and its dependencies...`)
35
+ await server.start()
36
+ await server.addEntrypoint()
110
37
  })
111
38
  }
package/cli/main.ts CHANGED
@@ -8,13 +8,14 @@ import { registerConfig } from "./config/register"
8
8
  import { registerConfigPrint } from "./config/print/register"
9
9
  import { registerClone } from "./clone/register"
10
10
  import { perfectCli } from "perfect-cli"
11
+ import pkg from "../package.json"
11
12
 
12
13
  const program = new Command()
13
14
 
14
15
  program
15
16
  .name("tsci")
16
17
  .description("CLI for developing tscircuit snippets")
17
- .version("1.0.0")
18
+ .version(pkg.version)
18
19
 
19
20
  registerDev(program)
20
21
  registerClone(program)
package/dist/main.js CHANGED
@@ -5,13 +5,74 @@ import { Command } from "commander";
5
5
 
6
6
  // cli/dev/register.ts
7
7
  import * as path4 from "node:path";
8
- import * as chokidar from "chokidar";
9
- import * as fs4 from "node:fs";
10
8
 
11
- // lib/server/createServer.ts
12
- import * as http from "node:http";
9
+ // lib/dependency-analysis/installNodeModuleTypesForSnippet.ts
13
10
  import * as fs from "node:fs";
14
11
  import * as path from "node:path";
12
+ import * as ts from "typescript";
13
+ async function installNodeModuleTypesForSnippet(snippetPath) {
14
+ const content = fs.readFileSync(snippetPath, "utf-8");
15
+ const sourceFile = ts.createSourceFile(
16
+ snippetPath,
17
+ content,
18
+ ts.ScriptTarget.Latest,
19
+ true
20
+ );
21
+ const imports = [];
22
+ function visit(node) {
23
+ if (ts.isImportDeclaration(node)) {
24
+ const moduleSpecifier = node.moduleSpecifier;
25
+ if (moduleSpecifier && ts.isStringLiteral(moduleSpecifier)) {
26
+ const importPath = moduleSpecifier.text;
27
+ if (importPath.startsWith("@tsci/")) {
28
+ imports.push(importPath);
29
+ }
30
+ }
31
+ }
32
+ ts.forEachChild(node, visit);
33
+ }
34
+ visit(sourceFile);
35
+ let projectRoot = path.dirname(snippetPath);
36
+ while (projectRoot !== path.parse(projectRoot).root) {
37
+ if (fs.existsSync(path.join(projectRoot, "package.json"))) {
38
+ break;
39
+ }
40
+ projectRoot = path.dirname(projectRoot);
41
+ }
42
+ for (const importPath of imports) {
43
+ const [owner, name] = importPath.replace("@tsci/", "").split(".");
44
+ try {
45
+ const response = await fetch(
46
+ `https://registry-api.tscircuit.com/snippets/get?owner_name=${owner}&unscoped_name=${name}`
47
+ );
48
+ if (!response.ok) {
49
+ console.warn(`Failed to fetch types for ${importPath}`);
50
+ continue;
51
+ }
52
+ const data = await response.json();
53
+ if (data.snippet.dts) {
54
+ const packageDir = path.join(
55
+ projectRoot,
56
+ "node_modules",
57
+ "@tsci",
58
+ `${owner}.${name}`
59
+ );
60
+ fs.mkdirSync(packageDir, { recursive: true });
61
+ fs.writeFileSync(path.join(packageDir, "index.d.ts"), data.snippet.dts);
62
+ }
63
+ } catch (error) {
64
+ console.warn(`Error fetching types for ${importPath}:`, error);
65
+ }
66
+ }
67
+ }
68
+
69
+ // cli/dev/DevServer.ts
70
+ import ky from "ky";
71
+
72
+ // lib/server/createHttpServer.ts
73
+ import * as http from "node:http";
74
+ import * as fs2 from "node:fs";
75
+ import * as path2 from "node:path";
15
76
  import { getNodeHandler } from "winterspec/adapters/node";
16
77
 
17
78
  // package.json
@@ -19,7 +80,7 @@ var package_default = {
19
80
  name: "@tscircuit/cli",
20
81
  main: "dist/main.js",
21
82
  type: "module",
22
- version: "0.1.2",
83
+ version: "0.1.4",
23
84
  bin: {
24
85
  tsci: "./dist/main.js"
25
86
  },
@@ -34,16 +95,19 @@ var package_default = {
34
95
  devDependencies: {
35
96
  "@biomejs/biome": "^1.9.4",
36
97
  "@tscircuit/core": "^0.0.249",
37
- "@types/bun": "latest",
98
+ "@types/bun": "^1.1.15",
38
99
  "@types/configstore": "^6.0.2",
39
100
  "@types/react": "^19.0.1",
40
- tsup: "^8.3.5"
101
+ "get-port": "^7.1.0",
102
+ tempy: "^3.1.0",
103
+ tsup: "^8.3.5",
104
+ "typed-ky": "^0.0.4"
41
105
  },
42
106
  peerDependencies: {
43
107
  typescript: "^5.0.0"
44
108
  },
45
109
  dependencies: {
46
- "@tscircuit/file-server": "^0.0.11",
110
+ "@tscircuit/file-server": "^0.0.13",
47
111
  "@tscircuit/runframe": "^0.0.47",
48
112
  chokidar: "^4.0.1",
49
113
  commander: "^12.1.0",
@@ -55,7 +119,7 @@ var package_default = {
55
119
  }
56
120
  };
57
121
 
58
- // lib/server/createServer.ts
122
+ // lib/server/createHttpServer.ts
59
123
  import winterspecBundle from "@tscircuit/file-server/dist/bundle.js";
60
124
 
61
125
  // lib/site/getIndex.ts
@@ -74,19 +138,19 @@ var getIndex = async () => {
74
138
  </html>`;
75
139
  };
76
140
 
77
- // lib/server/createServer.ts
78
- var createServer2 = async (port = 3e3) => {
141
+ // lib/server/createHttpServer.ts
142
+ var createHttpServer = async (port = 3e3) => {
79
143
  const fileServerHandler = getNodeHandler(winterspecBundle, {});
80
144
  const server = http.createServer(async (req, res) => {
81
145
  const url = new URL(req.url, `http://${req.headers.host}`);
82
146
  if (url.pathname === "/standalone.min.js") {
83
- const standaloneFilePath = process.env.RUNFRAME_STANDALONE_FILE_PATH || path.resolve(
147
+ const standaloneFilePath = process.env.RUNFRAME_STANDALONE_FILE_PATH || path2.resolve(
84
148
  process.cwd(),
85
149
  "node_modules",
86
150
  "@tscircuit/runframe/dist/standalone.min.js"
87
151
  );
88
152
  try {
89
- const content = fs.readFileSync(standaloneFilePath, "utf8");
153
+ const content = fs2.readFileSync(standaloneFilePath, "utf8");
90
154
  res.writeHead(200, {
91
155
  "Content-Type": "application/javascript; charset=utf-8"
92
156
  });
@@ -115,147 +179,14 @@ var createServer2 = async (port = 3e3) => {
115
179
  res.writeHead(404);
116
180
  res.end("Not found");
117
181
  });
118
- return new Promise((resolve4) => {
182
+ return new Promise((resolve3) => {
119
183
  server.listen(port, () => {
120
184
  console.log(`Server running at http://localhost:${port}`);
121
- resolve4();
185
+ resolve3({ server });
122
186
  });
123
187
  });
124
188
  };
125
189
 
126
- // lib/dependency-analysis/getLocalFileDependencies.ts
127
- import * as ts from "typescript";
128
- import * as path2 from "path";
129
- import * as fs2 from "fs";
130
- function getLocalFileDependencies(pathToTsxFile) {
131
- const absolutePath = path2.resolve(pathToTsxFile);
132
- const baseDir = path2.dirname(absolutePath);
133
- const content = fs2.readFileSync(absolutePath, "utf-8");
134
- const sourceFile = ts.createSourceFile(
135
- absolutePath,
136
- content,
137
- ts.ScriptTarget.Latest,
138
- true
139
- );
140
- const dependencies = /* @__PURE__ */ new Set();
141
- function visit(node) {
142
- if (ts.isImportDeclaration(node) || ts.isExportDeclaration(node)) {
143
- const moduleSpecifier = node.moduleSpecifier;
144
- if (moduleSpecifier && ts.isStringLiteral(moduleSpecifier)) {
145
- const importPath = moduleSpecifier.text;
146
- if (importPath.startsWith(".")) {
147
- resolveAndAddDependency(importPath);
148
- }
149
- }
150
- }
151
- if (ts.isCallExpression(node) && node.expression.kind === ts.SyntaxKind.ImportKeyword) {
152
- const argument = node.arguments[0];
153
- if (argument && ts.isStringLiteral(argument)) {
154
- const importPath = argument.text;
155
- if (importPath.startsWith(".")) {
156
- resolveAndAddDependency(importPath);
157
- }
158
- }
159
- }
160
- ts.forEachChild(node, visit);
161
- }
162
- function resolveAndAddDependency(importPath) {
163
- const extensions = [
164
- ".tsx",
165
- ".ts",
166
- ".jsx",
167
- ".js",
168
- ".css",
169
- ".scss",
170
- ".sass",
171
- ".less"
172
- ];
173
- let resolvedPath = path2.resolve(baseDir, importPath);
174
- if (fs2.existsSync(resolvedPath)) {
175
- dependencies.add(resolvedPath);
176
- return;
177
- }
178
- for (const ext of extensions) {
179
- const pathWithExt = resolvedPath + ext;
180
- if (fs2.existsSync(pathWithExt)) {
181
- dependencies.add(pathWithExt);
182
- return;
183
- }
184
- }
185
- if (fs2.existsSync(resolvedPath) && fs2.statSync(resolvedPath).isDirectory()) {
186
- for (const ext of extensions) {
187
- const indexPath = path2.join(resolvedPath, `index${ext}`);
188
- if (fs2.existsSync(indexPath)) {
189
- dependencies.add(indexPath);
190
- return;
191
- }
192
- }
193
- }
194
- }
195
- visit(sourceFile);
196
- return Array.from(dependencies);
197
- }
198
-
199
- // lib/dependency-analysis/installNodeModuleTypes.ts
200
- import * as fs3 from "node:fs";
201
- import * as path3 from "node:path";
202
- import * as ts2 from "typescript";
203
- async function installTypes(snippetPath) {
204
- const content = fs3.readFileSync(snippetPath, "utf-8");
205
- const sourceFile = ts2.createSourceFile(
206
- snippetPath,
207
- content,
208
- ts2.ScriptTarget.Latest,
209
- true
210
- );
211
- const imports = [];
212
- function visit(node) {
213
- if (ts2.isImportDeclaration(node)) {
214
- const moduleSpecifier = node.moduleSpecifier;
215
- if (moduleSpecifier && ts2.isStringLiteral(moduleSpecifier)) {
216
- const importPath = moduleSpecifier.text;
217
- if (importPath.startsWith("@tsci/")) {
218
- imports.push(importPath);
219
- }
220
- }
221
- }
222
- ts2.forEachChild(node, visit);
223
- }
224
- visit(sourceFile);
225
- let projectRoot = path3.dirname(snippetPath);
226
- while (projectRoot !== path3.parse(projectRoot).root) {
227
- if (fs3.existsSync(path3.join(projectRoot, "package.json"))) {
228
- break;
229
- }
230
- projectRoot = path3.dirname(projectRoot);
231
- }
232
- for (const importPath of imports) {
233
- const [owner, name] = importPath.replace("@tsci/", "").split(".");
234
- try {
235
- const response = await fetch(
236
- `https://registry-api.tscircuit.com/snippets/get?owner_name=${owner}&unscoped_name=${name}`
237
- );
238
- if (!response.ok) {
239
- console.warn(`Failed to fetch types for ${importPath}`);
240
- continue;
241
- }
242
- const data = await response.json();
243
- if (data.snippet.dts) {
244
- const packageDir = path3.join(
245
- projectRoot,
246
- "node_modules",
247
- "@tsci",
248
- `${owner}.${name}`
249
- );
250
- fs3.mkdirSync(packageDir, { recursive: true });
251
- fs3.writeFileSync(path3.join(packageDir, "index.d.ts"), data.snippet.dts);
252
- }
253
- } catch (error) {
254
- console.warn(`Error fetching types for ${importPath}:`, error);
255
- }
256
- }
257
- }
258
-
259
190
  // lib/server/EventsWatcher.ts
260
191
  import { EventEmitter } from "events";
261
192
  var EventsWatcher = class extends EventEmitter {
@@ -307,6 +238,130 @@ var EventsWatcher = class extends EventEmitter {
307
238
  }
308
239
  };
309
240
 
241
+ // cli/dev/DevServer.ts
242
+ import path3 from "node:path";
243
+ import fs3 from "node:fs";
244
+ import * as chokidar from "chokidar";
245
+ var DevServer = class {
246
+ port;
247
+ /**
248
+ * The path to a component that exports a <board /> or <group /> component
249
+ */
250
+ componentFilePath;
251
+ projectDir;
252
+ /**
253
+ * The HTTP server that hosts the file server and event bus. You can use
254
+ * fsKy to communicate with the file server/event bus
255
+ */
256
+ httpServer;
257
+ /**
258
+ * Watches for events on the event bus by polling `api/events/list`
259
+ */
260
+ eventsWatcher;
261
+ /**
262
+ * A ky instance that can be used to communicate with the file server and
263
+ * event bus
264
+ */
265
+ fsKy;
266
+ /**
267
+ * A chokidar instance that watches the project directory for file changes
268
+ */
269
+ filesystemWatcher;
270
+ constructor({
271
+ port,
272
+ componentFilePath
273
+ }) {
274
+ this.port = port;
275
+ this.componentFilePath = componentFilePath;
276
+ this.projectDir = path3.dirname(componentFilePath);
277
+ this.fsKy = ky.create({
278
+ prefixUrl: `http://localhost:${port}`
279
+ });
280
+ }
281
+ async start() {
282
+ const { server } = await createHttpServer(this.port);
283
+ this.httpServer = server;
284
+ this.eventsWatcher = new EventsWatcher(`http://localhost:${this.port}`);
285
+ this.eventsWatcher.start();
286
+ this.eventsWatcher.on(
287
+ "FILE_UPDATED",
288
+ this.handleFileUpdatedEventFromServer.bind(this)
289
+ );
290
+ this.filesystemWatcher = chokidar.watch(this.projectDir, {
291
+ persistent: true,
292
+ ignoreInitial: true
293
+ });
294
+ this.filesystemWatcher.on(
295
+ "change",
296
+ (filePath) => this.handleFileChangedOnFilesystem(filePath)
297
+ );
298
+ this.filesystemWatcher.on(
299
+ "add",
300
+ (filePath) => this.handleFileChangedOnFilesystem(filePath)
301
+ );
302
+ this.upsertInitialFiles();
303
+ }
304
+ async addEntrypoint() {
305
+ const relativeComponentFilePath = path3.relative(
306
+ this.projectDir,
307
+ this.componentFilePath
308
+ );
309
+ await this.fsKy.post("api/files/upsert", {
310
+ json: {
311
+ file_path: "entrypoint.tsx",
312
+ text_content: `
313
+ import MyCircuit from "./${relativeComponentFilePath}"
314
+
315
+ circuit.add(<MyCircuit />)
316
+ `
317
+ }
318
+ });
319
+ }
320
+ async handleFileUpdatedEventFromServer(ev) {
321
+ if (ev.initiator === "filesystem_change") return;
322
+ if (ev.file_path === "manual-edits.json") {
323
+ console.log("Manual edits updated, updating on filesystem...");
324
+ const { file } = await this.fsKy.get("api/files/get", {
325
+ searchParams: { file_path: ev.file_path }
326
+ }).json();
327
+ fs3.writeFileSync(
328
+ path3.join(this.projectDir, "manual-edits.json"),
329
+ file.text_content
330
+ );
331
+ }
332
+ }
333
+ async handleFileChangedOnFilesystem(absoluteFilePath) {
334
+ const relativeFilePath = path3.relative(this.projectDir, absoluteFilePath);
335
+ if (relativeFilePath.includes("manual-edits.json")) return;
336
+ await this.fsKy.post("api/files/upsert", {
337
+ json: {
338
+ file_path: relativeFilePath,
339
+ text_content: fs3.readFileSync(absoluteFilePath, "utf-8"),
340
+ initiator: "filesystem_change"
341
+ }
342
+ }).json();
343
+ }
344
+ async upsertInitialFiles() {
345
+ const fileNames = fs3.readdirSync(this.projectDir);
346
+ for (const fileName of fileNames) {
347
+ await this.fsKy.post("api/files/upsert", {
348
+ json: {
349
+ file_path: fileName,
350
+ text_content: fs3.readFileSync(
351
+ path3.join(this.projectDir, fileName),
352
+ "utf-8"
353
+ ),
354
+ initiator: "filesystem_change"
355
+ }
356
+ });
357
+ }
358
+ }
359
+ async stop() {
360
+ this.httpServer?.close();
361
+ this.eventsWatcher?.stop();
362
+ }
363
+ };
364
+
310
365
  // cli/dev/register.ts
311
366
  var registerDev = (program2) => {
312
367
  program2.command("dev").description("Start development server for a snippet").argument("<file>", "Path to the snippet file").option("-p, --port <number>", "Port to run server on", "3000").action(async (file, options) => {
@@ -315,79 +370,17 @@ var registerDev = (program2) => {
315
370
  const port = parseInt(options.port);
316
371
  try {
317
372
  console.log("Installing types for imported snippets...");
318
- await installTypes(absolutePath);
373
+ await installNodeModuleTypesForSnippet(absolutePath);
319
374
  console.log("Types installed successfully");
320
375
  } catch (error) {
321
376
  console.warn("Failed to install types:", error);
322
377
  }
323
- await createServer2(port);
324
- const eventsWatcher = new EventsWatcher(`http://localhost:${port}`);
325
- eventsWatcher.start();
326
- await fetch(`http://localhost:${port}/api/files/upsert`, {
327
- method: "POST",
328
- headers: { "Content-Type": "application/json" },
329
- body: JSON.stringify({
330
- file_path: "entrypoint.tsx",
331
- text_content: `
332
- import MyCircuit from "./snippet.tsx"
333
-
334
- circuit.add(<MyCircuit />)
335
- `
336
- })
337
- });
338
- const updateFile = async (filePath) => {
339
- try {
340
- const content = await fs4.promises.readFile(filePath, "utf-8");
341
- const response = await fetch(
342
- `http://localhost:${port}/api/files/upsert`,
343
- {
344
- method: "POST",
345
- headers: { "Content-Type": "application/json" },
346
- body: JSON.stringify({
347
- file_path: path4.relative(fileDir, filePath),
348
- text_content: content
349
- })
350
- }
351
- );
352
- if (!response.ok) {
353
- console.error(`Failed to update ${filePath}`);
354
- }
355
- } catch (error) {
356
- console.error(`Error updating ${filePath}:`, error);
357
- }
358
- };
359
- const dependencies = /* @__PURE__ */ new Set([absolutePath]);
360
- try {
361
- const deps = getLocalFileDependencies(absolutePath);
362
- deps.forEach((dep) => dependencies.add(dep));
363
- } catch (error) {
364
- console.warn("Failed to analyze dependencies:", error);
365
- }
366
- const filesystemWatcher = chokidar.watch(Array.from(dependencies), {
367
- persistent: true,
368
- ignoreInitial: false
369
- });
370
- filesystemWatcher.on("change", async (filePath) => {
371
- console.log(`File ${filePath} changed`);
372
- await updateFile(filePath);
373
- });
374
- filesystemWatcher.on("add", async (filePath) => {
375
- console.log(`File ${filePath} added`);
376
- await updateFile(filePath);
378
+ const server = new DevServer({
379
+ port,
380
+ componentFilePath: absolutePath
377
381
  });
378
- eventsWatcher.on("FILE_UPDATED", async (ev) => {
379
- if (ev.file_path === "manual-edits.json") {
380
- console.log("Manual edits updated, updating on filesystem...");
381
- const { file: file2 } = await fetch(
382
- `http://localhost:${port}/api/files/get?file_path=manual-edits.json`
383
- ).then((r) => r.json());
384
- fs4.writeFileSync(
385
- path4.join(fileDir, "manual-edits.json"),
386
- file2.text_content
387
- );
388
- }
389
- });
390
- console.log(`Watching ${file} and its dependencies...`);
382
+ await server.start();
383
+ await server.addEntrypoint();
391
384
  });
392
385
  };
393
386
 
@@ -404,7 +397,7 @@ var getRegistryApiUrl = () => {
404
397
  import delay from "delay";
405
398
 
406
399
  // lib/registry-api/get-ky.ts
407
- import ky from "ky";
400
+ import ky2 from "ky";
408
401
  var prettyResponseErrorHook = async (_request, _options, response) => {
409
402
  if (!response.ok) {
410
403
  try {
@@ -419,7 +412,7 @@ var prettyResponseErrorHook = async (_request, _options, response) => {
419
412
  }
420
413
  };
421
414
  var getKy = () => {
422
- return ky.create({
415
+ return ky2.create({
423
416
  prefixUrl: getRegistryApiUrl(),
424
417
  hooks: {
425
418
  afterResponse: [prettyResponseErrorHook]
@@ -430,8 +423,8 @@ var getKy = () => {
430
423
  // cli/auth/login/register.ts
431
424
  var registerAuthLogin = (program2) => {
432
425
  program2.commands.find((c) => c.name() === "auth").command("login").description("Authenticate CLI, login to registry").action(async (args) => {
433
- const ky2 = getKy();
434
- const { login_page } = await ky2.post(
426
+ const ky3 = getKy();
427
+ const { login_page } = await ky3.post(
435
428
  "sessions/login_page/create",
436
429
  {
437
430
  json: {}
@@ -440,7 +433,7 @@ var registerAuthLogin = (program2) => {
440
433
  console.log("Please visit the following URL to log in:");
441
434
  console.log(login_page.url);
442
435
  while (true) {
443
- const { login_page: new_login_page } = await ky2.post(
436
+ const { login_page: new_login_page } = await ky3.post(
444
437
  "sessions/login_page/get",
445
438
  {
446
439
  json: {
@@ -460,7 +453,7 @@ var registerAuthLogin = (program2) => {
460
453
  }
461
454
  await delay(1e3);
462
455
  }
463
- const { session } = await ky2.post(
456
+ const { session } = await ky3.post(
464
457
  "sessions/login_page/exchange_for_cli_session",
465
458
  {
466
459
  json: {
@@ -501,7 +494,7 @@ var registerConfigPrint = (program2) => {
501
494
  };
502
495
 
503
496
  // cli/clone/register.ts
504
- import * as fs5 from "node:fs";
497
+ import * as fs4 from "node:fs";
505
498
  import * as path5 from "node:path";
506
499
  var registerClone = (program2) => {
507
500
  program2.command("clone").description("Clone a snippet from the registry").argument("<snippet>", "Snippet to clone (e.g. author/snippetName)").action(async (snippetPath) => {
@@ -522,23 +515,23 @@ var registerClone = (program2) => {
522
515
  );
523
516
  process.exit(1);
524
517
  }
525
- const ky2 = getKy();
518
+ const ky3 = getKy();
526
519
  try {
527
520
  console.log(`Cloning ${author}/${snippetName}...`);
528
- const packageFileList = await ky2.post("package_files/list", {
521
+ const packageFileList = await ky3.post("package_files/list", {
529
522
  json: {
530
523
  package_name: `${author}/${snippetName}`,
531
524
  use_latest_version: true
532
525
  }
533
526
  }).json();
534
527
  const dirPath = `./${author}.${snippetName}`;
535
- if (!fs5.existsSync(dirPath)) {
536
- fs5.mkdirSync(dirPath);
528
+ if (!fs4.existsSync(dirPath)) {
529
+ fs4.mkdirSync(dirPath);
537
530
  }
538
531
  for (const fileInfo of packageFileList.package_files) {
539
532
  const filePath = fileInfo.file_path.startsWith("/") ? fileInfo.file_path.slice(1) : fileInfo.file_path;
540
533
  if (filePath.startsWith("dist/")) continue;
541
- const fileContent = await ky2.post("package_files/get", {
534
+ const fileContent = await ky3.post("package_files/get", {
542
535
  json: {
543
536
  package_name: `${author}/${snippetName}`,
544
537
  file_path: fileInfo.file_path
@@ -546,10 +539,10 @@ var registerClone = (program2) => {
546
539
  }).json();
547
540
  const fullPath = path5.join(dirPath, filePath);
548
541
  const dirName = path5.dirname(fullPath);
549
- if (!fs5.existsSync(dirName)) {
550
- fs5.mkdirSync(dirName, { recursive: true });
542
+ if (!fs4.existsSync(dirName)) {
543
+ fs4.mkdirSync(dirName, { recursive: true });
551
544
  }
552
- fs5.writeFileSync(fullPath, fileContent.package_file.content_text);
545
+ fs4.writeFileSync(fullPath, fileContent.package_file.content_text);
553
546
  }
554
547
  console.log(`Successfully cloned to ./${author}.${snippetName}/`);
555
548
  } catch (error) {
@@ -566,7 +559,7 @@ var registerClone = (program2) => {
566
559
  // cli/main.ts
567
560
  import { perfectCli } from "perfect-cli";
568
561
  var program = new Command();
569
- program.name("tsci").description("CLI for developing tscircuit snippets").version("1.0.0");
562
+ program.name("tsci").description("CLI for developing tscircuit snippets").version(package_default.version);
570
563
  registerDev(program);
571
564
  registerClone(program);
572
565
  registerAuth(program);
@@ -579,4 +572,4 @@ if (process.argv.length === 2) {
579
572
  } else {
580
573
  program.parse();
581
574
  }
582
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../cli/main.ts", "../cli/dev/register.ts", "../lib/server/createServer.ts", "../package.json", "../lib/site/getIndex.ts", "../lib/dependency-analysis/getLocalFileDependencies.ts", "../lib/dependency-analysis/installNodeModuleTypes.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"],
  "sourcesContent": ["#!/usr/bin/env node\nimport { Command } from \"commander\"\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\"\n\nconst program = new Command()\n\nprogram\n  .name(\"tsci\")\n  .description(\"CLI for developing tscircuit snippets\")\n  .version(\"1.0.0\")\n\nregisterDev(program)\nregisterClone(program)\n\nregisterAuth(program)\nregisterAuthLogin(program)\nregisterAuthLogout(program)\n\nregisterConfig(program)\nregisterConfigPrint(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 path from \"node:path\"\nimport * as chokidar from \"chokidar\"\nimport * as fs from \"node:fs\"\nimport { createServer } from \"lib/server/createServer\"\nimport { getLocalFileDependencies } from \"lib/dependency-analysis/getLocalFileDependencies\"\nimport { installTypes } from \"../../lib/dependency-analysis/installNodeModuleTypes\"\nimport { EventsWatcher } from \"../../lib/server/EventsWatcher\"\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 absolutePath = path.resolve(file)\n      const fileDir = path.dirname(absolutePath)\n      const port = parseInt(options.port)\n\n      try {\n        console.log(\"Installing types for imported snippets...\")\n        await installTypes(absolutePath)\n        console.log(\"Types installed successfully\")\n      } catch (error) {\n        console.warn(\"Failed to install types:\", error)\n      }\n\n      // Start the server\n      await createServer(port)\n\n      const eventsWatcher = new EventsWatcher(`http://localhost:${port}`)\n      eventsWatcher.start()\n\n      await fetch(`http://localhost:${port}/api/files/upsert`, {\n        method: \"POST\",\n        headers: { \"Content-Type\": \"application/json\" },\n        body: JSON.stringify({\n          file_path: \"entrypoint.tsx\",\n          text_content: `\nimport MyCircuit from \"./snippet.tsx\"\n\ncircuit.add(<MyCircuit />)\n`,\n        }),\n      })\n\n      // Function to update file content\n      const updateFile = async (filePath: string) => {\n        try {\n          const content = await fs.promises.readFile(filePath, \"utf-8\")\n          const response = await fetch(\n            `http://localhost:${port}/api/files/upsert`,\n            {\n              method: \"POST\",\n              headers: { \"Content-Type\": \"application/json\" },\n              body: JSON.stringify({\n                file_path: path.relative(fileDir, filePath),\n                text_content: content,\n              }),\n            },\n          )\n          if (!response.ok) {\n            console.error(`Failed to update ${filePath}`)\n          }\n        } catch (error) {\n          console.error(`Error updating ${filePath}:`, error)\n        }\n      }\n\n      // Get initial dependencies\n      const dependencies = new Set([absolutePath])\n      try {\n        const deps = getLocalFileDependencies(absolutePath)\n        deps.forEach((dep) => dependencies.add(dep))\n      } catch (error) {\n        console.warn(\"Failed to analyze dependencies:\", error)\n      }\n\n      // Watch the main file and its dependencies\n      const filesystemWatcher = chokidar.watch(Array.from(dependencies), {\n        persistent: true,\n        ignoreInitial: false,\n      })\n\n      filesystemWatcher.on(\"change\", async (filePath) => {\n        console.log(`File ${filePath} changed`)\n        await updateFile(filePath)\n      })\n\n      filesystemWatcher.on(\"add\", async (filePath) => {\n        console.log(`File ${filePath} added`)\n        await updateFile(filePath)\n      })\n\n      eventsWatcher.on(\"FILE_UPDATED\", async (ev) => {\n        if (ev.file_path === \"manual-edits.json\") {\n          console.log(\"Manual edits updated, updating on filesystem...\")\n          const { file } = await fetch(\n            `http://localhost:${port}/api/files/get?file_path=manual-edits.json`,\n          ).then((r) => r.json())\n          fs.writeFileSync(\n            path.join(fileDir, \"manual-edits.json\"),\n            file.text_content,\n          )\n        }\n      })\n\n      console.log(`Watching ${file} and its dependencies...`)\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 createServer = async (port: number = 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<void>((resolve) => {\n    server.listen(port, () => {\n      console.log(`Server running at http://localhost:${port}`)\n      resolve()\n    })\n  })\n}\n", "{\n  \"name\": \"@tscircuit/cli\",\n  \"main\": \"dist/main.js\",\n  \"type\": \"module\",\n  \"version\": \"0.1.2\",\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/snippet.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\": \"latest\",\n    \"@types/configstore\": \"^6.0.2\",\n    \"@types/react\": \"^19.0.1\",\n    \"tsup\": \"^8.3.5\"\n  },\n  \"peerDependencies\": {\n    \"typescript\": \"^5.0.0\"\n  },\n  \"dependencies\": {\n    \"@tscircuit/file-server\": \"^0.0.11\",\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    \"perfect-cli\": \"^1.0.20\"\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 * as ts from \"typescript\"\nimport * as path from \"path\"\nimport * as fs from \"fs\"\n\nfunction getLocalFileDependencies(pathToTsxFile: string): string[] {\n  // Ensure absolute path\n  const absolutePath = path.resolve(pathToTsxFile)\n  const baseDir = path.dirname(absolutePath)\n\n  // Read and parse the file\n  const content = fs.readFileSync(absolutePath, \"utf-8\")\n  const sourceFile = ts.createSourceFile(\n    absolutePath,\n    content,\n    ts.ScriptTarget.Latest,\n    true,\n  )\n\n  const dependencies = new Set<string>()\n\n  // Recursively visit nodes to find imports\n  function visit(node: ts.Node) {\n    if (ts.isImportDeclaration(node) || ts.isExportDeclaration(node)) {\n      const moduleSpecifier = node.moduleSpecifier\n      if (moduleSpecifier && ts.isStringLiteral(moduleSpecifier)) {\n        const importPath = moduleSpecifier.text\n        // Only process local imports (starting with . or ..)\n        if (importPath.startsWith(\".\")) {\n          resolveAndAddDependency(importPath)\n        }\n      }\n    }\n\n    // Handle dynamic imports\n    if (\n      ts.isCallExpression(node) &&\n      node.expression.kind === ts.SyntaxKind.ImportKeyword\n    ) {\n      const argument = node.arguments[0]\n      if (argument && ts.isStringLiteral(argument)) {\n        const importPath = argument.text\n        if (importPath.startsWith(\".\")) {\n          resolveAndAddDependency(importPath)\n        }\n      }\n    }\n\n    ts.forEachChild(node, visit)\n  }\n\n  // Helper to resolve and add dependency paths\n  function resolveAndAddDependency(importPath: string) {\n    const extensions = [\n      \".tsx\",\n      \".ts\",\n      \".jsx\",\n      \".js\",\n      \".css\",\n      \".scss\",\n      \".sass\",\n      \".less\",\n    ]\n    let resolvedPath = path.resolve(baseDir, importPath)\n\n    // Check if path exists as-is\n    if (fs.existsSync(resolvedPath)) {\n      dependencies.add(resolvedPath)\n      return\n    }\n\n    // Try with extensions\n    for (const ext of extensions) {\n      const pathWithExt = resolvedPath + ext\n      if (fs.existsSync(pathWithExt)) {\n        dependencies.add(pathWithExt)\n        return\n      }\n    }\n\n    // Check for index files in directories\n    if (\n      fs.existsSync(resolvedPath) &&\n      fs.statSync(resolvedPath).isDirectory()\n    ) {\n      for (const ext of extensions) {\n        const indexPath = path.join(resolvedPath, `index${ext}`)\n        if (fs.existsSync(indexPath)) {\n          dependencies.add(indexPath)\n          return\n        }\n      }\n    }\n  }\n\n  // Start the traversal\n  visit(sourceFile)\n\n  return Array.from(dependencies)\n}\n\nexport { getLocalFileDependencies }\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 installTypes(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 { 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  program.commands\n    .find((c) => c.name() === \"auth\")!\n    .command(\"login\")\n    .description(\"Authenticate CLI, login to registry\")\n    .action(async (args) => {\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", "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"],
  "mappings": "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",
  "names": ["path", "fs", "createServer", "resolve", "path", "fs", "fs", "path", "ts", "program", "createServer", "file", "program", "ky", "program", "program", "program", "program", "fs", "path", "program", "ky"]
}

575
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../cli/main.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"],
  "sourcesContent": ["#!/usr/bin/env node\nimport { Command } from \"commander\"\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\"\n\nconst program = new Command()\n\nprogram\n  .name(\"tsci\")\n  .description(\"CLI for developing tscircuit snippets\")\n  .version(pkg.version)\n\nregisterDev(program)\nregisterClone(program)\n\nregisterAuth(program)\nregisterAuthLogin(program)\nregisterAuthLogout(program)\n\nregisterConfig(program)\nregisterConfigPrint(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 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 absolutePath = path.resolve(file)\n      const fileDir = path.dirname(absolutePath)\n      const port = parseInt(options.port)\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\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.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      await this.fsKy.post(\"api/files/upsert\", {\n        json: {\n          file_path: fileName,\n          text_content: fs.readFileSync(\n            path.join(this.projectDir, fileName),\n            \"utf-8\",\n          ),\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.4\",\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/snippet.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    \"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    \"perfect-cli\": \"^1.0.20\"\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  program.commands\n    .find((c) => c.name() === \"auth\")!\n    .command(\"login\")\n    .description(\"Authenticate CLI, login to registry\")\n    .action(async (args) => {\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", "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"],
  "mappings": "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",
  "names": ["path", "fs", "path", "resolve", "path", "fs", "program", "ky", "program", "ky", "program", "program", "program", "program", "fs", "path", "program", "ky"]
}

@@ -8,7 +8,7 @@ interface SnippetApiResponse {
8
8
  }
9
9
  }
10
10
 
11
- export async function installTypes(snippetPath: string) {
11
+ export async function installNodeModuleTypesForSnippet(snippetPath: string) {
12
12
  const content = fs.readFileSync(snippetPath, "utf-8")
13
13
  const sourceFile = ts.createSourceFile(
14
14
  snippetPath,
@@ -0,0 +1,7 @@
1
+ export interface FileUpdatedEvent {
2
+ event_id: string
3
+ event_type: "FILE_UPDATED"
4
+ file_path: string
5
+ initiator?: "filesystem_change" | "browser_edit"
6
+ created_at: string
7
+ }
@@ -0,0 +1,38 @@
1
+ export interface FileServerRoutes {
2
+ "api/files/get": {
3
+ GET: {
4
+ searchParams: {
5
+ file_path: string
6
+ }
7
+ responseJson: {
8
+ file: {
9
+ file_id: string
10
+ file_path: string
11
+ text_content: string
12
+ }
13
+ }
14
+ }
15
+ }
16
+ "api/files/upsert": {
17
+ POST: {
18
+ requestJson: {
19
+ file_path: string
20
+ text_content: string
21
+ initiator?: "filesystem_change"
22
+ }
23
+ responseJson: {
24
+ file: {
25
+ file_id: string
26
+ file_path: string
27
+ }
28
+ }
29
+ }
30
+ }
31
+ "api/files/list": {
32
+ GET: {
33
+ responseJson: {
34
+ file_list: { file_id: string; file_path: string }[]
35
+ }
36
+ }
37
+ }
38
+ }
package/lib/index.ts CHANGED
@@ -1,2 +1,2 @@
1
- export { createServer } from "./server/createServer"
1
+ export { createHttpServer } from "./server/createHttpServer"
2
2
  export { getLocalFileDependencies } from "./dependency-analysis/getLocalFileDependencies"
@@ -8,7 +8,7 @@ import pkg from "../../package.json"
8
8
  import winterspecBundle from "@tscircuit/file-server/dist/bundle.js"
9
9
  import { getIndex } from "../site/getIndex"
10
10
 
11
- export const createServer = async (port: number = 3000) => {
11
+ export const createHttpServer = async (port = 3000) => {
12
12
  const fileServerHandler = getNodeHandler(winterspecBundle as any, {})
13
13
 
14
14
  const server = http.createServer(async (req, res) => {
@@ -58,10 +58,10 @@ export const createServer = async (port: number = 3000) => {
58
58
  res.end("Not found")
59
59
  })
60
60
 
61
- return new Promise<void>((resolve) => {
61
+ return new Promise<{ server: http.Server }>((resolve) => {
62
62
  server.listen(port, () => {
63
63
  console.log(`Server running at http://localhost:${port}`)
64
- resolve()
64
+ resolve({ server })
65
65
  })
66
66
  })
67
67
  }
package/package.json CHANGED
@@ -2,7 +2,7 @@
2
2
  "name": "@tscircuit/cli",
3
3
  "main": "dist/main.js",
4
4
  "type": "module",
5
- "version": "0.1.3",
5
+ "version": "0.1.5",
6
6
  "bin": {
7
7
  "tsci": "./dist/main.js"
8
8
  },
@@ -17,16 +17,19 @@
17
17
  "devDependencies": {
18
18
  "@biomejs/biome": "^1.9.4",
19
19
  "@tscircuit/core": "^0.0.249",
20
- "@types/bun": "latest",
20
+ "@types/bun": "^1.1.15",
21
21
  "@types/configstore": "^6.0.2",
22
22
  "@types/react": "^19.0.1",
23
- "tsup": "^8.3.5"
23
+ "get-port": "^7.1.0",
24
+ "tempy": "^3.1.0",
25
+ "tsup": "^8.3.5",
26
+ "typed-ky": "^0.0.4"
24
27
  },
25
28
  "peerDependencies": {
26
29
  "typescript": "^5.0.0"
27
30
  },
28
31
  "dependencies": {
29
- "@tscircuit/file-server": "^0.0.11",
32
+ "@tscircuit/file-server": "^0.0.13",
30
33
  "@tscircuit/runframe": "^0.0.47",
31
34
  "chokidar": "^4.0.1",
32
35
  "commander": "^12.1.0",
@@ -0,0 +1,25 @@
1
+ import * as tempy from "tempy"
2
+ import getPort from "get-port"
3
+
4
+ interface Params {
5
+ vfs?: Record<string, string>
6
+ }
7
+
8
+ export const getTestFixture = async (params: Params) => {
9
+ // Create temp directory
10
+ const tempDirPath = await tempy.temporaryDirectory()
11
+ const devServerPort = await getPort()
12
+
13
+ // Write virtual filesystem files
14
+ if (params.vfs) {
15
+ for (const [filePath, content] of Object.entries(params.vfs)) {
16
+ await Bun.write(`${tempDirPath}/${filePath}`, content)
17
+ }
18
+ }
19
+
20
+ return {
21
+ tempDirPath,
22
+ devServerPort,
23
+ devServerUrl: `http://localhost:${devServerPort}`,
24
+ }
25
+ }
@@ -0,0 +1,40 @@
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
+
5
+ test("test1 basic dev server filesystem watching", async () => {
6
+ const { tempDirPath, devServerPort, devServerUrl } = await getTestFixture({
7
+ vfs: {
8
+ "snippet.tsx": `
9
+ export const MyCircuit = () => (
10
+ <board width="10mm" height="10mm">
11
+ <chip name="U1" footprint="soic8" />
12
+ </board>
13
+ )
14
+ `,
15
+ "manual-edits.json": "{}",
16
+ },
17
+ })
18
+
19
+ const devServer = new DevServer({
20
+ port: devServerPort,
21
+ componentFilePath: `${tempDirPath}/snippet.tsx`,
22
+ })
23
+
24
+ await devServer.start()
25
+ await devServer.addEntrypoint()
26
+
27
+ const { file_list } = await devServer.fsKy.get("api/files/list").json()
28
+
29
+ expect(file_list.map((f) => f.file_path).sort()).toMatchInlineSnapshot(`
30
+ [
31
+ "entrypoint.tsx",
32
+ "manual-edits.json",
33
+ "snippet.tsx",
34
+ ]
35
+ `)
36
+
37
+ afterEach(async () => {
38
+ await devServer.stop()
39
+ })
40
+ })
package/tsconfig.json CHANGED
@@ -10,7 +10,8 @@
10
10
  "baseUrl": ".",
11
11
  "paths": {
12
12
  "lib/*": ["lib/*"],
13
- "cli/*": ["cli/*"]
13
+ "cli/*": ["cli/*"],
14
+ "tests/*": ["tests/*"]
14
15
  },
15
16
 
16
17
  // Bundler mode