All checks were successful
Build and Deploy / build-and-deploy (push) Successful in 5m35s
- Server sends sexp source text, client (sexp.js) renders everything - SexpExpr marker class for nested sexp composition in serialize() - sexp_page() HTML shell with data-mount="body" for full page loads - sexp_response() returns text/sexp for OOB/partial responses - ~app-body layout component replaces ~app-layout (no raw!) - ~rich-text is the only component using raw! (for CMS HTML content) - Fragment endpoints return text/sexp, auto-wrapped in SexpExpr - All _*_html() helpers converted to _*_sexp() returning sexp source - Head auto-hoist: sexp.js moves meta/title/link/script[ld+json] from rendered body to document.head automatically - Unknown components render warning box instead of crashing page - Component kwargs preserve AST for lazy rendering (fixes <> in kwargs) - Fix unterminated paren in events/sexp/tickets.sexpr Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
87 lines
3.0 KiB
Python
87 lines
3.0 KiB
Python
from __future__ import annotations
|
|
|
|
from quart import (
|
|
make_response, Blueprint, g, request
|
|
)
|
|
|
|
from shared.infrastructure.actions import call_action
|
|
from shared.infrastructure.data_client import fetch_data
|
|
from shared.browser.app.authz import require_admin
|
|
from shared.browser.app.utils.htmx import is_htmx_request
|
|
from shared.sexp.helpers import sexp_response
|
|
|
|
|
|
def register():
|
|
bp = Blueprint("page_admin", __name__)
|
|
|
|
@bp.get("/")
|
|
@require_admin
|
|
async def admin(**kwargs):
|
|
from shared.sexp.page import get_template_context
|
|
from sexp.sexp_components import render_cart_admin_page, render_cart_admin_oob
|
|
|
|
ctx = await get_template_context()
|
|
page_post = getattr(g, "page_post", None)
|
|
if not is_htmx_request():
|
|
html = await render_cart_admin_page(ctx, page_post)
|
|
return await make_response(html)
|
|
else:
|
|
sexp_src = await render_cart_admin_oob(ctx, page_post)
|
|
return sexp_response(sexp_src)
|
|
|
|
@bp.get("/payments/")
|
|
@require_admin
|
|
async def payments(**kwargs):
|
|
from shared.sexp.page import get_template_context
|
|
from sexp.sexp_components import render_cart_payments_page, render_cart_payments_oob
|
|
|
|
ctx = await get_template_context()
|
|
page_post = getattr(g, "page_post", None)
|
|
if not is_htmx_request():
|
|
html = await render_cart_payments_page(ctx, page_post)
|
|
return await make_response(html)
|
|
else:
|
|
sexp_src = await render_cart_payments_oob(ctx, page_post)
|
|
return sexp_response(sexp_src)
|
|
|
|
@bp.put("/payments/")
|
|
@require_admin
|
|
async def update_sumup(**kwargs):
|
|
"""Update SumUp credentials for this page (writes to blog's db_blog)."""
|
|
page_post = getattr(g, "page_post", None)
|
|
if not page_post:
|
|
return await make_response("Page not found", 404)
|
|
|
|
form = await request.form
|
|
merchant_code = (form.get("merchant_code") or "").strip()
|
|
api_key = (form.get("api_key") or "").strip()
|
|
checkout_prefix = (form.get("checkout_prefix") or "").strip()
|
|
|
|
payload = {
|
|
"container_type": "page",
|
|
"container_id": page_post.id,
|
|
"sumup_merchant_code": merchant_code,
|
|
"sumup_checkout_prefix": checkout_prefix,
|
|
}
|
|
if api_key:
|
|
payload["sumup_api_key"] = api_key
|
|
|
|
await call_action("blog", "update-page-config", payload=payload)
|
|
|
|
# Re-fetch page config to get fresh data
|
|
from types import SimpleNamespace
|
|
raw_pc = await fetch_data(
|
|
"blog", "page-config",
|
|
params={"container_type": "page", "container_id": page_post.id},
|
|
required=False,
|
|
)
|
|
g.page_config = SimpleNamespace(**raw_pc) if raw_pc else None
|
|
|
|
from shared.sexp.page import get_template_context
|
|
from sexp.sexp_components import render_cart_payments_panel
|
|
ctx = await get_template_context()
|
|
html = render_cart_payments_panel(ctx)
|
|
return sexp_response(html)
|
|
|
|
return bp
|