@lukaskj/ts-utils 1.0.2 → 1.0.3
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/README.md +1 -0
- package/dist/cache/cacheService.d.ts +19 -0
- package/dist/cache/constants.d.ts +1 -0
- package/dist/cache/index.cjs +182 -0
- package/dist/cache/index.d.ts +3 -0
- package/dist/cache/index.mjs +137 -0
- package/dist/cache/internal/types.d.ts +8 -0
- package/dist/cache/types.d.ts +10 -0
- package/dist/index.cjs +170 -11
- package/dist/index.d.ts +9 -5
- package/dist/index.mjs +162 -11
- package/dist/utils/isCallable.d.ts +1 -0
- package/dist/utils/isPromise.d.ts +6 -0
- package/dist/utils/sleep.d.ts +1 -0
- package/package.json +12 -1
- /package/dist/{chunksToLines.d.ts → utils/chunksToLines.d.ts} +0 -0
- /package/dist/{isNullOrUndefined.d.ts → utils/isNullOrUndefined.d.ts} +0 -0
- /package/dist/{isObject.d.ts → utils/isObject.d.ts} +0 -0
- /package/dist/{isPrimitive.d.ts → utils/isPrimitive.d.ts} +0 -0
- /package/dist/{jsonParser.d.ts → utils/jsonParser.d.ts} +0 -0
- /package/dist/{randomString.d.ts → utils/randomString.d.ts} +0 -0
- /package/dist/{readStdinAsync.d.ts → utils/readStdinAsync.d.ts} +0 -0
package/README.md
CHANGED
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import type { CacheMetadataOptions, CacheOptions, ValueLoader } from "./internal/types.ts";
|
|
2
|
+
import type { ICacheAdapter } from "./types.ts";
|
|
3
|
+
export declare class Cache<TAdapter extends ICacheAdapter | undefined = undefined> {
|
|
4
|
+
private inMemoryCache;
|
|
5
|
+
private readonly options;
|
|
6
|
+
private readonly adapter?;
|
|
7
|
+
constructor(defaultOptions?: Partial<CacheOptions>);
|
|
8
|
+
constructor(adapter?: TAdapter, defaultOptions?: Partial<CacheOptions>);
|
|
9
|
+
private getFromAdapter;
|
|
10
|
+
private saveToAdapter;
|
|
11
|
+
get<T>(cacheKey: string, valueLoader?: ValueLoader<T>, options?: Partial<CacheMetadataOptions>): Promise<T | undefined>;
|
|
12
|
+
private mergeOptions;
|
|
13
|
+
private tryGetFromMemory;
|
|
14
|
+
private tryGetFromAdapter;
|
|
15
|
+
private tryGetFromValueLoader;
|
|
16
|
+
private cacheValue;
|
|
17
|
+
private createCacheEntry;
|
|
18
|
+
private isExpired;
|
|
19
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare const CACHE_TTL_NEVER_EXPIRE = -1;
|
|
@@ -0,0 +1,182 @@
|
|
|
1
|
+
var __create = Object.create;
|
|
2
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
3
|
+
var __defProp = Object.defineProperty;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
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 __moduleCache = /* @__PURE__ */ new WeakMap;
|
|
19
|
+
var __toCommonJS = (from) => {
|
|
20
|
+
var entry = __moduleCache.get(from), desc;
|
|
21
|
+
if (entry)
|
|
22
|
+
return entry;
|
|
23
|
+
entry = __defProp({}, "__esModule", { value: true });
|
|
24
|
+
if (from && typeof from === "object" || typeof from === "function")
|
|
25
|
+
__getOwnPropNames(from).map((key) => !__hasOwnProp.call(entry, key) && __defProp(entry, key, {
|
|
26
|
+
get: () => from[key],
|
|
27
|
+
enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
|
|
28
|
+
}));
|
|
29
|
+
__moduleCache.set(from, entry);
|
|
30
|
+
return entry;
|
|
31
|
+
};
|
|
32
|
+
var __export = (target, all) => {
|
|
33
|
+
for (var name in all)
|
|
34
|
+
__defProp(target, name, {
|
|
35
|
+
get: all[name],
|
|
36
|
+
enumerable: true,
|
|
37
|
+
configurable: true,
|
|
38
|
+
set: (newValue) => all[name] = () => newValue
|
|
39
|
+
});
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
// src/cache/index.ts
|
|
43
|
+
var exports_cache = {};
|
|
44
|
+
__export(exports_cache, {
|
|
45
|
+
Cache: () => Cache,
|
|
46
|
+
CACHE_TTL_NEVER_EXPIRE: () => CACHE_TTL_NEVER_EXPIRE
|
|
47
|
+
});
|
|
48
|
+
module.exports = __toCommonJS(exports_cache);
|
|
49
|
+
|
|
50
|
+
// src/cache/cacheService.ts
|
|
51
|
+
var import_types = require("node:util/types");
|
|
52
|
+
|
|
53
|
+
// src/utils/isCallable.ts
|
|
54
|
+
function isCallable(value) {
|
|
55
|
+
return typeof value === "function";
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
// src/cache/cacheService.ts
|
|
59
|
+
var defaultCacheOptions = {
|
|
60
|
+
expirationThresholdMs: 0,
|
|
61
|
+
ttlMs: 60 * 60 * 1000
|
|
62
|
+
};
|
|
63
|
+
|
|
64
|
+
class Cache {
|
|
65
|
+
inMemoryCache = new Map;
|
|
66
|
+
options;
|
|
67
|
+
adapter;
|
|
68
|
+
constructor(adapter, defaultOptions) {
|
|
69
|
+
const options = adapter && !("getValue" in adapter) ? adapter : defaultOptions;
|
|
70
|
+
if (adapter && "getValue" in adapter && "setValue" in adapter) {
|
|
71
|
+
this.adapter = adapter;
|
|
72
|
+
}
|
|
73
|
+
this.options = {
|
|
74
|
+
...defaultCacheOptions,
|
|
75
|
+
...options
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
async getFromAdapter(cacheKey) {
|
|
79
|
+
if (!this.adapter) {
|
|
80
|
+
return;
|
|
81
|
+
}
|
|
82
|
+
return this.adapter.getValue(cacheKey);
|
|
83
|
+
}
|
|
84
|
+
async saveToAdapter(cacheKey, value) {
|
|
85
|
+
if (!this.adapter) {
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
return this.adapter.setValue(cacheKey, value);
|
|
89
|
+
}
|
|
90
|
+
async get(cacheKey, valueLoader, options = {}) {
|
|
91
|
+
const mergedOptions = this.mergeOptions(options);
|
|
92
|
+
const cachedValue = this.tryGetFromMemory(cacheKey);
|
|
93
|
+
if (cachedValue !== undefined) {
|
|
94
|
+
return cachedValue;
|
|
95
|
+
}
|
|
96
|
+
const freshValue = await this.tryGetFromValueLoader(valueLoader);
|
|
97
|
+
if (freshValue !== undefined) {
|
|
98
|
+
await this.cacheValue(cacheKey, freshValue, mergedOptions);
|
|
99
|
+
return freshValue;
|
|
100
|
+
}
|
|
101
|
+
const adapterValue = await this.tryGetFromAdapter(cacheKey);
|
|
102
|
+
if (adapterValue !== undefined) {
|
|
103
|
+
await this.cacheValue(cacheKey, adapterValue, mergedOptions);
|
|
104
|
+
return adapterValue;
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
mergeOptions(options) {
|
|
108
|
+
return {
|
|
109
|
+
ttlMs: this.options.ttlMs,
|
|
110
|
+
expirationThresholdMs: this.options.expirationThresholdMs,
|
|
111
|
+
...options
|
|
112
|
+
};
|
|
113
|
+
}
|
|
114
|
+
tryGetFromMemory(cacheKey) {
|
|
115
|
+
const cached = this.inMemoryCache.get(cacheKey);
|
|
116
|
+
if (cached && !this.isExpired(cached.metadata)) {
|
|
117
|
+
return cached.data;
|
|
118
|
+
}
|
|
119
|
+
return;
|
|
120
|
+
}
|
|
121
|
+
async tryGetFromAdapter(cacheKey) {
|
|
122
|
+
const cached = await this.getFromAdapter(cacheKey);
|
|
123
|
+
if (cached && !this.isExpired(cached.metadata)) {
|
|
124
|
+
this.inMemoryCache.set(cacheKey, cached);
|
|
125
|
+
return cached.data;
|
|
126
|
+
}
|
|
127
|
+
return;
|
|
128
|
+
}
|
|
129
|
+
async tryGetFromValueLoader(valueLoader) {
|
|
130
|
+
if (valueLoader === undefined) {
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
133
|
+
if (isCallable(valueLoader)) {
|
|
134
|
+
return await valueLoader();
|
|
135
|
+
}
|
|
136
|
+
if (import_types.isPromise(valueLoader)) {
|
|
137
|
+
return await valueLoader;
|
|
138
|
+
}
|
|
139
|
+
return valueLoader;
|
|
140
|
+
}
|
|
141
|
+
async cacheValue(cacheKey, value, options) {
|
|
142
|
+
const cacheData = this.createCacheEntry(value, options);
|
|
143
|
+
await this.saveToAdapter(cacheKey, cacheData);
|
|
144
|
+
this.inMemoryCache.set(cacheKey, cacheData);
|
|
145
|
+
}
|
|
146
|
+
createCacheEntry(value, options) {
|
|
147
|
+
const now = Date.now();
|
|
148
|
+
let expiresAt;
|
|
149
|
+
if (options.ttlMs < 0) {
|
|
150
|
+
expiresAt = -1;
|
|
151
|
+
} else {
|
|
152
|
+
expiresAt = now + options.ttlMs - options.expirationThresholdMs;
|
|
153
|
+
}
|
|
154
|
+
return {
|
|
155
|
+
data: value,
|
|
156
|
+
metadata: {
|
|
157
|
+
createdAt: now,
|
|
158
|
+
ttlMs: options.ttlMs,
|
|
159
|
+
expirationThresholdMs: options.expirationThresholdMs,
|
|
160
|
+
expiresAt
|
|
161
|
+
}
|
|
162
|
+
};
|
|
163
|
+
}
|
|
164
|
+
isExpired(metadata) {
|
|
165
|
+
if (!metadata) {
|
|
166
|
+
return true;
|
|
167
|
+
}
|
|
168
|
+
if (metadata.expiresAt < 0) {
|
|
169
|
+
return false;
|
|
170
|
+
}
|
|
171
|
+
const expirationTime = metadata.expiresAt ?? (metadata.createdAt && metadata.ttlMs ? metadata.createdAt + metadata.ttlMs : undefined);
|
|
172
|
+
if (expirationTime === undefined) {
|
|
173
|
+
return true;
|
|
174
|
+
}
|
|
175
|
+
return Date.now() > expirationTime;
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
// src/cache/constants.ts
|
|
179
|
+
var CACHE_TTL_NEVER_EXPIRE = -1;
|
|
180
|
+
|
|
181
|
+
//# debugId=F6FFA1F34A10A7B264756E2164756E21
|
|
182
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/cache/cacheService.ts", "../src/utils/isCallable.ts", "../src/cache/constants.ts"],
  "sourcesContent": [
    "import { isPromise } from \"node:util/types\";\nimport { isCallable } from \"../utils/isCallable.ts\";\nimport type { CacheMetadataOptions, CacheOptions, CacheWithMetadata, ValueLoader } from \"./internal/types.ts\";\nimport type { CacheMetadata, ICacheAdapter } from \"./types.ts\";\n\nconst defaultCacheOptions: CacheOptions = {\n  expirationThresholdMs: 0,\n  ttlMs: 60 * 60 * 1000,\n};\n\nexport class Cache<TAdapter extends ICacheAdapter | undefined = undefined> {\n  private inMemoryCache: Map<string, CacheWithMetadata> = new Map();\n  private readonly options: CacheOptions;\n  private readonly adapter?: TAdapter;\n\n  constructor(defaultOptions?: Partial<CacheOptions>);\n  constructor(adapter?: TAdapter, defaultOptions?: Partial<CacheOptions>);\n  constructor(adapter?: TAdapter | Partial<CacheOptions>, defaultOptions?: Partial<CacheOptions>) {\n    const options = adapter && !(\"getValue\" in adapter) ? (adapter as Partial<CacheOptions>) : defaultOptions;\n    if (adapter && \"getValue\" in adapter && \"setValue\" in adapter) {\n      this.adapter = adapter as TAdapter;\n    }\n\n    this.options = {\n      ...defaultCacheOptions,\n      ...options,\n    };\n  }\n\n  private async getFromAdapter<T>(cacheKey: string): Promise<T | undefined> {\n    if (!this.adapter) {\n      return undefined;\n    }\n\n    return this.adapter.getValue(cacheKey);\n  }\n\n  private async saveToAdapter<T>(cacheKey: string, value: T): Promise<void> {\n    if (!this.adapter) {\n      return;\n    }\n\n    return this.adapter.setValue<T>(cacheKey, value);\n  }\n\n  public async get<T>(\n    cacheKey: string,\n    valueLoader?: ValueLoader<T>,\n    options: Partial<CacheMetadataOptions> = {},\n  ): Promise<T | undefined> {\n    const mergedOptions = this.mergeOptions(options);\n\n    // Try in-memory cache first\n    const cachedValue = this.tryGetFromMemory<T>(cacheKey);\n    if (cachedValue !== undefined) {\n      return cachedValue;\n    }\n\n    // Load fresh value from loader function first\n    const freshValue = await this.tryGetFromValueLoader(valueLoader);\n    if (freshValue !== undefined) {\n      await this.cacheValue(cacheKey, freshValue, mergedOptions);\n\n      return freshValue;\n    }\n\n    // Try adapter cache second\n    const adapterValue = await this.tryGetFromAdapter<T>(cacheKey);\n    if (adapterValue !== undefined) {\n      await this.cacheValue(cacheKey, adapterValue, mergedOptions);\n\n      return adapterValue;\n    }\n  }\n\n  private mergeOptions(options: Partial<CacheMetadataOptions>): CacheMetadataOptions {\n    return {\n      ttlMs: this.options.ttlMs,\n      expirationThresholdMs: this.options.expirationThresholdMs,\n      ...options,\n    };\n  }\n\n  private tryGetFromMemory<T>(cacheKey: string): T | undefined {\n    const cached = this.inMemoryCache.get(cacheKey);\n\n    if (cached && !this.isExpired(cached.metadata)) {\n      return cached.data as T;\n    }\n\n    return undefined;\n  }\n\n  private async tryGetFromAdapter<T>(cacheKey: string): Promise<T | undefined> {\n    const cached = await this.getFromAdapter<CacheWithMetadata<T>>(cacheKey);\n\n    if (cached && !this.isExpired(cached.metadata)) {\n      this.inMemoryCache.set(cacheKey, cached);\n      return cached.data;\n    }\n\n    return undefined;\n  }\n\n  private async tryGetFromValueLoader<T>(valueLoader?: ValueLoader<T>): Promise<T | undefined> {\n    if (valueLoader === undefined) {\n      return undefined;\n    }\n\n    if (isCallable(valueLoader)) {\n      return await valueLoader();\n    }\n\n    if (isPromise(valueLoader)) {\n      return await valueLoader;\n    }\n\n    return valueLoader;\n  }\n\n  private async cacheValue<T>(cacheKey: string, value: T, options: CacheMetadataOptions): Promise<void> {\n    const cacheData = this.createCacheEntry(value, options);\n\n    await this.saveToAdapter(cacheKey, cacheData);\n    this.inMemoryCache.set(cacheKey, cacheData);\n  }\n\n  private createCacheEntry<T>(value: T, options: CacheMetadataOptions): CacheWithMetadata<T> {\n    const now = Date.now();\n    let expiresAt: number;\n    if (options.ttlMs < 0) {\n      expiresAt = -1;\n    } else {\n      expiresAt = now + options.ttlMs - options.expirationThresholdMs;\n    }\n\n    return {\n      data: value,\n      metadata: {\n        createdAt: now,\n        ttlMs: options.ttlMs,\n        expirationThresholdMs: options.expirationThresholdMs,\n        expiresAt,\n      },\n    };\n  }\n\n  private isExpired(metadata?: CacheMetadata): boolean {\n    if (!metadata) {\n      return true;\n    }\n\n    if (metadata.expiresAt < 0) {\n      return false;\n    }\n\n    const expirationTime =\n      metadata.expiresAt ?? (metadata.createdAt && metadata.ttlMs ? metadata.createdAt + metadata.ttlMs : undefined);\n\n    if (expirationTime === undefined) {\n      return true;\n    }\n\n    return Date.now() > expirationTime;\n  }\n}\n",
    "export function isCallable(value: any): value is CallableFunction {\n  return typeof value === \"function\";\n}\n",
    "export const CACHE_TTL_NEVER_EXPIRE = -1;\n"
  ],
  "mappings": "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",
  "debugId": "F6FFA1F34A10A7B264756E2164756E21",
  "names": []
}
|
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
// src/cache/cacheService.ts
|
|
2
|
+
import { isPromise } from "node:util/types";
|
|
3
|
+
|
|
4
|
+
// src/utils/isCallable.ts
|
|
5
|
+
function isCallable(value) {
|
|
6
|
+
return typeof value === "function";
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
// src/cache/cacheService.ts
|
|
10
|
+
var defaultCacheOptions = {
|
|
11
|
+
expirationThresholdMs: 0,
|
|
12
|
+
ttlMs: 60 * 60 * 1000
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
class Cache {
|
|
16
|
+
inMemoryCache = new Map;
|
|
17
|
+
options;
|
|
18
|
+
adapter;
|
|
19
|
+
constructor(adapter, defaultOptions) {
|
|
20
|
+
const options = adapter && !("getValue" in adapter) ? adapter : defaultOptions;
|
|
21
|
+
if (adapter && "getValue" in adapter && "setValue" in adapter) {
|
|
22
|
+
this.adapter = adapter;
|
|
23
|
+
}
|
|
24
|
+
this.options = {
|
|
25
|
+
...defaultCacheOptions,
|
|
26
|
+
...options
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
async getFromAdapter(cacheKey) {
|
|
30
|
+
if (!this.adapter) {
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
return this.adapter.getValue(cacheKey);
|
|
34
|
+
}
|
|
35
|
+
async saveToAdapter(cacheKey, value) {
|
|
36
|
+
if (!this.adapter) {
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
return this.adapter.setValue(cacheKey, value);
|
|
40
|
+
}
|
|
41
|
+
async get(cacheKey, valueLoader, options = {}) {
|
|
42
|
+
const mergedOptions = this.mergeOptions(options);
|
|
43
|
+
const cachedValue = this.tryGetFromMemory(cacheKey);
|
|
44
|
+
if (cachedValue !== undefined) {
|
|
45
|
+
return cachedValue;
|
|
46
|
+
}
|
|
47
|
+
const freshValue = await this.tryGetFromValueLoader(valueLoader);
|
|
48
|
+
if (freshValue !== undefined) {
|
|
49
|
+
await this.cacheValue(cacheKey, freshValue, mergedOptions);
|
|
50
|
+
return freshValue;
|
|
51
|
+
}
|
|
52
|
+
const adapterValue = await this.tryGetFromAdapter(cacheKey);
|
|
53
|
+
if (adapterValue !== undefined) {
|
|
54
|
+
await this.cacheValue(cacheKey, adapterValue, mergedOptions);
|
|
55
|
+
return adapterValue;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
mergeOptions(options) {
|
|
59
|
+
return {
|
|
60
|
+
ttlMs: this.options.ttlMs,
|
|
61
|
+
expirationThresholdMs: this.options.expirationThresholdMs,
|
|
62
|
+
...options
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
tryGetFromMemory(cacheKey) {
|
|
66
|
+
const cached = this.inMemoryCache.get(cacheKey);
|
|
67
|
+
if (cached && !this.isExpired(cached.metadata)) {
|
|
68
|
+
return cached.data;
|
|
69
|
+
}
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
async tryGetFromAdapter(cacheKey) {
|
|
73
|
+
const cached = await this.getFromAdapter(cacheKey);
|
|
74
|
+
if (cached && !this.isExpired(cached.metadata)) {
|
|
75
|
+
this.inMemoryCache.set(cacheKey, cached);
|
|
76
|
+
return cached.data;
|
|
77
|
+
}
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
async tryGetFromValueLoader(valueLoader) {
|
|
81
|
+
if (valueLoader === undefined) {
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
if (isCallable(valueLoader)) {
|
|
85
|
+
return await valueLoader();
|
|
86
|
+
}
|
|
87
|
+
if (isPromise(valueLoader)) {
|
|
88
|
+
return await valueLoader;
|
|
89
|
+
}
|
|
90
|
+
return valueLoader;
|
|
91
|
+
}
|
|
92
|
+
async cacheValue(cacheKey, value, options) {
|
|
93
|
+
const cacheData = this.createCacheEntry(value, options);
|
|
94
|
+
await this.saveToAdapter(cacheKey, cacheData);
|
|
95
|
+
this.inMemoryCache.set(cacheKey, cacheData);
|
|
96
|
+
}
|
|
97
|
+
createCacheEntry(value, options) {
|
|
98
|
+
const now = Date.now();
|
|
99
|
+
let expiresAt;
|
|
100
|
+
if (options.ttlMs < 0) {
|
|
101
|
+
expiresAt = -1;
|
|
102
|
+
} else {
|
|
103
|
+
expiresAt = now + options.ttlMs - options.expirationThresholdMs;
|
|
104
|
+
}
|
|
105
|
+
return {
|
|
106
|
+
data: value,
|
|
107
|
+
metadata: {
|
|
108
|
+
createdAt: now,
|
|
109
|
+
ttlMs: options.ttlMs,
|
|
110
|
+
expirationThresholdMs: options.expirationThresholdMs,
|
|
111
|
+
expiresAt
|
|
112
|
+
}
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
isExpired(metadata) {
|
|
116
|
+
if (!metadata) {
|
|
117
|
+
return true;
|
|
118
|
+
}
|
|
119
|
+
if (metadata.expiresAt < 0) {
|
|
120
|
+
return false;
|
|
121
|
+
}
|
|
122
|
+
const expirationTime = metadata.expiresAt ?? (metadata.createdAt && metadata.ttlMs ? metadata.createdAt + metadata.ttlMs : undefined);
|
|
123
|
+
if (expirationTime === undefined) {
|
|
124
|
+
return true;
|
|
125
|
+
}
|
|
126
|
+
return Date.now() > expirationTime;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
// src/cache/constants.ts
|
|
130
|
+
var CACHE_TTL_NEVER_EXPIRE = -1;
|
|
131
|
+
export {
|
|
132
|
+
Cache,
|
|
133
|
+
CACHE_TTL_NEVER_EXPIRE
|
|
134
|
+
};
|
|
135
|
+
|
|
136
|
+
//# debugId=46944600E0400D9A64756E2164756E21
|
|
137
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/cache/cacheService.ts", "../src/utils/isCallable.ts", "../src/cache/constants.ts"],
  "sourcesContent": [
    "import { isPromise } from \"node:util/types\";\nimport { isCallable } from \"../utils/isCallable.ts\";\nimport type { CacheMetadataOptions, CacheOptions, CacheWithMetadata, ValueLoader } from \"./internal/types.ts\";\nimport type { CacheMetadata, ICacheAdapter } from \"./types.ts\";\n\nconst defaultCacheOptions: CacheOptions = {\n  expirationThresholdMs: 0,\n  ttlMs: 60 * 60 * 1000,\n};\n\nexport class Cache<TAdapter extends ICacheAdapter | undefined = undefined> {\n  private inMemoryCache: Map<string, CacheWithMetadata> = new Map();\n  private readonly options: CacheOptions;\n  private readonly adapter?: TAdapter;\n\n  constructor(defaultOptions?: Partial<CacheOptions>);\n  constructor(adapter?: TAdapter, defaultOptions?: Partial<CacheOptions>);\n  constructor(adapter?: TAdapter | Partial<CacheOptions>, defaultOptions?: Partial<CacheOptions>) {\n    const options = adapter && !(\"getValue\" in adapter) ? (adapter as Partial<CacheOptions>) : defaultOptions;\n    if (adapter && \"getValue\" in adapter && \"setValue\" in adapter) {\n      this.adapter = adapter as TAdapter;\n    }\n\n    this.options = {\n      ...defaultCacheOptions,\n      ...options,\n    };\n  }\n\n  private async getFromAdapter<T>(cacheKey: string): Promise<T | undefined> {\n    if (!this.adapter) {\n      return undefined;\n    }\n\n    return this.adapter.getValue(cacheKey);\n  }\n\n  private async saveToAdapter<T>(cacheKey: string, value: T): Promise<void> {\n    if (!this.adapter) {\n      return;\n    }\n\n    return this.adapter.setValue<T>(cacheKey, value);\n  }\n\n  public async get<T>(\n    cacheKey: string,\n    valueLoader?: ValueLoader<T>,\n    options: Partial<CacheMetadataOptions> = {},\n  ): Promise<T | undefined> {\n    const mergedOptions = this.mergeOptions(options);\n\n    // Try in-memory cache first\n    const cachedValue = this.tryGetFromMemory<T>(cacheKey);\n    if (cachedValue !== undefined) {\n      return cachedValue;\n    }\n\n    // Load fresh value from loader function first\n    const freshValue = await this.tryGetFromValueLoader(valueLoader);\n    if (freshValue !== undefined) {\n      await this.cacheValue(cacheKey, freshValue, mergedOptions);\n\n      return freshValue;\n    }\n\n    // Try adapter cache second\n    const adapterValue = await this.tryGetFromAdapter<T>(cacheKey);\n    if (adapterValue !== undefined) {\n      await this.cacheValue(cacheKey, adapterValue, mergedOptions);\n\n      return adapterValue;\n    }\n  }\n\n  private mergeOptions(options: Partial<CacheMetadataOptions>): CacheMetadataOptions {\n    return {\n      ttlMs: this.options.ttlMs,\n      expirationThresholdMs: this.options.expirationThresholdMs,\n      ...options,\n    };\n  }\n\n  private tryGetFromMemory<T>(cacheKey: string): T | undefined {\n    const cached = this.inMemoryCache.get(cacheKey);\n\n    if (cached && !this.isExpired(cached.metadata)) {\n      return cached.data as T;\n    }\n\n    return undefined;\n  }\n\n  private async tryGetFromAdapter<T>(cacheKey: string): Promise<T | undefined> {\n    const cached = await this.getFromAdapter<CacheWithMetadata<T>>(cacheKey);\n\n    if (cached && !this.isExpired(cached.metadata)) {\n      this.inMemoryCache.set(cacheKey, cached);\n      return cached.data;\n    }\n\n    return undefined;\n  }\n\n  private async tryGetFromValueLoader<T>(valueLoader?: ValueLoader<T>): Promise<T | undefined> {\n    if (valueLoader === undefined) {\n      return undefined;\n    }\n\n    if (isCallable(valueLoader)) {\n      return await valueLoader();\n    }\n\n    if (isPromise(valueLoader)) {\n      return await valueLoader;\n    }\n\n    return valueLoader;\n  }\n\n  private async cacheValue<T>(cacheKey: string, value: T, options: CacheMetadataOptions): Promise<void> {\n    const cacheData = this.createCacheEntry(value, options);\n\n    await this.saveToAdapter(cacheKey, cacheData);\n    this.inMemoryCache.set(cacheKey, cacheData);\n  }\n\n  private createCacheEntry<T>(value: T, options: CacheMetadataOptions): CacheWithMetadata<T> {\n    const now = Date.now();\n    let expiresAt: number;\n    if (options.ttlMs < 0) {\n      expiresAt = -1;\n    } else {\n      expiresAt = now + options.ttlMs - options.expirationThresholdMs;\n    }\n\n    return {\n      data: value,\n      metadata: {\n        createdAt: now,\n        ttlMs: options.ttlMs,\n        expirationThresholdMs: options.expirationThresholdMs,\n        expiresAt,\n      },\n    };\n  }\n\n  private isExpired(metadata?: CacheMetadata): boolean {\n    if (!metadata) {\n      return true;\n    }\n\n    if (metadata.expiresAt < 0) {\n      return false;\n    }\n\n    const expirationTime =\n      metadata.expiresAt ?? (metadata.createdAt && metadata.ttlMs ? metadata.createdAt + metadata.ttlMs : undefined);\n\n    if (expirationTime === undefined) {\n      return true;\n    }\n\n    return Date.now() > expirationTime;\n  }\n}\n",
    "export function isCallable(value: any): value is CallableFunction {\n  return typeof value === \"function\";\n}\n",
    "export const CACHE_TTL_NEVER_EXPIRE = -1;\n"
  ],
  "mappings": "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",
  "debugId": "46944600E0400D9A64756E2164756E21",
  "names": []
}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import type { CacheMetadata } from "../types.ts";
|
|
2
|
+
export type CacheWithMetadata<T = any> = {
|
|
3
|
+
data: T;
|
|
4
|
+
metadata: CacheMetadata;
|
|
5
|
+
};
|
|
6
|
+
export type CacheMetadataOptions = Pick<CacheMetadata, "ttlMs" | "expirationThresholdMs">;
|
|
7
|
+
export type CacheOptions = CacheMetadataOptions & {};
|
|
8
|
+
export type ValueLoader<T> = T | Promise<T> | (() => Promise<T>);
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export interface ICacheAdapter {
|
|
2
|
+
getValue<T>(key: string): T | undefined | Promise<T | undefined>;
|
|
3
|
+
setValue<T>(key: string, value: T): void | Promise<void>;
|
|
4
|
+
}
|
|
5
|
+
export type CacheMetadata = {
|
|
6
|
+
createdAt: number;
|
|
7
|
+
ttlMs: number;
|
|
8
|
+
expiresAt: number;
|
|
9
|
+
expirationThresholdMs: number;
|
|
10
|
+
};
|
package/dist/index.cjs
CHANGED
|
@@ -39,20 +39,163 @@ var __export = (target, all) => {
|
|
|
39
39
|
});
|
|
40
40
|
};
|
|
41
41
|
|
|
42
|
+
// src/cache/index.ts
|
|
43
|
+
var exports_cache = {};
|
|
44
|
+
__export(exports_cache, {
|
|
45
|
+
Cache: () => Cache,
|
|
46
|
+
CACHE_TTL_NEVER_EXPIRE: () => CACHE_TTL_NEVER_EXPIRE
|
|
47
|
+
});
|
|
48
|
+
module.exports = __toCommonJS(exports_cache);
|
|
49
|
+
|
|
50
|
+
// src/cache/cacheService.ts
|
|
51
|
+
var import_types = require("node:util/types");
|
|
52
|
+
|
|
53
|
+
// src/utils/isCallable.ts
|
|
54
|
+
function isCallable(value) {
|
|
55
|
+
return typeof value === "function";
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
// src/cache/cacheService.ts
|
|
59
|
+
var defaultCacheOptions = {
|
|
60
|
+
expirationThresholdMs: 0,
|
|
61
|
+
ttlMs: 60 * 60 * 1000
|
|
62
|
+
};
|
|
63
|
+
|
|
64
|
+
class Cache {
|
|
65
|
+
inMemoryCache = new Map;
|
|
66
|
+
options;
|
|
67
|
+
adapter;
|
|
68
|
+
constructor(adapter, defaultOptions) {
|
|
69
|
+
const options = adapter && !("getValue" in adapter) ? adapter : defaultOptions;
|
|
70
|
+
if (adapter && "getValue" in adapter && "setValue" in adapter) {
|
|
71
|
+
this.adapter = adapter;
|
|
72
|
+
}
|
|
73
|
+
this.options = {
|
|
74
|
+
...defaultCacheOptions,
|
|
75
|
+
...options
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
async getFromAdapter(cacheKey) {
|
|
79
|
+
if (!this.adapter) {
|
|
80
|
+
return;
|
|
81
|
+
}
|
|
82
|
+
return this.adapter.getValue(cacheKey);
|
|
83
|
+
}
|
|
84
|
+
async saveToAdapter(cacheKey, value) {
|
|
85
|
+
if (!this.adapter) {
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
return this.adapter.setValue(cacheKey, value);
|
|
89
|
+
}
|
|
90
|
+
async get(cacheKey, valueLoader, options = {}) {
|
|
91
|
+
const mergedOptions = this.mergeOptions(options);
|
|
92
|
+
const cachedValue = this.tryGetFromMemory(cacheKey);
|
|
93
|
+
if (cachedValue !== undefined) {
|
|
94
|
+
return cachedValue;
|
|
95
|
+
}
|
|
96
|
+
const freshValue = await this.tryGetFromValueLoader(valueLoader);
|
|
97
|
+
if (freshValue !== undefined) {
|
|
98
|
+
await this.cacheValue(cacheKey, freshValue, mergedOptions);
|
|
99
|
+
return freshValue;
|
|
100
|
+
}
|
|
101
|
+
const adapterValue = await this.tryGetFromAdapter(cacheKey);
|
|
102
|
+
if (adapterValue !== undefined) {
|
|
103
|
+
await this.cacheValue(cacheKey, adapterValue, mergedOptions);
|
|
104
|
+
return adapterValue;
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
mergeOptions(options) {
|
|
108
|
+
return {
|
|
109
|
+
ttlMs: this.options.ttlMs,
|
|
110
|
+
expirationThresholdMs: this.options.expirationThresholdMs,
|
|
111
|
+
...options
|
|
112
|
+
};
|
|
113
|
+
}
|
|
114
|
+
tryGetFromMemory(cacheKey) {
|
|
115
|
+
const cached = this.inMemoryCache.get(cacheKey);
|
|
116
|
+
if (cached && !this.isExpired(cached.metadata)) {
|
|
117
|
+
return cached.data;
|
|
118
|
+
}
|
|
119
|
+
return;
|
|
120
|
+
}
|
|
121
|
+
async tryGetFromAdapter(cacheKey) {
|
|
122
|
+
const cached = await this.getFromAdapter(cacheKey);
|
|
123
|
+
if (cached && !this.isExpired(cached.metadata)) {
|
|
124
|
+
this.inMemoryCache.set(cacheKey, cached);
|
|
125
|
+
return cached.data;
|
|
126
|
+
}
|
|
127
|
+
return;
|
|
128
|
+
}
|
|
129
|
+
async tryGetFromValueLoader(valueLoader) {
|
|
130
|
+
if (valueLoader === undefined) {
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
133
|
+
if (isCallable(valueLoader)) {
|
|
134
|
+
return await valueLoader();
|
|
135
|
+
}
|
|
136
|
+
if (import_types.isPromise(valueLoader)) {
|
|
137
|
+
return await valueLoader;
|
|
138
|
+
}
|
|
139
|
+
return valueLoader;
|
|
140
|
+
}
|
|
141
|
+
async cacheValue(cacheKey, value, options) {
|
|
142
|
+
const cacheData = this.createCacheEntry(value, options);
|
|
143
|
+
await this.saveToAdapter(cacheKey, cacheData);
|
|
144
|
+
this.inMemoryCache.set(cacheKey, cacheData);
|
|
145
|
+
}
|
|
146
|
+
createCacheEntry(value, options) {
|
|
147
|
+
const now = Date.now();
|
|
148
|
+
let expiresAt;
|
|
149
|
+
if (options.ttlMs < 0) {
|
|
150
|
+
expiresAt = -1;
|
|
151
|
+
} else {
|
|
152
|
+
expiresAt = now + options.ttlMs - options.expirationThresholdMs;
|
|
153
|
+
}
|
|
154
|
+
return {
|
|
155
|
+
data: value,
|
|
156
|
+
metadata: {
|
|
157
|
+
createdAt: now,
|
|
158
|
+
ttlMs: options.ttlMs,
|
|
159
|
+
expirationThresholdMs: options.expirationThresholdMs,
|
|
160
|
+
expiresAt
|
|
161
|
+
}
|
|
162
|
+
};
|
|
163
|
+
}
|
|
164
|
+
isExpired(metadata) {
|
|
165
|
+
if (!metadata) {
|
|
166
|
+
return true;
|
|
167
|
+
}
|
|
168
|
+
if (metadata.expiresAt < 0) {
|
|
169
|
+
return false;
|
|
170
|
+
}
|
|
171
|
+
const expirationTime = metadata.expiresAt ?? (metadata.createdAt && metadata.ttlMs ? metadata.createdAt + metadata.ttlMs : undefined);
|
|
172
|
+
if (expirationTime === undefined) {
|
|
173
|
+
return true;
|
|
174
|
+
}
|
|
175
|
+
return Date.now() > expirationTime;
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
// src/cache/constants.ts
|
|
179
|
+
var CACHE_TTL_NEVER_EXPIRE = -1;
|
|
42
180
|
// src/index.ts
|
|
43
181
|
var exports_src = {};
|
|
44
182
|
__export(exports_src, {
|
|
45
183
|
tryCatchAsync: () => tryCatchAsync,
|
|
46
184
|
tryCatch: () => tryCatch,
|
|
185
|
+
sleep: () => sleep,
|
|
186
|
+
randomString: () => randomString,
|
|
47
187
|
jsonParser: () => jsonParser,
|
|
188
|
+
isPrimitive: () => isPrimitive,
|
|
48
189
|
isObject: () => isObject,
|
|
49
190
|
isNullOrUndefined: () => isNullOrUndefined,
|
|
50
191
|
isNullOrEmptyOrUndefined: () => isNullOrEmptyOrUndefined,
|
|
51
|
-
chunksToLines: () => chunksToLines
|
|
192
|
+
chunksToLines: () => chunksToLines,
|
|
193
|
+
Cache: () => Cache,
|
|
194
|
+
CACHE_TTL_NEVER_EXPIRE: () => CACHE_TTL_NEVER_EXPIRE
|
|
52
195
|
});
|
|
53
196
|
module.exports = __toCommonJS(exports_src);
|
|
54
197
|
|
|
55
|
-
// src/chunksToLines.ts
|
|
198
|
+
// src/utils/chunksToLines.ts
|
|
56
199
|
async function* chunksToLines(chunks) {
|
|
57
200
|
let previous = "";
|
|
58
201
|
for await (const chunk of chunks) {
|
|
@@ -69,11 +212,20 @@ async function* chunksToLines(chunks) {
|
|
|
69
212
|
yield previous;
|
|
70
213
|
}
|
|
71
214
|
}
|
|
72
|
-
// src/
|
|
215
|
+
// src/utils/isNullOrUndefined.ts
|
|
216
|
+
var isNullOrUndefined = (value) => value === null || value === undefined;
|
|
217
|
+
function isNullOrEmptyOrUndefined(value) {
|
|
218
|
+
return value === undefined || value === null || value === "" || value.toString().trim() === "";
|
|
219
|
+
}
|
|
220
|
+
// src/utils/isObject.ts
|
|
73
221
|
function isObject(input) {
|
|
74
222
|
return Object.prototype.toString.apply(input) === "[object Object]";
|
|
75
223
|
}
|
|
76
|
-
// src/
|
|
224
|
+
// src/utils/isPrimitive.ts
|
|
225
|
+
function isPrimitive(value) {
|
|
226
|
+
return typeof value !== "object" && typeof value !== "function" || value === null || value instanceof Date;
|
|
227
|
+
}
|
|
228
|
+
// src/utils/jsonParser.ts
|
|
77
229
|
var import_secure_json_parse = __toESM(require("secure-json-parse"));
|
|
78
230
|
function jsonParser(input) {
|
|
79
231
|
try {
|
|
@@ -82,6 +234,18 @@ function jsonParser(input) {
|
|
|
82
234
|
return { err };
|
|
83
235
|
}
|
|
84
236
|
}
|
|
237
|
+
// src/utils/randomString.ts
|
|
238
|
+
function randomString(length = 6) {
|
|
239
|
+
let result = "";
|
|
240
|
+
const characters = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
|
|
241
|
+
const charactersLength = characters.length;
|
|
242
|
+
for (let i = 0;i < length; i++) {
|
|
243
|
+
result += characters.charAt(Math.floor(Math.random() * charactersLength));
|
|
244
|
+
}
|
|
245
|
+
return result;
|
|
246
|
+
}
|
|
247
|
+
// src/utils/sleep.ts
|
|
248
|
+
var sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
|
|
85
249
|
// src/neverthrow/tryCatch.ts
|
|
86
250
|
function tryCatch(fn) {
|
|
87
251
|
try {
|
|
@@ -99,11 +263,6 @@ async function tryCatchAsync(promise) {
|
|
|
99
263
|
return [undefined, error];
|
|
100
264
|
}
|
|
101
265
|
}
|
|
102
|
-
// src/isNullOrUndefined.ts
|
|
103
|
-
var isNullOrUndefined = (value) => value === null || value === undefined;
|
|
104
|
-
function isNullOrEmptyOrUndefined(value) {
|
|
105
|
-
return value === undefined || value === null || value === "" || value.toString().trim() === "";
|
|
106
|
-
}
|
|
107
266
|
|
|
108
|
-
//# debugId=
|
|
109
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/chunksToLines.ts", "../src/isObject.ts", "../src/jsonParser.ts", "../src/neverthrow/tryCatch.ts", "../src/isNullOrUndefined.ts"],
  "sourcesContent": [
    "/**\n * Converts an async iterable of string or Uint8Array chunks into an async iterable of lines.\n * Handles cases where lines may be split across multiple chunks.\n * Usage:\n * ```ts\n * import { pipeline } from \"node:stream/promises\";\n * import { createReadStream } from \"node:fs\";\n * import { chunksToLines } from \"@lukaskj/ts-utils\";\n *\n * const readStream = createReadStream(\"path/to/file.txt\", { encoding: \"utf-8\" });\n * await pipeline(readStream, chunksToLines, process.stdout).catch((err) => {\n *   console.error(err);\n *   process.exit(1);\n * });\n * ```\n * @param chunks\n */\nexport async function* chunksToLines(chunks: AsyncIterable<string | Uint8Array>) {\n  let previous = \"\";\n\n  for await (const chunk of chunks) {\n    previous += chunk;\n    let eolIndex: number;\n\n    // biome-ignore lint/suspicious/noAssignInExpressions: \"explanation\"\n    while ((eolIndex = previous.indexOf(\"\\n\")) >= 0) {\n      // this line includes the EOL\n      const line = previous.slice(0, eolIndex + 1);\n      yield line?.trim() ?? \"\";\n      previous = previous.slice(eolIndex + 1);\n    }\n  }\n\n  if (previous.length > 0) {\n    yield previous;\n  }\n}\n",
    "/**\n * Type guard that checks if the input is a plain JavaScript object.\n * @param {any} input - The value to check\n * @returns {boolean} True if the input is a plain object, false otherwise\n * @typeParam {Record<string, any>} Type guard narrows the input type to a string-keyed object\n */\nexport function isObject(input: any): input is Record<string, any> {\n  return Object.prototype.toString.apply(input) === \"[object Object]\";\n}\n",
    "import sjs from \"secure-json-parse\";\n\n/**\n * Safely parses JSON input using secure-json-parse with prototype pollution protection.\n * @param {any} input - The input to be parsed as JSON\n * @returns {{ value?: any, err?: Error }} An object containing either the parsed value or an error if parsing failed\n */\nexport function jsonParser(input: any) {\n  try {\n    return { value: sjs.parse(input, { protoAction: \"remove\" }) };\n  } catch (err) {\n    return { err };\n  }\n}\n",
    "// https://gist.github.com/t3dotgg/a486c4ae66d32bf17c09c73609dacc5b\n// Types for the result object with discriminated union\n// type Success<T> = {\n//   data: T;\n//   error: null;\n// };\n\n// type Failure<E> = {\n//   data: null;\n//   error: E;\n// };\n\n// export type Result<T, E = Error> = Success<T> | Failure<E>;\n\nexport type Result<T, E> = [undefined, E] | [T, undefined];\nexport type PromiseResult<T, E> = Promise<Result<T, E>>;\n\n/**\n * Wraps a synchronous function in a try-catch block and returns a Result tuple containing either the returned value or the caught error\n *\n * @template T - The type of the successful result value\n * @template E - The type of the error value, must extend Error\n * @param {() => T} fn - The function to be executed\n * @returns {Result<T, E>} A tuple containing either [data, undefined] or [undefined, error]\n *\n * @example\n * const [data, error] = tryCatch(() => someRiskyOperation());\n * if (error) {\n *   console.error(error);\n * } else {\n *   console.log(data);\n * }\n */\nexport function tryCatch<T, E extends Error>(fn: () => T): Result<T, E> {\n  try {\n    const data = fn();\n    return [data as T, undefined];\n  } catch (error) {\n    return [undefined, error as E];\n  }\n}\n\n/**\n * Wraps a Promise in a try-catch block and returns a Result object containing either the resolved data or the caught error\n *\n * @template T - The type of the successful result value\n * @template E - The type of the error value, defaults to Error\n * @param {Promise<T>} promise - The promise to be executed\n * @returns {Promise<Result<T, E>>} A Promise that resolves to a Result object containing either the data or error\n *\n * @example\n * const {data, error} = await tryCatchAsync(somePromise);\n * if (error) {\n *   console.error(result.error);\n * } else {\n *   console.log(data);\n * }\n */\nexport async function tryCatchAsync<T, E extends Error>(promise: Promise<T>): PromiseResult<T, E> {\n  try {\n    const data = await promise;\n    return [data as T, undefined];\n  } catch (error) {\n    return [undefined, error as E];\n  }\n}\n",
    "export const isNullOrUndefined = (value: any): value is null | undefined => value === null || value === undefined;\n\nexport function isNullOrEmptyOrUndefined(value?: any | null): value is null | undefined {\n  return value === undefined || value === null || value === \"\" || value.toString().trim() === \"\";\n}\n"
  ],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAiBA,gBAAuB,aAAa,CAAC,QAA4C;AAAA,EAC/E,IAAI,WAAW;AAAA,EAEf,iBAAiB,SAAS,QAAQ;AAAA,IAChC,YAAY;AAAA,IACZ,IAAI;AAAA,IAGJ,QAAQ,WAAW,SAAS,QAAQ;AAAA,CAAI,MAAM,GAAG;AAAA,MAE/C,MAAM,OAAO,SAAS,MAAM,GAAG,WAAW,CAAC;AAAA,MAC3C,MAAM,MAAM,KAAK,KAAK;AAAA,MACtB,WAAW,SAAS,MAAM,WAAW,CAAC;AAAA,IACxC;AAAA,EACF;AAAA,EAEA,IAAI,SAAS,SAAS,GAAG;AAAA,IACvB,MAAM;AAAA,EACR;AAAA;;AC7BK,SAAS,QAAQ,CAAC,OAA0C;AAAA,EACjE,OAAO,OAAO,UAAU,SAAS,MAAM,KAAK,MAAM;AAAA;;ACPpC,IAAhB;AAOO,SAAS,UAAU,CAAC,OAAY;AAAA,EACrC,IAAI;AAAA,IACF,OAAO,EAAE,OAAO,iCAAI,MAAM,OAAO,EAAE,aAAa,SAAS,CAAC,EAAE;AAAA,IAC5D,OAAO,KAAK;AAAA,IACZ,OAAO,EAAE,IAAI;AAAA;AAAA;;ACsBV,SAAS,QAA4B,CAAC,IAA2B;AAAA,EACtE,IAAI;AAAA,IACF,MAAM,OAAO,GAAG;AAAA,IAChB,OAAO,CAAC,MAAW,SAAS;AAAA,IAC5B,OAAO,OAAO;AAAA,IACd,OAAO,CAAC,WAAW,KAAU;AAAA;AAAA;AAoBjC,eAAsB,aAAiC,CAAC,SAA0C;AAAA,EAChG,IAAI;AAAA,IACF,MAAM,OAAO,MAAM;AAAA,IACnB,OAAO,CAAC,MAAW,SAAS;AAAA,IAC5B,OAAO,OAAO;AAAA,IACd,OAAO,CAAC,WAAW,KAAU;AAAA;AAAA;;AC/D1B,IAAM,oBAAoB,CAAC,UAA0C,UAAU,QAAQ,UAAU;AAEjG,SAAS,wBAAwB,CAAC,OAA+C;AAAA,EACtF,OAAO,UAAU,aAAa,UAAU,QAAQ,UAAU,MAAM,MAAM,SAAS,EAAE,KAAK,MAAM;AAAA;",
  "debugId": "BE1349762FEF2BD064756E2164756E21",
  "names": []
}
|
|
267
|
+
//# debugId=095D83D51A0E4FF764756E2164756E21
|
|
268
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/cache/cacheService.ts", "../src/utils/isCallable.ts", "../src/cache/constants.ts", "../src/utils/chunksToLines.ts", "../src/utils/isNullOrUndefined.ts", "../src/utils/isObject.ts", "../src/utils/isPrimitive.ts", "../src/utils/jsonParser.ts", "../src/utils/randomString.ts", "../src/utils/sleep.ts", "../src/neverthrow/tryCatch.ts"],
  "sourcesContent": [
    "import { isPromise } from \"node:util/types\";\nimport { isCallable } from \"../utils/isCallable.ts\";\nimport type { CacheMetadataOptions, CacheOptions, CacheWithMetadata, ValueLoader } from \"./internal/types.ts\";\nimport type { CacheMetadata, ICacheAdapter } from \"./types.ts\";\n\nconst defaultCacheOptions: CacheOptions = {\n  expirationThresholdMs: 0,\n  ttlMs: 60 * 60 * 1000,\n};\n\nexport class Cache<TAdapter extends ICacheAdapter | undefined = undefined> {\n  private inMemoryCache: Map<string, CacheWithMetadata> = new Map();\n  private readonly options: CacheOptions;\n  private readonly adapter?: TAdapter;\n\n  constructor(defaultOptions?: Partial<CacheOptions>);\n  constructor(adapter?: TAdapter, defaultOptions?: Partial<CacheOptions>);\n  constructor(adapter?: TAdapter | Partial<CacheOptions>, defaultOptions?: Partial<CacheOptions>) {\n    const options = adapter && !(\"getValue\" in adapter) ? (adapter as Partial<CacheOptions>) : defaultOptions;\n    if (adapter && \"getValue\" in adapter && \"setValue\" in adapter) {\n      this.adapter = adapter as TAdapter;\n    }\n\n    this.options = {\n      ...defaultCacheOptions,\n      ...options,\n    };\n  }\n\n  private async getFromAdapter<T>(cacheKey: string): Promise<T | undefined> {\n    if (!this.adapter) {\n      return undefined;\n    }\n\n    return this.adapter.getValue(cacheKey);\n  }\n\n  private async saveToAdapter<T>(cacheKey: string, value: T): Promise<void> {\n    if (!this.adapter) {\n      return;\n    }\n\n    return this.adapter.setValue<T>(cacheKey, value);\n  }\n\n  public async get<T>(\n    cacheKey: string,\n    valueLoader?: ValueLoader<T>,\n    options: Partial<CacheMetadataOptions> = {},\n  ): Promise<T | undefined> {\n    const mergedOptions = this.mergeOptions(options);\n\n    // Try in-memory cache first\n    const cachedValue = this.tryGetFromMemory<T>(cacheKey);\n    if (cachedValue !== undefined) {\n      return cachedValue;\n    }\n\n    // Load fresh value from loader function first\n    const freshValue = await this.tryGetFromValueLoader(valueLoader);\n    if (freshValue !== undefined) {\n      await this.cacheValue(cacheKey, freshValue, mergedOptions);\n\n      return freshValue;\n    }\n\n    // Try adapter cache second\n    const adapterValue = await this.tryGetFromAdapter<T>(cacheKey);\n    if (adapterValue !== undefined) {\n      await this.cacheValue(cacheKey, adapterValue, mergedOptions);\n\n      return adapterValue;\n    }\n  }\n\n  private mergeOptions(options: Partial<CacheMetadataOptions>): CacheMetadataOptions {\n    return {\n      ttlMs: this.options.ttlMs,\n      expirationThresholdMs: this.options.expirationThresholdMs,\n      ...options,\n    };\n  }\n\n  private tryGetFromMemory<T>(cacheKey: string): T | undefined {\n    const cached = this.inMemoryCache.get(cacheKey);\n\n    if (cached && !this.isExpired(cached.metadata)) {\n      return cached.data as T;\n    }\n\n    return undefined;\n  }\n\n  private async tryGetFromAdapter<T>(cacheKey: string): Promise<T | undefined> {\n    const cached = await this.getFromAdapter<CacheWithMetadata<T>>(cacheKey);\n\n    if (cached && !this.isExpired(cached.metadata)) {\n      this.inMemoryCache.set(cacheKey, cached);\n      return cached.data;\n    }\n\n    return undefined;\n  }\n\n  private async tryGetFromValueLoader<T>(valueLoader?: ValueLoader<T>): Promise<T | undefined> {\n    if (valueLoader === undefined) {\n      return undefined;\n    }\n\n    if (isCallable(valueLoader)) {\n      return await valueLoader();\n    }\n\n    if (isPromise(valueLoader)) {\n      return await valueLoader;\n    }\n\n    return valueLoader;\n  }\n\n  private async cacheValue<T>(cacheKey: string, value: T, options: CacheMetadataOptions): Promise<void> {\n    const cacheData = this.createCacheEntry(value, options);\n\n    await this.saveToAdapter(cacheKey, cacheData);\n    this.inMemoryCache.set(cacheKey, cacheData);\n  }\n\n  private createCacheEntry<T>(value: T, options: CacheMetadataOptions): CacheWithMetadata<T> {\n    const now = Date.now();\n    let expiresAt: number;\n    if (options.ttlMs < 0) {\n      expiresAt = -1;\n    } else {\n      expiresAt = now + options.ttlMs - options.expirationThresholdMs;\n    }\n\n    return {\n      data: value,\n      metadata: {\n        createdAt: now,\n        ttlMs: options.ttlMs,\n        expirationThresholdMs: options.expirationThresholdMs,\n        expiresAt,\n      },\n    };\n  }\n\n  private isExpired(metadata?: CacheMetadata): boolean {\n    if (!metadata) {\n      return true;\n    }\n\n    if (metadata.expiresAt < 0) {\n      return false;\n    }\n\n    const expirationTime =\n      metadata.expiresAt ?? (metadata.createdAt && metadata.ttlMs ? metadata.createdAt + metadata.ttlMs : undefined);\n\n    if (expirationTime === undefined) {\n      return true;\n    }\n\n    return Date.now() > expirationTime;\n  }\n}\n",
    "export function isCallable(value: any): value is CallableFunction {\n  return typeof value === \"function\";\n}\n",
    "export const CACHE_TTL_NEVER_EXPIRE = -1;\n",
    "/**\n * Converts an async iterable of string or Uint8Array chunks into an async iterable of lines.\n * Handles cases where lines may be split across multiple chunks.\n * Usage:\n * ```ts\n * import { pipeline } from \"node:stream/promises\";\n * import { createReadStream } from \"node:fs\";\n * import { chunksToLines } from \"@lukaskj/ts-utils\";\n *\n * const readStream = createReadStream(\"path/to/file.txt\", { encoding: \"utf-8\" });\n * await pipeline(readStream, chunksToLines, process.stdout).catch((err) => {\n *   console.error(err);\n *   process.exit(1);\n * });\n * ```\n * @param chunks\n */\nexport async function* chunksToLines(chunks: AsyncIterable<string | Uint8Array>) {\n  let previous = \"\";\n\n  for await (const chunk of chunks) {\n    previous += chunk;\n    let eolIndex: number;\n\n    // biome-ignore lint/suspicious/noAssignInExpressions: \"explanation\"\n    while ((eolIndex = previous.indexOf(\"\\n\")) >= 0) {\n      // this line includes the EOL\n      const line = previous.slice(0, eolIndex + 1);\n      yield line?.trim() ?? \"\";\n      previous = previous.slice(eolIndex + 1);\n    }\n  }\n\n  if (previous.length > 0) {\n    yield previous;\n  }\n}\n",
    "export const isNullOrUndefined = (value: any): value is null | undefined => value === null || value === undefined;\n\nexport function isNullOrEmptyOrUndefined(value?: any | null): value is null | undefined {\n  return value === undefined || value === null || value === \"\" || value.toString().trim() === \"\";\n}\n",
    "/**\n * Type guard that checks if the input is a plain JavaScript object.\n * @param {any} input - The value to check\n * @returns {boolean} True if the input is a plain object, false otherwise\n * @typeParam {Record<string, any>} Type guard narrows the input type to a string-keyed object\n */\nexport function isObject(input: any): input is Record<string, any> {\n  return Object.prototype.toString.apply(input) === \"[object Object]\";\n}\n",
    "export function isPrimitive(value: unknown): boolean {\n  return (typeof value !== \"object\" && typeof value !== \"function\") || value === null || value instanceof Date;\n}\n",
    "import sjs from \"secure-json-parse\";\n\n/**\n * Safely parses JSON input using secure-json-parse with prototype pollution protection.\n * @param {any} input - The input to be parsed as JSON\n * @returns {{ value?: any, err?: Error }} An object containing either the parsed value or an error if parsing failed\n */\nexport function jsonParser(input: any) {\n  try {\n    return { value: sjs.parse(input, { protoAction: \"remove\" }) };\n  } catch (err) {\n    return { err };\n  }\n}\n",
    "export function randomString(length: number = 6): string {\n  let result = \"\";\n  const characters = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789\";\n  const charactersLength = characters.length;\n  for (let i = 0; i < length; i++) {\n    result += characters.charAt(Math.floor(Math.random() * charactersLength));\n  }\n  return result;\n}\n",
    "export const sleep = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms));\n",
    "// https://gist.github.com/t3dotgg/a486c4ae66d32bf17c09c73609dacc5b\n// Types for the result object with discriminated union\n// type Success<T> = {\n//   data: T;\n//   error: null;\n// };\n\n// type Failure<E> = {\n//   data: null;\n//   error: E;\n// };\n\n// export type Result<T, E = Error> = Success<T> | Failure<E>;\n\nexport type Result<T, E> = [undefined, E] | [T, undefined];\nexport type PromiseResult<T, E> = Promise<Result<T, E>>;\n\n/**\n * Wraps a synchronous function in a try-catch block and returns a Result tuple containing either the returned value or the caught error\n *\n * @template T - The type of the successful result value\n * @template E - The type of the error value, must extend Error\n * @param {() => T} fn - The function to be executed\n * @returns {Result<T, E>} A tuple containing either [data, undefined] or [undefined, error]\n *\n * @example\n * const [data, error] = tryCatch(() => someRiskyOperation());\n * if (error) {\n *   console.error(error);\n * } else {\n *   console.log(data);\n * }\n */\nexport function tryCatch<T, E extends Error>(fn: () => T): Result<T, E> {\n  try {\n    const data = fn();\n    return [data as T, undefined];\n  } catch (error) {\n    return [undefined, error as E];\n  }\n}\n\n/**\n * Wraps a Promise in a try-catch block and returns a Result object containing either the resolved data or the caught error\n *\n * @template T - The type of the successful result value\n * @template E - The type of the error value, defaults to Error\n * @param {Promise<T>} promise - The promise to be executed\n * @returns {Promise<Result<T, E>>} A Promise that resolves to a Result object containing either the data or error\n *\n * @example\n * const {data, error} = await tryCatchAsync(somePromise);\n * if (error) {\n *   console.error(result.error);\n * } else {\n *   console.log(data);\n * }\n */\nexport async function tryCatchAsync<T, E extends Error>(promise: Promise<T>): PromiseResult<T, E> {\n  try {\n    const data = await promise;\n    return [data as T, undefined];\n  } catch (error) {\n    return [undefined, error as E];\n  }\n}\n"
  ],
  "mappings": "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",
  "debugId": "095D83D51A0E4FF764756E2164756E21",
  "names": []
}
|
package/dist/index.d.ts
CHANGED
|
@@ -1,6 +1,10 @@
|
|
|
1
|
-
export * from "./chunksToLines.ts";
|
|
2
|
-
export * from "./isObject.ts";
|
|
3
|
-
export * from "./jsonParser.ts";
|
|
4
|
-
export * from "./neverthrow/tryCatch.ts";
|
|
5
1
|
export * from "./types.ts";
|
|
6
|
-
export * from "./
|
|
2
|
+
export * from "./utils/chunksToLines.ts";
|
|
3
|
+
export * from "./utils/isNullOrUndefined.ts";
|
|
4
|
+
export * from "./utils/isObject.ts";
|
|
5
|
+
export * from "./utils/isPrimitive.ts";
|
|
6
|
+
export * from "./utils/jsonParser.ts";
|
|
7
|
+
export * from "./utils/randomString.ts";
|
|
8
|
+
export * from "./utils/sleep.ts";
|
|
9
|
+
export * from "./neverthrow/tryCatch.ts";
|
|
10
|
+
export * from "./cache/index.ts";
|
package/dist/index.mjs
CHANGED
|
@@ -1,4 +1,134 @@
|
|
|
1
|
-
// src/
|
|
1
|
+
// src/cache/cacheService.ts
|
|
2
|
+
import { isPromise } from "node:util/types";
|
|
3
|
+
|
|
4
|
+
// src/utils/isCallable.ts
|
|
5
|
+
function isCallable(value) {
|
|
6
|
+
return typeof value === "function";
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
// src/cache/cacheService.ts
|
|
10
|
+
var defaultCacheOptions = {
|
|
11
|
+
expirationThresholdMs: 0,
|
|
12
|
+
ttlMs: 60 * 60 * 1000
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
class Cache {
|
|
16
|
+
inMemoryCache = new Map;
|
|
17
|
+
options;
|
|
18
|
+
adapter;
|
|
19
|
+
constructor(adapter, defaultOptions) {
|
|
20
|
+
const options = adapter && !("getValue" in adapter) ? adapter : defaultOptions;
|
|
21
|
+
if (adapter && "getValue" in adapter && "setValue" in adapter) {
|
|
22
|
+
this.adapter = adapter;
|
|
23
|
+
}
|
|
24
|
+
this.options = {
|
|
25
|
+
...defaultCacheOptions,
|
|
26
|
+
...options
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
async getFromAdapter(cacheKey) {
|
|
30
|
+
if (!this.adapter) {
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
return this.adapter.getValue(cacheKey);
|
|
34
|
+
}
|
|
35
|
+
async saveToAdapter(cacheKey, value) {
|
|
36
|
+
if (!this.adapter) {
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
return this.adapter.setValue(cacheKey, value);
|
|
40
|
+
}
|
|
41
|
+
async get(cacheKey, valueLoader, options = {}) {
|
|
42
|
+
const mergedOptions = this.mergeOptions(options);
|
|
43
|
+
const cachedValue = this.tryGetFromMemory(cacheKey);
|
|
44
|
+
if (cachedValue !== undefined) {
|
|
45
|
+
return cachedValue;
|
|
46
|
+
}
|
|
47
|
+
const freshValue = await this.tryGetFromValueLoader(valueLoader);
|
|
48
|
+
if (freshValue !== undefined) {
|
|
49
|
+
await this.cacheValue(cacheKey, freshValue, mergedOptions);
|
|
50
|
+
return freshValue;
|
|
51
|
+
}
|
|
52
|
+
const adapterValue = await this.tryGetFromAdapter(cacheKey);
|
|
53
|
+
if (adapterValue !== undefined) {
|
|
54
|
+
await this.cacheValue(cacheKey, adapterValue, mergedOptions);
|
|
55
|
+
return adapterValue;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
mergeOptions(options) {
|
|
59
|
+
return {
|
|
60
|
+
ttlMs: this.options.ttlMs,
|
|
61
|
+
expirationThresholdMs: this.options.expirationThresholdMs,
|
|
62
|
+
...options
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
tryGetFromMemory(cacheKey) {
|
|
66
|
+
const cached = this.inMemoryCache.get(cacheKey);
|
|
67
|
+
if (cached && !this.isExpired(cached.metadata)) {
|
|
68
|
+
return cached.data;
|
|
69
|
+
}
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
async tryGetFromAdapter(cacheKey) {
|
|
73
|
+
const cached = await this.getFromAdapter(cacheKey);
|
|
74
|
+
if (cached && !this.isExpired(cached.metadata)) {
|
|
75
|
+
this.inMemoryCache.set(cacheKey, cached);
|
|
76
|
+
return cached.data;
|
|
77
|
+
}
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
async tryGetFromValueLoader(valueLoader) {
|
|
81
|
+
if (valueLoader === undefined) {
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
if (isCallable(valueLoader)) {
|
|
85
|
+
return await valueLoader();
|
|
86
|
+
}
|
|
87
|
+
if (isPromise(valueLoader)) {
|
|
88
|
+
return await valueLoader;
|
|
89
|
+
}
|
|
90
|
+
return valueLoader;
|
|
91
|
+
}
|
|
92
|
+
async cacheValue(cacheKey, value, options) {
|
|
93
|
+
const cacheData = this.createCacheEntry(value, options);
|
|
94
|
+
await this.saveToAdapter(cacheKey, cacheData);
|
|
95
|
+
this.inMemoryCache.set(cacheKey, cacheData);
|
|
96
|
+
}
|
|
97
|
+
createCacheEntry(value, options) {
|
|
98
|
+
const now = Date.now();
|
|
99
|
+
let expiresAt;
|
|
100
|
+
if (options.ttlMs < 0) {
|
|
101
|
+
expiresAt = -1;
|
|
102
|
+
} else {
|
|
103
|
+
expiresAt = now + options.ttlMs - options.expirationThresholdMs;
|
|
104
|
+
}
|
|
105
|
+
return {
|
|
106
|
+
data: value,
|
|
107
|
+
metadata: {
|
|
108
|
+
createdAt: now,
|
|
109
|
+
ttlMs: options.ttlMs,
|
|
110
|
+
expirationThresholdMs: options.expirationThresholdMs,
|
|
111
|
+
expiresAt
|
|
112
|
+
}
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
isExpired(metadata) {
|
|
116
|
+
if (!metadata) {
|
|
117
|
+
return true;
|
|
118
|
+
}
|
|
119
|
+
if (metadata.expiresAt < 0) {
|
|
120
|
+
return false;
|
|
121
|
+
}
|
|
122
|
+
const expirationTime = metadata.expiresAt ?? (metadata.createdAt && metadata.ttlMs ? metadata.createdAt + metadata.ttlMs : undefined);
|
|
123
|
+
if (expirationTime === undefined) {
|
|
124
|
+
return true;
|
|
125
|
+
}
|
|
126
|
+
return Date.now() > expirationTime;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
// src/cache/constants.ts
|
|
130
|
+
var CACHE_TTL_NEVER_EXPIRE = -1;
|
|
131
|
+
// src/utils/chunksToLines.ts
|
|
2
132
|
async function* chunksToLines(chunks) {
|
|
3
133
|
let previous = "";
|
|
4
134
|
for await (const chunk of chunks) {
|
|
@@ -15,11 +145,20 @@ async function* chunksToLines(chunks) {
|
|
|
15
145
|
yield previous;
|
|
16
146
|
}
|
|
17
147
|
}
|
|
18
|
-
// src/
|
|
148
|
+
// src/utils/isNullOrUndefined.ts
|
|
149
|
+
var isNullOrUndefined = (value) => value === null || value === undefined;
|
|
150
|
+
function isNullOrEmptyOrUndefined(value) {
|
|
151
|
+
return value === undefined || value === null || value === "" || value.toString().trim() === "";
|
|
152
|
+
}
|
|
153
|
+
// src/utils/isObject.ts
|
|
19
154
|
function isObject(input) {
|
|
20
155
|
return Object.prototype.toString.apply(input) === "[object Object]";
|
|
21
156
|
}
|
|
22
|
-
// src/
|
|
157
|
+
// src/utils/isPrimitive.ts
|
|
158
|
+
function isPrimitive(value) {
|
|
159
|
+
return typeof value !== "object" && typeof value !== "function" || value === null || value instanceof Date;
|
|
160
|
+
}
|
|
161
|
+
// src/utils/jsonParser.ts
|
|
23
162
|
import sjs from "secure-json-parse";
|
|
24
163
|
function jsonParser(input) {
|
|
25
164
|
try {
|
|
@@ -28,6 +167,18 @@ function jsonParser(input) {
|
|
|
28
167
|
return { err };
|
|
29
168
|
}
|
|
30
169
|
}
|
|
170
|
+
// src/utils/randomString.ts
|
|
171
|
+
function randomString(length = 6) {
|
|
172
|
+
let result = "";
|
|
173
|
+
const characters = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
|
|
174
|
+
const charactersLength = characters.length;
|
|
175
|
+
for (let i = 0;i < length; i++) {
|
|
176
|
+
result += characters.charAt(Math.floor(Math.random() * charactersLength));
|
|
177
|
+
}
|
|
178
|
+
return result;
|
|
179
|
+
}
|
|
180
|
+
// src/utils/sleep.ts
|
|
181
|
+
var sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
|
|
31
182
|
// src/neverthrow/tryCatch.ts
|
|
32
183
|
function tryCatch(fn) {
|
|
33
184
|
try {
|
|
@@ -45,20 +196,20 @@ async function tryCatchAsync(promise) {
|
|
|
45
196
|
return [undefined, error];
|
|
46
197
|
}
|
|
47
198
|
}
|
|
48
|
-
// src/isNullOrUndefined.ts
|
|
49
|
-
var isNullOrUndefined = (value) => value === null || value === undefined;
|
|
50
|
-
function isNullOrEmptyOrUndefined(value) {
|
|
51
|
-
return value === undefined || value === null || value === "" || value.toString().trim() === "";
|
|
52
|
-
}
|
|
53
199
|
export {
|
|
54
200
|
tryCatchAsync,
|
|
55
201
|
tryCatch,
|
|
202
|
+
sleep,
|
|
203
|
+
randomString,
|
|
56
204
|
jsonParser,
|
|
205
|
+
isPrimitive,
|
|
57
206
|
isObject,
|
|
58
207
|
isNullOrUndefined,
|
|
59
208
|
isNullOrEmptyOrUndefined,
|
|
60
|
-
chunksToLines
|
|
209
|
+
chunksToLines,
|
|
210
|
+
Cache,
|
|
211
|
+
CACHE_TTL_NEVER_EXPIRE
|
|
61
212
|
};
|
|
62
213
|
|
|
63
|
-
//# debugId=
|
|
64
|
-
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vc3JjL2NodW5rc1RvTGluZXMudHMiLCAiLi4vc3JjL2lzT2JqZWN0LnRzIiwgIi4uL3NyYy9qc29uUGFyc2VyLnRzIiwgIi4uL3NyYy9uZXZlcnRocm93L3RyeUNhdGNoLnRzIiwgIi4uL3NyYy9pc051bGxPclVuZGVmaW5lZC50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsKICAgICIvKipcbiAqIENvbnZlcnRzIGFuIGFzeW5jIGl0ZXJhYmxlIG9mIHN0cmluZyBvciBVaW50OEFycmF5IGNodW5rcyBpbnRvIGFuIGFzeW5jIGl0ZXJhYmxlIG9mIGxpbmVzLlxuICogSGFuZGxlcyBjYXNlcyB3aGVyZSBsaW5lcyBtYXkgYmUgc3BsaXQgYWNyb3NzIG11bHRpcGxlIGNodW5rcy5cbiAqIFVzYWdlOlxuICogYGBgdHNcbiAqIGltcG9ydCB7IHBpcGVsaW5lIH0gZnJvbSBcIm5vZGU6c3RyZWFtL3Byb21pc2VzXCI7XG4gKiBpbXBvcnQgeyBjcmVhdGVSZWFkU3RyZWFtIH0gZnJvbSBcIm5vZGU6ZnNcIjtcbiAqIGltcG9ydCB7IGNodW5rc1RvTGluZXMgfSBmcm9tIFwiQGx1a2Fza2ovdHMtdXRpbHNcIjtcbiAqXG4gKiBjb25zdCByZWFkU3RyZWFtID0gY3JlYXRlUmVhZFN0cmVhbShcInBhdGgvdG8vZmlsZS50eHRcIiwgeyBlbmNvZGluZzogXCJ1dGYtOFwiIH0pO1xuICogYXdhaXQgcGlwZWxpbmUocmVhZFN0cmVhbSwgY2h1bmtzVG9MaW5lcywgcHJvY2Vzcy5zdGRvdXQpLmNhdGNoKChlcnIpID0+IHtcbiAqICAgY29uc29sZS5lcnJvcihlcnIpO1xuICogICBwcm9jZXNzLmV4aXQoMSk7XG4gKiB9KTtcbiAqIGBgYFxuICogQHBhcmFtIGNodW5rc1xuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24qIGNodW5rc1RvTGluZXMoY2h1bmtzOiBBc3luY0l0ZXJhYmxlPHN0cmluZyB8IFVpbnQ4QXJyYXk+KSB7XG4gIGxldCBwcmV2aW91cyA9IFwiXCI7XG5cbiAgZm9yIGF3YWl0IChjb25zdCBjaHVuayBvZiBjaHVua3MpIHtcbiAgICBwcmV2aW91cyArPSBjaHVuaztcbiAgICBsZXQgZW9sSW5kZXg6IG51bWJlcjtcblxuICAgIC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9Bc3NpZ25JbkV4cHJlc3Npb25zOiBcImV4cGxhbmF0aW9uXCJcbiAgICB3aGlsZSAoKGVvbEluZGV4ID0gcHJldmlvdXMuaW5kZXhPZihcIlxcblwiKSkgPj0gMCkge1xuICAgICAgLy8gdGhpcyBsaW5lIGluY2x1ZGVzIHRoZSBFT0xcbiAgICAgIGNvbnN0IGxpbmUgPSBwcmV2aW91cy5zbGljZSgwLCBlb2xJbmRleCArIDEpO1xuICAgICAgeWllbGQgbGluZT8udHJpbSgpID8/IFwiXCI7XG4gICAgICBwcmV2aW91cyA9IHByZXZpb3VzLnNsaWNlKGVvbEluZGV4ICsgMSk7XG4gICAgfVxuICB9XG5cbiAgaWYgKHByZXZpb3VzLmxlbmd0aCA+IDApIHtcbiAgICB5aWVsZCBwcmV2aW91cztcbiAgfVxufVxuIiwKICAgICIvKipcbiAqIFR5cGUgZ3VhcmQgdGhhdCBjaGVja3MgaWYgdGhlIGlucHV0IGlzIGEgcGxhaW4gSmF2YVNjcmlwdCBvYmplY3QuXG4gKiBAcGFyYW0ge2FueX0gaW5wdXQgLSBUaGUgdmFsdWUgdG8gY2hlY2tcbiAqIEByZXR1cm5zIHtib29sZWFufSBUcnVlIGlmIHRoZSBpbnB1dCBpcyBhIHBsYWluIG9iamVjdCwgZmFsc2Ugb3RoZXJ3aXNlXG4gKiBAdHlwZVBhcmFtIHtSZWNvcmQ8c3RyaW5nLCBhbnk+fSBUeXBlIGd1YXJkIG5hcnJvd3MgdGhlIGlucHV0IHR5cGUgdG8gYSBzdHJpbmcta2V5ZWQgb2JqZWN0XG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpc09iamVjdChpbnB1dDogYW55KTogaW5wdXQgaXMgUmVjb3JkPHN0cmluZywgYW55PiB7XG4gIHJldHVybiBPYmplY3QucHJvdG90eXBlLnRvU3RyaW5nLmFwcGx5KGlucHV0KSA9PT0gXCJbb2JqZWN0IE9iamVjdF1cIjtcbn1cbiIsCiAgICAiaW1wb3J0IHNqcyBmcm9tIFwic2VjdXJlLWpzb24tcGFyc2VcIjtcblxuLyoqXG4gKiBTYWZlbHkgcGFyc2VzIEpTT04gaW5wdXQgdXNpbmcgc2VjdXJlLWpzb24tcGFyc2Ugd2l0aCBwcm90b3R5cGUgcG9sbHV0aW9uIHByb3RlY3Rpb24uXG4gKiBAcGFyYW0ge2FueX0gaW5wdXQgLSBUaGUgaW5wdXQgdG8gYmUgcGFyc2VkIGFzIEpTT05cbiAqIEByZXR1cm5zIHt7IHZhbHVlPzogYW55LCBlcnI/OiBFcnJvciB9fSBBbiBvYmplY3QgY29udGFpbmluZyBlaXRoZXIgdGhlIHBhcnNlZCB2YWx1ZSBvciBhbiBlcnJvciBpZiBwYXJzaW5nIGZhaWxlZFxuICovXG5leHBvcnQgZnVuY3Rpb24ganNvblBhcnNlcihpbnB1dDogYW55KSB7XG4gIHRyeSB7XG4gICAgcmV0dXJuIHsgdmFsdWU6IHNqcy5wYXJzZShpbnB1dCwgeyBwcm90b0FjdGlvbjogXCJyZW1vdmVcIiB9KSB9O1xuICB9IGNhdGNoIChlcnIpIHtcbiAgICByZXR1cm4geyBlcnIgfTtcbiAgfVxufVxuIiwKICAgICIvLyBodHRwczovL2dpc3QuZ2l0aHViLmNvbS90M2RvdGdnL2E0ODZjNGFlNjZkMzJiZjE3YzA5YzczNjA5ZGFjYzViXG4vLyBUeXBlcyBmb3IgdGhlIHJlc3VsdCBvYmplY3Qgd2l0aCBkaXNjcmltaW5hdGVkIHVuaW9uXG4vLyB0eXBlIFN1Y2Nlc3M8VD4gPSB7XG4vLyAgIGRhdGE6IFQ7XG4vLyAgIGVycm9yOiBudWxsO1xuLy8gfTtcblxuLy8gdHlwZSBGYWlsdXJlPEU+ID0ge1xuLy8gICBkYXRhOiBudWxsO1xuLy8gICBlcnJvcjogRTtcbi8vIH07XG5cbi8vIGV4cG9ydCB0eXBlIFJlc3VsdDxULCBFID0gRXJyb3I+ID0gU3VjY2VzczxUPiB8IEZhaWx1cmU8RT47XG5cbmV4cG9ydCB0eXBlIFJlc3VsdDxULCBFPiA9IFt1bmRlZmluZWQsIEVdIHwgW1QsIHVuZGVmaW5lZF07XG5leHBvcnQgdHlwZSBQcm9taXNlUmVzdWx0PFQsIEU+ID0gUHJvbWlzZTxSZXN1bHQ8VCwgRT4+O1xuXG4vKipcbiAqIFdyYXBzIGEgc3luY2hyb25vdXMgZnVuY3Rpb24gaW4gYSB0cnktY2F0Y2ggYmxvY2sgYW5kIHJldHVybnMgYSBSZXN1bHQgdHVwbGUgY29udGFpbmluZyBlaXRoZXIgdGhlIHJldHVybmVkIHZhbHVlIG9yIHRoZSBjYXVnaHQgZXJyb3JcbiAqXG4gKiBAdGVtcGxhdGUgVCAtIFRoZSB0eXBlIG9mIHRoZSBzdWNjZXNzZnVsIHJlc3VsdCB2YWx1ZVxuICogQHRlbXBsYXRlIEUgLSBUaGUgdHlwZSBvZiB0aGUgZXJyb3IgdmFsdWUsIG11c3QgZXh0ZW5kIEVycm9yXG4gKiBAcGFyYW0geygpID0+IFR9IGZuIC0gVGhlIGZ1bmN0aW9uIHRvIGJlIGV4ZWN1dGVkXG4gKiBAcmV0dXJucyB7UmVzdWx0PFQsIEU+fSBBIHR1cGxlIGNvbnRhaW5pbmcgZWl0aGVyIFtkYXRhLCB1bmRlZmluZWRdIG9yIFt1bmRlZmluZWQsIGVycm9yXVxuICpcbiAqIEBleGFtcGxlXG4gKiBjb25zdCBbZGF0YSwgZXJyb3JdID0gdHJ5Q2F0Y2goKCkgPT4gc29tZVJpc2t5T3BlcmF0aW9uKCkpO1xuICogaWYgKGVycm9yKSB7XG4gKiAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICogfSBlbHNlIHtcbiAqICAgY29uc29sZS5sb2coZGF0YSk7XG4gKiB9XG4gKi9cbmV4cG9ydCBmdW5jdGlvbiB0cnlDYXRjaDxULCBFIGV4dGVuZHMgRXJyb3I+KGZuOiAoKSA9PiBUKTogUmVzdWx0PFQsIEU+IHtcbiAgdHJ5IHtcbiAgICBjb25zdCBkYXRhID0gZm4oKTtcbiAgICByZXR1cm4gW2RhdGEgYXMgVCwgdW5kZWZpbmVkXTtcbiAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICByZXR1cm4gW3VuZGVmaW5lZCwgZXJyb3IgYXMgRV07XG4gIH1cbn1cblxuLyoqXG4gKiBXcmFwcyBhIFByb21pc2UgaW4gYSB0cnktY2F0Y2ggYmxvY2sgYW5kIHJldHVybnMgYSBSZXN1bHQgb2JqZWN0IGNvbnRhaW5pbmcgZWl0aGVyIHRoZSByZXNvbHZlZCBkYXRhIG9yIHRoZSBjYXVnaHQgZXJyb3JcbiAqXG4gKiBAdGVtcGxhdGUgVCAtIFRoZSB0eXBlIG9mIHRoZSBzdWNjZXNzZnVsIHJlc3VsdCB2YWx1ZVxuICogQHRlbXBsYXRlIEUgLSBUaGUgdHlwZSBvZiB0aGUgZXJyb3IgdmFsdWUsIGRlZmF1bHRzIHRvIEVycm9yXG4gKiBAcGFyYW0ge1Byb21pc2U8VD59IHByb21pc2UgLSBUaGUgcHJvbWlzZSB0byBiZSBleGVjdXRlZFxuICogQHJldHVybnMge1Byb21pc2U8UmVzdWx0PFQsIEU+Pn0gQSBQcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gYSBSZXN1bHQgb2JqZWN0IGNvbnRhaW5pbmcgZWl0aGVyIHRoZSBkYXRhIG9yIGVycm9yXG4gKlxuICogQGV4YW1wbGVcbiAqIGNvbnN0IHtkYXRhLCBlcnJvcn0gPSBhd2FpdCB0cnlDYXRjaEFzeW5jKHNvbWVQcm9taXNlKTtcbiAqIGlmIChlcnJvcikge1xuICogICBjb25zb2xlLmVycm9yKHJlc3VsdC5lcnJvcik7XG4gKiB9IGVsc2Uge1xuICogICBjb25zb2xlLmxvZyhkYXRhKTtcbiAqIH1cbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHRyeUNhdGNoQXN5bmM8VCwgRSBleHRlbmRzIEVycm9yPihwcm9taXNlOiBQcm9taXNlPFQ+KTogUHJvbWlzZVJlc3VsdDxULCBFPiB7XG4gIHRyeSB7XG4gICAgY29uc3QgZGF0YSA9IGF3YWl0IHByb21pc2U7XG4gICAgcmV0dXJuIFtkYXRhIGFzIFQsIHVuZGVmaW5lZF07XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgcmV0dXJuIFt1bmRlZmluZWQsIGVycm9yIGFzIEVdO1xuICB9XG59XG4iLAogICAgImV4cG9ydCBjb25zdCBpc051bGxPclVuZGVmaW5lZCA9ICh2YWx1ZTogYW55KTogdmFsdWUgaXMgbnVsbCB8IHVuZGVmaW5lZCA9PiB2YWx1ZSA9PT0gbnVsbCB8fCB2YWx1ZSA9PT0gdW5kZWZpbmVkO1xuXG5leHBvcnQgZnVuY3Rpb24gaXNOdWxsT3JFbXB0eU9yVW5kZWZpbmVkKHZhbHVlPzogYW55IHwgbnVsbCk6IHZhbHVlIGlzIG51bGwgfCB1bmRlZmluZWQge1xuICByZXR1cm4gdmFsdWUgPT09IHVuZGVmaW5lZCB8fCB2YWx1ZSA9PT0gbnVsbCB8fCB2YWx1ZSA9PT0gXCJcIiB8fCB2YWx1ZS50b1N0cmluZygpLnRyaW0oKSA9PT0gXCJcIjtcbn1cbiIKICBdLAogICJtYXBwaW5ncyI6ICI7QUFpQkEsZ0JBQXVCLGFBQWEsQ0FBQyxRQUE0QztBQUFBLEVBQy9FLElBQUksV0FBVztBQUFBLEVBRWYsaUJBQWlCLFNBQVMsUUFBUTtBQUFBLElBQ2hDLFlBQVk7QUFBQSxJQUNaLElBQUk7QUFBQSxJQUdKLFFBQVEsV0FBVyxTQUFTLFFBQVE7QUFBQSxDQUFJLE1BQU0sR0FBRztBQUFBLE1BRS9DLE1BQU0sT0FBTyxTQUFTLE1BQU0sR0FBRyxXQUFXLENBQUM7QUFBQSxNQUMzQyxNQUFNLE1BQU0sS0FBSyxLQUFLO0FBQUEsTUFDdEIsV0FBVyxTQUFTLE1BQU0sV0FBVyxDQUFDO0FBQUEsSUFDeEM7QUFBQSxFQUNGO0FBQUEsRUFFQSxJQUFJLFNBQVMsU0FBUyxHQUFHO0FBQUEsSUFDdkIsTUFBTTtBQUFBLEVBQ1I7QUFBQTs7QUM3QkssU0FBUyxRQUFRLENBQUMsT0FBMEM7QUFBQSxFQUNqRSxPQUFPLE9BQU8sVUFBVSxTQUFTLE1BQU0sS0FBSyxNQUFNO0FBQUE7O0FDUHBEO0FBT08sU0FBUyxVQUFVLENBQUMsT0FBWTtBQUFBLEVBQ3JDLElBQUk7QUFBQSxJQUNGLE9BQU8sRUFBRSxPQUFPLElBQUksTUFBTSxPQUFPLEVBQUUsYUFBYSxTQUFTLENBQUMsRUFBRTtBQUFBLElBQzVELE9BQU8sS0FBSztBQUFBLElBQ1osT0FBTyxFQUFFLElBQUk7QUFBQTtBQUFBOztBQ3NCVixTQUFTLFFBQTRCLENBQUMsSUFBMkI7QUFBQSxFQUN0RSxJQUFJO0FBQUEsSUFDRixNQUFNLE9BQU8sR0FBRztBQUFBLElBQ2hCLE9BQU8sQ0FBQyxNQUFXLFNBQVM7QUFBQSxJQUM1QixPQUFPLE9BQU87QUFBQSxJQUNkLE9BQU8sQ0FBQyxXQUFXLEtBQVU7QUFBQTtBQUFBO0FBb0JqQyxlQUFzQixhQUFpQyxDQUFDLFNBQTBDO0FBQUEsRUFDaEcsSUFBSTtBQUFBLElBQ0YsTUFBTSxPQUFPLE1BQU07QUFBQSxJQUNuQixPQUFPLENBQUMsTUFBVyxTQUFTO0FBQUEsSUFDNUIsT0FBTyxPQUFPO0FBQUEsSUFDZCxPQUFPLENBQUMsV0FBVyxLQUFVO0FBQUE7QUFBQTs7QUMvRDFCLElBQU0sb0JBQW9CLENBQUMsVUFBMEMsVUFBVSxRQUFRLFVBQVU7QUFFakcsU0FBUyx3QkFBd0IsQ0FBQyxPQUErQztBQUFBLEVBQ3RGLE9BQU8sVUFBVSxhQUFhLFVBQVUsUUFBUSxVQUFVLE1BQU0sTUFBTSxTQUFTLEVBQUUsS0FBSyxNQUFNO0FBQUE7IiwKICAiZGVidWdJZCI6ICIyRjFEMjE2QzNENUM3NTFENjQ3NTZFMjE2NDc1NkUyMSIsCiAgIm5hbWVzIjogW10KfQ==
|
|
214
|
+
//# debugId=2329ACCD00D0C99964756E2164756E21
|
|
215
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/cache/cacheService.ts", "../src/utils/isCallable.ts", "../src/cache/constants.ts", "../src/utils/chunksToLines.ts", "../src/utils/isNullOrUndefined.ts", "../src/utils/isObject.ts", "../src/utils/isPrimitive.ts", "../src/utils/jsonParser.ts", "../src/utils/randomString.ts", "../src/utils/sleep.ts", "../src/neverthrow/tryCatch.ts"],
  "sourcesContent": [
    "import { isPromise } from \"node:util/types\";\nimport { isCallable } from \"../utils/isCallable.ts\";\nimport type { CacheMetadataOptions, CacheOptions, CacheWithMetadata, ValueLoader } from \"./internal/types.ts\";\nimport type { CacheMetadata, ICacheAdapter } from \"./types.ts\";\n\nconst defaultCacheOptions: CacheOptions = {\n  expirationThresholdMs: 0,\n  ttlMs: 60 * 60 * 1000,\n};\n\nexport class Cache<TAdapter extends ICacheAdapter | undefined = undefined> {\n  private inMemoryCache: Map<string, CacheWithMetadata> = new Map();\n  private readonly options: CacheOptions;\n  private readonly adapter?: TAdapter;\n\n  constructor(defaultOptions?: Partial<CacheOptions>);\n  constructor(adapter?: TAdapter, defaultOptions?: Partial<CacheOptions>);\n  constructor(adapter?: TAdapter | Partial<CacheOptions>, defaultOptions?: Partial<CacheOptions>) {\n    const options = adapter && !(\"getValue\" in adapter) ? (adapter as Partial<CacheOptions>) : defaultOptions;\n    if (adapter && \"getValue\" in adapter && \"setValue\" in adapter) {\n      this.adapter = adapter as TAdapter;\n    }\n\n    this.options = {\n      ...defaultCacheOptions,\n      ...options,\n    };\n  }\n\n  private async getFromAdapter<T>(cacheKey: string): Promise<T | undefined> {\n    if (!this.adapter) {\n      return undefined;\n    }\n\n    return this.adapter.getValue(cacheKey);\n  }\n\n  private async saveToAdapter<T>(cacheKey: string, value: T): Promise<void> {\n    if (!this.adapter) {\n      return;\n    }\n\n    return this.adapter.setValue<T>(cacheKey, value);\n  }\n\n  public async get<T>(\n    cacheKey: string,\n    valueLoader?: ValueLoader<T>,\n    options: Partial<CacheMetadataOptions> = {},\n  ): Promise<T | undefined> {\n    const mergedOptions = this.mergeOptions(options);\n\n    // Try in-memory cache first\n    const cachedValue = this.tryGetFromMemory<T>(cacheKey);\n    if (cachedValue !== undefined) {\n      return cachedValue;\n    }\n\n    // Load fresh value from loader function first\n    const freshValue = await this.tryGetFromValueLoader(valueLoader);\n    if (freshValue !== undefined) {\n      await this.cacheValue(cacheKey, freshValue, mergedOptions);\n\n      return freshValue;\n    }\n\n    // Try adapter cache second\n    const adapterValue = await this.tryGetFromAdapter<T>(cacheKey);\n    if (adapterValue !== undefined) {\n      await this.cacheValue(cacheKey, adapterValue, mergedOptions);\n\n      return adapterValue;\n    }\n  }\n\n  private mergeOptions(options: Partial<CacheMetadataOptions>): CacheMetadataOptions {\n    return {\n      ttlMs: this.options.ttlMs,\n      expirationThresholdMs: this.options.expirationThresholdMs,\n      ...options,\n    };\n  }\n\n  private tryGetFromMemory<T>(cacheKey: string): T | undefined {\n    const cached = this.inMemoryCache.get(cacheKey);\n\n    if (cached && !this.isExpired(cached.metadata)) {\n      return cached.data as T;\n    }\n\n    return undefined;\n  }\n\n  private async tryGetFromAdapter<T>(cacheKey: string): Promise<T | undefined> {\n    const cached = await this.getFromAdapter<CacheWithMetadata<T>>(cacheKey);\n\n    if (cached && !this.isExpired(cached.metadata)) {\n      this.inMemoryCache.set(cacheKey, cached);\n      return cached.data;\n    }\n\n    return undefined;\n  }\n\n  private async tryGetFromValueLoader<T>(valueLoader?: ValueLoader<T>): Promise<T | undefined> {\n    if (valueLoader === undefined) {\n      return undefined;\n    }\n\n    if (isCallable(valueLoader)) {\n      return await valueLoader();\n    }\n\n    if (isPromise(valueLoader)) {\n      return await valueLoader;\n    }\n\n    return valueLoader;\n  }\n\n  private async cacheValue<T>(cacheKey: string, value: T, options: CacheMetadataOptions): Promise<void> {\n    const cacheData = this.createCacheEntry(value, options);\n\n    await this.saveToAdapter(cacheKey, cacheData);\n    this.inMemoryCache.set(cacheKey, cacheData);\n  }\n\n  private createCacheEntry<T>(value: T, options: CacheMetadataOptions): CacheWithMetadata<T> {\n    const now = Date.now();\n    let expiresAt: number;\n    if (options.ttlMs < 0) {\n      expiresAt = -1;\n    } else {\n      expiresAt = now + options.ttlMs - options.expirationThresholdMs;\n    }\n\n    return {\n      data: value,\n      metadata: {\n        createdAt: now,\n        ttlMs: options.ttlMs,\n        expirationThresholdMs: options.expirationThresholdMs,\n        expiresAt,\n      },\n    };\n  }\n\n  private isExpired(metadata?: CacheMetadata): boolean {\n    if (!metadata) {\n      return true;\n    }\n\n    if (metadata.expiresAt < 0) {\n      return false;\n    }\n\n    const expirationTime =\n      metadata.expiresAt ?? (metadata.createdAt && metadata.ttlMs ? metadata.createdAt + metadata.ttlMs : undefined);\n\n    if (expirationTime === undefined) {\n      return true;\n    }\n\n    return Date.now() > expirationTime;\n  }\n}\n",
    "export function isCallable(value: any): value is CallableFunction {\n  return typeof value === \"function\";\n}\n",
    "export const CACHE_TTL_NEVER_EXPIRE = -1;\n",
    "/**\n * Converts an async iterable of string or Uint8Array chunks into an async iterable of lines.\n * Handles cases where lines may be split across multiple chunks.\n * Usage:\n * ```ts\n * import { pipeline } from \"node:stream/promises\";\n * import { createReadStream } from \"node:fs\";\n * import { chunksToLines } from \"@lukaskj/ts-utils\";\n *\n * const readStream = createReadStream(\"path/to/file.txt\", { encoding: \"utf-8\" });\n * await pipeline(readStream, chunksToLines, process.stdout).catch((err) => {\n *   console.error(err);\n *   process.exit(1);\n * });\n * ```\n * @param chunks\n */\nexport async function* chunksToLines(chunks: AsyncIterable<string | Uint8Array>) {\n  let previous = \"\";\n\n  for await (const chunk of chunks) {\n    previous += chunk;\n    let eolIndex: number;\n\n    // biome-ignore lint/suspicious/noAssignInExpressions: \"explanation\"\n    while ((eolIndex = previous.indexOf(\"\\n\")) >= 0) {\n      // this line includes the EOL\n      const line = previous.slice(0, eolIndex + 1);\n      yield line?.trim() ?? \"\";\n      previous = previous.slice(eolIndex + 1);\n    }\n  }\n\n  if (previous.length > 0) {\n    yield previous;\n  }\n}\n",
    "export const isNullOrUndefined = (value: any): value is null | undefined => value === null || value === undefined;\n\nexport function isNullOrEmptyOrUndefined(value?: any | null): value is null | undefined {\n  return value === undefined || value === null || value === \"\" || value.toString().trim() === \"\";\n}\n",
    "/**\n * Type guard that checks if the input is a plain JavaScript object.\n * @param {any} input - The value to check\n * @returns {boolean} True if the input is a plain object, false otherwise\n * @typeParam {Record<string, any>} Type guard narrows the input type to a string-keyed object\n */\nexport function isObject(input: any): input is Record<string, any> {\n  return Object.prototype.toString.apply(input) === \"[object Object]\";\n}\n",
    "export function isPrimitive(value: unknown): boolean {\n  return (typeof value !== \"object\" && typeof value !== \"function\") || value === null || value instanceof Date;\n}\n",
    "import sjs from \"secure-json-parse\";\n\n/**\n * Safely parses JSON input using secure-json-parse with prototype pollution protection.\n * @param {any} input - The input to be parsed as JSON\n * @returns {{ value?: any, err?: Error }} An object containing either the parsed value or an error if parsing failed\n */\nexport function jsonParser(input: any) {\n  try {\n    return { value: sjs.parse(input, { protoAction: \"remove\" }) };\n  } catch (err) {\n    return { err };\n  }\n}\n",
    "export function randomString(length: number = 6): string {\n  let result = \"\";\n  const characters = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789\";\n  const charactersLength = characters.length;\n  for (let i = 0; i < length; i++) {\n    result += characters.charAt(Math.floor(Math.random() * charactersLength));\n  }\n  return result;\n}\n",
    "export const sleep = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms));\n",
    "// https://gist.github.com/t3dotgg/a486c4ae66d32bf17c09c73609dacc5b\n// Types for the result object with discriminated union\n// type Success<T> = {\n//   data: T;\n//   error: null;\n// };\n\n// type Failure<E> = {\n//   data: null;\n//   error: E;\n// };\n\n// export type Result<T, E = Error> = Success<T> | Failure<E>;\n\nexport type Result<T, E> = [undefined, E] | [T, undefined];\nexport type PromiseResult<T, E> = Promise<Result<T, E>>;\n\n/**\n * Wraps a synchronous function in a try-catch block and returns a Result tuple containing either the returned value or the caught error\n *\n * @template T - The type of the successful result value\n * @template E - The type of the error value, must extend Error\n * @param {() => T} fn - The function to be executed\n * @returns {Result<T, E>} A tuple containing either [data, undefined] or [undefined, error]\n *\n * @example\n * const [data, error] = tryCatch(() => someRiskyOperation());\n * if (error) {\n *   console.error(error);\n * } else {\n *   console.log(data);\n * }\n */\nexport function tryCatch<T, E extends Error>(fn: () => T): Result<T, E> {\n  try {\n    const data = fn();\n    return [data as T, undefined];\n  } catch (error) {\n    return [undefined, error as E];\n  }\n}\n\n/**\n * Wraps a Promise in a try-catch block and returns a Result object containing either the resolved data or the caught error\n *\n * @template T - The type of the successful result value\n * @template E - The type of the error value, defaults to Error\n * @param {Promise<T>} promise - The promise to be executed\n * @returns {Promise<Result<T, E>>} A Promise that resolves to a Result object containing either the data or error\n *\n * @example\n * const {data, error} = await tryCatchAsync(somePromise);\n * if (error) {\n *   console.error(result.error);\n * } else {\n *   console.log(data);\n * }\n */\nexport async function tryCatchAsync<T, E extends Error>(promise: Promise<T>): PromiseResult<T, E> {\n  try {\n    const data = await promise;\n    return [data as T, undefined];\n  } catch (error) {\n    return [undefined, error as E];\n  }\n}\n"
  ],
  "mappings": "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",
  "debugId": "2329ACCD00D0C99964756E2164756E21",
  "names": []
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare function isCallable(value: any): value is CallableFunction;
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare const sleep: (ms: number) => Promise<unknown>;
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@lukaskj/ts-utils",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.3",
|
|
4
4
|
"module": "src/index.ts",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"scripts": {
|
|
@@ -23,6 +23,7 @@
|
|
|
23
23
|
"@faker-js/faker": "^9.9.0",
|
|
24
24
|
"@types/bun": "latest",
|
|
25
25
|
"@types/node": "^24.2.1",
|
|
26
|
+
"bun-mock-extended": "^3.0.5",
|
|
26
27
|
"husky": "^9.1.7",
|
|
27
28
|
"lint-staged": "^16.1.5",
|
|
28
29
|
"typescript": "^5.9.2"
|
|
@@ -65,6 +66,16 @@
|
|
|
65
66
|
"default": "./dist/colors/index.cjs",
|
|
66
67
|
"types": "./dist/colors/index.d.ts"
|
|
67
68
|
}
|
|
69
|
+
},
|
|
70
|
+
"./cache": {
|
|
71
|
+
"import": {
|
|
72
|
+
"default": "./dist/cache/index.mjs",
|
|
73
|
+
"types": "./dist/cache/index.d.ts"
|
|
74
|
+
},
|
|
75
|
+
"require": {
|
|
76
|
+
"default": "./dist/cache/index.cjs",
|
|
77
|
+
"types": "./dist/cache/index.d.ts"
|
|
78
|
+
}
|
|
68
79
|
}
|
|
69
80
|
},
|
|
70
81
|
"engines": {
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|