2024-05-17 12:22:39 -04:00
|
|
|
import { DateTime } from 'luxon';
|
2024-10-08 23:08:49 +02:00
|
|
|
import { SemVer } from 'semver';
|
2024-05-17 12:22:39 -04:00
|
|
|
import { serverVersion } from 'src/constants';
|
2025-02-11 17:15:56 -05:00
|
|
|
import { ImmichEnvironment, JobName, JobStatus, SystemMetadataKey } from 'src/enum';
|
2024-05-17 12:22:39 -04:00
|
|
|
import { VersionService } from 'src/services/version.service';
|
2024-10-02 10:54:35 -04:00
|
|
|
import { mockEnvData } from 'test/repositories/config.repository.mock';
|
2025-03-10 16:52:44 -04:00
|
|
|
import { factory } from 'test/small.factory';
|
2025-02-10 18:47:42 -05:00
|
|
|
import { newTestService, ServiceMocks } from 'test/utils';
|
2024-05-17 12:22:39 -04:00
|
|
|
|
2024-05-20 20:31:36 -04:00
|
|
|
const mockRelease = (version: string) => ({
|
2024-05-17 12:22:39 -04:00
|
|
|
id: 1,
|
|
|
|
|
url: 'https://api.github.com/repos/owner/repo/releases/1',
|
2024-05-20 20:31:36 -04:00
|
|
|
tag_name: version,
|
2024-05-17 12:22:39 -04:00
|
|
|
name: 'Release 1000',
|
|
|
|
|
created_at: DateTime.utc().toISO(),
|
|
|
|
|
published_at: DateTime.utc().toISO(),
|
|
|
|
|
body: '',
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
describe(VersionService.name, () => {
|
|
|
|
|
let sut: VersionService;
|
2025-02-10 18:47:42 -05:00
|
|
|
let mocks: ServiceMocks;
|
2024-05-17 12:22:39 -04:00
|
|
|
|
|
|
|
|
beforeEach(() => {
|
2025-02-10 18:47:42 -05:00
|
|
|
({ sut, mocks } = newTestService(VersionService));
|
2024-05-17 12:22:39 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should work', () => {
|
|
|
|
|
expect(sut).toBeDefined();
|
|
|
|
|
});
|
|
|
|
|
|
2024-10-01 13:33:58 -04:00
|
|
|
describe('onBootstrap', () => {
|
|
|
|
|
it('should record a new version', async () => {
|
2025-03-10 16:52:44 -04:00
|
|
|
mocks.versionHistory.getAll.mockResolvedValue([]);
|
|
|
|
|
mocks.versionHistory.getLatest.mockResolvedValue(void 0);
|
|
|
|
|
mocks.versionHistory.create.mockResolvedValue(factory.versionHistory());
|
|
|
|
|
|
2024-10-01 13:33:58 -04:00
|
|
|
await expect(sut.onBootstrap()).resolves.toBeUndefined();
|
2025-03-10 16:52:44 -04:00
|
|
|
|
2025-02-10 18:47:42 -05:00
|
|
|
expect(mocks.versionHistory.create).toHaveBeenCalledWith({ version: expect.any(String) });
|
2024-10-01 13:33:58 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should skip a duplicate version', async () => {
|
2025-02-10 18:47:42 -05:00
|
|
|
mocks.versionHistory.getLatest.mockResolvedValue({
|
2024-10-01 13:33:58 -04:00
|
|
|
id: 'version-1',
|
|
|
|
|
createdAt: new Date(),
|
|
|
|
|
version: serverVersion.toString(),
|
|
|
|
|
});
|
|
|
|
|
await expect(sut.onBootstrap()).resolves.toBeUndefined();
|
2025-02-10 18:47:42 -05:00
|
|
|
expect(mocks.versionHistory.create).not.toHaveBeenCalled();
|
2024-10-01 13:33:58 -04:00
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2024-05-17 12:22:39 -04:00
|
|
|
describe('getVersion', () => {
|
|
|
|
|
it('should respond the server version', () => {
|
2024-05-20 20:31:36 -04:00
|
|
|
expect(sut.getVersion()).toEqual({
|
|
|
|
|
major: serverVersion.major,
|
|
|
|
|
minor: serverVersion.minor,
|
|
|
|
|
patch: serverVersion.patch,
|
|
|
|
|
});
|
2024-05-17 12:22:39 -04:00
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2024-10-01 13:33:58 -04:00
|
|
|
describe('getVersionHistory', () => {
|
|
|
|
|
it('should respond the server version history', async () => {
|
|
|
|
|
const upgrade = { id: 'upgrade-1', createdAt: new Date(), version: '1.0.0' };
|
2025-02-10 18:47:42 -05:00
|
|
|
mocks.versionHistory.getAll.mockResolvedValue([upgrade]);
|
2024-10-01 13:33:58 -04:00
|
|
|
await expect(sut.getVersionHistory()).resolves.toEqual([upgrade]);
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2024-05-17 12:22:39 -04:00
|
|
|
describe('handQueueVersionCheck', () => {
|
|
|
|
|
it('should queue a version check job', async () => {
|
|
|
|
|
await expect(sut.handleQueueVersionCheck()).resolves.toBeUndefined();
|
2025-07-15 14:50:13 -04:00
|
|
|
expect(mocks.job.queue).toHaveBeenCalledWith({ name: JobName.VersionCheck, data: {} });
|
2024-05-17 12:22:39 -04:00
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
describe('handVersionCheck', () => {
|
|
|
|
|
beforeEach(() => {
|
2025-07-15 14:50:13 -04:00
|
|
|
mocks.config.getEnv.mockReturnValue(mockEnvData({ environment: ImmichEnvironment.Production }));
|
2024-05-17 12:22:39 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should not run in dev mode', async () => {
|
2025-07-15 14:50:13 -04:00
|
|
|
mocks.config.getEnv.mockReturnValue(mockEnvData({ environment: ImmichEnvironment.Development }));
|
|
|
|
|
await expect(sut.handleVersionCheck()).resolves.toEqual(JobStatus.Skipped);
|
2024-05-17 12:22:39 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should not run if the last check was < 60 minutes ago', async () => {
|
2025-02-10 18:47:42 -05:00
|
|
|
mocks.systemMetadata.get.mockResolvedValue({
|
2024-05-17 12:22:39 -04:00
|
|
|
checkedAt: DateTime.utc().minus({ minutes: 5 }).toISO(),
|
|
|
|
|
releaseVersion: '1.0.0',
|
|
|
|
|
});
|
2025-07-15 14:50:13 -04:00
|
|
|
await expect(sut.handleVersionCheck()).resolves.toEqual(JobStatus.Skipped);
|
2024-05-17 12:22:39 -04:00
|
|
|
});
|
|
|
|
|
|
2024-10-08 23:08:49 +02:00
|
|
|
it('should not run if version check is disabled', async () => {
|
2025-02-10 18:47:42 -05:00
|
|
|
mocks.systemMetadata.get.mockResolvedValue({ newVersionCheck: { enabled: false } });
|
2025-07-15 14:50:13 -04:00
|
|
|
await expect(sut.handleVersionCheck()).resolves.toEqual(JobStatus.Skipped);
|
2024-10-08 23:08:49 +02:00
|
|
|
});
|
|
|
|
|
|
2024-05-17 12:22:39 -04:00
|
|
|
it('should run if it has been > 60 minutes', async () => {
|
2025-02-10 18:47:42 -05:00
|
|
|
mocks.serverInfo.getGitHubRelease.mockResolvedValue(mockRelease('v100.0.0'));
|
|
|
|
|
mocks.systemMetadata.get.mockResolvedValue({
|
2024-05-17 12:22:39 -04:00
|
|
|
checkedAt: DateTime.utc().minus({ minutes: 65 }).toISO(),
|
|
|
|
|
releaseVersion: '1.0.0',
|
|
|
|
|
});
|
2025-07-15 14:50:13 -04:00
|
|
|
await expect(sut.handleVersionCheck()).resolves.toEqual(JobStatus.Success);
|
2025-02-10 18:47:42 -05:00
|
|
|
expect(mocks.systemMetadata.set).toHaveBeenCalled();
|
|
|
|
|
expect(mocks.logger.log).toHaveBeenCalled();
|
|
|
|
|
expect(mocks.event.clientBroadcast).toHaveBeenCalled();
|
2024-05-17 12:22:39 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should not notify if the version is equal', async () => {
|
2025-02-10 18:47:42 -05:00
|
|
|
mocks.serverInfo.getGitHubRelease.mockResolvedValue(mockRelease(serverVersion.toString()));
|
2025-07-15 14:50:13 -04:00
|
|
|
await expect(sut.handleVersionCheck()).resolves.toEqual(JobStatus.Success);
|
|
|
|
|
expect(mocks.systemMetadata.set).toHaveBeenCalledWith(SystemMetadataKey.VersionCheckState, {
|
2024-05-17 12:22:39 -04:00
|
|
|
checkedAt: expect.any(String),
|
|
|
|
|
releaseVersion: serverVersion.toString(),
|
|
|
|
|
});
|
2025-02-10 18:47:42 -05:00
|
|
|
expect(mocks.event.clientBroadcast).not.toHaveBeenCalled();
|
2024-05-17 12:22:39 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should handle a github error', async () => {
|
2025-02-10 18:47:42 -05:00
|
|
|
mocks.serverInfo.getGitHubRelease.mockRejectedValue(new Error('GitHub is down'));
|
2025-07-15 14:50:13 -04:00
|
|
|
await expect(sut.handleVersionCheck()).resolves.toEqual(JobStatus.Failed);
|
2025-02-10 18:47:42 -05:00
|
|
|
expect(mocks.systemMetadata.set).not.toHaveBeenCalled();
|
|
|
|
|
expect(mocks.event.clientBroadcast).not.toHaveBeenCalled();
|
|
|
|
|
expect(mocks.logger.warn).toHaveBeenCalled();
|
2024-05-17 12:22:39 -04:00
|
|
|
});
|
|
|
|
|
});
|
2024-10-08 23:08:49 +02:00
|
|
|
|
|
|
|
|
describe('onWebsocketConnectionEvent', () => {
|
|
|
|
|
it('should send on_server_version client event', async () => {
|
|
|
|
|
await sut.onWebsocketConnection({ userId: '42' });
|
2025-02-10 18:47:42 -05:00
|
|
|
expect(mocks.event.clientSend).toHaveBeenCalledWith('on_server_version', '42', expect.any(SemVer));
|
|
|
|
|
expect(mocks.event.clientSend).toHaveBeenCalledTimes(1);
|
2024-10-08 23:08:49 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should also send a new release notification', async () => {
|
2025-02-10 18:47:42 -05:00
|
|
|
mocks.systemMetadata.get.mockResolvedValue({ checkedAt: '2024-01-01', releaseVersion: 'v1.42.0' });
|
2024-10-08 23:08:49 +02:00
|
|
|
await sut.onWebsocketConnection({ userId: '42' });
|
2025-02-10 18:47:42 -05:00
|
|
|
expect(mocks.event.clientSend).toHaveBeenCalledWith('on_server_version', '42', expect.any(SemVer));
|
|
|
|
|
expect(mocks.event.clientSend).toHaveBeenCalledWith('on_new_release', '42', expect.any(Object));
|
2024-10-08 23:08:49 +02:00
|
|
|
});
|
|
|
|
|
});
|
2024-05-17 12:22:39 -04:00
|
|
|
});
|