When a non-main group is re-registered with --is-main, the existing CLAUDE.md (copied from global template) lacked admin context. Now register.ts detects this promotion case and replaces it with the main template. Files that already contain "## Admin Context" are preserved. Adds tests for: - promoting non-main to main upgrades the template - cross-channel promotion (e.g. Telegram non-main → main) - promotion with custom assistant name - re-registration preserves user-modified main CLAUDE.md - re-registration preserves user-modified non-main CLAUDE.md Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
215 lines
6.2 KiB
TypeScript
215 lines
6.2 KiB
TypeScript
/**
|
|
* Step: register — Write channel registration config, create group folders.
|
|
*
|
|
* Accepts --channel to specify the messaging platform (whatsapp, telegram, slack, discord).
|
|
* Uses parameterized SQL queries to prevent injection.
|
|
*/
|
|
import fs from 'fs';
|
|
import path from 'path';
|
|
|
|
import { STORE_DIR } from '../src/config.ts';
|
|
import { initDatabase, setRegisteredGroup } from '../src/db.ts';
|
|
import { isValidGroupFolder } from '../src/group-folder.ts';
|
|
import { logger } from '../src/logger.ts';
|
|
import { emitStatus } from './status.ts';
|
|
|
|
interface RegisterArgs {
|
|
jid: string;
|
|
name: string;
|
|
trigger: string;
|
|
folder: string;
|
|
channel: string;
|
|
requiresTrigger: boolean;
|
|
isMain: boolean;
|
|
assistantName: string;
|
|
}
|
|
|
|
function parseArgs(args: string[]): RegisterArgs {
|
|
const result: RegisterArgs = {
|
|
jid: '',
|
|
name: '',
|
|
trigger: '',
|
|
folder: '',
|
|
channel: 'whatsapp', // backward-compat: pre-refactor installs omit --channel
|
|
requiresTrigger: true,
|
|
isMain: false,
|
|
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 '--channel':
|
|
result.channel = (args[++i] || '').toLowerCase();
|
|
break;
|
|
case '--no-trigger-required':
|
|
result.requiresTrigger = false;
|
|
break;
|
|
case '--is-main':
|
|
result.isMain = true;
|
|
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);
|
|
}
|
|
|
|
if (!isValidGroupFolder(parsed.folder)) {
|
|
emitStatus('REGISTER_CHANNEL', {
|
|
STATUS: 'failed',
|
|
ERROR: 'invalid_folder',
|
|
LOG: 'logs/setup.log',
|
|
});
|
|
process.exit(4);
|
|
}
|
|
|
|
logger.info(parsed, 'Registering channel');
|
|
|
|
// Ensure data and store directories exist (store/ may not exist on
|
|
// fresh installs that skip WhatsApp auth, which normally creates it)
|
|
fs.mkdirSync(path.join(projectRoot, 'data'), { recursive: true });
|
|
fs.mkdirSync(STORE_DIR, { recursive: true });
|
|
|
|
// Initialize database (creates schema + runs migrations)
|
|
initDatabase();
|
|
|
|
setRegisteredGroup(parsed.jid, {
|
|
name: parsed.name,
|
|
folder: parsed.folder,
|
|
trigger: parsed.trigger,
|
|
added_at: new Date().toISOString(),
|
|
requiresTrigger: parsed.requiresTrigger,
|
|
isMain: parsed.isMain,
|
|
});
|
|
|
|
logger.info('Wrote registration to SQLite');
|
|
|
|
// Create group folders
|
|
fs.mkdirSync(path.join(projectRoot, 'groups', parsed.folder, 'logs'), {
|
|
recursive: true,
|
|
});
|
|
|
|
// Create or upgrade CLAUDE.md in the group folder from the appropriate template.
|
|
// The agent runs with CWD=/workspace/group and loads CLAUDE.md from there.
|
|
const groupClaudeMdPath = path.join(
|
|
projectRoot,
|
|
'groups',
|
|
parsed.folder,
|
|
'CLAUDE.md',
|
|
);
|
|
const mainTemplatePath = path.join(projectRoot, 'groups', 'main', 'CLAUDE.md');
|
|
const globalTemplatePath = path.join(projectRoot, 'groups', 'global', 'CLAUDE.md');
|
|
const templatePath = parsed.isMain ? mainTemplatePath : globalTemplatePath;
|
|
const fileExists = fs.existsSync(groupClaudeMdPath);
|
|
|
|
// Promotion case: group was registered as non-main (got global template)
|
|
// and is now being re-registered as main. Replace with main template.
|
|
const needsPromotion =
|
|
parsed.isMain &&
|
|
fileExists &&
|
|
!fs.readFileSync(groupClaudeMdPath, 'utf-8').includes('## Admin Context');
|
|
|
|
if (!fileExists || needsPromotion) {
|
|
if (fs.existsSync(templatePath)) {
|
|
fs.copyFileSync(templatePath, groupClaudeMdPath);
|
|
logger.info(
|
|
{
|
|
file: groupClaudeMdPath,
|
|
template: templatePath,
|
|
promoted: needsPromotion,
|
|
},
|
|
needsPromotion
|
|
? 'Promoted CLAUDE.md to main template'
|
|
: 'Created CLAUDE.md from template',
|
|
);
|
|
}
|
|
}
|
|
|
|
// 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 groupsDir = path.join(projectRoot, 'groups');
|
|
const mdFiles = fs
|
|
.readdirSync(groupsDir)
|
|
.map((d) => path.join(groupsDir, d, 'CLAUDE.md'))
|
|
.filter((f) => fs.existsSync(f));
|
|
|
|
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,
|
|
CHANNEL: parsed.channel,
|
|
TRIGGER: parsed.trigger,
|
|
REQUIRES_TRIGGER: parsed.requiresTrigger,
|
|
ASSISTANT_NAME: parsed.assistantName,
|
|
NAME_UPDATED: nameUpdated,
|
|
STATUS: 'success',
|
|
LOG: 'logs/setup.log',
|
|
});
|
|
}
|