@yuants/vendor-okx 0.16.9 → 0.17.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (90) hide show
  1. package/dist/account.js +198 -0
  2. package/dist/account.js.map +1 -0
  3. package/dist/api.js +463 -0
  4. package/dist/api.js.map +1 -0
  5. package/dist/cli.js +3 -0
  6. package/dist/cli.js.map +1 -0
  7. package/dist/cluster.js +80 -0
  8. package/dist/cluster.js.map +1 -0
  9. package/dist/extension.js +89 -0
  10. package/dist/extension.js.map +1 -0
  11. package/dist/index.js +7 -0
  12. package/dist/index.js.map +1 -0
  13. package/dist/interest_rate.js +133 -0
  14. package/dist/interest_rate.js.map +1 -0
  15. package/dist/legacy_index.js +554 -0
  16. package/dist/legacy_index.js.map +1 -0
  17. package/dist/logger.js +91 -0
  18. package/dist/logger.js.map +1 -0
  19. package/dist/ohlc.js +171 -0
  20. package/dist/ohlc.js.map +1 -0
  21. package/dist/order.js +96 -0
  22. package/dist/order.js.map +1 -0
  23. package/dist/product.js +85 -0
  24. package/dist/product.js.map +1 -0
  25. package/dist/quote.js +58 -0
  26. package/dist/quote.js.map +1 -0
  27. package/dist/vendor-okx.d.ts +1 -0
  28. package/dist/websocket.js +80 -0
  29. package/dist/websocket.js.map +1 -0
  30. package/lib/account.d.ts +102 -0
  31. package/lib/account.d.ts.map +1 -0
  32. package/lib/account.js +201 -0
  33. package/lib/account.js.map +1 -0
  34. package/lib/api.d.ts +1401 -0
  35. package/lib/api.d.ts.map +1 -0
  36. package/lib/api.js +470 -0
  37. package/lib/api.js.map +1 -0
  38. package/lib/cli.d.ts +3 -0
  39. package/lib/cli.d.ts.map +1 -0
  40. package/lib/cli.js +5 -0
  41. package/lib/cli.js.map +1 -0
  42. package/lib/cluster.d.ts +2 -0
  43. package/lib/cluster.d.ts.map +1 -0
  44. package/lib/cluster.js +108 -0
  45. package/lib/cluster.js.map +1 -0
  46. package/lib/extension.d.ts +4 -0
  47. package/lib/extension.d.ts.map +1 -0
  48. package/lib/extension.js +91 -0
  49. package/lib/extension.js.map +1 -0
  50. package/lib/index.d.ts +7 -0
  51. package/lib/index.d.ts.map +1 -0
  52. package/lib/index.js +9 -0
  53. package/lib/index.js.map +1 -0
  54. package/lib/interest_rate.d.ts +2 -0
  55. package/lib/interest_rate.d.ts.map +1 -0
  56. package/lib/interest_rate.js +135 -0
  57. package/lib/interest_rate.js.map +1 -0
  58. package/lib/legacy_index.d.ts +2 -0
  59. package/lib/legacy_index.d.ts.map +1 -0
  60. package/lib/legacy_index.js +556 -0
  61. package/lib/legacy_index.js.map +1 -0
  62. package/lib/logger.d.ts +21 -0
  63. package/lib/logger.d.ts.map +1 -0
  64. package/lib/logger.js +98 -0
  65. package/lib/logger.js.map +1 -0
  66. package/lib/ohlc.d.ts +2 -0
  67. package/lib/ohlc.d.ts.map +1 -0
  68. package/lib/ohlc.js +173 -0
  69. package/lib/ohlc.js.map +1 -0
  70. package/lib/order.d.ts +4 -0
  71. package/lib/order.d.ts.map +1 -0
  72. package/lib/order.js +99 -0
  73. package/lib/order.js.map +1 -0
  74. package/lib/product.d.ts +6 -0
  75. package/lib/product.d.ts.map +1 -0
  76. package/lib/product.js +88 -0
  77. package/lib/product.js.map +1 -0
  78. package/lib/quote.d.ts +42 -0
  79. package/lib/quote.d.ts.map +1 -0
  80. package/lib/quote.js +61 -0
  81. package/lib/quote.js.map +1 -0
  82. package/lib/websocket.d.ts +14 -0
  83. package/lib/websocket.d.ts.map +1 -0
  84. package/lib/websocket.js +83 -0
  85. package/lib/websocket.js.map +1 -0
  86. package/package.json +9 -4
  87. package/temp/image-tag +1 -0
  88. package/temp/package-deps.json +42 -0
  89. package/temp/vendor-okx.api.json +177 -0
  90. package/temp/vendor-okx.api.md +9 -0
@@ -0,0 +1,21 @@
1
+ /**
2
+ * 日志级别
3
+ */
4
+ export declare type LogLevel = 'info' | 'warn' | 'error' | 'debug';
5
+ /**
6
+ * 集群模式下的日志管理器
7
+ * 在worker进程中将日志发送到主进程,在主进程中直接使用console
8
+ */
9
+ declare class ClusterLogger {
10
+ private isWorker;
11
+ private sendToMaster;
12
+ info(message: string, ...args: any[]): void;
13
+ warn(message: string, ...args: any[]): void;
14
+ error(message: string, ...args: any[]): void;
15
+ debug(message: string, ...args: any[]): void;
16
+ log(message: string, ...args: any[]): void;
17
+ }
18
+ export declare const logger: ClusterLogger;
19
+ export declare const overrideConsole: () => () => void;
20
+ export {};
21
+ //# sourceMappingURL=logger.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logger.d.ts","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":"AAGA;;GAEG;AACH,oBAAY,QAAQ,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,OAAO,CAAC;AAE3D;;;GAGG;AACH,cAAM,aAAa;IACjB,OAAO,CAAC,QAAQ,CAAsB;IAEtC,OAAO,CAAC,YAAY;IA6CpB,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE;IAIpC,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE;IAIpC,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE;IAIrC,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE;IAIrC,GAAG,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE;CAGpC;AAGD,eAAO,MAAM,MAAM,eAAsB,CAAC;AAG1C,eAAO,MAAM,eAAe,kBAuB3B,CAAC"}
package/lib/logger.js ADDED
@@ -0,0 +1,98 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.overrideConsole = exports.logger = void 0;
7
+ const cluster_1 = __importDefault(require("cluster"));
8
+ /**
9
+ * 集群模式下的日志管理器
10
+ * 在worker进程中将日志发送到主进程,在主进程中直接使用console
11
+ */
12
+ class ClusterLogger {
13
+ constructor() {
14
+ this.isWorker = !cluster_1.default.isPrimary;
15
+ }
16
+ sendToMaster(level, message, ...args) {
17
+ if (this.isWorker && process.send) {
18
+ const formattedMessage = args.length > 0
19
+ ? `${message} ${args
20
+ .map((arg) => (typeof arg === 'object' ? JSON.stringify(arg) : String(arg)))
21
+ .join(' ')}`
22
+ : message;
23
+ process.send({
24
+ type: 'log',
25
+ level,
26
+ message: `[PID:${process.pid}] ${formattedMessage}`,
27
+ pid: process.pid,
28
+ label: process.env.WORKER_LABEL || 'unknown',
29
+ timestamp: Date.now(),
30
+ });
31
+ }
32
+ else {
33
+ // 在主进程中直接输出
34
+ const formattedMessage = args.length > 0
35
+ ? `${message} ${args
36
+ .map((arg) => (typeof arg === 'object' ? JSON.stringify(arg) : String(arg)))
37
+ .join(' ')}`
38
+ : message;
39
+ const logMessage = `[PID:${process.pid}] ${formattedMessage}`;
40
+ // 控制台输出
41
+ switch (level) {
42
+ case 'error':
43
+ console.error(logMessage);
44
+ break;
45
+ case 'warn':
46
+ console.warn(logMessage);
47
+ break;
48
+ case 'debug':
49
+ console.debug(logMessage);
50
+ break;
51
+ default:
52
+ console.log(logMessage);
53
+ }
54
+ }
55
+ }
56
+ info(message, ...args) {
57
+ this.sendToMaster('info', message, ...args);
58
+ }
59
+ warn(message, ...args) {
60
+ this.sendToMaster('warn', message, ...args);
61
+ }
62
+ error(message, ...args) {
63
+ this.sendToMaster('error', message, ...args);
64
+ }
65
+ debug(message, ...args) {
66
+ this.sendToMaster('debug', message, ...args);
67
+ }
68
+ log(message, ...args) {
69
+ this.info(message, ...args);
70
+ }
71
+ }
72
+ // 默认配置的logger实例
73
+ exports.logger = new ClusterLogger();
74
+ // 猛踩油门
75
+ const overrideConsole = () => {
76
+ if (!cluster_1.default.isPrimary) {
77
+ const originalConsole = {
78
+ log: console.log,
79
+ info: console.info,
80
+ warn: console.warn,
81
+ error: console.error,
82
+ };
83
+ console.log = (...args) => exports.logger.info(args.join(' '));
84
+ console.info = (...args) => exports.logger.info(args.join(' '));
85
+ console.warn = (...args) => exports.logger.warn(args.join(' '));
86
+ console.error = (...args) => exports.logger.error(args.join(' '));
87
+ // 提供恢复方法
88
+ return () => {
89
+ console.log = originalConsole.log;
90
+ console.info = originalConsole.info;
91
+ console.warn = originalConsole.warn;
92
+ console.error = originalConsole.error;
93
+ };
94
+ }
95
+ return () => { }; // 主进程中返回空函数
96
+ };
97
+ exports.overrideConsole = overrideConsole;
98
+ //# sourceMappingURL=logger.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logger.js","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":"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","sourcesContent":["import cluster from 'cluster';\nimport { formatTime } from '@yuants/utils';\n\n/**\n * 日志级别\n */\nexport type LogLevel = 'info' | 'warn' | 'error' | 'debug';\n\n/**\n * 集群模式下的日志管理器\n * 在worker进程中将日志发送到主进程,在主进程中直接使用console\n */\nclass ClusterLogger {\n private isWorker = !cluster.isPrimary;\n\n private sendToMaster(level: LogLevel, message: string, ...args: any[]) {\n if (this.isWorker && process.send) {\n const formattedMessage =\n args.length > 0\n ? `${message} ${args\n .map((arg) => (typeof arg === 'object' ? JSON.stringify(arg) : String(arg)))\n .join(' ')}`\n : message;\n\n process.send({\n type: 'log',\n level,\n message: `[PID:${process.pid}] ${formattedMessage}`,\n pid: process.pid,\n label: process.env.WORKER_LABEL || 'unknown',\n timestamp: Date.now(),\n });\n } else {\n // 在主进程中直接输出\n const formattedMessage =\n args.length > 0\n ? `${message} ${args\n .map((arg) => (typeof arg === 'object' ? JSON.stringify(arg) : String(arg)))\n .join(' ')}`\n : message;\n\n const logMessage = `[PID:${process.pid}] ${formattedMessage}`;\n\n // 控制台输出\n switch (level) {\n case 'error':\n console.error(logMessage);\n break;\n case 'warn':\n console.warn(logMessage);\n break;\n case 'debug':\n console.debug(logMessage);\n break;\n default:\n console.log(logMessage);\n }\n }\n }\n\n info(message: string, ...args: any[]) {\n this.sendToMaster('info', message, ...args);\n }\n\n warn(message: string, ...args: any[]) {\n this.sendToMaster('warn', message, ...args);\n }\n\n error(message: string, ...args: any[]) {\n this.sendToMaster('error', message, ...args);\n }\n\n debug(message: string, ...args: any[]) {\n this.sendToMaster('debug', message, ...args);\n }\n\n log(message: string, ...args: any[]) {\n this.info(message, ...args);\n }\n}\n\n// 默认配置的logger实例\nexport const logger = new ClusterLogger();\n\n// 猛踩油门\nexport const overrideConsole = () => {\n if (!cluster.isPrimary) {\n const originalConsole = {\n log: console.log,\n info: console.info,\n warn: console.warn,\n error: console.error,\n };\n\n console.log = (...args: any[]) => logger.info(args.join(' '));\n console.info = (...args: any[]) => logger.info(args.join(' '));\n console.warn = (...args: any[]) => logger.warn(args.join(' '));\n console.error = (...args: any[]) => logger.error(args.join(' '));\n\n // 提供恢复方法\n return () => {\n console.log = originalConsole.log;\n console.info = originalConsole.info;\n console.warn = originalConsole.warn;\n console.error = originalConsole.error;\n };\n }\n return () => {}; // 主进程中返回空函数\n};\n"]}
package/lib/ohlc.d.ts ADDED
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=ohlc.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ohlc.d.ts","sourceRoot":"","sources":["../src/ohlc.ts"],"names":[],"mappings":""}
package/lib/ohlc.js ADDED
@@ -0,0 +1,173 @@
1
+ "use strict";
2
+ var __await = (this && this.__await) || function (v) { return this instanceof __await ? (this.v = v, this) : new __await(v); }
3
+ var __asyncGenerator = (this && this.__asyncGenerator) || function (thisArg, _arguments, generator) {
4
+ if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined.");
5
+ var g = generator.apply(thisArg, _arguments || []), i, q = [];
6
+ return i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i;
7
+ function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }
8
+ function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }
9
+ function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }
10
+ function fulfill(value) { resume("next", value); }
11
+ function reject(value) { resume("throw", value); }
12
+ function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }
13
+ };
14
+ Object.defineProperty(exports, "__esModule", { value: true });
15
+ const data_series_1 = require("@yuants/data-series");
16
+ const protocol_1 = require("@yuants/protocol");
17
+ const utils_1 = require("@yuants/utils");
18
+ const rxjs_1 = require("rxjs");
19
+ const api_1 = require("./api");
20
+ const websocket_1 = require("./websocket");
21
+ const sql_1 = require("@yuants/sql");
22
+ // 时间粒度,默认值1m
23
+ // 如 [1m/3m/5m/15m/30m/1H/2H/4H]
24
+ // 香港时间开盘价k线:[6H/12H/1D/1W/1M]
25
+ // UTC时间开盘价k线:[6Hutc/12Hutc/1Dutc/1Wutc/1Mutc]
26
+ const DURATION_TO_OKX_BAR_TYPE = {
27
+ PT1M: '1m',
28
+ PT3M: '3m',
29
+ PT5M: '5m',
30
+ PT15M: '15m',
31
+ PT30M: '30m',
32
+ PT1H: '1H',
33
+ PT2H: '2H',
34
+ PT4H: '4H',
35
+ PT6H: '6H',
36
+ PT12H: '12H',
37
+ P1D: '1D',
38
+ P1W: '1W',
39
+ P1M: '1M',
40
+ };
41
+ const DURATION_TO_OKX_CANDLE_TYPE = {
42
+ PT1M: 'mark-price-candle1m',
43
+ PT3M: 'mark-price-candle3m',
44
+ PT5M: 'mark-price-candle5m',
45
+ PT15M: 'mark-price-candle15m',
46
+ PT30M: 'mark-price-candle30m',
47
+ PT1H: 'mark-price-candle1H',
48
+ PT2H: 'mark-price-candle2H',
49
+ PT4H: 'mark-price-candle4H',
50
+ PT6H: 'mark-price-candle6H',
51
+ PT12H: 'mark-price-candle12H',
52
+ P1D: 'mark-price-candle1D',
53
+ P1W: 'mark-price-candle1W',
54
+ P1M: 'mark-price-candle1M',
55
+ };
56
+ (0, data_series_1.createSeriesProvider)(protocol_1.Terminal.fromNodeEnv(), {
57
+ tableName: 'ohlc',
58
+ series_id_prefix_parts: ['OKX'],
59
+ reversed: true,
60
+ serviceOptions: { concurrent: 1 },
61
+ queryFn: function ({ series_id, ended_at }) {
62
+ return __asyncGenerator(this, arguments, function* () {
63
+ const [datasource_id, product_id, duration] = (0, utils_1.decodePath)(series_id);
64
+ const offset = (0, utils_1.convertDurationToOffset)(duration);
65
+ if (!datasource_id) {
66
+ throw 'datasource_id is required';
67
+ }
68
+ if (!product_id) {
69
+ throw 'product_id is required';
70
+ }
71
+ if (!offset) {
72
+ throw 'duration is invalid';
73
+ }
74
+ const [instType, instId] = (0, utils_1.decodePath)(product_id);
75
+ if (!instId) {
76
+ throw `invalid product_id: ${product_id}`;
77
+ }
78
+ const bar = DURATION_TO_OKX_BAR_TYPE[duration];
79
+ if (!bar) {
80
+ throw `unsupported duration: ${duration}`;
81
+ }
82
+ let currentStartTime = ended_at;
83
+ while (true) {
84
+ // 向前翻页,时间降序,不含 after 时间点
85
+ const res = yield __await(api_1.client.getHistoryCandles({
86
+ instId,
87
+ bar,
88
+ after: `${currentStartTime}`,
89
+ limit: '100',
90
+ }));
91
+ if (res.code !== '0') {
92
+ throw `API failed: ${res.code} ${res.msg}`;
93
+ }
94
+ if (res.data.length === 0)
95
+ break;
96
+ currentStartTime = +res.data[res.data.length - 1][0];
97
+ const data = res.data.map((x) => ({
98
+ series_id,
99
+ datasource_id,
100
+ product_id,
101
+ duration,
102
+ created_at: (0, utils_1.formatTime)(+x[0]),
103
+ closed_at: (0, utils_1.formatTime)(+x[0] + offset),
104
+ open: x[1],
105
+ high: x[2],
106
+ low: x[3],
107
+ close: x[4],
108
+ volume: x[5],
109
+ open_interest: '0',
110
+ }));
111
+ yield yield __await(data);
112
+ yield __await((0, rxjs_1.firstValueFrom)((0, rxjs_1.timer)(1000)));
113
+ }
114
+ });
115
+ },
116
+ });
117
+ protocol_1.Terminal.fromNodeEnv().channel.publishChannel('ohlc', { pattern: `^OKX/` }, (series_id) => {
118
+ // const [] = decodePath(args)
119
+ const [datasource_id, product_id, duration] = (0, utils_1.decodePath)(series_id);
120
+ const [, instId] = (0, utils_1.decodePath)(product_id);
121
+ const offset = (0, utils_1.convertDurationToOffset)(duration);
122
+ if (!datasource_id) {
123
+ throw 'datasource_id is required';
124
+ }
125
+ if (!product_id) {
126
+ throw 'product_id is required';
127
+ }
128
+ if (!offset) {
129
+ throw 'duration is invalid';
130
+ }
131
+ const candleType = DURATION_TO_OKX_CANDLE_TYPE[duration];
132
+ console.info((0, utils_1.formatTime)(Date.now()), `subscribe`, series_id);
133
+ return new rxjs_1.Observable((subscriber) => {
134
+ console.info((0, utils_1.formatTime)(Date.now()), `subscribe`, candleType, instId);
135
+ websocket_1.okxBusinessWsClient.subscribe(candleType, instId, async (data) => {
136
+ const created_at = Number(data[0]);
137
+ const closed_at = created_at + offset;
138
+ const open = data[1];
139
+ const high = data[2];
140
+ const low = data[3];
141
+ const close = data[4];
142
+ if (isNaN(closed_at)) {
143
+ return;
144
+ }
145
+ console.info((0, utils_1.formatTime)(Date.now()), `insertData`, data);
146
+ const cancelData = {
147
+ closed_at: (0, utils_1.formatTime)(closed_at),
148
+ created_at: (0, utils_1.formatTime)(created_at),
149
+ open,
150
+ high,
151
+ low,
152
+ close,
153
+ series_id,
154
+ datasource_id,
155
+ duration,
156
+ product_id,
157
+ volume: '0',
158
+ open_interest: '0',
159
+ };
160
+ subscriber.next(cancelData);
161
+ });
162
+ return () => {
163
+ websocket_1.okxBusinessWsClient.unsubscribe(candleType, instId);
164
+ };
165
+ }).pipe((0, sql_1.writeToSQL)({
166
+ tableName: 'ohlc',
167
+ conflictKeys: ['created_at', 'series_id'],
168
+ writeInterval: 1000,
169
+ terminal: protocol_1.Terminal.fromNodeEnv(),
170
+ keyFn: (x) => (0, utils_1.encodePath)(x.created_at, x.series_id),
171
+ }));
172
+ });
173
+ //# sourceMappingURL=ohlc.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ohlc.js","sourceRoot":"","sources":["../src/ohlc.ts"],"names":[],"mappings":"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","sourcesContent":["import { IOHLC } from '@yuants/data-ohlc';\nimport { createSeriesProvider } from '@yuants/data-series';\nimport { Terminal } from '@yuants/protocol';\nimport { convertDurationToOffset, decodePath, encodePath, formatTime } from '@yuants/utils';\nimport { firstValueFrom, Observable, of, timer } from 'rxjs';\nimport { client } from './api';\nimport { okxBusinessWsClient } from './websocket';\nimport { buildInsertManyIntoTableSQL, requestSQL, writeToSQL } from '@yuants/sql';\n\n// 时间粒度,默认值1m\n// 如 [1m/3m/5m/15m/30m/1H/2H/4H]\n// 香港时间开盘价k线:[6H/12H/1D/1W/1M]\n// UTC时间开盘价k线:[6Hutc/12Hutc/1Dutc/1Wutc/1Mutc]\n\nconst DURATION_TO_OKX_BAR_TYPE: Record<string, string> = {\n PT1M: '1m',\n PT3M: '3m',\n PT5M: '5m',\n PT15M: '15m',\n PT30M: '30m',\n\n PT1H: '1H',\n PT2H: '2H',\n PT4H: '4H',\n PT6H: '6H',\n PT12H: '12H',\n\n P1D: '1D',\n P1W: '1W',\n P1M: '1M',\n};\n\nconst DURATION_TO_OKX_CANDLE_TYPE: Record<string, string> = {\n PT1M: 'mark-price-candle1m',\n PT3M: 'mark-price-candle3m',\n PT5M: 'mark-price-candle5m',\n PT15M: 'mark-price-candle15m',\n PT30M: 'mark-price-candle30m',\n\n PT1H: 'mark-price-candle1H',\n PT2H: 'mark-price-candle2H',\n PT4H: 'mark-price-candle4H',\n PT6H: 'mark-price-candle6H',\n PT12H: 'mark-price-candle12H',\n\n P1D: 'mark-price-candle1D',\n P1W: 'mark-price-candle1W',\n P1M: 'mark-price-candle1M',\n};\n\ncreateSeriesProvider<IOHLC>(Terminal.fromNodeEnv(), {\n tableName: 'ohlc',\n series_id_prefix_parts: ['OKX'],\n reversed: true,\n serviceOptions: { concurrent: 1 },\n queryFn: async function* ({ series_id, ended_at }) {\n const [datasource_id, product_id, duration] = decodePath(series_id);\n const offset = convertDurationToOffset(duration);\n if (!datasource_id) {\n throw 'datasource_id is required';\n }\n if (!product_id) {\n throw 'product_id is required';\n }\n if (!offset) {\n throw 'duration is invalid';\n }\n const [instType, instId] = decodePath(product_id);\n if (!instId) {\n throw `invalid product_id: ${product_id}`;\n }\n\n const bar = DURATION_TO_OKX_BAR_TYPE[duration];\n if (!bar) {\n throw `unsupported duration: ${duration}`;\n }\n\n let currentStartTime = ended_at;\n\n while (true) {\n // 向前翻页,时间降序,不含 after 时间点\n const res = await client.getHistoryCandles({\n instId,\n bar,\n after: `${currentStartTime}`,\n limit: '100',\n });\n if (res.code !== '0') {\n throw `API failed: ${res.code} ${res.msg}`;\n }\n if (res.data.length === 0) break;\n currentStartTime = +res.data[res.data.length - 1][0];\n const data = res.data.map(\n (x): IOHLC => ({\n series_id,\n datasource_id,\n product_id,\n duration,\n created_at: formatTime(+x[0]),\n closed_at: formatTime(+x[0] + offset),\n open: x[1],\n high: x[2],\n low: x[3],\n close: x[4],\n volume: x[5],\n open_interest: '0',\n }),\n );\n yield data;\n await firstValueFrom(timer(1000));\n }\n },\n});\n\nTerminal.fromNodeEnv().channel.publishChannel('ohlc', { pattern: `^OKX/` }, (series_id) => {\n // const [] = decodePath(args)\n const [datasource_id, product_id, duration] = decodePath(series_id);\n const [, instId] = decodePath(product_id);\n const offset = convertDurationToOffset(duration);\n if (!datasource_id) {\n throw 'datasource_id is required';\n }\n if (!product_id) {\n throw 'product_id is required';\n }\n if (!offset) {\n throw 'duration is invalid';\n }\n const candleType = DURATION_TO_OKX_CANDLE_TYPE[duration];\n console.info(formatTime(Date.now()), `subscribe`, series_id);\n return new Observable<IOHLC>((subscriber) => {\n console.info(formatTime(Date.now()), `subscribe`, candleType, instId);\n okxBusinessWsClient.subscribe(candleType, instId, async (data: string[]) => {\n const created_at = Number(data[0]);\n const closed_at = created_at + offset;\n const open = data[1];\n const high = data[2];\n const low = data[3];\n const close = data[4];\n if (isNaN(closed_at)) {\n return;\n }\n console.info(formatTime(Date.now()), `insertData`, data);\n const cancelData: IOHLC = {\n closed_at: formatTime(closed_at),\n created_at: formatTime(created_at),\n open,\n high,\n low,\n close,\n series_id,\n datasource_id,\n duration,\n product_id,\n volume: '0',\n open_interest: '0',\n };\n subscriber.next(cancelData);\n });\n return () => {\n okxBusinessWsClient.unsubscribe(candleType, instId);\n };\n }).pipe(\n writeToSQL({\n tableName: 'ohlc',\n conflictKeys: ['created_at', 'series_id'],\n writeInterval: 1000,\n terminal: Terminal.fromNodeEnv(),\n keyFn: (x) => encodePath(x.created_at, x.series_id),\n }),\n );\n});\n"]}
package/lib/order.d.ts ADDED
@@ -0,0 +1,4 @@
1
+ import { IOrder } from '@yuants/data-order';
2
+ import { Subject } from 'rxjs';
3
+ export declare const order$: Subject<IOrder>;
4
+ //# sourceMappingURL=order.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"order.d.ts","sourceRoot":"","sources":["../src/order.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAI5C,OAAO,EAUL,OAAO,EAER,MAAM,MAAM,CAAC;AAId,eAAO,MAAM,MAAM,iBAAwB,CAAC"}
package/lib/order.js ADDED
@@ -0,0 +1,99 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.order$ = void 0;
4
+ const protocol_1 = require("@yuants/protocol");
5
+ const sql_1 = require("@yuants/sql");
6
+ const utils_1 = require("@yuants/utils");
7
+ const rxjs_1 = require("rxjs");
8
+ const account_1 = require("./account");
9
+ const api_1 = require("./api");
10
+ exports.order$ = new rxjs_1.Subject();
11
+ exports.order$
12
+ .pipe(
13
+ //
14
+ // mergeMap((x) => x),
15
+ (0, sql_1.writeToSQL)({
16
+ terminal: protocol_1.Terminal.fromNodeEnv(),
17
+ tableName: '"order"',
18
+ writeInterval: 1000,
19
+ keyFn: (order) => (0, utils_1.encodePath)(order.account_id, order.order_id),
20
+ columns: [
21
+ 'order_id',
22
+ 'account_id',
23
+ 'product_id',
24
+ 'position_id',
25
+ 'order_type',
26
+ 'order_direction',
27
+ 'volume',
28
+ 'submit_at',
29
+ 'updated_at',
30
+ 'filled_at',
31
+ 'price',
32
+ 'traded_volume',
33
+ 'traded_price',
34
+ 'order_status',
35
+ 'comment',
36
+ 'profit_correction',
37
+ 'real_profit',
38
+ 'inferred_base_currency_price',
39
+ ],
40
+ conflictKeys: ['account_id', 'order_id'],
41
+ }))
42
+ .subscribe();
43
+ const makeOrder = (x, account_id) => {
44
+ const order_type = x.ordType === 'market' ? 'MARKET' : x.ordType === 'limit' ? 'LIMIT' : 'UNKNOWN';
45
+ const order_direction = x.side === 'buy'
46
+ ? x.posSide === 'long'
47
+ ? 'OPEN_LONG'
48
+ : 'CLOSE_SHORT'
49
+ : x.posSide === 'short'
50
+ ? 'OPEN_SHORT'
51
+ : 'CLOSE_LONG';
52
+ const order_status = ['live', 'partially_filled'].includes(x.state)
53
+ ? 'ACCEPTED'
54
+ : x.state === 'filled'
55
+ ? 'TRADED'
56
+ : 'CANCELLED';
57
+ return {
58
+ order_id: x.clOrdId !== '' ? x.clOrdId : x.ordId,
59
+ account_id,
60
+ product_id: (0, utils_1.encodePath)(x.instType, x.instId),
61
+ submit_at: +x.cTime,
62
+ filled_at: +x.fillTime,
63
+ created_at: (0, utils_1.formatTime)(+x.cTime),
64
+ updated_at: (0, utils_1.formatTime)(+x.uTime),
65
+ order_type,
66
+ order_direction,
67
+ volume: +x.sz,
68
+ traded_volume: +x.accFillSz,
69
+ price: +x.px,
70
+ traded_price: +x.avgPx,
71
+ order_status,
72
+ };
73
+ };
74
+ (async () => {
75
+ const uid = await (0, rxjs_1.firstValueFrom)(account_1.accountUid$);
76
+ const TRADING_ACCOUNT_ID = `okx/${uid}/trading`;
77
+ const FUNDING_ACCOUNT_ID = `okx/${uid}/funding/USDT`;
78
+ const EARNING_ACCOUNT_ID = `okx/${uid}/earning/USDT`;
79
+ const swapHistoryOrders = (0, rxjs_1.defer)(() => api_1.client.getTradeOrdersHistory({ instType: 'SWAP' })).pipe((0, rxjs_1.repeat)({ delay: 1000 }), (0, rxjs_1.retry)({ delay: 1000 }), (0, rxjs_1.shareReplay)(1));
80
+ const swapPendingOrders = (0, rxjs_1.defer)(() => api_1.client.getTradeOrdersPending({ instType: 'SWAP' })).pipe((0, rxjs_1.repeat)({ delay: 1000 }), (0, rxjs_1.retry)({ delay: 1000 }), (0, rxjs_1.shareReplay)(1));
81
+ const ordersFromHistoryOrder$ = swapHistoryOrders.pipe(
82
+ //
83
+ (0, rxjs_1.mergeMap)((x) => (0, rxjs_1.from)(x.data || []).pipe(
84
+ //
85
+ (0, rxjs_1.map)((x) => makeOrder(x, TRADING_ACCOUNT_ID)))));
86
+ const ordersFromPendingOrder$ = swapPendingOrders.pipe(
87
+ //
88
+ (0, rxjs_1.mergeMap)((x) => (0, rxjs_1.from)(x.data || []).pipe(
89
+ //
90
+ (0, rxjs_1.map)((x) => makeOrder(x, TRADING_ACCOUNT_ID)))));
91
+ (0, rxjs_1.merge)(ordersFromHistoryOrder$, ordersFromPendingOrder$)
92
+ .pipe(
93
+ //
94
+ (0, rxjs_1.tap)((x) => {
95
+ exports.order$.next(x);
96
+ }))
97
+ .subscribe();
98
+ })();
99
+ //# sourceMappingURL=order.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"order.js","sourceRoot":"","sources":["../src/order.ts"],"names":[],"mappings":"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","sourcesContent":["import { IOrder } from '@yuants/data-order';\nimport { Terminal } from '@yuants/protocol';\nimport { writeToSQL } from '@yuants/sql';\nimport { encodePath, formatTime } from '@yuants/utils';\nimport {\n defer,\n firstValueFrom,\n from,\n map,\n merge,\n mergeMap,\n repeat,\n retry,\n shareReplay,\n Subject,\n tap,\n} from 'rxjs';\nimport { accountUid$ } from './account';\nimport { client } from './api';\n\nexport const order$ = new Subject<IOrder>();\n\norder$\n .pipe(\n //\n // mergeMap((x) => x),\n writeToSQL({\n terminal: Terminal.fromNodeEnv(),\n tableName: '\"order\"',\n writeInterval: 1000,\n keyFn: (order) => encodePath(order.account_id, order.order_id),\n columns: [\n 'order_id',\n 'account_id',\n 'product_id',\n 'position_id',\n 'order_type',\n 'order_direction',\n 'volume',\n 'submit_at',\n 'updated_at',\n 'filled_at',\n 'price',\n 'traded_volume',\n 'traded_price',\n 'order_status',\n 'comment',\n 'profit_correction',\n 'real_profit',\n 'inferred_base_currency_price',\n ],\n conflictKeys: ['account_id', 'order_id'],\n }),\n )\n .subscribe();\n\nconst makeOrder = (\n x: {\n ordType: string;\n side: string;\n posSide: string;\n instType: string;\n instId: string;\n cTime: string;\n uTime: string;\n fillTime: string;\n sz: string;\n accFillSz: string;\n px: string;\n avgPx: string;\n state: string;\n clOrdId: string;\n ordId: string;\n },\n account_id: string,\n): IOrder => {\n const order_type = x.ordType === 'market' ? 'MARKET' : x.ordType === 'limit' ? 'LIMIT' : 'UNKNOWN';\n const order_direction =\n x.side === 'buy'\n ? x.posSide === 'long'\n ? 'OPEN_LONG'\n : 'CLOSE_SHORT'\n : x.posSide === 'short'\n ? 'OPEN_SHORT'\n : 'CLOSE_LONG';\n const order_status = ['live', 'partially_filled'].includes(x.state)\n ? 'ACCEPTED'\n : x.state === 'filled'\n ? 'TRADED'\n : 'CANCELLED';\n return {\n order_id: x.clOrdId !== '' ? x.clOrdId : x.ordId,\n account_id,\n product_id: encodePath(x.instType, x.instId),\n submit_at: +x.cTime,\n filled_at: +x.fillTime,\n created_at: formatTime(+x.cTime),\n updated_at: formatTime(+x.uTime),\n order_type,\n order_direction,\n volume: +x.sz,\n traded_volume: +x.accFillSz,\n price: +x.px,\n traded_price: +x.avgPx,\n order_status,\n };\n};\n\n(async () => {\n const uid = await firstValueFrom(accountUid$);\n\n const TRADING_ACCOUNT_ID = `okx/${uid}/trading`;\n const FUNDING_ACCOUNT_ID = `okx/${uid}/funding/USDT`;\n const EARNING_ACCOUNT_ID = `okx/${uid}/earning/USDT`;\n\n const swapHistoryOrders = defer(() => client.getTradeOrdersHistory({ instType: 'SWAP' })).pipe(\n repeat({ delay: 1000 }),\n retry({ delay: 1000 }),\n shareReplay(1),\n );\n\n const swapPendingOrders = defer(() => client.getTradeOrdersPending({ instType: 'SWAP' })).pipe(\n repeat({ delay: 1000 }),\n retry({ delay: 1000 }),\n shareReplay(1),\n );\n\n const ordersFromHistoryOrder$ = swapHistoryOrders.pipe(\n //\n mergeMap((x) =>\n from(x.data || []).pipe(\n //\n map((x) => makeOrder(x, TRADING_ACCOUNT_ID)),\n ),\n ),\n );\n\n const ordersFromPendingOrder$ = swapPendingOrders.pipe(\n //\n mergeMap((x) =>\n from(x.data || []).pipe(\n //\n map((x) => makeOrder(x, TRADING_ACCOUNT_ID)),\n ),\n ),\n );\n\n merge(ordersFromHistoryOrder$, ordersFromPendingOrder$)\n .pipe(\n //\n tap((x) => {\n order$.next(x);\n }),\n )\n .subscribe();\n})();\n"]}
@@ -0,0 +1,6 @@
1
+ import { IProduct } from '@yuants/data-product';
2
+ export declare const usdtSwapProducts$: import("rxjs").Observable<IProduct[]>;
3
+ export declare const marginProducts$: import("rxjs").Observable<IProduct[]>;
4
+ export declare const mapProductIdToMarginProduct$: import("rxjs").Observable<Map<string, IProduct>>;
5
+ export declare const mapProductIdToUsdtSwapProduct$: import("rxjs").Observable<Map<string, IProduct>>;
6
+ //# sourceMappingURL=product.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"product.d.ts","sourceRoot":"","sources":["../src/product.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,QAAQ,EAAE,MAAM,sBAAsB,CAAC;AAchD,eAAO,MAAM,iBAAiB,uCAiC7B,CAAC;AAQF,eAAO,MAAM,eAAe,uCAgC3B,CAAC;AAIF,eAAO,MAAM,4BAA4B,kDAExC,CAAC;AA4CF,eAAO,MAAM,8BAA8B,kDAG1C,CAAC"}
package/lib/product.js ADDED
@@ -0,0 +1,88 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.mapProductIdToUsdtSwapProduct$ = exports.mapProductIdToMarginProduct$ = exports.marginProducts$ = exports.usdtSwapProducts$ = void 0;
4
+ const utils_1 = require("@yuants/utils");
5
+ const protocol_1 = require("@yuants/protocol");
6
+ const sql_1 = require("@yuants/sql");
7
+ const rxjs_1 = require("rxjs");
8
+ const api_1 = require("./api");
9
+ const product$ = new rxjs_1.Subject();
10
+ const swapInstruments$ = (0, rxjs_1.defer)(() => api_1.client.getInstruments({ instType: 'SWAP' })).pipe((0, rxjs_1.repeat)({ delay: 3600000 }), (0, rxjs_1.retry)({ delay: 10000 }), (0, rxjs_1.shareReplay)(1));
11
+ exports.usdtSwapProducts$ = swapInstruments$.pipe((0, rxjs_1.mergeMap)((x) => (0, rxjs_1.from)(x.data || []).pipe((0, rxjs_1.filter)((x) => x.ctType === 'linear' && x.settleCcy === 'USDT'),
12
+ // ISSUE: 可能有 level = '' 的错误情况 (SPK-USDT and RESOLV-USDT at 2025-06-19 08 UTC)
13
+ (0, rxjs_1.filter)((x) => +x.lever > 0), (0, rxjs_1.map)((x) => ({
14
+ datasource_id: 'OKX',
15
+ product_id: (0, utils_1.encodePath)(x.instType, x.instId),
16
+ name: `${x.ctValCcy}-${x.settleCcy}-PERP`,
17
+ base_currency: x.ctValCcy,
18
+ quote_currency: x.settleCcy,
19
+ value_scale: +x.ctVal,
20
+ volume_step: +x.lotSz,
21
+ price_step: +x.tickSz,
22
+ margin_rate: 1 / +x.lever,
23
+ value_scale_unit: '',
24
+ value_based_cost: 0,
25
+ volume_based_cost: 0,
26
+ max_position: 0,
27
+ max_volume: 0,
28
+ allow_long: true,
29
+ allow_short: true,
30
+ market_id: 'OKX',
31
+ no_interest_rate: false,
32
+ })), (0, rxjs_1.tap)((x) => product$.next(x)), (0, rxjs_1.toArray)())), (0, rxjs_1.shareReplay)(1));
33
+ exports.usdtSwapProducts$.subscribe();
34
+ const marginInstruments$ = (0, rxjs_1.defer)(() => api_1.client.getInstruments({ instType: 'MARGIN' })).pipe((0, rxjs_1.repeat)({ delay: 3600000 }), (0, rxjs_1.retry)({ delay: 10000 }), (0, rxjs_1.shareReplay)(1));
35
+ exports.marginProducts$ = marginInstruments$.pipe((0, rxjs_1.mergeMap)((x) => (0, rxjs_1.from)(x.data || []).pipe(
36
+ // ISSUE: 可能有 level = '' 的错误情况 (SPK-USDT and RESOLV-USDT at 2025-06-19 08 UTC)
37
+ (0, rxjs_1.filter)((x) => +x.lever > 0), (0, rxjs_1.map)((x) => ({
38
+ datasource_id: 'OKX',
39
+ product_id: (0, utils_1.encodePath)(x.instType, x.instId),
40
+ base_currency: x.baseCcy,
41
+ quote_currency: x.quoteCcy,
42
+ value_scale: 1,
43
+ volume_step: +x.lotSz,
44
+ price_step: +x.tickSz,
45
+ margin_rate: 1 / +x.lever,
46
+ name: `${x.baseCcy}-${x.quoteCcy}-MARGIN`,
47
+ value_scale_unit: '',
48
+ value_based_cost: 0,
49
+ volume_based_cost: 0,
50
+ max_position: 0,
51
+ max_volume: 0,
52
+ allow_long: true,
53
+ allow_short: true,
54
+ market_id: 'OKX',
55
+ no_interest_rate: false,
56
+ })), (0, rxjs_1.tap)((x) => product$.next(x)), (0, rxjs_1.toArray)())), (0, rxjs_1.shareReplay)(1));
57
+ exports.marginProducts$.subscribe();
58
+ exports.mapProductIdToMarginProduct$ = exports.marginProducts$.pipe((0, rxjs_1.map)((x) => new Map(x.map((x) => [x.product_id, x])), (0, rxjs_1.shareReplay)(1)));
59
+ (0, rxjs_1.defer)(() => api_1.client.getInstruments({ instType: 'SPOT' }))
60
+ .pipe((0, rxjs_1.repeat)({ delay: 3600000 }), (0, rxjs_1.retry)({ delay: 10000 }), (0, rxjs_1.mergeMap)((x) => (0, rxjs_1.from)(x.data || []).pipe((0, rxjs_1.map)((x) => ({
61
+ datasource_id: 'OKX',
62
+ product_id: (0, utils_1.encodePath)(x.instType, x.instId),
63
+ base_currency: x.baseCcy,
64
+ quote_currency: x.quoteCcy,
65
+ value_scale: 1,
66
+ volume_step: +x.lotSz,
67
+ price_step: +x.tickSz,
68
+ margin_rate: 1,
69
+ name: `${x.baseCcy}-${x.quoteCcy}-SPOT`,
70
+ value_scale_unit: '',
71
+ value_based_cost: 0,
72
+ volume_based_cost: 0,
73
+ max_position: 0,
74
+ max_volume: 0,
75
+ allow_long: true,
76
+ allow_short: true,
77
+ market_id: 'OKX',
78
+ no_interest_rate: true,
79
+ })), (0, rxjs_1.tap)((x) => product$.next(x)), (0, rxjs_1.toArray)())))
80
+ .subscribe();
81
+ (0, sql_1.createSQLWriter)(protocol_1.Terminal.fromNodeEnv(), {
82
+ data$: product$,
83
+ tableName: 'product',
84
+ writeInterval: 1000,
85
+ conflictKeys: ['datasource_id', 'product_id'],
86
+ });
87
+ exports.mapProductIdToUsdtSwapProduct$ = exports.usdtSwapProducts$.pipe((0, rxjs_1.map)((x) => new Map(x.map((x) => [x.product_id, x]))), (0, rxjs_1.shareReplay)(1));
88
+ //# sourceMappingURL=product.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"product.js","sourceRoot":"","sources":["../src/product.ts"],"names":[],"mappings":"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","sourcesContent":["import { encodePath } from '@yuants/utils';\nimport { IProduct } from '@yuants/data-product';\nimport { Terminal } from '@yuants/protocol';\nimport { createSQLWriter } from '@yuants/sql';\nimport { defer, filter, from, map, mergeMap, repeat, retry, shareReplay, Subject, tap, toArray } from 'rxjs';\nimport { client } from './api';\n\nconst product$ = new Subject<IProduct>();\n\nconst swapInstruments$ = defer(() => client.getInstruments({ instType: 'SWAP' })).pipe(\n repeat({ delay: 3600_000 }),\n retry({ delay: 10_000 }),\n shareReplay(1),\n);\n\nexport const usdtSwapProducts$ = swapInstruments$.pipe(\n mergeMap((x) =>\n from(x.data || []).pipe(\n filter((x) => x.ctType === 'linear' && x.settleCcy === 'USDT'),\n // ISSUE: 可能有 level = '' 的错误情况 (SPK-USDT and RESOLV-USDT at 2025-06-19 08 UTC)\n filter((x) => +x.lever > 0),\n map(\n (x): IProduct => ({\n datasource_id: 'OKX',\n product_id: encodePath(x.instType, x.instId),\n name: `${x.ctValCcy}-${x.settleCcy}-PERP`,\n base_currency: x.ctValCcy,\n quote_currency: x.settleCcy,\n value_scale: +x.ctVal,\n volume_step: +x.lotSz,\n price_step: +x.tickSz,\n margin_rate: 1 / +x.lever,\n value_scale_unit: '',\n value_based_cost: 0,\n volume_based_cost: 0,\n max_position: 0,\n max_volume: 0,\n allow_long: true,\n allow_short: true,\n market_id: 'OKX',\n no_interest_rate: false,\n }),\n ),\n tap((x) => product$.next(x)),\n toArray(),\n ),\n ),\n shareReplay(1),\n);\n\nusdtSwapProducts$.subscribe();\nconst marginInstruments$ = defer(() => client.getInstruments({ instType: 'MARGIN' })).pipe(\n repeat({ delay: 3600_000 }),\n retry({ delay: 10_000 }),\n shareReplay(1),\n);\nexport const marginProducts$ = marginInstruments$.pipe(\n mergeMap((x) =>\n from(x.data || []).pipe(\n // ISSUE: 可能有 level = '' 的错误情况 (SPK-USDT and RESOLV-USDT at 2025-06-19 08 UTC)\n filter((x) => +x.lever > 0),\n map(\n (x): IProduct => ({\n datasource_id: 'OKX',\n product_id: encodePath(x.instType, x.instId),\n base_currency: x.baseCcy,\n quote_currency: x.quoteCcy,\n value_scale: 1,\n volume_step: +x.lotSz,\n price_step: +x.tickSz,\n margin_rate: 1 / +x.lever,\n name: `${x.baseCcy}-${x.quoteCcy}-MARGIN`,\n value_scale_unit: '',\n value_based_cost: 0,\n volume_based_cost: 0,\n max_position: 0,\n max_volume: 0,\n allow_long: true,\n allow_short: true,\n market_id: 'OKX',\n no_interest_rate: false,\n }),\n ),\n tap((x) => product$.next(x)),\n toArray(),\n ),\n ),\n shareReplay(1),\n);\n\nmarginProducts$.subscribe();\n\nexport const mapProductIdToMarginProduct$ = marginProducts$.pipe(\n map((x) => new Map(x.map((x) => [x.product_id, x])), shareReplay(1)),\n);\n\ndefer(() => client.getInstruments({ instType: 'SPOT' }))\n .pipe(\n repeat({ delay: 3600_000 }),\n retry({ delay: 10_000 }),\n mergeMap((x) =>\n from(x.data || []).pipe(\n map(\n (x): IProduct => ({\n datasource_id: 'OKX',\n product_id: encodePath(x.instType, x.instId),\n base_currency: x.baseCcy,\n quote_currency: x.quoteCcy,\n value_scale: 1,\n volume_step: +x.lotSz,\n price_step: +x.tickSz,\n margin_rate: 1,\n name: `${x.baseCcy}-${x.quoteCcy}-SPOT`,\n value_scale_unit: '',\n value_based_cost: 0,\n volume_based_cost: 0,\n max_position: 0,\n max_volume: 0,\n allow_long: true,\n allow_short: true,\n market_id: 'OKX',\n no_interest_rate: true,\n }),\n ),\n tap((x) => product$.next(x)),\n toArray(),\n ),\n ),\n )\n .subscribe();\n\ncreateSQLWriter<IProduct>(Terminal.fromNodeEnv(), {\n data$: product$,\n tableName: 'product',\n writeInterval: 1_000,\n conflictKeys: ['datasource_id', 'product_id'],\n});\n\nexport const mapProductIdToUsdtSwapProduct$ = usdtSwapProducts$.pipe(\n map((x) => new Map(x.map((x) => [x.product_id, x]))),\n shareReplay(1),\n);\n"]}
package/lib/quote.d.ts ADDED
@@ -0,0 +1,42 @@
1
+ export declare const swapMarketTickers$: import("rxjs").Observable<{
2
+ [k: string]: {
3
+ instType: string;
4
+ instId: string;
5
+ last: string;
6
+ lastSz: string;
7
+ askPx: string;
8
+ askSz: string;
9
+ bidPx: string;
10
+ bidSz: string;
11
+ open24h: string;
12
+ high24h: string;
13
+ low24h: string;
14
+ volCcy24h: string;
15
+ vol24h: string;
16
+ sodUtc0: string;
17
+ sodUtc8: string;
18
+ ts: string;
19
+ };
20
+ }>;
21
+ export declare const spotMarketTickers$: import("rxjs").Observable<{
22
+ [k: string]: {
23
+ instType: string;
24
+ instId: string;
25
+ last: string;
26
+ lastSz: string;
27
+ askPx: string;
28
+ askSz: string;
29
+ bidPx: string;
30
+ bidSz: string;
31
+ open24h: string;
32
+ high24h: string;
33
+ low24h: string;
34
+ volCcy24h: string;
35
+ vol24h: string;
36
+ sodUtc0: string;
37
+ sodUtc8: string;
38
+ ts: string;
39
+ };
40
+ }>;
41
+ export declare const swapOpenInterest$: import("rxjs").Observable<Map<string, number>>;
42
+ //# sourceMappingURL=quote.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"quote.d.ts","sourceRoot":"","sources":["../src/quote.ts"],"names":[],"mappings":"AAmBA,eAAO,MAAM,kBAAkB;;;;;;;;;;;;;;;;;;;EAW9B,CAAC;AAEF,eAAO,MAAM,kBAAkB;;;;;;;;;;;;;;;;;;;EAW9B,CAAC;AAmFF,eAAO,MAAM,iBAAiB,gDAG7B,CAAC"}