@specific.dev/cli 0.1.37
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/dist/admin/404/index.html +1 -0
- package/dist/admin/404.html +1 -0
- package/dist/admin/__next.__PAGE__.txt +9 -0
- package/dist/admin/__next._full.txt +20 -0
- package/dist/admin/__next._head.txt +6 -0
- package/dist/admin/__next._index.txt +5 -0
- package/dist/admin/__next._tree.txt +4 -0
- package/dist/admin/_next/static/0zkv3YeV6IWOWVWE1S1A1/_buildManifest.js +11 -0
- package/dist/admin/_next/static/0zkv3YeV6IWOWVWE1S1A1/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/0zkv3YeV6IWOWVWE1S1A1/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/1o2O2QPOKhuZyxcNaCjq8/_buildManifest.js +11 -0
- package/dist/admin/_next/static/1o2O2QPOKhuZyxcNaCjq8/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/1o2O2QPOKhuZyxcNaCjq8/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/2ZtqtFX2EkuM82mTM7NWT/_buildManifest.js +11 -0
- package/dist/admin/_next/static/2ZtqtFX2EkuM82mTM7NWT/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/2ZtqtFX2EkuM82mTM7NWT/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/2vLF5bo4ZtUfhs8bdQBoF/_buildManifest.js +11 -0
- package/dist/admin/_next/static/2vLF5bo4ZtUfhs8bdQBoF/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/2vLF5bo4ZtUfhs8bdQBoF/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/8_8s51m7RbLVvbIechG-b/_buildManifest.js +11 -0
- package/dist/admin/_next/static/8_8s51m7RbLVvbIechG-b/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/8_8s51m7RbLVvbIechG-b/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/C4Y44KtLnP684m57gN3Ga/_buildManifest.js +11 -0
- package/dist/admin/_next/static/C4Y44KtLnP684m57gN3Ga/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/C4Y44KtLnP684m57gN3Ga/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/DBeh36kAJnZBeBMJSPBW8/_buildManifest.js +11 -0
- package/dist/admin/_next/static/DBeh36kAJnZBeBMJSPBW8/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/DBeh36kAJnZBeBMJSPBW8/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/REKEUu6DP2t99jKSAqNOu/_buildManifest.js +11 -0
- package/dist/admin/_next/static/REKEUu6DP2t99jKSAqNOu/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/REKEUu6DP2t99jKSAqNOu/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/TDGT3mkKtTe0w_CxujLxI/_buildManifest.js +11 -0
- package/dist/admin/_next/static/TDGT3mkKtTe0w_CxujLxI/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/TDGT3mkKtTe0w_CxujLxI/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/ZHMJ-g7mAlAMt_2uCXeEk/_buildManifest.js +11 -0
- package/dist/admin/_next/static/ZHMJ-g7mAlAMt_2uCXeEk/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/ZHMJ-g7mAlAMt_2uCXeEk/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/a6JDAB-CdvWPvS4sBgJji/_buildManifest.js +11 -0
- package/dist/admin/_next/static/a6JDAB-CdvWPvS4sBgJji/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/a6JDAB-CdvWPvS4sBgJji/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/abFUeBpymhlx-IxygVnM9/_buildManifest.js +11 -0
- package/dist/admin/_next/static/abFUeBpymhlx-IxygVnM9/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/abFUeBpymhlx-IxygVnM9/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/chunks/0afa5e999b6c353a.js +1 -0
- package/dist/admin/_next/static/chunks/1584f10ea1cebcb2.js +4 -0
- package/dist/admin/_next/static/chunks/195bbec70cfcd241.js +2 -0
- package/dist/admin/_next/static/chunks/2583656ea9ac4ad6.js +5 -0
- package/dist/admin/_next/static/chunks/465f799faf41e6df.js +1 -0
- package/dist/admin/_next/static/chunks/4ab079bdcb131778.js +5 -0
- package/dist/admin/_next/static/chunks/4b5ae5ebb2087f0d.js +2 -0
- package/dist/admin/_next/static/chunks/605800ff25160d05.js +1 -0
- package/dist/admin/_next/static/chunks/656b870f0567ed5f.js +1 -0
- package/dist/admin/_next/static/chunks/71098a6cd6181738.css +3 -0
- package/dist/admin/_next/static/chunks/806bdb8e4a6a9b95.js +4 -0
- package/dist/admin/_next/static/chunks/895a6f91f0b479fb.js +2 -0
- package/dist/admin/_next/static/chunks/9032f4a1aac1ca5d.css +3 -0
- package/dist/admin/_next/static/chunks/a28af2dc6f5fbaad.js +1 -0
- package/dist/admin/_next/static/chunks/a6dad97d9634a72d.js +1 -0
- package/dist/admin/_next/static/chunks/a6dad97d9634a72d.js.map +1 -0
- package/dist/admin/_next/static/chunks/b4205fc2f84bda68.css +3 -0
- package/dist/admin/_next/static/chunks/c1a750c25bc8d092.js +1 -0
- package/dist/admin/_next/static/chunks/c3d30f6f144dca51.js +2 -0
- package/dist/admin/_next/static/chunks/cbf55ce8731457ae.js +2 -0
- package/dist/admin/_next/static/chunks/d2be314c3ece3fbe.js +1 -0
- package/dist/admin/_next/static/chunks/de6af6d8adf8b50a.js +5 -0
- package/dist/admin/_next/static/chunks/f0c001244d275aab.js +5 -0
- package/dist/admin/_next/static/chunks/fde89fd76ad6a3d0.css +3 -0
- package/dist/admin/_next/static/chunks/ff1a16fafef87110.js +1 -0
- package/dist/admin/_next/static/chunks/turbopack-a3d691c83d4b1778.js +4 -0
- package/dist/admin/_next/static/chunks/turbopack-e5185af8e7c716ca.js +4 -0
- package/dist/admin/_next/static/dcjrfD44GuB6g8bZ6BcFm/_buildManifest.js +11 -0
- package/dist/admin/_next/static/dcjrfD44GuB6g8bZ6BcFm/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/dcjrfD44GuB6g8bZ6BcFm/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/fTEa8Scx922n-dMuqN3Vc/_buildManifest.js +11 -0
- package/dist/admin/_next/static/fTEa8Scx922n-dMuqN3Vc/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/fTEa8Scx922n-dMuqN3Vc/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/kZNwyhHft01wPtJ_AvqQT/_buildManifest.js +11 -0
- package/dist/admin/_next/static/kZNwyhHft01wPtJ_AvqQT/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/kZNwyhHft01wPtJ_AvqQT/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/kyVInC6N3DY6NWnjkLQ4J/_buildManifest.js +11 -0
- package/dist/admin/_next/static/kyVInC6N3DY6NWnjkLQ4J/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/kyVInC6N3DY6NWnjkLQ4J/_ssgManifest.js +1 -0
- package/dist/admin/_next/static/media/4fa387ec64143e14-s.c1fdd6c2.woff2 +0 -0
- package/dist/admin/_next/static/media/7178b3e590c64307-s.b97b3418.woff2 +0 -0
- package/dist/admin/_next/static/media/797e433ab948586e-s.p.dbea232f.woff2 +0 -0
- package/dist/admin/_next/static/media/8a480f0b521d4e75-s.8e0177b5.woff2 +0 -0
- package/dist/admin/_next/static/media/bbc41e54d2fcbd21-s.799d8ef8.woff2 +0 -0
- package/dist/admin/_next/static/media/caa3a2e1cccd8315-s.p.853070df.woff2 +0 -0
- package/dist/admin/_next/static/media/favicon.0b3bf435.ico +0 -0
- package/dist/admin/_next/static/twedXBdhyxpjS68UQoNAc/_buildManifest.js +11 -0
- package/dist/admin/_next/static/twedXBdhyxpjS68UQoNAc/_clientMiddlewareManifest.json +1 -0
- package/dist/admin/_next/static/twedXBdhyxpjS68UQoNAc/_ssgManifest.js +1 -0
- package/dist/admin/_not-found/__next._full.txt +14 -0
- package/dist/admin/_not-found/__next._head.txt +6 -0
- package/dist/admin/_not-found/__next._index.txt +5 -0
- package/dist/admin/_not-found/__next._not-found.__PAGE__.txt +5 -0
- package/dist/admin/_not-found/__next._not-found.txt +4 -0
- package/dist/admin/_not-found/__next._tree.txt +2 -0
- package/dist/admin/_not-found/index.html +1 -0
- package/dist/admin/_not-found/index.txt +14 -0
- package/dist/admin/databases/__next._full.txt +20 -0
- package/dist/admin/databases/__next._head.txt +6 -0
- package/dist/admin/databases/__next._index.txt +5 -0
- package/dist/admin/databases/__next._tree.txt +4 -0
- package/dist/admin/databases/__next.databases.__PAGE__.txt +9 -0
- package/dist/admin/databases/__next.databases.txt +4 -0
- package/dist/admin/databases/index.html +1 -0
- package/dist/admin/databases/index.txt +20 -0
- package/dist/admin/favicon.ico +0 -0
- package/dist/admin/file.svg +1 -0
- package/dist/admin/globe.svg +1 -0
- package/dist/admin/index.html +1 -0
- package/dist/admin/index.txt +20 -0
- package/dist/admin/next.svg +1 -0
- package/dist/admin/vercel.svg +1 -0
- package/dist/admin/window.svg +1 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +190523 -0
- package/dist/cli.js.map +1 -0
- package/dist/commands/check.d.ts +2 -0
- package/dist/commands/check.d.ts.map +1 -0
- package/dist/commands/check.js +104 -0
- package/dist/commands/check.js.map +1 -0
- package/dist/commands/clean.d.ts +2 -0
- package/dist/commands/clean.d.ts.map +1 -0
- package/dist/commands/clean.js +70 -0
- package/dist/commands/clean.js.map +1 -0
- package/dist/commands/deploy.d.ts +2 -0
- package/dist/commands/deploy.d.ts.map +1 -0
- package/dist/commands/deploy.js +11 -0
- package/dist/commands/deploy.js.map +1 -0
- package/dist/commands/dev.d.ts +2 -0
- package/dist/commands/dev.d.ts.map +1 -0
- package/dist/commands/dev.js +398 -0
- package/dist/commands/dev.js.map +1 -0
- package/dist/commands/docs.d.ts +2 -0
- package/dist/commands/docs.d.ts.map +1 -0
- package/dist/commands/docs.js +32 -0
- package/dist/commands/docs.js.map +1 -0
- package/dist/commands/exec.d.ts +2 -0
- package/dist/commands/exec.d.ts.map +1 -0
- package/dist/commands/exec.js +178 -0
- package/dist/commands/exec.js.map +1 -0
- package/dist/commands/init.d.ts +2 -0
- package/dist/commands/init.d.ts.map +1 -0
- package/dist/commands/init.js +339 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/psql.d.ts +2 -0
- package/dist/commands/psql.d.ts.map +1 -0
- package/dist/commands/psql.js +53 -0
- package/dist/commands/psql.js.map +1 -0
- package/dist/commands/secrets.d.ts +3 -0
- package/dist/commands/secrets.d.ts.map +1 -0
- package/dist/commands/secrets.js +136 -0
- package/dist/commands/secrets.js.map +1 -0
- package/dist/docs/builds.md +71 -0
- package/dist/docs/index.md +30 -0
- package/dist/docs/integrations/drizzle.md +83 -0
- package/dist/docs/integrations/nextjs.md +45 -0
- package/dist/docs/integrations/prisma.md +110 -0
- package/dist/docs/postgres.md +41 -0
- package/dist/docs/redis.md +33 -0
- package/dist/docs/secrets-config.md +141 -0
- package/dist/docs/services.md +325 -0
- package/dist/docs/storage.md +62 -0
- package/dist/docs/sync.md +66 -0
- package/dist/lib/dev/database-manager.d.ts +17 -0
- package/dist/lib/dev/database-manager.d.ts.map +1 -0
- package/dist/lib/dev/database-manager.js +114 -0
- package/dist/lib/dev/database-manager.js.map +1 -0
- package/dist/lib/dev/env-resolver.d.ts +14 -0
- package/dist/lib/dev/env-resolver.d.ts.map +1 -0
- package/dist/lib/dev/env-resolver.js +109 -0
- package/dist/lib/dev/env-resolver.js.map +1 -0
- package/dist/lib/dev/http-proxy.d.ts +11 -0
- package/dist/lib/dev/http-proxy.d.ts.map +1 -0
- package/dist/lib/dev/http-proxy.js +165 -0
- package/dist/lib/dev/http-proxy.js.map +1 -0
- package/dist/lib/dev/index.d.ts +11 -0
- package/dist/lib/dev/index.d.ts.map +1 -0
- package/dist/lib/dev/index.js +7 -0
- package/dist/lib/dev/index.js.map +1 -0
- package/dist/lib/dev/instance-state.d.ts +45 -0
- package/dist/lib/dev/instance-state.d.ts.map +1 -0
- package/dist/lib/dev/instance-state.js +213 -0
- package/dist/lib/dev/instance-state.js.map +1 -0
- package/dist/lib/dev/port-allocator.d.ts +5 -0
- package/dist/lib/dev/port-allocator.d.ts.map +1 -0
- package/dist/lib/dev/port-allocator.js +21 -0
- package/dist/lib/dev/port-allocator.js.map +1 -0
- package/dist/lib/dev/service-runner.d.ts +16 -0
- package/dist/lib/dev/service-runner.d.ts.map +1 -0
- package/dist/lib/dev/service-runner.js +61 -0
- package/dist/lib/dev/service-runner.js.map +1 -0
- package/dist/lib/secrets/index.d.ts +2 -0
- package/dist/lib/secrets/index.d.ts.map +1 -0
- package/dist/lib/secrets/index.js +2 -0
- package/dist/lib/secrets/index.js.map +1 -0
- package/dist/lib/secrets/parser.d.ts +37 -0
- package/dist/lib/secrets/parser.d.ts.map +1 -0
- package/dist/lib/secrets/parser.js +116 -0
- package/dist/lib/secrets/parser.js.map +1 -0
- package/package.json +54 -0
package/dist/cli.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.tsx"],"names":[],"mappings":";AACA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACjD,OAAO,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACjD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAC/C,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACjD,OAAO,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACjD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAEvD,MAAM,OAAO,GAAG,IAAI,OAAO,EAAE,CAAC;AAE9B,OAAO;KACJ,IAAI,CAAC,UAAU,CAAC;KAChB,WAAW,CAAC,0CAA0C,CAAC;KACvD,OAAO,CAAC,OAAO,CAAC;KAChB,uBAAuB,EAAE,CAAC;AAE7B,OAAO;KACJ,OAAO,CAAC,MAAM,CAAC;KACf,WAAW,CAAC,gDAAgD,CAAC;KAC7D,MAAM,CAAC,WAAW,CAAC,CAAC;AAEvB,OAAO;KACJ,OAAO,CAAC,cAAc,CAAC;KACvB,WAAW,CAAC,mCAAmC,CAAC;KAChD,MAAM,CAAC,WAAW,CAAC,CAAC;AAEvB,OAAO;KACJ,OAAO,CAAC,OAAO,CAAC;KAChB,WAAW,CAAC,qCAAqC,CAAC;KAClD,MAAM,CAAC,YAAY,CAAC,CAAC;AAExB,OAAO;KACJ,OAAO,CAAC,KAAK,CAAC;KACd,WAAW,CAAC,qCAAqC,CAAC;KAClD,MAAM,CAAC,UAAU,CAAC,CAAC;AAEtB,OAAO;KACJ,OAAO,CAAC,QAAQ,CAAC;KACjB,WAAW,CAAC,mCAAmC,CAAC;KAChD,MAAM,CAAC,aAAa,CAAC,CAAC;AAEzB,OAAO;KACJ,OAAO,CAAC,0BAA0B,CAAC;KACnC,WAAW,CAAC,gDAAgD,CAAC;KAC7D,kBAAkB,EAAE;KACpB,MAAM,CAAC,KAAK,EAAE,OAAe,EAAE,IAAc,EAAE,EAAE;IAChD,sDAAsD;IACtD,MAAM,YAAY,GAAG,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IAC7D,MAAM,WAAW,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;AAC3C,CAAC,CAAC,CAAC;AAEL,OAAO;KACJ,OAAO,CAAC,iBAAiB,CAAC;KAC1B,WAAW,CAAC,wCAAwC,CAAC;KACrD,MAAM,CAAC,WAAW,CAAC,CAAC;AAEvB,OAAO;KACJ,OAAO,CAAC,OAAO,CAAC;KAChB,WAAW,CAAC,8CAA8C,CAAC;KAC3D,MAAM,CAAC,YAAY,CAAC,CAAC;AAExB,OAAO;KACJ,OAAO,CAAC,yBAAyB,CAAC;KAClC,WAAW,CAAC,gBAAgB,CAAC;KAC7B,MAAM,CAAC,cAAc,CAAC,CAAC;AAE1B,OAAO,CAAC,KAAK,EAAE,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"check.d.ts","sourceRoot":"","sources":["../../src/commands/check.tsx"],"names":[],"mappings":"AAkIA,wBAAgB,YAAY,SAE3B"}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import React, { useState, useEffect } from "react";
|
|
2
|
+
import { render, Text, Box } from "ink";
|
|
3
|
+
import Spinner from "ink-spinner";
|
|
4
|
+
import * as fs from "fs";
|
|
5
|
+
import * as path from "path";
|
|
6
|
+
import { parseConfig } from "@specific/config";
|
|
7
|
+
function CheckUI() {
|
|
8
|
+
const [state, setState] = useState({ status: "loading" });
|
|
9
|
+
useEffect(() => {
|
|
10
|
+
async function load() {
|
|
11
|
+
const configPath = path.join(process.cwd(), "specific.hcl");
|
|
12
|
+
if (!fs.existsSync(configPath)) {
|
|
13
|
+
setState({
|
|
14
|
+
status: "error",
|
|
15
|
+
error: "No specific.hcl found in current directory",
|
|
16
|
+
});
|
|
17
|
+
return;
|
|
18
|
+
}
|
|
19
|
+
try {
|
|
20
|
+
const hcl = fs.readFileSync(configPath, "utf-8");
|
|
21
|
+
const config = await parseConfig(hcl);
|
|
22
|
+
setState({ status: "success", config });
|
|
23
|
+
}
|
|
24
|
+
catch (err) {
|
|
25
|
+
setState({
|
|
26
|
+
status: "error",
|
|
27
|
+
error: err instanceof Error ? err.message : String(err),
|
|
28
|
+
});
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
load();
|
|
32
|
+
}, []);
|
|
33
|
+
if (state.status === "loading") {
|
|
34
|
+
return (React.createElement(Box, null,
|
|
35
|
+
React.createElement(Text, { color: "blue" },
|
|
36
|
+
React.createElement(Spinner, { type: "dots" })),
|
|
37
|
+
React.createElement(Text, null, " Checking configuration...")));
|
|
38
|
+
}
|
|
39
|
+
if (state.status === "error") {
|
|
40
|
+
return (React.createElement(Box, { flexDirection: "column" },
|
|
41
|
+
React.createElement(Text, { color: "red" },
|
|
42
|
+
"Error: ",
|
|
43
|
+
state.error)));
|
|
44
|
+
}
|
|
45
|
+
const { config } = state;
|
|
46
|
+
if (!config) {
|
|
47
|
+
return null;
|
|
48
|
+
}
|
|
49
|
+
return (React.createElement(Box, { flexDirection: "column", gap: 1 },
|
|
50
|
+
React.createElement(Text, { color: "green" }, "Configuration is valid"),
|
|
51
|
+
config.builds.length > 0 && (React.createElement(Box, { flexDirection: "column" },
|
|
52
|
+
React.createElement(Text, { bold: true },
|
|
53
|
+
"Builds (",
|
|
54
|
+
config.builds.length,
|
|
55
|
+
"):"),
|
|
56
|
+
config.builds.map((b) => (React.createElement(Text, { key: b.name },
|
|
57
|
+
" ",
|
|
58
|
+
"- ",
|
|
59
|
+
b.name))))),
|
|
60
|
+
config.services.length > 0 && (React.createElement(Box, { flexDirection: "column" },
|
|
61
|
+
React.createElement(Text, { bold: true },
|
|
62
|
+
"Services (",
|
|
63
|
+
config.services.length,
|
|
64
|
+
"):"),
|
|
65
|
+
config.services.map((s) => (React.createElement(Text, { key: s.name },
|
|
66
|
+
" ",
|
|
67
|
+
"- ",
|
|
68
|
+
s.name))))),
|
|
69
|
+
config.databases.length > 0 && (React.createElement(Box, { flexDirection: "column" },
|
|
70
|
+
React.createElement(Text, { bold: true },
|
|
71
|
+
"Databases (",
|
|
72
|
+
config.databases.length,
|
|
73
|
+
"):"),
|
|
74
|
+
config.databases.map((d) => (React.createElement(Text, { key: d.name },
|
|
75
|
+
" ",
|
|
76
|
+
"- ",
|
|
77
|
+
d.name,
|
|
78
|
+
" (",
|
|
79
|
+
d.engine,
|
|
80
|
+
")"))))),
|
|
81
|
+
config.configs.length > 0 && (React.createElement(Box, { flexDirection: "column" },
|
|
82
|
+
React.createElement(Text, { bold: true },
|
|
83
|
+
"Config (",
|
|
84
|
+
config.configs.length,
|
|
85
|
+
"):"),
|
|
86
|
+
config.configs.map((c) => (React.createElement(Text, { key: c.name },
|
|
87
|
+
" ",
|
|
88
|
+
"- ",
|
|
89
|
+
c.name,
|
|
90
|
+
c.default ? ` (default: ${c.default})` : ""))))),
|
|
91
|
+
config.environments.length > 0 && (React.createElement(Box, { flexDirection: "column" },
|
|
92
|
+
React.createElement(Text, { bold: true },
|
|
93
|
+
"Environments (",
|
|
94
|
+
config.environments.length,
|
|
95
|
+
"):"),
|
|
96
|
+
config.environments.map((e) => (React.createElement(Text, { key: e.name },
|
|
97
|
+
" ",
|
|
98
|
+
"- ",
|
|
99
|
+
e.name)))))));
|
|
100
|
+
}
|
|
101
|
+
export function checkCommand() {
|
|
102
|
+
render(React.createElement(CheckUI, null));
|
|
103
|
+
}
|
|
104
|
+
//# sourceMappingURL=check.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"check.js","sourceRoot":"","sources":["../../src/commands/check.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"clean.d.ts","sourceRoot":"","sources":["../../src/commands/clean.tsx"],"names":[],"mappings":"AA0FA,wBAAgB,YAAY,SAE3B"}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import React, { useState, useEffect } from "react";
|
|
2
|
+
import { render, Text, Box } from "ink";
|
|
3
|
+
import Spinner from "ink-spinner";
|
|
4
|
+
import * as fs from "fs";
|
|
5
|
+
import * as path from "path";
|
|
6
|
+
import { InstanceStateManager } from "../lib/dev/instance-state.js";
|
|
7
|
+
function CleanUI() {
|
|
8
|
+
const [state, setState] = useState({ status: "checking" });
|
|
9
|
+
useEffect(() => {
|
|
10
|
+
async function clean() {
|
|
11
|
+
const projectRoot = process.cwd();
|
|
12
|
+
const specificDir = path.join(projectRoot, ".specific");
|
|
13
|
+
// Check if .specific directory exists
|
|
14
|
+
if (!fs.existsSync(specificDir)) {
|
|
15
|
+
setState({ status: "nothing" });
|
|
16
|
+
return;
|
|
17
|
+
}
|
|
18
|
+
// Check if specific dev is running
|
|
19
|
+
const stateManager = new InstanceStateManager(projectRoot);
|
|
20
|
+
const existingInstances = await stateManager.getExistingInstances();
|
|
21
|
+
if (existingInstances) {
|
|
22
|
+
setState({
|
|
23
|
+
status: "error",
|
|
24
|
+
error: `Cannot clean while 'specific dev' is running (PID ${existingInstances.owner.pid}). Please stop it first.`,
|
|
25
|
+
});
|
|
26
|
+
return;
|
|
27
|
+
}
|
|
28
|
+
// Clean stale state if any
|
|
29
|
+
await stateManager.cleanStaleState();
|
|
30
|
+
// Remove the .specific directory
|
|
31
|
+
setState({ status: "cleaning" });
|
|
32
|
+
try {
|
|
33
|
+
fs.rmSync(specificDir, { recursive: true, force: true });
|
|
34
|
+
setState({ status: "success" });
|
|
35
|
+
}
|
|
36
|
+
catch (err) {
|
|
37
|
+
setState({
|
|
38
|
+
status: "error",
|
|
39
|
+
error: err instanceof Error ? err.message : String(err),
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
clean();
|
|
44
|
+
}, []);
|
|
45
|
+
if (state.status === "checking") {
|
|
46
|
+
return (React.createElement(Box, null,
|
|
47
|
+
React.createElement(Text, { color: "blue" },
|
|
48
|
+
React.createElement(Spinner, { type: "dots" })),
|
|
49
|
+
React.createElement(Text, null, " Checking for running instances...")));
|
|
50
|
+
}
|
|
51
|
+
if (state.status === "cleaning") {
|
|
52
|
+
return (React.createElement(Box, null,
|
|
53
|
+
React.createElement(Text, { color: "blue" },
|
|
54
|
+
React.createElement(Spinner, { type: "dots" })),
|
|
55
|
+
React.createElement(Text, null, " Removing .specific directory...")));
|
|
56
|
+
}
|
|
57
|
+
if (state.status === "error") {
|
|
58
|
+
return React.createElement(Text, { color: "red" },
|
|
59
|
+
"Error: ",
|
|
60
|
+
state.error);
|
|
61
|
+
}
|
|
62
|
+
if (state.status === "nothing") {
|
|
63
|
+
return React.createElement(Text, { color: "yellow" }, "Nothing to clean (.specific directory does not exist)");
|
|
64
|
+
}
|
|
65
|
+
return React.createElement(Text, { color: "green" }, "Cleaned .specific directory");
|
|
66
|
+
}
|
|
67
|
+
export function cleanCommand() {
|
|
68
|
+
render(React.createElement(CleanUI, null));
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=clean.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"clean.js","sourceRoot":"","sources":["../../src/commands/clean.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"deploy.d.ts","sourceRoot":"","sources":["../../src/commands/deploy.tsx"],"names":[],"mappings":"AAYA,wBAAgB,aAAa,SAE5B"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import React from "react";
|
|
2
|
+
import { render, Text, Box } from "ink";
|
|
3
|
+
function DeployUI() {
|
|
4
|
+
return (React.createElement(Box, { flexDirection: "column" },
|
|
5
|
+
React.createElement(Text, { color: "cyan" }, "specific deploy"),
|
|
6
|
+
React.createElement(Text, { dimColor: true }, "Deploy to Specific infrastructure")));
|
|
7
|
+
}
|
|
8
|
+
export function deployCommand() {
|
|
9
|
+
render(React.createElement(DeployUI, null));
|
|
10
|
+
}
|
|
11
|
+
//# sourceMappingURL=deploy.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"deploy.js","sourceRoot":"","sources":["../../src/commands/deploy.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,GAAG,EAAE,MAAM,KAAK,CAAC;AAExC,SAAS,QAAQ;IACf,OAAO,CACL,oBAAC,GAAG,IAAC,aAAa,EAAC,QAAQ;QACzB,oBAAC,IAAI,IAAC,KAAK,EAAC,MAAM,sBAAuB;QACzC,oBAAC,IAAI,IAAC,QAAQ,8CAAyC,CACnD,CACP,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,aAAa;IAC3B,MAAM,CAAC,oBAAC,QAAQ,OAAG,CAAC,CAAC;AACvB,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"dev.d.ts","sourceRoot":"","sources":["../../src/commands/dev.tsx"],"names":[],"mappings":"AAmgBA,wBAAgB,UAAU,SAEzB"}
|
|
@@ -0,0 +1,398 @@
|
|
|
1
|
+
import React, { useState, useEffect, useRef } from "react";
|
|
2
|
+
import { render, Text, Box, useApp, Static } from "ink";
|
|
3
|
+
import Spinner from "ink-spinner";
|
|
4
|
+
import * as fs from "fs";
|
|
5
|
+
import * as path from "path";
|
|
6
|
+
import { parseConfig } from "@specific/config";
|
|
7
|
+
import { PortAllocator, startDatabase, startService, startHttpProxy, InstanceStateManager, } from "../lib/dev/index.js";
|
|
8
|
+
import { prepareSecrets } from "../lib/secrets/index.js";
|
|
9
|
+
// Colors for service prefixes
|
|
10
|
+
const COLORS = ["cyan", "yellow", "green", "magenta", "blue"];
|
|
11
|
+
function DevUI() {
|
|
12
|
+
const { exit } = useApp();
|
|
13
|
+
const [state, setState] = useState({
|
|
14
|
+
status: "loading",
|
|
15
|
+
databases: new Map(),
|
|
16
|
+
services: [],
|
|
17
|
+
output: [],
|
|
18
|
+
colorMap: new Map(),
|
|
19
|
+
});
|
|
20
|
+
// Track if shutdown is in progress
|
|
21
|
+
const shuttingDown = useRef(false);
|
|
22
|
+
const stateManagerRef = useRef(null);
|
|
23
|
+
const shutdown = async () => {
|
|
24
|
+
if (shuttingDown.current)
|
|
25
|
+
return;
|
|
26
|
+
shuttingDown.current = true;
|
|
27
|
+
setState((s) => ({ ...s, status: "stopping" }));
|
|
28
|
+
// Stop proxy first (stop accepting new requests)
|
|
29
|
+
if (state.proxy) {
|
|
30
|
+
await state.proxy.stop();
|
|
31
|
+
}
|
|
32
|
+
// Stop services
|
|
33
|
+
for (const service of state.services) {
|
|
34
|
+
await service.stop();
|
|
35
|
+
}
|
|
36
|
+
// Then stop databases
|
|
37
|
+
for (const db of state.databases.values()) {
|
|
38
|
+
await db.stop();
|
|
39
|
+
}
|
|
40
|
+
// Release ownership
|
|
41
|
+
if (stateManagerRef.current) {
|
|
42
|
+
await stateManagerRef.current.releaseOwnership();
|
|
43
|
+
}
|
|
44
|
+
exit();
|
|
45
|
+
};
|
|
46
|
+
// Handle SIGINT/SIGTERM for graceful shutdown
|
|
47
|
+
useEffect(() => {
|
|
48
|
+
const handleSignal = () => {
|
|
49
|
+
shutdown();
|
|
50
|
+
};
|
|
51
|
+
process.on("SIGINT", handleSignal);
|
|
52
|
+
process.on("SIGTERM", handleSignal);
|
|
53
|
+
return () => {
|
|
54
|
+
process.off("SIGINT", handleSignal);
|
|
55
|
+
process.off("SIGTERM", handleSignal);
|
|
56
|
+
};
|
|
57
|
+
}, [state.services, state.databases]);
|
|
58
|
+
const addLog = (line, colorMap) => {
|
|
59
|
+
const color = colorMap.get(`svc:${line.service}`) || "cyan";
|
|
60
|
+
setState((s) => ({
|
|
61
|
+
...s,
|
|
62
|
+
output: [
|
|
63
|
+
...s.output,
|
|
64
|
+
{ type: "log", service: line.service, text: line.text, color },
|
|
65
|
+
],
|
|
66
|
+
}));
|
|
67
|
+
};
|
|
68
|
+
useEffect(() => {
|
|
69
|
+
let cancelled = false;
|
|
70
|
+
const startedDatabases = [];
|
|
71
|
+
const startedServices = [];
|
|
72
|
+
let startedProxy = null;
|
|
73
|
+
const stateManager = new InstanceStateManager(process.cwd());
|
|
74
|
+
stateManagerRef.current = stateManager;
|
|
75
|
+
async function start() {
|
|
76
|
+
// Phase 0: Check for existing instances
|
|
77
|
+
await stateManager.cleanStaleState();
|
|
78
|
+
const existing = await stateManager.getExistingInstances();
|
|
79
|
+
if (existing) {
|
|
80
|
+
setState((s) => ({
|
|
81
|
+
...s,
|
|
82
|
+
status: "error",
|
|
83
|
+
error: `Development environment already running (owned by PID ${existing.owner.pid})`,
|
|
84
|
+
}));
|
|
85
|
+
return;
|
|
86
|
+
}
|
|
87
|
+
try {
|
|
88
|
+
await stateManager.claimOwnership("dev");
|
|
89
|
+
}
|
|
90
|
+
catch (err) {
|
|
91
|
+
setState((s) => ({
|
|
92
|
+
...s,
|
|
93
|
+
status: "error",
|
|
94
|
+
error: err instanceof Error ? err.message : String(err),
|
|
95
|
+
}));
|
|
96
|
+
return;
|
|
97
|
+
}
|
|
98
|
+
// Phase 1: Load config
|
|
99
|
+
const configPath = path.join(process.cwd(), "specific.hcl");
|
|
100
|
+
if (!fs.existsSync(configPath)) {
|
|
101
|
+
setState((s) => ({
|
|
102
|
+
...s,
|
|
103
|
+
status: "error",
|
|
104
|
+
error: "No specific.hcl found in current directory",
|
|
105
|
+
}));
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
let config;
|
|
109
|
+
try {
|
|
110
|
+
const hcl = fs.readFileSync(configPath, "utf-8");
|
|
111
|
+
config = await parseConfig(hcl);
|
|
112
|
+
}
|
|
113
|
+
catch (err) {
|
|
114
|
+
setState((s) => ({
|
|
115
|
+
...s,
|
|
116
|
+
status: "error",
|
|
117
|
+
error: err instanceof Error ? err.message : String(err),
|
|
118
|
+
}));
|
|
119
|
+
return;
|
|
120
|
+
}
|
|
121
|
+
if (cancelled)
|
|
122
|
+
return;
|
|
123
|
+
// Assign colors to databases and services
|
|
124
|
+
const colorMap = new Map();
|
|
125
|
+
let colorIndex = 0;
|
|
126
|
+
for (const db of config.databases) {
|
|
127
|
+
colorMap.set(`db:${db.name}`, COLORS[colorIndex % COLORS.length]);
|
|
128
|
+
colorIndex++;
|
|
129
|
+
}
|
|
130
|
+
for (const service of config.services) {
|
|
131
|
+
colorMap.set(`svc:${service.name}`, COLORS[colorIndex % COLORS.length]);
|
|
132
|
+
colorIndex++;
|
|
133
|
+
}
|
|
134
|
+
setState((s) => ({ ...s, status: "starting", config, colorMap }));
|
|
135
|
+
// Phase 2: Start databases
|
|
136
|
+
const portAllocator = new PortAllocator();
|
|
137
|
+
const databases = new Map();
|
|
138
|
+
for (const db of config.databases) {
|
|
139
|
+
if (cancelled)
|
|
140
|
+
break;
|
|
141
|
+
try {
|
|
142
|
+
const port = portAllocator.allocate();
|
|
143
|
+
const instance = await startDatabase(db, port);
|
|
144
|
+
databases.set(db.name, instance);
|
|
145
|
+
startedDatabases.push(instance);
|
|
146
|
+
// Update state immediately so UI shows this database as ready
|
|
147
|
+
setState((s) => ({ ...s, databases: new Map(databases) }));
|
|
148
|
+
const dbState = {
|
|
149
|
+
engine: instance.engine,
|
|
150
|
+
port: instance.port,
|
|
151
|
+
host: instance.host,
|
|
152
|
+
user: instance.user,
|
|
153
|
+
password: instance.password,
|
|
154
|
+
dbName: instance.dbName,
|
|
155
|
+
url: instance.url,
|
|
156
|
+
};
|
|
157
|
+
if (instance.endpoint)
|
|
158
|
+
dbState.endpoint = instance.endpoint;
|
|
159
|
+
if (instance.accessKey)
|
|
160
|
+
dbState.accessKey = instance.accessKey;
|
|
161
|
+
if (instance.secretKey)
|
|
162
|
+
dbState.secretKey = instance.secretKey;
|
|
163
|
+
await stateManager.registerDatabase(db.name, dbState);
|
|
164
|
+
}
|
|
165
|
+
catch (err) {
|
|
166
|
+
setState((s) => ({
|
|
167
|
+
...s,
|
|
168
|
+
status: "error",
|
|
169
|
+
error: `Failed to start database "${db.name}": ${err instanceof Error ? err.message : String(err)}`,
|
|
170
|
+
}));
|
|
171
|
+
return;
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
if (cancelled)
|
|
175
|
+
return;
|
|
176
|
+
// Phase 3: Prepare secrets (loads user secrets + handles generated ones)
|
|
177
|
+
const secrets = await prepareSecrets(config.secrets);
|
|
178
|
+
// Phase 4: Start services
|
|
179
|
+
const services = [];
|
|
180
|
+
for (const service of config.services) {
|
|
181
|
+
if (cancelled)
|
|
182
|
+
break;
|
|
183
|
+
// Skip static services for now
|
|
184
|
+
if (service.serve) {
|
|
185
|
+
continue;
|
|
186
|
+
}
|
|
187
|
+
// Skip services without a command
|
|
188
|
+
if (!service.command && !service.dev?.command) {
|
|
189
|
+
continue;
|
|
190
|
+
}
|
|
191
|
+
try {
|
|
192
|
+
// Allocate port for services with expose
|
|
193
|
+
const port = service.expose ? portAllocator.allocate() : undefined;
|
|
194
|
+
const running = startService(service, databases, secrets, port, (line) => addLog(line, colorMap));
|
|
195
|
+
services.push(running);
|
|
196
|
+
startedServices.push(running);
|
|
197
|
+
// Update state immediately so UI shows this service as ready
|
|
198
|
+
setState((s) => ({ ...s, services: [...services] }));
|
|
199
|
+
if (running.process.pid) {
|
|
200
|
+
const serviceState = {
|
|
201
|
+
pid: running.process.pid,
|
|
202
|
+
};
|
|
203
|
+
if (running.port !== undefined) {
|
|
204
|
+
serviceState.port = running.port;
|
|
205
|
+
}
|
|
206
|
+
await stateManager.registerService(service.name, serviceState);
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
catch (err) {
|
|
210
|
+
setState((s) => ({
|
|
211
|
+
...s,
|
|
212
|
+
status: "error",
|
|
213
|
+
error: `Failed to start service "${service.name}": ${err instanceof Error ? err.message : String(err)}`,
|
|
214
|
+
}));
|
|
215
|
+
return;
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
if (cancelled)
|
|
219
|
+
return;
|
|
220
|
+
// Phase 5: Start HTTP proxy for exposed services
|
|
221
|
+
const exposedServices = services
|
|
222
|
+
.filter((s) => s.port !== undefined)
|
|
223
|
+
.map((s) => ({ name: s.name, port: s.port }));
|
|
224
|
+
if (exposedServices.length > 0) {
|
|
225
|
+
try {
|
|
226
|
+
const proxy = await startHttpProxy(exposedServices);
|
|
227
|
+
startedProxy = proxy;
|
|
228
|
+
setState((s) => ({ ...s, proxy }));
|
|
229
|
+
}
|
|
230
|
+
catch (err) {
|
|
231
|
+
setState((s) => ({
|
|
232
|
+
...s,
|
|
233
|
+
status: "error",
|
|
234
|
+
error: `Failed to start HTTP proxy: ${err instanceof Error ? err.message : String(err)}`,
|
|
235
|
+
}));
|
|
236
|
+
return;
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
if (cancelled)
|
|
240
|
+
return;
|
|
241
|
+
setState((s) => ({ ...s, status: "running", services }));
|
|
242
|
+
}
|
|
243
|
+
start();
|
|
244
|
+
return () => {
|
|
245
|
+
cancelled = true;
|
|
246
|
+
// Cleanup on unmount
|
|
247
|
+
if (startedProxy) {
|
|
248
|
+
startedProxy.stop().catch(() => { });
|
|
249
|
+
}
|
|
250
|
+
for (const service of startedServices) {
|
|
251
|
+
service.stop().catch(() => { });
|
|
252
|
+
}
|
|
253
|
+
for (const db of startedDatabases) {
|
|
254
|
+
db.stop().catch(() => { });
|
|
255
|
+
}
|
|
256
|
+
stateManager.releaseOwnership().catch(() => { });
|
|
257
|
+
};
|
|
258
|
+
}, []);
|
|
259
|
+
if (state.status === "loading") {
|
|
260
|
+
return (React.createElement(Box, null,
|
|
261
|
+
React.createElement(Text, { color: "blue" },
|
|
262
|
+
React.createElement(Spinner, { type: "dots" }))));
|
|
263
|
+
}
|
|
264
|
+
if (state.status === "error") {
|
|
265
|
+
return (React.createElement(Box, { flexDirection: "column" },
|
|
266
|
+
React.createElement(Text, { color: "red" },
|
|
267
|
+
"Error: ",
|
|
268
|
+
state.error)));
|
|
269
|
+
}
|
|
270
|
+
if (state.status === "stopping") {
|
|
271
|
+
return (React.createElement(Box, null,
|
|
272
|
+
React.createElement(Text, { color: "yellow" },
|
|
273
|
+
React.createElement(Spinner, { type: "dots" })),
|
|
274
|
+
React.createElement(Text, null, " Shutting down...")));
|
|
275
|
+
}
|
|
276
|
+
const { config, databases, services, proxy, output } = state;
|
|
277
|
+
// Show spinner while starting if no config yet
|
|
278
|
+
if (!config) {
|
|
279
|
+
return (React.createElement(Box, null,
|
|
280
|
+
React.createElement(Text, { color: "blue" },
|
|
281
|
+
React.createElement(Spinner, { type: "dots" }))));
|
|
282
|
+
}
|
|
283
|
+
// During starting phase, show dynamic view (no logs yet)
|
|
284
|
+
if (state.status === "starting") {
|
|
285
|
+
return (React.createElement(Box, { flexDirection: "column" },
|
|
286
|
+
React.createElement(Text, null,
|
|
287
|
+
React.createElement(Text, { bold: true, color: "cyan" }, "Specific dev server"),
|
|
288
|
+
React.createElement(Text, { dimColor: true }, " (Ctrl+C to stop)")),
|
|
289
|
+
React.createElement(Text, null, " "),
|
|
290
|
+
config.databases.length > 0 && (React.createElement(React.Fragment, null,
|
|
291
|
+
React.createElement(Text, { bold: true }, "Databases:"),
|
|
292
|
+
config.databases.map((db) => {
|
|
293
|
+
const instance = databases.get(db.name);
|
|
294
|
+
const isReady = !!instance;
|
|
295
|
+
return (React.createElement(Text, { key: db.name },
|
|
296
|
+
React.createElement(Text, { color: isReady ? "green" : "gray" }, " \u25CF "),
|
|
297
|
+
React.createElement(Text, null,
|
|
298
|
+
db.name,
|
|
299
|
+
" (",
|
|
300
|
+
db.engine,
|
|
301
|
+
")"),
|
|
302
|
+
instance && React.createElement(Text, null,
|
|
303
|
+
" \u2192 localhost:",
|
|
304
|
+
instance.port)));
|
|
305
|
+
}),
|
|
306
|
+
React.createElement(Text, null, " "))),
|
|
307
|
+
config.services.length > 0 && (React.createElement(React.Fragment, null,
|
|
308
|
+
React.createElement(Text, { bold: true }, "Services:"),
|
|
309
|
+
config.services.map((svc) => {
|
|
310
|
+
const running = services.find((s) => s.name === svc.name);
|
|
311
|
+
const isReady = !!running;
|
|
312
|
+
const portInfo = running?.port
|
|
313
|
+
? ` → localhost:${running.port}`
|
|
314
|
+
: "";
|
|
315
|
+
return (React.createElement(Text, { key: svc.name },
|
|
316
|
+
React.createElement(Text, { color: isReady ? "green" : "gray" }, " \u25CF "),
|
|
317
|
+
React.createElement(Text, null,
|
|
318
|
+
svc.name,
|
|
319
|
+
portInfo)));
|
|
320
|
+
}),
|
|
321
|
+
React.createElement(Text, null, " "))),
|
|
322
|
+
React.createElement(Box, null,
|
|
323
|
+
React.createElement(Text, { color: "blue" },
|
|
324
|
+
React.createElement(Spinner, { type: "dots" })),
|
|
325
|
+
React.createElement(Text, null, " Starting..."))));
|
|
326
|
+
}
|
|
327
|
+
const staticItems = [
|
|
328
|
+
{
|
|
329
|
+
key: "title",
|
|
330
|
+
content: (React.createElement(Text, null,
|
|
331
|
+
React.createElement(Text, { bold: true, color: "cyan" }, "Specific dev server"),
|
|
332
|
+
React.createElement(Text, { dimColor: true }, " (Ctrl+C to stop)"))),
|
|
333
|
+
},
|
|
334
|
+
{ key: "space1", content: React.createElement(Text, null, " ") },
|
|
335
|
+
...(config.databases.length > 0
|
|
336
|
+
? [
|
|
337
|
+
{ key: "db-header", content: React.createElement(Text, { bold: true }, "Databases:") },
|
|
338
|
+
...config.databases.map((db) => {
|
|
339
|
+
const instance = databases.get(db.name);
|
|
340
|
+
return {
|
|
341
|
+
key: `db-${db.name}`,
|
|
342
|
+
content: (React.createElement(Text, null,
|
|
343
|
+
React.createElement(Text, { color: "green" }, " \u25CF "),
|
|
344
|
+
React.createElement(Text, null,
|
|
345
|
+
db.name,
|
|
346
|
+
" (",
|
|
347
|
+
db.engine,
|
|
348
|
+
")"),
|
|
349
|
+
instance && React.createElement(Text, null,
|
|
350
|
+
" \u2192 localhost:",
|
|
351
|
+
instance.port))),
|
|
352
|
+
};
|
|
353
|
+
}),
|
|
354
|
+
{ key: "space2", content: React.createElement(Text, null, " ") },
|
|
355
|
+
]
|
|
356
|
+
: []),
|
|
357
|
+
...(services.length > 0
|
|
358
|
+
? [
|
|
359
|
+
{ key: "svc-header", content: React.createElement(Text, { bold: true }, "Services:") },
|
|
360
|
+
...services.map((svc) => ({
|
|
361
|
+
key: `svc-${svc.name}`,
|
|
362
|
+
content: (React.createElement(Text, null,
|
|
363
|
+
React.createElement(Text, { color: "green" }, " \u25CF "),
|
|
364
|
+
React.createElement(Text, null,
|
|
365
|
+
svc.name,
|
|
366
|
+
svc.port ? (proxy ? (React.createElement(React.Fragment, null,
|
|
367
|
+
" → ",
|
|
368
|
+
React.createElement(Text, { bold: true },
|
|
369
|
+
"https://",
|
|
370
|
+
svc.name,
|
|
371
|
+
".local.spcf.app"),
|
|
372
|
+
React.createElement(Text, { dimColor: true },
|
|
373
|
+
" (localhost:",
|
|
374
|
+
svc.port,
|
|
375
|
+
")"))) : (` → localhost:${svc.port}`)) : ("")))),
|
|
376
|
+
})),
|
|
377
|
+
{ key: "space3", content: React.createElement(Text, null, " ") },
|
|
378
|
+
]
|
|
379
|
+
: []),
|
|
380
|
+
{ key: "separator", content: React.createElement(Text, { dimColor: true }, "─".repeat(50)) },
|
|
381
|
+
...output.map((line, i) => ({
|
|
382
|
+
key: `log-${i}`,
|
|
383
|
+
content: (React.createElement(Text, null,
|
|
384
|
+
React.createElement(Text, { color: line.color },
|
|
385
|
+
"[",
|
|
386
|
+
line.service,
|
|
387
|
+
"]"),
|
|
388
|
+
" ",
|
|
389
|
+
line.text)),
|
|
390
|
+
})),
|
|
391
|
+
];
|
|
392
|
+
return (React.createElement(Box, { flexDirection: "column" },
|
|
393
|
+
React.createElement(Static, { items: staticItems }, (item) => React.createElement(Box, { key: item.key }, item.content))));
|
|
394
|
+
}
|
|
395
|
+
export function devCommand() {
|
|
396
|
+
render(React.createElement(DevUI, null));
|
|
397
|
+
}
|
|
398
|
+
//# sourceMappingURL=dev.js.map
|