relayLeadInboxMessages only processes unread messages after provisioningComplete, but CLI marks permission_request messages as read after native delivery -- before our relay runs. Move permission_request inbox scan BEFORE provisioningComplete check. Scan ALL messages (including read=true), track processed IDs via processedPermissionRequestIds Set on ProvisioningRun to prevent re-emitting. Also look up both alive and provisioning runs so the scan works during team bootstrap.
156 lines
5.8 KiB
TypeScript
156 lines
5.8 KiB
TypeScript
import React, { act } from 'react';
|
|
import { createRoot } from 'react-dom/client';
|
|
import { afterEach, describe, expect, it, vi } from 'vitest';
|
|
|
|
vi.mock('@renderer/hooks/useTheme', () => ({
|
|
useTheme: () => ({ theme: 'dark', resolvedTheme: 'dark', isDark: true, isLight: false }),
|
|
}));
|
|
vi.mock('@renderer/components/chat/viewers/MarkdownViewer', () => ({
|
|
MarkdownViewer: ({ content }: { content: string }) => React.createElement('div', null, content),
|
|
}));
|
|
vi.mock('@renderer/components/common/CopyButton', () => ({
|
|
CopyButton: () => null,
|
|
}));
|
|
vi.mock('@renderer/components/team/attachments/AttachmentDisplay', () => ({
|
|
AttachmentDisplay: () => null,
|
|
}));
|
|
vi.mock('@renderer/components/team/MemberBadge', () => ({
|
|
MemberBadge: ({ name }: { name: string }) => React.createElement('span', null, name),
|
|
}));
|
|
vi.mock('@renderer/components/team/TaskTooltip', () => ({
|
|
TaskTooltip: ({ children }: { children: React.ReactNode }) => React.createElement(React.Fragment, null, children),
|
|
}));
|
|
vi.mock('@renderer/components/ui/ExpandableContent', () => ({
|
|
ExpandableContent: ({ children }: { children: React.ReactNode }) =>
|
|
React.createElement(React.Fragment, null, children),
|
|
}));
|
|
vi.mock('@renderer/components/ui/tooltip', () => ({
|
|
Tooltip: ({ children }: { children: React.ReactNode }) => React.createElement(React.Fragment, null, children),
|
|
TooltipTrigger: ({ children }: { children: React.ReactNode }) =>
|
|
React.createElement(React.Fragment, null, children),
|
|
TooltipContent: ({ children }: { children: React.ReactNode }) =>
|
|
React.createElement('div', null, children),
|
|
}));
|
|
vi.mock('@renderer/components/team/activity/ReplyQuoteBlock', () => ({
|
|
ReplyQuoteBlock: () => null,
|
|
}));
|
|
|
|
import {
|
|
ActivityItem,
|
|
getCrossTeamSentMemberName,
|
|
getCrossTeamSentTarget,
|
|
getSystemMessageLabel,
|
|
isQualifiedExternalRecipient,
|
|
} from '@renderer/components/team/activity/ActivityItem';
|
|
import type { InboxMessage } from '@shared/types';
|
|
|
|
describe('ActivityItem slash command rendering', () => {
|
|
afterEach(() => {
|
|
document.body.innerHTML = '';
|
|
vi.unstubAllGlobals();
|
|
});
|
|
|
|
it('renders standalone sent slash commands with command-specific styling content', async () => {
|
|
vi.stubGlobal('IS_REACT_ACT_ENVIRONMENT', true);
|
|
const host = document.createElement('div');
|
|
document.body.appendChild(host);
|
|
const root = createRoot(host);
|
|
|
|
const message: InboxMessage = {
|
|
from: 'user',
|
|
text: '/compact keep kanban aligned',
|
|
timestamp: new Date('2026-03-27T12:00:00.000Z').toISOString(),
|
|
read: true,
|
|
source: 'user_sent',
|
|
};
|
|
|
|
await act(async () => {
|
|
root.render(React.createElement(ActivityItem, { message, teamName: 'my-team' }));
|
|
await Promise.resolve();
|
|
});
|
|
|
|
expect(host.textContent).toContain('command');
|
|
expect(host.textContent).toContain('/compact');
|
|
expect(host.textContent).toContain('Compact conversation with optional focus instructions.');
|
|
expect(host.textContent).toContain('keep kanban aligned');
|
|
|
|
await act(async () => {
|
|
root.unmount();
|
|
await Promise.resolve();
|
|
});
|
|
});
|
|
|
|
it('renders slash command results as a distinct command output row', async () => {
|
|
vi.stubGlobal('IS_REACT_ACT_ENVIRONMENT', true);
|
|
const host = document.createElement('div');
|
|
document.body.appendChild(host);
|
|
const root = createRoot(host);
|
|
|
|
const message: InboxMessage = {
|
|
from: 'team-lead',
|
|
text: 'Model set to sonnet\nContext usage reset',
|
|
timestamp: new Date('2026-03-27T12:01:00.000Z').toISOString(),
|
|
read: true,
|
|
source: 'lead_session',
|
|
messageKind: 'slash_command_result',
|
|
commandOutput: {
|
|
stream: 'stdout',
|
|
commandLabel: '/model',
|
|
},
|
|
summary: 'Model set to sonnet',
|
|
};
|
|
|
|
await act(async () => {
|
|
root.render(React.createElement(ActivityItem, { message, teamName: 'my-team' }));
|
|
await Promise.resolve();
|
|
});
|
|
|
|
expect(host.textContent).toContain('result');
|
|
expect(host.textContent).toContain('stdout');
|
|
expect(host.textContent).toContain('/model');
|
|
expect(host.textContent).toContain('Model set to sonnet');
|
|
expect(host.textContent).toContain('Context usage reset');
|
|
expect(host.textContent).not.toContain('team-lead');
|
|
|
|
await act(async () => {
|
|
root.unmount();
|
|
await Promise.resolve();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('ActivityItem legacy system message fallback', () => {
|
|
it('recognizes historical assignment and review message wording', () => {
|
|
expect(getSystemMessageLabel('New task assigned to you: #abcd1234 "Implement feature".')).toBe(
|
|
'Task'
|
|
);
|
|
expect(getSystemMessageLabel('Task #abcd1234 approved by reviewer.')).toBe('Task approved');
|
|
expect(getSystemMessageLabel('Task #abcd1234 needs fixes before approval.')).toBe(
|
|
'Review changes requested'
|
|
);
|
|
});
|
|
|
|
it('does not treat new controller-authored summaries as legacy system noise', () => {
|
|
expect(getSystemMessageLabel('Review request for #abcd1234')).toBeNull();
|
|
expect(getSystemMessageLabel('Approved abcd1234')).toBeNull();
|
|
expect(getSystemMessageLabel('Fix request for abcd1234')).toBeNull();
|
|
});
|
|
|
|
it('does not classify dotted local teammates as external recipients', () => {
|
|
expect(isQualifiedExternalRecipient('ops.bot', 'my-team', new Set(['ops.bot']))).toBe(false);
|
|
expect(isQualifiedExternalRecipient('team-best.user', 'my-team', new Set(['ops.bot']))).toBe(
|
|
true
|
|
);
|
|
});
|
|
|
|
it('recognizes pseudo cross-team recipients in activity rows', () => {
|
|
expect(getCrossTeamSentTarget('cross-team:team-best', 'my-team', new Set(['ops.bot']))).toBe(
|
|
'team-best'
|
|
);
|
|
expect(getCrossTeamSentTarget('team-best.user', 'my-team', new Set(['ops.bot']))).toBe(
|
|
'team-best'
|
|
);
|
|
expect(getCrossTeamSentMemberName('team-best.user')).toBe('user');
|
|
expect(getCrossTeamSentMemberName('cross-team:team-best')).toBeNull();
|
|
});
|
|
});
|