@metamask-previews/perps-controller 1.2.0-preview-0b8553c95 → 1.3.0-preview-e5ed18206

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.
@@ -1,510 +0,0 @@
1
- /**
2
- * MYXProvider
3
- *
4
- * Stage 1 provider implementation for MYX protocol.
5
- * Implements the PerpsProvider interface with read-only operations.
6
- * Trading functionality will be added in Stage 3.
7
- *
8
- * Key differences from HyperLiquid:
9
- * - Uses USDT collateral on BNB chain (vs USDC on Arbitrum)
10
- * - Multi-Pool Model: multiple pools can exist per symbol
11
- * - Uses REST polling for prices (WebSocket deferred to Stage 3)
12
- */
13
- var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) {
14
- if (kind === "m") throw new TypeError("Private method is not writable");
15
- if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
16
- if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
17
- return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
18
- };
19
- var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
20
- if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
21
- if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
22
- return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
23
- };
24
- var _MYXProvider_instances, _MYXProvider_deps, _MYXProvider_clientService, _MYXProvider_isTestnet, _MYXProvider_poolsCache, _MYXProvider_poolSymbolMap, _MYXProvider_tickersCache, _MYXProvider_getErrorContext, _MYXProvider_getAdaptedPrice;
25
- import { PERPS_CONSTANTS } from "../constants/perpsConfig.mjs";
26
- import { MYXClientService } from "../services/MYXClientService.mjs";
27
- import { WebSocketConnectionState } from "../types/index.mjs";
28
- import { ensureError } from "../utils/errorUtils.mjs";
29
- import { adaptMarketFromMYX, adaptMarketDataFromMYX, adaptPriceFromMYX, filterMYXExclusiveMarkets, buildPoolSymbolMap } from "../utils/myxAdapter.mjs";
30
- // ============================================================================
31
- // Constants
32
- // ============================================================================
33
- const MYX_NOT_SUPPORTED_ERROR = 'MYX trading not yet supported';
34
- const MYX_BLOCK_EXPLORER_URL = 'https://bscscan.com';
35
- const MYX_TESTNET_EXPLORER_URL = 'https://testnet.bscscan.com';
36
- // ============================================================================
37
- // MYXProvider
38
- // ============================================================================
39
- /**
40
- * MYX provider implementation
41
- *
42
- * Stage 1: Read-only operations (markets, prices)
43
- * Trading operations return errors until Stage 3.
44
- */
45
- export class MYXProvider {
46
- constructor(options) {
47
- _MYXProvider_instances.add(this);
48
- this.protocolId = 'myx';
49
- // Platform dependencies
50
- _MYXProvider_deps.set(this, void 0);
51
- // Client service
52
- _MYXProvider_clientService.set(this, void 0);
53
- // Configuration
54
- _MYXProvider_isTestnet.set(this, void 0);
55
- // Cache for pools (freshness delegated to MYXClientService)
56
- _MYXProvider_poolsCache.set(this, []);
57
- _MYXProvider_poolSymbolMap.set(this, new Map());
58
- // Ticker cache for price data
59
- _MYXProvider_tickersCache.set(this, new Map());
60
- __classPrivateFieldSet(this, _MYXProvider_deps, options.platformDependencies, "f");
61
- __classPrivateFieldSet(this, _MYXProvider_isTestnet, options.isTestnet ?? true, "f"); // Force testnet in Stage 1
62
- // Initialize client service
63
- __classPrivateFieldSet(this, _MYXProvider_clientService, new MYXClientService(__classPrivateFieldGet(this, _MYXProvider_deps, "f"), {
64
- isTestnet: __classPrivateFieldGet(this, _MYXProvider_isTestnet, "f"),
65
- }), "f");
66
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").debugLogger.log('[MYXProvider] Constructor complete', {
67
- protocolId: this.protocolId,
68
- isTestnet: __classPrivateFieldGet(this, _MYXProvider_isTestnet, "f"),
69
- });
70
- }
71
- // ============================================================================
72
- // Initialization & Lifecycle
73
- // ============================================================================
74
- async initialize() {
75
- try {
76
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").debugLogger.log('[MYXProvider] Initializing...');
77
- // Fetch initial markets
78
- const pools = await __classPrivateFieldGet(this, _MYXProvider_clientService, "f").getMarkets();
79
- // Filter to MYX-exclusive markets
80
- __classPrivateFieldSet(this, _MYXProvider_poolsCache, filterMYXExclusiveMarkets(pools), "f");
81
- __classPrivateFieldSet(this, _MYXProvider_poolSymbolMap, buildPoolSymbolMap(__classPrivateFieldGet(this, _MYXProvider_poolsCache, "f")), "f");
82
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").debugLogger.log('[MYXProvider] Initialized successfully', {
83
- totalPools: pools.length,
84
- exclusivePools: __classPrivateFieldGet(this, _MYXProvider_poolsCache, "f").length,
85
- });
86
- return { success: true };
87
- }
88
- catch (caughtError) {
89
- const wrappedError = ensureError(caughtError, 'MYXProvider.initialize');
90
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").logger.error(wrappedError, __classPrivateFieldGet(this, _MYXProvider_instances, "m", _MYXProvider_getErrorContext).call(this, 'initialize'));
91
- return { success: false, error: wrappedError.message };
92
- }
93
- }
94
- async disconnect() {
95
- try {
96
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").debugLogger.log('[MYXProvider] Disconnecting...');
97
- __classPrivateFieldGet(this, _MYXProvider_clientService, "f").disconnect();
98
- __classPrivateFieldSet(this, _MYXProvider_poolsCache, [], "f");
99
- __classPrivateFieldGet(this, _MYXProvider_poolSymbolMap, "f").clear();
100
- __classPrivateFieldGet(this, _MYXProvider_tickersCache, "f").clear();
101
- return { success: true };
102
- }
103
- catch (caughtError) {
104
- const wrappedError = ensureError(caughtError, 'MYXProvider.disconnect');
105
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").logger.error(wrappedError, __classPrivateFieldGet(this, _MYXProvider_instances, "m", _MYXProvider_getErrorContext).call(this, 'disconnect'));
106
- return { success: false, error: wrappedError.message };
107
- }
108
- }
109
- async ping(timeoutMs) {
110
- await __classPrivateFieldGet(this, _MYXProvider_clientService, "f").ping(timeoutMs);
111
- }
112
- async toggleTestnet() {
113
- // Stage 1: Testnet only
114
- return {
115
- success: false,
116
- isTestnet: __classPrivateFieldGet(this, _MYXProvider_isTestnet, "f"),
117
- error: 'MYX mainnet not yet available',
118
- };
119
- }
120
- async isReadyToTrade() {
121
- // Stage 1: Trading not supported
122
- return {
123
- ready: false,
124
- error: 'MYX trading not yet supported',
125
- walletConnected: false,
126
- networkSupported: __classPrivateFieldGet(this, _MYXProvider_isTestnet, "f"),
127
- };
128
- }
129
- // ============================================================================
130
- // Market Data Operations (Stage 1 - Fully Implemented)
131
- // ============================================================================
132
- // TODO: Align error handling - read operations should return empty defaults
133
- // instead of throwing, matching HyperLiquid pattern
134
- async getMarkets(_params) {
135
- try {
136
- // Delegate cache freshness to MYXClientService
137
- const pools = await __classPrivateFieldGet(this, _MYXProvider_clientService, "f").getMarkets();
138
- __classPrivateFieldSet(this, _MYXProvider_poolsCache, filterMYXExclusiveMarkets(pools), "f");
139
- __classPrivateFieldSet(this, _MYXProvider_poolSymbolMap, buildPoolSymbolMap(__classPrivateFieldGet(this, _MYXProvider_poolsCache, "f")), "f");
140
- return __classPrivateFieldGet(this, _MYXProvider_poolsCache, "f").map((pool) => adaptMarketFromMYX(pool));
141
- }
142
- catch (caughtError) {
143
- const wrappedError = ensureError(caughtError, 'MYXProvider.getMarkets');
144
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").logger.error(wrappedError, __classPrivateFieldGet(this, _MYXProvider_instances, "m", _MYXProvider_getErrorContext).call(this, 'getMarkets'));
145
- throw wrappedError;
146
- }
147
- }
148
- async getMarketDataWithPrices() {
149
- try {
150
- // Ensure we have markets
151
- if (__classPrivateFieldGet(this, _MYXProvider_poolsCache, "f").length === 0) {
152
- await this.getMarkets();
153
- }
154
- // Fetch tickers for all pools
155
- const poolIds = __classPrivateFieldGet(this, _MYXProvider_poolsCache, "f").map((pool) => pool.poolId);
156
- const tickers = await __classPrivateFieldGet(this, _MYXProvider_clientService, "f").getTickers(poolIds);
157
- // Build ticker map
158
- const tickerMap = new Map();
159
- for (const ticker of tickers) {
160
- tickerMap.set(ticker.poolId, ticker);
161
- __classPrivateFieldGet(this, _MYXProvider_tickersCache, "f").set(ticker.poolId, ticker);
162
- }
163
- // Transform to PerpsMarketData
164
- return __classPrivateFieldGet(this, _MYXProvider_poolsCache, "f").map((pool) => {
165
- const ticker = tickerMap.get(pool.poolId);
166
- return adaptMarketDataFromMYX(pool, ticker, __classPrivateFieldGet(this, _MYXProvider_deps, "f").marketDataFormatters);
167
- });
168
- }
169
- catch (caughtError) {
170
- const wrappedError = ensureError(caughtError, 'MYXProvider.getMarketDataWithPrices');
171
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").logger.error(wrappedError, __classPrivateFieldGet(this, _MYXProvider_instances, "m", _MYXProvider_getErrorContext).call(this, 'getMarketDataWithPrices'));
172
- throw wrappedError;
173
- }
174
- }
175
- // ============================================================================
176
- // Price Subscriptions (Stage 1 - REST Polling)
177
- // ============================================================================
178
- subscribeToPrices(params) {
179
- const { symbols, callback, includeOrderBook } = params;
180
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").debugLogger.log('[MYXProvider] Setting up price subscription', {
181
- symbols: symbols.length,
182
- includeOrderBook,
183
- });
184
- // Map symbols to pool IDs
185
- const poolIds = [];
186
- for (const pool of __classPrivateFieldGet(this, _MYXProvider_poolsCache, "f")) {
187
- const symbol = pool.baseSymbol || pool.poolId;
188
- if (symbols.includes(symbol)) {
189
- poolIds.push(pool.poolId);
190
- }
191
- }
192
- if (poolIds.length === 0) {
193
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").debugLogger.log('[MYXProvider] subscribeToPrices: No pool IDs found. Ensure initialize() has been called.', { symbols });
194
- setTimeout(() => params.callback([]), 0);
195
- return () => {
196
- /* noop */
197
- };
198
- }
199
- // Start price polling
200
- __classPrivateFieldGet(this, _MYXProvider_clientService, "f").startPricePolling(poolIds, (tickers) => {
201
- // Convert tickers to PriceUpdate format
202
- const updates = tickers.map((ticker) => {
203
- const symbol = __classPrivateFieldGet(this, _MYXProvider_poolSymbolMap, "f").get(ticker.poolId) ?? ticker.poolId;
204
- const { price, change24h } = __classPrivateFieldGet(this, _MYXProvider_instances, "m", _MYXProvider_getAdaptedPrice).call(this, ticker);
205
- return {
206
- symbol,
207
- price,
208
- timestamp: Date.now(),
209
- percentChange24h: change24h.toFixed(2),
210
- providerId: 'myx',
211
- };
212
- });
213
- callback(updates);
214
- });
215
- // Return unsubscribe function
216
- return () => {
217
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").debugLogger.log('[MYXProvider] Unsubscribing from prices');
218
- __classPrivateFieldGet(this, _MYXProvider_clientService, "f").stopPricePolling();
219
- };
220
- }
221
- // ============================================================================
222
- // Asset Routes (Stage 1 - Stubbed)
223
- // ============================================================================
224
- getDepositRoutes(_params) {
225
- // Stage 1: No deposit support
226
- return [];
227
- }
228
- getWithdrawalRoutes(_params) {
229
- // Stage 1: No withdrawal support
230
- return [];
231
- }
232
- // ============================================================================
233
- // Trading Operations (Stage 1 - All Stubbed)
234
- // ============================================================================
235
- async placeOrder(_params) {
236
- return {
237
- success: false,
238
- error: MYX_NOT_SUPPORTED_ERROR,
239
- };
240
- }
241
- async editOrder(_params) {
242
- return {
243
- success: false,
244
- error: MYX_NOT_SUPPORTED_ERROR,
245
- };
246
- }
247
- async cancelOrder(_params) {
248
- return {
249
- success: false,
250
- error: MYX_NOT_SUPPORTED_ERROR,
251
- };
252
- }
253
- async cancelOrders(_params) {
254
- return {
255
- success: false,
256
- successCount: 0,
257
- failureCount: 0,
258
- results: [],
259
- };
260
- }
261
- async closePosition(_params) {
262
- return {
263
- success: false,
264
- error: MYX_NOT_SUPPORTED_ERROR,
265
- };
266
- }
267
- async closePositions(_params) {
268
- return {
269
- success: false,
270
- successCount: 0,
271
- failureCount: 0,
272
- results: [],
273
- };
274
- }
275
- async updatePositionTPSL(_params) {
276
- return {
277
- success: false,
278
- error: MYX_NOT_SUPPORTED_ERROR,
279
- };
280
- }
281
- async updateMargin(_params) {
282
- return {
283
- success: false,
284
- error: MYX_NOT_SUPPORTED_ERROR,
285
- };
286
- }
287
- async withdraw(_params) {
288
- return {
289
- success: false,
290
- error: MYX_NOT_SUPPORTED_ERROR,
291
- };
292
- }
293
- // ============================================================================
294
- // Account Operations (Stage 1 - Empty Returns)
295
- // ============================================================================
296
- async getPositions(_params) {
297
- // Stage 1: No position tracking
298
- return [];
299
- }
300
- async getAccountState(_params) {
301
- // Stage 1: Empty account state
302
- return {
303
- availableBalance: '0',
304
- totalBalance: '0',
305
- marginUsed: '0',
306
- unrealizedPnl: '0',
307
- returnOnEquity: '0',
308
- };
309
- }
310
- async getOrders(_params) {
311
- // Stage 1: No order tracking
312
- return [];
313
- }
314
- async getOpenOrders(_params) {
315
- // Stage 1: No order tracking
316
- return [];
317
- }
318
- async getOrderFills(_params) {
319
- // Stage 1: No fill tracking
320
- return [];
321
- }
322
- async getOrFetchFills(_params) {
323
- // Stage 1: No fill tracking
324
- return [];
325
- }
326
- async getFunding(_params) {
327
- // Stage 1: No funding tracking
328
- return [];
329
- }
330
- async getHistoricalPortfolio(_params) {
331
- return {
332
- accountValue1dAgo: '0',
333
- timestamp: Date.now(),
334
- };
335
- }
336
- async getUserNonFundingLedgerUpdates(_params) {
337
- return [];
338
- }
339
- async getUserHistory(_params) {
340
- return [];
341
- }
342
- // ============================================================================
343
- // Validation Operations (Stage 1 - All Invalid)
344
- // ============================================================================
345
- async validateDeposit(_params) {
346
- return { isValid: false, error: MYX_NOT_SUPPORTED_ERROR };
347
- }
348
- async validateOrder(_params) {
349
- return { isValid: false, error: MYX_NOT_SUPPORTED_ERROR };
350
- }
351
- async validateClosePosition(_params) {
352
- return { isValid: false, error: MYX_NOT_SUPPORTED_ERROR };
353
- }
354
- async validateWithdrawal(_params) {
355
- return { isValid: false, error: MYX_NOT_SUPPORTED_ERROR };
356
- }
357
- // ============================================================================
358
- // Protocol Calculations (Stage 1 - Default Values)
359
- // ============================================================================
360
- async calculateLiquidationPrice(_params) {
361
- return '0';
362
- }
363
- async calculateMaintenanceMargin(_params) {
364
- return 0;
365
- }
366
- async getMaxLeverage(_asset) {
367
- return 100; // MYX default max leverage
368
- }
369
- async calculateFees(_params) {
370
- // MYX fee structure (placeholder values)
371
- return {
372
- feeRate: 0.0005, // 0.05% total fee rate
373
- protocolFeeRate: 0.0005, // Protocol taker fee
374
- };
375
- }
376
- // ============================================================================
377
- // Subscriptions (Stage 1 - No-op)
378
- // ============================================================================
379
- subscribeToPositions(params) {
380
- // Stage 1: No position tracking - immediately call back with empty array
381
- // to signal loading is complete (no data to show)
382
- setTimeout(() => params.callback([]), 0);
383
- return () => {
384
- /* noop */
385
- };
386
- }
387
- subscribeToOrderFills(params) {
388
- // Stage 1: No fill tracking - immediately call back with empty array
389
- setTimeout(() => params.callback([]), 0);
390
- return () => {
391
- /* noop */
392
- };
393
- }
394
- subscribeToOrders(params) {
395
- // Stage 1: No order tracking - immediately call back with empty array
396
- setTimeout(() => params.callback([]), 0);
397
- return () => {
398
- /* noop */
399
- };
400
- }
401
- subscribeToAccount(params) {
402
- // Stage 1: Empty account state - immediately call back
403
- setTimeout(() => params.callback({
404
- availableBalance: '0',
405
- totalBalance: '0',
406
- marginUsed: '0',
407
- unrealizedPnl: '0',
408
- returnOnEquity: '0',
409
- }), 0);
410
- return () => {
411
- /* noop */
412
- };
413
- }
414
- subscribeToOICaps(params) {
415
- // Stage 1: No OI caps - immediately call back with empty array
416
- // (matches HyperLiquid pattern which calls callback with cached data)
417
- setTimeout(() => params.callback([]), 0);
418
- return () => {
419
- /* noop */
420
- };
421
- }
422
- subscribeToCandles(params) {
423
- // Stage 1: No candle data - immediately call back with empty candles
424
- // (matches HyperLiquid pattern which calls callback after initial fetch)
425
- setTimeout(() => params.callback({
426
- symbol: params.symbol,
427
- interval: params.interval,
428
- candles: [],
429
- }), 0);
430
- return () => {
431
- /* noop */
432
- };
433
- }
434
- subscribeToOrderBook(params) {
435
- // Stage 1: No order book - immediately call back with empty data
436
- setTimeout(() => params.callback({
437
- bids: [],
438
- asks: [],
439
- spread: '0',
440
- spreadPercentage: '0',
441
- midPrice: '0',
442
- lastUpdated: Date.now(),
443
- maxTotal: '0',
444
- }), 0);
445
- return () => {
446
- /* noop */
447
- };
448
- }
449
- setLiveDataConfig(_config) {
450
- // Stage 1: No-op
451
- }
452
- // ============================================================================
453
- // Connection State (Stage 1 - REST Only)
454
- // ============================================================================
455
- getWebSocketConnectionState() {
456
- // Stage 1: No WebSocket, report as connected (REST is always available)
457
- return WebSocketConnectionState.Connected;
458
- }
459
- subscribeToConnectionState(_listener) {
460
- // Stage 1: No WebSocket, no connection state changes
461
- return () => {
462
- /* noop */
463
- };
464
- }
465
- async reconnect() {
466
- // Stage 1: No WebSocket to reconnect
467
- __classPrivateFieldGet(this, _MYXProvider_deps, "f").debugLogger.log('[MYXProvider] reconnect() is no-op in Stage 1');
468
- }
469
- // ============================================================================
470
- // Block Explorer
471
- // ============================================================================
472
- getBlockExplorerUrl(address) {
473
- const baseUrl = __classPrivateFieldGet(this, _MYXProvider_isTestnet, "f")
474
- ? MYX_TESTNET_EXPLORER_URL
475
- : MYX_BLOCK_EXPLORER_URL;
476
- return address ? `${baseUrl}/address/${address}` : baseUrl;
477
- }
478
- // ============================================================================
479
- // Fee Discount (Stage 1 - No-op)
480
- // ============================================================================
481
- setUserFeeDiscount(_discountBips) {
482
- // Stage 1: No fee discount support
483
- }
484
- // ============================================================================
485
- // HIP-3 Operations (N/A for MYX)
486
- // ============================================================================
487
- async getAvailableDexs() {
488
- // MYX doesn't have HIP-3 equivalent
489
- return [];
490
- }
491
- }
492
- _MYXProvider_deps = new WeakMap(), _MYXProvider_clientService = new WeakMap(), _MYXProvider_isTestnet = new WeakMap(), _MYXProvider_poolsCache = new WeakMap(), _MYXProvider_poolSymbolMap = new WeakMap(), _MYXProvider_tickersCache = new WeakMap(), _MYXProvider_instances = new WeakSet(), _MYXProvider_getErrorContext = function _MYXProvider_getErrorContext(method, extra) {
493
- return {
494
- tags: {
495
- feature: PERPS_CONSTANTS.FeatureName,
496
- provider: 'MYXProvider',
497
- network: __classPrivateFieldGet(this, _MYXProvider_isTestnet, "f") ? 'testnet' : 'mainnet',
498
- },
499
- context: {
500
- name: `MYXProvider.${method}`,
501
- data: {
502
- isTestnet: __classPrivateFieldGet(this, _MYXProvider_isTestnet, "f"),
503
- ...extra,
504
- },
505
- },
506
- };
507
- }, _MYXProvider_getAdaptedPrice = function _MYXProvider_getAdaptedPrice(ticker) {
508
- return adaptPriceFromMYX(ticker);
509
- };
510
- //# sourceMappingURL=MYXProvider.mjs.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"MYXProvider.mjs","sourceRoot":"","sources":["../../src/providers/MYXProvider.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MYXProvider\n *\n * Stage 1 provider implementation for MYX protocol.\n * Implements the PerpsProvider interface with read-only operations.\n * Trading functionality will be added in Stage 3.\n *\n * Key differences from HyperLiquid:\n * - Uses USDT collateral on BNB chain (vs USDC on Arbitrum)\n * - Multi-Pool Model: multiple pools can exist per symbol\n * - Uses REST polling for prices (WebSocket deferred to Stage 3)\n */\n\nimport type { CaipAccountId } from '@metamask/utils';\n\nimport { PERPS_CONSTANTS } from '../constants/perpsConfig';\nimport { MYXClientService } from '../services/MYXClientService';\nimport { WebSocketConnectionState } from '../types';\nimport type {\n AccountState,\n AssetRoute,\n BatchCancelOrdersParams,\n CancelOrderParams,\n CancelOrderResult,\n CancelOrdersResult,\n ClosePositionParams,\n ClosePositionsParams,\n ClosePositionsResult,\n DepositParams,\n DisconnectResult,\n EditOrderParams,\n FeeCalculationParams,\n FeeCalculationResult,\n Funding,\n GetAccountStateParams,\n GetFundingParams,\n GetHistoricalPortfolioParams,\n GetMarketsParams,\n GetOrderFillsParams,\n GetOrdersParams,\n GetOrFetchFillsParams,\n GetPositionsParams,\n GetSupportedPathsParams,\n HistoricalPortfolioResult,\n InitializeResult,\n LiquidationPriceParams,\n LiveDataConfig,\n MaintenanceMarginParams,\n MarginResult,\n MarketInfo,\n Order,\n OrderFill,\n OrderParams,\n OrderResult,\n PerpsPlatformDependencies,\n PerpsMarketData,\n PerpsProvider,\n Position,\n PriceUpdate,\n ReadyToTradeResult,\n SubscribeAccountParams,\n SubscribeCandlesParams,\n SubscribeOICapsParams,\n SubscribeOrderBookParams,\n SubscribeOrderFillsParams,\n SubscribeOrdersParams,\n SubscribePositionsParams,\n SubscribePricesParams,\n ToggleTestnetResult,\n UpdatePositionTPSLParams,\n UserHistoryItem,\n WithdrawParams,\n WithdrawResult,\n RawLedgerUpdate,\n} from '../types';\nimport type { MYXPoolSymbol, MYXTicker } from '../types/myx-types';\nimport { ensureError } from '../utils/errorUtils';\nimport {\n adaptMarketFromMYX,\n adaptMarketDataFromMYX,\n adaptPriceFromMYX,\n filterMYXExclusiveMarkets,\n buildPoolSymbolMap,\n} from '../utils/myxAdapter';\n\n// ============================================================================\n// Constants\n// ============================================================================\n\nconst MYX_NOT_SUPPORTED_ERROR = 'MYX trading not yet supported';\nconst MYX_BLOCK_EXPLORER_URL = 'https://bscscan.com';\nconst MYX_TESTNET_EXPLORER_URL = 'https://testnet.bscscan.com';\n\n// ============================================================================\n// MYXProvider\n// ============================================================================\n\n/**\n * MYX provider implementation\n *\n * Stage 1: Read-only operations (markets, prices)\n * Trading operations return errors until Stage 3.\n */\nexport class MYXProvider implements PerpsProvider {\n readonly protocolId = 'myx';\n\n // Platform dependencies\n readonly #deps: PerpsPlatformDependencies;\n\n // Client service\n readonly #clientService: MYXClientService;\n\n // Configuration\n readonly #isTestnet: boolean;\n\n // Cache for pools (freshness delegated to MYXClientService)\n #poolsCache: MYXPoolSymbol[] = [];\n\n #poolSymbolMap: Map<string, string> = new Map();\n\n // Ticker cache for price data\n readonly #tickersCache: Map<string, MYXTicker> = new Map();\n\n constructor(options: {\n isTestnet?: boolean;\n platformDependencies: PerpsPlatformDependencies;\n }) {\n this.#deps = options.platformDependencies;\n this.#isTestnet = options.isTestnet ?? true; // Force testnet in Stage 1\n\n // Initialize client service\n this.#clientService = new MYXClientService(this.#deps, {\n isTestnet: this.#isTestnet,\n });\n\n this.#deps.debugLogger.log('[MYXProvider] Constructor complete', {\n protocolId: this.protocolId,\n isTestnet: this.#isTestnet,\n });\n }\n\n // ============================================================================\n // Error Context Helper\n // ============================================================================\n\n #getErrorContext(\n method: string,\n extra?: Record<string, unknown>,\n ): {\n tags?: Record<string, string | number>;\n context?: { name: string; data: Record<string, unknown> };\n } {\n return {\n tags: {\n feature: PERPS_CONSTANTS.FeatureName,\n provider: 'MYXProvider',\n network: this.#isTestnet ? 'testnet' : 'mainnet',\n },\n context: {\n name: `MYXProvider.${method}`,\n data: {\n isTestnet: this.#isTestnet,\n ...extra,\n },\n },\n };\n }\n\n // ============================================================================\n // Initialization & Lifecycle\n // ============================================================================\n\n async initialize(): Promise<InitializeResult> {\n try {\n this.#deps.debugLogger.log('[MYXProvider] Initializing...');\n\n // Fetch initial markets\n const pools = await this.#clientService.getMarkets();\n\n // Filter to MYX-exclusive markets\n this.#poolsCache = filterMYXExclusiveMarkets(pools);\n this.#poolSymbolMap = buildPoolSymbolMap(this.#poolsCache);\n\n this.#deps.debugLogger.log('[MYXProvider] Initialized successfully', {\n totalPools: pools.length,\n exclusivePools: this.#poolsCache.length,\n });\n\n return { success: true };\n } catch (caughtError) {\n const wrappedError = ensureError(caughtError, 'MYXProvider.initialize');\n this.#deps.logger.error(\n wrappedError,\n this.#getErrorContext('initialize'),\n );\n return { success: false, error: wrappedError.message };\n }\n }\n\n async disconnect(): Promise<DisconnectResult> {\n try {\n this.#deps.debugLogger.log('[MYXProvider] Disconnecting...');\n\n this.#clientService.disconnect();\n this.#poolsCache = [];\n this.#poolSymbolMap.clear();\n this.#tickersCache.clear();\n\n return { success: true };\n } catch (caughtError) {\n const wrappedError = ensureError(caughtError, 'MYXProvider.disconnect');\n this.#deps.logger.error(\n wrappedError,\n this.#getErrorContext('disconnect'),\n );\n return { success: false, error: wrappedError.message };\n }\n }\n\n async ping(timeoutMs?: number): Promise<void> {\n await this.#clientService.ping(timeoutMs);\n }\n\n async toggleTestnet(): Promise<ToggleTestnetResult> {\n // Stage 1: Testnet only\n return {\n success: false,\n isTestnet: this.#isTestnet,\n error: 'MYX mainnet not yet available',\n };\n }\n\n async isReadyToTrade(): Promise<ReadyToTradeResult> {\n // Stage 1: Trading not supported\n return {\n ready: false,\n error: 'MYX trading not yet supported',\n walletConnected: false,\n networkSupported: this.#isTestnet,\n };\n }\n\n // ============================================================================\n // Market Data Operations (Stage 1 - Fully Implemented)\n // ============================================================================\n\n // TODO: Align error handling - read operations should return empty defaults\n // instead of throwing, matching HyperLiquid pattern\n async getMarkets(_params?: GetMarketsParams): Promise<MarketInfo[]> {\n try {\n // Delegate cache freshness to MYXClientService\n const pools = await this.#clientService.getMarkets();\n this.#poolsCache = filterMYXExclusiveMarkets(pools);\n this.#poolSymbolMap = buildPoolSymbolMap(this.#poolsCache);\n\n return this.#poolsCache.map((pool) => adaptMarketFromMYX(pool));\n } catch (caughtError) {\n const wrappedError = ensureError(caughtError, 'MYXProvider.getMarkets');\n this.#deps.logger.error(\n wrappedError,\n this.#getErrorContext('getMarkets'),\n );\n throw wrappedError;\n }\n }\n\n async getMarketDataWithPrices(): Promise<PerpsMarketData[]> {\n try {\n // Ensure we have markets\n if (this.#poolsCache.length === 0) {\n await this.getMarkets();\n }\n\n // Fetch tickers for all pools\n const poolIds = this.#poolsCache.map((pool) => pool.poolId);\n const tickers = await this.#clientService.getTickers(poolIds);\n\n // Build ticker map\n const tickerMap = new Map<string, MYXTicker>();\n for (const ticker of tickers) {\n tickerMap.set(ticker.poolId, ticker);\n this.#tickersCache.set(ticker.poolId, ticker);\n }\n\n // Transform to PerpsMarketData\n return this.#poolsCache.map((pool) => {\n const ticker = tickerMap.get(pool.poolId);\n return adaptMarketDataFromMYX(\n pool,\n ticker,\n this.#deps.marketDataFormatters,\n );\n });\n } catch (caughtError) {\n const wrappedError = ensureError(\n caughtError,\n 'MYXProvider.getMarketDataWithPrices',\n );\n this.#deps.logger.error(\n wrappedError,\n this.#getErrorContext('getMarketDataWithPrices'),\n );\n throw wrappedError;\n }\n }\n\n // ============================================================================\n // Price Subscriptions (Stage 1 - REST Polling)\n // ============================================================================\n\n subscribeToPrices(params: SubscribePricesParams): () => void {\n const { symbols, callback, includeOrderBook } = params;\n\n this.#deps.debugLogger.log('[MYXProvider] Setting up price subscription', {\n symbols: symbols.length,\n includeOrderBook,\n });\n\n // Map symbols to pool IDs\n const poolIds: string[] = [];\n for (const pool of this.#poolsCache) {\n const symbol = pool.baseSymbol || pool.poolId;\n if (symbols.includes(symbol)) {\n poolIds.push(pool.poolId);\n }\n }\n\n if (poolIds.length === 0) {\n this.#deps.debugLogger.log(\n '[MYXProvider] subscribeToPrices: No pool IDs found. Ensure initialize() has been called.',\n { symbols },\n );\n setTimeout(() => params.callback([]), 0);\n return () => {\n /* noop */\n };\n }\n\n // Start price polling\n this.#clientService.startPricePolling(poolIds, (tickers) => {\n // Convert tickers to PriceUpdate format\n const updates: PriceUpdate[] = tickers.map((ticker) => {\n const symbol = this.#poolSymbolMap.get(ticker.poolId) ?? ticker.poolId;\n const { price, change24h } = this.#getAdaptedPrice(ticker);\n\n return {\n symbol,\n price,\n timestamp: Date.now(),\n percentChange24h: change24h.toFixed(2),\n providerId: 'myx',\n };\n });\n\n callback(updates);\n });\n\n // Return unsubscribe function\n return () => {\n this.#deps.debugLogger.log('[MYXProvider] Unsubscribing from prices');\n this.#clientService.stopPricePolling();\n };\n }\n\n #getAdaptedPrice(ticker: MYXTicker): {\n price: string;\n change24h: number;\n } {\n return adaptPriceFromMYX(ticker);\n }\n\n // ============================================================================\n // Asset Routes (Stage 1 - Stubbed)\n // ============================================================================\n\n getDepositRoutes(_params?: GetSupportedPathsParams): AssetRoute[] {\n // Stage 1: No deposit support\n return [];\n }\n\n getWithdrawalRoutes(_params?: GetSupportedPathsParams): AssetRoute[] {\n // Stage 1: No withdrawal support\n return [];\n }\n\n // ============================================================================\n // Trading Operations (Stage 1 - All Stubbed)\n // ============================================================================\n\n async placeOrder(_params: OrderParams): Promise<OrderResult> {\n return {\n success: false,\n error: MYX_NOT_SUPPORTED_ERROR,\n };\n }\n\n async editOrder(_params: EditOrderParams): Promise<OrderResult> {\n return {\n success: false,\n error: MYX_NOT_SUPPORTED_ERROR,\n };\n }\n\n async cancelOrder(_params: CancelOrderParams): Promise<CancelOrderResult> {\n return {\n success: false,\n error: MYX_NOT_SUPPORTED_ERROR,\n };\n }\n\n async cancelOrders(\n _params: BatchCancelOrdersParams,\n ): Promise<CancelOrdersResult> {\n return {\n success: false,\n successCount: 0,\n failureCount: 0,\n results: [],\n };\n }\n\n async closePosition(_params: ClosePositionParams): Promise<OrderResult> {\n return {\n success: false,\n error: MYX_NOT_SUPPORTED_ERROR,\n };\n }\n\n async closePositions(\n _params: ClosePositionsParams,\n ): Promise<ClosePositionsResult> {\n return {\n success: false,\n successCount: 0,\n failureCount: 0,\n results: [],\n };\n }\n\n async updatePositionTPSL(\n _params: UpdatePositionTPSLParams,\n ): Promise<OrderResult> {\n return {\n success: false,\n error: MYX_NOT_SUPPORTED_ERROR,\n };\n }\n\n async updateMargin(_params: {\n symbol: string;\n amount: string;\n isAdd: boolean;\n }): Promise<MarginResult> {\n return {\n success: false,\n error: MYX_NOT_SUPPORTED_ERROR,\n };\n }\n\n async withdraw(_params: WithdrawParams): Promise<WithdrawResult> {\n return {\n success: false,\n error: MYX_NOT_SUPPORTED_ERROR,\n };\n }\n\n // ============================================================================\n // Account Operations (Stage 1 - Empty Returns)\n // ============================================================================\n\n async getPositions(_params?: GetPositionsParams): Promise<Position[]> {\n // Stage 1: No position tracking\n return [];\n }\n\n async getAccountState(\n _params?: GetAccountStateParams,\n ): Promise<AccountState> {\n // Stage 1: Empty account state\n return {\n availableBalance: '0',\n totalBalance: '0',\n marginUsed: '0',\n unrealizedPnl: '0',\n returnOnEquity: '0',\n };\n }\n\n async getOrders(_params?: GetOrdersParams): Promise<Order[]> {\n // Stage 1: No order tracking\n return [];\n }\n\n async getOpenOrders(_params?: GetOrdersParams): Promise<Order[]> {\n // Stage 1: No order tracking\n return [];\n }\n\n async getOrderFills(_params?: GetOrderFillsParams): Promise<OrderFill[]> {\n // Stage 1: No fill tracking\n return [];\n }\n\n async getOrFetchFills(_params?: GetOrFetchFillsParams): Promise<OrderFill[]> {\n // Stage 1: No fill tracking\n return [];\n }\n\n async getFunding(_params?: GetFundingParams): Promise<Funding[]> {\n // Stage 1: No funding tracking\n return [];\n }\n\n async getHistoricalPortfolio(\n _params?: GetHistoricalPortfolioParams,\n ): Promise<HistoricalPortfolioResult> {\n return {\n accountValue1dAgo: '0',\n timestamp: Date.now(),\n };\n }\n\n async getUserNonFundingLedgerUpdates(_params?: {\n accountId?: string;\n startTime?: number;\n endTime?: number;\n }): Promise<RawLedgerUpdate[]> {\n return [];\n }\n\n async getUserHistory(_params?: {\n accountId?: CaipAccountId;\n startTime?: number;\n endTime?: number;\n }): Promise<UserHistoryItem[]> {\n return [];\n }\n\n // ============================================================================\n // Validation Operations (Stage 1 - All Invalid)\n // ============================================================================\n\n async validateDeposit(\n _params: DepositParams,\n ): Promise<{ isValid: boolean; error?: string }> {\n return { isValid: false, error: MYX_NOT_SUPPORTED_ERROR };\n }\n\n async validateOrder(\n _params: OrderParams,\n ): Promise<{ isValid: boolean; error?: string }> {\n return { isValid: false, error: MYX_NOT_SUPPORTED_ERROR };\n }\n\n async validateClosePosition(\n _params: ClosePositionParams,\n ): Promise<{ isValid: boolean; error?: string }> {\n return { isValid: false, error: MYX_NOT_SUPPORTED_ERROR };\n }\n\n async validateWithdrawal(\n _params: WithdrawParams,\n ): Promise<{ isValid: boolean; error?: string }> {\n return { isValid: false, error: MYX_NOT_SUPPORTED_ERROR };\n }\n\n // ============================================================================\n // Protocol Calculations (Stage 1 - Default Values)\n // ============================================================================\n\n async calculateLiquidationPrice(\n _params: LiquidationPriceParams,\n ): Promise<string> {\n return '0';\n }\n\n async calculateMaintenanceMargin(\n _params: MaintenanceMarginParams,\n ): Promise<number> {\n return 0;\n }\n\n async getMaxLeverage(_asset: string): Promise<number> {\n return 100; // MYX default max leverage\n }\n\n async calculateFees(\n _params: FeeCalculationParams,\n ): Promise<FeeCalculationResult> {\n // MYX fee structure (placeholder values)\n return {\n feeRate: 0.0005, // 0.05% total fee rate\n protocolFeeRate: 0.0005, // Protocol taker fee\n };\n }\n\n // ============================================================================\n // Subscriptions (Stage 1 - No-op)\n // ============================================================================\n\n subscribeToPositions(params: SubscribePositionsParams): () => void {\n // Stage 1: No position tracking - immediately call back with empty array\n // to signal loading is complete (no data to show)\n setTimeout(() => params.callback([]), 0);\n return () => {\n /* noop */\n };\n }\n\n subscribeToOrderFills(params: SubscribeOrderFillsParams): () => void {\n // Stage 1: No fill tracking - immediately call back with empty array\n setTimeout(() => params.callback([]), 0);\n return () => {\n /* noop */\n };\n }\n\n subscribeToOrders(params: SubscribeOrdersParams): () => void {\n // Stage 1: No order tracking - immediately call back with empty array\n setTimeout(() => params.callback([]), 0);\n return () => {\n /* noop */\n };\n }\n\n subscribeToAccount(params: SubscribeAccountParams): () => void {\n // Stage 1: Empty account state - immediately call back\n setTimeout(\n () =>\n params.callback({\n availableBalance: '0',\n totalBalance: '0',\n marginUsed: '0',\n unrealizedPnl: '0',\n returnOnEquity: '0',\n }),\n 0,\n );\n return () => {\n /* noop */\n };\n }\n\n subscribeToOICaps(params: SubscribeOICapsParams): () => void {\n // Stage 1: No OI caps - immediately call back with empty array\n // (matches HyperLiquid pattern which calls callback with cached data)\n setTimeout(() => params.callback([]), 0);\n return () => {\n /* noop */\n };\n }\n\n subscribeToCandles(params: SubscribeCandlesParams): () => void {\n // Stage 1: No candle data - immediately call back with empty candles\n // (matches HyperLiquid pattern which calls callback after initial fetch)\n setTimeout(\n () =>\n params.callback({\n symbol: params.symbol,\n interval: params.interval,\n candles: [],\n }),\n 0,\n );\n return () => {\n /* noop */\n };\n }\n\n subscribeToOrderBook(params: SubscribeOrderBookParams): () => void {\n // Stage 1: No order book - immediately call back with empty data\n setTimeout(\n () =>\n params.callback({\n bids: [],\n asks: [],\n spread: '0',\n spreadPercentage: '0',\n midPrice: '0',\n lastUpdated: Date.now(),\n maxTotal: '0',\n }),\n 0,\n );\n return () => {\n /* noop */\n };\n }\n\n setLiveDataConfig(_config: Partial<LiveDataConfig>): void {\n // Stage 1: No-op\n }\n\n // ============================================================================\n // Connection State (Stage 1 - REST Only)\n // ============================================================================\n\n getWebSocketConnectionState(): WebSocketConnectionState {\n // Stage 1: No WebSocket, report as connected (REST is always available)\n return WebSocketConnectionState.Connected;\n }\n\n subscribeToConnectionState(\n _listener: (\n state: WebSocketConnectionState,\n reconnectionAttempt: number,\n ) => void,\n ): () => void {\n // Stage 1: No WebSocket, no connection state changes\n return () => {\n /* noop */\n };\n }\n\n async reconnect(): Promise<void> {\n // Stage 1: No WebSocket to reconnect\n this.#deps.debugLogger.log('[MYXProvider] reconnect() is no-op in Stage 1');\n }\n\n // ============================================================================\n // Block Explorer\n // ============================================================================\n\n getBlockExplorerUrl(address?: string): string {\n const baseUrl = this.#isTestnet\n ? MYX_TESTNET_EXPLORER_URL\n : MYX_BLOCK_EXPLORER_URL;\n\n return address ? `${baseUrl}/address/${address}` : baseUrl;\n }\n\n // ============================================================================\n // Fee Discount (Stage 1 - No-op)\n // ============================================================================\n\n setUserFeeDiscount(_discountBips: number | undefined): void {\n // Stage 1: No fee discount support\n }\n\n // ============================================================================\n // HIP-3 Operations (N/A for MYX)\n // ============================================================================\n\n async getAvailableDexs(): Promise<string[]> {\n // MYX doesn't have HIP-3 equivalent\n return [];\n }\n}\n"]}