An alternative frontend for hot deals 🔥 https://rfd.davegallant.ca
  • TypeScript 29.4%
  • Vue 27.1%
  • JavaScript 25.4%
  • CSS 13.3%
  • HTML 2.5%
  • Other 2.3%
Find a file
2026-07-02 21:43:30 -04:00
.agents/skills/update-changelog Update skills directory 2026-04-11 19:06:38 -04:00
.github/workflows test(ci): expand coverage and add workflow 2026-06-28 08:36:27 -04:00
functions fix(worker): improve RFD fetch headers 2026-06-30 10:24:58 -04:00
public Migrate to Cloudflare Pages + Workers + KV (#429) 2026-06-22 22:22:30 -04:00
src fix(deals): refresh cached topics in UI 2026-06-29 08:29:50 -04:00
worker fix(worker): reduce refresh frequency 2026-07-02 21:43:30 -04:00
.browserslistrc Migrate to vuetify and vite (#251) 2025-08-29 20:57:27 -04:00
.editorconfig Add initial frontend and backend 2022-08-07 04:01:13 +00:00
.envrc Add various performance improvements 2026-02-14 08:44:40 -05:00
.eslintignore Cleanup frontend code 2026-02-15 20:27:12 -05:00
.gitignore Migrate to Cloudflare Pages + Workers + KV (#429) 2026-06-22 22:22:30 -04:00
AGENTS.md Migrate to Cloudflare Pages + Workers + KV (#429) 2026-06-22 22:22:30 -04:00
babel.config.js Add initial frontend and backend 2022-08-07 04:01:13 +00:00
CHANGELOG.md fix(worker): reduce refresh CPU usage 2026-06-28 08:29:45 -04:00
flake.lock Add various performance improvements 2026-02-14 08:44:40 -05:00
flake.nix Wrap InfoOverlay in <tbody> 2026-02-20 19:19:23 -05:00
index.html chore: release 0.7.0 — PWA accent color from orange to red 2026-05-22 09:02:50 -04:00
jsconfig.json Add initial frontend and backend 2022-08-07 04:01:13 +00:00
Justfile fix(ui): improve narrow mobile deal rows 2026-06-23 08:45:11 -04:00
Makefile Migrate to Cloudflare Pages + Workers + KV (#429) 2026-06-22 22:22:30 -04:00
package-lock.json Update vue monorepo (#387) 2026-06-22 23:06:48 -04:00
package.json fix(build): remove stale Vite plugin 2026-06-22 22:36:50 -04:00
README.md Update README to clarify Cloudflare Worker refresh 2026-06-28 08:40:24 -04:00
renovate.json Change Renovate schedule to run on the 1st of month 2026-06-06 07:29:34 -04:00
VERSION fix(worker): reduce refresh CPU usage 2026-06-28 08:29:45 -04:00
vite.config.mjs test(ci): expand coverage and add workflow 2026-06-28 08:36:27 -04:00
wrangler.toml Migrate to Cloudflare Pages + Workers + KV (#429) 2026-06-22 22:22:30 -04:00

rfd-fyi

This repository provides a simple, less-distracting frontend for Hot Deals posted on https://forums.redflagdeals.com.

The frontend is a Vite/Vue 3 app. Cloudflare Pages serves the static output, Pages Functions serve /topics.json, /html, and /admin/refresh from Cloudflare KV, and a scheduled Cloudflare Worker refreshes cached topics to avoid excessive requests to RedFlagDeals itself.

Architecture

flowchart TD
  Browser[Browser] -->|GET /| Pages[Cloudflare Pages static assets]
  Browser -->|GET /topics.json| TopicsFn[Pages Function: /topics.json]
  Browser -->|GET /html| HtmlFn[Pages Function: /html]

  TopicsFn -->|read topics.json| KV[(Cloudflare KV: TOPICS_KV)]
  HtmlFn -->|read topics.json| KV

  Cron[Cloudflare Cron Trigger<br/>every 5 minutes] --> RefreshWorker[Scheduled Worker: rfd-fyi-refresh]
  RefreshWorker -->|fetch topic pages| RFD[RedFlagDeals API]
  RefreshWorker -->|fetch redirect rules| Redirects[Redirect rules JSON]
  RefreshWorker -->|write topics.json| KV

  AdminPages[Manual refresh<br/>POST /admin/refresh] --> TopicsRefreshFn[Pages Function: /admin/refresh]
  AdminWorker[Manual refresh<br/>GET /refresh] --> RefreshWorker
  TopicsRefreshFn -->|fetch topic pages| RFD
  TopicsRefreshFn -->|fetch redirect rules| Redirects
  TopicsRefreshFn -->|write topics.json| KV

Cloudflare deployment

Install dependencies and log in to Cloudflare:

npm ci
npx wrangler login

Create a KV namespace and a preview namespace:

npx wrangler kv namespace create TOPICS_KV
npx wrangler kv namespace create TOPICS_KV --preview

Copy the returned namespace IDs into both:

  • wrangler.toml
  • worker/wrangler.toml

Build and deploy the Pages app:

npm run build
npm run pages:deploy
# or
npm run build && just deploy-pages

Deploy the scheduled refresh Worker:

npm run worker:deploy
# or
just deploy-worker

Deploy both:

just deploy

The Worker runs every 5 minutes and writes the latest topics to KV. Pages reads that cached JSON at /topics.json and renders a no-JavaScript view at /html.

Optional manual refresh endpoints:

# For the Pages /admin/refresh endpoint
npx wrangler pages secret put REFRESH_SECRET --project-name rfd-fyi
curl -X POST -H "Authorization: Bearer $REFRESH_SECRET" https://<your-pages-domain>/admin/refresh

# For the Worker /refresh endpoint
npx wrangler secret put REFRESH_SECRET --config worker/wrangler.toml
curl -H "Authorization: Bearer $REFRESH_SECRET" https://rfd-fyi-refresh.<your-subdomain>.workers.dev/refresh

Tests and checks

Run the unit test suite:

npm test -- --run

Run coverage:

npm run test:coverage -- --run

Run linting and a production build:

npm run lint
npm run build

GitHub Actions runs npm ci, tests, linting, and build on every push and pull request.

Local Development

Full Cloudflare Pages local dev

Run the Cloudflare Pages build locally, including Pages Functions:

npm run pages:dev

Wrangler serves the app at:

http://localhost:8788

Local Pages KV starts empty, so the app may initially show no deals. Seed local KV by calling the manual refresh endpoint in another shell:

curl -X POST -H "Authorization: Bearer dev" http://localhost:8788/admin/refresh

A successful refresh returns something like:

{"refreshed":1000}

After that, these local endpoints should return populated data:

http://localhost:8788/topics.json
http://localhost:8788/html

Refresh Worker local dev

To run the scheduled refresh Worker locally:

npm run worker:dev

Frontend-only Vite dev

For frontend-only Vite development:

npm run serve

The Vite dev server proxies /topics.json and /html to the configured production Pages origin by default, so it should show live deals without seeding local KV. Override with VITE_API_ORIGIN if needed:

VITE_API_ORIGIN=http://localhost:8788 npm run serve