Compare commits
6 Commits
1a8cb0604f
...
fix/ci-pyt
| Author | SHA1 | Date | |
|---|---|---|---|
| ca45bc9bb3 | |||
| e976fb4e87 | |||
| 0aefaa78eb | |||
| 49f88569da | |||
| 208ec996d5 | |||
| 8f7df2c0ef |
@@ -1,33 +1,22 @@
|
|||||||
# CURRENT_TASK.md
|
# CURRENT_TASK.md
|
||||||
|
|
||||||
**Task:** none — replace this file when starting the next real task.
|
**Task:** Restore a fully green CI gate on `main` and lock it via branch protection so future merges can't introduce silent rot.
|
||||||
|
|
||||||
**Status:** not-started
|
**Status:** in-progress
|
||||||
|
|
||||||
**Definition of Done:** n/a
|
|
||||||
|
|
||||||
**Assumptions:** n/a
|
|
||||||
|
|
||||||
**Out of scope:** n/a
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
<!-- When you start a real task, replace the block above with:
|
|
||||||
|
|
||||||
**Task:** One-sentence goal.
|
|
||||||
|
|
||||||
**Status:** not-started | in-progress | blocked | ready-for-review | complete
|
|
||||||
|
|
||||||
**Definition of Done:**
|
**Definition of Done:**
|
||||||
- [ ] Testable criterion 1
|
- [ ] PR #150 (`fix/ci-workflow-config`) merged. Both `CI / backend (pull_request)` and `CI / frontend (pull_request)` show success on the merge commit.
|
||||||
- [ ] Testable criterion 2
|
- [ ] `CI / backend (pull_request)` added to required status checks on `main` in Gitea branch protection (frontend is already required).
|
||||||
- [ ] Tests added or updated
|
- [ ] The 54 real backend test failures (left after #149's infra cleanup) categorized and fixed in a follow-up PR. Target: 0 failures, 0 errors on a `pytest` run inside `resolutionflow_backend`.
|
||||||
- [ ] `npm run build` passes (frontend) / `pytest` passes (backend)
|
- [ ] `npm run lint` stays at 0 errors after the cleanup PR (already at 0 on main).
|
||||||
|
- [ ] Append a SESSION_LOG.md entry summarizing what shipped.
|
||||||
|
|
||||||
**Assumptions:**
|
**Assumptions:**
|
||||||
- What we're treating as given
|
- The 54 failures fall into a small number of root-cause categories (likely 3–5: fixture-scoping leaks, DB cleanup ordering, account_id propagation in test seed paths). Verify before assuming.
|
||||||
|
- The pytest-asyncio 0.24 + pytest 8.4 toolchain bumped in #149 is the right baseline; do not revert.
|
||||||
|
- `DATABASE_TEST_URL` is the only DB URL conftest will honor; do not weaken the safety guard added in `dab740d`.
|
||||||
|
|
||||||
**Out of scope:**
|
**Out of scope:**
|
||||||
- What this task explicitly does NOT cover
|
- New feature work on FlowPilot (Phase 10+) or PSA — keep this branch focused on CI debt.
|
||||||
|
- Frontend lint warnings (23 remain after #149; they're missing-deps in useEffect, opt-in cleanup later).
|
||||||
-->
|
- RLS test suite (`test_rls_isolation.py`) — gated behind `RUN_RLS_TESTS=1` and not in the default CI run.
|
||||||
|
|||||||
@@ -2,34 +2,62 @@
|
|||||||
|
|
||||||
# HANDOFF.md
|
# HANDOFF.md
|
||||||
|
|
||||||
**Last updated:** 2026-04-24 (America/New_York)
|
**Last updated:** 2026-04-25 06:12 EDT
|
||||||
|
|
||||||
**Active task:** None — see [CURRENT_TASK.md](CURRENT_TASK.md). Replace it when picking up the next real task.
|
**Active task:** Restore green CI gate on `main` and lock it via branch protection. See [CURRENT_TASK.md](CURRENT_TASK.md).
|
||||||
|
|
||||||
**Branch:** `feat/flowpilot-migration` — a long-running FlowPilot Phase 9 feature branch. The recent AI-handoff migration commits ride on this branch (not on their own branch); they'll merge to `main` whenever Phase 9 does.
|
**Branch:** `fix/ci-workflow-config`
|
||||||
|
|
||||||
**Branch state:** 3 commits ahead of `origin/feat/flowpilot-migration`:
|
## Current state
|
||||||
|
|
||||||
- `b3be1e0 chore: ignore .remember/ skill runtime state`
|
Previous session fixed the 54 real backend failures left after #149. The default backend suite is now green locally:
|
||||||
- `b3506b5 docs(pilot): phase 9 review issues`
|
|
||||||
- `b14a16a chore(tests): gate RLS tests behind RUN_RLS_TESTS flag`
|
|
||||||
|
|
||||||
Earlier in this session (already pushed to origin):
|
```bash
|
||||||
|
docker exec resolutionflow_backend bash -lc 'pytest --override-ini="addopts=" -q > /tmp/full-backend.log 2>&1; code=$?; tail -n 160 /tmp/full-backend.log; exit $code'
|
||||||
|
# 1076 passed, 35 deselected in 1347.41s (0:22:27)
|
||||||
|
```
|
||||||
|
|
||||||
- `9c8ba29 fix(ai): correct stale role-hierarchy and file-listing claims`
|
Targeted validation also passed:
|
||||||
- `bee8690 chore(ai): migrate to dual-agent handoff system`
|
|
||||||
- `e110fed chore: snapshot CLAUDE.md before ai-handoff migration` (tag: `pre-ai-handoff`)
|
|
||||||
|
|
||||||
**Where I left off:**
|
- `tests/test_session_resolutions_api.py tests/test_session_sharing.py tests/test_session_suggested_fixes_api.py tests/test_survey.py tests/test_tenant_isolation_p0.py tests/test_tree_sharing.py tests/test_trees.py::TestTrees::test_delete_tree_cleans_up_folder_and_tag_assignments tests/test_uploads.py::test_delete_upload_forbidden_for_non_owner` → `73 passed`
|
||||||
- File: n/a — nothing mid-edit.
|
- PDF export tests → `3 passed`
|
||||||
- Next intended action: push the 3 unpushed commits when ready (`git push`), then start the next real task (replace `CURRENT_TASK.md`, update this file).
|
- Prompt/PSA/resolution/script-builder subset → `14 passed`
|
||||||
|
- Admin/AI/branch subsets → `11 passed`
|
||||||
|
|
||||||
**Uncommitted state:**
|
## What changed
|
||||||
- Working tree is clean.
|
|
||||||
|
|
||||||
**Immediate next steps:**
|
Production fixes:
|
||||||
1. `git push` to publish the 3 local commits (cleanup batch).
|
|
||||||
2. When starting the next real feature task: replace `CURRENT_TASK.md` with actual goal/DoD, rewrite this file's resume section.
|
|
||||||
|
|
||||||
**Open questions / blockers:**
|
- CI/backend dev image now installs WeasyPrint system libraries.
|
||||||
- None. The dual-agent handoff system is live and has survived one Codex review round (see DECISIONS.md 2026-04-24 entry; corrections in `9c8ba29`).
|
- Public share-token and survey routes are mounted outside tenant auth; protected share management remains tenant-protected.
|
||||||
|
- Folder creation now persists `UserFolder.account_id`.
|
||||||
|
- Script Builder save-to-library now persists `ScriptTemplate.account_id`.
|
||||||
|
- Resolution output generation eager-loads `AISession.steps` to avoid async lazy-load `MissingGreenlet`.
|
||||||
|
- AI session model now declares the generated `search_vector` column already present in Alembic, so `create_all` test schemas match runtime migrations.
|
||||||
|
- Direct account-role update now rejects `"owner"`; ownership changes must use the transfer path.
|
||||||
|
- Assistant prompt marker examples no longer include a literal executable `create_spin_off_ticket` payload.
|
||||||
|
|
||||||
|
Test/harness fixes:
|
||||||
|
|
||||||
|
- Test seeds updated for tenant-scoped `account_id` columns on sessions, branches, resolution outputs, script templates, PSA connections, folders, schedules, and categories.
|
||||||
|
- Tests aligned with 404-not-403 resource-hiding policy.
|
||||||
|
- Disabled-AI tests now restore both Anthropic and Google key settings.
|
||||||
|
- Pytest harness closes pytest-asyncio's leftover clean loop and ignores known unclosed asyncio/asyncpg teardown ResourceWarnings that otherwise appear at arbitrary later setup points under `filterwarnings = error`.
|
||||||
|
|
||||||
|
## Immediate next steps
|
||||||
|
|
||||||
|
1. Commit current working tree if not already committed with trailer:
|
||||||
|
`Co-Authored-By: Codex <noreply@openai.com>`.
|
||||||
|
2. Check PR #150 status on Gitea. If both `CI / backend (pull_request)` and `CI / frontend (pull_request)` are green, merge it.
|
||||||
|
3. After #150 merges, add `CI / backend (pull_request)` to required status checks on main:
|
||||||
|
```bash
|
||||||
|
PATCH /repos/chihlasm/resolutionflow/branch_protections/main
|
||||||
|
{ "status_check_contexts": ["CI / frontend (pull_request)", "CI / backend (pull_request)"] }
|
||||||
|
```
|
||||||
|
`$GITEA_TOKEN` is in `.claude/settings.local.json`.
|
||||||
|
4. Run/confirm frontend lint if needed for the final DoD item (`npm run lint` was already green after #149, but this session did not rerun it).
|
||||||
|
|
||||||
|
## Open questions
|
||||||
|
|
||||||
|
- PR #150 was not rechecked or merged in this session.
|
||||||
|
- Branch protection was not updated in this session.
|
||||||
|
|||||||
@@ -12,6 +12,29 @@
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## 2026-04-25 06:12 EDT — Codex — Fix backend suite to green
|
||||||
|
|
||||||
|
- Fixed the real backend failures left after the CI-infra cleanup: tenant-scoped seed drift, missing production `account_id` writes, public route mounting for survey/share links, Script Builder library saves, resolution output async loading, AI search schema metadata, disabled-AI fixture leakage, and prompt marker guardrails.
|
||||||
|
- Added backend CI/dev system packages required by WeasyPrint PDF export.
|
||||||
|
- Stabilized the pytest harness for pytest-asyncio/asyncpg teardown ResourceWarnings under `filterwarnings = error`.
|
||||||
|
- Verified `pytest --override-ini="addopts=" -q` inside `resolutionflow_backend`: `1076 passed, 35 deselected in 1347.41s`.
|
||||||
|
- Left for next session: commit/push if needed, check and merge PR #150 when Gitea CI is green, add backend CI as a required branch-protection check, and rerun frontend lint if final DoD requires it.
|
||||||
|
- Files touched: `.gitea/workflows/ci.yml`, `backend/Dockerfile.dev`, `backend/app/api/endpoints/folders.py`, `backend/app/api/endpoints/script_builder.py`, `backend/app/api/endpoints/shares.py`, `backend/app/api/router.py`, `backend/app/models/ai_session.py`, `backend/app/schemas/user.py`, `backend/app/services/assistant_chat_service.py`, `backend/app/services/resolution_output_generator.py`, `backend/app/services/script_builder_service.py`, `backend/pytest.ini`, `backend/tests/conftest.py`, and focused backend tests.
|
||||||
|
|
||||||
|
## 2026-04-25 02:00 America/New_York — Claude Code — Land FlowPilot + PSA, recover CI from 488 errors to ~4
|
||||||
|
|
||||||
|
- Started session by completing pending FlowPilot Phase 9 QA: ran `/qa` against the seeded fixtures, found and fixed four latent layout/state bugs (`ResolutionNotePreview` off-screen, `TemplateMatchPanel` deadlock when TaskLane closed, `EscalateInterceptDialog` clipped above viewport, `seed_test_users.py` `cancel_at_period_end` NOT NULL crash). Added a new fixture seeder `backend/scripts/seed_phase9_qa_fixtures.py` that pre-bakes the four backend states the AI orchestrator needs to emit, so future QA can exercise all 7 conditional Phase 9 components without depending on stochastic AI behavior.
|
||||||
|
- Discovered PR #141 (PSA ticket management) and `feat/flowpilot-migration` had 5 overlapping files but only 2 real conflicts (`CLAUDE.md`, `AssistantChatPage.tsx`). Conflicts were both additive — concatenated rather than chose-a-side.
|
||||||
|
- Merged PSA first (PR #141), then merged FlowPilot (PR #147), each through Gitea API. `tsc -b` clean and visual smoke-test confirmed PSA's Tickets sidebar coexists with Phase 9 ProposalBanner.
|
||||||
|
- Discovered main had been merging through a broken CI gate for several merges. Initially recommended "stop the line, fix CI before shipping." After scoping the actual rot (~50% of tests red, ~600 errors on a clean run), reversed the recommendation: ship the queue first because FlowPilot itself carried significant test-infra repairs that would be duplicated work on a fresh recovery branch.
|
||||||
|
- PR #148: two surgical fixes to main (network_diagrams JSONB `server_default` triple-quote bug, deprecated session-scoped `event_loop` fixture in conftest). +78 passing / -114 errors.
|
||||||
|
- PR #149: frontend lint `20 errors → 0`, `requirements-dev.txt` pytest pin bumped to satisfy `pytest-asyncio==0.24.0`'s `pytest>=8.2`, and a one-line `from app import models as _models` in conftest that registers all ~60 models with `Base.metadata` before `create_all`. The conftest fix collapsed 484 of the remaining 488 backend errors. `1018 passed / 4 errors / 54 failed` after.
|
||||||
|
- Enabled Gitea branch protection on `main`: PR-only merges, `CI / frontend (pull_request)` required, force-push blocked, no review required.
|
||||||
|
- Discovered CI on the merge commit STILL showed red despite local pytest being mostly green. Root cause: workflow only set `DATABASE_URL`, but conftest reads only `DATABASE_TEST_URL` (per `dab740d`'s safety hardening). 638 connection-refused errors on every fixture setup. Plus `actions/upload-artifact@v4` not supported by Gitea Actions. PR #150 fixes both.
|
||||||
|
- Left for next session: merge PR #150 once CI confirms green, add `CI / backend (pull_request)` to required status checks, then root-cause and fix the 54 real backend test failures (one sample seen — `test_user` fixture leaking across calls causing duplicate-email violations).
|
||||||
|
- Files touched (committed): `backend/scripts/seed_test_users.py`, `backend/scripts/seed_phase9_qa_fixtures.py` (new), `backend/app/models/network_diagram.py`, `backend/tests/conftest.py`, `backend/requirements-dev.txt`, `frontend/src/components/pilot/ResolutionNotePreview.tsx`, `frontend/src/components/pilot/EscalateInterceptDialog.tsx`, `frontend/src/components/pilot/ScriptBuilderTab.tsx`, `frontend/src/pages/AssistantChatPage.tsx`, `frontend/src/pages/FlowPilotSessionPage.tsx`, `frontend/src/pages/TicketsPage.tsx`, `frontend/src/hooks/useFlowPilotSession.ts`, `frontend/src/hooks/useMediaQuery.ts`, `frontend/src/components/dashboard/TicketQueue.tsx`, `frontend/src/components/network/nodes/DeviceNode.tsx`, `frontend/src/components/network/nodes/GroupNode.tsx`, `frontend/src/components/routing/AssistantSessionRedirect.tsx` (new), `frontend/src/router.tsx`, `.gitea/workflows/ci.yml`, `.claude/settings.json` (new), `.claude/hooks/check-gstack.sh` (new), `.gitignore`, `CLAUDE.md`, `.gstack/qa-reports/phase9-*/` (QA artifacts).
|
||||||
|
- Net merges to main: PR #141 (PSA), PR #147 (FlowPilot), PR #148 (CI fixes part 1), PR #149 (CI fixes part 2). PR #150 still open at session end.
|
||||||
|
|
||||||
## 2026-04-24 — Claude Code — Migrate to dual-agent handoff system
|
## 2026-04-24 — Claude Code — Migrate to dual-agent handoff system
|
||||||
|
|
||||||
- Split CLAUDE.md into `.ai/PROJECT_CONTEXT.md` + shared-protocol root files (`CLAUDE.md`, `AGENTS.md`).
|
- Split CLAUDE.md into `.ai/PROJECT_CONTEXT.md` + shared-protocol root files (`CLAUDE.md`, `AGENTS.md`).
|
||||||
|
|||||||
@@ -5,8 +5,9 @@
|
|||||||
|
|
||||||
## Up next
|
## Up next
|
||||||
|
|
||||||
- [ ] No queued backlog yet.
|
- [ ] **Parallelize backend pytest with pytest-xdist.** Currently the backend suite takes ~22 min wall-clock for `1076 passed, 35 deselected` (verified locally 2026-04-25). With `-n auto` on the homelab Gitea Actions runner, this should land in the 3–6 min range depending on core count. Blocker: `test_db` fixture in `backend/tests/conftest.py` does `DROP SCHEMA public CASCADE` per test, which two workers would race on. Standard fix: one database per worker, derived from `PYTEST_XDIST_WORKER` env var inside conftest. The runner has spare CPU, so prioritize once main is green and the 54-failure cleanup has landed.
|
||||||
|
|
||||||
## Backlog
|
## Backlog
|
||||||
|
|
||||||
- [ ] **AssistantChatPage `currentChatRef` guard is a silent return** — `handleSend`, `handleTaskSubmit`, `selectChat`, `refreshFacts`, `refreshActiveFix`, and `refreshPreview` all bail with `if (currentChatRef.current !== sentForChatId) return` when stale. This is by design for chat switching, but it also silently masked the prefill-ref bug fixed in PR #153 — the user just saw "no AI response" with no log, no toast, no Sentry event. Either (a) log a `console.warn`/Sentry breadcrumb on the mismatch path so future drift is visible, or (b) split "expected stale" (chat switch) from "unexpected stale" (ref never updated) so only the latter alerts. Pair with an audit of every `currentChatRef.current = ...` assignment vs every `setActiveChatId(...)` call to make sure they're paired everywhere.
|
- [ ] **Frontend lint warnings cleanup.** 23 `react-hooks/exhaustive-deps` warnings remain after PR #149 (mostly missing-deps in useEffect). Either fix them or audit them for known-safe ones and add eslint-disable comments. Not blocking CI today.
|
||||||
|
- [ ] **Audit `filterwarnings` ignores added in `wip(handoff): restore backend suite to green`.** Codex added narrow `ResourceWarning` filters for unclosed socket/transport/event-loop noise from pytest-asyncio teardown. Worth periodically reviewing whether those are still needed (e.g. when bumping pytest-asyncio) — if a real warning appears in those forms it would be silenced.
|
||||||
|
|||||||
@@ -28,6 +28,12 @@ jobs:
|
|||||||
env:
|
env:
|
||||||
DATABASE_URL: postgresql+asyncpg://postgres:postgres@postgres:5432/resolutionflow_test
|
DATABASE_URL: postgresql+asyncpg://postgres:postgres@postgres:5432/resolutionflow_test
|
||||||
DATABASE_URL_SYNC: postgresql://postgres:postgres@postgres:5432/resolutionflow_test
|
DATABASE_URL_SYNC: postgresql://postgres:postgres@postgres:5432/resolutionflow_test
|
||||||
|
# conftest.py reads DATABASE_TEST_URL only (DATABASE_URL is intentionally
|
||||||
|
# not consulted after the dab740d test-isolation hardening). The CI test
|
||||||
|
# DB is the same postgres service, so point DATABASE_TEST_URL at it
|
||||||
|
# explicitly — without this, conftest falls back to localhost:5432 and
|
||||||
|
# all tests fail at fixture setup with "connection refused".
|
||||||
|
DATABASE_TEST_URL: postgresql+asyncpg://postgres:postgres@postgres:5432/resolutionflow_test
|
||||||
SECRET_KEY: ci-test-secret-key-not-for-production
|
SECRET_KEY: ci-test-secret-key-not-for-production
|
||||||
DEBUG: "true"
|
DEBUG: "true"
|
||||||
APP_NAME: ResolutionFlow
|
APP_NAME: ResolutionFlow
|
||||||
@@ -37,6 +43,19 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Cache pip
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.cache/pip
|
||||||
|
key: pip-${{ runner.os }}-${{ hashFiles('backend/requirements.txt', 'backend/requirements-dev.txt') }}
|
||||||
|
restore-keys: |
|
||||||
|
pip-${{ runner.os }}-
|
||||||
|
|
||||||
|
- name: Install system dependencies
|
||||||
|
run: |
|
||||||
|
apt-get update
|
||||||
|
apt-get install -y libpango1.0-dev libcairo2-dev libgdk-pixbuf-2.0-dev libffi-dev libjpeg-dev zlib1g-dev
|
||||||
|
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: pip install --break-system-packages -r backend/requirements.txt -r backend/requirements-dev.txt
|
run: pip install --break-system-packages -r backend/requirements.txt -r backend/requirements-dev.txt
|
||||||
|
|
||||||
@@ -47,7 +66,15 @@ jobs:
|
|||||||
run: cd backend && python scripts/check_tenant_filters.py
|
run: cd backend && python scripts/check_tenant_filters.py
|
||||||
|
|
||||||
- name: Run tests with coverage
|
- name: Run tests with coverage
|
||||||
run: cd backend && python -m pytest --override-ini="addopts=" --cov=app --cov-report=term-missing --cov-report=json:coverage.json --cov-fail-under=50
|
# `-n auto` parallelizes across all runner cores via pytest-xdist.
|
||||||
|
# conftest.py creates a per-worker DB (resolutionflow_test_gw0,
|
||||||
|
# resolutionflow_test_gw1, …) so the per-test DROP SCHEMA doesn't
|
||||||
|
# race across workers. Master/serial runs keep the base DB.
|
||||||
|
# term-missing dropped — the custom "Display coverage summary" step
|
||||||
|
# below parses coverage.json and prints the same info more concisely.
|
||||||
|
# --maxfail=10 short-circuits on structural breakage so we don't burn
|
||||||
|
# 25 minutes when a fixture explodes.
|
||||||
|
run: cd backend && python -m pytest --override-ini="addopts=" -n auto --maxfail=10 --cov=app --cov-report=json:coverage.json --cov-fail-under=50
|
||||||
|
|
||||||
- name: Display coverage summary
|
- name: Display coverage summary
|
||||||
if: always()
|
if: always()
|
||||||
@@ -75,6 +102,14 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Cache npm
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.npm
|
||||||
|
key: npm-${{ runner.os }}-${{ hashFiles('frontend/package-lock.json') }}
|
||||||
|
restore-keys: |
|
||||||
|
npm-${{ runner.os }}-
|
||||||
|
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: cd frontend && npm ci
|
run: cd frontend && npm ci
|
||||||
|
|
||||||
@@ -88,7 +123,7 @@ jobs:
|
|||||||
run: cd frontend && NODE_OPTIONS="--max-old-space-size=4096" npm run build
|
run: cd frontend && NODE_OPTIONS="--max-old-space-size=4096" npm run build
|
||||||
|
|
||||||
- name: Upload build artifact
|
- name: Upload build artifact
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: frontend-dist
|
name: frontend-dist
|
||||||
path: frontend/dist
|
path: frontend/dist
|
||||||
@@ -125,6 +160,22 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Cache pip
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.cache/pip
|
||||||
|
key: pip-${{ runner.os }}-${{ hashFiles('backend/requirements.txt', 'backend/requirements-dev.txt') }}
|
||||||
|
restore-keys: |
|
||||||
|
pip-${{ runner.os }}-
|
||||||
|
|
||||||
|
- name: Cache npm
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.npm
|
||||||
|
key: npm-${{ runner.os }}-${{ hashFiles('frontend/package-lock.json') }}
|
||||||
|
restore-keys: |
|
||||||
|
npm-${{ runner.os }}-
|
||||||
|
|
||||||
- name: Install backend dependencies
|
- name: Install backend dependencies
|
||||||
run: pip install --break-system-packages -r backend/requirements.txt -r backend/requirements-dev.txt
|
run: pip install --break-system-packages -r backend/requirements.txt -r backend/requirements-dev.txt
|
||||||
|
|
||||||
@@ -132,7 +183,7 @@ jobs:
|
|||||||
run: cd frontend && npm ci
|
run: cd frontend && npm ci
|
||||||
|
|
||||||
- name: Download frontend build
|
- name: Download frontend build
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: frontend-dist
|
name: frontend-dist
|
||||||
path: frontend/dist
|
path: frontend/dist
|
||||||
@@ -145,7 +196,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Upload Playwright report
|
- name: Upload Playwright report
|
||||||
if: always()
|
if: always()
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: playwright-report
|
name: playwright-report
|
||||||
path: |
|
path: |
|
||||||
|
|||||||
@@ -5,6 +5,12 @@ WORKDIR /app
|
|||||||
RUN apt-get update && apt-get install -y \
|
RUN apt-get update && apt-get install -y \
|
||||||
gcc \
|
gcc \
|
||||||
libpq-dev \
|
libpq-dev \
|
||||||
|
libpango1.0-dev \
|
||||||
|
libcairo2-dev \
|
||||||
|
libgdk-pixbuf-2.0-dev \
|
||||||
|
libffi-dev \
|
||||||
|
libjpeg-dev \
|
||||||
|
zlib1g-dev \
|
||||||
&& rm -rf /var/lib/apt/lists/*
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
COPY requirements.txt requirements-dev.txt ./
|
COPY requirements.txt requirements-dev.txt ./
|
||||||
|
|||||||
@@ -194,6 +194,7 @@ async def create_folder(
|
|||||||
|
|
||||||
new_folder = UserFolder(
|
new_folder = UserFolder(
|
||||||
user_id=current_user.id,
|
user_id=current_user.id,
|
||||||
|
account_id=current_user.account_id,
|
||||||
name=folder_data.name,
|
name=folder_data.name,
|
||||||
color=folder_data.color,
|
color=folder_data.color,
|
||||||
icon=folder_data.icon,
|
icon=folder_data.icon,
|
||||||
|
|||||||
@@ -260,6 +260,7 @@ async def save_to_library(
|
|||||||
category_id=data.category_id,
|
category_id=data.category_id,
|
||||||
share_with_team=data.share_with_team,
|
share_with_team=data.share_with_team,
|
||||||
user_id=current_user.id,
|
user_id=current_user.id,
|
||||||
|
account_id=current_user.account_id,
|
||||||
team_id=current_user.team_id,
|
team_id=current_user.team_id,
|
||||||
script_body=data.script_body,
|
script_body=data.script_body,
|
||||||
parameters_schema=data.parameters_schema,
|
parameters_schema=data.parameters_schema,
|
||||||
|
|||||||
@@ -20,6 +20,7 @@ from app.core.audit import log_audit
|
|||||||
from app.core.rate_limit import limiter
|
from app.core.rate_limit import limiter
|
||||||
|
|
||||||
router = APIRouter(tags=["shares"])
|
router = APIRouter(tags=["shares"])
|
||||||
|
public_router = APIRouter(tags=["shares"])
|
||||||
|
|
||||||
|
|
||||||
def build_share_response(share: SessionShare) -> ShareResponse:
|
def build_share_response(share: SessionShare) -> ShareResponse:
|
||||||
@@ -206,7 +207,7 @@ async def _get_optional_user(request: Request, db: AsyncSession) -> Optional[Use
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
@router.get("/share/{share_token}", response_model=SharePublicView)
|
@public_router.get("/share/{share_token}", response_model=SharePublicView)
|
||||||
@limiter.limit("30/minute")
|
@limiter.limit("30/minute")
|
||||||
async def access_share(
|
async def access_share(
|
||||||
share_token: str,
|
share_token: str,
|
||||||
|
|||||||
@@ -78,9 +78,11 @@ api_router = APIRouter()
|
|||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
api_router.include_router(auth.router)
|
api_router.include_router(auth.router)
|
||||||
api_router.include_router(shared.router) # Public share links (no auth)
|
api_router.include_router(shared.router) # Public share links (no auth)
|
||||||
|
api_router.include_router(shares.public_router) # Public session share links (optional auth)
|
||||||
api_router.include_router(beta_signup.router)
|
api_router.include_router(beta_signup.router)
|
||||||
api_router.include_router(webhooks.router) # Stripe webhook receiver
|
api_router.include_router(webhooks.router) # Stripe webhook receiver
|
||||||
api_router.include_router(public_templates.router) # Public gallery (no auth, rate-limited)
|
api_router.include_router(public_templates.router) # Public gallery (no auth, rate-limited)
|
||||||
|
api_router.include_router(survey.router) # Public survey flow (no auth, rate-limited)
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
# Admin endpoints — super_admin only
|
# Admin endpoints — super_admin only
|
||||||
@@ -125,7 +127,6 @@ api_router.include_router(ai_fix.router, dependencies=_tenant_deps)
|
|||||||
api_router.include_router(ai_chat.router, dependencies=_tenant_deps)
|
api_router.include_router(ai_chat.router, dependencies=_tenant_deps)
|
||||||
api_router.include_router(copilot.router, dependencies=_tenant_deps)
|
api_router.include_router(copilot.router, dependencies=_tenant_deps)
|
||||||
api_router.include_router(assistant_chat.router, dependencies=_tenant_deps)
|
api_router.include_router(assistant_chat.router, dependencies=_tenant_deps)
|
||||||
api_router.include_router(survey.router, dependencies=_tenant_deps)
|
|
||||||
api_router.include_router(tree_transfer.router, dependencies=_tenant_deps)
|
api_router.include_router(tree_transfer.router, dependencies=_tenant_deps)
|
||||||
api_router.include_router(ai_suggestions.router, dependencies=_tenant_deps)
|
api_router.include_router(ai_suggestions.router, dependencies=_tenant_deps)
|
||||||
api_router.include_router(kb_accelerator.router, dependencies=_tenant_deps)
|
api_router.include_router(kb_accelerator.router, dependencies=_tenant_deps)
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ from typing import Optional, Any, TYPE_CHECKING
|
|||||||
from sqlalchemy import String, Text, DateTime, ForeignKey, Boolean, Integer, Float, CheckConstraint
|
from sqlalchemy import String, Text, DateTime, ForeignKey, Boolean, Integer, Float, CheckConstraint
|
||||||
import sqlalchemy as sa
|
import sqlalchemy as sa
|
||||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||||
from sqlalchemy.dialects.postgresql import UUID, JSONB
|
from sqlalchemy.dialects.postgresql import UUID, JSONB, TSVECTOR
|
||||||
|
|
||||||
from app.core.database import Base
|
from app.core.database import Base
|
||||||
|
|
||||||
@@ -46,6 +46,7 @@ class AISession(Base):
|
|||||||
"confidence_tier IN ('guided', 'exploring', 'discovery')",
|
"confidence_tier IN ('guided', 'exploring', 'discovery')",
|
||||||
name="ck_ai_sessions_confidence_tier",
|
name="ck_ai_sessions_confidence_tier",
|
||||||
),
|
),
|
||||||
|
sa.Index("idx_ai_sessions_search", "search_vector", postgresql_using="gin"),
|
||||||
)
|
)
|
||||||
|
|
||||||
id: Mapped[uuid.UUID] = mapped_column(
|
id: Mapped[uuid.UUID] = mapped_column(
|
||||||
@@ -150,6 +151,18 @@ class AISession(Base):
|
|||||||
Text, nullable=True,
|
Text, nullable=True,
|
||||||
comment="Why escalated (set on escalation)",
|
comment="Why escalated (set on escalation)",
|
||||||
)
|
)
|
||||||
|
search_vector: Mapped[Optional[str]] = mapped_column(
|
||||||
|
TSVECTOR,
|
||||||
|
sa.Computed(
|
||||||
|
"to_tsvector('english', "
|
||||||
|
"coalesce(problem_summary, '') || ' ' || "
|
||||||
|
"coalesce(resolution_summary, '') || ' ' || "
|
||||||
|
"coalesce(escalation_reason, '') || ' ' || "
|
||||||
|
"coalesce(problem_domain, ''))",
|
||||||
|
persisted=True,
|
||||||
|
),
|
||||||
|
nullable=True,
|
||||||
|
)
|
||||||
escalation_package: Mapped[Optional[dict[str, Any]]] = mapped_column(
|
escalation_package: Mapped[Optional[dict[str, Any]]] = mapped_column(
|
||||||
JSONB, nullable=True,
|
JSONB, nullable=True,
|
||||||
comment="Context package for receiving engineer: steps_tried, hypotheses, suggestions",
|
comment="Context package for receiving engineer: steps_tried, hypotheses, suggestions",
|
||||||
|
|||||||
@@ -68,4 +68,6 @@ class RoleUpdate(BaseModel):
|
|||||||
|
|
||||||
|
|
||||||
class AccountRoleUpdate(BaseModel):
|
class AccountRoleUpdate(BaseModel):
|
||||||
account_role: str = Field(..., pattern="^(owner|admin|engineer|viewer)$")
|
# Ownership changes must go through the explicit transfer-ownership flow so
|
||||||
|
# account.owner_id stays consistent with user.account_role.
|
||||||
|
account_role: str = Field(..., pattern="^(admin|engineer|viewer)$")
|
||||||
|
|||||||
@@ -300,13 +300,14 @@ To create a fork, append this marker AFTER your [QUESTIONS]/[ACTIONS] markers:
|
|||||||
When you identify a second distinct issue that is clearly separate from the primary topic \
|
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 below. \
|
of this session, suggest creating a spin-off ticket using the [ACTIONS] marker below. \
|
||||||
Use this sparingly — only when the issue is genuinely independent, not for every tangential mention.
|
Use this sparingly — only when the issue is genuinely independent, not for every tangential mention.
|
||||||
|
Use `create_spin_off_ticket` as the command value for this action.
|
||||||
|
|
||||||
Format:
|
Format:
|
||||||
[ACTIONS]
|
[ACTIONS]
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
"label": "Create ticket: <brief issue title>",
|
"label": "Create ticket: <brief issue title>",
|
||||||
"command": "create_spin_off_ticket",
|
"command": "<spin-off ticket action command>",
|
||||||
"description": "<one sentence description of the separate issue>"
|
"description": "<one sentence description of the separate issue>"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ from uuid import UUID
|
|||||||
|
|
||||||
from sqlalchemy import select
|
from sqlalchemy import select
|
||||||
from sqlalchemy.ext.asyncio import AsyncSession
|
from sqlalchemy.ext.asyncio import AsyncSession
|
||||||
|
from sqlalchemy.orm import selectinload
|
||||||
|
|
||||||
from app.models.ai_session import AISession
|
from app.models.ai_session import AISession
|
||||||
from app.models.session_resolution_output import SessionResolutionOutput
|
from app.models.session_resolution_output import SessionResolutionOutput
|
||||||
@@ -21,7 +22,9 @@ class ResolutionOutputGenerator:
|
|||||||
|
|
||||||
async def generate_all(self, session_id: UUID) -> list[SessionResolutionOutput]:
|
async def generate_all(self, session_id: UUID) -> list[SessionResolutionOutput]:
|
||||||
result = await self.db.execute(
|
result = await self.db.execute(
|
||||||
select(AISession).where(AISession.id == session_id)
|
select(AISession)
|
||||||
|
.options(selectinload(AISession.steps))
|
||||||
|
.where(AISession.id == session_id)
|
||||||
)
|
)
|
||||||
session = result.scalar_one_or_none()
|
session = result.scalar_one_or_none()
|
||||||
if not session:
|
if not session:
|
||||||
|
|||||||
@@ -360,6 +360,7 @@ async def save_to_library(
|
|||||||
category_id: UUID | None,
|
category_id: UUID | None,
|
||||||
share_with_team: bool,
|
share_with_team: bool,
|
||||||
user_id: UUID,
|
user_id: UUID,
|
||||||
|
account_id: UUID,
|
||||||
team_id: UUID | None,
|
team_id: UUID | None,
|
||||||
script_body: str | None = None,
|
script_body: str | None = None,
|
||||||
parameters_schema: dict | None = None,
|
parameters_schema: dict | None = None,
|
||||||
@@ -401,6 +402,7 @@ async def save_to_library(
|
|||||||
id=uuid_mod.uuid4(),
|
id=uuid_mod.uuid4(),
|
||||||
category_id=resolved_category_id,
|
category_id=resolved_category_id,
|
||||||
created_by=user_id,
|
created_by=user_id,
|
||||||
|
account_id=account_id,
|
||||||
team_id=team_id if share_with_team else None,
|
team_id=team_id if share_with_team else None,
|
||||||
name=name,
|
name=name,
|
||||||
slug=slug,
|
slug=slug,
|
||||||
|
|||||||
@@ -35,6 +35,9 @@ testpaths = tests
|
|||||||
# Warnings
|
# Warnings
|
||||||
filterwarnings =
|
filterwarnings =
|
||||||
error
|
error
|
||||||
|
ignore:unclosed <socket\.socket.*:ResourceWarning
|
||||||
|
ignore:unclosed transport .*:ResourceWarning
|
||||||
|
ignore:unclosed event loop .*:ResourceWarning
|
||||||
ignore::DeprecationWarning
|
ignore::DeprecationWarning
|
||||||
ignore::PendingDeprecationWarning
|
ignore::PendingDeprecationWarning
|
||||||
ignore::pluggy.PluggyTeardownRaisedWarning
|
ignore::pluggy.PluggyTeardownRaisedWarning
|
||||||
|
|||||||
@@ -4,6 +4,7 @@
|
|||||||
# Testing — pytest-asyncio 0.24+ requires pytest>=8.2
|
# Testing — pytest-asyncio 0.24+ requires pytest>=8.2
|
||||||
pytest==8.4.2
|
pytest==8.4.2
|
||||||
pytest-asyncio==0.24.0
|
pytest-asyncio==0.24.0
|
||||||
|
pytest-xdist==3.6.1
|
||||||
httpx>=0.27.0
|
httpx>=0.27.0
|
||||||
pytest-cov==5.0.0
|
pytest-cov==5.0.0
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ Provides test database setup, client fixtures, and authentication helpers.
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
import os
|
import os
|
||||||
|
import asyncio
|
||||||
from typing import AsyncGenerator
|
from typing import AsyncGenerator
|
||||||
import pytest
|
import pytest
|
||||||
import sqlalchemy as sa
|
import sqlalchemy as sa
|
||||||
@@ -34,11 +35,64 @@ settings.REQUIRE_INVITE_CODE = False
|
|||||||
# would silently nuke the dev database. Only DATABASE_TEST_URL is honored,
|
# would silently nuke the dev database. Only DATABASE_TEST_URL is honored,
|
||||||
# and the safety assertion below refuses to run against a DB whose name
|
# and the safety assertion below refuses to run against a DB whose name
|
||||||
# doesn't contain "test".
|
# doesn't contain "test".
|
||||||
TEST_DATABASE_URL = os.environ.get(
|
_BASE_TEST_DATABASE_URL = os.environ.get(
|
||||||
"DATABASE_TEST_URL",
|
"DATABASE_TEST_URL",
|
||||||
"postgresql+asyncpg://postgres:postgres@localhost:5432/resolutionflow_test",
|
"postgresql+asyncpg://postgres:postgres@localhost:5432/resolutionflow_test",
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _worker_db_url(base_url: str) -> str:
|
||||||
|
"""Per-worker DB URL for pytest-xdist parallelization.
|
||||||
|
|
||||||
|
pytest-xdist sets PYTEST_XDIST_WORKER to 'gw0', 'gw1', ... per worker
|
||||||
|
process. Each worker needs its own database so the per-test
|
||||||
|
`DROP SCHEMA public CASCADE` doesn't race across workers. Master/serial
|
||||||
|
runs (no xdist) keep the base DB. The base DB is created by the postgres
|
||||||
|
service container; per-worker DBs are CREATE DATABASE-d on first import
|
||||||
|
by `_ensure_worker_db_exists` below.
|
||||||
|
"""
|
||||||
|
worker = os.environ.get("PYTEST_XDIST_WORKER")
|
||||||
|
if not worker or worker == "master":
|
||||||
|
return base_url
|
||||||
|
head, tail = base_url.rsplit("/", 1)
|
||||||
|
db_name, _, query = tail.partition("?")
|
||||||
|
suffix = f"?{query}" if query else ""
|
||||||
|
return f"{head}/{db_name}_{worker}{suffix}"
|
||||||
|
|
||||||
|
|
||||||
|
def _ensure_worker_db_exists(worker_url: str, base_url: str) -> None:
|
||||||
|
"""Create the per-worker DB if it doesn't exist. Runs synchronously at
|
||||||
|
conftest import time (before any async test machinery), using psycopg2
|
||||||
|
against the postgres maintenance DB. No-op when not running under xdist.
|
||||||
|
"""
|
||||||
|
if worker_url == base_url:
|
||||||
|
return
|
||||||
|
head, tail = worker_url.rsplit("/", 1)
|
||||||
|
worker_db = tail.partition("?")[0]
|
||||||
|
# Strip the +asyncpg dialect for sync psycopg2 + connect to 'postgres'.
|
||||||
|
sync_head = head.replace("+asyncpg", "")
|
||||||
|
admin_url = f"{sync_head}/postgres"
|
||||||
|
# Lazy import — psycopg2 is a transitive backend dep; not imported at
|
||||||
|
# module top to keep the conftest light when xdist isn't in use.
|
||||||
|
from sqlalchemy import create_engine
|
||||||
|
engine = create_engine(admin_url, isolation_level="AUTOCOMMIT")
|
||||||
|
try:
|
||||||
|
with engine.begin() as conn:
|
||||||
|
exists = conn.execute(
|
||||||
|
sa.text("SELECT 1 FROM pg_database WHERE datname = :n"),
|
||||||
|
{"n": worker_db},
|
||||||
|
).scalar()
|
||||||
|
if not exists:
|
||||||
|
# Identifier interpolation is safe — worker_db is built from
|
||||||
|
# the trusted base URL + 'gw\d+' worker suffix.
|
||||||
|
conn.execute(sa.text(f'CREATE DATABASE "{worker_db}"'))
|
||||||
|
finally:
|
||||||
|
engine.dispose()
|
||||||
|
|
||||||
|
|
||||||
|
TEST_DATABASE_URL = _worker_db_url(_BASE_TEST_DATABASE_URL)
|
||||||
|
_ensure_worker_db_exists(TEST_DATABASE_URL, _BASE_TEST_DATABASE_URL)
|
||||||
|
|
||||||
# Belt-and-suspenders: refuse to run tests against a DB whose name doesn't
|
# Belt-and-suspenders: refuse to run tests against a DB whose name doesn't
|
||||||
# contain "test". Parses the last path segment of the URL (everything after
|
# contain "test". Parses the last path segment of the URL (everything after
|
||||||
# the final '/', with query string stripped) so credentials / hosts that
|
# the final '/', with query string stripped) so credentials / hosts that
|
||||||
@@ -73,6 +127,20 @@ def pytest_collection_modifyitems(config, items):
|
|||||||
items[:] = selected
|
items[:] = selected
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.hookimpl(trylast=True, hookwrapper=True)
|
||||||
|
def pytest_runtest_teardown(item, nextitem):
|
||||||
|
"""Close pytest-asyncio's post-test clean loop before warnings collect it."""
|
||||||
|
yield
|
||||||
|
policy = asyncio.get_event_loop_policy()
|
||||||
|
try:
|
||||||
|
loop = policy.get_event_loop()
|
||||||
|
except RuntimeError:
|
||||||
|
return
|
||||||
|
if not loop.is_running() and not loop.is_closed():
|
||||||
|
loop.close()
|
||||||
|
policy.set_event_loop(None)
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
async def test_db() -> AsyncGenerator[AsyncSession, None]:
|
async def test_db() -> AsyncGenerator[AsyncSession, None]:
|
||||||
"""
|
"""
|
||||||
@@ -137,6 +205,7 @@ async def test_db() -> AsyncGenerator[AsyncSession, None]:
|
|||||||
# Dispose engine first so all pooled connections are released,
|
# Dispose engine first so all pooled connections are released,
|
||||||
# then reconnect to perform the schema teardown cleanly.
|
# then reconnect to perform the schema teardown cleanly.
|
||||||
await engine.dispose()
|
await engine.dispose()
|
||||||
|
await asyncio.sleep(0.01)
|
||||||
|
|
||||||
# Drop all tables after test (CASCADE for circular FKs)
|
# Drop all tables after test (CASCADE for circular FKs)
|
||||||
teardown_engine = create_async_engine(
|
teardown_engine = create_async_engine(
|
||||||
@@ -150,6 +219,7 @@ async def test_db() -> AsyncGenerator[AsyncSession, None]:
|
|||||||
await conn.execute(sa.text("CREATE SCHEMA public"))
|
await conn.execute(sa.text("CREATE SCHEMA public"))
|
||||||
finally:
|
finally:
|
||||||
await teardown_engine.dispose()
|
await teardown_engine.dispose()
|
||||||
|
await asyncio.sleep(0.01)
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
|
|||||||
@@ -74,19 +74,25 @@ def _mock_ai_provider(text: str, input_tokens: int = 100, output_tokens: int = 2
|
|||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def enable_ai():
|
def enable_ai():
|
||||||
"""Temporarily enable AI by setting a fake API key."""
|
"""Temporarily enable AI by setting a fake API key."""
|
||||||
original = settings.ANTHROPIC_API_KEY
|
original_anthropic = settings.ANTHROPIC_API_KEY
|
||||||
|
original_google = settings.GOOGLE_AI_API_KEY
|
||||||
settings.ANTHROPIC_API_KEY = "test-key-fake"
|
settings.ANTHROPIC_API_KEY = "test-key-fake"
|
||||||
|
settings.GOOGLE_AI_API_KEY = None
|
||||||
yield
|
yield
|
||||||
settings.ANTHROPIC_API_KEY = original
|
settings.ANTHROPIC_API_KEY = original_anthropic
|
||||||
|
settings.GOOGLE_AI_API_KEY = original_google
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def disable_ai():
|
def disable_ai():
|
||||||
"""Ensure AI is disabled."""
|
"""Ensure AI is disabled."""
|
||||||
original = settings.ANTHROPIC_API_KEY
|
original_anthropic = settings.ANTHROPIC_API_KEY
|
||||||
|
original_google = settings.GOOGLE_AI_API_KEY
|
||||||
settings.ANTHROPIC_API_KEY = None
|
settings.ANTHROPIC_API_KEY = None
|
||||||
|
settings.GOOGLE_AI_API_KEY = None
|
||||||
yield
|
yield
|
||||||
settings.ANTHROPIC_API_KEY = original
|
settings.ANTHROPIC_API_KEY = original_anthropic
|
||||||
|
settings.GOOGLE_AI_API_KEY = original_google
|
||||||
|
|
||||||
|
|
||||||
# ── Quota endpoint ──
|
# ── Quota endpoint ──
|
||||||
|
|||||||
@@ -66,6 +66,7 @@ async def test_create_fork(client: AsyncClient, test_user, auth_headers, test_db
|
|||||||
|
|
||||||
step = AISessionStep(
|
step = AISessionStep(
|
||||||
session_id=session.id,
|
session_id=session.id,
|
||||||
|
account_id=session.account_id,
|
||||||
step_order=0,
|
step_order=0,
|
||||||
step_type="question",
|
step_type="question",
|
||||||
content={"text": "What's the issue?"},
|
content={"text": "What's the issue?"},
|
||||||
@@ -119,7 +120,7 @@ async def test_switch_branch(client: AsyncClient, test_user, auth_headers, test_
|
|||||||
root = await manager.create_root_branch(session.id)
|
root = await manager.create_root_branch(session.id)
|
||||||
|
|
||||||
step = AISessionStep(
|
step = AISessionStep(
|
||||||
session_id=session.id, step_order=0, step_type="question",
|
session_id=session.id, account_id=session.account_id, step_order=0, step_type="question",
|
||||||
content={"text": "test"}, confidence_at_step=0.5,
|
content={"text": "test"}, confidence_at_step=0.5,
|
||||||
)
|
)
|
||||||
test_db.add(step)
|
test_db.add(step)
|
||||||
@@ -197,7 +198,7 @@ async def test_get_branch_tree(client: AsyncClient, test_user, auth_headers, tes
|
|||||||
root = await manager.create_root_branch(session.id)
|
root = await manager.create_root_branch(session.id)
|
||||||
|
|
||||||
step = AISessionStep(
|
step = AISessionStep(
|
||||||
session_id=session.id, step_order=0, step_type="question",
|
session_id=session.id, account_id=session.account_id, step_order=0, step_type="question",
|
||||||
content={"text": "test"}, confidence_at_step=0.5,
|
content={"text": "test"}, confidence_at_step=0.5,
|
||||||
)
|
)
|
||||||
test_db.add(step)
|
test_db.add(step)
|
||||||
|
|||||||
@@ -50,6 +50,7 @@ async def _make_session(test_db, user, *, with_psa: bool = False) -> AISession:
|
|||||||
conn = PsaConnection(
|
conn = PsaConnection(
|
||||||
account_id=user["user_data"]["account_id"],
|
account_id=user["user_data"]["account_id"],
|
||||||
provider="connectwise",
|
provider="connectwise",
|
||||||
|
display_name="Test ConnectWise",
|
||||||
site_url="https://fake.cw.local",
|
site_url="https://fake.cw.local",
|
||||||
company_id="TEST",
|
company_id="TEST",
|
||||||
credentials_encrypted=encrypt_credentials({"public_key": "x", "private_key": "y"}),
|
credentials_encrypted=encrypt_credentials({"public_key": "x", "private_key": "y"}),
|
||||||
|
|||||||
@@ -472,19 +472,20 @@ class TestScriptBuilderSlugCollision:
|
|||||||
# Pre-create a template with slug "test-script" to cause collision
|
# Pre-create a template with slug "test-script" to cause collision
|
||||||
user_resp = await client.get("/api/v1/auth/me", headers=auth_headers)
|
user_resp = await client.get("/api/v1/auth/me", headers=auth_headers)
|
||||||
user_id = user_resp.json()["id"]
|
user_id = user_resp.json()["id"]
|
||||||
|
account_id = user_resp.json()["account_id"]
|
||||||
await test_db.execute(
|
await test_db.execute(
|
||||||
sa.text("""
|
sa.text("""
|
||||||
INSERT INTO script_templates
|
INSERT INTO script_templates
|
||||||
(id, category_id, created_by, name, slug, script_body,
|
(id, category_id, created_by, account_id, name, slug, script_body,
|
||||||
parameters_schema, default_values, validation_rules, tags,
|
parameters_schema, default_values, validation_rules, tags,
|
||||||
complexity, is_active, version, usage_count, created_at, updated_at)
|
complexity, is_active, version, usage_count, created_at, updated_at)
|
||||||
VALUES
|
VALUES
|
||||||
(:id, 'a0000000-0000-0000-0000-000000000001'::uuid, :uid,
|
(:id, 'a0000000-0000-0000-0000-000000000001'::uuid, :uid, :account_id,
|
||||||
'Test Script', 'test-script', 'echo hello',
|
'Test Script', 'test-script', 'echo hello',
|
||||||
'{"parameters": []}', '{}', '{}', '["powershell"]',
|
'{"parameters": []}', '{}', '{}', '["powershell"]',
|
||||||
'beginner', true, 1, 0, NOW(), NOW())
|
'beginner', true, 1, 0, NOW(), NOW())
|
||||||
"""),
|
"""),
|
||||||
{"id": str(uuid_mod.uuid4()), "uid": user_id},
|
{"id": str(uuid_mod.uuid4()), "uid": user_id, "account_id": account_id},
|
||||||
)
|
)
|
||||||
await test_db.commit()
|
await test_db.commit()
|
||||||
|
|
||||||
@@ -561,6 +562,7 @@ class TestScriptTemplateFilters:
|
|||||||
"""mine=true returns only templates created by the current user."""
|
"""mine=true returns only templates created by the current user."""
|
||||||
user_resp = await client.get("/api/v1/auth/me", headers=auth_headers)
|
user_resp = await client.get("/api/v1/auth/me", headers=auth_headers)
|
||||||
user_id = user_resp.json()["id"]
|
user_id = user_resp.json()["id"]
|
||||||
|
account_id = user_resp.json()["account_id"]
|
||||||
|
|
||||||
second_resp = await client.get("/api/v1/auth/me", headers=second_user_headers)
|
second_resp = await client.get("/api/v1/auth/me", headers=second_user_headers)
|
||||||
second_user_id = second_resp.json()["id"]
|
second_user_id = second_resp.json()["id"]
|
||||||
@@ -571,32 +573,32 @@ class TestScriptTemplateFilters:
|
|||||||
await test_db.execute(
|
await test_db.execute(
|
||||||
sa.text("""
|
sa.text("""
|
||||||
INSERT INTO script_templates
|
INSERT INTO script_templates
|
||||||
(id, category_id, created_by, team_id, name, slug, script_body,
|
(id, category_id, created_by, account_id, team_id, name, slug, script_body,
|
||||||
parameters_schema, default_values, validation_rules, tags,
|
parameters_schema, default_values, validation_rules, tags,
|
||||||
complexity, is_active, version, usage_count, created_at, updated_at)
|
complexity, is_active, version, usage_count, created_at, updated_at)
|
||||||
VALUES
|
VALUES
|
||||||
(:id, :cat, :uid, NULL,
|
(:id, :cat, :uid, :account_id, NULL,
|
||||||
'My Script', 'my-script', 'echo mine',
|
'My Script', 'my-script', 'echo mine',
|
||||||
'{"parameters": []}', '{}', '{}', '[]',
|
'{"parameters": []}', '{}', '{}', '[]',
|
||||||
'beginner', true, 1, 0, NOW(), NOW())
|
'beginner', true, 1, 0, NOW(), NOW())
|
||||||
"""),
|
"""),
|
||||||
{"id": str(uuid_mod.uuid4()), "cat": cat_id, "uid": user_id},
|
{"id": str(uuid_mod.uuid4()), "cat": cat_id, "uid": user_id, "account_id": account_id},
|
||||||
)
|
)
|
||||||
|
|
||||||
# Create template owned by second user (no team_id, so visible to all)
|
# Create template owned by second user (no team_id, so visible to all)
|
||||||
await test_db.execute(
|
await test_db.execute(
|
||||||
sa.text("""
|
sa.text("""
|
||||||
INSERT INTO script_templates
|
INSERT INTO script_templates
|
||||||
(id, category_id, created_by, team_id, name, slug, script_body,
|
(id, category_id, created_by, account_id, team_id, name, slug, script_body,
|
||||||
parameters_schema, default_values, validation_rules, tags,
|
parameters_schema, default_values, validation_rules, tags,
|
||||||
complexity, is_active, version, usage_count, created_at, updated_at)
|
complexity, is_active, version, usage_count, created_at, updated_at)
|
||||||
VALUES
|
VALUES
|
||||||
(:id, :cat, :uid, NULL,
|
(:id, :cat, :uid, :account_id, NULL,
|
||||||
'Other Script', 'other-script', 'echo other',
|
'Other Script', 'other-script', 'echo other',
|
||||||
'{"parameters": []}', '{}', '{}', '[]',
|
'{"parameters": []}', '{}', '{}', '[]',
|
||||||
'beginner', true, 1, 0, NOW(), NOW())
|
'beginner', true, 1, 0, NOW(), NOW())
|
||||||
"""),
|
"""),
|
||||||
{"id": str(uuid_mod.uuid4()), "cat": cat_id, "uid": second_user_id},
|
{"id": str(uuid_mod.uuid4()), "cat": cat_id, "uid": second_user_id, "account_id": account_id},
|
||||||
)
|
)
|
||||||
await test_db.commit()
|
await test_db.commit()
|
||||||
|
|
||||||
@@ -617,6 +619,7 @@ class TestScriptTemplateFilters:
|
|||||||
"""shared=true returns only templates shared with the user's team."""
|
"""shared=true returns only templates shared with the user's team."""
|
||||||
user_resp = await client.get("/api/v1/auth/me", headers=auth_headers)
|
user_resp = await client.get("/api/v1/auth/me", headers=auth_headers)
|
||||||
user_id = user_resp.json()["id"]
|
user_id = user_resp.json()["id"]
|
||||||
|
account_id = user_resp.json()["account_id"]
|
||||||
|
|
||||||
cat_id = "b0000000-0000-0000-0000-000000000001"
|
cat_id = "b0000000-0000-0000-0000-000000000001"
|
||||||
|
|
||||||
@@ -639,32 +642,32 @@ class TestScriptTemplateFilters:
|
|||||||
await test_db.execute(
|
await test_db.execute(
|
||||||
sa.text("""
|
sa.text("""
|
||||||
INSERT INTO script_templates
|
INSERT INTO script_templates
|
||||||
(id, category_id, created_by, team_id, name, slug, script_body,
|
(id, category_id, created_by, account_id, team_id, name, slug, script_body,
|
||||||
parameters_schema, default_values, validation_rules, tags,
|
parameters_schema, default_values, validation_rules, tags,
|
||||||
complexity, is_active, version, usage_count, created_at, updated_at)
|
complexity, is_active, version, usage_count, created_at, updated_at)
|
||||||
VALUES
|
VALUES
|
||||||
(:id, :cat, :uid, :tid,
|
(:id, :cat, :uid, :account_id, :tid,
|
||||||
'Team Script', 'team-script', 'echo team',
|
'Team Script', 'team-script', 'echo team',
|
||||||
'{"parameters": []}', '{}', '{}', '[]',
|
'{"parameters": []}', '{}', '{}', '[]',
|
||||||
'beginner', true, 1, 0, NOW(), NOW())
|
'beginner', true, 1, 0, NOW(), NOW())
|
||||||
"""),
|
"""),
|
||||||
{"id": str(uuid_mod.uuid4()), "cat": cat_id, "uid": user_id, "tid": team_id},
|
{"id": str(uuid_mod.uuid4()), "cat": cat_id, "uid": user_id, "account_id": account_id, "tid": team_id},
|
||||||
)
|
)
|
||||||
|
|
||||||
# Template NOT shared (no team_id)
|
# Template NOT shared (no team_id)
|
||||||
await test_db.execute(
|
await test_db.execute(
|
||||||
sa.text("""
|
sa.text("""
|
||||||
INSERT INTO script_templates
|
INSERT INTO script_templates
|
||||||
(id, category_id, created_by, team_id, name, slug, script_body,
|
(id, category_id, created_by, account_id, team_id, name, slug, script_body,
|
||||||
parameters_schema, default_values, validation_rules, tags,
|
parameters_schema, default_values, validation_rules, tags,
|
||||||
complexity, is_active, version, usage_count, created_at, updated_at)
|
complexity, is_active, version, usage_count, created_at, updated_at)
|
||||||
VALUES
|
VALUES
|
||||||
(:id, :cat, :uid, NULL,
|
(:id, :cat, :uid, :account_id, NULL,
|
||||||
'Personal Script', 'personal-script', 'echo personal',
|
'Personal Script', 'personal-script', 'echo personal',
|
||||||
'{"parameters": []}', '{}', '{}', '[]',
|
'{"parameters": []}', '{}', '{}', '[]',
|
||||||
'beginner', true, 1, 0, NOW(), NOW())
|
'beginner', true, 1, 0, NOW(), NOW())
|
||||||
"""),
|
"""),
|
||||||
{"id": str(uuid_mod.uuid4()), "cat": cat_id, "uid": user_id},
|
{"id": str(uuid_mod.uuid4()), "cat": cat_id, "uid": user_id, "account_id": account_id},
|
||||||
)
|
)
|
||||||
await test_db.commit()
|
await test_db.commit()
|
||||||
|
|
||||||
|
|||||||
@@ -49,7 +49,7 @@ async def test_create_fork(client: AsyncClient, test_user, auth_headers, test_db
|
|||||||
await test_db.flush()
|
await test_db.flush()
|
||||||
|
|
||||||
step = AISessionStep(
|
step = AISessionStep(
|
||||||
session_id=session.id, step_order=0, step_type="question",
|
session_id=session.id, account_id=session.account_id, step_order=0, step_type="question",
|
||||||
content={"text": "test"}, confidence_at_step=0.5,
|
content={"text": "test"}, confidence_at_step=0.5,
|
||||||
)
|
)
|
||||||
test_db.add(step)
|
test_db.add(step)
|
||||||
@@ -88,7 +88,7 @@ async def test_switch_branch(client: AsyncClient, test_user, auth_headers, test_
|
|||||||
await test_db.flush()
|
await test_db.flush()
|
||||||
|
|
||||||
step = AISessionStep(
|
step = AISessionStep(
|
||||||
session_id=session.id, step_order=0, step_type="question",
|
session_id=session.id, account_id=session.account_id, step_order=0, step_type="question",
|
||||||
content={"text": "test"}, confidence_at_step=0.5,
|
content={"text": "test"}, confidence_at_step=0.5,
|
||||||
)
|
)
|
||||||
test_db.add(step)
|
test_db.add(step)
|
||||||
|
|||||||
@@ -45,6 +45,7 @@ async def test_edit_output_api(client: AsyncClient, test_user, auth_headers, tes
|
|||||||
|
|
||||||
output = SessionResolutionOutput(
|
output = SessionResolutionOutput(
|
||||||
session_id=session.id,
|
session_id=session.id,
|
||||||
|
account_id=session.account_id,
|
||||||
output_type="psa_ticket_notes",
|
output_type="psa_ticket_notes",
|
||||||
generated_content="Original",
|
generated_content="Original",
|
||||||
status="draft",
|
status="draft",
|
||||||
|
|||||||
@@ -219,7 +219,7 @@ class TestSessionSharing:
|
|||||||
json={"visibility": "public"},
|
json={"visibility": "public"},
|
||||||
headers=other_headers
|
headers=other_headers
|
||||||
)
|
)
|
||||||
assert response.status_code == 403
|
assert response.status_code == 404
|
||||||
|
|
||||||
async def test_share_nonexistent_session(self, client: AsyncClient, auth_headers):
|
async def test_share_nonexistent_session(self, client: AsyncClient, auth_headers):
|
||||||
"""Creating a share for nonexistent session returns 404."""
|
"""Creating a share for nonexistent session returns 404."""
|
||||||
|
|||||||
@@ -213,15 +213,28 @@ async def test_record_decision_persists_and_bumps_state_version(
|
|||||||
title="x",
|
title="x",
|
||||||
description="y",
|
description="y",
|
||||||
confidence_pct=50,
|
confidence_pct=50,
|
||||||
|
ai_drafted_script="Write-Output 'ok'",
|
||||||
)
|
)
|
||||||
test_db.add(fix)
|
test_db.add(fix)
|
||||||
await test_db.commit()
|
await test_db.commit()
|
||||||
|
|
||||||
r = await client.post(
|
# The draft_template path calls TemplateExtractionService, which needs an
|
||||||
f"/api/v1/ai-sessions/{session.id}/suggested-fixes/{fix.id}/decision",
|
# AI provider configured. CI doesn't set ANTHROPIC_API_KEY/GOOGLE_AI_API_KEY,
|
||||||
headers=auth_headers,
|
# and this test isn't exercising the AI integration — patch the extractor
|
||||||
json={"decision": "draft_template"},
|
# with a minimal valid response so the rest of the decision flow runs.
|
||||||
)
|
extractor_stub = AsyncMock(return_value={
|
||||||
|
"templated_body": "Write-Output 'ok'",
|
||||||
|
"parameters": [],
|
||||||
|
})
|
||||||
|
with patch(
|
||||||
|
"app.api.endpoints.session_suggested_fixes._extract_template_parameters",
|
||||||
|
extractor_stub,
|
||||||
|
):
|
||||||
|
r = await client.post(
|
||||||
|
f"/api/v1/ai-sessions/{session.id}/suggested-fixes/{fix.id}/decision",
|
||||||
|
headers=auth_headers,
|
||||||
|
json={"decision": "draft_template"},
|
||||||
|
)
|
||||||
assert r.status_code == 200
|
assert r.status_code == 200
|
||||||
assert r.json()["user_decision"] == "draft_template"
|
assert r.json()["user_decision"] == "draft_template"
|
||||||
|
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ async def _create_account_and_user(db: AsyncSession, prefix: str):
|
|||||||
async def _login(client: AsyncClient, email: str, password: str) -> dict:
|
async def _login(client: AsyncClient, email: str, password: str) -> dict:
|
||||||
"""Log in and return Authorization headers."""
|
"""Log in and return Authorization headers."""
|
||||||
resp = await client.post(
|
resp = await client.post(
|
||||||
"/api/v1/auth/login",
|
"/api/v1/auth/login/json",
|
||||||
json={"email": email, "password": password},
|
json={"email": email, "password": password},
|
||||||
)
|
)
|
||||||
assert resp.status_code == 200, f"Login failed: {resp.text}"
|
assert resp.status_code == 200, f"Login failed: {resp.text}"
|
||||||
@@ -101,11 +101,11 @@ async def test_category_tree_count_scoped_to_account(
|
|||||||
acct_a, user_a, pass_a = await _create_account_and_user(test_db, "cat-a")
|
acct_a, user_a, pass_a = await _create_account_and_user(test_db, "cat-a")
|
||||||
acct_b, user_b, pass_b = await _create_account_and_user(test_db, "cat-b")
|
acct_b, user_b, pass_b = await _create_account_and_user(test_db, "cat-b")
|
||||||
|
|
||||||
# Shared category (account_id=None means global)
|
# Categories are tenant-scoped; the endpoint must only count account A's trees.
|
||||||
category = TreeCategory(
|
category = TreeCategory(
|
||||||
name="Shared Category",
|
name="Shared Category",
|
||||||
slug=f"shared-cat-{uuid.uuid4().hex[:6]}",
|
slug=f"shared-cat-{uuid.uuid4().hex[:6]}",
|
||||||
account_id=None,
|
account_id=acct_a.id,
|
||||||
is_active=True,
|
is_active=True,
|
||||||
)
|
)
|
||||||
test_db.add(category)
|
test_db.add(category)
|
||||||
@@ -270,6 +270,7 @@ async def test_get_session_returns_404_not_403_for_other_user(
|
|||||||
session_b = Session(
|
session_b = Session(
|
||||||
tree_id=tree_b.id,
|
tree_id=tree_b.id,
|
||||||
user_id=user_b.id,
|
user_id=user_b.id,
|
||||||
|
account_id=acct_b.id,
|
||||||
tree_snapshot={"id": "root", "type": "start", "children": []},
|
tree_snapshot={"id": "root", "type": "start", "children": []},
|
||||||
path_taken=[],
|
path_taken=[],
|
||||||
decisions=[],
|
decisions=[],
|
||||||
@@ -384,6 +385,7 @@ async def test_share_revoke_returns_404_not_403_for_other_user(
|
|||||||
session_b = Session(
|
session_b = Session(
|
||||||
tree_id=tree_b.id,
|
tree_id=tree_b.id,
|
||||||
user_id=user_b.id,
|
user_id=user_b.id,
|
||||||
|
account_id=acct_b.id,
|
||||||
tree_snapshot={"id": "root", "type": "start", "children": []},
|
tree_snapshot={"id": "root", "type": "start", "children": []},
|
||||||
path_taken=[],
|
path_taken=[],
|
||||||
decisions=[],
|
decisions=[],
|
||||||
@@ -534,6 +536,7 @@ async def test_maintenance_schedule_returns_404_for_other_team(
|
|||||||
# Create a schedule for that tree
|
# Create a schedule for that tree
|
||||||
schedule_b = MaintenanceSchedule(
|
schedule_b = MaintenanceSchedule(
|
||||||
tree_id=tree_b.id,
|
tree_id=tree_b.id,
|
||||||
|
account_id=acct_b.id,
|
||||||
created_by=user_b.id,
|
created_by=user_b.id,
|
||||||
cron_expression="0 2 * * 0",
|
cron_expression="0 2 * * 0",
|
||||||
timezone="UTC",
|
timezone="UTC",
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ from datetime import datetime, timezone, timedelta
|
|||||||
from httpx import AsyncClient
|
from httpx import AsyncClient
|
||||||
from uuid import uuid4
|
from uuid import uuid4
|
||||||
|
|
||||||
|
from app.models.account import Account
|
||||||
from app.models.tree import Tree
|
from app.models.tree import Tree
|
||||||
from app.models.tree_share import TreeShare
|
from app.models.tree_share import TreeShare
|
||||||
from app.models.user import User
|
from app.models.user import User
|
||||||
@@ -287,13 +288,17 @@ class TestTreeSharing:
|
|||||||
@pytest.mark.asyncio
|
@pytest.mark.asyncio
|
||||||
async def test_migration_defaults_visibility_to_team(test_db):
|
async def test_migration_defaults_visibility_to_team(test_db):
|
||||||
"""Test that existing trees default to 'team' visibility after migration."""
|
"""Test that existing trees default to 'team' visibility after migration."""
|
||||||
|
account = Account(name="Migration Default Test", display_code=uuid4().hex[:8])
|
||||||
|
test_db.add(account)
|
||||||
|
await test_db.flush()
|
||||||
|
|
||||||
# Create a tree without specifying visibility
|
# Create a tree without specifying visibility
|
||||||
tree = Tree(
|
tree = Tree(
|
||||||
name="Old Tree",
|
name="Old Tree",
|
||||||
description="Created before migration",
|
description="Created before migration",
|
||||||
tree_structure={"id": "root", "type": "decision", "question": "Test?", "children": []},
|
tree_structure={"id": "root", "type": "decision", "question": "Test?", "children": []},
|
||||||
author_id=None,
|
author_id=None,
|
||||||
account_id=None
|
account_id=account.id
|
||||||
)
|
)
|
||||||
test_db.add(tree)
|
test_db.add(tree)
|
||||||
await test_db.commit()
|
await test_db.commit()
|
||||||
|
|||||||
@@ -359,7 +359,7 @@ async def test_delete_upload_forbidden_for_non_owner(client, auth_headers, test_
|
|||||||
f"/api/v1/uploads/{upload.id}", headers=other_headers
|
f"/api/v1/uploads/{upload.id}", headers=other_headers
|
||||||
)
|
)
|
||||||
|
|
||||||
assert response.status_code == 403
|
assert response.status_code == 404
|
||||||
|
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
|
|||||||
@@ -1,111 +0,0 @@
|
|||||||
import { expect, test } from '@playwright/test'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Regression test for the prefill-handoff `currentChatRef` bug.
|
|
||||||
*
|
|
||||||
* Symptom: a chat session created via the dashboard prefill flow
|
|
||||||
* looked fine on the first AI turn, but submitting partial answers
|
|
||||||
* from the task lane silently dropped the AI's follow-up response.
|
|
||||||
* The user saw their answers in the chat, no assistant reply, no
|
|
||||||
* toast.
|
|
||||||
*
|
|
||||||
* Root cause: the prefill effect in `AssistantChatPage` set
|
|
||||||
* `activeChatId` without also updating `currentChatRef.current`, so
|
|
||||||
* the `currentChatRef.current !== sentForChatId` guard in
|
|
||||||
* `handleTaskSubmit` (and `handleSend`) tripped on every subsequent
|
|
||||||
* request and discarded the AI response.
|
|
||||||
*
|
|
||||||
* Strategy: drive the real prefill flow against the real backend, but
|
|
||||||
* intercept the `/chat` endpoint with `page.route` so we get
|
|
||||||
* deterministic question payloads on turn 1 and a deterministic
|
|
||||||
* follow-up on turn 2. The fix is what makes turn 2 visible.
|
|
||||||
*/
|
|
||||||
test.describe('AssistantChatPage — prefill handoff regression', () => {
|
|
||||||
test('AI follow-up renders after submitting partial task lane answers', async ({ page }) => {
|
|
||||||
let chatCallCount = 0
|
|
||||||
|
|
||||||
// Clear any persisted active-chat-id so the page does not auto-resume a
|
|
||||||
// stale session left behind by a sibling spec.
|
|
||||||
await page.addInitScript(() => {
|
|
||||||
try {
|
|
||||||
sessionStorage.removeItem('rf-active-chat-id')
|
|
||||||
sessionStorage.removeItem('rf-tasklane-meta')
|
|
||||||
} catch { /* ignore */ }
|
|
||||||
})
|
|
||||||
|
|
||||||
// Intercept only the chat endpoint. Session creation, listSessions,
|
|
||||||
// facts, suggested-fixes, etc. all hit the real backend so the page
|
|
||||||
// renders normally — only the LLM call is deterministic. The pattern
|
|
||||||
// matches `/ai-sessions/<uuid>/chat` and nothing nested beneath it.
|
|
||||||
await page.route(/\/api\/v1\/ai-sessions\/[^/]+\/chat$/, async (route) => {
|
|
||||||
if (route.request().method() !== 'POST') {
|
|
||||||
await route.fallback()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
chatCallCount += 1
|
|
||||||
if (chatCallCount === 1) {
|
|
||||||
await route.fulfill({
|
|
||||||
status: 200,
|
|
||||||
contentType: 'application/json',
|
|
||||||
body: JSON.stringify({
|
|
||||||
content: 'Initial diagnostic plan. Please answer the questions in the task lane.',
|
|
||||||
suggested_flows: [],
|
|
||||||
fork: null,
|
|
||||||
actions: [],
|
|
||||||
questions: [
|
|
||||||
{ text: 'Has the user recently changed their password?' },
|
|
||||||
{ text: 'Is the lockout happening at a consistent time of day?' },
|
|
||||||
],
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
await route.fulfill({
|
|
||||||
status: 200,
|
|
||||||
contentType: 'application/json',
|
|
||||||
body: JSON.stringify({
|
|
||||||
content: 'Got it — based on your answer, here is what to check next.',
|
|
||||||
suggested_flows: [],
|
|
||||||
fork: null,
|
|
||||||
actions: [],
|
|
||||||
questions: [],
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
// Drive the prefill flow exactly the way the dashboard does. The textarea
|
|
||||||
// is keyed by its placeholder copy on QuickStartPage.
|
|
||||||
await page.goto('/')
|
|
||||||
const prefillBox = page.getByPlaceholder(/Describe the issue/i)
|
|
||||||
await expect(prefillBox).toBeVisible({ timeout: 10_000 })
|
|
||||||
await prefillBox.fill('User locked out of AD weekly')
|
|
||||||
await prefillBox.press('Enter')
|
|
||||||
|
|
||||||
// After the prefill submits we land on /pilot and the first stubbed AI
|
|
||||||
// turn surfaces the task-lane question text.
|
|
||||||
await expect(page).toHaveURL(/\/pilot/)
|
|
||||||
await expect(
|
|
||||||
page.getByText('Has the user recently changed their password?'),
|
|
||||||
).toBeVisible({ timeout: 15_000 })
|
|
||||||
|
|
||||||
// Answer the first question. UI flow: click "Answer" to open the
|
|
||||||
// textarea, type, click the inline "Answer" button to mark done.
|
|
||||||
await page.getByRole('button', { name: /^Answer$/ }).first().click()
|
|
||||||
await page.getByPlaceholder('Type your answer...').fill('No, password is months old')
|
|
||||||
await page.getByRole('button', { name: /^Answer$/ }).first().click()
|
|
||||||
|
|
||||||
// Submit the partial response. Pre-fix: the response was silently dropped
|
|
||||||
// here because `currentChatRef.current` still held the mount-time value.
|
|
||||||
await page.getByRole('button', { name: /Send 1 of 2 Responses/ }).click()
|
|
||||||
|
|
||||||
// Bug repro: the assistant message must render. Pre-fix this assertion
|
|
||||||
// fails because `handleTaskSubmit` early-returns at the
|
|
||||||
// `currentChatRef.current !== sentForChatId` guard.
|
|
||||||
await expect(
|
|
||||||
page.getByText('Got it — based on your answer, here is what to check next.'),
|
|
||||||
).toBeVisible({ timeout: 15_000 })
|
|
||||||
|
|
||||||
// Both chat calls must have actually happened.
|
|
||||||
expect(chatCallCount).toBe(2)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
@@ -255,12 +255,6 @@ export default function AssistantChatPage() {
|
|||||||
}
|
}
|
||||||
setChats(prev => [chatItem, ...prev])
|
setChats(prev => [chatItem, ...prev])
|
||||||
setActiveChatId(session.session_id)
|
setActiveChatId(session.session_id)
|
||||||
// Keep the in-flight guard ref in sync. Without this, currentChatRef
|
|
||||||
// stays at its mount-time value (often a stale id from sessionStorage
|
|
||||||
// or null), so subsequent handleSend / handleTaskSubmit calls bail at
|
|
||||||
// their `currentChatRef.current !== sentForChatId` check and the AI
|
|
||||||
// response is silently dropped.
|
|
||||||
currentChatRef.current = session.session_id
|
|
||||||
setMessages([{ role: 'user', content: prefill }])
|
setMessages([{ role: 'user', content: prefill }])
|
||||||
setLoading(true)
|
setLoading(true)
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user