@vanyamate/sec 0.1.4 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/index.ts DELETED
@@ -1,166 +0,0 @@
1
- export type EffectFunction<Args extends any[], Result> = (...args: Args) => Promise<Result>;
2
- export type EffectEvent =
3
- 'onBefore'
4
- | 'onSuccess'
5
- | 'onError'
6
- | 'onFinally';
7
-
8
- export type Effect<Args extends any[], Result> = {
9
- (...args: Args): Promise<Result>;
10
- onSuccess: (callback: (result: Result, ...args: Args) => void) => void;
11
- onError: (callback: (error: unknown, ...args: Args) => void) => void;
12
- onFinally: (callback: (...args: Args) => void) => void;
13
- onBefore: (callback: (...args: Args) => void) => void;
14
- };
15
-
16
- export type Listener<State> = (state: State) => void;
17
- export type Payload<Result, Args> = {
18
- result?: Result;
19
- error?: unknown;
20
- args: Args;
21
- };
22
- export type Handler<State, Args, Result> = (state: State, payload: Payload<Result, Args>) => State;
23
-
24
- export function effect<Args extends any[], Result> (fn: EffectFunction<Args, Result>): Effect<Args, Result> {
25
- let beforeListeners: ((...args: Args) => void)[] = [];
26
- let successListeners: ((result: Result, ...args: Args) => void)[] = [];
27
- let errorListeners: ((error: unknown, ...args: Args) => void)[] = [];
28
- let finallyListeners: ((...args: Args) => void)[] = [];
29
-
30
- const wrappedEffect: Effect<Args, Result> = async (...args: Args) => {
31
- beforeListeners.forEach(listener => listener(...args));
32
- try {
33
- const result = await fn(...args);
34
- successListeners.forEach(listener => listener(result, ...args));
35
- return result;
36
- } catch (error) {
37
- errorListeners.forEach(listener => listener(error, ...args));
38
- throw error;
39
- } finally {
40
- finallyListeners.forEach(listener => listener(...args));
41
- }
42
- };
43
-
44
- wrappedEffect.onBefore = (callback: (...args: Args) => void) => {
45
- beforeListeners.push(callback);
46
- };
47
-
48
- wrappedEffect.onSuccess = (callback: (result: Result, ...args: Args) => void) => {
49
- successListeners.push(callback);
50
- };
51
-
52
- wrappedEffect.onError = (callback: (error: unknown, ...args: Args) => void) => {
53
- errorListeners.push(callback);
54
- };
55
-
56
- wrappedEffect.onFinally = (callback: (...args: Args) => void) => {
57
- finallyListeners.push(callback);
58
- };
59
-
60
- return wrappedEffect;
61
- }
62
-
63
- export type Store<State> = {
64
- get: () => State;
65
- set: (newState: State) => void;
66
- on: <Args extends any[], Result>(
67
- effect: Effect<Args, Result>,
68
- event: EffectEvent,
69
- handler: Handler<State, Args, Result>,
70
- ) => Store<State>;
71
- subscribe: (listener: Listener<State>) => () => void;
72
- };
73
-
74
- export function store<State> (initialState: State): Store<State> {
75
- let state = initialState;
76
- let listeners: Listener<State>[] = [];
77
-
78
- const get = () => state;
79
- const set = (newState: State) => {
80
- state = newState;
81
- listeners.forEach(listener => listener(state));
82
- };
83
-
84
- const on = <Args extends any[], Result> (
85
- effect: Effect<Args, Result>,
86
- event: EffectEvent,
87
- handler: Handler<State, Args, Result>,
88
- ): Store<State> => {
89
- const callback = (payload: Payload<Result, Args>) => storeApi.set(handler(state, payload));
90
-
91
- if (event === 'onBefore') {
92
- effect.onBefore((...args) => callback({ args }));
93
- } else if (event === 'onSuccess') {
94
- effect.onSuccess((result, ...args) => callback({
95
- result,
96
- args,
97
- }));
98
- } else if (event === 'onError') {
99
- effect.onError((error, ...args) => callback({ error, args }));
100
- } else if (event === 'onFinally') {
101
- effect.onFinally((...args) => callback({ args }));
102
- }
103
-
104
- return storeApi;
105
- };
106
-
107
- const subscribe = (listener: Listener<State>) => {
108
- listeners.push(listener);
109
- return () => {
110
- const index = listeners.indexOf(listener);
111
- if (index !== -1) {
112
- listeners.splice(index, 1);
113
- }
114
- };
115
- };
116
-
117
- const storeApi = {
118
- get,
119
- set,
120
- on,
121
- subscribe,
122
- };
123
-
124
- return storeApi;
125
- }
126
-
127
- export function combine<States extends any[], Result> (
128
- stores: { [K in keyof States]: Store<States[K]> },
129
- callback: (...stores: { [K in keyof States]: Store<States[K]> }) => Result,
130
- ): Store<Result> {
131
- let combinedState: Result = callback(...stores);
132
-
133
- stores.forEach((store) => {
134
- store.subscribe(() => {
135
- combinedState = callback(...stores);
136
- listeners.forEach(listener => listener(combinedState));
137
- });
138
- });
139
-
140
- const listeners: Listener<Result>[] = [];
141
-
142
- const get = () => combinedState;
143
-
144
- const subscribe = (listener: Listener<Result>) => {
145
- listeners.push(listener);
146
- return () => {
147
- const index = listeners.indexOf(listener);
148
- if (index !== -1) {
149
- listeners.splice(index, 1);
150
- }
151
- };
152
- };
153
-
154
- const storeApi: Store<Result> = {
155
- get,
156
- set: () => {
157
- throw new Error('Cannot call \'set\' on combined store');
158
- },
159
- on : () => {
160
- throw new Error('Cannot call \'on\' on combined store');
161
- },
162
- subscribe,
163
- };
164
-
165
- return storeApi;
166
- }