# News relaunch — link/redirect map + interim-routing plan Scope: stand up `/news` (the feed's new home) and cut `/home3` → `/` (hub) without breaking the feed, deep links, or SEO. Verified against the codebase 2026-06-28. Settled with the user + Codex; amendments folded in. **Next build = the extraction (§A, pure refactor).** ## A. Interim routing — no broken window, no duplicated impl The feed currently IS `routes/+page.svelte` (~1,065 lines: views, BottomNav, MoodNav, LanePicker, SavedFlyout, search). Don't clone it. Extract once, mount twice. 1. **Extract** the feed UI from `routes/+page.svelte` → `lib/components/NewsFeed.svelte` (verbatim move; URL base becomes `/news`). `parseView`/`urlForView` switch base `/` → `/news`. 2. **Mount at both** during transition: - `routes/news/+page.svelte` → `` - `routes/+page.svelte` → `` (interim; identical component) One implementation, both routes live, zero interim breakage. - **`/news` stays hidden during transition**: `X-Robots-Tag: noindex, follow` (so we don't publish a duplicate indexable feed). Removed at cutover, when `/news` is added to the sitemap. 3. **Parity test** at `/news` (every view, deep link, Back/Forward, account action) — see §F. 4. **Restyle** `NewsFeed` once (CD editorial + reusable card — replaces `ArticleCard`). 5. **Behavior fix** (separate, deliberate, post-parity): Latest stays chronological; Highlights gets prefs + geo scope. 6. **Cutover**: `routes/+page.svelte` → hub content (move from `home3`); add the legacy-query redirect shim (§C); drop `/news` noindex + add to sitemap; `/news` stays the feed. ## B. Link/redirect map Rule of thumb: **brand/logo/"home"/"back to UB" → `/` (hub)**; **anything "more news / browse / a feed view" → `/news`**; **feed-internal view URLs → `/news?…` base**; **`/home3` fallbacks → `/`**. ### B1. Brand / home / back → `/` (KEEP) - `Header.svelte:9` brand `/`; `account:103` brand `/`, `:108` back `/`; `admin:678` brand `/`, `:28` non-admin bounce `/`; `auth/verify:24` post-signin `/` (DECIDED: always hub; future optional start-screen setting), `:38` back `/`; `zen:59` brand `/`; `share.py:197,327` brand logo `/`, `:354` "Back to Upbeat Bytes" `/`; `api.py:781` unsubscribe back `/`. ### B2. News CTAs → `/news` (CHANGE) - `home3:198` "Read more good news" `/` → **`/news`** (the loop bug) - `share.py:210` "Explore Upbeat Bytes →" → **`/news`** - `share.py:332` "Browse more on Upbeat Bytes →" → **`/news`** ### B3. Feed-internal view URLs → `/news?…` base (CHANGE, via the extraction) - `routes/+page.svelte` (→ NewsFeed): `urlForView` builds `/?source=`,`/?tag=`,`/?view=` (`:39-41`), search `goto('/?q=')` (`:50`), clear `goto('/')` (`:689`), `urlForView('today')` returns `/` (`:344,503`) → all rebase to `/news`. - `account:196` Following `/?view=following` → **`/news?view=following`** - `share.py:30` source link `/?source={id}` → **`/news?source={id}`** ### B4. `/home3` fallbacks → `/` (CHANGE) - `HubBar:18` home nav `/home3` → **`/`**; `:28` brand `/home3` → **`/`** - `HubShell:17` back fallback `goto('/home3')` → **`/`** - `art:41`, `play:213` back fallback `goto('/home3')` → **`/`** ## C. Redirects - **Legacy root-query shim** on the new hub `/` — runs **before the hub fetches/renders** (no hub flash), `replaceState`: - `/?view=today` → `/news?view=highlights` - `/?view=latest` → `/news` (Latest is the default) - `?tag=`, `?source=`, `?q=`, other `?view=` → carry across to `/news?…` - `/news?view=today` remains an accepted **alias** (old links never break) - **`/home2`, `/home2.html`, `/home3`, `/home3.html` → `/` permanent (301)** (Caddy). A bare route delete would just serve the SPA fallback — redirect explicitly. ## D. Infra - **Caddy `@hidden`** (currently `/home2 /home3 /word /word.html /quote /quote.html /onthisday /onthisday.html /admin /admin.html`): - REMOVE `/word* /quote* /onthisday*` (indexable at launch) - `/home3*`,`/home2*` → 301 redirects (out of @hidden) - **`/news` carries `X-Robots-Tag: noindex, follow` until cutover**, then removed - KEEP `/admin*`; `/a/*` still routed to FastAPI - **Sitemap** (`api.py sitemap()`): raise `LIMIT 5000` → ~50000; gate on having a real summary (skip ~31 incomplete; ~+1,000 URLs); add static `/news` (at cutover) + `/art /play /word /quote /onthisday` (keep `/`,`/today`). Fix **`HEAD /sitemap.xml`** (currently 404). - **Head patcher** (`patch-static-heads.mjs`): add `/news` (title/desc/canonical/OG). - **PWA description** (`manifest.webmanifest` + `app.html` description/og/twitter): currently news-only — broaden to the hub (news + daily art + games + small resets). - **Footer**: ONE shared `Footer.svelte` — consistent core (**motto + Send feedback**) + a default **slot for per-section extras**. **`FeedbackModal` stays in the global layout**; only the layout's `` *markup* is removed (and HubShell's ``). **Coverage inventory** — the shared footer must be explicitly added to every public surface: **Hub, News, Play, Art, HubShell details, Account, Zen** (if dev-visible). Admin/auth get a deliberate **minimal** treatment (explicit, not by omission). ## E. Behavior (deliberate, post-parity) - **Latest** = newest accepted after safety/boundary exclusions; **stop passing `home`** (or add an explicit "Local first" lane) so Latest ≠ local-first. - **Highlights** = ranked around interests + geo scope dial (kept at launch). - `trackVisit()` stays global; `markBriefSeen()` only inside Highlights. ## F. Cutover checklist (rehearsal first, hidden) 1. **GSC review** (coverage / manual-actions / crawl) BEFORE rehearsal. 2. **Parity** at `/news`: each view (today/latest/following/tag/source/search/mood/topic), deep links, Back/Forward single-history, account actions (save/follow/hide/replace), PWA. 3. **Legacy redirects**: `/?view=today`→`/news?view=highlights`, `/?view=latest`→`/news`, `/?tag=…`,`/?source=…`,`/?q=…`→`/news?…`; `/home2*`,`/home3*`→`/`. 4. **SEO**: `/` 200 + indexable + canonical; `/a/*` 200 + self-canonical (unchanged); `/today` 200 indexable; `/news` noindex dropped + in sitemap; joy/art/play noindex removed + in sitemap; sitemap GET + HEAD 200. 5. **Caches**: anon Latest/Brief edge-cacheable (45s) intact; personalized private/no-store. SW is a kill-switch (no bump) — just verify. 6. **Promote** → live 200/301/canonical/cache checks; resubmit sitemap in GSC.