Adds api

This commit is contained in:
2026-06-11 15:03:28 +02:00
parent c000517fd1
commit 97983aff87
4 changed files with 77 additions and 11 deletions
+5 -10
View File
@@ -10,21 +10,16 @@ The plan is derived from:
- Build scripts in `bs/` only linting is required at this stage. - Build scripts in `bs/` only linting is required at this stage.
## Tasks ## Tasks
1. **Environment handling** (TODO) 1. **API client** (DONE)
- Create a small helper (`@core/env.ts`) that reads `.env.ts` (or falls back to process.env).
- Expose constants: `COMMAFEED_URL`, `USERNAME`, `PASSWORD`.
1. **API client** (TODO)
- Adds `src/api` folder to the project structure (update also @src/README.md). - Adds `src/api` folder to the project structure (update also @src/README.md).
- Implement a lightweight wrapper around `fetch` in `src/api/index.ts`. - Implement a lightweight wrapper around `fetch` in `src/api/fetchAPI.ts`.
- Build `authHeader()` that returns the BasicAuth header using env vars. - Build `authHeader()` that returns the BasicAuth header using for now hardcoded vars.
1. **Episode service** (TODO) 1. **Episode service** (DONE)
- Add `src/api/episodes.ts` with `getEpisodes(feedId)` and `getEpisode(id)`. - Add `src/api/episodes.ts` with `getEpisodes(feedId)` and `getEpisode(id)`.
- Add data types - Add data types
- Use `/rest/feed/entries?feed_id=${feedId}` endpoint. - Use `/rest/feed/entries?feed_id=${feedId}` endpoint.
1. **Route integration** (TODO)
- In `app-episodes/routes.ts`, call the service during `enter()` to prefetch episodes.
- Pass fetched data via route context or set a global state (e.g., using Lit Labs signals).
1. **Page update** (TODO) 1. **Page update** (TODO)
- Add fetching all episodes
- Update `<app-episodes>` to reflect loading state(s). - Update `<app-episodes>` to reflect loading state(s).
- Add episode list item component. - Add episode list item component.
- Update `<app-episodes>` to render the episode list received from the route context. - Update `<app-episodes>` to render the episode list received from the route context.
+36
View File
@@ -0,0 +1,36 @@
import { fetchAPI } from "./fetchAPI.js";
export interface Episode {
id: string;
title: string;
published: string; // ISO string or similar
audio_url?: string;
description?: 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(`feed/entries?feed_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;
}
+34
View File
@@ -0,0 +1,34 @@
const url_server = "https://rss.jaandrle.cz";
const url_base = `${url_server}/rest`;
// @ts-expect-error S7016
import { users } from "ENV";
const [ username, password ] = users[0].split(':');
console.log(import.meta);
/**
* 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();
const headers = new Headers(options.headers);
headers.set('Authorization', auth);
return fetch(url, {
...options,
headers,
});
}
+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"]