feat: initial shared library extraction
Contains shared infrastructure for all coop services: - shared/ (factory, urls, user_loader, context, internal_api, jinja_setup) - models/ (User, Order, Calendar, Ticket, Product, Ghost CMS) - db/ (SQLAlchemy async session, base) - suma_browser/app/ (csrf, middleware, errors, authz, redis_cacher, payments, filters, utils) - suma_browser/templates/ (shared base layouts, macros, error pages) - static/ (CSS, JS, fonts, images) - alembic/ (database migrations) - config/ (app-config.yaml) - editor/ (Lexical editor Node.js build) - requirements.txt Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
58
suma_browser/app/middleware.py
Normal file
58
suma_browser/app/middleware.py
Normal file
@@ -0,0 +1,58 @@
|
||||
|
||||
def register(app):
|
||||
import json
|
||||
from typing import Any
|
||||
|
||||
def _decode_headers(scope) -> dict[str, str]:
|
||||
out = {}
|
||||
for k, v in scope.get("headers", []):
|
||||
try:
|
||||
out[k.decode("latin1")] = v.decode("latin1")
|
||||
except Exception:
|
||||
out[repr(k)] = repr(v)
|
||||
return out
|
||||
|
||||
def _safe(obj: Any):
|
||||
# make scope json-serialisable; fall back to repr()
|
||||
try:
|
||||
json.dumps(obj)
|
||||
return obj
|
||||
except Exception:
|
||||
return repr(obj)
|
||||
|
||||
class ScopeDumpMiddleware:
|
||||
def __init__(self, app, *, log_bodies: bool = False):
|
||||
self.app = app
|
||||
self.log_bodies = log_bodies # keep False; bodies aren't needed for routing
|
||||
|
||||
async def __call__(self, scope, receive, send):
|
||||
if scope["type"] in ("http", "websocket"):
|
||||
# Build a compact view of keys relevant to routing
|
||||
scope_view = {
|
||||
"type": scope.get("type"),
|
||||
"asgi": scope.get("asgi"),
|
||||
"http_version": scope.get("http_version"),
|
||||
"scheme": scope.get("scheme"),
|
||||
"method": scope.get("method"),
|
||||
"server": scope.get("server"),
|
||||
"client": scope.get("client"),
|
||||
"root_path": scope.get("root_path"),
|
||||
"path": scope.get("path"),
|
||||
"raw_path": scope.get("raw_path").decode("latin1") if scope.get("raw_path") else None,
|
||||
"query_string": scope.get("query_string", b"").decode("latin1"),
|
||||
"headers": _decode_headers(scope),
|
||||
}
|
||||
|
||||
print("\n=== ASGI SCOPE (routing) ===")
|
||||
print(json.dumps({_safe(k): _safe(v) for k, v in scope_view.items()}, indent=2))
|
||||
print("=== END SCOPE ===\n", flush=True)
|
||||
|
||||
return await self.app(scope, receive, send)
|
||||
|
||||
# wrap LAST so you see what hits Quart
|
||||
#app.asgi_app = ScopeDumpMiddleware(app.asgi_app)
|
||||
|
||||
|
||||
from hypercorn.middleware import ProxyFixMiddleware
|
||||
# trust a single proxy hop; use legacy X-Forwarded-* headers
|
||||
app.asgi_app = ProxyFixMiddleware(app.asgi_app, mode="legacy", trusted_hops=1)
|
||||
Reference in New Issue
Block a user