Merge pull request #3 from gavrielc/claude/secure-ipc-access-Ni9l4

Secure IPC with per-group namespaces to prevent privilege escalation
This commit is contained in:
gavrielc
2026-02-01 20:40:27 +02:00
committed by GitHub
3 changed files with 141 additions and 83 deletions

View File

@@ -110,11 +110,13 @@ function buildVolumeMounts(group: RegisteredGroup, isMain: boolean): VolumeMount
}); });
} }
const ipcDir = path.join(DATA_DIR, 'ipc'); // Per-group IPC namespace: each group gets its own IPC directory
fs.mkdirSync(path.join(ipcDir, 'messages'), { recursive: true }); // This prevents cross-group privilege escalation via IPC
fs.mkdirSync(path.join(ipcDir, 'tasks'), { recursive: true }); const groupIpcDir = path.join(DATA_DIR, 'ipc', group.folder);
fs.mkdirSync(path.join(groupIpcDir, 'messages'), { recursive: true });
fs.mkdirSync(path.join(groupIpcDir, 'tasks'), { recursive: true });
mounts.push({ mounts.push({
hostPath: ipcDir, hostPath: groupIpcDir,
containerPath: '/workspace/ipc', containerPath: '/workspace/ipc',
readonly: false readonly: false
}); });
@@ -359,7 +361,10 @@ export async function runContainerAgent(
}); });
} }
export function writeTasksSnapshot(tasks: Array<{ export function writeTasksSnapshot(
groupFolder: string,
isMain: boolean,
tasks: Array<{
id: string; id: string;
groupFolder: string; groupFolder: string;
prompt: string; prompt: string;
@@ -367,9 +372,17 @@ export function writeTasksSnapshot(tasks: Array<{
schedule_value: string; schedule_value: string;
status: string; status: string;
next_run: string | null; next_run: string | null;
}>): void { }>
const ipcDir = path.join(DATA_DIR, 'ipc'); ): void {
fs.mkdirSync(ipcDir, { recursive: true }); // Write filtered tasks to the group's IPC directory
const tasksFile = path.join(ipcDir, 'current_tasks.json'); const groupIpcDir = path.join(DATA_DIR, 'ipc', groupFolder);
fs.writeFileSync(tasksFile, JSON.stringify(tasks, null, 2)); fs.mkdirSync(groupIpcDir, { recursive: true });
// Main sees all tasks, others only see their own
const filteredTasks = isMain
? tasks
: tasks.filter(t => t.groupFolder === groupFolder);
const tasksFile = path.join(groupIpcDir, 'current_tasks.json');
fs.writeFileSync(tasksFile, JSON.stringify(filteredTasks, null, 2));
} }

View File

@@ -96,9 +96,9 @@ async function runAgent(group: RegisteredGroup, prompt: string, chatJid: string)
const isMain = group.folder === MAIN_GROUP_FOLDER; const isMain = group.folder === MAIN_GROUP_FOLDER;
const sessionId = sessions[group.folder]; const sessionId = sessions[group.folder];
// Update tasks snapshot for container to read // Update tasks snapshot for container to read (filtered by group)
const tasks = getAllTasks(); const tasks = getAllTasks();
writeTasksSnapshot(tasks.map(t => ({ writeTasksSnapshot(group.folder, isMain, tasks.map(t => ({
id: t.id, id: t.id,
groupFolder: t.group_folder, groupFolder: t.group_folder,
prompt: t.prompt, prompt: t.prompt,
@@ -144,63 +144,92 @@ async function sendMessage(jid: string, text: string): Promise<void> {
} }
function startIpcWatcher(): void { function startIpcWatcher(): void {
const messagesDir = path.join(DATA_DIR, 'ipc', 'messages'); const ipcBaseDir = path.join(DATA_DIR, 'ipc');
const tasksDir = path.join(DATA_DIR, 'ipc', 'tasks'); fs.mkdirSync(ipcBaseDir, { recursive: true });
fs.mkdirSync(messagesDir, { recursive: true });
fs.mkdirSync(tasksDir, { recursive: true });
const processIpcFiles = async () => { const processIpcFiles = async () => {
// Scan all group IPC directories (identity determined by directory)
let groupFolders: string[];
try { try {
groupFolders = fs.readdirSync(ipcBaseDir).filter(f => {
const stat = fs.statSync(path.join(ipcBaseDir, f));
return stat.isDirectory() && f !== 'errors';
});
} catch (err) {
logger.error({ err }, 'Error reading IPC base directory');
setTimeout(processIpcFiles, IPC_POLL_INTERVAL);
return;
}
for (const sourceGroup of groupFolders) {
const isMain = sourceGroup === MAIN_GROUP_FOLDER;
const messagesDir = path.join(ipcBaseDir, sourceGroup, 'messages');
const tasksDir = path.join(ipcBaseDir, sourceGroup, 'tasks');
// Process messages from this group's IPC directory
try {
if (fs.existsSync(messagesDir)) {
const messageFiles = fs.readdirSync(messagesDir).filter(f => f.endsWith('.json')); const messageFiles = fs.readdirSync(messagesDir).filter(f => f.endsWith('.json'));
for (const file of messageFiles) { for (const file of messageFiles) {
const filePath = path.join(messagesDir, file); const filePath = path.join(messagesDir, file);
try { try {
const data = JSON.parse(fs.readFileSync(filePath, 'utf-8')); const data = JSON.parse(fs.readFileSync(filePath, 'utf-8'));
if (data.type === 'message' && data.chatJid && data.text) { if (data.type === 'message' && data.chatJid && data.text) {
// Authorization: verify this group can send to this chatJid
const targetGroup = registeredGroups[data.chatJid];
if (isMain || (targetGroup && targetGroup.folder === sourceGroup)) {
await sendMessage(data.chatJid, `${ASSISTANT_NAME}: ${data.text}`); await sendMessage(data.chatJid, `${ASSISTANT_NAME}: ${data.text}`);
logger.info({ chatJid: data.chatJid }, 'IPC message sent'); logger.info({ chatJid: data.chatJid, sourceGroup }, 'IPC message sent');
} else {
logger.warn({ chatJid: data.chatJid, sourceGroup }, 'Unauthorized IPC message attempt blocked');
}
} }
fs.unlinkSync(filePath); fs.unlinkSync(filePath);
} catch (err) { } catch (err) {
logger.error({ file, err }, 'Error processing IPC message'); logger.error({ file, sourceGroup, err }, 'Error processing IPC message');
// Move to error directory instead of deleting const errorDir = path.join(ipcBaseDir, 'errors');
const errorDir = path.join(DATA_DIR, 'ipc', 'errors');
fs.mkdirSync(errorDir, { recursive: true }); fs.mkdirSync(errorDir, { recursive: true });
fs.renameSync(filePath, path.join(errorDir, file)); fs.renameSync(filePath, path.join(errorDir, `${sourceGroup}-${file}`));
}
} }
} }
} catch (err) { } catch (err) {
logger.error({ err }, 'Error reading IPC messages directory'); logger.error({ err, sourceGroup }, 'Error reading IPC messages directory');
} }
// Process tasks from this group's IPC directory
try { try {
if (fs.existsSync(tasksDir)) {
const taskFiles = fs.readdirSync(tasksDir).filter(f => f.endsWith('.json')); const taskFiles = fs.readdirSync(tasksDir).filter(f => f.endsWith('.json'));
for (const file of taskFiles) { for (const file of taskFiles) {
const filePath = path.join(tasksDir, file); const filePath = path.join(tasksDir, file);
try { try {
const data = JSON.parse(fs.readFileSync(filePath, 'utf-8')); const data = JSON.parse(fs.readFileSync(filePath, 'utf-8'));
await processTaskIpc(data); // Pass source group identity to processTaskIpc for authorization
await processTaskIpc(data, sourceGroup, isMain);
fs.unlinkSync(filePath); fs.unlinkSync(filePath);
} catch (err) { } catch (err) {
logger.error({ file, err }, 'Error processing IPC task'); logger.error({ file, sourceGroup, err }, 'Error processing IPC task');
const errorDir = path.join(DATA_DIR, 'ipc', 'errors'); const errorDir = path.join(ipcBaseDir, 'errors');
fs.mkdirSync(errorDir, { recursive: true }); fs.mkdirSync(errorDir, { recursive: true });
fs.renameSync(filePath, path.join(errorDir, file)); fs.renameSync(filePath, path.join(errorDir, `${sourceGroup}-${file}`));
}
} }
} }
} catch (err) { } catch (err) {
logger.error({ err }, 'Error reading IPC tasks directory'); logger.error({ err, sourceGroup }, 'Error reading IPC tasks directory');
}
} }
setTimeout(processIpcFiles, IPC_POLL_INTERVAL); setTimeout(processIpcFiles, IPC_POLL_INTERVAL);
}; };
processIpcFiles(); processIpcFiles();
logger.info('IPC watcher started'); logger.info('IPC watcher started (per-group namespaces)');
} }
async function processTaskIpc(data: { async function processTaskIpc(
data: {
type: string; type: string;
taskId?: string; taskId?: string;
prompt?: string; prompt?: string;
@@ -208,8 +237,10 @@ async function processTaskIpc(data: {
schedule_value?: string; schedule_value?: string;
groupFolder?: string; groupFolder?: string;
chatJid?: string; chatJid?: string;
isMain?: boolean; },
}): Promise<void> { sourceGroup: string, // Verified identity from IPC directory
isMain: boolean // Verified from directory path
): Promise<void> {
// Import db functions dynamically to avoid circular deps // Import db functions dynamically to avoid circular deps
const { createTask, updateTask, deleteTask, getTaskById: getTask } = await import('./db.js'); const { createTask, updateTask, deleteTask, getTaskById: getTask } = await import('./db.js');
const { CronExpressionParser } = await import('cron-parser'); const { CronExpressionParser } = await import('cron-parser');
@@ -217,6 +248,20 @@ async function processTaskIpc(data: {
switch (data.type) { switch (data.type) {
case 'schedule_task': case 'schedule_task':
if (data.prompt && data.schedule_type && data.schedule_value && data.groupFolder && data.chatJid) { if (data.prompt && data.schedule_type && data.schedule_value && data.groupFolder && data.chatJid) {
// Authorization: non-main groups can only schedule for themselves
const targetGroup = data.groupFolder;
if (!isMain && targetGroup !== sourceGroup) {
logger.warn({ sourceGroup, targetGroup, chatJid: data.chatJid }, 'Unauthorized schedule_task attempt blocked');
break;
}
// Authorization: verify the chatJid belongs to the target group
const chatGroup = registeredGroups[data.chatJid];
if (!isMain && (!chatGroup || chatGroup.folder !== targetGroup)) {
logger.warn({ sourceGroup, targetGroup, chatJid: data.chatJid }, 'Unauthorized schedule_task chatJid blocked');
break;
}
const scheduleType = data.schedule_type as 'cron' | 'interval' | 'once'; const scheduleType = data.schedule_type as 'cron' | 'interval' | 'once';
let nextRun: string | null = null; let nextRun: string | null = null;
@@ -233,7 +278,7 @@ async function processTaskIpc(data: {
const taskId = `task-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`; const taskId = `task-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`;
createTask({ createTask({
id: taskId, id: taskId,
group_folder: data.groupFolder, group_folder: targetGroup,
chat_jid: data.chatJid, chat_jid: data.chatJid,
prompt: data.prompt, prompt: data.prompt,
schedule_type: scheduleType, schedule_type: scheduleType,
@@ -242,18 +287,18 @@ async function processTaskIpc(data: {
status: 'active', status: 'active',
created_at: new Date().toISOString() created_at: new Date().toISOString()
}); });
logger.info({ taskId, groupFolder: data.groupFolder }, 'Task created via IPC'); logger.info({ taskId, sourceGroup, targetGroup }, 'Task created via IPC');
} }
break; break;
case 'pause_task': case 'pause_task':
if (data.taskId) { if (data.taskId) {
const task = getTask(data.taskId); const task = getTask(data.taskId);
if (task && (data.isMain || task.group_folder === data.groupFolder)) { if (task && (isMain || task.group_folder === sourceGroup)) {
updateTask(data.taskId, { status: 'paused' }); updateTask(data.taskId, { status: 'paused' });
logger.info({ taskId: data.taskId }, 'Task paused via IPC'); logger.info({ taskId: data.taskId, sourceGroup }, 'Task paused via IPC');
} else { } else {
logger.warn({ taskId: data.taskId, groupFolder: data.groupFolder }, 'Unauthorized task pause attempt'); logger.warn({ taskId: data.taskId, sourceGroup }, 'Unauthorized task pause attempt');
} }
} }
break; break;
@@ -261,11 +306,11 @@ async function processTaskIpc(data: {
case 'resume_task': case 'resume_task':
if (data.taskId) { if (data.taskId) {
const task = getTask(data.taskId); const task = getTask(data.taskId);
if (task && (data.isMain || task.group_folder === data.groupFolder)) { if (task && (isMain || task.group_folder === sourceGroup)) {
updateTask(data.taskId, { status: 'active' }); updateTask(data.taskId, { status: 'active' });
logger.info({ taskId: data.taskId }, 'Task resumed via IPC'); logger.info({ taskId: data.taskId, sourceGroup }, 'Task resumed via IPC');
} else { } else {
logger.warn({ taskId: data.taskId, groupFolder: data.groupFolder }, 'Unauthorized task resume attempt'); logger.warn({ taskId: data.taskId, sourceGroup }, 'Unauthorized task resume attempt');
} }
} }
break; break;
@@ -273,11 +318,11 @@ async function processTaskIpc(data: {
case 'cancel_task': case 'cancel_task':
if (data.taskId) { if (data.taskId) {
const task = getTask(data.taskId); const task = getTask(data.taskId);
if (task && (data.isMain || task.group_folder === data.groupFolder)) { if (task && (isMain || task.group_folder === sourceGroup)) {
deleteTask(data.taskId); deleteTask(data.taskId);
logger.info({ taskId: data.taskId }, 'Task cancelled via IPC'); logger.info({ taskId: data.taskId, sourceGroup }, 'Task cancelled via IPC');
} else { } else {
logger.warn({ taskId: data.taskId, groupFolder: data.groupFolder }, 'Unauthorized task cancel attempt'); logger.warn({ taskId: data.taskId, sourceGroup }, 'Unauthorized task cancel attempt');
} }
} }
break; break;

View File

@@ -40,9 +40,10 @@ async function runTask(task: ScheduledTask, deps: SchedulerDependencies): Promis
return; return;
} }
// Update tasks snapshot for container to read // Update tasks snapshot for container to read (filtered by group)
const isMain = task.group_folder === MAIN_GROUP_FOLDER;
const tasks = getAllTasks(); const tasks = getAllTasks();
writeTasksSnapshot(tasks.map(t => ({ writeTasksSnapshot(task.group_folder, isMain, tasks.map(t => ({
id: t.id, id: t.id,
groupFolder: t.group_folder, groupFolder: t.group_folder,
prompt: t.prompt, prompt: t.prompt,
@@ -56,7 +57,6 @@ async function runTask(task: ScheduledTask, deps: SchedulerDependencies): Promis
let error: string | null = null; let error: string | null = null;
try { try {
const isMain = task.group_folder === MAIN_GROUP_FOLDER;
const output = await runContainerAgent(group, { const output = await runContainerAgent(group, {
prompt: task.prompt, prompt: task.prompt,
groupFolder: task.group_folder, groupFolder: task.group_folder,