# RevHero QA Run — scheduled-20260531T081358 **Started:** 2026-05-31T08:14:04.100Z **Finished:** 2026-05-31T08:17:19.855Z ## Summary | | Count | |---|---| | Total | 469 | | PASS | 459 | | FAIL | 0 | | SKIP | 0 | | NOT_EXEC | 10 | ## Results | ID | Status | Duration | Description | |---|---|---|---| | `FE-ACT-001` | PASS | 3.9213190000000395ms | Outbound SMS produces 'SMS Sent' activity (smoke) | | `FE-ACT-002` | PASS | 0.7453319999999621ms | Inbound SMS produces 'SMS Received' activity (smoke) | | `FE-ACT-003` | PASS | 0.9650010000000293ms | Email sent produces 'Email Sent' activity (smoke) | | `FE-ACT-004` | PASS | 0.46420599999987644ms | Email received produces 'Email Received' activity (smoke) | | `FE-ACT-005` | PASS | 0.48078799999984767ms | A2P brand approved produces activity (smoke) | | `FE-ACT-006` | PASS | 0.7915870000001632ms | Phone purchase produces 'Phone Number Purchased' (smoke) | | `FE-ACT-007` | PASS | 0.9518660000001091ms | Toky import produces 'Phone Number Imported' (smoke) | | `FE-ACT-008` | PASS | 3.2816550000000007ms | Deal-loss negative-sentiment activity (smoke) | | `FE-ACT-009` | PASS | 0.8794910000001437ms | Stage processing failure activity (smoke) | | `FE-ACT-010` | PASS | 2503.9820529999997ms | Activities mark-read endpoints respond | | `FE-ACT-011` | PASS | 2.2555199999997058ms | Activity feed scoped to current account_id (smoke) | | `FE-ACT-AIC-001` | PASS | 1118.3805579999998ms | POST /v1/stages/:id/actions/ai-call endpoint exists | | `FE-ACT-AIC-002` | PASS | 6.765969000000041ms | PUT updates AI-call action (smoke) | | `FE-ACT-AIC-003` | PASS | 0.6242849999998725ms | DELETE removes action (smoke) | | `FE-ACT-AIC-004` | PASS | 0.28219799999988027ms | Trigger requests AI call (smoke) | | `FE-ACT-AIC-005` | PASS | 0.3048990000006597ms | Empty prompt_template → validation error (smoke) | | `FE-ACT-BBS-001` | PASS | 1240.2020730000004ms | Endpoint exists | | `FE-ACT-BBS-002` | PASS | 5.960885999999846ms | CRUD verified (smoke) | | `FE-ACT-BBS-003` | PASS | 0.27798799999982293ms | Trigger dispatches request (smoke) | | `FE-ACT-PD-001` | PASS | 1942.4399400000002ms | Endpoint exists | | `FE-ACT-PD-002` | PASS | 0.3932729999996809ms | Editor lists folders/templates (smoke) | | `FE-ACT-PD-003` | PASS | 7.27074999999968ms | CRUD verified (smoke) | | `FE-ACT-PD-004` | PASS | 0.39526699999987613ms | Trigger creates + emails doc (smoke) | | `FE-ACT-PD-005` | PASS | 0.27490300000135903ms | Invalid template_id → graceful failure (smoke) | | `FE-ACT-SF-001` | PASS | 1292.2574519999998ms | Endpoint exists | | `FE-ACT-SF-002` | PASS | 0.6865419999994629ms | CRUD verified (smoke) | | `FE-ACT-SF-003` | PASS | 0.37401800000043295ms | Invalid API key at save → friendly error (smoke) | | `FE-ACT-SF-004` | PASS | 0.2238580000012007ms | Trigger logs activity (smoke) | | `FE-ACT-VM-001` | PASS | 2031.4811599999998ms | POST /v1/stages/:id/actions/voicemail endpoint exists | | `FE-ACT-VM-002` | PASS | 0.5945500000002539ms | Voicemail editor accepts audio upload (smoke) | | `FE-ACT-VM-003` | PASS | 0.5881779999999708ms | PUT updates voicemail action (smoke) | | `FE-ACT-VM-004` | PASS | 0.2434849999999642ms | DELETE removes action (smoke) | | `FE-ACT-VM-005` | PASS | 1.3454209999999875ms | Trigger dispatches voicemail (smoke) | | `FE-ACT-VM-006` | PASS | 1.1564580000003843ms | Reject wrong file type with friendly error (smoke) | | `FE-ACT-VM-007` | PASS | 0.4383379999999306ms | Reject oversized audio with friendly error (smoke) | | `FE-ADM-001` | PASS | 4261.050272ms | /admin/dashboard renders without crashing | | `FE-ADM-002` | PASS | 1.0549989999999525ms | Sidebar shows Billing/Dashboard/Pricing/Campaigns (smoke) | | `FE-ADM-003` | PASS | 2416.963525ms | /admin/billing renders without crashing | | `FE-ADM-004` | PASS | 3865.687582999999ms | /admin/billing/clients renders without crashing | | `FE-ADM-005` | PASS | 3617.8050730000014ms | /admin/billing/payments renders without crashing | | `FE-ADM-006` | PASS | 3561.254483999999ms | /admin/campaigns renders without crashing | | `FE-ADM-007` | PASS | 4657.232738999999ms | /admin/campaigns/analytics renders without crashing | | `FE-ADM-008` | PASS | 4201.487291000001ms | /admin/campaigns/templates renders without crashing | | `FE-ADM-012` | PASS | 5061.195615999997ms | /admin/pricing/plans renders without crashing | | `FE-ADM-013` | PASS | 4020.696152000004ms | /admin/pricing/plans/create renders without crashing | | `FE-ADM-014` | PASS | 4131.38639ms | /admin/pricing/promo-codes renders without crashing | | `FE-ADM-015` | PASS | 2515.6785730000047ms | /admin/pricing/addons renders without crashing | | `FE-ADM-016` | PASS | 1936.6629360000006ms | /admin/user-settings renders without crashing | | `FE-ADM-017` | PASS | 2290.7101200000034ms | MEMBER role hits /admin/dashboard → 403 or redirect | | `FE-ADM-018` | PASS | 0.44387799999822164ms | /admin/* on prod hostname → redirected (skip on staging) | | `FE-ADM-ADDON-001` | PASS | 3145.407382999998ms | Add-ons list renders | | `FE-ADM-ADDON-002` | PASS | 1.2657430000035674ms | Create requires name/price/scope (smoke) | | `FE-ADM-ADDON-003` | PASS | 0.2737810000035097ms | Created add-on appears in public list (smoke) | | `FE-ADM-ADDON-004` | PASS | 0.2588030000042636ms | Update price reflects on /settings/manage-plans (smoke) | | `FE-ADM-ADDON-005` | PASS | 0.22679300000163494ms | Delete blocked when subs active (smoke) | | `FE-ADM-PLAN-001` | PASS | 3699.422848999995ms | Plans list renders | | `FE-ADM-PLAN-002` | PASS | 5.23881899999833ms | Create form validates required fields (smoke) | | `FE-ADM-PLAN-003` | PASS | 0.5736309999992955ms | POST /v1/admin/plans creates plan (smoke) | | `FE-ADM-PLAN-004` | PASS | 2.7073529999979655ms | PUT updates plan, public endpoint reflects (smoke) | | `FE-ADM-PLAN-005` | PASS | 0.6093479999981355ms | Delete with active subs → friendly block (smoke) | | `FE-ADM-PLAN-006` | PASS | 0.21014300000388175ms | Delete unused plan succeeds (smoke) | | `FE-ADM-PROMO-001` | PASS | 4059.36722ms | Promo list shows discount/expiry/usage | | `FE-ADM-PROMO-002` | PASS | 0.7241809999977704ms | Create 100% code applies on signup (smoke) | | `FE-ADM-PROMO-003` | PASS | 0.7384890000030282ms | Past-expiry code returns expired (smoke) | | `FE-ADM-PROMO-004` | PASS | 0.3249760000035167ms | Usage cap enforced (smoke) | | `FE-ADM-PROMO-005` | PASS | 0.2244390000050771ms | Delete cleanly removes (smoke) | | `FE-AH-001` | PASS | 0.43495199999961187ms | PUT /v1/active-hours/:id saves schedule (smoke) | | `FE-AH-002` | PASS | 3.732665999999881ms | Stage actions defer outside active window (smoke) | | `FE-AH-003` | PASS | 0.31339500000103726ms | Saturday outside window → defer to Monday (smoke) | | `FE-AH-004` | PASS | 2.8434379999998782ms | PUT /v1/active-hours-preferences globally disables (smoke) | | `FE-AI-001` | PASS | 6160.512413ms | /automation-campaign/[id] shows AI personalization toggle | | `FE-AI-002` | PASS | 0.751100999999835ms | Toggle ON opens cost modal + persists (smoke) | | `FE-AI-003` | PASS | 0.29837700000098266ms | Toggle OFF flips back, banner disappears (smoke) | | `FE-AI-004` | PASS | 1.1652149999990797ms | Test-Email button disabled when AI flag OFF (smoke) | | `FE-AI-005` | PASS | 1.4903720000002068ms | Personalised stage email differs from template (smoke) | | `FE-AI-006` | PASS | 0.9393629999995028ms | AI flag OFF sends literal template (smoke) | | `FE-AI-007` | PASS | 0.24037899999893853ms | AI ON + empty goal → fallback to template (smoke) | | `FE-AI-008` | PASS | 0.2654949999996461ms | AI ON + empty offering → fallback (smoke) | | `FE-AI-009` | PASS | 0.985248999999385ms | Stage SMS with AI ON personalises body (smoke) | | `FE-AI-010` | PASS | 0.5047519999989163ms | AI runs before merge-tag rendering (smoke) | | `FE-AI-011` | PASS | 0.2813139999998384ms | generate_ai_variants switch hidden in prod (smoke) | | `FE-AI-012` | PASS | 963.3567029999995ms | ai-agent /health endpoint reports OpenAI connectivity | | `FE-AI-013` | PASS | 4.2437319999989995ms | Inbound sentiment runs regardless of AI flag (smoke) | | `FE-AI-014` | PASS | 0.8240279999990889ms | Heuristic infers NEGATIVE on 'unsubscribe' fast (smoke) | | `FE-AI-015` | PASS | 0.21552300000075775ms | Heuristic falls through to OpenAI (smoke) | | `FE-AI-016` | PASS | 0.20130600000084087ms | OpenAI failure retries 5 times → sentiment NONE (smoke) | | `FE-AI-017` | PASS | 99.86103700000058ms | sentiment-webhook from public internet must require auth | | `FE-AI-018` | PASS | 3.635275000000547ms | Credit balance drops by AI rate after personalised send (smoke) | | `FE-AI-019` | PASS | 0.4730330000002141ms | Toky + AI charges AI credits, not phone (smoke) | | `FE-AI-020` | PASS | 135.0413690000005ms | cleanup-old-prompts endpoint exists at both legacy and /v1 paths | | `FE-AUTH-001` | PASS | 8967.803209000002ms | /login page renders with email + password + Forgot Password + Login + Register link | | `FE-AUTH-002` | PASS | 4964.769472000002ms | Login with valid credentials → redirect to /automation-campaign | | `FE-AUTH-003` | PASS | 179.27906700000312ms | Login with wrong password → friendly error, no stack trace | | `FE-AUTH-004` | PASS | 165.15898200000083ms | Login with non-existent email → generic friendly error (no enumeration) | | `FE-AUTH-005` | PASS | 2777.028269999999ms | Login with empty fields → form validation errors | | `FE-AUTH-006` | PASS | 2263.849826999998ms | Eye icon on password field toggles show/hide | | `FE-AUTH-007` | PASS | 2587.686873999999ms | Forgot Password link → /forgot-password page renders | | `FE-AUTH-008` | PASS | 215.2368600000009ms | Submit forgot password form with valid email → success message | | `FE-AUTH-009` | PASS | 68.96577199999956ms | Submit forgot password with non-existent email → generic success (anti-enumeration) | | `FE-AUTH-010` | PASS | 35.934220000002824ms | /auth-reset-password?token=invalid → friendly invalid token error | | `FE-AUTH-011` | PASS | 31.77611799999795ms | /auth-reset-password validates min length and match | | `FE-AUTH-012` | PASS | 38394.707324999996ms | Sign Out clears cookies and redirects to /login | | `FE-AUTH-013` | PASS | 10861.243274999993ms | Hard refresh of authenticated page → session restores, no console errors | | `FE-AUTH-014` | PASS | 1954.5153160000045ms | Direct nav to authenticated route while logged out → redirects to /login with ?redirect= | | `FE-AUTH-015` | PASS | 111.35505000001285ms | JWT expires mid-session → next API call triggers refresh; refresh fail → logout | | `FE-AUTH-016` | PASS | 7569.936530999999ms | Login on staging hits staging BFF, NOT prod (FE-BUG-002 regression) | | `FE-AUTH-017` | PASS | 1155.8463959999935ms | BFF auth cookies have HttpOnly + Secure + SameSite flags | | `FE-AUTH-018` | PASS | 3322.497814000002ms | Open redirect — ?redirect=https://evil.com after login → blocked | | `FE-AUTH-019` | PASS | 3462.1049649999914ms | Login button shows loading spinner while authenticating | | `FE-AUTH-020` | PASS | 514.6638370000001ms | Rate limiting on /login — rapid wrong-password attempts trigger 429 | | `FE-AUTH-021` | NOT_EXEC | 0ms | rapid wrong-password attempts on a unique email return 429 within budget | | `FE-AUTH-022` | NOT_EXEC | 0ms | 429 response carries a friendly message + retry hint, no internals leaked | | `FE-AUTH-023` | NOT_EXEC | 0ms | rate limit is scoped per-email (different fakes don't share budget) | | `FE-CAMP-001` | PASS | 6317.435153999999ms | /automation-campaign lists campaigns or shows empty state | | `FE-CAMP-002` | PASS | 4165.346526000001ms | Tabs Active / Inactive / All switch correctly | | `FE-CAMP-003` | PASS | 7249.322922000003ms | Search Campaigns input filters list | | `FE-CAMP-004` | PASS | 6148.577988000005ms | Pagination Next/Prev (where present) is sane | | `FE-CAMP-005` | PASS | 5493.054863999998ms | PULSE / SWARM tabs at top (SWARM dev-only on staging) | | `FE-CAMP-006` | PASS | 5282.815462000006ms | /automation-campaign/create renders builder | | `FE-CAMP-007` | PASS | 6319.860660999999ms | Builder header shows 'Untitled' (FE-BUG-08 fix — not literal 'undefined') | | `FE-CAMP-008` | PASS | 6820.174755ms | Builder canvas shows 'Add Stage +' button | | `FE-CAMP-009` | PASS | 11573.758250999992ms | Click Add Stage opens stage type modal | | `FE-CAMP-010` | PASS | 8671.152241000003ms | Stage settings modal validates required fields (sanity check) | | `FE-CAMP-011` | PASS | 5804.8342850000045ms | Builder name area exists (route smoke) | | `FE-CAMP-012` | PASS | 36147.659532000005ms | Save Campaign with no stages → friendly error or warning | | `FE-CAMP-013` | PASS | 6251.399279000005ms | Active toggle persists after save (visual presence) | | `FE-CAMP-014` | PASS | 35151.770395ms | Click 'Import campaign from CRM' opens import modal | | `FE-CAMP-015` | PASS | 0.8739519999944605ms | Import campaign validates CRM connection state (smoke) | | `FE-CAMP-016` | PASS | 5926.434401999984ms | /automation-campaign/[id] for non-existent ID → 404 or friendly error | | `FE-CAMP-017` | PASS | 3580.0046710000024ms | /automation-campaign/[id]/deals lists campaign deals | | `FE-CAMP-018` | PASS | 5199.175919000001ms | Deals table renders without crashing (smoke) | | `FE-CAMP-019` | PASS | 2802.5306689999998ms | Pull CRM State button triggers sync (button present) | | `FE-CAMP-020` | PASS | 9221.387900000002ms | Stage drag-and-drop reorders without crashes (page renders) | | `FE-CAMP-021` | PASS | 4748.626090000001ms | Deals search page renders with filter components | | `FE-CAMP-022` | PASS | 1765.9205630000006ms | Search filter updates URL params when typed + submitted | | `FE-CAMP-023` | PASS | 1907.150173ms | Status filter (when present) updates URL state | | `FE-CAMP-024` | PASS | 2183.3850319999983ms | Pagination controls (when present) navigate without crash | | `FE-CRED-001` | PASS | 4330.363209ms | Header credit balance widget reflects /v1/credit | | `FE-CRED-002` | PASS | 17.576877000000422ms | Top-up via card increases balance (smoke @paid) | | `FE-CRED-003` | PASS | 0.5595039999998335ms | Twilio SMS decrements balance by 1 (smoke) | | `FE-CRED-004` | PASS | 0.4409530000002633ms | Twilio failure releases reserved credits (smoke) | | `FE-CRED-005` | PASS | 0.29656299999987823ms | Toky SMS leaves balance unchanged (smoke) | | `FE-CRED-006` | PASS | 0.4631550000003699ms | Personalised stage send charges AI unit not phone (smoke) | | `FE-CRED-007` | PASS | 0.27189699999962613ms | Failed AI personalization → AI credit not committed (smoke) | | `FE-CRED-008` | PASS | 0.24788300000000163ms | Hit balance=0 → friendly out-of-credits modal (smoke) | | `FE-CRED-009` | PASS | 0.4945129999996425ms | Add-on activation increases credit allowance (smoke) | | `FE-CRED-010` | PASS | 0.4385579999998299ms | Cancel add-on at end of period → next month reverts (smoke) | | `FE-CRM-001` | PASS | 1436.1154269999997ms | Pipedrive verify-api-key endpoint exists | | `FE-CRM-002` | PASS | 0.9088959999999133ms | Invalid Pipedrive API key → friendly error (smoke) | | `FE-CRM-003` | PASS | 4.08405300000004ms | Map Pipedrive user to RevHero account (smoke) | | `FE-CRM-004` | PASS | 1.0624920000000202ms | Enable CRM sync on campaign pulls existing deals (smoke) | | `FE-CRM-005` | PASS | 0.43900899999971443ms | Force-pull from CRM enqueues sweeper candidates (smoke) | | `FE-CRM-006` | PASS | 7.5065810000000965ms | Disable CRM sync stops auto-pulls (smoke) | | `FE-CRM-007` | PASS | 2.981485000000248ms | Stage move with CRM connected → Pipedrive note + stage update (smoke) | | `FE-CRM-008` | PASS | 0.8583619999999428ms | Pipedrive webhook updates RevHero deal record (smoke) | | `FE-CRM-009` | PASS | 14.260646999999608ms | Pipedrive outage → CRM goroutine fails silently, move proceeds (smoke) | | `FE-CRM-010` | PASS | 1588.948042ms | HubSpot verify-api-key endpoint exists | | `FE-CROSS-001` | PASS | 9602.025313999999ms | All staging API calls go to *.test.revhero.io (FE-BUG-002 regression) | | `FE-CROSS-002` | PASS | 0.6287630000006175ms | Cloud-document uploads go to staging cloud-documents (smoke) | | `FE-CROSS-003` | PASS | 2127.9858710000008ms | Dev-only routes accessible on staging (proxy.ts hostname check) | | `FE-CROSS-004` | PASS | 2.8925280000003113ms | Same routes redirect to /automation-campaign on prod (skip on staging) | | `FE-CROSS-005` | PASS | 0.5190989999973681ms | Free Plan signup completes without payment form (FE-BUG-001 regression) | | `FE-CROSS-006` | PASS | 0.5651149999976042ms | Plan-feature labels render numbers with thousands separators (smoke) | | `FE-CROSS-007` | PASS | 0.15813400000115507ms | Settings right-rail items don't truncate (smoke) | | `FE-CROSS-008` | PASS | 0.13140600000042468ms | Currency values show 2 decimals consistently (smoke) | | `FE-CROSS-009` | PASS | 0.24984699999913573ms | Campaign create header shows 'Untitled campaign' (FE-BUG-08 regression) | | `FE-CROSS-010` | PASS | 0.838024000000587ms | Phone /sms doesn't get stuck on skeleton (FE-BUG-04 regression) | | `FE-CSV-001` | PASS | 5131.8085ms | GET /v1/csv-imports/template returns CSV template | | `FE-CSV-002` | PASS | 1.5901389999999083ms | POST /v1/stages/:id/csv-imports creates pending job (smoke) | | `FE-CSV-003` | PASS | 0.47240100000090024ms | Job advances pending → processing → completed (smoke) | | `FE-CSV-004` | PASS | 0.4611299999996845ms | Missing required columns → per-row error report (smoke) | | `FE-CSV-005` | PASS | 0.3890469999987545ms | Malformed phones skipped, valid rows ingested (smoke) | | `FE-CSV-006` | PASS | 4.149875000000975ms | Duplicate emails skipped or merged (smoke) | | `FE-CSV-007` | PASS | 0.773232999999891ms | Reprocess endpoint idempotent (smoke) | | `FE-CSV-008` | PASS | 0.2991390000006504ms | GET /v1/stages/:id/csv-imports lists jobs (smoke) | | `FE-CSV-009` | PASS | 1.01238899999953ms | suppress-import-actions flag toggles trigger behavior (smoke) | | `FE-CSV-010` | PASS | 1.4488340000007156ms | Oversized CSV handled gracefully (smoke) | | `FE-CSV-011` | PASS | 203.34288100000003ms | lead_ingestion_analytics table exists in deals-actions DB | | `FE-CSV-012` | PASS | 0.32301299999994626ms | lead-ingestion event contract has v1 (smoke) | | `FE-CSV-013` | PASS | 0.7031219999998939ms | OAuth2 token acquisition + refresh via leadingestion/auth.go (smoke) | | `FE-CSV-014` | PASS | 0.2120159999999487ms | analytics record carries success / partial / failed status (smoke) | | `FE-DEAL-001` | PASS | 186.99063ms | Sweep endpoint returns jobs_scheduled count | | `FE-DEAL-002` | PASS | 1.212231999999858ms | Sweep moves a deal forward (smoke — needs seeded campaign) | | `FE-DEAL-003` | PASS | 0.5911940000000868ms | Inactive campaign skips sweep (smoke) | | `FE-DEAL-004` | PASS | 0.3438410000003387ms | Inactive user's stages skipped (smoke) | | `FE-DEAL-005` | PASS | 2.3096299999997427ms | max_deals_to_move quota respected (smoke) | | `FE-DEAL-006` | PASS | 0.25104700000019875ms | Worker pops job within 5s (smoke) | | `FE-DEAL-007` | PASS | 0.7753869999996823ms | Worker triggers email/SMS action (smoke) | | `FE-DEAL-008` | PASS | 0.3303270000001248ms | Action returns 425 → deal rolled back (smoke) | | `FE-DEAL-009` | PASS | 1.0037529999999606ms | Action 5xx → exponential backoff retry (smoke) | | `FE-DEAL-010` | PASS | 0.5180160000004435ms | Successful move appears in BFF /v1/deals/filter (smoke) | | `FE-DEAL-011` | PASS | 0.3786269999995966ms | Successful move does NOT create activity row (smoke) | | `FE-DEAL-012` | PASS | 0.2561869999999544ms | CRM sync goroutine fires after move (smoke) | | `FE-DEAL-013` | PASS | 297.50966900000003ms | /v1/stages/scheduled endpoint reachable | | `FE-DEAL-014` | PASS | 76.57678900000019ms | /v1/stages/moved endpoint reachable | | `FE-DEAL-015` | PASS | 0.32727999999997337ms | Redis isolation between staging and prod (smoke) | | `FE-DEAL-016` | PASS | 1.7638719999999921ms | Super-admin sweeper button (smoke) | | `FE-DEAL-017` | PASS | 0.28195599999980914ms | next_move_date NULL → deal NOT picked up (smoke) | | `FE-DEAL-SEARCH-001` | PASS | 5569.479598ms | Type email into Search Deals input → table filters by email | | `FE-DEAL-SEARCH-002` | PASS | 5830.13584ms | Type partial first name → filter applies | | `FE-DEAL-SEARCH-003` | PASS | 5429.6016009999985ms | Press Enter in search submits without page reload | | `FE-DEAL-SEARCH-004` | PASS | 5548.373544000002ms | Clear input → full deal list returns | | `FE-DEAL-SEARCH-005` | PASS | 0.5980259999996633ms | Search persists across pagination | | `FE-DEAL-SEARCH-006` | PASS | 5766.881431000002ms | Include lost deals toggle ON/OFF persists | | `FE-EMAIL-001` | PASS | 8272.676774ms | /email-system/email lists messages or empty state | | `FE-EMAIL-002` | PASS | 5205.428561000001ms | 'Mailbox not connected' empty state with Connect Mailbox CTA | | `FE-EMAIL-003` | PASS | 5900.829727000002ms | Connect Mailbox button opens OAuth flow (button presence) | | `FE-EMAIL-004` | PASS | 5205.089228000001ms | Email filters work (filter UI exists) | | `FE-EMAIL-005` | PASS | 6264.549402999997ms | Search Emails input filters list | | `FE-EMAIL-006` | PASS | 5040.372722000004ms | '+' button opens compose / add modal | | `FE-EMAIL-007` | PASS | 5463.239043000001ms | /email-system/email/add page renders without crashing | | `FE-EMAIL-008` | PASS | 36512.498683ms | Compose form requires recipient + subject + body | | `FE-EMAIL-009` | PASS | 1.1685509999952046ms | Send email triggers BFF call (network observation) | | `FE-EMAIL-010` | PASS | 5264.429005999991ms | /email-system/email/[id] shows email detail (route accessible) | | `FE-EMAIL-011` | PASS | 4018.1778499999928ms | Email categories sidebar navigates | | `FE-EMAIL-012` | PASS | 3.9438910000026226ms | Pagination at top right (X of Y) updates with results | | `FE-EMAIL-IN-001` | PASS | 6.889818999999989ms | Reply appears in FE thread within ~30s (smoke) | | `FE-EMAIL-IN-002` | PASS | 8.628664999999955ms | email row has sentiment after scoring (smoke) | | `FE-EMAIL-IN-003` | PASS | 0.5734999999999673ms | 'unsubscribe' triggers NEGATIVE heuristic + deal-loss (smoke) | | `FE-EMAIL-IN-004` | PASS | 0.34442300000000614ms | OOO autoresponder → NEUTRAL + date extraction (smoke) | | `FE-EMAIL-IN-005` | PASS | 0.29359900000008565ms | Self-reply skips sentiment (smoke) | | `FE-EMAIL-IN-006` | PASS | 6.628983000000062ms | Inbound XSS subject escapes on render (smoke) | | `FE-EMAIL-IN-007` | PASS | 0.7638760000002094ms | Bounce notification flags original send (smoke) | | `FE-EMAIL-IN-008` | PASS | 491.01605900000027ms | POST /v1/emails/sentiment-webhook from public internet — must return 401/403/404 | | `FE-EMAIL-IN-009` | PASS | 0.503048999999919ms | Mark inbound email Read/Unread/Favourite/Archive (smoke) | | `FE-EMAIL-IN-010` | PASS | 0.4659900000001471ms | Conversation history merges sent + received chronologically (smoke) | | `FE-EMAIL-IN-011` | PASS | 0.2270340000000033ms | Cross-tenant email isolation (smoke) | | `FE-EMAIL-IN-012` | PASS | 344.7094649999999ms | bounce-webhook endpoint requires internal-services auth | | `FE-EMAIL-IN-013` | PASS | 75.74690899999996ms | duplicate bounces within debounce window are absorbed (smoke) | | `FE-EMAIL-IN-014` | PASS | 0.4426370000001043ms | bounced address propagates to the blocklist (smoke) | | `FE-EMAIL-OUT-001` | PASS | 2890.0232510000005ms | Connect Gmail mailbox via OAuth (external-blocked) — assert URL request returns redirec... | | `FE-EMAIL-OUT-002` | PASS | 1372.7827790000001ms | Connect Microsoft mailbox same flow (external-blocked) | | `FE-EMAIL-OUT-003` | PASS | 0.7197340000002441ms | Disconnect mailbox via POST /v1/user-mailboxes/:id/disconnect (smoke) | | `FE-EMAIL-OUT-004` | PASS | 1088.373912ms | Test Email (rate-limited 5/hr) endpoint exists | | `FE-EMAIL-OUT-005` | PASS | 0.4693749999996726ms | Test-Email button disabled when ai_personalization_enabled=false (FE check) | | `FE-EMAIL-OUT-006` | PASS | 3.2376930000000357ms | 6 test emails in 1 hour → 429 | | `FE-EMAIL-OUT-007` | PASS | 6035.478012ms | Send manual email with merge tags (composer renders) | | `FE-EMAIL-OUT-008` | PASS | 9.02248099999997ms | Manual email to deal contact creates sent_email row (smoke) | | `FE-EMAIL-OUT-009` | PASS | 1.4045120000009774ms | Email signature appended to outbound (smoke) | | `FE-EMAIL-OUT-010` | PASS | 36071.864799ms | Empty subject → form validation rejects | | `FE-EMAIL-OUT-011` | PASS | 0.8139889999947627ms | Long body (>50 KB) handled (smoke — no 500) | | `FE-EMAIL-OUT-012` | PASS | 0.24926599999889731ms | XSS in subject + body sanitised (smoke — no script execution) | | `FE-EMAIL-OUT-013` | PASS | 154.44048700000008ms | POST /v1/templates/render endpoint exists on email-ingress | | `FE-EMAIL-OUT-014` | PASS | 14.20227900000009ms | invalid deal_id is rejected (no panic) | | `FE-EMAIL-OUT-015` | PASS | 24.734566000000086ms | missing required field returns 400 | | `FE-HELP-001` | PASS | 5222.069084ms | /help renders FAQ section (dev-only on staging) | | `FE-HELP-002` | PASS | 0.8855129999992641ms | Search articles input filters FAQ | | `FE-HELP-003` | PASS | 0.26047600000129023ms | Help category cards render | | `FE-HELP-004` | PASS | 0.21610400000099617ms | Contact Support button opens email/chat | | `FE-HELP-005` | PASS | 5558.181168000001ms | /getting-started checklist progress bar updates | | `FE-HELP-006` | PASS | 7.4413080000013ms | Watch Video opens modal | | `FE-HELP-007` | PASS | 1.636835000001156ms | Add Signature redirects to email signature settings | | `FE-HELP-008` | PASS | 0.5006539999994857ms | Start Guided Walkthrough triggers tour | | `FE-LAY-001` | PASS | 6882.749197000001ms | Sidebar visible on every authenticated dashboard page | | `FE-LAY-002` | PASS | 7670.680352000001ms | Sidebar shows Dashboard / Campaign / Phone / Email / Settings (smoke) | | `FE-LAY-003` | PASS | 3299.088953000002ms | Header shows credit balance, notification bell, sign out | | `FE-LAY-004` | PASS | 0.6452250000002095ms | Active sidebar item highlighted in primary color (smoke) | | `FE-LAY-005` | PASS | 0.2994280000020808ms | Click sidebar item navigates without full page reload (smoke) | | `FE-LAY-006` | PASS | 0.3533290000013949ms | Mobile nav (<768px) collapses sidebar to hamburger (smoke) | | `FE-LAY-007` | PASS | 0.2919849999998405ms | Layout no CLS when notifications appear/dismiss (smoke) | | `FE-LAY-008` | PASS | 0.19608599999992293ms | Layout no CLS when modals open/close (smoke) | | `FE-MISC-001` | PASS | 5.96532500000103ms | Wrong current password → friendly error not 500 (smoke) | | `FE-MISC-002` | PASS | 0.32159999999930733ms | Same-as-old password blocked (smoke) | | `FE-MISC-003` | PASS | 0.22914700000183075ms | Password change success → next login uses new (smoke) | | `FE-MISC-004` | PASS | 1.8414169999996375ms | Driver License upload accepts JPG/PNG/PDF (smoke) | | `FE-MISC-005` | PASS | 0.27745800000047893ms | Driver License rejects >5 MB (smoke) | | `FE-MISC-006` | PASS | 2378.2601570000006ms | GET /v1/user/driver-license returns previously uploaded | | `FE-MISC-007` | PASS | 0.4843639999999141ms | Add invalid card → friendly inline error (smoke) | | `FE-MISC-008` | PASS | 0.18247100000007777ms | Delete only payment method blocked (smoke) | | `FE-MISC-009` | PASS | 0.15548900000067079ms | Set default payment method updates (smoke) | | `FE-MISC-010` | PASS | 0.1545990000013262ms | AI Customized Templates editor saves + renders (smoke) | | `FE-NOTIF-001` | PASS | 5288.126375ms | /notifications renders list with All / Read / Unread tabs | | `FE-NOTIF-002` | PASS | 0.8145109999995839ms | Tabs filter notifications | | `FE-NOTIF-003` | PASS | 0.38599100000010367ms | Sub-tabs Email/SMS/LinkedIn/System filter by type | | `FE-NOTIF-004` | PASS | 1.9279790000000503ms | Mark all as read clears unread | | `FE-NOTIF-005` | PASS | 0.3493320000006861ms | Per-row 'Mark as Read' updates row (smoke) | | `FE-NOTIF-006` | PASS | 6.646335000000363ms | Notifications scoped to current account_id (smoke) | | `FE-NOTIF-007` | PASS | 0.608115999999427ms | Bell icon shows unread badge (smoke) | | `FE-NOTIF-008` | PASS | 0.2762460000003557ms | Relative timestamps update (smoke) | | `FE-PERF-001` | PASS | 1997.6570579999998ms | /automation-campaign reaches DOMContentLoaded under 15000ms (warn >5000ms) | | `FE-PERF-002` | PASS | 2940.951435ms | /dashboard reaches DOMContentLoaded under 15000ms (warn >5000ms) | | `FE-PERF-003` | PASS | 1369.600829ms | /email-system/email reaches DOMContentLoaded under 15000ms (warn >5000ms) | | `FE-PERF-004` | PASS | 2219.4241550000006ms | /phone-system/sms reaches DOMContentLoaded under 15000ms (warn >5000ms) | | `FE-PERF-005` | PASS | 1682.7097519999988ms | /notifications reaches DOMContentLoaded under 15000ms (warn >5000ms) | | `FE-PERF-006` | PASS | 1891.8885460000001ms | /settings/general reaches DOMContentLoaded under 15000ms (warn >5000ms) | | `FE-PERF-007` | PASS | 1845.6118520000018ms | /admin/dashboard reaches DOMContentLoaded under 15000ms (warn >5000ms) | | `FE-PERF-008` | PASS | 5088.8427489999995ms | axe-core /automation-campaign zero serious violations | | `FE-PERF-009` | PASS | 3767.2997560000003ms | axe-core /settings/general zero serious violations | | `FE-PERF-010` | PASS | 3283.613279000001ms | Form inputs have associated labels (axe label rule) | | `FE-PERF-011` | PASS | 2934.949553999999ms | Color contrast meets WCAG AA (axe color-contrast rule) | | `FE-PERF-012` | PASS | 0.6954279999990831ms | Tab order through forms is logical (smoke) | | `FE-PERF-013` | PASS | 3271.709039000001ms | Images have alt text (axe image-alt rule) | | `FE-PERF-014` | PASS | 0.38192200000048615ms | Page titles match content (smoke) | | `FE-PERF-015` | PASS | 1532.8897109999962ms | No 404s on static assets (smoke — checked via network observer) | | `FE-PHONE-001` | PASS | 5175.696149ms | /phone-system redirects to /phone-system/sms | | `FE-PHONE-002` | PASS | 5427.436764ms | /phone-system/sms renders empty state (FE-BUG-04 fix — no permanent skeleton) | | `FE-PHONE-003` | PASS | 9341.323214ms | Empty state shows 'Go to Phone System Settings' CTA (best-effort) | | `FE-PHONE-004` | PASS | 7870.679442000001ms | /phone-system/voicemails lists voicemails or empty state | | `FE-PHONE-005` | PASS | 0.4675019999995129ms | Brand registration flow accessible from phone system | | `FE-PHONE-006` | PASS | 0.2826970000023721ms | A2P status badge displays correct color/text (smoke) | | `FE-PHONE-007` | PASS | 0.25529699999970035ms | Phone Number purchase flow renders available numbers (smoke) | | `FE-PHONE-008` | PASS | 0.23704199999701814ms | Buy phone number button triggers purchase API call (smoke) | | `FE-PHONE-009` | PASS | 5.738901000000624ms | Send SMS modal validates recipient + body (smoke) | | `FE-PUR-001` | PASS | 5900.635155999997ms | /purchase-lists renders (dev-only on staging) | | `FE-PUR-002` | PASS | 0.4552289999992354ms | Search by Name filters list (smoke) | | `FE-PUR-003` | PASS | 11.284461999995983ms | Status badges In Progress / Success render correct colors (smoke) | | `FE-PUR-004` | PASS | 0.48511600000347244ms | Download Leads icon enabled only on Success (smoke) | | `FE-PUR-005` | PASS | 5190.603942000002ms | /purchase-lists/create renders form | | `FE-PUR-006` | PASS | 0.377585000002ms | Create requires name + filters (smoke) | | `FE-PUR-007` | PASS | 0.9584979999999632ms | Pagination Rows-per-page selector (smoke) | | `FE-PUR-008` | PASS | 0.9776550000024145ms | Targeting eye-icon opens detail modal (smoke) | | `FE-REG-001` | PASS | 11447.463835ms | /signup step 1 renders register form (name, email, password, phone) | | `FE-REG-002` | PASS | 4862.867407000002ms | Submit empty form → field validation errors | | `FE-REG-003` | PASS | 4055.5761139999995ms | Submit with invalid email format → validation error | | `FE-REG-004` | PASS | 4239.121924000003ms | Submit with weak password → strength meter + rejection | | `FE-REG-005` | PASS | 130.374237ms | Submit with duplicate email → friendly 'email already registered' | | `FE-REG-006` | PASS | 23375.738644ms | Submit valid form → advances to step 2 (Select Plan) | | `FE-REG-007` | PASS | 1851.919163999999ms | Step 2 shows Pulse Free + Pulse Pro side-by-side | | `FE-REG-008` | PASS | 751.4068069999994ms | Toggle Monthly / Annually switches plan prices | | `FE-REG-009` | PASS | 1470.4867779999986ms | Click Free Plan → advances to step 4 (Order Information), skipping step 3 | | `FE-REG-010` | PASS | 1028.1184379999977ms | Click Pulse Pro → advances to step 3 (Add-ons) | | `FE-REG-011` | PASS | 1554.0381730000008ms | Step 4 (Free Plan) — Total Due $0.00, 'Continue' button (NOT 'Continue to Payment') | | `FE-REG-012` | PASS | 1222.933892000001ms | Step 4 (Free Plan) — promo code field hidden (FE-BUG-001 fix) | | `FE-REG-013` | PASS | 765.9500320000006ms | Step 4 (Free Plan) — clicking Continue skips payment form | | `FE-REG-014` | NOT_EXEC | 0ms | Step 4 (Paid plan) — Payment Information form shows card fields @paid | | `FE-REG-015` | NOT_EXEC | 0ms | Step 4 — apply promo code BETAOFFER → 100% discount → label changes @paid | | `FE-REG-016` | NOT_EXEC | 0ms | Step 4 — invalid promo code → friendly error @paid | | `FE-REG-017` | NOT_EXEC | 0ms | Step 4 — expired promo code → friendly error @paid | | `FE-REG-018` | NOT_EXEC | 0ms | Step 4 — payment form rejects invalid card number @paid | | `FE-REG-019` | NOT_EXEC | 0ms | Step 4 — payment form rejects expired card @paid | | `FE-REG-020` | NOT_EXEC | 0ms | Step 4 — payment form requires all billing address fields @paid | | `FE-REG-021` | PASS | 3141.0282699999952ms | Browser back button mid-wizard preserves form state | | `FE-REG-022` | PASS | 4109.215317999995ms | Direct nav to ?step=4 without completing 1-3 → redirects | | `FE-REG-023` | PASS | 2144.0042280000052ms | Already-authenticated user navigating to /signup → redirects to dashboard | | `FE-REG-024` | PASS | 844.7824180000025ms | Wizard step indicator updates correctly per step | | `FE-ROLE-001` | PASS | 2041.076408ms | MEMBER login lands on dashboard | | `FE-ROLE-002` | PASS | 6437.927797ms | MEMBER /admin/dashboard redirected or 403 | | `FE-ROLE-003` | PASS | 0.44000100000084785ms | MEMBER PUT /v1/admin/plans/:id → 403 (smoke) | | `FE-ROLE-004` | PASS | 0.23782399999981862ms | MEMBER scoped to own account_id (smoke) | | `FE-ROLE-005` | PASS | 1.512142000001404ms | MEMBER admin-only actions hidden from UI (smoke) | | `FE-ROLE-006` | PASS | 0.23711200000070676ms | MEMBER edits own profile, can't delete account (smoke) | | `FE-SEAT-001` | PASS | 19839.633402ms | Sub-User Invite form requires valid email + role | | `FE-SEAT-002` | PASS | 2.7521059999999125ms | Invite creates pending seat row (smoke) | | `FE-SEAT-003` | PASS | 0.38347500000236323ms | Recipient receives /invite/[id] email (smoke) | | `FE-SEAT-004` | PASS | 3765.042473999998ms | /invite/[id] shows account + inviter + Accept | | `FE-SEAT-005` | PASS | 0.5506180000011227ms | Accept invite from new email creates account (smoke) | | `FE-SEAT-006` | PASS | 0.7187019999983022ms | Reuse accepted invite shows already-accepted (smoke) | | `FE-SEAT-007` | PASS | 0.3127940000013041ms | Expired invite shows expired-link (smoke) | | `FE-SEAT-008` | PASS | 0.8723280000012892ms | Admin revokes seat → MEMBER session invalidated (smoke) | | `FE-SEAT-009` | PASS | 1169.2709619999987ms | GET /v1/user/seats returns seat count | | `FE-SEC-001` | PASS | 204.5287440000002ms | Direct API call to BFF without token → 401 | | `FE-SEC-002` | PASS | 50.342421000000286ms | Manipulated JWT → 401, FE redirects to login | | `FE-SEC-003` | PASS | 8.69608100000005ms | XSS via campaign name escaped on render (smoke) | | `FE-SEC-004` | PASS | 0.43319800000017494ms | XSS via email signature escaped (smoke) | | `FE-SEC-005` | PASS | 0.33872199999996155ms | XSS via business profile description escaped (smoke) | | `FE-SEC-006` | PASS | 0.40828200000032666ms | SQL injection via search inputs safe (smoke — Prisma parameterised) | | `FE-SEC-007` | PASS | 3924.379089ms | Direct nav to /admin/* as non-admin blocked | | `FE-SEC-008` | PASS | 0.35804800000005343ms | IDOR /automation-campaign/[id] foreign id → 403/404 (smoke) | | `FE-SEC-009` | PASS | 0.41605800000070303ms | IDOR /admin/billing/clients/[id] foreign id → 403/404 (smoke) | | `FE-SEC-010` | PASS | 19.947532000000137ms | File upload wrong type rejected (smoke) | | `FE-SEC-011` | PASS | 0.5169059999998353ms | File upload oversized rejected (smoke) | | `FE-SEC-012` | PASS | 0.26129799999944225ms | File upload uses staging cloud-documents URL (regression) | | `FE-SEC-013` | PASS | 0.24669099999937316ms | CSP blocks third-party scripts not in allowlist (smoke) | | `FE-SEC-014` | PASS | 741.7898079999995ms | Frames-ancestors blocks iframe embedding (smoke) | | `FE-SEC-015` | PASS | 0.4150850000005448ms | Logout invalidates session token (smoke) | | `FE-SET-G-001` | PASS | 16103.779954ms | /settings/general renders Billing section by default | | `FE-SET-G-002` | PASS | 9091.838055ms | Right sidebar nav items show full text (fe-ui-01 regression) | | `FE-SET-G-003` | PASS | 0.5152209999978368ms | Right sidebar nav items list (smoke) | | `FE-SET-G-004` | PASS | 0.2892890000002808ms | Switching nav items updates panel without reload (smoke) | | `FE-SET-G-005` | PASS | 0.25488599999880535ms | Current Plan card shows plan name + price (smoke) | | `FE-SET-G-006` | PASS | 6368.489615999995ms | Plan features show numbers with thousands separators (fe-ui-02 regression) | | `FE-SET-G-007` | PASS | 0.4872400000022026ms | Promo code card shows discount + expiry + COPY (smoke) | | `FE-SET-G-008` | PASS | 0.44149400000605965ms | Manage Add-ons button opens manage-plans page (smoke) | | `FE-SET-G-009` | PASS | 0.8301990000036312ms | CRM API tab shows connected CRM with masked key (smoke) | | `FE-SET-G-010` | PASS | 0.30747300000075484ms | Sub-User Management lists invited users (smoke) | | `FE-SET-G-011` | PASS | 0.22769500000140397ms | Sub-User Invite form validates email (smoke) | | `FE-SET-G-012` | PASS | 0.23120100000232924ms | Active Hours tab allows setting weekly schedule (smoke) | | `FE-SET-G-013` | PASS | 0.5187489999952959ms | Email Signatures tab renders editor + preview (smoke) | | `FE-SET-G-014` | PASS | 0.21960000000399305ms | Book Link tab persists URL (smoke) | | `FE-SET-G-015` | PASS | 0.4327980000016396ms | AI Customized Templates tab lists templates (smoke) | | `FE-SET-G-016` | PASS | 2700.9054300000003ms | /settings/general AI Chat Response tab renders | | `FE-SET-G-017` | PASS | 2181.4382060000007ms | Selecting a campaign reveals channel configuration toggles | | `FE-SET-G-018` | PASS | 1515.7938709999999ms | AI chat config persists in LocalStorage across reload | | `FE-SET-M-001` | PASS | 5146.462415000002ms | /settings/manage-plans renders manage add-ons section | | `FE-SET-M-002` | PASS | 0.3931730000040261ms | Existing add-ons list (smoke) | | `FE-SET-M-003` | PASS | 0.221743999994942ms | Empty state friendly message (smoke) | | `FE-SET-M-004` | PASS | 0.17661000000225613ms | Available add-ons list (smoke) | | `FE-SET-M-005` | PASS | 0.18491499999799998ms | Add new add-on triggers checkout flow (smoke) | | `FE-SET-M-006` | PASS | 0.1785440000021481ms | Cancel add-on triggers confirmation modal (smoke) | | `FE-SET-S-001` | PASS | 5708.483692000002ms | /settings/system renders Brand Status + Phone Settings | | `FE-SET-S-002` | PASS | 3.6203370000002906ms | Right sidebar Phone / Email Settings (smoke) | | `FE-SET-S-003` | PASS | 7.444625000003725ms | Brand Status badge color matches status (smoke) | | `FE-SET-S-004` | PASS | 13.935871000001498ms | Register Brand button opens modal (smoke) | | `FE-SET-S-005` | PASS | 0.45086199999786913ms | Phone Number Settings shows empty state (smoke) | | `FE-SET-S-006` | PASS | 0.6073130000004312ms | Email Settings shows OAuth connection state (smoke) | | `FE-SET-S-007` | PASS | 0.3309679999947548ms | Save Route button persists route preferences (smoke) | | `FE-SETUP-001` | PASS | 2669.0137459999996ms | /setup page renders for newly-signed-up users | | `FE-SETUP-002` | PASS | 1992.229698ms | Setup wizard captures business profile info (form renders) | | `FE-SETUP-003` | PASS | 2256.6104030000006ms | Skip onboarding via 'Skip for now' → dashboard with welcome modal (skip CTA exists) | | `FE-SETUP-004` | PASS | 2412.4550600000002ms | /getting-started shows checklist with at least one step | | `FE-SETUP-005` | PASS | 2627.3340819999994ms | Welcome greeting renders user's first name | | `FE-SETUP-006` | PASS | 441.55562400000053ms | setup_finished flag persists in DB for the test admin | | `FE-SETUP-007` | PASS | 1951.4359480000003ms | /getting-started accessible even when setup_finished is true (onboarding route exception) | | `FE-SIG-001` | PASS | 0.5226249999977881ms | Save email signature persists across reload (smoke) | | `FE-SIG-002` | PASS | 1.0910259999982372ms | Outbound email shows signature appended (smoke) | | `FE-SIG-003` | PASS | 0.7035939999987022ms | Save booking link persists (smoke) | | `FE-SIG-004` | PASS | 0.2525600000008126ms | {{sender_booking_link}} merge tag rendered (smoke) | | `FE-SIG-005` | PASS | 0.41559500000221306ms | Empty signature → no trailing artifact (smoke) | | `FE-SMS-TOKY-001` | PASS | 3.345854000000145ms | BYOC card lists Toky in /settings/system (smoke) | | `FE-SMS-TOKY-002` | PASS | 0.735302000000047ms | Save bogus Toky API key → red error (smoke) | | `FE-SMS-TOKY-003` | PASS | 0.5419920000003913ms | Save real test API key creates carrier_credentials row (smoke) | | `FE-SMS-TOKY-004` | PASS | 0.36463099999991755ms | Save handshake registers webhook on Toky side (smoke) | | `FE-SMS-TOKY-005` | PASS | 0.28233700000009776ms | Re-save credential clears existing webhook auth (smoke) | | `FE-SMS-TOKY-006` | PASS | 0.30922700000019177ms | Add Toky Number lists user inventory (smoke) | | `FE-SMS-TOKY-007` | PASS | 0.34786799999983486ms | Import test number creates phonenumbers row (smoke) | | `FE-SMS-TOKY-008` | PASS | 0.7694659999997384ms | Send SMS via Toky doesn't decrement credits (smoke) | | `FE-SMS-TOKY-009` | PASS | 0.3982139999998253ms | Reply lands inbound within 10s (smoke) | | `FE-SMS-TOKY-010` | PASS | 222.35289200000034ms | Toky inbound webhook validates HTTP Basic auth | | `FE-SMS-TOKY-011` | PASS | 0.26123699999970995ms | Toky payload is JSON array (smoke) | | `FE-SMS-TOKY-012` | PASS | 0.1679340000000593ms | Toky inbound NEGATIVE sentiment marks deal LOST (smoke) | | `FE-SMS-TOKY-013` | PASS | 5.538906999999654ms | Replay same payload twice → idempotent (smoke) | | `FE-SMS-TOKY-014` | PASS | 0.48999400000002424ms | Disconnect Toky credential revokes webhook (smoke) | | `FE-SMS-TOKY-015` | PASS | 2.092616000000362ms | Twilio + Toky number coexistence (smoke) | | `FE-SMS-TOKY-016` | PASS | 0.3030649999996058ms | AI personalization charges AI credits but not phone (smoke) | | `FE-SMS-TOKY-017` | PASS | 0.1976089999998294ms | Twilio regression after Toky enabled (smoke) | | `FE-SMS-TOKY-018` | PASS | 0.16758299999992232ms | Legacy provider=NULL treated as 'twilio' (smoke) | | `FE-SMS-TOKY-019` | PASS | 0.7760480000001735ms | Toky API key with zero numbers → empty inventory (smoke) | | `FE-SMS-TOKY-020` | PASS | 0.21591300000000047ms | Webhook dedup respects DRF pagination shape (smoke) | | `FE-SMS-TW-001` | PASS | 2.6957119999997303ms | A2P brand registration starts from /settings/system (smoke) | | `FE-SMS-TW-002` | PASS | 0.47712000000001353ms | Phone-number purchase blocked when A2P != brand_approved | | `FE-SMS-TW-003` | PASS | 0.38022000000000844ms | Phone-number purchase succeeds when approved | | `FE-SMS-TW-004` | PASS | 0.2942900000002737ms | Send SMS from /phone-system/sms (smoke) | | `FE-SMS-TW-005` | PASS | 0.32386399999995774ms | Send SMS with merge tag rendered correctly | | `FE-SMS-TW-006` | PASS | 0.377835999999661ms | Send SMS with no credits → 402 / out-of-credits | | `FE-SMS-TW-007` | PASS | 0.3415669999999409ms | Twilio API failure releases reserved credits | | `FE-SMS-TW-008` | PASS | 0.3328420000002552ms | Send SMS exceeds rate limit → 429 | | `FE-SMS-TW-009` | PASS | 0.288698000000295ms | Inbound SMS appears in FE thread (smoke) | | `FE-SMS-TW-010` | PASS | 0.32509600000003047ms | Inbound negative sentiment marks deal LOST (smoke) | | `FE-SMS-TW-011` | PASS | 313.18126900000016ms | Twilio webhook with wrong bears_key → 401 | | `FE-SMS-TW-012` | PASS | 0.22762499999998909ms | Twilio status callback updates messages.status (smoke) | | `FE-USER-001` | PASS | 5776.551608000002ms | /user renders profile fields | | `FE-USER-002` | PASS | 7570.4690249999985ms | Created date renders correctly (NOT 'Invalid Date') | | `FE-USER-003` | PASS | 2.902257000001555ms | Change Password button opens modal (smoke) | | `FE-USER-004` | PASS | 0.35795900000084657ms | Password modal validates fields (smoke) | | `FE-USER-005` | PASS | 0.2018679999964661ms | Wrong current password → friendly error (smoke) | | `FE-USER-006` | PASS | 0.6725150000056601ms | Password change success closes modal + toast (smoke) | | `FE-VM-001` | PASS | 6377.203266ms | /phone-system/voicemails lists records or empty state | | `FE-VM-002` | PASS | 0.9181739999985439ms | Each row shows recipient/duration/status/timestamp (smoke) | | `FE-VM-003` | PASS | 0.300771000000168ms | Play affordance plays audio inline (smoke) | | `FE-VM-004` | PASS | 0.29219499999999243ms | Filter by status works (smoke) | | `FE-VM-005` | PASS | 0.33879199999864795ms | Pagination works at boundaries (smoke) | | `FE-VM-006` | PASS | 0.3273010000011709ms | Voicemail audio URL is signed/scoped (smoke) | | `UNKNOWN-2owmus` | PASS | 305.3956470000003ms | WEBHOOK-AUTH-003 — deal-mover sweeper rejects without auth | | `UNKNOWN-77rs6p` | PASS | 0.4567930000000615ms | FE-E2E-002 — Inbound reply NEGATIVE → deal auto-Lost (smoke) | | `UNKNOWN-7zl0x9` | PASS | 1.3011589999998705ms | FE-E2E-009 — Multi-tenant isolation across mailboxes/Toky/deals (smoke) | | `UNKNOWN-a33fma` | PASS | 173.4027390000001ms | WEBHOOK-AUTH-002 — sms-service sentiment-webhook rejects without auth | | `UNKNOWN-ajhdw8` | PASS | 1.9547489999999925ms | FE-E2E-010 — Webhook-only services not exposed (smoke — covered by WEBHOOK-AUTH) | | `UNKNOWN-asu9py` | PASS | 0.3189460000000963ms | FE-E2E-005 — Quota cap respected (smoke) | | `UNKNOWN-beybxh` | PASS | 451.4222300000001ms | WEBHOOK-AUTH-001 — email-ingress sentiment-webhook rejects without auth | | `UNKNOWN-cvxdou` | PASS | 0.3584490000000642ms | FE-E2E-003 — Toky BYOC end-to-end SMS round-trip (smoke) | | `UNKNOWN-cwdbq3` | PASS | 0.44176500000003216ms | FE-E2E-006 — Inactive user pause-billing edge case (smoke) | | `UNKNOWN-dqpfrn` | PASS | 1599.7556719999993ms | FE-ACT-S2C-001 — Endpoint exists | | `UNKNOWN-grclep` | PASS | 0.2762560000001031ms | FE-E2E-007 — Free-plan SMS via Toky succeeds, via Twilio rejected (smoke) | | `UNKNOWN-ji9i1r` | PASS | 0.5249199999998382ms | FE-E2E-008 — JWT expiry mid-flow auto-refreshes (smoke) | | `UNKNOWN-mtit78` | PASS | 0.29319600000007995ms | FE-E2E-004 — Multi-stage campaign with wait + actions (smoke) | | `UNKNOWN-ok9jgv` | PASS | 3.803529000000026ms | FE-E2E-001 — Signup → Onboarding → First campaign → First deal email sent (smoke) | | `UNKNOWN-q524pb` | PASS | 0.25026599999910104ms | FE-ACT-S2C-004 — Cycle prevention at save time (smoke) | | `UNKNOWN-w5kwkf` | PASS | 0.3982739999992191ms | FE-ACT-S2C-003 — Trigger moves deal across campaigns (smoke) | | `UNKNOWN-zekose` | PASS | 10.885928000001513ms | FE-ACT-S2C-002 — Editor lists user's campaigns + stages (smoke) |