fix(team): avoid epoch bootstrap timestamps

This commit is contained in:
777genius 2026-06-06 20:47:16 +03:00
parent 08b1de7fa2
commit 0551c9c363
2 changed files with 74 additions and 15 deletions

View file

@ -85,18 +85,21 @@ function resolveLeadName(config: TeamConfig): string {
return lead?.name?.trim() || 'team-lead';
}
function resolveSyntheticBootstrapTimestamp(config: TeamConfig, member: TeamConfigMember): string {
function resolveSyntheticBootstrapTimestamp(
config: TeamConfig,
member: TeamConfigMember
): string | null {
const raw = member.joinedAt ?? (config as { createdAt?: unknown }).createdAt;
if (typeof raw === 'number' && Number.isFinite(raw)) {
if (typeof raw === 'number' && Number.isFinite(raw) && raw > 0) {
return new Date(raw).toISOString();
}
if (typeof raw === 'string') {
const parsed = Date.parse(raw);
if (Number.isFinite(parsed)) {
if (Number.isFinite(parsed) && parsed > 0) {
return new Date(parsed).toISOString();
}
}
return new Date(0).toISOString();
return null;
}
function buildSyntheticBootstrapDisplayPrompt(
@ -122,7 +125,10 @@ Call member_briefing directly yourself. Do NOT use Agent, any subagent, or a del
After member_briefing succeeds, wait for instructions from the lead and use team mailbox/task tools normally.`;
}
function buildSyntheticBootstrapMessages(config: TeamConfig): InboxMessage[] {
function buildSyntheticBootstrapMessages(
config: TeamConfig,
fallbackTimestampForMessage: (messageId: string) => string
): InboxMessage[] {
const members = Array.isArray(config.members) ? config.members : [];
const leadName = resolveLeadName(config);
const normalizedLeadName = leadName.trim().toLowerCase();
@ -134,15 +140,20 @@ function buildSyntheticBootstrapMessages(config: TeamConfig): InboxMessage[] {
member.name.trim().toLowerCase() !== normalizedLeadName &&
member.removedAt == null
)
.map((member) => ({
from: leadName,
to: member.name,
text: buildSyntheticBootstrapDisplayPrompt(config, member),
timestamp: resolveSyntheticBootstrapTimestamp(config, member),
read: true,
source: 'system_notification' as const,
messageId: `bootstrap-start:${config.name}:${member.name}`,
}));
.map((member) => {
const messageId = `bootstrap-start:${config.name}:${member.name}`;
return {
from: leadName,
to: member.name,
text: buildSyntheticBootstrapDisplayPrompt(config, member),
timestamp:
resolveSyntheticBootstrapTimestamp(config, member) ??
fallbackTimestampForMessage(messageId),
read: true,
source: 'system_notification' as const,
messageId,
};
});
}
function isVisibleTeamMessage(message: InboxMessage): boolean {
@ -429,6 +440,7 @@ export class TeamMessageFeedService {
private readonly dirtyTeams = new Set<string>();
private readonly inFlightByTeam = new Map<string, InFlightTeamMessageFeed>();
private readonly generationByTeam = new Map<string, number>();
private readonly syntheticBootstrapTimestampByMessageId = new Map<string, string>();
constructor(private readonly deps: TeamMessageFeedDeps) {}
@ -487,6 +499,17 @@ export class TeamMessageFeedService {
return this.generationByTeam.get(teamName) ?? 0;
}
private getSyntheticBootstrapFallbackTimestamp(messageId: string): string {
const existing = this.syntheticBootstrapTimestampByMessageId.get(messageId);
if (existing) {
return existing;
}
const timestamp = new Date(Date.now()).toISOString();
this.syntheticBootstrapTimestampByMessageId.set(messageId, timestamp);
return timestamp;
}
private refreshCleanExpiredCacheInBackground(
teamName: string,
cached: TeamMessageFeedCacheEntry,
@ -554,7 +577,9 @@ export class TeamMessageFeedService {
const sourceMs = Date.now() - sourceStartedAt;
const normalizeStartedAt = Date.now();
const syntheticMessages = buildSyntheticBootstrapMessages(config);
const syntheticMessages = buildSyntheticBootstrapMessages(config, (messageId) =>
this.getSyntheticBootstrapFallbackTimestamp(messageId)
);
let messages = [...inboxMessages, ...leadTexts, ...sentMessages, ...syntheticMessages].filter(
isVisibleTeamMessage
);

View file

@ -120,6 +120,40 @@ describe('TeamMessageFeedService', () => {
expect(feed.messages[0].text).toContain('member_briefing');
});
it('does not stamp synthetic bootstrap prompts with Unix epoch when config has no join time', async () => {
const service = new TeamMessageFeedService({
getConfig: vi.fn(async () => ({
name: 'opencode-test',
members: [
{ name: 'team-lead', role: 'Lead' },
{
name: 'alice',
role: 'Developer',
providerId: 'opencode' as const,
model: 'openrouter/big-pickle',
},
],
})),
getInboxMessages: vi.fn(async () => []),
getLeadSessionMessages: vi.fn(async () => []),
getSentMessages: vi.fn(async () => []),
});
const first = await service.getFeed('opencode-test');
expect(first.messages).toHaveLength(1);
expect(first.messages[0].messageId).toBe('bootstrap-start:opencode-test:alice');
expect(first.messages[0].timestamp).toBe('2026-04-19T18:46:40.000Z');
expect(first.messages[0].timestamp).not.toBe('1970-01-01T00:00:00.000Z');
vi.setSystemTime(new Date('2026-04-19T18:47:00.000Z'));
service.invalidate('opencode-test');
const refreshed = await service.getFeed('opencode-test');
expect(refreshed.messages[0].timestamp).toBe(first.messages[0].timestamp);
expect(refreshed.feedRevision).toBe(first.feedRevision);
});
it('does not hide user-authored text just because it resembles an internal prompt', async () => {
const service = new TeamMessageFeedService({
getConfig: vi.fn(async () => config),