Phase 4 Stage 1: backend foundation (additive endpoints + schema)
Schema (idempotent migration): - payment_sessions.is_free_trial -> is_first_session_discount (data copied) - payment_sessions.mode TEXT NOT NULL DEFAULT 'chat' CHECK (chat|call) - chat_sessions.topics TEXT[] for ESP picks (info-only) New endpoints: - GET /api/client/onboarding-state (drives verif sheet + S6 paywall gate) - GET /api/client/chat-pricing (rewrite: chat+call groups + first-session discount block, per-customer eligibility) - GET /api/shared/auth-providers (env-probed; replaces ENABLE_SOCIAL_AUTH build flag — frontend cutover lands in stage 2) - GET /api/client/support-handles (Tanya Admin handles, CC-config-driven) session_warning WS event fires once at 180s remaining. app_config seeds (mock pricing tiers, first-session discount, support handles, payment method order, end-session 2-step toggle). CC SettingsPage: 3 new sections (first-session discount, pricing tiers JSON editors, support handles). 15/15 Vitest passing. chat_sessions.is_free_trial also renamed for consistency (plan only specified payment_sessions; pairing.service.js read both). Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -34,7 +34,11 @@ describe('POST /api/client/payment-sessions', () => {
|
||||
|
||||
beforeEach(async () => {
|
||||
await resetDb()
|
||||
customer = await createCustomer({ callName: 'PaymentTester' })
|
||||
// Phone-verified customer (phone non-null) is required for first-session-discount
|
||||
// eligibility under the Phase 4 predicate.
|
||||
// Random suffix avoids the unique-phone constraint clashing with parallel test files.
|
||||
const phone = `+628${Math.floor(Math.random() * 1e10).toString().padStart(10, '0')}`
|
||||
customer = await createCustomer({ callName: 'PaymentTester', phone })
|
||||
token = customerJwt(customer.id)
|
||||
})
|
||||
|
||||
@@ -42,25 +46,25 @@ describe('POST /api/client/payment-sessions', () => {
|
||||
await app?.close()
|
||||
})
|
||||
|
||||
it('happy path returns 201 + a pending payment-session row', async () => {
|
||||
it('happy path returns 201 + a pending payment-session row at the discounted price for an eligible customer', async () => {
|
||||
const res = await app.inject({
|
||||
method: 'POST',
|
||||
url: '/api/client/payment-sessions',
|
||||
headers: authHeader(token),
|
||||
payload: { duration_minutes: 15 },
|
||||
// Discount duration default is 12 minutes (config seed). Eligible customer →
|
||||
// amount forced to actual_price_idr (2000), is_first_session_discount=true.
|
||||
payload: { duration_minutes: 12 },
|
||||
})
|
||||
|
||||
expect(res.statusCode).toBe(201)
|
||||
const body = res.json()
|
||||
expect(body.success).toBe(true)
|
||||
expect(body.data.status).toBe(PaymentSessionStatus.PENDING)
|
||||
expect(body.data.duration_minutes).toBe(15)
|
||||
// Default tier for 15min from migrate.js is 30000 — but the eligibility logic
|
||||
// also needs `free_trial_enabled` to be true (default) AND no prior tx. Customer is
|
||||
// brand-new so they get the trial → amount=0, is_free_trial=true. Verify accordingly.
|
||||
expect(body.data.is_free_trial).toBe(true)
|
||||
expect(body.data.amount).toBe(0)
|
||||
expect(body.data.duration_minutes).toBe(12)
|
||||
expect(body.data.is_first_session_discount).toBe(true)
|
||||
expect(body.data.amount).toBe(2000)
|
||||
expect(body.data.is_extension).toBe(false)
|
||||
expect(body.data.mode).toBe('chat')
|
||||
|
||||
// Verify persistence
|
||||
const sql = db()
|
||||
@@ -69,35 +73,41 @@ describe('POST /api/client/payment-sessions', () => {
|
||||
expect(row.customer_id).toBe(customer.id)
|
||||
})
|
||||
|
||||
it('POST /:id/confirm transitions the row and returns 200', async () => {
|
||||
// Create a paid (non-trial) tier so the row has a non-zero amount and we exercise the
|
||||
// confirm path with a "real" payment. Insert a transaction first so the customer is
|
||||
// ineligible for the free trial.
|
||||
it('non-eligible customer pays the standard tier price', async () => {
|
||||
// Drop first-session-discount eligibility by inserting a completed session.
|
||||
const sql = db()
|
||||
// Bootstrap: create a fake prior chat session + transaction so the customer is no
|
||||
// longer eligible for the free trial. (The simpler alternative — flipping
|
||||
// free_trial_enabled in app_config — would impact other tests.)
|
||||
const [prior] = await sql`
|
||||
INSERT INTO chat_sessions (customer_id, status, duration_minutes, price)
|
||||
VALUES (${customer.id}, 'completed', 15, 30000)
|
||||
RETURNING id
|
||||
`
|
||||
await sql`
|
||||
INSERT INTO customer_transactions (customer_id, session_id, type, amount)
|
||||
VALUES (${customer.id}, ${prior.id}, 'paid', 30000)
|
||||
INSERT INTO chat_sessions (customer_id, status, duration_minutes, price)
|
||||
VALUES (${customer.id}, 'completed', 12, 12000)
|
||||
`
|
||||
|
||||
const res = await app.inject({
|
||||
method: 'POST',
|
||||
url: '/api/client/payment-sessions',
|
||||
headers: authHeader(token),
|
||||
payload: { duration_minutes: 12 },
|
||||
})
|
||||
|
||||
expect(res.statusCode).toBe(201)
|
||||
const body = res.json()
|
||||
expect(body.data.is_first_session_discount).toBe(false)
|
||||
// 12-minute tier in Phase 4 chat tiers = 12000 IDR.
|
||||
expect(body.data.amount).toBe(12000)
|
||||
})
|
||||
|
||||
it('POST /:id/confirm transitions the row and returns 200', async () => {
|
||||
// Use a non-discount tier (5 min @ 5000 IDR) so we exercise the regular confirm path.
|
||||
const createRes = await app.inject({
|
||||
method: 'POST',
|
||||
url: '/api/client/payment-sessions',
|
||||
headers: authHeader(token),
|
||||
payload: { duration_minutes: 15 },
|
||||
payload: { duration_minutes: 5 },
|
||||
})
|
||||
expect(createRes.statusCode).toBe(201)
|
||||
const created = createRes.json().data
|
||||
expect(created.status).toBe(PaymentSessionStatus.PENDING)
|
||||
expect(created.is_free_trial).toBe(false)
|
||||
expect(created.amount).toBe(30000)
|
||||
expect(created.is_first_session_discount).toBe(false)
|
||||
expect(created.amount).toBe(5000)
|
||||
|
||||
const confirmRes = await app.inject({
|
||||
method: 'POST',
|
||||
@@ -112,4 +122,21 @@ describe('POST /api/client/payment-sessions', () => {
|
||||
expect(confirmed.status).toBe(PaymentSessionStatus.CONFIRMED)
|
||||
expect(confirmed.confirmed_at).toBeTruthy()
|
||||
})
|
||||
|
||||
it('call-mode payment session uses the call tier price group', async () => {
|
||||
// 20-minute call tier in Phase 4 = 17000 IDR.
|
||||
const res = await app.inject({
|
||||
method: 'POST',
|
||||
url: '/api/client/payment-sessions',
|
||||
headers: authHeader(token),
|
||||
payload: { duration_minutes: 20, mode: 'call' },
|
||||
})
|
||||
|
||||
expect(res.statusCode).toBe(201)
|
||||
const body = res.json()
|
||||
expect(body.data.mode).toBe('call')
|
||||
// Eligible customer but discount modes default = ['chat'], so call is full price.
|
||||
expect(body.data.is_first_session_discount).toBe(false)
|
||||
expect(body.data.amount).toBe(17000)
|
||||
})
|
||||
})
|
||||
|
||||
Reference in New Issue
Block a user