projax 1.3.5 → 1.3.6

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/prxi.js ADDED
@@ -0,0 +1,297 @@
1
+ #!/usr/bin/env node
2
+ "use strict";
3
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
4
+ if (k2 === undefined) k2 = k;
5
+ var desc = Object.getOwnPropertyDescriptor(m, k);
6
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
7
+ desc = { enumerable: true, get: function() { return m[k]; } };
8
+ }
9
+ Object.defineProperty(o, k2, desc);
10
+ }) : (function(o, m, k, k2) {
11
+ if (k2 === undefined) k2 = k;
12
+ o[k2] = m[k];
13
+ }));
14
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
15
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
16
+ }) : function(o, v) {
17
+ o["default"] = v;
18
+ });
19
+ var __importStar = (this && this.__importStar) || (function () {
20
+ var ownKeys = function(o) {
21
+ ownKeys = Object.getOwnPropertyNames || function (o) {
22
+ var ar = [];
23
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
24
+ return ar;
25
+ };
26
+ return ownKeys(o);
27
+ };
28
+ return function (mod) {
29
+ if (mod && mod.__esModule) return mod;
30
+ var result = {};
31
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
32
+ __setModuleDefault(result, mod);
33
+ return result;
34
+ };
35
+ })();
36
+ Object.defineProperty(exports, "__esModule", { value: true });
37
+ const react_1 = __importStar(require("react"));
38
+ const ink_1 = require("ink");
39
+ const core_bridge_1 = require("./core-bridge");
40
+ const script_runner_1 = require("./script-runner");
41
+ // Color scheme matching desktop app
42
+ const colors = {
43
+ bgPrimary: '#0d1117',
44
+ bgSecondary: '#161b22',
45
+ bgTertiary: '#1c2128',
46
+ bgHover: '#21262d',
47
+ borderColor: '#30363d',
48
+ textPrimary: '#c9d1d9',
49
+ textSecondary: '#8b949e',
50
+ textTertiary: '#6e7681',
51
+ accentCyan: '#39c5cf',
52
+ accentBlue: '#58a6ff',
53
+ accentGreen: '#3fb950',
54
+ accentPurple: '#bc8cff',
55
+ accentOrange: '#ffa657',
56
+ };
57
+ const HelpModal = ({ onClose }) => {
58
+ (0, ink_1.useInput)((input, key) => {
59
+ if (input === 'q' || key.escape || key.return) {
60
+ onClose();
61
+ }
62
+ });
63
+ return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", borderStyle: "round", borderColor: colors.accentCyan, padding: 1, width: 60 },
64
+ react_1.default.createElement(ink_1.Text, { bold: true, color: colors.accentCyan }, "PROJAX Terminal UI - Help"),
65
+ react_1.default.createElement(ink_1.Text, null, " "),
66
+ react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, "Navigation:"),
67
+ react_1.default.createElement(ink_1.Text, null, " \u2191/k Move up in project list"),
68
+ react_1.default.createElement(ink_1.Text, null, " \u2193/j Move down in project list"),
69
+ react_1.default.createElement(ink_1.Text, null, " Enter Select project"),
70
+ react_1.default.createElement(ink_1.Text, null, " "),
71
+ react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, "Actions:"),
72
+ react_1.default.createElement(ink_1.Text, null, " s Scan selected project"),
73
+ react_1.default.createElement(ink_1.Text, null, " p Scan ports"),
74
+ react_1.default.createElement(ink_1.Text, null, " "),
75
+ react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, "General:"),
76
+ react_1.default.createElement(ink_1.Text, null, " q/Esc Quit"),
77
+ react_1.default.createElement(ink_1.Text, null, " ? Show this help"),
78
+ react_1.default.createElement(ink_1.Text, null, " "),
79
+ react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, "Press any key to close...")));
80
+ };
81
+ const LoadingModal = ({ message }) => {
82
+ return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", borderStyle: "round", borderColor: colors.accentCyan, padding: 1, width: 40 },
83
+ react_1.default.createElement(ink_1.Text, null, message),
84
+ react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, "Please wait...")));
85
+ };
86
+ const ErrorModal = ({ message, onClose }) => {
87
+ (0, ink_1.useInput)((input, key) => {
88
+ if (key.escape || key.return) {
89
+ onClose();
90
+ }
91
+ });
92
+ return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", borderStyle: "round", borderColor: "#f85149", padding: 1, width: 50 },
93
+ react_1.default.createElement(ink_1.Text, { color: "#f85149", bold: true }, "Error"),
94
+ react_1.default.createElement(ink_1.Text, null, " "),
95
+ react_1.default.createElement(ink_1.Text, null, message),
96
+ react_1.default.createElement(ink_1.Text, null, " "),
97
+ react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, "Press any key to close...")));
98
+ };
99
+ const ProjectListComponent = ({ projects, selectedIndex }) => {
100
+ return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", width: "30%", borderStyle: "round", borderColor: colors.borderColor, padding: 1 },
101
+ react_1.default.createElement(ink_1.Text, { bold: true, color: colors.textPrimary }, "Projects"),
102
+ react_1.default.createElement(ink_1.Text, null, " "),
103
+ projects.length === 0 ? (react_1.default.createElement(ink_1.Text, { color: colors.textTertiary }, "No projects found")) : (projects.map((project, index) => {
104
+ const isSelected = index === selectedIndex;
105
+ const desc = project.description || project.path;
106
+ const shortDesc = desc.length > 30 ? desc.substring(0, 27) + '...' : desc;
107
+ return (react_1.default.createElement(ink_1.Text, { key: project.id, color: isSelected ? colors.accentCyan : colors.textPrimary, bold: isSelected },
108
+ isSelected ? '▶ ' : ' ',
109
+ project.name,
110
+ " - ",
111
+ shortDesc));
112
+ }))));
113
+ };
114
+ const ProjectDetailsComponent = ({ project }) => {
115
+ const [scripts, setScripts] = (0, react_1.useState)(null);
116
+ const [ports, setPorts] = (0, react_1.useState)([]);
117
+ (0, react_1.useEffect)(() => {
118
+ if (!project) {
119
+ setScripts(null);
120
+ setPorts([]);
121
+ return;
122
+ }
123
+ // Load scripts
124
+ try {
125
+ const projectScripts = (0, script_runner_1.getProjectScripts)(project.path);
126
+ setScripts(projectScripts);
127
+ }
128
+ catch (error) {
129
+ setScripts(null);
130
+ }
131
+ // Load ports
132
+ try {
133
+ const db = (0, core_bridge_1.getDatabaseManager)();
134
+ const projectPorts = db.getProjectPorts(project.id);
135
+ setPorts(projectPorts);
136
+ }
137
+ catch (error) {
138
+ setPorts([]);
139
+ }
140
+ }, [project]);
141
+ if (!project) {
142
+ return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", flexGrow: 1, borderStyle: "round", borderColor: colors.borderColor, padding: 1 },
143
+ react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, "Select a project to view details")));
144
+ }
145
+ const lastScanned = project.last_scanned
146
+ ? new Date(project.last_scanned * 1000).toLocaleString()
147
+ : 'Never';
148
+ return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", flexGrow: 1, borderStyle: "round", borderColor: colors.borderColor, padding: 1 },
149
+ react_1.default.createElement(ink_1.Text, { bold: true, color: colors.textPrimary }, project.name),
150
+ react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, project.description || project.path),
151
+ react_1.default.createElement(ink_1.Text, { color: colors.textTertiary }, project.path),
152
+ react_1.default.createElement(ink_1.Text, null, " "),
153
+ project.framework && (react_1.default.createElement(react_1.default.Fragment, null,
154
+ react_1.default.createElement(ink_1.Text, null,
155
+ "Framework: ",
156
+ react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, project.framework)))),
157
+ react_1.default.createElement(ink_1.Text, null,
158
+ "Last Scanned: ",
159
+ lastScanned),
160
+ react_1.default.createElement(ink_1.Text, null, " "),
161
+ scripts && scripts.scripts && scripts.scripts.size > 0 && (react_1.default.createElement(react_1.default.Fragment, null,
162
+ react_1.default.createElement(ink_1.Text, { bold: true },
163
+ "Available Scripts (",
164
+ react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, scripts.scripts.size),
165
+ "):"),
166
+ Array.from(scripts.scripts.entries()).map(([name, script]) => (react_1.default.createElement(ink_1.Text, { key: name },
167
+ ' ',
168
+ react_1.default.createElement(ink_1.Text, { color: colors.accentGreen }, name),
169
+ ' - ',
170
+ script.command))),
171
+ react_1.default.createElement(ink_1.Text, null, " "))),
172
+ ports.length > 0 && (react_1.default.createElement(react_1.default.Fragment, null,
173
+ react_1.default.createElement(ink_1.Text, { bold: true },
174
+ "Detected Ports (",
175
+ react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, ports.length),
176
+ "):"),
177
+ ports.map((port) => (react_1.default.createElement(ink_1.Text, { key: port.id },
178
+ ' ',
179
+ "Port ",
180
+ react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, port.port),
181
+ " - ",
182
+ port.config_source))),
183
+ react_1.default.createElement(ink_1.Text, null, " ")))));
184
+ };
185
+ const StatusBar = () => {
186
+ return (react_1.default.createElement(ink_1.Box, null,
187
+ react_1.default.createElement(ink_1.Text, { color: colors.accentGreen }, "\u25CF API CONNECTED"),
188
+ react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, " | Press "),
189
+ react_1.default.createElement(ink_1.Text, { bold: true }, "q"),
190
+ react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, " to quit, "),
191
+ react_1.default.createElement(ink_1.Text, { bold: true }, "?"),
192
+ react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, " for help")));
193
+ };
194
+ const App = () => {
195
+ const { exit } = (0, ink_1.useApp)();
196
+ const [projects, setProjects] = (0, react_1.useState)([]);
197
+ const [selectedIndex, setSelectedIndex] = (0, react_1.useState)(0);
198
+ const [showHelp, setShowHelp] = (0, react_1.useState)(false);
199
+ const [isLoading, setIsLoading] = (0, react_1.useState)(false);
200
+ const [loadingMessage, setLoadingMessage] = (0, react_1.useState)('');
201
+ const [error, setError] = (0, react_1.useState)(null);
202
+ (0, react_1.useEffect)(() => {
203
+ loadProjects();
204
+ }, []);
205
+ const loadProjects = () => {
206
+ const allProjects = (0, core_bridge_1.getAllProjects)();
207
+ setProjects(allProjects);
208
+ if (selectedIndex >= allProjects.length) {
209
+ setSelectedIndex(Math.max(0, allProjects.length - 1));
210
+ }
211
+ };
212
+ const selectedProject = projects.length > 0 ? projects[selectedIndex] : null;
213
+ (0, ink_1.useInput)((input, key) => {
214
+ // Don't process input if modal is showing
215
+ if (showHelp || isLoading || error) {
216
+ return;
217
+ }
218
+ // Quit
219
+ if (input === 'q' || key.escape) {
220
+ exit();
221
+ return;
222
+ }
223
+ // Help
224
+ if (input === '?') {
225
+ setShowHelp(true);
226
+ return;
227
+ }
228
+ // Navigation
229
+ if (key.upArrow || input === 'k') {
230
+ setSelectedIndex((prev) => Math.max(0, prev - 1));
231
+ return;
232
+ }
233
+ if (key.downArrow || input === 'j') {
234
+ setSelectedIndex((prev) => Math.min(projects.length - 1, prev + 1));
235
+ return;
236
+ }
237
+ // Scan project
238
+ if (input === 's' && selectedProject) {
239
+ setIsLoading(true);
240
+ setLoadingMessage(`Scanning ${selectedProject.name}...`);
241
+ setTimeout(async () => {
242
+ try {
243
+ await (0, core_bridge_1.scanProject)(selectedProject.id);
244
+ loadProjects();
245
+ setIsLoading(false);
246
+ }
247
+ catch (err) {
248
+ setIsLoading(false);
249
+ setError(err instanceof Error ? err.message : String(err));
250
+ }
251
+ }, 100);
252
+ return;
253
+ }
254
+ // Scan ports
255
+ if (input === 'p' && selectedProject) {
256
+ setIsLoading(true);
257
+ setLoadingMessage(`Scanning ports for ${selectedProject.name}...`);
258
+ setTimeout(async () => {
259
+ try {
260
+ // Import port scanner dynamically
261
+ const { scanProjectPorts } = await Promise.resolve().then(() => __importStar(require('./port-scanner')));
262
+ await scanProjectPorts(selectedProject.id);
263
+ loadProjects();
264
+ setIsLoading(false);
265
+ }
266
+ catch (err) {
267
+ setIsLoading(false);
268
+ setError(err instanceof Error ? err.message : String(err));
269
+ }
270
+ }, 100);
271
+ return;
272
+ }
273
+ });
274
+ if (showHelp) {
275
+ return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", padding: 1 },
276
+ react_1.default.createElement(HelpModal, { onClose: () => setShowHelp(false) })));
277
+ }
278
+ if (isLoading) {
279
+ return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", padding: 1 },
280
+ react_1.default.createElement(LoadingModal, { message: loadingMessage })));
281
+ }
282
+ if (error) {
283
+ return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", padding: 1 },
284
+ react_1.default.createElement(ErrorModal, { message: error, onClose: () => setError(null) })));
285
+ }
286
+ return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", padding: 1 },
287
+ react_1.default.createElement(ink_1.Box, { borderStyle: "round", borderColor: colors.borderColor, padding: 1 },
288
+ react_1.default.createElement(ink_1.Text, { bold: true, color: colors.accentCyan }, "PROJAX")),
289
+ react_1.default.createElement(ink_1.Box, { flexDirection: "row", marginTop: 1 },
290
+ react_1.default.createElement(ProjectListComponent, { projects: projects, selectedIndex: selectedIndex }),
291
+ react_1.default.createElement(ink_1.Box, { width: 1 }),
292
+ react_1.default.createElement(ProjectDetailsComponent, { project: selectedProject })),
293
+ react_1.default.createElement(ink_1.Box, { marginTop: 1 },
294
+ react_1.default.createElement(StatusBar, null))));
295
+ };
296
+ // Render the app
297
+ (0, ink_1.render)(react_1.default.createElement(App, null));