@jetstart/cli 1.2.0 → 1.5.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 (67) hide show
  1. package/dist/cli.js +18 -1
  2. package/dist/commands/android-emulator.d.ts +8 -0
  3. package/dist/commands/android-emulator.js +280 -0
  4. package/dist/commands/create.d.ts +1 -6
  5. package/dist/commands/create.js +119 -0
  6. package/dist/commands/dev.d.ts +1 -7
  7. package/dist/commands/dev.js +69 -10
  8. package/dist/commands/index.d.ts +2 -0
  9. package/dist/commands/index.js +5 -1
  10. package/dist/commands/install-audit.d.ts +9 -0
  11. package/dist/commands/install-audit.js +185 -0
  12. package/dist/types/index.d.ts +22 -0
  13. package/dist/types/index.js +8 -0
  14. package/dist/utils/android-sdk.d.ts +81 -0
  15. package/dist/utils/android-sdk.js +432 -0
  16. package/dist/utils/downloader.d.ts +35 -0
  17. package/dist/utils/downloader.js +214 -0
  18. package/dist/utils/emulator-deployer.d.ts +29 -0
  19. package/dist/utils/emulator-deployer.js +224 -0
  20. package/dist/utils/emulator.d.ts +101 -0
  21. package/dist/utils/emulator.js +410 -0
  22. package/dist/utils/java.d.ts +25 -0
  23. package/dist/utils/java.js +363 -0
  24. package/dist/utils/system-tools.d.ts +93 -0
  25. package/dist/utils/system-tools.js +599 -0
  26. package/dist/utils/template.js +777 -748
  27. package/package.json +7 -3
  28. package/src/cli.ts +20 -2
  29. package/src/commands/android-emulator.ts +304 -0
  30. package/src/commands/create.ts +128 -5
  31. package/src/commands/dev.ts +71 -18
  32. package/src/commands/index.ts +3 -1
  33. package/src/commands/install-audit.ts +227 -0
  34. package/src/types/index.ts +30 -0
  35. package/src/utils/android-sdk.ts +478 -0
  36. package/src/utils/downloader.ts +201 -0
  37. package/src/utils/emulator-deployer.ts +210 -0
  38. package/src/utils/emulator.ts +463 -0
  39. package/src/utils/java.ts +369 -0
  40. package/src/utils/system-tools.ts +648 -0
  41. package/src/utils/template.ts +875 -867
  42. package/dist/cli.d.ts.map +0 -1
  43. package/dist/cli.js.map +0 -1
  44. package/dist/commands/build.d.ts.map +0 -1
  45. package/dist/commands/build.js.map +0 -1
  46. package/dist/commands/create.d.ts.map +0 -1
  47. package/dist/commands/create.js.map +0 -1
  48. package/dist/commands/dev.d.ts.map +0 -1
  49. package/dist/commands/dev.js.map +0 -1
  50. package/dist/commands/index.d.ts.map +0 -1
  51. package/dist/commands/index.js.map +0 -1
  52. package/dist/commands/logs.d.ts.map +0 -1
  53. package/dist/commands/logs.js.map +0 -1
  54. package/dist/index.d.ts.map +0 -1
  55. package/dist/index.js.map +0 -1
  56. package/dist/types/index.d.ts.map +0 -1
  57. package/dist/types/index.js.map +0 -1
  58. package/dist/utils/index.d.ts.map +0 -1
  59. package/dist/utils/index.js.map +0 -1
  60. package/dist/utils/logger.d.ts.map +0 -1
  61. package/dist/utils/logger.js.map +0 -1
  62. package/dist/utils/prompt.d.ts.map +0 -1
  63. package/dist/utils/prompt.js.map +0 -1
  64. package/dist/utils/spinner.d.ts.map +0 -1
  65. package/dist/utils/spinner.js.map +0 -1
  66. package/dist/utils/template.d.ts.map +0 -1
  67. package/dist/utils/template.js.map +0 -1
package/dist/cli.js CHANGED
@@ -14,12 +14,15 @@ const create_1 = require("./commands/create");
14
14
  const dev_1 = require("./commands/dev");
15
15
  const build_1 = require("./commands/build");
16
16
  const logs_1 = require("./commands/logs");
17
+ const install_audit_1 = require("./commands/install-audit");
18
+ const android_emulator_1 = require("./commands/android-emulator");
17
19
  const shared_1 = require("@jetstart/shared");
20
+ const package_json_1 = require("../package.json");
18
21
  const program = new commander_1.Command();
19
22
  program
20
23
  .name('jetstart')
21
24
  .description('Fast, wireless Android development with Kotlin and Jetpack Compose')
22
- .version(shared_1.JETSTART_VERSION);
25
+ .version(package_json_1.version || shared_1.JETSTART_VERSION);
23
26
  // Create command
24
27
  program
25
28
  .command('create <name>')
@@ -27,6 +30,7 @@ program
27
30
  .option('-p, --package <name>', 'Package name (e.g., com.example.app)')
28
31
  .option('-t, --template <name>', 'Template to use', 'default')
29
32
  .option('--skip-install', 'Skip npm install')
33
+ .option('--full-install', 'Automatically install all required Android dependencies')
30
34
  .action(create_1.createCommand);
31
35
  // Dev command
32
36
  program
@@ -36,6 +40,8 @@ program
36
40
  .option('-H, --host <host>', 'Host address (defaults to auto-detected network IP)')
37
41
  .option('--no-qr', 'Do not show QR code')
38
42
  .option('--no-open', 'Do not open browser')
43
+ .option('--emulator', 'Deploy to running Android emulator')
44
+ .option('--avd <name>', 'Target specific emulator by name')
39
45
  .action(dev_1.devCommand);
40
46
  // Build command
41
47
  program
@@ -54,6 +60,17 @@ program
54
60
  .option('-s, --source <source>', 'Filter by log source')
55
61
  .option('-n, --lines <number>', 'Number of lines to show', '100')
56
62
  .action(logs_1.logsCommand);
63
+ // Install audit command
64
+ program
65
+ .command('install-audit')
66
+ .description('Audit installation of required tools and dependencies')
67
+ .option('--json', 'Output results in JSON format')
68
+ .action(install_audit_1.installAuditCommand);
69
+ // Android emulator command
70
+ program
71
+ .command('android-emulator')
72
+ .description('Manage Android emulators (AVDs)')
73
+ .action(android_emulator_1.androidEmulatorCommand);
57
74
  // Error handling
58
75
  program.on('command:*', (operands) => {
59
76
  console.error(chalk_1.default.red(`Error: Unknown command '${operands[0]}'`));
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Android emulator management command
3
+ */
4
+ /**
5
+ * Android emulator command handler
6
+ */
7
+ export declare function androidEmulatorCommand(): Promise<void>;
8
+ //# sourceMappingURL=android-emulator.d.ts.map
@@ -0,0 +1,280 @@
1
+ "use strict";
2
+ /**
3
+ * Android emulator management command
4
+ */
5
+ var __importDefault = (this && this.__importDefault) || function (mod) {
6
+ return (mod && mod.__esModule) ? mod : { "default": mod };
7
+ };
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ exports.androidEmulatorCommand = androidEmulatorCommand;
10
+ const inquirer_1 = __importDefault(require("inquirer"));
11
+ const chalk_1 = __importDefault(require("chalk"));
12
+ const emulator_1 = require("../utils/emulator");
13
+ const logger_1 = require("../utils/logger");
14
+ /**
15
+ * Display list of AVDs
16
+ */
17
+ async function handleListAVDs() {
18
+ const avdManager = (0, emulator_1.createAVDManager)();
19
+ const avds = await avdManager.listAVDs();
20
+ if (avds.length === 0) {
21
+ console.log();
22
+ (0, logger_1.warning)('No Android emulators found.');
23
+ console.log();
24
+ const { shouldCreate } = await inquirer_1.default.prompt([
25
+ {
26
+ type: 'confirm',
27
+ name: 'shouldCreate',
28
+ message: 'Would you like to create a JetStart-optimized emulator?',
29
+ default: true,
30
+ },
31
+ ]);
32
+ if (shouldCreate) {
33
+ await handleCreateJetStartAVD();
34
+ }
35
+ return;
36
+ }
37
+ console.log();
38
+ console.log(chalk_1.default.bold('Available Android Virtual Devices:'));
39
+ console.log('━'.repeat(70));
40
+ console.log();
41
+ for (const avd of avds) {
42
+ const statusIcon = avd.running ? chalk_1.default.green('✓') : chalk_1.default.gray('○');
43
+ const statusText = avd.running ? chalk_1.default.green('(Running)') : '';
44
+ console.log(` ${statusIcon} ${chalk_1.default.bold(avd.name)} ${statusText}`);
45
+ console.log(` - Device: ${avd.device || 'Unknown'}`);
46
+ console.log(` - Target: ${avd.target || 'Unknown'}`);
47
+ if (avd.basedOn) {
48
+ console.log(` - Based on: ${avd.basedOn}`);
49
+ }
50
+ console.log();
51
+ }
52
+ }
53
+ /**
54
+ * Start an emulator
55
+ */
56
+ async function handleStartEmulator() {
57
+ const avdManager = (0, emulator_1.createAVDManager)();
58
+ const avds = await avdManager.listAVDs();
59
+ if (avds.length === 0) {
60
+ (0, logger_1.warning)('No emulators available. Create one first.');
61
+ return;
62
+ }
63
+ // Filter out running emulators
64
+ const availableAVDs = avds.filter((avd) => !avd.running);
65
+ if (availableAVDs.length === 0) {
66
+ (0, logger_1.info)('All emulators are already running.');
67
+ return;
68
+ }
69
+ const { avdName } = await inquirer_1.default.prompt([
70
+ {
71
+ type: 'list',
72
+ name: 'avdName',
73
+ message: 'Select emulator to start:',
74
+ choices: availableAVDs.map((avd) => ({ name: `${avd.name} (${avd.target})`, value: avd.name })),
75
+ },
76
+ ]);
77
+ await avdManager.startEmulator({ avdName });
78
+ }
79
+ /**
80
+ * Stop an emulator
81
+ */
82
+ async function handleStopEmulator() {
83
+ const avdManager = (0, emulator_1.createAVDManager)();
84
+ const avds = await avdManager.listAVDs();
85
+ // Filter running emulators
86
+ const runningAVDs = avds.filter((avd) => avd.running);
87
+ if (runningAVDs.length === 0) {
88
+ (0, logger_1.warning)('No emulators are currently running.');
89
+ return;
90
+ }
91
+ const { avdName } = await inquirer_1.default.prompt([
92
+ {
93
+ type: 'list',
94
+ name: 'avdName',
95
+ message: 'Select emulator to stop:',
96
+ choices: runningAVDs.map((avd) => ({ name: `${avd.name} (${avd.target})`, value: avd.name })),
97
+ },
98
+ ]);
99
+ await avdManager.stopEmulator(avdName);
100
+ }
101
+ /**
102
+ * Create JetStart-optimized AVD
103
+ */
104
+ async function handleCreateJetStartAVD() {
105
+ const avdManager = (0, emulator_1.createAVDManager)();
106
+ try {
107
+ await avdManager.createJetStartAVD();
108
+ }
109
+ catch (err) {
110
+ (0, logger_1.error)(`Failed to create AVD: ${err.message}`);
111
+ }
112
+ }
113
+ /**
114
+ * Create custom AVD
115
+ */
116
+ async function handleCreateCustomAVD() {
117
+ const avdManager = (0, emulator_1.createAVDManager)();
118
+ const answers = await inquirer_1.default.prompt([
119
+ {
120
+ type: 'input',
121
+ name: 'name',
122
+ message: 'Enter AVD name:',
123
+ default: 'my-android-emulator',
124
+ validate: (input) => {
125
+ if (!input || input.trim().length === 0) {
126
+ return 'Name cannot be empty';
127
+ }
128
+ if (!/^[a-zA-Z0-9_-]+$/.test(input)) {
129
+ return 'Name can only contain letters, numbers, hyphens, and underscores';
130
+ }
131
+ return true;
132
+ },
133
+ },
134
+ {
135
+ type: 'list',
136
+ name: 'device',
137
+ message: 'Select device profile:',
138
+ choices: [
139
+ { name: 'Pixel 5', value: 'pixel_5' },
140
+ { name: 'Pixel 6', value: 'pixel_6' },
141
+ { name: 'Pixel 8', value: 'pixel_8' },
142
+ { name: 'Nexus 5', value: 'Nexus 5' },
143
+ ],
144
+ default: 'pixel_5',
145
+ },
146
+ {
147
+ type: 'list',
148
+ name: 'apiLevel',
149
+ message: 'Select API level:',
150
+ choices: [
151
+ { name: 'API 34 (Android 14) - Recommended', value: 34 },
152
+ { name: 'API 33 (Android 13)', value: 33 },
153
+ { name: 'API 31 (Android 12)', value: 31 },
154
+ { name: 'API 29 (Android 10)', value: 29 },
155
+ { name: 'API 24 (Android 7.0) - Minimum', value: 24 },
156
+ ],
157
+ default: 34,
158
+ },
159
+ {
160
+ type: 'list',
161
+ name: 'abi',
162
+ message: 'Select ABI (architecture):',
163
+ choices: [
164
+ { name: 'x86_64 (Intel/AMD 64-bit)', value: 'x86_64' },
165
+ { name: 'arm64-v8a (ARM 64-bit)', value: 'arm64-v8a' },
166
+ ],
167
+ default: process.arch === 'arm64' ? 'arm64-v8a' : 'x86_64',
168
+ },
169
+ ]);
170
+ try {
171
+ await avdManager.createAVD({
172
+ name: answers.name,
173
+ device: answers.device,
174
+ apiLevel: answers.apiLevel,
175
+ abi: answers.abi,
176
+ });
177
+ }
178
+ catch (err) {
179
+ (0, logger_1.error)(`Failed to create AVD: ${err.message}`);
180
+ }
181
+ }
182
+ /**
183
+ * Delete an AVD
184
+ */
185
+ async function handleDeleteAVD() {
186
+ const avdManager = (0, emulator_1.createAVDManager)();
187
+ const avds = await avdManager.listAVDs();
188
+ if (avds.length === 0) {
189
+ (0, logger_1.warning)('No emulators available to delete.');
190
+ return;
191
+ }
192
+ const { avdName } = await inquirer_1.default.prompt([
193
+ {
194
+ type: 'list',
195
+ name: 'avdName',
196
+ message: 'Select emulator to delete:',
197
+ choices: avds.map((avd) => ({ name: `${avd.name} (${avd.target})`, value: avd.name })),
198
+ },
199
+ ]);
200
+ const { confirm } = await inquirer_1.default.prompt([
201
+ {
202
+ type: 'confirm',
203
+ name: 'confirm',
204
+ message: `Are you sure you want to delete "${avdName}"?`,
205
+ default: false,
206
+ },
207
+ ]);
208
+ if (confirm) {
209
+ await avdManager.deleteAVD(avdName);
210
+ }
211
+ else {
212
+ (0, logger_1.info)('Deletion cancelled');
213
+ }
214
+ }
215
+ /**
216
+ * Android emulator command handler
217
+ */
218
+ async function androidEmulatorCommand() {
219
+ try {
220
+ console.log();
221
+ console.log(chalk_1.default.cyan.bold(' JetStart Android Emulator Manager'));
222
+ console.log();
223
+ let running = true;
224
+ while (running) {
225
+ const { action } = await inquirer_1.default.prompt([
226
+ {
227
+ type: 'list',
228
+ name: 'action',
229
+ message: 'What would you like to do?',
230
+ choices: [
231
+ { name: 'List existing emulators', value: 'list' },
232
+ { name: 'Start emulator', value: 'start' },
233
+ { name: 'Stop emulator', value: 'stop' },
234
+ new inquirer_1.default.Separator(),
235
+ { name: 'Create JetStart-optimized emulator', value: 'create-jetstart' },
236
+ { name: 'Create custom emulator', value: 'create-custom' },
237
+ { name: 'Delete emulator', value: 'delete' },
238
+ new inquirer_1.default.Separator(),
239
+ { name: 'Exit', value: 'exit' },
240
+ ],
241
+ },
242
+ ]);
243
+ switch (action) {
244
+ case 'list':
245
+ await handleListAVDs();
246
+ break;
247
+ case 'start':
248
+ await handleStartEmulator();
249
+ break;
250
+ case 'stop':
251
+ await handleStopEmulator();
252
+ break;
253
+ case 'create-jetstart':
254
+ await handleCreateJetStartAVD();
255
+ break;
256
+ case 'create-custom':
257
+ await handleCreateCustomAVD();
258
+ break;
259
+ case 'delete':
260
+ await handleDeleteAVD();
261
+ break;
262
+ case 'exit':
263
+ running = false;
264
+ console.log();
265
+ (0, logger_1.success)('Goodbye!');
266
+ console.log();
267
+ break;
268
+ }
269
+ if (running && action !== 'list') {
270
+ console.log();
271
+ }
272
+ }
273
+ }
274
+ catch (err) {
275
+ console.log();
276
+ (0, logger_1.error)(`Emulator management failed: ${err.message}`);
277
+ process.exit(1);
278
+ }
279
+ }
280
+ //# sourceMappingURL=android-emulator.js.map
@@ -2,11 +2,6 @@
2
2
  * Create Command
3
3
  * Scaffolds a new JetStart project
4
4
  */
5
- interface CreateOptions {
6
- package?: string;
7
- template?: string;
8
- skipInstall?: boolean;
9
- }
5
+ import { CreateOptions } from '../types';
10
6
  export declare function createCommand(name: string, options: CreateOptions): Promise<void>;
11
- export {};
12
7
  //# sourceMappingURL=create.d.ts.map
@@ -11,11 +11,112 @@ exports.createCommand = createCommand;
11
11
  const path_1 = __importDefault(require("path"));
12
12
  const fs_extra_1 = __importDefault(require("fs-extra"));
13
13
  const chalk_1 = __importDefault(require("chalk"));
14
+ const inquirer_1 = __importDefault(require("inquirer"));
14
15
  const logger_1 = require("../utils/logger");
15
16
  const spinner_1 = require("../utils/spinner");
16
17
  const prompt_1 = require("../utils/prompt");
17
18
  const template_1 = require("../utils/template");
18
19
  const shared_1 = require("@jetstart/shared");
20
+ const java_1 = require("../utils/java");
21
+ const android_sdk_1 = require("../utils/android-sdk");
22
+ const system_tools_1 = require("../utils/system-tools");
23
+ /**
24
+ * Run full installation (automated mode)
25
+ */
26
+ async function runFullInstallation() {
27
+ (0, logger_1.info)('Starting full automated installation...');
28
+ console.log();
29
+ // 1. Check and install Java
30
+ const java = await (0, java_1.detectJava)();
31
+ if (!java || !(await (0, java_1.isJavaCompatible)(java.version))) {
32
+ await (0, java_1.installJava)();
33
+ }
34
+ else {
35
+ (0, logger_1.success)(`Java ${java.version} already installed`);
36
+ }
37
+ // 2. Check and install Android SDK
38
+ const sdkManager = (0, android_sdk_1.createSDKManager)();
39
+ const sdkRoot = await (0, system_tools_1.findAndroidSDK)();
40
+ if (!sdkRoot) {
41
+ (0, logger_1.info)('Installing Android SDK components...');
42
+ await sdkManager.installCmdlineTools();
43
+ }
44
+ else {
45
+ (0, logger_1.success)(`Android SDK found at ${sdkRoot}`);
46
+ }
47
+ // 3. Install required SDK components
48
+ for (const component of android_sdk_1.REQUIRED_SDK_COMPONENTS) {
49
+ await sdkManager.installComponent(component);
50
+ }
51
+ console.log();
52
+ (0, logger_1.success)('All dependencies installed successfully!');
53
+ console.log();
54
+ }
55
+ /**
56
+ * Run interactive installation
57
+ */
58
+ async function runInteractiveInstallation() {
59
+ (0, logger_1.info)('Checking dependencies...');
60
+ console.log();
61
+ // Check Java
62
+ const java = await (0, java_1.detectJava)();
63
+ if (!java || !(await (0, java_1.isJavaCompatible)(java.version))) {
64
+ const { shouldInstall } = await inquirer_1.default.prompt([
65
+ {
66
+ type: 'confirm',
67
+ name: 'shouldInstall',
68
+ message: 'Java 17+ not found. Would you like to install it?',
69
+ default: true,
70
+ },
71
+ ]);
72
+ if (shouldInstall) {
73
+ await (0, java_1.installJava)();
74
+ }
75
+ else {
76
+ (0, logger_1.warning)('Java installation skipped. You may need to install it manually.');
77
+ }
78
+ }
79
+ else {
80
+ (0, logger_1.success)(`Java ${java.version} detected`);
81
+ }
82
+ // Check Android SDK
83
+ const sdkRoot = await (0, system_tools_1.findAndroidSDK)();
84
+ if (!sdkRoot) {
85
+ const { shouldInstall } = await inquirer_1.default.prompt([
86
+ {
87
+ type: 'confirm',
88
+ name: 'shouldInstall',
89
+ message: 'Android SDK not found. Would you like to install it?',
90
+ default: true,
91
+ },
92
+ ]);
93
+ if (shouldInstall) {
94
+ const sdkManager = (0, android_sdk_1.createSDKManager)();
95
+ await sdkManager.installCmdlineTools();
96
+ // Ask about components
97
+ const { installComponents } = await inquirer_1.default.prompt([
98
+ {
99
+ type: 'confirm',
100
+ name: 'installComponents',
101
+ message: 'Install required SDK components?',
102
+ default: true,
103
+ },
104
+ ]);
105
+ if (installComponents) {
106
+ for (const component of android_sdk_1.REQUIRED_SDK_COMPONENTS) {
107
+ await sdkManager.installComponent(component);
108
+ }
109
+ }
110
+ }
111
+ else {
112
+ (0, logger_1.warning)('Android SDK installation skipped. Project creation may fail without SDK.');
113
+ }
114
+ }
115
+ else {
116
+ (0, logger_1.success)(`Android SDK found at ${sdkRoot}`);
117
+ }
118
+ console.log();
119
+ }
19
120
  async function createCommand(name, options) {
20
121
  try {
21
122
  // Validate project name
@@ -32,6 +133,24 @@ async function createCommand(name, options) {
32
133
  }
33
134
  (0, logger_1.log)(`Creating JetStart project: ${chalk_1.default.cyan(name)}`);
34
135
  console.log();
136
+ // Run installation if requested
137
+ if (options.fullInstall) {
138
+ await runFullInstallation();
139
+ }
140
+ else {
141
+ // Interactive mode: ask user
142
+ const { shouldCheckDeps } = await inquirer_1.default.prompt([
143
+ {
144
+ type: 'confirm',
145
+ name: 'shouldCheckDeps',
146
+ message: 'Check and install dependencies?',
147
+ default: true,
148
+ },
149
+ ]);
150
+ if (shouldCheckDeps) {
151
+ await runInteractiveInstallation();
152
+ }
153
+ }
35
154
  // Get package name
36
155
  let packageName = options.package;
37
156
  if (!packageName) {
@@ -2,12 +2,6 @@
2
2
  * Dev Command
3
3
  * Starts the development server
4
4
  */
5
- interface DevOptions {
6
- port?: string;
7
- host?: string;
8
- qr?: boolean;
9
- open?: boolean;
10
- }
5
+ import { DevOptions } from '../types';
11
6
  export declare function devCommand(options: DevOptions): Promise<void>;
12
- export {};
13
7
  //# sourceMappingURL=dev.d.ts.map
@@ -43,11 +43,13 @@ Object.defineProperty(exports, "__esModule", { value: true });
43
43
  exports.devCommand = devCommand;
44
44
  const path = __importStar(require("path"));
45
45
  const os = __importStar(require("os"));
46
+ const fs = __importStar(require("fs-extra"));
46
47
  const chalk_1 = __importDefault(require("chalk"));
47
48
  const qrcode_terminal_1 = __importDefault(require("qrcode-terminal"));
48
49
  const logger_1 = require("../utils/logger");
49
50
  const core_1 = require("@jetstart/core");
50
51
  const shared_1 = require("@jetstart/shared");
52
+ const emulator_deployer_1 = require("../utils/emulator-deployer");
51
53
  async function devCommand(options) {
52
54
  try {
53
55
  const port = parseInt(options.port || String(shared_1.DEFAULT_CORE_PORT));
@@ -59,6 +61,39 @@ async function devCommand(options) {
59
61
  // Get project path and name
60
62
  const projectPath = process.cwd();
61
63
  const projectName = path.basename(projectPath);
64
+ // Setup emulator deployment if requested
65
+ let deployer = null;
66
+ let packageName = null;
67
+ if (options.emulator) {
68
+ try {
69
+ deployer = await emulator_deployer_1.EmulatorDeployer.findOrSelectEmulator(options.avd);
70
+ // Try to read package name from build.gradle (modern Android)
71
+ const buildGradlePath = path.join(projectPath, 'app/build.gradle');
72
+ (0, logger_1.info)(`Looking for build.gradle at: ${buildGradlePath}`);
73
+ if (await fs.pathExists(buildGradlePath)) {
74
+ const buildGradleContent = await fs.readFile(buildGradlePath, 'utf8');
75
+ // Match both applicationId "..." and applicationId '...'
76
+ const packageMatch = buildGradleContent.match(/applicationId\s+["']([^"']+)["']/);
77
+ packageName = packageMatch ? packageMatch[1] : null;
78
+ (0, logger_1.info)(`Package name: ${packageName || 'NOT FOUND'}`);
79
+ }
80
+ else {
81
+ (0, logger_1.warning)('build.gradle not found - emulator deployment requires package name');
82
+ }
83
+ if (packageName) {
84
+ (0, logger_1.success)('Emulator deployment enabled');
85
+ }
86
+ else {
87
+ (0, logger_1.warning)('Emulator deployment disabled: package name not found');
88
+ }
89
+ console.log();
90
+ }
91
+ catch (err) {
92
+ (0, logger_1.error)(`Emulator setup failed: ${err.message}`);
93
+ (0, logger_1.info)('Continuing without emulator deployment...');
94
+ console.log();
95
+ }
96
+ }
62
97
  // Create and start Core server
63
98
  // Bind to 0.0.0.0 to accept connections on all interfaces,
64
99
  // but pass the detected IP for display and client connections
@@ -71,9 +106,38 @@ async function devCommand(options) {
71
106
  projectName,
72
107
  });
73
108
  const session = await server.start();
109
+ // Listen for build completion and deploy to emulator
110
+ if (deployer && packageName) {
111
+ (0, logger_1.info)(`Setting up build:complete listener for package: ${packageName}`);
112
+ let hasDeployed = false; // Track if we've already deployed to prevent reinstall loops
113
+ server.on('build:complete', async (result) => {
114
+ (0, logger_1.info)(`Build complete event received! APK: ${result.apkPath || 'NO APK PATH'}`);
115
+ // Only deploy if this is the first time OR if it's a file-change build (not initial client connection)
116
+ if (!hasDeployed && result.apkPath && deployer) {
117
+ try {
118
+ (0, logger_1.info)('Deploying to emulator (initial deployment)...');
119
+ await deployer.installAPK(result.apkPath, packageName);
120
+ await deployer.launchApp(packageName);
121
+ hasDeployed = true; // Mark as deployed to prevent reinstall loop
122
+ (0, logger_1.success)('Initial deployment complete. Future builds will be sent via hot reload.');
123
+ }
124
+ catch (err) {
125
+ (0, logger_1.warning)(`Emulator deployment failed: ${err.message}`);
126
+ }
127
+ }
128
+ else if (hasDeployed) {
129
+ (0, logger_1.info)('Skipping deployment (app already installed, using hot reload)');
130
+ }
131
+ else {
132
+ (0, logger_1.warning)(`Skipping deployment: apkPath=${result.apkPath}, deployer=${!!deployer}`);
133
+ }
134
+ });
135
+ }
136
+ else {
137
+ (0, logger_1.info)(`Emulator deployment not configured: deployer=${!!deployer}, packageName=${packageName}`);
138
+ }
74
139
  // Get URLs
75
140
  const serverUrl = `http://${host}:${port}`;
76
- const wsUrl = `ws://${host}:${wsPort}`;
77
141
  const localUrl = `http://localhost:${port}`;
78
142
  console.log();
79
143
  (0, logger_1.success)('JetStart dev server is running!');
@@ -84,16 +148,11 @@ async function devCommand(options) {
84
148
  console.log();
85
149
  // Generate QR code for mobile connection
86
150
  if (showQR) {
87
- const qrData = {
88
- serverUrl,
89
- wsUrl,
90
- sessionId: session.id,
91
- token: session.token,
92
- projectName,
93
- version: '0.1.0',
94
- };
151
+ // Ultra-compact format: host|port|wsPort|sessionId|token|projectName
152
+ // Using short alphanumeric IDs for minimal QR code size
153
+ const qrData = `${host}|${port}|${wsPort}|${session.id}|${session.token}|${projectName}`;
95
154
  console.log(chalk_1.default.bold('Scan QR or connect manually:'));
96
- qrcode_terminal_1.default.generate(JSON.stringify(qrData), { small: true });
155
+ qrcode_terminal_1.default.generate(qrData, { small: true });
97
156
  console.log();
98
157
  (0, logger_1.info)(`IP: ${chalk_1.default.cyan(host)}`);
99
158
  (0, logger_1.info)(`Session: ${chalk_1.default.dim(session.id)}`);
@@ -5,4 +5,6 @@ export { createCommand } from './create';
5
5
  export { devCommand } from './dev';
6
6
  export { buildCommand } from './build';
7
7
  export { logsCommand } from './logs';
8
+ export { installAuditCommand } from './install-audit';
9
+ export { androidEmulatorCommand } from './android-emulator';
8
10
  //# sourceMappingURL=index.d.ts.map
@@ -3,7 +3,7 @@
3
3
  * Export all commands
4
4
  */
5
5
  Object.defineProperty(exports, "__esModule", { value: true });
6
- exports.logsCommand = exports.buildCommand = exports.devCommand = exports.createCommand = void 0;
6
+ exports.androidEmulatorCommand = exports.installAuditCommand = exports.logsCommand = exports.buildCommand = exports.devCommand = exports.createCommand = void 0;
7
7
  var create_1 = require("./create");
8
8
  Object.defineProperty(exports, "createCommand", { enumerable: true, get: function () { return create_1.createCommand; } });
9
9
  var dev_1 = require("./dev");
@@ -12,4 +12,8 @@ var build_1 = require("./build");
12
12
  Object.defineProperty(exports, "buildCommand", { enumerable: true, get: function () { return build_1.buildCommand; } });
13
13
  var logs_1 = require("./logs");
14
14
  Object.defineProperty(exports, "logsCommand", { enumerable: true, get: function () { return logs_1.logsCommand; } });
15
+ var install_audit_1 = require("./install-audit");
16
+ Object.defineProperty(exports, "installAuditCommand", { enumerable: true, get: function () { return install_audit_1.installAuditCommand; } });
17
+ var android_emulator_1 = require("./android-emulator");
18
+ Object.defineProperty(exports, "androidEmulatorCommand", { enumerable: true, get: function () { return android_emulator_1.androidEmulatorCommand; } });
15
19
  //# sourceMappingURL=index.js.map
@@ -0,0 +1,9 @@
1
+ /**
2
+ * Installation audit command - checks all required tools and dependencies
3
+ */
4
+ import { AuditOptions } from '../types';
5
+ /**
6
+ * Install audit command handler
7
+ */
8
+ export declare function installAuditCommand(options: AuditOptions): Promise<void>;
9
+ //# sourceMappingURL=install-audit.d.ts.map