summaryrefslogtreecommitdiff
path: root/packages/backend/src/core
diff options
context:
space:
mode:
authorPrivateGER <privateger@privateger.me>2025-05-29 20:29:42 +0200
committerPrivateGER <privateger@privateger.me>2025-05-29 20:29:42 +0200
commit7cba9c11d43c6f5929b9a2b046a18ddab738e651 (patch)
treeb4fdcb15c0da3735711cd8e1404ff69f0867c379 /packages/backend/src/core
parentmerge: Merge stable into develop (!1052) (diff)
downloadsharkey-7cba9c11d43c6f5929b9a2b046a18ddab738e651.tar.gz
sharkey-7cba9c11d43c6f5929b9a2b046a18ddab738e651.tar.bz2
sharkey-7cba9c11d43c6f5929b9a2b046a18ddab738e651.zip
Add web optimization for video files during processing
Diffstat (limited to 'packages/backend/src/core')
-rw-r--r--packages/backend/src/core/DriveService.ts4
-rw-r--r--packages/backend/src/core/VideoProcessingService.ts64
2 files changed, 67 insertions, 1 deletions
diff --git a/packages/backend/src/core/DriveService.ts b/packages/backend/src/core/DriveService.ts
index 82c447baaa..abe9131089 100644
--- a/packages/backend/src/core/DriveService.ts
+++ b/packages/backend/src/core/DriveService.ts
@@ -159,6 +159,10 @@ export class DriveService {
// thunbnail, webpublic を必要なら生成
const alts = await this.generateAlts(path, type, !file.uri);
+ if (type && type.startsWith('video/')) {
+ await this.videoProcessingService.webOptimizeVideo(path, type);
+ }
+
if (this.meta.useObjectStorage) {
//#region ObjectStorage params
let [ext] = (name.match(/\.([a-zA-Z0-9_-]+)$/) ?? ['']);
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();
+ }
+ }
}