2024-04-29 09:48:28 -04:00
|
|
|
import { SetMetadata, applyDecorators } from '@nestjs/common';
|
|
|
|
import { ApiExtension, ApiOperation, ApiProperty, ApiTags } from '@nestjs/swagger';
|
2024-03-20 15:04:03 -05:00
|
|
|
import _ from 'lodash';
|
2024-04-29 09:48:28 -04:00
|
|
|
import { ADDED_IN_PREFIX, DEPRECATED_IN_PREFIX, LIFECYCLE_EXTENSION } from 'src/constants';
|
2024-09-27 10:28:42 -04:00
|
|
|
import { MetadataKey } from 'src/enum';
|
2024-09-30 10:35:11 -04:00
|
|
|
import { EmitEvent } from 'src/interfaces/event.interface';
|
2024-10-31 13:42:58 -04:00
|
|
|
import { JobName, QueueName } from 'src/interfaces/job.interface';
|
2024-03-20 22:15:09 -05:00
|
|
|
import { setUnion } from 'src/utils/set';
|
2024-03-20 15:04:03 -05:00
|
|
|
|
|
|
|
// PostgreSQL uses a 16-bit integer to indicate the number of bound parameters. This means that the
|
|
|
|
// maximum number of parameters is 65535. Any query that tries to bind more than that (e.g. searching
|
|
|
|
// by a list of IDs) requires splitting the query into multiple chunks.
|
|
|
|
// We are rounding down this limit, as queries commonly include other filters and parameters.
|
|
|
|
export const DATABASE_PARAMETER_CHUNK_SIZE = 65_500;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Chunks an array or set into smaller collections of the same type and specified size.
|
|
|
|
*
|
|
|
|
* @param collection The collection to chunk.
|
|
|
|
* @param size The size of each chunk.
|
|
|
|
*/
|
|
|
|
function chunks<T>(collection: Array<T>, size: number): Array<Array<T>>;
|
|
|
|
function chunks<T>(collection: Set<T>, size: number): Array<Set<T>>;
|
|
|
|
function chunks<T>(collection: Array<T> | Set<T>, size: number): Array<Array<T>> | Array<Set<T>> {
|
|
|
|
if (collection instanceof Set) {
|
|
|
|
const result = [];
|
|
|
|
let chunk = new Set<T>();
|
|
|
|
for (const element of collection) {
|
|
|
|
chunk.add(element);
|
|
|
|
if (chunk.size === size) {
|
|
|
|
result.push(chunk);
|
|
|
|
chunk = new Set<T>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (chunk.size > 0) {
|
|
|
|
result.push(chunk);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
} else {
|
|
|
|
return _.chunk(collection, size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Wraps a method that takes a collection of parameters and sequentially calls it with chunks of the collection,
|
|
|
|
* to overcome the maximum number of parameters allowed by the database driver.
|
|
|
|
*
|
|
|
|
* @param options.paramIndex The index of the function parameter to chunk. Defaults to 0.
|
|
|
|
* @param options.flatten Whether to flatten the results. Defaults to false.
|
|
|
|
*/
|
2024-07-17 07:43:35 -04:00
|
|
|
export function Chunked(
|
|
|
|
options: { paramIndex?: number; chunkSize?: number; mergeFn?: (results: any) => any } = {},
|
|
|
|
): MethodDecorator {
|
2024-03-20 15:04:03 -05:00
|
|
|
return (target: any, propertyKey: string | symbol, descriptor: PropertyDescriptor) => {
|
|
|
|
const originalMethod = descriptor.value;
|
|
|
|
const parameterIndex = options.paramIndex ?? 0;
|
2024-07-17 07:43:35 -04:00
|
|
|
const chunkSize = options.chunkSize || DATABASE_PARAMETER_CHUNK_SIZE;
|
2024-03-20 15:04:03 -05:00
|
|
|
descriptor.value = async function (...arguments_: any[]) {
|
|
|
|
const argument = arguments_[parameterIndex];
|
|
|
|
|
|
|
|
// Early return if argument length is less than or equal to the chunk size.
|
|
|
|
if (
|
2024-07-17 07:43:35 -04:00
|
|
|
(Array.isArray(argument) && argument.length <= chunkSize) ||
|
|
|
|
(argument instanceof Set && argument.size <= chunkSize)
|
2024-03-20 15:04:03 -05:00
|
|
|
) {
|
|
|
|
return await originalMethod.apply(this, arguments_);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Promise.all(
|
2024-07-17 07:43:35 -04:00
|
|
|
chunks(argument, chunkSize).map(async (chunk) => {
|
2024-03-20 15:04:03 -05:00
|
|
|
await Reflect.apply(originalMethod, this, [
|
|
|
|
...arguments_.slice(0, parameterIndex),
|
|
|
|
chunk,
|
|
|
|
...arguments_.slice(parameterIndex + 1),
|
|
|
|
]);
|
|
|
|
}),
|
|
|
|
).then((results) => (options.mergeFn ? options.mergeFn(results) : results));
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export function ChunkedArray(options?: { paramIndex?: number }): MethodDecorator {
|
|
|
|
return Chunked({ ...options, mergeFn: _.flatten });
|
|
|
|
}
|
|
|
|
|
|
|
|
export function ChunkedSet(options?: { paramIndex?: number }): MethodDecorator {
|
|
|
|
return Chunked({ ...options, mergeFn: setUnion });
|
|
|
|
}
|
|
|
|
|
|
|
|
const UUID = '00000000-0000-4000-a000-000000000000';
|
|
|
|
|
|
|
|
export const DummyValue = {
|
|
|
|
UUID,
|
|
|
|
UUID_SET: new Set([UUID]),
|
|
|
|
PAGINATION: { take: 10, skip: 0 },
|
|
|
|
EMAIL: 'user@immich.app',
|
|
|
|
STRING: 'abcdefghi',
|
2024-05-03 15:34:57 -04:00
|
|
|
NUMBER: 50,
|
2024-03-20 15:04:03 -05:00
|
|
|
BUFFER: Buffer.from('abcdefghi'),
|
|
|
|
DATE: new Date(),
|
|
|
|
TIME_BUCKET: '2024-01-01T00:00:00.000Z',
|
|
|
|
};
|
|
|
|
|
|
|
|
export const GENERATE_SQL_KEY = 'generate-sql-key';
|
|
|
|
|
|
|
|
export interface GenerateSqlQueries {
|
|
|
|
name?: string;
|
|
|
|
params: unknown[];
|
|
|
|
}
|
|
|
|
|
2024-10-21 19:52:30 -04:00
|
|
|
export const Telemetry = (options: { enabled?: boolean }) =>
|
|
|
|
SetMetadata(MetadataKey.TELEMETRY_ENABLED, options?.enabled ?? true);
|
|
|
|
|
2024-03-20 15:04:03 -05:00
|
|
|
/** Decorator to enable versioning/tracking of generated Sql */
|
|
|
|
export const GenerateSql = (...options: GenerateSqlQueries[]) => SetMetadata(GENERATE_SQL_KEY, options);
|
2024-03-21 23:59:21 +01:00
|
|
|
|
2024-09-30 10:35:11 -04:00
|
|
|
export type EventConfig = {
|
|
|
|
name: EmitEvent;
|
|
|
|
/** handle socket.io server events as well */
|
|
|
|
server?: boolean;
|
2024-06-27 15:54:20 -04:00
|
|
|
/** lower value has higher priority, defaults to 0 */
|
2024-08-15 16:12:41 -04:00
|
|
|
priority?: number;
|
2024-06-27 15:54:20 -04:00
|
|
|
};
|
2024-09-30 10:35:11 -04:00
|
|
|
export const OnEvent = (config: EventConfig) => SetMetadata(MetadataKey.EVENT_CONFIG, config);
|
2024-06-27 15:54:20 -04:00
|
|
|
|
2024-10-31 13:42:58 -04:00
|
|
|
export type JobConfig = {
|
|
|
|
name: JobName;
|
|
|
|
queue: QueueName;
|
|
|
|
};
|
|
|
|
export const OnJob = (config: JobConfig) => SetMetadata(MetadataKey.JOB_CONFIG, config);
|
|
|
|
|
2024-04-29 09:48:28 -04:00
|
|
|
type LifecycleRelease = 'NEXT_RELEASE' | string;
|
|
|
|
type LifecycleMetadata = {
|
|
|
|
addedAt?: LifecycleRelease;
|
|
|
|
deprecatedAt?: LifecycleRelease;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const EndpointLifecycle = ({ addedAt, deprecatedAt }: LifecycleMetadata) => {
|
|
|
|
const decorators: MethodDecorator[] = [ApiExtension(LIFECYCLE_EXTENSION, { addedAt, deprecatedAt })];
|
|
|
|
if (deprecatedAt) {
|
|
|
|
decorators.push(
|
|
|
|
ApiTags('Deprecated'),
|
|
|
|
ApiOperation({ deprecated: true, description: DEPRECATED_IN_PREFIX + deprecatedAt }),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return applyDecorators(...decorators);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const PropertyLifecycle = ({ addedAt, deprecatedAt }: LifecycleMetadata) => {
|
|
|
|
const decorators: PropertyDecorator[] = [];
|
|
|
|
decorators.push(ApiProperty({ description: ADDED_IN_PREFIX + addedAt }));
|
|
|
|
if (deprecatedAt) {
|
|
|
|
decorators.push(ApiProperty({ deprecated: true, description: DEPRECATED_IN_PREFIX + deprecatedAt }));
|
|
|
|
}
|
|
|
|
|
|
|
|
return applyDecorators(...decorators);
|
|
|
|
};
|