@milkio/redis 1.0.0-beta.99 → 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/LICENSE +20 -20
- package/index.d.ts +19 -303
- package/index.js +103 -51
- package/package.json +12 -1
- package/chunk-6sxyen7j.js +0 -24
- package/chunk-xt8gwvgg.js +0 -16021
package/index.js
CHANGED
|
@@ -1,30 +1,28 @@
|
|
|
1
|
-
import {
|
|
2
|
-
__require,
|
|
3
|
-
__toESM
|
|
4
|
-
} from "./chunk-6sxyen7j.js";
|
|
5
|
-
|
|
6
1
|
// packages/milkio-redis/index.ts
|
|
7
|
-
async function createRedis(
|
|
8
|
-
const
|
|
9
|
-
const redis = await NodeRedis.default.createClient(options).connect();
|
|
2
|
+
async function createRedis(redisClient) {
|
|
3
|
+
const redis = redisClient;
|
|
10
4
|
const milkioRedis = {
|
|
11
5
|
redis,
|
|
12
|
-
useCache: (key,
|
|
6
|
+
useCache: (key, options) => ({
|
|
13
7
|
set: async (value, expireMs) => {
|
|
14
|
-
|
|
8
|
+
if (expireMs === null) {
|
|
9
|
+
await redis.SET(key, JSON.stringify(value));
|
|
10
|
+
} else {
|
|
11
|
+
await redis.PSETEX(key, expireMs, JSON.stringify(value));
|
|
12
|
+
}
|
|
15
13
|
return value;
|
|
16
14
|
},
|
|
17
15
|
get: async () => {
|
|
18
16
|
const result = await redis.GET(key);
|
|
19
17
|
if (result === null)
|
|
20
|
-
return
|
|
18
|
+
return options?.defaultValue;
|
|
21
19
|
return reviveJSONParse(JSON.parse(result));
|
|
22
20
|
},
|
|
23
21
|
pull: async () => {
|
|
24
22
|
const resultRaw = await redis.MULTI().GET(key).DEL(key).EXEC();
|
|
25
23
|
const result = resultRaw[0];
|
|
26
24
|
if (result === null)
|
|
27
|
-
return
|
|
25
|
+
return options?.defaultValue;
|
|
28
26
|
return reviveJSONParse(JSON.parse(result));
|
|
29
27
|
},
|
|
30
28
|
has: async () => {
|
|
@@ -35,29 +33,29 @@ async function createRedis(options) {
|
|
|
35
33
|
await redis.DEL(key);
|
|
36
34
|
}
|
|
37
35
|
}),
|
|
38
|
-
useFetch: (key,
|
|
39
|
-
if (typeof
|
|
40
|
-
throw new Error("expireMs must be a positive number");
|
|
36
|
+
useFetch: (key, options) => {
|
|
37
|
+
if (options.expireMs !== null && (typeof options.expireMs !== "number" || options.expireMs <= 0)) {
|
|
38
|
+
throw new Error("expireMs must be a positive number or null");
|
|
41
39
|
}
|
|
42
|
-
if (
|
|
40
|
+
if (options.lockInterval !== undefined && (typeof options.lockInterval !== "number" || options.lockInterval <= 0)) {
|
|
43
41
|
throw new Error("lockInterval must be a positive number if provided");
|
|
44
42
|
}
|
|
45
|
-
if (
|
|
46
|
-
throw new Error("realExpireMs must be a positive number if provided");
|
|
43
|
+
if (options.realExpireMs !== undefined && options.realExpireMs !== null && (typeof options.realExpireMs !== "number" || options.realExpireMs <= 0)) {
|
|
44
|
+
throw new Error("realExpireMs must be a positive number or null if provided");
|
|
47
45
|
}
|
|
48
|
-
if (
|
|
49
|
-
throw new Error("notFoundExpireMs must be a positive number if provided");
|
|
46
|
+
if (options.notFoundExpireMs !== undefined && options.notFoundExpireMs !== null && (typeof options.notFoundExpireMs !== "number" || options.notFoundExpireMs <= 0)) {
|
|
47
|
+
throw new Error("notFoundExpireMs must be a positive number or null if provided");
|
|
50
48
|
}
|
|
51
|
-
if (
|
|
52
|
-
if (typeof
|
|
49
|
+
if (options.refreshLockInterval !== undefined) {
|
|
50
|
+
if (typeof options.refreshLockInterval !== "number" || options.refreshLockInterval <= 0) {
|
|
53
51
|
throw new Error("refreshLockInterval must be a positive number if provided");
|
|
54
52
|
}
|
|
55
|
-
const lockInterval =
|
|
56
|
-
if (
|
|
53
|
+
const lockInterval = options.lockInterval ?? 8192;
|
|
54
|
+
if (options.refreshLockInterval <= lockInterval) {
|
|
57
55
|
throw new Error("refreshLockInterval must be greater than lockInterval");
|
|
58
56
|
}
|
|
59
57
|
}
|
|
60
|
-
if (typeof
|
|
58
|
+
if (typeof options.fetch !== "function") {
|
|
61
59
|
throw new Error("fetch must be a function");
|
|
62
60
|
}
|
|
63
61
|
return {
|
|
@@ -69,9 +67,12 @@ async function createRedis(options) {
|
|
|
69
67
|
await redis.DEL(key);
|
|
70
68
|
},
|
|
71
69
|
fetch: async () => {
|
|
72
|
-
const lockInterval =
|
|
73
|
-
|
|
74
|
-
|
|
70
|
+
const lockInterval = options.lockInterval ?? 8192;
|
|
71
|
+
let realExpireMs = options.realExpireMs ?? null;
|
|
72
|
+
if (options.expireMs !== null) {
|
|
73
|
+
realExpireMs = options.realExpireMs ?? Math.floor(options.expireMs * (Math.random() + 0.5)) + 8192;
|
|
74
|
+
}
|
|
75
|
+
const notFoundExpireMs = options.notFoundExpireMs ?? (options.expireMs !== null ? Math.min(options.expireMs, 16384) : 86400000);
|
|
75
76
|
const resultRaw = await redis.GET(key);
|
|
76
77
|
const result = resultRaw ? reviveJSONParse(JSON.parse(resultRaw)) : undefined;
|
|
77
78
|
const now = Date.now();
|
|
@@ -80,13 +81,34 @@ async function createRedis(options) {
|
|
|
80
81
|
}
|
|
81
82
|
const refreshLockKey = `${key}:refresh-lock`;
|
|
82
83
|
if (await redis.EXISTS(refreshLockKey)) {
|
|
83
|
-
return result ? result.R :
|
|
84
|
+
return result ? result.R : options?.defaultValue;
|
|
84
85
|
}
|
|
85
86
|
const lockKey = `${key}:lock`;
|
|
86
87
|
const lockSet = await redis.SET(lockKey, "1", { PX: lockInterval, NX: true });
|
|
87
88
|
const gotLock = lockSet === "OK";
|
|
88
|
-
if (!gotLock)
|
|
89
|
-
|
|
89
|
+
if (!gotLock) {
|
|
90
|
+
const maxWaitTime = lockInterval;
|
|
91
|
+
const waitInterval = 42;
|
|
92
|
+
let waited = 0;
|
|
93
|
+
while (waited < maxWaitTime) {
|
|
94
|
+
await new Promise((resolve) => setTimeout(resolve, waitInterval));
|
|
95
|
+
waited += waitInterval;
|
|
96
|
+
const waitedResultRaw = await redis.GET(key);
|
|
97
|
+
const waitedResult = waitedResultRaw ? reviveJSONParse(JSON.parse(waitedResultRaw)) : undefined;
|
|
98
|
+
if (waitedResult && waitedResult.T > now) {
|
|
99
|
+
return waitedResult.R;
|
|
100
|
+
}
|
|
101
|
+
if (!await redis.EXISTS(lockKey)) {
|
|
102
|
+
break;
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
const finalResultRaw = await redis.GET(key);
|
|
106
|
+
const finalResult = finalResultRaw ? reviveJSONParse(JSON.parse(finalResultRaw)) : undefined;
|
|
107
|
+
if (finalResult && finalResult.T > now) {
|
|
108
|
+
return finalResult.R;
|
|
109
|
+
}
|
|
110
|
+
return result ? result.R : options.defaultValue;
|
|
111
|
+
}
|
|
90
112
|
const recheckRaw = await redis.GET(key);
|
|
91
113
|
const recheck = recheckRaw ? reviveJSONParse(JSON.parse(recheckRaw)) : undefined;
|
|
92
114
|
if (recheck && recheck.T > now) {
|
|
@@ -94,14 +116,21 @@ async function createRedis(options) {
|
|
|
94
116
|
}
|
|
95
117
|
let data;
|
|
96
118
|
try {
|
|
97
|
-
data = await
|
|
119
|
+
data = await options.fetch();
|
|
98
120
|
} catch (error) {
|
|
99
121
|
await redis.DEL(lockKey);
|
|
100
122
|
throw error;
|
|
101
123
|
}
|
|
102
|
-
|
|
124
|
+
let effectiveExpireMs = null;
|
|
125
|
+
if (data !== undefined) {
|
|
126
|
+
if (options.expireMs !== null) {
|
|
127
|
+
effectiveExpireMs = options.expireMs + (realExpireMs ?? 0);
|
|
128
|
+
}
|
|
129
|
+
} else {
|
|
130
|
+
effectiveExpireMs = notFoundExpireMs;
|
|
131
|
+
}
|
|
103
132
|
const cacheValue = {
|
|
104
|
-
T: now + (data !== undefined ?
|
|
133
|
+
T: now + (data !== undefined ? options.expireMs ?? Number.MAX_SAFE_INTEGER : notFoundExpireMs),
|
|
105
134
|
R: data
|
|
106
135
|
};
|
|
107
136
|
const refreshLockExists = await redis.EXISTS(refreshLockKey);
|
|
@@ -109,30 +138,48 @@ async function createRedis(options) {
|
|
|
109
138
|
await redis.DEL(lockKey);
|
|
110
139
|
return data;
|
|
111
140
|
}
|
|
112
|
-
|
|
141
|
+
if (effectiveExpireMs === null) {
|
|
142
|
+
await redis.MULTI().SET(key, JSON.stringify(cacheValue)).DEL(lockKey).EXEC();
|
|
143
|
+
} else {
|
|
144
|
+
await redis.MULTI().PSETEX(key, effectiveExpireMs, JSON.stringify(cacheValue)).DEL(lockKey).EXEC();
|
|
145
|
+
}
|
|
113
146
|
return data;
|
|
114
147
|
},
|
|
115
148
|
refresh: async () => {
|
|
116
149
|
const refreshLockKey = `${key}:refresh-lock`;
|
|
117
|
-
const refreshLockInterval =
|
|
150
|
+
const refreshLockInterval = options.refreshLockInterval ?? (options.lockInterval ?? 8192) + 1024;
|
|
118
151
|
await redis.SET(refreshLockKey, "1", {
|
|
119
152
|
PX: refreshLockInterval
|
|
120
153
|
});
|
|
121
154
|
const now = Date.now();
|
|
122
155
|
let data;
|
|
123
156
|
try {
|
|
124
|
-
data = await
|
|
157
|
+
data = await options.fetch();
|
|
125
158
|
} catch (error) {
|
|
126
159
|
await redis.DEL(refreshLockKey);
|
|
127
160
|
throw error;
|
|
128
161
|
}
|
|
129
|
-
|
|
130
|
-
|
|
162
|
+
let realExpireMs = options.realExpireMs ?? null;
|
|
163
|
+
if (options.expireMs !== null) {
|
|
164
|
+
realExpireMs = options.realExpireMs ?? Math.floor(options.expireMs * (Math.random() + 0.5)) + 8192;
|
|
165
|
+
}
|
|
166
|
+
let effectiveExpireMs = null;
|
|
167
|
+
if (data !== undefined) {
|
|
168
|
+
if (options.expireMs !== null) {
|
|
169
|
+
effectiveExpireMs = options.expireMs + (realExpireMs ?? 0);
|
|
170
|
+
}
|
|
171
|
+
} else {
|
|
172
|
+
effectiveExpireMs = options.notFoundExpireMs ?? (options.expireMs !== null ? Math.min(options.expireMs, 16384) : 86400000);
|
|
173
|
+
}
|
|
131
174
|
const cacheValue = {
|
|
132
|
-
T: now + (data !== undefined ?
|
|
175
|
+
T: now + (data !== undefined ? options.expireMs ?? Number.MAX_SAFE_INTEGER : options.notFoundExpireMs ?? (options.expireMs !== null ? Math.min(options.expireMs, 16384) : 86400000)),
|
|
133
176
|
R: data
|
|
134
177
|
};
|
|
135
|
-
|
|
178
|
+
if (effectiveExpireMs === null) {
|
|
179
|
+
await redis.SET(key, JSON.stringify(cacheValue));
|
|
180
|
+
} else {
|
|
181
|
+
await redis.PSETEX(key, effectiveExpireMs, JSON.stringify(cacheValue));
|
|
182
|
+
}
|
|
136
183
|
}
|
|
137
184
|
};
|
|
138
185
|
},
|
|
@@ -142,7 +189,10 @@ async function createRedis(options) {
|
|
|
142
189
|
return result ? Number(result) : 0;
|
|
143
190
|
},
|
|
144
191
|
add: async (amount, expireMs) => {
|
|
145
|
-
if (
|
|
192
|
+
if (expireMs === null) {
|
|
193
|
+
const result = await redis.INCRBY(key, amount);
|
|
194
|
+
return result;
|
|
195
|
+
} else if (!expireMs) {
|
|
146
196
|
const result = await redis.INCRBY(key, amount);
|
|
147
197
|
return result;
|
|
148
198
|
} else {
|
|
@@ -196,14 +246,15 @@ async function createRedis(options) {
|
|
|
196
246
|
};
|
|
197
247
|
return milkioRedis;
|
|
198
248
|
}
|
|
249
|
+
var isoDatePattern = /^(\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(?:\.\d{1,3})?)(Z|[+-]\d{2}:?\d{2})?$/;
|
|
199
250
|
function reviveJSONParse(json) {
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
251
|
+
if (json !== null && typeof json === "object") {
|
|
252
|
+
if (json instanceof Date || typeof json.getTime === "function" && typeof json.toISOString === "function") {
|
|
253
|
+
return json;
|
|
254
|
+
}
|
|
255
|
+
if (Array.isArray(json)) {
|
|
256
|
+
return json.map((item) => reviveJSONParse(item));
|
|
257
|
+
}
|
|
207
258
|
return Object.entries(json).reduce((acc, [key, value]) => {
|
|
208
259
|
acc[key] = reviveJSONParse(value);
|
|
209
260
|
return acc;
|
|
@@ -219,8 +270,9 @@ function reviveJSONParse(json) {
|
|
|
219
270
|
return json;
|
|
220
271
|
}
|
|
221
272
|
export {
|
|
273
|
+
reviveJSONParse,
|
|
222
274
|
createRedis
|
|
223
275
|
};
|
|
224
276
|
|
|
225
|
-
//# debugId=
|
|
226
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["..\\index.ts"],
  "sourcesContent": [
    "import type { RedisClientOptions } from \"redis\";\n\nexport type MilkioRedisCacheOptions<T> = {\n  defaultValue?: T | undefined;\n  expireMs: number;\n};\n\nexport type MilkioRedisFetchOptions<T> = {\n  defaultValue?: T | undefined;\n  expireMs: number;\n  realExpireMs?: number;\n  lockInterval?: number;\n  refreshLockInterval?: number;\n  notFoundExpireMs?: number;\n  fetch: () => T | undefined | Promise<T | undefined>;\n};\n\nexport async function createRedis<Options extends RedisClientOptions>(options: Options) {\n  const NodeRedis = await import(\"redis\");\n  const redis = await NodeRedis.default.createClient(options).connect();\n\n  const milkioRedis = {\n    redis,\n    useCache: <T>(key: string, options?: MilkioRedisCacheOptions<T>) => ({\n      set: async (value: T, expireMs: number): Promise<T> => {\n        await redis.PSETEX(key, expireMs, JSON.stringify(value));\n        return value;\n      },\n      get: async (): Promise<T | undefined> => {\n        const result = await redis.GET(key);\n        if (result === null) return options?.defaultValue;\n        return reviveJSONParse(JSON.parse(result));\n      },\n      pull: async () => {\n        const resultRaw = await redis.MULTI().GET(key).DEL(key).EXEC();\n        const result = resultRaw[0];\n        if (result === null) return options?.defaultValue;\n        return reviveJSONParse(JSON.parse(result as string));\n      },\n      has: async (): Promise<boolean> => {\n        const result = await redis.GET(key);\n        return result !== null;\n      },\n      del: async () => {\n        await redis.DEL(key);\n      },\n    }),\n\n    useFetch: <Options extends MilkioRedisFetchOptions<any>>(key: string, options: Options) => {\n      if (typeof options.expireMs !== \"number\" || options.expireMs <= 0) {\n        throw new Error(\"expireMs must be a positive number\");\n      }\n\n      if (options.lockInterval !== undefined && (typeof options.lockInterval !== \"number\" || options.lockInterval <= 0)) {\n        throw new Error(\"lockInterval must be a positive number if provided\");\n      }\n\n      if (options.realExpireMs !== undefined && (typeof options.realExpireMs !== \"number\" || options.realExpireMs <= 0)) {\n        throw new Error(\"realExpireMs must be a positive number if provided\");\n      }\n\n      if (options.notFoundExpireMs !== undefined && (typeof options.notFoundExpireMs !== \"number\" || options.notFoundExpireMs <= 0)) {\n        throw new Error(\"notFoundExpireMs must be a positive number if provided\");\n      }\n\n      if (options.refreshLockInterval !== undefined) {\n        if (typeof options.refreshLockInterval !== \"number\" || options.refreshLockInterval <= 0) {\n          throw new Error(\"refreshLockInterval must be a positive number if provided\");\n        }\n\n        const lockInterval = options.lockInterval ?? 8192;\n        if (options.refreshLockInterval <= lockInterval) {\n          throw new Error(\"refreshLockInterval must be greater than lockInterval\");\n        }\n      }\n\n      if (typeof options.fetch !== \"function\") {\n        throw new Error(\"fetch must be a function\");\n      }\n\n      return {\n        has: async (): Promise<boolean> => {\n          const result = await redis.GET(key);\n          return result !== null;\n        },\n        del: async () => {\n          await redis.DEL(key);\n        },\n        fetch: async (): Promise<Awaited<ReturnType<Options[\"fetch\"]>>> => {\n          const lockInterval = options.lockInterval ?? 8192;\n          const realExpireMs = options.realExpireMs ?? Math.floor(options.expireMs * (Math.random() + 0.5)) + 8192;\n\n          const notFoundExpireMs = options.notFoundExpireMs ?? Math.min(options.expireMs, 16384);\n\n          const resultRaw = await redis.GET(key);\n          const result = resultRaw ? (reviveJSONParse(JSON.parse(resultRaw)) as { T: number; R: Awaited<ReturnType<Options[\"fetch\"]>> }) : undefined;\n\n          const now = Date.now();\n          if (result && result.T > now) {\n            return result.R;\n          }\n\n          const refreshLockKey = `${key}:refresh-lock`;\n          if (await redis.EXISTS(refreshLockKey)) {\n            return result ? result.R : options?.defaultValue;\n          }\n\n          const lockKey = `${key}:lock`;\n          const lockSet = await redis.SET(lockKey, \"1\", { PX: lockInterval, NX: true });\n          const gotLock = lockSet === \"OK\";\n\n          if (!gotLock) return result ? result.R : options.defaultValue;\n\n          const recheckRaw = await redis.GET(key);\n          const recheck = recheckRaw ? (reviveJSONParse(JSON.parse(recheckRaw)) as { T: number; R: Awaited<ReturnType<Options[\"fetch\"]>> }) : undefined;\n\n          if (recheck && recheck.T > now) {\n            return recheck.R;\n          }\n\n          let data: Awaited<ReturnType<Options[\"fetch\"]>>;\n          try {\n            data = await options.fetch();\n          } catch (error) {\n            await redis.DEL(lockKey);\n            throw error;\n          }\n\n          const effectiveExpireMs = data !== undefined ? options.expireMs + realExpireMs : notFoundExpireMs;\n\n          const cacheValue = {\n            T: now + (data !== undefined ? options.expireMs : notFoundExpireMs),\n            R: data,\n          };\n\n          const refreshLockExists = await redis.EXISTS(refreshLockKey);\n          if (refreshLockExists) {\n            await redis.DEL(lockKey);\n            return data;\n          }\n\n          await redis.MULTI().PSETEX(key, effectiveExpireMs, JSON.stringify(cacheValue)).DEL(lockKey).EXEC();\n\n          return data;\n        },\n        refresh: async (): Promise<void> => {\n          const refreshLockKey = `${key}:refresh-lock`;\n          const refreshLockInterval = options.refreshLockInterval ?? (options.lockInterval ?? 8192) + 1024;\n\n          await redis.SET(refreshLockKey, \"1\", {\n            PX: refreshLockInterval,\n          });\n\n          const now = Date.now();\n          let data: Awaited<ReturnType<Options[\"fetch\"]>> | undefined;\n          try {\n            data = await options.fetch();\n          } catch (error) {\n            await redis.DEL(refreshLockKey);\n            throw error;\n          }\n\n          const realExpireMs = options.realExpireMs ?? Math.floor(options.expireMs * (Math.random() + 0.5)) + 8192;\n          const effectiveExpireMs = data !== undefined ? options.expireMs + realExpireMs : (options.notFoundExpireMs ?? Math.min(options.expireMs, 16384));\n\n          const cacheValue = {\n            T: now + (data !== undefined ? options.expireMs : (options.notFoundExpireMs ?? Math.min(options.expireMs, 16384))),\n            R: data,\n          };\n\n          await redis.PSETEX(key, effectiveExpireMs, JSON.stringify(cacheValue)); // Do not clean the update lock! Instead, wait for it to expire naturally\n        },\n      };\n    },\n\n    useCount: (key: string) => ({\n      get: async (): Promise<number> => {\n        const result = await redis.GET(key);\n        return result ? Number(result) : 0;\n      },\n      add: async (amount: number, expireMs?: number): Promise<number> => {\n        if (!expireMs) {\n          const result = await redis.INCRBY(key, amount);\n          return result;\n        } else {\n          const result = await redis.MULTI().INCRBY(key, amount).PEXPIRE(key, expireMs).EXEC();\n          return Number(result[0]);\n        }\n      },\n      sub: async (amount: number): Promise<number> => {\n        const result = await redis.DECRBY(key, amount);\n        return result;\n      },\n    }),\n\n    useClockIn: (key: string, cleanDate: Date) => ({\n      clockIn: async (offset: number): Promise<void> => {\n        await redis.MULTI().SETBIT(key, offset, 1).PEXPIREAT(key, cleanDate.getTime()).EXEC();\n      },\n      check: async (offset: number): Promise<boolean> => {\n        const result = await redis.GETBIT(key, offset);\n        return result === 1;\n      },\n      firstClockIn: async (): Promise<number> => {\n        const result = await redis.BITPOS(key, 1);\n        return result;\n      },\n      lastClockIn: async (): Promise<number> => {\n        const result = await redis.BITPOS(key, 1, -1);\n        return result;\n      },\n      toArray: async (length: number): Promise<boolean[]> => {\n        const resultRaw = await redis.BITFIELD(key, [\n          {\n            operation: \"GET\",\n            encoding: `u${length}`,\n            offset: \"#0\",\n          },\n        ]);\n        const result = Number.parseInt(`${resultRaw}`).toString(2).split(\"\");\n        const fill = [];\n        for (let i = 0; i < length - result.length; i++) fill.push(\"0\");\n        return [...fill, ...result].map((v) => v === \"1\");\n      },\n      count: async (): Promise<number> => {\n        const result = await redis.BITCOUNT(key);\n        return result;\n      },\n      clean: async (): Promise<void> => {\n        await redis.DEL(key);\n      },\n    }),\n  };\n\n  return milkioRedis;\n}\n\nfunction reviveJSONParse<T>(json: T): T {\n  const isoDatePattern = /^(\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}(?:\\.\\d{1,3})?)(Z|[+-]\\d{2}:?\\d{2})?$/;\n\n  if (json instanceof Date) return json;\n  if (Array.isArray(json)) {\n    return json.map((item) => reviveJSONParse(item)) as any;\n  }\n  if (typeof json === \"object\" && json !== null) {\n    return Object.entries(json).reduce((acc, [key, value]) => {\n      acc[key as keyof T] = reviveJSONParse(value);\n      return acc;\n    }, {} as T);\n  }\n  if (typeof json === \"string\") {\n    const match = json.match(isoDatePattern);\n    if (match) {\n      const normalizedDateString = match[2] ? `${match[1]}${match[2].replace(\":\", \"\")}` : `${match[1]}Z`;\n      return new Date(normalizedDateString) as any;\n    }\n  }\n  return json;\n}\n\nexport type Redis = Awaited<ReturnType<typeof createRedis>>;\n"
  ],
  "mappings": "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",
  "debugId": "6F6F9ACDCC3531F264756E2164756E21",
  "names": []
}
|
|
277
|
+
//# debugId=512607529B76886364756E2164756E21
|
|
278
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../index.ts"],
  "sourcesContent": [
    "export type MilkioRedisCacheOptions<T> = {\n  defaultValue?: T | undefined;\n  expireMs: number | null;\n};\n\nexport type MilkioRedisFetchOptions<T> = {\n  defaultValue?: T | undefined;\n  expireMs: number | null;\n  realExpireMs?: number | null;\n  lockInterval?: number;\n  refreshLockInterval?: number;\n  notFoundExpireMs?: number | null;\n  fetch: () => T | undefined | Promise<T | undefined>;\n};\n\nexport type MilkioRedis<RedisT = any> = {\n  redis: RedisT;\n  useCache: <T>(\n    key: string,\n    options?: MilkioRedisCacheOptions<T>,\n  ) => {\n    set: (value: T, expireMs: number | null) => Promise<T>;\n    get: () => Promise<T | undefined>;\n    pull: () => Promise<T | undefined>;\n    has: () => Promise<boolean>;\n    del: () => Promise<void>;\n  };\n  useFetch: <T>(\n    key: string,\n    options: MilkioRedisFetchOptions<T>,\n  ) => {\n    has: () => Promise<boolean>;\n    del: () => Promise<void>;\n    fetch: () => Promise<T>;\n    refresh: () => Promise<void>;\n  };\n  useCount: (key: string) => {\n    get: () => Promise<number>;\n    add: (amount: number, expireMs?: number | null) => Promise<number>;\n    sub: (amount: number) => Promise<number>;\n  };\n  useClockIn: (\n    key: string,\n    cleanDate: Date,\n  ) => {\n    clockIn: (offset: number) => Promise<void>;\n    check: (offset: number) => Promise<boolean>;\n    firstClockIn: () => Promise<number>;\n    lastClockIn: () => Promise<number>;\n    toArray: (length: number) => Promise<boolean[]>;\n    count: () => Promise<number>;\n    clean: () => Promise<void>;\n  };\n};\n\nexport async function createRedis<RedisT extends { PSETEX: any; SET: any; [others: string | number | symbol]: any }>(redisClient: RedisT): Promise<MilkioRedis<RedisT>> {\n  const redis = redisClient as any;\n\n  const milkioRedis: any = {\n    redis,\n    useCache: <T>(key: string, options?: MilkioRedisCacheOptions<T>) => ({\n      set: async (value: T, expireMs: number | null): Promise<T> => {\n        if (expireMs === null) {\n          // 永久保存\n          await redis.SET(key, JSON.stringify(value));\n        } else {\n          await redis.PSETEX(key, expireMs, JSON.stringify(value));\n        }\n        return value;\n      },\n      get: async (): Promise<T | undefined> => {\n        const result = await redis.GET(key);\n        if (result === null) return options?.defaultValue;\n        return reviveJSONParse(JSON.parse(result as any as string));\n      },\n      pull: async () => {\n        const resultRaw = await redis.MULTI().GET(key).DEL(key).EXEC();\n        const result = resultRaw[0];\n        if (result === null) return options?.defaultValue;\n        return reviveJSONParse(JSON.parse(result as any as string));\n      },\n      has: async (): Promise<boolean> => {\n        const result = await redis.GET(key);\n        return result !== null;\n      },\n      del: async () => {\n        await redis.DEL(key);\n      },\n    }),\n\n    useFetch: <Options extends MilkioRedisFetchOptions<any>>(key: string, options: Options) => {\n      if (options.expireMs !== null && (typeof options.expireMs !== 'number' || options.expireMs <= 0)) {\n        throw new Error('expireMs must be a positive number or null');\n      }\n\n      if (options.lockInterval !== undefined && (typeof options.lockInterval !== 'number' || options.lockInterval <= 0)) {\n        throw new Error('lockInterval must be a positive number if provided');\n      }\n\n      if (options.realExpireMs !== undefined && options.realExpireMs !== null && (typeof options.realExpireMs !== 'number' || options.realExpireMs <= 0)) {\n        throw new Error('realExpireMs must be a positive number or null if provided');\n      }\n\n      if (options.notFoundExpireMs !== undefined && options.notFoundExpireMs !== null && (typeof options.notFoundExpireMs !== 'number' || options.notFoundExpireMs <= 0)) {\n        throw new Error('notFoundExpireMs must be a positive number or null if provided');\n      }\n\n      if (options.refreshLockInterval !== undefined) {\n        if (typeof options.refreshLockInterval !== 'number' || options.refreshLockInterval <= 0) {\n          throw new Error('refreshLockInterval must be a positive number if provided');\n        }\n\n        const lockInterval = options.lockInterval ?? 8192;\n        if (options.refreshLockInterval <= lockInterval) {\n          throw new Error('refreshLockInterval must be greater than lockInterval');\n        }\n      }\n\n      if (typeof options.fetch !== 'function') {\n        throw new Error('fetch must be a function');\n      }\n\n      return {\n        has: async (): Promise<boolean> => {\n          const result = await redis.GET(key);\n          return result !== null;\n        },\n        del: async () => {\n          await redis.DEL(key);\n        },\n        fetch: async (): Promise<Awaited<ReturnType<Options['fetch']>>> => {\n          const lockInterval = options.lockInterval ?? 8192;\n\n          // 处理永久保存的情况\n          let realExpireMs: number | null = options.realExpireMs ?? null;\n          if (options.expireMs !== null) {\n            realExpireMs = options.realExpireMs ?? Math.floor(options.expireMs * (Math.random() + 0.5)) + 8192;\n          }\n\n          const notFoundExpireMs = options.notFoundExpireMs ?? (options.expireMs !== null ? Math.min(options.expireMs, 16384) : 86400000); // 默认1天\n\n          const resultRaw = await redis.GET(key);\n          const result = resultRaw ? (reviveJSONParse(JSON.parse(resultRaw as string)) as { T: number; R: Awaited<ReturnType<Options['fetch']>> }) : undefined;\n\n          const now = Date.now();\n          if (result && result.T > now) {\n            return result.R;\n          }\n\n          const refreshLockKey = `${key}:refresh-lock`;\n          if (await redis.EXISTS(refreshLockKey)) {\n            return result ? result.R : options?.defaultValue;\n          }\n\n          const lockKey = `${key}:lock`;\n          const lockSet = await redis.SET(lockKey, '1', { PX: lockInterval, NX: true });\n          const gotLock = lockSet === 'OK';\n\n          if (!gotLock) {\n            const maxWaitTime = lockInterval;\n            const waitInterval = 42;\n            let waited = 0;\n            while (waited < maxWaitTime) {\n              await new Promise((resolve) => setTimeout(resolve, waitInterval));\n              waited += waitInterval;\n              const waitedResultRaw = await redis.GET(key);\n              const waitedResult = waitedResultRaw ? (reviveJSONParse(JSON.parse(waitedResultRaw as string)) as { T: number; R: Awaited<ReturnType<Options['fetch']>> }) : undefined;\n              if (waitedResult && waitedResult.T > now) {\n                return waitedResult.R;\n              }\n              if (!(await redis.EXISTS(lockKey))) {\n                break;\n              }\n            }\n            const finalResultRaw = await redis.GET(key);\n            const finalResult = finalResultRaw ? (reviveJSONParse(JSON.parse(finalResultRaw as string)) as { T: number; R: Awaited<ReturnType<Options['fetch']>> }) : undefined;\n            if (finalResult && finalResult.T > now) {\n              return finalResult.R;\n            }\n            return result ? result.R : options.defaultValue;\n          }\n\n          const recheckRaw = await redis.GET(key);\n          const recheck = recheckRaw ? (reviveJSONParse(JSON.parse(recheckRaw as string)) as { T: number; R: Awaited<ReturnType<Options['fetch']>> }) : undefined;\n\n          if (recheck && recheck.T > now) {\n            return recheck.R;\n          }\n\n          let data: Awaited<ReturnType<Options['fetch']>>;\n          try {\n            data = await options.fetch();\n          } catch (error) {\n            await redis.DEL(lockKey);\n            throw error;\n          }\n\n          // 计算过期时间\n          let effectiveExpireMs: number | null = null;\n          if (data !== undefined) {\n            if (options.expireMs !== null) {\n              effectiveExpireMs = options.expireMs + (realExpireMs ?? 0);\n            }\n            // 如果 expireMs 为 null，effectiveExpireMs 保持 null（永久保存）\n          } else {\n            effectiveExpireMs = notFoundExpireMs;\n          }\n\n          const cacheValue = {\n            T: now + (data !== undefined ? (options.expireMs ?? Number.MAX_SAFE_INTEGER) : notFoundExpireMs),\n            R: data,\n          };\n\n          const refreshLockExists = await redis.EXISTS(refreshLockKey);\n          if (refreshLockExists) {\n            await redis.DEL(lockKey);\n            return data;\n          }\n\n          if (effectiveExpireMs === null) {\n            // 永久保存\n            await redis.MULTI().SET(key, JSON.stringify(cacheValue)).DEL(lockKey).EXEC();\n          } else {\n            await redis.MULTI().PSETEX(key, effectiveExpireMs, JSON.stringify(cacheValue)).DEL(lockKey).EXEC();\n          }\n\n          return data;\n        },\n        refresh: async (): Promise<void> => {\n          const refreshLockKey = `${key}:refresh-lock`;\n          const refreshLockInterval = options.refreshLockInterval ?? (options.lockInterval ?? 8192) + 1024;\n\n          await redis.SET(refreshLockKey, '1', {\n            PX: refreshLockInterval,\n          });\n\n          const now = Date.now();\n          let data: Awaited<ReturnType<Options['fetch']>> | undefined;\n          try {\n            data = await options.fetch();\n          } catch (error) {\n            await redis.DEL(refreshLockKey);\n            throw error;\n          }\n\n          // 处理永久保存的情况\n          let realExpireMs: number | null = options.realExpireMs ?? null;\n          if (options.expireMs !== null) {\n            realExpireMs = options.realExpireMs ?? Math.floor(options.expireMs * (Math.random() + 0.5)) + 8192;\n          }\n\n          // 计算过期时间\n          let effectiveExpireMs: number | null = null;\n          if (data !== undefined) {\n            if (options.expireMs !== null) {\n              effectiveExpireMs = options.expireMs + (realExpireMs ?? 0);\n            }\n            // 如果 expireMs 为 null，effectiveExpireMs 保持 null（永久保存）\n          } else {\n            effectiveExpireMs = options.notFoundExpireMs ?? (options.expireMs !== null ? Math.min(options.expireMs, 16384) : 86400000);\n          }\n\n          const cacheValue = {\n            T: now + (data !== undefined ? (options.expireMs ?? Number.MAX_SAFE_INTEGER) : (options.notFoundExpireMs ?? (options.expireMs !== null ? Math.min(options.expireMs, 16384) : 86400000))),\n            R: data,\n          };\n\n          if (effectiveExpireMs === null) {\n            // 永久保存\n            await redis.SET(key, JSON.stringify(cacheValue));\n          } else {\n            await redis.PSETEX(key, effectiveExpireMs, JSON.stringify(cacheValue));\n          }\n          // Do not clean the update lock! Instead, wait for it to expire naturally\n        },\n      };\n    },\n\n    useCount: (key: string) => ({\n      get: async (): Promise<number> => {\n        const result = await redis.GET(key);\n        return result ? Number(result) : 0;\n      },\n      add: async (amount: number, expireMs?: number | null): Promise<number> => {\n        if (expireMs === null) {\n          // 永久保存\n          const result = await redis.INCRBY(key, amount);\n          return result as number;\n        } else if (!expireMs) {\n          const result = await redis.INCRBY(key, amount);\n          return result as number;\n        } else {\n          const result = await redis.MULTI().INCRBY(key, amount).PEXPIRE(key, expireMs).EXEC();\n          return Number(result[0]) as number;\n        }\n      },\n      sub: async (amount: number): Promise<number> => {\n        const result = await redis.DECRBY(key, amount);\n        return result as number;\n      },\n    }),\n\n    useClockIn: (key: string, cleanDate: Date) => ({\n      clockIn: async (offset: number): Promise<void> => {\n        await redis.MULTI().SETBIT(key, offset, 1).PEXPIREAT(key, cleanDate.getTime()).EXEC();\n      },\n      check: async (offset: number): Promise<boolean> => {\n        const result = await redis.GETBIT(key, offset);\n        return result === 1;\n      },\n      firstClockIn: async (): Promise<number> => {\n        const result = await redis.BITPOS(key, 1);\n        return result as number;\n      },\n      lastClockIn: async (): Promise<number> => {\n        const result = await redis.BITPOS(key, 1, -1);\n        return result as number;\n      },\n      toArray: async (length: number): Promise<boolean[]> => {\n        const resultRaw = await redis.BITFIELD(key, [\n          {\n            operation: 'GET',\n            encoding: `u${length}`,\n            offset: '#0',\n          },\n        ]);\n        const result = Number.parseInt(`${resultRaw}`).toString(2).split('');\n        const fill = [];\n        for (let i = 0; i < length - result.length; i++) fill.push('0');\n        return [...fill, ...result].map((v) => v === '1');\n      },\n      count: async (): Promise<number> => {\n        const result = await redis.BITCOUNT(key);\n        return result as number;\n      },\n      clean: async (): Promise<void> => {\n        await redis.DEL(key);\n      },\n    }),\n  };\n\n  return milkioRedis;\n}\n\nconst isoDatePattern = /^(\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}(?:\\.\\d{1,3})?)(Z|[+-]\\d{2}:?\\d{2})?$/;\n\nexport function reviveJSONParse<T>(json: T): T {\n  if (json !== null && typeof json === 'object') {\n    if (json instanceof Date || (typeof (json as any).getTime === 'function' && typeof (json as any).toISOString === 'function')) {\n      return json;\n    }\n    if (Array.isArray(json)) {\n      return json.map((item) => reviveJSONParse(item)) as any;\n    }\n    return Object.entries(json).reduce((acc, [key, value]) => {\n      acc[key as keyof T] = reviveJSONParse(value);\n      return acc;\n    }, {} as T);\n  }\n  if (typeof json === 'string') {\n    const match = json.match(isoDatePattern);\n    if (match) {\n      const normalizedDateString = match[2] ? `${match[1]}${match[2].replace(':', '')}` : `${match[1]}Z`;\n      return new Date(normalizedDateString) as any;\n    }\n  }\n  return json;\n}\n\nexport type Redis = MilkioRedis;\n"
  ],
  "mappings": "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",
  "debugId": "512607529B76886364756E2164756E21",
  "names": []
}
|
package/package.json
CHANGED
|
@@ -1 +1,12 @@
|
|
|
1
|
-
{
|
|
1
|
+
{
|
|
2
|
+
"name": "@milkio/redis",
|
|
3
|
+
"version": "1.0.0",
|
|
4
|
+
"type": "module",
|
|
5
|
+
"exports": "./index.js",
|
|
6
|
+
"types": "./index.d.ts",
|
|
7
|
+
"dependencies": {},
|
|
8
|
+
"repository": {
|
|
9
|
+
"type": "git",
|
|
10
|
+
"url": "https://github.com/akirarika/milkio"
|
|
11
|
+
}
|
|
12
|
+
}
|
package/chunk-6sxyen7j.js
DELETED
|
@@ -1,24 +0,0 @@
|
|
|
1
|
-
import { createRequire } from "node:module";
|
|
2
|
-
var __create = Object.create;
|
|
3
|
-
var __getProtoOf = Object.getPrototypeOf;
|
|
4
|
-
var __defProp = Object.defineProperty;
|
|
5
|
-
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
-
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
7
|
-
var __toESM = (mod, isNodeMode, target) => {
|
|
8
|
-
target = mod != null ? __create(__getProtoOf(mod)) : {};
|
|
9
|
-
const to = isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target;
|
|
10
|
-
for (let key of __getOwnPropNames(mod))
|
|
11
|
-
if (!__hasOwnProp.call(to, key))
|
|
12
|
-
__defProp(to, key, {
|
|
13
|
-
get: () => mod[key],
|
|
14
|
-
enumerable: true
|
|
15
|
-
});
|
|
16
|
-
return to;
|
|
17
|
-
};
|
|
18
|
-
var __commonJS = (cb, mod) => () => (mod || cb((mod = { exports: {} }).exports, mod), mod.exports);
|
|
19
|
-
var __require = /* @__PURE__ */ createRequire(import.meta.url);
|
|
20
|
-
|
|
21
|
-
export { __toESM, __commonJS, __require };
|
|
22
|
-
|
|
23
|
-
//# debugId=21F72A06ECE4C28164756E2164756E21
|
|
24
|
-
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFtdLAogICJzb3VyY2VzQ29udGVudCI6IFsKICBdLAogICJtYXBwaW5ncyI6ICIiLAogICJkZWJ1Z0lkIjogIjIxRjcyQTA2RUNFNEMyODE2NDc1NkUyMTY0NzU2RTIxIiwKICAibmFtZXMiOiBbXQp9
|