@tscircuit/cli 0.1.4 → 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 +1 -0
- package/bun.lockb +0 -0
- package/cli/dev/DevServer.ts +155 -0
- package/cli/dev/register.ts +9 -82
- package/dist/main.js +225 -232
- package/lib/dependency-analysis/{installNodeModuleTypes.ts → installNodeModuleTypesForSnippet.ts} +1 -1
- package/lib/file-server/FileServerEvent.ts +7 -0
- package/lib/file-server/FileServerRoutes.ts +38 -0
- package/lib/index.ts +1 -1
- package/lib/server/{createServer.ts → createHttpServer.ts} +3 -3
- package/package.json +7 -4
- package/tests/fixtures/get-test-fixture.ts +25 -0
- package/tests/test1-dev-server-basic.test.ts +40 -0
- package/tsconfig.json +2 -1
package/biome.json
CHANGED
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
|
+
}
|
package/cli/dev/register.ts
CHANGED
|
@@ -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 {
|
|
5
|
+
import { createHttpServer } from "lib/server/createHttpServer"
|
|
6
6
|
import { getLocalFileDependencies } from "lib/dependency-analysis/getLocalFileDependencies"
|
|
7
|
-
import {
|
|
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
|
|
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
|
-
|
|
30
|
-
|
|
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
|
-
|
|
35
|
+
await server.start()
|
|
36
|
+
await server.addEntrypoint()
|
|
110
37
|
})
|
|
111
38
|
}
|
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/
|
|
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.
|
|
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": "
|
|
98
|
+
"@types/bun": "^1.1.15",
|
|
38
99
|
"@types/configstore": "^6.0.2",
|
|
39
100
|
"@types/react": "^19.0.1",
|
|
40
|
-
|
|
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.
|
|
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/
|
|
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/
|
|
78
|
-
var
|
|
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 ||
|
|
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 =
|
|
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((
|
|
182
|
+
return new Promise((resolve3) => {
|
|
119
183
|
server.listen(port, () => {
|
|
120
184
|
console.log(`Server running at http://localhost:${port}`);
|
|
121
|
-
|
|
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
|
|
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
|
-
|
|
324
|
-
|
|
325
|
-
|
|
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
|
-
|
|
379
|
-
|
|
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
|
|
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
|
|
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
|
|
434
|
-
const { login_page } = await
|
|
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
|
|
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
|
|
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
|
|
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
|
|
518
|
+
const ky3 = getKy();
|
|
526
519
|
try {
|
|
527
520
|
console.log(`Cloning ${author}/${snippetName}...`);
|
|
528
|
-
const packageFileList = await
|
|
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 (!
|
|
536
|
-
|
|
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
|
|
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 (!
|
|
550
|
-
|
|
542
|
+
if (!fs4.existsSync(dirName)) {
|
|
543
|
+
fs4.mkdirSync(dirName, { recursive: true });
|
|
551
544
|
}
|
|
552
|
-
|
|
545
|
+
fs4.writeFileSync(fullPath, fileContent.package_file.content_text);
|
|
553
546
|
}
|
|
554
547
|
console.log(`Successfully cloned to ./${author}.${snippetName}/`);
|
|
555
548
|
} catch (error) {
|
|
@@ -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\"\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 { 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.3\",\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"]
}

|
package/lib/dependency-analysis/{installNodeModuleTypes.ts → installNodeModuleTypesForSnippet.ts}
RENAMED
|
@@ -8,7 +8,7 @@ interface SnippetApiResponse {
|
|
|
8
8
|
}
|
|
9
9
|
}
|
|
10
10
|
|
|
11
|
-
export async function
|
|
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,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 {
|
|
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
|
|
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<
|
|
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.
|
|
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": "
|
|
20
|
+
"@types/bun": "^1.1.15",
|
|
21
21
|
"@types/configstore": "^6.0.2",
|
|
22
22
|
"@types/react": "^19.0.1",
|
|
23
|
-
"
|
|
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.
|
|
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
|
+
})
|