@tscircuit/cli 0.0.394 → 0.1.2

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.
Files changed (246) hide show
  1. package/.github/workflows/bun-formatcheck.yml +26 -0
  2. package/.github/workflows/bun-pver-release.yml +25 -0
  3. package/.github/workflows/{typecheck.yml → bun-typecheck.yml} +0 -3
  4. package/LICENSE +21 -0
  5. package/README.md +33 -36
  6. package/biome.json +9 -9
  7. package/bun.lockb +0 -0
  8. package/cli/CliContext.ts +8 -0
  9. package/cli/auth/login/register.ts +73 -0
  10. package/cli/auth/logout/register.ts +11 -0
  11. package/cli/auth/register.ts +5 -0
  12. package/cli/clone/register.ts +99 -0
  13. package/cli/config/print/register.ts +12 -0
  14. package/cli/config/register.ts +5 -0
  15. package/cli/dev/register.ts +111 -0
  16. package/cli/main.ts +33 -0
  17. package/dist/main.js +537 -0
  18. package/docs/file-server-api-usage.md +57 -0
  19. package/docs/run-frame-usage.md +14 -0
  20. package/example-dir/manual-edits.json +1 -0
  21. package/example-dir/snippet.d.ts +13 -0
  22. package/example-dir/snippet.tsx +20 -0
  23. package/example-dir/types.d.ts +11 -0
  24. package/lib/cli-config/TypedConfigStore.ts +50 -0
  25. package/lib/cli-config/index.ts +16 -0
  26. package/lib/dependency-analysis/DependencyAnalyzer.ts +129 -0
  27. package/lib/dependency-analysis/getLocalFileDependencies.ts +101 -0
  28. package/lib/dependency-analysis/installNodeModuleTypes.ts +74 -0
  29. package/lib/index.ts +2 -0
  30. package/lib/project-config/index.ts +5 -0
  31. package/lib/registry-api/endpoint-types.ts +20 -0
  32. package/lib/registry-api/get-ky.ts +30 -0
  33. package/lib/server/EventsWatcher.ts +75 -0
  34. package/lib/server/createServer.ts +62 -0
  35. package/lib/site/getIndex.ts +18 -0
  36. package/package.json +27 -144
  37. package/tsconfig.json +25 -22
  38. package/.github/workflows/formatbot.yml +0 -63
  39. package/.github/workflows/release.yml +0 -40
  40. package/.github/workflows/test.yml +0 -32
  41. package/.github/workflows/windows-tests.yml +0 -32
  42. package/.prettierrc +0 -1
  43. package/DEVELOPMENT.md +0 -7
  44. package/api/README.md +0 -3
  45. package/api/db/generic-json-level.ts +0 -123
  46. package/api/db/get-db.ts +0 -26
  47. package/api/db/schema.ts +0 -65
  48. package/api/db/zod-level-db.ts +0 -148
  49. package/api/index.ts +0 -4
  50. package/api/lib/middlewares/with-db.ts +0 -18
  51. package/api/lib/middlewares/with-debug-request-logging.ts +0 -13
  52. package/api/lib/middlewares/with-error-response.ts +0 -37
  53. package/api/lib/with-winter-spec.ts +0 -9
  54. package/api/lib/zod/export_parameters.ts +0 -25
  55. package/api/routes/api/db/download.ts +0 -25
  56. package/api/routes/api/dev_package_examples/create.ts +0 -43
  57. package/api/routes/api/dev_package_examples/get.ts +0 -46
  58. package/api/routes/api/dev_package_examples/list.ts +0 -36
  59. package/api/routes/api/dev_package_examples/update.ts +0 -59
  60. package/api/routes/api/dev_server/reset.ts +0 -13
  61. package/api/routes/api/export_files/create.ts +0 -27
  62. package/api/routes/api/export_files/download.ts +0 -25
  63. package/api/routes/api/export_requests/create.ts +0 -30
  64. package/api/routes/api/export_requests/get.ts +0 -43
  65. package/api/routes/api/export_requests/list.ts +0 -26
  66. package/api/routes/api/export_requests/update.ts +0 -34
  67. package/api/routes/api/health.ts +0 -11
  68. package/api/routes/api/package_info/create.ts +0 -26
  69. package/api/routes/api/package_info/get.ts +0 -16
  70. package/api/routes/health.ts +0 -11
  71. package/api/routes/index.ts +0 -16
  72. package/api/server.ts +0 -20
  73. package/api/static-routes.ts +0 -24
  74. package/api/tests/fixtures/get-test-server.ts +0 -31
  75. package/api/tests/fixtures/start-server.ts +0 -41
  76. package/api/tests/routes/dev_package_examples/create.test.ts +0 -19
  77. package/api/tests/routes/dev_package_examples/get.test.ts +0 -25
  78. package/api/tests/routes/dev_package_examples/list.test.ts +0 -32
  79. package/api/tests/routes/dev_package_examples/update.test.ts +0 -38
  80. package/api/tests/routes/export_files/create.test.ts +0 -18
  81. package/api/tests/routes/export_files/download.test.ts +0 -29
  82. package/api/tests/routes/export_requests/create.test.ts +0 -24
  83. package/api/tests/routes/export_requests/get.test.ts +0 -41
  84. package/api/tests/routes/export_requests/list.test.ts +0 -35
  85. package/api/tests/routes/export_requests/update.test.ts +0 -50
  86. package/api/tests/routes/health.test.ts +0 -10
  87. package/bunfig.toml +0 -2
  88. package/cli/cli.ts +0 -13
  89. package/cli/lib/cmd-fns/add.ts +0 -34
  90. package/cli/lib/cmd-fns/auth-login.ts +0 -59
  91. package/cli/lib/cmd-fns/auth-logout.ts +0 -7
  92. package/cli/lib/cmd-fns/auth-sessions-create.ts +0 -3
  93. package/cli/lib/cmd-fns/auth-sessions-get.ts +0 -3
  94. package/cli/lib/cmd-fns/auth-sessions-list.ts +0 -5
  95. package/cli/lib/cmd-fns/config-clear.ts +0 -5
  96. package/cli/lib/cmd-fns/config-print-config.ts +0 -6
  97. package/cli/lib/cmd-fns/config-reveal-location.ts +0 -5
  98. package/cli/lib/cmd-fns/config-set-log-requests.ts +0 -7
  99. package/cli/lib/cmd-fns/config-set-registry.ts +0 -9
  100. package/cli/lib/cmd-fns/config-set-runtime.ts +0 -7
  101. package/cli/lib/cmd-fns/config-set-session.ts +0 -7
  102. package/cli/lib/cmd-fns/dev/check-if-initialized.ts +0 -22
  103. package/cli/lib/cmd-fns/dev/derive-selector-from-pcb-component-id.ts +0 -23
  104. package/cli/lib/cmd-fns/dev/dev-server-request-handler.ts +0 -61
  105. package/cli/lib/cmd-fns/dev/find-available-port.ts +0 -32
  106. package/cli/lib/cmd-fns/dev/fulfill-export-requests.ts +0 -162
  107. package/cli/lib/cmd-fns/dev/get-dev-server-axios.ts +0 -29
  108. package/cli/lib/cmd-fns/dev/index.ts +0 -168
  109. package/cli/lib/cmd-fns/dev/infer-export-name-from-source.ts +0 -17
  110. package/cli/lib/cmd-fns/dev/mark-all-examples-loading.ts +0 -18
  111. package/cli/lib/cmd-fns/dev/soupify-and-upload-example-file.ts +0 -62
  112. package/cli/lib/cmd-fns/dev/start-dev-server.ts +0 -34
  113. package/cli/lib/cmd-fns/dev/start-edit-event-watcher.ts +0 -347
  114. package/cli/lib/cmd-fns/dev/start-export-request-watcher.ts +0 -33
  115. package/cli/lib/cmd-fns/dev/start-fs-watcher.ts +0 -54
  116. package/cli/lib/cmd-fns/dev/upload-examples-from-directory.ts +0 -42
  117. package/cli/lib/cmd-fns/dev-server-fulfill-export-requests.ts +0 -43
  118. package/cli/lib/cmd-fns/dev-server-upload.ts +0 -56
  119. package/cli/lib/cmd-fns/export-gerbers.ts +0 -28
  120. package/cli/lib/cmd-fns/export-kicad-pcb.ts +0 -36
  121. package/cli/lib/cmd-fns/export-pnp-csv.ts +0 -32
  122. package/cli/lib/cmd-fns/gen-jlcpcb-component.ts +0 -64
  123. package/cli/lib/cmd-fns/go.ts +0 -14
  124. package/cli/lib/cmd-fns/index.ts +0 -46
  125. package/cli/lib/cmd-fns/init/create-or-modify-npmrc.ts +0 -21
  126. package/cli/lib/cmd-fns/init/get-generated-npmrc.ts +0 -8
  127. package/cli/lib/cmd-fns/init/get-generated-readme.ts +0 -41
  128. package/cli/lib/cmd-fns/init/get-generated-tsconfig.ts +0 -34
  129. package/cli/lib/cmd-fns/init/index.ts +0 -193
  130. package/cli/lib/cmd-fns/install.ts +0 -34
  131. package/cli/lib/cmd-fns/lint.ts +0 -43
  132. package/cli/lib/cmd-fns/open.ts +0 -19
  133. package/cli/lib/cmd-fns/package-examples-create.ts +0 -36
  134. package/cli/lib/cmd-fns/package-examples-get.ts +0 -20
  135. package/cli/lib/cmd-fns/package-examples-list.ts +0 -18
  136. package/cli/lib/cmd-fns/package-files-create.ts +0 -31
  137. package/cli/lib/cmd-fns/package-files-download.ts +0 -29
  138. package/cli/lib/cmd-fns/package-files-get.ts +0 -3
  139. package/cli/lib/cmd-fns/package-files-list.ts +0 -28
  140. package/cli/lib/cmd-fns/package-files-upload-directory.ts +0 -6
  141. package/cli/lib/cmd-fns/package-releases-create.ts +0 -35
  142. package/cli/lib/cmd-fns/package-releases-get.ts +0 -3
  143. package/cli/lib/cmd-fns/package-releases-list.ts +0 -32
  144. package/cli/lib/cmd-fns/package-releases-update.ts +0 -45
  145. package/cli/lib/cmd-fns/packages-create.ts +0 -16
  146. package/cli/lib/cmd-fns/packages-get.ts +0 -16
  147. package/cli/lib/cmd-fns/packages-list.ts +0 -16
  148. package/cli/lib/cmd-fns/publish/index.ts +0 -336
  149. package/cli/lib/cmd-fns/remove.ts +0 -31
  150. package/cli/lib/cmd-fns/render.ts +0 -45
  151. package/cli/lib/cmd-fns/soupify.ts +0 -31
  152. package/cli/lib/cmd-fns/uninstall.ts +0 -31
  153. package/cli/lib/cmd-fns/version.ts +0 -38
  154. package/cli/lib/create-config-manager.ts +0 -97
  155. package/cli/lib/export-fns/export-bom-csv.ts +0 -32
  156. package/cli/lib/export-fns/export-gerbers.ts +0 -108
  157. package/cli/lib/export-fns/export-kicad-pcb.ts +0 -32
  158. package/cli/lib/export-fns/export-pnp-csv.ts +0 -31
  159. package/cli/lib/get-program.ts +0 -387
  160. package/cli/lib/param-handlers/index.ts +0 -21
  161. package/cli/lib/param-handlers/interact-for-local-directory.ts +0 -58
  162. package/cli/lib/param-handlers/interact-for-local-file.ts +0 -59
  163. package/cli/lib/param-handlers/interact-for-package-example-id.ts +0 -25
  164. package/cli/lib/param-handlers/interact-for-package-name-with-version.ts +0 -63
  165. package/cli/lib/param-handlers/interact-for-package-name.ts +0 -45
  166. package/cli/lib/param-handlers/interact-for-package-release-id.ts +0 -15
  167. package/cli/lib/param-handlers/interact-for-registry-url.ts +0 -27
  168. package/cli/lib/param-handlers/interact-for-runtime.ts +0 -33
  169. package/cli/lib/param-handlers/param-handler-type.ts +0 -7
  170. package/cli/lib/posthog.ts +0 -23
  171. package/cli/lib/soupify/get-export-name-from-file.ts +0 -29
  172. package/cli/lib/soupify/get-tmp-entrpoint-filepath.ts +0 -15
  173. package/cli/lib/soupify/index.ts +0 -1
  174. package/cli/lib/soupify/run-entrypoint-file.ts +0 -59
  175. package/cli/lib/soupify/soupify-with-core.ts +0 -74
  176. package/cli/lib/soupify/soupify.ts +0 -6
  177. package/cli/lib/util/app-context.ts +0 -17
  178. package/cli/lib/util/create-context-and-run-program.ts +0 -168
  179. package/cli/lib/util/get-all-package-files.ts +0 -66
  180. package/cli/lib/util/lint-project.ts +0 -137
  181. package/cli/tests/export-gerber-keyboard.test.ts +0 -16
  182. package/cli/tests/export-gerber.test.ts +0 -49
  183. package/cli/tests/export-kicad-pcb.test.ts +0 -23
  184. package/cli/tests/export-pnp-csv.test.ts +0 -24
  185. package/cli/tests/fixtures/preload.ts +0 -54
  186. package/cli/tests/init.test.ts +0 -9
  187. package/cli/tests/open.test.ts +0 -9
  188. package/cli/tests/soupify-builder.test.ts +0 -9
  189. package/cli/tests/soupify-core.test.ts +0 -9
  190. package/dist/cli.js +0 -3676
  191. package/docs/EDIT_EVENT_PIPELINE.md +0 -34
  192. package/example-project/README.md +0 -18
  193. package/example-project/examples/basic-capacitor.tsx +0 -5
  194. package/example-project/examples/basic-chip.tsx +0 -26
  195. package/example-project/examples/basic-resistor.tsx +0 -3
  196. package/example-project/examples/macrokeypad.tsx +0 -59
  197. package/example-project/index.ts +0 -1
  198. package/example-project/package.json +0 -5
  199. package/example-project/src/ArduinoProMicroBreakout.tsx +0 -37
  200. package/example-project/src/Key.tsx +0 -46
  201. package/example-project/src/Keyswitch.tsx +0 -26
  202. package/example-project/src/KeyswitchSocket.tsx +0 -56
  203. package/example-project/src/MyCircuit.tsx +0 -38
  204. package/example-project/src/manual-edits.ts +0 -93
  205. package/frontend/README.md +0 -3
  206. package/frontend/bun.lockb +0 -0
  207. package/frontend/components/command-k.tsx +0 -86
  208. package/frontend/components/dialogs/generic-export-dialog.tsx +0 -189
  209. package/frontend/components/dialogs/gerber-export-dialog.tsx +0 -168
  210. package/frontend/components/global-context-providers.tsx +0 -11
  211. package/frontend/components/select-example-search.tsx +0 -118
  212. package/frontend/components/ui/alert-dialog.tsx +0 -139
  213. package/frontend/components/ui/alert.tsx +0 -59
  214. package/frontend/components/ui/breadcrumb.tsx +0 -115
  215. package/frontend/components/ui/button.tsx +0 -57
  216. package/frontend/components/ui/card.tsx +0 -76
  217. package/frontend/components/ui/command.tsx +0 -153
  218. package/frontend/components/ui/context-menu.tsx +0 -202
  219. package/frontend/components/ui/dialog.tsx +0 -120
  220. package/frontend/components/ui/menubar.tsx +0 -238
  221. package/frontend/components/ui/navigation-menu.tsx +0 -128
  222. package/frontend/components/ui/popover.tsx +0 -31
  223. package/frontend/components/ui/select.tsx +0 -162
  224. package/frontend/components/ui/tabs.tsx +0 -53
  225. package/frontend/components/ui/toggle-group.tsx +0 -59
  226. package/frontend/components/ui/toggle.tsx +0 -43
  227. package/frontend/components/ui/tooltip.tsx +0 -28
  228. package/frontend/components.json +0 -17
  229. package/frontend/hooks/toast-if-api-not-connected.ts +0 -23
  230. package/frontend/hooks/use-active-dev-package-example-lite.ts +0 -39
  231. package/frontend/hooks/use-dev-package-examples.tsx +0 -18
  232. package/frontend/hooks/use-global-store.ts +0 -42
  233. package/frontend/index.css +0 -76
  234. package/frontend/index.html +0 -13
  235. package/frontend/lib/utils.ts +0 -6
  236. package/frontend/main.tsx +0 -13
  237. package/frontend/tailwind.config.js +0 -74
  238. package/frontend/views/App.tsx +0 -22
  239. package/frontend/views/Header.tsx +0 -55
  240. package/frontend/views/HeaderMenu.tsx +0 -326
  241. package/frontend/views/MainContentView.tsx +0 -172
  242. package/frontend/vite-env.d.ts +0 -1
  243. package/frontend/vite.config.ts +0 -50
  244. package/renovate.json +0 -16
  245. package/scripts/build-cli.ts +0 -12
  246. package/tsup.config.ts +0 -7
package/dist/main.js ADDED
@@ -0,0 +1,537 @@
1
+ #!/usr/bin/env node
2
+
3
+ // cli/main.ts
4
+ import { Command } from "commander";
5
+
6
+ // cli/dev/register.ts
7
+ import * as path4 from "node:path";
8
+ import * as chokidar from "chokidar";
9
+ import * as fs4 from "node:fs";
10
+
11
+ // lib/server/createServer.ts
12
+ import * as http from "node:http";
13
+ import * as fs from "node:fs";
14
+ import * as path from "node:path";
15
+ import { getNodeHandler } from "winterspec/adapters/node";
16
+ import winterspecBundle from "@tscircuit/file-server/dist/bundle.js";
17
+
18
+ // lib/site/getIndex.ts
19
+ var getIndex = async () => {
20
+ return `<html>
21
+ <head>
22
+ </head>
23
+ <body>
24
+ <script src="https://cdn.tailwindcss.com"></script>
25
+ <div id="root">loading...</div>
26
+ <script>
27
+ globalThis.process = { env: { NODE_ENV: "production" } }
28
+ </script>
29
+ <script src="/standalone.min.js"></script>
30
+ </body>
31
+ </html>`;
32
+ };
33
+
34
+ // lib/server/createServer.ts
35
+ var createServer2 = async (port = 3e3) => {
36
+ const fileServerHandler = getNodeHandler(winterspecBundle, {});
37
+ const server = http.createServer(async (req, res) => {
38
+ const url = new URL(req.url, `http://${req.headers.host}`);
39
+ if (url.pathname === "/standalone.min.js") {
40
+ const standaloneFilePath = process.env.RUNFRAME_STANDALONE_FILE_PATH || path.resolve(
41
+ process.cwd(),
42
+ "node_modules",
43
+ "@tscircuit/runframe/dist/standalone.min.js"
44
+ );
45
+ try {
46
+ const content = fs.readFileSync(standaloneFilePath, "utf8");
47
+ res.writeHead(200, {
48
+ "Content-Type": "application/javascript; charset=utf-8"
49
+ });
50
+ res.end(content);
51
+ return;
52
+ } catch (error) {
53
+ console.error("Error serving standalone.min.js:", error);
54
+ res.writeHead(404);
55
+ res.end("File not found");
56
+ return;
57
+ }
58
+ }
59
+ if (url.pathname === "/") {
60
+ const html = await getIndex();
61
+ res.writeHead(200, { "Content-Type": "text/html" });
62
+ res.end(html);
63
+ return;
64
+ }
65
+ if (url.pathname.startsWith("/api/")) {
66
+ req.url = req.url.replace("/api/", "/");
67
+ fileServerHandler(req, res);
68
+ return;
69
+ }
70
+ res.writeHead(404);
71
+ res.end("Not found");
72
+ });
73
+ return new Promise((resolve4) => {
74
+ server.listen(port, () => {
75
+ console.log(`Server running at http://localhost:${port}`);
76
+ resolve4();
77
+ });
78
+ });
79
+ };
80
+
81
+ // lib/dependency-analysis/getLocalFileDependencies.ts
82
+ import * as ts from "typescript";
83
+ import * as path2 from "path";
84
+ import * as fs2 from "fs";
85
+ function getLocalFileDependencies(pathToTsxFile) {
86
+ const absolutePath = path2.resolve(pathToTsxFile);
87
+ const baseDir = path2.dirname(absolutePath);
88
+ const content = fs2.readFileSync(absolutePath, "utf-8");
89
+ const sourceFile = ts.createSourceFile(
90
+ absolutePath,
91
+ content,
92
+ ts.ScriptTarget.Latest,
93
+ true
94
+ );
95
+ const dependencies = /* @__PURE__ */ new Set();
96
+ function visit(node) {
97
+ if (ts.isImportDeclaration(node) || ts.isExportDeclaration(node)) {
98
+ const moduleSpecifier = node.moduleSpecifier;
99
+ if (moduleSpecifier && ts.isStringLiteral(moduleSpecifier)) {
100
+ const importPath = moduleSpecifier.text;
101
+ if (importPath.startsWith(".")) {
102
+ resolveAndAddDependency(importPath);
103
+ }
104
+ }
105
+ }
106
+ if (ts.isCallExpression(node) && node.expression.kind === ts.SyntaxKind.ImportKeyword) {
107
+ const argument = node.arguments[0];
108
+ if (argument && ts.isStringLiteral(argument)) {
109
+ const importPath = argument.text;
110
+ if (importPath.startsWith(".")) {
111
+ resolveAndAddDependency(importPath);
112
+ }
113
+ }
114
+ }
115
+ ts.forEachChild(node, visit);
116
+ }
117
+ function resolveAndAddDependency(importPath) {
118
+ const extensions = [
119
+ ".tsx",
120
+ ".ts",
121
+ ".jsx",
122
+ ".js",
123
+ ".css",
124
+ ".scss",
125
+ ".sass",
126
+ ".less"
127
+ ];
128
+ let resolvedPath = path2.resolve(baseDir, importPath);
129
+ if (fs2.existsSync(resolvedPath)) {
130
+ dependencies.add(resolvedPath);
131
+ return;
132
+ }
133
+ for (const ext of extensions) {
134
+ const pathWithExt = resolvedPath + ext;
135
+ if (fs2.existsSync(pathWithExt)) {
136
+ dependencies.add(pathWithExt);
137
+ return;
138
+ }
139
+ }
140
+ if (fs2.existsSync(resolvedPath) && fs2.statSync(resolvedPath).isDirectory()) {
141
+ for (const ext of extensions) {
142
+ const indexPath = path2.join(resolvedPath, `index${ext}`);
143
+ if (fs2.existsSync(indexPath)) {
144
+ dependencies.add(indexPath);
145
+ return;
146
+ }
147
+ }
148
+ }
149
+ }
150
+ visit(sourceFile);
151
+ return Array.from(dependencies);
152
+ }
153
+
154
+ // lib/dependency-analysis/installNodeModuleTypes.ts
155
+ import * as fs3 from "node:fs";
156
+ import * as path3 from "node:path";
157
+ import * as ts2 from "typescript";
158
+ async function installTypes(snippetPath) {
159
+ const content = fs3.readFileSync(snippetPath, "utf-8");
160
+ const sourceFile = ts2.createSourceFile(
161
+ snippetPath,
162
+ content,
163
+ ts2.ScriptTarget.Latest,
164
+ true
165
+ );
166
+ const imports = [];
167
+ function visit(node) {
168
+ if (ts2.isImportDeclaration(node)) {
169
+ const moduleSpecifier = node.moduleSpecifier;
170
+ if (moduleSpecifier && ts2.isStringLiteral(moduleSpecifier)) {
171
+ const importPath = moduleSpecifier.text;
172
+ if (importPath.startsWith("@tsci/")) {
173
+ imports.push(importPath);
174
+ }
175
+ }
176
+ }
177
+ ts2.forEachChild(node, visit);
178
+ }
179
+ visit(sourceFile);
180
+ let projectRoot = path3.dirname(snippetPath);
181
+ while (projectRoot !== path3.parse(projectRoot).root) {
182
+ if (fs3.existsSync(path3.join(projectRoot, "package.json"))) {
183
+ break;
184
+ }
185
+ projectRoot = path3.dirname(projectRoot);
186
+ }
187
+ for (const importPath of imports) {
188
+ const [owner, name] = importPath.replace("@tsci/", "").split(".");
189
+ try {
190
+ const response = await fetch(
191
+ `https://registry-api.tscircuit.com/snippets/get?owner_name=${owner}&unscoped_name=${name}`
192
+ );
193
+ if (!response.ok) {
194
+ console.warn(`Failed to fetch types for ${importPath}`);
195
+ continue;
196
+ }
197
+ const data = await response.json();
198
+ if (data.snippet.dts) {
199
+ const packageDir = path3.join(
200
+ projectRoot,
201
+ "node_modules",
202
+ "@tsci",
203
+ `${owner}.${name}`
204
+ );
205
+ fs3.mkdirSync(packageDir, { recursive: true });
206
+ fs3.writeFileSync(path3.join(packageDir, "index.d.ts"), data.snippet.dts);
207
+ }
208
+ } catch (error) {
209
+ console.warn(`Error fetching types for ${importPath}:`, error);
210
+ }
211
+ }
212
+ }
213
+
214
+ // lib/server/EventsWatcher.ts
215
+ import { EventEmitter } from "events";
216
+ var EventsWatcher = class extends EventEmitter {
217
+ lastPollTime;
218
+ pollInterval;
219
+ baseUrl;
220
+ polling = false;
221
+ timeoutId;
222
+ constructor(baseUrl = "http://localhost:3000", pollInterval = 1e3) {
223
+ super();
224
+ this.baseUrl = baseUrl;
225
+ this.pollInterval = pollInterval;
226
+ this.lastPollTime = (/* @__PURE__ */ new Date()).toISOString();
227
+ }
228
+ async start() {
229
+ if (this.polling) return;
230
+ this.polling = true;
231
+ await this.poll();
232
+ }
233
+ stop() {
234
+ this.polling = false;
235
+ if (this.timeoutId) {
236
+ clearTimeout(this.timeoutId);
237
+ }
238
+ }
239
+ async poll() {
240
+ if (!this.polling) return;
241
+ try {
242
+ const response = await fetch(
243
+ `${this.baseUrl}/api/events/list?since=${encodeURIComponent(this.lastPollTime)}`
244
+ );
245
+ if (!response.ok) {
246
+ throw new Error(`HTTP error! status: ${response.status}`);
247
+ }
248
+ const data = await response.json();
249
+ const latestEvent = data.event_list[data.event_list.length - 1];
250
+ this.lastPollTime = latestEvent ? latestEvent.created_at : (/* @__PURE__ */ new Date()).toISOString();
251
+ data.event_list.forEach((event) => {
252
+ this.emit(event.event_type, event);
253
+ this.emit("*", event);
254
+ });
255
+ } catch (error) {
256
+ this.emit("error", error);
257
+ }
258
+ this.timeoutId = globalThis.setTimeout(
259
+ () => this.poll(),
260
+ this.pollInterval
261
+ );
262
+ }
263
+ };
264
+
265
+ // cli/dev/register.ts
266
+ var registerDev = (program2) => {
267
+ 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) => {
268
+ const absolutePath = path4.resolve(file);
269
+ const fileDir = path4.dirname(absolutePath);
270
+ const port = parseInt(options.port);
271
+ try {
272
+ console.log("Installing types for imported snippets...");
273
+ await installTypes(absolutePath);
274
+ console.log("Types installed successfully");
275
+ } catch (error) {
276
+ console.warn("Failed to install types:", error);
277
+ }
278
+ await createServer2(port);
279
+ const eventsWatcher = new EventsWatcher(`http://localhost:${port}`);
280
+ eventsWatcher.start();
281
+ await fetch(`http://localhost:${port}/api/files/upsert`, {
282
+ method: "POST",
283
+ headers: { "Content-Type": "application/json" },
284
+ body: JSON.stringify({
285
+ file_path: "entrypoint.tsx",
286
+ text_content: `
287
+ import MyCircuit from "./snippet.tsx"
288
+
289
+ circuit.add(<MyCircuit />)
290
+ `
291
+ })
292
+ });
293
+ const updateFile = async (filePath) => {
294
+ try {
295
+ const content = await fs4.promises.readFile(filePath, "utf-8");
296
+ const response = await fetch(
297
+ `http://localhost:${port}/api/files/upsert`,
298
+ {
299
+ method: "POST",
300
+ headers: { "Content-Type": "application/json" },
301
+ body: JSON.stringify({
302
+ file_path: path4.relative(fileDir, filePath),
303
+ text_content: content
304
+ })
305
+ }
306
+ );
307
+ if (!response.ok) {
308
+ console.error(`Failed to update ${filePath}`);
309
+ }
310
+ } catch (error) {
311
+ console.error(`Error updating ${filePath}:`, error);
312
+ }
313
+ };
314
+ const dependencies = /* @__PURE__ */ new Set([absolutePath]);
315
+ try {
316
+ const deps = getLocalFileDependencies(absolutePath);
317
+ deps.forEach((dep) => dependencies.add(dep));
318
+ } catch (error) {
319
+ console.warn("Failed to analyze dependencies:", error);
320
+ }
321
+ const filesystemWatcher = chokidar.watch(Array.from(dependencies), {
322
+ persistent: true,
323
+ ignoreInitial: false
324
+ });
325
+ filesystemWatcher.on("change", async (filePath) => {
326
+ console.log(`File ${filePath} changed`);
327
+ await updateFile(filePath);
328
+ });
329
+ filesystemWatcher.on("add", async (filePath) => {
330
+ console.log(`File ${filePath} added`);
331
+ await updateFile(filePath);
332
+ });
333
+ eventsWatcher.on("FILE_UPDATED", async (ev) => {
334
+ if (ev.file_path === "manual-edits.json") {
335
+ console.log("Manual edits updated, updating on filesystem...");
336
+ const { file: file2 } = await fetch(
337
+ `http://localhost:${port}/api/files/get?file_path=manual-edits.json`
338
+ ).then((r) => r.json());
339
+ fs4.writeFileSync(
340
+ path4.join(fileDir, "manual-edits.json"),
341
+ file2.text_content
342
+ );
343
+ }
344
+ });
345
+ console.log(`Watching ${file} and its dependencies...`);
346
+ });
347
+ };
348
+
349
+ // lib/cli-config/index.ts
350
+ import Configstore from "configstore";
351
+ var cliConfig = new Configstore(
352
+ "tscircuit"
353
+ );
354
+ var getRegistryApiUrl = () => {
355
+ return cliConfig.get("registryApiUrl") ?? "https://registry-api.tscircuit.com";
356
+ };
357
+
358
+ // cli/auth/login/register.ts
359
+ import delay from "delay";
360
+
361
+ // lib/registry-api/get-ky.ts
362
+ import ky from "ky";
363
+ var prettyResponseErrorHook = async (_request, _options, response) => {
364
+ if (!response.ok) {
365
+ try {
366
+ const errorData = await response.json();
367
+ throw new Error(
368
+ `FAIL [${response.status}]: ${_request.method} ${new URL(_request.url).pathname}
369
+
370
+ ${JSON.stringify(errorData, null, 2)}`
371
+ );
372
+ } catch (e) {
373
+ }
374
+ }
375
+ };
376
+ var getKy = () => {
377
+ return ky.create({
378
+ prefixUrl: getRegistryApiUrl(),
379
+ hooks: {
380
+ afterResponse: [prettyResponseErrorHook]
381
+ }
382
+ });
383
+ };
384
+
385
+ // cli/auth/login/register.ts
386
+ var registerAuthLogin = (program2) => {
387
+ program2.commands.find((c) => c.name() === "auth").command("login").description("Authenticate CLI, login to registry").action(async (args) => {
388
+ const ky2 = getKy();
389
+ const { login_page } = await ky2.post(
390
+ "sessions/login_page/create",
391
+ {
392
+ json: {}
393
+ }
394
+ ).json();
395
+ console.log("Please visit the following URL to log in:");
396
+ console.log(login_page.url);
397
+ while (true) {
398
+ const { login_page: new_login_page } = await ky2.post(
399
+ "sessions/login_page/get",
400
+ {
401
+ json: {
402
+ login_page_id: login_page.login_page_id
403
+ },
404
+ headers: {
405
+ Authorization: `Bearer ${login_page.login_page_auth_token}`
406
+ }
407
+ }
408
+ ).json();
409
+ if (new_login_page.was_login_successful) {
410
+ console.log("Logged in! Generating token...");
411
+ break;
412
+ }
413
+ if (new_login_page.is_expired) {
414
+ throw new Error("Login page expired");
415
+ }
416
+ await delay(1e3);
417
+ }
418
+ const { session } = await ky2.post(
419
+ "sessions/login_page/exchange_for_cli_session",
420
+ {
421
+ json: {
422
+ login_page_id: login_page.login_page_id
423
+ },
424
+ headers: {
425
+ Authorization: `Bearer ${login_page.login_page_auth_token}`
426
+ }
427
+ }
428
+ ).json();
429
+ cliConfig.set("sessionToken", session.token);
430
+ console.log("Ready to use!");
431
+ });
432
+ };
433
+
434
+ // cli/auth/logout/register.ts
435
+ var registerAuthLogout = (program2) => {
436
+ program2.commands.find((c) => c.name() === "auth").command("logout").description("Logout from registry").action((args) => {
437
+ console.log("logout");
438
+ });
439
+ };
440
+
441
+ // cli/auth/register.ts
442
+ var registerAuth = (program2) => {
443
+ program2.command("auth").description("Login/logout");
444
+ };
445
+
446
+ // cli/config/register.ts
447
+ var registerConfig = (program2) => {
448
+ program2.command("config").description("Manage tscircuit CLI configuration");
449
+ };
450
+
451
+ // cli/config/print/register.ts
452
+ var registerConfigPrint = (program2) => {
453
+ program2.commands.find((c) => c.name() === "config").command("print").description("Print the current config").action(() => {
454
+ console.log(JSON.stringify(cliConfig.all, null, 2));
455
+ });
456
+ };
457
+
458
+ // cli/clone/register.ts
459
+ import * as fs5 from "node:fs";
460
+ import * as path5 from "node:path";
461
+ var registerClone = (program2) => {
462
+ program2.command("clone").description("Clone a snippet from the registry").argument("<snippet>", "Snippet to clone (e.g. author/snippetName)").action(async (snippetPath) => {
463
+ let author;
464
+ let snippetName;
465
+ if (!snippetPath.startsWith("@tsci/") && snippetPath.includes("/")) {
466
+ ;
467
+ [author, snippetName] = snippetPath.split("/");
468
+ } else {
469
+ const trimmedPath = snippetPath.replace("@tsci/", "");
470
+ const firstDotIndex = trimmedPath.indexOf(".");
471
+ author = trimmedPath.slice(0, firstDotIndex);
472
+ snippetName = trimmedPath.slice(firstDotIndex + 1);
473
+ }
474
+ if (!author || !snippetName) {
475
+ console.error(
476
+ "Invalid snippet path. Use format: author/snippetName, author.snippetName or @tsci/author.snippetName"
477
+ );
478
+ process.exit(1);
479
+ }
480
+ const ky2 = getKy();
481
+ try {
482
+ console.log(`Cloning ${author}/${snippetName}...`);
483
+ const packageFileList = await ky2.post("package_files/list", {
484
+ json: {
485
+ package_name: `${author}/${snippetName}`,
486
+ use_latest_version: true
487
+ }
488
+ }).json();
489
+ const dirPath = `./${author}.${snippetName}`;
490
+ if (!fs5.existsSync(dirPath)) {
491
+ fs5.mkdirSync(dirPath);
492
+ }
493
+ for (const fileInfo of packageFileList.package_files) {
494
+ const filePath = fileInfo.file_path.startsWith("/") ? fileInfo.file_path.slice(1) : fileInfo.file_path;
495
+ if (filePath.startsWith("dist/")) continue;
496
+ const fileContent = await ky2.post("package_files/get", {
497
+ json: {
498
+ package_name: `${author}/${snippetName}`,
499
+ file_path: fileInfo.file_path
500
+ }
501
+ }).json();
502
+ const fullPath = path5.join(dirPath, filePath);
503
+ const dirName = path5.dirname(fullPath);
504
+ if (!fs5.existsSync(dirName)) {
505
+ fs5.mkdirSync(dirName, { recursive: true });
506
+ }
507
+ fs5.writeFileSync(fullPath, fileContent.package_file.content_text);
508
+ }
509
+ console.log(`Successfully cloned to ./${author}.${snippetName}/`);
510
+ } catch (error) {
511
+ if (error instanceof Error) {
512
+ console.error("Failed to clone snippet:", error.message);
513
+ } else {
514
+ console.error("Failed to clone snippet:", error);
515
+ }
516
+ process.exit(1);
517
+ }
518
+ });
519
+ };
520
+
521
+ // cli/main.ts
522
+ import { perfectCli } from "perfect-cli";
523
+ var program = new Command();
524
+ program.name("tsci").description("CLI for developing tscircuit snippets").version("1.0.0");
525
+ registerDev(program);
526
+ registerClone(program);
527
+ registerAuth(program);
528
+ registerAuthLogin(program);
529
+ registerAuthLogout(program);
530
+ registerConfig(program);
531
+ registerConfigPrint(program);
532
+ if (process.argv.length === 2) {
533
+ perfectCli(program, process.argv);
534
+ } else {
535
+ program.parse();
536
+ }
537
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../cli/main.ts", "../cli/dev/register.ts", "../lib/server/createServer.ts", "../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\"\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        res.writeHead(404)\n        res.end(\"File not found\")\n        return\n      }\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", "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"]
}

@@ -0,0 +1,57 @@
1
+ # File Server API Usage
2
+
3
+ The file server is a simple API that allows you to upload files and get their contents.
4
+
5
+ It emits events whenever a file is updated.
6
+
7
+ The browser preview will typically poll for event
8
+ changes.
9
+
10
+ ```
11
+ POST /files/upsert
12
+ REQUEST: { file_id?, text_content, file_path }
13
+ RESPONSE: { file: { file_id, file_path, text_content } }
14
+
15
+ GET /files/get?file_id?&file_path?
16
+ RESPONSE: { file: { file_id, file_path, text_content } }
17
+
18
+ GET /files/list
19
+ RESPONSE { file_list: Array<{ file_id, file_path } }
20
+
21
+ GET /events/list?since=<iso timestamp>
22
+ RESPONSE { event_list: Array<{ event_id, created_at, event_type, ... }> }
23
+
24
+ POST /events/create
25
+ REQUEST { event_type: "..." }
26
+ RESPONSE { event: { event_id, ... } }
27
+ ```
28
+
29
+ ## Starting the File Server
30
+
31
+ Here's an example of how to start the file server from a vite plugin that starts the file server:
32
+
33
+ ```tsx
34
+ import { defineConfig, type Plugin } from "vite"
35
+ import react from "@vitejs/plugin-react"
36
+ import { resolve } from "node:path"
37
+ import winterspecBundle from "@tscircuit/file-server/dist/bundle"
38
+ import { getNodeHandler } from "winterspec/adapters/node"
39
+
40
+ const fakeHandler = getNodeHandler(winterspecBundle as any, {})
41
+
42
+ function apiServerPlugin(): Plugin {
43
+ return {
44
+ name: "api-server",
45
+ configureServer(server) {
46
+ server.middlewares.use(async (req, res, next) => {
47
+ if (req.url?.startsWith("/api/")) {
48
+ req.url = req.url.replace("/api/", "/")
49
+ fakeHandler(req, res)
50
+ } else {
51
+ next()
52
+ }
53
+ })
54
+ },
55
+ }
56
+ }
57
+ ```