orbital-command 0.3.0 → 1.0.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 (160) hide show
  1. package/README.md +67 -42
  2. package/bin/commands/config.js +19 -0
  3. package/bin/commands/events.js +40 -0
  4. package/bin/commands/launch.js +126 -0
  5. package/bin/commands/manifest.js +283 -0
  6. package/bin/commands/registry.js +104 -0
  7. package/bin/commands/update.js +24 -0
  8. package/bin/lib/helpers.js +229 -0
  9. package/bin/orbital.js +95 -870
  10. package/dist/assets/Landing-CfQdHR0N.js +11 -0
  11. package/dist/assets/PrimitivesConfig-DThSipFy.js +32 -0
  12. package/dist/assets/QualityGates-B4kxM5UU.js +26 -0
  13. package/dist/assets/SessionTimeline-Bz1iZnmg.js +1 -0
  14. package/dist/assets/Settings-DLcZwbCT.js +12 -0
  15. package/dist/assets/SourceControl-BMNIz7Lt.js +36 -0
  16. package/dist/assets/WorkflowVisualizer-CxuSBOYu.js +69 -0
  17. package/dist/assets/{arrow-down-CPy85_J6.js → arrow-down-DVPp6_qp.js} +1 -1
  18. package/dist/assets/bot-NFaJBDn_.js +6 -0
  19. package/dist/assets/{charts-DbDg0Psc.js → charts-LGLb8hyU.js} +1 -1
  20. package/dist/assets/{circle-x-Cwz6ZQDV.js → circle-x-IsFCkBZu.js} +1 -1
  21. package/dist/assets/{file-text-C46Xr65c.js → file-text-J1cebZXF.js} +1 -1
  22. package/dist/assets/{globe-Cn2yNZUD.js → globe-WzeyHsUc.js} +1 -1
  23. package/dist/assets/index-BdJ57EhC.css +1 -0
  24. package/dist/assets/index-o4ScMAuR.js +349 -0
  25. package/dist/assets/{key-OPaNTWJ5.js → key-CKR8JJSj.js} +1 -1
  26. package/dist/assets/{minus-GMsbpKym.js → minus-CHBsJyjp.js} +1 -1
  27. package/dist/assets/radio-xqZaR-Uk.js +6 -0
  28. package/dist/assets/rocket-D_xvvNG6.js +6 -0
  29. package/dist/assets/{shield-DwAFkDYI.js → shield-TdB1yv_a.js} +1 -1
  30. package/dist/assets/useSocketListener-0L5yiN5i.js +1 -0
  31. package/dist/assets/useWorkflowEditor-CqeRWVQX.js +11 -0
  32. package/dist/assets/workflow-constants-Rw-GmgHZ.js +6 -0
  33. package/dist/assets/zap-C9wqYMpl.js +6 -0
  34. package/dist/index.html +3 -3
  35. package/dist/server/server/__tests__/data-routes.test.js +2 -0
  36. package/dist/server/server/__tests__/scope-routes.test.js +1 -0
  37. package/dist/server/server/config-migrator.js +0 -3
  38. package/dist/server/server/config.js +35 -6
  39. package/dist/server/server/database.js +0 -22
  40. package/dist/server/server/index.js +26 -814
  41. package/dist/server/server/init.js +32 -399
  42. package/dist/server/server/launch.js +1 -1
  43. package/dist/server/server/parsers/event-parser.js +4 -1
  44. package/dist/server/server/project-context.js +19 -9
  45. package/dist/server/server/project-manager.js +6 -6
  46. package/dist/server/server/routes/aggregate-routes.js +871 -0
  47. package/dist/server/server/routes/config-routes.js +41 -88
  48. package/dist/server/server/routes/data-routes.js +5 -15
  49. package/dist/server/server/routes/dispatch-routes.js +24 -8
  50. package/dist/server/server/routes/manifest-routes.js +1 -1
  51. package/dist/server/server/routes/scope-routes.js +10 -7
  52. package/dist/server/server/schema.js +1 -0
  53. package/dist/server/server/services/batch-orchestrator.js +17 -3
  54. package/dist/server/server/services/config-service.js +10 -1
  55. package/dist/server/server/services/scope-service.js +7 -7
  56. package/dist/server/server/services/sprint-orchestrator.js +24 -11
  57. package/dist/server/server/services/sprint-service.js +2 -2
  58. package/dist/server/server/uninstall.js +195 -0
  59. package/dist/server/server/update.js +212 -0
  60. package/dist/server/server/utils/dispatch-utils.js +8 -6
  61. package/dist/server/server/utils/flag-builder.js +54 -0
  62. package/dist/server/server/utils/json-fields.js +14 -0
  63. package/dist/server/server/utils/json-fields.test.js +73 -0
  64. package/dist/server/server/utils/route-helpers.js +37 -0
  65. package/dist/server/server/utils/route-helpers.test.js +115 -0
  66. package/dist/server/server/watchers/event-watcher.js +28 -13
  67. package/dist/server/server/wizard/config-editor.js +4 -4
  68. package/dist/server/server/wizard/doctor.js +2 -2
  69. package/dist/server/server/wizard/index.js +224 -39
  70. package/dist/server/server/wizard/phases/welcome.js +1 -4
  71. package/dist/server/server/wizard/ui.js +6 -7
  72. package/dist/server/shared/api-types.js +80 -1
  73. package/dist/server/shared/workflow-engine.js +1 -1
  74. package/package.json +20 -20
  75. package/schemas/orbital.config.schema.json +1 -19
  76. package/scripts/postinstall.js +6 -42
  77. package/scripts/release.sh +53 -0
  78. package/server/__tests__/data-routes.test.ts +2 -0
  79. package/server/__tests__/scope-routes.test.ts +1 -0
  80. package/server/config-migrator.ts +0 -3
  81. package/server/config.ts +39 -11
  82. package/server/database.ts +0 -26
  83. package/server/global-config.ts +4 -0
  84. package/server/index.ts +29 -894
  85. package/server/init.ts +32 -443
  86. package/server/launch.ts +1 -1
  87. package/server/parsers/event-parser.ts +4 -1
  88. package/server/project-context.ts +26 -10
  89. package/server/project-manager.ts +5 -6
  90. package/server/routes/aggregate-routes.ts +968 -0
  91. package/server/routes/config-routes.ts +41 -81
  92. package/server/routes/data-routes.ts +7 -16
  93. package/server/routes/dispatch-routes.ts +29 -8
  94. package/server/routes/manifest-routes.ts +1 -1
  95. package/server/routes/scope-routes.ts +12 -7
  96. package/server/schema.ts +1 -0
  97. package/server/services/batch-orchestrator.ts +18 -2
  98. package/server/services/config-service.ts +10 -1
  99. package/server/services/scope-service.ts +6 -6
  100. package/server/services/sprint-orchestrator.ts +24 -9
  101. package/server/services/sprint-service.ts +2 -2
  102. package/server/uninstall.ts +214 -0
  103. package/server/update.ts +263 -0
  104. package/server/utils/dispatch-utils.ts +8 -6
  105. package/server/utils/flag-builder.ts +56 -0
  106. package/server/utils/json-fields.test.ts +83 -0
  107. package/server/utils/json-fields.ts +14 -0
  108. package/server/utils/route-helpers.test.ts +144 -0
  109. package/server/utils/route-helpers.ts +38 -0
  110. package/server/watchers/event-watcher.ts +24 -12
  111. package/server/wizard/config-editor.ts +4 -4
  112. package/server/wizard/doctor.ts +2 -2
  113. package/server/wizard/index.ts +291 -40
  114. package/server/wizard/phases/welcome.ts +1 -5
  115. package/server/wizard/ui.ts +6 -7
  116. package/shared/api-types.ts +106 -0
  117. package/shared/workflow-engine.ts +1 -1
  118. package/templates/agents/QUICK-REFERENCE.md +1 -0
  119. package/templates/agents/README.md +1 -0
  120. package/templates/agents/SKILL-TRIGGERS.md +11 -0
  121. package/templates/agents/green-team/deep-dive.md +361 -0
  122. package/templates/hooks/end-session.sh +1 -0
  123. package/templates/hooks/init-session.sh +1 -0
  124. package/templates/hooks/scope-commit-logger.sh +2 -2
  125. package/templates/hooks/scope-create-gate.sh +2 -4
  126. package/templates/hooks/scope-gate.sh +4 -6
  127. package/templates/hooks/scope-helpers.sh +10 -1
  128. package/templates/hooks/scope-lifecycle-gate.sh +14 -5
  129. package/templates/hooks/scope-prepare.sh +1 -1
  130. package/templates/hooks/scope-transition.sh +14 -6
  131. package/templates/hooks/time-tracker.sh +2 -5
  132. package/templates/orbital.config.json +1 -4
  133. package/templates/presets/development.json +4 -4
  134. package/templates/presets/gitflow.json +7 -0
  135. package/templates/prompts/README.md +23 -0
  136. package/templates/prompts/deep-dive-audit.md +94 -0
  137. package/templates/quick/rules.md +56 -5
  138. package/templates/skills/git-commit/SKILL.md +21 -6
  139. package/templates/skills/git-dev/SKILL.md +8 -4
  140. package/templates/skills/git-main/SKILL.md +8 -4
  141. package/templates/skills/git-production/SKILL.md +6 -3
  142. package/templates/skills/git-staging/SKILL.md +6 -3
  143. package/templates/skills/scope-fix-review/SKILL.md +8 -4
  144. package/templates/skills/scope-implement/SKILL.md +13 -5
  145. package/templates/skills/scope-post-review/SKILL.md +16 -4
  146. package/templates/skills/scope-pre-review/SKILL.md +6 -2
  147. package/dist/assets/PrimitivesConfig-CrmQXYh4.js +0 -32
  148. package/dist/assets/QualityGates-BbasOsF3.js +0 -21
  149. package/dist/assets/SessionTimeline-CGeJsVvy.js +0 -1
  150. package/dist/assets/Settings-oiM496mc.js +0 -12
  151. package/dist/assets/SourceControl-B1fP2nJL.js +0 -41
  152. package/dist/assets/WorkflowVisualizer-CWLYf-f0.js +0 -74
  153. package/dist/assets/formatDistanceToNow-BMqsSP44.js +0 -1
  154. package/dist/assets/index-Aj4sV8Al.css +0 -1
  155. package/dist/assets/index-Bc9dK3MW.js +0 -354
  156. package/dist/assets/useWorkflowEditor-BJkTX_NR.js +0 -16
  157. package/dist/assets/zap-DfbUoOty.js +0 -11
  158. package/dist/server/server/services/telemetry-service.js +0 -143
  159. package/server/services/telemetry-service.ts +0 -195
  160. /package/{shared/default-workflow.json → templates/presets/default.json} +0 -0
@@ -0,0 +1,104 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import crypto from 'crypto';
4
+ import {
5
+ ORBITAL_HOME,
6
+ detectProjectRoot,
7
+ loadRegistry,
8
+ writeRegistryAtomic,
9
+ } from '../lib/helpers.js';
10
+
11
+ export function cmdRegister(args) {
12
+ const targetPath = args[0] ? path.resolve(args[0]) : detectProjectRoot();
13
+ const nameFlag = args.indexOf('--alias');
14
+ const name = nameFlag >= 0 ? args[nameFlag + 1] : path.basename(targetPath);
15
+
16
+ if (!fs.existsSync(ORBITAL_HOME)) fs.mkdirSync(ORBITAL_HOME, { recursive: true });
17
+
18
+ if (!fs.existsSync(path.join(targetPath, '.claude'))) {
19
+ console.error(`Error: ${targetPath} has not been initialized with Orbital Command.`);
20
+ console.error(`Run \`orbital\` in that directory first.`);
21
+ process.exit(1);
22
+ }
23
+
24
+ const registry = loadRegistry();
25
+
26
+ if (registry.projects?.some(p => p.path === targetPath)) {
27
+ console.log(`Project already registered: ${targetPath}`);
28
+ return;
29
+ }
30
+
31
+ const COLORS = [
32
+ '210 80% 55%', '340 75% 55%', '160 60% 45%', '30 90% 55%',
33
+ '270 65% 55%', '50 85% 50%', '180 55% 45%', '0 70% 55%',
34
+ '120 50% 42%', '300 60% 50%', '200 70% 50%', '15 80% 55%',
35
+ ];
36
+ const usedColors = (registry.projects || []).map(p => p.color);
37
+ const color = COLORS.find(c => !usedColors.includes(c)) || COLORS[0];
38
+
39
+ const baseSlug = path.basename(targetPath).toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/-+/g, '-').replace(/^-|-$/g, '') || 'project';
40
+ const existingIds = (registry.projects || []).map(p => p.id);
41
+ const slug = existingIds.includes(baseSlug)
42
+ ? `${baseSlug}-${crypto.createHash('sha256').update(targetPath).digest('hex').slice(0, 4)}`
43
+ : baseSlug;
44
+
45
+ const project = {
46
+ id: slug,
47
+ path: targetPath,
48
+ name,
49
+ color,
50
+ registeredAt: new Date().toISOString(),
51
+ enabled: true,
52
+ };
53
+
54
+ if (!registry.projects) registry.projects = [];
55
+ registry.projects.push(project);
56
+ writeRegistryAtomic(registry);
57
+
58
+ console.log(`Registered project: ${name}`);
59
+ console.log(` ID: ${slug}`);
60
+ console.log(` Path: ${targetPath}`);
61
+ console.log(` Color: ${color}`);
62
+ }
63
+
64
+ export function cmdUnregister(args) {
65
+ const idOrPath = args[0];
66
+ if (!idOrPath) {
67
+ console.error('Usage: orbital unregister <id-or-path>');
68
+ process.exit(1);
69
+ }
70
+
71
+ const absPath = path.isAbsolute(idOrPath) ? idOrPath : path.resolve(idOrPath);
72
+ const registry = loadRegistry();
73
+ const idx = (registry.projects || []).findIndex(p => p.id === idOrPath || p.path === absPath);
74
+
75
+ if (idx === -1) {
76
+ console.error(`Project not found: ${idOrPath}`);
77
+ process.exit(1);
78
+ }
79
+
80
+ const removed = registry.projects.splice(idx, 1)[0];
81
+ writeRegistryAtomic(registry);
82
+
83
+ console.log(`Unregistered project: ${removed.name} (${removed.id})`);
84
+ console.log(` Project files in ${removed.path} are preserved.`);
85
+ }
86
+
87
+ export function cmdProjects() {
88
+ const registry = loadRegistry();
89
+ const projects = registry.projects || [];
90
+
91
+ if (projects.length === 0) {
92
+ console.log('\nNo projects registered.');
93
+ console.log('Run `orbital` in a project directory to get started.\n');
94
+ return;
95
+ }
96
+
97
+ console.log(`\n ${'ID'.padEnd(22)} ${'NAME'.padEnd(22)} ${'STATUS'.padEnd(10)} PATH`);
98
+ console.log(` ${'─'.repeat(22)} ${'─'.repeat(22)} ${'─'.repeat(10)} ${'─'.repeat(30)}`);
99
+ for (const p of projects) {
100
+ const status = p.enabled ? (fs.existsSync(p.path) ? 'active' : 'offline') : 'disabled';
101
+ console.log(` ${p.id.padEnd(22)} ${p.name.padEnd(22)} ${status.padEnd(10)} ${p.path}`);
102
+ }
103
+ console.log();
104
+ }
@@ -0,0 +1,24 @@
1
+ import {
2
+ detectProjectRoot,
3
+ stampTemplateVersion,
4
+ loadSharedModule,
5
+ } from '../lib/helpers.js';
6
+
7
+ export async function cmdUpdate(args) {
8
+ const projectRoot = detectProjectRoot();
9
+ const dryRun = args.includes('--dry-run');
10
+
11
+ const { runUpdate } = await loadSharedModule();
12
+ runUpdate(projectRoot, { dryRun });
13
+
14
+ if (!dryRun) stampTemplateVersion(projectRoot);
15
+ }
16
+
17
+ export async function cmdUninstall(args) {
18
+ const projectRoot = detectProjectRoot();
19
+ const dryRun = args.includes('--dry-run');
20
+ const keepConfig = args.includes('--keep-config');
21
+
22
+ const { runUninstall } = await loadSharedModule();
23
+ runUninstall(projectRoot, { dryRun, keepConfig });
24
+ }
@@ -0,0 +1,229 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import { execFileSync, spawn } from 'child_process';
4
+ import { fileURLToPath } from 'url';
5
+
6
+ // ---------------------------------------------------------------------------
7
+ // Paths
8
+ // ---------------------------------------------------------------------------
9
+ const __filename = fileURLToPath(import.meta.url);
10
+ const __dirname = path.dirname(__filename);
11
+ export const PACKAGE_ROOT = path.resolve(__dirname, '..', '..');
12
+
13
+ // ---------------------------------------------------------------------------
14
+ // Multi-project registry paths
15
+ // ---------------------------------------------------------------------------
16
+ export const ORBITAL_HOME = path.join(process.env.HOME || process.env.USERPROFILE || '~', '.orbital');
17
+ export const REGISTRY_PATH = path.join(ORBITAL_HOME, 'config.json');
18
+
19
+ // ---------------------------------------------------------------------------
20
+ // CLI Helpers
21
+ // ---------------------------------------------------------------------------
22
+
23
+ export function getFlagValue(args, flag) {
24
+ const idx = args.indexOf(flag);
25
+ return idx !== -1 && idx + 1 < args.length ? args[idx + 1] : undefined;
26
+ }
27
+
28
+ /**
29
+ * Resolve a package binary (e.g. 'tsx', 'vite') to an absolute path.
30
+ * Checks PACKAGE_ROOT/node_modules/.bin first (global installs, non-hoisted),
31
+ * then the parent node_modules/.bin (hoisted local installs where deps are
32
+ * lifted to <project>/node_modules/.bin/). Returns null to fall back to npx.
33
+ */
34
+ export function resolveBin(name) {
35
+ const local = path.join(PACKAGE_ROOT, 'node_modules', '.bin', name);
36
+ if (fs.existsSync(local)) return local;
37
+ const hoisted = path.join(PACKAGE_ROOT, '..', '.bin', name);
38
+ if (fs.existsSync(hoisted)) return path.resolve(hoisted);
39
+ return null;
40
+ }
41
+
42
+ export function isGitRepo() {
43
+ try {
44
+ execFileSync('git', ['rev-parse', '--show-toplevel'], { encoding: 'utf8', stdio: 'pipe' });
45
+ return true;
46
+ } catch {
47
+ return false;
48
+ }
49
+ }
50
+
51
+ export function detectProjectRoot() {
52
+ try {
53
+ return execFileSync('git', ['rev-parse', '--show-toplevel'], { encoding: 'utf8', stdio: 'pipe' }).trim();
54
+ } catch {
55
+ return process.cwd();
56
+ }
57
+ }
58
+
59
+ export function requireGitRepo() {
60
+ if (!isGitRepo()) {
61
+ console.error('Not a git repository. Run `orbital` from inside a project directory.');
62
+ process.exit(1);
63
+ }
64
+ }
65
+
66
+ export function loadConfig(projectRoot) {
67
+ const configPath = path.join(projectRoot, '.claude', 'orbital.config.json');
68
+ if (fs.existsSync(configPath)) {
69
+ try {
70
+ return JSON.parse(fs.readFileSync(configPath, 'utf8'));
71
+ } catch (err) {
72
+ console.warn(`Warning: could not parse ${configPath}: ${err.message}`);
73
+ }
74
+ }
75
+ return { serverPort: 4444, clientPort: 4445 };
76
+ }
77
+
78
+ export function getPackageVersion() {
79
+ try {
80
+ const pkg = JSON.parse(fs.readFileSync(path.join(PACKAGE_ROOT, 'package.json'), 'utf8'));
81
+ return pkg.version || '0.0.0';
82
+ } catch {
83
+ return '0.0.0';
84
+ }
85
+ }
86
+
87
+ export function stampTemplateVersion(projectRoot) {
88
+ const configPath = path.join(projectRoot, '.claude', 'orbital.config.json');
89
+ if (!fs.existsSync(configPath)) return;
90
+
91
+ try {
92
+ const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));
93
+ const version = getPackageVersion();
94
+ if (config.templateVersion !== version) {
95
+ config.templateVersion = version;
96
+ fs.writeFileSync(configPath, JSON.stringify(config, null, 2) + '\n', 'utf8');
97
+ console.log(` Stamped templateVersion: ${version}`);
98
+ }
99
+ } catch { /* ignore malformed config */ }
100
+ }
101
+
102
+ export function openBrowser(url) {
103
+ const platform = process.platform;
104
+ if (platform === 'darwin') {
105
+ spawn('open', [url], { detached: true, stdio: 'ignore' }).unref();
106
+ } else if (platform === 'win32') {
107
+ spawn('cmd', ['/c', 'start', url], { detached: true, stdio: 'ignore' }).unref();
108
+ } else {
109
+ spawn('xdg-open', [url], { detached: true, stdio: 'ignore' }).unref();
110
+ }
111
+ }
112
+
113
+ // ---------------------------------------------------------------------------
114
+ // Shared Module Loaders
115
+ // ---------------------------------------------------------------------------
116
+
117
+ /**
118
+ * Load the shared init module. Tries compiled JS first (production/global
119
+ * installs via npm publish), then TypeScript source (dev via tsx).
120
+ */
121
+ export async function loadSharedModule() {
122
+ try {
123
+ return await import('../../dist/server/server/init.js');
124
+ } catch {
125
+ try {
126
+ return await import('../../server/init.js');
127
+ } catch {
128
+ console.error('Error: Orbital Command server module not found.');
129
+ console.error('Try reinstalling: npm install -g orbital-command');
130
+ console.error('For local development: npm run build:server');
131
+ process.exit(1);
132
+ }
133
+ }
134
+ }
135
+
136
+ /**
137
+ * Load the interactive wizard module.
138
+ */
139
+ export async function loadWizardModule() {
140
+ try {
141
+ return await import('../../dist/server/server/wizard/index.js');
142
+ } catch {
143
+ try {
144
+ return await import('../../server/wizard/index.js');
145
+ } catch {
146
+ console.error('Error: Wizard module not found.');
147
+ console.error('Try reinstalling: npm install -g orbital-command');
148
+ process.exit(1);
149
+ }
150
+ }
151
+ }
152
+
153
+ // ---------------------------------------------------------------------------
154
+ // Registry
155
+ // ---------------------------------------------------------------------------
156
+
157
+ export function loadRegistry() {
158
+ if (!fs.existsSync(REGISTRY_PATH)) return { version: 1, projects: [] };
159
+ try {
160
+ return JSON.parse(fs.readFileSync(REGISTRY_PATH, 'utf8'));
161
+ } catch {
162
+ return { version: 1, projects: [] };
163
+ }
164
+ }
165
+
166
+ export function writeRegistryAtomic(registry) {
167
+ if (!fs.existsSync(ORBITAL_HOME)) fs.mkdirSync(ORBITAL_HOME, { recursive: true });
168
+ const tmp = REGISTRY_PATH + `.tmp.${process.pid}`;
169
+ try {
170
+ fs.writeFileSync(tmp, JSON.stringify(registry, null, 2), 'utf8');
171
+ fs.renameSync(tmp, REGISTRY_PATH);
172
+ } catch (err) {
173
+ try { fs.unlinkSync(tmp); } catch { /* best-effort */ }
174
+ throw err;
175
+ }
176
+ }
177
+
178
+ export function orbitalSetupDone() {
179
+ return fs.existsSync(path.join(ORBITAL_HOME, 'config.json'));
180
+ }
181
+
182
+ export function printHelp() {
183
+ console.log(`
184
+ Orbital Command — mission control for Claude Code projects
185
+
186
+ Usage:
187
+ orbital Context-aware hub (setup, launch, config, etc.)
188
+ orbital <command> Run a specific command directly
189
+
190
+ Commands:
191
+ config Modify project settings interactively
192
+ doctor Health check and version diagnostics
193
+ update Sync templates and apply migrations
194
+ status Show template sync status
195
+
196
+ Template Management:
197
+ validate Check cross-references and consistency
198
+ pin <path> Lock a file from updates
199
+ unpin <path> Unlock a pinned file
200
+ pins List all pinned files
201
+ diff <path> Show diff between template and local file
202
+ reset <path> Restore a file from the current template
203
+
204
+ Project Management:
205
+ register [path] Register a project with the dashboard
206
+ unregister <id> Remove a project from the dashboard
207
+ projects List all registered projects
208
+
209
+ Development:
210
+ dev Start with Vite dev server (HMR)
211
+ build Production build of the dashboard frontend
212
+
213
+ Other:
214
+ emit <TYPE> <JSON> Emit an orbital event
215
+ uninstall Remove Orbital artifacts from the project
216
+
217
+ Update Options:
218
+ --dry-run Preview changes without applying them
219
+
220
+ Uninstall Options:
221
+ --dry-run Preview removal without applying
222
+ --keep-config Keep orbital.config.json for re-initialization
223
+
224
+ Examples:
225
+ orbital # hub menu — setup, launch, config, etc.
226
+ orbital config # modify project settings directly
227
+ orbital update # sync templates to latest version
228
+ `);
229
+ }