Setup scripts are standalone CLI tools run via tsx with no runtime imports from the main app. Moving them out of src/ excludes them from the tsc build output and reduces the compiled bundle size. - git mv src/setup/ setup/ - Fix imports to use ../src/logger.js and ../src/config.js - Update package.json, vitest.config.ts, SKILL.md references - Fix platform tests to be cross-platform (macOS + Linux) Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
148 lines
4.5 KiB
TypeScript
148 lines
4.5 KiB
TypeScript
/**
|
|
* Step: register — Write channel registration config, create group folders.
|
|
* Replaces 06-register-channel.sh
|
|
*
|
|
* Fixes: SQL injection (parameterized queries), sed -i '' (uses fs directly).
|
|
*/
|
|
import fs from 'fs';
|
|
import path from 'path';
|
|
|
|
import Database from 'better-sqlite3';
|
|
|
|
import { STORE_DIR } from '../src/config.js';
|
|
import { logger } from '../src/logger.js';
|
|
import { emitStatus } from './status.js';
|
|
|
|
interface RegisterArgs {
|
|
jid: string;
|
|
name: string;
|
|
trigger: string;
|
|
folder: string;
|
|
requiresTrigger: boolean;
|
|
assistantName: string;
|
|
}
|
|
|
|
function parseArgs(args: string[]): RegisterArgs {
|
|
const result: RegisterArgs = {
|
|
jid: '',
|
|
name: '',
|
|
trigger: '',
|
|
folder: '',
|
|
requiresTrigger: true,
|
|
assistantName: 'Andy',
|
|
};
|
|
|
|
for (let i = 0; i < args.length; i++) {
|
|
switch (args[i]) {
|
|
case '--jid': result.jid = args[++i] || ''; break;
|
|
case '--name': result.name = args[++i] || ''; break;
|
|
case '--trigger': result.trigger = args[++i] || ''; break;
|
|
case '--folder': result.folder = args[++i] || ''; break;
|
|
case '--no-trigger-required': result.requiresTrigger = false; break;
|
|
case '--assistant-name': result.assistantName = args[++i] || 'Andy'; break;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
export async function run(args: string[]): Promise<void> {
|
|
const projectRoot = process.cwd();
|
|
const parsed = parseArgs(args);
|
|
|
|
if (!parsed.jid || !parsed.name || !parsed.trigger || !parsed.folder) {
|
|
emitStatus('REGISTER_CHANNEL', {
|
|
STATUS: 'failed',
|
|
ERROR: 'missing_required_args',
|
|
LOG: 'logs/setup.log',
|
|
});
|
|
process.exit(4);
|
|
}
|
|
|
|
logger.info(parsed, 'Registering channel');
|
|
|
|
// Ensure data directory exists
|
|
fs.mkdirSync(path.join(projectRoot, 'data'), { recursive: true });
|
|
|
|
// Write to SQLite using parameterized queries (no SQL injection)
|
|
const dbPath = path.join(STORE_DIR, 'messages.db');
|
|
const timestamp = new Date().toISOString();
|
|
const requiresTriggerInt = parsed.requiresTrigger ? 1 : 0;
|
|
|
|
const db = new Database(dbPath);
|
|
// Ensure schema exists
|
|
db.exec(`CREATE TABLE IF NOT EXISTS registered_groups (
|
|
jid TEXT PRIMARY KEY,
|
|
name TEXT NOT NULL,
|
|
folder TEXT NOT NULL UNIQUE,
|
|
trigger_pattern TEXT NOT NULL,
|
|
added_at TEXT NOT NULL,
|
|
container_config TEXT,
|
|
requires_trigger INTEGER DEFAULT 1
|
|
)`);
|
|
|
|
db.prepare(
|
|
`INSERT OR REPLACE INTO registered_groups
|
|
(jid, name, folder, trigger_pattern, added_at, container_config, requires_trigger)
|
|
VALUES (?, ?, ?, ?, ?, NULL, ?)`,
|
|
).run(parsed.jid, parsed.name, parsed.folder, parsed.trigger, timestamp, requiresTriggerInt);
|
|
|
|
db.close();
|
|
logger.info('Wrote registration to SQLite');
|
|
|
|
// Create group folders
|
|
fs.mkdirSync(path.join(projectRoot, 'groups', parsed.folder, 'logs'), { recursive: true });
|
|
|
|
// Update assistant name in CLAUDE.md files if different from default
|
|
let nameUpdated = false;
|
|
if (parsed.assistantName !== 'Andy') {
|
|
logger.info({ from: 'Andy', to: parsed.assistantName }, 'Updating assistant name');
|
|
|
|
const mdFiles = [
|
|
path.join(projectRoot, 'groups', 'global', 'CLAUDE.md'),
|
|
path.join(projectRoot, 'groups', 'main', 'CLAUDE.md'),
|
|
];
|
|
|
|
for (const mdFile of mdFiles) {
|
|
if (fs.existsSync(mdFile)) {
|
|
let content = fs.readFileSync(mdFile, 'utf-8');
|
|
content = content.replace(/^# Andy$/m, `# ${parsed.assistantName}`);
|
|
content = content.replace(/You are Andy/g, `You are ${parsed.assistantName}`);
|
|
fs.writeFileSync(mdFile, content);
|
|
logger.info({ file: mdFile }, 'Updated CLAUDE.md');
|
|
}
|
|
}
|
|
|
|
// Update .env
|
|
const envFile = path.join(projectRoot, '.env');
|
|
if (fs.existsSync(envFile)) {
|
|
let envContent = fs.readFileSync(envFile, 'utf-8');
|
|
if (envContent.includes('ASSISTANT_NAME=')) {
|
|
envContent = envContent.replace(
|
|
/^ASSISTANT_NAME=.*$/m,
|
|
`ASSISTANT_NAME="${parsed.assistantName}"`,
|
|
);
|
|
} else {
|
|
envContent += `\nASSISTANT_NAME="${parsed.assistantName}"`;
|
|
}
|
|
fs.writeFileSync(envFile, envContent);
|
|
} else {
|
|
fs.writeFileSync(envFile, `ASSISTANT_NAME="${parsed.assistantName}"\n`);
|
|
}
|
|
logger.info('Set ASSISTANT_NAME in .env');
|
|
nameUpdated = true;
|
|
}
|
|
|
|
emitStatus('REGISTER_CHANNEL', {
|
|
JID: parsed.jid,
|
|
NAME: parsed.name,
|
|
FOLDER: parsed.folder,
|
|
TRIGGER: parsed.trigger,
|
|
REQUIRES_TRIGGER: parsed.requiresTrigger,
|
|
ASSISTANT_NAME: parsed.assistantName,
|
|
NAME_UPDATED: nameUpdated,
|
|
STATUS: 'success',
|
|
LOG: 'logs/setup.log',
|
|
});
|
|
}
|