summaryrefslogtreecommitdiff
path: root/packages/backend/src/core/VideoProcessingService.ts
diff options
context:
space:
mode:
Diffstat (limited to 'packages/backend/src/core/VideoProcessingService.ts')
-rw-r--r--packages/backend/src/core/VideoProcessingService.ts64
1 files changed, 63 insertions, 1 deletions
diff --git a/packages/backend/src/core/VideoProcessingService.ts b/packages/backend/src/core/VideoProcessingService.ts
index 747fe4fc7e..809934e1a9 100644
--- a/packages/backend/src/core/VideoProcessingService.ts
+++ b/packages/backend/src/core/VideoProcessingService.ts
@@ -9,18 +9,25 @@ import { DI } from '@/di-symbols.js';
import type { Config } from '@/config.js';
import { ImageProcessingService } from '@/core/ImageProcessingService.js';
import type { IImage } from '@/core/ImageProcessingService.js';
-import { createTempDir } from '@/misc/create-temp.js';
+import {createTemp, createTempDir } from '@/misc/create-temp.js';
import { bindThis } from '@/decorators.js';
import { appendQuery, query } from '@/misc/prelude/url.js';
+import {LoggerService} from "@/core/LoggerService.js";
+import type Logger from "@/logger.js";
@Injectable()
export class VideoProcessingService {
+ private logger: Logger;
+
constructor(
@Inject(DI.config)
private config: Config,
private imageProcessingService: ImageProcessingService,
+
+ private loggerService: LoggerService,
) {
+ this.logger = this.loggerService.getLogger("video-processing");
}
@bindThis
@@ -60,5 +67,60 @@ export class VideoProcessingService {
}),
);
}
+
+ /**
+ * Optimize video for web playback by adding faststart flag.
+ * This allows the video to start playing before it is fully downloaded.
+ * The original file is modified in-place.
+ * @param source Path to the video file
+ * @param mimeType The MIME type of the video
+ * @returns Promise that resolves when optimization is complete
+ */
+ @bindThis
+ public async webOptimizeVideo(source: string, mimeType: string): Promise<void> {
+ // faststart is only supported for MP4, M4A, M4W and MOV files (the MOV family).
+ // WebM (and Matroska) files always support faststart-like behavior.
+ const supportedMimeTypes = new Map([
+ ['video/mp4', 'mp4'],
+ ['video/m4a', 'mp4'],
+ ['video/m4v', 'mp4'],
+ ['video/quicktime', 'mov']
+ ]);
+
+ const outputFormat = supportedMimeTypes.get(mimeType);
+ if (!outputFormat) {
+ this.logger.debug(`Skipping web optimization for unsupported MIME type: ${mimeType}`);
+ return;
+ }
+
+ const [tempPath, cleanup] = await createTemp();
+
+ try {
+ await new Promise<void>((resolve, reject) => {
+ FFmpeg(source)
+ .format(outputFormat) // Specify output format
+ .addOutputOptions('-c copy') // Copy streams without re-encoding
+ .addOutputOptions('-movflags +faststart')
+ .on('error', reject)
+ .on('end', async () => {
+ try {
+ // Replace original file with optimized version
+ const fs = await import('node:fs/promises');
+ await fs.copyFile(tempPath, source);
+ this.logger.info(`Web-optimized video: ${source}`);
+ resolve();
+ } catch (copyError) {
+ reject(copyError);
+ }
+ })
+ .save(tempPath);
+ });
+ } catch (error) {
+ this.logger.warn(`Failed to web-optimize video: ${source}`, { error });
+ throw error;
+ } finally {
+ cleanup();
+ }
+ }
}