Integrate videos directly from Bunny.net CDN for faster streaming

Implements Bunny.net CDN integration for video streaming and utilizes its API to fetch and display videos.

Replit-Commit-Author: Agent
Replit-Commit-Session-Id: 50814a1e-92e4-4968-856f-7bc7eedf5e8f
Replit-Commit-Checkpoint-Type: full_checkpoint
Replit-Commit-Screenshot-Url: https://storage.googleapis.com/screenshot-production-us-central1/8cc42625-c1f5-4e43-99bd-77f2c4dedee2/50814a1e-92e4-4968-856f-7bc7eedf5e8f/DXDnT5V
This commit is contained in:
sebastjanartic 2025-08-04 18:32:51 +00:00
parent edf1e36795
commit b77e18915f
4 changed files with 221 additions and 2 deletions

View File

@ -109,8 +109,8 @@ export default function VideoModal({ video, isOpen, onClose }: VideoModalProps)
preload="metadata"
onPlay={handleVideoPlay}
data-testid="video-player"
src={video.videoUrl}
>
<source src={video.videoUrl} type="video/mp4" />
Your browser does not support the video tag.
</video>

122
server/bunny.ts Normal file
View File

@ -0,0 +1,122 @@
import { type Video, type InsertVideo } from "@shared/schema";
interface BunnyVideo {
guid: string;
title: string;
length: number;
status: number;
dateUploaded: string;
views: number;
thumbnailFileName?: string;
category?: string;
}
interface BunnyLibraryResponse {
items: BunnyVideo[];
currentPage: number;
itemsPerPage: number;
totalItems: number;
}
export class BunnyService {
private apiKey: string;
private libraryId: string;
private hostname: string;
constructor() {
this.apiKey = process.env.BUNNY_API_KEY!;
this.libraryId = process.env.BUNNY_LIBRARY_ID!;
this.hostname = process.env.BUNNY_HOSTNAME!;
if (!this.apiKey || !this.libraryId || !this.hostname) {
throw new Error("Missing Bunny.net configuration");
}
}
private async makeRequest(endpoint: string): Promise<any> {
const url = `https://video.bunnycdn.com/library/${this.libraryId}/${endpoint}`;
const response = await fetch(url, {
headers: {
'AccessKey': this.apiKey,
'Accept': 'application/json',
'Content-Type': 'application/json'
}
});
if (!response.ok) {
throw new Error(`Bunny API error: ${response.status} ${response.statusText}`);
}
return response.json();
}
private bunnyVideoToVideo(bunnyVideo: BunnyVideo): Video {
// Generate thumbnail URL from Bunny CDN
const thumbnailUrl = bunnyVideo.thumbnailFileName
? `https://${this.hostname}/${bunnyVideo.guid}/${bunnyVideo.thumbnailFileName}`
: `https://${this.hostname}/${bunnyVideo.guid}/thumbnail.jpg`;
// Generate video URL for streaming
const videoUrl = `https://${this.hostname}/${bunnyVideo.guid}/playlist.m3u8`;
return {
id: bunnyVideo.guid,
title: bunnyVideo.title || 'Untitled Video',
description: null, // Bunny API doesn't return description in list view
thumbnailUrl,
videoUrl,
duration: Math.floor(bunnyVideo.length || 0),
views: bunnyVideo.views || 0,
category: bunnyVideo.category || null,
createdAt: new Date(bunnyVideo.dateUploaded)
};
}
async getVideos(page: number = 1, itemsPerPage: number = 20, search?: string): Promise<{ videos: Video[], total: number }> {
try {
let endpoint = `videos?page=${page}&itemsPerPage=${itemsPerPage}&orderBy=date`;
if (search) {
endpoint += `&search=${encodeURIComponent(search)}`;
}
const response: BunnyLibraryResponse = await this.makeRequest(endpoint);
// Filter only successfully processed videos (status 4 = finished)
const processedVideos = response.items.filter(video => video.status === 4);
const videos = processedVideos.map(video => this.bunnyVideoToVideo(video));
return {
videos,
total: response.totalItems
};
} catch (error) {
console.error('Error fetching videos from Bunny:', error);
throw error;
}
}
async getVideo(guid: string): Promise<Video | null> {
try {
const bunnyVideo: BunnyVideo = await this.makeRequest(`videos/${guid}`);
return this.bunnyVideoToVideo(bunnyVideo);
} catch (error) {
console.error(`Error fetching video ${guid} from Bunny:`, error);
return null;
}
}
async getCategories(): Promise<string[]> {
try {
// Get all videos and extract unique categories
const { videos } = await this.getVideos(1, 1000);
const categories = Array.from(new Set(videos.map(v => v.category).filter(Boolean) as string[]));
return categories;
} catch (error) {
console.error('Error fetching categories from Bunny:', error);
return [];
}
}
}

View File

@ -55,6 +55,7 @@ export async function registerRoutes(app: Express): Promise<Server> {
const categories = Array.from(new Set(videos.map(v => v.category).filter(Boolean)));
res.json(categories);
} catch (error) {
console.error("Error fetching categories:", error);
res.status(500).json({ message: "Failed to fetch categories" });
}
});

View File

@ -1,5 +1,6 @@
import { type Video, type InsertVideo } from "@shared/schema";
import { randomUUID } from "crypto";
import { BunnyService } from "./bunny";
export interface IStorage {
getVideos(limit?: number, offset?: number, search?: string, category?: string): Promise<Video[]>;
@ -146,4 +147,99 @@ export class MemStorage implements IStorage {
}
}
export const storage = new MemStorage();
// Use Bunny.net storage if API keys are available, otherwise fallback to memory storage
class BunnyStorage implements IStorage {
private bunnyService: BunnyService;
private viewsCache: Map<string, number> = new Map();
constructor() {
this.bunnyService = new BunnyService();
}
async getVideos(limit = 20, offset = 0, search?: string, category?: string): Promise<Video[]> {
try {
const page = Math.floor(offset / limit) + 1;
const { videos } = await this.bunnyService.getVideos(page, limit, search);
// Filter by category if specified
let filteredVideos = videos;
if (category && category !== "All Categories") {
filteredVideos = videos.filter(video => video.category === category);
}
// Apply cached view counts
filteredVideos.forEach(video => {
if (this.viewsCache.has(video.id)) {
video.views += this.viewsCache.get(video.id)!;
}
});
return filteredVideos;
} catch (error) {
console.error('Error fetching videos from Bunny:', error);
// Fallback to empty array on error
return [];
}
}
async getVideo(id: string): Promise<Video | undefined> {
try {
const video = await this.bunnyService.getVideo(id);
if (!video) return undefined;
// Apply cached view counts
if (this.viewsCache.has(video.id)) {
video.views += this.viewsCache.get(video.id)!;
}
return video;
} catch (error) {
console.error(`Error fetching video ${id} from Bunny:`, error);
return undefined;
}
}
async createVideo(video: InsertVideo): Promise<Video> {
// Note: Creating videos would require uploading to Bunny.net
// For now, we'll throw an error as this operation is not supported
throw new Error("Creating videos is not supported with Bunny.net integration");
}
async updateVideoViews(id: string): Promise<void> {
// Since we can't update views in Bunny.net directly, we'll cache them locally
const currentViews = this.viewsCache.get(id) || 0;
this.viewsCache.set(id, currentViews + 1);
}
async getVideoCount(search?: string, category?: string): Promise<number> {
try {
const { total } = await this.bunnyService.getVideos(1, 1, search);
return total;
} catch (error) {
console.error('Error getting video count from Bunny:', error);
return 0;
}
}
}
// Try to use Bunny.net storage, fallback to memory storage if not configured
let storage: IStorage;
// Check if Bunny.net environment variables are available
const hasBunnyConfig = process.env.BUNNY_API_KEY && process.env.BUNNY_LIBRARY_ID && process.env.BUNNY_HOSTNAME;
if (hasBunnyConfig) {
try {
storage = new BunnyStorage();
console.log('✅ Using Bunny.net storage with library ID:', process.env.BUNNY_LIBRARY_ID);
} catch (error) {
console.error('❌ Failed to initialize Bunny.net storage:', error);
console.log('📁 Falling back to memory storage');
storage = new MemStorage();
}
} else {
console.log('📁 Bunny.net environment variables not found, using memory storage');
storage = new MemStorage();
}
export { storage };