2023-03-20 16:55:28 +01:00
|
|
|
import { BadRequestException } from '@nestjs/common';
|
2023-05-26 21:43:24 +02:00
|
|
|
import { newAssetRepositoryMock, newCommunicationRepositoryMock, newJobRepositoryMock } from '../../test';
|
|
|
|
import { IAssetRepository } from '../asset';
|
|
|
|
import { ICommunicationRepository } from '../communication';
|
2023-03-20 16:55:28 +01:00
|
|
|
import { IJobRepository, JobCommand, JobName, JobService, QueueName } from '../job';
|
|
|
|
|
|
|
|
describe(JobService.name, () => {
|
|
|
|
let sut: JobService;
|
2023-05-26 21:43:24 +02:00
|
|
|
let assetMock: jest.Mocked<IAssetRepository>;
|
|
|
|
let communicationMock: jest.Mocked<ICommunicationRepository>;
|
2023-03-20 16:55:28 +01:00
|
|
|
let jobMock: jest.Mocked<IJobRepository>;
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
2023-05-26 21:43:24 +02:00
|
|
|
assetMock = newAssetRepositoryMock();
|
|
|
|
communicationMock = newCommunicationRepositoryMock();
|
2023-03-20 16:55:28 +01:00
|
|
|
jobMock = newJobRepositoryMock();
|
2023-05-26 21:43:24 +02:00
|
|
|
sut = new JobService(assetMock, communicationMock, jobMock);
|
2023-03-20 16:55:28 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should work', () => {
|
|
|
|
expect(sut).toBeDefined();
|
|
|
|
});
|
|
|
|
|
2023-05-17 19:07:17 +02:00
|
|
|
describe('handleNightlyJobs', () => {
|
|
|
|
it('should run the scheduled jobs', async () => {
|
|
|
|
await sut.handleNightlyJobs();
|
|
|
|
|
|
|
|
expect(jobMock.queue.mock.calls).toEqual([
|
|
|
|
[{ name: JobName.USER_DELETE_CHECK }],
|
|
|
|
[{ name: JobName.PERSON_CLEANUP }],
|
2023-05-22 04:24:21 +02:00
|
|
|
[{ name: JobName.QUEUE_GENERATE_THUMBNAILS, data: { force: false } }],
|
2023-05-17 19:07:17 +02:00
|
|
|
]);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2023-03-20 16:55:28 +01:00
|
|
|
describe('getAllJobStatus', () => {
|
|
|
|
it('should get all job statuses', async () => {
|
|
|
|
jobMock.getJobCounts.mockResolvedValue({
|
|
|
|
active: 1,
|
|
|
|
completed: 1,
|
|
|
|
failed: 1,
|
|
|
|
delayed: 1,
|
|
|
|
waiting: 1,
|
2023-03-29 17:33:03 +02:00
|
|
|
paused: 1,
|
2023-03-20 16:55:28 +01:00
|
|
|
});
|
2023-04-01 22:46:07 +02:00
|
|
|
jobMock.getQueueStatus.mockResolvedValue({
|
|
|
|
isActive: true,
|
|
|
|
isPaused: true,
|
|
|
|
});
|
2023-03-20 16:55:28 +01:00
|
|
|
|
2023-04-01 22:46:07 +02:00
|
|
|
const expectedJobStatus = {
|
|
|
|
jobCounts: {
|
2023-03-20 16:55:28 +01:00
|
|
|
active: 1,
|
|
|
|
completed: 1,
|
|
|
|
delayed: 1,
|
|
|
|
failed: 1,
|
|
|
|
waiting: 1,
|
2023-03-29 17:33:03 +02:00
|
|
|
paused: 1,
|
2023-03-20 16:55:28 +01:00
|
|
|
},
|
2023-04-01 22:46:07 +02:00
|
|
|
queueStatus: {
|
|
|
|
isActive: true,
|
|
|
|
isPaused: true,
|
2023-03-20 16:55:28 +01:00
|
|
|
},
|
2023-04-01 22:46:07 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
await expect(sut.getAllJobsStatus()).resolves.toEqual({
|
|
|
|
'background-task-queue': expectedJobStatus,
|
|
|
|
'clip-encoding-queue': expectedJobStatus,
|
|
|
|
'metadata-extraction-queue': expectedJobStatus,
|
|
|
|
'object-tagging-queue': expectedJobStatus,
|
|
|
|
'search-queue': expectedJobStatus,
|
|
|
|
'storage-template-migration-queue': expectedJobStatus,
|
|
|
|
'thumbnail-generation-queue': expectedJobStatus,
|
|
|
|
'video-conversion-queue': expectedJobStatus,
|
2023-05-17 19:07:17 +02:00
|
|
|
'recognize-faces-queue': expectedJobStatus,
|
feat(server): xmp sidecar metadata (#2466)
* initial commit for XMP sidecar support
* Added support for 'missing' metadata files to include those without sidecar files, now detects sidecar files in the filesystem for media already ingested but the sidecar was created afterwards
* didn't mean to commit default log level during testing
* new sidecar logic for video metadata as well
* Added xml mimetype for sidecars only
* don't need capture group for this regex
* wrong default value reverted
* simplified the move here - keep it in the same try catch since the outcome is to move the media back anyway
* simplified setter logic
Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
* simplified logic per suggestions
* sidecar is now its own queue with a discover and sync, updated UI for the new job queueing
* queue a sidecar job for every asset based on discovery or sync, though the logic is almost identical aside from linking the sidecar
* now queue sidecar jobs for each assset, though logic is mostly the same between discovery and sync
* simplified logic of filename extraction and asset instantiation
* not sure how that got deleted..
* updated code per suggestions and comments in the PR
* stat was not being used, removed the variable set
* better type checking, using in-scope variables for exif getter instead of passing in every time
* removed commented out test
* ran and resolved all lints, formats, checks, and tests
* resolved suggested change in PR
* made getExifProperty more dynamic with multiple possible args for fallbacks, fixed typo, used generic in function for better type checking
* better error handling and moving files back to positions on move or save failure
* regenerated api
* format fixes
* Added XMP documentation
* documentation typo
* Merged in main
* missed merge conflict
* more changes due to a merge
* Resolving conflicts
* added icon for sidecar jobs
---------
Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
2023-05-25 03:59:30 +02:00
|
|
|
'sidecar-queue': expectedJobStatus,
|
2023-03-20 16:55:28 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('handleCommand', () => {
|
|
|
|
it('should handle a pause command', async () => {
|
|
|
|
await sut.handleCommand(QueueName.METADATA_EXTRACTION, { command: JobCommand.PAUSE, force: false });
|
|
|
|
|
|
|
|
expect(jobMock.pause).toHaveBeenCalledWith(QueueName.METADATA_EXTRACTION);
|
|
|
|
});
|
|
|
|
|
2023-03-28 20:25:22 +02:00
|
|
|
it('should handle a resume command', async () => {
|
|
|
|
await sut.handleCommand(QueueName.METADATA_EXTRACTION, { command: JobCommand.RESUME, force: false });
|
|
|
|
|
|
|
|
expect(jobMock.resume).toHaveBeenCalledWith(QueueName.METADATA_EXTRACTION);
|
|
|
|
});
|
|
|
|
|
2023-03-20 16:55:28 +01:00
|
|
|
it('should handle an empty command', async () => {
|
|
|
|
await sut.handleCommand(QueueName.METADATA_EXTRACTION, { command: JobCommand.EMPTY, force: false });
|
|
|
|
|
|
|
|
expect(jobMock.empty).toHaveBeenCalledWith(QueueName.METADATA_EXTRACTION);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not start a job that is already running', async () => {
|
2023-04-01 22:46:07 +02:00
|
|
|
jobMock.getQueueStatus.mockResolvedValue({ isActive: true, isPaused: false });
|
2023-03-20 16:55:28 +01:00
|
|
|
|
|
|
|
await expect(
|
|
|
|
sut.handleCommand(QueueName.VIDEO_CONVERSION, { command: JobCommand.START, force: false }),
|
|
|
|
).rejects.toBeInstanceOf(BadRequestException);
|
|
|
|
|
|
|
|
expect(jobMock.queue).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle a start video conversion command', async () => {
|
2023-04-01 22:46:07 +02:00
|
|
|
jobMock.getQueueStatus.mockResolvedValue({ isActive: false, isPaused: false });
|
2023-03-20 16:55:28 +01:00
|
|
|
|
|
|
|
await sut.handleCommand(QueueName.VIDEO_CONVERSION, { command: JobCommand.START, force: false });
|
|
|
|
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({ name: JobName.QUEUE_VIDEO_CONVERSION, data: { force: false } });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle a start storage template migration command', async () => {
|
2023-04-01 22:46:07 +02:00
|
|
|
jobMock.getQueueStatus.mockResolvedValue({ isActive: false, isPaused: false });
|
2023-03-20 16:55:28 +01:00
|
|
|
|
|
|
|
await sut.handleCommand(QueueName.STORAGE_TEMPLATE_MIGRATION, { command: JobCommand.START, force: false });
|
|
|
|
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({ name: JobName.STORAGE_TEMPLATE_MIGRATION });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle a start object tagging command', async () => {
|
2023-04-01 22:46:07 +02:00
|
|
|
jobMock.getQueueStatus.mockResolvedValue({ isActive: false, isPaused: false });
|
2023-03-20 16:55:28 +01:00
|
|
|
|
|
|
|
await sut.handleCommand(QueueName.OBJECT_TAGGING, { command: JobCommand.START, force: false });
|
|
|
|
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({ name: JobName.QUEUE_OBJECT_TAGGING, data: { force: false } });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle a start clip encoding command', async () => {
|
2023-04-01 22:46:07 +02:00
|
|
|
jobMock.getQueueStatus.mockResolvedValue({ isActive: false, isPaused: false });
|
2023-03-20 16:55:28 +01:00
|
|
|
|
|
|
|
await sut.handleCommand(QueueName.CLIP_ENCODING, { command: JobCommand.START, force: false });
|
|
|
|
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({ name: JobName.QUEUE_ENCODE_CLIP, data: { force: false } });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle a start metadata extraction command', async () => {
|
2023-04-01 22:46:07 +02:00
|
|
|
jobMock.getQueueStatus.mockResolvedValue({ isActive: false, isPaused: false });
|
2023-03-20 16:55:28 +01:00
|
|
|
|
|
|
|
await sut.handleCommand(QueueName.METADATA_EXTRACTION, { command: JobCommand.START, force: false });
|
|
|
|
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({ name: JobName.QUEUE_METADATA_EXTRACTION, data: { force: false } });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle a start thumbnail generation command', async () => {
|
2023-04-01 22:46:07 +02:00
|
|
|
jobMock.getQueueStatus.mockResolvedValue({ isActive: false, isPaused: false });
|
2023-03-20 16:55:28 +01:00
|
|
|
|
|
|
|
await sut.handleCommand(QueueName.THUMBNAIL_GENERATION, { command: JobCommand.START, force: false });
|
|
|
|
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({ name: JobName.QUEUE_GENERATE_THUMBNAILS, data: { force: false } });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should throw a bad request when an invalid queue is used', async () => {
|
2023-04-01 22:46:07 +02:00
|
|
|
jobMock.getQueueStatus.mockResolvedValue({ isActive: false, isPaused: false });
|
2023-03-20 16:55:28 +01:00
|
|
|
|
|
|
|
await expect(
|
|
|
|
sut.handleCommand(QueueName.BACKGROUND_TASK, { command: JobCommand.START, force: false }),
|
|
|
|
).rejects.toBeInstanceOf(BadRequestException);
|
|
|
|
|
|
|
|
expect(jobMock.queue).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|