initial commit — cassandra v0.1

Containerised macro-strategy dashboard: 4-panel web UI (indicators,
portfolio, flash news, AI strategic log), MariaDB store, hourly
ingestion jobs, OpenRouter-backed AI analysis.

Ports the four prototype scripts in the parent dir (market_pulse,
flash_news, trading212, strategic_log) into async services backed by a
persistent DB and served via FastAPI + Jinja2 + HTMX. APScheduler runs
as a separate compose service for crash-safety and easier restarts.

Portfolio composition + position names come live from Trading 212;
news per-ticker headlines reuse those names. Tone (NOVICE/INTERMEDIATE/
PRO) and analysis style (DRY/SPECULATIVE) are env-configurable and
stored on each log row so historical entries show what produced them.

Default model is deepseek/deepseek-v4-flash (overridable via env).
Light/dark theme toggle, sans-serif for prose surfaces, monospace for
data. Bearer-token auth, OpenRouter monthly cost cap, RSS feeds auto-
disabled on consecutive failures.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Giorgio Gilestro 2026-05-15 21:56:10 +01:00
commit a10409c02b
61 changed files with 4890 additions and 0 deletions

View file

@ -0,0 +1,40 @@
"""default.toml and portfolio.toml must load cleanly into the expected shapes."""
from __future__ import annotations
import pytest
pytest.importorskip("pydantic_settings")
from pathlib import Path
from app.config import load_feeds, load_groups, load_presets
ROOT = Path(__file__).resolve().parent.parent
DEFAULT = ROOT / "config" / "default.toml"
PORTFOLIO = ROOT / "config" / "portfolio.toml"
def test_default_groups_present():
g = load_groups(DEFAULT, PORTFOLIO)
for expected in ("equity", "mag7", "rates", "macro", "commodities", "fx", "pie"):
assert expected in g, f"missing group: {expected}"
# Every item is a 3-tuple of strings.
for items in g.values():
for sym, lab, note in items:
assert isinstance(sym, str) and sym
assert isinstance(lab, str)
assert isinstance(note, str)
def test_default_feeds_present():
f = load_feeds(DEFAULT, PORTFOLIO)
for cat in ("markets", "world", "energy", "asia"):
assert cat in f, f"missing feed category: {cat}"
assert all(name and url for name, url in f[cat])
def test_presets_thesis_present():
p = load_presets(DEFAULT, PORTFOLIO)
assert "thesis" in p
assert "hormuz" in p["thesis"]