@holoscript/core 2.0.1 → 2.1.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/dist/chunk-2XXE34KS.js +344 -0
- package/dist/chunk-2XXE34KS.js.map +1 -0
- package/dist/chunk-3X2EGU7Z.cjs +52 -0
- package/dist/chunk-3X2EGU7Z.cjs.map +1 -0
- package/dist/chunk-AFFVFO4D.js +1689 -0
- package/dist/chunk-AFFVFO4D.js.map +1 -0
- package/dist/chunk-DGUM43GV.js +10 -0
- package/dist/chunk-DGUM43GV.js.map +1 -0
- package/{src/HoloScriptDebugger.ts → dist/chunk-DOY73HDH.js} +118 -257
- package/dist/chunk-DOY73HDH.js.map +1 -0
- package/dist/chunk-JEQ2X3Z6.cjs +12 -0
- package/dist/chunk-JEQ2X3Z6.cjs.map +1 -0
- package/dist/chunk-L6VLNVKP.cjs +1691 -0
- package/dist/chunk-L6VLNVKP.cjs.map +1 -0
- package/dist/chunk-MFNO57XL.cjs +347 -0
- package/dist/chunk-MFNO57XL.cjs.map +1 -0
- package/dist/chunk-R75MREOS.cjs +424 -0
- package/dist/chunk-R75MREOS.cjs.map +1 -0
- package/dist/chunk-SATNCODL.js +45 -0
- package/dist/chunk-SATNCODL.js.map +1 -0
- package/dist/chunk-T57ZL7KR.cjs +1281 -0
- package/dist/chunk-T57ZL7KR.cjs.map +1 -0
- package/dist/chunk-U72GEJZT.js +1279 -0
- package/dist/chunk-U72GEJZT.js.map +1 -0
- package/dist/debugger.cjs +20 -0
- package/dist/debugger.cjs.map +1 -0
- package/dist/debugger.d.cts +171 -0
- package/dist/debugger.d.ts +171 -0
- package/dist/debugger.js +7 -0
- package/dist/debugger.js.map +1 -0
- package/dist/index.cjs +6803 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +4093 -0
- package/dist/index.d.ts +4093 -0
- package/dist/index.js +6715 -0
- package/dist/index.js.map +1 -0
- package/dist/parser.cjs +14 -0
- package/dist/parser.cjs.map +1 -0
- package/dist/parser.d.cts +172 -0
- package/dist/parser.d.ts +172 -0
- package/dist/parser.js +5 -0
- package/dist/parser.js.map +1 -0
- package/dist/runtime.cjs +14 -0
- package/dist/runtime.cjs.map +1 -0
- package/dist/runtime.d.cts +200 -0
- package/dist/runtime.d.ts +200 -0
- package/dist/runtime.js +5 -0
- package/dist/runtime.js.map +1 -0
- package/dist/type-checker.cjs +17 -0
- package/dist/type-checker.cjs.map +1 -0
- package/dist/type-checker.d.cts +105 -0
- package/dist/type-checker.d.ts +105 -0
- package/dist/type-checker.js +4 -0
- package/dist/type-checker.js.map +1 -0
- package/dist/types-4h8cbtF_.d.cts +329 -0
- package/dist/types-4h8cbtF_.d.ts +329 -0
- package/package.json +17 -13
- package/src/HoloScript2DParser.js +0 -227
- package/src/HoloScript2DParser.ts +0 -261
- package/src/HoloScriptCodeParser.js +0 -1102
- package/src/HoloScriptCodeParser.ts +0 -1188
- package/src/HoloScriptDebugger.js +0 -458
- package/src/HoloScriptParser.js +0 -338
- package/src/HoloScriptParser.ts +0 -397
- package/src/HoloScriptPlusParser.js +0 -371
- package/src/HoloScriptPlusParser.ts +0 -543
- package/src/HoloScriptRuntime.js +0 -1399
- package/src/HoloScriptRuntime.test.js +0 -351
- package/src/HoloScriptRuntime.test.ts +0 -436
- package/src/HoloScriptRuntime.ts +0 -1653
- package/src/HoloScriptTypeChecker.js +0 -356
- package/src/HoloScriptTypeChecker.ts +0 -475
- package/src/__tests__/GraphicsServices.test.js +0 -357
- package/src/__tests__/GraphicsServices.test.ts +0 -427
- package/src/__tests__/HoloScriptPlusParser.test.js +0 -317
- package/src/__tests__/HoloScriptPlusParser.test.ts +0 -392
- package/src/__tests__/integration.test.js +0 -336
- package/src/__tests__/integration.test.ts +0 -416
- package/src/__tests__/performance.bench.js +0 -218
- package/src/__tests__/performance.bench.ts +0 -262
- package/src/__tests__/type-checker.test.js +0 -60
- package/src/__tests__/type-checker.test.ts +0 -73
- package/src/index.js +0 -217
- package/src/index.ts +0 -426
- package/src/interop/Interoperability.js +0 -413
- package/src/interop/Interoperability.ts +0 -494
- package/src/logger.js +0 -42
- package/src/logger.ts +0 -57
- package/src/parser/EnhancedParser.js +0 -205
- package/src/parser/EnhancedParser.ts +0 -251
- package/src/parser/HoloScriptPlusParser.js +0 -928
- package/src/parser/HoloScriptPlusParser.ts +0 -1089
- package/src/runtime/HoloScriptPlusRuntime.js +0 -674
- package/src/runtime/HoloScriptPlusRuntime.ts +0 -861
- package/src/runtime/PerformanceTelemetry.js +0 -323
- package/src/runtime/PerformanceTelemetry.ts +0 -467
- package/src/runtime/RuntimeOptimization.js +0 -361
- package/src/runtime/RuntimeOptimization.ts +0 -416
- package/src/services/HololandGraphicsPipelineService.js +0 -506
- package/src/services/HololandGraphicsPipelineService.ts +0 -662
- package/src/services/PlatformPerformanceOptimizer.js +0 -356
- package/src/services/PlatformPerformanceOptimizer.ts +0 -503
- package/src/state/ReactiveState.js +0 -427
- package/src/state/ReactiveState.ts +0 -572
- package/src/tools/DeveloperExperience.js +0 -376
- package/src/tools/DeveloperExperience.ts +0 -438
- package/src/traits/AIDriverTrait.js +0 -322
- package/src/traits/AIDriverTrait.test.js +0 -329
- package/src/traits/AIDriverTrait.test.ts +0 -357
- package/src/traits/AIDriverTrait.ts +0 -474
- package/src/traits/LightingTrait.js +0 -313
- package/src/traits/LightingTrait.test.js +0 -410
- package/src/traits/LightingTrait.test.ts +0 -462
- package/src/traits/LightingTrait.ts +0 -505
- package/src/traits/MaterialTrait.js +0 -194
- package/src/traits/MaterialTrait.test.js +0 -286
- package/src/traits/MaterialTrait.test.ts +0 -329
- package/src/traits/MaterialTrait.ts +0 -324
- package/src/traits/RenderingTrait.js +0 -356
- package/src/traits/RenderingTrait.test.js +0 -363
- package/src/traits/RenderingTrait.test.ts +0 -427
- package/src/traits/RenderingTrait.ts +0 -555
- package/src/traits/VRTraitSystem.js +0 -740
- package/src/traits/VRTraitSystem.ts +0 -1040
- package/src/traits/VoiceInputTrait.js +0 -284
- package/src/traits/VoiceInputTrait.test.js +0 -226
- package/src/traits/VoiceInputTrait.test.ts +0 -252
- package/src/traits/VoiceInputTrait.ts +0 -401
- package/src/types/AdvancedTypeSystem.js +0 -226
- package/src/types/AdvancedTypeSystem.ts +0 -494
- package/src/types/HoloScriptPlus.d.ts +0 -853
- package/src/types.js +0 -6
- package/src/types.ts +0 -369
- package/tsconfig.json +0 -23
- package/tsup.config.d.ts +0 -2
- package/tsup.config.js +0 -18
- package/tsup.config.ts +0 -19
|
@@ -1,436 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* HoloScript Runtime Tests
|
|
3
|
-
*
|
|
4
|
-
* Comprehensive test suite for the HoloScript runtime engine.
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { describe, it, expect, beforeEach } from 'vitest';
|
|
8
|
-
import { HoloScriptRuntime } from './HoloScriptRuntime';
|
|
9
|
-
import { HoloScriptCodeParser } from './HoloScriptCodeParser';
|
|
10
|
-
import type { OrbNode, MethodNode, ConnectionNode, GateNode, StreamNode } from './types';
|
|
11
|
-
|
|
12
|
-
describe('HoloScriptRuntime', () => {
|
|
13
|
-
let runtime: HoloScriptRuntime;
|
|
14
|
-
let parser: HoloScriptCodeParser;
|
|
15
|
-
|
|
16
|
-
beforeEach(() => {
|
|
17
|
-
runtime = new HoloScriptRuntime();
|
|
18
|
-
parser = new HoloScriptCodeParser();
|
|
19
|
-
});
|
|
20
|
-
|
|
21
|
-
describe('Basic Execution', () => {
|
|
22
|
-
it('should create an orb', async () => {
|
|
23
|
-
const orbNode: OrbNode = {
|
|
24
|
-
type: 'orb',
|
|
25
|
-
name: 'testOrb',
|
|
26
|
-
properties: { message: 'Hello World' },
|
|
27
|
-
methods: [],
|
|
28
|
-
position: { x: 1, y: 2, z: 3 },
|
|
29
|
-
hologram: { shape: 'orb', color: '#00ffff', size: 1, glow: true, interactive: true },
|
|
30
|
-
};
|
|
31
|
-
|
|
32
|
-
const result = await runtime.executeNode(orbNode);
|
|
33
|
-
|
|
34
|
-
expect(result.success).toBe(true);
|
|
35
|
-
expect(result.output).toBeDefined();
|
|
36
|
-
expect(result.spatialPosition).toEqual({ x: 1, y: 2, z: 3 });
|
|
37
|
-
});
|
|
38
|
-
|
|
39
|
-
it('should define a function', async () => {
|
|
40
|
-
const funcNode: MethodNode = {
|
|
41
|
-
type: 'method',
|
|
42
|
-
name: 'testFunc',
|
|
43
|
-
parameters: [{ type: 'parameter', name: 'x', dataType: 'number' }],
|
|
44
|
-
body: [],
|
|
45
|
-
position: { x: 0, y: 0, z: 0 },
|
|
46
|
-
};
|
|
47
|
-
|
|
48
|
-
const result = await runtime.executeNode(funcNode);
|
|
49
|
-
|
|
50
|
-
expect(result.success).toBe(true);
|
|
51
|
-
expect(result.output).toContain('testFunc');
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
it('should create a connection', async () => {
|
|
55
|
-
// First create two orbs
|
|
56
|
-
const orb1: OrbNode = { type: 'orb', name: 'source', properties: {}, methods: [], position: { x: 0, y: 0, z: 0 } };
|
|
57
|
-
const orb2: OrbNode = { type: 'orb', name: 'target', properties: {}, methods: [], position: { x: 5, y: 0, z: 0 } };
|
|
58
|
-
|
|
59
|
-
await runtime.executeNode(orb1);
|
|
60
|
-
await runtime.executeNode(orb2);
|
|
61
|
-
|
|
62
|
-
const connNode: ConnectionNode = {
|
|
63
|
-
type: 'connection',
|
|
64
|
-
from: 'source',
|
|
65
|
-
to: 'target',
|
|
66
|
-
dataType: 'number',
|
|
67
|
-
bidirectional: false,
|
|
68
|
-
};
|
|
69
|
-
|
|
70
|
-
const result = await runtime.executeNode(connNode);
|
|
71
|
-
|
|
72
|
-
expect(result.success).toBe(true);
|
|
73
|
-
expect(result.output).toContain('source');
|
|
74
|
-
expect(result.output).toContain('target');
|
|
75
|
-
});
|
|
76
|
-
});
|
|
77
|
-
|
|
78
|
-
describe('Expression Evaluation', () => {
|
|
79
|
-
it('should evaluate number literals', () => {
|
|
80
|
-
expect(runtime.evaluateExpression('42')).toBe(42);
|
|
81
|
-
expect(runtime.evaluateExpression('-3.14')).toBe(-3.14);
|
|
82
|
-
expect(runtime.evaluateExpression('0')).toBe(0);
|
|
83
|
-
});
|
|
84
|
-
|
|
85
|
-
it('should evaluate string literals', () => {
|
|
86
|
-
expect(runtime.evaluateExpression('"hello"')).toBe('hello');
|
|
87
|
-
expect(runtime.evaluateExpression("'world'")).toBe('world');
|
|
88
|
-
});
|
|
89
|
-
|
|
90
|
-
it('should evaluate boolean literals', () => {
|
|
91
|
-
expect(runtime.evaluateExpression('true')).toBe(true);
|
|
92
|
-
expect(runtime.evaluateExpression('false')).toBe(false);
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
it('should evaluate array literals', () => {
|
|
96
|
-
expect(runtime.evaluateExpression('[]')).toEqual([]);
|
|
97
|
-
expect(runtime.evaluateExpression('[1, 2, 3]')).toEqual([1, 2, 3]);
|
|
98
|
-
expect(runtime.evaluateExpression('["a", "b"]')).toEqual(['a', 'b']);
|
|
99
|
-
});
|
|
100
|
-
|
|
101
|
-
it('should evaluate object literals', () => {
|
|
102
|
-
expect(runtime.evaluateExpression('{}')).toEqual({});
|
|
103
|
-
expect(runtime.evaluateExpression('{x: 1, y: 2}')).toEqual({ x: 1, y: 2 });
|
|
104
|
-
});
|
|
105
|
-
|
|
106
|
-
it('should evaluate arithmetic operations', () => {
|
|
107
|
-
runtime.setVariable('a', 10);
|
|
108
|
-
runtime.setVariable('b', 3);
|
|
109
|
-
|
|
110
|
-
expect(runtime.evaluateExpression('a + b')).toBe(13);
|
|
111
|
-
expect(runtime.evaluateExpression('a - b')).toBe(7);
|
|
112
|
-
expect(runtime.evaluateExpression('a * b')).toBe(30);
|
|
113
|
-
expect(runtime.evaluateExpression('a / b')).toBeCloseTo(3.33, 1);
|
|
114
|
-
});
|
|
115
|
-
});
|
|
116
|
-
|
|
117
|
-
describe('Built-in Functions', () => {
|
|
118
|
-
it('should execute math functions', async () => {
|
|
119
|
-
const result1 = await runtime.callFunction('add', [5, 3]);
|
|
120
|
-
expect(result1.success).toBe(true);
|
|
121
|
-
expect(result1.output).toBe(8);
|
|
122
|
-
|
|
123
|
-
const result2 = await runtime.callFunction('multiply', [4, 7]);
|
|
124
|
-
expect(result2.success).toBe(true);
|
|
125
|
-
expect(result2.output).toBe(28);
|
|
126
|
-
|
|
127
|
-
const result3 = await runtime.callFunction('abs', [-42]);
|
|
128
|
-
expect(result3.success).toBe(true);
|
|
129
|
-
expect(result3.output).toBe(42);
|
|
130
|
-
});
|
|
131
|
-
|
|
132
|
-
it('should execute string functions', async () => {
|
|
133
|
-
const result1 = await runtime.callFunction('concat', ['Hello', ' ', 'World']);
|
|
134
|
-
expect(result1.success).toBe(true);
|
|
135
|
-
expect(result1.output).toBe('Hello World');
|
|
136
|
-
|
|
137
|
-
const result2 = await runtime.callFunction('uppercase', ['test']);
|
|
138
|
-
expect(result2.success).toBe(true);
|
|
139
|
-
expect(result2.output).toBe('TEST');
|
|
140
|
-
|
|
141
|
-
const result3 = await runtime.callFunction('length', ['hello']);
|
|
142
|
-
expect(result3.success).toBe(true);
|
|
143
|
-
expect(result3.output).toBe(5);
|
|
144
|
-
});
|
|
145
|
-
|
|
146
|
-
it('should execute array functions', async () => {
|
|
147
|
-
const arr = [1, 2, 3];
|
|
148
|
-
|
|
149
|
-
const result1 = await runtime.callFunction('push', [arr, 4]);
|
|
150
|
-
expect(result1.success).toBe(true);
|
|
151
|
-
expect(result1.output).toEqual([1, 2, 3, 4]);
|
|
152
|
-
|
|
153
|
-
const result2 = await runtime.callFunction('length', [[1, 2, 3, 4, 5]]);
|
|
154
|
-
expect(result2.success).toBe(true);
|
|
155
|
-
expect(result2.output).toBe(5);
|
|
156
|
-
});
|
|
157
|
-
|
|
158
|
-
it('should execute type checking functions', async () => {
|
|
159
|
-
expect((await runtime.callFunction('isNumber', [42])).output).toBe(true);
|
|
160
|
-
expect((await runtime.callFunction('isNumber', ['hello'])).output).toBe(false);
|
|
161
|
-
expect((await runtime.callFunction('isString', ['hello'])).output).toBe(true);
|
|
162
|
-
expect((await runtime.callFunction('isArray', [[1, 2, 3]])).output).toBe(true);
|
|
163
|
-
});
|
|
164
|
-
});
|
|
165
|
-
|
|
166
|
-
describe('Variable Scoping', () => {
|
|
167
|
-
it('should set and get variables', () => {
|
|
168
|
-
runtime.setVariable('x', 100);
|
|
169
|
-
expect(runtime.getVariable('x')).toBe(100);
|
|
170
|
-
|
|
171
|
-
runtime.setVariable('y', 'hello');
|
|
172
|
-
expect(runtime.getVariable('y')).toBe('hello');
|
|
173
|
-
});
|
|
174
|
-
|
|
175
|
-
it('should handle nested property access', () => {
|
|
176
|
-
runtime.setVariable('obj', { nested: { value: 42 } });
|
|
177
|
-
expect(runtime.getVariable('obj.nested.value')).toBe(42);
|
|
178
|
-
});
|
|
179
|
-
|
|
180
|
-
it('should set nested properties', () => {
|
|
181
|
-
runtime.setVariable('data.x', 10);
|
|
182
|
-
runtime.setVariable('data.y', 20);
|
|
183
|
-
expect(runtime.getVariable('data')).toEqual({ x: 10, y: 20 });
|
|
184
|
-
});
|
|
185
|
-
});
|
|
186
|
-
|
|
187
|
-
describe('Gate Execution', () => {
|
|
188
|
-
it('should execute true path when condition is true', async () => {
|
|
189
|
-
runtime.setVariable('value', 10);
|
|
190
|
-
|
|
191
|
-
const gateNode: GateNode = {
|
|
192
|
-
type: 'gate',
|
|
193
|
-
condition: 'value > 5',
|
|
194
|
-
truePath: [],
|
|
195
|
-
falsePath: [],
|
|
196
|
-
position: { x: 0, y: 0, z: 0 },
|
|
197
|
-
};
|
|
198
|
-
|
|
199
|
-
const result = await runtime.executeNode(gateNode);
|
|
200
|
-
|
|
201
|
-
expect(result.success).toBe(true);
|
|
202
|
-
expect(result.output).toContain('true');
|
|
203
|
-
});
|
|
204
|
-
|
|
205
|
-
it('should execute false path when condition is false', async () => {
|
|
206
|
-
runtime.setVariable('value', 3);
|
|
207
|
-
|
|
208
|
-
const gateNode: GateNode = {
|
|
209
|
-
type: 'gate',
|
|
210
|
-
condition: 'value > 5',
|
|
211
|
-
truePath: [],
|
|
212
|
-
falsePath: [],
|
|
213
|
-
position: { x: 0, y: 0, z: 0 },
|
|
214
|
-
};
|
|
215
|
-
|
|
216
|
-
const result = await runtime.executeNode(gateNode);
|
|
217
|
-
|
|
218
|
-
expect(result.success).toBe(true);
|
|
219
|
-
expect(result.output).toContain('false');
|
|
220
|
-
});
|
|
221
|
-
|
|
222
|
-
it('should handle boolean literals in conditions', async () => {
|
|
223
|
-
const trueGate: GateNode = {
|
|
224
|
-
type: 'gate',
|
|
225
|
-
condition: 'true',
|
|
226
|
-
truePath: [],
|
|
227
|
-
falsePath: [],
|
|
228
|
-
};
|
|
229
|
-
|
|
230
|
-
const falseGate: GateNode = {
|
|
231
|
-
type: 'gate',
|
|
232
|
-
condition: 'false',
|
|
233
|
-
truePath: [],
|
|
234
|
-
falsePath: [],
|
|
235
|
-
};
|
|
236
|
-
|
|
237
|
-
const result1 = await runtime.executeNode(trueGate);
|
|
238
|
-
expect(result1.output).toContain('true');
|
|
239
|
-
|
|
240
|
-
const result2 = await runtime.executeNode(falseGate);
|
|
241
|
-
expect(result2.output).toContain('false');
|
|
242
|
-
});
|
|
243
|
-
|
|
244
|
-
it('should handle comparison operators', async () => {
|
|
245
|
-
runtime.setVariable('a', 5);
|
|
246
|
-
runtime.setVariable('b', 5);
|
|
247
|
-
|
|
248
|
-
const tests = [
|
|
249
|
-
{ condition: 'a == b', expected: true },
|
|
250
|
-
{ condition: 'a != b', expected: false },
|
|
251
|
-
{ condition: 'a >= b', expected: true },
|
|
252
|
-
{ condition: 'a <= b', expected: true },
|
|
253
|
-
{ condition: 'a > b', expected: false },
|
|
254
|
-
{ condition: 'a < b', expected: false },
|
|
255
|
-
];
|
|
256
|
-
|
|
257
|
-
for (const { condition, expected } of tests) {
|
|
258
|
-
const gate: GateNode = { type: 'gate', condition, truePath: [], falsePath: [] };
|
|
259
|
-
const result = await runtime.executeNode(gate);
|
|
260
|
-
expect(result.output).toContain(expected ? 'true' : 'false');
|
|
261
|
-
}
|
|
262
|
-
});
|
|
263
|
-
});
|
|
264
|
-
|
|
265
|
-
describe('Stream Execution', () => {
|
|
266
|
-
it('should process stream with transformations', async () => {
|
|
267
|
-
runtime.setVariable('numbers', [1, 2, 3, 4, 5]);
|
|
268
|
-
|
|
269
|
-
const streamNode: StreamNode = {
|
|
270
|
-
type: 'stream',
|
|
271
|
-
name: 'testStream',
|
|
272
|
-
source: 'numbers',
|
|
273
|
-
transformations: [
|
|
274
|
-
{ type: 'transformation', operation: 'sum', parameters: {} },
|
|
275
|
-
],
|
|
276
|
-
position: { x: 0, y: 0, z: 0 },
|
|
277
|
-
};
|
|
278
|
-
|
|
279
|
-
const result = await runtime.executeNode(streamNode);
|
|
280
|
-
|
|
281
|
-
expect(result.success).toBe(true);
|
|
282
|
-
expect(runtime.getVariable('testStream_result')).toBe(15);
|
|
283
|
-
});
|
|
284
|
-
|
|
285
|
-
it('should handle multiple transformations', async () => {
|
|
286
|
-
runtime.setVariable('data', [5, 3, 8, 1, 9]);
|
|
287
|
-
|
|
288
|
-
const streamNode: StreamNode = {
|
|
289
|
-
type: 'stream',
|
|
290
|
-
name: 'multiStream',
|
|
291
|
-
source: 'data',
|
|
292
|
-
transformations: [
|
|
293
|
-
{ type: 'transformation', operation: 'sort', parameters: {} },
|
|
294
|
-
{ type: 'transformation', operation: 'take', parameters: { count: 3 } },
|
|
295
|
-
],
|
|
296
|
-
position: { x: 0, y: 0, z: 0 },
|
|
297
|
-
};
|
|
298
|
-
|
|
299
|
-
const result = await runtime.executeNode(streamNode);
|
|
300
|
-
|
|
301
|
-
expect(result.success).toBe(true);
|
|
302
|
-
expect(runtime.getVariable('multiStream_result')).toEqual([1, 3, 5]);
|
|
303
|
-
});
|
|
304
|
-
});
|
|
305
|
-
|
|
306
|
-
describe('Event System', () => {
|
|
307
|
-
it('should register and emit events', async () => {
|
|
308
|
-
let eventFired = false;
|
|
309
|
-
let eventData: unknown = null;
|
|
310
|
-
|
|
311
|
-
runtime.on('test.event', (data) => {
|
|
312
|
-
eventFired = true;
|
|
313
|
-
eventData = data;
|
|
314
|
-
});
|
|
315
|
-
|
|
316
|
-
await runtime.emit('test.event', { value: 42 });
|
|
317
|
-
|
|
318
|
-
expect(eventFired).toBe(true);
|
|
319
|
-
expect(eventData).toEqual({ value: 42 });
|
|
320
|
-
});
|
|
321
|
-
|
|
322
|
-
it('should remove event handlers', async () => {
|
|
323
|
-
let count = 0;
|
|
324
|
-
const handler = () => { count++; };
|
|
325
|
-
|
|
326
|
-
runtime.on('counter', handler);
|
|
327
|
-
await runtime.emit('counter');
|
|
328
|
-
expect(count).toBe(1);
|
|
329
|
-
|
|
330
|
-
runtime.off('counter', handler);
|
|
331
|
-
await runtime.emit('counter');
|
|
332
|
-
expect(count).toBe(1); // Should not increment
|
|
333
|
-
});
|
|
334
|
-
});
|
|
335
|
-
|
|
336
|
-
describe('Code Parser Integration', () => {
|
|
337
|
-
it('should parse and execute orb declaration', async () => {
|
|
338
|
-
const code = `
|
|
339
|
-
orb greeting {
|
|
340
|
-
message: "Hello"
|
|
341
|
-
color: "#ff0000"
|
|
342
|
-
}
|
|
343
|
-
`;
|
|
344
|
-
|
|
345
|
-
const parseResult = parser.parse(code);
|
|
346
|
-
expect(parseResult.success).toBe(true);
|
|
347
|
-
expect(parseResult.ast.length).toBe(1);
|
|
348
|
-
|
|
349
|
-
const results = await runtime.executeProgram(parseResult.ast);
|
|
350
|
-
expect(results[0].success).toBe(true);
|
|
351
|
-
});
|
|
352
|
-
|
|
353
|
-
it('should parse and execute function declaration', async () => {
|
|
354
|
-
const code = `
|
|
355
|
-
function greet(name: string): string {
|
|
356
|
-
return name
|
|
357
|
-
}
|
|
358
|
-
`;
|
|
359
|
-
|
|
360
|
-
const parseResult = parser.parse(code);
|
|
361
|
-
expect(parseResult.success).toBe(true);
|
|
362
|
-
|
|
363
|
-
const results = await runtime.executeProgram(parseResult.ast);
|
|
364
|
-
expect(results[0].success).toBe(true);
|
|
365
|
-
|
|
366
|
-
const context = runtime.getContext();
|
|
367
|
-
expect(context.functions.has('greet')).toBe(true);
|
|
368
|
-
});
|
|
369
|
-
|
|
370
|
-
it('should parse and execute connection', async () => {
|
|
371
|
-
const code = `
|
|
372
|
-
orb source { value: 10 }
|
|
373
|
-
orb target { value: 0 }
|
|
374
|
-
connect source to target as "number"
|
|
375
|
-
`;
|
|
376
|
-
|
|
377
|
-
const parseResult = parser.parse(code);
|
|
378
|
-
expect(parseResult.success).toBe(true);
|
|
379
|
-
expect(parseResult.ast.length).toBe(3);
|
|
380
|
-
|
|
381
|
-
const results = await runtime.executeProgram(parseResult.ast);
|
|
382
|
-
expect(results.every(r => r.success)).toBe(true);
|
|
383
|
-
|
|
384
|
-
const context = runtime.getContext();
|
|
385
|
-
expect(context.connections.length).toBe(1);
|
|
386
|
-
});
|
|
387
|
-
});
|
|
388
|
-
|
|
389
|
-
describe('Security', () => {
|
|
390
|
-
it('should block suspicious keywords in expressions', () => {
|
|
391
|
-
expect(runtime.evaluateExpression('eval("1+1")')).toBeUndefined();
|
|
392
|
-
expect(runtime.evaluateExpression('process.exit()')).toBeUndefined();
|
|
393
|
-
expect(runtime.evaluateExpression('require("fs")')).toBeUndefined();
|
|
394
|
-
});
|
|
395
|
-
|
|
396
|
-
it('should respect execution limits', async () => {
|
|
397
|
-
// Create a very deep recursion scenario
|
|
398
|
-
const nodes = Array(2000).fill(null).map((_, i) => ({
|
|
399
|
-
type: 'orb',
|
|
400
|
-
name: `orb${i}`,
|
|
401
|
-
properties: {},
|
|
402
|
-
methods: [],
|
|
403
|
-
} as OrbNode));
|
|
404
|
-
|
|
405
|
-
const results = await runtime.executeProgram(nodes);
|
|
406
|
-
|
|
407
|
-
// Should have stopped before completing all
|
|
408
|
-
const lastResult = results[results.length - 1];
|
|
409
|
-
expect(lastResult.error).toContain('Max total nodes exceeded');
|
|
410
|
-
});
|
|
411
|
-
});
|
|
412
|
-
|
|
413
|
-
describe('Reset', () => {
|
|
414
|
-
it('should clear all state on reset', async () => {
|
|
415
|
-
// Set up some state
|
|
416
|
-
runtime.setVariable('x', 100);
|
|
417
|
-
await runtime.executeNode({
|
|
418
|
-
type: 'orb',
|
|
419
|
-
name: 'testOrb',
|
|
420
|
-
properties: {},
|
|
421
|
-
methods: [],
|
|
422
|
-
} as OrbNode);
|
|
423
|
-
|
|
424
|
-
runtime.on('test', () => {});
|
|
425
|
-
|
|
426
|
-
// Reset
|
|
427
|
-
runtime.reset();
|
|
428
|
-
|
|
429
|
-
// Verify state is cleared
|
|
430
|
-
expect(runtime.getVariable('x')).toBeUndefined();
|
|
431
|
-
expect(runtime.getContext().variables.size).toBe(0);
|
|
432
|
-
expect(runtime.getContext().functions.size).toBe(0);
|
|
433
|
-
expect(runtime.getExecutionHistory()).toHaveLength(0);
|
|
434
|
-
});
|
|
435
|
-
});
|
|
436
|
-
});
|