Compare commits
7 Commits
feat/insta
...
mosaic-v0.
| Author | SHA1 | Date | |
|---|---|---|---|
| 1bfd8570d6 | |||
| 312acd8bad | |||
| d08b969918 | |||
| 051de0d8a9 | |||
| bd76df1a50 | |||
| 62b2ce2da1 | |||
| 172bacb30f |
@@ -103,12 +103,12 @@ steps:
|
|||||||
- mkdir -p /kaniko/.docker
|
- mkdir -p /kaniko/.docker
|
||||||
- echo "{\"auths\":{\"git.mosaicstack.dev\":{\"username\":\"$REGISTRY_USER\",\"password\":\"$REGISTRY_PASS\"}}}" > /kaniko/.docker/config.json
|
- echo "{\"auths\":{\"git.mosaicstack.dev\":{\"username\":\"$REGISTRY_USER\",\"password\":\"$REGISTRY_PASS\"}}}" > /kaniko/.docker/config.json
|
||||||
- |
|
- |
|
||||||
DESTINATIONS="--destination git.mosaicstack.dev/mosaicstack/mosaic-stack/gateway:sha-${CI_COMMIT_SHA:0:7}"
|
DESTINATIONS="--destination git.mosaicstack.dev/mosaicstack/stack/gateway:sha-${CI_COMMIT_SHA:0:7}"
|
||||||
if [ "$CI_COMMIT_BRANCH" = "main" ]; then
|
if [ "$CI_COMMIT_BRANCH" = "main" ]; then
|
||||||
DESTINATIONS="$DESTINATIONS --destination git.mosaicstack.dev/mosaicstack/mosaic-stack/gateway:latest"
|
DESTINATIONS="$DESTINATIONS --destination git.mosaicstack.dev/mosaicstack/stack/gateway:latest"
|
||||||
fi
|
fi
|
||||||
if [ -n "$CI_COMMIT_TAG" ]; then
|
if [ -n "$CI_COMMIT_TAG" ]; then
|
||||||
DESTINATIONS="$DESTINATIONS --destination git.mosaicstack.dev/mosaicstack/mosaic-stack/gateway:$CI_COMMIT_TAG"
|
DESTINATIONS="$DESTINATIONS --destination git.mosaicstack.dev/mosaicstack/stack/gateway:$CI_COMMIT_TAG"
|
||||||
fi
|
fi
|
||||||
/kaniko/executor --context . --dockerfile docker/gateway.Dockerfile $DESTINATIONS
|
/kaniko/executor --context . --dockerfile docker/gateway.Dockerfile $DESTINATIONS
|
||||||
depends_on:
|
depends_on:
|
||||||
@@ -128,12 +128,12 @@ steps:
|
|||||||
- mkdir -p /kaniko/.docker
|
- mkdir -p /kaniko/.docker
|
||||||
- echo "{\"auths\":{\"git.mosaicstack.dev\":{\"username\":\"$REGISTRY_USER\",\"password\":\"$REGISTRY_PASS\"}}}" > /kaniko/.docker/config.json
|
- echo "{\"auths\":{\"git.mosaicstack.dev\":{\"username\":\"$REGISTRY_USER\",\"password\":\"$REGISTRY_PASS\"}}}" > /kaniko/.docker/config.json
|
||||||
- |
|
- |
|
||||||
DESTINATIONS="--destination git.mosaicstack.dev/mosaicstack/mosaic-stack/web:sha-${CI_COMMIT_SHA:0:7}"
|
DESTINATIONS="--destination git.mosaicstack.dev/mosaicstack/stack/web:sha-${CI_COMMIT_SHA:0:7}"
|
||||||
if [ "$CI_COMMIT_BRANCH" = "main" ]; then
|
if [ "$CI_COMMIT_BRANCH" = "main" ]; then
|
||||||
DESTINATIONS="$DESTINATIONS --destination git.mosaicstack.dev/mosaicstack/mosaic-stack/web:latest"
|
DESTINATIONS="$DESTINATIONS --destination git.mosaicstack.dev/mosaicstack/stack/web:latest"
|
||||||
fi
|
fi
|
||||||
if [ -n "$CI_COMMIT_TAG" ]; then
|
if [ -n "$CI_COMMIT_TAG" ]; then
|
||||||
DESTINATIONS="$DESTINATIONS --destination git.mosaicstack.dev/mosaicstack/mosaic-stack/web:$CI_COMMIT_TAG"
|
DESTINATIONS="$DESTINATIONS --destination git.mosaicstack.dev/mosaicstack/stack/web:$CI_COMMIT_TAG"
|
||||||
fi
|
fi
|
||||||
/kaniko/executor --context . --dockerfile docker/web.Dockerfile $DESTINATIONS
|
/kaniko/executor --context . --dockerfile docker/web.Dockerfile $DESTINATIONS
|
||||||
depends_on:
|
depends_on:
|
||||||
|
|||||||
22
README.md
22
README.md
@@ -7,7 +7,13 @@ Mosaic gives you a unified launcher for Claude Code, Codex, OpenCode, and Pi —
|
|||||||
## Quick Install
|
## Quick Install
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
bash <(curl -fsSL https://git.mosaicstack.dev/mosaicstack/mosaic-stack/raw/branch/main/tools/install.sh)
|
curl -fsSL https://mosaicstack.dev/install.sh | bash
|
||||||
|
```
|
||||||
|
|
||||||
|
Or use the direct URL:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
bash <(curl -fsSL https://git.mosaicstack.dev/mosaicstack/stack/raw/branch/main/tools/install.sh)
|
||||||
```
|
```
|
||||||
|
|
||||||
The installer auto-launches the setup wizard, which walks you through gateway install and verification. Flags for non-interactive use:
|
The installer auto-launches the setup wizard, which walks you through gateway install and verification. Flags for non-interactive use:
|
||||||
@@ -179,8 +185,8 @@ Consent state is persisted in config. Remote upload is a no-op until you run `mo
|
|||||||
### Setup
|
### Setup
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone git@git.mosaicstack.dev:mosaicstack/mosaic-stack.git
|
git clone git@git.mosaicstack.dev:mosaicstack/stack.git
|
||||||
cd mosaic-stack
|
cd stack
|
||||||
|
|
||||||
# Start infrastructure (Postgres, Valkey, Jaeger)
|
# Start infrastructure (Postgres, Valkey, Jaeger)
|
||||||
docker compose up -d
|
docker compose up -d
|
||||||
@@ -229,7 +235,7 @@ npm packages are published to the Gitea package registry on main merges.
|
|||||||
## Architecture
|
## Architecture
|
||||||
|
|
||||||
```
|
```
|
||||||
mosaic-stack/
|
stack/
|
||||||
├── apps/
|
├── apps/
|
||||||
│ ├── gateway/ NestJS API + WebSocket hub (Fastify, Socket.IO, OTEL)
|
│ ├── gateway/ NestJS API + WebSocket hub (Fastify, Socket.IO, OTEL)
|
||||||
│ └── web/ Next.js dashboard (React 19, Tailwind)
|
│ └── web/ Next.js dashboard (React 19, Tailwind)
|
||||||
@@ -302,7 +308,13 @@ Each stage has a dispatch mode (`exec` for research/review, `yolo` for coding),
|
|||||||
Run the installer again — it handles upgrades automatically:
|
Run the installer again — it handles upgrades automatically:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
bash <(curl -fsSL https://git.mosaicstack.dev/mosaicstack/mosaic-stack/raw/branch/main/tools/install.sh)
|
curl -fsSL https://mosaicstack.dev/install.sh | bash
|
||||||
|
```
|
||||||
|
|
||||||
|
Or use the direct URL:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
bash <(curl -fsSL https://git.mosaicstack.dev/mosaicstack/stack/raw/branch/main/tools/install.sh)
|
||||||
```
|
```
|
||||||
|
|
||||||
Or use the CLI:
|
Or use the CLI:
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.6",
|
"version": "0.0.6",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "apps/gateway"
|
"directory": "apps/gateway"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -8,10 +8,10 @@
|
|||||||
**ID:** install-ux-v2-20260405
|
**ID:** install-ux-v2-20260405
|
||||||
**Statement:** The install-ux-hardening mission shipped the plumbing (uninstall, masked password, hooks consent, unified flow, headless path), but the first real end-to-end run surfaced a critical regression and a collection of UX failings that make the wizard feel neither quick nor intelligent. This mission closes the bootstrap regression as a hotfix, then rethinks the first-run experience around a provider-first, intent-driven flow with a drill-down main menu and a genuinely fast quick-start.
|
**Statement:** The install-ux-hardening mission shipped the plumbing (uninstall, masked password, hooks consent, unified flow, headless path), but the first real end-to-end run surfaced a critical regression and a collection of UX failings that make the wizard feel neither quick nor intelligent. This mission closes the bootstrap regression as a hotfix, then rethinks the first-run experience around a provider-first, intent-driven flow with a drill-down main menu and a genuinely fast quick-start.
|
||||||
**Phase:** Execution
|
**Phase:** Execution
|
||||||
**Current Milestone:** IUV-M02
|
**Current Milestone:** IUV-M03
|
||||||
**Progress:** 1 / 3 milestones
|
**Progress:** 2 / 3 milestones
|
||||||
**Status:** active
|
**Status:** active
|
||||||
**Last Updated:** 2026-04-05 (IUV-M01 complete — mosaic-v0.0.26 released)
|
**Last Updated:** 2026-04-05 (IUV-M02 complete — CORS/FQDN + skill installer rework)
|
||||||
**Parent Mission:** [install-ux-hardening-20260405](./archive/missions/install-ux-hardening-20260405/MISSION-MANIFEST.md) (complete — `mosaic-v0.0.25`)
|
**Parent Mission:** [install-ux-hardening-20260405](./archive/missions/install-ux-hardening-20260405/MISSION-MANIFEST.md) (complete — `mosaic-v0.0.25`)
|
||||||
|
|
||||||
## Context
|
## Context
|
||||||
@@ -47,7 +47,7 @@ Real-run testing of `@mosaicstack/mosaic@0.0.25` uncovered:
|
|||||||
| # | ID | Name | Status | Branch | Issue | Started | Completed |
|
| # | ID | Name | Status | Branch | Issue | Started | Completed |
|
||||||
| --- | ------- | ------------------------------------------------------------ | ----------- | ---------------------- | ----- | ---------- | ---------- |
|
| --- | ------- | ------------------------------------------------------------ | ----------- | ---------------------- | ----- | ---------- | ---------- |
|
||||||
| 1 | IUV-M01 | Hotfix: bootstrap DTO + wizard failure + port prefill + copy | complete | fix/bootstrap-hotfix | #436 | 2026-04-05 | 2026-04-05 |
|
| 1 | IUV-M01 | Hotfix: bootstrap DTO + wizard failure + port prefill + copy | complete | fix/bootstrap-hotfix | #436 | 2026-04-05 | 2026-04-05 |
|
||||||
| 2 | IUV-M02 | UX polish: CORS/FQDN, skill installer rework | not-started | feat/install-ux-polish | #437 | — | — |
|
| 2 | IUV-M02 | UX polish: CORS/FQDN, skill installer rework | complete | feat/install-ux-polish | #437 | 2026-04-05 | 2026-04-05 |
|
||||||
| 3 | IUV-M03 | Provider-first intelligent flow + drill-down main menu | not-started | feat/install-ux-intent | #438 | — | — |
|
| 3 | IUV-M03 | Provider-first intelligent flow + drill-down main menu | not-started | feat/install-ux-intent | #438 | — | — |
|
||||||
|
|
||||||
## Subagent Delegation Plan
|
## Subagent Delegation Plan
|
||||||
|
|||||||
@@ -20,18 +20,18 @@
|
|||||||
|
|
||||||
## Milestone 2 — UX polish: CORS/FQDN, skill installer rework (IUV-M02)
|
## Milestone 2 — UX polish: CORS/FQDN, skill installer rework (IUV-M02)
|
||||||
|
|
||||||
| id | status | description | issue | agent | branch | depends_on | estimate | notes |
|
| id | status | description | issue | agent | branch | depends_on | estimate | notes |
|
||||||
| --------- | ----------- | ------------------------------------------------------------------------------------------------------------------------------------ | ----- | ------ | ---------------------- | ---------- | -------- | --------------------------- |
|
| --------- | ------ | ------------------------------------------------------------------------------------------------------------------------------------ | ----- | ------ | ---------------------- | ---------- | -------- | ---------------------------------------------------------------------- |
|
||||||
| IUV-02-01 | not-started | Replace CORS origin prompt with FQDN / hostname input; derive the CORS value internally; default to `localhost` with clear help text | #437 | sonnet | feat/install-ux-polish | — | 10K | |
|
| IUV-02-01 | done | Replace CORS origin prompt with FQDN / hostname input; derive the CORS value internally; default to `localhost` with clear help text | #437 | sonnet | feat/install-ux-polish | — | 10K | `deriveCorsOrigin()` pure fn; MOSAIC_HOSTNAME headless var; PR #444 |
|
||||||
| IUV-02-02 | not-started | Diagnose and document the concrete failure modes of the current skill / additional feature install section end-to-end | #437 | sonnet | feat/install-ux-polish | IUV-02-01 | 8K | needs real-run reproduction |
|
| IUV-02-02 | done | Diagnose and document the concrete failure modes of the current skill / additional feature install section end-to-end | #437 | sonnet | feat/install-ux-polish | IUV-02-01 | 8K | selection→install gap, silent catch{}, no whitelist concept |
|
||||||
| IUV-02-03 | not-started | Rework the skill installer so it is usable end-to-end (selection, install, verify, failure reporting) | #437 | sonnet | feat/install-ux-polish | IUV-02-02 | 20K | |
|
| IUV-02-03 | done | Rework the skill installer so it is usable end-to-end (selection, install, verify, failure reporting) | #437 | sonnet | feat/install-ux-polish | IUV-02-02 | 20K | MOSAIC_INSTALL_SKILLS env var whitelist; SyncSkillsResult typed return |
|
||||||
| IUV-02-04 | not-started | Tests + code review + PR merge | #437 | sonnet | feat/install-ux-polish | IUV-02-03 | 10K | |
|
| IUV-02-04 | done | Tests + code review + PR merge | #437 | sonnet | feat/install-ux-polish | IUV-02-03 | 10K | 18 new tests (13 CORS + 5 skills); PR #444 merged `172bacb3` |
|
||||||
|
|
||||||
## Milestone 3 — Provider-first intelligent flow + drill-down main menu (IUV-M03)
|
## Milestone 3 — Provider-first intelligent flow + drill-down main menu (IUV-M03)
|
||||||
|
|
||||||
| id | status | description | issue | agent | branch | depends_on | estimate | notes |
|
| id | status | description | issue | agent | branch | depends_on | estimate | notes |
|
||||||
| --------- | ----------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------ | ----- | ----- | ---------------------- | ---------- | -------- | ------------------------------------------------------------- |
|
| --------- | ----------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------ | ----- | ----- | ---------------------- | ---------- | -------- | ------------------------------------------------------------- |
|
||||||
| IUV-03-01 | not-started | Design doc: new first-run state machine — main menu (Plugins / Providers / …), Quick Start vs Custom paths, provider-first flow, intent intake + naming loop | #438 | opus | feat/install-ux-intent | IUV-02-04 | 15K | scratchpad + explicit non-goals |
|
| IUV-03-01 | not-started | Design doc: new first-run state machine — main menu (Plugins / Providers / …), Quick Start vs Custom paths, provider-first flow, intent intake + naming loop | #438 | opus | feat/install-ux-intent | — | 15K | scratchpad + explicit non-goals |
|
||||||
| IUV-03-02 | not-started | Implement drill-down main menu (Plugins: Recommended / Custom, Providers, …) as the top-level entry point of `mosaic wizard` | #438 | opus | feat/install-ux-intent | IUV-03-01 | 25K | |
|
| IUV-03-02 | not-started | Implement drill-down main menu (Plugins: Recommended / Custom, Providers, …) as the top-level entry point of `mosaic wizard` | #438 | opus | feat/install-ux-intent | IUV-03-01 | 25K | |
|
||||||
| IUV-03-03 | not-started | Quick Start path: curated minimum question set — define the exact baseline, delete everything else from the fast path | #438 | opus | feat/install-ux-intent | IUV-03-02 | 15K | |
|
| IUV-03-03 | not-started | Quick Start path: curated minimum question set — define the exact baseline, delete everything else from the fast path | #438 | opus | feat/install-ux-intent | IUV-03-02 | 15K | |
|
||||||
| IUV-03-04 | not-started | Provider-first natural-language intake: user describes intent → agent expounds → agent proposes a name (confirmable / overridable) — OpenClaw-style | #438 | opus | feat/install-ux-intent | IUV-03-03 | 25K | offline fallback required (deterministic default name + path) |
|
| IUV-03-04 | not-started | Provider-first natural-language intake: user describes intent → agent expounds → agent proposes a name (confirmable / overridable) — OpenClaw-style | #438 | opus | feat/install-ux-intent | IUV-03-03 | 25K | offline fallback required (deterministic default name + path) |
|
||||||
|
|||||||
@@ -165,7 +165,13 @@ The `mosaic` CLI provides a terminal interface to the same gateway API.
|
|||||||
Install via the Mosaic installer:
|
Install via the Mosaic installer:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
bash <(curl -fsSL https://git.mosaicstack.dev/mosaicstack/mosaic-stack/raw/branch/main/tools/install.sh)
|
curl -fsSL https://mosaicstack.dev/install.sh | bash
|
||||||
|
```
|
||||||
|
|
||||||
|
Or use the direct URL:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
bash <(curl -fsSL https://git.mosaicstack.dev/mosaicstack/stack/raw/branch/main/tools/install.sh)
|
||||||
```
|
```
|
||||||
|
|
||||||
The installer places the `mosaic` binary at `~/.npm-global/bin/mosaic`. Flags for
|
The installer places the `mosaic` binary at `~/.npm-global/bin/mosaic`. Flags for
|
||||||
|
|||||||
@@ -146,3 +146,28 @@ Worker's initial approach added `vitest.config.ts` to `apps/gateway/tsconfig.jso
|
|||||||
|
|
||||||
- Close out orchestrator doc sync (this commit): mark M01 subtasks done in `TASKS.md`, update manifest phase to Execution, commit scratchpad session 2, PR to main.
|
- Close out orchestrator doc sync (this commit): mark M01 subtasks done in `TASKS.md`, update manifest phase to Execution, commit scratchpad session 2, PR to main.
|
||||||
- After merge, delegate IUV-M02 (sonnet, isolated worktree). Dependencies: IUV-02-01 (CORS→FQDN) starts unblocked since M01 is released; first real task for the M02 worker is diagnosing the skill installer failure modes (IUV-02-02) against the fresh 0.0.26 install.
|
- After merge, delegate IUV-M02 (sonnet, isolated worktree). Dependencies: IUV-02-01 (CORS→FQDN) starts unblocked since M01 is released; first real task for the M02 worker is diagnosing the skill installer failure modes (IUV-02-02) against the fresh 0.0.26 install.
|
||||||
|
|
||||||
|
## Session 3 — 2026-04-05 (IUV-M02 delivery + close-out)
|
||||||
|
|
||||||
|
### Outcome
|
||||||
|
|
||||||
|
IUV-M02 shipped. PR #444 merged (`172bacb3`), issue #437 closed. 18 new tests (13 CORS derivation, 5 skill sync).
|
||||||
|
|
||||||
|
### Changes
|
||||||
|
|
||||||
|
**CORS → FQDN (IUV-02-01):**
|
||||||
|
|
||||||
|
- `packages/mosaic/src/stages/gateway-config.ts` — replaced raw "CORS origin" text prompt with "Web UI hostname" (default: `localhost`). Added HTTPS follow-up for remote hosts. Pure `deriveCorsOrigin(hostname, port, useHttps?)` function exported for testability.
|
||||||
|
- Headless: `MOSAIC_HOSTNAME` env var as friendly alternative; `MOSAIC_CORS_ORIGIN` still works as full override.
|
||||||
|
- `packages/mosaic/src/types.ts` — added `hostname?: string` to `GatewayState`.
|
||||||
|
|
||||||
|
**Skill installer rework (IUV-02-02 + IUV-02-03):**
|
||||||
|
|
||||||
|
- Root cause confirmed: `syncSkills()` in `finalize.ts` ignored `state.selectedSkills` entirely. The multiselect UI was a no-op.
|
||||||
|
- `packages/mosaic/src/stages/finalize.ts` — `syncSkills()` rewritten to accept `selectedSkills[]`, returns typed `SyncSkillsResult`, passes `MOSAIC_INSTALL_SKILLS` (colon-separated) as env var to the bash script.
|
||||||
|
- `packages/mosaic/framework/tools/_scripts/mosaic-sync-skills` — added bash associative array whitelist filter keyed on `MOSAIC_INSTALL_SKILLS`. When set, only whitelisted skills are linked. Empty/unset = all skills (legacy behavior preserved for `mosaic sync` outside wizard).
|
||||||
|
- Failure surfaces: silent `catch {}` replaced with typed error reporting through `p.warn()`.
|
||||||
|
|
||||||
|
### Next action
|
||||||
|
|
||||||
|
- Delegate IUV-M03 (opus, isolated worktree) — the architectural milestone: provider-first intelligent flow, drill-down main menu, Quick Start fast path, agent self-naming. This is the biggest piece of the mission.
|
||||||
|
|||||||
227
docs/scratchpads/iuv-m03-design.md
Normal file
227
docs/scratchpads/iuv-m03-design.md
Normal file
@@ -0,0 +1,227 @@
|
|||||||
|
# IUV-M03 Design: Provider-first intelligent flow + drill-down main menu
|
||||||
|
|
||||||
|
**Issue:** #438
|
||||||
|
**Branch:** `feat/install-ux-intent`
|
||||||
|
**Date:** 2026-04-05
|
||||||
|
|
||||||
|
## 1. New first-run state machine
|
||||||
|
|
||||||
|
The linear 12-stage interrogation is replaced with a menu-driven architecture.
|
||||||
|
|
||||||
|
### Flow overview
|
||||||
|
|
||||||
|
```
|
||||||
|
Welcome banner
|
||||||
|
|
|
||||||
|
v
|
||||||
|
Detect existing install (auto)
|
||||||
|
|
|
||||||
|
v
|
||||||
|
Main Menu (loop)
|
||||||
|
|-- Quick Start -> provider key + admin creds -> finalize
|
||||||
|
|-- Providers -> LLM API key config
|
||||||
|
|-- Agent Identity -> intent intake + naming (deterministic)
|
||||||
|
|-- Skills -> recommended / custom selection
|
||||||
|
|-- Gateway -> port, storage tier, hostname, CORS
|
||||||
|
|-- Advanced -> SOUL.md, USER.md, TOOLS.md, runtimes, hooks
|
||||||
|
|-- Finish & Apply -> finalize + gateway bootstrap
|
||||||
|
v
|
||||||
|
Done
|
||||||
|
```
|
||||||
|
|
||||||
|
### Menu navigation
|
||||||
|
|
||||||
|
- Main menu is a `select` prompt. Each option drills into a sub-flow.
|
||||||
|
- Completing a section returns to the main menu.
|
||||||
|
- Menu items show completion state: `[done]` hint after configuration.
|
||||||
|
- `Finish & Apply` is always last and requires at minimum a provider key (or explicit skip).
|
||||||
|
- The menu tracks configured sections in `WizardState.completedSections`.
|
||||||
|
|
||||||
|
### Headless bypass
|
||||||
|
|
||||||
|
When `MOSAIC_ASSUME_YES=1` or `!process.stdin.isTTY`, the entire menu is skipped.
|
||||||
|
The wizard runs: defaults + env var overrides -> finalize -> gateway config -> bootstrap.
|
||||||
|
This preserves full backward compatibility with `tools/install.sh --yes`.
|
||||||
|
|
||||||
|
## 2. Quick Start path
|
||||||
|
|
||||||
|
Target: 3-5 questions max. Under 90 seconds for a returning user.
|
||||||
|
|
||||||
|
### Questions asked
|
||||||
|
|
||||||
|
1. **Provider API key** (Anthropic/OpenAI) - `text` prompt with paste support
|
||||||
|
2. **Admin email** - `text` prompt
|
||||||
|
3. **Admin password** - masked + confirmed
|
||||||
|
|
||||||
|
### Questions skipped (with defaults)
|
||||||
|
|
||||||
|
| Setting | Default | Rationale |
|
||||||
|
| ---------------------------- | ------------------------------- | ---------------------- |
|
||||||
|
| Agent name | "Mosaic" | Generic but branded |
|
||||||
|
| Port | 14242 | Standard default |
|
||||||
|
| Storage tier | local | No external deps |
|
||||||
|
| Hostname | localhost | Dev-first |
|
||||||
|
| CORS origin | http://localhost:3000 | Standard web UI port |
|
||||||
|
| Skills | recommended set | Curated by maintainers |
|
||||||
|
| Runtimes | auto-detected | No user input needed |
|
||||||
|
| Communication style | direct | Most popular choice |
|
||||||
|
| SOUL.md / USER.md / TOOLS.md | template defaults | Can customize later |
|
||||||
|
| Hooks | auto-install if Claude detected | Safe default |
|
||||||
|
|
||||||
|
### Flow
|
||||||
|
|
||||||
|
```
|
||||||
|
Quick Start selected
|
||||||
|
-> "Paste your LLM API key (Anthropic recommended):"
|
||||||
|
-> [auto-detect provider from key prefix: sk-ant-* = Anthropic, sk-* = OpenAI]
|
||||||
|
-> Apply all defaults
|
||||||
|
-> Run finalize (sync framework, write configs, link assets, sync skills)
|
||||||
|
-> Run gateway config (headless-style with defaults + provided key)
|
||||||
|
-> "Admin email:"
|
||||||
|
-> "Admin password:" (masked + confirm)
|
||||||
|
-> Run gateway bootstrap
|
||||||
|
-> Done
|
||||||
|
```
|
||||||
|
|
||||||
|
## 3. Provider-first flow
|
||||||
|
|
||||||
|
Provider configuration (currently buried in gateway-config stage as "ANTHROPIC_API_KEY")
|
||||||
|
moves to a dedicated top-level menu item and is the first question in Quick Start.
|
||||||
|
|
||||||
|
### Provider detection
|
||||||
|
|
||||||
|
The API key prefix determines the provider:
|
||||||
|
|
||||||
|
- `sk-ant-api03-*` -> Anthropic (Claude)
|
||||||
|
- `sk-*` -> OpenAI
|
||||||
|
- Empty/skipped -> no provider (gateway starts without LLM access)
|
||||||
|
|
||||||
|
### Storage
|
||||||
|
|
||||||
|
The provider key is stored in the gateway `.env` as `ANTHROPIC_API_KEY` or `OPENAI_API_KEY`.
|
||||||
|
For Quick Start, this replaces the old interactive prompt in `collectAndWriteConfig`.
|
||||||
|
|
||||||
|
### Menu section: "Providers"
|
||||||
|
|
||||||
|
In the drill-down menu, "Providers" lets users:
|
||||||
|
|
||||||
|
1. Enter/change their API key
|
||||||
|
2. See which provider was detected
|
||||||
|
3. Optionally configure a second provider
|
||||||
|
|
||||||
|
For v0.0.27, we support Anthropic and OpenAI keys only. The key is stored
|
||||||
|
in `WizardState` and written during finalize.
|
||||||
|
|
||||||
|
## 4. Intent intake + naming (deterministic fallback - Option B)
|
||||||
|
|
||||||
|
### Rationale
|
||||||
|
|
||||||
|
At install time, the LLM provider may not be configured yet (chicken-and-egg).
|
||||||
|
We use **Option B: deterministic advisor** for the install wizard.
|
||||||
|
|
||||||
|
### Flow (Agent Identity menu section)
|
||||||
|
|
||||||
|
```
|
||||||
|
1. "What will this agent primarily help you with?"
|
||||||
|
-> Select from presets:
|
||||||
|
- General purpose assistant
|
||||||
|
- Software development
|
||||||
|
- DevOps & infrastructure
|
||||||
|
- Research & analysis
|
||||||
|
- Content & writing
|
||||||
|
- Custom (free text description)
|
||||||
|
|
||||||
|
2. System proposes a thematic name based on selection:
|
||||||
|
- General purpose -> "Mosaic"
|
||||||
|
- Software development -> "Forge"
|
||||||
|
- DevOps & infrastructure -> "Sentinel"
|
||||||
|
- Research & analysis -> "Atlas"
|
||||||
|
- Content & writing -> "Muse"
|
||||||
|
- Custom -> "Mosaic" (default)
|
||||||
|
|
||||||
|
3. "Your agent will be named 'Forge'. Press Enter to accept or type a new name:"
|
||||||
|
-> User confirms or overrides
|
||||||
|
```
|
||||||
|
|
||||||
|
### Storage
|
||||||
|
|
||||||
|
- Agent name -> `WizardState.soul.agentName` -> written to SOUL.md
|
||||||
|
- Intent category -> `WizardState.agentIntent` (new field) -> written to `~/.config/mosaic/agent.json`
|
||||||
|
|
||||||
|
### Post-install LLM-powered intake (future)
|
||||||
|
|
||||||
|
A future `mosaic configure identity` command can use the configured LLM to:
|
||||||
|
|
||||||
|
- Accept free-text intent description
|
||||||
|
- Generate an expounded persona
|
||||||
|
- Propose a contextual name
|
||||||
|
|
||||||
|
This is explicitly out of scope for the install wizard.
|
||||||
|
|
||||||
|
## 5. Headless backward-compat
|
||||||
|
|
||||||
|
### Supported env vars (unchanged)
|
||||||
|
|
||||||
|
| Variable | Used by |
|
||||||
|
| -------------------------- | ---------------------------------------------- |
|
||||||
|
| `MOSAIC_ASSUME_YES=1` | Skip all prompts, use defaults + env overrides |
|
||||||
|
| `MOSAIC_ADMIN_NAME` | Gateway bootstrap |
|
||||||
|
| `MOSAIC_ADMIN_EMAIL` | Gateway bootstrap |
|
||||||
|
| `MOSAIC_ADMIN_PASSWORD` | Gateway bootstrap |
|
||||||
|
| `MOSAIC_GATEWAY_PORT` | Gateway config |
|
||||||
|
| `MOSAIC_HOSTNAME` | Gateway config (CORS derivation) |
|
||||||
|
| `MOSAIC_CORS_ORIGIN` | Gateway config (full override) |
|
||||||
|
| `MOSAIC_STORAGE_TIER` | Gateway config (local/team) |
|
||||||
|
| `MOSAIC_DATABASE_URL` | Gateway config (team tier) |
|
||||||
|
| `MOSAIC_VALKEY_URL` | Gateway config (team tier) |
|
||||||
|
| `MOSAIC_ANTHROPIC_API_KEY` | Provider config |
|
||||||
|
|
||||||
|
### New env vars
|
||||||
|
|
||||||
|
| Variable | Purpose |
|
||||||
|
| --------------------- | ----------------------------------------- |
|
||||||
|
| `MOSAIC_AGENT_NAME` | Override agent name in headless mode |
|
||||||
|
| `MOSAIC_AGENT_INTENT` | Override intent category in headless mode |
|
||||||
|
|
||||||
|
### `tools/install.sh --yes`
|
||||||
|
|
||||||
|
The install script sets `MOSAIC_ASSUME_YES=1` and passes through env vars.
|
||||||
|
No changes needed to the script itself. The new wizard detects headless mode
|
||||||
|
at the top of `runWizard` and runs a linear path identical to the old flow.
|
||||||
|
|
||||||
|
## 6. Explicit non-goals
|
||||||
|
|
||||||
|
- **No GUI** — this is a terminal wizard only
|
||||||
|
- **No multi-user install** — single-user, single-machine
|
||||||
|
- **No registry changes** — npm publish flow is unchanged
|
||||||
|
- **No LLM calls during install** — deterministic fallback only
|
||||||
|
- **No new dependencies** — uses existing @clack/prompts and picocolors
|
||||||
|
- **No changes to gateway API** — only the wizard orchestration changes
|
||||||
|
- **No changes to tools/install.sh** — headless compat maintained via env vars
|
||||||
|
|
||||||
|
## 7. Implementation plan
|
||||||
|
|
||||||
|
### Files to modify
|
||||||
|
|
||||||
|
1. `packages/mosaic/src/types.ts` — add `MenuSection`, `AgentIntent`, `completedSections`, `agentIntent`, `providerKey`, `providerType` to WizardState
|
||||||
|
2. `packages/mosaic/src/wizard.ts` — replace linear flow with menu loop
|
||||||
|
3. `packages/mosaic/src/stages/mode-select.ts` — becomes the main menu
|
||||||
|
4. `packages/mosaic/src/stages/provider-setup.ts` — new: provider key collection
|
||||||
|
5. `packages/mosaic/src/stages/agent-intent.ts` — new: intent intake + naming
|
||||||
|
6. `packages/mosaic/src/stages/menu-gateway.ts` — new: gateway sub-menu wrapper
|
||||||
|
7. `packages/mosaic/src/stages/quick-start.ts` — new: quick start linear path
|
||||||
|
8. `packages/mosaic/src/constants.ts` — add intent presets and name mappings
|
||||||
|
9. `packages/mosaic/package.json` — version bump 0.0.26 -> 0.0.27
|
||||||
|
|
||||||
|
### Files to add (tests)
|
||||||
|
|
||||||
|
1. `packages/mosaic/src/stages/wizard-menu.spec.ts` — menu navigation tests
|
||||||
|
2. `packages/mosaic/src/stages/quick-start.spec.ts` — quick start path tests
|
||||||
|
3. `packages/mosaic/src/stages/agent-intent.spec.ts` — intent + naming tests
|
||||||
|
4. `packages/mosaic/src/stages/provider-setup.spec.ts` — provider detection tests
|
||||||
|
|
||||||
|
### Migration strategy
|
||||||
|
|
||||||
|
The existing stage functions remain intact. The menu system wraps them —
|
||||||
|
each menu item calls the appropriate stage function(s). The linear headless
|
||||||
|
path calls them in the same order as before.
|
||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/agent"
|
"directory": "packages/agent"
|
||||||
},
|
},
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/auth"
|
"directory": "packages/auth"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.3",
|
"version": "0.0.3",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/brain"
|
"directory": "packages/brain"
|
||||||
},
|
},
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/config"
|
"directory": "packages/config"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/coord"
|
"directory": "packages/coord"
|
||||||
},
|
},
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.3",
|
"version": "0.0.3",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/db"
|
"directory": "packages/db"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/design-tokens"
|
"directory": "packages/design-tokens"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.3",
|
"version": "0.0.3",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/forge"
|
"directory": "packages/forge"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.3",
|
"version": "0.0.3",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/log"
|
"directory": "packages/log"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.3",
|
"version": "0.0.3",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/macp"
|
"directory": "packages/macp"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.4",
|
"version": "0.0.4",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/memory"
|
"directory": "packages/memory"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ import { runWizard } from '../../src/wizard.js';
|
|||||||
describe('Full Wizard (headless)', () => {
|
describe('Full Wizard (headless)', () => {
|
||||||
let tmpDir: string;
|
let tmpDir: string;
|
||||||
const repoRoot = join(import.meta.dirname, '..', '..');
|
const repoRoot = join(import.meta.dirname, '..', '..');
|
||||||
|
const originalEnv = { ...process.env };
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
tmpDir = mkdtempSync(join(tmpdir(), 'mosaic-wizard-test-'));
|
tmpDir = mkdtempSync(join(tmpdir(), 'mosaic-wizard-test-'));
|
||||||
@@ -32,12 +33,16 @@ describe('Full Wizard (headless)', () => {
|
|||||||
|
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
rmSync(tmpDir, { recursive: true, force: true });
|
rmSync(tmpDir, { recursive: true, force: true });
|
||||||
|
process.env = { ...originalEnv };
|
||||||
});
|
});
|
||||||
|
|
||||||
it('quick start produces valid SOUL.md', async () => {
|
it('quick start produces valid SOUL.md', async () => {
|
||||||
|
// The headless path reads agent name from MOSAIC_AGENT_NAME env var
|
||||||
|
// (via agentIntentStage) rather than prompting interactively.
|
||||||
|
process.env['MOSAIC_AGENT_NAME'] = 'TestBot';
|
||||||
|
|
||||||
const prompter = new HeadlessPrompter({
|
const prompter = new HeadlessPrompter({
|
||||||
'Installation mode': 'quick',
|
'Installation mode': 'quick',
|
||||||
'What name should agents use?': 'TestBot',
|
|
||||||
'Communication style': 'direct',
|
'Communication style': 'direct',
|
||||||
'Your name': 'Tester',
|
'Your name': 'Tester',
|
||||||
'Your pronouns': 'They/Them',
|
'Your pronouns': 'They/Them',
|
||||||
@@ -62,9 +67,10 @@ describe('Full Wizard (headless)', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('quick start produces valid USER.md', async () => {
|
it('quick start produces valid USER.md', async () => {
|
||||||
|
process.env['MOSAIC_AGENT_NAME'] = 'TestBot';
|
||||||
|
|
||||||
const prompter = new HeadlessPrompter({
|
const prompter = new HeadlessPrompter({
|
||||||
'Installation mode': 'quick',
|
'Installation mode': 'quick',
|
||||||
'What name should agents use?': 'TestBot',
|
|
||||||
'Communication style': 'direct',
|
'Communication style': 'direct',
|
||||||
'Your name': 'Tester',
|
'Your name': 'Tester',
|
||||||
'Your pronouns': 'He/Him',
|
'Your pronouns': 'He/Him',
|
||||||
|
|||||||
@@ -4,14 +4,20 @@ Universal agent standards layer for Claude Code, Codex, OpenCode, and Pi.
|
|||||||
|
|
||||||
One config, every runtime, same standards.
|
One config, every runtime, same standards.
|
||||||
|
|
||||||
> **This is the framework component of [mosaic-stack](https://git.mosaicstack.dev/mosaic/mosaic-stack).** No personal data, credentials, user-specific preferences, or machine-specific paths should be committed. All personalization happens at install time via `mosaic init` or by editing files in `~/.config/mosaic/` after installation.
|
> **This is the framework component of [mosaic-stack](https://git.mosaicstack.dev/mosaicstack/stack).** No personal data, credentials, user-specific preferences, or machine-specific paths should be committed. All personalization happens at install time via `mosaic init` or by editing files in `~/.config/mosaic/` after installation.
|
||||||
|
|
||||||
## Quick Install
|
## Quick Install
|
||||||
|
|
||||||
### Mac / Linux
|
### Mac / Linux
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
bash <(curl -fsSL https://git.mosaicstack.dev/mosaic/mosaic-stack/raw/branch/main/tools/install.sh)
|
curl -fsSL https://mosaicstack.dev/install.sh | bash
|
||||||
|
```
|
||||||
|
|
||||||
|
Or use the direct URL:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
bash <(curl -fsSL https://git.mosaicstack.dev/mosaicstack/stack/raw/branch/main/tools/install.sh)
|
||||||
```
|
```
|
||||||
|
|
||||||
### Windows (PowerShell)
|
### Windows (PowerShell)
|
||||||
@@ -23,8 +29,8 @@ bash <(curl -fsSL https://git.mosaicstack.dev/mosaic/mosaic-stack/raw/branch/mai
|
|||||||
### From Source (any platform)
|
### From Source (any platform)
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone git@git.mosaicstack.dev:mosaic/mosaic-stack.git ~/src/mosaic-stack
|
git clone git@git.mosaicstack.dev:mosaicstack/stack.git ~/src/stack
|
||||||
cd ~/src/mosaic-stack && bash tools/install.sh
|
cd ~/src/stack && bash tools/install.sh
|
||||||
```
|
```
|
||||||
|
|
||||||
The installer:
|
The installer:
|
||||||
@@ -145,13 +151,19 @@ mosaic upgrade check # Check upgrade status (no changes)
|
|||||||
Run the installer again — it handles upgrades automatically:
|
Run the installer again — it handles upgrades automatically:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
bash <(curl -fsSL https://git.mosaicstack.dev/mosaic/mosaic-stack/raw/branch/main/tools/install.sh)
|
curl -fsSL https://mosaicstack.dev/install.sh | bash
|
||||||
|
```
|
||||||
|
|
||||||
|
Or use the direct URL:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
bash <(curl -fsSL https://git.mosaicstack.dev/mosaicstack/stack/raw/branch/main/tools/install.sh)
|
||||||
```
|
```
|
||||||
|
|
||||||
Or from a local checkout:
|
Or from a local checkout:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd ~/src/mosaic-stack && git pull && bash tools/install.sh
|
cd ~/src/stack && git pull && bash tools/install.sh
|
||||||
```
|
```
|
||||||
|
|
||||||
The installer preserves local `SOUL.md`, `USER.md`, `TOOLS.md`, and `memory/` by default.
|
The installer preserves local `SOUL.md`, `USER.md`, `TOOLS.md`, and `memory/` by default.
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
{
|
{
|
||||||
"name": "@mosaicstack/mosaic",
|
"name": "@mosaicstack/mosaic",
|
||||||
"version": "0.0.26",
|
"version": "0.0.28",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/mosaic"
|
"directory": "packages/mosaic"
|
||||||
},
|
},
|
||||||
"description": "Mosaic agent framework — installation wizard and meta package",
|
"description": "Mosaic agent framework — installation wizard and meta package",
|
||||||
|
|||||||
@@ -135,15 +135,11 @@ program
|
|||||||
|
|
||||||
// No valid session — prompt for credentials
|
// No valid session — prompt for credentials
|
||||||
if (!session) {
|
if (!session) {
|
||||||
const readline = await import('node:readline');
|
const { promptLine, promptSecret } = await import('./commands/gateway/login.js');
|
||||||
const rl = readline.createInterface({ input: process.stdin, output: process.stdout });
|
|
||||||
const ask = (q: string): Promise<string> =>
|
|
||||||
new Promise((resolve) => rl.question(q, resolve));
|
|
||||||
|
|
||||||
console.log(`Sign in to ${opts.gateway}`);
|
console.log(`Sign in to ${opts.gateway}`);
|
||||||
const email = await ask('Email: ');
|
const email = await promptLine('Email: ');
|
||||||
const password = await ask('Password: ');
|
const password = await promptSecret('Password: ');
|
||||||
rl.close();
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const auth = await signIn(opts.gateway, email, password);
|
const auth = await signIn(opts.gateway, email, password);
|
||||||
|
|||||||
@@ -26,6 +26,53 @@ export const DEFAULTS = {
|
|||||||
| (add your git providers here) | | | |`,
|
| (add your git providers here) | | | |`,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/** Preset intent categories with display labels and suggested agent names. */
|
||||||
|
export const INTENT_PRESETS: Record<
|
||||||
|
string,
|
||||||
|
{ label: string; hint: string; suggestedName: string }
|
||||||
|
> = {
|
||||||
|
general: {
|
||||||
|
label: 'General purpose assistant',
|
||||||
|
hint: 'Versatile helper for any task',
|
||||||
|
suggestedName: 'Mosaic',
|
||||||
|
},
|
||||||
|
'software-dev': {
|
||||||
|
label: 'Software development',
|
||||||
|
hint: 'Coding, debugging, architecture',
|
||||||
|
suggestedName: 'Forge',
|
||||||
|
},
|
||||||
|
devops: {
|
||||||
|
label: 'DevOps & infrastructure',
|
||||||
|
hint: 'CI/CD, containers, monitoring',
|
||||||
|
suggestedName: 'Sentinel',
|
||||||
|
},
|
||||||
|
research: {
|
||||||
|
label: 'Research & analysis',
|
||||||
|
hint: 'Data analysis, literature review',
|
||||||
|
suggestedName: 'Atlas',
|
||||||
|
},
|
||||||
|
content: {
|
||||||
|
label: 'Content & writing',
|
||||||
|
hint: 'Documentation, copywriting, editing',
|
||||||
|
suggestedName: 'Muse',
|
||||||
|
},
|
||||||
|
custom: {
|
||||||
|
label: 'Custom',
|
||||||
|
hint: 'Describe your own use case',
|
||||||
|
suggestedName: 'Mosaic',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Detect LLM provider type from an API key prefix.
|
||||||
|
*/
|
||||||
|
export function detectProviderType(key: string): 'anthropic' | 'openai' | 'none' {
|
||||||
|
if (!key) return 'none';
|
||||||
|
if (key.startsWith('sk-ant-')) return 'anthropic';
|
||||||
|
if (key.startsWith('sk-')) return 'openai';
|
||||||
|
return 'none';
|
||||||
|
}
|
||||||
|
|
||||||
export const RECOMMENDED_SKILLS = new Set([
|
export const RECOMMENDED_SKILLS = new Set([
|
||||||
'brainstorming',
|
'brainstorming',
|
||||||
'code-review-excellence',
|
'code-review-excellence',
|
||||||
|
|||||||
129
packages/mosaic/src/stages/agent-intent.spec.ts
Normal file
129
packages/mosaic/src/stages/agent-intent.spec.ts
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
import { describe, it, expect, vi, afterEach } from 'vitest';
|
||||||
|
import type { WizardState } from '../types.js';
|
||||||
|
import { agentIntentStage } from './agent-intent.js';
|
||||||
|
|
||||||
|
function buildPrompter(overrides: Partial<Record<string, unknown>> = {}) {
|
||||||
|
return {
|
||||||
|
intro: vi.fn(),
|
||||||
|
outro: vi.fn(),
|
||||||
|
note: vi.fn(),
|
||||||
|
log: vi.fn(),
|
||||||
|
warn: vi.fn(),
|
||||||
|
text: vi.fn().mockResolvedValue('Mosaic'),
|
||||||
|
confirm: vi.fn().mockResolvedValue(false),
|
||||||
|
select: vi.fn().mockResolvedValue('general'),
|
||||||
|
multiselect: vi.fn(),
|
||||||
|
groupMultiselect: vi.fn(),
|
||||||
|
spinner: vi.fn().mockReturnValue({ update: vi.fn(), stop: vi.fn() }),
|
||||||
|
separator: vi.fn(),
|
||||||
|
...overrides,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function makeState(): WizardState {
|
||||||
|
return {
|
||||||
|
mosaicHome: '/tmp/mosaic',
|
||||||
|
sourceDir: '/tmp/mosaic',
|
||||||
|
mode: 'quick',
|
||||||
|
installAction: 'fresh',
|
||||||
|
soul: {},
|
||||||
|
user: {},
|
||||||
|
tools: {},
|
||||||
|
runtimes: { detected: [], mcpConfigured: false },
|
||||||
|
selectedSkills: [],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('agentIntentStage', () => {
|
||||||
|
const originalEnv = { ...process.env };
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
process.env = { ...originalEnv };
|
||||||
|
});
|
||||||
|
|
||||||
|
it('uses default intent and name in headless mode', async () => {
|
||||||
|
process.env['MOSAIC_ASSUME_YES'] = '1';
|
||||||
|
delete process.env['MOSAIC_AGENT_INTENT'];
|
||||||
|
delete process.env['MOSAIC_AGENT_NAME'];
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter();
|
||||||
|
|
||||||
|
await agentIntentStage(p, state);
|
||||||
|
|
||||||
|
expect(state.agentIntent).toBe('general');
|
||||||
|
expect(state.soul.agentName).toBe('Mosaic');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('reads intent from MOSAIC_AGENT_INTENT env var', async () => {
|
||||||
|
process.env['MOSAIC_ASSUME_YES'] = '1';
|
||||||
|
process.env['MOSAIC_AGENT_INTENT'] = 'software-dev';
|
||||||
|
delete process.env['MOSAIC_AGENT_NAME'];
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter();
|
||||||
|
|
||||||
|
await agentIntentStage(p, state);
|
||||||
|
|
||||||
|
expect(state.agentIntent).toBe('software-dev');
|
||||||
|
expect(state.soul.agentName).toBe('Forge');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('honors MOSAIC_AGENT_NAME env var override', async () => {
|
||||||
|
process.env['MOSAIC_ASSUME_YES'] = '1';
|
||||||
|
process.env['MOSAIC_AGENT_INTENT'] = 'devops';
|
||||||
|
process.env['MOSAIC_AGENT_NAME'] = 'MyBot';
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter();
|
||||||
|
|
||||||
|
await agentIntentStage(p, state);
|
||||||
|
|
||||||
|
expect(state.agentIntent).toBe('devops');
|
||||||
|
expect(state.soul.agentName).toBe('MyBot');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('falls back to general for unknown intent values', async () => {
|
||||||
|
process.env['MOSAIC_ASSUME_YES'] = '1';
|
||||||
|
process.env['MOSAIC_AGENT_INTENT'] = 'nonexistent';
|
||||||
|
delete process.env['MOSAIC_AGENT_NAME'];
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter();
|
||||||
|
|
||||||
|
await agentIntentStage(p, state);
|
||||||
|
|
||||||
|
expect(state.agentIntent).toBe('general');
|
||||||
|
expect(state.soul.agentName).toBe('Mosaic');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('prompts for intent and name in interactive mode', async () => {
|
||||||
|
delete process.env['MOSAIC_ASSUME_YES'];
|
||||||
|
const origIsTTY = process.stdin.isTTY;
|
||||||
|
Object.defineProperty(process.stdin, 'isTTY', { value: true, configurable: true });
|
||||||
|
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter({
|
||||||
|
select: vi.fn().mockResolvedValue('research'),
|
||||||
|
text: vi.fn().mockResolvedValue('Atlas'),
|
||||||
|
});
|
||||||
|
|
||||||
|
await agentIntentStage(p, state);
|
||||||
|
|
||||||
|
expect(state.agentIntent).toBe('research');
|
||||||
|
expect(state.soul.agentName).toBe('Atlas');
|
||||||
|
expect(p.select).toHaveBeenCalled();
|
||||||
|
expect(p.text).toHaveBeenCalled();
|
||||||
|
|
||||||
|
Object.defineProperty(process.stdin, 'isTTY', { value: origIsTTY, configurable: true });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('maps content intent to Muse suggested name', async () => {
|
||||||
|
process.env['MOSAIC_ASSUME_YES'] = '1';
|
||||||
|
process.env['MOSAIC_AGENT_INTENT'] = 'content';
|
||||||
|
delete process.env['MOSAIC_AGENT_NAME'];
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter();
|
||||||
|
|
||||||
|
await agentIntentStage(p, state);
|
||||||
|
|
||||||
|
expect(state.agentIntent).toBe('content');
|
||||||
|
expect(state.soul.agentName).toBe('Muse');
|
||||||
|
});
|
||||||
|
});
|
||||||
64
packages/mosaic/src/stages/agent-intent.ts
Normal file
64
packages/mosaic/src/stages/agent-intent.ts
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
import type { WizardPrompter } from '../prompter/interface.js';
|
||||||
|
import type { AgentIntent, WizardState } from '../types.js';
|
||||||
|
import { INTENT_PRESETS } from '../constants.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Agent intent + naming stage — deterministic (no LLM required).
|
||||||
|
*
|
||||||
|
* The user picks an intent category from presets, the system proposes a
|
||||||
|
* thematic name, and the user confirms or overrides it.
|
||||||
|
*
|
||||||
|
* In headless mode, reads from `MOSAIC_AGENT_INTENT` and `MOSAIC_AGENT_NAME`.
|
||||||
|
*/
|
||||||
|
export async function agentIntentStage(p: WizardPrompter, state: WizardState): Promise<void> {
|
||||||
|
const isHeadless = process.env['MOSAIC_ASSUME_YES'] === '1' || !process.stdin.isTTY;
|
||||||
|
|
||||||
|
if (isHeadless) {
|
||||||
|
const intentEnv = process.env['MOSAIC_AGENT_INTENT'] ?? 'general';
|
||||||
|
const nameEnv = process.env['MOSAIC_AGENT_NAME'];
|
||||||
|
const preset = INTENT_PRESETS[intentEnv] ?? INTENT_PRESETS['general']!;
|
||||||
|
state.agentIntent ??= (intentEnv in INTENT_PRESETS ? intentEnv : 'general') as AgentIntent;
|
||||||
|
// Respect existing agentName (e.g. from CLI overrides) — only set from
|
||||||
|
// env/preset if not already populated.
|
||||||
|
state.soul.agentName ??= nameEnv ?? preset.suggestedName;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
p.separator();
|
||||||
|
p.note(
|
||||||
|
'Tell us what this agent will primarily help you with.\n' +
|
||||||
|
"We'll suggest a name based on your choice — you can always change it.",
|
||||||
|
'Agent Identity',
|
||||||
|
);
|
||||||
|
|
||||||
|
const intentOptions = Object.entries(INTENT_PRESETS).map(([value, info]) => ({
|
||||||
|
value: value as AgentIntent,
|
||||||
|
label: info.label,
|
||||||
|
hint: info.hint,
|
||||||
|
}));
|
||||||
|
|
||||||
|
const intent = await p.select<AgentIntent>({
|
||||||
|
message: 'What will this agent primarily help you with?',
|
||||||
|
options: intentOptions,
|
||||||
|
initialValue: 'general' as AgentIntent,
|
||||||
|
});
|
||||||
|
|
||||||
|
state.agentIntent = intent;
|
||||||
|
|
||||||
|
const preset = INTENT_PRESETS[intent];
|
||||||
|
const suggestedName = preset?.suggestedName ?? 'Mosaic';
|
||||||
|
|
||||||
|
const name = await p.text({
|
||||||
|
message: `Your agent will be named "${suggestedName}". Press Enter to accept or type a new name`,
|
||||||
|
initialValue: suggestedName,
|
||||||
|
defaultValue: suggestedName,
|
||||||
|
validate: (v) => {
|
||||||
|
if (v.length === 0) return 'Name cannot be empty';
|
||||||
|
if (v.length > 50) return 'Name must be under 50 characters';
|
||||||
|
return undefined;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
state.soul.agentName = name;
|
||||||
|
p.log(`Agent name set to: ${name}`);
|
||||||
|
}
|
||||||
@@ -126,6 +126,14 @@ export interface GatewayConfigStageOptions {
|
|||||||
portOverride?: number;
|
portOverride?: number;
|
||||||
/** Skip the `npm install -g @mosaicstack/gateway` step (local build / tests). */
|
/** Skip the `npm install -g @mosaicstack/gateway` step (local build / tests). */
|
||||||
skipInstall?: boolean;
|
skipInstall?: boolean;
|
||||||
|
/**
|
||||||
|
* Pre-collected provider API key (from the provider-setup stage or Quick
|
||||||
|
* Start path). When set, the gateway-config stage will skip the interactive
|
||||||
|
* API key prompt and use this value directly.
|
||||||
|
*/
|
||||||
|
providerKey?: string;
|
||||||
|
/** Provider type detected from the key prefix. */
|
||||||
|
providerType?: 'anthropic' | 'openai' | 'none';
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface GatewayConfigStageResult {
|
export interface GatewayConfigStageResult {
|
||||||
@@ -314,6 +322,8 @@ export async function gatewayConfigStage(
|
|||||||
envFile: ENV_FILE,
|
envFile: ENV_FILE,
|
||||||
mosaicConfigFile: MOSAIC_CONFIG_FILE,
|
mosaicConfigFile: MOSAIC_CONFIG_FILE,
|
||||||
gatewayHome: GATEWAY_HOME,
|
gatewayHome: GATEWAY_HOME,
|
||||||
|
providerKey: opts.providerKey,
|
||||||
|
providerType: opts.providerType,
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
if (err instanceof GatewayConfigValidationError) {
|
if (err instanceof GatewayConfigValidationError) {
|
||||||
@@ -389,6 +399,10 @@ interface CollectOptions {
|
|||||||
envFile: string;
|
envFile: string;
|
||||||
mosaicConfigFile: string;
|
mosaicConfigFile: string;
|
||||||
gatewayHome: string;
|
gatewayHome: string;
|
||||||
|
/** Pre-collected API key — skips the interactive prompt when set. */
|
||||||
|
providerKey?: string;
|
||||||
|
/** Provider type — determines the env var name for the key. */
|
||||||
|
providerType?: 'anthropic' | 'openai' | 'none';
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Raised by the config stage when headless env validation fails. */
|
/** Raised by the config stage when headless env validation fails. */
|
||||||
@@ -466,10 +480,15 @@ async function collectAndWriteConfig(
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
anthropicKey = await p.text({
|
if (opts.providerKey) {
|
||||||
message: 'ANTHROPIC_API_KEY (optional, press Enter to skip)',
|
anthropicKey = opts.providerKey;
|
||||||
defaultValue: '',
|
p.log(`Using API key from provider setup (${opts.providerType ?? 'unknown'}).`);
|
||||||
});
|
} else {
|
||||||
|
anthropicKey = await p.text({
|
||||||
|
message: 'ANTHROPIC_API_KEY (optional, press Enter to skip)',
|
||||||
|
defaultValue: '',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
hostname = await p.text({
|
hostname = await p.text({
|
||||||
message: 'Web UI hostname (for browser access)',
|
message: 'Web UI hostname (for browser access)',
|
||||||
@@ -508,7 +527,11 @@ async function collectAndWriteConfig(
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (anthropicKey) {
|
if (anthropicKey) {
|
||||||
envLines.push(`ANTHROPIC_API_KEY=${anthropicKey}`);
|
if (opts.providerType === 'openai') {
|
||||||
|
envLines.push(`OPENAI_API_KEY=${anthropicKey}`);
|
||||||
|
} else {
|
||||||
|
envLines.push(`ANTHROPIC_API_KEY=${anthropicKey}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
writeFileSync(opts.envFile, envLines.join('\n') + '\n', { mode: 0o600 });
|
writeFileSync(opts.envFile, envLines.join('\n') + '\n', { mode: 0o600 });
|
||||||
|
|||||||
118
packages/mosaic/src/stages/provider-setup.spec.ts
Normal file
118
packages/mosaic/src/stages/provider-setup.spec.ts
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
import { describe, it, expect, vi, afterEach } from 'vitest';
|
||||||
|
import type { WizardState } from '../types.js';
|
||||||
|
import { providerSetupStage } from './provider-setup.js';
|
||||||
|
|
||||||
|
function buildPrompter(overrides: Partial<Record<string, unknown>> = {}) {
|
||||||
|
return {
|
||||||
|
intro: vi.fn(),
|
||||||
|
outro: vi.fn(),
|
||||||
|
note: vi.fn(),
|
||||||
|
log: vi.fn(),
|
||||||
|
warn: vi.fn(),
|
||||||
|
text: vi.fn().mockResolvedValue(''),
|
||||||
|
confirm: vi.fn().mockResolvedValue(false),
|
||||||
|
select: vi.fn().mockResolvedValue('general'),
|
||||||
|
multiselect: vi.fn(),
|
||||||
|
groupMultiselect: vi.fn(),
|
||||||
|
spinner: vi.fn().mockReturnValue({ update: vi.fn(), stop: vi.fn() }),
|
||||||
|
separator: vi.fn(),
|
||||||
|
...overrides,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function makeState(): WizardState {
|
||||||
|
return {
|
||||||
|
mosaicHome: '/tmp/mosaic',
|
||||||
|
sourceDir: '/tmp/mosaic',
|
||||||
|
mode: 'quick',
|
||||||
|
installAction: 'fresh',
|
||||||
|
soul: {},
|
||||||
|
user: {},
|
||||||
|
tools: {},
|
||||||
|
runtimes: { detected: [], mcpConfigured: false },
|
||||||
|
selectedSkills: [],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('providerSetupStage', () => {
|
||||||
|
const originalEnv = { ...process.env };
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
process.env = { ...originalEnv };
|
||||||
|
});
|
||||||
|
|
||||||
|
it('detects Anthropic key from prefix in headless mode', async () => {
|
||||||
|
process.env['MOSAIC_ASSUME_YES'] = '1';
|
||||||
|
process.env['MOSAIC_ANTHROPIC_API_KEY'] = 'sk-ant-api03-test123';
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter();
|
||||||
|
|
||||||
|
await providerSetupStage(p, state);
|
||||||
|
|
||||||
|
expect(state.providerKey).toBe('sk-ant-api03-test123');
|
||||||
|
expect(state.providerType).toBe('anthropic');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('detects OpenAI key from prefix in headless mode', async () => {
|
||||||
|
process.env['MOSAIC_ASSUME_YES'] = '1';
|
||||||
|
process.env['MOSAIC_OPENAI_API_KEY'] = 'sk-proj-test123';
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter();
|
||||||
|
|
||||||
|
await providerSetupStage(p, state);
|
||||||
|
|
||||||
|
expect(state.providerKey).toBe('sk-proj-test123');
|
||||||
|
expect(state.providerType).toBe('openai');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('sets provider type to none when no key is provided in headless mode', async () => {
|
||||||
|
process.env['MOSAIC_ASSUME_YES'] = '1';
|
||||||
|
delete process.env['MOSAIC_ANTHROPIC_API_KEY'];
|
||||||
|
delete process.env['MOSAIC_OPENAI_API_KEY'];
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter();
|
||||||
|
|
||||||
|
await providerSetupStage(p, state);
|
||||||
|
|
||||||
|
expect(state.providerKey).toBeUndefined();
|
||||||
|
expect(state.providerType).toBe('none');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('prompts for key in interactive mode', async () => {
|
||||||
|
delete process.env['MOSAIC_ASSUME_YES'];
|
||||||
|
// Simulate a TTY
|
||||||
|
const origIsTTY = process.stdin.isTTY;
|
||||||
|
Object.defineProperty(process.stdin, 'isTTY', { value: true, configurable: true });
|
||||||
|
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter({
|
||||||
|
text: vi.fn().mockResolvedValue('sk-ant-api03-interactive'),
|
||||||
|
});
|
||||||
|
|
||||||
|
await providerSetupStage(p, state);
|
||||||
|
|
||||||
|
expect(p.text).toHaveBeenCalled();
|
||||||
|
expect(state.providerKey).toBe('sk-ant-api03-interactive');
|
||||||
|
expect(state.providerType).toBe('anthropic');
|
||||||
|
|
||||||
|
Object.defineProperty(process.stdin, 'isTTY', { value: origIsTTY, configurable: true });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('handles empty key in interactive mode', async () => {
|
||||||
|
delete process.env['MOSAIC_ASSUME_YES'];
|
||||||
|
const origIsTTY = process.stdin.isTTY;
|
||||||
|
Object.defineProperty(process.stdin, 'isTTY', { value: true, configurable: true });
|
||||||
|
|
||||||
|
const state = makeState();
|
||||||
|
const p = buildPrompter({
|
||||||
|
text: vi.fn().mockResolvedValue(''),
|
||||||
|
});
|
||||||
|
|
||||||
|
await providerSetupStage(p, state);
|
||||||
|
|
||||||
|
expect(state.providerType).toBe('none');
|
||||||
|
expect(state.providerKey).toBeUndefined();
|
||||||
|
|
||||||
|
Object.defineProperty(process.stdin, 'isTTY', { value: origIsTTY, configurable: true });
|
||||||
|
});
|
||||||
|
});
|
||||||
54
packages/mosaic/src/stages/provider-setup.ts
Normal file
54
packages/mosaic/src/stages/provider-setup.ts
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import type { WizardPrompter } from '../prompter/interface.js';
|
||||||
|
import type { WizardState } from '../types.js';
|
||||||
|
import { detectProviderType } from '../constants.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provider setup stage — collects the user's LLM API key and detects the
|
||||||
|
* provider type from the key prefix.
|
||||||
|
*
|
||||||
|
* In headless mode, reads from `MOSAIC_ANTHROPIC_API_KEY` or `MOSAIC_OPENAI_API_KEY`.
|
||||||
|
*/
|
||||||
|
export async function providerSetupStage(p: WizardPrompter, state: WizardState): Promise<void> {
|
||||||
|
const isHeadless = process.env['MOSAIC_ASSUME_YES'] === '1' || !process.stdin.isTTY;
|
||||||
|
|
||||||
|
if (isHeadless) {
|
||||||
|
const anthropicKey = process.env['MOSAIC_ANTHROPIC_API_KEY'] ?? '';
|
||||||
|
const openaiKey = process.env['MOSAIC_OPENAI_API_KEY'] ?? '';
|
||||||
|
const key = anthropicKey || openaiKey;
|
||||||
|
state.providerKey = key || undefined;
|
||||||
|
state.providerType = detectProviderType(key);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
p.separator();
|
||||||
|
p.note(
|
||||||
|
'Configure your LLM provider so the agent has a brain.\n' +
|
||||||
|
'Anthropic (Claude) and OpenAI are supported.\n' +
|
||||||
|
'You can skip this and add a key later via `mosaic configure`.',
|
||||||
|
'LLM Provider',
|
||||||
|
);
|
||||||
|
|
||||||
|
const key = await p.text({
|
||||||
|
message: 'API key (paste your Anthropic or OpenAI key, or press Enter to skip)',
|
||||||
|
defaultValue: '',
|
||||||
|
placeholder: 'sk-ant-api03-... or sk-...',
|
||||||
|
});
|
||||||
|
|
||||||
|
if (key) {
|
||||||
|
const provider = detectProviderType(key);
|
||||||
|
state.providerKey = key;
|
||||||
|
state.providerType = provider;
|
||||||
|
|
||||||
|
if (provider === 'anthropic') {
|
||||||
|
p.log('Detected provider: Anthropic (Claude)');
|
||||||
|
} else if (provider === 'openai') {
|
||||||
|
p.log('Detected provider: OpenAI');
|
||||||
|
} else {
|
||||||
|
p.log('Provider auto-detection failed. Key will be stored as ANTHROPIC_API_KEY.');
|
||||||
|
state.providerType = 'anthropic';
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
state.providerType = 'none';
|
||||||
|
p.log('No API key provided. You can add one later with `mosaic configure`.');
|
||||||
|
}
|
||||||
|
}
|
||||||
98
packages/mosaic/src/stages/quick-start.ts
Normal file
98
packages/mosaic/src/stages/quick-start.ts
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
import type { WizardPrompter } from '../prompter/interface.js';
|
||||||
|
import type { ConfigService } from '../config/config-service.js';
|
||||||
|
import type { WizardState } from '../types.js';
|
||||||
|
import { DEFAULTS } from '../constants.js';
|
||||||
|
import { providerSetupStage } from './provider-setup.js';
|
||||||
|
import { runtimeSetupStage } from './runtime-setup.js';
|
||||||
|
import { hooksPreviewStage } from './hooks-preview.js';
|
||||||
|
import { skillsSelectStage } from './skills-select.js';
|
||||||
|
import { finalizeStage } from './finalize.js';
|
||||||
|
import { gatewayConfigStage } from './gateway-config.js';
|
||||||
|
import { gatewayBootstrapStage } from './gateway-bootstrap.js';
|
||||||
|
|
||||||
|
export interface QuickStartOptions {
|
||||||
|
skipGateway?: boolean;
|
||||||
|
gatewayHost?: string;
|
||||||
|
gatewayPort?: number;
|
||||||
|
gatewayPortOverride?: number;
|
||||||
|
skipGatewayNpmInstall?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Quick Start path — minimal questions to get a working agent.
|
||||||
|
*
|
||||||
|
* 1. Provider API key
|
||||||
|
* 2. Admin email + password (via gateway bootstrap)
|
||||||
|
* 3. Everything else uses defaults.
|
||||||
|
*
|
||||||
|
* Target: under 90 seconds for a returning user.
|
||||||
|
*/
|
||||||
|
export async function quickStartPath(
|
||||||
|
prompter: WizardPrompter,
|
||||||
|
state: WizardState,
|
||||||
|
configService: ConfigService,
|
||||||
|
options: QuickStartOptions,
|
||||||
|
): Promise<void> {
|
||||||
|
state.mode = 'quick';
|
||||||
|
|
||||||
|
// 1. Provider setup (first question)
|
||||||
|
await providerSetupStage(prompter, state);
|
||||||
|
|
||||||
|
// Apply sensible defaults for everything else
|
||||||
|
state.soul.agentName ??= 'Mosaic';
|
||||||
|
state.soul.roleDescription ??= DEFAULTS.roleDescription;
|
||||||
|
state.soul.communicationStyle ??= 'direct';
|
||||||
|
state.user.background = DEFAULTS.background;
|
||||||
|
state.user.accessibilitySection = DEFAULTS.accessibilitySection;
|
||||||
|
state.user.personalBoundaries = DEFAULTS.personalBoundaries;
|
||||||
|
state.tools.gitProviders = [];
|
||||||
|
state.tools.credentialsLocation = DEFAULTS.credentialsLocation;
|
||||||
|
state.tools.customToolsSection = DEFAULTS.customToolsSection;
|
||||||
|
|
||||||
|
// Runtime detection (auto, no user input in quick mode)
|
||||||
|
await runtimeSetupStage(prompter, state);
|
||||||
|
|
||||||
|
// Hooks (auto-accept in quick mode for Claude)
|
||||||
|
await hooksPreviewStage(prompter, state);
|
||||||
|
|
||||||
|
// Skills (recommended set, no user input in quick mode)
|
||||||
|
await skillsSelectStage(prompter, state);
|
||||||
|
|
||||||
|
// Finalize (writes configs, links runtime assets, syncs skills)
|
||||||
|
await finalizeStage(prompter, state, configService);
|
||||||
|
|
||||||
|
// Gateway config + bootstrap
|
||||||
|
if (!options.skipGateway) {
|
||||||
|
const headlessRun = process.env['MOSAIC_ASSUME_YES'] === '1' || !process.stdin.isTTY;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const configResult = await gatewayConfigStage(prompter, state, {
|
||||||
|
host: options.gatewayHost ?? 'localhost',
|
||||||
|
defaultPort: options.gatewayPort ?? 14242,
|
||||||
|
portOverride: options.gatewayPortOverride,
|
||||||
|
skipInstall: options.skipGatewayNpmInstall,
|
||||||
|
providerKey: state.providerKey,
|
||||||
|
providerType: state.providerType ?? 'none',
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!configResult.ready || !configResult.host || !configResult.port) {
|
||||||
|
if (headlessRun) {
|
||||||
|
prompter.warn('Gateway configuration failed in headless mode — aborting wizard.');
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const bootstrapResult = await gatewayBootstrapStage(prompter, state, {
|
||||||
|
host: configResult.host,
|
||||||
|
port: configResult.port,
|
||||||
|
});
|
||||||
|
if (!bootstrapResult.completed) {
|
||||||
|
prompter.warn('Admin bootstrap failed — aborting wizard.');
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
prompter.warn(`Gateway setup failed: ${err instanceof Error ? err.message : String(err)}`);
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
118
packages/mosaic/src/stages/wizard-menu.spec.ts
Normal file
118
packages/mosaic/src/stages/wizard-menu.spec.ts
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
import { describe, it, expect, afterEach } from 'vitest';
|
||||||
|
import type { MenuSection } from '../types.js';
|
||||||
|
import { detectProviderType, INTENT_PRESETS } from '../constants.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for the drill-down menu system and its supporting utilities.
|
||||||
|
*
|
||||||
|
* The menu loop itself is in wizard.ts and is hard to unit test in isolation
|
||||||
|
* because it orchestrates many async stages. These tests verify the building
|
||||||
|
* blocks: provider detection, intent presets, and the WizardState shape.
|
||||||
|
*/
|
||||||
|
|
||||||
|
describe('detectProviderType', () => {
|
||||||
|
it('detects Anthropic from sk-ant- prefix', () => {
|
||||||
|
expect(detectProviderType('sk-ant-api03-abc123')).toBe('anthropic');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('detects OpenAI from sk- prefix', () => {
|
||||||
|
expect(detectProviderType('sk-proj-abc123')).toBe('openai');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns none for empty string', () => {
|
||||||
|
expect(detectProviderType('')).toBe('none');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns none for unrecognized prefix', () => {
|
||||||
|
expect(detectProviderType('gsk_abc123')).toBe('none');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('INTENT_PRESETS', () => {
|
||||||
|
it('has all expected intent categories', () => {
|
||||||
|
expect(Object.keys(INTENT_PRESETS)).toEqual(
|
||||||
|
expect.arrayContaining([
|
||||||
|
'general',
|
||||||
|
'software-dev',
|
||||||
|
'devops',
|
||||||
|
'research',
|
||||||
|
'content',
|
||||||
|
'custom',
|
||||||
|
]),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('each preset has label, hint, and suggestedName', () => {
|
||||||
|
for (const [key, preset] of Object.entries(INTENT_PRESETS)) {
|
||||||
|
expect(preset.label, `${key}.label`).toBeTruthy();
|
||||||
|
expect(preset.hint, `${key}.hint`).toBeTruthy();
|
||||||
|
expect(preset.suggestedName, `${key}.suggestedName`).toBeTruthy();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('maps software-dev to Forge', () => {
|
||||||
|
expect(INTENT_PRESETS['software-dev']?.suggestedName).toBe('Forge');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('maps devops to Sentinel', () => {
|
||||||
|
expect(INTENT_PRESETS['devops']?.suggestedName).toBe('Sentinel');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('WizardState completedSections', () => {
|
||||||
|
it('tracks completed sections as a Set', () => {
|
||||||
|
const completed = new Set<MenuSection>();
|
||||||
|
completed.add('providers');
|
||||||
|
completed.add('identity');
|
||||||
|
|
||||||
|
expect(completed.has('providers')).toBe(true);
|
||||||
|
expect(completed.has('identity')).toBe(true);
|
||||||
|
expect(completed.has('skills')).toBe(false);
|
||||||
|
expect(completed.size).toBe(2);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('headless backward compat', () => {
|
||||||
|
const originalEnv = { ...process.env };
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
process.env = { ...originalEnv };
|
||||||
|
});
|
||||||
|
|
||||||
|
it('MOSAIC_ASSUME_YES=1 triggers headless path', () => {
|
||||||
|
process.env['MOSAIC_ASSUME_YES'] = '1';
|
||||||
|
const isHeadless = process.env['MOSAIC_ASSUME_YES'] === '1' || !process.stdin.isTTY;
|
||||||
|
expect(isHeadless).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('non-TTY triggers headless path', () => {
|
||||||
|
delete process.env['MOSAIC_ASSUME_YES'];
|
||||||
|
// In test environments, process.stdin.isTTY is typically undefined (falsy)
|
||||||
|
const isHeadless = process.env['MOSAIC_ASSUME_YES'] === '1' || !process.stdin.isTTY;
|
||||||
|
expect(isHeadless).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('all headless env vars are recognized', () => {
|
||||||
|
// This test documents the expected env vars for headless installs.
|
||||||
|
const headlessVars = [
|
||||||
|
'MOSAIC_ASSUME_YES',
|
||||||
|
'MOSAIC_ADMIN_NAME',
|
||||||
|
'MOSAIC_ADMIN_EMAIL',
|
||||||
|
'MOSAIC_ADMIN_PASSWORD',
|
||||||
|
'MOSAIC_GATEWAY_PORT',
|
||||||
|
'MOSAIC_HOSTNAME',
|
||||||
|
'MOSAIC_CORS_ORIGIN',
|
||||||
|
'MOSAIC_STORAGE_TIER',
|
||||||
|
'MOSAIC_DATABASE_URL',
|
||||||
|
'MOSAIC_VALKEY_URL',
|
||||||
|
'MOSAIC_ANTHROPIC_API_KEY',
|
||||||
|
'MOSAIC_AGENT_NAME',
|
||||||
|
'MOSAIC_AGENT_INTENT',
|
||||||
|
];
|
||||||
|
|
||||||
|
// Just verify none of them throw when accessed
|
||||||
|
for (const v of headlessVars) {
|
||||||
|
expect(() => process.env[v]).not.toThrow();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -3,6 +3,19 @@ export type InstallAction = 'fresh' | 'keep' | 'reconfigure' | 'reset';
|
|||||||
export type CommunicationStyle = 'direct' | 'friendly' | 'formal';
|
export type CommunicationStyle = 'direct' | 'friendly' | 'formal';
|
||||||
export type RuntimeName = 'claude' | 'codex' | 'opencode' | 'pi';
|
export type RuntimeName = 'claude' | 'codex' | 'opencode' | 'pi';
|
||||||
|
|
||||||
|
export type MenuSection =
|
||||||
|
| 'quick-start'
|
||||||
|
| 'providers'
|
||||||
|
| 'identity'
|
||||||
|
| 'skills'
|
||||||
|
| 'gateway'
|
||||||
|
| 'advanced'
|
||||||
|
| 'finish';
|
||||||
|
|
||||||
|
export type AgentIntent = 'general' | 'software-dev' | 'devops' | 'research' | 'content' | 'custom';
|
||||||
|
|
||||||
|
export type ProviderType = 'anthropic' | 'openai' | 'none';
|
||||||
|
|
||||||
export interface SoulConfig {
|
export interface SoulConfig {
|
||||||
agentName?: string;
|
agentName?: string;
|
||||||
roleDescription?: string;
|
roleDescription?: string;
|
||||||
@@ -86,4 +99,12 @@ export interface WizardState {
|
|||||||
selectedSkills: string[];
|
selectedSkills: string[];
|
||||||
hooks?: HooksState;
|
hooks?: HooksState;
|
||||||
gateway?: GatewayState;
|
gateway?: GatewayState;
|
||||||
|
/** Tracks which menu sections have been completed in drill-down mode. */
|
||||||
|
completedSections?: Set<MenuSection>;
|
||||||
|
/** The user's chosen agent intent category. */
|
||||||
|
agentIntent?: AgentIntent;
|
||||||
|
/** The LLM provider API key entered during setup. */
|
||||||
|
providerKey?: string;
|
||||||
|
/** Detected provider type based on API key prefix. */
|
||||||
|
providerType?: ProviderType;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
import type { WizardPrompter } from './prompter/interface.js';
|
import type { WizardPrompter } from './prompter/interface.js';
|
||||||
import type { ConfigService } from './config/config-service.js';
|
import type { ConfigService } from './config/config-service.js';
|
||||||
import type { WizardState } from './types.js';
|
import type { MenuSection, WizardState } from './types.js';
|
||||||
import { welcomeStage } from './stages/welcome.js';
|
import { welcomeStage } from './stages/welcome.js';
|
||||||
import { detectInstallStage } from './stages/detect-install.js';
|
import { detectInstallStage } from './stages/detect-install.js';
|
||||||
import { modeSelectStage } from './stages/mode-select.js';
|
|
||||||
import { soulSetupStage } from './stages/soul-setup.js';
|
import { soulSetupStage } from './stages/soul-setup.js';
|
||||||
import { userSetupStage } from './stages/user-setup.js';
|
import { userSetupStage } from './stages/user-setup.js';
|
||||||
import { toolsSetupStage } from './stages/tools-setup.js';
|
import { toolsSetupStage } from './stages/tools-setup.js';
|
||||||
@@ -13,6 +12,10 @@ import { skillsSelectStage } from './stages/skills-select.js';
|
|||||||
import { finalizeStage } from './stages/finalize.js';
|
import { finalizeStage } from './stages/finalize.js';
|
||||||
import { gatewayConfigStage } from './stages/gateway-config.js';
|
import { gatewayConfigStage } from './stages/gateway-config.js';
|
||||||
import { gatewayBootstrapStage } from './stages/gateway-bootstrap.js';
|
import { gatewayBootstrapStage } from './stages/gateway-bootstrap.js';
|
||||||
|
import { providerSetupStage } from './stages/provider-setup.js';
|
||||||
|
import { agentIntentStage } from './stages/agent-intent.js';
|
||||||
|
import { quickStartPath } from './stages/quick-start.js';
|
||||||
|
import { DEFAULTS } from './constants.js';
|
||||||
|
|
||||||
export interface WizardOptions {
|
export interface WizardOptions {
|
||||||
mosaicHome: string;
|
mosaicHome: string;
|
||||||
@@ -54,6 +57,7 @@ export async function runWizard(options: WizardOptions): Promise<void> {
|
|||||||
tools: {},
|
tools: {},
|
||||||
runtimes: { detected: [], mcpConfigured: false },
|
runtimes: { detected: [], mcpConfigured: false },
|
||||||
selectedSkills: [],
|
selectedSkills: [],
|
||||||
|
completedSections: new Set<MenuSection>(),
|
||||||
};
|
};
|
||||||
|
|
||||||
// Apply CLI overrides (strip undefined values)
|
// Apply CLI overrides (strip undefined values)
|
||||||
@@ -90,55 +94,304 @@ export async function runWizard(options: WizardOptions): Promise<void> {
|
|||||||
// Stage 2: Existing Install Detection
|
// Stage 2: Existing Install Detection
|
||||||
await detectInstallStage(prompter, state, configService);
|
await detectInstallStage(prompter, state, configService);
|
||||||
|
|
||||||
// Stage 3: Quick Start vs Advanced (skip if keeping existing)
|
// ── Headless bypass ────────────────────────────────────────────────────────
|
||||||
if (state.installAction === 'fresh' || state.installAction === 'reset') {
|
// When MOSAIC_ASSUME_YES=1 or no TTY, run the linear headless path.
|
||||||
await modeSelectStage(prompter, state);
|
// This preserves full backward compatibility with tools/install.sh --yes.
|
||||||
} else if (state.installAction === 'reconfigure') {
|
const headlessRun = process.env['MOSAIC_ASSUME_YES'] === '1' || !process.stdin.isTTY;
|
||||||
state.mode = 'advanced';
|
if (headlessRun) {
|
||||||
|
await runHeadlessPath(prompter, state, configService, options);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Stage 4: SOUL.md
|
// ── Interactive: Main Menu ─────────────────────────────────────────────────
|
||||||
|
if (state.installAction === 'fresh' || state.installAction === 'reset') {
|
||||||
|
await runMenuLoop(prompter, state, configService, options);
|
||||||
|
} else if (state.installAction === 'reconfigure') {
|
||||||
|
state.mode = 'advanced';
|
||||||
|
await runMenuLoop(prompter, state, configService, options);
|
||||||
|
} else {
|
||||||
|
// 'keep' — skip identity setup, go straight to finalize + gateway
|
||||||
|
await runKeepPath(prompter, state, configService, options);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Menu-driven interactive flow ────────────────────────────────────────────
|
||||||
|
|
||||||
|
type MenuChoice =
|
||||||
|
| 'quick-start'
|
||||||
|
| 'providers'
|
||||||
|
| 'identity'
|
||||||
|
| 'skills'
|
||||||
|
| 'gateway-config'
|
||||||
|
| 'advanced'
|
||||||
|
| 'finish';
|
||||||
|
|
||||||
|
function menuLabel(section: MenuChoice, completed: Set<MenuSection>): string {
|
||||||
|
const labels: Record<MenuChoice, string> = {
|
||||||
|
'quick-start': 'Quick Start',
|
||||||
|
providers: 'Providers',
|
||||||
|
identity: 'Agent Identity',
|
||||||
|
skills: 'Skills',
|
||||||
|
'gateway-config': 'Gateway',
|
||||||
|
advanced: 'Advanced',
|
||||||
|
finish: 'Finish & Apply',
|
||||||
|
};
|
||||||
|
const base = labels[section];
|
||||||
|
const sectionKey: MenuSection =
|
||||||
|
section === 'gateway-config' ? 'gateway' : (section as MenuSection);
|
||||||
|
if (completed.has(sectionKey)) {
|
||||||
|
return `${base} [done]`;
|
||||||
|
}
|
||||||
|
return base;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function runMenuLoop(
|
||||||
|
prompter: WizardPrompter,
|
||||||
|
state: WizardState,
|
||||||
|
configService: ConfigService,
|
||||||
|
options: WizardOptions,
|
||||||
|
): Promise<void> {
|
||||||
|
const completed = state.completedSections!;
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
const choice = await prompter.select<MenuChoice>({
|
||||||
|
message: 'What would you like to configure?',
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
value: 'quick-start',
|
||||||
|
label: menuLabel('quick-start', completed),
|
||||||
|
hint: 'Recommended defaults, minimal questions',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 'providers',
|
||||||
|
label: menuLabel('providers', completed),
|
||||||
|
hint: 'LLM API keys (Anthropic, OpenAI)',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 'identity',
|
||||||
|
label: menuLabel('identity', completed),
|
||||||
|
hint: 'Agent name, intent, persona',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 'skills',
|
||||||
|
label: menuLabel('skills', completed),
|
||||||
|
hint: 'Install agent skills',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 'gateway-config',
|
||||||
|
label: menuLabel('gateway-config', completed),
|
||||||
|
hint: 'Port, storage, database',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 'advanced',
|
||||||
|
label: menuLabel('advanced', completed),
|
||||||
|
hint: 'SOUL.md, USER.md, TOOLS.md, runtimes, hooks',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 'finish',
|
||||||
|
label: menuLabel('finish', completed),
|
||||||
|
hint: 'Write configs and start gateway',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
switch (choice) {
|
||||||
|
case 'quick-start':
|
||||||
|
await quickStartPath(prompter, state, configService, options);
|
||||||
|
return; // Quick start is a complete flow — exit menu
|
||||||
|
|
||||||
|
case 'providers':
|
||||||
|
await providerSetupStage(prompter, state);
|
||||||
|
completed.add('providers');
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'identity':
|
||||||
|
await agentIntentStage(prompter, state);
|
||||||
|
completed.add('identity');
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'skills':
|
||||||
|
await skillsSelectStage(prompter, state);
|
||||||
|
completed.add('skills');
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'gateway-config':
|
||||||
|
// Gateway config is handled during Finish — mark as "configured"
|
||||||
|
// after user reviews settings.
|
||||||
|
await runGatewaySubMenu(prompter, state, options);
|
||||||
|
completed.add('gateway');
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'advanced':
|
||||||
|
await runAdvancedSubMenu(prompter, state);
|
||||||
|
completed.add('advanced');
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'finish':
|
||||||
|
await runFinishPath(prompter, state, configService, options);
|
||||||
|
return; // Done
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Gateway sub-menu ─────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
async function runGatewaySubMenu(
|
||||||
|
prompter: WizardPrompter,
|
||||||
|
state: WizardState,
|
||||||
|
_options: WizardOptions,
|
||||||
|
): Promise<void> {
|
||||||
|
prompter.note(
|
||||||
|
'Gateway settings will be applied when you select "Finish & Apply".\n' +
|
||||||
|
'Configure the settings you want to customize here.',
|
||||||
|
'Gateway Configuration',
|
||||||
|
);
|
||||||
|
|
||||||
|
// For now, just let them know defaults will be used and they can
|
||||||
|
// override during finish. The actual gateway config stage runs
|
||||||
|
// during Finish & Apply. This menu item exists so users know
|
||||||
|
// the gateway is part of the wizard.
|
||||||
|
const port = await prompter.text({
|
||||||
|
message: 'Gateway port',
|
||||||
|
initialValue: (_options.gatewayPort ?? 14242).toString(),
|
||||||
|
defaultValue: (_options.gatewayPort ?? 14242).toString(),
|
||||||
|
validate: (v) => {
|
||||||
|
const n = parseInt(v, 10);
|
||||||
|
if (Number.isNaN(n) || n < 1 || n > 65535) return 'Port must be 1-65535';
|
||||||
|
return undefined;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// Store for later use in the gateway config stage
|
||||||
|
_options.gatewayPort = parseInt(port, 10);
|
||||||
|
prompter.log(`Gateway port set to ${port}. Will be applied during Finish & Apply.`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Advanced sub-menu ────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
async function runAdvancedSubMenu(prompter: WizardPrompter, state: WizardState): Promise<void> {
|
||||||
|
state.mode = 'advanced';
|
||||||
|
|
||||||
|
// Run the detailed setup stages
|
||||||
await soulSetupStage(prompter, state);
|
await soulSetupStage(prompter, state);
|
||||||
|
|
||||||
// Stage 5: USER.md
|
|
||||||
await userSetupStage(prompter, state);
|
await userSetupStage(prompter, state);
|
||||||
|
|
||||||
// Stage 6: TOOLS.md
|
|
||||||
await toolsSetupStage(prompter, state);
|
await toolsSetupStage(prompter, state);
|
||||||
|
|
||||||
// Stage 7: Runtime Detection & Installation
|
|
||||||
await runtimeSetupStage(prompter, state);
|
await runtimeSetupStage(prompter, state);
|
||||||
|
|
||||||
// Stage 8: Hooks preview (Claude only — skipped if Claude not detected)
|
|
||||||
await hooksPreviewStage(prompter, state);
|
await hooksPreviewStage(prompter, state);
|
||||||
|
}
|
||||||
|
|
||||||
// Stage 9: Skills Selection
|
// ── Finish & Apply ──────────────────────────────────────────────────────────
|
||||||
await skillsSelectStage(prompter, state);
|
|
||||||
|
|
||||||
// Stage 10: Finalize (writes configs, links runtime assets, runs doctor)
|
async function runFinishPath(
|
||||||
|
prompter: WizardPrompter,
|
||||||
|
state: WizardState,
|
||||||
|
configService: ConfigService,
|
||||||
|
options: WizardOptions,
|
||||||
|
): Promise<void> {
|
||||||
|
// Apply defaults for anything not explicitly configured
|
||||||
|
state.soul.agentName ??= 'Mosaic';
|
||||||
|
state.soul.roleDescription ??= DEFAULTS.roleDescription;
|
||||||
|
state.soul.communicationStyle ??= 'direct';
|
||||||
|
state.user.background ??= DEFAULTS.background;
|
||||||
|
state.user.accessibilitySection ??= DEFAULTS.accessibilitySection;
|
||||||
|
state.user.personalBoundaries ??= DEFAULTS.personalBoundaries;
|
||||||
|
state.tools.gitProviders ??= [];
|
||||||
|
state.tools.credentialsLocation ??= DEFAULTS.credentialsLocation;
|
||||||
|
state.tools.customToolsSection ??= DEFAULTS.customToolsSection;
|
||||||
|
|
||||||
|
// Runtime detection if not already done
|
||||||
|
if (state.runtimes.detected.length === 0 && !state.completedSections?.has('advanced')) {
|
||||||
|
await runtimeSetupStage(prompter, state);
|
||||||
|
await hooksPreviewStage(prompter, state);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Skills defaults if not already configured
|
||||||
|
if (!state.completedSections?.has('skills')) {
|
||||||
|
await skillsSelectStage(prompter, state);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Finalize (writes configs, links runtime assets, syncs skills)
|
||||||
await finalizeStage(prompter, state, configService);
|
await finalizeStage(prompter, state, configService);
|
||||||
|
|
||||||
// Stages 11 & 12: Gateway config + admin bootstrap.
|
// Gateway stages
|
||||||
// The unified first-run flow runs these as terminal stages so the user
|
|
||||||
// goes from "welcome" through "admin user created" in a single cohesive
|
|
||||||
// experience. Callers that only want the framework portion pass
|
|
||||||
// `skipGateway: true`.
|
|
||||||
if (!options.skipGateway) {
|
if (!options.skipGateway) {
|
||||||
const headlessRun = process.env['MOSAIC_ASSUME_YES'] === '1' || !process.stdin.isTTY;
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const configResult = await gatewayConfigStage(prompter, state, {
|
const configResult = await gatewayConfigStage(prompter, state, {
|
||||||
host: options.gatewayHost ?? 'localhost',
|
host: options.gatewayHost ?? 'localhost',
|
||||||
defaultPort: options.gatewayPort ?? 14242,
|
defaultPort: options.gatewayPort ?? 14242,
|
||||||
portOverride: options.gatewayPortOverride,
|
portOverride: options.gatewayPortOverride,
|
||||||
skipInstall: options.skipGatewayNpmInstall,
|
skipInstall: options.skipGatewayNpmInstall,
|
||||||
|
providerKey: state.providerKey,
|
||||||
|
providerType: state.providerType ?? 'none',
|
||||||
|
});
|
||||||
|
|
||||||
|
if (configResult.ready && configResult.host && configResult.port) {
|
||||||
|
const bootstrapResult = await gatewayBootstrapStage(prompter, state, {
|
||||||
|
host: configResult.host,
|
||||||
|
port: configResult.port,
|
||||||
|
});
|
||||||
|
if (!bootstrapResult.completed) {
|
||||||
|
prompter.warn('Admin bootstrap failed — aborting wizard.');
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
prompter.warn(`Gateway setup failed: ${err instanceof Error ? err.message : String(err)}`);
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Headless linear path (backward compat) ──────────────────────────────────
|
||||||
|
|
||||||
|
async function runHeadlessPath(
|
||||||
|
prompter: WizardPrompter,
|
||||||
|
state: WizardState,
|
||||||
|
configService: ConfigService,
|
||||||
|
options: WizardOptions,
|
||||||
|
): Promise<void> {
|
||||||
|
// Provider setup from env vars
|
||||||
|
await providerSetupStage(prompter, state);
|
||||||
|
|
||||||
|
// Agent intent from env vars
|
||||||
|
await agentIntentStage(prompter, state);
|
||||||
|
|
||||||
|
// SOUL.md
|
||||||
|
await soulSetupStage(prompter, state);
|
||||||
|
|
||||||
|
// USER.md
|
||||||
|
await userSetupStage(prompter, state);
|
||||||
|
|
||||||
|
// TOOLS.md
|
||||||
|
await toolsSetupStage(prompter, state);
|
||||||
|
|
||||||
|
// Runtime Detection
|
||||||
|
await runtimeSetupStage(prompter, state);
|
||||||
|
|
||||||
|
// Hooks
|
||||||
|
await hooksPreviewStage(prompter, state);
|
||||||
|
|
||||||
|
// Skills
|
||||||
|
await skillsSelectStage(prompter, state);
|
||||||
|
|
||||||
|
// Finalize
|
||||||
|
await finalizeStage(prompter, state, configService);
|
||||||
|
|
||||||
|
// Gateway stages
|
||||||
|
if (!options.skipGateway) {
|
||||||
|
try {
|
||||||
|
const configResult = await gatewayConfigStage(prompter, state, {
|
||||||
|
host: options.gatewayHost ?? 'localhost',
|
||||||
|
defaultPort: options.gatewayPort ?? 14242,
|
||||||
|
portOverride: options.gatewayPortOverride,
|
||||||
|
skipInstall: options.skipGatewayNpmInstall,
|
||||||
|
providerKey: state.providerKey,
|
||||||
|
providerType: state.providerType ?? 'none',
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!configResult.ready || !configResult.host || !configResult.port) {
|
if (!configResult.ready || !configResult.host || !configResult.port) {
|
||||||
if (headlessRun) {
|
prompter.warn('Gateway configuration failed in headless mode — aborting wizard.');
|
||||||
prompter.warn('Gateway configuration failed in headless mode — aborting wizard.');
|
process.exit(1);
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
const bootstrapResult = await gatewayBootstrapStage(prompter, state, {
|
const bootstrapResult = await gatewayBootstrapStage(prompter, state, {
|
||||||
host: configResult.host,
|
host: configResult.host,
|
||||||
@@ -150,12 +403,53 @@ export async function runWizard(options: WizardOptions): Promise<void> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
// Stages normally return structured `ready: false` results for
|
prompter.warn(`Gateway setup failed: ${err instanceof Error ? err.message : String(err)}`);
|
||||||
// expected failures. Anything that reaches here is an unexpected
|
throw err;
|
||||||
// runtime error — render a concise warning for UX AND re-throw so
|
}
|
||||||
// the CLI (and `tools/install.sh` auto-launch) sees a non-zero exit.
|
}
|
||||||
// Swallowing here would let headless installs report success even
|
}
|
||||||
// when the gateway stage crashed.
|
|
||||||
|
// ── Keep path (preserve existing identity) ──────────────────────────────────
|
||||||
|
|
||||||
|
async function runKeepPath(
|
||||||
|
prompter: WizardPrompter,
|
||||||
|
state: WizardState,
|
||||||
|
configService: ConfigService,
|
||||||
|
options: WizardOptions,
|
||||||
|
): Promise<void> {
|
||||||
|
// Runtime detection
|
||||||
|
await runtimeSetupStage(prompter, state);
|
||||||
|
|
||||||
|
// Hooks
|
||||||
|
await hooksPreviewStage(prompter, state);
|
||||||
|
|
||||||
|
// Skills
|
||||||
|
await skillsSelectStage(prompter, state);
|
||||||
|
|
||||||
|
// Finalize
|
||||||
|
await finalizeStage(prompter, state, configService);
|
||||||
|
|
||||||
|
// Gateway stages
|
||||||
|
if (!options.skipGateway) {
|
||||||
|
try {
|
||||||
|
const configResult = await gatewayConfigStage(prompter, state, {
|
||||||
|
host: options.gatewayHost ?? 'localhost',
|
||||||
|
defaultPort: options.gatewayPort ?? 14242,
|
||||||
|
portOverride: options.gatewayPortOverride,
|
||||||
|
skipInstall: options.skipGatewayNpmInstall,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (configResult.ready && configResult.host && configResult.port) {
|
||||||
|
const bootstrapResult = await gatewayBootstrapStage(prompter, state, {
|
||||||
|
host: configResult.host,
|
||||||
|
port: configResult.port,
|
||||||
|
});
|
||||||
|
if (!bootstrapResult.completed) {
|
||||||
|
prompter.warn('Admin bootstrap failed — aborting wizard.');
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
prompter.warn(`Gateway setup failed: ${err instanceof Error ? err.message : String(err)}`);
|
prompter.warn(`Gateway setup failed: ${err instanceof Error ? err.message : String(err)}`);
|
||||||
throw err;
|
throw err;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/prdy"
|
"directory": "packages/prdy"
|
||||||
},
|
},
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.3",
|
"version": "0.0.3",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/quality-rails"
|
"directory": "packages/quality-rails"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.4",
|
"version": "0.0.4",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/queue"
|
"directory": "packages/queue"
|
||||||
},
|
},
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.4",
|
"version": "0.0.4",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/storage"
|
"directory": "packages/storage"
|
||||||
},
|
},
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "packages/types"
|
"directory": "packages/types"
|
||||||
},
|
},
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "plugins/discord"
|
"directory": "plugins/discord"
|
||||||
},
|
},
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "plugins/macp"
|
"directory": "plugins/macp"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "plugins/mosaic-framework"
|
"directory": "plugins/mosaic-framework"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"version": "0.0.2",
|
"version": "0.0.2",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.mosaicstack.dev/mosaicstack/mosaic-stack.git",
|
"url": "https://git.mosaicstack.dev/mosaicstack/stack.git",
|
||||||
"directory": "plugins/telegram"
|
"directory": "plugins/telegram"
|
||||||
},
|
},
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
|
|||||||
@@ -5,11 +5,11 @@
|
|||||||
# 1. Mosaic framework → ~/.config/mosaic/ (bash launcher, guides, runtime configs, tools)
|
# 1. Mosaic framework → ~/.config/mosaic/ (bash launcher, guides, runtime configs, tools)
|
||||||
# 2. @mosaicstack/mosaic (npm) → ~/.npm-global/ (CLI, TUI, gateway client, wizard)
|
# 2. @mosaicstack/mosaic (npm) → ~/.npm-global/ (CLI, TUI, gateway client, wizard)
|
||||||
#
|
#
|
||||||
# Remote install (recommended):
|
# Quick: curl -fsSL https://mosaicstack.dev/install.sh | bash
|
||||||
# bash <(curl -fsSL https://git.mosaicstack.dev/mosaicstack/mosaic-stack/raw/branch/main/tools/install.sh)
|
# Direct: bash <(curl -fsSL https://git.mosaicstack.dev/mosaicstack/stack/raw/branch/main/tools/install.sh)
|
||||||
#
|
#
|
||||||
# Remote install (alternative — use -s -- to pass flags):
|
# Remote install (alternative — use -s -- to pass flags):
|
||||||
# curl -fsSL https://git.mosaicstack.dev/mosaicstack/mosaic-stack/raw/branch/main/tools/install.sh | bash -s --
|
# curl -fsSL https://git.mosaicstack.dev/mosaicstack/stack/raw/branch/main/tools/install.sh | bash -s --
|
||||||
#
|
#
|
||||||
# Flags:
|
# Flags:
|
||||||
# --check Version check only, no install
|
# --check Version check only, no install
|
||||||
@@ -69,7 +69,7 @@ REGISTRY="${MOSAIC_REGISTRY:-https://git.mosaicstack.dev/api/packages/mosaicstac
|
|||||||
SCOPE="${MOSAIC_SCOPE:-@mosaicstack}"
|
SCOPE="${MOSAIC_SCOPE:-@mosaicstack}"
|
||||||
PREFIX="${MOSAIC_PREFIX:-$HOME/.npm-global}"
|
PREFIX="${MOSAIC_PREFIX:-$HOME/.npm-global}"
|
||||||
CLI_PKG="${SCOPE}/mosaic"
|
CLI_PKG="${SCOPE}/mosaic"
|
||||||
REPO_BASE="https://git.mosaicstack.dev/mosaicstack/mosaic-stack"
|
REPO_BASE="https://git.mosaicstack.dev/mosaicstack/stack"
|
||||||
ARCHIVE_URL="${REPO_BASE}/archive/${GIT_REF}.tar.gz"
|
ARCHIVE_URL="${REPO_BASE}/archive/${GIT_REF}.tar.gz"
|
||||||
|
|
||||||
# ─── uninstall path ───────────────────────────────────────────────────────────
|
# ─── uninstall path ───────────────────────────────────────────────────────────
|
||||||
|
|||||||
Reference in New Issue
Block a user