Files
indiekit-endpoint-youtube/index.js
Ricardo 2b8de8027b Initial commit: YouTube channel endpoint for Indiekit
Features:
- Display latest videos from any YouTube channel
- Live streaming status with animated badge
- Upcoming stream detection
- Admin dashboard with video grid
- Public JSON API for Eleventy integration
- Quota-efficient API usage (playlist method)
- Smart caching (5min videos, 1min live status)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-23 23:02:13 +01:00

89 lines
2.2 KiB
JavaScript

import express from "express";
import { fileURLToPath } from "node:url";
import path from "node:path";
import { dashboardController } from "./lib/controllers/dashboard.js";
import { videosController } from "./lib/controllers/videos.js";
import { channelController } from "./lib/controllers/channel.js";
import { liveController } from "./lib/controllers/live.js";
const __dirname = path.dirname(fileURLToPath(import.meta.url));
const protectedRouter = express.Router();
const publicRouter = express.Router();
const defaults = {
mountPath: "/youtube",
apiKey: process.env.YOUTUBE_API_KEY,
channelId: process.env.YOUTUBE_CHANNEL_ID,
channelHandle: process.env.YOUTUBE_CHANNEL_HANDLE,
cacheTtl: 300_000, // 5 minutes
liveCacheTtl: 60_000, // 1 minute for live status
limits: {
videos: 10,
},
};
export default class YouTubeEndpoint {
name = "YouTube channel endpoint";
constructor(options = {}) {
this.options = { ...defaults, ...options };
this.mountPath = this.options.mountPath;
}
get environment() {
return ["YOUTUBE_API_KEY", "YOUTUBE_CHANNEL_ID", "YOUTUBE_CHANNEL_HANDLE"];
}
get localesDirectory() {
return path.join(__dirname, "locales");
}
get navigationItems() {
return {
href: this.options.mountPath,
text: "youtube.title",
};
}
get shortcutItems() {
return {
url: this.options.mountPath,
name: "youtube.videos",
iconName: "syndicate",
};
}
/**
* Protected routes (require authentication)
* Admin dashboard
*/
get routes() {
protectedRouter.get("/", dashboardController.get);
protectedRouter.post("/refresh", dashboardController.refresh);
return protectedRouter;
}
/**
* Public routes (no authentication required)
* JSON API endpoints for Eleventy frontend
*/
get routesPublic() {
publicRouter.get("/api/videos", videosController.api);
publicRouter.get("/api/channel", channelController.api);
publicRouter.get("/api/live", liveController.api);
return publicRouter;
}
init(Indiekit) {
Indiekit.addEndpoint(this);
// Store YouTube config in application for controller access
Indiekit.config.application.youtubeConfig = this.options;
Indiekit.config.application.youtubeEndpoint = this.mountPath;
}
}