Push builds have been RED on the lint step since ~build 209 from accumulated formatting drift. This is the mechanical cleanup: ruff format + ruff --fix (UP038 isinstance unions, SIM105 contextlib.suppress, UP031 f-strings, SIM115 tempfile context manager), shfmt -i 2 -ci, nixpkgs-fmt/statix/deadnix (merged attrsets, dropped unused lib args), yamllint, and shell quoting fixes in tests/lasuite-docs/setup_custom_tests.sh. No behaviour changes intended; lint: PASS, unit tests: 138 passed.
118 lines
4.9 KiB
Python
118 lines
4.9 KiB
Python
"""Shared ghost test helper — a cookie-aware Ghost Admin API client.
|
|
|
|
Ghost's Admin API authenticates a human session via a cookie (`ghost-admin-api-session`) set by
|
|
`POST /ghost/api/admin/session/`, and enforces a CSRF check requiring the `Origin` header to match
|
|
the site's configured `url` on state-changing requests. The shared `runner/harness/http` helpers are
|
|
deliberately cookie-less (stateless status+json), so this helper builds a stdlib
|
|
`urllib` opener with an `HTTPCookieProcessor` so the session cookie persists across
|
|
setup → login → create → read within the test process.
|
|
|
|
Auth path (version-independent — no `/v3/`/`/v5/` in the URL; Ghost negotiates the API version, and
|
|
the recipe under test may be any 5.x/6.x):
|
|
1. `POST /authentication/setup/` — creates the owner on a FRESH deploy (idempotent: a re-run finds
|
|
"setup already completed" and we ignore it).
|
|
2. `POST /session/` — establishes the admin session cookie (always done, so the client is
|
|
authenticated whether or not THIS process ran setup).
|
|
3. `POST /posts/?source=html` / `GET /posts/<id>/` — create + read back.
|
|
|
|
Admin credentials are class-B run-scoped (deterministic within a run; the whole app — DB + secrets —
|
|
is destroyed at teardown). Password is ≥10 chars per Ghost's setup requirement.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import http.cookiejar
|
|
import json
|
|
import ssl
|
|
import urllib.error
|
|
import urllib.request
|
|
|
|
# Per-run *.ci.commoninternet.net domains use the operator wildcard cert via Traefik file provider;
|
|
# the real-cert check is done once in the generic install assertion, so content/API calls skip the
|
|
# chain check (same rationale as runner/harness/http._CTX).
|
|
_CTX = ssl.create_default_context()
|
|
_CTX.check_hostname = False
|
|
_CTX.verify_mode = ssl.CERT_NONE
|
|
|
|
ADMIN_NAME = "CCCI Admin"
|
|
ADMIN_EMAIL = "ccci-admin@ccci.example.com"
|
|
ADMIN_PW = "Ccci-Test-Pw-2026!" # >=10 chars (Ghost setup requirement)
|
|
BLOG_TITLE = "CCCI Test Blog"
|
|
|
|
|
|
def _json(raw: bytes) -> object | None:
|
|
try:
|
|
return json.loads(raw)
|
|
except (json.JSONDecodeError, ValueError):
|
|
return None
|
|
|
|
|
|
class GhostAdmin:
|
|
def __init__(self, domain: str):
|
|
self.base = f"https://{domain}/ghost/api/admin"
|
|
self.origin = f"https://{domain}"
|
|
self._jar = http.cookiejar.CookieJar()
|
|
self._opener = urllib.request.build_opener(
|
|
urllib.request.HTTPCookieProcessor(self._jar),
|
|
urllib.request.HTTPSHandler(context=_CTX),
|
|
)
|
|
|
|
def req(self, method: str, path: str, body: dict | None = None, timeout: int = 60):
|
|
url = f"{self.base}{path}"
|
|
data = json.dumps(body).encode() if body is not None else None
|
|
req = urllib.request.Request(url, data=data, method=method)
|
|
if data is not None:
|
|
req.add_header("Content-Type", "application/json")
|
|
# CSRF: Ghost requires Origin to match the configured site url on state-changing requests.
|
|
req.add_header("Origin", self.origin)
|
|
try:
|
|
with self._opener.open(req, timeout=timeout) as resp:
|
|
return resp.getcode(), _json(resp.read())
|
|
except urllib.error.HTTPError as e:
|
|
return e.code, _json(e.read())
|
|
except Exception as e: # noqa: BLE001 — transport-level: surface as status 0
|
|
return 0, {"transport_error": str(e)}
|
|
|
|
def ensure_authenticated(self) -> None:
|
|
# Ensure the owner exists (fresh deploy → 201; already set up → 4xx, ignored).
|
|
self.req(
|
|
"POST",
|
|
"/authentication/setup/",
|
|
{
|
|
"setup": [
|
|
{
|
|
"name": ADMIN_NAME,
|
|
"email": ADMIN_EMAIL,
|
|
"password": ADMIN_PW,
|
|
"blogTitle": BLOG_TITLE,
|
|
}
|
|
]
|
|
},
|
|
)
|
|
# Always establish a fresh admin session (cookie persists in self._jar).
|
|
status, body = self.req(
|
|
"POST", "/session/", {"username": ADMIN_EMAIL, "password": ADMIN_PW}
|
|
)
|
|
assert status in (
|
|
200,
|
|
201,
|
|
), f"ghost admin session login failed: HTTP {status}, body={body!r}"
|
|
|
|
def create_post(self, title: str, html: str) -> dict:
|
|
status, body = self.req(
|
|
"POST",
|
|
"/posts/?source=html",
|
|
{"posts": [{"title": title, "html": html, "status": "published"}]},
|
|
)
|
|
assert status in (200, 201), f"create post failed: HTTP {status}, body={body!r}"
|
|
posts = (body or {}).get("posts") or []
|
|
assert posts and posts[0].get("id"), f"create post returned no id: {body!r}"
|
|
return posts[0]
|
|
|
|
def get_post(self, post_id: str) -> dict:
|
|
status, body = self.req("GET", f"/posts/{post_id}/?formats=html")
|
|
assert status == 200, f"read post failed: HTTP {status}, body={body!r}"
|
|
posts = (body or {}).get("posts") or []
|
|
assert posts, f"read post returned empty: {body!r}"
|
|
return posts[0]
|