@flowconsole/cli 0.0.0-beta-20260128114951 → 0.0.0-beta-20260428172441
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/README.md +121 -0
- package/bin/fc.js +63 -0
- package/package.json +40 -35
- package/postinstall.js +269 -0
- package/dist/index.d.ts +0 -2
- package/dist/index.js +0 -104
package/README.md
ADDED
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
# @flowconsole/cli
|
|
2
|
+
|
|
3
|
+
npm wrapper for the FlowConsole CLI (`fcon`) — architecture-as-code scanning, validation, diff, and push.
|
|
4
|
+
|
|
5
|
+
> **Note:** This package (`@flowconsole/cli@2.x`) replaces the legacy TypeScript CLI (`@flowconsole/cli@1.x`). The new CLI is a .NET self-contained binary distributed via this npm wrapper. `@flowconsole/cli@1.x` is deprecated but remains published on npm.
|
|
6
|
+
|
|
7
|
+
## Installation
|
|
8
|
+
|
|
9
|
+
```bash
|
|
10
|
+
npm install -g @flowconsole/cli
|
|
11
|
+
```
|
|
12
|
+
|
|
13
|
+
After installation, two equivalent commands are available globally:
|
|
14
|
+
|
|
15
|
+
| Command | When to use |
|
|
16
|
+
|---------|-------------|
|
|
17
|
+
| `fcon` | Short alias without the shell-builtin conflict. Recommended when scripting in zsh/bash. |
|
|
18
|
+
| `flowconsole` | Explicit, fully-qualified name. Best for CI scripts and documentation where readability matters more than brevity. |
|
|
19
|
+
|
|
20
|
+
All resolve to the same underlying binary.
|
|
21
|
+
|
|
22
|
+
## Platform Support
|
|
23
|
+
|
|
24
|
+
| Platform | Architecture | RID |
|
|
25
|
+
|----------|-------------|-----|
|
|
26
|
+
| Linux | x64 | linux-x64 |
|
|
27
|
+
| Linux | arm64 | linux-arm64 |
|
|
28
|
+
| macOS | x64 (Intel) | osx-x64 |
|
|
29
|
+
| macOS | arm64 (Apple Silicon) | osx-arm64 |
|
|
30
|
+
| Windows | x64 | win-x64 |
|
|
31
|
+
| Windows | arm64 | win-arm64 |
|
|
32
|
+
|
|
33
|
+
The `postinstall` script automatically detects your platform and downloads the appropriate binary from GitHub Releases.
|
|
34
|
+
|
|
35
|
+
## Usage
|
|
36
|
+
|
|
37
|
+
```bash
|
|
38
|
+
# Scan a project directory
|
|
39
|
+
fcon scan ./my-project
|
|
40
|
+
|
|
41
|
+
# Validate a snapshot against rules
|
|
42
|
+
fcon validate snapshot.json rules/
|
|
43
|
+
|
|
44
|
+
# Compare two snapshots
|
|
45
|
+
fcon diff before.json after.json --format markdown
|
|
46
|
+
|
|
47
|
+
# Push snapshot to FlowConsole backend
|
|
48
|
+
export FLOWCONSOLE_API_KEY=fcp_your_token_here
|
|
49
|
+
fcon push snapshot snapshot.json --model <model-id>
|
|
50
|
+
|
|
51
|
+
# Push validation findings
|
|
52
|
+
fcon push findings findings.json --model <model-id>
|
|
53
|
+
|
|
54
|
+
# Manage telemetry
|
|
55
|
+
fcon telemetry status
|
|
56
|
+
fcon telemetry off
|
|
57
|
+
|
|
58
|
+
# CI pipeline example
|
|
59
|
+
fcon scan ./project -o snapshot.json && \
|
|
60
|
+
fcon validate snapshot.json rules/ && \
|
|
61
|
+
fcon push snapshot snapshot.json --model <model-id>
|
|
62
|
+
```
|
|
63
|
+
|
|
64
|
+
## Environment Variables
|
|
65
|
+
|
|
66
|
+
| Variable | Description |
|
|
67
|
+
|----------|-------------|
|
|
68
|
+
| `FLOWCONSOLE_API_KEY` | Personal Access Token for `fcon push` (required; `--api-key` flag is refused for security) |
|
|
69
|
+
| `FLOWCONSOLE_API_URL` | Backend API URL (default from `.flowconsole.yaml`) |
|
|
70
|
+
| `FLOWCONSOLE_TELEMETRY` | Set to `off` to disable telemetry |
|
|
71
|
+
| `DO_NOT_TRACK` | Set to `1` to disable telemetry (standard) |
|
|
72
|
+
| `FLOWCONSOLE_CLI_DOWNLOAD_URL` | Override binary download base URL (for mirrors/air-gap) |
|
|
73
|
+
| `FLOWCONSOLE_CLI_SKIP_DOWNLOAD` | Set to `1` to skip binary download in postinstall |
|
|
74
|
+
|
|
75
|
+
## Troubleshooting
|
|
76
|
+
|
|
77
|
+
### postinstall fails
|
|
78
|
+
|
|
79
|
+
- Check your internet connection — the binary is downloaded from GitHub Releases
|
|
80
|
+
- If behind a proxy, configure `https_proxy` / `HTTPS_PROXY` environment variable
|
|
81
|
+
- If on an unsupported platform, download the binary manually from the [Releases page](https://github.com/flowconsole/flowconsole/releases)
|
|
82
|
+
- Set `FLOWCONSOLE_CLI_DOWNLOAD_URL` to point to a mirror if GitHub is blocked
|
|
83
|
+
|
|
84
|
+
### Binary not found after install
|
|
85
|
+
|
|
86
|
+
Run `npm rebuild @flowconsole/cli` to re-trigger the download.
|
|
87
|
+
|
|
88
|
+
### macOS Gatekeeper warning
|
|
89
|
+
|
|
90
|
+
The alpha binaries are unsigned. On first run, macOS may block execution. To allow:
|
|
91
|
+
|
|
92
|
+
```bash
|
|
93
|
+
xattr -d com.apple.quarantine $(which fc)
|
|
94
|
+
```
|
|
95
|
+
|
|
96
|
+
Or go to System Settings > Privacy & Security and click "Allow Anyway".
|
|
97
|
+
|
|
98
|
+
### Windows SmartScreen warning
|
|
99
|
+
|
|
100
|
+
The alpha binaries are unsigned. Windows SmartScreen may show "Windows protected your PC". Click "More info" then "Run anyway".
|
|
101
|
+
|
|
102
|
+
> Signing and notarization are planned for v1.0.0. Alpha binaries are unsigned.
|
|
103
|
+
|
|
104
|
+
## Deprecation of v1
|
|
105
|
+
|
|
106
|
+
`@flowconsole/cli@1.x` was a TypeScript-based CLI. `@flowconsole/cli@2.x` is a complete rewrite as a .NET self-contained binary, offering:
|
|
107
|
+
|
|
108
|
+
- Faster scanning via Tree-sitter native parsers
|
|
109
|
+
- Built-in rule engine with CEL expressions
|
|
110
|
+
- Schema validation against the model-snapshot contract
|
|
111
|
+
- `fcon push` for CI/CD integration with PAT authentication
|
|
112
|
+
- `fcon diff` for offline snapshot comparison
|
|
113
|
+
- Anonymous telemetry (opt-out via `fcon telemetry off`)
|
|
114
|
+
|
|
115
|
+
To upgrade: `npm install -g @flowconsole/cli@next`
|
|
116
|
+
|
|
117
|
+
The v1 package remains published for backward compatibility but receives no updates.
|
|
118
|
+
|
|
119
|
+
## License
|
|
120
|
+
|
|
121
|
+
Apache-2.0
|
package/bin/fc.js
ADDED
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
"use strict";
|
|
4
|
+
|
|
5
|
+
const { execFileSync } = require("child_process");
|
|
6
|
+
const path = require("path");
|
|
7
|
+
const fs = require("fs");
|
|
8
|
+
|
|
9
|
+
const PLATFORM_MAP = {
|
|
10
|
+
darwin: "osx",
|
|
11
|
+
linux: "linux",
|
|
12
|
+
win32: "win",
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
const ARCH_MAP = {
|
|
16
|
+
x64: "x64",
|
|
17
|
+
arm64: "arm64",
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
function getBinaryPath() {
|
|
21
|
+
const platform = PLATFORM_MAP[process.platform];
|
|
22
|
+
const arch = ARCH_MAP[process.arch];
|
|
23
|
+
|
|
24
|
+
if (!platform || !arch) {
|
|
25
|
+
console.error(
|
|
26
|
+
`Unsupported platform: ${process.platform}-${process.arch}\n` +
|
|
27
|
+
`Run 'npm rebuild @flowconsole/cli' or download the binary manually.`
|
|
28
|
+
);
|
|
29
|
+
process.exit(1);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
const rid = `${platform}-${arch}`;
|
|
33
|
+
const binaryName = process.platform === "win32" ? "fcon.exe" : "fcon";
|
|
34
|
+
const binaryPath = path.join(__dirname, "..", "binaries", rid, binaryName);
|
|
35
|
+
|
|
36
|
+
if (!fs.existsSync(binaryPath)) {
|
|
37
|
+
console.error(
|
|
38
|
+
`FlowConsole CLI binary not found at ${binaryPath}\n` +
|
|
39
|
+
`Run 'npm rebuild @flowconsole/cli' to download it, or set\n` +
|
|
40
|
+
`FLOWCONSOLE_CLI_DOWNLOAD_URL to a custom mirror.`
|
|
41
|
+
);
|
|
42
|
+
process.exit(1);
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
return binaryPath;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
const binaryPath = getBinaryPath();
|
|
49
|
+
|
|
50
|
+
try {
|
|
51
|
+
const result = execFileSync(binaryPath, process.argv.slice(2), {
|
|
52
|
+
stdio: "inherit",
|
|
53
|
+
env: process.env,
|
|
54
|
+
});
|
|
55
|
+
} catch (err) {
|
|
56
|
+
// execFileSync throws on non-zero exit code when stdio is inherit
|
|
57
|
+
if (err.status != null) {
|
|
58
|
+
process.exitCode = err.status;
|
|
59
|
+
} else {
|
|
60
|
+
console.error(`Failed to execute FlowConsole CLI: ${err.message}`);
|
|
61
|
+
process.exitCode = 1;
|
|
62
|
+
}
|
|
63
|
+
}
|
package/package.json
CHANGED
|
@@ -1,46 +1,51 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@flowconsole/cli",
|
|
3
|
-
"version": "0.0.0-beta-
|
|
4
|
-
"description": "FlowConsole CLI for
|
|
5
|
-
"
|
|
6
|
-
"types": "dist/index.d.ts",
|
|
3
|
+
"version": "0.0.0-beta-20260428172441",
|
|
4
|
+
"description": "FlowConsole CLI (fc / fcon / flowconsole) — architecture scanner and rule engine for your CI/CD",
|
|
5
|
+
"license": "Apache-2.0",
|
|
7
6
|
"bin": {
|
|
8
|
-
"
|
|
7
|
+
"fc": "./bin/fc.js",
|
|
8
|
+
"fcon": "./bin/fc.js",
|
|
9
|
+
"flowconsole": "./bin/fc.js"
|
|
10
|
+
},
|
|
11
|
+
"cliVersion": "0.2.0-beta-20260428172441",
|
|
12
|
+
"engines": {
|
|
13
|
+
"node": ">=18.0.0"
|
|
9
14
|
},
|
|
15
|
+
"os": [
|
|
16
|
+
"darwin",
|
|
17
|
+
"linux",
|
|
18
|
+
"win32"
|
|
19
|
+
],
|
|
20
|
+
"cpu": [
|
|
21
|
+
"x64",
|
|
22
|
+
"arm64"
|
|
23
|
+
],
|
|
10
24
|
"files": [
|
|
11
|
-
"
|
|
12
|
-
"
|
|
13
|
-
"
|
|
14
|
-
"
|
|
15
|
-
"!dist/test_sample.*"
|
|
25
|
+
"bin/",
|
|
26
|
+
"binaries/",
|
|
27
|
+
"postinstall.js",
|
|
28
|
+
"README.md"
|
|
16
29
|
],
|
|
17
|
-
"
|
|
18
|
-
"
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
"
|
|
27
|
-
"
|
|
28
|
-
"
|
|
29
|
-
"uuidv7": "^1.1.0",
|
|
30
|
-
"yargs": "^18.0.0",
|
|
31
|
-
"@flowconsole/core": "0.0.0-beta-20260128114951"
|
|
30
|
+
"keywords": [
|
|
31
|
+
"architecture",
|
|
32
|
+
"architecture-as-code",
|
|
33
|
+
"diagrams",
|
|
34
|
+
"cli",
|
|
35
|
+
"flowconsole"
|
|
36
|
+
],
|
|
37
|
+
"author": "FlowConsole <v@flowconsole.tech> (https://flowconsole.tech)",
|
|
38
|
+
"repository": {
|
|
39
|
+
"type": "git",
|
|
40
|
+
"url": "https://github.com/flowconsole/flowconsole.git",
|
|
41
|
+
"directory": "packages/cli"
|
|
32
42
|
},
|
|
33
|
-
"
|
|
34
|
-
|
|
35
|
-
"
|
|
36
|
-
"ts-node": "^10.9.2",
|
|
37
|
-
"typescript": "^5.9.3",
|
|
38
|
-
"vitest": "^4.0.17",
|
|
39
|
-
"@flowconsole/sdk": "0.0.0-beta-20260128114951"
|
|
43
|
+
"homepage": "https://flowconsole.tech",
|
|
44
|
+
"bugs": {
|
|
45
|
+
"url": "https://github.com/flowconsole/flowconsole/issues"
|
|
40
46
|
},
|
|
41
47
|
"scripts": {
|
|
42
|
-
"
|
|
43
|
-
"
|
|
44
|
-
"test:watch": "vitest --config vitest.config.ts"
|
|
48
|
+
"postinstall": "node postinstall.js",
|
|
49
|
+
"dev:reinstall": "bash scripts/dev-reinstall.sh"
|
|
45
50
|
}
|
|
46
51
|
}
|
package/postinstall.js
ADDED
|
@@ -0,0 +1,269 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
"use strict";
|
|
4
|
+
|
|
5
|
+
const https = require("https");
|
|
6
|
+
const http = require("http");
|
|
7
|
+
const fs = require("fs");
|
|
8
|
+
const path = require("path");
|
|
9
|
+
const { createHash } = require("crypto");
|
|
10
|
+
const { execFileSync } = require("child_process");
|
|
11
|
+
const { pipeline } = require("stream/promises");
|
|
12
|
+
const zlib = require("zlib");
|
|
13
|
+
|
|
14
|
+
const PLATFORM_MAP = {
|
|
15
|
+
darwin: "osx",
|
|
16
|
+
linux: "linux",
|
|
17
|
+
win32: "win",
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
const ARCH_MAP = {
|
|
21
|
+
x64: "x64",
|
|
22
|
+
arm64: "arm64",
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
const SUPPORTED_RIDS = [
|
|
26
|
+
"linux-x64",
|
|
27
|
+
"linux-arm64",
|
|
28
|
+
"osx-x64",
|
|
29
|
+
"osx-arm64",
|
|
30
|
+
"win-x64",
|
|
31
|
+
"win-arm64",
|
|
32
|
+
];
|
|
33
|
+
|
|
34
|
+
function getRid() {
|
|
35
|
+
const platform = PLATFORM_MAP[process.platform];
|
|
36
|
+
const arch = ARCH_MAP[process.arch];
|
|
37
|
+
|
|
38
|
+
if (!platform || !arch) {
|
|
39
|
+
const supported = SUPPORTED_RIDS.join(", ");
|
|
40
|
+
console.error(
|
|
41
|
+
`Unsupported platform: ${process.platform}-${process.arch}\n` +
|
|
42
|
+
`Supported platforms: ${supported}\n` +
|
|
43
|
+
`You can download the binary manually from the GitHub Releases page.`
|
|
44
|
+
);
|
|
45
|
+
process.exit(1);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
return `${platform}-${arch}`;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
function getVersion() {
|
|
52
|
+
const pkg = JSON.parse(
|
|
53
|
+
fs.readFileSync(path.join(__dirname, "package.json"), "utf8")
|
|
54
|
+
);
|
|
55
|
+
return pkg.cliVersion || "0.2.0-alpha";
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
function getDownloadBaseUrl() {
|
|
59
|
+
// Allow override via env for mirrors or private registries
|
|
60
|
+
if (process.env.FLOWCONSOLE_CLI_DOWNLOAD_URL) {
|
|
61
|
+
return process.env.FLOWCONSOLE_CLI_DOWNLOAD_URL;
|
|
62
|
+
}
|
|
63
|
+
const pkg = JSON.parse(
|
|
64
|
+
fs.readFileSync(path.join(__dirname, "package.json"), "utf8")
|
|
65
|
+
);
|
|
66
|
+
const repoUrl =
|
|
67
|
+
(pkg.repository && pkg.repository.url) || "";
|
|
68
|
+
// Extract GitHub owner/repo from git URL
|
|
69
|
+
const match = repoUrl.match(
|
|
70
|
+
/github\.com[/:]([^/]+\/[^/.]+?)(?:\.git)?$/
|
|
71
|
+
);
|
|
72
|
+
if (match) {
|
|
73
|
+
return `https://github.com/${match[1]}/releases/download`;
|
|
74
|
+
}
|
|
75
|
+
return "https://github.com/flowconsole/flowconsole/releases/download";
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
function fetch(url, maxRedirects = 5) {
|
|
79
|
+
return new Promise((resolve, reject) => {
|
|
80
|
+
const client = url.startsWith("https") ? https : http;
|
|
81
|
+
client
|
|
82
|
+
.get(url, { headers: { "User-Agent": "flowconsole-cli-npm" } }, (res) => {
|
|
83
|
+
//Follow redirects (GitHub releases redirect to S3)
|
|
84
|
+
if (
|
|
85
|
+
(res.statusCode === 301 ||
|
|
86
|
+
res.statusCode === 302 ||
|
|
87
|
+
res.statusCode === 307) &&
|
|
88
|
+
res.headers.location
|
|
89
|
+
) {
|
|
90
|
+
if (maxRedirects <= 0) {
|
|
91
|
+
reject(new Error(`Too many redirects following ${url}`));
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
return fetch(res.headers.location, maxRedirects - 1).then(resolve, reject);
|
|
95
|
+
}
|
|
96
|
+
if (res.statusCode !== 200) {
|
|
97
|
+
reject(
|
|
98
|
+
new Error(
|
|
99
|
+
`Failed to download ${url}: HTTP ${res.statusCode}`
|
|
100
|
+
)
|
|
101
|
+
);
|
|
102
|
+
res.resume();
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
resolve(res);
|
|
106
|
+
})
|
|
107
|
+
.on("error", reject);
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
async function downloadToBuffer(url) {
|
|
112
|
+
const res = await fetch(url);
|
|
113
|
+
const chunks = [];
|
|
114
|
+
for await (const chunk of res) {
|
|
115
|
+
chunks.push(chunk);
|
|
116
|
+
}
|
|
117
|
+
return Buffer.concat(chunks);
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
async function downloadAndVerify(rid, version) {
|
|
121
|
+
const baseUrl = getDownloadBaseUrl();
|
|
122
|
+
const tag = `cli-v${version}`;
|
|
123
|
+
const isWindows = rid.startsWith("win-");
|
|
124
|
+
const archiveExt = isWindows ? "zip" : "tar.gz";
|
|
125
|
+
const archiveName = `fcon-${rid}.${archiveExt}`;
|
|
126
|
+
const checksumName = `fcon-${rid}.sha256`;
|
|
127
|
+
|
|
128
|
+
const archiveUrl = `${baseUrl}/${tag}/${archiveName}`;
|
|
129
|
+
const checksumUrl = `${baseUrl}/${tag}/${checksumName}`;
|
|
130
|
+
|
|
131
|
+
console.log(`Downloading FlowConsole CLI ${version} for ${rid}...`);
|
|
132
|
+
console.log(` Archive: ${archiveUrl}`);
|
|
133
|
+
|
|
134
|
+
// Download checksum first
|
|
135
|
+
let expectedChecksum;
|
|
136
|
+
try {
|
|
137
|
+
const checksumBuf = await downloadToBuffer(checksumUrl);
|
|
138
|
+
// Format: "<hash> <filename>" or "<hash> <filename>"
|
|
139
|
+
expectedChecksum = checksumBuf.toString("utf8").trim().split(/\s+/)[0];
|
|
140
|
+
} catch (err) {
|
|
141
|
+
console.error(
|
|
142
|
+
`Failed to download checksum file: ${err.message}\n` +
|
|
143
|
+
`Cannot verify binary integrity. Aborting installation.\n` +
|
|
144
|
+
`Download the binary manually from: ${archiveUrl}`
|
|
145
|
+
);
|
|
146
|
+
process.exit(1);
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// Download archive
|
|
150
|
+
const archiveBuf = await downloadToBuffer(archiveUrl);
|
|
151
|
+
|
|
152
|
+
// Verify checksum
|
|
153
|
+
const actualChecksum = createHash("sha256")
|
|
154
|
+
.update(archiveBuf)
|
|
155
|
+
.digest("hex");
|
|
156
|
+
|
|
157
|
+
if (actualChecksum !== expectedChecksum) {
|
|
158
|
+
console.error(
|
|
159
|
+
`SHA-256 checksum mismatch!\n` +
|
|
160
|
+
` Expected: ${expectedChecksum}\n` +
|
|
161
|
+
` Actual: ${actualChecksum}\n` +
|
|
162
|
+
`The downloaded binary may be corrupted or tampered with.\n` +
|
|
163
|
+
`Aborting installation.`
|
|
164
|
+
);
|
|
165
|
+
process.exit(1);
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
console.log(` Checksum verified: ${actualChecksum.substring(0, 16)}...`);
|
|
169
|
+
|
|
170
|
+
return { archiveBuf, isWindows };
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
function extractTarGz(buf, destDir) {
|
|
174
|
+
// Write to temp file and use system tar (Node.js has no built-in tar extraction)
|
|
175
|
+
const tmpFile = path.join(destDir, "_archive.tar.gz");
|
|
176
|
+
fs.writeFileSync(tmpFile, buf);
|
|
177
|
+
try {
|
|
178
|
+
execFileSync("tar", ["xzf", tmpFile, "-C", destDir], {
|
|
179
|
+
stdio: "pipe",
|
|
180
|
+
});
|
|
181
|
+
} finally {
|
|
182
|
+
try {
|
|
183
|
+
fs.unlinkSync(tmpFile);
|
|
184
|
+
} catch (_) {}
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
function extractZip(buf, destDir) {
|
|
189
|
+
// Write to temp file and use system tools
|
|
190
|
+
const tmpFile = path.join(destDir, "_archive.zip");
|
|
191
|
+
fs.writeFileSync(tmpFile, buf);
|
|
192
|
+
try {
|
|
193
|
+
// Try unzip first (available on most systems including Windows via Git Bash)
|
|
194
|
+
try {
|
|
195
|
+
execFileSync("unzip", ["-o", tmpFile, "-d", destDir], {
|
|
196
|
+
stdio: "pipe",
|
|
197
|
+
});
|
|
198
|
+
} catch (_) {
|
|
199
|
+
// Fallback to PowerShell on Windows
|
|
200
|
+
// Escape single quotes for PowerShell (double them inside single-quoted strings)
|
|
201
|
+
const escapedTmpFile = tmpFile.replace(/'/g, "''");
|
|
202
|
+
const escapedDestDir = destDir.replace(/'/g, "''");
|
|
203
|
+
execFileSync(
|
|
204
|
+
"powershell",
|
|
205
|
+
[
|
|
206
|
+
"-NoProfile",
|
|
207
|
+
"-Command",
|
|
208
|
+
`Expand-Archive -Path '${escapedTmpFile}' -DestinationPath '${escapedDestDir}' -Force`,
|
|
209
|
+
],
|
|
210
|
+
{ stdio: "pipe" }
|
|
211
|
+
);
|
|
212
|
+
}
|
|
213
|
+
} finally {
|
|
214
|
+
try {
|
|
215
|
+
fs.unlinkSync(tmpFile);
|
|
216
|
+
} catch (_) {}
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
async function main() {
|
|
221
|
+
// Skip in CI if FLOWCONSOLE_CLI_SKIP_DOWNLOAD is set
|
|
222
|
+
if (process.env.FLOWCONSOLE_CLI_SKIP_DOWNLOAD === "1") {
|
|
223
|
+
console.log("FLOWCONSOLE_CLI_SKIP_DOWNLOAD=1, skipping binary download.");
|
|
224
|
+
return;
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
const rid = getRid();
|
|
228
|
+
const version = getVersion();
|
|
229
|
+
const binDir = path.join(__dirname, "binaries", rid);
|
|
230
|
+
|
|
231
|
+
// Check if already downloaded
|
|
232
|
+
const binaryName = rid.startsWith("win-") ? "fcon.exe" : "fcon";
|
|
233
|
+
const binaryPath = path.join(binDir, binaryName);
|
|
234
|
+
if (fs.existsSync(binaryPath)) {
|
|
235
|
+
console.log(`FlowConsole CLI binary already exists at ${binaryPath}`);
|
|
236
|
+
return;
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
const { archiveBuf, isWindows } = await downloadAndVerify(rid, version);
|
|
240
|
+
|
|
241
|
+
// Extract
|
|
242
|
+
fs.mkdirSync(binDir, { recursive: true });
|
|
243
|
+
|
|
244
|
+
if (isWindows) {
|
|
245
|
+
extractZip(archiveBuf, binDir);
|
|
246
|
+
} else {
|
|
247
|
+
extractTarGz(archiveBuf, binDir);
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
// chmod +x on POSIX
|
|
251
|
+
if (!isWindows && fs.existsSync(binaryPath)) {
|
|
252
|
+
fs.chmodSync(binaryPath, 0o755);
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
if (!fs.existsSync(binaryPath)) {
|
|
256
|
+
console.error(
|
|
257
|
+
`Binary not found after extraction at ${binaryPath}.\n` +
|
|
258
|
+
`Archive may have a different structure. Please report this issue.`
|
|
259
|
+
);
|
|
260
|
+
process.exit(1);
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
console.log(`FlowConsole CLI ${version} installed successfully for ${rid}.`);
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
main().catch((err) => {
|
|
267
|
+
console.error(`Failed to install FlowConsole CLI: ${err.message}`);
|
|
268
|
+
process.exit(1);
|
|
269
|
+
});
|
package/dist/index.d.ts
DELETED
package/dist/index.js
DELETED
|
@@ -1,104 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
"use strict";
|
|
3
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
-
const fs = require("node:fs");
|
|
5
|
-
const path = require("node:path");
|
|
6
|
-
const yargs = require("yargs");
|
|
7
|
-
const helpers = require("yargs/helpers");
|
|
8
|
-
const napi_1 = require("@ast-grep/napi");
|
|
9
|
-
const parser_registry_1 = require("@flowconsole/core/infra/parser-registry");
|
|
10
|
-
const language_detector_1 = require("@flowconsole/core/infra/language-detector");
|
|
11
|
-
const typescript_parser_1 = require("@flowconsole/core/parsers/typescript-parser");
|
|
12
|
-
const python_parser_1 = require("@flowconsole/core/parsers/python-parser");
|
|
13
|
-
const csharp_parser_1 = require("@flowconsole/core/parsers/csharp-parser");
|
|
14
|
-
const java_parser_1 = require("@flowconsole/core/parsers/java-parser");
|
|
15
|
-
const go_parser_1 = require("@flowconsole/core/parsers/go-parser");
|
|
16
|
-
// ast-grep lang packages export CommonJS objects without a default field; make sure we pass the raw object.
|
|
17
|
-
function loadLang(mod) {
|
|
18
|
-
return (mod?.default ?? mod);
|
|
19
|
-
}
|
|
20
|
-
const csharp = loadLang(require('@ast-grep/lang-csharp'));
|
|
21
|
-
const python = loadLang(require('@ast-grep/lang-python'));
|
|
22
|
-
const java = loadLang(require('@ast-grep/lang-java'));
|
|
23
|
-
const go = loadLang(require('@ast-grep/lang-go'));
|
|
24
|
-
(0, napi_1.registerDynamicLanguage)({
|
|
25
|
-
csharp,
|
|
26
|
-
python,
|
|
27
|
-
java,
|
|
28
|
-
go,
|
|
29
|
-
});
|
|
30
|
-
const registry = new parser_registry_1.ParserRegistry();
|
|
31
|
-
registry.register(new typescript_parser_1.TypeScriptParser());
|
|
32
|
-
registry.register(new python_parser_1.PythonParser());
|
|
33
|
-
registry.register(new csharp_parser_1.CSharpParser());
|
|
34
|
-
registry.register(new java_parser_1.JavaParser());
|
|
35
|
-
registry.register(new go_parser_1.GoParser());
|
|
36
|
-
async function parseFile(options) {
|
|
37
|
-
const { file, lang, output } = options;
|
|
38
|
-
const source = fs.readFileSync(path.resolve(file), 'utf8');
|
|
39
|
-
const detectedLang = lang || language_detector_1.LanguageDetector.detectFromFilePath(file);
|
|
40
|
-
if (!detectedLang) {
|
|
41
|
-
console.error(`Error: Could not detect language for ${file}. Please specify with --lang`);
|
|
42
|
-
console.error(`\nSupported languages: ${registry.getSupportedLanguages().join(', ')}`);
|
|
43
|
-
process.exit(1);
|
|
44
|
-
}
|
|
45
|
-
const parser = registry.getParser(detectedLang);
|
|
46
|
-
if (!parser) {
|
|
47
|
-
console.error(`Error: No parser available for language: ${detectedLang}`);
|
|
48
|
-
console.error(`Supported languages: ${registry.getSupportedLanguages().join(', ')}`);
|
|
49
|
-
process.exit(1);
|
|
50
|
-
}
|
|
51
|
-
try {
|
|
52
|
-
const result = parser.parse(source);
|
|
53
|
-
const json = JSON.stringify(result, null, 2);
|
|
54
|
-
if (output) {
|
|
55
|
-
fs.writeFileSync(output, json, 'utf8');
|
|
56
|
-
console.log(`Output written to: ${output}`);
|
|
57
|
-
}
|
|
58
|
-
else {
|
|
59
|
-
console.log(json);
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
catch (error) {
|
|
63
|
-
console.error(`Error parsing file:`, error);
|
|
64
|
-
process.exit(1);
|
|
65
|
-
}
|
|
66
|
-
}
|
|
67
|
-
yargs(helpers.hideBin(process.argv))
|
|
68
|
-
.command('parse <file>', 'Parse architecture file and extract nodes and flows', (yargs) => {
|
|
69
|
-
return yargs
|
|
70
|
-
.positional('file', {
|
|
71
|
-
describe: 'Path to architecture file',
|
|
72
|
-
type: 'string',
|
|
73
|
-
demandOption: true,
|
|
74
|
-
})
|
|
75
|
-
.option('lang', {
|
|
76
|
-
alias: 'l',
|
|
77
|
-
describe: 'Language of the source file',
|
|
78
|
-
choices: ['typescript', 'python', 'csharp', 'java', 'go'],
|
|
79
|
-
type: 'string',
|
|
80
|
-
})
|
|
81
|
-
.option('output', {
|
|
82
|
-
alias: 'o',
|
|
83
|
-
describe: 'Output file path (default: stdout)',
|
|
84
|
-
type: 'string',
|
|
85
|
-
});
|
|
86
|
-
}, (argv) => {
|
|
87
|
-
parseFile(argv).catch((error) => {
|
|
88
|
-
console.error(error);
|
|
89
|
-
process.exit(1);
|
|
90
|
-
});
|
|
91
|
-
})
|
|
92
|
-
.command('languages', 'List supported languages', () => { }, () => {
|
|
93
|
-
console.log('Supported languages:');
|
|
94
|
-
registry.getSupportedLanguages().forEach((lang) => {
|
|
95
|
-
console.log(` - ${lang}`);
|
|
96
|
-
});
|
|
97
|
-
})
|
|
98
|
-
.demandCommand(1, 'You must provide a command')
|
|
99
|
-
.help()
|
|
100
|
-
.alias('help', 'h')
|
|
101
|
-
.version('0.1.0')
|
|
102
|
-
.alias('version', 'v')
|
|
103
|
-
.parse();
|
|
104
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../index.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\nimport * as fs from 'node:fs';\nimport * as path from 'node:path';\nimport * as yargs from 'yargs';\nimport * as helpers from 'yargs/helpers';\nimport { registerDynamicLanguage } from '@ast-grep/napi';\nimport { ParserRegistry } from '@flowconsole/core/infra/parser-registry';\nimport { LanguageDetector } from '@flowconsole/core/infra/language-detector';\nimport { TypeScriptParser } from '@flowconsole/core/parsers/typescript-parser';\nimport { PythonParser } from '@flowconsole/core/parsers/python-parser';\nimport { CSharpParser } from '@flowconsole/core/parsers/csharp-parser';\nimport { JavaParser } from '@flowconsole/core/parsers/java-parser';\nimport { GoParser } from '@flowconsole/core/parsers/go-parser';\nimport type { SupportedLanguage } from '@flowconsole/core/types/common';\ntype LangRegistration = {\n  libraryPath: string;\n  extensions: string[];\n  languageSymbol?: string;\n  metaVarChar?: string;\n  expandoChar?: string;\n};\n\n// ast-grep lang packages export CommonJS objects without a default field; make sure we pass the raw object.\nfunction loadLang(mod: any): LangRegistration {\n  return (mod?.default ?? mod) as LangRegistration;\n}\n\nconst csharp = loadLang(require('@ast-grep/lang-csharp'));\nconst python = loadLang(require('@ast-grep/lang-python'));\nconst java = loadLang(require('@ast-grep/lang-java'));\nconst go = loadLang(require('@ast-grep/lang-go'));\n\ninterface ParseOptions {\n  file: string;\n  lang?: SupportedLanguage;\n  output?: string;\n}\n\nregisterDynamicLanguage({\n  csharp,\n  python,\n  java,\n  go,\n});\n\nconst registry = new ParserRegistry();\n\nregistry.register(new TypeScriptParser());\nregistry.register(new PythonParser());\nregistry.register(new CSharpParser());\nregistry.register(new JavaParser());\nregistry.register(new GoParser());\n\nasync function parseFile(options: ParseOptions): Promise<void> {\n  const { file, lang, output } = options;\n\n  const source = fs.readFileSync(path.resolve(file), 'utf8');\n\n  const detectedLang = lang || LanguageDetector.detectFromFilePath(file);\n\n  if (!detectedLang) {\n    console.error(`Error: Could not detect language for ${file}. Please specify with --lang`);\n    console.error(`\\nSupported languages: ${registry.getSupportedLanguages().join(', ')}`);\n    process.exit(1);\n  }\n\n  const parser = registry.getParser(detectedLang);\n\n  if (!parser) {\n    console.error(`Error: No parser available for language: ${detectedLang}`);\n    console.error(`Supported languages: ${registry.getSupportedLanguages().join(', ')}`);\n    process.exit(1);\n  }\n\n  try {\n    const result = parser.parse(source);\n    const json = JSON.stringify(result, null, 2);\n\n    if (output) {\n      fs.writeFileSync(output, json, 'utf8');\n      console.log(`Output written to: ${output}`);\n    } else {\n      console.log(json);\n    }\n  } catch (error) {\n    console.error(`Error parsing file:`, error);\n    process.exit(1);\n  }\n}\n\nyargs(helpers.hideBin(process.argv))\n  .command(\n    'parse <file>',\n    'Parse architecture file and extract nodes and flows',\n    (yargs) => {\n      return yargs\n        .positional('file', {\n          describe: 'Path to architecture file',\n          type: 'string',\n          demandOption: true,\n        })\n        .option('lang', {\n          alias: 'l',\n          describe: 'Language of the source file',\n          choices: ['typescript', 'python', 'csharp', 'java', 'go'] as const,\n          type: 'string',\n        })\n        .option('output', {\n          alias: 'o',\n          describe: 'Output file path (default: stdout)',\n          type: 'string',\n        });\n    },\n    (argv) => {\n      parseFile(argv as ParseOptions).catch((error) => {\n        console.error(error);\n        process.exit(1);\n      });\n    }\n  )\n  .command(\n    'languages',\n    'List supported languages',\n    () => {},\n    () => {\n      console.log('Supported languages:');\n      registry.getSupportedLanguages().forEach((lang) => {\n        console.log(`  - ${lang}`);\n      });\n    }\n  )\n  .demandCommand(1, 'You must provide a command')\n  .help()\n  .alias('help', 'h')\n  .version('0.1.0')\n  .alias('version', 'v')\n  .parse();\n"]}
|