import { useEffect, useMemo, useCallback } from "react"; import { useLocation, useSearchParams } from "@/lib/router"; import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query"; import { issuesApi } from "../api/issues"; import { agentsApi } from "../api/agents"; import { projectsApi } from "../api/projects"; import { heartbeatsApi } from "../api/heartbeats"; import { useCompany } from "../context/CompanyContext"; import { useBreadcrumbs } from "../context/BreadcrumbContext"; import { collectLiveIssueIds } from "../lib/liveIssueIds"; import { queryKeys } from "../lib/queryKeys"; import { createIssueDetailLocationState } from "../lib/issueDetailBreadcrumb"; import { EmptyState } from "../components/EmptyState"; import { IssuesList } from "../components/IssuesList"; import { CircleDot } from "lucide-react"; const WORKSPACE_FILTER_ISSUE_LIMIT = 1000; export function buildIssuesSearchUrl(currentHref: string, search: string): string | null { const url = new URL(currentHref); const currentSearch = url.searchParams.get("q") ?? ""; if (currentSearch === search) return null; if (search.length > 0) { url.searchParams.set("q", search); } else { url.searchParams.delete("q"); } return `${url.pathname}${url.search}${url.hash}`; } export function Issues() { const { selectedCompanyId } = useCompany(); const { setBreadcrumbs } = useBreadcrumbs(); const location = useLocation(); const [searchParams] = useSearchParams(); const queryClient = useQueryClient(); const initialSearch = searchParams.get("q") ?? ""; const participantAgentId = searchParams.get("participantAgentId") ?? undefined; const initialWorkspaces = searchParams.getAll("workspace").filter((workspaceId) => workspaceId.length > 0); const workspaceIdFilter = initialWorkspaces.length === 1 ? initialWorkspaces[0] : undefined; const handleSearchChange = useCallback((search: string) => { const nextUrl = buildIssuesSearchUrl(window.location.href, search); if (!nextUrl) return; window.history.replaceState(window.history.state, "", nextUrl); }, []); const { data: agents } = useQuery({ queryKey: queryKeys.agents.list(selectedCompanyId!), queryFn: () => agentsApi.list(selectedCompanyId!), enabled: !!selectedCompanyId, }); const { data: projects } = useQuery({ queryKey: queryKeys.projects.list(selectedCompanyId!), queryFn: () => projectsApi.list(selectedCompanyId!), enabled: !!selectedCompanyId, }); const { data: liveRuns } = useQuery({ queryKey: queryKeys.liveRuns(selectedCompanyId!), queryFn: () => heartbeatsApi.liveRunsForCompany(selectedCompanyId!), enabled: !!selectedCompanyId, refetchInterval: 5000, }); const liveIssueIds = useMemo(() => collectLiveIssueIds(liveRuns), [liveRuns]); const issueLinkState = useMemo( () => createIssueDetailLocationState( "Issues", `${location.pathname}${location.search}${location.hash}`, "issues", ), [location.pathname, location.search, location.hash], ); useEffect(() => { setBreadcrumbs([{ label: "Issues" }]); }, [setBreadcrumbs]); const { data: issues, isLoading, error } = useQuery({ queryKey: [ ...queryKeys.issues.list(selectedCompanyId!), "participant-agent", participantAgentId ?? "__all__", "workspace", workspaceIdFilter ?? "__all__", "with-routine-executions", ], queryFn: () => issuesApi.list(selectedCompanyId!, { participantAgentId, workspaceId: workspaceIdFilter, includeRoutineExecutions: true, ...(workspaceIdFilter ? { limit: WORKSPACE_FILTER_ISSUE_LIMIT } : {}), }), enabled: !!selectedCompanyId, }); const updateIssue = useMutation({ mutationFn: ({ id, data }: { id: string; data: Record }) => issuesApi.update(id, data), onSuccess: () => { queryClient.invalidateQueries({ queryKey: queryKeys.issues.list(selectedCompanyId!) }); }, }); if (!selectedCompanyId) { return ; } return ( 0 ? initialWorkspaces : undefined} initialSearch={initialSearch} onSearchChange={handleSearchChange} enableRoutineVisibilityFilter onUpdateIssue={(id, data) => updateIssue.mutate({ id, data })} searchFilters={participantAgentId || workspaceIdFilter ? { participantAgentId, workspaceId: workspaceIdFilter } : undefined} /> ); }