feat(config): support assistant briefing runtime edits and setup defaults

This commit is contained in:
William Valentin
2026-02-18 12:15:37 -08:00
parent 4267eae16c
commit 182827d612
5 changed files with 91 additions and 0 deletions
+11
View File
@@ -68,6 +68,16 @@ describe('setupAutomation', () => {
expect(briefing.enabled).toBe(true);
expect(briefing.schedule).toBe('0 7 * * 1-5');
expect(minioSync.enabled).toBe(true);
expect(automation.delivery_mode).toBe('announce');
expect((config.memory as Record<string, unknown>).daily_log).toEqual({
enabled: true,
namespace_prefix: 'daily',
});
expect((config.memory as Record<string, unknown>).proactive_extract).toEqual({
enabled: true,
min_tool_calls: 1,
namespace: 'global',
});
});
it('allows overriding operator pack output routing', async () => {
@@ -102,6 +112,7 @@ describe('setupAutomation', () => {
expect((heartbeat.notify as Record<string, unknown>).channel).toBe('discord');
expect((briefing.output as Record<string, unknown>).peer).toBe('ops-room');
expect(automation.minio_sync).toBeUndefined();
expect(automation.delivery_mode).toBe('announce');
});
it('leaves operator pack disabled when not selected', async () => {
+10
View File
@@ -116,5 +116,15 @@ describe('ConfigBuilder', () => {
expect((obj.automation as Record<string, unknown>)?.heartbeat).toBeDefined();
expect((obj.automation as Record<string, unknown>)?.daily_briefing).toBeDefined();
expect((obj.automation as Record<string, unknown>)?.minio_sync).toBeDefined();
expect((obj.automation as Record<string, unknown>)?.delivery_mode).toBe('announce');
expect((obj.memory as Record<string, unknown>)?.daily_log).toEqual({
enabled: true,
namespace_prefix: 'daily',
});
expect((obj.memory as Record<string, unknown>)?.proactive_extract).toEqual({
enabled: true,
min_tool_calls: 1,
namespace: 'global',
});
});
});
+15
View File
@@ -257,12 +257,16 @@ export class ConfigBuilder {
applyOperatorPack(options: OperatorPackOptions): void {
const automation = (this.config.automation ?? {}) as Record<string, unknown>;
const backup = (this.config.backup ?? {}) as Record<string, unknown>;
const memory = (this.config.memory ?? {}) as Record<string, unknown>;
backup.enabled = true;
backup.schedule = options.backupSchedule;
backup.run_on_start = true;
backup.notify = { channel: options.outputChannel, peer: options.outputPeer };
// Personal-assistant mode defaults: proactive push and continuity-first memory cadence.
automation.delivery_mode = 'announce';
automation.heartbeat = {
enabled: true,
notify: { channel: options.outputChannel, peer: options.outputPeer },
@@ -298,8 +302,19 @@ export class ConfigBuilder {
};
}
memory.daily_log = {
enabled: true,
namespace_prefix: 'daily',
};
memory.proactive_extract = {
enabled: true,
min_tool_calls: 1,
namespace: 'global',
};
this.config.automation = automation;
this.config.backup = backup;
this.config.memory = memory;
}
build(): SetupConfig {
+44
View File
@@ -176,6 +176,50 @@ const PATCHABLE_KEYS: Record<string, (config: Config, value: unknown) => boolean
config.automation.daily_briefing.enabled = value;
return true;
},
'automation.daily_briefing.output.channel': (config, value) => {
if (typeof value !== 'string' || value.trim().length === 0) {return false;}
config.automation ??= {} as Config['automation'];
config.automation.daily_briefing ??= {} as Config['automation']['daily_briefing'];
config.automation.daily_briefing.output ??= { channel: 'webchat', peer: 'assistant' };
config.automation.daily_briefing.output.channel = value.trim();
return true;
},
'automation.daily_briefing.output.peer': (config, value) => {
if (typeof value !== 'string' || value.trim().length === 0) {return false;}
config.automation ??= {} as Config['automation'];
config.automation.daily_briefing ??= {} as Config['automation']['daily_briefing'];
config.automation.daily_briefing.output ??= { channel: 'webchat', peer: 'assistant' };
config.automation.daily_briefing.output.peer = value.trim();
return true;
},
'automation.daily_briefing.prompt': (config, value) => {
if (typeof value !== 'string' || value.trim().length === 0) {return false;}
config.automation ??= {} as Config['automation'];
config.automation.daily_briefing ??= {} as Config['automation']['daily_briefing'];
config.automation.daily_briefing.prompt = value;
return true;
},
'automation.daily_briefing.schedule': (config, value) => {
if (typeof value !== 'string' || value.trim().length === 0) {return false;}
config.automation ??= {} as Config['automation'];
config.automation.daily_briefing ??= {} as Config['automation']['daily_briefing'];
config.automation.daily_briefing.schedule = value.trim();
return true;
},
'automation.daily_briefing.timezone': (config, value) => {
if (typeof value !== 'string' || value.trim().length === 0) {return false;}
config.automation ??= {} as Config['automation'];
config.automation.daily_briefing ??= {} as Config['automation']['daily_briefing'];
config.automation.daily_briefing.timezone = value.trim();
return true;
},
'automation.daily_briefing.model_tier': (config, value) => {
if (value !== 'fast' && value !== 'default' && value !== 'complex' && value !== 'local') {return false;}
config.automation ??= {} as Config['automation'];
config.automation.daily_briefing ??= {} as Config['automation']['daily_briefing'];
config.automation.daily_briefing.model_tier = value;
return true;
},
'memory.daily_log.enabled': (config, value) => {
if (typeof value !== 'boolean') {return false;}
config.memory ??= {} as Config['memory'];
+11
View File
@@ -1145,6 +1145,9 @@ describe('config handlers', () => {
'server.nodes.push.enabled': true,
'automation.delivery_mode': 'announce',
'automation.daily_briefing.enabled': true,
'automation.daily_briefing.output.channel': 'telegram',
'automation.daily_briefing.output.peer': '12345',
'automation.daily_briefing.model_tier': 'fast',
'memory.daily_log.enabled': true,
'memory.proactive_extract.enabled': true,
'memory.proactive_extract.min_tool_calls': 2,
@@ -1165,6 +1168,9 @@ describe('config handlers', () => {
'server.nodes.push.enabled',
'automation.delivery_mode',
'automation.daily_briefing.enabled',
'automation.daily_briefing.output.channel',
'automation.daily_briefing.output.peer',
'automation.daily_briefing.model_tier',
'memory.daily_log.enabled',
'memory.proactive_extract.enabled',
'memory.proactive_extract.min_tool_calls',
@@ -1182,6 +1188,9 @@ describe('config handlers', () => {
expect(config.server.nodes.push.enabled).toBe(true);
expect(getPath(config, 'automation', 'delivery_mode')).toBe('announce');
expect(getPath(config, 'automation', 'daily_briefing', 'enabled')).toBe(true);
expect(getPath(config, 'automation', 'daily_briefing', 'output', 'channel')).toBe('telegram');
expect(getPath(config, 'automation', 'daily_briefing', 'output', 'peer')).toBe('12345');
expect(getPath(config, 'automation', 'daily_briefing', 'model_tier')).toBe('fast');
expect(getPath(config, 'memory', 'daily_log', 'enabled')).toBe(true);
expect(getPath(config, 'memory', 'proactive_extract', 'enabled')).toBe(true);
expect(getPath(config, 'memory', 'proactive_extract', 'min_tool_calls')).toBe(2);
@@ -1222,6 +1231,7 @@ describe('config handlers', () => {
'server.queue.cap': 0,
'memory.proactive_extract.min_tool_calls': 99,
'tts.enabled_channels': [1, 2, 3],
'automation.daily_briefing.model_tier': 'ultra',
},
},
};
@@ -1234,6 +1244,7 @@ describe('config handlers', () => {
'server.queue.cap',
'memory.proactive_extract.min_tool_calls',
'tts.enabled_channels',
'automation.daily_briefing.model_tier',
]);
expect(r.persisted).toBe(false);
});