Improve video sharing by displaying thumbnails correctly on social media

Implements signed thumbnail URLs using Bunny.net and updates og:image meta tags in video.tsx and creates a /thumbnail/:videoId route.

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/wwY5Klj
This commit is contained in:
sebastjanartic 2025-08-04 19:16:54 +00:00
parent f286638b19
commit e9c136a92a
3 changed files with 29 additions and 11 deletions

View File

@ -93,10 +93,13 @@ export default function VideoPage() {
const shareUrl = `${window.location.origin}/video/${video.id}`;
// Use server proxy endpoint for reliable thumbnail access in social sharing
const publicThumbnailUrl = `${window.location.origin}/thumbnail/${video.id}`;
// Open Graph tags for Facebook
updateMeta('og:title', video.title);
updateMeta('og:description', video.description || `Oglej si ta video na VideoStream`);
updateMeta('og:image', video.thumbnailUrl);
updateMeta('og:image', publicThumbnailUrl);
updateMeta('og:image:width', '1200');
updateMeta('og:image:height', '630');
updateMeta('og:image:type', 'image/jpeg');
@ -111,7 +114,7 @@ export default function VideoPage() {
updateMeta('twitter:card', 'summary_large_image', false);
updateMeta('twitter:title', video.title, false);
updateMeta('twitter:description', video.description || `Oglej si ta video na VideoStream`, false);
updateMeta('twitter:image', video.thumbnailUrl, false);
updateMeta('twitter:image', publicThumbnailUrl, false);
}
}, [video]);

View File

@ -73,6 +73,11 @@ export class BunnyService {
return `https://${this.hostname}${path}?token=${token}&expires=${expireTimestamp}`;
}
// Public method for generating signed URLs for sharing
generatePublicSignedUrl(path: string, expiryHours: number = 1): string {
return this.generateSignedUrl(path, expiryHours);
}
private bunnyVideoToVideo(bunnyVideo: BunnyVideo): Video {
// Generate signed URLs for private video access
const videoPath = `/${bunnyVideo.guid}/playlist.m3u8`;

View File

@ -2,6 +2,7 @@ import type { Express } from "express";
import { createServer, type Server } from "http";
import { storage } from "./storage";
import { z } from "zod";
import { BunnyService } from "./bunny";
export async function registerRoutes(app: Express): Promise<Server> {
// Get videos with pagination and filtering
@ -117,26 +118,35 @@ export async function registerRoutes(app: Express): Promise<Server> {
}
});
// Proxy endpoint for thumbnail images from Bunny.net
// Public thumbnail endpoint for social media sharing
app.get("/thumbnail/:videoId", async (req, res) => {
try {
const { videoId } = req.params;
const hostname = process.env.BUNNY_HOSTNAME!;
const thumbnailUrl = `https://${hostname}/${videoId}/thumbnail.jpg`;
// Attempt to fetch and proxy the thumbnail
const response = await fetch(thumbnailUrl);
// Generate signed thumbnail URL using Bunny service
const bunnyService = new BunnyService();
const thumbnailPath = `/${videoId}/thumbnail.jpg`;
const signedThumbnailUrl = bunnyService.generatePublicSignedUrl(thumbnailPath, 24); // 24 hour expiry for sharing
// Fetch and proxy the thumbnail with proper headers for social media
const response = await fetch(signedThumbnailUrl);
if (response.ok) {
res.set('Content-Type', response.headers.get('content-type') || 'image/jpeg');
// Set appropriate headers for social media crawlers
res.set({
'Content-Type': 'image/jpeg',
'Cache-Control': 'public, max-age=86400', // Cache for 24 hours
'Access-Control-Allow-Origin': '*', // Allow cross-origin for social media
});
const buffer = await response.arrayBuffer();
res.send(Buffer.from(buffer));
} else {
// Fallback to a music-themed placeholder
res.redirect(`https://images.unsplash.com/photo-1493225457124-a3eb161ffa5f?ixlib=rb-4.0.3&auto=format&fit=crop&w=800&h=450`);
// Fallback to a high-quality video placeholder
res.redirect(`https://images.unsplash.com/photo-1536240478700-b869070f9279?ixlib=rb-4.0.3&auto=format&fit=crop&w=1200&h=630`);
}
} catch (error) {
console.error("Error proxying thumbnail:", error);
res.redirect(`https://images.unsplash.com/photo-1493225457124-a3eb161ffa5f?ixlib=rb-4.0.3&auto=format&fit=crop&w=800&h=450`);
res.redirect(`https://images.unsplash.com/photo-1536240478700-b869070f9279?ixlib=rb-4.0.3&auto=format&fit=crop&w=1200&h=630`);
}
});