@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.
Files changed (204) hide show
  1. package/dist/admin/404/index.html +1 -0
  2. package/dist/admin/404.html +1 -0
  3. package/dist/admin/__next.__PAGE__.txt +9 -0
  4. package/dist/admin/__next._full.txt +20 -0
  5. package/dist/admin/__next._head.txt +6 -0
  6. package/dist/admin/__next._index.txt +5 -0
  7. package/dist/admin/__next._tree.txt +4 -0
  8. package/dist/admin/_next/static/0zkv3YeV6IWOWVWE1S1A1/_buildManifest.js +11 -0
  9. package/dist/admin/_next/static/0zkv3YeV6IWOWVWE1S1A1/_clientMiddlewareManifest.json +1 -0
  10. package/dist/admin/_next/static/0zkv3YeV6IWOWVWE1S1A1/_ssgManifest.js +1 -0
  11. package/dist/admin/_next/static/1o2O2QPOKhuZyxcNaCjq8/_buildManifest.js +11 -0
  12. package/dist/admin/_next/static/1o2O2QPOKhuZyxcNaCjq8/_clientMiddlewareManifest.json +1 -0
  13. package/dist/admin/_next/static/1o2O2QPOKhuZyxcNaCjq8/_ssgManifest.js +1 -0
  14. package/dist/admin/_next/static/2ZtqtFX2EkuM82mTM7NWT/_buildManifest.js +11 -0
  15. package/dist/admin/_next/static/2ZtqtFX2EkuM82mTM7NWT/_clientMiddlewareManifest.json +1 -0
  16. package/dist/admin/_next/static/2ZtqtFX2EkuM82mTM7NWT/_ssgManifest.js +1 -0
  17. package/dist/admin/_next/static/2vLF5bo4ZtUfhs8bdQBoF/_buildManifest.js +11 -0
  18. package/dist/admin/_next/static/2vLF5bo4ZtUfhs8bdQBoF/_clientMiddlewareManifest.json +1 -0
  19. package/dist/admin/_next/static/2vLF5bo4ZtUfhs8bdQBoF/_ssgManifest.js +1 -0
  20. package/dist/admin/_next/static/8_8s51m7RbLVvbIechG-b/_buildManifest.js +11 -0
  21. package/dist/admin/_next/static/8_8s51m7RbLVvbIechG-b/_clientMiddlewareManifest.json +1 -0
  22. package/dist/admin/_next/static/8_8s51m7RbLVvbIechG-b/_ssgManifest.js +1 -0
  23. package/dist/admin/_next/static/C4Y44KtLnP684m57gN3Ga/_buildManifest.js +11 -0
  24. package/dist/admin/_next/static/C4Y44KtLnP684m57gN3Ga/_clientMiddlewareManifest.json +1 -0
  25. package/dist/admin/_next/static/C4Y44KtLnP684m57gN3Ga/_ssgManifest.js +1 -0
  26. package/dist/admin/_next/static/DBeh36kAJnZBeBMJSPBW8/_buildManifest.js +11 -0
  27. package/dist/admin/_next/static/DBeh36kAJnZBeBMJSPBW8/_clientMiddlewareManifest.json +1 -0
  28. package/dist/admin/_next/static/DBeh36kAJnZBeBMJSPBW8/_ssgManifest.js +1 -0
  29. package/dist/admin/_next/static/REKEUu6DP2t99jKSAqNOu/_buildManifest.js +11 -0
  30. package/dist/admin/_next/static/REKEUu6DP2t99jKSAqNOu/_clientMiddlewareManifest.json +1 -0
  31. package/dist/admin/_next/static/REKEUu6DP2t99jKSAqNOu/_ssgManifest.js +1 -0
  32. package/dist/admin/_next/static/TDGT3mkKtTe0w_CxujLxI/_buildManifest.js +11 -0
  33. package/dist/admin/_next/static/TDGT3mkKtTe0w_CxujLxI/_clientMiddlewareManifest.json +1 -0
  34. package/dist/admin/_next/static/TDGT3mkKtTe0w_CxujLxI/_ssgManifest.js +1 -0
  35. package/dist/admin/_next/static/ZHMJ-g7mAlAMt_2uCXeEk/_buildManifest.js +11 -0
  36. package/dist/admin/_next/static/ZHMJ-g7mAlAMt_2uCXeEk/_clientMiddlewareManifest.json +1 -0
  37. package/dist/admin/_next/static/ZHMJ-g7mAlAMt_2uCXeEk/_ssgManifest.js +1 -0
  38. package/dist/admin/_next/static/a6JDAB-CdvWPvS4sBgJji/_buildManifest.js +11 -0
  39. package/dist/admin/_next/static/a6JDAB-CdvWPvS4sBgJji/_clientMiddlewareManifest.json +1 -0
  40. package/dist/admin/_next/static/a6JDAB-CdvWPvS4sBgJji/_ssgManifest.js +1 -0
  41. package/dist/admin/_next/static/abFUeBpymhlx-IxygVnM9/_buildManifest.js +11 -0
  42. package/dist/admin/_next/static/abFUeBpymhlx-IxygVnM9/_clientMiddlewareManifest.json +1 -0
  43. package/dist/admin/_next/static/abFUeBpymhlx-IxygVnM9/_ssgManifest.js +1 -0
  44. package/dist/admin/_next/static/chunks/0afa5e999b6c353a.js +1 -0
  45. package/dist/admin/_next/static/chunks/1584f10ea1cebcb2.js +4 -0
  46. package/dist/admin/_next/static/chunks/195bbec70cfcd241.js +2 -0
  47. package/dist/admin/_next/static/chunks/2583656ea9ac4ad6.js +5 -0
  48. package/dist/admin/_next/static/chunks/465f799faf41e6df.js +1 -0
  49. package/dist/admin/_next/static/chunks/4ab079bdcb131778.js +5 -0
  50. package/dist/admin/_next/static/chunks/4b5ae5ebb2087f0d.js +2 -0
  51. package/dist/admin/_next/static/chunks/605800ff25160d05.js +1 -0
  52. package/dist/admin/_next/static/chunks/656b870f0567ed5f.js +1 -0
  53. package/dist/admin/_next/static/chunks/71098a6cd6181738.css +3 -0
  54. package/dist/admin/_next/static/chunks/806bdb8e4a6a9b95.js +4 -0
  55. package/dist/admin/_next/static/chunks/895a6f91f0b479fb.js +2 -0
  56. package/dist/admin/_next/static/chunks/9032f4a1aac1ca5d.css +3 -0
  57. package/dist/admin/_next/static/chunks/a28af2dc6f5fbaad.js +1 -0
  58. package/dist/admin/_next/static/chunks/a6dad97d9634a72d.js +1 -0
  59. package/dist/admin/_next/static/chunks/a6dad97d9634a72d.js.map +1 -0
  60. package/dist/admin/_next/static/chunks/b4205fc2f84bda68.css +3 -0
  61. package/dist/admin/_next/static/chunks/c1a750c25bc8d092.js +1 -0
  62. package/dist/admin/_next/static/chunks/c3d30f6f144dca51.js +2 -0
  63. package/dist/admin/_next/static/chunks/cbf55ce8731457ae.js +2 -0
  64. package/dist/admin/_next/static/chunks/d2be314c3ece3fbe.js +1 -0
  65. package/dist/admin/_next/static/chunks/de6af6d8adf8b50a.js +5 -0
  66. package/dist/admin/_next/static/chunks/f0c001244d275aab.js +5 -0
  67. package/dist/admin/_next/static/chunks/fde89fd76ad6a3d0.css +3 -0
  68. package/dist/admin/_next/static/chunks/ff1a16fafef87110.js +1 -0
  69. package/dist/admin/_next/static/chunks/turbopack-a3d691c83d4b1778.js +4 -0
  70. package/dist/admin/_next/static/chunks/turbopack-e5185af8e7c716ca.js +4 -0
  71. package/dist/admin/_next/static/dcjrfD44GuB6g8bZ6BcFm/_buildManifest.js +11 -0
  72. package/dist/admin/_next/static/dcjrfD44GuB6g8bZ6BcFm/_clientMiddlewareManifest.json +1 -0
  73. package/dist/admin/_next/static/dcjrfD44GuB6g8bZ6BcFm/_ssgManifest.js +1 -0
  74. package/dist/admin/_next/static/fTEa8Scx922n-dMuqN3Vc/_buildManifest.js +11 -0
  75. package/dist/admin/_next/static/fTEa8Scx922n-dMuqN3Vc/_clientMiddlewareManifest.json +1 -0
  76. package/dist/admin/_next/static/fTEa8Scx922n-dMuqN3Vc/_ssgManifest.js +1 -0
  77. package/dist/admin/_next/static/kZNwyhHft01wPtJ_AvqQT/_buildManifest.js +11 -0
  78. package/dist/admin/_next/static/kZNwyhHft01wPtJ_AvqQT/_clientMiddlewareManifest.json +1 -0
  79. package/dist/admin/_next/static/kZNwyhHft01wPtJ_AvqQT/_ssgManifest.js +1 -0
  80. package/dist/admin/_next/static/kyVInC6N3DY6NWnjkLQ4J/_buildManifest.js +11 -0
  81. package/dist/admin/_next/static/kyVInC6N3DY6NWnjkLQ4J/_clientMiddlewareManifest.json +1 -0
  82. package/dist/admin/_next/static/kyVInC6N3DY6NWnjkLQ4J/_ssgManifest.js +1 -0
  83. package/dist/admin/_next/static/media/4fa387ec64143e14-s.c1fdd6c2.woff2 +0 -0
  84. package/dist/admin/_next/static/media/7178b3e590c64307-s.b97b3418.woff2 +0 -0
  85. package/dist/admin/_next/static/media/797e433ab948586e-s.p.dbea232f.woff2 +0 -0
  86. package/dist/admin/_next/static/media/8a480f0b521d4e75-s.8e0177b5.woff2 +0 -0
  87. package/dist/admin/_next/static/media/bbc41e54d2fcbd21-s.799d8ef8.woff2 +0 -0
  88. package/dist/admin/_next/static/media/caa3a2e1cccd8315-s.p.853070df.woff2 +0 -0
  89. package/dist/admin/_next/static/media/favicon.0b3bf435.ico +0 -0
  90. package/dist/admin/_next/static/twedXBdhyxpjS68UQoNAc/_buildManifest.js +11 -0
  91. package/dist/admin/_next/static/twedXBdhyxpjS68UQoNAc/_clientMiddlewareManifest.json +1 -0
  92. package/dist/admin/_next/static/twedXBdhyxpjS68UQoNAc/_ssgManifest.js +1 -0
  93. package/dist/admin/_not-found/__next._full.txt +14 -0
  94. package/dist/admin/_not-found/__next._head.txt +6 -0
  95. package/dist/admin/_not-found/__next._index.txt +5 -0
  96. package/dist/admin/_not-found/__next._not-found.__PAGE__.txt +5 -0
  97. package/dist/admin/_not-found/__next._not-found.txt +4 -0
  98. package/dist/admin/_not-found/__next._tree.txt +2 -0
  99. package/dist/admin/_not-found/index.html +1 -0
  100. package/dist/admin/_not-found/index.txt +14 -0
  101. package/dist/admin/databases/__next._full.txt +20 -0
  102. package/dist/admin/databases/__next._head.txt +6 -0
  103. package/dist/admin/databases/__next._index.txt +5 -0
  104. package/dist/admin/databases/__next._tree.txt +4 -0
  105. package/dist/admin/databases/__next.databases.__PAGE__.txt +9 -0
  106. package/dist/admin/databases/__next.databases.txt +4 -0
  107. package/dist/admin/databases/index.html +1 -0
  108. package/dist/admin/databases/index.txt +20 -0
  109. package/dist/admin/favicon.ico +0 -0
  110. package/dist/admin/file.svg +1 -0
  111. package/dist/admin/globe.svg +1 -0
  112. package/dist/admin/index.html +1 -0
  113. package/dist/admin/index.txt +20 -0
  114. package/dist/admin/next.svg +1 -0
  115. package/dist/admin/vercel.svg +1 -0
  116. package/dist/admin/window.svg +1 -0
  117. package/dist/cli.d.ts +3 -0
  118. package/dist/cli.d.ts.map +1 -0
  119. package/dist/cli.js +190523 -0
  120. package/dist/cli.js.map +1 -0
  121. package/dist/commands/check.d.ts +2 -0
  122. package/dist/commands/check.d.ts.map +1 -0
  123. package/dist/commands/check.js +104 -0
  124. package/dist/commands/check.js.map +1 -0
  125. package/dist/commands/clean.d.ts +2 -0
  126. package/dist/commands/clean.d.ts.map +1 -0
  127. package/dist/commands/clean.js +70 -0
  128. package/dist/commands/clean.js.map +1 -0
  129. package/dist/commands/deploy.d.ts +2 -0
  130. package/dist/commands/deploy.d.ts.map +1 -0
  131. package/dist/commands/deploy.js +11 -0
  132. package/dist/commands/deploy.js.map +1 -0
  133. package/dist/commands/dev.d.ts +2 -0
  134. package/dist/commands/dev.d.ts.map +1 -0
  135. package/dist/commands/dev.js +398 -0
  136. package/dist/commands/dev.js.map +1 -0
  137. package/dist/commands/docs.d.ts +2 -0
  138. package/dist/commands/docs.d.ts.map +1 -0
  139. package/dist/commands/docs.js +32 -0
  140. package/dist/commands/docs.js.map +1 -0
  141. package/dist/commands/exec.d.ts +2 -0
  142. package/dist/commands/exec.d.ts.map +1 -0
  143. package/dist/commands/exec.js +178 -0
  144. package/dist/commands/exec.js.map +1 -0
  145. package/dist/commands/init.d.ts +2 -0
  146. package/dist/commands/init.d.ts.map +1 -0
  147. package/dist/commands/init.js +339 -0
  148. package/dist/commands/init.js.map +1 -0
  149. package/dist/commands/psql.d.ts +2 -0
  150. package/dist/commands/psql.d.ts.map +1 -0
  151. package/dist/commands/psql.js +53 -0
  152. package/dist/commands/psql.js.map +1 -0
  153. package/dist/commands/secrets.d.ts +3 -0
  154. package/dist/commands/secrets.d.ts.map +1 -0
  155. package/dist/commands/secrets.js +136 -0
  156. package/dist/commands/secrets.js.map +1 -0
  157. package/dist/docs/builds.md +71 -0
  158. package/dist/docs/index.md +30 -0
  159. package/dist/docs/integrations/drizzle.md +83 -0
  160. package/dist/docs/integrations/nextjs.md +45 -0
  161. package/dist/docs/integrations/prisma.md +110 -0
  162. package/dist/docs/postgres.md +41 -0
  163. package/dist/docs/redis.md +33 -0
  164. package/dist/docs/secrets-config.md +141 -0
  165. package/dist/docs/services.md +325 -0
  166. package/dist/docs/storage.md +62 -0
  167. package/dist/docs/sync.md +66 -0
  168. package/dist/lib/dev/database-manager.d.ts +17 -0
  169. package/dist/lib/dev/database-manager.d.ts.map +1 -0
  170. package/dist/lib/dev/database-manager.js +114 -0
  171. package/dist/lib/dev/database-manager.js.map +1 -0
  172. package/dist/lib/dev/env-resolver.d.ts +14 -0
  173. package/dist/lib/dev/env-resolver.d.ts.map +1 -0
  174. package/dist/lib/dev/env-resolver.js +109 -0
  175. package/dist/lib/dev/env-resolver.js.map +1 -0
  176. package/dist/lib/dev/http-proxy.d.ts +11 -0
  177. package/dist/lib/dev/http-proxy.d.ts.map +1 -0
  178. package/dist/lib/dev/http-proxy.js +165 -0
  179. package/dist/lib/dev/http-proxy.js.map +1 -0
  180. package/dist/lib/dev/index.d.ts +11 -0
  181. package/dist/lib/dev/index.d.ts.map +1 -0
  182. package/dist/lib/dev/index.js +7 -0
  183. package/dist/lib/dev/index.js.map +1 -0
  184. package/dist/lib/dev/instance-state.d.ts +45 -0
  185. package/dist/lib/dev/instance-state.d.ts.map +1 -0
  186. package/dist/lib/dev/instance-state.js +213 -0
  187. package/dist/lib/dev/instance-state.js.map +1 -0
  188. package/dist/lib/dev/port-allocator.d.ts +5 -0
  189. package/dist/lib/dev/port-allocator.d.ts.map +1 -0
  190. package/dist/lib/dev/port-allocator.js +21 -0
  191. package/dist/lib/dev/port-allocator.js.map +1 -0
  192. package/dist/lib/dev/service-runner.d.ts +16 -0
  193. package/dist/lib/dev/service-runner.d.ts.map +1 -0
  194. package/dist/lib/dev/service-runner.js +61 -0
  195. package/dist/lib/dev/service-runner.js.map +1 -0
  196. package/dist/lib/secrets/index.d.ts +2 -0
  197. package/dist/lib/secrets/index.d.ts.map +1 -0
  198. package/dist/lib/secrets/index.js +2 -0
  199. package/dist/lib/secrets/index.js.map +1 -0
  200. package/dist/lib/secrets/parser.d.ts +37 -0
  201. package/dist/lib/secrets/parser.d.ts.map +1 -0
  202. package/dist/lib/secrets/parser.js +116 -0
  203. package/dist/lib/secrets/parser.js.map +1 -0
  204. package/package.json +54 -0
@@ -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,2 @@
1
+ export declare function checkCommand(): void;
2
+ //# sourceMappingURL=check.d.ts.map
@@ -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,2 @@
1
+ export declare function cleanCommand(): void;
2
+ //# sourceMappingURL=clean.d.ts.map
@@ -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,2 @@
1
+ export declare function deployCommand(): void;
2
+ //# sourceMappingURL=deploy.d.ts.map
@@ -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,2 @@
1
+ export declare function devCommand(): void;
2
+ //# sourceMappingURL=dev.d.ts.map
@@ -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