@milkio/redis 1.0.0-beta.205 → 1.0.0-beta.206
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/index.d.ts +1 -0
- package/index.js +33 -10
- package/package.json +1 -1
package/index.d.ts
CHANGED
package/index.js
CHANGED
|
@@ -86,8 +86,29 @@ async function createRedis(redisClient) {
|
|
|
86
86
|
const lockKey = `${key}:lock`;
|
|
87
87
|
const lockSet = await redis.SET(lockKey, "1", { PX: lockInterval, NX: true });
|
|
88
88
|
const gotLock = lockSet === "OK";
|
|
89
|
-
if (!gotLock)
|
|
89
|
+
if (!gotLock) {
|
|
90
|
+
const maxWaitTime = lockInterval;
|
|
91
|
+
const waitInterval = 16;
|
|
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
|
+
}
|
|
90
110
|
return result ? result.R : options.defaultValue;
|
|
111
|
+
}
|
|
91
112
|
const recheckRaw = await redis.GET(key);
|
|
92
113
|
const recheck = recheckRaw ? reviveJSONParse(JSON.parse(recheckRaw)) : undefined;
|
|
93
114
|
if (recheck && recheck.T > now) {
|
|
@@ -225,14 +246,15 @@ async function createRedis(redisClient) {
|
|
|
225
246
|
};
|
|
226
247
|
return milkioRedis;
|
|
227
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})?$/;
|
|
228
250
|
function reviveJSONParse(json) {
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
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
|
+
}
|
|
236
258
|
return Object.entries(json).reduce((acc, [key, value]) => {
|
|
237
259
|
acc[key] = reviveJSONParse(value);
|
|
238
260
|
return acc;
|
|
@@ -248,8 +270,9 @@ function reviveJSONParse(json) {
|
|
|
248
270
|
return json;
|
|
249
271
|
}
|
|
250
272
|
export {
|
|
273
|
+
reviveJSONParse,
|
|
251
274
|
createRedis
|
|
252
275
|
};
|
|
253
276
|
|
|
254
|
-
//# debugId=
|
|
255
|
-
//# 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>(key: string, options?: MilkioRedisCacheOptions<T>) => {\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>(key: string, options: MilkioRedisFetchOptions<T>) => {\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: (key: string, cleanDate: Date) => {\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) return result ? result.R : options.defaultValue;\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\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 = MilkioRedis;"
  ],
  "mappings": "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",
  "debugId": "840E4CE7CC3CCA9864756E2164756E21",
  "names": []
}
|
|
277
|
+
//# debugId=17D453A4610A3E8E64756E2164756E21
|
|
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>(key: string, options?: MilkioRedisCacheOptions<T>) => {\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>(key: string, options: MilkioRedisFetchOptions<T>) => {\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: (key: string, cleanDate: Date) => {\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 = 16;\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\n\nexport type Redis = MilkioRedis;"
  ],
  "mappings": "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",
  "debugId": "17D453A4610A3E8E64756E2164756E21",
  "names": []
}
|