mirror of
https://github.com/immich-app/immich.git
synced 2025-01-10 05:46:46 +01:00
9539a361e4
* custom `IsOptional` * added link to source * formatting * Update server/src/domain/domain.util.ts Co-authored-by: Jason Rasmussen <jrasm91@gmail.com> * nullable birth date endpoint * made `nullable` a property * formatting * removed unused dto * updated decorator arg * fixed album e2e tests * add null tests for auth e2e * add null test for person e2e * fixed tests * added null test for user e2e * removed unusued import * log key in test name * chore: add note about mobile not being able to use the endpoint --------- Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
114 lines
3 KiB
TypeScript
114 lines
3 KiB
TypeScript
import { applyDecorators } from '@nestjs/common';
|
|
import { ApiProperty } from '@nestjs/swagger';
|
|
import { IsArray, IsNotEmpty, IsOptional, IsString, IsUUID, ValidationOptions, ValidateIf } from 'class-validator';
|
|
import { basename, extname } from 'node:path';
|
|
import sanitize from 'sanitize-filename';
|
|
|
|
export type Options = {
|
|
optional?: boolean;
|
|
each?: boolean;
|
|
};
|
|
|
|
export function ValidateUUID({ optional, each }: Options = { optional: false, each: false }) {
|
|
return applyDecorators(
|
|
IsUUID('4', { each }),
|
|
ApiProperty({ format: 'uuid' }),
|
|
optional ? Optional() : IsNotEmpty(),
|
|
each ? IsArray() : IsString(),
|
|
);
|
|
}
|
|
|
|
interface IValue {
|
|
value?: string;
|
|
}
|
|
|
|
export const toBoolean = ({ value }: IValue) => {
|
|
if (value == 'true') {
|
|
return true;
|
|
} else if (value == 'false') {
|
|
return false;
|
|
}
|
|
return value;
|
|
};
|
|
|
|
export const toEmail = ({ value }: IValue) => value?.toLowerCase();
|
|
|
|
export const toSanitized = ({ value }: IValue) => sanitize((value || '').replace(/\./g, ''));
|
|
|
|
export function getFileNameWithoutExtension(path: string): string {
|
|
return basename(path, extname(path));
|
|
}
|
|
|
|
export function getLivePhotoMotionFilename(stillName: string, motionName: string) {
|
|
return getFileNameWithoutExtension(stillName) + extname(motionName);
|
|
}
|
|
|
|
const KiB = Math.pow(1024, 1);
|
|
const MiB = Math.pow(1024, 2);
|
|
const GiB = Math.pow(1024, 3);
|
|
const TiB = Math.pow(1024, 4);
|
|
const PiB = Math.pow(1024, 5);
|
|
|
|
export const HumanReadableSize = { KiB, MiB, GiB, TiB, PiB };
|
|
|
|
export function asHumanReadable(bytes: number, precision = 1): string {
|
|
const units = ['B', 'KiB', 'MiB', 'GiB', 'TiB', 'PiB', 'EiB'];
|
|
|
|
let magnitude = 0;
|
|
let remainder = bytes;
|
|
while (remainder >= 1024) {
|
|
if (magnitude + 1 < units.length) {
|
|
magnitude++;
|
|
remainder /= 1024;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
return `${remainder.toFixed(magnitude == 0 ? 0 : precision)} ${units[magnitude]}`;
|
|
}
|
|
|
|
export interface PaginationOptions {
|
|
take: number;
|
|
skip?: number;
|
|
}
|
|
|
|
export interface PaginationResult<T> {
|
|
items: T[];
|
|
hasNextPage: boolean;
|
|
}
|
|
|
|
export type Paginated<T> = Promise<PaginationResult<T>>;
|
|
|
|
export async function* usePagination<T>(
|
|
pageSize: number,
|
|
getNextPage: (pagination: PaginationOptions) => Paginated<T>,
|
|
) {
|
|
let hasNextPage = true;
|
|
|
|
for (let skip = 0; hasNextPage; skip += pageSize) {
|
|
const result = await getNextPage({ take: pageSize, skip });
|
|
hasNextPage = result.hasNextPage;
|
|
yield result.items;
|
|
}
|
|
}
|
|
|
|
export interface OptionalOptions extends ValidationOptions {
|
|
nullable?: boolean;
|
|
}
|
|
|
|
/**
|
|
* Checks if value is missing and if so, ignores all validators.
|
|
*
|
|
* @param validationOptions {@link OptionalOptions}
|
|
*
|
|
* @see IsOptional exported from `class-validator.
|
|
*/
|
|
// https://stackoverflow.com/a/71353929
|
|
export function Optional({ nullable, ...validationOptions }: OptionalOptions = {}) {
|
|
if (nullable === true) {
|
|
return IsOptional(validationOptions);
|
|
}
|
|
|
|
return ValidateIf((obj: any, v: any) => v !== undefined, validationOptions);
|
|
}
|