2023-12-14 11:55:40 -05:00
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
2024-06-26 08:25:09 -04:00
|
|
|
import { getBuildMetadata } from 'src/config';
|
|
|
|
|
import { serverVersion } from 'src/constants';
|
2024-03-20 21:20:38 +01:00
|
|
|
import { StorageCore, StorageFolder } from 'src/cores/storage.core';
|
|
|
|
|
import { SystemConfigCore } from 'src/cores/system-config.core';
|
2023-07-15 20:24:46 -05:00
|
|
|
import {
|
2024-06-26 08:25:09 -04:00
|
|
|
ServerAboutResponseDto,
|
2023-09-08 22:51:46 -04:00
|
|
|
ServerConfigDto,
|
2023-08-18 00:55:26 -04:00
|
|
|
ServerFeaturesDto,
|
2023-07-15 20:24:46 -05:00
|
|
|
ServerMediaTypesResponseDto,
|
|
|
|
|
ServerPingResponse,
|
|
|
|
|
ServerStatsResponseDto,
|
2024-05-22 10:25:55 +01:00
|
|
|
ServerStorageResponseDto,
|
2023-07-15 20:24:46 -05:00
|
|
|
UsageByUserDto,
|
2024-03-20 23:53:07 +01:00
|
|
|
} from 'src/dtos/server-info.dto';
|
2024-03-20 16:02:51 -05:00
|
|
|
import { SystemMetadataKey } from 'src/entities/system-metadata.entity';
|
2024-06-27 15:54:20 -04:00
|
|
|
import { OnEvents } from 'src/interfaces/event.interface';
|
2024-04-17 03:00:31 +05:30
|
|
|
import { ILoggerRepository } from 'src/interfaces/logger.interface';
|
2024-06-26 08:25:09 -04:00
|
|
|
import { IServerInfoRepository } from 'src/interfaces/server-info.interface';
|
2024-03-21 12:59:49 +01:00
|
|
|
import { IStorageRepository } from 'src/interfaces/storage.interface';
|
|
|
|
|
import { ISystemMetadataRepository } from 'src/interfaces/system-metadata.interface';
|
|
|
|
|
import { IUserRepository, UserStatsQueryResponse } from 'src/interfaces/user.interface';
|
2024-03-20 22:15:09 -05:00
|
|
|
import { asHumanReadable } from 'src/utils/bytes';
|
|
|
|
|
import { mimeTypes } from 'src/utils/mime-types';
|
2024-05-16 13:08:37 -04:00
|
|
|
import { isDuplicateDetectionEnabled, isFacialRecognitionEnabled, isSmartSearchEnabled } from 'src/utils/misc';
|
2023-03-21 22:49:19 -04:00
|
|
|
|
|
|
|
|
@Injectable()
|
2024-06-27 15:54:20 -04:00
|
|
|
export class ServerInfoService implements OnEvents {
|
2023-08-18 00:55:26 -04:00
|
|
|
private configCore: SystemConfigCore;
|
2023-05-24 22:05:31 -04:00
|
|
|
|
2023-03-21 22:49:19 -04:00
|
|
|
constructor(
|
|
|
|
|
@Inject(IUserRepository) private userRepository: IUserRepository,
|
|
|
|
|
@Inject(IStorageRepository) private storageRepository: IStorageRepository,
|
2024-04-17 03:00:31 +05:30
|
|
|
@Inject(ISystemMetadataRepository) private systemMetadataRepository: ISystemMetadataRepository,
|
2024-06-26 08:25:09 -04:00
|
|
|
@Inject(IServerInfoRepository) private serverInfoRepository: IServerInfoRepository,
|
2024-04-17 03:00:31 +05:30
|
|
|
@Inject(ILoggerRepository) private logger: ILoggerRepository,
|
2023-08-18 00:55:26 -04:00
|
|
|
) {
|
2024-04-17 03:00:31 +05:30
|
|
|
this.logger.setContext(ServerInfoService.name);
|
2024-05-15 18:58:23 -04:00
|
|
|
this.configCore = SystemConfigCore.create(systemMetadataRepository, this.logger);
|
2023-08-18 00:55:26 -04:00
|
|
|
}
|
2023-03-21 22:49:19 -04:00
|
|
|
|
2024-06-27 15:54:20 -04:00
|
|
|
async onBootstrapEvent(): Promise<void> {
|
2024-01-08 16:22:26 +01:00
|
|
|
const featureFlags = await this.getFeatures();
|
|
|
|
|
if (featureFlags.configFile) {
|
2024-04-19 20:36:15 -04:00
|
|
|
await this.systemMetadataRepository.set(SystemMetadataKey.ADMIN_ONBOARDING, {
|
|
|
|
|
isOnboarded: true,
|
|
|
|
|
});
|
2024-01-08 16:22:26 +01:00
|
|
|
}
|
2024-06-27 15:54:20 -04:00
|
|
|
this.logger.log(`Feature Flags: ${JSON.stringify(await this.getFeatures(), null, 2)}`);
|
2024-01-08 16:22:26 +01:00
|
|
|
}
|
|
|
|
|
|
2024-06-26 08:25:09 -04:00
|
|
|
async getAboutInfo(): Promise<ServerAboutResponseDto> {
|
2024-06-27 19:36:25 +02:00
|
|
|
const version = `v${serverVersion.toString()}`;
|
2024-06-26 08:25:09 -04:00
|
|
|
const buildMetadata = getBuildMetadata();
|
|
|
|
|
const buildVersions = await this.serverInfoRepository.getBuildVersions();
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
version,
|
|
|
|
|
versionUrl: `https://github.com/immich-app/immich/releases/tag/${version}`,
|
|
|
|
|
...buildMetadata,
|
|
|
|
|
...buildVersions,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-22 10:25:55 +01:00
|
|
|
async getStorage(): Promise<ServerStorageResponseDto> {
|
2023-10-14 13:12:59 -04:00
|
|
|
const libraryBase = StorageCore.getBaseFolder(StorageFolder.LIBRARY);
|
2023-05-24 22:05:31 -04:00
|
|
|
const diskInfo = await this.storageRepository.checkDiskUsage(libraryBase);
|
2023-03-21 22:49:19 -04:00
|
|
|
|
|
|
|
|
const usagePercentage = (((diskInfo.total - diskInfo.free) / diskInfo.total) * 100).toFixed(2);
|
|
|
|
|
|
2024-05-22 10:25:55 +01:00
|
|
|
const serverInfo = new ServerStorageResponseDto();
|
2023-03-21 22:49:19 -04:00
|
|
|
serverInfo.diskAvailable = asHumanReadable(diskInfo.available);
|
|
|
|
|
serverInfo.diskSize = asHumanReadable(diskInfo.total);
|
|
|
|
|
serverInfo.diskUse = asHumanReadable(diskInfo.total - diskInfo.free);
|
|
|
|
|
serverInfo.diskAvailableRaw = diskInfo.available;
|
|
|
|
|
serverInfo.diskSizeRaw = diskInfo.total;
|
|
|
|
|
serverInfo.diskUseRaw = diskInfo.total - diskInfo.free;
|
2024-02-02 04:18:00 +01:00
|
|
|
serverInfo.diskUsagePercentage = Number.parseFloat(usagePercentage);
|
2023-03-21 22:49:19 -04:00
|
|
|
return serverInfo;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ping(): ServerPingResponse {
|
2023-08-18 00:55:26 -04:00
|
|
|
return { res: 'pong' };
|
2023-03-21 22:49:19 -04:00
|
|
|
}
|
|
|
|
|
|
2024-05-14 14:43:49 -04:00
|
|
|
async getFeatures(): Promise<ServerFeaturesDto> {
|
2024-05-14 15:31:36 -04:00
|
|
|
const { reverseGeocoding, map, machineLearning, trash, oauth, passwordLogin, notifications } =
|
2024-06-12 07:07:35 -04:00
|
|
|
await this.configCore.getConfig({ withCache: false });
|
2024-05-14 15:31:36 -04:00
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
smartSearch: isSmartSearchEnabled(machineLearning),
|
|
|
|
|
facialRecognition: isFacialRecognitionEnabled(machineLearning),
|
2024-05-16 13:08:37 -04:00
|
|
|
duplicateDetection: isDuplicateDetectionEnabled(machineLearning),
|
2024-05-14 15:31:36 -04:00
|
|
|
map: map.enabled,
|
|
|
|
|
reverseGeocoding: reverseGeocoding.enabled,
|
|
|
|
|
sidecar: true,
|
|
|
|
|
search: true,
|
|
|
|
|
trash: trash.enabled,
|
|
|
|
|
oauth: oauth.enabled,
|
|
|
|
|
oauthAutoLaunch: oauth.autoLaunch,
|
|
|
|
|
passwordLogin: passwordLogin.enabled,
|
|
|
|
|
configFile: this.configCore.isUsingConfigFile(),
|
|
|
|
|
email: notifications.smtp.enabled,
|
|
|
|
|
};
|
2023-08-18 00:55:26 -04:00
|
|
|
}
|
|
|
|
|
|
2023-10-25 15:13:05 -07:00
|
|
|
async getTheme() {
|
2024-06-12 07:07:35 -04:00
|
|
|
const { theme } = await this.configCore.getConfig({ withCache: false });
|
2023-10-26 19:32:33 -07:00
|
|
|
return theme;
|
2023-10-25 15:13:05 -07:00
|
|
|
}
|
|
|
|
|
|
2023-09-08 22:51:46 -04:00
|
|
|
async getConfig(): Promise<ServerConfigDto> {
|
2024-06-12 07:07:35 -04:00
|
|
|
const config = await this.configCore.getConfig({ withCache: false });
|
2023-10-11 04:37:13 +02:00
|
|
|
const isInitialized = await this.userRepository.hasAdmin();
|
2024-01-03 23:28:32 -06:00
|
|
|
const onboarding = await this.systemMetadataRepository.get(SystemMetadataKey.ADMIN_ONBOARDING);
|
2023-10-11 04:37:13 +02:00
|
|
|
|
2023-09-08 22:51:46 -04:00
|
|
|
return {
|
2024-01-04 00:00:17 -05:00
|
|
|
loginPageMessage: config.server.loginPageMessage,
|
2023-10-06 07:01:14 +00:00
|
|
|
trashDays: config.trash.days,
|
2024-03-06 00:45:40 -05:00
|
|
|
userDeleteDelay: config.user.deleteDelay,
|
2023-09-08 22:51:46 -04:00
|
|
|
oauthButtonText: config.oauth.buttonText,
|
2023-10-11 04:37:13 +02:00
|
|
|
isInitialized,
|
2024-01-03 23:28:32 -06:00
|
|
|
isOnboarded: onboarding?.isOnboarded || false,
|
2024-01-03 21:54:48 -05:00
|
|
|
externalDomain: config.server.externalDomain,
|
2023-09-08 22:51:46 -04:00
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-03 21:33:15 -04:00
|
|
|
async getStatistics(): Promise<ServerStatsResponseDto> {
|
2023-03-21 22:49:19 -04:00
|
|
|
const userStats: UserStatsQueryResponse[] = await this.userRepository.getUserStats();
|
|
|
|
|
const serverStats = new ServerStatsResponseDto();
|
|
|
|
|
|
|
|
|
|
for (const user of userStats) {
|
|
|
|
|
const usage = new UsageByUserDto();
|
|
|
|
|
usage.userId = user.userId;
|
2023-11-11 20:03:32 -05:00
|
|
|
usage.userName = user.userName;
|
2023-03-21 22:49:19 -04:00
|
|
|
usage.photos = user.photos;
|
|
|
|
|
usage.videos = user.videos;
|
|
|
|
|
usage.usage = user.usage;
|
2024-01-12 18:43:36 -06:00
|
|
|
usage.quotaSizeInBytes = user.quotaSizeInBytes;
|
2023-03-21 22:49:19 -04:00
|
|
|
|
|
|
|
|
serverStats.photos += usage.photos;
|
|
|
|
|
serverStats.videos += usage.videos;
|
|
|
|
|
serverStats.usage += usage.usage;
|
|
|
|
|
serverStats.usageByUser.push(usage);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return serverStats;
|
|
|
|
|
}
|
2023-07-15 20:24:46 -05:00
|
|
|
|
|
|
|
|
getSupportedMediaTypes(): ServerMediaTypesResponseDto {
|
|
|
|
|
return {
|
2023-08-15 12:05:32 -04:00
|
|
|
video: Object.keys(mimeTypes.video),
|
|
|
|
|
image: Object.keys(mimeTypes.image),
|
|
|
|
|
sidecar: Object.keys(mimeTypes.sidecar),
|
2023-07-15 20:24:46 -05:00
|
|
|
};
|
|
|
|
|
}
|
2023-03-21 22:49:19 -04:00
|
|
|
}
|