5 Commits

Author SHA1 Message Date
jaandrle ed632ad3fb Episodes (no paging) 2026-06-12 14:55:38 +02:00
jaandrle 5a0a2de0f0 🐛 Fixes API development 2026-06-12 13:45:24 +02:00
jaandrle bcb51d5397 Adds basic fetch
- 🐛 cors
- 🐛 `.env.js`/`ENV`
2026-06-11 16:37:10 +02:00
jaandrle 97983aff87 Adds api 2026-06-11 15:07:51 +02:00
jaandrle c000517fd1 🔤 Adds plan 2026-06-11 15:07:17 +02:00
14 changed files with 403 additions and 6 deletions
+39
View File
@@ -0,0 +1,39 @@
# Basic Episodes Fetch
## Overview
This plan focuses on adding a minimal **episodesfetch** feature that pulls podcast episodes from the CommaFeed backend
using environment variables for configuration.
The plan is derived from:
- `docs/dev/README.md` (API endpoints, authentication)
- `src/` structure and existing route system
- Build scripts in `bs/` only linting is required at this stage.
## Tasks
1. **API client** (DONE)
- Adds `src/api` folder to the project structure (update also @src/README.md).
- Implement a lightweight wrapper around `fetch` in `src/api/fetchAPI.ts`.
- Build `authHeader()` that returns the BasicAuth header using for now hardcoded vars.
1. **Episode service** (DONE)
- Add `src/api/episodes.ts` with `getEpisodes(feedId)` and `getEpisode(id)`.
- Add data types
- Use `/rest/feed/entries?feed_id=${feedId}` endpoint.
1. **Page update** (TODO)
- Add fetching all episodes
- Update `<app-episodes>` to reflect loading state(s).
- Add episode list item component.
- Update `<app-episodes>` to render the episode list received from the route context.
1. **Testing** (TODO)
- Add unit tests for `api/*.ts` using project test setup.
- Add unit test for created component(s)
1. **Linting** (TODO)
- Ensure all new files satisfy the lint rule defined in `bs/dev/lint`.
## Expected Output
- Visiting `/episodes/` will display a list of episodes fetched from CommaFeed.
- Episodes are identified by their `id` and displayed with title, publish date, and an audio link if available.
- Authentication uses the env variables; no credentials are hardcoded in source.
---
> **Note**: This plan intentionally keeps external dependencies minimal. It can be extended later for pagination, caching, or offline support.
+3
View File
@@ -2,3 +2,6 @@
- [x] initial setup (`bs`, app structure, …) - [x] initial setup (`bs`, app structure, …)
- [x] adds routing support - [x] adds routing support
- [ ] [episodes basic fetch](./plans/plan-basic-episodes.md)
- [ ] :bug: cors
- [ ] :bug: `.env.js`/`ENV`
+127
View File
@@ -24,6 +24,7 @@
"@web/test-runner": "~0.20", "@web/test-runner": "~0.20",
"babel-plugin-template-html-minifier": "~4.1", "babel-plugin-template-html-minifier": "~4.1",
"deepmerge": "~4.3", "deepmerge": "~4.3",
"koa-proxies": "^0.12.4",
"rollup": "~4.61", "rollup": "~4.61",
"rollup-plugin-esbuild": "~6.2", "rollup-plugin-esbuild": "~6.2",
"rollup-plugin-workbox": "~8.1", "rollup-plugin-workbox": "~8.1",
@@ -6034,6 +6035,13 @@
"node": ">= 0.6" "node": ">= 0.6"
} }
}, },
"node_modules/eventemitter3": {
"version": "4.0.7",
"resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz",
"integrity": "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==",
"dev": true,
"license": "MIT"
},
"node_modules/events-universal": { "node_modules/events-universal": {
"version": "1.0.1", "version": "1.0.1",
"resolved": "https://registry.npmjs.org/events-universal/-/events-universal-1.0.1.tgz", "resolved": "https://registry.npmjs.org/events-universal/-/events-universal-1.0.1.tgz",
@@ -6246,6 +6254,27 @@
"node": ">=6" "node": ">=6"
} }
}, },
"node_modules/follow-redirects": {
"version": "1.16.0",
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.16.0.tgz",
"integrity": "sha512-y5rN/uOsadFT/JfYwhxRS5R7Qce+g3zG97+JrtFZlC9klX/W5hD7iiLzScI4nZqUS7DNUdhPgw4xI8W2LuXlUw==",
"dev": true,
"funding": [
{
"type": "individual",
"url": "https://github.com/sponsors/RubenVerborgh"
}
],
"license": "MIT",
"engines": {
"node": ">=4.0"
},
"peerDependenciesMeta": {
"debug": {
"optional": true
}
}
},
"node_modules/for-each": { "node_modules/for-each": {
"version": "0.3.5", "version": "0.3.5",
"resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.5.tgz", "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.5.tgz",
@@ -6786,6 +6815,21 @@
"node": ">= 0.6" "node": ">= 0.6"
} }
}, },
"node_modules/http-proxy": {
"version": "1.18.1",
"resolved": "https://registry.npmjs.org/http-proxy/-/http-proxy-1.18.1.tgz",
"integrity": "sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ==",
"dev": true,
"license": "MIT",
"dependencies": {
"eventemitter3": "^4.0.0",
"follow-redirects": "^1.0.0",
"requires-port": "^1.0.0"
},
"engines": {
"node": ">=8.0.0"
}
},
"node_modules/http-proxy-agent": { "node_modules/http-proxy-agent": {
"version": "7.0.2", "version": "7.0.2",
"resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-7.0.2.tgz", "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-7.0.2.tgz",
@@ -7690,6 +7734,21 @@
"etag": "^1.8.1" "etag": "^1.8.1"
} }
}, },
"node_modules/koa-proxies": {
"version": "0.12.4",
"resolved": "https://registry.npmjs.org/koa-proxies/-/koa-proxies-0.12.4.tgz",
"integrity": "sha512-xxrEtN0e7s7/gNRoOMUltCbuIaCWqTQUTZNWQqet/8MoxSW0hG422lx2Al9FfYO3nCeA+b5c5/YmILRzavivDA==",
"dev": true,
"license": "MIT",
"dependencies": {
"http-proxy": "^1.18.1",
"path-match": "^1.2.4",
"uuid": "^8.3.2"
},
"peerDependencies": {
"koa": ">=2"
}
},
"node_modules/koa-send": { "node_modules/koa-send": {
"version": "5.0.1", "version": "5.0.1",
"resolved": "https://registry.npmjs.org/koa-send/-/koa-send-5.0.1.tgz", "resolved": "https://registry.npmjs.org/koa-send/-/koa-send-5.0.1.tgz",
@@ -8779,6 +8838,39 @@
"node": ">=8" "node": ">=8"
} }
}, },
"node_modules/path-match": {
"version": "1.2.4",
"resolved": "https://registry.npmjs.org/path-match/-/path-match-1.2.4.tgz",
"integrity": "sha512-UWlehEdqu36jmh4h5CWJ7tARp1OEVKGHKm6+dg9qMq5RKUTV5WJrGgaZ3dN2m7WFAXDbjlHzvJvL/IUpy84Ktw==",
"deprecated": "This package is archived and no longer maintained. For support, visit https://github.com/expressjs/express/discussions",
"dev": true,
"license": "MIT",
"dependencies": {
"http-errors": "~1.4.0",
"path-to-regexp": "^1.0.0"
}
},
"node_modules/path-match/node_modules/http-errors": {
"version": "1.4.0",
"resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.4.0.tgz",
"integrity": "sha512-oLjPqve1tuOl5aRhv8GK5eHpqP1C9fb+Ol+XTLjKfLltE44zdDbEdjPSbU7Ch5rSNsVFqZn97SrMmZLdu1/YMw==",
"dev": true,
"license": "MIT",
"dependencies": {
"inherits": "2.0.1",
"statuses": ">= 1.2.1 < 2"
},
"engines": {
"node": ">= 0.6"
}
},
"node_modules/path-match/node_modules/inherits": {
"version": "2.0.1",
"resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz",
"integrity": "sha512-8nWq2nLTAwd02jTqJExUYFSD/fKq6VH9Y/oG2accc/kdI0V98Bag8d5a4gi3XHz73rDWa2PvTtvcWYquKqSENA==",
"dev": true,
"license": "ISC"
},
"node_modules/path-parse": { "node_modules/path-parse": {
"version": "1.0.7", "version": "1.0.7",
"resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz",
@@ -8810,6 +8902,23 @@
"dev": true, "dev": true,
"license": "ISC" "license": "ISC"
}, },
"node_modules/path-to-regexp": {
"version": "1.9.0",
"resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-1.9.0.tgz",
"integrity": "sha512-xIp7/apCFJuUHdDLWe8O1HIkb0kQrOMb/0u6FXQjemHn/ii5LrIzU6bdECnsiTF/GjZkMEKg1xdiZwNqDYlZ6g==",
"dev": true,
"license": "MIT",
"dependencies": {
"isarray": "0.0.1"
}
},
"node_modules/path-to-regexp/node_modules/isarray": {
"version": "0.0.1",
"resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz",
"integrity": "sha512-D2S+3GLxWH+uhrNEcoh/fnmYeP8E8/zHl644d/jdA0g2uyXvy3sb0qxotE+ne0LtccHknQzWwZEzhak7oJ0COQ==",
"dev": true,
"license": "MIT"
},
"node_modules/path-type": { "node_modules/path-type": {
"version": "4.0.0", "version": "4.0.0",
"resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz",
@@ -9275,6 +9384,13 @@
"node": ">=0.10.0" "node": ">=0.10.0"
} }
}, },
"node_modules/requires-port": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz",
"integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==",
"dev": true,
"license": "MIT"
},
"node_modules/resolve": { "node_modules/resolve": {
"version": "1.22.11", "version": "1.22.11",
"resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.11.tgz", "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.11.tgz",
@@ -10730,6 +10846,17 @@
"browserslist": ">= 4.21.0" "browserslist": ">= 4.21.0"
} }
}, },
"node_modules/uuid": {
"version": "8.3.2",
"resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
"integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==",
"deprecated": "uuid@10 and below is no longer supported. For ESM codebases, update to uuid@latest. For CommonJS codebases, use uuid@11 (but be aware this version will likely be deprecated in 2028).",
"dev": true,
"license": "MIT",
"bin": {
"uuid": "dist/bin/uuid"
}
},
"node_modules/v8-to-istanbul": { "node_modules/v8-to-istanbul": {
"version": "9.3.0", "version": "9.3.0",
"resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-9.3.0.tgz", "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-9.3.0.tgz",
+1
View File
@@ -26,6 +26,7 @@
"@web/test-runner": "~0.20", "@web/test-runner": "~0.20",
"babel-plugin-template-html-minifier": "~4.1", "babel-plugin-template-html-minifier": "~4.1",
"deepmerge": "~4.3", "deepmerge": "~4.3",
"koa-proxies": "^0.12.4",
"rollup": "~4.61", "rollup": "~4.61",
"rollup-plugin-esbuild": "~6.2", "rollup-plugin-esbuild": "~6.2",
"rollup-plugin-workbox": "~8.1", "rollup-plugin-workbox": "~8.1",
+37
View File
@@ -0,0 +1,37 @@
import { fetchAPI } from "./fetchAPI.js";
export interface Episode {
id: string;
title: string;
content: string; // HTML
date: number;
feedId: string;
feedName: string;
}
export interface FeedEntriesResponse {
entries: Episode[];
}
/**
* Fetches all entries for a given feed.
* @param feedId - The ID of the feed.
*/
export async function getEpisodes(feedId: string= "all"): Promise<Episode[]> {
const response = await fetchAPI(`category/entries?id=${feedId}`);
if (!response.ok)
throw new Error(`Failed to fetch episodes: ${response.statusText}`);
const data = await response.json() as FeedEntriesResponse;
return data.entries || [];
}
/**
* Fetches a single episode by its ID.
* @param id - The ID of the episode.
*/
export async function getEpisode(id: string): Promise<Episode> {
const response = await fetchAPI(`feed/entry/${id}/`);
if (!response.ok)
throw new Error(`Failed to fetch episode: ${response.statusText}`);
return await response.json() as Episode;
}
+35
View File
@@ -0,0 +1,35 @@
const url_server = "";
const url_base = `${url_server}/rest`;
const [ username, password ] = ":".split(':');
/**
* Generates the Basic Authentication header.
*/
export function authHeader(): string {
const credentials = btoa(`${username}:${password}`);
return `Basic ${credentials}`;
}
/**
* A lightweight wrapper around fetch for the CommaFeed API.
* @param endpoint - The API endpoint (e.g., '/rest/feed/')
* @param options - Fetch options
*/
export async function fetchAPI(endpoint: string, options: RequestInit = {}): Promise<Response> {
const url = `${url_base}/${endpoint}`;
const auth = authHeader();
options = { ...options };
const { headers = {} } = options;
Reflect.deleteProperty(options, "headers");
return fetch(url, {
...options,
headers: {
Authorization: auth,
Accept: "application/json",
...headers,
}
});
}
+6 -1
View File
@@ -4,6 +4,11 @@ export const styles = css`
display: flex; display: flex;
flex-direction: column; flex-direction: column;
align-items: center; align-items: center;
justify-content: flex-start; --width: 33ch;
/* internal */
margin-inline: max(7.5ch, calc(50% - var(--width)));
}
.error {
color: red;
} }
`; `;
+39 -3
View File
@@ -1,14 +1,50 @@
import { LitElement, html } from "lit"; import { LitElement, html } from "lit";
import { customElement } from "lit/decorators.js"; import { customElement, state, property } from "lit/decorators.js";
import { styles } from "./index.css.js"; import { styles } from "./index.css.js";
import { getEpisodes, type Episode } from "../api/episodes.js";
import "../components/c-episode-list-card/index.js";
@customElement("app-episodes") @customElement("app-episodes")
export class AppEpisodes extends LitElement { export class AppEpisodes extends LitElement {
static override styles = styles; static override styles = styles;
@property({ type: String }) feedId: string = "all";
@state() private episodes: Episode[] = [];
@state() private loading = true;
@state() private error: string | null = null;
override async firstUpdated() {
await this.fetchEpisodes();
}
private async fetchEpisodes() {
this.loading = true;
this.error = null;
try {
this.episodes = await getEpisodes(this.feedId);
} catch (e) {
this.error = e instanceof Error ? e.message : "An unknown error occurred";
} finally {
this.loading = false;
}
}
override render() { override render() {
if (this.loading) {
return html`<p>Loading episodes...</p>`;
}
if (this.error) {
return html`<p class="error">Error: ${this.error}</p>`;
}
if (this.episodes.length === 0) {
return html`<p>No episodes found.</p>`;
}
return html` return html`
<div class="logo"></div> <h1>Episodes</h1>
<a href="/episodes/1">Episode 1</a> ${this.episodes.map((episode) => html`<c-episode-list-card .episode=${episode}></c-episode-list-card>`)}
`; `;
} }
} }
+1 -1
View File
@@ -9,7 +9,7 @@ export class AppHome extends LitElement {
return html` return html`
<h1>My app</h1> <h1>My app</h1>
<p>Hello world</p> <p>Hello world</p>
<a href="/episodes">Episode</a> <a href="/episodes">Episodes</a>
`; `;
} }
} }
@@ -0,0 +1,31 @@
import { css } from "lit";
export const styles = css`
:host {
display: grid;
grid-template-areas:
"title title"
"date feed"
"content content";
grid-template-rows:
fit-content
fit-content
1fr;
}
h2 {
grid-area: title;
text-wrap: balance;
text-wrap: pretty;
}
time, .feed {
color: #666;
}
time { grid-area: date; }
.feed { grid-area: feed; }
.content {
grid-area: content;
max-height: 3.5lh;
overflow: auto;
font-size: .9em;
}
`;
@@ -0,0 +1,33 @@
import { html } from "lit";
import { fixture, expect } from "@open-wc/testing";
import { type Episode } from "@/api/episodes.js";
import "./index.js";
describe("EpisodeListItem", () => {
const mockEpisode: Episode = {
id: "1",
title: "Test Episode",
date: 1781258812000,
content: "<b>Test content</b>",
feedId: "2",
feedName: "Test Feed",
};
it("renders correctly with episode data", async () => {
const el = await fixture(html`<episode-list-card .episode=${mockEpisode}></episode-list-card>`);
const title = el.shadowRoot!.querySelector("h2");
expect(title?.textContent).to.equal("Test Episode");
const time = el.shadowRoot!.querySelector("time");
expect(time?.textContent).to.contain("2023");
const feed = el.shadowRoot!.querySelector(".feed");
expect(feed?.textContent).to.equal("Test Feed");
const content = el.shadowRoot!.querySelector(".content");
expect(content?.innerHTML.trim()).to.equal("<b>Test content</b>");
});
it("renders empty when no episode is provided", async () => {
const el = await fixture(html`<episode-list-card></episode-list-card>`);
expect(el.shadowRoot!.innerHTML.trim()).to.equal("");
});
});
@@ -0,0 +1,35 @@
import { LitElement, html } from "lit";
import { property, customElement } from "lit/decorators.js";
import { type Episode } from "@/api/episodes.js";
import { styles } from "./index.css.js";
import { templateContent } from "lit/directives/template-content.js";
@customElement("c-episode-list-card")
export class EpisodeListCard extends LitElement {
static override styles = styles;
@property({ type: Object }) episode!: Episode;
override render() {
if (!this.episode) return html``;
const { id, title, content, date, feedId, feedName }= this.episode;
const dateString = new Date(date).toLocaleDateString();
const template = document.createElement("template");
try {
// @ts-expect-error 2551
template.setHTML(content);
} catch (e) {
template.innerText = content;
}
return html`
<h2>
<a href="/episodes/${id}">${title}</a>
</h2>
<time datetime="${date}">${dateString}</time>
<a class="feed" href="/feeds/${feedId}">${feedName}</a>
<div class="content">
${templateContent(template)}
</div>
`;
}
}
+2 -1
View File
@@ -34,7 +34,8 @@
"types": ["mocha"], "types": ["mocha"],
"lib": ["es2021", "dom", "DOM.Iterable"], "lib": ["es2021", "dom", "DOM.Iterable"],
"paths": { "paths": {
"@/*": ["./src/*"] "@/*": ["./src/*"],
"ENV": ["./.env.js"]
} }
}, },
"include": ["**/*.ts"] "include": ["**/*.ts"]
+14
View File
@@ -17,7 +17,21 @@ const target = mode !== "dist" // OR "src"
appIndex: pathDist("index.html"), appIndex: pathDist("index.html"),
}; };
import proxy from "koa-proxies";
import { users } from "./.env.js";
const middleware = [
proxy("/rest", {
target: "https://rss.jaandrle.cz",
headers: {
Authorization: "Basic " + Buffer.from(users[0]).toString("base64"),
},
logs: true,
changeOrigin: true,
}),
];
export default /** @type {import("@web/dev-server").DevServerConfig} */ ({ export default /** @type {import("@web/dev-server").DevServerConfig} */ ({
middleware,
open: "/", // SPA routing open: "/", // SPA routing
nodeResolve: { // Resolve bare module imports nodeResolve: { // Resolve bare module imports
exportConditions: ["browser", "development"], exportConditions: ["browser", "development"],