Replaces the previous "one row, N seats via extra.quantity" model
with proper one-row-per-attendee semantics. Each attendee gets a
unique scannable id; the door PUT /register/{ticket_id} marks
them registered independently — so a buyer can purchase 3 tickets,
hand 2 QRs to friends arriving separately, and each attendee can
enter on their own schedule.
Schema (migrations_fork.py m002):
- ticket.payment_hash: new TEXT column shared across all rows of
a multi-ticket purchase. Backfilled `payment_hash = id` for
pre-migration rows (id WAS the payment_hash by invariant).
Wire:
- TicketPaymentRequest grows `ticket_ids: list[str]` so the
webapp gets every scannable id back in the create response.
- POST /tickets/{event_id}/{payment_hash} polling endpoint now
reports `ticket_ids` (every row) + keeps `ticket_id` for
back-compat.
- api_ticket_create loops quantity times; the first row reuses
payment_hash as id (preserves legacy `id == payment_hash`
invariant for single-ticket purchases), the rest get
urlsafe_short_hash() uuids.
Payment flow:
- on_invoice_paid fetches all rows by payment_hash and marks each
paid via set_ticket_paid, which now increments event.sold by 1
per row (was N per row via extra.quantity — simpler now). The
per-event asyncio lock still serializes counter + republish so
concurrent multi-ticket purchases for the same event don't
reorder the published Nostr state.
- Each paid row triggers its own send_ticket_notification_in_
background call — no-op for buyers without nostr_identifier /
email, useful when the buyer set those on the row.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
130 lines
5.1 KiB
Python
130 lines
5.1 KiB
Python
"""
|
|
Fork-specific database migrations for the aiolabs events extension.
|
|
|
|
These migrations are tracked separately under `events_fork` in the
|
|
`dbversions` table (loaded by `lnbits/core/helpers.py:migrate_extension_database`),
|
|
so they do not collide with upstream's `m{NNN}_*` numbering in
|
|
`migrations.py`. Keeping the upstream-tracked file untouched means
|
|
`git pull upstream` stays rebase-clean for schema changes.
|
|
|
|
Conventions:
|
|
- Sequential numbering starting from m001.
|
|
- Each migration is `async def m{NNN}_<description>(db)`.
|
|
- DDL must be idempotent: a fresh install runs every migration; an
|
|
install that previously ran the OLD versions of these as
|
|
`m007-m011` in `migrations.py` has the columns/tables already.
|
|
Use `_alter_add_column_safe` / `_create_table_safe` so re-runs are
|
|
no-ops instead of crashes.
|
|
|
|
History compressed into m001 (was m007-m011 in migrations.py pre-v1.6
|
|
rebase):
|
|
- m007 add_user_id_support (ticket.user_id column)
|
|
- m008 add_event_status (events.status column)
|
|
- m009 add_nostr_columns (events.nostr_event_id + created_at)
|
|
- m010 add_events_settings (events.settings singleton table)
|
|
- m011 add_location_and_categories (events.location + categories)
|
|
"""
|
|
|
|
|
|
async def _alter_add_column_safe(db, sql: str) -> None:
|
|
"""ALTER TABLE ADD COLUMN that swallows duplicate-column errors.
|
|
|
|
Re-running the squashed migration on a database that already has
|
|
these columns (from the pre-squash `m007-m011` in migrations.py)
|
|
must be a silent no-op. Same swallow we used in the old migrations.
|
|
"""
|
|
try:
|
|
await db.execute(sql)
|
|
except Exception as exc:
|
|
msg = str(exc).lower()
|
|
if "duplicate column" in msg or "already exists" in msg:
|
|
return
|
|
raise
|
|
|
|
|
|
async def m001_aio_event_schema(db):
|
|
"""
|
|
Apply every aiolabs schema delta on top of upstream events v1.3.0.
|
|
|
|
This is the squashed equivalent of the pre-v1.6 sequence
|
|
m007 → m011. Order matters for the settings table seed insert
|
|
but the individual column adds are independent and idempotent.
|
|
"""
|
|
|
|
# --- ticket.user_id ----------------------------------------------
|
|
# Lets a ticket reference an LNbits user id instead of (name, email).
|
|
# Application logic enforces that exactly one identifier scheme is
|
|
# used per ticket.
|
|
await _alter_add_column_safe(
|
|
db, "ALTER TABLE events.ticket ADD COLUMN user_id TEXT"
|
|
)
|
|
|
|
# --- events.status -----------------------------------------------
|
|
# Proposal / approval workflow. Existing rows default to 'approved'
|
|
# so they stay visible after upgrade.
|
|
await _alter_add_column_safe(
|
|
db,
|
|
"ALTER TABLE events.events ADD COLUMN status TEXT NOT NULL DEFAULT 'approved'",
|
|
)
|
|
|
|
# --- events.nostr_event_id, nostr_event_created_at ---------------
|
|
# Track the most recent NIP-52 calendar event we published, so
|
|
# subsequent edits can issue replaceable updates and NIP-09 deletes
|
|
# against the right addressable coordinate.
|
|
await _alter_add_column_safe(
|
|
db, "ALTER TABLE events.events ADD COLUMN nostr_event_id TEXT"
|
|
)
|
|
await _alter_add_column_safe(
|
|
db, "ALTER TABLE events.events ADD COLUMN nostr_event_created_at INTEGER"
|
|
)
|
|
|
|
# --- events.settings ---------------------------------------------
|
|
# Singleton settings row used by the admin UI to toggle e.g.
|
|
# auto_approve. CREATE TABLE IF NOT EXISTS + a guarded seed keeps
|
|
# this idempotent.
|
|
await db.execute("""
|
|
CREATE TABLE IF NOT EXISTS events.settings (
|
|
id INTEGER PRIMARY KEY DEFAULT 1,
|
|
auto_approve BOOLEAN NOT NULL DEFAULT FALSE
|
|
)
|
|
""")
|
|
await db.execute(
|
|
"INSERT INTO events.settings (id, auto_approve) "
|
|
"SELECT 1, FALSE WHERE NOT EXISTS "
|
|
"(SELECT 1 FROM events.settings WHERE id = 1)"
|
|
)
|
|
|
|
# --- events.location, events.categories --------------------------
|
|
# NIP-52 calendar metadata. `categories` carries a JSON-encoded
|
|
# list of hashtags (the NIP-52 `t` tags).
|
|
await _alter_add_column_safe(
|
|
db, "ALTER TABLE events.events ADD COLUMN location TEXT"
|
|
)
|
|
await _alter_add_column_safe(
|
|
db, "ALTER TABLE events.events ADD COLUMN categories TEXT"
|
|
)
|
|
|
|
|
|
async def m002_ticket_payment_hash(db):
|
|
"""
|
|
Add `ticket.payment_hash` for multi-ticket purchases.
|
|
|
|
Multi-ticket purchases land as N rows sharing one LNbits invoice
|
|
(so each attendee gets a distinct scannable QR but the buyer
|
|
pays once). `ticket.id` stays the row primary key — for legacy
|
|
single-purchase rows it equals payment_hash; for multi-purchase
|
|
children it's a uuid generated at create-time. `payment_hash`
|
|
is the new join key for invoice lookup.
|
|
|
|
Backfill existing rows from id so the
|
|
GET-tickets-by-payment-hash path keeps working for pre-migration
|
|
data (id was the payment_hash by invariant before this column).
|
|
"""
|
|
await _alter_add_column_safe(
|
|
db, "ALTER TABLE events.ticket ADD COLUMN payment_hash TEXT"
|
|
)
|
|
await db.execute(
|
|
"UPDATE events.ticket SET payment_hash = id "
|
|
"WHERE payment_hash IS NULL OR payment_hash = ''"
|
|
)
|
|
|