@agentica/core 0.44.0-dev.20260313 → 0.44.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -21
- package/README.md +218 -218
- package/lib/context/internal/__IChatInitialApplication.d.ts +1 -2
- package/lib/errors/AgenticaJsonParseError.js +6 -6
- package/lib/index.mjs +47 -1
- package/lib/index.mjs.map +1 -1
- package/lib/orchestrate/call.js +16 -16
- package/lib/orchestrate/initialize.js +43 -1
- package/lib/orchestrate/initialize.js.map +1 -1
- package/lib/structures/IAgenticaController.d.ts +143 -143
- package/lib/utils/ChatGptCompletionMessageUtil.js +6 -6
- package/package.json +6 -6
- package/prompts/cancel.md +5 -5
- package/prompts/common.md +3 -3
- package/prompts/describe.md +7 -7
- package/prompts/execute.md +122 -122
- package/prompts/initialize.md +3 -3
- package/prompts/json_parse_error.md +35 -35
- package/prompts/select.md +7 -7
- package/prompts/validate.md +123 -123
- package/prompts/validate_repeated.md +31 -31
- package/src/Agentica.ts +367 -367
- package/src/MicroAgentica.ts +357 -357
- package/src/constants/AgenticaConstant.ts +4 -4
- package/src/constants/AgenticaDefaultPrompt.ts +44 -44
- package/src/constants/index.ts +2 -2
- package/src/context/AgenticaContext.ts +136 -136
- package/src/context/AgenticaContextRequestResult.ts +14 -14
- package/src/context/AgenticaOperation.ts +73 -73
- package/src/context/AgenticaOperationCollection.ts +49 -49
- package/src/context/AgenticaOperationSelection.ts +9 -9
- package/src/context/AgenticaTokenUsage.ts +186 -186
- package/src/context/MicroAgenticaContext.ts +99 -99
- package/src/context/index.ts +5 -5
- package/src/context/internal/AgenticaOperationComposer.ts +177 -177
- package/src/context/internal/AgenticaTokenUsageAggregator.ts +66 -66
- package/src/context/internal/__IChatCancelFunctionsApplication.ts +23 -23
- package/src/context/internal/__IChatFunctionReference.ts +21 -21
- package/src/context/internal/__IChatInitialApplication.ts +13 -15
- package/src/context/internal/__IChatSelectFunctionsApplication.ts +24 -24
- package/src/context/internal/isAgenticaContext.ts +11 -11
- package/src/errors/AgenticaJsonParseError.ts +52 -52
- package/src/errors/AgenticaValidationError.ts +49 -49
- package/src/errors/index.ts +2 -2
- package/src/events/AgenticaAssistantMessageEvent.ts +12 -12
- package/src/events/AgenticaCallEvent.ts +27 -27
- package/src/events/AgenticaCancelEvent.ts +9 -9
- package/src/events/AgenticaDescribeEvent.ts +14 -14
- package/src/events/AgenticaEvent.ts +59 -59
- package/src/events/AgenticaEvent.type.ts +19 -19
- package/src/events/AgenticaEventBase.ts +18 -18
- package/src/events/AgenticaEventSource.ts +6 -6
- package/src/events/AgenticaExecuteEvent.ts +45 -45
- package/src/events/AgenticaInitializeEvent.ts +7 -7
- package/src/events/AgenticaJsonParseErrorEvent.ts +16 -16
- package/src/events/AgenticaRequestEvent.ts +27 -27
- package/src/events/AgenticaResponseEvent.ts +32 -32
- package/src/events/AgenticaSelectEvent.ts +11 -11
- package/src/events/AgenticaUserMessageEvent.ts +12 -12
- package/src/events/AgenticaValidateEvent.ts +32 -32
- package/src/events/MicroAgenticaEvent.ts +45 -45
- package/src/events/index.ts +15 -15
- package/src/factory/events.ts +357 -357
- package/src/factory/histories.ts +348 -348
- package/src/factory/index.ts +3 -3
- package/src/factory/operations.ts +16 -16
- package/src/functional/assertHttpController.ts +106 -106
- package/src/functional/assertHttpLlmApplication.ts +52 -52
- package/src/functional/assertMcpController.ts +47 -47
- package/src/functional/createMcpLlmApplication.ts +72 -72
- package/src/functional/index.ts +7 -7
- package/src/functional/validateHttpController.ts +113 -113
- package/src/functional/validateHttpLlmApplication.ts +65 -65
- package/src/functional/validateMcpController.ts +53 -53
- package/src/histories/AgenticaAssistantMessageHistory.ts +10 -10
- package/src/histories/AgenticaCancelHistory.ts +8 -8
- package/src/histories/AgenticaDescribeHistory.ts +18 -18
- package/src/histories/AgenticaExecuteHistory.ts +64 -64
- package/src/histories/AgenticaHistory.ts +28 -28
- package/src/histories/AgenticaHistoryBase.ts +35 -35
- package/src/histories/AgenticaSelectHistory.ts +8 -8
- package/src/histories/AgenticaSystemMessageHistory.ts +10 -10
- package/src/histories/AgenticaUserMessageHistory.ts +11 -11
- package/src/histories/MicroAgenticaHistory.ts +19 -19
- package/src/histories/contents/AgenticaUserMessageAudioContent.ts +21 -21
- package/src/histories/contents/AgenticaUserMessageContent.ts +19 -19
- package/src/histories/contents/AgenticaUserMessageContentBase.ts +6 -6
- package/src/histories/contents/AgenticaUserMessageFileContent.ts +25 -25
- package/src/histories/contents/AgenticaUserMessageImageContent.ts +33 -33
- package/src/histories/contents/AgenticaUserMessageTextContent.ts +15 -15
- package/src/histories/contents/index.ts +5 -5
- package/src/histories/index.ts +10 -10
- package/src/index.ts +15 -15
- package/src/json/IAgenticaEventJson.ts +265 -265
- package/src/json/IAgenticaEventJson.type.ts +19 -19
- package/src/json/IAgenticaHistoryJson.ts +165 -165
- package/src/json/IAgenticaHistoryJson.type.ts +19 -19
- package/src/json/IAgenticaOperationJson.ts +36 -36
- package/src/json/IAgenticaOperationSelectionJson.ts +26 -26
- package/src/json/IAgenticaTokenUsageJson.ts +107 -107
- package/src/json/IMicroAgenticaEventJson.ts +22 -22
- package/src/json/IMicroAgenticaHistoryJson.ts +25 -25
- package/src/json/index.ts +7 -7
- package/src/orchestrate/call.ts +542 -542
- package/src/orchestrate/cancel.ts +265 -265
- package/src/orchestrate/describe.ts +66 -66
- package/src/orchestrate/execute.ts +61 -61
- package/src/orchestrate/index.ts +6 -6
- package/src/orchestrate/initialize.ts +102 -102
- package/src/orchestrate/internal/cancelFunctionFromContext.ts +33 -33
- package/src/orchestrate/internal/selectFunctionFromContext.ts +34 -34
- package/src/orchestrate/select.ts +320 -320
- package/src/structures/IAgenticaConfig.ts +83 -83
- package/src/structures/IAgenticaConfigBase.ts +87 -87
- package/src/structures/IAgenticaController.ts +143 -143
- package/src/structures/IAgenticaExecutor.ts +167 -167
- package/src/structures/IAgenticaProps.ts +78 -78
- package/src/structures/IAgenticaSystemPrompt.ts +236 -236
- package/src/structures/IAgenticaVendor.ts +54 -54
- package/src/structures/IMcpTool.ts +60 -60
- package/src/structures/IMicroAgenticaConfig.ts +56 -56
- package/src/structures/IMicroAgenticaExecutor.ts +67 -67
- package/src/structures/IMicroAgenticaProps.ts +77 -77
- package/src/structures/IMicroAgenticaSystemPrompt.ts +169 -169
- package/src/structures/index.ts +10 -10
- package/src/transformers/transformHistory.ts +172 -172
- package/src/utils/AssistantMessageEmptyError.ts +20 -20
- package/src/utils/AsyncQueue.spec.ts +355 -355
- package/src/utils/AsyncQueue.ts +95 -95
- package/src/utils/ByteArrayUtil.ts +5 -5
- package/src/utils/ChatGptCompletionMessageUtil.spec.ts +314 -314
- package/src/utils/ChatGptCompletionMessageUtil.ts +210 -210
- package/src/utils/ChatGptCompletionStreamingUtil.spec.ts +909 -909
- package/src/utils/ChatGptCompletionStreamingUtil.ts +91 -91
- package/src/utils/ChatGptTokenUsageAggregator.spec.ts +226 -226
- package/src/utils/ChatGptTokenUsageAggregator.ts +57 -57
- package/src/utils/MPSC.spec.ts +276 -276
- package/src/utils/MPSC.ts +42 -42
- package/src/utils/Singleton.spec.ts +138 -138
- package/src/utils/Singleton.ts +42 -42
- package/src/utils/StreamUtil.spec.ts +512 -512
- package/src/utils/StreamUtil.ts +87 -87
- package/src/utils/__map_take.spec.ts +140 -140
- package/src/utils/__map_take.ts +13 -13
- package/src/utils/__retry.spec.ts +198 -198
- package/src/utils/__retry.ts +18 -18
- package/src/utils/assertExecuteFailure.ts +16 -16
- package/src/utils/index.ts +4 -4
- package/src/utils/request.ts +140 -140
- package/src/utils/types.ts +50 -50
package/src/utils/StreamUtil.ts
CHANGED
|
@@ -1,87 +1,87 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @module StreamUtil
|
|
3
|
-
*
|
|
4
|
-
* Utility functions for streams.
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
async function readAll<T>(stream: ReadableStream<T>, abortSignal?: AbortSignal): Promise<T[]> {
|
|
8
|
-
const reader = stream.getReader();
|
|
9
|
-
const result: T[] = [];
|
|
10
|
-
while (true) {
|
|
11
|
-
const { done, value } = await reader.read();
|
|
12
|
-
if (done || abortSignal?.aborted === true) {
|
|
13
|
-
break;
|
|
14
|
-
}
|
|
15
|
-
result.push(value);
|
|
16
|
-
}
|
|
17
|
-
return result;
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
async function reduce<T, R = T>(stream: ReadableStream<T>, reducer: (acc: T | R, cur: T) => R, options: { initial?: R; abortSignal?: AbortSignal }): Promise<R | null> {
|
|
21
|
-
const reader = stream.getReader();
|
|
22
|
-
const iterator = streamDefaultReaderToAsyncGenerator(reader);
|
|
23
|
-
let acc = (options.initial ?? null) as R | null | T;
|
|
24
|
-
|
|
25
|
-
for await (const value of iterator) {
|
|
26
|
-
if (options.abortSignal?.aborted === true) {
|
|
27
|
-
break;
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
if (acc === null) {
|
|
31
|
-
acc = value;
|
|
32
|
-
continue;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
acc = reducer(acc, value);
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
return acc as R;
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
function from<T>(...value: T[]): ReadableStream<T> {
|
|
42
|
-
const stream = new ReadableStream<T>({
|
|
43
|
-
start: (controller) => {
|
|
44
|
-
value.forEach(v => controller.enqueue(v));
|
|
45
|
-
controller.close();
|
|
46
|
-
},
|
|
47
|
-
});
|
|
48
|
-
|
|
49
|
-
return stream;
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
export async function* toAsyncGenerator<T>(value: T): AsyncGenerator<T, undefined, undefined> {
|
|
53
|
-
yield value;
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
export async function* streamDefaultReaderToAsyncGenerator<T>(reader: ReadableStreamDefaultReader<T>, abortSignal?: AbortSignal): AsyncGenerator<Awaited<T>, undefined, undefined> {
|
|
57
|
-
while (true) {
|
|
58
|
-
const { done, value } = await reader.read();
|
|
59
|
-
if (done || abortSignal?.aborted === true) {
|
|
60
|
-
break;
|
|
61
|
-
}
|
|
62
|
-
yield value;
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
function transform<T, R>(stream: ReadableStream<T>, transformer: (value: T) => R, abortSignal?: AbortSignal): ReadableStream<R> {
|
|
67
|
-
const reader = stream.getReader();
|
|
68
|
-
|
|
69
|
-
return new ReadableStream<R>({
|
|
70
|
-
pull: async (controller) => {
|
|
71
|
-
const { done, value } = await reader.read();
|
|
72
|
-
if (done === true || abortSignal?.aborted === true) {
|
|
73
|
-
controller.close();
|
|
74
|
-
return;
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
controller.enqueue(transformer(value));
|
|
78
|
-
},
|
|
79
|
-
});
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
export const StreamUtil = {
|
|
83
|
-
readAll,
|
|
84
|
-
reduce,
|
|
85
|
-
from,
|
|
86
|
-
transform,
|
|
87
|
-
};
|
|
1
|
+
/**
|
|
2
|
+
* @module StreamUtil
|
|
3
|
+
*
|
|
4
|
+
* Utility functions for streams.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
async function readAll<T>(stream: ReadableStream<T>, abortSignal?: AbortSignal): Promise<T[]> {
|
|
8
|
+
const reader = stream.getReader();
|
|
9
|
+
const result: T[] = [];
|
|
10
|
+
while (true) {
|
|
11
|
+
const { done, value } = await reader.read();
|
|
12
|
+
if (done || abortSignal?.aborted === true) {
|
|
13
|
+
break;
|
|
14
|
+
}
|
|
15
|
+
result.push(value);
|
|
16
|
+
}
|
|
17
|
+
return result;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
async function reduce<T, R = T>(stream: ReadableStream<T>, reducer: (acc: T | R, cur: T) => R, options: { initial?: R; abortSignal?: AbortSignal }): Promise<R | null> {
|
|
21
|
+
const reader = stream.getReader();
|
|
22
|
+
const iterator = streamDefaultReaderToAsyncGenerator(reader);
|
|
23
|
+
let acc = (options.initial ?? null) as R | null | T;
|
|
24
|
+
|
|
25
|
+
for await (const value of iterator) {
|
|
26
|
+
if (options.abortSignal?.aborted === true) {
|
|
27
|
+
break;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
if (acc === null) {
|
|
31
|
+
acc = value;
|
|
32
|
+
continue;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
acc = reducer(acc, value);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
return acc as R;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
function from<T>(...value: T[]): ReadableStream<T> {
|
|
42
|
+
const stream = new ReadableStream<T>({
|
|
43
|
+
start: (controller) => {
|
|
44
|
+
value.forEach(v => controller.enqueue(v));
|
|
45
|
+
controller.close();
|
|
46
|
+
},
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
return stream;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
export async function* toAsyncGenerator<T>(value: T): AsyncGenerator<T, undefined, undefined> {
|
|
53
|
+
yield value;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
export async function* streamDefaultReaderToAsyncGenerator<T>(reader: ReadableStreamDefaultReader<T>, abortSignal?: AbortSignal): AsyncGenerator<Awaited<T>, undefined, undefined> {
|
|
57
|
+
while (true) {
|
|
58
|
+
const { done, value } = await reader.read();
|
|
59
|
+
if (done || abortSignal?.aborted === true) {
|
|
60
|
+
break;
|
|
61
|
+
}
|
|
62
|
+
yield value;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
function transform<T, R>(stream: ReadableStream<T>, transformer: (value: T) => R, abortSignal?: AbortSignal): ReadableStream<R> {
|
|
67
|
+
const reader = stream.getReader();
|
|
68
|
+
|
|
69
|
+
return new ReadableStream<R>({
|
|
70
|
+
pull: async (controller) => {
|
|
71
|
+
const { done, value } = await reader.read();
|
|
72
|
+
if (done === true || abortSignal?.aborted === true) {
|
|
73
|
+
controller.close();
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
controller.enqueue(transformer(value));
|
|
78
|
+
},
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
export const StreamUtil = {
|
|
83
|
+
readAll,
|
|
84
|
+
reduce,
|
|
85
|
+
from,
|
|
86
|
+
transform,
|
|
87
|
+
};
|
|
@@ -1,140 +1,140 @@
|
|
|
1
|
-
import { __map_take } from "./__map_take";
|
|
2
|
-
|
|
3
|
-
describe("__map_take", () => {
|
|
4
|
-
describe("basic functionality", () => {
|
|
5
|
-
it("should generate value for new key", () => {
|
|
6
|
-
const map = new Map<string, number>();
|
|
7
|
-
const generator = () => 42;
|
|
8
|
-
|
|
9
|
-
const result = __map_take(map, "test", generator);
|
|
10
|
-
|
|
11
|
-
expect(result).toBe(42);
|
|
12
|
-
expect(map.get("test")).toBe(42);
|
|
13
|
-
});
|
|
14
|
-
|
|
15
|
-
it("should return existing value for existing key", () => {
|
|
16
|
-
const map = new Map<string, number>();
|
|
17
|
-
map.set("test", 100);
|
|
18
|
-
|
|
19
|
-
const generator = () => 42;
|
|
20
|
-
const result = __map_take(map, "test", generator);
|
|
21
|
-
|
|
22
|
-
expect(result).toBe(100);
|
|
23
|
-
expect(map.get("test")).toBe(100);
|
|
24
|
-
});
|
|
25
|
-
});
|
|
26
|
-
|
|
27
|
-
describe("various type tests", () => {
|
|
28
|
-
it("should handle object type", () => {
|
|
29
|
-
const map = new Map<string, { value: number }>();
|
|
30
|
-
const generator = () => ({ value: 42 });
|
|
31
|
-
|
|
32
|
-
const result = __map_take(map, "test", generator);
|
|
33
|
-
|
|
34
|
-
expect(result).toEqual({ value: 42 });
|
|
35
|
-
expect(map.get("test")).toEqual({ value: 42 });
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
it("should handle array type", () => {
|
|
39
|
-
const map = new Map<string, number[]>();
|
|
40
|
-
const generator = () => [1, 2, 3];
|
|
41
|
-
|
|
42
|
-
const result = __map_take(map, "test", generator);
|
|
43
|
-
|
|
44
|
-
expect(result).toEqual([1, 2, 3]);
|
|
45
|
-
expect(map.get("test")).toEqual([1, 2, 3]);
|
|
46
|
-
});
|
|
47
|
-
|
|
48
|
-
it("should handle function type", () => {
|
|
49
|
-
const map = new Map<string, () => number>();
|
|
50
|
-
const generator = () => () => 42;
|
|
51
|
-
|
|
52
|
-
const result = __map_take(map, "test", generator);
|
|
53
|
-
|
|
54
|
-
expect(result()).toBe(42);
|
|
55
|
-
expect(map.get("test")?.()).toBe(42);
|
|
56
|
-
});
|
|
57
|
-
});
|
|
58
|
-
|
|
59
|
-
describe("edge cases", () => {
|
|
60
|
-
it("should handle null key", () => {
|
|
61
|
-
const map = new Map<null, string>();
|
|
62
|
-
const generator = () => "test";
|
|
63
|
-
|
|
64
|
-
const result = __map_take(map, null, generator);
|
|
65
|
-
|
|
66
|
-
expect(result).toBe("test");
|
|
67
|
-
expect(map.get(null)).toBe("test");
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
it("should handle undefined key", () => {
|
|
71
|
-
const map = new Map<undefined, string>();
|
|
72
|
-
const generator = () => "test";
|
|
73
|
-
|
|
74
|
-
const result = __map_take(map, undefined, generator);
|
|
75
|
-
|
|
76
|
-
expect(result).toBe("test");
|
|
77
|
-
expect(map.get(undefined)).toBe("test");
|
|
78
|
-
});
|
|
79
|
-
|
|
80
|
-
it("should handle empty string key", () => {
|
|
81
|
-
const map = new Map<string, string>();
|
|
82
|
-
const generator = () => "test";
|
|
83
|
-
|
|
84
|
-
const result = __map_take(map, "", generator);
|
|
85
|
-
|
|
86
|
-
expect(result).toBe("test");
|
|
87
|
-
expect(map.get("")).toBe("test");
|
|
88
|
-
});
|
|
89
|
-
});
|
|
90
|
-
|
|
91
|
-
describe("generator function tests", () => {
|
|
92
|
-
it("should not call generator multiple times", () => {
|
|
93
|
-
const map = new Map<string, number>();
|
|
94
|
-
let callCount = 0;
|
|
95
|
-
|
|
96
|
-
const generator = () => {
|
|
97
|
-
callCount++;
|
|
98
|
-
return 42;
|
|
99
|
-
};
|
|
100
|
-
|
|
101
|
-
__map_take(map, "test", generator);
|
|
102
|
-
__map_take(map, "test", generator);
|
|
103
|
-
|
|
104
|
-
expect(callCount).toBe(1);
|
|
105
|
-
});
|
|
106
|
-
|
|
107
|
-
it("should handle generator throwing error", () => {
|
|
108
|
-
const map = new Map<string, number>();
|
|
109
|
-
const generator = () => {
|
|
110
|
-
throw new Error("Generator error");
|
|
111
|
-
};
|
|
112
|
-
|
|
113
|
-
expect(() => __map_take(map, "test", generator)).toThrow("Generator error");
|
|
114
|
-
});
|
|
115
|
-
|
|
116
|
-
it("should handle generator returning undefined", () => {
|
|
117
|
-
const map = new Map<string, undefined>();
|
|
118
|
-
const generator = () => undefined;
|
|
119
|
-
|
|
120
|
-
const result = __map_take(map, "test", generator);
|
|
121
|
-
|
|
122
|
-
expect(result).toBeUndefined();
|
|
123
|
-
expect(map.get("test")).toBeUndefined();
|
|
124
|
-
});
|
|
125
|
-
});
|
|
126
|
-
|
|
127
|
-
describe("concurrency tests", () => {
|
|
128
|
-
it("should handle concurrent access to same key", () => {
|
|
129
|
-
const map = new Map<string, number>();
|
|
130
|
-
const generator = () => 42;
|
|
131
|
-
|
|
132
|
-
const result1 = __map_take(map, "test", generator);
|
|
133
|
-
const result2 = __map_take(map, "test", generator);
|
|
134
|
-
|
|
135
|
-
expect(result1).toBe(42);
|
|
136
|
-
expect(result2).toBe(42);
|
|
137
|
-
expect(map.get("test")).toBe(42);
|
|
138
|
-
});
|
|
139
|
-
});
|
|
140
|
-
});
|
|
1
|
+
import { __map_take } from "./__map_take";
|
|
2
|
+
|
|
3
|
+
describe("__map_take", () => {
|
|
4
|
+
describe("basic functionality", () => {
|
|
5
|
+
it("should generate value for new key", () => {
|
|
6
|
+
const map = new Map<string, number>();
|
|
7
|
+
const generator = () => 42;
|
|
8
|
+
|
|
9
|
+
const result = __map_take(map, "test", generator);
|
|
10
|
+
|
|
11
|
+
expect(result).toBe(42);
|
|
12
|
+
expect(map.get("test")).toBe(42);
|
|
13
|
+
});
|
|
14
|
+
|
|
15
|
+
it("should return existing value for existing key", () => {
|
|
16
|
+
const map = new Map<string, number>();
|
|
17
|
+
map.set("test", 100);
|
|
18
|
+
|
|
19
|
+
const generator = () => 42;
|
|
20
|
+
const result = __map_take(map, "test", generator);
|
|
21
|
+
|
|
22
|
+
expect(result).toBe(100);
|
|
23
|
+
expect(map.get("test")).toBe(100);
|
|
24
|
+
});
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
describe("various type tests", () => {
|
|
28
|
+
it("should handle object type", () => {
|
|
29
|
+
const map = new Map<string, { value: number }>();
|
|
30
|
+
const generator = () => ({ value: 42 });
|
|
31
|
+
|
|
32
|
+
const result = __map_take(map, "test", generator);
|
|
33
|
+
|
|
34
|
+
expect(result).toEqual({ value: 42 });
|
|
35
|
+
expect(map.get("test")).toEqual({ value: 42 });
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
it("should handle array type", () => {
|
|
39
|
+
const map = new Map<string, number[]>();
|
|
40
|
+
const generator = () => [1, 2, 3];
|
|
41
|
+
|
|
42
|
+
const result = __map_take(map, "test", generator);
|
|
43
|
+
|
|
44
|
+
expect(result).toEqual([1, 2, 3]);
|
|
45
|
+
expect(map.get("test")).toEqual([1, 2, 3]);
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
it("should handle function type", () => {
|
|
49
|
+
const map = new Map<string, () => number>();
|
|
50
|
+
const generator = () => () => 42;
|
|
51
|
+
|
|
52
|
+
const result = __map_take(map, "test", generator);
|
|
53
|
+
|
|
54
|
+
expect(result()).toBe(42);
|
|
55
|
+
expect(map.get("test")?.()).toBe(42);
|
|
56
|
+
});
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
describe("edge cases", () => {
|
|
60
|
+
it("should handle null key", () => {
|
|
61
|
+
const map = new Map<null, string>();
|
|
62
|
+
const generator = () => "test";
|
|
63
|
+
|
|
64
|
+
const result = __map_take(map, null, generator);
|
|
65
|
+
|
|
66
|
+
expect(result).toBe("test");
|
|
67
|
+
expect(map.get(null)).toBe("test");
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
it("should handle undefined key", () => {
|
|
71
|
+
const map = new Map<undefined, string>();
|
|
72
|
+
const generator = () => "test";
|
|
73
|
+
|
|
74
|
+
const result = __map_take(map, undefined, generator);
|
|
75
|
+
|
|
76
|
+
expect(result).toBe("test");
|
|
77
|
+
expect(map.get(undefined)).toBe("test");
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
it("should handle empty string key", () => {
|
|
81
|
+
const map = new Map<string, string>();
|
|
82
|
+
const generator = () => "test";
|
|
83
|
+
|
|
84
|
+
const result = __map_take(map, "", generator);
|
|
85
|
+
|
|
86
|
+
expect(result).toBe("test");
|
|
87
|
+
expect(map.get("")).toBe("test");
|
|
88
|
+
});
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
describe("generator function tests", () => {
|
|
92
|
+
it("should not call generator multiple times", () => {
|
|
93
|
+
const map = new Map<string, number>();
|
|
94
|
+
let callCount = 0;
|
|
95
|
+
|
|
96
|
+
const generator = () => {
|
|
97
|
+
callCount++;
|
|
98
|
+
return 42;
|
|
99
|
+
};
|
|
100
|
+
|
|
101
|
+
__map_take(map, "test", generator);
|
|
102
|
+
__map_take(map, "test", generator);
|
|
103
|
+
|
|
104
|
+
expect(callCount).toBe(1);
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
it("should handle generator throwing error", () => {
|
|
108
|
+
const map = new Map<string, number>();
|
|
109
|
+
const generator = () => {
|
|
110
|
+
throw new Error("Generator error");
|
|
111
|
+
};
|
|
112
|
+
|
|
113
|
+
expect(() => __map_take(map, "test", generator)).toThrow("Generator error");
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
it("should handle generator returning undefined", () => {
|
|
117
|
+
const map = new Map<string, undefined>();
|
|
118
|
+
const generator = () => undefined;
|
|
119
|
+
|
|
120
|
+
const result = __map_take(map, "test", generator);
|
|
121
|
+
|
|
122
|
+
expect(result).toBeUndefined();
|
|
123
|
+
expect(map.get("test")).toBeUndefined();
|
|
124
|
+
});
|
|
125
|
+
});
|
|
126
|
+
|
|
127
|
+
describe("concurrency tests", () => {
|
|
128
|
+
it("should handle concurrent access to same key", () => {
|
|
129
|
+
const map = new Map<string, number>();
|
|
130
|
+
const generator = () => 42;
|
|
131
|
+
|
|
132
|
+
const result1 = __map_take(map, "test", generator);
|
|
133
|
+
const result2 = __map_take(map, "test", generator);
|
|
134
|
+
|
|
135
|
+
expect(result1).toBe(42);
|
|
136
|
+
expect(result2).toBe(42);
|
|
137
|
+
expect(map.get("test")).toBe(42);
|
|
138
|
+
});
|
|
139
|
+
});
|
|
140
|
+
});
|
package/src/utils/__map_take.ts
CHANGED
|
@@ -1,13 +1,13 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @internal
|
|
3
|
-
*/
|
|
4
|
-
export function __map_take<Key, T>(dict: Map<Key, T>, key: Key, generator: () => T): T {
|
|
5
|
-
const oldbie: T | undefined = dict.get(key);
|
|
6
|
-
if (oldbie !== undefined) {
|
|
7
|
-
return oldbie;
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
const value: T = generator();
|
|
11
|
-
dict.set(key, value);
|
|
12
|
-
return value;
|
|
13
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* @internal
|
|
3
|
+
*/
|
|
4
|
+
export function __map_take<Key, T>(dict: Map<Key, T>, key: Key, generator: () => T): T {
|
|
5
|
+
const oldbie: T | undefined = dict.get(key);
|
|
6
|
+
if (oldbie !== undefined) {
|
|
7
|
+
return oldbie;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
const value: T = generator();
|
|
11
|
+
dict.set(key, value);
|
|
12
|
+
return value;
|
|
13
|
+
}
|