- Replace false claim about linkedTicket state with explicit fetch step on modal open - Remove MyQueueWidget references; TicketQueue is the existing component being updated Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
486 lines
22 KiB
Markdown
486 lines
22 KiB
Markdown
# PSA Ticket Management — Design Spec
|
||
|
||
**Date:** 2026-04-16
|
||
**Status:** Approved
|
||
**Author:** Michael Chihlas + Claude
|
||
|
||
---
|
||
|
||
## Overview
|
||
|
||
Add PSA ticket management to ResolutionFlow so MSP engineers can view, manage, and create ConnectWise tickets without leaving the app. The feature surfaces in three places: a dedicated Tickets page, a dashboard widget on QuickStartPage, and a spin-off ticket flow inside ResolutionAssist sessions.
|
||
|
||
---
|
||
|
||
## Decisions Made
|
||
|
||
| Question | Decision |
|
||
|----------|----------|
|
||
| Where does ticket management live? | Both: dedicated `/tickets` page + dashboard widget on QuickStartPage |
|
||
| List layout | Flat list with rich filters + pagination |
|
||
| Row density | Compact single-line rows |
|
||
| Ticket detail | Right-side slide-out panel (~50% width) |
|
||
| Ticket creation | Two-tab modal: Quick Create (AI) + Full Form |
|
||
| Resource member list | All CW members, RF-mapped users visually highlighted |
|
||
| Architecture | Dedicated `ticket_service.py` + normalized DTOs |
|
||
|
||
---
|
||
|
||
## Section 1 — Backend
|
||
|
||
### New Endpoints
|
||
|
||
All added to `backend/app/api/endpoints/integrations.py`, backed by `backend/app/services/ticket_service.py`.
|
||
|
||
| Method | Path | Purpose |
|
||
|--------|------|---------|
|
||
| `POST` | `/integrations/psa/tickets` | Create a ticket |
|
||
| `PATCH` | `/integrations/psa/tickets/{id}/status` | Update ticket status |
|
||
| `GET` | `/integrations/psa/tickets/{id}/resources` | List current assignees |
|
||
| `POST` | `/integrations/psa/tickets/{id}/resources` | Add a resource (member) |
|
||
| `DELETE` | `/integrations/psa/tickets/{id}/resources/{member_id}` | Remove a resource |
|
||
| `POST` | `/integrations/psa/tickets/ai-parse` | Natural language → structured pre-fill payload |
|
||
|
||
**Breaking change — `search_tickets` response shape updated to `TicketListResponse`.**
|
||
The existing `/integrations/psa/tickets/search` endpoint currently returns `list[PSATicketSearchResult]`. This spec changes it to return `TicketListResponse` (adds `total`, `page`, `page_size` wrapper).
|
||
|
||
Current callers that must be migrated:
|
||
- `integrationsApi.searchTickets()` in `frontend/src/api/integrations.ts` (line 18) — update return type
|
||
- `integrationsApi.searchTicketsQueue()` in `frontend/src/api/integrations.ts` (line 20) — update return type
|
||
- `frontend/src/components/dashboard/TicketQueue.tsx` — update to read `.items` from response
|
||
- `frontend/src/components/session/TicketPickerModal.tsx` — update to read `.items` from response
|
||
|
||
All other existing endpoints (`get_ticket`, `get_ticket_statuses`, `list_members`, `list_boards`) are unchanged.
|
||
|
||
### ticket_service.py
|
||
|
||
New service wrapping the PSA provider for ticket mutations. Keeps `integrations.py` clean and PSA-agnostic for future Autotask support.
|
||
|
||
Methods:
|
||
- `create_ticket(account_id, payload) → PSATicketCreated`
|
||
- `add_resource(account_id, ticket_id, member_id) → PSAResource`
|
||
- `remove_resource(account_id, ticket_id, member_id) → None`
|
||
- `update_status(account_id, ticket_id, status_id) → PSATicketStatusUpdate`
|
||
- `list_resources(account_id, ticket_id) → list[PSAResource]`
|
||
|
||
### PSA Provider — New Abstract Methods and Paginated Result Type
|
||
|
||
**New type in `backend/app/services/psa/types.py`:**
|
||
```python
|
||
@dataclass
|
||
class PaginatedTicketResult:
|
||
items: list[PSATicket]
|
||
total: int
|
||
page: int
|
||
page_size: int
|
||
```
|
||
|
||
**`search_tickets` signature change** — updated on both the abstract base and `ConnectWiseProvider` to return `PaginatedTicketResult` instead of `list[PSATicket]`:
|
||
```python
|
||
# base.py
|
||
@abstractmethod
|
||
async def search_tickets(self, query: str, **filters) -> PaginatedTicketResult: ...
|
||
```
|
||
|
||
**How `total` is fetched** — ConnectWise provides `GET /service/tickets/count?conditions=...` which accepts the same conditions string as the page fetch. The `ConnectWiseProvider.search_tickets()` implementation fires two parallel requests:
|
||
1. `GET /service/tickets?conditions=...&pageSize=N&page=N` — the current page
|
||
2. `GET /service/tickets/count?conditions=...` — returns `{ "count": 142 }`
|
||
|
||
Both use the same built conditions string. `asyncio.gather()` runs them in parallel. The count result is used to populate `PaginatedTicketResult.total`.
|
||
|
||
**New abstract methods** added to `PSAProvider` base and `ConnectWiseProvider`:
|
||
```python
|
||
async def list_resources(self, ticket_id: int) -> list[PSAResource]: ...
|
||
async def add_resource(self, ticket_id: int, member_id: int) -> PSAResource: ...
|
||
async def remove_resource(self, ticket_id: int, member_id: int) -> None: ...
|
||
async def create_ticket(self, payload: TicketCreatePayload) -> PSATicketCreated: ...
|
||
```
|
||
|
||
`update_status` already exists on the provider — no change needed there.
|
||
|
||
ConnectWise implementation:
|
||
- `list_resources` → `GET /service/tickets/{id}/members`
|
||
- `add_resource` → `POST /service/tickets/{id}/members`
|
||
- `remove_resource` → `DELETE /service/tickets/{id}/members/{member_id}`
|
||
- `create_ticket` → `POST /service/tickets`
|
||
|
||
### Normalized DTOs (Pydantic Schemas)
|
||
|
||
New schemas in `backend/app/schemas/psa_tickets.py`:
|
||
|
||
```python
|
||
class PSAResource(BaseModel):
|
||
member_id: int
|
||
member_name: str
|
||
member_identifier: str # CW username
|
||
is_rf_user: bool # True if mapped in RF member mappings
|
||
|
||
class PSATicketCreated(BaseModel):
|
||
id: int
|
||
summary: str
|
||
board_name: str
|
||
status_name: str
|
||
priority_name: str
|
||
company_name: str
|
||
resources: list[PSAResource]
|
||
|
||
class PSATicketStatusUpdate(BaseModel):
|
||
ticket_id: int
|
||
previous_status: str
|
||
new_status: str
|
||
|
||
class TicketCreatePayload(BaseModel):
|
||
summary: str
|
||
company_id: int
|
||
board_id: int
|
||
status_id: int
|
||
priority_id: int
|
||
description: str | None = None
|
||
assigned_member_id: int | None = None
|
||
|
||
class TicketListResponse(BaseModel):
|
||
items: list[PSATicketSearchResult] # existing schema
|
||
total: int
|
||
page: int
|
||
page_size: int
|
||
```
|
||
|
||
`search_tickets` endpoint updated to return `TicketListResponse` (was a plain list). Backend sorts results by `priority desc, dateEntered desc` via CW `orderBy` param.
|
||
|
||
### AI Parse Endpoint
|
||
|
||
`POST /integrations/psa/tickets/ai-parse`
|
||
|
||
Request:
|
||
```json
|
||
{ "prompt": "New ticket for Acme Corp, Outlook not syncing, high priority, assign to me" }
|
||
```
|
||
|
||
Response — all pre-fill fields nullable, explicit `missing_fields` and `warnings`:
|
||
```json
|
||
{
|
||
"summary": "Outlook not syncing",
|
||
"company_id": 42,
|
||
"board_id": null,
|
||
"priority_id": null,
|
||
"status_id": null,
|
||
"assigned_member_id": 17,
|
||
"description": "User reports Outlook calendar not syncing since yesterday morning.",
|
||
"missing_fields": ["board_id", "priority_id", "status_id"],
|
||
"warnings": ["Could not determine board from context"]
|
||
}
|
||
```
|
||
|
||
Frontend uses `missing_fields` to highlight required fields still needing engineer input. No ticket is created at this step — it is a parse-only endpoint.
|
||
|
||
---
|
||
|
||
## Section 2 — Frontend Architecture
|
||
|
||
### New Files
|
||
|
||
| File | Purpose |
|
||
|------|---------|
|
||
| `pages/TicketsPage.tsx` | Main tickets page — filter bar + paginated list |
|
||
| `components/tickets/TicketListRow.tsx` | Compact single-line row |
|
||
| `components/tickets/TicketFilterBar.tsx` | Config-driven filter bar (7 filters) |
|
||
| `components/tickets/TicketDetailPanel.tsx` | Slide-out panel orchestrator |
|
||
| `components/tickets/detail/TicketDetailHeader.tsx` | ID, summary, company, board, SLA |
|
||
| `components/tickets/detail/TicketResourceManager.tsx` | Assignee list + add/remove |
|
||
| `components/tickets/detail/TicketNotesFeed.tsx` | Chronological notes history |
|
||
| `components/tickets/detail/TicketAddNote.tsx` | Inline note composer |
|
||
| `components/tickets/detail/TicketConfigs.tsx` | Attached devices/configs |
|
||
| `components/tickets/detail/TicketRelated.tsx` | Related tickets list |
|
||
| `components/tickets/NewTicketModal.tsx` | Two-tab modal (owns draft state) |
|
||
| `components/tickets/AiTicketParseForm.tsx` | Prompt input → emits parsed values upward |
|
||
| `api/tickets.ts` | All ticket API calls (typed, `.then(r => r.data)` pattern) |
|
||
| `types/tickets.ts` | TypeScript interfaces mirroring normalized DTOs |
|
||
|
||
### Existing Files Touched
|
||
|
||
- `router.tsx` — add `/tickets` route (lazy, via `lazyWithRetry`)
|
||
- `AppLayout.tsx` — add "Tickets" nav item in sidebar under RESOLVE section
|
||
- `AssistantChatPage.tsx` — handle `create_spin_off_ticket` action type in TaskLane + add "New Ticket" button to session header
|
||
- `QuickStartPage.tsx` — no structural change needed; `TicketQueue` already renders at line 64. The existing component is updated in place (see Section 4).
|
||
|
||
### Shared Types (`types/tickets.ts`)
|
||
|
||
```typescript
|
||
export interface TicketFilters {
|
||
search: string;
|
||
board_id: number | null;
|
||
status_id: number | null;
|
||
priority: string | null;
|
||
company_id: number | null;
|
||
assigned: 'me' | 'unassigned' | 'all' | number; // number = specific member_id
|
||
include_closed: boolean;
|
||
}
|
||
|
||
export interface TicketCreationPayload {
|
||
summary: string;
|
||
company_id: number | null;
|
||
board_id: number | null;
|
||
status_id: number | null;
|
||
priority_id: number | null;
|
||
description: string;
|
||
assigned_member_id: number | null;
|
||
}
|
||
|
||
export interface AiParseResponse {
|
||
summary: string | null;
|
||
company_id: number | null;
|
||
board_id: number | null;
|
||
priority_id: number | null;
|
||
status_id: number | null;
|
||
assigned_member_id: number | null;
|
||
description: string | null;
|
||
missing_fields: string[];
|
||
warnings: string[];
|
||
}
|
||
|
||
export interface PSAResource {
|
||
member_id: number;
|
||
member_name: string;
|
||
member_identifier: string;
|
||
is_rf_user: boolean;
|
||
}
|
||
|
||
// TicketSearchResult is the existing PSATicketSearchResult type from types/integrations.ts
|
||
// Re-export or import from there — do not redefine
|
||
export interface TicketListResponse {
|
||
items: PSATicketSearchResult[];
|
||
total: number;
|
||
page: number;
|
||
page_size: number;
|
||
}
|
||
```
|
||
|
||
### TicketsPage — Filter & Pagination State
|
||
|
||
All filter and pagination state lives in URL query params via `useSearchParams`:
|
||
|
||
| Param | Type | Default |
|
||
|-------|------|---------|
|
||
| `search` | string | `""` |
|
||
| `board` | number | — |
|
||
| `status` | number | — |
|
||
| `priority` | string | — |
|
||
| `company` | number | — |
|
||
| `assigned` | `me \| unassigned \| all \| {id}` | `all` |
|
||
| `closed` | boolean | `false` |
|
||
| `page` | number | `1` |
|
||
|
||
Filter changes reset `page` to 1. Pagination: page size of 25. Controls show "Showing X–Y of Z tickets". Next disabled when `page * 25 >= total`.
|
||
|
||
### TicketFilterBar — Config-Driven
|
||
|
||
Filters defined as a `FILTER_CONFIG` array. Each entry:
|
||
```typescript
|
||
{ key: keyof TicketFilters, label: string, type: 'text' | 'select' | 'toggle', loadOptions?: () => Promise<Option[]> }
|
||
```
|
||
Adding or removing a filter is a one-line config change, not a component edit.
|
||
|
||
### TicketDetailPanel — Optimistic Hydration
|
||
|
||
The panel uses the **existing** `/integrations/psa/tickets/{id}/context` endpoint (client: `psaContextApi.getTicketContext()` in `frontend/src/api/psaContext.ts`) which already returns company, contact, configurations, notes, and related tickets in one call. This avoids creating redundant endpoints.
|
||
|
||
1. Panel opens immediately with list row data (id, summary, company, board, status, priority) — no loading state for these fields
|
||
2. Two parallel fetches fire on open:
|
||
- `psaContextApi.getTicketContext(ticketId)` — hydrates contact, notes, configs, related tickets
|
||
- `ticketsApi.listResources(ticketId)` — hydrates assignees (new endpoint)
|
||
3. All detail sections (contact, notes, configs, related) render skeletons until `getTicketContext` resolves
|
||
4. Resources section renders skeleton until `listResources` resolves
|
||
|
||
`get_ticket` (the simpler single-ticket endpoint) is **not** used by the panel — `getTicketContext` is a strict superset of the data needed.
|
||
|
||
### NewTicketModal — State Ownership
|
||
|
||
- `NewTicketModal` owns the `TicketCreationPayload` draft state
|
||
- `AiTicketParseForm` is a pure emitter: accepts a prompt string, calls `ai-parse`, fires `onParsed(Partial<TicketCreationPayload>)` upward
|
||
- Modal merges parsed values into draft, highlights `missing_fields` with visual indicators
|
||
- Two tabs: **Quick Create** (AI prompt → review) | **Full Form** (manual entry)
|
||
- Default tab: Quick Create if AI-triggered, Full Form if engineer-initiated
|
||
- Initial props: `initialValues?: Partial<TicketCreationPayload>` — used for spin-off pre-population
|
||
|
||
---
|
||
|
||
## Section 3 — ResolutionAssist Integration
|
||
|
||
### Two Trigger Paths
|
||
|
||
**1. AI-suggested (via `[ACTIONS]` marker)**
|
||
|
||
When the AI identifies a second distinct issue during a session, it emits a JSON array inside the `[ACTIONS]` marker — matching the exact format `_parse_actions_marker()` in `unified_chat_service.py` expects (a list of objects with `label`, `command`, `description`):
|
||
|
||
```
|
||
[ACTIONS]
|
||
[
|
||
{
|
||
"label": "Create ticket: Printer offline on 2nd floor",
|
||
"command": "create_spin_off_ticket",
|
||
"description": "Printer offline on 2nd floor"
|
||
}
|
||
]
|
||
[/ACTIONS]
|
||
```
|
||
|
||
The existing `_parse_actions_marker()` parser in `unified_chat_service.py` already handles this format — no parser changes needed. The frontend reads `action.command === "create_spin_off_ticket"` to render the "Create Ticket" button in TaskLane, and uses `action.description` as the `summary_hint` pre-populated into the Quick Create prompt input.
|
||
|
||
`summary_hint` (from `action.description`) populates the AI prompt input only, not the summary field directly. The engineer still runs the AI parse step and reviews all output. This prevents bypassing review with potentially hallucinated values.
|
||
|
||
**2. Engineer-initiated**
|
||
|
||
A "New Ticket" button in the ResolutionAssist session header. Always visible regardless of AI suggestion. Opens `NewTicketModal` with Full Form tab as default.
|
||
|
||
### Both Paths — NewTicketModal Pre-population
|
||
|
||
**The linked ticket IDs problem:** The current `PSATicketInfo` type in `frontend/src/types/integrations.ts` only exposes `company_name` and `board_name` — not `company_id` or `board_id`. The modal needs the numeric IDs to pre-populate the form selects.
|
||
|
||
**Fix:** Expand `PSATicketInfo` in `types/integrations.ts` to add the optional ID fields:
|
||
```typescript
|
||
export interface PSATicketInfo {
|
||
id: string
|
||
summary: string
|
||
company_name: string | null
|
||
board_name: string | null
|
||
status_name: string | null
|
||
priority_name: string | null
|
||
company_id: number | null // add
|
||
board_id: number | null // add
|
||
}
|
||
```
|
||
|
||
These fields are already returned by the CW API in `get_ticket()` — update `_map_ticket()` in `ConnectWiseProvider` and the `PSATicketInfo` Pydantic schema to pass them through.
|
||
|
||
**`AssistantChatPage` state change required:** The current page only tracks `activePsaTicketId: string | null` (line 76) — it does not hold a `PSATicketInfo` object. Add a new state field:
|
||
```typescript
|
||
const [linkedTicket, setLinkedTicket] = useState<PSATicketInfo | null>(null)
|
||
```
|
||
|
||
When the modal is opened (either via AI suggestion or the "New Ticket" button), if `activePsaTicketId` is set and `linkedTicket` is null, fire `integrationsApi.getTicket(activePsaTicketId)` to fetch the full ticket (which now includes `company_id` and `board_id`) and store it in `linkedTicket`. The modal opens immediately — `initialValues` is populated once the fetch resolves and the form fields update. If the fetch is still in flight when the modal opens, `company_id` and `board_id` start empty and fill in when ready.
|
||
|
||
Once `linkedTicket` is populated, the modal receives:
|
||
```typescript
|
||
initialValues: {
|
||
company_id: linkedTicket.company_id,
|
||
board_id: linkedTicket.board_id,
|
||
}
|
||
```
|
||
|
||
When no linked ticket exists (`activePsaTicketId === null`): `initialValues` is omitted. `company_id` and `board_id` render empty, requiring manual selection. No silent defaults, no errors.
|
||
|
||
### TaskLane Action Lifecycle
|
||
|
||
- Opening the modal does **not** remove the action from TaskLane
|
||
- Dismissing the modal without submitting leaves the action visible
|
||
- Successful ticket creation removes the action and shows a success toast: `"Ticket #1042 created in ConnectWise"`
|
||
|
||
### System Prompt Addition
|
||
|
||
New rule added to `ASSISTANT_SYSTEM_PROMPT` in `backend/app/services/assistant_chat_service.py`:
|
||
|
||
> When you identify a second distinct issue that is clearly separate from the primary topic of this session, suggest creating a spin-off ticket using the `[ACTIONS]` marker. Use `"command": "create_spin_off_ticket"` and put the issue description in `"description"`. Only suggest this when the issue is genuinely separate — do not suggest for every tangential mention.
|
||
|
||
### Backend
|
||
|
||
- **`assistant_chat_service.py`** — system prompt updated with spin-off ticket instruction (above)
|
||
- **`unified_chat_service.py`** — no parser changes needed; the existing `_parse_actions_marker()` already handles the JSON array format. The frontend reads `command === "create_spin_off_ticket"` to route the action
|
||
- **`flowpilot_engine.py`** — no changes needed for this feature; guided FlowPilot sessions do not use this action type in the current scope
|
||
|
||
No new backend endpoints — the modal reuses `POST /integrations/psa/tickets` and `POST /integrations/psa/tickets/ai-parse`.
|
||
|
||
---
|
||
|
||
## Section 4 — Dashboard Widget (QuickStartPage)
|
||
|
||
### Placement
|
||
|
||
`TicketQueue` **already exists** in `QuickStartPage` (line 64, below `ActiveFlowPilotSessions`, above the Dashboard section). It currently auto-hides if no PSA connection exists. This spec updates the existing `TicketQueue` component — it is **not** a new widget and does not need to be added to `QuickStartPage`. The Dashboard section below it is not collapsible.
|
||
|
||
### Data Fetching
|
||
|
||
On mount: `GET /integrations/psa/member-mappings` first to detect mapping state, then `integrationsApi.searchTicketsQueue({ assigned_to_me: true, include_closed: false, page_size: 5 })` if a mapping exists for the current user.
|
||
|
||
`searchTicketsQueue` is used (not `searchTickets`) because it already accepts `assigned_to_me` and `page_size` params. Its return type will be updated to `TicketListResponse` as part of the search endpoint migration, so the widget reads `.items` after that change.
|
||
|
||
Member mapping detection is explicit — the widget checks the mappings response, not the ticket result. "No mapping" and "no tickets" are distinct states.
|
||
|
||
### Widget States
|
||
|
||
| State | Condition | Display |
|
||
|-------|-----------|---------|
|
||
| Hidden | No PSA connection | Widget not rendered |
|
||
| Prompt | PSA connected, no member mapping | "Map your PSA member to see your queue" → `/account/integrations` |
|
||
| Loading | Fetching | 3 skeleton rows |
|
||
| Populated | Tickets returned | Up to 5 compact rows + "View All Tickets →" |
|
||
| Empty | No assigned open tickets | "No open tickets assigned to you" — muted, no CTA |
|
||
| Error | PSA fetch fails | Silent — returns `[]`, no toast (per Lesson 111) |
|
||
|
||
### Row Display
|
||
|
||
Compact row matching Tickets page style: `#ID · Summary · Status badge · Priority dot`
|
||
|
||
Clicking a row opens `TicketDetailPanel` as a right-side sheet rendered at the `QuickStartPage` level. Does **not** navigate away.
|
||
|
||
### "View All Tickets" Link
|
||
|
||
Links to `/tickets?assigned=me`. `TicketsPage` reads `assigned` from `useSearchParams` on mount and applies it as the initial filter state — consistent with Section 2 URL param contract.
|
||
|
||
### Sorting
|
||
|
||
Backend `search_tickets()` adds `orderBy=priority desc,dateEntered desc` to the CW API query. Widget does not sort client-side.
|
||
|
||
---
|
||
|
||
## Files Changed Summary
|
||
|
||
### New Backend Files
|
||
- `backend/app/services/ticket_service.py`
|
||
- `backend/app/schemas/psa_tickets.py`
|
||
|
||
### Modified Backend Files
|
||
- `backend/app/api/endpoints/integrations.py` — 6 new endpoints, update search to return `TicketListResponse`
|
||
- `backend/app/services/psa/types.py` — add `PaginatedTicketResult` dataclass
|
||
- `backend/app/services/psa/base.py` — 4 new abstract methods; update `search_tickets` return type to `PaginatedTicketResult`
|
||
- `backend/app/services/psa/connectwise/provider.py` — implement 4 new methods; update `search_tickets` to fire parallel count request and return `PaginatedTicketResult`; update `_map_ticket()` to pass through `company_id` and `board_id`
|
||
- `backend/app/schemas/psa_connection.py` — add `company_id` and `board_id` to `PSATicketInfo` Pydantic schema
|
||
- `backend/app/services/assistant_chat_service.py` — add spin-off ticket rule to `ASSISTANT_SYSTEM_PROMPT`
|
||
- ~~`backend/app/services/flowpilot_engine.py`~~ — no changes (FlowPilot out of scope for this feature)
|
||
- ~~`backend/app/services/unified_chat_service.py`~~ — no changes (existing `[ACTIONS]` parser handles the format)
|
||
|
||
### New Frontend Files
|
||
- `frontend/src/pages/TicketsPage.tsx`
|
||
- `frontend/src/api/tickets.ts`
|
||
- `frontend/src/types/tickets.ts`
|
||
- `frontend/src/components/tickets/TicketListRow.tsx`
|
||
- `frontend/src/components/tickets/TicketFilterBar.tsx`
|
||
- `frontend/src/components/tickets/TicketDetailPanel.tsx`
|
||
- `frontend/src/components/tickets/NewTicketModal.tsx`
|
||
- `frontend/src/components/tickets/AiTicketParseForm.tsx`
|
||
- `frontend/src/components/tickets/detail/TicketDetailHeader.tsx`
|
||
- `frontend/src/components/tickets/detail/TicketResourceManager.tsx`
|
||
- `frontend/src/components/tickets/detail/TicketNotesFeed.tsx`
|
||
- `frontend/src/components/tickets/detail/TicketAddNote.tsx`
|
||
- `frontend/src/components/tickets/detail/TicketConfigs.tsx`
|
||
- `frontend/src/components/tickets/detail/TicketRelated.tsx`
|
||
|
||
### Modified Frontend Files
|
||
- `frontend/src/router.tsx` — `/tickets` route
|
||
- `frontend/src/components/layout/AppLayout.tsx` — Tickets nav item
|
||
- `frontend/src/pages/AssistantChatPage.tsx` — handle `create_spin_off_ticket` command in action renderer + add "New Ticket" button to session header
|
||
- `frontend/src/components/dashboard/TicketQueue.tsx` — update existing component (see Section 4 — not a new file)
|
||
- `frontend/src/api/integrations.ts` — update `searchTickets()` and `searchTicketsQueue()` return types to `TicketListResponse`
|
||
- `frontend/src/types/integrations.ts` — add `company_id: number | null` and `board_id: number | null` to `PSATicketInfo`
|
||
- `frontend/src/components/dashboard/TicketQueue.tsx` — update existing component: read `.items`, add mapping-state detection, member-mapping check, and 5-item cap
|
||
- `frontend/src/components/session/TicketPickerModal.tsx` — read `.items` from paginated response
|
||
|
||
---
|
||
|
||
## Out of Scope
|
||
|
||
- Autotask provider implementation (schema-ready, not implemented)
|
||
- Time entry creation from ticket detail (provider method exists, no UI)
|
||
- Ticket editing beyond status (summary, description, priority changes)
|
||
- Bulk ticket operations
|
||
- Real-time ticket updates / polling
|