gigafibre-fsm/services/targo-hub/lib/referral.js
louispaulb 01bb99857f refactor(targo-hub): add erp.js wrapper + migrate 7 lib files to it
Replaces hand-rolled `erpFetch` + `encodeURIComponent(JSON.stringify(...))`
URL building with a structured wrapper: erp.get/list/listRaw/create/update/
remove/getMany/hydrateLabels/raw.

Key wins:
- erp.list auto-retries up to 5 times when v16 rejects a fetched/linked
  field with "Field not permitted in query" — the field is dropped and the
  call continues, so callers don't have to know which fields v16 allows.
- erp.hydrateLabels batches link-label resolution (customer_name,
  service_location_name, …) in one query per link field — no N+1, no
  v16 breakage.
- Consistent {ok, error, status} shape for mutations.

Migrated call sites:
- otp.js: Customer email lookup + verifyOTP customer detail fetch +
  Contact email fallback + Service Location listing
- referral.js: Referral Credit fetch / update / generate
- tech-absence-sms.js: lookupTechByPhone, set/clear absence
- conversation.js: Issue archive create
- magic-link.js: Tech lookup for /refresh
- ical.js: Tech lookup + jobs listing for iCal feed
- tech-mobile.js: 13 erpFetch sites → erp wrapper

Remaining erpFetch callers (dispatch.js, acceptance.js, payments.js,
contracts.js, checkout.js, …) deliberately left untouched this pass —
they each have 10+ sites and need individual smoke-tests.

Live-tested against production ERPNext: tech-mobile page renders 54K
bytes, no runtime errors in targo-hub logs post-restart.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-04-22 23:01:27 -04:00

137 lines
5.4 KiB
JavaScript

'use strict'
// Referral Credit — validation + application endpoints backing the wizard's
// referral code field. Codes are stored in the ERPNext "Referral Credit"
// doctype (code unique, status Active/Used/Fulfilled/Cancelled).
//
// POST /api/referral/validate { code } → { ok, referrer, error? }
// POST /api/referral/apply { code, quotation, customer } → { ok, error? }
// POST /api/referral/generate { customer } → { ok, code }
//
// Apply is called from the Vue wizard at publish time (after quote creation)
// so the code flips to "Used" and links to the quotation. Fulfilled happens
// server-side once the service invoice is submitted (not here).
const { log, json, parseBody } = require('./helpers')
const erp = require('./erp')
const DT = 'Referral Credit'
async function fetchByCode (code) {
const rows = await erp.list(DT, {
filters: [['code', '=', code]],
fields: ['name', 'code', 'referrer', 'referrer_customer_name', 'status', 'new_subscriber_credit', 'referrer_credit'],
limit: 1,
})
return rows[0] || null
}
async function validateCode (code) {
if (!code || !code.trim()) return { ok: false, error: 'Code manquant' }
const normalized = code.trim().toUpperCase()
const row = await fetchByCode(normalized)
if (!row) return { ok: false, error: 'Code introuvable' }
if (row.status !== 'Active') {
return { ok: false, error: `Code déjà ${row.status === 'Used' ? 'utilisé' : row.status === 'Fulfilled' ? 'complété' : 'annulé'}` }
}
return {
ok: true,
code: row.code,
referrer: row.referrer,
referrer_name: row.referrer_customer_name || '',
new_subscriber_credit: Number(row.new_subscriber_credit) || 50,
referrer_credit: Number(row.referrer_credit) || 50,
}
}
async function applyCode (code, quotation, customer) {
if (!code) return { ok: false, error: 'Code manquant' }
const normalized = String(code).trim().toUpperCase()
const row = await fetchByCode(normalized)
if (!row) return { ok: false, error: 'Code introuvable' }
if (row.status !== 'Active') return { ok: false, error: `Code non applicable (statut: ${row.status})` }
if (row.referrer === customer) return { ok: false, error: "On ne peut pas utiliser son propre code" }
const patch = {
status: 'Used',
applied_on: new Date().toISOString().slice(0, 10),
}
if (quotation) patch.referred_quotation = quotation
if (customer) patch.referred_customer = customer
const res = await erp.update(DT, row.name, patch)
if (!res.ok) {
log('referral.apply PUT failed:', res.status, res.error)
return { ok: false, error: 'Erreur de mise à jour (' + (res.status || 'err') + ')' }
}
return { ok: true, code: row.code }
}
// Generate a new code for a customer. Idempotent — returns the existing
// Active code if one already exists for that referrer (customers can share
// one code perpetually until explicitly cancelled).
async function generateCode (customer) {
if (!customer) return { ok: false, error: 'customer manquant' }
const existing = await erp.list(DT, {
filters: [['referrer', '=', customer], ['status', '=', 'Active']],
fields: ['name', 'code'],
limit: 1,
})
if (existing.length) {
return { ok: true, code: existing[0].code, created: false }
}
// New code — 6 uppercase alphanumerics, prefixed with GIGA-. Retry on
// collision (the unique constraint guarantees we don't double-insert).
const alphabet = 'ABCDEFGHJKLMNPQRSTUVWXYZ23456789' // ambiguous chars omitted
for (let attempt = 0; attempt < 8; attempt++) {
let suffix = ''
for (let i = 0; i < 6; i++) suffix += alphabet[Math.floor(Math.random() * alphabet.length)]
const code = `GIGA-${suffix}`
const res = await erp.create(DT, { code, referrer: customer, status: 'Active' })
if (res.ok) return { ok: true, code, created: true }
// 409/417 on duplicate code or field rejection — retry with new suffix
if (res.status !== 409 && res.status !== 417) {
log('referral.generate POST failed:', res.status, res.error)
return { ok: false, error: 'Erreur ERP (' + (res.status || 'err') + ')' }
}
}
return { ok: false, error: 'Génération échouée après 8 tentatives' }
}
async function handle (req, res, method, path) {
if (path === '/api/referral/validate' && method === 'POST') {
try {
const body = await parseBody(req)
const result = await validateCode(body.code || '')
return json(res, 200, result)
} catch (e) {
log('referral.validate error:', e.message)
return json(res, 500, { ok: false, error: e.message })
}
}
if (path === '/api/referral/apply' && method === 'POST') {
try {
const body = await parseBody(req)
const result = await applyCode(body.code || '', body.quotation || '', body.customer || '')
return json(res, result.ok ? 200 : 400, result)
} catch (e) {
log('referral.apply error:', e.message)
return json(res, 500, { ok: false, error: e.message })
}
}
if (path === '/api/referral/generate' && method === 'POST') {
try {
const body = await parseBody(req)
const result = await generateCode(body.customer || '')
return json(res, result.ok ? 200 : 400, result)
} catch (e) {
log('referral.generate error:', e.message)
return json(res, 500, { ok: false, error: e.message })
}
}
return json(res, 404, { error: 'Referral route not found: ' + path })
}
module.exports = { handle, validateCode, applyCode, generateCode }