@tradejs/infra 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/redis.mjs ADDED
@@ -0,0 +1,296 @@
1
+ // src/redis.ts
2
+ import Redis from "ioredis";
3
+ var TTL_1D = 86400;
4
+ var toJson = (value) => JSON.stringify(value);
5
+ var logger = {
6
+ log: (level, message, ...args) => {
7
+ const method = level === "error" ? "error" : level === "warn" ? "warn" : "log";
8
+ console[method](`[infra:redis] ${message}`, ...args);
9
+ }
10
+ };
11
+ var redisConnectionWarningShown = false;
12
+ var redisUnavailable = false;
13
+ var isRedisConnectivityError = (error) => /ECONNREFUSED|ENOTFOUND|EAI_AGAIN|ETIMEDOUT|MaxRetriesPerRequestError|Connection is closed|Stream isn't writeable/i.test(
14
+ error.message
15
+ );
16
+ var toNonNegativeInt = (value, fallback) => {
17
+ const parsed = Number.parseInt(String(value ?? ""), 10);
18
+ return Number.isFinite(parsed) && parsed >= 0 ? parsed : fallback;
19
+ };
20
+ var toPositiveInt = (value, fallback) => {
21
+ const parsed = Number.parseInt(String(value ?? ""), 10);
22
+ return Number.isFinite(parsed) && parsed > 0 ? parsed : fallback;
23
+ };
24
+ var markRedisUnavailable = (error) => {
25
+ redisUnavailable = true;
26
+ if (redisConnectionWarningShown) return;
27
+ redisConnectionWarningShown = true;
28
+ logger.log(
29
+ "warn",
30
+ "Redis is unavailable: %s. Cache-dependent features are temporarily disabled.",
31
+ error.message
32
+ );
33
+ };
34
+ var getRedis = () => {
35
+ if (!global.__redis__) {
36
+ const host = process.env.REDIS_HOST || "127.0.0.1";
37
+ const port = toPositiveInt(process.env.REDIS_PORT, 6379);
38
+ const connectTimeout = toPositiveInt(
39
+ process.env.REDIS_CONNECT_TIMEOUT_MS,
40
+ 3e3
41
+ );
42
+ const maxRetriesPerRequest = toNonNegativeInt(
43
+ process.env.REDIS_MAX_RETRIES_PER_REQUEST,
44
+ 1
45
+ );
46
+ global.__redis__ = new Redis({
47
+ host,
48
+ port,
49
+ connectTimeout,
50
+ maxRetriesPerRequest,
51
+ enableOfflineQueue: false,
52
+ retryStrategy: (attempt) => Math.min(attempt * 200, 2e3)
53
+ });
54
+ global.__redis__.on("error", (error) => {
55
+ if (isRedisConnectivityError(error)) {
56
+ markRedisUnavailable(error);
57
+ return;
58
+ }
59
+ logger.log("error", "Redis client error: %s", String(error));
60
+ });
61
+ global.__redis__.on("ready", () => {
62
+ redisUnavailable = false;
63
+ if (redisConnectionWarningShown) {
64
+ redisConnectionWarningShown = false;
65
+ logger.log("info", "Redis connection restored");
66
+ }
67
+ });
68
+ }
69
+ return global.__redis__;
70
+ };
71
+ var sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
72
+ var getRedisStatus = (redis) => String(redis.status ?? "ready");
73
+ var waitForRedisReady = async (redis) => {
74
+ if (getRedisStatus(redis) === "ready") {
75
+ return true;
76
+ }
77
+ const readyTimeoutMs = toPositiveInt(
78
+ process.env.REDIS_READY_TIMEOUT_MS,
79
+ toPositiveInt(process.env.REDIS_CONNECT_TIMEOUT_MS, 3e3)
80
+ );
81
+ const startedAt = Date.now();
82
+ while (Date.now() - startedAt < readyTimeoutMs) {
83
+ if (getRedisStatus(redis) === "ready") {
84
+ return true;
85
+ }
86
+ if (getRedisStatus(redis) === "end") {
87
+ return false;
88
+ }
89
+ await sleep(50);
90
+ }
91
+ return getRedisStatus(redis) === "ready";
92
+ };
93
+ var getReadyRedis = async () => {
94
+ if (redisUnavailable) return null;
95
+ const redis = getRedis();
96
+ const ready = await waitForRedisReady(redis);
97
+ if (ready) {
98
+ return redis;
99
+ }
100
+ markRedisUnavailable(
101
+ new Error(`Redis is not ready (status=${getRedisStatus(redis)})`)
102
+ );
103
+ return null;
104
+ };
105
+ var toResultString = (result) => {
106
+ if (result == null) return null;
107
+ if (typeof result === "string") return result;
108
+ if (Buffer.isBuffer(result)) return result.toString("utf8");
109
+ return String(result);
110
+ };
111
+ var DEFAULT_OPTIONS = {
112
+ expire: TTL_1D
113
+ };
114
+ var parseJsonOrDeleteKey = async (redis, key, raw, fallback) => {
115
+ try {
116
+ return JSON.parse(raw);
117
+ } catch (e) {
118
+ logger.log("error", "failed JSON.parse(%s): %s", key, String(e));
119
+ await redis.del(key);
120
+ return fallback;
121
+ }
122
+ };
123
+ var getKeys = async (prefix) => {
124
+ if (redisUnavailable) return [];
125
+ const redis = await getReadyRedis();
126
+ if (!redis) return [];
127
+ const keys = [];
128
+ try {
129
+ let cursor = "0";
130
+ do {
131
+ const [nextCursor, batch] = await redis.scan(
132
+ cursor,
133
+ "MATCH",
134
+ `${prefix}*`,
135
+ "COUNT",
136
+ "200"
137
+ );
138
+ cursor = nextCursor;
139
+ for (const key of batch) {
140
+ if (key.startsWith(prefix)) {
141
+ keys.push(key);
142
+ }
143
+ }
144
+ } while (cursor !== "0");
145
+ } catch (e) {
146
+ if (e instanceof Error && isRedisConnectivityError(e)) {
147
+ markRedisUnavailable(e);
148
+ return [];
149
+ }
150
+ logger.log("warn", "failed SCAN for %s: %s", prefix, String(e));
151
+ return [];
152
+ }
153
+ return keys;
154
+ };
155
+ var getData = async (key, fallback = []) => {
156
+ if (redisUnavailable) return fallback;
157
+ const redis = await getReadyRedis();
158
+ if (!redis) return fallback;
159
+ try {
160
+ const rawJson = await redis.call("JSON.GET", key);
161
+ const raw = toResultString(rawJson);
162
+ if (raw == null) return fallback;
163
+ return parseJsonOrDeleteKey(redis, key, raw, fallback);
164
+ } catch (e) {
165
+ if (e instanceof Error && isRedisConnectivityError(e)) {
166
+ markRedisUnavailable(e);
167
+ return fallback;
168
+ }
169
+ logger.log(
170
+ "error",
171
+ "failed JSON.GET %s: %s (fallback to GET)",
172
+ key,
173
+ String(e)
174
+ );
175
+ }
176
+ try {
177
+ const raw = await redis.get(key);
178
+ if (raw == null) return fallback;
179
+ return parseJsonOrDeleteKey(redis, key, raw, fallback);
180
+ } catch (e) {
181
+ if (e instanceof Error && isRedisConnectivityError(e)) {
182
+ markRedisUnavailable(e);
183
+ return fallback;
184
+ }
185
+ logger.log("error", "failed GET %s: %s", key, String(e));
186
+ return fallback;
187
+ }
188
+ };
189
+ var delKey = async (key) => {
190
+ return delKeyWithOptions(key);
191
+ };
192
+ var RedisWriteBlockedError = class extends Error {
193
+ constructor(message) {
194
+ super(message);
195
+ this.name = "RedisWriteBlockedError";
196
+ }
197
+ };
198
+ var delKeyWithOptions = async (key, options = {}) => {
199
+ if (redisUnavailable) return false;
200
+ const { raiseOnMisconf = false } = options;
201
+ const redis = await getReadyRedis();
202
+ if (!redis) return false;
203
+ try {
204
+ const result = await redis.del(key);
205
+ if (result === 1) {
206
+ return true;
207
+ }
208
+ return false;
209
+ } catch (e) {
210
+ if (e instanceof Error && isRedisConnectivityError(e)) {
211
+ markRedisUnavailable(e);
212
+ return false;
213
+ }
214
+ const msg = String(e);
215
+ if (raiseOnMisconf && msg.includes("MISCONF")) {
216
+ throw new RedisWriteBlockedError(msg);
217
+ }
218
+ logger.log("error", "failed DEL %s: %s", key, String(e));
219
+ return false;
220
+ }
221
+ };
222
+ var setData = async (key, data, options = {}) => {
223
+ if (redisUnavailable) return;
224
+ const { expire } = { ...DEFAULT_OPTIONS, ...options };
225
+ const redis = await getReadyRedis();
226
+ if (!redis) return;
227
+ const value = toJson(data);
228
+ try {
229
+ await redis.call("JSON.SET", key, "$", value);
230
+ if (expire) {
231
+ await redis.expire(key, expire);
232
+ }
233
+ } catch (e) {
234
+ if (e instanceof Error && isRedisConnectivityError(e)) {
235
+ markRedisUnavailable(e);
236
+ return;
237
+ }
238
+ logger.log(
239
+ "error",
240
+ "failed JSON.SET %s: %s (fallback to SET)",
241
+ key,
242
+ String(e)
243
+ );
244
+ try {
245
+ if (expire) {
246
+ await redis.set(key, value, "EX", expire);
247
+ } else {
248
+ await redis.set(key, value);
249
+ }
250
+ } catch (e2) {
251
+ if (e2 instanceof Error && isRedisConnectivityError(e2)) {
252
+ markRedisUnavailable(e2);
253
+ return;
254
+ }
255
+ logger.log("error", "failed SET %s: %s", key, String(e2));
256
+ }
257
+ }
258
+ };
259
+ var redisKeys = {
260
+ users: () => "users:index:",
261
+ user: (userName) => `users:index:${userName}`,
262
+ bots: (userName) => `users:${userName}:bots`,
263
+ botsPrefix: () => "users:",
264
+ bot: (userName, botId) => `users:${userName}:bots:${botId}`,
265
+ backtestConfig: (userName, config) => `users:${userName}:backtests:configs:${config}`,
266
+ strategies: (userName) => `users:${userName}:strategies`,
267
+ strategyConfig: (userName, strategyName) => `users:${userName}:strategies:${strategyName}:config`,
268
+ strategyResults: (userName, strategyName) => `users:${userName}:strategies:${strategyName}:results`,
269
+ tests: (userName, strategyName) => strategyName ? `users:${userName}:tests:${strategyName}` : `users:${userName}:tests:`,
270
+ testOrders: (userName, strategyName, testName) => `users:${userName}:tests:${strategyName}:${testName}:orders`,
271
+ testConfig: (userName, strategyName, testName) => `users:${userName}:tests:${strategyName}:${testName}:config`,
272
+ testStat: (userName, strategyName, testName) => `users:${userName}:tests:${strategyName}:${testName}:stat`,
273
+ cacheChunk: (userName, chunkId) => `users:${userName}:cache:tests:chunks:${chunkId}`,
274
+ cacheOrders: (userName, orderLogId) => `users:${userName}:cache:tests:orders:${orderLogId}`,
275
+ cachePositions: (userName, orderLogId) => `users:${userName}:cache:tests:positions:${orderLogId}`,
276
+ signal: (symbol, signalId) => `signals:${symbol}:${signalId}`,
277
+ signalsBySymbol: (symbol) => `signals:${symbol}:`,
278
+ storeSignal: (symbol, signalId) => `store:signals:${symbol}:${signalId}`,
279
+ analysis: (symbol, signalId) => `analysis:${symbol}:${signalId}`,
280
+ backtestResults: (userName, config, timestamp) => `users:${userName}:backtests:results:${config}:${timestamp}`,
281
+ mlSignalsByStrategy: (strategyName) => `ml:${strategyName}:signals:`,
282
+ mlSignals: () => "ml:",
283
+ mlSignal: (strategyName, signalId) => `ml:${strategyName}:signals:${signalId}`,
284
+ mlResultsByStrategy: (strategyName) => `ml:${strategyName}:results:`,
285
+ mlResults: () => "ml:",
286
+ mlResult: (strategyName, signalId) => `ml:${strategyName}:results:${signalId}`
287
+ };
288
+ export {
289
+ RedisWriteBlockedError,
290
+ delKey,
291
+ delKeyWithOptions,
292
+ getData,
293
+ getKeys,
294
+ redisKeys,
295
+ setData
296
+ };
@@ -0,0 +1,68 @@
1
+ import { Pool } from 'pg';
2
+ import { KlineChartData } from '@tradejs/types';
3
+
4
+ declare global {
5
+ var __pgPool__: Pool | undefined;
6
+ }
7
+ type CandleRow = {
8
+ symbol: string;
9
+ interval: number;
10
+ ts: Date;
11
+ open: number;
12
+ high: number;
13
+ low: number;
14
+ close: number;
15
+ volume?: number | null;
16
+ turnover?: number | null;
17
+ };
18
+ type DerivativesInterval = '15m' | '1h';
19
+ type DerivativesRow = {
20
+ symbol: string;
21
+ interval: DerivativesInterval;
22
+ ts: Date;
23
+ openInterest?: number | null;
24
+ fundingRate?: number | null;
25
+ liqLong?: number | null;
26
+ liqShort?: number | null;
27
+ liqTotal?: number | null;
28
+ source?: string | null;
29
+ };
30
+ type SpreadRow = {
31
+ symbol: string;
32
+ interval: DerivativesInterval;
33
+ ts: Date;
34
+ binancePrice?: number | null;
35
+ coinbasePrice?: number | null;
36
+ spread?: number | null;
37
+ source?: string | null;
38
+ };
39
+ declare const toRows: (symbol: string, interval: number, data: KlineChartData) => CandleRow[];
40
+ declare function upsertCandles(rows: CandleRow[]): Promise<void>;
41
+ declare function upsertDerivatives(rows: DerivativesRow[]): Promise<void>;
42
+ declare function getDerivativesRangeForSymbols(symbols: string[], interval: DerivativesInterval, startMs: number, endMs: number): Promise<any[]>;
43
+ declare function getDerivativesSummary(hours?: number, limit?: number): Promise<{
44
+ rows: any[];
45
+ aggregates: any[];
46
+ hours: number;
47
+ }>;
48
+ declare function upsertSpreadRows(rows: SpreadRow[]): Promise<void>;
49
+ declare function getSpreadRangeForSymbols(symbols: string[], interval: DerivativesInterval, startMs: number, endMs: number): Promise<any[]>;
50
+ declare function getSpreadSummary(hours?: number, limit?: number): Promise<{
51
+ rows: any[];
52
+ aggregates: any[];
53
+ hours: number;
54
+ }>;
55
+ declare function getCandlesRange(symbol: string, interval: number, startMs: number, endMs: number): Promise<any[]>;
56
+ declare function getDataEdges(symbol: string, interval: number): Promise<{
57
+ min: number | undefined;
58
+ max: number | undefined;
59
+ }>;
60
+ declare function waitForDbReady(attempts?: number, delayMs?: number): Promise<void>;
61
+ declare function deleteCandles(symbol: string, interval: number): Promise<void>;
62
+ declare function findContinuityGap(symbol: string, interval: number): Promise<{
63
+ ts: number;
64
+ prevTs: number;
65
+ diffSeconds: number;
66
+ } | null>;
67
+
68
+ export { type CandleRow, type DerivativesInterval, type DerivativesRow, type SpreadRow, deleteCandles, findContinuityGap, getCandlesRange, getDataEdges, getDerivativesRangeForSymbols, getDerivativesSummary, getSpreadRangeForSymbols, getSpreadSummary, toRows, upsertCandles, upsertDerivatives, upsertSpreadRows, waitForDbReady };
@@ -0,0 +1,68 @@
1
+ import { Pool } from 'pg';
2
+ import { KlineChartData } from '@tradejs/types';
3
+
4
+ declare global {
5
+ var __pgPool__: Pool | undefined;
6
+ }
7
+ type CandleRow = {
8
+ symbol: string;
9
+ interval: number;
10
+ ts: Date;
11
+ open: number;
12
+ high: number;
13
+ low: number;
14
+ close: number;
15
+ volume?: number | null;
16
+ turnover?: number | null;
17
+ };
18
+ type DerivativesInterval = '15m' | '1h';
19
+ type DerivativesRow = {
20
+ symbol: string;
21
+ interval: DerivativesInterval;
22
+ ts: Date;
23
+ openInterest?: number | null;
24
+ fundingRate?: number | null;
25
+ liqLong?: number | null;
26
+ liqShort?: number | null;
27
+ liqTotal?: number | null;
28
+ source?: string | null;
29
+ };
30
+ type SpreadRow = {
31
+ symbol: string;
32
+ interval: DerivativesInterval;
33
+ ts: Date;
34
+ binancePrice?: number | null;
35
+ coinbasePrice?: number | null;
36
+ spread?: number | null;
37
+ source?: string | null;
38
+ };
39
+ declare const toRows: (symbol: string, interval: number, data: KlineChartData) => CandleRow[];
40
+ declare function upsertCandles(rows: CandleRow[]): Promise<void>;
41
+ declare function upsertDerivatives(rows: DerivativesRow[]): Promise<void>;
42
+ declare function getDerivativesRangeForSymbols(symbols: string[], interval: DerivativesInterval, startMs: number, endMs: number): Promise<any[]>;
43
+ declare function getDerivativesSummary(hours?: number, limit?: number): Promise<{
44
+ rows: any[];
45
+ aggregates: any[];
46
+ hours: number;
47
+ }>;
48
+ declare function upsertSpreadRows(rows: SpreadRow[]): Promise<void>;
49
+ declare function getSpreadRangeForSymbols(symbols: string[], interval: DerivativesInterval, startMs: number, endMs: number): Promise<any[]>;
50
+ declare function getSpreadSummary(hours?: number, limit?: number): Promise<{
51
+ rows: any[];
52
+ aggregates: any[];
53
+ hours: number;
54
+ }>;
55
+ declare function getCandlesRange(symbol: string, interval: number, startMs: number, endMs: number): Promise<any[]>;
56
+ declare function getDataEdges(symbol: string, interval: number): Promise<{
57
+ min: number | undefined;
58
+ max: number | undefined;
59
+ }>;
60
+ declare function waitForDbReady(attempts?: number, delayMs?: number): Promise<void>;
61
+ declare function deleteCandles(symbol: string, interval: number): Promise<void>;
62
+ declare function findContinuityGap(symbol: string, interval: number): Promise<{
63
+ ts: number;
64
+ prevTs: number;
65
+ diffSeconds: number;
66
+ } | null>;
67
+
68
+ export { type CandleRow, type DerivativesInterval, type DerivativesRow, type SpreadRow, deleteCandles, findContinuityGap, getCandlesRange, getDataEdges, getDerivativesRangeForSymbols, getDerivativesSummary, getSpreadRangeForSymbols, getSpreadSummary, toRows, upsertCandles, upsertDerivatives, upsertSpreadRows, waitForDbReady };