2024-03-20 19:32:04 +01:00
|
|
|
import { Stats } from 'node:fs';
|
2024-03-20 22:02:51 +01:00
|
|
|
import { AssetType } from 'src/entities/asset.entity';
|
|
|
|
import { ExifEntity } from 'src/entities/exif.entity';
|
2023-08-07 22:35:25 +02:00
|
|
|
import {
|
2024-02-21 06:25:30 +01:00
|
|
|
AudioCodec,
|
2023-09-03 08:21:51 +02:00
|
|
|
Colorspace,
|
2023-08-07 22:35:25 +02:00
|
|
|
SystemConfigKey,
|
|
|
|
ToneMapping,
|
|
|
|
TranscodeHWAccel,
|
|
|
|
TranscodePolicy,
|
|
|
|
VideoCodec,
|
2024-03-20 22:02:51 +01:00
|
|
|
} from 'src/entities/system-config.entity';
|
2024-03-21 12:59:49 +01:00
|
|
|
import { IAssetRepository, WithoutProperty } from 'src/interfaces/asset.interface';
|
|
|
|
import { ICryptoRepository } from 'src/interfaces/crypto.interface';
|
|
|
|
import { IJobRepository, JobName, JobStatus } from 'src/interfaces/job.interface';
|
|
|
|
import { IMediaRepository } from 'src/interfaces/media.interface';
|
|
|
|
import { IMoveRepository } from 'src/interfaces/move.interface';
|
|
|
|
import { IPersonRepository } from 'src/interfaces/person.interface';
|
|
|
|
import { IStorageRepository } from 'src/interfaces/storage.interface';
|
|
|
|
import { ISystemConfigRepository } from 'src/interfaces/system-config.interface';
|
2024-03-21 00:07:30 +01:00
|
|
|
import { MediaService } from 'src/services/media.service';
|
2024-03-20 19:32:04 +01:00
|
|
|
import { assetStub } from 'test/fixtures/asset.stub';
|
|
|
|
import { faceStub } from 'test/fixtures/face.stub';
|
|
|
|
import { probeStub } from 'test/fixtures/media.stub';
|
|
|
|
import { personStub } from 'test/fixtures/person.stub';
|
|
|
|
import { newAssetRepositoryMock } from 'test/repositories/asset.repository.mock';
|
|
|
|
import { newCryptoRepositoryMock } from 'test/repositories/crypto.repository.mock';
|
|
|
|
import { newJobRepositoryMock } from 'test/repositories/job.repository.mock';
|
|
|
|
import { newMediaRepositoryMock } from 'test/repositories/media.repository.mock';
|
|
|
|
import { newMoveRepositoryMock } from 'test/repositories/move.repository.mock';
|
|
|
|
import { newPersonRepositoryMock } from 'test/repositories/person.repository.mock';
|
|
|
|
import { newStorageRepositoryMock } from 'test/repositories/storage.repository.mock';
|
|
|
|
import { newSystemConfigRepositoryMock } from 'test/repositories/system-config.repository.mock';
|
2023-03-24 03:40:46 +01:00
|
|
|
|
|
|
|
describe(MediaService.name, () => {
|
|
|
|
let sut: MediaService;
|
|
|
|
let assetMock: jest.Mocked<IAssetRepository>;
|
2023-04-04 16:48:02 +02:00
|
|
|
let configMock: jest.Mocked<ISystemConfigRepository>;
|
2023-03-24 03:40:46 +01:00
|
|
|
let jobMock: jest.Mocked<IJobRepository>;
|
|
|
|
let mediaMock: jest.Mocked<IMediaRepository>;
|
2023-10-11 04:14:44 +02:00
|
|
|
let moveMock: jest.Mocked<IMoveRepository>;
|
2023-09-08 08:49:43 +02:00
|
|
|
let personMock: jest.Mocked<IPersonRepository>;
|
2023-03-24 03:40:46 +01:00
|
|
|
let storageMock: jest.Mocked<IStorageRepository>;
|
2023-12-29 19:41:33 +01:00
|
|
|
let cryptoMock: jest.Mocked<ICryptoRepository>;
|
2023-03-24 03:40:46 +01:00
|
|
|
|
2024-03-05 23:23:06 +01:00
|
|
|
beforeEach(() => {
|
2023-03-24 03:40:46 +01:00
|
|
|
assetMock = newAssetRepositoryMock();
|
2023-04-04 16:48:02 +02:00
|
|
|
configMock = newSystemConfigRepositoryMock();
|
2023-03-24 03:40:46 +01:00
|
|
|
jobMock = newJobRepositoryMock();
|
|
|
|
mediaMock = newMediaRepositoryMock();
|
2023-10-11 04:14:44 +02:00
|
|
|
moveMock = newMoveRepositoryMock();
|
2023-09-08 08:49:43 +02:00
|
|
|
personMock = newPersonRepositoryMock();
|
2023-03-24 03:40:46 +01:00
|
|
|
storageMock = newStorageRepositoryMock();
|
2023-12-29 19:41:33 +01:00
|
|
|
cryptoMock = newCryptoRepositoryMock();
|
2023-05-17 19:07:17 +02:00
|
|
|
|
2023-12-29 19:41:33 +01:00
|
|
|
sut = new MediaService(assetMock, personMock, jobMock, mediaMock, storageMock, configMock, moveMock, cryptoMock);
|
2023-03-24 03:40:46 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should be defined', () => {
|
|
|
|
expect(sut).toBeDefined();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('handleQueueGenerateThumbnails', () => {
|
|
|
|
it('should queue all assets', async () => {
|
2023-05-22 20:05:06 +02:00
|
|
|
assetMock.getAll.mockResolvedValue({
|
2023-08-01 03:28:07 +02:00
|
|
|
items: [assetStub.image],
|
2023-05-22 20:05:06 +02:00
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2024-01-18 06:08:48 +01:00
|
|
|
personMock.getAll.mockResolvedValue({
|
|
|
|
items: [personStub.newThumbnail],
|
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2023-09-27 22:46:46 +02:00
|
|
|
personMock.getFacesByIds.mockResolvedValue([faceStub.face1]);
|
2023-03-24 03:40:46 +01:00
|
|
|
|
|
|
|
await sut.handleQueueGenerateThumbnails({ force: true });
|
|
|
|
|
|
|
|
expect(assetMock.getAll).toHaveBeenCalled();
|
|
|
|
expect(assetMock.getWithout).not.toHaveBeenCalled();
|
2024-01-01 21:45:42 +01:00
|
|
|
expect(jobMock.queueAll).toHaveBeenCalledWith([
|
|
|
|
{
|
|
|
|
name: JobName.GENERATE_JPEG_THUMBNAIL,
|
|
|
|
data: { id: assetStub.image.id },
|
|
|
|
},
|
|
|
|
]);
|
2023-09-08 08:49:43 +02:00
|
|
|
|
2024-01-18 06:08:48 +01:00
|
|
|
expect(personMock.getAll).toHaveBeenCalledWith({ skip: 0, take: 1000 }, {});
|
2024-01-01 21:45:42 +01:00
|
|
|
expect(jobMock.queueAll).toHaveBeenCalledWith([
|
|
|
|
{
|
|
|
|
name: JobName.GENERATE_PERSON_THUMBNAIL,
|
|
|
|
data: { id: personStub.newThumbnail.id },
|
|
|
|
},
|
|
|
|
]);
|
2023-09-08 08:49:43 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should queue all people with missing thumbnail path', async () => {
|
|
|
|
assetMock.getWithout.mockResolvedValue({
|
|
|
|
items: [assetStub.image],
|
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2024-01-18 06:08:48 +01:00
|
|
|
personMock.getAll.mockResolvedValue({
|
|
|
|
items: [personStub.noThumbnail],
|
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2023-09-08 08:49:43 +02:00
|
|
|
personMock.getRandomFace.mockResolvedValue(faceStub.face1);
|
|
|
|
|
|
|
|
await sut.handleQueueGenerateThumbnails({ force: false });
|
|
|
|
|
|
|
|
expect(assetMock.getAll).not.toHaveBeenCalled();
|
|
|
|
expect(assetMock.getWithout).toHaveBeenCalledWith({ skip: 0, take: 1000 }, WithoutProperty.THUMBNAIL);
|
|
|
|
|
2024-01-18 06:08:48 +01:00
|
|
|
expect(personMock.getAll).toHaveBeenCalledWith({ skip: 0, take: 1000 }, { where: { thumbnailPath: '' } });
|
2023-09-08 08:49:43 +02:00
|
|
|
expect(personMock.getRandomFace).toHaveBeenCalled();
|
2024-01-01 21:45:42 +01:00
|
|
|
expect(jobMock.queueAll).toHaveBeenCalledWith([
|
|
|
|
{
|
|
|
|
name: JobName.GENERATE_PERSON_THUMBNAIL,
|
|
|
|
data: {
|
|
|
|
id: personStub.newThumbnail.id,
|
|
|
|
},
|
2023-09-08 08:49:43 +02:00
|
|
|
},
|
2024-01-01 21:45:42 +01:00
|
|
|
]);
|
2023-03-24 03:40:46 +01:00
|
|
|
});
|
|
|
|
|
2023-06-18 05:22:31 +02:00
|
|
|
it('should queue all assets with missing resize path', async () => {
|
2023-05-22 20:05:06 +02:00
|
|
|
assetMock.getWithout.mockResolvedValue({
|
2023-08-01 03:28:07 +02:00
|
|
|
items: [assetStub.noResizePath],
|
2023-05-22 20:05:06 +02:00
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2024-01-18 06:08:48 +01:00
|
|
|
personMock.getAll.mockResolvedValue({
|
|
|
|
items: [],
|
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2023-03-24 03:40:46 +01:00
|
|
|
|
|
|
|
await sut.handleQueueGenerateThumbnails({ force: false });
|
|
|
|
|
|
|
|
expect(assetMock.getAll).not.toHaveBeenCalled();
|
2023-05-22 20:05:06 +02:00
|
|
|
expect(assetMock.getWithout).toHaveBeenCalledWith({ skip: 0, take: 1000 }, WithoutProperty.THUMBNAIL);
|
2024-01-01 21:45:42 +01:00
|
|
|
expect(jobMock.queueAll).toHaveBeenCalledWith([
|
|
|
|
{
|
|
|
|
name: JobName.GENERATE_JPEG_THUMBNAIL,
|
|
|
|
data: { id: assetStub.image.id },
|
|
|
|
},
|
|
|
|
]);
|
2023-09-08 08:49:43 +02:00
|
|
|
|
2024-01-18 06:08:48 +01:00
|
|
|
expect(personMock.getAll).toHaveBeenCalledWith({ skip: 0, take: 1000 }, { where: { thumbnailPath: '' } });
|
2023-03-24 03:40:46 +01:00
|
|
|
});
|
2023-06-18 05:22:31 +02:00
|
|
|
|
|
|
|
it('should queue all assets with missing webp path', async () => {
|
|
|
|
assetMock.getWithout.mockResolvedValue({
|
2023-08-01 03:28:07 +02:00
|
|
|
items: [assetStub.noWebpPath],
|
2023-06-18 05:22:31 +02:00
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2024-01-18 06:08:48 +01:00
|
|
|
personMock.getAll.mockResolvedValue({
|
|
|
|
items: [],
|
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2023-06-18 05:22:31 +02:00
|
|
|
|
|
|
|
await sut.handleQueueGenerateThumbnails({ force: false });
|
|
|
|
|
|
|
|
expect(assetMock.getAll).not.toHaveBeenCalled();
|
|
|
|
expect(assetMock.getWithout).toHaveBeenCalledWith({ skip: 0, take: 1000 }, WithoutProperty.THUMBNAIL);
|
2024-01-01 21:45:42 +01:00
|
|
|
expect(jobMock.queueAll).toHaveBeenCalledWith([
|
|
|
|
{
|
|
|
|
name: JobName.GENERATE_WEBP_THUMBNAIL,
|
|
|
|
data: { id: assetStub.image.id },
|
|
|
|
},
|
|
|
|
]);
|
2023-09-08 08:49:43 +02:00
|
|
|
|
2024-01-18 06:08:48 +01:00
|
|
|
expect(personMock.getAll).toHaveBeenCalledWith({ skip: 0, take: 1000 }, { where: { thumbnailPath: '' } });
|
2023-06-18 05:22:31 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should queue all assets with missing thumbhash', async () => {
|
|
|
|
assetMock.getWithout.mockResolvedValue({
|
2023-08-01 03:28:07 +02:00
|
|
|
items: [assetStub.noThumbhash],
|
2023-06-18 05:22:31 +02:00
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2024-01-18 06:08:48 +01:00
|
|
|
personMock.getAll.mockResolvedValue({
|
|
|
|
items: [],
|
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2023-06-18 05:22:31 +02:00
|
|
|
|
|
|
|
await sut.handleQueueGenerateThumbnails({ force: false });
|
|
|
|
|
|
|
|
expect(assetMock.getAll).not.toHaveBeenCalled();
|
|
|
|
expect(assetMock.getWithout).toHaveBeenCalledWith({ skip: 0, take: 1000 }, WithoutProperty.THUMBNAIL);
|
2024-01-01 21:45:42 +01:00
|
|
|
expect(jobMock.queueAll).toHaveBeenCalledWith([
|
|
|
|
{
|
|
|
|
name: JobName.GENERATE_THUMBHASH_THUMBNAIL,
|
|
|
|
data: { id: assetStub.image.id },
|
|
|
|
},
|
|
|
|
]);
|
2023-09-08 08:49:43 +02:00
|
|
|
|
2024-01-18 06:08:48 +01:00
|
|
|
expect(personMock.getAll).toHaveBeenCalledWith({ skip: 0, take: 1000 }, { where: { thumbnailPath: '' } });
|
2023-06-18 05:22:31 +02:00
|
|
|
});
|
2023-03-24 03:40:46 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('handleGenerateJpegThumbnail', () => {
|
2023-07-09 04:43:11 +02:00
|
|
|
it('should skip thumbnail generation if asset not found', async () => {
|
|
|
|
assetMock.getByIds.mockResolvedValue([]);
|
2023-08-01 03:28:07 +02:00
|
|
|
await sut.handleGenerateJpegThumbnail({ id: assetStub.image.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.resize).not.toHaveBeenCalled();
|
2024-03-20 03:42:10 +01:00
|
|
|
expect(assetMock.update).not.toHaveBeenCalledWith();
|
2023-07-09 04:43:11 +02:00
|
|
|
});
|
|
|
|
|
2023-08-07 22:35:25 +02:00
|
|
|
it('should skip video thumbnail generation if no video stream', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.noVideoStreams);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleGenerateJpegThumbnail({ id: assetStub.image.id });
|
|
|
|
expect(mediaMock.resize).not.toHaveBeenCalled();
|
2024-03-20 03:42:10 +01:00
|
|
|
expect(assetMock.update).not.toHaveBeenCalledWith();
|
2023-08-07 22:35:25 +02:00
|
|
|
});
|
|
|
|
|
2023-03-24 03:40:46 +01:00
|
|
|
it('should generate a thumbnail for an image', async () => {
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image]);
|
|
|
|
await sut.handleGenerateJpegThumbnail({ id: assetStub.image.id });
|
2023-03-24 03:40:46 +01:00
|
|
|
|
2023-09-25 17:07:21 +02:00
|
|
|
expect(storageMock.mkdirSync).toHaveBeenCalledWith('upload/thumbs/user-id/as/se');
|
|
|
|
expect(mediaMock.resize).toHaveBeenCalledWith('/original/path.jpg', 'upload/thumbs/user-id/as/se/asset-id.jpeg', {
|
2023-09-26 01:18:47 +02:00
|
|
|
size: 1440,
|
|
|
|
format: 'jpeg',
|
|
|
|
quality: 80,
|
|
|
|
colorspace: Colorspace.SRGB,
|
|
|
|
});
|
2024-03-20 03:42:10 +01:00
|
|
|
expect(assetMock.update).toHaveBeenCalledWith({
|
2023-09-26 01:18:47 +02:00
|
|
|
id: 'asset-id',
|
2023-09-26 04:17:53 +02:00
|
|
|
resizePath: 'upload/thumbs/user-id/as/se/asset-id.jpeg',
|
2023-09-26 01:18:47 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should generate a P3 thumbnail for a wide gamut image', async () => {
|
|
|
|
assetMock.getByIds.mockResolvedValue([
|
|
|
|
{ ...assetStub.image, exifInfo: { profileDescription: 'Adobe RGB', bitsPerSample: 14 } as ExifEntity },
|
|
|
|
]);
|
|
|
|
await sut.handleGenerateJpegThumbnail({ id: assetStub.image.id });
|
|
|
|
|
2023-09-26 04:17:53 +02:00
|
|
|
expect(storageMock.mkdirSync).toHaveBeenCalledWith('upload/thumbs/user-id/as/se');
|
|
|
|
expect(mediaMock.resize).toHaveBeenCalledWith('/original/path.jpg', 'upload/thumbs/user-id/as/se/asset-id.jpeg', {
|
2023-03-25 15:50:57 +01:00
|
|
|
size: 1440,
|
|
|
|
format: 'jpeg',
|
2023-09-03 08:21:51 +02:00
|
|
|
quality: 80,
|
|
|
|
colorspace: Colorspace.P3,
|
2023-03-25 15:50:57 +01:00
|
|
|
});
|
2024-03-20 03:42:10 +01:00
|
|
|
expect(assetMock.update).toHaveBeenCalledWith({
|
2023-03-24 03:40:46 +01:00
|
|
|
id: 'asset-id',
|
2023-09-25 17:07:21 +02:00
|
|
|
resizePath: 'upload/thumbs/user-id/as/se/asset-id.jpeg',
|
2023-03-24 03:40:46 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should generate a thumbnail for a video', async () => {
|
2023-08-07 22:35:25 +02:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStream2160p);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleGenerateJpegThumbnail({ id: assetStub.video.id });
|
2023-03-24 03:40:46 +01:00
|
|
|
|
2023-09-25 17:07:21 +02:00
|
|
|
expect(storageMock.mkdirSync).toHaveBeenCalledWith('upload/thumbs/user-id/as/se');
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/thumbs/user-id/as/se/asset-id.jpeg',
|
|
|
|
{
|
|
|
|
inputOptions: ['-ss 00:00:00', '-sws_flags accurate_rnd+bitexact+full_chroma_int'],
|
|
|
|
outputOptions: [
|
|
|
|
'-frames:v 1',
|
|
|
|
'-v verbose',
|
|
|
|
'-vf scale=-2:1440:flags=lanczos+accurate_rnd+bitexact+full_chroma_int:out_color_matrix=601:out_range=pc,format=yuv420p',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
2024-03-20 03:42:10 +01:00
|
|
|
expect(assetMock.update).toHaveBeenCalledWith({
|
2023-08-07 22:35:25 +02:00
|
|
|
id: 'asset-id',
|
2023-09-25 17:07:21 +02:00
|
|
|
resizePath: 'upload/thumbs/user-id/as/se/asset-id.jpeg',
|
2023-08-07 22:35:25 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should tonemap thumbnail for hdr video', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamHDR);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleGenerateJpegThumbnail({ id: assetStub.video.id });
|
|
|
|
|
2023-09-25 17:07:21 +02:00
|
|
|
expect(storageMock.mkdirSync).toHaveBeenCalledWith('upload/thumbs/user-id/as/se');
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/thumbs/user-id/as/se/asset-id.jpeg',
|
|
|
|
{
|
|
|
|
inputOptions: ['-ss 00:00:00', '-sws_flags accurate_rnd+bitexact+full_chroma_int'],
|
|
|
|
outputOptions: [
|
|
|
|
'-frames:v 1',
|
|
|
|
'-v verbose',
|
|
|
|
'-vf zscale=t=linear:npl=100,tonemap=hable:desat=0,zscale=p=bt709:t=601:m=bt470bg:range=pc,format=yuv420p',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
2024-03-20 03:42:10 +01:00
|
|
|
expect(assetMock.update).toHaveBeenCalledWith({
|
2023-03-24 03:40:46 +01:00
|
|
|
id: 'asset-id',
|
2023-09-25 17:07:21 +02:00
|
|
|
resizePath: 'upload/thumbs/user-id/as/se/asset-id.jpeg',
|
2023-03-24 03:40:46 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2023-09-28 14:29:31 +02:00
|
|
|
it('should always generate video thumbnail in one pass', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamHDR);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TWO_PASS, value: true },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '5000k' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleGenerateJpegThumbnail({ id: assetStub.video.id });
|
|
|
|
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/thumbs/user-id/as/se/asset-id.jpeg',
|
|
|
|
{
|
|
|
|
inputOptions: ['-ss 00:00:00', '-sws_flags accurate_rnd+bitexact+full_chroma_int'],
|
|
|
|
outputOptions: [
|
|
|
|
'-frames:v 1',
|
|
|
|
'-v verbose',
|
|
|
|
'-vf zscale=t=linear:npl=100,tonemap=hable:desat=0,zscale=p=bt709:t=601:m=bt470bg:range=pc,format=yuv420p',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-05-26 21:43:24 +02:00
|
|
|
it('should run successfully', async () => {
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image]);
|
|
|
|
await sut.handleGenerateJpegThumbnail({ id: assetStub.image.id });
|
2023-03-24 03:40:46 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('handleGenerateWebpThumbnail', () => {
|
2023-07-09 04:43:11 +02:00
|
|
|
it('should skip thumbnail generation if asset not found', async () => {
|
|
|
|
assetMock.getByIds.mockResolvedValue([]);
|
2023-08-01 03:28:07 +02:00
|
|
|
await sut.handleGenerateWebpThumbnail({ id: assetStub.image.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.resize).not.toHaveBeenCalled();
|
2024-03-20 03:42:10 +01:00
|
|
|
expect(assetMock.update).not.toHaveBeenCalledWith();
|
2023-07-09 04:43:11 +02:00
|
|
|
});
|
|
|
|
|
2023-03-24 03:40:46 +01:00
|
|
|
it('should generate a thumbnail', async () => {
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image]);
|
|
|
|
await sut.handleGenerateWebpThumbnail({ id: assetStub.image.id });
|
2023-03-24 03:40:46 +01:00
|
|
|
|
2023-09-25 17:07:21 +02:00
|
|
|
expect(mediaMock.resize).toHaveBeenCalledWith('/original/path.jpg', 'upload/thumbs/user-id/as/se/asset-id.webp', {
|
2023-09-03 08:21:51 +02:00
|
|
|
format: 'webp',
|
|
|
|
size: 250,
|
|
|
|
quality: 80,
|
2023-09-26 01:18:47 +02:00
|
|
|
colorspace: Colorspace.SRGB,
|
2023-09-03 08:21:51 +02:00
|
|
|
});
|
2024-03-20 03:42:10 +01:00
|
|
|
expect(assetMock.update).toHaveBeenCalledWith({
|
2023-09-25 17:07:21 +02:00
|
|
|
id: 'asset-id',
|
|
|
|
webpPath: 'upload/thumbs/user-id/as/se/asset-id.webp',
|
|
|
|
});
|
2023-03-24 03:40:46 +01:00
|
|
|
});
|
|
|
|
});
|
2023-04-04 16:48:02 +02:00
|
|
|
|
2023-09-26 01:18:47 +02:00
|
|
|
it('should generate a P3 thumbnail for a wide gamut image', async () => {
|
|
|
|
assetMock.getByIds.mockResolvedValue([
|
|
|
|
{ ...assetStub.image, exifInfo: { profileDescription: 'Adobe RGB', bitsPerSample: 14 } as ExifEntity },
|
|
|
|
]);
|
|
|
|
await sut.handleGenerateWebpThumbnail({ id: assetStub.image.id });
|
|
|
|
|
2023-09-26 04:17:53 +02:00
|
|
|
expect(storageMock.mkdirSync).toHaveBeenCalledWith('upload/thumbs/user-id/as/se');
|
|
|
|
expect(mediaMock.resize).toHaveBeenCalledWith('/original/path.jpg', 'upload/thumbs/user-id/as/se/asset-id.webp', {
|
2023-09-26 01:18:47 +02:00
|
|
|
format: 'webp',
|
|
|
|
size: 250,
|
|
|
|
quality: 80,
|
|
|
|
colorspace: Colorspace.P3,
|
|
|
|
});
|
2024-03-20 03:42:10 +01:00
|
|
|
expect(assetMock.update).toHaveBeenCalledWith({
|
2023-09-26 04:17:53 +02:00
|
|
|
id: 'asset-id',
|
|
|
|
webpPath: 'upload/thumbs/user-id/as/se/asset-id.webp',
|
|
|
|
});
|
2023-09-26 01:18:47 +02:00
|
|
|
});
|
|
|
|
|
2023-06-18 05:22:31 +02:00
|
|
|
describe('handleGenerateThumbhashThumbnail', () => {
|
2023-07-09 04:43:11 +02:00
|
|
|
it('should skip thumbhash generation if asset not found', async () => {
|
|
|
|
assetMock.getByIds.mockResolvedValue([]);
|
2023-08-01 03:28:07 +02:00
|
|
|
await sut.handleGenerateThumbhashThumbnail({ id: assetStub.image.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.generateThumbhash).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
2023-06-18 05:22:31 +02:00
|
|
|
it('should skip thumbhash generation if resize path is missing', async () => {
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.noResizePath]);
|
|
|
|
await sut.handleGenerateThumbhashThumbnail({ id: assetStub.noResizePath.id });
|
2023-06-18 05:22:31 +02:00
|
|
|
expect(mediaMock.generateThumbhash).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should generate a thumbhash', async () => {
|
|
|
|
const thumbhashBuffer = Buffer.from('a thumbhash', 'utf8');
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image]);
|
2023-06-18 05:22:31 +02:00
|
|
|
mediaMock.generateThumbhash.mockResolvedValue(thumbhashBuffer);
|
|
|
|
|
2023-08-01 03:28:07 +02:00
|
|
|
await sut.handleGenerateThumbhashThumbnail({ id: assetStub.image.id });
|
2023-06-18 05:22:31 +02:00
|
|
|
|
2023-07-10 19:56:45 +02:00
|
|
|
expect(mediaMock.generateThumbhash).toHaveBeenCalledWith('/uploads/user-id/thumbs/path.jpg');
|
2024-03-20 03:42:10 +01:00
|
|
|
expect(assetMock.update).toHaveBeenCalledWith({ id: 'asset-id', thumbhash: thumbhashBuffer });
|
2023-06-18 05:22:31 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2023-04-04 16:48:02 +02:00
|
|
|
describe('handleQueueVideoConversion', () => {
|
|
|
|
it('should queue all video assets', async () => {
|
2023-05-22 20:05:06 +02:00
|
|
|
assetMock.getAll.mockResolvedValue({
|
2023-08-01 03:28:07 +02:00
|
|
|
items: [assetStub.video],
|
2023-05-22 20:05:06 +02:00
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2024-01-18 06:08:48 +01:00
|
|
|
personMock.getAll.mockResolvedValue({
|
|
|
|
items: [],
|
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2023-04-04 16:48:02 +02:00
|
|
|
|
|
|
|
await sut.handleQueueVideoConversion({ force: true });
|
|
|
|
|
2023-05-22 20:05:06 +02:00
|
|
|
expect(assetMock.getAll).toHaveBeenCalledWith({ skip: 0, take: 1000 }, { type: AssetType.VIDEO });
|
2023-04-04 16:48:02 +02:00
|
|
|
expect(assetMock.getWithout).not.toHaveBeenCalled();
|
2024-01-01 21:45:42 +01:00
|
|
|
expect(jobMock.queueAll).toHaveBeenCalledWith([
|
|
|
|
{
|
|
|
|
name: JobName.VIDEO_CONVERSION,
|
|
|
|
data: { id: assetStub.video.id },
|
|
|
|
},
|
|
|
|
]);
|
2023-04-04 16:48:02 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should queue all video assets without encoded videos', async () => {
|
2023-05-22 20:05:06 +02:00
|
|
|
assetMock.getWithout.mockResolvedValue({
|
2023-08-01 03:28:07 +02:00
|
|
|
items: [assetStub.video],
|
2023-05-22 20:05:06 +02:00
|
|
|
hasNextPage: false,
|
|
|
|
});
|
2023-04-04 16:48:02 +02:00
|
|
|
|
|
|
|
await sut.handleQueueVideoConversion({});
|
|
|
|
|
|
|
|
expect(assetMock.getAll).not.toHaveBeenCalled();
|
2023-05-22 20:05:06 +02:00
|
|
|
expect(assetMock.getWithout).toHaveBeenCalledWith({ skip: 0, take: 1000 }, WithoutProperty.ENCODED_VIDEO);
|
2024-01-01 21:45:42 +01:00
|
|
|
expect(jobMock.queueAll).toHaveBeenCalledWith([
|
|
|
|
{
|
|
|
|
name: JobName.VIDEO_CONVERSION,
|
|
|
|
data: { id: assetStub.video.id },
|
|
|
|
},
|
|
|
|
]);
|
2023-04-04 16:48:02 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('handleVideoConversion', () => {
|
2023-04-11 15:56:52 +02:00
|
|
|
beforeEach(() => {
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
2023-04-11 15:56:52 +02:00
|
|
|
});
|
|
|
|
|
2023-07-09 04:43:11 +02:00
|
|
|
it('should skip transcoding if asset not found', async () => {
|
|
|
|
assetMock.getByIds.mockResolvedValue([]);
|
2023-08-01 03:28:07 +02:00
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.probe).not.toHaveBeenCalled();
|
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should skip transcoding if non-video asset', async () => {
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.image.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.probe).not.toHaveBeenCalled();
|
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
2023-04-04 16:48:02 +02:00
|
|
|
it('should transcode the longest stream', async () => {
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
2023-04-06 05:32:59 +02:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.multipleVideoStreams);
|
2023-04-04 16:48:02 +02:00
|
|
|
|
2023-08-01 03:28:07 +02:00
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-04-04 16:48:02 +02:00
|
|
|
|
|
|
|
expect(mediaMock.probe).toHaveBeenCalledWith('/original/path.ext');
|
|
|
|
expect(configMock.load).toHaveBeenCalled();
|
|
|
|
expect(storageMock.mkdirSync).toHaveBeenCalled();
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-07-01 03:48:40 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf format=yuv420p',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
2023-05-22 20:07:43 +02:00
|
|
|
twoPass: false,
|
|
|
|
},
|
2023-04-04 16:48:02 +02:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should skip a video without any streams', async () => {
|
2023-04-06 05:32:59 +02:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.noVideoStreams);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-04-04 16:48:02 +02:00
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should skip a video without any height', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.noHeight);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-04-04 16:48:02 +02:00
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should transcode when set to all', async () => {
|
2023-04-06 05:32:59 +02:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.multipleVideoStreams);
|
2023-07-09 04:43:11 +02:00
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.ALL }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-04-04 16:48:02 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-07-01 03:48:40 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
|
|
|
'-c:a aac',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf format=yuv420p',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
2023-05-22 20:07:43 +02:00
|
|
|
twoPass: false,
|
|
|
|
},
|
2023-04-04 16:48:02 +02:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should transcode when optimal and too big', async () => {
|
2023-04-06 05:32:59 +02:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStream2160p);
|
2023-07-09 04:43:11 +02:00
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.OPTIMAL }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-04-06 05:32:59 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-05-22 20:07:43 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-movflags faststart',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
2023-04-06 05:32:59 +02:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-01-31 02:25:07 +01:00
|
|
|
it('should transcode when policy Bitrate and bitrate higher than max bitrate', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStream40Mbps);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.BITRATE },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '30M' },
|
|
|
|
]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2024-01-31 02:25:07 +01:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-v verbose',
|
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
'-maxrate 30M',
|
|
|
|
'-bufsize 60M',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-06-10 06:15:12 +02:00
|
|
|
it('should not scale resolution if no target resolution', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStream2160p);
|
|
|
|
configMock.load.mockResolvedValue([
|
2023-07-09 04:43:11 +02:00
|
|
|
{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.ALL },
|
2023-06-10 06:15:12 +02:00
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_RESOLUTION, value: 'original' },
|
|
|
|
]);
|
2023-08-01 03:28:07 +02:00
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-06-10 06:15:12 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-06-10 06:15:12 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-07-01 03:48:40 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
|
|
|
'-c:a aac',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf format=yuv420p',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
2023-06-10 06:15:12 +02:00
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-04-06 05:32:59 +02:00
|
|
|
it('should transcode with alternate scaling video is vertical', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamVertical2160p);
|
2023-07-09 04:43:11 +02:00
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.OPTIMAL }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-04-06 05:32:59 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-05-22 20:07:43 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-movflags faststart',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=720:-2,format=yuv420p',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
2023-04-06 05:32:59 +02:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-01-18 04:16:44 +01:00
|
|
|
it('should always scale video if height is uneven', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamOddHeight);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.ALL },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_RESOLUTION, value: 'original' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
|
|
|
'-c:v h264',
|
|
|
|
'-c:a aac',
|
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-v verbose',
|
|
|
|
`-vf scale=-2:354,format=yuv420p`,
|
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should always scale video if width is uneven', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamOddWidth);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.ALL },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_RESOLUTION, value: 'original' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
|
|
|
'-c:v h264',
|
|
|
|
'-c:a aac',
|
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-v verbose',
|
|
|
|
`-vf scale=354:-2,format=yuv420p`,
|
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-02-14 17:24:39 +01:00
|
|
|
it('should copy video stream when video matches target', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
2024-02-21 06:25:30 +01:00
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.HEVC },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEPTED_AUDIO_CODECS, value: [AudioCodec.AAC] },
|
|
|
|
]);
|
2024-02-14 17:24:39 +01:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
|
|
|
'-c:v copy',
|
|
|
|
'-c:a aac',
|
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-tag:v hvc1',
|
|
|
|
'-v verbose',
|
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should copy audio stream when audio matches target', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.audioStreamAac);
|
2023-07-09 04:43:11 +02:00
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.OPTIMAL }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-04-06 05:32:59 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-05-22 20:07:43 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-14 17:24:39 +01:00
|
|
|
'-c:a copy',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-movflags faststart',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
2023-04-06 05:32:59 +02:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should transcode when container doesnt match target', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
2023-07-09 04:43:11 +02:00
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.OPTIMAL }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-04-04 16:48:02 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-05-22 20:07:43 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-movflags faststart',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
2023-04-04 16:48:02 +02:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-02-14 17:24:39 +01:00
|
|
|
it('should throw an exception if transcode value is invalid', async () => {
|
2023-04-06 05:32:59 +02:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStream2160p);
|
2023-04-04 16:48:02 +02:00
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: 'invalid' }]);
|
2024-02-14 17:24:39 +01:00
|
|
|
|
|
|
|
await expect(sut.handleVideoConversion({ id: assetStub.video.id })).rejects.toThrow();
|
2023-04-04 16:48:02 +02:00
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
2023-05-22 20:07:43 +02:00
|
|
|
|
2023-07-09 04:43:11 +02:00
|
|
|
it('should not transcode if transcoding is disabled', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStream2160p);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.DISABLED }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not transcode if target codec is invalid', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStream2160p);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: 'invalid' }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
2023-12-28 06:34:00 +01:00
|
|
|
it('should delete existing transcode if current policy does not require transcoding', async () => {
|
|
|
|
const asset = assetStub.hasEncodedVideo;
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStream2160p);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.DISABLED }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([asset]);
|
|
|
|
|
|
|
|
await sut.handleVideoConversion({ id: asset.id });
|
|
|
|
|
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
|
|
name: JobName.DELETE_FILES,
|
|
|
|
data: { files: [asset.encodedVideoPath] },
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2023-05-22 20:07:43 +02:00
|
|
|
it('should set max bitrate if above 0', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '4500k' }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-05-22 20:07:43 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-05-22 20:07:43 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-movflags faststart',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
'-maxrate 4500k',
|
2023-07-01 03:48:05 +02:00
|
|
|
'-bufsize 9000k',
|
2023-05-22 20:07:43 +02:00
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should transcode in two passes for h264/h265 when enabled and max bitrate is above 0', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '4500k' },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TWO_PASS, value: true },
|
|
|
|
]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-05-22 20:07:43 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-05-22 20:07:43 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-movflags faststart',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-b:v 3104k',
|
|
|
|
'-minrate 1552k',
|
|
|
|
'-maxrate 4500k',
|
|
|
|
],
|
|
|
|
twoPass: true,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should fallback to one pass for h264/h265 if two-pass is enabled but no max bitrate is set', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TWO_PASS, value: true }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-05-22 20:07:43 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-05-22 20:07:43 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-movflags faststart',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-07-09 04:43:11 +02:00
|
|
|
it('should transcode by bitrate in two passes for vp9 when two pass mode and max bitrate are enabled', async () => {
|
2023-05-22 20:07:43 +02:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
2023-07-09 04:43:11 +02:00
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '4500k' },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TWO_PASS, value: true },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.VP9 },
|
2023-05-22 20:07:43 +02:00
|
|
|
]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-05-22 20:07:43 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-05-22 20:07:43 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v vp9',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-movflags faststart',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-cpu-used 5',
|
|
|
|
'-row-mt 1',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-b:v 3104k',
|
|
|
|
'-minrate 1552k',
|
|
|
|
'-maxrate 4500k',
|
|
|
|
],
|
|
|
|
twoPass: true,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-01-20 21:05:08 +01:00
|
|
|
it('should transcode by crf in two passes for vp9 when two pass mode is enabled and max bitrate is disabled', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '0' },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TWO_PASS, value: true },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.VP9 },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
|
|
|
'-c:v vp9',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2024-01-20 21:05:08 +01:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-v verbose',
|
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
|
|
|
'-cpu-used 5',
|
|
|
|
'-row-mt 1',
|
|
|
|
'-crf 23',
|
|
|
|
'-b:v 0',
|
|
|
|
],
|
|
|
|
twoPass: true,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-07-09 04:43:11 +02:00
|
|
|
it('should configure preset for vp9', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.VP9 },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_PRESET, value: 'slow' },
|
|
|
|
]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-07-09 04:43:11 +02:00
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v vp9',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-cpu-used 2',
|
|
|
|
'-row-mt 1',
|
|
|
|
'-crf 23',
|
|
|
|
'-b:v 0',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not configure preset for vp9 if invalid', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.VP9 },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_PRESET, value: 'invalid' },
|
|
|
|
]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-07-09 04:43:11 +02:00
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v vp9',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-row-mt 1',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-crf 23',
|
|
|
|
'-b:v 0',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should configure threads if above 0', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
2023-07-09 04:43:11 +02:00
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.VP9 },
|
2023-05-22 20:07:43 +02:00
|
|
|
{ key: SystemConfigKey.FFMPEG_THREADS, value: 2 },
|
|
|
|
]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-05-22 20:07:43 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-05-22 20:07:43 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v vp9',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-movflags faststart',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-cpu-used 5',
|
|
|
|
'-row-mt 1',
|
|
|
|
'-threads 2',
|
|
|
|
'-crf 23',
|
|
|
|
'-b:v 0',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-07-09 04:43:11 +02:00
|
|
|
it('should disable thread pooling for h264 if thread limit is above 0', async () => {
|
2023-05-22 20:07:43 +02:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_THREADS, value: 2 }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-05-22 20:07:43 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-05-22 20:07:43 +02:00
|
|
|
{
|
2023-07-09 04:43:11 +02:00
|
|
|
inputOptions: [],
|
2023-05-22 20:07:43 +02:00
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-movflags faststart',
|
2023-07-01 03:48:40 +02:00
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-05-22 20:07:43 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-threads 2',
|
|
|
|
'-x264-params "pools=none"',
|
|
|
|
'-x264-params "frame-threads=2"',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
2023-07-09 04:43:11 +02:00
|
|
|
|
|
|
|
it('should omit thread flags for h264 if thread limit is at or below 0', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_THREADS, value: 0 }]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-07-09 04:43:11 +02:00
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should disable thread pooling for hevc if thread limit is above 0', async () => {
|
2024-02-14 17:24:39 +01:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamVp9);
|
2023-07-09 04:43:11 +02:00
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_THREADS, value: 2 },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.HEVC },
|
|
|
|
]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-07-09 04:43:11 +02:00
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v hevc',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2024-01-29 03:17:20 +01:00
|
|
|
'-tag:v hvc1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-threads 2',
|
|
|
|
'-x265-params "pools=none"',
|
|
|
|
'-x265-params "frame-threads=2"',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should omit thread flags for hevc if thread limit is at or below 0', async () => {
|
2024-02-14 17:24:39 +01:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamVp9);
|
2023-07-09 04:43:11 +02:00
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_THREADS, value: 0 },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.HEVC },
|
|
|
|
]);
|
2023-08-01 03:28:07 +02:00
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
2023-07-09 04:43:11 +02:00
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-07-09 04:43:11 +02:00
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v hevc',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2024-01-29 03:17:20 +01:00
|
|
|
'-tag:v hvc1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-07-09 04:43:11 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
2023-08-02 03:56:10 +02:00
|
|
|
|
|
|
|
it('should skip transcoding for audioless videos with optimal policy if video codec is correct', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.noAudioStreams);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.HEVC },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.OPTIMAL },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_RESOLUTION, value: '1080p' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
it('should fail if hwaccel is enabled for an unsupported codec', async () => {
|
2023-08-02 03:56:10 +02:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.NVENC },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.VP9 },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
2024-03-15 14:16:54 +01:00
|
|
|
await expect(sut.handleVideoConversion({ id: assetStub.video.id })).resolves.toBe(JobStatus.FAILED);
|
2023-08-02 03:56:10 +02:00
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
it('should fail if hwaccel option is invalid', async () => {
|
2023-08-02 03:56:10 +02:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_ACCEL, value: 'invalid' }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
2024-03-15 14:16:54 +01:00
|
|
|
await expect(sut.handleVideoConversion({ id: assetStub.video.id })).resolves.toBe(JobStatus.FAILED);
|
2023-08-02 03:56:10 +02:00
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set two pass options for nvenc when enabled', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.NVENC },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '10000k' },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TWO_PASS, value: true },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device cuda=cuda:0', '-filter_hw_device cuda'],
|
|
|
|
outputOptions: [
|
|
|
|
'-tune hq',
|
|
|
|
'-qmin 0',
|
|
|
|
'-rc-lookahead 20',
|
|
|
|
'-i_qfactor 0.75',
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_nvenc`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf format=nv12,hwupload_cuda,scale_cuda=-2:720',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-preset p1',
|
|
|
|
'-b:v 6897k',
|
|
|
|
'-maxrate 10000k',
|
|
|
|
'-bufsize 6897k',
|
|
|
|
'-multipass 2',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set vbr options for nvenc when max bitrate is enabled', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.NVENC },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '10000k' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device cuda=cuda:0', '-filter_hw_device cuda'],
|
|
|
|
outputOptions: [
|
|
|
|
'-tune hq',
|
|
|
|
'-qmin 0',
|
|
|
|
'-rc-lookahead 20',
|
|
|
|
'-i_qfactor 0.75',
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_nvenc`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf format=nv12,hwupload_cuda,scale_cuda=-2:720',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-preset p1',
|
|
|
|
'-cq:v 23',
|
|
|
|
'-maxrate 10000k',
|
|
|
|
'-bufsize 6897k',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set cq options for nvenc when max bitrate is disabled', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.NVENC }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device cuda=cuda:0', '-filter_hw_device cuda'],
|
|
|
|
outputOptions: [
|
|
|
|
'-tune hq',
|
|
|
|
'-qmin 0',
|
|
|
|
'-rc-lookahead 20',
|
|
|
|
'-i_qfactor 0.75',
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_nvenc`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf format=nv12,hwupload_cuda,scale_cuda=-2:720',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-preset p1',
|
|
|
|
'-cq:v 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should omit preset for nvenc if invalid', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.NVENC },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_PRESET, value: 'invalid' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device cuda=cuda:0', '-filter_hw_device cuda'],
|
|
|
|
outputOptions: [
|
|
|
|
'-tune hq',
|
|
|
|
'-qmin 0',
|
|
|
|
'-rc-lookahead 20',
|
|
|
|
'-i_qfactor 0.75',
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_nvenc`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf format=nv12,hwupload_cuda,scale_cuda=-2:720',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-cq:v 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should ignore two pass for nvenc if max bitrate is disabled', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.NVENC }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device cuda=cuda:0', '-filter_hw_device cuda'],
|
|
|
|
outputOptions: [
|
|
|
|
'-tune hq',
|
|
|
|
'-qmin 0',
|
|
|
|
'-rc-lookahead 20',
|
|
|
|
'-i_qfactor 0.75',
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_nvenc`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf format=nv12,hwupload_cuda,scale_cuda=-2:720',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-preset p1',
|
|
|
|
'-cq:v 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set options for qsv', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.QSV },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '10000k' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device qsv=hw', '-filter_hw_device hw'],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_qsv`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-bf 7',
|
|
|
|
'-refs 5',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf format=nv12,hwupload=extra_hw_frames=64,scale_qsv=-1:720',
|
|
|
|
'-preset 7',
|
|
|
|
'-global_quality 23',
|
|
|
|
'-maxrate 10000k',
|
|
|
|
'-bufsize 20000k',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-01-30 02:40:02 +01:00
|
|
|
it('should set options for qsv with custom dri node', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.QSV },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '10000k' },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_PREFERRED_HW_DEVICE, value: '/dev/dri/renderD128' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device qsv=hw,child_device=/dev/dri/renderD128', '-filter_hw_device hw'],
|
|
|
|
outputOptions: [
|
|
|
|
`-c:v h264_qsv`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2024-01-30 02:40:02 +01:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-bf 7',
|
|
|
|
'-refs 5',
|
|
|
|
'-g 256',
|
|
|
|
'-v verbose',
|
|
|
|
'-vf format=nv12,hwupload=extra_hw_frames=64,scale_qsv=-1:720',
|
|
|
|
'-preset 7',
|
|
|
|
'-global_quality 23',
|
|
|
|
'-maxrate 10000k',
|
|
|
|
'-bufsize 20000k',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-08-02 03:56:10 +02:00
|
|
|
it('should omit preset for qsv if invalid', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.QSV },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_PRESET, value: 'invalid' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device qsv=hw', '-filter_hw_device hw'],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_qsv`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-bf 7',
|
|
|
|
'-refs 5',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf format=nv12,hwupload=extra_hw_frames=64,scale_qsv=-1:720',
|
|
|
|
'-global_quality 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set low power mode for qsv if target video codec is vp9', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.QSV },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.VP9 },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device qsv=hw', '-filter_hw_device hw'],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v vp9_qsv`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-bf 7',
|
|
|
|
'-refs 5',
|
|
|
|
'-g 256',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-low_power 1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf format=nv12,hwupload=extra_hw_frames=64,scale_qsv=-1:720',
|
|
|
|
'-preset 7',
|
|
|
|
'-q:v 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
it('should fail for qsv if no hw devices', async () => {
|
2023-08-02 03:56:10 +02:00
|
|
|
storageMock.readdir.mockResolvedValue([]);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.QSV }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
2024-03-15 14:16:54 +01:00
|
|
|
await expect(sut.handleVideoConversion({ id: assetStub.video.id })).resolves.toBe(JobStatus.FAILED);
|
2023-08-02 03:56:10 +02:00
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set vbr options for vaapi when max bitrate is enabled', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.VAAPI },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '10000k' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device vaapi=accel:/dev/dri/renderD128', '-filter_hw_device accel'],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_vaapi`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf format=nv12,hwupload,scale_vaapi=-2:720',
|
|
|
|
'-compression_level 7',
|
|
|
|
'-b:v 6897k',
|
|
|
|
'-maxrate 10000k',
|
|
|
|
'-minrate 3448.5k',
|
|
|
|
'-rc_mode 3',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set cq options for vaapi when max bitrate is disabled', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.VAAPI }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device vaapi=accel:/dev/dri/renderD128', '-filter_hw_device accel'],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_vaapi`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf format=nv12,hwupload,scale_vaapi=-2:720',
|
|
|
|
'-compression_level 7',
|
|
|
|
'-qp 23',
|
|
|
|
'-global_quality 23',
|
|
|
|
'-rc_mode 1',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should omit preset for vaapi if invalid', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.VAAPI },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_PRESET, value: 'invalid' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device vaapi=accel:/dev/dri/renderD128', '-filter_hw_device accel'],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_vaapi`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf format=nv12,hwupload,scale_vaapi=-2:720',
|
|
|
|
'-qp 23',
|
|
|
|
'-global_quality 23',
|
|
|
|
'-rc_mode 1',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should prefer gpu for vaapi if available', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD129', 'card1', 'card0', 'renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.VAAPI }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device vaapi=accel:/dev/dri/card1', '-filter_hw_device accel'],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_vaapi`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf format=nv12,hwupload,scale_vaapi=-2:720',
|
|
|
|
'-compression_level 7',
|
|
|
|
'-qp 23',
|
|
|
|
'-global_quality 23',
|
|
|
|
'-rc_mode 1',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD129', 'renderD128']);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device vaapi=accel:/dev/dri/renderD129', '-filter_hw_device accel'],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
`-c:v h264_vaapi`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf format=nv12,hwupload,scale_vaapi=-2:720',
|
|
|
|
'-compression_level 7',
|
|
|
|
'-qp 23',
|
|
|
|
'-global_quality 23',
|
|
|
|
'-rc_mode 1',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-01-30 02:40:02 +01:00
|
|
|
it('should select specific gpu node if selected', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD129', 'card1', 'card0', 'renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.VAAPI },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_PREFERRED_HW_DEVICE, value: '/dev/dri/renderD128' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
|
|
|
{
|
|
|
|
inputOptions: ['-init_hw_device vaapi=accel:/dev/dri/renderD128', '-filter_hw_device accel'],
|
|
|
|
outputOptions: [
|
|
|
|
`-c:v h264_vaapi`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2024-01-30 02:40:02 +01:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
|
|
|
'-v verbose',
|
|
|
|
'-vf format=nv12,hwupload,scale_vaapi=-2:720',
|
|
|
|
'-compression_level 7',
|
|
|
|
'-qp 23',
|
|
|
|
'-global_quality 23',
|
|
|
|
'-rc_mode 1',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-08-02 03:56:10 +02:00
|
|
|
it('should fallback to sw transcoding if hw transcoding fails', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.VAAPI }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
mediaMock.transcode.mockRejectedValueOnce(new Error('error'));
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledTimes(2);
|
|
|
|
expect(mediaMock.transcode).toHaveBeenLastCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-02 03:56:10 +02:00
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-v verbose',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-vf scale=-2:720,format=yuv420p',
|
2023-08-02 03:56:10 +02:00
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
it('should fail for vaapi if no hw devices', async () => {
|
2023-08-02 03:56:10 +02:00
|
|
|
storageMock.readdir.mockResolvedValue([]);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.VAAPI }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
2024-03-15 14:16:54 +01:00
|
|
|
await expect(sut.handleVideoConversion({ id: assetStub.video.id })).resolves.toBe(JobStatus.FAILED);
|
2023-08-02 03:56:10 +02:00
|
|
|
expect(mediaMock.transcode).not.toHaveBeenCalled();
|
|
|
|
});
|
2023-10-30 15:39:37 +01:00
|
|
|
|
|
|
|
it('should set vbr options for rkmpp when max bitrate is enabled', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
2024-02-14 17:24:39 +01:00
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamVp9);
|
2023-10-30 15:39:37 +01:00
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.RKMPP },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '10000k' },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_TARGET_VIDEO_CODEC, value: VideoCodec.HEVC },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
|
|
|
{
|
2024-02-28 02:32:07 +01:00
|
|
|
inputOptions: ['-hwaccel rkmpp', '-hwaccel_output_format drm_prime', '-afbc rga'],
|
2023-10-30 15:39:37 +01:00
|
|
|
outputOptions: [
|
2024-02-27 16:47:04 +01:00
|
|
|
`-c:v hevc_rkmpp`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-10-30 15:39:37 +01:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
2024-01-29 03:17:20 +01:00
|
|
|
'-tag:v hvc1',
|
2023-10-30 15:39:37 +01:00
|
|
|
'-v verbose',
|
2024-02-28 02:32:07 +01:00
|
|
|
'-vf scale_rkrga=-2:720:format=nv12:afbc=1',
|
2023-10-30 15:39:37 +01:00
|
|
|
'-level 153',
|
2024-02-28 02:32:07 +01:00
|
|
|
'-rc_mode AVBR',
|
2023-10-30 15:39:37 +01:00
|
|
|
'-b:v 10000k',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set cqp options for rkmpp when max bitrate is disabled', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.matroskaContainer);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.RKMPP },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_CRF, value: 30 },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '0' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
|
|
|
{
|
2024-02-28 02:32:07 +01:00
|
|
|
inputOptions: ['-hwaccel rkmpp', '-hwaccel_output_format drm_prime', '-afbc rga'],
|
2023-10-30 15:39:37 +01:00
|
|
|
outputOptions: [
|
2024-02-27 16:47:04 +01:00
|
|
|
`-c:v h264_rkmpp`,
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-10-30 15:39:37 +01:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
|
|
|
'-v verbose',
|
2024-02-28 02:32:07 +01:00
|
|
|
'-vf scale_rkrga=-2:720:format=nv12:afbc=1',
|
2023-10-30 15:39:37 +01:00
|
|
|
'-level 51',
|
2024-02-28 02:32:07 +01:00
|
|
|
'-rc_mode CQP',
|
2024-02-27 16:47:04 +01:00
|
|
|
'-qp_init 30',
|
2023-10-30 15:39:37 +01:00
|
|
|
],
|
|
|
|
twoPass: false,
|
2024-03-09 03:17:26 +01:00
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set OpenCL tonemapping options for rkmpp when OpenCL is available', async () => {
|
|
|
|
storageMock.readdir.mockResolvedValue(['renderD128']);
|
|
|
|
storageMock.stat.mockResolvedValue({ ...new Stats(), isFile: () => true, isCharacterDevice: () => true });
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamHDR);
|
|
|
|
configMock.load.mockResolvedValue([
|
|
|
|
{ key: SystemConfigKey.FFMPEG_ACCEL, value: TranscodeHWAccel.RKMPP },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_CRF, value: 30 },
|
|
|
|
{ key: SystemConfigKey.FFMPEG_MAX_BITRATE, value: '0' },
|
|
|
|
]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
|
|
|
{
|
|
|
|
inputOptions: ['-hwaccel rkmpp', '-hwaccel_output_format drm_prime', '-afbc rga'],
|
|
|
|
outputOptions: [
|
|
|
|
`-c:v h264_rkmpp`,
|
|
|
|
'-c:a copy',
|
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
|
|
|
'-g 256',
|
|
|
|
'-v verbose',
|
|
|
|
'-vf scale_rkrga=-2:720:format=p010:afbc=1,hwmap=derive_device=opencl:mode=read,tonemap_opencl=format=nv12:r=pc:p=bt709:t=bt709:m=bt709:tonemap=hable:desat=0,hwmap=derive_device=rkmpp:mode=write:reverse=1,format=drm_prime',
|
|
|
|
'-level 51',
|
|
|
|
'-rc_mode CQP',
|
|
|
|
'-qp_init 30',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
2023-10-30 15:39:37 +01:00
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
2023-04-04 16:48:02 +02:00
|
|
|
});
|
2023-08-07 22:35:25 +02:00
|
|
|
|
|
|
|
it('should tonemap when policy is required and video is hdr', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamHDR);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.REQUIRED }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-07 22:35:25 +02:00
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf zscale=t=linear:npl=100,tonemap=hable:desat=0,zscale=p=bt709:t=bt709:m=bt709:range=pc,format=yuv420p',
|
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should tonemap when policy is optimal and video is hdr', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamHDR);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TRANSCODE, value: TranscodePolicy.OPTIMAL }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-07 22:35:25 +02:00
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf zscale=t=linear:npl=100,tonemap=hable:desat=0,zscale=p=bt709:t=bt709:m=bt709:range=pc,format=yuv420p',
|
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set npl to 250 for reinhard and mobius tone-mapping algorithms', async () => {
|
|
|
|
mediaMock.probe.mockResolvedValue(probeStub.videoStreamHDR);
|
|
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.FFMPEG_TONEMAP, value: ToneMapping.MOBIUS }]);
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.video]);
|
|
|
|
await sut.handleVideoConversion({ id: assetStub.video.id });
|
|
|
|
expect(mediaMock.transcode).toHaveBeenCalledWith(
|
|
|
|
'/original/path.ext',
|
2023-09-25 17:07:21 +02:00
|
|
|
'upload/encoded-video/user-id/as/se/asset-id.mp4',
|
2023-08-07 22:35:25 +02:00
|
|
|
{
|
|
|
|
inputOptions: [],
|
|
|
|
outputOptions: [
|
2023-09-03 03:22:42 +02:00
|
|
|
'-c:v h264',
|
2024-02-21 06:25:30 +01:00
|
|
|
'-c:a copy',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-movflags faststart',
|
|
|
|
'-fps_mode passthrough',
|
2023-09-03 03:22:42 +02:00
|
|
|
'-map 0:0',
|
|
|
|
'-map 0:1',
|
2023-08-07 22:35:25 +02:00
|
|
|
'-v verbose',
|
|
|
|
'-vf zscale=t=linear:npl=250,tonemap=mobius:desat=0,zscale=p=bt709:t=bt709:m=bt709:range=pc,format=yuv420p',
|
|
|
|
'-preset ultrafast',
|
|
|
|
'-crf 23',
|
|
|
|
],
|
|
|
|
twoPass: false,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
2023-09-26 01:18:47 +02:00
|
|
|
|
|
|
|
describe('isSRGB', () => {
|
|
|
|
it('should return true for srgb colorspace', () => {
|
|
|
|
const asset = { ...assetStub.image, exifInfo: { colorspace: 'sRGB' } as ExifEntity };
|
|
|
|
expect(sut.isSRGB(asset)).toEqual(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return true for srgb profile description', () => {
|
|
|
|
const asset = { ...assetStub.image, exifInfo: { profileDescription: 'sRGB v1.31' } as ExifEntity };
|
|
|
|
expect(sut.isSRGB(asset)).toEqual(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return true for 8-bit image with no colorspace metadata', () => {
|
|
|
|
const asset = { ...assetStub.image, exifInfo: { bitsPerSample: 8 } as ExifEntity };
|
|
|
|
expect(sut.isSRGB(asset)).toEqual(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return true for image with no colorspace or bit depth metadata', () => {
|
|
|
|
const asset = { ...assetStub.image, exifInfo: {} as ExifEntity };
|
|
|
|
expect(sut.isSRGB(asset)).toEqual(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return false for non-srgb colorspace', () => {
|
|
|
|
const asset = { ...assetStub.image, exifInfo: { colorspace: 'Adobe RGB' } as ExifEntity };
|
|
|
|
expect(sut.isSRGB(asset)).toEqual(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return false for non-srgb profile description', () => {
|
|
|
|
const asset = { ...assetStub.image, exifInfo: { profileDescription: 'sP3C' } as ExifEntity };
|
|
|
|
expect(sut.isSRGB(asset)).toEqual(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return false for 16-bit image with no colorspace metadata', () => {
|
|
|
|
const asset = { ...assetStub.image, exifInfo: { bitsPerSample: 16 } as ExifEntity };
|
|
|
|
expect(sut.isSRGB(asset)).toEqual(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return true for 16-bit image with sRGB colorspace', () => {
|
|
|
|
const asset = { ...assetStub.image, exifInfo: { colorspace: 'sRGB', bitsPerSample: 16 } as ExifEntity };
|
|
|
|
expect(sut.isSRGB(asset)).toEqual(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return true for 16-bit image with sRGB profile', () => {
|
|
|
|
const asset = { ...assetStub.image, exifInfo: { profileDescription: 'sRGB', bitsPerSample: 16 } as ExifEntity };
|
|
|
|
expect(sut.isSRGB(asset)).toEqual(true);
|
|
|
|
});
|
|
|
|
});
|
2023-03-24 03:40:46 +01:00
|
|
|
});
|