Adds two managed system overlays (l4d2center-maps, cedapug-maps) that fetch curated map archives from upstream sources and reconcile addons symlinks for non-Steam maps. A daily systemd timer enqueues a coalesced refresh_global_overlays worker job; downloads, extraction, and rebuilds run in the existing job worker and surface in the job log UI. Schema: GlobalOverlaySource / GlobalOverlayItem / GlobalOverlayItemFile plus nullable Job.user_id so system jobs render as "system" in the UI. The new builder reconciles symlinks against the per-source vpk cache and leaves foreign symlinks untouched. Initialize-time guard refuses to mount a partial overlay if any expected vpk is missing from cache. Refresh service uses shutil.move to handle EXDEV when /tmp and the cache live on different filesystems. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
60 lines
2 KiB
Python
60 lines
2 KiB
Python
import os
|
|
|
|
import click
|
|
from sqlalchemy.exc import IntegrityError
|
|
from sqlalchemy import select
|
|
|
|
from l4d2web.auth import hash_password
|
|
from l4d2web.db import session_scope
|
|
from l4d2web.models import User
|
|
|
|
|
|
@click.command("promote-admin")
|
|
@click.argument("username")
|
|
def promote_admin(username: str) -> None:
|
|
with session_scope() as db:
|
|
user = db.scalar(select(User).where(User.username == username))
|
|
if user is None:
|
|
raise click.ClickException("user not found")
|
|
user.admin = True
|
|
|
|
|
|
@click.command("create-user")
|
|
@click.argument("username")
|
|
@click.option("--admin", is_flag=True, default=False)
|
|
def create_user(username: str, admin: bool) -> None:
|
|
password = os.getenv("LEFT4ME_ADMIN_PASSWORD")
|
|
if password is None:
|
|
password = click.prompt("Password", hide_input=True, confirmation_prompt=True)
|
|
if password == "":
|
|
raise click.ClickException("password must not be empty")
|
|
|
|
try:
|
|
with session_scope() as db:
|
|
existing = db.scalar(select(User).where(User.username == username))
|
|
if existing is not None:
|
|
raise click.ClickException("user already exists")
|
|
db.add(User(username=username, password_digest=hash_password(password), admin=admin))
|
|
except IntegrityError as exc:
|
|
raise click.ClickException("user already exists") from exc
|
|
|
|
click.echo(f"created user {username}")
|
|
|
|
|
|
@click.command("refresh-global-overlays")
|
|
def refresh_global_overlays_command() -> None:
|
|
from l4d2web.services.global_overlays import (
|
|
ensure_global_overlays,
|
|
enqueue_refresh_global_overlays,
|
|
)
|
|
|
|
with session_scope() as db:
|
|
ensure_global_overlays(db)
|
|
job = enqueue_refresh_global_overlays(db, user_id=None)
|
|
click.echo(f"queued refresh_global_overlays job #{job.id}")
|
|
|
|
|
|
def register_cli(app) -> None:
|
|
app.cli.add_command(promote_admin)
|
|
app.cli.add_command(create_user)
|
|
app.cli.add_command(refresh_global_overlays_command)
|