@forgecharts/sdk 1.1.23

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.
Files changed (101) hide show
  1. package/package.json +50 -0
  2. package/src/__tests__/backwardCompatibility.test.ts +191 -0
  3. package/src/__tests__/candleInvariant.test.ts +500 -0
  4. package/src/__tests__/public-api-surface.ts +76 -0
  5. package/src/__tests__/timeframeBoundary.test.ts +583 -0
  6. package/src/api/DrawingManager.ts +188 -0
  7. package/src/api/EventBus.ts +53 -0
  8. package/src/api/IndicatorDAG.ts +389 -0
  9. package/src/api/IndicatorRegistry.ts +47 -0
  10. package/src/api/LayoutManager.ts +72 -0
  11. package/src/api/PaneManager.ts +129 -0
  12. package/src/api/ReferenceAPI.ts +195 -0
  13. package/src/api/TChart.ts +881 -0
  14. package/src/api/createChart.ts +43 -0
  15. package/src/api/drawing tools/fib gann menu/fibRetracement.ts +27 -0
  16. package/src/api/drawing tools/lines menu/crossLine.ts +21 -0
  17. package/src/api/drawing tools/lines menu/disjointChannel.ts +74 -0
  18. package/src/api/drawing tools/lines menu/extendedLine.ts +22 -0
  19. package/src/api/drawing tools/lines menu/flatTopBottom.ts +45 -0
  20. package/src/api/drawing tools/lines menu/horizontal.ts +24 -0
  21. package/src/api/drawing tools/lines menu/horizontalRay.ts +25 -0
  22. package/src/api/drawing tools/lines menu/infoLine.ts +127 -0
  23. package/src/api/drawing tools/lines menu/insidePitchfork.ts +21 -0
  24. package/src/api/drawing tools/lines menu/modifiedSchiffPitchfork.ts +18 -0
  25. package/src/api/drawing tools/lines menu/parallelChannel.ts +47 -0
  26. package/src/api/drawing tools/lines menu/pitchfork.ts +15 -0
  27. package/src/api/drawing tools/lines menu/ray.ts +28 -0
  28. package/src/api/drawing tools/lines menu/regressionTrend.ts +157 -0
  29. package/src/api/drawing tools/lines menu/schiffPitchfork.ts +18 -0
  30. package/src/api/drawing tools/lines menu/trendAngle.ts +64 -0
  31. package/src/api/drawing tools/lines menu/trendline.ts +16 -0
  32. package/src/api/drawing tools/lines menu/vertical.ts +16 -0
  33. package/src/api/drawing tools/pointers menu/crosshair.ts +17 -0
  34. package/src/api/drawing tools/pointers menu/cursor.ts +16 -0
  35. package/src/api/drawing tools/pointers menu/demonstration.ts +35 -0
  36. package/src/api/drawing tools/pointers menu/dot.ts +26 -0
  37. package/src/api/drawing tools/shapes menu/rectangle.ts +24 -0
  38. package/src/api/drawing tools/shapes menu/text.ts +30 -0
  39. package/src/api/drawingUtils.ts +82 -0
  40. package/src/core/CanvasLayer.ts +77 -0
  41. package/src/core/Chart.ts +917 -0
  42. package/src/core/CoordTransform.ts +282 -0
  43. package/src/core/Crosshair.ts +207 -0
  44. package/src/core/IndicatorEngine.ts +216 -0
  45. package/src/core/InteractionManager.ts +899 -0
  46. package/src/core/PriceScale.ts +133 -0
  47. package/src/core/Series.ts +132 -0
  48. package/src/core/TimeScale.ts +175 -0
  49. package/src/datafeed/DatafeedConnector.ts +300 -0
  50. package/src/engine/CandleEngine.ts +458 -0
  51. package/src/engine/__tests__/CandleEngine.test.ts +402 -0
  52. package/src/engine/candleInvariants.ts +172 -0
  53. package/src/engine/mergeUtils.ts +93 -0
  54. package/src/engine/timeframeUtils.ts +118 -0
  55. package/src/index.ts +190 -0
  56. package/src/internal.ts +41 -0
  57. package/src/licensing/ChartRuntimeResolver.ts +380 -0
  58. package/src/licensing/LicenseManager.ts +131 -0
  59. package/src/licensing/__tests__/ChartRuntimeResolver.test.ts +207 -0
  60. package/src/licensing/__tests__/LicenseManager.test.ts +180 -0
  61. package/src/licensing/licenseTypes.ts +19 -0
  62. package/src/pine/PineCompiler.ts +68 -0
  63. package/src/pine/diagnostics.ts +30 -0
  64. package/src/pine/index.ts +7 -0
  65. package/src/pine/pine-ast.ts +163 -0
  66. package/src/pine/pine-lexer.ts +265 -0
  67. package/src/pine/pine-parser.ts +439 -0
  68. package/src/pine/pine-transpiler.ts +301 -0
  69. package/src/pixi/LayerName.ts +35 -0
  70. package/src/pixi/PixiCandlestickRenderer.ts +125 -0
  71. package/src/pixi/PixiChart.ts +425 -0
  72. package/src/pixi/PixiCrosshairRenderer.ts +134 -0
  73. package/src/pixi/PixiDrawingRenderer.ts +121 -0
  74. package/src/pixi/PixiGridRenderer.ts +136 -0
  75. package/src/pixi/PixiLayerManager.ts +102 -0
  76. package/src/renderers/CandlestickRenderer.ts +130 -0
  77. package/src/renderers/HistogramRenderer.ts +63 -0
  78. package/src/renderers/LineRenderer.ts +77 -0
  79. package/src/theme/colors.ts +21 -0
  80. package/src/tools/barDivergenceCheck.ts +305 -0
  81. package/src/trading/TradingOverlayStore.ts +161 -0
  82. package/src/trading/UnmanagedIngestion.ts +156 -0
  83. package/src/trading/__tests__/ManagedTradingController.test.ts +338 -0
  84. package/src/trading/__tests__/TradingOverlayStore.test.ts +323 -0
  85. package/src/trading/__tests__/UnmanagedIngestion.test.ts +205 -0
  86. package/src/trading/managed/ManagedTradingController.ts +292 -0
  87. package/src/trading/managed/managedCapabilities.ts +98 -0
  88. package/src/trading/managed/managedTypes.ts +151 -0
  89. package/src/trading/tradingTypes.ts +135 -0
  90. package/src/tscript/TScriptIndicator.ts +54 -0
  91. package/src/tscript/ast.ts +105 -0
  92. package/src/tscript/lexer.ts +190 -0
  93. package/src/tscript/parser.ts +334 -0
  94. package/src/tscript/runtime.ts +525 -0
  95. package/src/tscript/series.ts +84 -0
  96. package/src/types/IChart.ts +56 -0
  97. package/src/types/IRenderer.ts +16 -0
  98. package/src/types/ISeries.ts +30 -0
  99. package/tsconfig.json +22 -0
  100. package/tsup.config.ts +15 -0
  101. package/vitest.config.ts +25 -0
@@ -0,0 +1,338 @@
1
+ /**
2
+ * ManagedTradingController — unit tests
3
+ *
4
+ * Covers:
5
+ * - Cannot initialize (registerProvider) in unmanaged mode
6
+ * - Can initialize in managed mode
7
+ * - isReady() reflects provider registration state
8
+ * - placeOrder() throws without a provider
9
+ * - placeOrder() with a mock provider: writes optimistic overlay entry,
10
+ * updates with provider id on success, rolls back on failure
11
+ * - cancelOrder() updates overlay status to 'cancelled'
12
+ * - modifyOrder() patches price / qty / label in overlay
13
+ * - placeBracketOrder() creates 3 orders with the same groupId in the overlay
14
+ * - reportFill() pushes to overlay + updates order status
15
+ * - All methods blocked in unmanaged mode
16
+ */
17
+
18
+ import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
19
+ import { LicenseManager } from '../../licensing/LicenseManager';
20
+ import { TradingOverlayStore } from '../TradingOverlayStore';
21
+ import { ManagedTradingController } from '../managed/ManagedTradingController';
22
+ import type { IExecutionProvider, PlaceOrderInput, PlaceBracketOrderInput } from '../managed/managedTypes';
23
+ import type { ExecutionFill } from '../tradingTypes';
24
+
25
+ // ─── License helpers ──────────────────────────────────────────────────────────
26
+
27
+ function installManaged() {
28
+ LicenseManager.getInstance().loadLicense({ licenseKey: 'm', mode: 'managed' });
29
+ }
30
+
31
+ function installUnmanaged() {
32
+ LicenseManager.getInstance().loadLicense({ licenseKey: 'u', mode: 'unmanaged' });
33
+ }
34
+
35
+ function clearLicense() {
36
+ LicenseManager.getInstance().clear();
37
+ }
38
+
39
+ // ─── Mock provider factory ────────────────────────────────────────────────────
40
+
41
+ function makeMockProvider(overrides: Partial<IExecutionProvider> = {}): IExecutionProvider {
42
+ return {
43
+ providerId: 'mock-provider',
44
+ placeOrder: vi.fn().mockResolvedValue({ orderId: 'prov-001', status: 'open' }),
45
+ cancelOrder: vi.fn().mockResolvedValue(undefined),
46
+ modifyOrder: vi.fn().mockResolvedValue({ orderId: 'prov-001', status: 'open' }),
47
+ placeBracketOrder: vi.fn().mockResolvedValue({
48
+ entry: { orderId: 'prov-entry', timestamp: 1000 },
49
+ stopLoss: { orderId: 'prov-sl', timestamp: 1000 },
50
+ takeProfit: { orderId: 'prov-tp', timestamp: 1000 },
51
+ groupId: 'bracket-test',
52
+ }),
53
+ ...overrides,
54
+ };
55
+ }
56
+
57
+ function makeOrderInput(overrides: Partial<PlaceOrderInput> = {}): PlaceOrderInput {
58
+ return {
59
+ symbol: 'BTCUSDT',
60
+ side: 'buy',
61
+ type: 'limit',
62
+ limitPrice: 50_000,
63
+ qty: 1,
64
+ ...overrides,
65
+ };
66
+ }
67
+
68
+ // ─── Mode gate ────────────────────────────────────────────────────────────────
69
+
70
+ describe('ManagedTradingController — mode gate', () => {
71
+ afterEach(clearLicense);
72
+
73
+ it('registerProvider() throws in unmanaged mode', () => {
74
+ installUnmanaged();
75
+ const store = new TradingOverlayStore();
76
+ const ctrl = new ManagedTradingController(store);
77
+ expect(() => ctrl.registerProvider(makeMockProvider())).toThrow(/managed/i);
78
+ });
79
+
80
+ it('registerProvider() throws when no license (defaults to unmanaged)', () => {
81
+ clearLicense();
82
+ const store = new TradingOverlayStore();
83
+ const ctrl = new ManagedTradingController(store);
84
+ expect(() => ctrl.registerProvider(makeMockProvider())).toThrow(/managed/i);
85
+ });
86
+
87
+ it('registerProvider() succeeds in managed mode', () => {
88
+ installManaged();
89
+ const store = new TradingOverlayStore();
90
+ const ctrl = new ManagedTradingController(store);
91
+ expect(() => ctrl.registerProvider(makeMockProvider())).not.toThrow();
92
+ });
93
+
94
+ it('placeOrder() throws in unmanaged mode', async () => {
95
+ installUnmanaged();
96
+ const store = new TradingOverlayStore();
97
+ const ctrl = new ManagedTradingController(store);
98
+ await expect(ctrl.placeOrder(makeOrderInput())).rejects.toThrow(/managed/i);
99
+ });
100
+ });
101
+
102
+ // ─── isReady / getProviderId ──────────────────────────────────────────────────
103
+
104
+ describe('ManagedTradingController — readiness', () => {
105
+ beforeEach(installManaged);
106
+ afterEach(clearLicense);
107
+
108
+ it('isReady() is false before provider registration', () => {
109
+ const ctrl = new ManagedTradingController(new TradingOverlayStore());
110
+ expect(ctrl.isReady()).toBe(false);
111
+ });
112
+
113
+ it('isReady() is true after registration', () => {
114
+ const ctrl = new ManagedTradingController(new TradingOverlayStore());
115
+ ctrl.registerProvider(makeMockProvider());
116
+ expect(ctrl.isReady()).toBe(true);
117
+ });
118
+
119
+ it('getProviderId() returns null before registration', () => {
120
+ const ctrl = new ManagedTradingController(new TradingOverlayStore());
121
+ expect(ctrl.getProviderId()).toBeNull();
122
+ });
123
+
124
+ it('getProviderId() returns the provider id after registration', () => {
125
+ const ctrl = new ManagedTradingController(new TradingOverlayStore());
126
+ ctrl.registerProvider(makeMockProvider({ providerId: 'rithmic' }));
127
+ expect(ctrl.getProviderId()).toBe('rithmic');
128
+ });
129
+ });
130
+
131
+ // ─── placeOrder ───────────────────────────────────────────────────────────────
132
+
133
+ describe('ManagedTradingController — placeOrder', () => {
134
+ beforeEach(installManaged);
135
+ afterEach(clearLicense);
136
+
137
+ it('throws when no provider is registered', async () => {
138
+ const ctrl = new ManagedTradingController(new TradingOverlayStore());
139
+ await expect(ctrl.placeOrder(makeOrderInput())).rejects.toThrow(/no execution provider/i);
140
+ });
141
+
142
+ it('writes a pending order to overlay before provider responds', async () => {
143
+ const store = new TradingOverlayStore();
144
+ const ctrl = new ManagedTradingController(store);
145
+ let resolveAck: () => void;
146
+ const provider = makeMockProvider({
147
+ placeOrder: vi.fn().mockImplementation(
148
+ () => new Promise<{ orderId: string; status: 'open' }>((res) => {
149
+ resolveAck = () => res({ orderId: 'prov-001', status: 'open' });
150
+ }),
151
+ ),
152
+ });
153
+ ctrl.registerProvider(provider);
154
+
155
+ const promise = ctrl.placeOrder(makeOrderInput());
156
+ // Overlay should have a pending entry immediately (optimistic)
157
+ expect(store.getOrders().some(o => o.status === 'pending')).toBe(true);
158
+
159
+ resolveAck!();
160
+ await promise;
161
+ });
162
+
163
+ it('updates overlay with provider-assigned id on success', async () => {
164
+ const store = new TradingOverlayStore();
165
+ const ctrl = new ManagedTradingController(store);
166
+ const provider = makeMockProvider({
167
+ placeOrder: vi.fn().mockResolvedValue({ orderId: 'server-id-1', status: 'open' }),
168
+ });
169
+ ctrl.registerProvider(provider);
170
+
171
+ await ctrl.placeOrder(makeOrderInput());
172
+
173
+ const orders = store.getOrders();
174
+ expect(orders.some(o => o.id === 'server-id-1' && o.status === 'open')).toBe(true);
175
+ });
176
+
177
+ it('rolls back to rejected status when provider throws', async () => {
178
+ const store = new TradingOverlayStore();
179
+ const ctrl = new ManagedTradingController(store);
180
+ const provider = makeMockProvider({
181
+ placeOrder: vi.fn().mockRejectedValue(new Error('venue rejected')),
182
+ });
183
+ ctrl.registerProvider(provider);
184
+
185
+ await expect(ctrl.placeOrder(makeOrderInput())).rejects.toThrow('venue rejected');
186
+
187
+ const orders = store.getOrders();
188
+ expect(orders.some(o => o.status === 'rejected')).toBe(true);
189
+ });
190
+
191
+ it('delegates to the registered provider', async () => {
192
+ const store = new TradingOverlayStore();
193
+ const ctrl = new ManagedTradingController(store);
194
+ const mockFn = vi.fn().mockResolvedValue({ orderId: 'x', status: 'open' });
195
+ ctrl.registerProvider(makeMockProvider({ placeOrder: mockFn }));
196
+
197
+ const input = makeOrderInput({ side: 'sell', qty: 3 });
198
+ await ctrl.placeOrder(input);
199
+
200
+ expect(mockFn).toHaveBeenCalledOnce();
201
+ expect(mockFn).toHaveBeenCalledWith(input);
202
+ });
203
+ });
204
+
205
+ // ─── cancelOrder ─────────────────────────────────────────────────────────────
206
+
207
+ describe('ManagedTradingController — cancelOrder', () => {
208
+ beforeEach(installManaged);
209
+ afterEach(clearLicense);
210
+
211
+ it('updates overlay status to cancelled after provider confirms', async () => {
212
+ const store = new TradingOverlayStore();
213
+ const ctrl = new ManagedTradingController(store);
214
+ ctrl.registerProvider(makeMockProvider());
215
+
216
+ await ctrl.placeOrder(makeOrderInput());
217
+ // Find the confirmed order id
218
+ const orderId = store.getOrders().find(o => o.status === 'open')?.id;
219
+ expect(orderId).toBeDefined();
220
+
221
+ await ctrl.cancelOrder(orderId!);
222
+ expect(store.getOrders().find(o => o.id === orderId)!.status).toBe('cancelled');
223
+ });
224
+ });
225
+
226
+ // ─── modifyOrder ──────────────────────────────────────────────────────────────
227
+
228
+ describe('ManagedTradingController — modifyOrder', () => {
229
+ beforeEach(installManaged);
230
+ afterEach(clearLicense);
231
+
232
+ it('patches price in overlay when limitPrice is provided', async () => {
233
+ const store = new TradingOverlayStore();
234
+ const ctrl = new ManagedTradingController(store);
235
+ ctrl.registerProvider(makeMockProvider());
236
+
237
+ await ctrl.placeOrder(makeOrderInput({ limitPrice: 50_000 }));
238
+ const orderId = store.getOrders().find(o => o.status === 'open')?.id;
239
+ expect(orderId).toBeDefined();
240
+
241
+ await ctrl.modifyOrder(orderId!, { limitPrice: 51_000 });
242
+ expect(store.getOrders().find(o => o.id === orderId)!.price).toBe(51_000);
243
+ });
244
+
245
+ it('patches qty in overlay when qty is provided', async () => {
246
+ const store = new TradingOverlayStore();
247
+ const ctrl = new ManagedTradingController(store);
248
+ ctrl.registerProvider(makeMockProvider());
249
+
250
+ await ctrl.placeOrder(makeOrderInput({ qty: 1 }));
251
+ const orderId = store.getOrders().find(o => o.status === 'open')?.id;
252
+ await ctrl.modifyOrder(orderId!, { qty: 5 });
253
+ expect(store.getOrders().find(o => o.id === orderId)!.qty).toBe(5);
254
+ });
255
+ });
256
+
257
+ // ─── placeBracketOrder ────────────────────────────────────────────────────────
258
+
259
+ describe('ManagedTradingController — placeBracketOrder', () => {
260
+ beforeEach(installManaged);
261
+ afterEach(clearLicense);
262
+
263
+ it('throws without a provider', async () => {
264
+ const ctrl = new ManagedTradingController(new TradingOverlayStore());
265
+ const input: PlaceBracketOrderInput = {
266
+ entry: makeOrderInput(),
267
+ stopLossPrice: 49_000,
268
+ takeProfitPrice: 55_000,
269
+ };
270
+ await expect(ctrl.placeBracketOrder(input)).rejects.toThrow(/no execution provider/i);
271
+ });
272
+
273
+ it('creates 3 orders sharing a groupId in the overlay', async () => {
274
+ const store = new TradingOverlayStore();
275
+ const ctrl = new ManagedTradingController(store);
276
+ ctrl.registerProvider(makeMockProvider());
277
+
278
+ const groupId = 'bracket-test';
279
+ const input: PlaceBracketOrderInput = {
280
+ entry: makeOrderInput({ groupId }),
281
+ stopLossPrice: 49_000,
282
+ takeProfitPrice: 55_000,
283
+ };
284
+ await ctrl.placeBracketOrder(input);
285
+
286
+ const group = store.getOrdersByGroup(groupId);
287
+ expect(group).toHaveLength(3);
288
+
289
+ const roles = group.map(o => o.role).sort();
290
+ expect(roles).toContain('entry');
291
+ expect(roles).toContain('stop_loss');
292
+ expect(roles).toContain('take_profit');
293
+ });
294
+ });
295
+
296
+ // ─── reportFill ───────────────────────────────────────────────────────────────
297
+
298
+ describe('ManagedTradingController — reportFill', () => {
299
+ beforeEach(installManaged);
300
+ afterEach(clearLicense);
301
+
302
+ it('pushes fill into overlay store', async () => {
303
+ const store = new TradingOverlayStore();
304
+ const ctrl = new ManagedTradingController(store);
305
+ ctrl.registerProvider(makeMockProvider());
306
+
307
+ await ctrl.placeOrder(makeOrderInput());
308
+ const orderId = store.getOrders().find(o => o.status === 'open')?.id ?? 'oid';
309
+
310
+ const fill: ExecutionFill = {
311
+ id: 'fill-1',
312
+ orderId,
313
+ side: 'buy',
314
+ price: 50_000,
315
+ qty: 1,
316
+ timestamp: Date.now(),
317
+ };
318
+ ctrl.reportFill(fill);
319
+
320
+ expect(store.getFills()).toHaveLength(1);
321
+ expect(store.getFills()[0]!.id).toBe('fill-1');
322
+ });
323
+
324
+ it('updates order status to filled after a fill report', async () => {
325
+ const store = new TradingOverlayStore();
326
+ const ctrl = new ManagedTradingController(store);
327
+ ctrl.registerProvider(makeMockProvider());
328
+
329
+ await ctrl.placeOrder(makeOrderInput());
330
+ const orderId = store.getOrders().find(o => o.status === 'open')?.id ?? 'oid';
331
+
332
+ ctrl.reportFill({
333
+ id: 'fill-2', orderId, side: 'buy', price: 50_000, qty: 1, timestamp: Date.now(),
334
+ });
335
+
336
+ expect(store.getOrders().find(o => o.id === orderId)!.status).toBe('filled');
337
+ });
338
+ });
@@ -0,0 +1,323 @@
1
+ /**
2
+ * TradingOverlayStore — unit tests
3
+ *
4
+ * Covers:
5
+ * - Orders: set, upsert, remove, getByGroup
6
+ * - Positions: set, upsert, remove
7
+ * - Fills: push, getFills, getFillsByOrder
8
+ * - clear() resets everything
9
+ * - snapshot() reflects current state
10
+ * - onChange subscriber fires on every mutation
11
+ * - Both unmanaged and managed pathways can write into the same store
12
+ * - groupId-based linked orders coexist correctly
13
+ * - TP / SL / entry roles are stored verbatim
14
+ */
15
+
16
+ import { describe, it, expect, beforeEach, vi } from 'vitest';
17
+ import { TradingOverlayStore } from '../TradingOverlayStore';
18
+ import type { ChartOrder, ChartPosition, ExecutionFill } from '../tradingTypes';
19
+
20
+ // ─── Helpers ──────────────────────────────────────────────────────────────────
21
+
22
+ function makeOrder(id: string, overrides: Partial<ChartOrder> = {}): ChartOrder {
23
+ return {
24
+ id,
25
+ role: 'entry',
26
+ side: 'buy',
27
+ status: 'open',
28
+ price: 100,
29
+ qty: 1,
30
+ ...overrides,
31
+ };
32
+ }
33
+
34
+ function makePosition(id: string, overrides: Partial<ChartPosition> = {}): ChartPosition {
35
+ return {
36
+ id,
37
+ side: 'long',
38
+ status: 'open',
39
+ entryPrice: 100,
40
+ qty: 1,
41
+ ...overrides,
42
+ };
43
+ }
44
+
45
+ function makeFill(id: string, orderId?: string): ExecutionFill {
46
+ return {
47
+ id,
48
+ side: 'buy',
49
+ price: 100,
50
+ qty: 1,
51
+ timestamp: Date.now(),
52
+ ...(orderId !== undefined ? { orderId } : {}),
53
+ };
54
+ }
55
+
56
+ // ─── Orders ───────────────────────────────────────────────────────────────────
57
+
58
+ describe('TradingOverlayStore — orders', () => {
59
+ let store: TradingOverlayStore;
60
+ beforeEach(() => { store = new TradingOverlayStore(); });
61
+
62
+ it('starts empty', () => {
63
+ expect(store.getOrders()).toHaveLength(0);
64
+ });
65
+
66
+ it('setOrders replaces the entire set', () => {
67
+ store.setOrders([makeOrder('a'), makeOrder('b')]);
68
+ expect(store.getOrders()).toHaveLength(2);
69
+
70
+ store.setOrders([makeOrder('c')]);
71
+ expect(store.getOrders()).toHaveLength(1);
72
+ expect(store.getOrders()[0]!.id).toBe('c');
73
+ });
74
+
75
+ it('upsertOrder inserts a new order', () => {
76
+ store.upsertOrder(makeOrder('o1'));
77
+ expect(store.getOrders()).toHaveLength(1);
78
+ expect(store.getOrders()[0]!.id).toBe('o1');
79
+ });
80
+
81
+ it('upsertOrder replaces an existing order by id', () => {
82
+ store.upsertOrder(makeOrder('o1', { price: 100 }));
83
+ store.upsertOrder(makeOrder('o1', { price: 200 }));
84
+ expect(store.getOrders()).toHaveLength(1);
85
+ expect(store.getOrders()[0]!.price).toBe(200);
86
+ });
87
+
88
+ it('removeOrder removes by id', () => {
89
+ store.upsertOrder(makeOrder('o1'));
90
+ store.upsertOrder(makeOrder('o2'));
91
+ store.removeOrder('o1');
92
+ expect(store.getOrders()).toHaveLength(1);
93
+ expect(store.getOrders()[0]!.id).toBe('o2');
94
+ });
95
+
96
+ it('removeOrder is a silent no-op for unknown id', () => {
97
+ store.upsertOrder(makeOrder('o1'));
98
+ store.removeOrder('nonexistent');
99
+ expect(store.getOrders()).toHaveLength(1);
100
+ });
101
+
102
+ it('getOrdersByGroup filters correctly', () => {
103
+ store.setOrders([
104
+ makeOrder('entry', { groupId: 'bracket1', role: 'entry' }),
105
+ makeOrder('sl', { groupId: 'bracket1', role: 'stop_loss' }),
106
+ makeOrder('tp', { groupId: 'bracket1', role: 'take_profit' }),
107
+ makeOrder('lone', { role: 'limit' }), // no groupId
108
+ ]);
109
+ const group = store.getOrdersByGroup('bracket1');
110
+ expect(group).toHaveLength(3);
111
+ const roles = group.map(o => o.role).sort();
112
+ expect(roles).toEqual(['entry', 'stop_loss', 'take_profit']);
113
+ });
114
+
115
+ it('getOrdersByGroup returns [] for unknown groupId', () => {
116
+ store.upsertOrder(makeOrder('o1', { groupId: 'g1' }));
117
+ expect(store.getOrdersByGroup('g2')).toHaveLength(0);
118
+ });
119
+ });
120
+
121
+ // ─── Positions ────────────────────────────────────────────────────────────────
122
+
123
+ describe('TradingOverlayStore — positions', () => {
124
+ let store: TradingOverlayStore;
125
+ beforeEach(() => { store = new TradingOverlayStore(); });
126
+
127
+ it('starts empty', () => {
128
+ expect(store.getPositions()).toHaveLength(0);
129
+ });
130
+
131
+ it('setPositions replaces the entire set', () => {
132
+ store.setPositions([makePosition('p1'), makePosition('p2')]);
133
+ expect(store.getPositions()).toHaveLength(2);
134
+ store.setPositions([]);
135
+ expect(store.getPositions()).toHaveLength(0);
136
+ });
137
+
138
+ it('upsertPosition inserts when missing', () => {
139
+ store.upsertPosition(makePosition('p1'));
140
+ expect(store.getPositions()).toHaveLength(1);
141
+ });
142
+
143
+ it('upsertPosition updates when id matches', () => {
144
+ store.upsertPosition(makePosition('p1', { qty: 1 }));
145
+ store.upsertPosition(makePosition('p1', { qty: 5 }));
146
+ expect(store.getPositions()).toHaveLength(1);
147
+ expect(store.getPositions()[0]!.qty).toBe(5);
148
+ });
149
+
150
+ it('removePosition removes by id', () => {
151
+ store.upsertPosition(makePosition('p1'));
152
+ store.removePosition('p1');
153
+ expect(store.getPositions()).toHaveLength(0);
154
+ });
155
+
156
+ it('removePosition is a silent no-op for unknown id', () => {
157
+ expect(() => store.removePosition('ghost')).not.toThrow();
158
+ });
159
+ });
160
+
161
+ // ─── Fills ────────────────────────────────────────────────────────────────────
162
+
163
+ describe('TradingOverlayStore — fills', () => {
164
+ let store: TradingOverlayStore;
165
+ beforeEach(() => { store = new TradingOverlayStore(); });
166
+
167
+ it('starts empty', () => {
168
+ expect(store.getFills()).toHaveLength(0);
169
+ });
170
+
171
+ it('pushExecution appends fills in order', () => {
172
+ store.pushExecution(makeFill('f1'));
173
+ store.pushExecution(makeFill('f2'));
174
+ expect(store.getFills()).toHaveLength(2);
175
+ expect(store.getFills()[0]!.id).toBe('f1');
176
+ expect(store.getFills()[1]!.id).toBe('f2');
177
+ });
178
+
179
+ it('getFillsByOrder filters by orderId', () => {
180
+ store.pushExecution(makeFill('f1', 'orderA'));
181
+ store.pushExecution(makeFill('f2', 'orderB'));
182
+ store.pushExecution(makeFill('f3', 'orderA'));
183
+ const fills = store.getFillsByOrder('orderA');
184
+ expect(fills).toHaveLength(2);
185
+ expect(fills.map(f => f.id)).toEqual(['f1', 'f3']);
186
+ });
187
+
188
+ it('getFillsByOrder returns [] for unknown orderId', () => {
189
+ store.pushExecution(makeFill('f1', 'o1'));
190
+ expect(store.getFillsByOrder('unknown')).toHaveLength(0);
191
+ });
192
+ });
193
+
194
+ // ─── clear & snapshot ─────────────────────────────────────────────────────────
195
+
196
+ describe('TradingOverlayStore — clear and snapshot', () => {
197
+ let store: TradingOverlayStore;
198
+ beforeEach(() => { store = new TradingOverlayStore(); });
199
+
200
+ it('clear() removes all state', () => {
201
+ store.upsertOrder(makeOrder('o1'));
202
+ store.upsertPosition(makePosition('p1'));
203
+ store.pushExecution(makeFill('f1'));
204
+ store.clear();
205
+ expect(store.getOrders()).toHaveLength(0);
206
+ expect(store.getPositions()).toHaveLength(0);
207
+ expect(store.getFills()).toHaveLength(0);
208
+ });
209
+
210
+ it('snapshot() reflects the current state', () => {
211
+ store.upsertOrder(makeOrder('o1'));
212
+ store.upsertPosition(makePosition('p1'));
213
+ store.pushExecution(makeFill('f1'));
214
+ const snap = store.snapshot();
215
+ expect(snap.orders).toHaveLength(1);
216
+ expect(snap.positions).toHaveLength(1);
217
+ expect(snap.fills).toHaveLength(1);
218
+ });
219
+ });
220
+
221
+ // ─── onChange subscription ────────────────────────────────────────────────────
222
+
223
+ describe('TradingOverlayStore — onChange notifications', () => {
224
+ let store: TradingOverlayStore;
225
+ beforeEach(() => { store = new TradingOverlayStore(); });
226
+
227
+ function countCalls(fn: () => void): number {
228
+ const spy = vi.fn();
229
+ const unsub = store.onChange(spy);
230
+ fn();
231
+ unsub();
232
+ return spy.mock.calls.length;
233
+ }
234
+
235
+ it('fires on setOrders', () => {
236
+ expect(countCalls(() => store.setOrders([makeOrder('o1')]))).toBe(1);
237
+ });
238
+
239
+ it('fires on upsertOrder', () => {
240
+ expect(countCalls(() => store.upsertOrder(makeOrder('o1')))).toBe(1);
241
+ });
242
+
243
+ it('fires on removeOrder (when item exists)', () => {
244
+ store.upsertOrder(makeOrder('o1'));
245
+ expect(countCalls(() => store.removeOrder('o1'))).toBe(1);
246
+ });
247
+
248
+ it('does NOT fire on removeOrder for unknown id', () => {
249
+ expect(countCalls(() => store.removeOrder('ghost'))).toBe(0);
250
+ });
251
+
252
+ it('fires on setPositions', () => {
253
+ expect(countCalls(() => store.setPositions([makePosition('p1')]))).toBe(1);
254
+ });
255
+
256
+ it('fires on upsertPosition', () => {
257
+ expect(countCalls(() => store.upsertPosition(makePosition('p1')))).toBe(1);
258
+ });
259
+
260
+ it('fires on pushExecution', () => {
261
+ expect(countCalls(() => store.pushExecution(makeFill('f1')))).toBe(1);
262
+ });
263
+
264
+ it('fires on clear', () => {
265
+ expect(countCalls(() => store.clear())).toBe(1);
266
+ });
267
+
268
+ it('unsubscribe stops future notifications', () => {
269
+ const spy = vi.fn();
270
+ const unsub = store.onChange(spy);
271
+ unsub();
272
+ store.upsertOrder(makeOrder('o1'));
273
+ expect(spy).not.toHaveBeenCalled();
274
+ });
275
+ });
276
+
277
+ // ─── Shared overlay: both modes write into the same store ─────────────────────
278
+
279
+ describe('TradingOverlayStore — shared between unmanaged and managed pathways', () => {
280
+ it('unmanaged push and managed push coexist in the same store', () => {
281
+ const store = new TradingOverlayStore();
282
+
283
+ // Simulate unmanaged host pushing an external position
284
+ store.upsertPosition(makePosition('ext-pos', { qty: 2 }));
285
+
286
+ // Simulate managed controller pushing an order (e.g. bracket entry)
287
+ store.upsertOrder(makeOrder('mgd-entry', { groupId: 'brk1', role: 'entry' }));
288
+ store.upsertOrder(makeOrder('mgd-sl', { groupId: 'brk1', role: 'stop_loss', price: 90 }));
289
+ store.upsertOrder(makeOrder('mgd-tp', { groupId: 'brk1', role: 'take_profit', price: 115 }));
290
+
291
+ expect(store.getPositions()).toHaveLength(1);
292
+ expect(store.getOrders()).toHaveLength(3);
293
+ expect(store.getOrdersByGroup('brk1')).toHaveLength(3);
294
+ });
295
+
296
+ it('groupId-based bracket legs can coexist alongside unlinked orders', () => {
297
+ const store = new TradingOverlayStore();
298
+
299
+ store.upsertOrder(makeOrder('solo', { role: 'limit' }));
300
+ store.upsertOrder(makeOrder('b1-ent', { groupId: 'brk-A', role: 'entry' }));
301
+ store.upsertOrder(makeOrder('b1-sl', { groupId: 'brk-A', role: 'stop_loss' }));
302
+ store.upsertOrder(makeOrder('b1-tp', { groupId: 'brk-A', role: 'take_profit' }));
303
+ store.upsertOrder(makeOrder('b2-ent', { groupId: 'brk-B', role: 'entry' }));
304
+ store.upsertOrder(makeOrder('b2-sl', { groupId: 'brk-B', role: 'stop_loss' }));
305
+
306
+ expect(store.getOrders()).toHaveLength(6);
307
+ expect(store.getOrdersByGroup('brk-A')).toHaveLength(3);
308
+ expect(store.getOrdersByGroup('brk-B')).toHaveLength(2);
309
+ });
310
+
311
+ it('TP / SL / entry roles are stored verbatim', () => {
312
+ const store = new TradingOverlayStore();
313
+ const entry = makeOrder('entry-1', { role: 'entry', price: 100 });
314
+ const sl = makeOrder('sl-1', { role: 'stop_loss', price: 90 });
315
+ const tp = makeOrder('tp-1', { role: 'take_profit', price: 115 });
316
+ store.setOrders([entry, sl, tp]);
317
+
318
+ const orders = store.getOrders();
319
+ expect(orders.find(o => o.id === 'entry-1')!.role).toBe('entry');
320
+ expect(orders.find(o => o.id === 'sl-1')!.role).toBe('stop_loss');
321
+ expect(orders.find(o => o.id === 'tp-1')!.role).toBe('take_profit');
322
+ });
323
+ });