@restforgejs/mcp-server 1.2.0

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 (135) hide show
  1. package/LICENSE.md +21 -0
  2. package/README.md +149 -0
  3. package/dist/index.d.ts +2 -0
  4. package/dist/index.js +7 -0
  5. package/dist/index.js.map +1 -0
  6. package/dist/lib/env-parser.d.ts +30 -0
  7. package/dist/lib/env-parser.js +150 -0
  8. package/dist/lib/env-parser.js.map +1 -0
  9. package/dist/lib/exec.d.ts +29 -0
  10. package/dist/lib/exec.js +38 -0
  11. package/dist/lib/exec.js.map +1 -0
  12. package/dist/server.d.ts +1 -0
  13. package/dist/server.js +220 -0
  14. package/dist/server.js.map +1 -0
  15. package/dist/tools/codegen/create-dashboard.d.ts +2 -0
  16. package/dist/tools/codegen/create-dashboard.js +256 -0
  17. package/dist/tools/codegen/create-dashboard.js.map +1 -0
  18. package/dist/tools/codegen/create-endpoint.d.ts +2 -0
  19. package/dist/tools/codegen/create-endpoint.js +263 -0
  20. package/dist/tools/codegen/create-endpoint.js.map +1 -0
  21. package/dist/tools/codegen/dbschema-generate-ddl.d.ts +2 -0
  22. package/dist/tools/codegen/dbschema-generate-ddl.js +187 -0
  23. package/dist/tools/codegen/dbschema-generate-ddl.js.map +1 -0
  24. package/dist/tools/codegen/dbschema-init.d.ts +2 -0
  25. package/dist/tools/codegen/dbschema-init.js +158 -0
  26. package/dist/tools/codegen/dbschema-init.js.map +1 -0
  27. package/dist/tools/codegen/dbschema-introspect.d.ts +2 -0
  28. package/dist/tools/codegen/dbschema-introspect.js +241 -0
  29. package/dist/tools/codegen/dbschema-introspect.js.map +1 -0
  30. package/dist/tools/codegen/dbschema-migrate.d.ts +2 -0
  31. package/dist/tools/codegen/dbschema-migrate.js +219 -0
  32. package/dist/tools/codegen/dbschema-migrate.js.map +1 -0
  33. package/dist/tools/codegen/dbschema-models.d.ts +2 -0
  34. package/dist/tools/codegen/dbschema-models.js +146 -0
  35. package/dist/tools/codegen/dbschema-models.js.map +1 -0
  36. package/dist/tools/codegen/dbschema-validate.d.ts +2 -0
  37. package/dist/tools/codegen/dbschema-validate.js +153 -0
  38. package/dist/tools/codegen/dbschema-validate.js.map +1 -0
  39. package/dist/tools/codegen/describe-table.d.ts +2 -0
  40. package/dist/tools/codegen/describe-table.js +259 -0
  41. package/dist/tools/codegen/describe-table.js.map +1 -0
  42. package/dist/tools/codegen/diff-payload.d.ts +2 -0
  43. package/dist/tools/codegen/diff-payload.js +165 -0
  44. package/dist/tools/codegen/diff-payload.js.map +1 -0
  45. package/dist/tools/codegen/generate-payload.d.ts +2 -0
  46. package/dist/tools/codegen/generate-payload.js +145 -0
  47. package/dist/tools/codegen/generate-payload.js.map +1 -0
  48. package/dist/tools/codegen/get-dashboard-catalog.d.ts +2 -0
  49. package/dist/tools/codegen/get-dashboard-catalog.js +213 -0
  50. package/dist/tools/codegen/get-dashboard-catalog.js.map +1 -0
  51. package/dist/tools/codegen/get-dbschema-catalog.d.ts +2 -0
  52. package/dist/tools/codegen/get-dbschema-catalog.js +244 -0
  53. package/dist/tools/codegen/get-dbschema-catalog.js.map +1 -0
  54. package/dist/tools/codegen/get-field-validation-catalog.d.ts +2 -0
  55. package/dist/tools/codegen/get-field-validation-catalog.js +186 -0
  56. package/dist/tools/codegen/get-field-validation-catalog.js.map +1 -0
  57. package/dist/tools/codegen/get-query-declarative-catalog.d.ts +2 -0
  58. package/dist/tools/codegen/get-query-declarative-catalog.js +200 -0
  59. package/dist/tools/codegen/get-query-declarative-catalog.js.map +1 -0
  60. package/dist/tools/codegen/index.d.ts +2 -0
  61. package/dist/tools/codegen/index.js +43 -0
  62. package/dist/tools/codegen/index.js.map +1 -0
  63. package/dist/tools/codegen/list-tables.d.ts +2 -0
  64. package/dist/tools/codegen/list-tables.js +220 -0
  65. package/dist/tools/codegen/list-tables.js.map +1 -0
  66. package/dist/tools/codegen/sync-payload.d.ts +2 -0
  67. package/dist/tools/codegen/sync-payload.js +177 -0
  68. package/dist/tools/codegen/sync-payload.js.map +1 -0
  69. package/dist/tools/codegen/validate-dashboard-payload.d.ts +2 -0
  70. package/dist/tools/codegen/validate-dashboard-payload.js +239 -0
  71. package/dist/tools/codegen/validate-dashboard-payload.js.map +1 -0
  72. package/dist/tools/codegen/validate-payload.d.ts +2 -0
  73. package/dist/tools/codegen/validate-payload.js +166 -0
  74. package/dist/tools/codegen/validate-payload.js.map +1 -0
  75. package/dist/tools/codegen/validate-sql.d.ts +2 -0
  76. package/dist/tools/codegen/validate-sql.js +270 -0
  77. package/dist/tools/codegen/validate-sql.js.map +1 -0
  78. package/dist/tools/health/index.d.ts +2 -0
  79. package/dist/tools/health/index.js +5 -0
  80. package/dist/tools/health/index.js.map +1 -0
  81. package/dist/tools/health/ping.d.ts +2 -0
  82. package/dist/tools/health/ping.js +68 -0
  83. package/dist/tools/health/ping.js.map +1 -0
  84. package/dist/tools/runtime/check-launcher-exists.d.ts +2 -0
  85. package/dist/tools/runtime/check-launcher-exists.js +111 -0
  86. package/dist/tools/runtime/check-launcher-exists.js.map +1 -0
  87. package/dist/tools/runtime/check-status.d.ts +2 -0
  88. package/dist/tools/runtime/check-status.js +417 -0
  89. package/dist/tools/runtime/check-status.js.map +1 -0
  90. package/dist/tools/runtime/detect-config.d.ts +2 -0
  91. package/dist/tools/runtime/detect-config.js +130 -0
  92. package/dist/tools/runtime/detect-config.js.map +1 -0
  93. package/dist/tools/runtime/detect-project.d.ts +2 -0
  94. package/dist/tools/runtime/detect-project.js +132 -0
  95. package/dist/tools/runtime/detect-project.js.map +1 -0
  96. package/dist/tools/runtime/generate-launcher.d.ts +2 -0
  97. package/dist/tools/runtime/generate-launcher.js +438 -0
  98. package/dist/tools/runtime/generate-launcher.js.map +1 -0
  99. package/dist/tools/runtime/index.d.ts +2 -0
  100. package/dist/tools/runtime/index.js +15 -0
  101. package/dist/tools/runtime/index.js.map +1 -0
  102. package/dist/tools/runtime/validate-preflight.d.ts +2 -0
  103. package/dist/tools/runtime/validate-preflight.js +209 -0
  104. package/dist/tools/runtime/validate-preflight.js.map +1 -0
  105. package/dist/tools/setup/create-folder.d.ts +2 -0
  106. package/dist/tools/setup/create-folder.js +138 -0
  107. package/dist/tools/setup/create-folder.js.map +1 -0
  108. package/dist/tools/setup/get-config-schema.d.ts +2 -0
  109. package/dist/tools/setup/get-config-schema.js +158 -0
  110. package/dist/tools/setup/get-config-schema.js.map +1 -0
  111. package/dist/tools/setup/get-init-template.d.ts +2 -0
  112. package/dist/tools/setup/get-init-template.js +130 -0
  113. package/dist/tools/setup/get-init-template.js.map +1 -0
  114. package/dist/tools/setup/index.d.ts +2 -0
  115. package/dist/tools/setup/index.js +21 -0
  116. package/dist/tools/setup/index.js.map +1 -0
  117. package/dist/tools/setup/init-config.d.ts +2 -0
  118. package/dist/tools/setup/init-config.js +120 -0
  119. package/dist/tools/setup/init-config.js.map +1 -0
  120. package/dist/tools/setup/install-package.d.ts +2 -0
  121. package/dist/tools/setup/install-package.js +133 -0
  122. package/dist/tools/setup/install-package.js.map +1 -0
  123. package/dist/tools/setup/read-env.d.ts +2 -0
  124. package/dist/tools/setup/read-env.js +138 -0
  125. package/dist/tools/setup/read-env.js.map +1 -0
  126. package/dist/tools/setup/update-env.d.ts +2 -0
  127. package/dist/tools/setup/update-env.js +176 -0
  128. package/dist/tools/setup/update-env.js.map +1 -0
  129. package/dist/tools/setup/validate-config.d.ts +2 -0
  130. package/dist/tools/setup/validate-config.js +138 -0
  131. package/dist/tools/setup/validate-config.js.map +1 -0
  132. package/dist/tools/setup/write-env.d.ts +2 -0
  133. package/dist/tools/setup/write-env.js +168 -0
  134. package/dist/tools/setup/write-env.js.map +1 -0
  135. package/package.json +60 -0
package/LICENSE.md ADDED
@@ -0,0 +1,21 @@
1
+ # MIT License
2
+
3
+ Copyright (c) 2026 restforge
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,149 @@
1
+ # @restforgejs/mcp-server
2
+
3
+ MCP (Model Context Protocol) server for the RESTForge framework. Exposes RESTForge capabilities to AI Agents (Claude Desktop, Cursor, Claude CLI, and other MCP clients) so agents can operate RESTForge through natural language without manually invoking CLI commands.
4
+
5
+ ## Requirements
6
+
7
+ - Node.js >= 18
8
+ - npm >= 9
9
+ - For full setup workflow: PostgreSQL / MySQL / Oracle / SQLite, RESTForge license key
10
+
11
+ ## Installation
12
+
13
+ ```bash
14
+ npm install -g @restforgejs/mcp-server
15
+ ```
16
+
17
+ After installation, the `restforge-mcp` command is available in PATH.
18
+
19
+ ## Quick Start
20
+
21
+ ### 1. Verify Install
22
+
23
+ ```bash
24
+ echo '{"jsonrpc":"2.0","id":1,"method":"tools/list"}' | restforge-mcp
25
+ ```
26
+
27
+ Output should list 29 tools across the `health_*`, `setup_*`, `codegen_*`, and `runtime_*` domains.
28
+
29
+ ### 2. Register with MCP Client
30
+
31
+ **Claude CLI** (user scope, applies to all projects):
32
+
33
+ ```bash
34
+ claude mcp add --transport stdio --scope user restforge -- restforge-mcp
35
+ ```
36
+
37
+ **Cursor** (`.cursor/mcp.json` in project root):
38
+
39
+ ```json
40
+ {
41
+ "mcpServers": {
42
+ "restforge": {
43
+ "command": "restforge-mcp"
44
+ }
45
+ }
46
+ }
47
+ ```
48
+
49
+ **Claude Desktop** (`claude_desktop_config.json`):
50
+
51
+ ```json
52
+ {
53
+ "mcpServers": {
54
+ "restforge": {
55
+ "command": "restforge-mcp"
56
+ }
57
+ }
58
+ }
59
+ ```
60
+
61
+ ### 3. Use via Natural Language
62
+
63
+ In your AI client chat, type prompts like:
64
+
65
+ > Setup a new RESTForge project at `d:/projects/api-test` with PostgreSQL on localhost:5432, license `XXXX-XXXX-XXXX-XXXX`
66
+
67
+ > Generate a CRUD endpoint for the `customer` table
68
+
69
+ > Run my RESTForge server (the agent generates a launcher script for the user to execute)
70
+
71
+ The agent orchestrates the appropriate tools to fulfill the request end-to-end.
72
+
73
+ ## Available Tools
74
+
75
+ 29 tools organized by domain. AI agents call these via the MCP protocol; end users do not invoke them directly.
76
+
77
+ ### Health Domain (1 tool)
78
+
79
+ | Tool | Description |
80
+ |------|-------------|
81
+ | `health_ping` | Smoke test MCP transport. Returns `pong` + ISO timestamp + server version |
82
+
83
+ ### Setup Domain (9 tools)
84
+
85
+ | Tool | Description |
86
+ |------|-------------|
87
+ | `setup_create_folder` | Create a new project folder for RESTForge |
88
+ | `setup_install_package` | Install `@restforgejs/platform` into the project's `node_modules` via npm |
89
+ | `setup_init_config` | Generate skeleton config and sample payloads via `restforge init` |
90
+ | `setup_write_env` | Write `config/db-connection.env` with license, server, and database settings |
91
+ | `setup_read_env` | Read current values from `config/db-connection.env` |
92
+ | `setup_update_env` | Update individual fields in `config/db-connection.env` |
93
+ | `setup_validate_config` | Validate license and connections to database, redis, and kafka |
94
+ | `setup_get_config_schema` | Get JSON schema of all 63 parameters available in `db-connection.env` |
95
+ | `setup_get_init_template` | Get raw `db-connection.env` template content |
96
+
97
+ ### Codegen Domain (13 tools)
98
+
99
+ | Tool | Description |
100
+ |------|-------------|
101
+ | `codegen_list_tables` | List all tables in the project's database (live introspection) |
102
+ | `codegen_describe_table` | Describe columns, primary key, and foreign keys of a specific table |
103
+ | `codegen_generate_payload` | Generate payload JSON from a database table |
104
+ | `codegen_validate_payload` | Validate payload JSON structure and constraints |
105
+ | `codegen_validate_dashboard_payload` | Validate dashboard payload structure |
106
+ | `codegen_diff_payload` | Diff payload JSON against the database schema |
107
+ | `codegen_sync_payload` | Sync payload JSON with the database schema |
108
+ | `codegen_create_endpoint` | Scaffold an endpoint module from a payload spec |
109
+ | `codegen_create_dashboard` | Scaffold a dashboard module from a payload spec |
110
+ | `codegen_validate_sql` | Validate a SELECT or WITH (CTE) SQL statement via EXPLAIN against the live database |
111
+ | `codegen_get_field_validation_catalog` | Get the field validation catalog (for grounding payload constraints) |
112
+ | `codegen_get_query_declarative_catalog` | Get the query declarative catalog (for grounding query JSON) |
113
+ | `codegen_get_dashboard_catalog` | Get the dashboard widget catalog (for grounding dashboard config) |
114
+
115
+ ### Runtime Domain (6 tools)
116
+
117
+ | Tool | Description |
118
+ |------|-------------|
119
+ | `runtime_detect_project` | Scan `src/modules/*.js` to list project names |
120
+ | `runtime_detect_config` | Scan `config/*.env` to list available config files |
121
+ | `runtime_validate_preflight` | Validate config + check PID file + check port availability before launch |
122
+ | `runtime_check_launcher_exists` | Check if launcher files (`server-start.bat`/`.sh`, `ecosystem.config.js`) exist in the project root |
123
+ | `runtime_generate_launcher` | Generate `server-start.bat`/`.sh` + `server-stop.bat`/`.sh` (and `ecosystem.config.js` for PM2 mode) |
124
+ | `runtime_check_status` | Detect if the server is running (host or PM2 mode) with optional HTTP health probe |
125
+
126
+ > **Runtime principle**: AI agents never start, stop, or restart the server directly. The runtime tools only generate launcher scripts that the user executes themselves, so the running server lives independently of the AI session.
127
+
128
+ ## Compatibility
129
+
130
+ This MCP server works with any MCP client that supports the stdio transport, including but not limited to:
131
+
132
+ - Claude Desktop
133
+ - Claude CLI (Claude Code)
134
+ - Cursor
135
+ - Windsurf
136
+ - Cline (VS Code extension)
137
+ - Continue (VS Code/JetBrains extension)
138
+ - Zed
139
+
140
+ The model used (Claude, GPT, Gemini, etc.) depends on the client configuration. Tool selection accuracy is best with frontier models that have mature tool-calling support.
141
+
142
+ ## Repository
143
+
144
+ - Source: [https://github.com/restforge-dev/mcp-server](https://github.com/restforge-dev/mcp-server)
145
+ - Issues: [https://github.com/restforge-dev/mcp-server/issues](https://github.com/restforge-dev/mcp-server/issues)
146
+
147
+ ## License
148
+
149
+ MIT — see [LICENSE.md](LICENSE.md).
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ export {};
package/dist/index.js ADDED
@@ -0,0 +1,7 @@
1
+ #!/usr/bin/env node
2
+ import { startServer } from './server.js';
3
+ startServer().catch((error) => {
4
+ console.error('[restforge-mcp] Fatal error:', error);
5
+ process.exit(1);
6
+ });
7
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAE1C,WAAW,EAAE,CAAC,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;IAC5B,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;IACrD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC"}
@@ -0,0 +1,30 @@
1
+ export type EnvEntry = {
2
+ kind: 'kv';
3
+ key: string;
4
+ value: string;
5
+ commentInline?: string;
6
+ } | {
7
+ kind: 'comment';
8
+ text: string;
9
+ } | {
10
+ kind: 'blank';
11
+ };
12
+ export type EnvFieldValue = string | number | boolean;
13
+ export declare function parseEnvFile(content: string): EnvEntry[];
14
+ export declare function serializeEnvFile(entries: EnvEntry[]): string;
15
+ export interface MergeResult {
16
+ entries: EnvEntry[];
17
+ updated: Array<{
18
+ key: string;
19
+ before: string;
20
+ after: string;
21
+ }>;
22
+ added: Array<{
23
+ key: string;
24
+ value: string;
25
+ }>;
26
+ unchanged: string[];
27
+ }
28
+ export declare function mergeEnvEntries(existing: EnvEntry[], updates: Record<string, EnvFieldValue>): MergeResult;
29
+ export declare function isSensitiveKey(key: string): boolean;
30
+ export declare function maskValue(value: string): string;
@@ -0,0 +1,150 @@
1
+ const KV_REGEX = /^\s*([A-Za-z_][A-Za-z0-9_]*)\s*=(.*)$/;
2
+ function stripQuotes(raw) {
3
+ const trimmed = raw.trim();
4
+ if (trimmed.length >= 2) {
5
+ const first = trimmed[0];
6
+ const last = trimmed[trimmed.length - 1];
7
+ if ((first === '"' && last === '"') || (first === "'" && last === "'")) {
8
+ return trimmed.slice(1, -1);
9
+ }
10
+ }
11
+ return trimmed;
12
+ }
13
+ function splitInlineComment(rawValue) {
14
+ const trimmed = rawValue.trimStart();
15
+ if (trimmed.startsWith('"') || trimmed.startsWith("'")) {
16
+ const quote = trimmed[0];
17
+ const closingIdx = trimmed.indexOf(quote, 1);
18
+ if (closingIdx !== -1) {
19
+ const afterQuote = trimmed.slice(closingIdx + 1);
20
+ const hashIdx = afterQuote.indexOf('#');
21
+ if (hashIdx !== -1) {
22
+ return {
23
+ value: trimmed.slice(0, closingIdx + 1),
24
+ comment: afterQuote.slice(hashIdx + 1).trim(),
25
+ };
26
+ }
27
+ return { value: trimmed };
28
+ }
29
+ return { value: trimmed };
30
+ }
31
+ const hashIdx = trimmed.indexOf(' #');
32
+ if (hashIdx !== -1) {
33
+ return {
34
+ value: trimmed.slice(0, hashIdx).trimEnd(),
35
+ comment: trimmed.slice(hashIdx + 2).trim(),
36
+ };
37
+ }
38
+ return { value: trimmed.trimEnd() };
39
+ }
40
+ export function parseEnvFile(content) {
41
+ const lines = content.split(/\r?\n/);
42
+ const entries = [];
43
+ const lastIsTrailingNewline = lines.length > 0 && lines[lines.length - 1] === '';
44
+ const effectiveLines = lastIsTrailingNewline ? lines.slice(0, -1) : lines;
45
+ for (const rawLine of effectiveLines) {
46
+ const trimmed = rawLine.trim();
47
+ if (trimmed === '') {
48
+ entries.push({ kind: 'blank' });
49
+ continue;
50
+ }
51
+ if (trimmed.startsWith('#')) {
52
+ entries.push({ kind: 'comment', text: rawLine });
53
+ continue;
54
+ }
55
+ const match = KV_REGEX.exec(rawLine);
56
+ if (!match) {
57
+ entries.push({ kind: 'comment', text: rawLine });
58
+ continue;
59
+ }
60
+ const key = match[1];
61
+ const { value: rawValue, comment } = splitInlineComment(match[2]);
62
+ const value = stripQuotes(rawValue);
63
+ const entry = { kind: 'kv', key, value };
64
+ if (comment !== undefined)
65
+ entry.commentInline = comment;
66
+ entries.push(entry);
67
+ }
68
+ return entries;
69
+ }
70
+ function needsQuoting(value) {
71
+ if (value === '')
72
+ return false;
73
+ return /[\s#"'=]/.test(value);
74
+ }
75
+ function quoteValue(value) {
76
+ const escaped = value.replace(/\\/g, '\\\\').replace(/"/g, '\\"');
77
+ return `"${escaped}"`;
78
+ }
79
+ function formatValue(value) {
80
+ return needsQuoting(value) ? quoteValue(value) : value;
81
+ }
82
+ export function serializeEnvFile(entries) {
83
+ const out = [];
84
+ for (const entry of entries) {
85
+ if (entry.kind === 'blank') {
86
+ out.push('');
87
+ }
88
+ else if (entry.kind === 'comment') {
89
+ out.push(entry.text);
90
+ }
91
+ else {
92
+ const valuePart = formatValue(entry.value);
93
+ const line = entry.commentInline
94
+ ? `${entry.key}=${valuePart} # ${entry.commentInline}`
95
+ : `${entry.key}=${valuePart}`;
96
+ out.push(line);
97
+ }
98
+ }
99
+ return out.join('\n') + '\n';
100
+ }
101
+ function normalizeFieldValue(value) {
102
+ if (typeof value === 'boolean')
103
+ return value ? 'true' : 'false';
104
+ if (typeof value === 'number')
105
+ return String(value);
106
+ return value;
107
+ }
108
+ export function mergeEnvEntries(existing, updates) {
109
+ const updateKeys = new Set(Object.keys(updates));
110
+ const updated = [];
111
+ const added = [];
112
+ const unchanged = [];
113
+ const next = existing.map((entry) => {
114
+ if (entry.kind !== 'kv')
115
+ return entry;
116
+ if (!updateKeys.has(entry.key)) {
117
+ unchanged.push(entry.key);
118
+ return entry;
119
+ }
120
+ const newValue = normalizeFieldValue(updates[entry.key]);
121
+ updateKeys.delete(entry.key);
122
+ if (entry.value === newValue) {
123
+ unchanged.push(entry.key);
124
+ return entry;
125
+ }
126
+ updated.push({ key: entry.key, before: entry.value, after: newValue });
127
+ return { ...entry, value: newValue };
128
+ });
129
+ for (const remainingKey of updateKeys) {
130
+ const value = normalizeFieldValue(updates[remainingKey]);
131
+ next.push({ kind: 'kv', key: remainingKey, value });
132
+ added.push({ key: remainingKey, value });
133
+ }
134
+ return { entries: next, updated, added, unchanged };
135
+ }
136
+ const SENSITIVE_KEYS = new Set([
137
+ 'LICENSE',
138
+ 'DB_PASSWORD',
139
+ 'REDIS_PASSWORD',
140
+ 'KAFKA_SASL_PASSWORD',
141
+ ]);
142
+ export function isSensitiveKey(key) {
143
+ return SENSITIVE_KEYS.has(key);
144
+ }
145
+ export function maskValue(value) {
146
+ if (value === '')
147
+ return '';
148
+ return '****';
149
+ }
150
+ //# sourceMappingURL=env-parser.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"env-parser.js","sourceRoot":"","sources":["../../src/lib/env-parser.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,29 @@
1
+ export interface ExecResult {
2
+ success: boolean;
3
+ stdout: string;
4
+ stderr: string;
5
+ exitCode: number;
6
+ command: string;
7
+ }
8
+ export interface ExecOptions {
9
+ cwd?: string;
10
+ timeout?: number;
11
+ /**
12
+ * Environment variables for the subprocess. Merged with parent process env
13
+ * via spread: { ...process.env, ...options.env }. Use this to suppress
14
+ * tooling banners (e.g. NODE_ENV: 'production') or pass tool-specific config.
15
+ */
16
+ env?: NodeJS.ProcessEnv;
17
+ /**
18
+ * Whether to strip the trailing newline from stdout/stderr. Default true
19
+ * (matches execa default). Set to false when byte-perfect passthrough is
20
+ * required (e.g. CLI commands that output template files where the final
21
+ * newline matters).
22
+ */
23
+ stripFinalNewline?: boolean;
24
+ }
25
+ /**
26
+ * Execute subprocess dengan structured result. Tidak throw — selalu return result.
27
+ * Cocok untuk wrapping CLI calls seperti npm, npx, restforge.
28
+ */
29
+ export declare function execProcess(command: string, args: string[], options?: ExecOptions): Promise<ExecResult>;
@@ -0,0 +1,38 @@
1
+ import { execa } from 'execa';
2
+ /**
3
+ * Execute subprocess dengan structured result. Tidak throw — selalu return result.
4
+ * Cocok untuk wrapping CLI calls seperti npm, npx, restforge.
5
+ */
6
+ export async function execProcess(command, args, options = {}) {
7
+ const { cwd = process.cwd(), timeout = 60_000, env, stripFinalNewline = true } = options;
8
+ const fullCommand = `${command} ${args.join(' ')}`;
9
+ // Merge env: parent env first, custom env overrides
10
+ const mergedEnv = env ? { ...process.env, ...env } : undefined;
11
+ try {
12
+ const result = await execa(command, args, {
13
+ cwd,
14
+ timeout,
15
+ reject: false,
16
+ stripFinalNewline,
17
+ ...(mergedEnv ? { env: mergedEnv } : {}),
18
+ });
19
+ return {
20
+ success: result.exitCode === 0,
21
+ stdout: result.stdout,
22
+ stderr: result.stderr,
23
+ exitCode: result.exitCode ?? -1,
24
+ command: fullCommand,
25
+ };
26
+ }
27
+ catch (error) {
28
+ const e = error;
29
+ return {
30
+ success: false,
31
+ stdout: e.stdout?.toString() ?? '',
32
+ stderr: e.stderr?.toString() ?? e.message,
33
+ exitCode: e.exitCode ?? -1,
34
+ command: fullCommand,
35
+ };
36
+ }
37
+ }
38
+ //# sourceMappingURL=exec.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"exec.js","sourceRoot":"","sources":["../../src/lib/exec.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAmB,MAAM,OAAO,CAAC;AA4B/C;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,WAAW,CAC/B,OAAe,EACf,IAAc,EACd,UAAuB,EAAE;IAEzB,MAAM,EAAE,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE,EAAE,OAAO,GAAG,MAAM,EAAE,GAAG,EAAE,iBAAiB,GAAG,IAAI,EAAE,GAAG,OAAO,CAAC;IACzF,MAAM,WAAW,GAAG,GAAG,OAAO,IAAI,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;IAEnD,oDAAoD;IACpD,MAAM,SAAS,GAAG,GAAG,CAAC,CAAC,CAAC,EAAE,GAAG,OAAO,CAAC,GAAG,EAAE,GAAG,GAAG,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;IAE/D,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,MAAM,KAAK,CAAC,OAAO,EAAE,IAAI,EAAE;YACxC,GAAG;YACH,OAAO;YACP,MAAM,EAAE,KAAK;YACb,iBAAiB;YACjB,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,GAAG,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;SACzC,CAAC,CAAC;QACH,OAAO;YACL,OAAO,EAAE,MAAM,CAAC,QAAQ,KAAK,CAAC;YAC9B,MAAM,EAAE,MAAM,CAAC,MAAM;YACrB,MAAM,EAAE,MAAM,CAAC,MAAM;YACrB,QAAQ,EAAE,MAAM,CAAC,QAAQ,IAAI,CAAC,CAAC;YAC/B,OAAO,EAAE,WAAW;SACrB,CAAC;IACJ,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,CAAC,GAAG,KAAmB,CAAC;QAC9B,OAAO;YACL,OAAO,EAAE,KAAK;YACd,MAAM,EAAE,CAAC,CAAC,MAAM,EAAE,QAAQ,EAAE,IAAI,EAAE;YAClC,MAAM,EAAE,CAAC,CAAC,MAAM,EAAE,QAAQ,EAAE,IAAI,CAAC,CAAC,OAAO;YACzC,QAAQ,EAAE,CAAC,CAAC,QAAQ,IAAI,CAAC,CAAC;YAC1B,OAAO,EAAE,WAAW;SACrB,CAAC;IACJ,CAAC;AACH,CAAC"}
@@ -0,0 +1 @@
1
+ export declare function startServer(): Promise<void>;