mirror of
https://github.com/qwibitai/nanoclaw.git
synced 2026-06-04 10:14:47 +08:00
6e0d742a7f
Wraps the scripted setup flow in a branded, friendly UI. Each step runs
under a clack spinner with elapsed time; child stdout/stderr is captured
quietly and dumped only on failure. Interactive children (token paste,
Anthropic OAuth) bypass the spinner and inherit the TTY.
- intro: NanoClaw wordmark + brand-cyan accent chip, truecolor with
kleur fallback and NO_COLOR / non-TTY awareness
- pair-telegram: emits PAIR_TELEGRAM_CODE / _ATTEMPT status blocks only;
auto.ts renders clack notes + "received X — doesn't match" checkpoints
- streaming status-block parser handles mid-step events without waiting
for the child to exit
- terminal-block detection now finds any block with a STATUS field
(handles MOUNTS emitting CONFIGURE_MOUNTS, etc.) and treats 'skipped'
as a success variant with an optional friendlier label
Also fixes a latent bash bug where `$VAR…` (unbraced followed by a
multi-byte Unicode character) pulled ellipsis bytes into the variable
name lookup and tripped `set -u`. Braced `${VAR}` in add-telegram.sh
and register-claude-token.sh.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
117 lines
3.9 KiB
TypeScript
117 lines
3.9 KiB
TypeScript
/**
|
|
* Step: pair-telegram — issue a one-time pairing code and wait for the
|
|
* operator to send the code from the chat they want to register.
|
|
*
|
|
* Emits machine-readable status blocks only. The parent driver
|
|
* (`setup:auto`) renders the code / attempt / success UI with clack. Running
|
|
* this step directly will look sparse — that's intentional.
|
|
*
|
|
* Blocks emitted:
|
|
* PAIR_TELEGRAM_CODE { CODE, REASON=initial|regenerated }
|
|
* PAIR_TELEGRAM_ATTEMPT { CANDIDATE }
|
|
* PAIR_TELEGRAM (final) { STATUS=success, CODE, INTENT, PLATFORM_ID,
|
|
* IS_GROUP, PAIRED_USER_ID }
|
|
* or { STATUS=failed, CODE, ERROR }
|
|
*
|
|
* Depends on src/channels/telegram-pairing.js, which setup/add-telegram.sh
|
|
* copies in from the `channels` branch before this step runs. setup/ is
|
|
* excluded from the host tsconfig, so this file's import resolves only at
|
|
* runtime — tsc won't complain on branches that haven't run add-telegram yet.
|
|
*/
|
|
import path from 'path';
|
|
|
|
import {
|
|
createPairing,
|
|
waitForPairing,
|
|
type PairingIntent,
|
|
} from '../src/channels/telegram-pairing.js';
|
|
import { DATA_DIR } from '../src/config.js';
|
|
import { initDb } from '../src/db/connection.js';
|
|
import { runMigrations } from '../src/db/migrations/index.js';
|
|
|
|
import { emitStatus } from './status.js';
|
|
|
|
function parseArgs(args: string[]): PairingIntent {
|
|
let intent: PairingIntent = 'main';
|
|
for (let i = 0; i < args.length; i++) {
|
|
if (args[i] === '--intent') {
|
|
const raw = args[++i] || 'main';
|
|
if (raw === 'main') {
|
|
intent = 'main';
|
|
} else if (raw.startsWith('wire-to:')) {
|
|
intent = { kind: 'wire-to', folder: raw.slice('wire-to:'.length) };
|
|
} else if (raw.startsWith('new-agent:')) {
|
|
intent = { kind: 'new-agent', folder: raw.slice('new-agent:'.length) };
|
|
} else {
|
|
throw new Error(`Unknown intent: ${raw}`);
|
|
}
|
|
}
|
|
}
|
|
return intent;
|
|
}
|
|
|
|
function intentToString(intent: PairingIntent): string {
|
|
if (intent === 'main') return 'main';
|
|
return `${intent.kind}:${intent.folder}`;
|
|
}
|
|
|
|
export async function run(args: string[]): Promise<void> {
|
|
const intent = parseArgs(args);
|
|
|
|
// Pairing stores state under DATA_DIR; the DB isn't strictly needed for the
|
|
// pairing primitive itself, but the inbound interceptor running inside the
|
|
// live service needs migrations applied. Touch it here so a fresh install
|
|
// doesn't fail on the first code match.
|
|
const db = initDb(path.join(DATA_DIR, 'v2.db'));
|
|
runMigrations(db);
|
|
|
|
const MAX_REGENERATIONS = 5;
|
|
let record = await createPairing(intent);
|
|
emitStatus('PAIR_TELEGRAM_CODE', {
|
|
CODE: record.code,
|
|
REASON: 'initial',
|
|
});
|
|
|
|
for (let regen = 0; regen <= MAX_REGENERATIONS; regen++) {
|
|
try {
|
|
const consumed = await waitForPairing(record.code, {
|
|
onAttempt: (a) => {
|
|
emitStatus('PAIR_TELEGRAM_ATTEMPT', {
|
|
CANDIDATE: a.candidate,
|
|
});
|
|
},
|
|
});
|
|
|
|
emitStatus('PAIR_TELEGRAM', {
|
|
STATUS: 'success',
|
|
CODE: record.code,
|
|
INTENT: intentToString(consumed.intent),
|
|
PLATFORM_ID: consumed.consumed!.platformId,
|
|
IS_GROUP: consumed.consumed!.isGroup,
|
|
PAIRED_USER_ID: consumed.consumed!.adminUserId
|
|
? `telegram:${consumed.consumed!.adminUserId}`
|
|
: '',
|
|
});
|
|
return;
|
|
} catch (err) {
|
|
const message = err instanceof Error ? err.message : String(err);
|
|
const invalidated = /invalidated by wrong code/.test(message);
|
|
if (invalidated && regen < MAX_REGENERATIONS) {
|
|
record = await createPairing(intent);
|
|
emitStatus('PAIR_TELEGRAM_CODE', {
|
|
CODE: record.code,
|
|
REASON: 'regenerated',
|
|
});
|
|
continue;
|
|
}
|
|
const reason = invalidated ? 'max-regenerations-exceeded' : message;
|
|
emitStatus('PAIR_TELEGRAM', {
|
|
STATUS: 'failed',
|
|
CODE: record.code,
|
|
ERROR: reason,
|
|
});
|
|
process.exit(2);
|
|
}
|
|
}
|
|
}
|