strfry-policies/src/pipeline.ts

39 lines
1.1 KiB
TypeScript
Raw Normal View History

import { InputMessage, OutputMessage, Policy } from './types.ts';
2023-03-24 14:36:11 -05:00
2023-03-24 20:17:51 -05:00
/** A policy function with opts to run it with. Used by the pipeline. */
2023-03-24 20:06:03 -05:00
type PolicyTuple<Opts = unknown> = [policy: Policy<Opts>, opts?: Opts];
2023-03-24 19:55:58 -05:00
2023-03-24 20:17:51 -05:00
/** Helper type for proper type inference of PolicyTuples in the pipeline. */
2023-03-24 20:06:03 -05:00
// https://stackoverflow.com/a/75806165
// https://stackoverflow.com/a/54608401
2023-03-24 19:55:58 -05:00
type PolicyTuplesRest<T extends PolicyTuple[]> = {
2023-03-24 20:17:51 -05:00
[K in keyof T]: PolicyTuple<T[K]> | Policy<T[K]>;
};
2023-03-24 19:55:58 -05:00
/** Processes messages through multiple policies, bailing early on rejection. */
2023-03-24 20:17:51 -05:00
async function pipeline<P extends any[]>(
msg: InputMessage,
policies: [...PolicyTuplesRest<P>],
): Promise<OutputMessage> {
2023-03-24 20:06:03 -05:00
for (const item of policies) {
const [policy, opts] = toTuple(item);
2023-03-24 19:55:58 -05:00
const result = await policy(msg, opts);
if (result.action !== 'accept') {
return result;
}
}
return {
id: msg.event.id,
action: 'accept',
msg: '',
};
}
2023-03-24 20:06:03 -05:00
/** Coerce item into a tuple if it isn't already. */
function toTuple<T>(item: PolicyTuple<T> | Policy<T>): PolicyTuple<T> {
return typeof item === 'function' ? [item] : item;
}
export default pipeline;