@variojs/core 0.0.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/README.md +270 -0
- package/dist/errors.d.ts +100 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +132 -0
- package/dist/errors.js.map +1 -0
- package/dist/expression/cache.d.ts +53 -0
- package/dist/expression/cache.d.ts.map +1 -0
- package/dist/expression/cache.js +158 -0
- package/dist/expression/cache.js.map +1 -0
- package/dist/expression/compiler.d.ts +34 -0
- package/dist/expression/compiler.d.ts.map +1 -0
- package/dist/expression/compiler.js +141 -0
- package/dist/expression/compiler.js.map +1 -0
- package/dist/expression/dependencies.d.ts +17 -0
- package/dist/expression/dependencies.d.ts.map +1 -0
- package/dist/expression/dependencies.js +106 -0
- package/dist/expression/dependencies.js.map +1 -0
- package/dist/expression/evaluate.d.ts +22 -0
- package/dist/expression/evaluate.d.ts.map +1 -0
- package/dist/expression/evaluate.js +75 -0
- package/dist/expression/evaluate.js.map +1 -0
- package/dist/expression/evaluator.d.ts +22 -0
- package/dist/expression/evaluator.d.ts.map +1 -0
- package/dist/expression/evaluator.js +533 -0
- package/dist/expression/evaluator.js.map +1 -0
- package/dist/expression/index.d.ts +12 -0
- package/dist/expression/index.d.ts.map +1 -0
- package/dist/expression/index.js +12 -0
- package/dist/expression/index.js.map +1 -0
- package/dist/expression/parser.d.ts +15 -0
- package/dist/expression/parser.d.ts.map +1 -0
- package/dist/expression/parser.js +42 -0
- package/dist/expression/parser.js.map +1 -0
- package/dist/expression/utils.d.ts +46 -0
- package/dist/expression/utils.d.ts.map +1 -0
- package/dist/expression/utils.js +78 -0
- package/dist/expression/utils.js.map +1 -0
- package/dist/expression/whitelist.d.ts +24 -0
- package/dist/expression/whitelist.d.ts.map +1 -0
- package/dist/expression/whitelist.js +198 -0
- package/dist/expression/whitelist.js.map +1 -0
- package/dist/index.d.ts +18 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +21 -0
- package/dist/index.js.map +1 -0
- package/dist/runtime/context.d.ts +8 -0
- package/dist/runtime/context.d.ts.map +1 -0
- package/dist/runtime/context.js +7 -0
- package/dist/runtime/context.js.map +1 -0
- package/dist/runtime/create-context.d.ts +20 -0
- package/dist/runtime/create-context.d.ts.map +1 -0
- package/dist/runtime/create-context.js +62 -0
- package/dist/runtime/create-context.js.map +1 -0
- package/dist/runtime/index.d.ts +10 -0
- package/dist/runtime/index.d.ts.map +1 -0
- package/dist/runtime/index.js +10 -0
- package/dist/runtime/index.js.map +1 -0
- package/dist/runtime/loop-context-pool.d.ts +58 -0
- package/dist/runtime/loop-context-pool.d.ts.map +1 -0
- package/dist/runtime/loop-context-pool.js +114 -0
- package/dist/runtime/loop-context-pool.js.map +1 -0
- package/dist/runtime/path.d.ts +114 -0
- package/dist/runtime/path.d.ts.map +1 -0
- package/dist/runtime/path.js +302 -0
- package/dist/runtime/path.js.map +1 -0
- package/dist/runtime/proxy.d.ts +18 -0
- package/dist/runtime/proxy.d.ts.map +1 -0
- package/dist/runtime/proxy.js +54 -0
- package/dist/runtime/proxy.js.map +1 -0
- package/dist/runtime/sandbox.d.ts +20 -0
- package/dist/runtime/sandbox.d.ts.map +1 -0
- package/dist/runtime/sandbox.js +32 -0
- package/dist/runtime/sandbox.js.map +1 -0
- package/dist/types.d.ts +191 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +20 -0
- package/dist/types.js.map +1 -0
- package/dist/vm/errors.d.ts +5 -0
- package/dist/vm/errors.d.ts.map +1 -0
- package/dist/vm/errors.js +5 -0
- package/dist/vm/errors.js.map +1 -0
- package/dist/vm/executor.d.ts +35 -0
- package/dist/vm/executor.d.ts.map +1 -0
- package/dist/vm/executor.js +137 -0
- package/dist/vm/executor.js.map +1 -0
- package/dist/vm/handlers/array/pop.d.ts +12 -0
- package/dist/vm/handlers/array/pop.d.ts.map +1 -0
- package/dist/vm/handlers/array/pop.js +28 -0
- package/dist/vm/handlers/array/pop.js.map +1 -0
- package/dist/vm/handlers/array/push.d.ts +13 -0
- package/dist/vm/handlers/array/push.d.ts.map +1 -0
- package/dist/vm/handlers/array/push.js +42 -0
- package/dist/vm/handlers/array/push.js.map +1 -0
- package/dist/vm/handlers/array/shift.d.ts +12 -0
- package/dist/vm/handlers/array/shift.d.ts.map +1 -0
- package/dist/vm/handlers/array/shift.js +28 -0
- package/dist/vm/handlers/array/shift.js.map +1 -0
- package/dist/vm/handlers/array/splice.d.ts +12 -0
- package/dist/vm/handlers/array/splice.d.ts.map +1 -0
- package/dist/vm/handlers/array/splice.js +59 -0
- package/dist/vm/handlers/array/splice.js.map +1 -0
- package/dist/vm/handlers/array/unshift.d.ts +13 -0
- package/dist/vm/handlers/array/unshift.d.ts.map +1 -0
- package/dist/vm/handlers/array/unshift.js +42 -0
- package/dist/vm/handlers/array/unshift.js.map +1 -0
- package/dist/vm/handlers/array/utils.d.ts +10 -0
- package/dist/vm/handlers/array/utils.d.ts.map +1 -0
- package/dist/vm/handlers/array/utils.js +33 -0
- package/dist/vm/handlers/array/utils.js.map +1 -0
- package/dist/vm/handlers/batch.d.ts +12 -0
- package/dist/vm/handlers/batch.d.ts.map +1 -0
- package/dist/vm/handlers/batch.js +40 -0
- package/dist/vm/handlers/batch.js.map +1 -0
- package/dist/vm/handlers/call.d.ts +14 -0
- package/dist/vm/handlers/call.d.ts.map +1 -0
- package/dist/vm/handlers/call.js +65 -0
- package/dist/vm/handlers/call.js.map +1 -0
- package/dist/vm/handlers/emit.d.ts +12 -0
- package/dist/vm/handlers/emit.d.ts.map +1 -0
- package/dist/vm/handlers/emit.js +26 -0
- package/dist/vm/handlers/emit.js.map +1 -0
- package/dist/vm/handlers/if.d.ts +13 -0
- package/dist/vm/handlers/if.d.ts.map +1 -0
- package/dist/vm/handlers/if.js +35 -0
- package/dist/vm/handlers/if.js.map +1 -0
- package/dist/vm/handlers/index.d.ts +14 -0
- package/dist/vm/handlers/index.d.ts.map +1 -0
- package/dist/vm/handlers/index.js +55 -0
- package/dist/vm/handlers/index.js.map +1 -0
- package/dist/vm/handlers/log.d.ts +12 -0
- package/dist/vm/handlers/log.d.ts.map +1 -0
- package/dist/vm/handlers/log.js +41 -0
- package/dist/vm/handlers/log.js.map +1 -0
- package/dist/vm/handlers/loop.d.ts +12 -0
- package/dist/vm/handlers/loop.d.ts.map +1 -0
- package/dist/vm/handlers/loop.js +71 -0
- package/dist/vm/handlers/loop.js.map +1 -0
- package/dist/vm/handlers/navigate.d.ts +12 -0
- package/dist/vm/handlers/navigate.d.ts.map +1 -0
- package/dist/vm/handlers/navigate.js +43 -0
- package/dist/vm/handlers/navigate.js.map +1 -0
- package/dist/vm/handlers/set.d.ts +15 -0
- package/dist/vm/handlers/set.d.ts.map +1 -0
- package/dist/vm/handlers/set.js +30 -0
- package/dist/vm/handlers/set.js.map +1 -0
- package/dist/vm/index.d.ts +8 -0
- package/dist/vm/index.d.ts.map +1 -0
- package/dist/vm/index.js +7 -0
- package/dist/vm/index.js.map +1 -0
- package/package.json +38 -0
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 内置动作处理器
|
|
3
|
+
*
|
|
4
|
+
* 所有内置动作统一注册到 ctx.$methods
|
|
5
|
+
*/
|
|
6
|
+
import { handleSet } from './set.js';
|
|
7
|
+
import { handleEmit } from './emit.js';
|
|
8
|
+
import { handleIf } from './if.js';
|
|
9
|
+
import { handleLoop } from './loop.js';
|
|
10
|
+
import { handleCall } from './call.js';
|
|
11
|
+
import { handleBatch } from './batch.js';
|
|
12
|
+
import { handleNavigate } from './navigate.js';
|
|
13
|
+
import { handleLog } from './log.js';
|
|
14
|
+
import { handlePush } from './array/push.js';
|
|
15
|
+
import { handlePop } from './array/pop.js';
|
|
16
|
+
import { handleShift } from './array/shift.js';
|
|
17
|
+
import { handleUnshift } from './array/unshift.js';
|
|
18
|
+
import { handleSplice } from './array/splice.js';
|
|
19
|
+
/**
|
|
20
|
+
* 内置动作处理器映射
|
|
21
|
+
*/
|
|
22
|
+
const BUILTIN_METHODS = {
|
|
23
|
+
// 原子动作
|
|
24
|
+
'set': handleSet,
|
|
25
|
+
'emit': handleEmit,
|
|
26
|
+
'navigate': handleNavigate,
|
|
27
|
+
'log': handleLog,
|
|
28
|
+
// 控制流动作
|
|
29
|
+
'if': handleIf,
|
|
30
|
+
'loop': handleLoop,
|
|
31
|
+
// 复合动作
|
|
32
|
+
'call': handleCall,
|
|
33
|
+
'batch': handleBatch,
|
|
34
|
+
// 数组操作动作
|
|
35
|
+
'push': handlePush,
|
|
36
|
+
'pop': handlePop,
|
|
37
|
+
'shift': handleShift,
|
|
38
|
+
'unshift': handleUnshift,
|
|
39
|
+
'splice': handleSplice,
|
|
40
|
+
};
|
|
41
|
+
/**
|
|
42
|
+
* 注册所有内置动作到 $methods
|
|
43
|
+
*
|
|
44
|
+
* 注意:此函数通过向现有 $methods 对象添加属性来注册方法,
|
|
45
|
+
* 而不是整体覆盖 $methods,以确保在 Proxy 保护下也能正常工作。
|
|
46
|
+
*/
|
|
47
|
+
export function registerBuiltinMethods(ctx) {
|
|
48
|
+
// 向现有 $methods 对象添加内置方法,不覆盖已有的同名方法
|
|
49
|
+
for (const [name, handler] of Object.entries(BUILTIN_METHODS)) {
|
|
50
|
+
if (!(name in ctx.$methods)) {
|
|
51
|
+
ctx.$methods[name] = handler;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/vm/handlers/index.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAGH,OAAO,EAAE,SAAS,EAAE,MAAM,UAAU,CAAA;AACpC,OAAO,EAAE,UAAU,EAAE,MAAM,WAAW,CAAA;AACtC,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAA;AAClC,OAAO,EAAE,UAAU,EAAE,MAAM,WAAW,CAAA;AACtC,OAAO,EAAE,UAAU,EAAE,MAAM,WAAW,CAAA;AACtC,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAA;AACxC,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAA;AAC9C,OAAO,EAAE,SAAS,EAAE,MAAM,UAAU,CAAA;AACpC,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAA;AAC5C,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAA;AAC1C,OAAO,EAAE,WAAW,EAAE,MAAM,kBAAkB,CAAA;AAC9C,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAA;AAClD,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAA;AAEhD;;GAEG;AACH,MAAM,eAAe,GAAkC;IACrD,OAAO;IACP,KAAK,EAAE,SAA0B;IACjC,MAAM,EAAE,UAA2B;IACnC,UAAU,EAAE,cAA+B;IAC3C,KAAK,EAAE,SAA0B;IAEjC,QAAQ;IACR,IAAI,EAAE,QAAyB;IAC/B,MAAM,EAAE,UAA2B;IAEnC,OAAO;IACP,MAAM,EAAE,UAA2B;IACnC,OAAO,EAAE,WAA4B;IAErC,SAAS;IACT,MAAM,EAAE,UAA2B;IACnC,KAAK,EAAE,SAA0B;IACjC,OAAO,EAAE,WAA4B;IACrC,SAAS,EAAE,aAA8B;IACzC,QAAQ,EAAE,YAA6B;CACxC,CAAA;AAED;;;;;GAKG;AACH,MAAM,UAAU,sBAAsB,CAAC,GAAmB;IACxD,mCAAmC;IACnC,KAAK,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,eAAe,CAAC,EAAE,CAAC;QAC9D,IAAI,CAAC,CAAC,IAAI,IAAI,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC5B,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,GAAG,OAAO,CAAA;QAC9B,CAAC;IACH,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* log 动作处理器
|
|
3
|
+
*
|
|
4
|
+
* 功能:调试输出
|
|
5
|
+
* 示例:{ "type": "log", "level": "info", "message": "User logged in" }
|
|
6
|
+
*/
|
|
7
|
+
import type { RuntimeContext, Action } from '../../types.js';
|
|
8
|
+
/**
|
|
9
|
+
* 处理 log 动作
|
|
10
|
+
*/
|
|
11
|
+
export declare function handleLog(ctx: RuntimeContext, action: Action): Promise<void>;
|
|
12
|
+
//# sourceMappingURL=log.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"log.d.ts","sourceRoot":"","sources":["../../../src/vm/handlers/log.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,EAAE,MAAM,YAAY,CAAA;AAIxD;;GAEG;AACH,wBAAsB,SAAS,CAC7B,GAAG,EAAE,cAAc,EACnB,MAAM,EAAE,MAAM,GACb,OAAO,CAAC,IAAI,CAAC,CAoCf"}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* log 动作处理器
|
|
3
|
+
*
|
|
4
|
+
* 功能:调试输出
|
|
5
|
+
* 示例:{ "type": "log", "level": "info", "message": "User logged in" }
|
|
6
|
+
*/
|
|
7
|
+
import { ActionError, ErrorCodes } from '../../errors.js';
|
|
8
|
+
import { evaluate } from '../../expression/evaluate.js';
|
|
9
|
+
/**
|
|
10
|
+
* 处理 log 动作
|
|
11
|
+
*/
|
|
12
|
+
export async function handleLog(ctx, action) {
|
|
13
|
+
const { level = 'info', message } = action;
|
|
14
|
+
if (!message) {
|
|
15
|
+
throw new ActionError(action, 'log action requires "message" parameter', ErrorCodes.ACTION_MISSING_PARAM, { metadata: { param: 'message' } });
|
|
16
|
+
}
|
|
17
|
+
// 求值 message(支持表达式)
|
|
18
|
+
let finalMessage = message;
|
|
19
|
+
if (typeof message === 'string' && message.startsWith('{{') && message.endsWith('}}')) {
|
|
20
|
+
const expr = message.slice(2, -2).trim();
|
|
21
|
+
finalMessage = evaluate(expr, ctx);
|
|
22
|
+
}
|
|
23
|
+
// 输出日志(将 finalMessage 转换为字符串)
|
|
24
|
+
const logLevel = String(level).toLowerCase();
|
|
25
|
+
const messageStr = String(finalMessage);
|
|
26
|
+
if (typeof console !== 'undefined') {
|
|
27
|
+
switch (logLevel) {
|
|
28
|
+
case 'error':
|
|
29
|
+
console.error('[Vario]', messageStr);
|
|
30
|
+
break;
|
|
31
|
+
case 'warn':
|
|
32
|
+
console.warn('[Vario]', messageStr);
|
|
33
|
+
break;
|
|
34
|
+
case 'info':
|
|
35
|
+
default:
|
|
36
|
+
console.log('[Vario]', messageStr);
|
|
37
|
+
break;
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
//# sourceMappingURL=log.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"log.js","sourceRoot":"","sources":["../../../src/vm/handlers/log.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,aAAa,CAAA;AACrD,OAAO,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAA;AAEnD;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,SAAS,CAC7B,GAAmB,EACnB,MAAc;IAEd,MAAM,EAAE,KAAK,GAAG,MAAM,EAAE,OAAO,EAAE,GAAG,MAAM,CAAA;IAE1C,IAAI,CAAC,OAAO,EAAE,CAAC;QACb,MAAM,IAAI,WAAW,CACnB,MAAM,EACN,yCAAyC,EACzC,UAAU,CAAC,oBAAoB,EAC/B,EAAE,QAAQ,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,EAAE,CACnC,CAAA;IACH,CAAC;IAED,oBAAoB;IACpB,IAAI,YAAY,GAAY,OAAO,CAAA;IACnC,IAAI,OAAO,OAAO,KAAK,QAAQ,IAAI,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QACtF,MAAM,IAAI,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAA;QACxC,YAAY,GAAG,QAAQ,CAAC,IAAI,EAAE,GAAG,CAAC,CAAA;IACpC,CAAC;IAED,8BAA8B;IAC9B,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,WAAW,EAAE,CAAA;IAC5C,MAAM,UAAU,GAAG,MAAM,CAAC,YAAY,CAAC,CAAA;IACvC,IAAI,OAAO,OAAO,KAAK,WAAW,EAAE,CAAC;QACnC,QAAQ,QAAQ,EAAE,CAAC;YACjB,KAAK,OAAO;gBACV,OAAO,CAAC,KAAK,CAAC,SAAS,EAAE,UAAU,CAAC,CAAA;gBACpC,MAAK;YACP,KAAK,MAAM;gBACT,OAAO,CAAC,IAAI,CAAC,SAAS,EAAE,UAAU,CAAC,CAAA;gBACnC,MAAK;YACP,KAAK,MAAM,CAAC;YACZ;gBACE,OAAO,CAAC,GAAG,CAAC,SAAS,EAAE,UAAU,CAAC,CAAA;gBAClC,MAAK;QACT,CAAC;IACH,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* loop 动作处理器
|
|
3
|
+
*
|
|
4
|
+
* 功能:循环执行
|
|
5
|
+
* 示例:{ "type": "loop", "var": "item", "in": "items", "body": [...] }
|
|
6
|
+
*/
|
|
7
|
+
import type { RuntimeContext, Action } from '../../types.js';
|
|
8
|
+
/**
|
|
9
|
+
* 处理 loop 动作
|
|
10
|
+
*/
|
|
11
|
+
export declare function handleLoop(ctx: RuntimeContext, action: Action): Promise<void>;
|
|
12
|
+
//# sourceMappingURL=loop.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"loop.d.ts","sourceRoot":"","sources":["../../../src/vm/handlers/loop.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,EAAE,MAAM,YAAY,CAAA;AAOxD;;GAEG;AACH,wBAAsB,UAAU,CAC9B,GAAG,EAAE,cAAc,EACnB,MAAM,EAAE,MAAM,GACb,OAAO,CAAC,IAAI,CAAC,CAiFf"}
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* loop 动作处理器
|
|
3
|
+
*
|
|
4
|
+
* 功能:循环执行
|
|
5
|
+
* 示例:{ "type": "loop", "var": "item", "in": "items", "body": [...] }
|
|
6
|
+
*/
|
|
7
|
+
import { ActionError, ErrorCodes } from '../../errors.js';
|
|
8
|
+
import { evaluate } from '../../expression/evaluate.js';
|
|
9
|
+
import { invalidateCache } from '../../expression/cache.js';
|
|
10
|
+
import { execute } from '../executor.js';
|
|
11
|
+
import { createLoopContext, releaseLoopContext } from '../../runtime/loop-context-pool.js';
|
|
12
|
+
/**
|
|
13
|
+
* 处理 loop 动作
|
|
14
|
+
*/
|
|
15
|
+
export async function handleLoop(ctx, action) {
|
|
16
|
+
const { var: varName, in: inExpr, body } = action;
|
|
17
|
+
if (!varName || typeof varName !== 'string') {
|
|
18
|
+
throw new ActionError(action, 'loop action requires "var" parameter', ErrorCodes.ACTION_MISSING_PARAM, { metadata: { param: 'var' } });
|
|
19
|
+
}
|
|
20
|
+
if (!inExpr || typeof inExpr !== 'string') {
|
|
21
|
+
throw new ActionError(action, 'loop action requires "in" parameter', ErrorCodes.ACTION_MISSING_PARAM, { metadata: { param: 'in' } });
|
|
22
|
+
}
|
|
23
|
+
if (!body || !Array.isArray(body)) {
|
|
24
|
+
throw new ActionError(action, 'loop action requires "body" parameter (array of actions)', ErrorCodes.ACTION_MISSING_PARAM, { metadata: { param: 'body' } });
|
|
25
|
+
}
|
|
26
|
+
// 求值 in 表达式,获取要遍历的数组或对象
|
|
27
|
+
const iterable = evaluate(inExpr, ctx);
|
|
28
|
+
if (iterable == null) {
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
// 遍历数组
|
|
32
|
+
if (Array.isArray(iterable)) {
|
|
33
|
+
for (let i = 0; i < iterable.length; i++) {
|
|
34
|
+
// 使用对象池创建循环上下文
|
|
35
|
+
const loopCtx = createLoopContext(ctx, iterable[i], i);
|
|
36
|
+
loopCtx[varName] = iterable[i];
|
|
37
|
+
try {
|
|
38
|
+
// 触发缓存失效,确保表达式重新求值
|
|
39
|
+
invalidateCache(varName, loopCtx);
|
|
40
|
+
await execute(body, loopCtx);
|
|
41
|
+
}
|
|
42
|
+
finally {
|
|
43
|
+
// 释放循环上下文回对象池
|
|
44
|
+
releaseLoopContext(loopCtx);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
// 遍历对象
|
|
49
|
+
else if (typeof iterable === 'object' && iterable !== null) {
|
|
50
|
+
const entries = Object.entries(iterable);
|
|
51
|
+
for (let i = 0; i < entries.length; i++) {
|
|
52
|
+
const [, value] = entries[i];
|
|
53
|
+
// 使用对象池创建循环上下文
|
|
54
|
+
const loopCtx = createLoopContext(ctx, value, i);
|
|
55
|
+
loopCtx[varName] = value;
|
|
56
|
+
try {
|
|
57
|
+
// 触发缓存失效,确保表达式重新求值
|
|
58
|
+
invalidateCache(varName, loopCtx);
|
|
59
|
+
await execute(body, loopCtx);
|
|
60
|
+
}
|
|
61
|
+
finally {
|
|
62
|
+
// 释放循环上下文回对象池
|
|
63
|
+
releaseLoopContext(loopCtx);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
else {
|
|
68
|
+
throw new ActionError(action, `loop "in" expression must evaluate to an array or object, got ${typeof iterable}`, ErrorCodes.ACTION_INVALID_PARAM, { metadata: { param: 'in', actualType: typeof iterable } });
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
//# sourceMappingURL=loop.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"loop.js","sourceRoot":"","sources":["../../../src/vm/handlers/loop.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* navigate 动作处理器
|
|
3
|
+
*
|
|
4
|
+
* 功能:路由导航
|
|
5
|
+
* 示例:{ "type": "navigate", "to": "/users/123" }
|
|
6
|
+
*/
|
|
7
|
+
import type { RuntimeContext, Action } from '../../types.js';
|
|
8
|
+
/**
|
|
9
|
+
* 处理 navigate 动作
|
|
10
|
+
*/
|
|
11
|
+
export declare function handleNavigate(ctx: RuntimeContext, action: Action): Promise<void>;
|
|
12
|
+
//# sourceMappingURL=navigate.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"navigate.d.ts","sourceRoot":"","sources":["../../../src/vm/handlers/navigate.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,EAAE,MAAM,YAAY,CAAA;AAIxD;;GAEG;AACH,wBAAsB,cAAc,CAClC,GAAG,EAAE,cAAc,EACnB,MAAM,EAAE,MAAM,GACb,OAAO,CAAC,IAAI,CAAC,CA8Cf"}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* navigate 动作处理器
|
|
3
|
+
*
|
|
4
|
+
* 功能:路由导航
|
|
5
|
+
* 示例:{ "type": "navigate", "to": "/users/123" }
|
|
6
|
+
*/
|
|
7
|
+
import { ActionError, ErrorCodes } from '../../errors.js';
|
|
8
|
+
import { evaluate } from '../../expression/evaluate.js';
|
|
9
|
+
/**
|
|
10
|
+
* 处理 navigate 动作
|
|
11
|
+
*/
|
|
12
|
+
export async function handleNavigate(ctx, action) {
|
|
13
|
+
const { to } = action;
|
|
14
|
+
if (!to || typeof to !== 'string') {
|
|
15
|
+
throw new ActionError(action, 'navigate action requires "to" parameter', ErrorCodes.ACTION_MISSING_PARAM, { metadata: { param: 'to' } });
|
|
16
|
+
}
|
|
17
|
+
// 求值 to(支持表达式)
|
|
18
|
+
let finalTo = to;
|
|
19
|
+
if (to.startsWith('{{') && to.endsWith('}}')) {
|
|
20
|
+
const expr = to.slice(2, -2).trim();
|
|
21
|
+
const result = evaluate(expr, ctx);
|
|
22
|
+
if (typeof result !== 'string') {
|
|
23
|
+
throw new ActionError(action, `navigate "to" expression must evaluate to a string, got ${typeof result}`, ErrorCodes.ACTION_INVALID_PARAM, { metadata: { param: 'to', actualType: typeof result } });
|
|
24
|
+
}
|
|
25
|
+
finalTo = result;
|
|
26
|
+
}
|
|
27
|
+
// 调用路由导航方法(如果已注册)
|
|
28
|
+
const navigateHandler = ctx.$methods['navigate'] || ctx.$methods['$navigate'];
|
|
29
|
+
if (navigateHandler) {
|
|
30
|
+
await navigateHandler(ctx, { to: finalTo });
|
|
31
|
+
}
|
|
32
|
+
else {
|
|
33
|
+
// 如果没有注册导航方法,使用默认行为(浏览器导航)
|
|
34
|
+
// 使用类型安全的 window 访问
|
|
35
|
+
if (typeof window !== 'undefined' && window.location) {
|
|
36
|
+
window.location.href = finalTo;
|
|
37
|
+
}
|
|
38
|
+
else {
|
|
39
|
+
throw new ActionError(action, 'navigate method not registered and window.location is not available', ErrorCodes.ACTION_EXECUTION_ERROR, { metadata: { reason: 'navigate_not_available' } });
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
//# sourceMappingURL=navigate.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"navigate.js","sourceRoot":"","sources":["../../../src/vm/handlers/navigate.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,aAAa,CAAA;AACrD,OAAO,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAA;AAEnD;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,cAAc,CAClC,GAAmB,EACnB,MAAc;IAEd,MAAM,EAAE,EAAE,EAAE,GAAG,MAAM,CAAA;IAErB,IAAI,CAAC,EAAE,IAAI,OAAO,EAAE,KAAK,QAAQ,EAAE,CAAC;QAClC,MAAM,IAAI,WAAW,CACnB,MAAM,EACN,yCAAyC,EACzC,UAAU,CAAC,oBAAoB,EAC/B,EAAE,QAAQ,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,CAC9B,CAAA;IACH,CAAC;IAED,eAAe;IACf,IAAI,OAAO,GAAW,EAAE,CAAA;IACxB,IAAI,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QAC7C,MAAM,IAAI,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAA;QACnC,MAAM,MAAM,GAAG,QAAQ,CAAC,IAAI,EAAE,GAAG,CAAC,CAAA;QAClC,IAAI,OAAO,MAAM,KAAK,QAAQ,EAAE,CAAC;YAC/B,MAAM,IAAI,WAAW,CACnB,MAAM,EACN,2DAA2D,OAAO,MAAM,EAAE,EAC1E,UAAU,CAAC,oBAAoB,EAC/B,EAAE,QAAQ,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,UAAU,EAAE,OAAO,MAAM,EAAE,EAAE,CACzD,CAAA;QACH,CAAC;QACD,OAAO,GAAG,MAAM,CAAA;IAClB,CAAC;IAED,kBAAkB;IAClB,MAAM,eAAe,GAAG,GAAG,CAAC,QAAQ,CAAC,UAAU,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAA;IAC7E,IAAI,eAAe,EAAE,CAAC;QACpB,MAAM,eAAe,CAAC,GAAG,EAAE,EAAE,EAAE,EAAE,OAAO,EAAE,CAAC,CAAA;IAC7C,CAAC;SAAM,CAAC;QACN,2BAA2B;QAC3B,oBAAoB;QACpB,IAAI,OAAO,MAAM,KAAK,WAAW,IAAK,MAA0C,CAAC,QAAQ,EAAE,CAAC;YACxF,MAAqD,CAAC,QAAQ,CAAC,IAAI,GAAG,OAAO,CAAA;QACjF,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,WAAW,CACnB,MAAM,EACN,qEAAqE,EACrE,UAAU,CAAC,sBAAsB,EACjC,EAAE,QAAQ,EAAE,EAAE,MAAM,EAAE,wBAAwB,EAAE,EAAE,CACnD,CAAA;QACH,CAAC;IACH,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* set 动作处理器
|
|
3
|
+
*
|
|
4
|
+
* 功能:修改状态
|
|
5
|
+
* 示例:{ "type": "set", "path": "user.name", "value": "张三" }
|
|
6
|
+
*
|
|
7
|
+
* 注意:缓存失效通过 RuntimeContext 的 onStateChange 钩子处理
|
|
8
|
+
* 框架集成层应在创建上下文时注册该钩子
|
|
9
|
+
*/
|
|
10
|
+
import type { RuntimeContext, Action } from '../../types.js';
|
|
11
|
+
/**
|
|
12
|
+
* 处理 set 动作
|
|
13
|
+
*/
|
|
14
|
+
export declare function handleSet(ctx: RuntimeContext, action: Action): Promise<void>;
|
|
15
|
+
//# sourceMappingURL=set.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"set.d.ts","sourceRoot":"","sources":["../../../src/vm/handlers/set.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,EAAE,MAAM,YAAY,CAAA;AAIxD;;GAEG;AACH,wBAAsB,SAAS,CAC7B,GAAG,EAAE,cAAc,EACnB,MAAM,EAAE,MAAM,GACb,OAAO,CAAC,IAAI,CAAC,CAsBf"}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* set 动作处理器
|
|
3
|
+
*
|
|
4
|
+
* 功能:修改状态
|
|
5
|
+
* 示例:{ "type": "set", "path": "user.name", "value": "张三" }
|
|
6
|
+
*
|
|
7
|
+
* 注意:缓存失效通过 RuntimeContext 的 onStateChange 钩子处理
|
|
8
|
+
* 框架集成层应在创建上下文时注册该钩子
|
|
9
|
+
*/
|
|
10
|
+
import { ActionError, ErrorCodes } from '../../errors.js';
|
|
11
|
+
import { evaluate } from '../../expression/evaluate.js';
|
|
12
|
+
/**
|
|
13
|
+
* 处理 set 动作
|
|
14
|
+
*/
|
|
15
|
+
export async function handleSet(ctx, action) {
|
|
16
|
+
const { path, value } = action;
|
|
17
|
+
if (!path || typeof path !== 'string') {
|
|
18
|
+
throw new ActionError(action, 'set action requires "path" parameter', ErrorCodes.ACTION_MISSING_PARAM, { metadata: { param: 'path' } });
|
|
19
|
+
}
|
|
20
|
+
// 求值 value(支持表达式)
|
|
21
|
+
let finalValue = value;
|
|
22
|
+
if (typeof value === 'string' && value.startsWith('{{') && value.endsWith('}}')) {
|
|
23
|
+
// 表达式插值:{{ user.age + 1 }}
|
|
24
|
+
const expr = value.slice(2, -2).trim();
|
|
25
|
+
finalValue = evaluate(expr, ctx);
|
|
26
|
+
}
|
|
27
|
+
// 设置状态(缓存失效通过 onStateChange 钩子自动处理)
|
|
28
|
+
ctx._set(path, finalValue);
|
|
29
|
+
}
|
|
30
|
+
//# sourceMappingURL=set.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"set.js","sourceRoot":"","sources":["../../../src/vm/handlers/set.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAGH,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,aAAa,CAAA;AACrD,OAAO,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAA;AAEnD;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,SAAS,CAC7B,GAAmB,EACnB,MAAc;IAEd,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,MAAM,CAAA;IAE9B,IAAI,CAAC,IAAI,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;QACtC,MAAM,IAAI,WAAW,CACnB,MAAM,EACN,sCAAsC,EACtC,UAAU,CAAC,oBAAoB,EAC/B,EAAE,QAAQ,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,EAAE,CAChC,CAAA;IACH,CAAC;IAED,kBAAkB;IAClB,IAAI,UAAU,GAAG,KAAK,CAAA;IACtB,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QAChF,2BAA2B;QAC3B,MAAM,IAAI,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAA;QACtC,UAAU,GAAG,QAAQ,CAAC,IAAI,EAAE,GAAG,CAAC,CAAA;IAClC,CAAC;IAED,oCAAoC;IACpC,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,UAAU,CAAC,CAAA;AAC5B,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/vm/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,eAAe,CAAA;AACvC,YAAY,EAAE,cAAc,EAAE,MAAM,eAAe,CAAA;AACnD,OAAO,EAAE,sBAAsB,EAAE,MAAM,qBAAqB,CAAA;AAC5D,cAAc,aAAa,CAAA"}
|
package/dist/vm/index.js
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/vm/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,eAAe,CAAA;AAEvC,OAAO,EAAE,sBAAsB,EAAE,MAAM,qBAAqB,CAAA;AAC5D,cAAc,aAAa,CAAA"}
|
package/package.json
ADDED
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@variojs/core",
|
|
3
|
+
"version": "0.0.1",
|
|
4
|
+
"description": "Vario Core Runtime - Instruction VM, Expression System, Runtime Context",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"main": "./dist/index.js",
|
|
7
|
+
"module": "./dist/index.js",
|
|
8
|
+
"types": "./dist/index.d.ts",
|
|
9
|
+
"exports": {
|
|
10
|
+
".": {
|
|
11
|
+
"import": "./dist/index.js",
|
|
12
|
+
"types": "./dist/index.d.ts"
|
|
13
|
+
}
|
|
14
|
+
},
|
|
15
|
+
"scripts": {
|
|
16
|
+
"build": "tsc",
|
|
17
|
+
"postbuild": "tsc-alias",
|
|
18
|
+
"dev": "tsc --watch",
|
|
19
|
+
"test": "vitest run",
|
|
20
|
+
"test:watch": "vitest",
|
|
21
|
+
"clean": "rm -rf dist"
|
|
22
|
+
},
|
|
23
|
+
"dependencies": {
|
|
24
|
+
"@babel/parser": "^7.23.6",
|
|
25
|
+
"@babel/types": "^7.23.6"
|
|
26
|
+
},
|
|
27
|
+
"devDependencies": {
|
|
28
|
+
"typescript": "^5.3.3",
|
|
29
|
+
"vitest": "^1.2.0"
|
|
30
|
+
},
|
|
31
|
+
"files": [
|
|
32
|
+
"dist",
|
|
33
|
+
"README.md"
|
|
34
|
+
],
|
|
35
|
+
"publishConfig": {
|
|
36
|
+
"access": "public"
|
|
37
|
+
}
|
|
38
|
+
}
|