All checks were successful
Build and Deploy / build-and-deploy (push) Successful in 1m5s
Combines shared, blog, market, cart, events, federation, and account into a single repository. Eliminates submodule sync, sibling model copying at build time, and per-app CI orchestration. Changes: - Remove per-app .git, .gitmodules, .gitea, submodule shared/ dirs - Remove stale sibling model copies from each app - Update all 6 Dockerfiles for monorepo build context (root = .) - Add build directives to docker-compose.yml - Add single .gitea/workflows/ci.yml with change detection - Add .dockerignore for monorepo build context - Create __init__.py for federation and account (cross-app imports)
121 lines
4.6 KiB
Python
121 lines
4.6 KiB
Python
from __future__ import annotations
|
|
|
|
import hashlib
|
|
import re
|
|
from pathlib import Path
|
|
|
|
from quart import Quart, g, url_for
|
|
|
|
from shared.config import config
|
|
from shared.utils import host_url
|
|
|
|
from shared.browser.app.csrf import generate_csrf_token
|
|
from shared.browser.app.authz import has_access
|
|
from shared.browser.app.filters import register as register_filters
|
|
|
|
from .urls import blog_url, market_url, cart_url, events_url, federation_url, account_url, login_url, page_cart_url, market_product_url
|
|
|
|
|
|
def setup_jinja(app: Quart) -> None:
|
|
app.jinja_env.add_extension("jinja2.ext.do")
|
|
|
|
# --- template globals ---
|
|
app.add_template_global(generate_csrf_token, "csrf_token")
|
|
app.add_template_global(has_access, "has_access")
|
|
|
|
def level():
|
|
if not hasattr(g, "_level_counter"):
|
|
g._level_counter = 0
|
|
return g._level_counter
|
|
|
|
def level_up():
|
|
if not hasattr(g, "_level_counter"):
|
|
g._level_counter = 0
|
|
g._level_counter += 1
|
|
return ""
|
|
|
|
app.jinja_env.globals["level"] = level
|
|
app.jinja_env.globals["level_up"] = level_up
|
|
app.jinja_env.globals["menu_colour"] = "sky"
|
|
app.jinja_env.globals["app_name"] = app.name
|
|
|
|
select_colours = """
|
|
[.hover-capable_&]:hover:bg-yellow-300
|
|
aria-selected:bg-stone-500 aria-selected:text-white
|
|
[.hover-capable_&[aria-selected=true]:hover]:bg-orange-500"""
|
|
app.jinja_env.globals["select_colours"] = select_colours
|
|
|
|
nav_button = f"""justify-center cursor-pointer flex flex-row items-center gap-2 rounded bg-stone-200 text-black
|
|
{select_colours}"""
|
|
|
|
styles = {
|
|
"pill": """
|
|
inline-flex items-center px-3 py-1 rounded-full bg-stone-200 text-stone-700 text-sm
|
|
hover:bg-stone-300 hover:text-stone-900
|
|
focus:outline-none focus-visible:ring-2 focus-visible:ring-stone-400
|
|
""",
|
|
"tr": "odd:bg-slate-50 even:bg-white hover:bg-slate-100",
|
|
"action_button": "px-2 py-1 border rounded text-sm bg-sky-300 hover:bg-sky-400 flex gap-1 items-center",
|
|
"pre_action_button": "px-2 py-1 border rounded text-sm bg-green-200 hover:bg-green-300",
|
|
"cancel_button": "px-3 py-1.5 rounded-full text-sm border border-stone-300 text-stone-700 hover:bg-stone-100",
|
|
"list_container": "border border-stone-200 rounded-lg p-3 mb-3 bg-white space-y-3 bg-yellow-200",
|
|
"nav_button": f"{nav_button} p-3",
|
|
"nav_button_less_pad": f"{nav_button} p-2",
|
|
}
|
|
app.jinja_env.globals["styles"] = styles
|
|
|
|
def _asset_url(path: str) -> str:
|
|
def squash_double_slashes(url: str) -> str:
|
|
m = re.match(r"(?:[A-Za-z][\w+.-]*:)?//", url)
|
|
prefix = m.group(0) if m else ""
|
|
rest = re.sub(r"/+", "/", url[len(prefix):])
|
|
return prefix + rest
|
|
|
|
file_path = Path("static") / path
|
|
try:
|
|
digest = hashlib.md5(file_path.read_bytes()).hexdigest()[:8]
|
|
except Exception:
|
|
digest = "dev"
|
|
return squash_double_slashes(
|
|
f"{g.scheme}://{g.host}{g.root}/{url_for('static', filename=path, v=digest)}"
|
|
)
|
|
|
|
app.jinja_env.globals["asset_url"] = _asset_url
|
|
|
|
def site():
|
|
return {
|
|
"url": host_url(),
|
|
"logo": _asset_url("img/logo.jpg"),
|
|
"default_image": _asset_url("img/logo.jpg"),
|
|
"title": config()["title"],
|
|
}
|
|
|
|
app.jinja_env.globals["site"] = site
|
|
|
|
# cross-app URL helpers available in all templates
|
|
app.jinja_env.globals["blog_url"] = blog_url
|
|
app.jinja_env.globals["market_url"] = market_url
|
|
app.jinja_env.globals["cart_url"] = cart_url
|
|
app.jinja_env.globals["events_url"] = events_url
|
|
app.jinja_env.globals["federation_url"] = federation_url
|
|
app.jinja_env.globals["account_url"] = account_url
|
|
app.jinja_env.globals["login_url"] = login_url
|
|
app.jinja_env.globals["page_cart_url"] = page_cart_url
|
|
app.jinja_env.globals["market_product_url"] = market_product_url
|
|
|
|
# widget registry available in all templates
|
|
from shared.services.widget_registry import widgets as _widget_registry
|
|
app.jinja_env.globals["widgets"] = _widget_registry
|
|
|
|
# fragment composition helper — fetch HTML from another app's fragment API
|
|
from shared.infrastructure.fragments import fetch_fragment_cached
|
|
|
|
async def _fragment(app_name: str, fragment_type: str, ttl: int = 30, **params) -> str:
|
|
p = params if params else None
|
|
return await fetch_fragment_cached(app_name, fragment_type, params=p, ttl=ttl)
|
|
|
|
app.jinja_env.globals["fragment"] = _fragment
|
|
|
|
# register jinja filters
|
|
register_filters(app)
|