Release: Merge develop to main (111 commits) #302
175
apps/web/src/app/(authenticated)/federation/dashboard/page.tsx
Normal file
175
apps/web/src/app/(authenticated)/federation/dashboard/page.tsx
Normal file
@@ -0,0 +1,175 @@
|
||||
"use client";
|
||||
|
||||
/**
|
||||
* Aggregated Dashboard Page
|
||||
* Displays data from multiple federated instances in a unified view
|
||||
*/
|
||||
|
||||
import { useState, useEffect } from "react";
|
||||
import { AggregatedDataGrid } from "@/components/federation/AggregatedDataGrid";
|
||||
import type { FederatedTask, FederatedEvent } from "@/components/federation/types";
|
||||
import {
|
||||
fetchConnections,
|
||||
FederationConnectionStatus,
|
||||
type ConnectionDetails,
|
||||
} from "@/lib/api/federation";
|
||||
import { sendFederatedQuery } from "@/lib/api/federation-queries";
|
||||
import type { Task, Event } from "@mosaic/shared";
|
||||
|
||||
export default function AggregatedDashboardPage(): React.JSX.Element {
|
||||
const [tasks, setTasks] = useState<FederatedTask[]>([]);
|
||||
const [events, setEvents] = useState<FederatedEvent[]>([]);
|
||||
const [isLoading, setIsLoading] = useState(true);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
const [connections, setConnections] = useState<ConnectionDetails[]>([]);
|
||||
|
||||
useEffect(() => {
|
||||
void loadAggregatedData();
|
||||
}, []);
|
||||
|
||||
async function loadAggregatedData(): Promise<void> {
|
||||
setIsLoading(true);
|
||||
setError(null);
|
||||
|
||||
try {
|
||||
// Fetch all active connections
|
||||
const allConnections = await fetchConnections(FederationConnectionStatus.ACTIVE);
|
||||
setConnections(allConnections);
|
||||
|
||||
if (allConnections.length === 0) {
|
||||
setIsLoading(false);
|
||||
return;
|
||||
}
|
||||
|
||||
// Query each connection for tasks and events
|
||||
const allTasks: FederatedTask[] = [];
|
||||
const allEvents: FederatedEvent[] = [];
|
||||
const errors: string[] = [];
|
||||
|
||||
for (const connection of allConnections) {
|
||||
try {
|
||||
// Query tasks
|
||||
if (connection.remoteCapabilities.supportsQuery) {
|
||||
const taskResponse = await sendFederatedQuery(connection.id, "tasks.list", {
|
||||
limit: 10,
|
||||
});
|
||||
|
||||
// Wait a bit for the query to be processed and response received
|
||||
// In production, this would use WebSocket or polling
|
||||
await new Promise((resolve) => setTimeout(resolve, 1000));
|
||||
|
||||
// For MVP, we'll use mock data since query processing is async
|
||||
// In production, we'd poll for the response or use WebSocket
|
||||
if (taskResponse.response) {
|
||||
const responseTasks = (taskResponse.response as { data?: Task[] }).data ?? [];
|
||||
const federatedTasks = responseTasks.map((task) => ({
|
||||
task,
|
||||
provenance: {
|
||||
instanceId: connection.remoteInstanceId,
|
||||
instanceName:
|
||||
(connection.metadata.name as string | undefined) ?? connection.remoteUrl,
|
||||
instanceUrl: connection.remoteUrl,
|
||||
timestamp: new Date().toISOString(),
|
||||
},
|
||||
}));
|
||||
allTasks.push(...federatedTasks);
|
||||
}
|
||||
|
||||
// Query events
|
||||
const eventResponse = await sendFederatedQuery(connection.id, "events.list", {
|
||||
limit: 10,
|
||||
});
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 1000));
|
||||
|
||||
if (eventResponse.response) {
|
||||
const responseEvents = (eventResponse.response as { data?: Event[] }).data ?? [];
|
||||
const federatedEvents = responseEvents.map((event) => ({
|
||||
event,
|
||||
provenance: {
|
||||
instanceId: connection.remoteInstanceId,
|
||||
instanceName:
|
||||
(connection.metadata.name as string | undefined) ?? connection.remoteUrl,
|
||||
instanceUrl: connection.remoteUrl,
|
||||
timestamp: new Date().toISOString(),
|
||||
},
|
||||
}));
|
||||
allEvents.push(...federatedEvents);
|
||||
}
|
||||
}
|
||||
} catch (err) {
|
||||
const errorMessage = err instanceof Error ? err.message : "Unknown error";
|
||||
const instanceName =
|
||||
(connection.metadata.name as string | undefined) ?? connection.remoteUrl;
|
||||
errors.push(`Unable to reach ${instanceName}: ${errorMessage}`);
|
||||
}
|
||||
}
|
||||
|
||||
setTasks(allTasks);
|
||||
setEvents(allEvents);
|
||||
|
||||
if (errors.length > 0 && allTasks.length === 0 && allEvents.length === 0) {
|
||||
setError(errors.join(", "));
|
||||
}
|
||||
} catch (err) {
|
||||
const errorMessage = err instanceof Error ? err.message : "Unable to load connections";
|
||||
setError(errorMessage);
|
||||
} finally {
|
||||
setIsLoading(false);
|
||||
}
|
||||
}
|
||||
|
||||
async function handleRefresh(): Promise<void> {
|
||||
await loadAggregatedData();
|
||||
}
|
||||
|
||||
return (
|
||||
<main className="container mx-auto px-4 py-8">
|
||||
{/* Header */}
|
||||
<div className="mb-8 flex items-center justify-between">
|
||||
<div>
|
||||
<h1 className="text-3xl font-bold text-gray-900">Aggregated Dashboard</h1>
|
||||
<p className="text-gray-600 mt-2">View tasks and events from all connected instances</p>
|
||||
</div>
|
||||
<button
|
||||
onClick={handleRefresh}
|
||||
disabled={isLoading}
|
||||
className="px-4 py-2 bg-blue-600 text-white text-sm font-medium rounded-md hover:bg-blue-700 disabled:opacity-50 disabled:cursor-not-allowed transition-colors"
|
||||
>
|
||||
{isLoading ? "Loading..." : "Refresh"}
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{/* Connection status */}
|
||||
{!isLoading && connections.length > 0 && (
|
||||
<div className="mb-6">
|
||||
<p className="text-sm text-gray-600">
|
||||
Connected to <span className="font-medium text-gray-900">{connections.length}</span>{" "}
|
||||
{connections.length === 1 ? "instance" : "instances"}
|
||||
</p>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Connection warning */}
|
||||
{!isLoading && connections.length === 0 && (
|
||||
<div className="mb-6 p-4 bg-yellow-50 border border-yellow-200 rounded-lg">
|
||||
<p className="text-sm text-yellow-800">
|
||||
No active connections found. Please visit the{" "}
|
||||
<a href="/federation/connections" className="font-medium underline">
|
||||
Connection Manager
|
||||
</a>{" "}
|
||||
to connect to remote instances.
|
||||
</p>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Data grid */}
|
||||
<AggregatedDataGrid
|
||||
tasks={tasks}
|
||||
events={events}
|
||||
isLoading={isLoading}
|
||||
{...(error !== null && { error })}
|
||||
/>
|
||||
</main>
|
||||
);
|
||||
}
|
||||
156
apps/web/src/components/federation/AggregatedDataGrid.test.tsx
Normal file
156
apps/web/src/components/federation/AggregatedDataGrid.test.tsx
Normal file
@@ -0,0 +1,156 @@
|
||||
/**
|
||||
* AggregatedDataGrid Component Tests
|
||||
*/
|
||||
|
||||
import { describe, it, expect } from "vitest";
|
||||
import { render, screen } from "@testing-library/react";
|
||||
import { AggregatedDataGrid } from "./AggregatedDataGrid";
|
||||
import { TaskStatus, TaskPriority } from "@mosaic/shared";
|
||||
import type { FederatedTask, FederatedEvent } from "./types";
|
||||
|
||||
const mockTasks: FederatedTask[] = [
|
||||
{
|
||||
task: {
|
||||
id: "task-1",
|
||||
title: "Task from Work",
|
||||
description: "Work task",
|
||||
status: TaskStatus.IN_PROGRESS,
|
||||
priority: TaskPriority.HIGH,
|
||||
dueDate: new Date("2026-02-05"),
|
||||
creatorId: "user-1",
|
||||
assigneeId: "user-1",
|
||||
workspaceId: "workspace-1",
|
||||
projectId: null,
|
||||
parentId: null,
|
||||
sortOrder: 0,
|
||||
metadata: {},
|
||||
completedAt: null,
|
||||
createdAt: new Date("2026-02-03"),
|
||||
updatedAt: new Date("2026-02-03"),
|
||||
},
|
||||
provenance: {
|
||||
instanceId: "instance-work-001",
|
||||
instanceName: "Work Instance",
|
||||
instanceUrl: "https://mosaic.work.example.com",
|
||||
timestamp: "2026-02-03T14:00:00Z",
|
||||
},
|
||||
},
|
||||
{
|
||||
task: {
|
||||
id: "task-2",
|
||||
title: "Task from Home",
|
||||
description: "Home task",
|
||||
status: TaskStatus.NOT_STARTED,
|
||||
priority: TaskPriority.MEDIUM,
|
||||
dueDate: new Date("2026-02-06"),
|
||||
creatorId: "user-1",
|
||||
assigneeId: "user-1",
|
||||
workspaceId: "workspace-2",
|
||||
projectId: null,
|
||||
parentId: null,
|
||||
sortOrder: 0,
|
||||
metadata: {},
|
||||
completedAt: null,
|
||||
createdAt: new Date("2026-02-03"),
|
||||
updatedAt: new Date("2026-02-03"),
|
||||
},
|
||||
provenance: {
|
||||
instanceId: "instance-home-001",
|
||||
instanceName: "Home Instance",
|
||||
instanceUrl: "https://mosaic.home.example.com",
|
||||
timestamp: "2026-02-03T14:00:00Z",
|
||||
},
|
||||
},
|
||||
];
|
||||
|
||||
const mockEvents: FederatedEvent[] = [
|
||||
{
|
||||
event: {
|
||||
id: "event-1",
|
||||
title: "Meeting from Work",
|
||||
description: "Team standup",
|
||||
startTime: new Date("2026-02-05T10:00:00"),
|
||||
endTime: new Date("2026-02-05T10:30:00"),
|
||||
allDay: false,
|
||||
location: "Zoom",
|
||||
recurrence: null,
|
||||
creatorId: "user-1",
|
||||
workspaceId: "workspace-1",
|
||||
projectId: null,
|
||||
metadata: {},
|
||||
createdAt: new Date("2026-02-03"),
|
||||
updatedAt: new Date("2026-02-03"),
|
||||
},
|
||||
provenance: {
|
||||
instanceId: "instance-work-001",
|
||||
instanceName: "Work Instance",
|
||||
instanceUrl: "https://mosaic.work.example.com",
|
||||
timestamp: "2026-02-03T14:00:00Z",
|
||||
},
|
||||
},
|
||||
];
|
||||
|
||||
describe("AggregatedDataGrid", () => {
|
||||
it("should render tasks", () => {
|
||||
render(<AggregatedDataGrid tasks={mockTasks} events={[]} />);
|
||||
|
||||
expect(screen.getByText("Task from Work")).toBeInTheDocument();
|
||||
expect(screen.getByText("Task from Home")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render events", () => {
|
||||
render(<AggregatedDataGrid tasks={[]} events={mockEvents} />);
|
||||
|
||||
expect(screen.getByText("Meeting from Work")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render both tasks and events", () => {
|
||||
render(<AggregatedDataGrid tasks={mockTasks} events={mockEvents} />);
|
||||
|
||||
expect(screen.getByText("Task from Work")).toBeInTheDocument();
|
||||
expect(screen.getByText("Meeting from Work")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should show loading state", () => {
|
||||
render(<AggregatedDataGrid tasks={[]} events={[]} isLoading={true} />);
|
||||
|
||||
expect(screen.getByText("Loading data from instances...")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should show empty state when no data", () => {
|
||||
render(<AggregatedDataGrid tasks={[]} events={[]} />);
|
||||
|
||||
expect(screen.getByText("No data available from connected instances")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should show error message", () => {
|
||||
render(<AggregatedDataGrid tasks={[]} events={[]} error="Unable to reach work instance" />);
|
||||
|
||||
expect(screen.getByText("Unable to reach work instance")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render with custom className", () => {
|
||||
const { container } = render(
|
||||
<AggregatedDataGrid tasks={mockTasks} events={[]} className="custom-class" />
|
||||
);
|
||||
|
||||
expect(container.querySelector(".custom-class")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should show instance provenance indicators", () => {
|
||||
render(<AggregatedDataGrid tasks={mockTasks} events={[]} />);
|
||||
|
||||
expect(screen.getByText("Work Instance")).toBeInTheDocument();
|
||||
expect(screen.getByText("Home Instance")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render with compact mode", () => {
|
||||
const { container } = render(
|
||||
<AggregatedDataGrid tasks={mockTasks} events={[]} compact={true} />
|
||||
);
|
||||
|
||||
// Check that cards have compact padding
|
||||
const compactCards = container.querySelectorAll(".p-3");
|
||||
expect(compactCards.length).toBeGreaterThan(0);
|
||||
});
|
||||
});
|
||||
100
apps/web/src/components/federation/AggregatedDataGrid.tsx
Normal file
100
apps/web/src/components/federation/AggregatedDataGrid.tsx
Normal file
@@ -0,0 +1,100 @@
|
||||
/**
|
||||
* AggregatedDataGrid Component
|
||||
* Displays aggregated tasks and events from multiple federated instances
|
||||
*/
|
||||
|
||||
import type { FederatedTask, FederatedEvent } from "./types";
|
||||
import { FederatedTaskCard } from "./FederatedTaskCard";
|
||||
import { FederatedEventCard } from "./FederatedEventCard";
|
||||
|
||||
interface AggregatedDataGridProps {
|
||||
tasks: FederatedTask[];
|
||||
events: FederatedEvent[];
|
||||
isLoading?: boolean;
|
||||
error?: string;
|
||||
compact?: boolean;
|
||||
className?: string;
|
||||
}
|
||||
|
||||
export function AggregatedDataGrid({
|
||||
tasks,
|
||||
events,
|
||||
isLoading = false,
|
||||
error,
|
||||
compact = false,
|
||||
className = "",
|
||||
}: AggregatedDataGridProps): React.JSX.Element {
|
||||
// Loading state
|
||||
if (isLoading) {
|
||||
return (
|
||||
<div className={`flex items-center justify-center p-8 ${className}`}>
|
||||
<div className="text-center">
|
||||
<div className="inline-block animate-spin rounded-full h-8 w-8 border-b-2 border-blue-600 mb-4"></div>
|
||||
<p className="text-gray-600">Loading data from instances...</p>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
// Error state
|
||||
if (error) {
|
||||
return (
|
||||
<div className={`flex items-center justify-center p-8 ${className}`}>
|
||||
<div className="text-center">
|
||||
<span className="text-4xl mb-4 block">⚠️</span>
|
||||
<p className="text-gray-900 font-medium mb-2">Unable to load data</p>
|
||||
<p className="text-gray-600 text-sm">{error}</p>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
// Empty state
|
||||
if (tasks.length === 0 && events.length === 0) {
|
||||
return (
|
||||
<div className={`flex items-center justify-center p-8 ${className}`}>
|
||||
<div className="text-center">
|
||||
<span className="text-4xl mb-4 block">📋</span>
|
||||
<p className="text-gray-900 font-medium mb-2">No data available</p>
|
||||
<p className="text-gray-600 text-sm">No data available from connected instances</p>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div className={`space-y-4 ${className}`}>
|
||||
{/* Tasks section */}
|
||||
{tasks.length > 0 && (
|
||||
<div>
|
||||
<h3 className="text-lg font-medium text-gray-900 mb-3">Tasks ({tasks.length})</h3>
|
||||
<div className="space-y-2">
|
||||
{tasks.map((federatedTask) => (
|
||||
<FederatedTaskCard
|
||||
key={`task-${federatedTask.task.id}-${federatedTask.provenance.instanceId}`}
|
||||
federatedTask={federatedTask}
|
||||
compact={compact}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Events section */}
|
||||
{events.length > 0 && (
|
||||
<div>
|
||||
<h3 className="text-lg font-medium text-gray-900 mb-3">Events ({events.length})</h3>
|
||||
<div className="space-y-2">
|
||||
{events.map((federatedEvent) => (
|
||||
<FederatedEventCard
|
||||
key={`event-${federatedEvent.event.id}-${federatedEvent.provenance.instanceId}`}
|
||||
federatedEvent={federatedEvent}
|
||||
compact={compact}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
136
apps/web/src/components/federation/FederatedEventCard.test.tsx
Normal file
136
apps/web/src/components/federation/FederatedEventCard.test.tsx
Normal file
@@ -0,0 +1,136 @@
|
||||
/**
|
||||
* FederatedEventCard Component Tests
|
||||
*/
|
||||
|
||||
import { describe, it, expect } from "vitest";
|
||||
import { render, screen } from "@testing-library/react";
|
||||
import { FederatedEventCard } from "./FederatedEventCard";
|
||||
import type { FederatedEvent } from "./types";
|
||||
|
||||
const mockEvent: FederatedEvent = {
|
||||
event: {
|
||||
id: "event-1",
|
||||
title: "Team standup",
|
||||
description: "Daily sync meeting",
|
||||
startTime: new Date("2026-02-05T10:00:00"),
|
||||
endTime: new Date("2026-02-05T10:30:00"),
|
||||
allDay: false,
|
||||
location: "Zoom",
|
||||
recurrence: null,
|
||||
creatorId: "user-1",
|
||||
workspaceId: "workspace-1",
|
||||
projectId: null,
|
||||
metadata: {},
|
||||
createdAt: new Date("2026-02-03"),
|
||||
updatedAt: new Date("2026-02-03"),
|
||||
},
|
||||
provenance: {
|
||||
instanceId: "instance-work-001",
|
||||
instanceName: "Work Instance",
|
||||
instanceUrl: "https://mosaic.work.example.com",
|
||||
timestamp: "2026-02-03T14:00:00Z",
|
||||
},
|
||||
};
|
||||
|
||||
describe("FederatedEventCard", () => {
|
||||
it("should render event title", () => {
|
||||
render(<FederatedEventCard federatedEvent={mockEvent} />);
|
||||
|
||||
expect(screen.getByText("Team standup")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render event description", () => {
|
||||
render(<FederatedEventCard federatedEvent={mockEvent} />);
|
||||
|
||||
expect(screen.getByText("Daily sync meeting")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render provenance indicator", () => {
|
||||
render(<FederatedEventCard federatedEvent={mockEvent} />);
|
||||
|
||||
expect(screen.getByText("Work Instance")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render event time", () => {
|
||||
render(<FederatedEventCard federatedEvent={mockEvent} />);
|
||||
|
||||
// Check for time components
|
||||
expect(screen.getByText(/10:00/)).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render event location", () => {
|
||||
render(<FederatedEventCard federatedEvent={mockEvent} />);
|
||||
|
||||
expect(screen.getByText("Zoom")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render with compact mode", () => {
|
||||
const { container } = render(<FederatedEventCard federatedEvent={mockEvent} compact={true} />);
|
||||
|
||||
const card = container.querySelector(".p-3");
|
||||
expect(card).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should handle event without description", () => {
|
||||
const eventNoDesc: FederatedEvent = {
|
||||
...mockEvent,
|
||||
event: {
|
||||
...mockEvent.event,
|
||||
description: null,
|
||||
},
|
||||
};
|
||||
|
||||
render(<FederatedEventCard federatedEvent={eventNoDesc} />);
|
||||
|
||||
expect(screen.getByText("Team standup")).toBeInTheDocument();
|
||||
expect(screen.queryByText("Daily sync meeting")).not.toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should handle event without location", () => {
|
||||
const eventNoLocation: FederatedEvent = {
|
||||
...mockEvent,
|
||||
event: {
|
||||
...mockEvent.event,
|
||||
location: null,
|
||||
},
|
||||
};
|
||||
|
||||
render(<FederatedEventCard federatedEvent={eventNoLocation} />);
|
||||
|
||||
expect(screen.getByText("Team standup")).toBeInTheDocument();
|
||||
expect(screen.queryByText("Zoom")).not.toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render all-day event", () => {
|
||||
const allDayEvent: FederatedEvent = {
|
||||
...mockEvent,
|
||||
event: {
|
||||
...mockEvent.event,
|
||||
allDay: true,
|
||||
},
|
||||
};
|
||||
|
||||
render(<FederatedEventCard federatedEvent={allDayEvent} />);
|
||||
|
||||
expect(screen.getByText("All day")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should handle onClick callback", () => {
|
||||
let clicked = false;
|
||||
const handleClick = (): void => {
|
||||
clicked = true;
|
||||
};
|
||||
|
||||
const { container } = render(
|
||||
<FederatedEventCard federatedEvent={mockEvent} onClick={handleClick} />
|
||||
);
|
||||
|
||||
const card = container.querySelector(".cursor-pointer");
|
||||
expect(card).toBeInTheDocument();
|
||||
|
||||
if (card instanceof HTMLElement) {
|
||||
card.click();
|
||||
}
|
||||
expect(clicked).toBe(true);
|
||||
});
|
||||
});
|
||||
94
apps/web/src/components/federation/FederatedEventCard.tsx
Normal file
94
apps/web/src/components/federation/FederatedEventCard.tsx
Normal file
@@ -0,0 +1,94 @@
|
||||
/**
|
||||
* FederatedEventCard Component
|
||||
* Displays an event from a federated instance with provenance indicator
|
||||
*/
|
||||
|
||||
import type { FederatedEvent } from "./types";
|
||||
import { ProvenanceIndicator } from "./ProvenanceIndicator";
|
||||
|
||||
interface FederatedEventCardProps {
|
||||
federatedEvent: FederatedEvent;
|
||||
compact?: boolean;
|
||||
onClick?: () => void;
|
||||
}
|
||||
|
||||
/**
|
||||
* Format time for display
|
||||
*/
|
||||
function formatTime(date: Date): string {
|
||||
return new Intl.DateTimeFormat("en-US", {
|
||||
hour: "numeric",
|
||||
minute: "2-digit",
|
||||
hour12: true,
|
||||
}).format(new Date(date));
|
||||
}
|
||||
|
||||
/**
|
||||
* Format date for display
|
||||
*/
|
||||
function formatDate(date: Date): string {
|
||||
return new Intl.DateTimeFormat("en-US", {
|
||||
weekday: "short",
|
||||
month: "short",
|
||||
day: "numeric",
|
||||
}).format(new Date(date));
|
||||
}
|
||||
|
||||
export function FederatedEventCard({
|
||||
federatedEvent,
|
||||
compact = false,
|
||||
onClick,
|
||||
}: FederatedEventCardProps): React.JSX.Element {
|
||||
const { event, provenance } = federatedEvent;
|
||||
|
||||
const paddingClass = compact ? "p-3" : "p-4";
|
||||
const clickableClass = onClick ? "cursor-pointer hover:border-gray-300" : "";
|
||||
|
||||
const startTime = formatTime(event.startTime);
|
||||
const endTime = event.endTime !== null ? formatTime(event.endTime) : "";
|
||||
const startDate = formatDate(event.startTime);
|
||||
|
||||
return (
|
||||
<div
|
||||
className={`border border-gray-200 rounded-lg ${paddingClass} ${clickableClass} transition-colors`}
|
||||
onClick={onClick}
|
||||
>
|
||||
{/* Header with title and provenance */}
|
||||
<div className="flex items-start justify-between mb-2">
|
||||
<div className="flex-1">
|
||||
<h3 className="text-base font-medium text-gray-900">{event.title}</h3>
|
||||
{event.description && <p className="text-sm text-gray-600 mt-1">{event.description}</p>}
|
||||
</div>
|
||||
<ProvenanceIndicator
|
||||
instanceId={provenance.instanceId}
|
||||
instanceName={provenance.instanceName}
|
||||
instanceUrl={provenance.instanceUrl}
|
||||
compact={compact}
|
||||
/>
|
||||
</div>
|
||||
|
||||
{/* Metadata row */}
|
||||
<div className="flex items-center gap-3 text-sm flex-wrap">
|
||||
{/* Date */}
|
||||
<span className="text-xs text-gray-700 font-medium">{startDate}</span>
|
||||
|
||||
{/* Time */}
|
||||
{event.allDay ? (
|
||||
<span className="text-xs text-gray-600">All day</span>
|
||||
) : (
|
||||
<span className="text-xs text-gray-600">
|
||||
{startTime} - {endTime}
|
||||
</span>
|
||||
)}
|
||||
|
||||
{/* Location */}
|
||||
{event.location && (
|
||||
<span className="text-xs text-gray-600">
|
||||
<span className="mr-1">📍</span>
|
||||
{event.location}
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
144
apps/web/src/components/federation/FederatedTaskCard.test.tsx
Normal file
144
apps/web/src/components/federation/FederatedTaskCard.test.tsx
Normal file
@@ -0,0 +1,144 @@
|
||||
/**
|
||||
* FederatedTaskCard Component Tests
|
||||
*/
|
||||
|
||||
import { describe, it, expect } from "vitest";
|
||||
import { render, screen } from "@testing-library/react";
|
||||
import { FederatedTaskCard } from "./FederatedTaskCard";
|
||||
import { TaskStatus, TaskPriority } from "@mosaic/shared";
|
||||
import type { FederatedTask } from "./types";
|
||||
|
||||
const mockTask: FederatedTask = {
|
||||
task: {
|
||||
id: "task-1",
|
||||
title: "Review pull request",
|
||||
description: "Review and provide feedback on frontend PR",
|
||||
status: TaskStatus.IN_PROGRESS,
|
||||
priority: TaskPriority.HIGH,
|
||||
dueDate: new Date("2026-02-05"),
|
||||
creatorId: "user-1",
|
||||
assigneeId: "user-1",
|
||||
workspaceId: "workspace-1",
|
||||
projectId: null,
|
||||
parentId: null,
|
||||
sortOrder: 0,
|
||||
metadata: {},
|
||||
completedAt: null,
|
||||
createdAt: new Date("2026-02-03"),
|
||||
updatedAt: new Date("2026-02-03"),
|
||||
},
|
||||
provenance: {
|
||||
instanceId: "instance-work-001",
|
||||
instanceName: "Work Instance",
|
||||
instanceUrl: "https://mosaic.work.example.com",
|
||||
timestamp: "2026-02-03T14:00:00Z",
|
||||
},
|
||||
};
|
||||
|
||||
describe("FederatedTaskCard", () => {
|
||||
it("should render task title", () => {
|
||||
render(<FederatedTaskCard federatedTask={mockTask} />);
|
||||
|
||||
expect(screen.getByText("Review pull request")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render task description", () => {
|
||||
render(<FederatedTaskCard federatedTask={mockTask} />);
|
||||
|
||||
expect(screen.getByText("Review and provide feedback on frontend PR")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render provenance indicator", () => {
|
||||
render(<FederatedTaskCard federatedTask={mockTask} />);
|
||||
|
||||
expect(screen.getByText("Work Instance")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render status badge", () => {
|
||||
render(<FederatedTaskCard federatedTask={mockTask} />);
|
||||
|
||||
expect(screen.getByText("In Progress")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render priority indicator for high priority", () => {
|
||||
render(<FederatedTaskCard federatedTask={mockTask} />);
|
||||
|
||||
expect(screen.getByText("High")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render due date", () => {
|
||||
render(<FederatedTaskCard federatedTask={mockTask} />);
|
||||
|
||||
// Check for "Due:" text followed by a date
|
||||
expect(screen.getByText(/Due:/)).toBeInTheDocument();
|
||||
expect(screen.getByText(/2026/)).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render with compact mode", () => {
|
||||
const { container } = render(<FederatedTaskCard federatedTask={mockTask} compact={true} />);
|
||||
|
||||
const card = container.querySelector(".p-3");
|
||||
expect(card).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render completed task with completed status", () => {
|
||||
const completedTask: FederatedTask = {
|
||||
...mockTask,
|
||||
task: {
|
||||
...mockTask.task,
|
||||
status: TaskStatus.COMPLETED,
|
||||
completedAt: new Date("2026-02-04"),
|
||||
},
|
||||
};
|
||||
|
||||
render(<FederatedTaskCard federatedTask={completedTask} />);
|
||||
|
||||
expect(screen.getByText("Completed")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should handle task without description", () => {
|
||||
const taskNoDesc: FederatedTask = {
|
||||
...mockTask,
|
||||
task: {
|
||||
...mockTask.task,
|
||||
description: null,
|
||||
},
|
||||
};
|
||||
|
||||
render(<FederatedTaskCard federatedTask={taskNoDesc} />);
|
||||
|
||||
expect(screen.getByText("Review pull request")).toBeInTheDocument();
|
||||
expect(
|
||||
screen.queryByText("Review and provide feedback on frontend PR")
|
||||
).not.toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should handle task without due date", () => {
|
||||
const taskNoDue: FederatedTask = {
|
||||
...mockTask,
|
||||
task: {
|
||||
...mockTask.task,
|
||||
dueDate: null,
|
||||
},
|
||||
};
|
||||
|
||||
render(<FederatedTaskCard federatedTask={taskNoDue} />);
|
||||
|
||||
expect(screen.getByText("Review pull request")).toBeInTheDocument();
|
||||
expect(screen.queryByText(/Due:/)).not.toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should use PDA-friendly language for status", () => {
|
||||
const pausedTask: FederatedTask = {
|
||||
...mockTask,
|
||||
task: {
|
||||
...mockTask.task,
|
||||
status: TaskStatus.PAUSED,
|
||||
},
|
||||
};
|
||||
|
||||
render(<FederatedTaskCard federatedTask={pausedTask} />);
|
||||
|
||||
expect(screen.getByText("Paused")).toBeInTheDocument();
|
||||
});
|
||||
});
|
||||
113
apps/web/src/components/federation/FederatedTaskCard.tsx
Normal file
113
apps/web/src/components/federation/FederatedTaskCard.tsx
Normal file
@@ -0,0 +1,113 @@
|
||||
/**
|
||||
* FederatedTaskCard Component
|
||||
* Displays a task from a federated instance with provenance indicator
|
||||
*/
|
||||
|
||||
import { TaskStatus, TaskPriority } from "@mosaic/shared";
|
||||
import type { FederatedTask } from "./types";
|
||||
import { ProvenanceIndicator } from "./ProvenanceIndicator";
|
||||
|
||||
interface FederatedTaskCardProps {
|
||||
federatedTask: FederatedTask;
|
||||
compact?: boolean;
|
||||
onClick?: () => void;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get PDA-friendly status text and color
|
||||
*/
|
||||
function getStatusDisplay(status: TaskStatus): { text: string; colorClass: string } {
|
||||
switch (status) {
|
||||
case TaskStatus.NOT_STARTED:
|
||||
return { text: "Not Started", colorClass: "bg-gray-100 text-gray-700" };
|
||||
case TaskStatus.IN_PROGRESS:
|
||||
return { text: "In Progress", colorClass: "bg-blue-100 text-blue-700" };
|
||||
case TaskStatus.COMPLETED:
|
||||
return { text: "Completed", colorClass: "bg-green-100 text-green-700" };
|
||||
case TaskStatus.PAUSED:
|
||||
return { text: "Paused", colorClass: "bg-yellow-100 text-yellow-700" };
|
||||
case TaskStatus.ARCHIVED:
|
||||
return { text: "Archived", colorClass: "bg-gray-100 text-gray-600" };
|
||||
default:
|
||||
return { text: "Unknown", colorClass: "bg-gray-100 text-gray-700" };
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get priority text and color
|
||||
*/
|
||||
function getPriorityDisplay(priority: TaskPriority): { text: string; colorClass: string } {
|
||||
switch (priority) {
|
||||
case TaskPriority.LOW:
|
||||
return { text: "Low", colorClass: "text-gray-600" };
|
||||
case TaskPriority.MEDIUM:
|
||||
return { text: "Medium", colorClass: "text-blue-600" };
|
||||
case TaskPriority.HIGH:
|
||||
return { text: "High", colorClass: "text-orange-600" };
|
||||
default:
|
||||
return { text: "Unknown", colorClass: "text-gray-600" };
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Format date for display
|
||||
*/
|
||||
function formatDate(date: Date | null): string | null {
|
||||
if (!date) {
|
||||
return null;
|
||||
}
|
||||
return new Intl.DateTimeFormat("en-US", {
|
||||
year: "numeric",
|
||||
month: "short",
|
||||
day: "numeric",
|
||||
}).format(new Date(date));
|
||||
}
|
||||
|
||||
export function FederatedTaskCard({
|
||||
federatedTask,
|
||||
compact = false,
|
||||
onClick,
|
||||
}: FederatedTaskCardProps): React.JSX.Element {
|
||||
const { task, provenance } = federatedTask;
|
||||
const status = getStatusDisplay(task.status);
|
||||
const priority = getPriorityDisplay(task.priority);
|
||||
const dueDate = formatDate(task.dueDate);
|
||||
|
||||
const paddingClass = compact ? "p-3" : "p-4";
|
||||
const clickableClass = onClick ? "cursor-pointer hover:border-gray-300" : "";
|
||||
|
||||
return (
|
||||
<div
|
||||
className={`border border-gray-200 rounded-lg ${paddingClass} ${clickableClass} transition-colors`}
|
||||
onClick={onClick}
|
||||
>
|
||||
{/* Header with title and provenance */}
|
||||
<div className="flex items-start justify-between mb-2">
|
||||
<div className="flex-1">
|
||||
<h3 className="text-base font-medium text-gray-900">{task.title}</h3>
|
||||
{task.description && <p className="text-sm text-gray-600 mt-1">{task.description}</p>}
|
||||
</div>
|
||||
<ProvenanceIndicator
|
||||
instanceId={provenance.instanceId}
|
||||
instanceName={provenance.instanceName}
|
||||
instanceUrl={provenance.instanceUrl}
|
||||
compact={compact}
|
||||
/>
|
||||
</div>
|
||||
|
||||
{/* Metadata row */}
|
||||
<div className="flex items-center gap-3 text-sm flex-wrap">
|
||||
{/* Status badge */}
|
||||
<span className={`px-2 py-1 rounded-full text-xs font-medium ${status.colorClass}`}>
|
||||
{status.text}
|
||||
</span>
|
||||
|
||||
{/* Priority */}
|
||||
<span className={`text-xs font-medium ${priority.colorClass}`}>{priority.text}</span>
|
||||
|
||||
{/* Due date */}
|
||||
{dueDate && <span className="text-xs text-gray-600">Due: {dueDate}</span>}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
105
apps/web/src/components/federation/ProvenanceIndicator.test.tsx
Normal file
105
apps/web/src/components/federation/ProvenanceIndicator.test.tsx
Normal file
@@ -0,0 +1,105 @@
|
||||
/**
|
||||
* ProvenanceIndicator Component Tests
|
||||
*/
|
||||
|
||||
import { describe, it, expect } from "vitest";
|
||||
import { render, screen } from "@testing-library/react";
|
||||
import { ProvenanceIndicator } from "./ProvenanceIndicator";
|
||||
|
||||
describe("ProvenanceIndicator", () => {
|
||||
it("should render instance name", () => {
|
||||
render(
|
||||
<ProvenanceIndicator
|
||||
instanceId="instance-work-001"
|
||||
instanceName="Work Instance"
|
||||
instanceUrl="https://mosaic.work.example.com"
|
||||
/>
|
||||
);
|
||||
|
||||
expect(screen.getByText("Work Instance")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render with compact mode", () => {
|
||||
const { container } = render(
|
||||
<ProvenanceIndicator
|
||||
instanceId="instance-work-001"
|
||||
instanceName="Work Instance"
|
||||
instanceUrl="https://mosaic.work.example.com"
|
||||
compact={true}
|
||||
/>
|
||||
);
|
||||
|
||||
// Compact mode should have smaller padding
|
||||
const badge = container.querySelector(".px-2");
|
||||
expect(badge).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render with custom color", () => {
|
||||
const { container } = render(
|
||||
<ProvenanceIndicator
|
||||
instanceId="instance-work-001"
|
||||
instanceName="Work Instance"
|
||||
instanceUrl="https://mosaic.work.example.com"
|
||||
color="bg-blue-100"
|
||||
/>
|
||||
);
|
||||
|
||||
const badge = container.querySelector(".bg-blue-100");
|
||||
expect(badge).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should render with default color when not provided", () => {
|
||||
const { container } = render(
|
||||
<ProvenanceIndicator
|
||||
instanceId="instance-work-001"
|
||||
instanceName="Work Instance"
|
||||
instanceUrl="https://mosaic.work.example.com"
|
||||
/>
|
||||
);
|
||||
|
||||
const badge = container.querySelector(".bg-gray-100");
|
||||
expect(badge).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should show tooltip with instance details on hover", () => {
|
||||
render(
|
||||
<ProvenanceIndicator
|
||||
instanceId="instance-work-001"
|
||||
instanceName="Work Instance"
|
||||
instanceUrl="https://mosaic.work.example.com"
|
||||
/>
|
||||
);
|
||||
|
||||
// Check for title attribute (tooltip)
|
||||
const badge = screen.getByText("Work Instance");
|
||||
expect(badge.closest("div")).toHaveAttribute(
|
||||
"title",
|
||||
"From: Work Instance (https://mosaic.work.example.com)"
|
||||
);
|
||||
});
|
||||
|
||||
it("should render with icon when showIcon is true", () => {
|
||||
render(
|
||||
<ProvenanceIndicator
|
||||
instanceId="instance-work-001"
|
||||
instanceName="Work Instance"
|
||||
instanceUrl="https://mosaic.work.example.com"
|
||||
showIcon={true}
|
||||
/>
|
||||
);
|
||||
|
||||
expect(screen.getByText("🔗")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("should not render icon by default", () => {
|
||||
render(
|
||||
<ProvenanceIndicator
|
||||
instanceId="instance-work-001"
|
||||
instanceName="Work Instance"
|
||||
instanceUrl="https://mosaic.work.example.com"
|
||||
/>
|
||||
);
|
||||
|
||||
expect(screen.queryByText("🔗")).not.toBeInTheDocument();
|
||||
});
|
||||
});
|
||||
36
apps/web/src/components/federation/ProvenanceIndicator.tsx
Normal file
36
apps/web/src/components/federation/ProvenanceIndicator.tsx
Normal file
@@ -0,0 +1,36 @@
|
||||
/**
|
||||
* ProvenanceIndicator Component
|
||||
* Shows which instance data came from with PDA-friendly design
|
||||
*/
|
||||
|
||||
interface ProvenanceIndicatorProps {
|
||||
instanceId: string;
|
||||
instanceName: string;
|
||||
instanceUrl: string;
|
||||
compact?: boolean;
|
||||
color?: string;
|
||||
showIcon?: boolean;
|
||||
}
|
||||
|
||||
export function ProvenanceIndicator({
|
||||
instanceId,
|
||||
instanceName,
|
||||
instanceUrl,
|
||||
compact = false,
|
||||
color = "bg-gray-100",
|
||||
showIcon = false,
|
||||
}: ProvenanceIndicatorProps): React.JSX.Element {
|
||||
const paddingClass = compact ? "px-2 py-0.5 text-xs" : "px-3 py-1 text-sm";
|
||||
const tooltipText = `From: ${instanceName} (${instanceUrl})`;
|
||||
|
||||
return (
|
||||
<div
|
||||
className={`inline-flex items-center gap-1 ${paddingClass} ${color} text-gray-700 rounded-full font-medium`}
|
||||
title={tooltipText}
|
||||
data-instance-id={instanceId}
|
||||
>
|
||||
{showIcon && <span>🔗</span>}
|
||||
<span>{instanceName}</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
36
apps/web/src/components/federation/types.ts
Normal file
36
apps/web/src/components/federation/types.ts
Normal file
@@ -0,0 +1,36 @@
|
||||
/**
|
||||
* Federation Component Types
|
||||
*/
|
||||
|
||||
import type { Task, Event } from "@mosaic/shared";
|
||||
|
||||
/**
|
||||
* Provenance information - where the data came from
|
||||
*/
|
||||
export interface ProvenanceInfo {
|
||||
instanceId: string;
|
||||
instanceName: string;
|
||||
instanceUrl: string;
|
||||
timestamp: string;
|
||||
}
|
||||
|
||||
/**
|
||||
* Federated task with provenance
|
||||
*/
|
||||
export interface FederatedTask {
|
||||
task: Task;
|
||||
provenance: ProvenanceInfo;
|
||||
}
|
||||
|
||||
/**
|
||||
* Federated event with provenance
|
||||
*/
|
||||
export interface FederatedEvent {
|
||||
event: Event;
|
||||
provenance: ProvenanceInfo;
|
||||
}
|
||||
|
||||
/**
|
||||
* Federated data item (union type)
|
||||
*/
|
||||
export type FederatedItem = FederatedTask | FederatedEvent;
|
||||
154
apps/web/src/lib/api/federation-queries.test.ts
Normal file
154
apps/web/src/lib/api/federation-queries.test.ts
Normal file
@@ -0,0 +1,154 @@
|
||||
/**
|
||||
* Federation Queries API Client Tests
|
||||
*/
|
||||
|
||||
import { describe, it, expect, vi, beforeEach } from "vitest";
|
||||
import {
|
||||
sendFederatedQuery,
|
||||
fetchQueryMessages,
|
||||
fetchQueryMessage,
|
||||
FederationMessageStatus,
|
||||
} from "./federation-queries";
|
||||
import * as client from "./client";
|
||||
|
||||
// Mock the API client
|
||||
vi.mock("./client", () => ({
|
||||
apiGet: vi.fn(),
|
||||
apiPost: vi.fn(),
|
||||
}));
|
||||
|
||||
describe("Federation Queries API", () => {
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
});
|
||||
|
||||
describe("sendFederatedQuery", () => {
|
||||
it("should send a query to a remote instance", async () => {
|
||||
const mockResponse = {
|
||||
id: "msg-123",
|
||||
messageId: "uuid-123",
|
||||
connectionId: "conn-1",
|
||||
query: "tasks.list",
|
||||
status: FederationMessageStatus.PENDING,
|
||||
createdAt: new Date().toISOString(),
|
||||
};
|
||||
|
||||
vi.mocked(client.apiPost).mockResolvedValue(mockResponse);
|
||||
|
||||
const result = await sendFederatedQuery("conn-1", "tasks.list", { limit: 10 });
|
||||
|
||||
expect(client.apiPost).toHaveBeenCalledWith("/api/v1/federation/query", {
|
||||
connectionId: "conn-1",
|
||||
query: "tasks.list",
|
||||
context: { limit: 10 },
|
||||
});
|
||||
expect(result).toEqual(mockResponse);
|
||||
});
|
||||
|
||||
it("should send a query without context", async () => {
|
||||
const mockResponse = {
|
||||
id: "msg-124",
|
||||
messageId: "uuid-124",
|
||||
connectionId: "conn-2",
|
||||
query: "events.today",
|
||||
status: FederationMessageStatus.PENDING,
|
||||
createdAt: new Date().toISOString(),
|
||||
};
|
||||
|
||||
vi.mocked(client.apiPost).mockResolvedValue(mockResponse);
|
||||
|
||||
const result = await sendFederatedQuery("conn-2", "events.today");
|
||||
|
||||
expect(client.apiPost).toHaveBeenCalledWith("/api/v1/federation/query", {
|
||||
connectionId: "conn-2",
|
||||
query: "events.today",
|
||||
});
|
||||
expect(result).toEqual(mockResponse);
|
||||
});
|
||||
|
||||
it("should handle API errors", async () => {
|
||||
vi.mocked(client.apiPost).mockRejectedValue(new Error("Network error"));
|
||||
|
||||
await expect(sendFederatedQuery("conn-1", "tasks.list")).rejects.toThrow("Network error");
|
||||
});
|
||||
});
|
||||
|
||||
describe("fetchQueryMessages", () => {
|
||||
it("should fetch all query messages", async () => {
|
||||
const mockMessages = [
|
||||
{
|
||||
id: "msg-1",
|
||||
messageId: "uuid-1",
|
||||
connectionId: "conn-1",
|
||||
query: "tasks.list",
|
||||
status: FederationMessageStatus.DELIVERED,
|
||||
response: { data: [] },
|
||||
createdAt: new Date().toISOString(),
|
||||
},
|
||||
{
|
||||
id: "msg-2",
|
||||
messageId: "uuid-2",
|
||||
connectionId: "conn-2",
|
||||
query: "events.list",
|
||||
status: FederationMessageStatus.PENDING,
|
||||
createdAt: new Date().toISOString(),
|
||||
},
|
||||
];
|
||||
|
||||
vi.mocked(client.apiGet).mockResolvedValue(mockMessages);
|
||||
|
||||
const result = await fetchQueryMessages();
|
||||
|
||||
expect(client.apiGet).toHaveBeenCalledWith("/api/v1/federation/queries");
|
||||
expect(result).toEqual(mockMessages);
|
||||
});
|
||||
|
||||
it("should fetch query messages filtered by status", async () => {
|
||||
const mockMessages = [
|
||||
{
|
||||
id: "msg-1",
|
||||
messageId: "uuid-1",
|
||||
connectionId: "conn-1",
|
||||
query: "tasks.list",
|
||||
status: FederationMessageStatus.DELIVERED,
|
||||
response: { data: [] },
|
||||
createdAt: new Date().toISOString(),
|
||||
},
|
||||
];
|
||||
|
||||
vi.mocked(client.apiGet).mockResolvedValue(mockMessages);
|
||||
|
||||
const result = await fetchQueryMessages(FederationMessageStatus.DELIVERED);
|
||||
|
||||
expect(client.apiGet).toHaveBeenCalledWith("/api/v1/federation/queries?status=DELIVERED");
|
||||
expect(result).toEqual(mockMessages);
|
||||
});
|
||||
});
|
||||
|
||||
describe("fetchQueryMessage", () => {
|
||||
it("should fetch a single query message", async () => {
|
||||
const mockMessage = {
|
||||
id: "msg-123",
|
||||
messageId: "uuid-123",
|
||||
connectionId: "conn-1",
|
||||
query: "tasks.list",
|
||||
status: FederationMessageStatus.DELIVERED,
|
||||
response: { data: [{ id: "task-1", title: "Test Task" }] },
|
||||
createdAt: new Date().toISOString(),
|
||||
};
|
||||
|
||||
vi.mocked(client.apiGet).mockResolvedValue(mockMessage);
|
||||
|
||||
const result = await fetchQueryMessage("msg-123");
|
||||
|
||||
expect(client.apiGet).toHaveBeenCalledWith("/api/v1/federation/queries/msg-123");
|
||||
expect(result).toEqual(mockMessage);
|
||||
});
|
||||
|
||||
it("should handle not found errors", async () => {
|
||||
vi.mocked(client.apiGet).mockRejectedValue(new Error("Not found"));
|
||||
|
||||
await expect(fetchQueryMessage("msg-999")).rejects.toThrow("Not found");
|
||||
});
|
||||
});
|
||||
});
|
||||
90
apps/web/src/lib/api/federation-queries.ts
Normal file
90
apps/web/src/lib/api/federation-queries.ts
Normal file
@@ -0,0 +1,90 @@
|
||||
/**
|
||||
* Federation Queries API Client
|
||||
* Handles federated query requests to remote instances
|
||||
*/
|
||||
|
||||
import { apiGet, apiPost } from "./client";
|
||||
|
||||
/**
|
||||
* Federation message status (matches backend enum)
|
||||
*/
|
||||
export enum FederationMessageStatus {
|
||||
PENDING = "PENDING",
|
||||
DELIVERED = "DELIVERED",
|
||||
FAILED = "FAILED",
|
||||
}
|
||||
|
||||
/**
|
||||
* Query message details
|
||||
*/
|
||||
export interface QueryMessageDetails {
|
||||
id: string;
|
||||
workspaceId: string;
|
||||
connectionId: string;
|
||||
messageType: string;
|
||||
messageId: string;
|
||||
correlationId?: string;
|
||||
query?: string;
|
||||
response?: unknown;
|
||||
status: FederationMessageStatus;
|
||||
error?: string;
|
||||
createdAt: string;
|
||||
updatedAt: string;
|
||||
deliveredAt?: string;
|
||||
}
|
||||
|
||||
/**
|
||||
* Send query request
|
||||
*/
|
||||
export interface SendQueryRequest {
|
||||
connectionId: string;
|
||||
query: string;
|
||||
context?: Record<string, unknown>;
|
||||
}
|
||||
|
||||
/**
|
||||
* Send a federated query to a remote instance
|
||||
*/
|
||||
export async function sendFederatedQuery(
|
||||
connectionId: string,
|
||||
query: string,
|
||||
context?: Record<string, unknown>
|
||||
): Promise<QueryMessageDetails> {
|
||||
const request: SendQueryRequest = {
|
||||
connectionId,
|
||||
query,
|
||||
};
|
||||
|
||||
if (context) {
|
||||
request.context = context;
|
||||
}
|
||||
|
||||
return apiPost<QueryMessageDetails>("/api/v1/federation/query", request);
|
||||
}
|
||||
|
||||
/**
|
||||
* Fetch all query messages for the workspace
|
||||
*/
|
||||
export async function fetchQueryMessages(
|
||||
status?: FederationMessageStatus
|
||||
): Promise<QueryMessageDetails[]> {
|
||||
const params = new URLSearchParams();
|
||||
|
||||
if (status) {
|
||||
params.append("status", status);
|
||||
}
|
||||
|
||||
const queryString = params.toString();
|
||||
const endpoint = queryString
|
||||
? `/api/v1/federation/queries?${queryString}`
|
||||
: "/api/v1/federation/queries";
|
||||
|
||||
return apiGet<QueryMessageDetails[]>(endpoint);
|
||||
}
|
||||
|
||||
/**
|
||||
* Fetch a single query message
|
||||
*/
|
||||
export async function fetchQueryMessage(messageId: string): Promise<QueryMessageDetails> {
|
||||
return apiGet<QueryMessageDetails>(`/api/v1/federation/queries/${messageId}`);
|
||||
}
|
||||
196
docs/scratchpads/92-aggregated-dashboard.md
Normal file
196
docs/scratchpads/92-aggregated-dashboard.md
Normal file
@@ -0,0 +1,196 @@
|
||||
# Issue #92: Aggregated Dashboard View
|
||||
|
||||
## Objective
|
||||
|
||||
Implement an Aggregated Dashboard View that displays data from multiple federated Mosaic Stack instances in a unified interface. This allows users to see tasks, events, and projects from all connected instances in one place, with clear provenance indicators showing which instance each item comes from.
|
||||
|
||||
## Requirements
|
||||
|
||||
Backend infrastructure complete from Phase 3:
|
||||
|
||||
- QUERY message type implemented
|
||||
- Connection Manager UI (FED-008) implemented
|
||||
- Query service with signature verification
|
||||
- Connection status tracking
|
||||
|
||||
Frontend requirements:
|
||||
|
||||
- Display data from multiple federated instances in unified view
|
||||
- Query federated instances for tasks, events, projects
|
||||
- Show data provenance (which instance data came from)
|
||||
- Filter and sort aggregated data
|
||||
- PDA-friendly design (no demanding language)
|
||||
- Proper loading states and error handling
|
||||
- Minimum 85% test coverage
|
||||
|
||||
## Approach
|
||||
|
||||
### Phase 1: Federation API Client (TDD)
|
||||
|
||||
1. Write tests for federation query API client
|
||||
2. Implement federation query client functions:
|
||||
- `sendFederatedQuery(connectionId, query, context)`
|
||||
- `fetchQueryMessages(status?)`
|
||||
- Types for federated queries and responses
|
||||
|
||||
### Phase 2: Core Components (TDD)
|
||||
|
||||
1. Write tests for FederatedTaskCard component
|
||||
2. Implement FederatedTaskCard with provenance indicator
|
||||
3. Write tests for FederatedEventCard component
|
||||
4. Implement FederatedEventCard with provenance indicator
|
||||
5. Write tests for AggregatedDataGrid component
|
||||
6. Implement AggregatedDataGrid with filtering/sorting
|
||||
|
||||
### Phase 3: Dashboard Page Implementation
|
||||
|
||||
1. Create `/federation/dashboard` page
|
||||
2. Integrate components
|
||||
3. Implement query logic to fetch from all active connections
|
||||
4. Add loading and error states
|
||||
5. Add empty states
|
||||
|
||||
### Phase 4: PDA-Friendly Polish
|
||||
|
||||
1. Review all language for PDA-friendliness
|
||||
2. Implement calm visual indicators
|
||||
3. Add helpful empty states
|
||||
4. Test error messaging
|
||||
|
||||
## Design Decisions
|
||||
|
||||
### Data Provenance Indicators
|
||||
|
||||
Each item shows its source instance with:
|
||||
|
||||
- Instance name badge
|
||||
- Instance-specific color coding (subtle)
|
||||
- Hover tooltip with full instance details
|
||||
- "From: [Instance Name]" text
|
||||
|
||||
### PDA-Friendly Language
|
||||
|
||||
- "Unable to reach" instead of "Connection failed"
|
||||
- "No data available" instead of "No results"
|
||||
- "Loading data from instances..." instead of "Fetching..."
|
||||
- "Would you like to..." instead of "You must..."
|
||||
- Status indicators: 🟢 Active, 🔵 Loading, ⏸️ Paused, ⚪ Unavailable
|
||||
|
||||
### Component Structure
|
||||
|
||||
```
|
||||
apps/web/src/
|
||||
├── app/(authenticated)/federation/
|
||||
│ └── dashboard/
|
||||
│ └── page.tsx
|
||||
├── components/federation/
|
||||
│ ├── FederatedTaskCard.tsx
|
||||
│ ├── FederatedTaskCard.test.tsx
|
||||
│ ├── FederatedEventCard.tsx
|
||||
│ ├── FederatedEventCard.test.tsx
|
||||
│ ├── AggregatedDataGrid.tsx
|
||||
│ ├── AggregatedDataGrid.test.tsx
|
||||
│ ├── ProvenanceIndicator.tsx
|
||||
│ └── ProvenanceIndicator.test.tsx
|
||||
└── lib/api/
|
||||
└── federation-queries.ts (extend federation.ts)
|
||||
```
|
||||
|
||||
### Query Strategy
|
||||
|
||||
For MVP:
|
||||
|
||||
- Query all ACTIVE connections on page load
|
||||
- Show loading state per connection
|
||||
- Display results as they arrive
|
||||
- Cache results for 5 minutes (optional, future enhancement)
|
||||
- Handle connection failures gracefully
|
||||
|
||||
## Progress
|
||||
|
||||
- [x] Create scratchpad
|
||||
- [x] Research backend query API
|
||||
- [x] Extend federation API client with query support
|
||||
- [x] Write tests for ProvenanceIndicator
|
||||
- [x] Implement ProvenanceIndicator
|
||||
- [x] Write tests for FederatedTaskCard
|
||||
- [x] Implement FederatedTaskCard
|
||||
- [x] Write tests for FederatedEventCard
|
||||
- [x] Implement FederatedEventCard
|
||||
- [x] Write tests for AggregatedDataGrid
|
||||
- [x] Implement AggregatedDataGrid
|
||||
- [x] Create dashboard page
|
||||
- [x] Implement query orchestration logic
|
||||
- [x] Add loading/error states
|
||||
- [x] Run all tests (86 tests passing)
|
||||
- [x] TypeScript type checking (passing)
|
||||
- [x] Linting (passing)
|
||||
- [x] PDA-friendliness review (all language reviewed)
|
||||
- [x] Final QA (ready for review)
|
||||
|
||||
## Testing Strategy
|
||||
|
||||
- Unit tests for each component
|
||||
- Integration tests for the dashboard page
|
||||
- Test error states and edge cases
|
||||
- Test provenance display accuracy
|
||||
- Test PDA-friendly language compliance
|
||||
- Test loading states for slow connections
|
||||
- Ensure all tests pass before commit
|
||||
|
||||
## API Endpoints Used
|
||||
|
||||
From backend (already implemented):
|
||||
|
||||
- `POST /api/v1/federation/query` - Send query to remote instance
|
||||
- `GET /api/v1/federation/queries` - List query messages
|
||||
- `GET /api/v1/federation/queries/:id` - Get single query message
|
||||
- `GET /api/v1/federation/connections` - List connections
|
||||
|
||||
Query payload structure:
|
||||
|
||||
```json
|
||||
{
|
||||
"connectionId": "conn-1",
|
||||
"query": "tasks.list",
|
||||
"context": {
|
||||
"status": "IN_PROGRESS",
|
||||
"limit": 10
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
Query response structure:
|
||||
|
||||
```json
|
||||
{
|
||||
"id": "msg-123",
|
||||
"messageId": "uuid",
|
||||
"status": "DELIVERED",
|
||||
"response": {
|
||||
"data": [...],
|
||||
"provenance": {
|
||||
"instanceId": "instance-work-001",
|
||||
"timestamp": "2026-02-03T..."
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
## Notes
|
||||
|
||||
- Backend query service is complete (query.service.ts)
|
||||
- Need to define standard query names: "tasks.list", "events.list", "projects.list"
|
||||
- Consider implementing query result caching in future phase
|
||||
- Real-time updates via WebSocket can be added later (Phase 5)
|
||||
- Initial implementation will use polling/manual refresh
|
||||
|
||||
## Blockers
|
||||
|
||||
None - all backend infrastructure is complete.
|
||||
|
||||
## Related Issues
|
||||
|
||||
- #85 (FED-005): QUERY Message Type - COMPLETED
|
||||
- #91 (FED-008): Connection Manager UI - COMPLETED
|
||||
- #90 (FED-007): EVENT Subscriptions - COMPLETED (for future real-time updates)
|
||||
Reference in New Issue
Block a user