import { Config, ExtractConfigType } from "../config/builder/config" import { ActionMetadata, ActionResult, Effects, ExportedAction } from "../types" import { createUtils } from "../util" import { Utils, utils } from "../util/utils" export class CreatedAction< Store, Vault, ConfigType extends | Record | Config | Config, Type extends Record = ExtractConfigType, > { private constructor( public readonly myMetaData: Omit, readonly fn: (options: { effects: Effects utils: Utils input: Type }) => Promise, readonly input: Config, ) {} public validator = this.input.validator static of< Store, Vault, ConfigType extends | Record | Config | Config, Type extends Record = ExtractConfigType, >( metaData: Omit & { input: Config | Config }, fn: (options: { effects: Effects utils: Utils input: Type }) => Promise, ) { const { input, ...rest } = metaData return new CreatedAction( rest, fn, input as Config, ) } exportedAction: ExportedAction = ({ effects, input }) => { return this.fn({ effects, utils: createUtils(effects), input: this.validator.unsafeCast(input), }) } run = async ({ effects, input }: { effects: Effects; input?: Type }) => { return this.fn({ effects, utils: createUtils(effects), input: this.validator.unsafeCast(input), }) } async ActionMetadata(options: { effects: Effects utils: Utils }): Promise { return { ...this.myMetaData, input: await this.input.build(options), } } async getConfig({ effects }: { effects: Effects }) { return this.input.build({ effects, utils: createUtils(effects) as any, }) } } export const createAction = CreatedAction.of