Some checks failed
lint.yml / feat: publish ticket counts in NIP-52 tags + republish on sale (pull_request) Failing after 0s
Inventory sync over Nostr, mirroring how nostrmarket republishes kind 30018 product events when stock changes. Connected webapp / other-client subscriptions pick up the new state via their existing relay subscription — no REST polling needed. build_nip52_event grows four AIO custom tags on every published kind 31922/31923 event: - tickets_available — current remaining (omitted when amount_tickets is 0, the schema's "unlimited" sentinel, so clients can tell the difference between unlimited and sold-out) - tickets_sold — running count, always emitted (clients derive original_capacity = available + sold for progress bars) - tickets_price — price_per_ticket (0 means free) - tickets_currency — the currency string Tags are AIO additions outside the NIP-52 spec; spec-compliant clients MUST ignore unknown tags so this stays backwards-compatible. set_ticket_paid calls publish_or_delete_nostr_event after the counter update so the new state lands on relays. The whole sequence (counter update + republish) is wrapped in a per-event-id asyncio lock to address the existing # todo: lock and to ensure two paid invoices for the same event can't reorder the published state. Failures inside the Nostr publish are logged + swallowed by the existing wrapper, so a relay outage can never break the payment flow. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
188 lines
6.1 KiB
Python
188 lines
6.1 KiB
Python
from __future__ import annotations
|
|
|
|
import asyncio
|
|
from asyncio.tasks import create_task
|
|
|
|
from lnbits.core.models.users import UserNotifications
|
|
from lnbits.core.services.nostr import send_nostr_dm
|
|
from lnbits.core.services.notifications import (
|
|
send_email_notification,
|
|
send_user_notification,
|
|
)
|
|
from lnbits.settings import settings
|
|
from lnbits.utils.nostr import normalize_private_key, normalize_public_key
|
|
from lnurl import execute
|
|
from loguru import logger
|
|
|
|
from .crud import (
|
|
get_event,
|
|
get_event_tickets,
|
|
purge_unpaid_tickets,
|
|
update_event,
|
|
update_ticket,
|
|
)
|
|
from .models import Event, Ticket
|
|
from .nostr_hooks import publish_or_delete_nostr_event
|
|
|
|
DEFAULT_NOSTR_RELAYS = [
|
|
"wss://relay.damus.io",
|
|
"wss://relay.primal.net",
|
|
"wss://relay.nostr.band",
|
|
]
|
|
|
|
# Per-event lock: serializes the counter-update + Nostr republish for a
|
|
# single event_id so two paid invoices landing on the listener queue back-
|
|
# to-back can't reorder the published state. Lazy-populated; entries are
|
|
# left in memory for the lifetime of the process (cheap — one asyncio.Lock
|
|
# object per event ever sold).
|
|
_event_paid_locks: dict[str, asyncio.Lock] = {}
|
|
|
|
|
|
def _event_paid_lock(event_id: str) -> asyncio.Lock:
|
|
lock = _event_paid_locks.get(event_id)
|
|
if lock is None:
|
|
lock = asyncio.Lock()
|
|
_event_paid_locks[event_id] = lock
|
|
return lock
|
|
|
|
|
|
async def set_ticket_paid(ticket: Ticket) -> Ticket:
|
|
if ticket.paid:
|
|
return ticket
|
|
|
|
async with _event_paid_lock(ticket.event):
|
|
ticket.paid = True
|
|
await update_ticket(ticket)
|
|
|
|
event = await get_event(ticket.event)
|
|
assert event, "Couldn't get event from ticket being paid"
|
|
event.sold += 1
|
|
event.amount_tickets -= 1
|
|
await update_event(event)
|
|
|
|
# Republish the NIP-52 calendar event so connected clients see
|
|
# the new tickets_available / tickets_sold counters via their
|
|
# existing relay subscription. Failures are logged + swallowed
|
|
# inside publish_or_delete_nostr_event so a Nostr outage doesn't
|
|
# break the payment flow.
|
|
await publish_or_delete_nostr_event(event)
|
|
|
|
return ticket
|
|
|
|
|
|
def send_ticket_notification_in_background(ticket: Ticket) -> None:
|
|
create_task(_send_ticket_notification(ticket))
|
|
|
|
|
|
async def _send_ticket_notification(ticket: Ticket) -> None:
|
|
event = await get_event(ticket.event)
|
|
if not event:
|
|
logger.warning(f"Event {ticket.event} not found for ticket notification.")
|
|
return
|
|
|
|
subject, message = _ticket_notification_message(ticket, event)
|
|
updated = False
|
|
|
|
if (
|
|
event.extra.email_notifications
|
|
and settings.lnbits_email_notifications_enabled
|
|
and ticket.email
|
|
):
|
|
try:
|
|
await send_email_notification([ticket.email], message, subject)
|
|
ticket.extra.email_notification_sent = True
|
|
updated = True
|
|
except Exception as exc:
|
|
logger.warning(f"Failed to email ticket {ticket.id}: {exc}")
|
|
|
|
if (
|
|
event.extra.nostr_notifications
|
|
and settings.is_nostr_notifications_configured()
|
|
and ticket.extra.nostr_identifier
|
|
):
|
|
try:
|
|
await _send_nostr_ticket_notification(
|
|
ticket.extra.nostr_identifier, message
|
|
)
|
|
ticket.extra.nostr_notification_sent = True
|
|
updated = True
|
|
except Exception as exc:
|
|
logger.warning(f"Failed to send nostr DM for ticket {ticket.id}: {exc}")
|
|
|
|
if updated:
|
|
await update_ticket(ticket)
|
|
|
|
|
|
async def resend_ticket_email_notification(ticket: Ticket) -> Ticket:
|
|
event = await get_event(ticket.event)
|
|
if not event:
|
|
raise ValueError("Event does not exist.")
|
|
if not settings.lnbits_email_notifications_enabled:
|
|
raise ValueError("Email notifications are not enabled.")
|
|
if not ticket.email:
|
|
raise ValueError("Ticket does not have an email address.")
|
|
|
|
subject, message = _ticket_notification_message(ticket, event)
|
|
await send_email_notification([ticket.email], message, subject)
|
|
ticket.extra.email_notification_sent = True
|
|
return await update_ticket(ticket)
|
|
|
|
|
|
def _ticket_notification_message(ticket: Ticket, event: Event) -> tuple[str, str]:
|
|
ticket_url = _ticket_url(ticket)
|
|
subject = (
|
|
event.extra.notification_subject.strip()
|
|
or f"Your ticket for '{event.name}' is ready"
|
|
)
|
|
body = (
|
|
event.extra.notification_body.strip()
|
|
or f"Your ticket for '{event.name}' is ready."
|
|
)
|
|
|
|
return subject, f"{body}\n\nOpen it here: {ticket_url}"
|
|
|
|
|
|
async def _send_nostr_ticket_notification(identifier: str, message: str) -> None:
|
|
if "@" in identifier:
|
|
await send_user_notification(
|
|
UserNotifications(nostr_identifier=identifier),
|
|
message,
|
|
"text_message",
|
|
)
|
|
return
|
|
|
|
private_key = normalize_private_key(settings.lnbits_nostr_notifications_private_key)
|
|
public_key = normalize_public_key(identifier)
|
|
await send_nostr_dm(private_key, public_key, message, DEFAULT_NOSTR_RELAYS)
|
|
|
|
|
|
def _ticket_url(ticket: Ticket) -> str:
|
|
base_url = (ticket.extra.ticket_base_url or settings.lnbits_baseurl).rstrip("/")
|
|
return f"{base_url}/events/ticket/{ticket.id}"
|
|
|
|
|
|
async def refund_tickets(event_id: str):
|
|
"""
|
|
Refund tickets for an event that has not met the minimum ticket requirement.
|
|
This function should be called when the event is closed and the minimum ticket
|
|
condition is not met.
|
|
"""
|
|
await purge_unpaid_tickets(event_id)
|
|
tickets = await get_event_tickets(event_id)
|
|
|
|
if not tickets:
|
|
return
|
|
|
|
for ticket in tickets:
|
|
if ticket.extra.refunded:
|
|
continue
|
|
if ticket.paid and ticket.extra.refund_address and ticket.extra.sats_paid:
|
|
try:
|
|
res = await execute(
|
|
ticket.extra.refund_address, str(ticket.extra.sats_paid)
|
|
)
|
|
if res:
|
|
ticket.extra.refunded = True
|
|
await update_ticket(ticket)
|
|
except Exception as e:
|
|
logger.error(f"Error refunding ticket {ticket.id}: {e}")
|