Brings the mitra app to figma-bestie parity for Home (§1), Undangan
inbox with Curhat Baru + Perpanjang tabs (§2), and the incoming-popup
+ active-chat flow (§3). Home now lives inside a StatefulShellRoute
with BestieTabBar so Profil + Undangan + Home share one shell.
- Shell: features/shell/ (StatefulShellRoute, BestieTabBar, 3 branches)
- Undangan: features/undangan/ — Curhat Baru reads
chatRequestProvider.pendingInvites; row Terima delegates accept to
the notifier and ChatRequestOverlay owns nav (no double-push).
Perpanjang tab stubbed (empty state) until backend exposes
pendingExtensionsProvider.
- Profil: features/profile/ — Bestie-styled stub
- Home: refactored to body-only (shell owns chrome)
- Popup: chat_request_overlay + chat_request_notifier updated to
serve the list rows, not just the modal
- Chat: mitra_chat_screen polish
- Theme: accentAmber tokens for the Perpanjang tab + halo_orb widget
(loading spinner used by undangan list states)
- Login: replace broken GoRouterState location guard with
_expectOtpPush flag — was stacking duplicate /otp pages on OTP
resend (see project-otp-nav-bug-fixed-2026-05-21)
Maestro:
- 17 new flows under .maestro/flows/ts-mitra-{1,2,3}-* covering home
online/offline variants, undangan empty/populated/tolak states,
popup curhat-baru → accept → chat → ended banner, plus popup
dismiss/expire/cancelled edge cases
- 4 new §A OTP flows (07/08/09/10) for invalid/mismatch/expired/cooldown
- Helper scripts: force_mitra_online/offline, force_pairing_timeout,
force_session_expires_at, delete_mitra_status_row,
customer_blast_now (js), customer_cancel_latest_blast
- Backend: POST /internal/_test/delete-mitra-status-row supports the
"fresh mitra with no status row" test setup
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
65 lines
2.8 KiB
JavaScript
65 lines
2.8 KiB
JavaScript
// Seed a confirmed payment_session for the test customer and fire a general blast.
|
|
// Used by Maestro flows that drive the mitra side and need a customer's
|
|
// request to arrive without running a second app.
|
|
//
|
|
// Required env: BACKEND_URL, TEST_CUSTOMER_JWT (from .maestro/config.yaml)
|
|
//
|
|
// Replaces customer_blast_now.sh (Maestro's runScript only supports JS, not shell).
|
|
|
|
const backend = BACKEND_URL || 'http://localhost:3000'
|
|
const internal = BACKEND_INTERNAL_URL || 'http://localhost:3001'
|
|
|
|
if (!TEST_CUSTOMER_JWT || TEST_CUSTOMER_JWT.startsWith('REPLACE')) {
|
|
// Test customer creds aren't set — create an anonymous customer instead so the
|
|
// suite still works on a fresh dev machine.
|
|
const auth = http.post(`${backend}/api/shared/auth/anonymous`, {
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: '{}',
|
|
})
|
|
if (auth.status !== 201 && auth.status !== 200) {
|
|
throw new Error(`anonymous auth failed (${auth.status}): ${auth.body}`)
|
|
}
|
|
const ad = json(auth.body)
|
|
output.TEST_CUSTOMER_JWT = ad.data.access_token
|
|
// give the anonymous customer a display name so the chat-request endpoint accepts
|
|
http.post(`${backend}/api/client/auth/profile`, {
|
|
headers: { 'Content-Type': 'application/json', Authorization: `Bearer ${ad.data.access_token}` },
|
|
body: JSON.stringify({ display_name: 'BlastTester' }),
|
|
method: 'PATCH',
|
|
})
|
|
}
|
|
|
|
const token = output.TEST_CUSTOMER_JWT || TEST_CUSTOMER_JWT
|
|
|
|
// Step 1: create a payment session (5 min chat)
|
|
const psResp = http.post(`${backend}/api/client/payment-sessions`, {
|
|
headers: { 'Content-Type': 'application/json', Authorization: `Bearer ${token}` },
|
|
body: JSON.stringify({ duration_minutes: 5, mode: 'chat' }),
|
|
})
|
|
if (psResp.status !== 200 && psResp.status !== 201) {
|
|
throw new Error(`create-payment failed (${psResp.status}): ${psResp.body}`)
|
|
}
|
|
const ps = json(psResp.body)
|
|
output.PAYMENT_SESSION_ID = ps.data.id
|
|
|
|
// Step 2: force-confirm via internal test endpoint (skip real Xendit)
|
|
const confirmResp = http.post(`${internal}/internal/_test/force-confirm-payment`, {
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify({ latest: true }),
|
|
})
|
|
if (confirmResp.status !== 200) {
|
|
throw new Error(`force-confirm-payment failed (${confirmResp.status}): ${confirmResp.body}`)
|
|
}
|
|
|
|
// Step 3: fire the chat request (general blast)
|
|
const brResp = http.post(`${backend}/api/client/chat/request`, {
|
|
headers: { 'Content-Type': 'application/json', Authorization: `Bearer ${token}` },
|
|
body: JSON.stringify({ payment_session_id: ps.data.id, topic_sensitivity: 'regular' }),
|
|
})
|
|
if (brResp.status !== 200 && brResp.status !== 201) {
|
|
throw new Error(`fire-blast failed (${brResp.status}): ${brResp.body}`)
|
|
}
|
|
const br = json(brResp.body)
|
|
output.SESSION_ID = br.data.id
|
|
console.log('blast fired — session_id:', br.data.id)
|