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>
137 lines
3.7 KiB
YAML
137 lines
3.7 KiB
YAML
# ts-mitra-A-05 — §A.1 phone-format normalization
|
|
# Spec ref: requirement/flow_mitra.mermaid.md §A.1
|
|
#
|
|
# Indonesian users type phone numbers in many shapes. The login screen's
|
|
# _subscriberDigits() in login_screen.dart must normalize ALL of these to
|
|
# the same +628200000501 (subscriber digits = 8200000501):
|
|
#
|
|
# 8200000501 — subscriber only
|
|
# 08200000501 — local format with leading 0
|
|
# 628200000501 — country code without +
|
|
# +628200000501 — full E.164
|
|
# 0628200000501 — typo combo with leading 0 before country code
|
|
#
|
|
# Strategy: seed ONE mitra at +628200000501. For each variant, do a fresh
|
|
# launchApp clearState, type the variant, tap "kirim kode", and assert the
|
|
# S3b screen shows the correctly normalized phone "+628200000501". A fresh
|
|
# launch per variant is more reliable than back-navigation across maestro
|
|
# / IME / keyboard state.
|
|
appId: com.mybestie.mitra
|
|
env:
|
|
TEST_PHONE: "+628200000501"
|
|
MITRA_DISPLAY_NAME: "Maestro Variants"
|
|
BACKEND_INTERNAL_URL: http://localhost:3001
|
|
---
|
|
- runScript:
|
|
file: ../scripts/seed_mitra.js
|
|
env:
|
|
TEST_PHONE: ${TEST_PHONE}
|
|
MITRA_DISPLAY_NAME: ${MITRA_DISPLAY_NAME}
|
|
IS_ACTIVE: "true"
|
|
BACKEND_INTERNAL_URL: ${BACKEND_INTERNAL_URL}
|
|
|
|
# ── Variant 1: 8xxxxxxxxx (subscriber only, 10 digits) ──
|
|
- runScript:
|
|
file: ../scripts/reset_phone.js
|
|
env:
|
|
TEST_PHONE: ${TEST_PHONE}
|
|
BACKEND_INTERNAL_URL: ${BACKEND_INTERNAL_URL}
|
|
- launchApp:
|
|
clearState: true
|
|
- extendedWaitUntil:
|
|
visible:
|
|
text: "(?s).*Halo Mitra Bestie.*"
|
|
timeout: 10000
|
|
- tapOn:
|
|
point: "50%, 53%"
|
|
- inputText: "8200000501"
|
|
- tapOn: "(?s).*kirim kode.*"
|
|
- extendedWaitUntil:
|
|
visible:
|
|
text: "(?s).*\\+628200000501.*"
|
|
timeout: 10000
|
|
|
|
# ── Variant 2: 08xxxxxxxxx (local format with leading 0) ──
|
|
- runScript:
|
|
file: ../scripts/reset_phone.js
|
|
env:
|
|
TEST_PHONE: ${TEST_PHONE}
|
|
BACKEND_INTERNAL_URL: ${BACKEND_INTERNAL_URL}
|
|
- launchApp:
|
|
clearState: true
|
|
- extendedWaitUntil:
|
|
visible:
|
|
text: "(?s).*Halo Mitra Bestie.*"
|
|
timeout: 10000
|
|
- tapOn:
|
|
point: "50%, 53%"
|
|
- inputText: "08200000501"
|
|
- tapOn: "(?s).*kirim kode.*"
|
|
- extendedWaitUntil:
|
|
visible:
|
|
text: "(?s).*\\+628200000501.*"
|
|
timeout: 10000
|
|
|
|
# ── Variant 3: 628xxxxxxxxx (country code without +) ──
|
|
- runScript:
|
|
file: ../scripts/reset_phone.js
|
|
env:
|
|
TEST_PHONE: ${TEST_PHONE}
|
|
BACKEND_INTERNAL_URL: ${BACKEND_INTERNAL_URL}
|
|
- launchApp:
|
|
clearState: true
|
|
- extendedWaitUntil:
|
|
visible:
|
|
text: "(?s).*Halo Mitra Bestie.*"
|
|
timeout: 10000
|
|
- tapOn:
|
|
point: "50%, 53%"
|
|
- inputText: "628200000501"
|
|
- tapOn: "(?s).*kirim kode.*"
|
|
- extendedWaitUntil:
|
|
visible:
|
|
text: "(?s).*\\+628200000501.*"
|
|
timeout: 10000
|
|
|
|
# ── Variant 4: +628xxxxxxxxx (full E.164 with +) ──
|
|
- runScript:
|
|
file: ../scripts/reset_phone.js
|
|
env:
|
|
TEST_PHONE: ${TEST_PHONE}
|
|
BACKEND_INTERNAL_URL: ${BACKEND_INTERNAL_URL}
|
|
- launchApp:
|
|
clearState: true
|
|
- extendedWaitUntil:
|
|
visible:
|
|
text: "(?s).*Halo Mitra Bestie.*"
|
|
timeout: 10000
|
|
- tapOn:
|
|
point: "50%, 53%"
|
|
- inputText: "+628200000501"
|
|
- tapOn: "(?s).*kirim kode.*"
|
|
- extendedWaitUntil:
|
|
visible:
|
|
text: "(?s).*\\+628200000501.*"
|
|
timeout: 10000
|
|
|
|
# ── Variant 5: 0628xxxxxxxxx (typo combo) ──
|
|
- runScript:
|
|
file: ../scripts/reset_phone.js
|
|
env:
|
|
TEST_PHONE: ${TEST_PHONE}
|
|
BACKEND_INTERNAL_URL: ${BACKEND_INTERNAL_URL}
|
|
- launchApp:
|
|
clearState: true
|
|
- extendedWaitUntil:
|
|
visible:
|
|
text: "(?s).*Halo Mitra Bestie.*"
|
|
timeout: 10000
|
|
- tapOn:
|
|
point: "50%, 53%"
|
|
- inputText: "0628200000501"
|
|
- tapOn: "(?s).*kirim kode.*"
|
|
- extendedWaitUntil:
|
|
visible:
|
|
text: "(?s).*\\+628200000501.*"
|
|
timeout: 10000
|