2023-04-27 23:18:48 +02:00
|
|
|
import { eq, remove, cloneDeep } from 'lodash/fp';
|
2021-03-25 14:59:44 +01:00
|
|
|
|
2023-06-23 10:04:03 +02:00
|
|
|
export type Handler = (...args: any[]) => any;
|
2021-03-25 14:59:44 +01:00
|
|
|
|
2023-06-26 17:24:32 +02:00
|
|
|
export interface Hook<T extends Handler = Handler> {
|
|
|
|
getHandlers(): Handler[];
|
|
|
|
register(handler: T): Hook<T>;
|
|
|
|
delete(handler: T): Hook<T>;
|
|
|
|
call(...args: any[]): void;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface AsyncSeriesHook extends Hook {
|
|
|
|
call(...args: any[]): Promise<void>;
|
|
|
|
}
|
|
|
|
export interface AsyncSeriesWaterfallHook extends Hook {
|
|
|
|
call(...args: any[]): Promise<any>;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface AsyncParallelHook extends Hook {
|
|
|
|
call(...args: any[]): Promise<any[]>;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface AsyncBailHook extends Hook {
|
|
|
|
call(...args: any[]): Promise<any>;
|
|
|
|
}
|
|
|
|
|
2021-03-25 14:59:44 +01:00
|
|
|
/**
|
|
|
|
* Create a default Strapi hook
|
|
|
|
*/
|
2023-06-26 17:24:32 +02:00
|
|
|
const createHook = <T extends Handler = Handler>(): Hook<T> => {
|
2023-04-27 23:18:48 +02:00
|
|
|
type State = {
|
|
|
|
handlers: T[];
|
|
|
|
};
|
|
|
|
|
|
|
|
const state: State = {
|
2021-03-25 14:59:44 +01:00
|
|
|
handlers: [],
|
|
|
|
};
|
|
|
|
|
|
|
|
return {
|
2022-05-09 20:55:21 +02:00
|
|
|
getHandlers() {
|
2021-03-25 14:59:44 +01:00
|
|
|
return state.handlers;
|
|
|
|
},
|
|
|
|
|
2023-04-27 23:18:48 +02:00
|
|
|
register(handler: T) {
|
2021-03-25 14:59:44 +01:00
|
|
|
state.handlers.push(handler);
|
2022-05-09 20:55:21 +02:00
|
|
|
|
2021-09-13 12:03:12 +02:00
|
|
|
return this;
|
2021-03-25 14:59:44 +01:00
|
|
|
},
|
|
|
|
|
2023-04-27 23:18:48 +02:00
|
|
|
delete(handler: T) {
|
2021-03-25 14:59:44 +01:00
|
|
|
state.handlers = remove(eq(handler), state.handlers);
|
2022-05-09 20:55:21 +02:00
|
|
|
|
2021-09-13 12:03:12 +02:00
|
|
|
return this;
|
2021-03-25 14:59:44 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
call() {
|
|
|
|
throw new Error('Method not implemented');
|
|
|
|
},
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create an async series hook.
|
|
|
|
* Upon execution, it will execute every handler in order with the same context
|
|
|
|
*/
|
2023-04-27 23:18:48 +02:00
|
|
|
const createAsyncSeriesHook = <T extends Handler = Handler>() => ({
|
|
|
|
...createHook<T>(),
|
2021-03-25 14:59:44 +01:00
|
|
|
|
2023-04-27 23:18:48 +02:00
|
|
|
async call(context: unknown) {
|
2022-05-09 20:55:21 +02:00
|
|
|
for (const handler of this.getHandlers()) {
|
2021-03-25 14:59:44 +01:00
|
|
|
await handler(context);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create an async series waterfall hook.
|
|
|
|
* Upon execution, it will execute every handler in order and pass the return value of the last handler to the next one
|
|
|
|
*/
|
2023-04-27 23:18:48 +02:00
|
|
|
const createAsyncSeriesWaterfallHook = <T extends Handler = Handler>() => ({
|
|
|
|
...createHook<T>(),
|
2021-03-25 14:59:44 +01:00
|
|
|
|
2023-04-27 23:18:48 +02:00
|
|
|
async call(param: unknown) {
|
2021-03-25 14:59:44 +01:00
|
|
|
let res = param;
|
|
|
|
|
2022-05-09 20:55:21 +02:00
|
|
|
for (const handler of this.getHandlers()) {
|
2021-03-25 14:59:44 +01:00
|
|
|
res = await handler(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create an async parallel hook.
|
|
|
|
* Upon execution, it will execute every registered handler in band.
|
|
|
|
*/
|
2023-04-27 23:18:48 +02:00
|
|
|
const createAsyncParallelHook = <T extends Handler = Handler>() => ({
|
|
|
|
...createHook<T>(),
|
2021-03-25 14:59:44 +01:00
|
|
|
|
2023-04-27 23:18:48 +02:00
|
|
|
async call(context: unknown) {
|
2022-08-08 23:33:39 +02:00
|
|
|
const promises = this.getHandlers().map((handler) => handler(cloneDeep(context)));
|
2021-03-25 14:59:44 +01:00
|
|
|
|
|
|
|
return Promise.all(promises);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2022-07-21 10:42:30 +02:00
|
|
|
/**
|
|
|
|
* Create an async parallel hook.
|
|
|
|
* Upon execution, it will execute every registered handler in serie and return the first result found.
|
|
|
|
*/
|
2023-04-27 23:18:48 +02:00
|
|
|
const createAsyncBailHook = <T extends Handler = Handler>() => ({
|
|
|
|
...createHook<T>(),
|
2022-07-21 10:42:30 +02:00
|
|
|
|
2023-04-27 23:18:48 +02:00
|
|
|
async call(context: unknown) {
|
2022-07-21 10:42:30 +02:00
|
|
|
for (const handler of this.getHandlers()) {
|
|
|
|
const result = await handler(context);
|
|
|
|
|
|
|
|
if (result !== undefined) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2023-04-27 23:18:48 +02:00
|
|
|
export const internals = {
|
2021-03-25 14:59:44 +01:00
|
|
|
// Internal utils
|
2023-04-27 23:18:48 +02:00
|
|
|
createHook,
|
|
|
|
};
|
|
|
|
|
|
|
|
export {
|
2021-03-25 14:59:44 +01:00
|
|
|
createAsyncSeriesHook,
|
|
|
|
createAsyncSeriesWaterfallHook,
|
|
|
|
createAsyncParallelHook,
|
2022-07-21 10:42:30 +02:00
|
|
|
createAsyncBailHook,
|
2021-03-25 14:59:44 +01:00
|
|
|
};
|