⚡ Adds api
This commit is contained in:
@@ -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 Basic‑Auth header using env vars.
|
- Build `authHeader()` that returns the Basic‑Auth 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 pre‑fetch 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.
|
||||||
|
|||||||
@@ -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;
|
||||||
|
}
|
||||||
@@ -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
@@ -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"]
|
||||||
|
|||||||
Reference in New Issue
Block a user