feat: add per-DM history limit overrides
This commit is contained in:
@@ -431,6 +431,71 @@ describe("getDmHistoryLimitFromSessionKey", () => {
|
|||||||
).toBe(5);
|
).toBe(5);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("returns per-DM override when set", () => {
|
||||||
|
const config = {
|
||||||
|
telegram: {
|
||||||
|
dmHistoryLimit: 15,
|
||||||
|
dms: { "123": { historyLimit: 5 } },
|
||||||
|
},
|
||||||
|
} as ClawdbotConfig;
|
||||||
|
expect(getDmHistoryLimitFromSessionKey("telegram:dm:123", config)).toBe(5);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("falls back to provider default when per-DM not set", () => {
|
||||||
|
const config = {
|
||||||
|
telegram: {
|
||||||
|
dmHistoryLimit: 15,
|
||||||
|
dms: { "456": { historyLimit: 5 } },
|
||||||
|
},
|
||||||
|
} as ClawdbotConfig;
|
||||||
|
expect(getDmHistoryLimitFromSessionKey("telegram:dm:123", config)).toBe(15);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns per-DM override for agent-prefixed keys", () => {
|
||||||
|
const config = {
|
||||||
|
telegram: {
|
||||||
|
dmHistoryLimit: 20,
|
||||||
|
dms: { "789": { historyLimit: 3 } },
|
||||||
|
},
|
||||||
|
} as ClawdbotConfig;
|
||||||
|
expect(
|
||||||
|
getDmHistoryLimitFromSessionKey("agent:main:telegram:dm:789", config),
|
||||||
|
).toBe(3);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("handles userId with colons (e.g., email)", () => {
|
||||||
|
const config = {
|
||||||
|
msteams: {
|
||||||
|
dmHistoryLimit: 10,
|
||||||
|
dms: { "user@example.com": { historyLimit: 7 } },
|
||||||
|
},
|
||||||
|
} as ClawdbotConfig;
|
||||||
|
expect(
|
||||||
|
getDmHistoryLimitFromSessionKey("msteams:dm:user@example.com", config),
|
||||||
|
).toBe(7);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns undefined when per-DM historyLimit is not set", () => {
|
||||||
|
const config = {
|
||||||
|
telegram: {
|
||||||
|
dms: { "123": {} },
|
||||||
|
},
|
||||||
|
} as ClawdbotConfig;
|
||||||
|
expect(
|
||||||
|
getDmHistoryLimitFromSessionKey("telegram:dm:123", config),
|
||||||
|
).toBeUndefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 0 when per-DM historyLimit is explicitly 0 (unlimited)", () => {
|
||||||
|
const config = {
|
||||||
|
telegram: {
|
||||||
|
dmHistoryLimit: 15,
|
||||||
|
dms: { "123": { historyLimit: 0 } },
|
||||||
|
},
|
||||||
|
} as ClawdbotConfig;
|
||||||
|
expect(getDmHistoryLimitFromSessionKey("telegram:dm:123", config)).toBe(0);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("runEmbeddedPiAgent", () => {
|
describe("runEmbeddedPiAgent", () => {
|
||||||
|
|||||||
@@ -446,12 +446,16 @@ export function limitHistoryTurns(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Extracts the provider name from a session key and looks up dmHistoryLimit
|
* Extracts the provider name and user ID from a session key and looks up
|
||||||
* from the provider config.
|
* dmHistoryLimit from the provider config, with per-DM override support.
|
||||||
*
|
*
|
||||||
* Session key formats:
|
* Session key formats:
|
||||||
* - `telegram:dm:123` → provider = telegram
|
* - `telegram:dm:123` → provider = telegram, userId = 123
|
||||||
* - `agent:main:telegram:dm:123` → provider = telegram (skip "agent:<id>:")
|
* - `agent:main:telegram:dm:123` → provider = telegram, userId = 123
|
||||||
|
*
|
||||||
|
* Resolution order:
|
||||||
|
* 1. Per-DM override: provider.dms[userId].historyLimit
|
||||||
|
* 2. Provider default: provider.dmHistoryLimit
|
||||||
*/
|
*/
|
||||||
export function getDmHistoryLimitFromSessionKey(
|
export function getDmHistoryLimitFromSessionKey(
|
||||||
sessionKey: string | undefined,
|
sessionKey: string | undefined,
|
||||||
@@ -467,22 +471,49 @@ export function getDmHistoryLimitFromSessionKey(
|
|||||||
const provider = providerParts[0]?.toLowerCase();
|
const provider = providerParts[0]?.toLowerCase();
|
||||||
if (!provider) return undefined;
|
if (!provider) return undefined;
|
||||||
|
|
||||||
|
// Extract userId: format is provider:dm:userId or provider:dm:userId:...
|
||||||
|
// The userId may contain colons (e.g., email addresses), so join remaining parts
|
||||||
|
const kind = providerParts[1]?.toLowerCase();
|
||||||
|
const userId = providerParts.slice(2).join(":");
|
||||||
|
|
||||||
|
// Helper to get limit with per-DM override support
|
||||||
|
const getLimit = (
|
||||||
|
providerConfig:
|
||||||
|
| {
|
||||||
|
dmHistoryLimit?: number;
|
||||||
|
dms?: Record<string, { historyLimit?: number }>;
|
||||||
|
}
|
||||||
|
| undefined,
|
||||||
|
): number | undefined => {
|
||||||
|
if (!providerConfig) return undefined;
|
||||||
|
// Check per-DM override first
|
||||||
|
if (
|
||||||
|
userId &&
|
||||||
|
kind === "dm" &&
|
||||||
|
providerConfig.dms?.[userId]?.historyLimit !== undefined
|
||||||
|
) {
|
||||||
|
return providerConfig.dms[userId].historyLimit;
|
||||||
|
}
|
||||||
|
// Fall back to provider default
|
||||||
|
return providerConfig.dmHistoryLimit;
|
||||||
|
};
|
||||||
|
|
||||||
// Map provider to config key
|
// Map provider to config key
|
||||||
switch (provider) {
|
switch (provider) {
|
||||||
case "telegram":
|
case "telegram":
|
||||||
return config.telegram?.dmHistoryLimit;
|
return getLimit(config.telegram);
|
||||||
case "whatsapp":
|
case "whatsapp":
|
||||||
return config.whatsapp?.dmHistoryLimit;
|
return getLimit(config.whatsapp);
|
||||||
case "discord":
|
case "discord":
|
||||||
return config.discord?.dmHistoryLimit;
|
return getLimit(config.discord);
|
||||||
case "slack":
|
case "slack":
|
||||||
return config.slack?.dmHistoryLimit;
|
return getLimit(config.slack);
|
||||||
case "signal":
|
case "signal":
|
||||||
return config.signal?.dmHistoryLimit;
|
return getLimit(config.signal);
|
||||||
case "imessage":
|
case "imessage":
|
||||||
return config.imessage?.dmHistoryLimit;
|
return getLimit(config.imessage);
|
||||||
case "msteams":
|
case "msteams":
|
||||||
return config.msteams?.dmHistoryLimit;
|
return getLimit(config.msteams);
|
||||||
default:
|
default:
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -149,6 +149,8 @@ export type WhatsAppConfig = {
|
|||||||
historyLimit?: number;
|
historyLimit?: number;
|
||||||
/** Max DM turns to keep as history context. */
|
/** Max DM turns to keep as history context. */
|
||||||
dmHistoryLimit?: number;
|
dmHistoryLimit?: number;
|
||||||
|
/** Per-DM config overrides keyed by user ID. */
|
||||||
|
dms?: Record<string, DmConfig>;
|
||||||
/** Outbound text chunk size (chars). Default: 4000. */
|
/** Outbound text chunk size (chars). Default: 4000. */
|
||||||
textChunkLimit?: number;
|
textChunkLimit?: number;
|
||||||
/** Maximum media file size in MB. Default: 50. */
|
/** Maximum media file size in MB. Default: 50. */
|
||||||
@@ -204,6 +206,8 @@ export type WhatsAppAccountConfig = {
|
|||||||
historyLimit?: number;
|
historyLimit?: number;
|
||||||
/** Max DM turns to keep as history context. */
|
/** Max DM turns to keep as history context. */
|
||||||
dmHistoryLimit?: number;
|
dmHistoryLimit?: number;
|
||||||
|
/** Per-DM config overrides keyed by user ID. */
|
||||||
|
dms?: Record<string, DmConfig>;
|
||||||
textChunkLimit?: number;
|
textChunkLimit?: number;
|
||||||
mediaMaxMb?: number;
|
mediaMaxMb?: number;
|
||||||
blockStreaming?: boolean;
|
blockStreaming?: boolean;
|
||||||
@@ -385,6 +389,8 @@ export type TelegramAccountConfig = {
|
|||||||
historyLimit?: number;
|
historyLimit?: number;
|
||||||
/** Max DM turns to keep as history context. */
|
/** Max DM turns to keep as history context. */
|
||||||
dmHistoryLimit?: number;
|
dmHistoryLimit?: number;
|
||||||
|
/** Per-DM config overrides keyed by user ID. */
|
||||||
|
dms?: Record<string, DmConfig>;
|
||||||
/** Outbound text chunk size (chars). Default: 4000. */
|
/** Outbound text chunk size (chars). Default: 4000. */
|
||||||
textChunkLimit?: number;
|
textChunkLimit?: number;
|
||||||
/** Disable block streaming for this account. */
|
/** Disable block streaming for this account. */
|
||||||
@@ -530,6 +536,8 @@ export type DiscordAccountConfig = {
|
|||||||
historyLimit?: number;
|
historyLimit?: number;
|
||||||
/** Max DM turns to keep as history context. */
|
/** Max DM turns to keep as history context. */
|
||||||
dmHistoryLimit?: number;
|
dmHistoryLimit?: number;
|
||||||
|
/** Per-DM config overrides keyed by user ID. */
|
||||||
|
dms?: Record<string, DmConfig>;
|
||||||
/** Retry policy for outbound Discord API calls. */
|
/** Retry policy for outbound Discord API calls. */
|
||||||
retry?: OutboundRetryConfig;
|
retry?: OutboundRetryConfig;
|
||||||
/** Per-action tool gating (default: true for all). */
|
/** Per-action tool gating (default: true for all). */
|
||||||
@@ -628,6 +636,8 @@ export type SlackAccountConfig = {
|
|||||||
historyLimit?: number;
|
historyLimit?: number;
|
||||||
/** Max DM turns to keep as history context. */
|
/** Max DM turns to keep as history context. */
|
||||||
dmHistoryLimit?: number;
|
dmHistoryLimit?: number;
|
||||||
|
/** Per-DM config overrides keyed by user ID. */
|
||||||
|
dms?: Record<string, DmConfig>;
|
||||||
textChunkLimit?: number;
|
textChunkLimit?: number;
|
||||||
blockStreaming?: boolean;
|
blockStreaming?: boolean;
|
||||||
/** Merge streamed block replies before sending. */
|
/** Merge streamed block replies before sending. */
|
||||||
@@ -689,6 +699,8 @@ export type SignalAccountConfig = {
|
|||||||
historyLimit?: number;
|
historyLimit?: number;
|
||||||
/** Max DM turns to keep as history context. */
|
/** Max DM turns to keep as history context. */
|
||||||
dmHistoryLimit?: number;
|
dmHistoryLimit?: number;
|
||||||
|
/** Per-DM config overrides keyed by user ID. */
|
||||||
|
dms?: Record<string, DmConfig>;
|
||||||
/** Outbound text chunk size (chars). Default: 4000. */
|
/** Outbound text chunk size (chars). Default: 4000. */
|
||||||
textChunkLimit?: number;
|
textChunkLimit?: number;
|
||||||
blockStreaming?: boolean;
|
blockStreaming?: boolean;
|
||||||
@@ -766,6 +778,8 @@ export type MSTeamsConfig = {
|
|||||||
historyLimit?: number;
|
historyLimit?: number;
|
||||||
/** Max DM turns to keep as history context. */
|
/** Max DM turns to keep as history context. */
|
||||||
dmHistoryLimit?: number;
|
dmHistoryLimit?: number;
|
||||||
|
/** Per-DM config overrides keyed by user ID. */
|
||||||
|
dms?: Record<string, DmConfig>;
|
||||||
/** Default reply style: "thread" replies to the message, "top-level" posts a new message. */
|
/** Default reply style: "thread" replies to the message, "top-level" posts a new message. */
|
||||||
replyStyle?: MSTeamsReplyStyle;
|
replyStyle?: MSTeamsReplyStyle;
|
||||||
/** Per-team config. Key is team ID (from the /team/ URL path segment). */
|
/** Per-team config. Key is team ID (from the /team/ URL path segment). */
|
||||||
@@ -804,6 +818,8 @@ export type IMessageAccountConfig = {
|
|||||||
historyLimit?: number;
|
historyLimit?: number;
|
||||||
/** Max DM turns to keep as history context. */
|
/** Max DM turns to keep as history context. */
|
||||||
dmHistoryLimit?: number;
|
dmHistoryLimit?: number;
|
||||||
|
/** Per-DM config overrides keyed by user ID. */
|
||||||
|
dms?: Record<string, DmConfig>;
|
||||||
/** Include attachments + reactions in watch payloads. */
|
/** Include attachments + reactions in watch payloads. */
|
||||||
includeAttachments?: boolean;
|
includeAttachments?: boolean;
|
||||||
/** Max outbound media size in MB. */
|
/** Max outbound media size in MB. */
|
||||||
@@ -941,6 +957,10 @@ export type GroupChatConfig = {
|
|||||||
historyLimit?: number;
|
historyLimit?: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type DmConfig = {
|
||||||
|
historyLimit?: number;
|
||||||
|
};
|
||||||
|
|
||||||
export type QueueConfig = {
|
export type QueueConfig = {
|
||||||
mode?: QueueMode;
|
mode?: QueueMode;
|
||||||
byProvider?: QueueModeByProvider;
|
byProvider?: QueueModeByProvider;
|
||||||
|
|||||||
@@ -62,6 +62,10 @@ const GroupChatSchema = z
|
|||||||
})
|
})
|
||||||
.optional();
|
.optional();
|
||||||
|
|
||||||
|
const DmConfigSchema = z.object({
|
||||||
|
historyLimit: z.number().int().min(0).optional(),
|
||||||
|
});
|
||||||
|
|
||||||
const IdentitySchema = z
|
const IdentitySchema = z
|
||||||
.object({
|
.object({
|
||||||
name: z.string().optional(),
|
name: z.string().optional(),
|
||||||
@@ -274,6 +278,7 @@ const TelegramAccountSchemaBase = z.object({
|
|||||||
groupPolicy: GroupPolicySchema.optional().default("open"),
|
groupPolicy: GroupPolicySchema.optional().default("open"),
|
||||||
historyLimit: z.number().int().min(0).optional(),
|
historyLimit: z.number().int().min(0).optional(),
|
||||||
dmHistoryLimit: z.number().int().min(0).optional(),
|
dmHistoryLimit: z.number().int().min(0).optional(),
|
||||||
|
dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
|
||||||
textChunkLimit: z.number().int().positive().optional(),
|
textChunkLimit: z.number().int().positive().optional(),
|
||||||
blockStreaming: z.boolean().optional(),
|
blockStreaming: z.boolean().optional(),
|
||||||
draftChunk: BlockStreamingChunkSchema.optional(),
|
draftChunk: BlockStreamingChunkSchema.optional(),
|
||||||
@@ -364,6 +369,7 @@ const DiscordAccountSchema = z.object({
|
|||||||
groupPolicy: GroupPolicySchema.optional().default("open"),
|
groupPolicy: GroupPolicySchema.optional().default("open"),
|
||||||
historyLimit: z.number().int().min(0).optional(),
|
historyLimit: z.number().int().min(0).optional(),
|
||||||
dmHistoryLimit: z.number().int().min(0).optional(),
|
dmHistoryLimit: z.number().int().min(0).optional(),
|
||||||
|
dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
|
||||||
textChunkLimit: z.number().int().positive().optional(),
|
textChunkLimit: z.number().int().positive().optional(),
|
||||||
blockStreaming: z.boolean().optional(),
|
blockStreaming: z.boolean().optional(),
|
||||||
blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
|
blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
|
||||||
@@ -437,6 +443,7 @@ const SlackAccountSchema = z.object({
|
|||||||
groupPolicy: GroupPolicySchema.optional().default("open"),
|
groupPolicy: GroupPolicySchema.optional().default("open"),
|
||||||
historyLimit: z.number().int().min(0).optional(),
|
historyLimit: z.number().int().min(0).optional(),
|
||||||
dmHistoryLimit: z.number().int().min(0).optional(),
|
dmHistoryLimit: z.number().int().min(0).optional(),
|
||||||
|
dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
|
||||||
textChunkLimit: z.number().int().positive().optional(),
|
textChunkLimit: z.number().int().positive().optional(),
|
||||||
blockStreaming: z.boolean().optional(),
|
blockStreaming: z.boolean().optional(),
|
||||||
blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
|
blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
|
||||||
@@ -492,6 +499,7 @@ const SignalAccountSchemaBase = z.object({
|
|||||||
groupPolicy: GroupPolicySchema.optional().default("open"),
|
groupPolicy: GroupPolicySchema.optional().default("open"),
|
||||||
historyLimit: z.number().int().min(0).optional(),
|
historyLimit: z.number().int().min(0).optional(),
|
||||||
dmHistoryLimit: z.number().int().min(0).optional(),
|
dmHistoryLimit: z.number().int().min(0).optional(),
|
||||||
|
dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
|
||||||
textChunkLimit: z.number().int().positive().optional(),
|
textChunkLimit: z.number().int().positive().optional(),
|
||||||
blockStreaming: z.boolean().optional(),
|
blockStreaming: z.boolean().optional(),
|
||||||
blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
|
blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
|
||||||
@@ -541,6 +549,7 @@ const IMessageAccountSchemaBase = z.object({
|
|||||||
groupPolicy: GroupPolicySchema.optional().default("open"),
|
groupPolicy: GroupPolicySchema.optional().default("open"),
|
||||||
historyLimit: z.number().int().min(0).optional(),
|
historyLimit: z.number().int().min(0).optional(),
|
||||||
dmHistoryLimit: z.number().int().min(0).optional(),
|
dmHistoryLimit: z.number().int().min(0).optional(),
|
||||||
|
dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
|
||||||
includeAttachments: z.boolean().optional(),
|
includeAttachments: z.boolean().optional(),
|
||||||
mediaMaxMb: z.number().int().positive().optional(),
|
mediaMaxMb: z.number().int().positive().optional(),
|
||||||
textChunkLimit: z.number().int().positive().optional(),
|
textChunkLimit: z.number().int().positive().optional(),
|
||||||
@@ -616,6 +625,7 @@ const MSTeamsConfigSchema = z
|
|||||||
requireMention: z.boolean().optional(),
|
requireMention: z.boolean().optional(),
|
||||||
historyLimit: z.number().int().min(0).optional(),
|
historyLimit: z.number().int().min(0).optional(),
|
||||||
dmHistoryLimit: z.number().int().min(0).optional(),
|
dmHistoryLimit: z.number().int().min(0).optional(),
|
||||||
|
dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
|
||||||
replyStyle: MSTeamsReplyStyleSchema.optional(),
|
replyStyle: MSTeamsReplyStyleSchema.optional(),
|
||||||
teams: z.record(z.string(), MSTeamsTeamSchema.optional()).optional(),
|
teams: z.record(z.string(), MSTeamsTeamSchema.optional()).optional(),
|
||||||
})
|
})
|
||||||
@@ -1361,6 +1371,7 @@ export const ClawdbotSchema = z
|
|||||||
groupPolicy: GroupPolicySchema.optional().default("open"),
|
groupPolicy: GroupPolicySchema.optional().default("open"),
|
||||||
historyLimit: z.number().int().min(0).optional(),
|
historyLimit: z.number().int().min(0).optional(),
|
||||||
dmHistoryLimit: z.number().int().min(0).optional(),
|
dmHistoryLimit: z.number().int().min(0).optional(),
|
||||||
|
dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
|
||||||
textChunkLimit: z.number().int().positive().optional(),
|
textChunkLimit: z.number().int().positive().optional(),
|
||||||
mediaMaxMb: z.number().int().positive().optional(),
|
mediaMaxMb: z.number().int().positive().optional(),
|
||||||
blockStreaming: z.boolean().optional(),
|
blockStreaming: z.boolean().optional(),
|
||||||
@@ -1411,6 +1422,7 @@ export const ClawdbotSchema = z
|
|||||||
groupPolicy: GroupPolicySchema.optional().default("open"),
|
groupPolicy: GroupPolicySchema.optional().default("open"),
|
||||||
historyLimit: z.number().int().min(0).optional(),
|
historyLimit: z.number().int().min(0).optional(),
|
||||||
dmHistoryLimit: z.number().int().min(0).optional(),
|
dmHistoryLimit: z.number().int().min(0).optional(),
|
||||||
|
dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
|
||||||
textChunkLimit: z.number().int().positive().optional(),
|
textChunkLimit: z.number().int().positive().optional(),
|
||||||
mediaMaxMb: z.number().int().positive().optional().default(50),
|
mediaMaxMb: z.number().int().positive().optional().default(50),
|
||||||
blockStreaming: z.boolean().optional(),
|
blockStreaming: z.boolean().optional(),
|
||||||
|
|||||||
Reference in New Issue
Block a user