agent-ecosystem/test/main/services/team/TeamMemberResolver.test.ts
iliya 4e82102ceb feat: enhance team member resolution and activity components
- Introduced a new method to avoid duplicate member names in TeamMemberResolver, ensuring case-insensitive uniqueness.
- Updated ActivityItem and ActivityTimeline components to utilize local member names for improved recipient qualification checks.
- Added tests to validate the handling of dotted names and deduplication in cross-team messaging scenarios.
2026-03-10 01:16:01 +02:00

196 lines
6.9 KiB
TypeScript

import { describe, expect, it } from 'vitest';
import { TeamMemberResolver } from '../../../../src/main/services/team/TeamMemberResolver';
import type {
InboxMessage,
TeamConfig,
TeamTask,
TeamTaskWithKanban,
} from '../../../../src/shared/types/team';
describe('TeamMemberResolver', () => {
it('builds roster from config + meta + inbox only', () => {
const resolver = new TeamMemberResolver();
const config: TeamConfig = {
name: 'My Team',
members: [{ name: 'team-lead', agentType: 'team-lead', role: 'lead' }],
};
const metaMembers: TeamConfig['members'] = [
{ name: 'alice', role: 'developer', agentType: 'general-purpose', color: 'blue' },
];
const inboxNames = ['bob'];
const tasks: TeamTask[] = [
{ id: '1', subject: 'Visible task', status: 'pending', owner: 'alice' },
{ id: '2', subject: 'Ghost task', status: 'pending', owner: 'stranger' },
];
const now = new Date().toISOString();
const messages: InboxMessage[] = [
{ from: 'bob', text: 'ready', timestamp: now, read: false, color: 'green' },
{ from: 'user', text: 'system note', timestamp: now, read: false },
];
const members = resolver.resolveMembers(config, metaMembers, inboxNames, tasks, messages);
const names = members.map((member) => member.name);
expect(names).toEqual(['alice', 'bob', 'team-lead']);
expect(names).not.toContain('stranger');
expect(names).not.toContain('user');
const alice = members.find((member) => member.name === 'alice');
expect(alice?.role).toBe('developer');
expect(alice?.color).toBe('blue');
const lead = members.find((member) => member.name === 'team-lead');
expect(lead?.role).toBe('lead');
expect(lead?.agentType).toBe('team-lead');
});
it('filters out "user" pseudo-member even when present in config, meta, or inboxNames', () => {
const resolver = new TeamMemberResolver();
const config: TeamConfig = {
name: 'Team',
members: [
{ name: 'team-lead', agentType: 'team-lead', role: 'lead' },
{ name: 'user', agentType: 'general-purpose' },
],
};
const metaMembers: TeamConfig['members'] = [
{ name: 'user', agentType: 'general-purpose' },
{ name: 'alice', role: 'dev', agentType: 'general-purpose' },
];
const inboxNames = ['user', 'alice'];
const tasks: TeamTask[] = [];
const messages: InboxMessage[] = [];
const members = resolver.resolveMembers(config, metaMembers, inboxNames, tasks, messages);
const names = members.map((m) => m.name);
expect(names).not.toContain('user');
expect(names).toContain('team-lead');
expect(names).toContain('alice');
});
it('ignores qualified external inbox names unless explicitly configured', () => {
const resolver = new TeamMemberResolver();
const config: TeamConfig = {
name: 'Team',
members: [{ name: 'team-lead', agentType: 'team-lead', role: 'lead' }],
};
const metaMembers: TeamConfig['members'] = [{ name: 'alice', agentType: 'general-purpose' }];
const inboxNames = ['alice', 'team-best.user', 'dream-team.team-lead'];
const tasks: TeamTask[] = [];
const messages: InboxMessage[] = [];
const members = resolver.resolveMembers(config, metaMembers, inboxNames, tasks, messages);
const names = members.map((m) => m.name);
expect(names).toContain('alice');
expect(names).toContain('team-lead');
expect(names).not.toContain('team-best.user');
expect(names).not.toContain('dream-team.team-lead');
});
it('keeps dotted names when they are explicitly configured members', () => {
const resolver = new TeamMemberResolver();
const config: TeamConfig = {
name: 'Team',
members: [
{ name: 'team-lead', agentType: 'team-lead', role: 'lead' },
{ name: 'ops.bot', agentType: 'general-purpose' },
],
};
const members = resolver.resolveMembers(config, [], ['ops.bot'], [], []);
const names = members.map((m) => m.name);
expect(names).toContain('ops.bot');
});
it('keeps dotted names when config casing differs from inbox casing', () => {
const resolver = new TeamMemberResolver();
const config: TeamConfig = {
name: 'Team',
members: [
{ name: 'team-lead', agentType: 'team-lead', role: 'lead' },
{ name: 'Ops.Bot', agentType: 'general-purpose' },
],
};
const members = resolver.resolveMembers(config, [], ['ops.bot'], [], []);
const names = members.map((m) => m.name);
expect(names).toContain('Ops.Bot');
expect(names).not.toContain('ops.bot');
});
it('sets currentTaskId for in_progress task', () => {
const resolver = new TeamMemberResolver();
const config: TeamConfig = {
name: 'Team',
members: [{ name: 'bob', agentType: 'general-purpose' }],
};
const tasks: TeamTaskWithKanban[] = [
{ id: 't1', subject: 'Work', status: 'in_progress', owner: 'bob' },
];
const members = resolver.resolveMembers(config, [], [], tasks, []);
const bob = members.find((m) => m.name === 'bob');
expect(bob?.currentTaskId).toBe('t1');
});
it('clears currentTaskId when task is approved via kanbanColumn', () => {
const resolver = new TeamMemberResolver();
const config: TeamConfig = {
name: 'Team',
members: [{ name: 'bob', agentType: 'general-purpose' }],
};
const tasks: TeamTaskWithKanban[] = [
{
id: 't1',
subject: 'Work',
status: 'in_progress',
owner: 'bob',
reviewState: 'approved',
kanbanColumn: 'approved',
},
];
const members = resolver.resolveMembers(config, [], [], tasks, []);
const bob = members.find((m) => m.name === 'bob');
expect(bob?.currentTaskId).toBeNull();
});
it('clears currentTaskId when task reviewState is approved even without kanbanColumn', () => {
const resolver = new TeamMemberResolver();
const config: TeamConfig = {
name: 'Team',
members: [{ name: 'bob', agentType: 'general-purpose' }],
};
const tasks: TeamTaskWithKanban[] = [
{
id: 't1',
subject: 'Work',
status: 'in_progress',
owner: 'bob',
reviewState: 'approved',
// kanbanColumn not set — stale data scenario
},
];
const members = resolver.resolveMembers(config, [], [], tasks, []);
const bob = members.find((m) => m.name === 'bob');
expect(bob?.currentTaskId).toBeNull();
});
it('clears currentTaskId when task status is completed', () => {
const resolver = new TeamMemberResolver();
const config: TeamConfig = {
name: 'Team',
members: [{ name: 'bob', agentType: 'general-purpose' }],
};
const tasks: TeamTaskWithKanban[] = [
{ id: 't1', subject: 'Work', status: 'completed', owner: 'bob' },
];
const members = resolver.resolveMembers(config, [], [], tasks, []);
const bob = members.find((m) => m.name === 'bob');
expect(bob?.currentTaskId).toBeNull();
});
});