brand: rename product to "Read the Markets" (read.markets)
The product is now "Read the Markets" served at https://read.markets, with the app at https://app.read.markets. "Cassandra" survives only as the in-product AI persona (system prompt + "Ask Cassandra" chat label). Centralised the brand in app/branding.py: BRAND_NAME, BRAND_SHORT, DOMAIN, SITE_URL, APP_URL, EMAIL_FROM_DEFAULT. Jinja templates pull {{ BRAND_NAME }} via globals registered in templates_env.py; Python code reads branding.BRAND_NAME directly. The future-rename surface is now a one-liner. Updated: FastAPI app title, every page title (dashboard, news, log, settings, upload, login, verify), header brand div, auth-card brands, OTP email subject + HTML + plain-text bodies (incl. uppercase header tag), OpenRouter X-Title + HTTP-Referer attribution headers, README. Email tests now assert against branding.BRAND_NAME rather than the literal string. Internal identifiers deliberately kept on the legacy "cassandra" name to avoid invalidating live sessions / advisory locks / configs: cookies (cassandra_session, cassandra_pending) + itsdangerous salts, MariaDB GET_LOCK keys, CASSANDRA_TOKEN env var, cassandra.css filename, pyproject package name, localStorage prefs, outbound User-Agent strings. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
parent
9759080134
commit
824d849c63
15 changed files with 82 additions and 39 deletions
|
|
@ -3,7 +3,7 @@
|
|||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>{% block title %}Cassandra{% endblock %}</title>
|
||||
<title>{% block title %}{{ BRAND_NAME }}{% endblock %}</title>
|
||||
{# Apply saved theme before stylesheet renders to avoid a flash. #}
|
||||
<script>
|
||||
(function() {
|
||||
|
|
@ -137,7 +137,7 @@
|
|||
<body>
|
||||
<div class="app">
|
||||
<header class="app-header">
|
||||
<div class="brand">Cassandra</div>
|
||||
<div class="brand">{{ BRAND_NAME }}</div>
|
||||
<nav>
|
||||
<a href="/" class="{% if request.url.path == '/' %}active{% endif %}">Dashboard</a>
|
||||
<a href="/news" class="{% if request.url.path == '/news' %}active{% endif %}">News</a>
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
{% extends "base.html" %}
|
||||
{% block title %}Cassandra · Dashboard{% endblock %}
|
||||
{% block title %}{{ BRAND_NAME }} · Dashboard{% endblock %}
|
||||
|
||||
{% block main %}
|
||||
<div id="dash-header-container"
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
{% extends "base.html" %}
|
||||
{% block title %}Cassandra · Strategic Log{% endblock %}
|
||||
{% block title %}{{ BRAND_NAME }} · Strategic Log{% endblock %}
|
||||
|
||||
{% block main %}
|
||||
<section class="panel log-page" style="grid-column: 1 / -1;">
|
||||
|
|
|
|||
|
|
@ -3,7 +3,7 @@
|
|||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>Cassandra · Sign in</title>
|
||||
<title>{{ BRAND_NAME }} · Sign in</title>
|
||||
<script>
|
||||
(function() {
|
||||
try { document.documentElement.dataset.theme = localStorage.getItem('cassandra.theme') || 'dark'; }
|
||||
|
|
@ -15,7 +15,7 @@
|
|||
<body>
|
||||
<div class="auth-shell">
|
||||
<div class="auth-card">
|
||||
<div class="auth-card__brand">Cassandra</div>
|
||||
<div class="auth-card__brand">{{ BRAND_NAME }}</div>
|
||||
<div class="auth-card__hint">sign in with email</div>
|
||||
|
||||
{% if referrer_present %}
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
{% extends "base.html" %}
|
||||
{% block title %}Cassandra · News{% endblock %}
|
||||
{% block title %}{{ BRAND_NAME }} · News{% endblock %}
|
||||
|
||||
{% block main %}
|
||||
<section class="panel" style="grid-column: 1 / -1;">
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
{% extends "base.html" %}
|
||||
{% block title %}Cassandra · Settings{% endblock %}
|
||||
{% block title %}{{ BRAND_NAME }} · Settings{% endblock %}
|
||||
|
||||
{% block main %}
|
||||
<section class="panel" style="grid-column: 1 / -1; max-width: 760px; margin: 0 auto;">
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
{% extends "base.html" %}
|
||||
{% block title %}Cassandra · Import Portfolio{% endblock %}
|
||||
{% block title %}{{ BRAND_NAME }} · Import Portfolio{% endblock %}
|
||||
|
||||
{% block main %}
|
||||
<section class="panel" style="grid-column: 1 / -1; max-width: 760px; margin: 0 auto;">
|
||||
|
|
@ -12,10 +12,10 @@
|
|||
<p style="color: var(--muted); font-size: 12.5px; margin: 0 0 14px; line-height: 1.6;">
|
||||
Export your pie from the T212 web app
|
||||
(<span class="neu">Trading 212 → Investing → Your Pie → ⋯ → Export</span>)
|
||||
and drop the CSV here. Cassandra resolves each Slice to its Yahoo
|
||||
ticker; the parsed pie is kept in <em>this browser's localStorage</em>
|
||||
only. The server learns just which tickers exist (anonymously) so it
|
||||
can fetch their prices.
|
||||
and drop the CSV here. Each Slice is resolved to its Yahoo ticker;
|
||||
the parsed pie is kept in <em>this browser's localStorage</em> only.
|
||||
The server learns just which tickers exist (anonymously) so it can
|
||||
fetch their prices.
|
||||
</p>
|
||||
|
||||
<form id="upload-form" autocomplete="off">
|
||||
|
|
|
|||
|
|
@ -3,7 +3,7 @@
|
|||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>Cassandra · Verify email</title>
|
||||
<title>{{ BRAND_NAME }} · Verify email</title>
|
||||
<script>
|
||||
(function() {
|
||||
try { document.documentElement.dataset.theme = localStorage.getItem('cassandra.theme') || 'dark'; }
|
||||
|
|
@ -15,7 +15,7 @@
|
|||
<body>
|
||||
<div class="auth-shell">
|
||||
<div class="auth-card">
|
||||
<div class="auth-card__brand">Cassandra</div>
|
||||
<div class="auth-card__brand">{{ BRAND_NAME }}</div>
|
||||
<div class="auth-card__hint">verify your email</div>
|
||||
|
||||
<p class="auth-card__lede">
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue