Day-of-launch deep-dive audit and remediation. 11 issues fixed across security, correctness, and reliability. Tests: 698 → 758 passing (+60), 0 failing, 2 skipped. CRITICAL (security) Fix #1 — HELCIM_API_TOKEN removed from runtimeConfig.public; dead useHelcim.js deleted. Production token MUST BE ROTATED post-deploy (was previously exposed in window.__NUXT__ payload). Fix #2 — /api/helcim/customer gated with origin check + per-IP/email rate limit + magic-link email verification (replaces unauthenticated setAuthCookie). Adds payment-bridge token for paid-tier signup so users can complete Helcim checkout before email verify. New utils: server/utils/{magicLink,rateLimit}.js. UX: signup success copy now prompts user to check email. Fix #3 — /api/events/[id]/payment deleted (dead code with unauth member-spoof bypass — processHelcimPayment was a permanent stub). Removes processHelcimPayment export and eventPaymentSchema. Fix #4 — /api/helcim/initialize-payment re-derives ticket amount server-side via calculateTicketPrice and calculateSeriesTicketPrice. Adds new series_ticket metadata type (was being shoved through event_ticket with seriesId in metadata.eventId). Fix #5 — /api/helcim/customer upgrades existing status:guest members in place rather than rejecting with 409. Lowercases email at lookup; preserves _id so prior event registrations stay linked. HIGH (correctness / reliability) Fix #6 — Daily reconciliation cron via Netlify scheduled function (@daily). New: netlify.toml, netlify/functions/reconcile-payments.mjs, server/api/internal/reconcile-payments.post.js. Shared-secret auth via NUXT_RECONCILE_TOKEN env var. Inline 3-retry exponential backoff on Helcim transactions API. Fix #7 — validateBeforeSave: false on event subdoc saves (waitlist endpoints) to dodge legacy location validators. Fix #8 — /api/series/[id]/tickets/purchase always upserts a guest Member when caller is unauthenticated, mirrors event-ticket flow byte-for-byte. SeriesPassPurchase.vue adds guest-account hint and client auth refresh on signedIn:true response. Fix #9 — /api/members/cancel-subscription leaves status active per ratified bylaws (was pending_payment). Adds lastCancelledAt audit field on Member model. Indirectly fixes false-positive detectStuckPendingPayment admin alert for cancelled members. Fix #10 — /api/auth/verify uses validateBody with strict() Zod schema (verifyMagicLinkSchema, max 2000 chars). Fix #11 — 8 vitest cases for cancel-subscription handler (was uncovered). Specs and audit at docs/superpowers/specs/2026-04-25-fix-*.md and docs/superpowers/plans/2026-04-25-launch-readiness-fixes.md. LAUNCH_READINESS.md updated with new test count, 3 deploy-time tasks (rotate Helcim token, set NUXT_RECONCILE_TOKEN, verify Netlify scheduled function), and Fixed-2026-04-25 fix log.
186 lines
4.6 KiB
Vue
186 lines
4.6 KiB
Vue
<template>
|
|
<Teleport to="body">
|
|
<div v-if="state !== 'idle'" class="signup-flow-overlay">
|
|
<div class="signup-flow-card">
|
|
<div class="signup-flow-step">{{ stepLabel }}</div>
|
|
|
|
<template v-if="isProgress">
|
|
<h2 class="signup-flow-heading">{{ progressHeading }}</h2>
|
|
<p class="signup-flow-body">
|
|
Please don't close this window. This usually takes a few seconds.
|
|
</p>
|
|
</template>
|
|
|
|
<template v-if="state === 'success'">
|
|
<h2 class="signup-flow-heading">Welcome to Ghost Guild!</h2>
|
|
<DashedBox :hoverable="false">
|
|
<div class="section-label" style="margin-bottom: 12px">
|
|
Membership Details
|
|
</div>
|
|
<dl class="details-list">
|
|
<div class="details-row">
|
|
<dt>Name</dt><dd>{{ summary?.name }}</dd>
|
|
</div>
|
|
<div class="details-row">
|
|
<dt>Email</dt><dd>{{ summary?.email }}</dd>
|
|
</div>
|
|
<div class="details-row">
|
|
<dt>Circle</dt><dd class="capitalize">{{ summary?.circle }}</dd>
|
|
</div>
|
|
<div class="details-row">
|
|
<dt>Contribution</dt><dd>{{ summary?.contribution }}</dd>
|
|
</div>
|
|
</dl>
|
|
</DashedBox>
|
|
<p class="signup-flow-body" style="margin-top: 16px">
|
|
Check {{ summary?.email }} for a sign-in link to finish setting up
|
|
your account. The link expires in 15 minutes.
|
|
</p>
|
|
</template>
|
|
|
|
<template v-if="state === 'error'">
|
|
<h2 class="signup-flow-heading">We couldn't complete your signup</h2>
|
|
<div v-if="errorMessage" class="error-box">
|
|
{{ errorMessage }}
|
|
</div>
|
|
<div class="button-row" style="margin-top: 20px">
|
|
<button class="btn" @click="$emit('close')">
|
|
Back to form
|
|
</button>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
</div>
|
|
</Teleport>
|
|
</template>
|
|
|
|
<script setup>
|
|
import { computed } from "vue";
|
|
|
|
const props = defineProps({
|
|
state: { type: String, required: true },
|
|
summary: { type: Object, default: null },
|
|
errorMessage: { type: String, default: "" },
|
|
dashboardHref: { type: String, default: "/welcome" },
|
|
});
|
|
|
|
defineEmits(["close"]);
|
|
|
|
const PROGRESS_STATES = [
|
|
"creating-customer",
|
|
"opening-payment",
|
|
"processing-payment",
|
|
"creating-subscription",
|
|
];
|
|
|
|
const isProgress = computed(() => PROGRESS_STATES.includes(props.state));
|
|
|
|
const progressHeading = computed(() => {
|
|
switch (props.state) {
|
|
case "creating-customer": return "Creating your account...";
|
|
case "opening-payment": return "Opening secure payment...";
|
|
case "processing-payment": return "Confirming your card...";
|
|
case "creating-subscription": return "Activating your membership...";
|
|
default: return "";
|
|
}
|
|
});
|
|
|
|
const stepLabel = computed(() => {
|
|
switch (props.state) {
|
|
case "creating-customer":
|
|
case "opening-payment":
|
|
return "Step 2 of 3 — Payment";
|
|
case "processing-payment":
|
|
case "creating-subscription":
|
|
return "Step 2 of 3 — Finalizing";
|
|
case "success":
|
|
return "Step 3 of 3 — Welcome";
|
|
case "error":
|
|
return "Something went wrong";
|
|
default:
|
|
return "";
|
|
}
|
|
});
|
|
</script>
|
|
|
|
<style scoped>
|
|
.signup-flow-overlay {
|
|
position: fixed;
|
|
inset: 0;
|
|
z-index: 50;
|
|
background: rgba(42, 32, 21, 0.72);
|
|
backdrop-filter: blur(4px);
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
padding: 24px;
|
|
}
|
|
|
|
.signup-flow-card {
|
|
background: var(--bg);
|
|
border: 1px dashed var(--border);
|
|
padding: 32px;
|
|
max-width: 520px;
|
|
width: 100%;
|
|
max-height: calc(100vh - 48px);
|
|
overflow-y: auto;
|
|
}
|
|
|
|
.signup-flow-step {
|
|
font-family: var(--font-body);
|
|
font-size: 0.75rem;
|
|
letter-spacing: 0.08em;
|
|
text-transform: uppercase;
|
|
color: var(--text-dim);
|
|
margin-bottom: 12px;
|
|
}
|
|
|
|
.signup-flow-heading {
|
|
font-family: var(--font-display);
|
|
font-size: 1.5rem;
|
|
color: var(--text-bright);
|
|
margin: 0 0 16px;
|
|
}
|
|
|
|
.signup-flow-body {
|
|
font-family: var(--font-body);
|
|
color: var(--text);
|
|
line-height: 1.5;
|
|
margin: 0;
|
|
}
|
|
|
|
.details-list {
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 8px;
|
|
}
|
|
|
|
.details-row {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: baseline;
|
|
font-size: 13px;
|
|
}
|
|
|
|
.details-row dt {
|
|
color: var(--text-faint);
|
|
}
|
|
|
|
.details-row dd {
|
|
color: var(--text-bright);
|
|
font-weight: 500;
|
|
}
|
|
|
|
.error-box {
|
|
border: 1px dashed var(--ember);
|
|
color: var(--ember);
|
|
padding: 12px 16px;
|
|
font-size: 12px;
|
|
}
|
|
|
|
.button-row {
|
|
display: flex;
|
|
gap: 12px;
|
|
align-items: center;
|
|
}
|
|
</style>
|