@jbrowse/mobx-state-tree 5.6.10 → 5.7.1
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/dist/core/action.js.map +1 -1
- package/dist/core/actionContext.js.map +1 -1
- package/dist/core/flow.js.map +1 -1
- package/dist/core/json-patch.js.map +1 -1
- package/dist/core/mst-operations.js.map +1 -1
- package/dist/core/node/BaseNode.js.map +1 -1
- package/dist/core/node/Hook.js.map +1 -1
- package/dist/core/node/create-node.js.map +1 -1
- package/dist/core/node/identifier-cache.js.map +1 -1
- package/dist/core/node/livelinessChecking.js.map +1 -1
- package/dist/core/node/node-utils.js.map +1 -1
- package/dist/core/node/object-node.d.ts +3 -1
- package/dist/core/node/object-node.js +2 -1
- package/dist/core/node/object-node.js.map +1 -1
- package/dist/core/node/scalar-node.d.ts +3 -1
- package/dist/core/node/scalar-node.js +2 -1
- package/dist/core/node/scalar-node.js.map +1 -1
- package/dist/core/process.js.map +1 -1
- package/dist/core/type/type-checker.js.map +1 -1
- package/dist/core/type/type.js.map +1 -1
- package/dist/index.js.map +1 -1
- package/dist/internal.js.map +1 -1
- package/dist/middlewares/create-action-tracking-middleware.js.map +1 -1
- package/dist/middlewares/createActionTrackingMiddleware2.js.map +1 -1
- package/dist/middlewares/on-action.js.map +1 -1
- package/dist/package.json +1 -0
- package/dist/types/complex-types/array.d.ts +2 -1
- package/dist/types/complex-types/array.js +2 -1
- package/dist/types/complex-types/array.js.map +1 -1
- package/dist/types/complex-types/map.d.ts +2 -1
- package/dist/types/complex-types/map.js +2 -1
- package/dist/types/complex-types/map.js.map +1 -1
- package/dist/types/complex-types/model.d.ts +2 -1
- package/dist/types/complex-types/model.js +2 -1
- package/dist/types/complex-types/model.js.map +1 -1
- package/dist/types/index.js.map +1 -1
- package/dist/types/primitives.d.ts +2 -1
- package/dist/types/primitives.js +2 -1
- package/dist/types/primitives.js.map +1 -1
- package/dist/types/utility-types/custom.d.ts +2 -1
- package/dist/types/utility-types/custom.js +2 -1
- package/dist/types/utility-types/custom.js.map +1 -1
- package/dist/types/utility-types/enumeration.js.map +1 -1
- package/dist/types/utility-types/frozen.d.ts +2 -1
- package/dist/types/utility-types/frozen.js +2 -1
- package/dist/types/utility-types/frozen.js.map +1 -1
- package/dist/types/utility-types/identifier.d.ts +2 -1
- package/dist/types/utility-types/identifier.js +2 -1
- package/dist/types/utility-types/identifier.js.map +1 -1
- package/dist/types/utility-types/late.js +2 -1
- package/dist/types/utility-types/late.js.map +1 -1
- package/dist/types/utility-types/lazy.d.ts +2 -1
- package/dist/types/utility-types/lazy.js +2 -1
- package/dist/types/utility-types/lazy.js.map +1 -1
- package/dist/types/utility-types/literal.d.ts +2 -1
- package/dist/types/utility-types/literal.js +2 -1
- package/dist/types/utility-types/literal.js.map +1 -1
- package/dist/types/utility-types/maybe.js.map +1 -1
- package/dist/types/utility-types/optional.d.ts +2 -1
- package/dist/types/utility-types/optional.js +2 -1
- package/dist/types/utility-types/optional.js.map +1 -1
- package/dist/types/utility-types/reference.d.ts +2 -1
- package/dist/types/utility-types/reference.js +2 -1
- package/dist/types/utility-types/reference.js.map +1 -1
- package/dist/types/utility-types/refinement.js +2 -1
- package/dist/types/utility-types/refinement.js.map +1 -1
- package/dist/types/utility-types/resilient.js +2 -1
- package/dist/types/utility-types/resilient.js.map +1 -1
- package/dist/types/utility-types/snapshotProcessor.js +3 -2
- package/dist/types/utility-types/snapshotProcessor.js.map +1 -1
- package/dist/types/utility-types/union.d.ts +2 -1
- package/dist/types/utility-types/union.js +2 -1
- package/dist/types/utility-types/union.js.map +1 -1
- package/dist/utils.js.map +1 -1
- package/esm/core/action.d.ts +87 -0
- package/esm/core/action.js +219 -0
- package/esm/core/action.js.map +1 -0
- package/esm/core/actionContext.d.ts +27 -0
- package/esm/core/actionContext.js +37 -0
- package/esm/core/actionContext.js.map +1 -0
- package/esm/core/flow.d.ts +69 -0
- package/esm/core/flow.js +173 -0
- package/esm/core/flow.js.map +1 -0
- package/esm/core/json-patch.d.ts +46 -0
- package/esm/core/json-patch.js +125 -0
- package/esm/core/json-patch.js.map +1 -0
- package/esm/core/mst-operations.d.ts +459 -0
- package/esm/core/mst-operations.js +844 -0
- package/esm/core/mst-operations.js.map +1 -0
- package/esm/core/node/BaseNode.d.ts +62 -0
- package/esm/core/node/BaseNode.js +148 -0
- package/esm/core/node/BaseNode.js.map +1 -0
- package/esm/core/node/Hook.d.ts +17 -0
- package/esm/core/node/Hook.js +12 -0
- package/esm/core/node/Hook.js.map +1 -0
- package/esm/core/node/create-node.d.ts +16 -0
- package/esm/core/node/create-node.js +36 -0
- package/esm/core/node/create-node.js.map +1 -0
- package/esm/core/node/identifier-cache.d.ts +19 -0
- package/esm/core/node/identifier-cache.js +111 -0
- package/esm/core/node/identifier-cache.js.map +1 -0
- package/esm/core/node/livelinessChecking.d.ts +37 -0
- package/esm/core/node/livelinessChecking.js +33 -0
- package/esm/core/node/livelinessChecking.js.map +1 -0
- package/esm/core/node/node-utils.d.ts +83 -0
- package/esm/core/node/node-utils.js +153 -0
- package/esm/core/node/node-utils.js.map +1 -0
- package/esm/core/node/object-node.d.ts +101 -0
- package/esm/core/node/object-node.js +542 -0
- package/esm/core/node/object-node.js.map +1 -0
- package/esm/core/node/scalar-node.d.ts +21 -0
- package/esm/core/node/scalar-node.js +86 -0
- package/esm/core/node/scalar-node.js.map +1 -0
- package/esm/core/process.d.ts +50 -0
- package/esm/core/process.js +35 -0
- package/esm/core/process.js.map +1 -0
- package/esm/core/type/type-checker.d.ts +69 -0
- package/esm/core/type/type-checker.js +144 -0
- package/esm/core/type/type-checker.js.map +1 -0
- package/esm/core/type/type.d.ts +317 -0
- package/esm/core/type/type.js +243 -0
- package/esm/core/type/type.js.map +1 -0
- package/esm/index.d.ts +2 -0
- package/esm/index.js +2 -0
- package/esm/index.js.map +1 -0
- package/esm/internal.d.ts +39 -0
- package/esm/internal.js +44 -0
- package/esm/internal.js.map +1 -0
- package/esm/middlewares/create-action-tracking-middleware.d.ts +24 -0
- package/esm/middlewares/create-action-tracking-middleware.js +78 -0
- package/esm/middlewares/create-action-tracking-middleware.js.map +1 -0
- package/esm/middlewares/createActionTrackingMiddleware2.d.ts +34 -0
- package/esm/middlewares/createActionTrackingMiddleware2.js +130 -0
- package/esm/middlewares/createActionTrackingMiddleware2.js.map +1 -0
- package/esm/middlewares/on-action.d.ts +87 -0
- package/esm/middlewares/on-action.js +210 -0
- package/esm/middlewares/on-action.js.map +1 -0
- package/esm/types/complex-types/array.d.ts +81 -0
- package/esm/types/complex-types/array.js +347 -0
- package/esm/types/complex-types/array.js.map +1 -0
- package/esm/types/complex-types/map.d.ts +111 -0
- package/esm/types/complex-types/map.js +356 -0
- package/esm/types/complex-types/map.js.map +1 -0
- package/esm/types/complex-types/model.d.ts +193 -0
- package/esm/types/complex-types/model.js +471 -0
- package/esm/types/complex-types/model.js.map +1 -0
- package/esm/types/index.d.ts +33 -0
- package/esm/types/index.js +35 -0
- package/esm/types/index.js.map +1 -0
- package/esm/types/primitives.d.ts +125 -0
- package/esm/types/primitives.js +177 -0
- package/esm/types/primitives.js.map +1 -0
- package/esm/types/utility-types/custom.d.ts +75 -0
- package/esm/types/utility-types/custom.js +106 -0
- package/esm/types/utility-types/custom.js.map +1 -0
- package/esm/types/utility-types/enumeration.d.ts +5 -0
- package/esm/types/utility-types/enumeration.js +31 -0
- package/esm/types/utility-types/enumeration.js.map +1 -0
- package/esm/types/utility-types/frozen.d.ts +24 -0
- package/esm/types/utility-types/frozen.js +92 -0
- package/esm/types/utility-types/frozen.js.map +1 -0
- package/esm/types/utility-types/identifier.d.ts +87 -0
- package/esm/types/utility-types/identifier.js +121 -0
- package/esm/types/utility-types/identifier.js.map +1 -0
- package/esm/types/utility-types/late.d.ts +10 -0
- package/esm/types/utility-types/late.js +106 -0
- package/esm/types/utility-types/late.js.map +1 -0
- package/esm/types/utility-types/lazy.d.ts +23 -0
- package/esm/types/utility-types/lazy.js +72 -0
- package/esm/types/utility-types/lazy.js.map +1 -0
- package/esm/types/utility-types/literal.d.ts +38 -0
- package/esm/types/utility-types/literal.js +58 -0
- package/esm/types/utility-types/literal.js.map +1 -0
- package/esm/types/utility-types/maybe.d.ts +26 -0
- package/esm/types/utility-types/maybe.js +26 -0
- package/esm/types/utility-types/maybe.js.map +1 -0
- package/esm/types/utility-types/optional.d.ts +42 -0
- package/esm/types/utility-types/optional.js +135 -0
- package/esm/types/utility-types/optional.js.map +1 -0
- package/esm/types/utility-types/reference.d.ts +90 -0
- package/esm/types/utility-types/reference.js +383 -0
- package/esm/types/utility-types/reference.js.map +1 -0
- package/esm/types/utility-types/refinement.d.ts +10 -0
- package/esm/types/utility-types/refinement.js +82 -0
- package/esm/types/utility-types/refinement.js.map +1 -0
- package/esm/types/utility-types/resilient.d.ts +18 -0
- package/esm/types/utility-types/resilient.js +118 -0
- package/esm/types/utility-types/resilient.js.map +1 -0
- package/esm/types/utility-types/snapshotProcessor.d.ts +63 -0
- package/esm/types/utility-types/snapshotProcessor.js +159 -0
- package/esm/types/utility-types/snapshotProcessor.js.map +1 -0
- package/esm/types/utility-types/union.d.ts +78 -0
- package/esm/types/utility-types/union.js +240 -0
- package/esm/types/utility-types/union.js.map +1 -0
- package/esm/utils.d.ts +230 -0
- package/esm/utils.js +449 -0
- package/esm/utils.js.map +1 -0
- package/package.json +23 -22
- package/dist/mobx-state-tree.js +0 -86
- package/dist/mobx-state-tree.js.map +0 -1
- package/dist/mobx-state-tree.module.js +0 -84
- package/dist/mobx-state-tree.module.js.map +0 -1
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import type { IActionContext, IAnyStateTreeNode, IDisposer } from "../internal.ts";
|
|
2
|
+
export type IMiddlewareEventType = "action" | "flow_spawn" | "flow_resume" | "flow_resume_error" | "flow_return" | "flow_throw";
|
|
3
|
+
export interface IMiddlewareEvent extends IActionContext {
|
|
4
|
+
/** Event type */
|
|
5
|
+
readonly type: IMiddlewareEventType;
|
|
6
|
+
/** Parent event unique id */
|
|
7
|
+
readonly parentId: number;
|
|
8
|
+
/** Parent event object */
|
|
9
|
+
readonly parentEvent: IMiddlewareEvent | undefined;
|
|
10
|
+
/** Root event unique id */
|
|
11
|
+
readonly rootId: number;
|
|
12
|
+
/** Id of all events, from root until current (excluding current) */
|
|
13
|
+
readonly allParentIds: number[];
|
|
14
|
+
}
|
|
15
|
+
export interface FunctionWithFlag extends Function {
|
|
16
|
+
_isMSTAction?: boolean;
|
|
17
|
+
_isFlowAction?: boolean;
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* @internal
|
|
21
|
+
* @hidden
|
|
22
|
+
*/
|
|
23
|
+
export type IMiddleware = {
|
|
24
|
+
handler: IMiddlewareHandler;
|
|
25
|
+
includeHooks: boolean;
|
|
26
|
+
};
|
|
27
|
+
export type IMiddlewareHandler = (actionCall: IMiddlewareEvent, next: (actionCall: IMiddlewareEvent, callback?: (value: any) => any) => void, abort: (value: any) => void) => any;
|
|
28
|
+
/**
|
|
29
|
+
* @internal
|
|
30
|
+
* @hidden
|
|
31
|
+
*/
|
|
32
|
+
export declare function getCurrentActionContext(): IMiddlewareEvent | undefined;
|
|
33
|
+
/**
|
|
34
|
+
* @internal
|
|
35
|
+
* @hidden
|
|
36
|
+
*/
|
|
37
|
+
export declare function getNextActionId(): number;
|
|
38
|
+
/**
|
|
39
|
+
* @internal
|
|
40
|
+
* @hidden
|
|
41
|
+
*/
|
|
42
|
+
export declare function runWithActionContext(context: IMiddlewareEvent, fn: Function): any;
|
|
43
|
+
/**
|
|
44
|
+
* @internal
|
|
45
|
+
* @hidden
|
|
46
|
+
*/
|
|
47
|
+
export declare function getParentActionContext(parentContext: IMiddlewareEvent | undefined): IMiddlewareEvent | undefined;
|
|
48
|
+
/**
|
|
49
|
+
* @internal
|
|
50
|
+
* @hidden
|
|
51
|
+
*/
|
|
52
|
+
export declare function createActionInvoker<T extends FunctionWithFlag>(target: IAnyStateTreeNode, name: string, fn: T): (...args: any[]) => any;
|
|
53
|
+
/**
|
|
54
|
+
* Middleware can be used to intercept any action is invoked on the subtree where it is attached.
|
|
55
|
+
* If a tree is protected (by default), this means that any mutation of the tree will pass through your middleware.
|
|
56
|
+
*
|
|
57
|
+
* For more details, see the [middleware docs](concepts/middleware.md)
|
|
58
|
+
*
|
|
59
|
+
* @param target Node to apply the middleware to.
|
|
60
|
+
* @param middleware Middleware to apply.
|
|
61
|
+
* @returns A callable function to dispose the middleware.
|
|
62
|
+
*/
|
|
63
|
+
export declare function addMiddleware(target: IAnyStateTreeNode, handler: IMiddlewareHandler, includeHooks?: boolean): IDisposer;
|
|
64
|
+
/**
|
|
65
|
+
* Binds middleware to a specific action.
|
|
66
|
+
*
|
|
67
|
+
* Example:
|
|
68
|
+
* ```ts
|
|
69
|
+
* type.actions(self => {
|
|
70
|
+
* function takeA____() {
|
|
71
|
+
* self.toilet.donate()
|
|
72
|
+
* self.wipe()
|
|
73
|
+
* self.wipe()
|
|
74
|
+
* self.toilet.flush()
|
|
75
|
+
* }
|
|
76
|
+
* return {
|
|
77
|
+
* takeA____: decorate(atomic, takeA____)
|
|
78
|
+
* }
|
|
79
|
+
* })
|
|
80
|
+
* ```
|
|
81
|
+
*
|
|
82
|
+
* @param handler
|
|
83
|
+
* @param fn
|
|
84
|
+
* @param includeHooks
|
|
85
|
+
* @returns The original function
|
|
86
|
+
*/
|
|
87
|
+
export declare function decorate<T extends Function>(handler: IMiddlewareHandler, fn: T, includeHooks?: boolean): T;
|
|
@@ -0,0 +1,219 @@
|
|
|
1
|
+
import { action as mobxAction } from "mobx";
|
|
2
|
+
import { Hook, devMode, fail, getRoot, getStateTreeNode, warnError } from "../internal.js";
|
|
3
|
+
let nextActionId = 1;
|
|
4
|
+
let currentActionContext;
|
|
5
|
+
/**
|
|
6
|
+
* @internal
|
|
7
|
+
* @hidden
|
|
8
|
+
*/
|
|
9
|
+
export function getCurrentActionContext() {
|
|
10
|
+
return currentActionContext;
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* @internal
|
|
14
|
+
* @hidden
|
|
15
|
+
*/
|
|
16
|
+
export function getNextActionId() {
|
|
17
|
+
return nextActionId++;
|
|
18
|
+
}
|
|
19
|
+
// TODO: optimize away entire action context if there is no middleware in tree?
|
|
20
|
+
/**
|
|
21
|
+
* @internal
|
|
22
|
+
* @hidden
|
|
23
|
+
*/
|
|
24
|
+
export function runWithActionContext(context, fn) {
|
|
25
|
+
const node = getStateTreeNode(context.context);
|
|
26
|
+
if (context.type === "action") {
|
|
27
|
+
node.assertAlive({
|
|
28
|
+
actionContext: context
|
|
29
|
+
});
|
|
30
|
+
}
|
|
31
|
+
const baseIsRunningAction = node._isRunningAction;
|
|
32
|
+
node._isRunningAction = true;
|
|
33
|
+
const previousContext = currentActionContext;
|
|
34
|
+
currentActionContext = context;
|
|
35
|
+
try {
|
|
36
|
+
return runMiddleWares(node, context, fn);
|
|
37
|
+
}
|
|
38
|
+
finally {
|
|
39
|
+
currentActionContext = previousContext;
|
|
40
|
+
node._isRunningAction = baseIsRunningAction;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* @internal
|
|
45
|
+
* @hidden
|
|
46
|
+
*/
|
|
47
|
+
export function getParentActionContext(parentContext) {
|
|
48
|
+
if (!parentContext) {
|
|
49
|
+
return undefined;
|
|
50
|
+
}
|
|
51
|
+
if (parentContext.type === "action") {
|
|
52
|
+
return parentContext;
|
|
53
|
+
}
|
|
54
|
+
return parentContext.parentActionEvent;
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* @internal
|
|
58
|
+
* @hidden
|
|
59
|
+
*/
|
|
60
|
+
export function createActionInvoker(target, name, fn) {
|
|
61
|
+
const res = function (...args) {
|
|
62
|
+
const id = getNextActionId();
|
|
63
|
+
const parentContext = currentActionContext;
|
|
64
|
+
const parentActionContext = getParentActionContext(parentContext);
|
|
65
|
+
return runWithActionContext({
|
|
66
|
+
type: "action",
|
|
67
|
+
name,
|
|
68
|
+
id,
|
|
69
|
+
args,
|
|
70
|
+
context: target,
|
|
71
|
+
tree: getRoot(target),
|
|
72
|
+
rootId: parentContext ? parentContext.rootId : id,
|
|
73
|
+
parentId: parentContext ? parentContext.id : 0,
|
|
74
|
+
allParentIds: parentContext
|
|
75
|
+
? [...parentContext.allParentIds, parentContext.id]
|
|
76
|
+
: [],
|
|
77
|
+
parentEvent: parentContext,
|
|
78
|
+
parentActionEvent: parentActionContext
|
|
79
|
+
}, fn);
|
|
80
|
+
};
|
|
81
|
+
res._isMSTAction = true;
|
|
82
|
+
res._isFlowAction = fn._isFlowAction;
|
|
83
|
+
return res;
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Middleware can be used to intercept any action is invoked on the subtree where it is attached.
|
|
87
|
+
* If a tree is protected (by default), this means that any mutation of the tree will pass through your middleware.
|
|
88
|
+
*
|
|
89
|
+
* For more details, see the [middleware docs](concepts/middleware.md)
|
|
90
|
+
*
|
|
91
|
+
* @param target Node to apply the middleware to.
|
|
92
|
+
* @param middleware Middleware to apply.
|
|
93
|
+
* @returns A callable function to dispose the middleware.
|
|
94
|
+
*/
|
|
95
|
+
export function addMiddleware(target, handler, includeHooks = true) {
|
|
96
|
+
const node = getStateTreeNode(target);
|
|
97
|
+
if (devMode()) {
|
|
98
|
+
if (!node.isProtectionEnabled) {
|
|
99
|
+
warnError("It is recommended to protect the state tree before attaching action middleware, as otherwise it cannot be guaranteed that all changes are passed through middleware. See `protect`");
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
return node.addMiddleWare(handler, includeHooks);
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* Binds middleware to a specific action.
|
|
106
|
+
*
|
|
107
|
+
* Example:
|
|
108
|
+
* ```ts
|
|
109
|
+
* type.actions(self => {
|
|
110
|
+
* function takeA____() {
|
|
111
|
+
* self.toilet.donate()
|
|
112
|
+
* self.wipe()
|
|
113
|
+
* self.wipe()
|
|
114
|
+
* self.toilet.flush()
|
|
115
|
+
* }
|
|
116
|
+
* return {
|
|
117
|
+
* takeA____: decorate(atomic, takeA____)
|
|
118
|
+
* }
|
|
119
|
+
* })
|
|
120
|
+
* ```
|
|
121
|
+
*
|
|
122
|
+
* @param handler
|
|
123
|
+
* @param fn
|
|
124
|
+
* @param includeHooks
|
|
125
|
+
* @returns The original function
|
|
126
|
+
*/
|
|
127
|
+
export function decorate(handler, fn, includeHooks = true) {
|
|
128
|
+
const middleware = { handler, includeHooks };
|
|
129
|
+
fn.$mst_middleware = fn.$mst_middleware || [];
|
|
130
|
+
fn.$mst_middleware.push(middleware);
|
|
131
|
+
return fn;
|
|
132
|
+
}
|
|
133
|
+
class CollectedMiddlewares {
|
|
134
|
+
arrayIndex = 0;
|
|
135
|
+
inArrayIndex = 0;
|
|
136
|
+
middlewares = [];
|
|
137
|
+
constructor(node, fn) {
|
|
138
|
+
// we just push middleware arrays into an array of arrays to avoid making copies
|
|
139
|
+
if (fn.$mst_middleware) {
|
|
140
|
+
this.middlewares.push(fn.$mst_middleware);
|
|
141
|
+
}
|
|
142
|
+
let n = node;
|
|
143
|
+
// Find all middlewares. Optimization: cache this?
|
|
144
|
+
while (n) {
|
|
145
|
+
if (n.middlewares) {
|
|
146
|
+
this.middlewares.push(n.middlewares);
|
|
147
|
+
}
|
|
148
|
+
n = n.parent;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
get isEmpty() {
|
|
152
|
+
return this.middlewares.length <= 0;
|
|
153
|
+
}
|
|
154
|
+
getNextMiddleware() {
|
|
155
|
+
const array = this.middlewares[this.arrayIndex];
|
|
156
|
+
if (!array) {
|
|
157
|
+
return undefined;
|
|
158
|
+
}
|
|
159
|
+
const item = array[this.inArrayIndex++];
|
|
160
|
+
if (!item) {
|
|
161
|
+
this.arrayIndex++;
|
|
162
|
+
this.inArrayIndex = 0;
|
|
163
|
+
return this.getNextMiddleware();
|
|
164
|
+
}
|
|
165
|
+
return item;
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
function runMiddleWares(node, baseCall, originalFn) {
|
|
169
|
+
const middlewares = new CollectedMiddlewares(node, originalFn);
|
|
170
|
+
// Short circuit
|
|
171
|
+
if (middlewares.isEmpty) {
|
|
172
|
+
return mobxAction(originalFn)(...baseCall.args);
|
|
173
|
+
}
|
|
174
|
+
let result = null;
|
|
175
|
+
function runNextMiddleware(call) {
|
|
176
|
+
const middleware = middlewares.getNextMiddleware();
|
|
177
|
+
const handler = middleware && middleware.handler;
|
|
178
|
+
if (!handler) {
|
|
179
|
+
return mobxAction(originalFn)(...call.args);
|
|
180
|
+
}
|
|
181
|
+
// skip hooks if asked to
|
|
182
|
+
if (!middleware.includeHooks && Hook[call.name]) {
|
|
183
|
+
return runNextMiddleware(call);
|
|
184
|
+
}
|
|
185
|
+
let nextInvoked = false;
|
|
186
|
+
function next(call2, callback) {
|
|
187
|
+
nextInvoked = true;
|
|
188
|
+
// the result can contain
|
|
189
|
+
// - the non manipulated return value from an action
|
|
190
|
+
// - the non manipulated abort value
|
|
191
|
+
// - one of the above but manipulated through the callback function
|
|
192
|
+
result = runNextMiddleware(call2);
|
|
193
|
+
if (callback) {
|
|
194
|
+
result = callback(result);
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
let abortInvoked = false;
|
|
198
|
+
function abort(value) {
|
|
199
|
+
abortInvoked = true;
|
|
200
|
+
// overwrite the result
|
|
201
|
+
// can be manipulated through middlewares earlier in the queue using the callback fn
|
|
202
|
+
result = value;
|
|
203
|
+
}
|
|
204
|
+
handler(call, next, abort);
|
|
205
|
+
if (devMode()) {
|
|
206
|
+
if (!nextInvoked && !abortInvoked) {
|
|
207
|
+
const node2 = getStateTreeNode(call.tree);
|
|
208
|
+
throw fail(`Neither the next() nor the abort() callback within the middleware ${handler.name} for the action: "${call.name}" on the node: ${node2.type.name} was invoked.`);
|
|
209
|
+
}
|
|
210
|
+
else if (nextInvoked && abortInvoked) {
|
|
211
|
+
const node2 = getStateTreeNode(call.tree);
|
|
212
|
+
throw fail(`The next() and abort() callback within the middleware ${handler.name} for the action: "${call.name}" on the node: ${node2.type.name} were invoked.`);
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
return result;
|
|
216
|
+
}
|
|
217
|
+
return runNextMiddleware(baseCall);
|
|
218
|
+
}
|
|
219
|
+
//# sourceMappingURL=action.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"action.js","sourceRoot":"","sources":["../../src/core/action.ts"],"names":[],"mappings":"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","sourcesContent":["import { action as mobxAction } from \"mobx\"\n\nimport {\n Hook,\n devMode,\n fail,\n getRoot,\n getStateTreeNode,\n warnError\n} from \"../internal.ts\"\n\nimport type {\n AnyObjectNode,\n IActionContext,\n IAnyStateTreeNode,\n IDisposer\n} from \"../internal.ts\"\n\nexport type IMiddlewareEventType =\n | \"action\"\n | \"flow_spawn\"\n | \"flow_resume\"\n | \"flow_resume_error\"\n | \"flow_return\"\n | \"flow_throw\"\n// | \"task_spawn TODO, see #273\"\n\nexport interface IMiddlewareEvent extends IActionContext {\n /** Event type */\n readonly type: IMiddlewareEventType\n\n /** Parent event unique id */\n readonly parentId: number\n /** Parent event object */\n readonly parentEvent: IMiddlewareEvent | undefined\n\n /** Root event unique id */\n readonly rootId: number\n /** Id of all events, from root until current (excluding current) */\n readonly allParentIds: number[]\n}\n\nexport interface FunctionWithFlag extends Function {\n _isMSTAction?: boolean\n _isFlowAction?: boolean\n}\n\n/**\n * @internal\n * @hidden\n */\nexport type IMiddleware = {\n handler: IMiddlewareHandler\n includeHooks: boolean\n}\n\nexport type IMiddlewareHandler = (\n actionCall: IMiddlewareEvent,\n next: (actionCall: IMiddlewareEvent, callback?: (value: any) => any) => void,\n abort: (value: any) => void\n) => any\n\nlet nextActionId = 1\nlet currentActionContext: IMiddlewareEvent | undefined\n\n/**\n * @internal\n * @hidden\n */\nexport function getCurrentActionContext() {\n return currentActionContext\n}\n\n/**\n * @internal\n * @hidden\n */\nexport function getNextActionId() {\n return nextActionId++\n}\n\n// TODO: optimize away entire action context if there is no middleware in tree?\n/**\n * @internal\n * @hidden\n */\nexport function runWithActionContext(context: IMiddlewareEvent, fn: Function) {\n const node = getStateTreeNode(context.context)\n\n if (context.type === \"action\") {\n node.assertAlive({\n actionContext: context\n })\n }\n\n const baseIsRunningAction = node._isRunningAction\n node._isRunningAction = true\n const previousContext = currentActionContext\n currentActionContext = context\n try {\n return runMiddleWares(node, context, fn)\n } finally {\n currentActionContext = previousContext\n node._isRunningAction = baseIsRunningAction\n }\n}\n\n/**\n * @internal\n * @hidden\n */\nexport function getParentActionContext(\n parentContext: IMiddlewareEvent | undefined\n) {\n if (!parentContext) {\n return undefined\n }\n if (parentContext.type === \"action\") {\n return parentContext\n }\n return parentContext.parentActionEvent\n}\n\n/**\n * @internal\n * @hidden\n */\nexport function createActionInvoker<T extends FunctionWithFlag>(\n target: IAnyStateTreeNode,\n name: string,\n fn: T\n) {\n const res = function (...args: any[]) {\n const id = getNextActionId()\n const parentContext = currentActionContext\n const parentActionContext = getParentActionContext(parentContext)\n\n return runWithActionContext(\n {\n type: \"action\",\n name,\n id,\n args,\n context: target,\n tree: getRoot(target),\n rootId: parentContext ? parentContext.rootId : id,\n parentId: parentContext ? parentContext.id : 0,\n allParentIds: parentContext\n ? [...parentContext.allParentIds, parentContext.id]\n : [],\n parentEvent: parentContext,\n parentActionEvent: parentActionContext\n },\n fn\n )\n }\n ;(res as FunctionWithFlag)._isMSTAction = true\n ;(res as FunctionWithFlag)._isFlowAction = fn._isFlowAction\n return res\n}\n\n/**\n * Middleware can be used to intercept any action is invoked on the subtree where it is attached.\n * If a tree is protected (by default), this means that any mutation of the tree will pass through your middleware.\n *\n * For more details, see the [middleware docs](concepts/middleware.md)\n *\n * @param target Node to apply the middleware to.\n * @param middleware Middleware to apply.\n * @returns A callable function to dispose the middleware.\n */\nexport function addMiddleware(\n target: IAnyStateTreeNode,\n handler: IMiddlewareHandler,\n includeHooks: boolean = true\n): IDisposer {\n const node = getStateTreeNode(target)\n if (devMode()) {\n if (!node.isProtectionEnabled) {\n warnError(\n \"It is recommended to protect the state tree before attaching action middleware, as otherwise it cannot be guaranteed that all changes are passed through middleware. See `protect`\"\n )\n }\n }\n return node.addMiddleWare(handler, includeHooks)\n}\n\n/**\n * Binds middleware to a specific action.\n *\n * Example:\n * ```ts\n * type.actions(self => {\n * function takeA____() {\n * self.toilet.donate()\n * self.wipe()\n * self.wipe()\n * self.toilet.flush()\n * }\n * return {\n * takeA____: decorate(atomic, takeA____)\n * }\n * })\n * ```\n *\n * @param handler\n * @param fn\n * @param includeHooks\n * @returns The original function\n */\nexport function decorate<T extends Function>(\n handler: IMiddlewareHandler,\n fn: T,\n includeHooks = true\n): T {\n const middleware: IMiddleware = { handler, includeHooks }\n ;(fn as any).$mst_middleware = (fn as any).$mst_middleware || []\n ;(fn as any).$mst_middleware.push(middleware)\n return fn\n}\n\nclass CollectedMiddlewares {\n private arrayIndex = 0\n private inArrayIndex = 0\n private middlewares: IMiddleware[][] = []\n\n constructor(node: AnyObjectNode, fn: Function) {\n // we just push middleware arrays into an array of arrays to avoid making copies\n if ((fn as any).$mst_middleware) {\n this.middlewares.push((fn as any).$mst_middleware)\n }\n let n: AnyObjectNode | null = node\n // Find all middlewares. Optimization: cache this?\n while (n) {\n if (n.middlewares) {\n this.middlewares.push(n.middlewares)\n }\n n = n.parent\n }\n }\n\n get isEmpty() {\n return this.middlewares.length <= 0\n }\n\n getNextMiddleware(): IMiddleware | undefined {\n const array = this.middlewares[this.arrayIndex]\n if (!array) {\n return undefined\n }\n const item = array[this.inArrayIndex++]\n if (!item) {\n this.arrayIndex++\n this.inArrayIndex = 0\n return this.getNextMiddleware()\n }\n return item\n }\n}\n\nfunction runMiddleWares(\n node: AnyObjectNode,\n baseCall: IMiddlewareEvent,\n originalFn: Function\n): any {\n const middlewares = new CollectedMiddlewares(node, originalFn)\n // Short circuit\n if (middlewares.isEmpty) {\n return mobxAction(originalFn)(...baseCall.args)\n }\n\n let result: any = null\n\n function runNextMiddleware(call: IMiddlewareEvent): any {\n const middleware = middlewares.getNextMiddleware()\n const handler = middleware && middleware.handler\n\n if (!handler) {\n return mobxAction(originalFn)(...call.args)\n }\n\n // skip hooks if asked to\n if (!middleware!.includeHooks && (Hook as any)[call.name]) {\n return runNextMiddleware(call)\n }\n\n let nextInvoked = false\n function next(\n call2: IMiddlewareEvent,\n callback?: (value: any) => any\n ): void {\n nextInvoked = true\n // the result can contain\n // - the non manipulated return value from an action\n // - the non manipulated abort value\n // - one of the above but manipulated through the callback function\n result = runNextMiddleware(call2)\n if (callback) {\n result = callback(result)\n }\n }\n\n let abortInvoked = false\n function abort(value: any) {\n abortInvoked = true\n // overwrite the result\n // can be manipulated through middlewares earlier in the queue using the callback fn\n result = value\n }\n\n handler(call, next, abort)\n if (devMode()) {\n if (!nextInvoked && !abortInvoked) {\n const node2 = getStateTreeNode(call.tree)\n throw fail(\n `Neither the next() nor the abort() callback within the middleware ${handler.name} for the action: \"${call.name}\" on the node: ${node2.type.name} was invoked.`\n )\n } else if (nextInvoked && abortInvoked) {\n const node2 = getStateTreeNode(call.tree)\n throw fail(\n `The next() and abort() callback within the middleware ${handler.name} for the action: \"${call.name}\" on the node: ${node2.type.name} were invoked.`\n )\n }\n }\n return result\n }\n return runNextMiddleware(baseCall)\n}\n"]}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { type IAnyStateTreeNode, type IMiddlewareEvent } from "../internal.ts";
|
|
2
|
+
export interface IActionContext {
|
|
3
|
+
/** Event name (action name for actions) */
|
|
4
|
+
readonly name: string;
|
|
5
|
+
/** Event unique id */
|
|
6
|
+
readonly id: number;
|
|
7
|
+
/** Parent action event object */
|
|
8
|
+
readonly parentActionEvent: IMiddlewareEvent | undefined;
|
|
9
|
+
/** Event context (node where the action was invoked) */
|
|
10
|
+
readonly context: IAnyStateTreeNode;
|
|
11
|
+
/** Event tree (root node of the node where the action was invoked) */
|
|
12
|
+
readonly tree: IAnyStateTreeNode;
|
|
13
|
+
/** Event arguments in an array (action arguments for actions) */
|
|
14
|
+
readonly args: any[];
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Returns the currently executing MST action context, or undefined if none.
|
|
18
|
+
*/
|
|
19
|
+
export declare function getRunningActionContext(): IActionContext | undefined;
|
|
20
|
+
/**
|
|
21
|
+
* Returns if the given action context is a parent of this action context.
|
|
22
|
+
*/
|
|
23
|
+
export declare function isActionContextChildOf(actionContext: IActionContext, parent: number | IActionContext | IMiddlewareEvent): boolean;
|
|
24
|
+
/**
|
|
25
|
+
* Returns if the given action context is this or a parent of this action context.
|
|
26
|
+
*/
|
|
27
|
+
export declare function isActionContextThisOrChildOf(actionContext: IActionContext, parentOrThis: number | IActionContext | IMiddlewareEvent): boolean;
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { getCurrentActionContext } from "./action.js";
|
|
2
|
+
/**
|
|
3
|
+
* Returns the currently executing MST action context, or undefined if none.
|
|
4
|
+
*/
|
|
5
|
+
export function getRunningActionContext() {
|
|
6
|
+
let current = getCurrentActionContext();
|
|
7
|
+
while (current && current.type !== "action") {
|
|
8
|
+
current = current.parentActionEvent;
|
|
9
|
+
}
|
|
10
|
+
return current;
|
|
11
|
+
}
|
|
12
|
+
function _isActionContextThisOrChildOf(actionContext, sameOrParent, includeSame) {
|
|
13
|
+
const parentId = typeof sameOrParent === "number" ? sameOrParent : sameOrParent.id;
|
|
14
|
+
let current = includeSame
|
|
15
|
+
? actionContext
|
|
16
|
+
: actionContext.parentActionEvent;
|
|
17
|
+
while (current) {
|
|
18
|
+
if (current.id === parentId) {
|
|
19
|
+
return true;
|
|
20
|
+
}
|
|
21
|
+
current = current.parentActionEvent;
|
|
22
|
+
}
|
|
23
|
+
return false;
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* Returns if the given action context is a parent of this action context.
|
|
27
|
+
*/
|
|
28
|
+
export function isActionContextChildOf(actionContext, parent) {
|
|
29
|
+
return _isActionContextThisOrChildOf(actionContext, parent, false);
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Returns if the given action context is this or a parent of this action context.
|
|
33
|
+
*/
|
|
34
|
+
export function isActionContextThisOrChildOf(actionContext, parentOrThis) {
|
|
35
|
+
return _isActionContextThisOrChildOf(actionContext, parentOrThis, true);
|
|
36
|
+
}
|
|
37
|
+
//# sourceMappingURL=actionContext.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"actionContext.js","sourceRoot":"","sources":["../../src/core/actionContext.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,uBAAuB,EAAE,MAAM,aAAa,CAAA;AAqBrD;;GAEG;AACH,MAAM,UAAU,uBAAuB;IACrC,IAAI,OAAO,GAAG,uBAAuB,EAAE,CAAA;IACvC,OAAO,OAAO,IAAI,OAAO,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;QAC5C,OAAO,GAAG,OAAO,CAAC,iBAAiB,CAAA;IACrC,CAAC;IACD,OAAO,OAAO,CAAA;AAChB,CAAC;AAED,SAAS,6BAA6B,CACpC,aAA6B,EAC7B,YAAwD,EACxD,WAAoB;IAEpB,MAAM,QAAQ,GACZ,OAAO,YAAY,KAAK,QAAQ,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,YAAY,CAAC,EAAE,CAAA;IAEnE,IAAI,OAAO,GAAkD,WAAW;QACtE,CAAC,CAAC,aAAa;QACf,CAAC,CAAC,aAAa,CAAC,iBAAiB,CAAA;IACnC,OAAO,OAAO,EAAE,CAAC;QACf,IAAI,OAAO,CAAC,EAAE,KAAK,QAAQ,EAAE,CAAC;YAC5B,OAAO,IAAI,CAAA;QACb,CAAC;QACD,OAAO,GAAG,OAAO,CAAC,iBAAiB,CAAA;IACrC,CAAC;IACD,OAAO,KAAK,CAAA;AACd,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,sBAAsB,CACpC,aAA6B,EAC7B,MAAkD;IAElD,OAAO,6BAA6B,CAAC,aAAa,EAAE,MAAM,EAAE,KAAK,CAAC,CAAA;AACpE,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,4BAA4B,CAC1C,aAA6B,EAC7B,YAAwD;IAExD,OAAO,6BAA6B,CAAC,aAAa,EAAE,YAAY,EAAE,IAAI,CAAC,CAAA;AACzE,CAAC","sourcesContent":["import { type IAnyStateTreeNode, type IMiddlewareEvent } from \"../internal.ts\"\nimport { getCurrentActionContext } from \"./action.ts\"\n\nexport interface IActionContext {\n /** Event name (action name for actions) */\n readonly name: string\n\n /** Event unique id */\n readonly id: number\n\n /** Parent action event object */\n readonly parentActionEvent: IMiddlewareEvent | undefined\n\n /** Event context (node where the action was invoked) */\n readonly context: IAnyStateTreeNode\n /** Event tree (root node of the node where the action was invoked) */\n readonly tree: IAnyStateTreeNode\n\n /** Event arguments in an array (action arguments for actions) */\n readonly args: any[]\n}\n\n/**\n * Returns the currently executing MST action context, or undefined if none.\n */\nexport function getRunningActionContext(): IActionContext | undefined {\n let current = getCurrentActionContext()\n while (current && current.type !== \"action\") {\n current = current.parentActionEvent\n }\n return current\n}\n\nfunction _isActionContextThisOrChildOf(\n actionContext: IActionContext,\n sameOrParent: number | IActionContext | IMiddlewareEvent,\n includeSame: boolean\n) {\n const parentId =\n typeof sameOrParent === \"number\" ? sameOrParent : sameOrParent.id\n\n let current: IActionContext | IMiddlewareEvent | undefined = includeSame\n ? actionContext\n : actionContext.parentActionEvent\n while (current) {\n if (current.id === parentId) {\n return true\n }\n current = current.parentActionEvent\n }\n return false\n}\n\n/**\n * Returns if the given action context is a parent of this action context.\n */\nexport function isActionContextChildOf(\n actionContext: IActionContext,\n parent: number | IActionContext | IMiddlewareEvent\n) {\n return _isActionContextThisOrChildOf(actionContext, parent, false)\n}\n\n/**\n * Returns if the given action context is this or a parent of this action context.\n */\nexport function isActionContextThisOrChildOf(\n actionContext: IActionContext,\n parentOrThis: number | IActionContext | IMiddlewareEvent\n) {\n return _isActionContextThisOrChildOf(actionContext, parentOrThis, true)\n}\n"]}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import type { FunctionWithFlag } from "./action.ts";
|
|
2
|
+
/**
|
|
3
|
+
* @hidden
|
|
4
|
+
*/
|
|
5
|
+
export type FlowReturn<R> = R extends Promise<infer T> ? T : R;
|
|
6
|
+
/**
|
|
7
|
+
* See [asynchronous actions](concepts/async-actions.md).
|
|
8
|
+
*
|
|
9
|
+
* @returns The flow as a promise.
|
|
10
|
+
*/
|
|
11
|
+
export declare function flow<R, Args extends any[]>(generator: (...args: Args) => Generator<PromiseLike<any>, R, any>): (...args: Args) => Promise<FlowReturn<R>>;
|
|
12
|
+
/**
|
|
13
|
+
* @deprecated Not needed since TS3.6.
|
|
14
|
+
* Used for TypeScript to make flows that return a promise return the actual promise result.
|
|
15
|
+
*
|
|
16
|
+
* @param val
|
|
17
|
+
* @returns
|
|
18
|
+
*/
|
|
19
|
+
export declare function castFlowReturn<T>(val: T): T;
|
|
20
|
+
/**
|
|
21
|
+
* @experimental
|
|
22
|
+
* experimental api - might change on minor/patch releases
|
|
23
|
+
*
|
|
24
|
+
* Convert a promise-returning function to a generator-returning one.
|
|
25
|
+
* This is intended to allow for usage of `yield*` in async actions to
|
|
26
|
+
* retain the promise return type.
|
|
27
|
+
*
|
|
28
|
+
* Example:
|
|
29
|
+
* ```ts
|
|
30
|
+
* function getDataAsync(input: string): Promise<number> { ... }
|
|
31
|
+
* const getDataGen = toGeneratorFunction(getDataAsync);
|
|
32
|
+
*
|
|
33
|
+
* const someModel.actions(self => ({
|
|
34
|
+
* someAction: flow(function*() {
|
|
35
|
+
* // value is typed as number
|
|
36
|
+
* const value = yield* getDataGen("input value");
|
|
37
|
+
* ...
|
|
38
|
+
* })
|
|
39
|
+
* }))
|
|
40
|
+
* ```
|
|
41
|
+
*/
|
|
42
|
+
export declare function toGeneratorFunction<R, Args extends any[]>(p: (...args: Args) => Promise<R>): (...args: Args) => Generator<Promise<R>, R, R>;
|
|
43
|
+
/**
|
|
44
|
+
* @experimental
|
|
45
|
+
* experimental api - might change on minor/patch releases
|
|
46
|
+
*
|
|
47
|
+
* Convert a promise to a generator yielding that promise
|
|
48
|
+
* This is intended to allow for usage of `yield*` in async actions to
|
|
49
|
+
* retain the promise return type.
|
|
50
|
+
*
|
|
51
|
+
* Example:
|
|
52
|
+
* ```ts
|
|
53
|
+
* function getDataAsync(input: string): Promise<number> { ... }
|
|
54
|
+
*
|
|
55
|
+
* const someModel.actions(self => ({
|
|
56
|
+
* someAction: flow(function*() {
|
|
57
|
+
* // value is typed as number
|
|
58
|
+
* const value = yield* toGenerator(getDataAsync("input value"));
|
|
59
|
+
* ...
|
|
60
|
+
* })
|
|
61
|
+
* }))
|
|
62
|
+
* ```
|
|
63
|
+
*/
|
|
64
|
+
export declare function toGenerator<R>(p: Promise<R>): Generator<Promise<R>, R, R>;
|
|
65
|
+
/**
|
|
66
|
+
* @internal
|
|
67
|
+
* @hidden
|
|
68
|
+
*/
|
|
69
|
+
export declare function createFlowSpawner(name: string, generator: FunctionWithFlag): (this: any, ...flowArgs: any[]) => Promise<unknown>;
|
package/esm/core/flow.js
ADDED
|
@@ -0,0 +1,173 @@
|
|
|
1
|
+
import { fail, setImmediateWithFallback } from "../utils.js";
|
|
2
|
+
import { getCurrentActionContext, getNextActionId, getParentActionContext, runWithActionContext } from "./action.js";
|
|
3
|
+
/**
|
|
4
|
+
* See [asynchronous actions](concepts/async-actions.md).
|
|
5
|
+
*
|
|
6
|
+
* @returns The flow as a promise.
|
|
7
|
+
*/
|
|
8
|
+
export function flow(generator) {
|
|
9
|
+
return createFlowSpawner(generator.name, generator);
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* @deprecated Not needed since TS3.6.
|
|
13
|
+
* Used for TypeScript to make flows that return a promise return the actual promise result.
|
|
14
|
+
*
|
|
15
|
+
* @param val
|
|
16
|
+
* @returns
|
|
17
|
+
*/
|
|
18
|
+
export function castFlowReturn(val) {
|
|
19
|
+
return val;
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* @experimental
|
|
23
|
+
* experimental api - might change on minor/patch releases
|
|
24
|
+
*
|
|
25
|
+
* Convert a promise-returning function to a generator-returning one.
|
|
26
|
+
* This is intended to allow for usage of `yield*` in async actions to
|
|
27
|
+
* retain the promise return type.
|
|
28
|
+
*
|
|
29
|
+
* Example:
|
|
30
|
+
* ```ts
|
|
31
|
+
* function getDataAsync(input: string): Promise<number> { ... }
|
|
32
|
+
* const getDataGen = toGeneratorFunction(getDataAsync);
|
|
33
|
+
*
|
|
34
|
+
* const someModel.actions(self => ({
|
|
35
|
+
* someAction: flow(function*() {
|
|
36
|
+
* // value is typed as number
|
|
37
|
+
* const value = yield* getDataGen("input value");
|
|
38
|
+
* ...
|
|
39
|
+
* })
|
|
40
|
+
* }))
|
|
41
|
+
* ```
|
|
42
|
+
*/
|
|
43
|
+
export function toGeneratorFunction(p) {
|
|
44
|
+
return function* (...args) {
|
|
45
|
+
return (yield p(...args));
|
|
46
|
+
};
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* @experimental
|
|
50
|
+
* experimental api - might change on minor/patch releases
|
|
51
|
+
*
|
|
52
|
+
* Convert a promise to a generator yielding that promise
|
|
53
|
+
* This is intended to allow for usage of `yield*` in async actions to
|
|
54
|
+
* retain the promise return type.
|
|
55
|
+
*
|
|
56
|
+
* Example:
|
|
57
|
+
* ```ts
|
|
58
|
+
* function getDataAsync(input: string): Promise<number> { ... }
|
|
59
|
+
*
|
|
60
|
+
* const someModel.actions(self => ({
|
|
61
|
+
* someAction: flow(function*() {
|
|
62
|
+
* // value is typed as number
|
|
63
|
+
* const value = yield* toGenerator(getDataAsync("input value"));
|
|
64
|
+
* ...
|
|
65
|
+
* })
|
|
66
|
+
* }))
|
|
67
|
+
* ```
|
|
68
|
+
*/
|
|
69
|
+
export function* toGenerator(p) {
|
|
70
|
+
return (yield p);
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* @internal
|
|
74
|
+
* @hidden
|
|
75
|
+
*/
|
|
76
|
+
export function createFlowSpawner(name, generator) {
|
|
77
|
+
const spawner = function flowSpawner(...flowArgs) {
|
|
78
|
+
// Implementation based on https://github.com/tj/co/blob/master/index.js
|
|
79
|
+
const runId = getNextActionId();
|
|
80
|
+
const parentContext = getCurrentActionContext();
|
|
81
|
+
if (!parentContext) {
|
|
82
|
+
throw fail("a mst flow must always have a parent context");
|
|
83
|
+
}
|
|
84
|
+
const parentActionContext = getParentActionContext(parentContext);
|
|
85
|
+
if (!parentActionContext) {
|
|
86
|
+
throw fail("a mst flow must always have a parent action context");
|
|
87
|
+
}
|
|
88
|
+
const contextBase = {
|
|
89
|
+
name,
|
|
90
|
+
id: runId,
|
|
91
|
+
tree: parentContext.tree,
|
|
92
|
+
context: parentContext.context,
|
|
93
|
+
parentId: parentContext.id,
|
|
94
|
+
allParentIds: [...parentContext.allParentIds, parentContext.id],
|
|
95
|
+
rootId: parentContext.rootId,
|
|
96
|
+
parentEvent: parentContext,
|
|
97
|
+
parentActionEvent: parentActionContext
|
|
98
|
+
};
|
|
99
|
+
function wrap(fn, type, arg) {
|
|
100
|
+
fn.$mst_middleware = spawner.$mst_middleware; // pick up any middleware attached to the flow
|
|
101
|
+
return runWithActionContext({
|
|
102
|
+
...contextBase,
|
|
103
|
+
type,
|
|
104
|
+
args: [arg]
|
|
105
|
+
}, fn);
|
|
106
|
+
}
|
|
107
|
+
return new Promise(function (resolve, reject) {
|
|
108
|
+
let gen;
|
|
109
|
+
const init = function asyncActionInit(...initArgs) {
|
|
110
|
+
gen = generator(...initArgs);
|
|
111
|
+
onFulfilled(undefined); // kick off the flow
|
|
112
|
+
};
|
|
113
|
+
init.$mst_middleware = spawner.$mst_middleware;
|
|
114
|
+
runWithActionContext({
|
|
115
|
+
...contextBase,
|
|
116
|
+
type: "flow_spawn",
|
|
117
|
+
args: flowArgs
|
|
118
|
+
}, init);
|
|
119
|
+
function onFulfilled(res) {
|
|
120
|
+
let ret;
|
|
121
|
+
try {
|
|
122
|
+
// prettier-ignore
|
|
123
|
+
const cancelError = wrap((_r) => { ret = gen.next(_r); }, "flow_resume", res);
|
|
124
|
+
if (cancelError instanceof Error) {
|
|
125
|
+
ret = gen.throw(cancelError);
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
catch (e) {
|
|
129
|
+
// prettier-ignore
|
|
130
|
+
setImmediateWithFallback(() => {
|
|
131
|
+
wrap((_r) => { reject(e); }, "flow_throw", e);
|
|
132
|
+
});
|
|
133
|
+
return;
|
|
134
|
+
}
|
|
135
|
+
next(ret);
|
|
136
|
+
return;
|
|
137
|
+
}
|
|
138
|
+
function onRejected(err) {
|
|
139
|
+
let ret;
|
|
140
|
+
try {
|
|
141
|
+
// prettier-ignore
|
|
142
|
+
wrap((_r) => { ret = gen.throw(_r); }, "flow_resume_error", err); // or yieldError?
|
|
143
|
+
}
|
|
144
|
+
catch (e) {
|
|
145
|
+
// prettier-ignore
|
|
146
|
+
setImmediateWithFallback(() => {
|
|
147
|
+
wrap((_r) => { reject(e); }, "flow_throw", e);
|
|
148
|
+
});
|
|
149
|
+
return;
|
|
150
|
+
}
|
|
151
|
+
next(ret);
|
|
152
|
+
}
|
|
153
|
+
function next(ret) {
|
|
154
|
+
if (ret.done) {
|
|
155
|
+
// prettier-ignore
|
|
156
|
+
setImmediateWithFallback(() => {
|
|
157
|
+
wrap((r) => { resolve(r); }, "flow_return", ret.value);
|
|
158
|
+
});
|
|
159
|
+
return;
|
|
160
|
+
}
|
|
161
|
+
// TODO: support more type of values? See https://github.com/tj/co/blob/249bbdc72da24ae44076afd716349d2089b31c4c/index.js#L100
|
|
162
|
+
if (!ret.value || typeof ret.value.then !== "function") {
|
|
163
|
+
// istanbul ignore next
|
|
164
|
+
throw fail("Only promises can be yielded to `async`, got: " + ret);
|
|
165
|
+
}
|
|
166
|
+
return ret.value.then(onFulfilled, onRejected);
|
|
167
|
+
}
|
|
168
|
+
});
|
|
169
|
+
};
|
|
170
|
+
spawner._isFlowAction = true;
|
|
171
|
+
return spawner;
|
|
172
|
+
}
|
|
173
|
+
//# sourceMappingURL=flow.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"flow.js","sourceRoot":"","sources":["../../src/core/flow.ts"],"names":[],"mappings":"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","sourcesContent":["import { fail, setImmediateWithFallback } from \"../utils.ts\"\nimport {\n getCurrentActionContext,\n getNextActionId,\n getParentActionContext,\n runWithActionContext\n} from \"./action.ts\"\n\nimport type { FunctionWithFlag, IMiddlewareEventType } from \"./action.ts\"\n\n/**\n * @hidden\n */\nexport type FlowReturn<R> = R extends Promise<infer T> ? T : R\n\n/**\n * See [asynchronous actions](concepts/async-actions.md).\n *\n * @returns The flow as a promise.\n */\nexport function flow<R, Args extends any[]>(\n generator: (...args: Args) => Generator<PromiseLike<any>, R, any>\n): (...args: Args) => Promise<FlowReturn<R>> {\n return createFlowSpawner(generator.name, generator) as any\n}\n\n/**\n * @deprecated Not needed since TS3.6.\n * Used for TypeScript to make flows that return a promise return the actual promise result.\n *\n * @param val\n * @returns\n */\nexport function castFlowReturn<T>(val: T): T {\n return val as any\n}\n\n/**\n * @experimental\n * experimental api - might change on minor/patch releases\n *\n * Convert a promise-returning function to a generator-returning one.\n * This is intended to allow for usage of `yield*` in async actions to\n * retain the promise return type.\n *\n * Example:\n * ```ts\n * function getDataAsync(input: string): Promise<number> { ... }\n * const getDataGen = toGeneratorFunction(getDataAsync);\n *\n * const someModel.actions(self => ({\n * someAction: flow(function*() {\n * // value is typed as number\n * const value = yield* getDataGen(\"input value\");\n * ...\n * })\n * }))\n * ```\n */\nexport function toGeneratorFunction<R, Args extends any[]>(\n p: (...args: Args) => Promise<R>\n) {\n return function* (...args: Args) {\n return (yield p(...args)) as R\n }\n}\n\n/**\n * @experimental\n * experimental api - might change on minor/patch releases\n *\n * Convert a promise to a generator yielding that promise\n * This is intended to allow for usage of `yield*` in async actions to\n * retain the promise return type.\n *\n * Example:\n * ```ts\n * function getDataAsync(input: string): Promise<number> { ... }\n *\n * const someModel.actions(self => ({\n * someAction: flow(function*() {\n * // value is typed as number\n * const value = yield* toGenerator(getDataAsync(\"input value\"));\n * ...\n * })\n * }))\n * ```\n */\nexport function* toGenerator<R>(p: Promise<R>) {\n return (yield p) as R\n}\n\n/**\n * @internal\n * @hidden\n */\nexport function createFlowSpawner(name: string, generator: FunctionWithFlag) {\n const spawner = function flowSpawner(this: any, ...flowArgs: any[]) {\n // Implementation based on https://github.com/tj/co/blob/master/index.js\n const runId = getNextActionId()\n const parentContext = getCurrentActionContext()!\n if (!parentContext) {\n throw fail(\"a mst flow must always have a parent context\")\n }\n const parentActionContext = getParentActionContext(parentContext)\n if (!parentActionContext) {\n throw fail(\"a mst flow must always have a parent action context\")\n }\n\n const contextBase = {\n name,\n id: runId,\n tree: parentContext.tree,\n context: parentContext.context,\n parentId: parentContext.id,\n allParentIds: [...parentContext.allParentIds, parentContext.id],\n rootId: parentContext.rootId,\n parentEvent: parentContext,\n parentActionEvent: parentActionContext\n }\n\n function wrap(fn: any, type: IMiddlewareEventType, arg: any) {\n fn.$mst_middleware = (spawner as any).$mst_middleware // pick up any middleware attached to the flow\n return runWithActionContext(\n {\n ...contextBase,\n type,\n args: [arg]\n },\n fn\n )\n }\n\n return new Promise(function (resolve, reject) {\n let gen: any\n const init = function asyncActionInit(...initArgs: any[]) {\n gen = generator(...initArgs)\n onFulfilled(undefined) // kick off the flow\n }\n ;(init as any).$mst_middleware = (spawner as any).$mst_middleware\n\n runWithActionContext(\n {\n ...contextBase,\n type: \"flow_spawn\",\n args: flowArgs\n },\n init\n )\n\n function onFulfilled(res: any) {\n let ret\n try {\n // prettier-ignore\n const cancelError: any = wrap((_r: any) => { ret = gen.next(_r) }, \"flow_resume\", res)\n if (cancelError instanceof Error) {\n ret = gen.throw(cancelError)\n }\n } catch (e) {\n // prettier-ignore\n setImmediateWithFallback(() => {\n wrap((_r: any) => { reject(e) }, \"flow_throw\", e)\n })\n return\n }\n next(ret)\n return\n }\n\n function onRejected(err: any) {\n let ret\n try {\n // prettier-ignore\n wrap((_r: any) => { ret = gen.throw(_r) }, \"flow_resume_error\", err) // or yieldError?\n } catch (e) {\n // prettier-ignore\n setImmediateWithFallback(() => {\n wrap((_r: any) => { reject(e) }, \"flow_throw\", e)\n })\n return\n }\n next(ret)\n }\n\n function next(ret: any) {\n if (ret.done) {\n // prettier-ignore\n setImmediateWithFallback(() => {\n wrap((r: any) => { resolve(r) }, \"flow_return\", ret.value)\n })\n return\n }\n // TODO: support more type of values? See https://github.com/tj/co/blob/249bbdc72da24ae44076afd716349d2089b31c4c/index.js#L100\n if (!ret.value || typeof ret.value.then !== \"function\") {\n // istanbul ignore next\n throw fail(\"Only promises can be yielded to `async`, got: \" + ret)\n }\n return ret.value.then(onFulfilled, onRejected)\n }\n })\n }\n ;(spawner as FunctionWithFlag)._isFlowAction = true\n return spawner\n}\n"]}
|