@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,380 @@
1
+ /**
2
+ * ChartRuntimeResolver
3
+ *
4
+ * Single source of truth for runtime capability queries.
5
+ * Mode is always derived from the installed LicenseManager state — it is
6
+ * never passed in by chart config, ensuring clients cannot bypass the
7
+ * license-driven mode selection.
8
+ *
9
+ * Usage:
10
+ * import { ChartRuntimeResolver } from '@forgecharts/sdk';
11
+ * const resolver = ChartRuntimeResolver.getInstance();
12
+ *
13
+ * resolver.isManagedMode() // true when a managed license is active
14
+ * resolver.isUnmanagedMode() // true when no license or unmanaged license
15
+ * resolver.canUseOrderEntry() // built-in order entry UI
16
+ * resolver.canUseExternalIngestion() // external data ingestion APIs
17
+ * resolver.canRenderOverlays() // overlay rendering
18
+ * resolver.canUseIndicators() // indicator panel
19
+ * resolver.canUseManagedTrading() // managed broker / execution hooks
20
+ * resolver.canUseDraggableOrders() // drag-to-price order placement
21
+ * resolver.canUseBracketOrders() // bracket / OCO order support
22
+ *
23
+ * Capability matrix
24
+ * ─────────────────────────────────────────────────────
25
+ * Capability unmanaged managed
26
+ * ─────────────────────────────────────────────────────
27
+ * externalIngestion ✓ ✗ (managed feed owns data)
28
+ * overlays ✓ ✓ (both modes render overlays)
29
+ * indicators ✓ ✓
30
+ * orderEntry ✗ ✓ (requires managed license)
31
+ * managedTrading ✗ ✓
32
+ * draggableOrders ✗ ✓
33
+ * bracketOrders ✗ ✓
34
+ * ─────────────────────────────────────────────────────
35
+ *
36
+ * Feature flags in the LicensePayload can further restrict capabilities
37
+ * within a mode. A managed license with `orderEntry: false` will still
38
+ * return false from canUseOrderEntry().
39
+ */
40
+
41
+ import { LicenseManager } from './LicenseManager';
42
+ import type { LicenseFeatures } from './licenseTypes';
43
+
44
+ export class ChartRuntimeResolver {
45
+ private static instance: ChartRuntimeResolver | null = null;
46
+
47
+ /** Always read from the singleton LicenseManager — never store a copy. */
48
+ private get lm(): LicenseManager {
49
+ return LicenseManager.getInstance();
50
+ }
51
+
52
+ private constructor() {}
53
+
54
+ static getInstance(): ChartRuntimeResolver {
55
+ if (!ChartRuntimeResolver.instance) {
56
+ ChartRuntimeResolver.instance = new ChartRuntimeResolver();
57
+ }
58
+ return ChartRuntimeResolver.instance;
59
+ }
60
+
61
+ // ── Mode checks ─────────────────────────────────────────────────────────────
62
+
63
+ /** True when the active license is in managed mode. */
64
+ isManagedMode(): boolean {
65
+ return this.lm.getMode() === 'managed';
66
+ }
67
+
68
+ /** True when no license is installed or the license is in unmanaged mode. */
69
+ isUnmanagedMode(): boolean {
70
+ return this.lm.getMode() === 'unmanaged';
71
+ }
72
+
73
+ // ── Capability queries ───────────────────────────────────────────────────────
74
+
75
+ /**
76
+ * External data ingestion APIs (e.g. pushing tick data from an external broker).
77
+ * Available in unmanaged mode only — in managed mode the built-in feed owns data.
78
+ * Override via `unmanagedIngestion` feature flag on a managed license if needed.
79
+ */
80
+ canUseExternalIngestion(): boolean {
81
+ if (this.isUnmanagedMode()) return true;
82
+ // A managed license can explicitly opt-in to external ingestion
83
+ return this.#featureOrDefault('unmanagedIngestion', false);
84
+ }
85
+
86
+ /**
87
+ * Overlay rendering from external state (price levels, zones, signals).
88
+ * Enabled in both modes by default; can be restricted via feature flag.
89
+ */
90
+ canRenderOverlays(): boolean {
91
+ return this.#featureOrDefault('overlays', true);
92
+ }
93
+
94
+ /**
95
+ * Indicator panel / computations.
96
+ * Enabled in both modes by default; can be restricted via feature flag.
97
+ */
98
+ canUseIndicators(): boolean {
99
+ return this.#featureOrDefault('indicators', true);
100
+ }
101
+
102
+ /**
103
+ * Built-in order entry UI hooks.
104
+ * Requires managed mode AND the `orderEntry` feature flag (or no flag restriction).
105
+ */
106
+ canUseOrderEntry(): boolean {
107
+ if (this.isUnmanagedMode()) return false;
108
+ return this.#featureOrDefault('orderEntry', true);
109
+ }
110
+
111
+ /**
112
+ * Managed trading service hooks (broker execution, position management).
113
+ * Requires managed mode AND the `managedTrading` feature flag (or no flag restriction).
114
+ */
115
+ canUseManagedTrading(): boolean {
116
+ if (this.isUnmanagedMode()) return false;
117
+ return this.#featureOrDefault('managedTrading', true);
118
+ }
119
+
120
+ /**
121
+ * Drag-to-price order placement.
122
+ * Requires managed mode AND the `draggableOrders` feature flag (or no flag restriction).
123
+ */
124
+ canUseDraggableOrders(): boolean {
125
+ if (this.isUnmanagedMode()) return false;
126
+ return this.#featureOrDefault('draggableOrders', true);
127
+ }
128
+
129
+ /**
130
+ * Bracket / OCO order support.
131
+ * Requires managed mode AND the `bracketOrders` feature flag (or no flag restriction).
132
+ */
133
+ canUseBracketOrders(): boolean {
134
+ if (this.isUnmanagedMode()) return false;
135
+ return this.#featureOrDefault('bracketOrders', true);
136
+ }
137
+
138
+ // ── UI render capability checks ──────────────────────────────────────────────
139
+ // These are component-level gates. They derive from the feature-level checks
140
+ // above and are the checks that toolbar / panel components should consume.
141
+
142
+ /**
143
+ * Whether to render the order entry tool in the toolbar.
144
+ * Hidden in unmanaged mode — requires managed license with order entry.
145
+ */
146
+ canRenderOrderEntry(): boolean {
147
+ return this.canUseOrderEntry();
148
+ }
149
+
150
+ /**
151
+ * Whether to render Buy / Sell action buttons on the chart or toolbar.
152
+ * Hidden in unmanaged mode — order placement requires managed license.
153
+ */
154
+ canRenderBuySellButtons(): boolean {
155
+ return this.canUseOrderEntry();
156
+ }
157
+
158
+ /**
159
+ * Whether to render the order ticket panel (entry form / DOM entry).
160
+ * Requires managed mode with order entry enabled.
161
+ */
162
+ canRenderOrderTicket(): boolean {
163
+ return this.canUseOrderEntry();
164
+ }
165
+
166
+ /**
167
+ * Whether to render bracket order controls (TP, SL, OCO).
168
+ * Requires managed mode with order entry AND the bracketOrders feature flag.
169
+ * If order entry itself is disabled, bracket controls are also hidden.
170
+ */
171
+ canRenderBracketControls(): boolean {
172
+ if (!this.canUseOrderEntry()) return false;
173
+ return this.canUseBracketOrders();
174
+ }
175
+
176
+ /**
177
+ * Whether to render order modification controls (drag-to-modify, edit panel).
178
+ * Requires managed mode with order entry AND the draggableOrders feature flag.
179
+ * If order entry itself is disabled, modification controls are also hidden.
180
+ */
181
+ canRenderOrderModificationControls(): boolean {
182
+ if (!this.canUseOrderEntry()) return false;
183
+ return this.canUseDraggableOrders();
184
+ }
185
+
186
+ /**
187
+ * Whether to render the managed trading control panel as a whole.
188
+ * Guards the container that holds buy/sell, order ticket, bracket controls, etc.
189
+ */
190
+ canRenderManagedTradingControls(): boolean {
191
+ return this.canUseManagedTrading();
192
+ }
193
+
194
+ /**
195
+ * True when the chart is in external-overlay-only (unmanaged) mode.
196
+ * Drives rendering of the external data overlay panel while suppressing
197
+ * all order entry UI elements.
198
+ */
199
+ canRenderExternalOverlayOnlyMode(): boolean {
200
+ return this.isUnmanagedMode();
201
+ }
202
+
203
+ /**
204
+ * Positions are always rendered regardless of license mode.
205
+ * Data flows from TradingOverlayStore (host-pushed in unmanaged,
206
+ * ManagedTradingController-driven in managed).
207
+ */
208
+ canRenderPositions(): boolean {
209
+ return true;
210
+ }
211
+
212
+ /**
213
+ * Execution fills are always rendered regardless of license mode.
214
+ */
215
+ canRenderFills(): boolean {
216
+ return true;
217
+ }
218
+
219
+ /**
220
+ * Candles are always rendered — unaffected by license mode.
221
+ * The RapidAPI datafeed remains active in all modes.
222
+ */
223
+ canRenderCandles(): boolean {
224
+ return true;
225
+ }
226
+
227
+ /**
228
+ * Indicator panel follows the `indicators` feature flag.
229
+ * Enabled in both modes by default.
230
+ */
231
+ canRenderIndicators(): boolean {
232
+ return this.canUseIndicators();
233
+ }
234
+
235
+ /**
236
+ * Drawing tools are always available regardless of license mode.
237
+ */
238
+ canRenderDrawings(): boolean {
239
+ return true;
240
+ }
241
+
242
+ // ── Convenience snapshot ────────────────────────────────────────────────────
243
+
244
+ /**
245
+ * Returns the full capability set as a plain object — useful for logging,
246
+ * debug panels, or passing state to React context in one call.
247
+ */
248
+ getCapabilities(): ChartCapabilities {
249
+ return {
250
+ // ── Mode & feature-level flags ──────────────────────────────────────
251
+ mode: this.lm.getMode(),
252
+ externalIngestion: this.canUseExternalIngestion(),
253
+ overlays: this.canRenderOverlays(),
254
+ indicators: this.canUseIndicators(),
255
+ orderEntry: this.canUseOrderEntry(),
256
+ managedTrading: this.canUseManagedTrading(),
257
+ draggableOrders: this.canUseDraggableOrders(),
258
+ bracketOrders: this.canUseBracketOrders(),
259
+ // ── UI render flags ─────────────────────────────────────────────────
260
+ renderOrderEntry: this.canRenderOrderEntry(),
261
+ renderBuySellButtons: this.canRenderBuySellButtons(),
262
+ renderOrderTicket: this.canRenderOrderTicket(),
263
+ renderBracketControls: this.canRenderBracketControls(),
264
+ renderOrderModificationControls: this.canRenderOrderModificationControls(),
265
+ renderManagedTradingControls: this.canRenderManagedTradingControls(),
266
+ renderExternalOverlayOnly: this.canRenderExternalOverlayOnlyMode(),
267
+ renderPositions: this.canRenderPositions(),
268
+ renderFills: this.canRenderFills(),
269
+ renderCandles: this.canRenderCandles(),
270
+ renderIndicators: this.canRenderIndicators(),
271
+ renderDrawings: this.canRenderDrawings(),
272
+ };
273
+ }
274
+
275
+ // ── Private helpers ─────────────────────────────────────────────────────────
276
+
277
+ /**
278
+ * If the active license has an explicit feature flag, use it.
279
+ * Otherwise fall back to `defaultValue` (allows capability even without an
280
+ * explicit flag, matching the "omitted = allowed" convention).
281
+ */
282
+ #featureOrDefault(name: keyof LicenseFeatures, defaultValue: boolean): boolean {
283
+ const features = this.lm.getFeatures();
284
+ const flag = features[name];
285
+ return flag === undefined ? defaultValue : flag === true;
286
+ }
287
+ }
288
+
289
+ // ── Capability snapshot type ─────────────────────────────────────────────────
290
+
291
+ export interface ChartCapabilities {
292
+ // ── Mode & feature-level flags ────────────────────────────────────────────
293
+ mode: 'managed' | 'unmanaged';
294
+ externalIngestion: boolean;
295
+ overlays: boolean;
296
+ indicators: boolean;
297
+ orderEntry: boolean;
298
+ managedTrading: boolean;
299
+ draggableOrders: boolean;
300
+ bracketOrders: boolean;
301
+ // ── UI render flags ───────────────────────────────────────────────────────
302
+ /** Show the order entry tool in toolbars (managed only). */
303
+ renderOrderEntry: boolean;
304
+ /** Show Buy / Sell action buttons (managed only). */
305
+ renderBuySellButtons: boolean;
306
+ /** Show the order ticket / entry form (managed only). */
307
+ renderOrderTicket: boolean;
308
+ /** Show bracket order controls — TP, SL, OCO (managed + bracketOrders). */
309
+ renderBracketControls: boolean;
310
+ /** Show order modification controls — drag handles, edit panel (managed + draggableOrders). */
311
+ renderOrderModificationControls: boolean;
312
+ /** Show the entire managed trading panel as a composite gate. */
313
+ renderManagedTradingControls: boolean;
314
+ /** True when operating in external-overlay-only (unmanaged) mode. */
315
+ renderExternalOverlayOnly: boolean;
316
+ /** Positions always rendered — data controlled by TradingOverlayStore. */
317
+ renderPositions: boolean;
318
+ /** Execution fills always rendered. */
319
+ renderFills: boolean;
320
+ /** Candles always rendered — RapidAPI datafeed unaffected by mode. */
321
+ renderCandles: boolean;
322
+ /** Indicators follow the `indicators` feature flag (both modes). */
323
+ renderIndicators: boolean;
324
+ /** Drawing tools always available regardless of license mode. */
325
+ renderDrawings: boolean;
326
+ }
327
+
328
+ // ── Module-level convenience functions ───────────────────────────────────────
329
+ // These mirror the instance methods and are the recommended import style for
330
+ // non-class consumers.
331
+
332
+ const resolver = () => ChartRuntimeResolver.getInstance();
333
+
334
+ /** True when a managed license is active. */
335
+ export const isManagedMode = (): boolean => resolver().isManagedMode();
336
+ /** True when no license or unmanaged license is active (safe default). */
337
+ export const isUnmanagedMode = (): boolean => resolver().isUnmanagedMode();
338
+ /** External data ingestion APIs (unmanaged mode only by default). */
339
+ export const canUseExternalIngestion = (): boolean => resolver().canUseExternalIngestion();
340
+ /** Overlay rendering — enabled in both modes. */
341
+ export const canRenderOverlays = (): boolean => resolver().canRenderOverlays();
342
+ /** Indicator computations — enabled in both modes. */
343
+ export const canUseIndicators = (): boolean => resolver().canUseIndicators();
344
+ /** Built-in order entry UI — managed mode only. */
345
+ export const canUseOrderEntry = (): boolean => resolver().canUseOrderEntry();
346
+ /** Managed trading service hooks — managed mode only. */
347
+ export const canUseManagedTrading = (): boolean => resolver().canUseManagedTrading();
348
+ /** Drag-to-price order placement — managed mode only. */
349
+ export const canUseDraggableOrders = (): boolean => resolver().canUseDraggableOrders();
350
+ /** Bracket / OCO order support — managed mode only. */
351
+ export const canUseBracketOrders = (): boolean => resolver().canUseBracketOrders();
352
+ /** Full capability snapshot as a plain object. */
353
+ export const getCapabilities = (): ChartCapabilities => resolver().getCapabilities();
354
+
355
+ // ── UI render convenience functions ──────────────────────────────────────────
356
+
357
+ /** Whether to render the order entry tool in toolbars (managed only). */
358
+ export const canRenderOrderEntry = (): boolean => resolver().canRenderOrderEntry();
359
+ /** Whether to render Buy/Sell buttons (managed only). */
360
+ export const canRenderBuySellButtons = (): boolean => resolver().canRenderBuySellButtons();
361
+ /** Whether to render the order ticket panel (managed only). */
362
+ export const canRenderOrderTicket = (): boolean => resolver().canRenderOrderTicket();
363
+ /** Whether to render bracket / OCO controls (managed + bracketOrders feature). */
364
+ export const canRenderBracketControls = (): boolean => resolver().canRenderBracketControls();
365
+ /** Whether to render drag/edit order modification controls (managed + draggableOrders). */
366
+ export const canRenderOrderModificationControls = (): boolean => resolver().canRenderOrderModificationControls();
367
+ /** Whether to render the managed trading control panel (managed only). */
368
+ export const canRenderManagedTradingControls = (): boolean => resolver().canRenderManagedTradingControls();
369
+ /** True when in external-overlay-only (unmanaged) mode — hides all order-entry UI. */
370
+ export const canRenderExternalOverlayOnlyMode = (): boolean => resolver().canRenderExternalOverlayOnlyMode();
371
+ /** Positions always rendered regardless of mode. */
372
+ export const canRenderPositions = (): boolean => resolver().canRenderPositions();
373
+ /** Execution fills always rendered regardless of mode. */
374
+ export const canRenderFills = (): boolean => resolver().canRenderFills();
375
+ /** Candles always rendered — RapidAPI datafeed unaffected by mode. */
376
+ export const canRenderCandles = (): boolean => resolver().canRenderCandles();
377
+ /** Indicators follow the indicators feature flag (both modes by default). */
378
+ export const canRenderIndicators = (): boolean => resolver().canRenderIndicators();
379
+ /** Drawing tools always available regardless of license mode. */
380
+ export const canRenderDrawings = (): boolean => resolver().canRenderDrawings();
@@ -0,0 +1,131 @@
1
+ import type { LicenseFeatures, LicenseMode, LicensePayload } from './licenseTypes';
2
+
3
+ const DEFAULT_MODE: LicenseMode = 'unmanaged';
4
+ const DEFAULT_FEATURES: LicenseFeatures = {};
5
+
6
+ /**
7
+ * LicenseManager — single source of truth for chart mode and feature access.
8
+ *
9
+ * Usage:
10
+ * const lm = LicenseManager.getInstance();
11
+ * await lm.validateLicense('my-key');
12
+ * lm.getMode(); // 'managed' | 'unmanaged'
13
+ * lm.hasFeature('orderEntry');
14
+ */
15
+ export class LicenseManager {
16
+ private static instance: LicenseManager | null = null;
17
+
18
+ private payload: LicensePayload | null = null;
19
+ private readonly subscribers: Set<() => void> = new Set();
20
+
21
+ private constructor() {}
22
+
23
+ /**
24
+ * Subscribe to license state changes (loadLicense, validateLicense, clear).
25
+ * Returns an unsubscribe function — call it to stop receiving notifications.
26
+ *
27
+ * @example
28
+ * const unsub = LicenseManager.getInstance().subscribe(() => {
29
+ * console.log('license changed', lm.getMode());
30
+ * });
31
+ * // later:
32
+ * unsub();
33
+ */
34
+ subscribe(cb: () => void): () => void {
35
+ this.subscribers.add(cb);
36
+ return () => this.subscribers.delete(cb);
37
+ }
38
+
39
+ private notify(): void {
40
+ this.subscribers.forEach((cb) => {
41
+ try { cb(); } catch { /* subscriber errors must not affect license state */ }
42
+ });
43
+ }
44
+
45
+ static getInstance(): LicenseManager {
46
+ if (!LicenseManager.instance) {
47
+ LicenseManager.instance = new LicenseManager();
48
+ }
49
+ return LicenseManager.instance;
50
+ }
51
+
52
+ /**
53
+ * Validate a license key against the licensing endpoint.
54
+ *
55
+ * @param key The raw license key entered by the user.
56
+ * @param verifyUrl Full URL of the forgecharts-admin verify endpoint,
57
+ * e.g. "http://licenses.example.com:4100/api/public/license/verify"
58
+ */
59
+ async validateLicense(key: string, verifyUrl: string): Promise<LicensePayload> {
60
+ if (!key || !key.trim()) {
61
+ throw new Error('LicenseManager: license key must not be empty');
62
+ }
63
+ if (!verifyUrl) {
64
+ throw new Error('LicenseManager: verifyUrl is required');
65
+ }
66
+
67
+ const res = await fetch(verifyUrl, {
68
+ method: 'POST',
69
+ headers: { 'Content-Type': 'application/json' },
70
+ body: JSON.stringify({ licenseKey: key.trim().toUpperCase() }),
71
+ });
72
+
73
+ if (!res.ok) {
74
+ throw new Error(`License server error: ${res.status} ${res.statusText}`);
75
+ }
76
+
77
+ const data = await res.json() as { valid: boolean; reason?: string; licenseKey?: string; mode?: string; plan?: string; expiresAt?: string | null; features?: Record<string, boolean> };
78
+
79
+ if (!data.valid) {
80
+ throw new Error(data.reason ?? 'License invalid');
81
+ }
82
+
83
+ const payload: LicensePayload = {
84
+ licenseKey: data.licenseKey ?? key.trim().toUpperCase(),
85
+ mode: (data.mode ?? 'unmanaged') as LicensePayload['mode'],
86
+ ...(data.plan != null && { plan: data.plan }),
87
+ ...(data.expiresAt != null && { expires: data.expiresAt }),
88
+ ...(data.features != null && { features: data.features as LicenseFeatures }),
89
+ };
90
+
91
+ this.payload = payload;
92
+ this.notify();
93
+ return payload;
94
+ }
95
+
96
+ /** Load a pre-validated payload (e.g. from cache / localStorage). */
97
+ loadLicense(payload: LicensePayload): void {
98
+ this.payload = payload;
99
+ this.notify();
100
+ }
101
+
102
+ /** Return the currently loaded payload, or null if none. */
103
+ getLicense(): LicensePayload | null {
104
+ return this.payload;
105
+ }
106
+
107
+ /**
108
+ * The chart mode driven by the active license.
109
+ * Defaults to 'unmanaged' so no managed features appear without a valid
110
+ * managed license.
111
+ */
112
+ getMode(): LicenseMode {
113
+ return this.payload?.mode ?? DEFAULT_MODE;
114
+ }
115
+
116
+ /** Return the feature flags for the active license. */
117
+ getFeatures(): LicenseFeatures {
118
+ return this.payload?.features ?? DEFAULT_FEATURES;
119
+ }
120
+
121
+ /** Check whether a specific feature is enabled on the active license. */
122
+ hasFeature(name: keyof LicenseFeatures): boolean {
123
+ return this.getFeatures()[name] === true;
124
+ }
125
+
126
+ /** Clear the current license state (returns to safe unmanaged defaults). */
127
+ clear(): void {
128
+ this.payload = null;
129
+ this.notify();
130
+ }
131
+ }