@agentica/core 0.43.3 → 0.44.0-dev.20260313
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/constants/AgenticaSystemPrompt.js +1 -1
- package/lib/constants/AgenticaSystemPrompt.js.map +1 -1
- package/lib/context/AgenticaOperation.d.ts +3 -2
- package/lib/context/internal/AgenticaOperationComposer.js.map +1 -1
- package/lib/context/internal/AgenticaTokenUsageAggregator.js.map +1 -1
- package/lib/context/internal/__IChatInitialApplication.d.ts +1 -1
- package/lib/errors/AgenticaJsonParseError.d.ts +5 -8
- package/lib/errors/AgenticaJsonParseError.js +11 -6
- package/lib/errors/AgenticaJsonParseError.js.map +1 -1
- package/lib/errors/AgenticaValidationError.d.ts +1 -1
- package/lib/events/AgenticaExecuteEvent.d.ts +1 -1
- package/lib/events/AgenticaJsonParseErrorEvent.d.ts +2 -2
- package/lib/factory/events.d.ts +2 -3
- package/lib/factory/events.js +1 -2
- package/lib/factory/events.js.map +1 -1
- package/lib/functional/assertHttpController.d.ts +2 -3
- package/lib/functional/assertHttpController.js +3939 -2656
- package/lib/functional/assertHttpController.js.map +1 -1
- package/lib/functional/assertHttpLlmApplication.d.ts +3 -3
- package/lib/functional/assertHttpLlmApplication.js +3939 -2656
- package/lib/functional/assertHttpLlmApplication.js.map +1 -1
- package/lib/functional/assertMcpController.d.ts +2 -2
- package/lib/functional/assertMcpController.js +202 -629
- package/lib/functional/assertMcpController.js.map +1 -1
- package/lib/functional/createMcpLlmApplication.d.ts +6 -0
- package/lib/functional/createMcpLlmApplication.js +56 -0
- package/lib/functional/createMcpLlmApplication.js.map +1 -0
- package/lib/functional/validateHttpController.d.ts +3 -3
- package/lib/functional/validateHttpController.js +3367 -2268
- package/lib/functional/validateHttpController.js.map +1 -1
- package/lib/functional/validateHttpLlmApplication.d.ts +3 -4
- package/lib/functional/validateHttpLlmApplication.js +3367 -2268
- package/lib/functional/validateHttpLlmApplication.js.map +1 -1
- package/lib/functional/validateMcpController.d.ts +2 -2
- package/lib/functional/validateMcpController.js +388 -1161
- package/lib/functional/validateMcpController.js.map +1 -1
- package/lib/histories/AgenticaExecuteHistory.d.ts +1 -1
- package/lib/index.mjs +21256 -18946
- package/lib/index.mjs.map +1 -1
- package/lib/orchestrate/call.js +43 -56
- package/lib/orchestrate/call.js.map +1 -1
- package/lib/orchestrate/cancel.js +9 -66
- package/lib/orchestrate/cancel.js.map +1 -1
- package/lib/orchestrate/initialize.js +4 -947
- package/lib/orchestrate/initialize.js.map +1 -1
- package/lib/orchestrate/select.js +11 -68
- package/lib/orchestrate/select.js.map +1 -1
- package/lib/structures/IAgenticaController.d.ts +143 -151
- package/lib/structures/IMcpTool.d.ts +52 -0
- package/lib/structures/IMcpTool.js +3 -0
- package/lib/structures/IMcpTool.js.map +1 -0
- package/lib/utils/ChatGptCompletionMessageUtil.js +16 -5
- package/lib/utils/ChatGptCompletionMessageUtil.js.map +1 -1
- package/lib/utils/ChatGptCompletionMessageUtil.spec.js +0 -5
- package/lib/utils/ChatGptCompletionMessageUtil.spec.js.map +1 -1
- package/package.json +7 -9
- 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 -33
- 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/AgenticaSystemPrompt.ts +1 -1
- 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 -72
- 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 +15 -15
- package/src/context/internal/__IChatSelectFunctionsApplication.ts +24 -24
- package/src/context/internal/isAgenticaContext.ts +11 -11
- package/src/errors/AgenticaJsonParseError.ts +52 -47
- 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 -15
- 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 -359
- 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 -104
- package/src/functional/assertHttpLlmApplication.ts +52 -57
- package/src/functional/assertMcpController.ts +47 -44
- package/src/functional/createMcpLlmApplication.ts +72 -0
- package/src/functional/index.ts +7 -7
- package/src/functional/validateHttpController.ts +113 -110
- package/src/functional/validateHttpLlmApplication.ts +65 -70
- package/src/functional/validateMcpController.ts +53 -50
- 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 -558
- package/src/orchestrate/cancel.ts +265 -269
- 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 -322
- package/src/structures/IAgenticaConfig.ts +83 -83
- package/src/structures/IAgenticaConfigBase.ts +87 -87
- package/src/structures/IAgenticaController.ts +143 -151
- 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 -0
- 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 -320
- package/src/utils/ChatGptCompletionMessageUtil.ts +210 -195
- 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/lib/context/internal/AgenticaOperationComposer.spec.d.ts +0 -1
- package/lib/context/internal/AgenticaOperationComposer.spec.js +0 -353
- package/lib/context/internal/AgenticaOperationComposer.spec.js.map +0 -1
- package/lib/utils/JsonUtil.d.ts +0 -8
- package/lib/utils/JsonUtil.js +0 -349
- package/lib/utils/JsonUtil.js.map +0 -1
- package/src/context/internal/AgenticaOperationComposer.spec.ts +0 -401
- package/src/utils/JsonUtil.ts +0 -460
package/src/utils/MPSC.ts
CHANGED
|
@@ -1,42 +1,42 @@
|
|
|
1
|
-
import { AsyncQueue } from "./AsyncQueue";
|
|
2
|
-
|
|
3
|
-
export class MPSC<T> {
|
|
4
|
-
private readonly queue: AsyncQueue<T>;
|
|
5
|
-
public readonly consumer: ReadableStream<T>;
|
|
6
|
-
|
|
7
|
-
public constructor() {
|
|
8
|
-
this.queue = new AsyncQueue<T>();
|
|
9
|
-
this.consumer = new ReadableStream<T>({
|
|
10
|
-
start: async (controller) => {
|
|
11
|
-
while (true) {
|
|
12
|
-
const { value, done } = await this.queue.dequeue();
|
|
13
|
-
if (done === true) {
|
|
14
|
-
controller.close();
|
|
15
|
-
return;
|
|
16
|
-
}
|
|
17
|
-
controller.enqueue(value);
|
|
18
|
-
}
|
|
19
|
-
},
|
|
20
|
-
});
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
produce(chunk: T) {
|
|
24
|
-
this.queue.enqueue(chunk);
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
close() {
|
|
28
|
-
this.queue.close();
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
done() {
|
|
32
|
-
return this.queue.done();
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
async waitClosed() {
|
|
36
|
-
await this.queue.waitClosed();
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
async waitUntilEmpty() {
|
|
40
|
-
await this.queue.waitUntilEmpty();
|
|
41
|
-
}
|
|
42
|
-
}
|
|
1
|
+
import { AsyncQueue } from "./AsyncQueue";
|
|
2
|
+
|
|
3
|
+
export class MPSC<T> {
|
|
4
|
+
private readonly queue: AsyncQueue<T>;
|
|
5
|
+
public readonly consumer: ReadableStream<T>;
|
|
6
|
+
|
|
7
|
+
public constructor() {
|
|
8
|
+
this.queue = new AsyncQueue<T>();
|
|
9
|
+
this.consumer = new ReadableStream<T>({
|
|
10
|
+
start: async (controller) => {
|
|
11
|
+
while (true) {
|
|
12
|
+
const { value, done } = await this.queue.dequeue();
|
|
13
|
+
if (done === true) {
|
|
14
|
+
controller.close();
|
|
15
|
+
return;
|
|
16
|
+
}
|
|
17
|
+
controller.enqueue(value);
|
|
18
|
+
}
|
|
19
|
+
},
|
|
20
|
+
});
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
produce(chunk: T) {
|
|
24
|
+
this.queue.enqueue(chunk);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
close() {
|
|
28
|
+
this.queue.close();
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
done() {
|
|
32
|
+
return this.queue.done();
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
async waitClosed() {
|
|
36
|
+
await this.queue.waitClosed();
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
async waitUntilEmpty() {
|
|
40
|
+
await this.queue.waitUntilEmpty();
|
|
41
|
+
}
|
|
42
|
+
}
|
|
@@ -1,138 +1,138 @@
|
|
|
1
|
-
import { Singleton } from "./Singleton";
|
|
2
|
-
|
|
3
|
-
describe("singleton", () => {
|
|
4
|
-
describe("basic functionality", () => {
|
|
5
|
-
it("should create instance only once", () => {
|
|
6
|
-
const factory = () => ({ value: 42 });
|
|
7
|
-
const singleton = new Singleton(factory);
|
|
8
|
-
|
|
9
|
-
const instance1 = singleton.get();
|
|
10
|
-
const instance2 = singleton.get();
|
|
11
|
-
|
|
12
|
-
expect(instance1).toBe(instance2);
|
|
13
|
-
expect(instance1.value).toBe(42);
|
|
14
|
-
});
|
|
15
|
-
|
|
16
|
-
it("should create different instances for different singletons", () => {
|
|
17
|
-
const factory1 = () => ({ value: 42 });
|
|
18
|
-
const factory2 = () => ({ value: 24 });
|
|
19
|
-
|
|
20
|
-
const singleton1 = new Singleton(factory1);
|
|
21
|
-
const singleton2 = new Singleton(factory2);
|
|
22
|
-
|
|
23
|
-
const instance1 = singleton1.get();
|
|
24
|
-
const instance2 = singleton2.get();
|
|
25
|
-
|
|
26
|
-
expect(instance1).not.toBe(instance2);
|
|
27
|
-
expect(instance1.value).toBe(42);
|
|
28
|
-
expect(instance2.value).toBe(24);
|
|
29
|
-
});
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
describe("constructor arguments", () => {
|
|
33
|
-
it("should pass constructor arguments to factory", () => {
|
|
34
|
-
const factory = (value: number) => ({ value });
|
|
35
|
-
const singleton = new Singleton(factory);
|
|
36
|
-
|
|
37
|
-
const instance = singleton.get(42);
|
|
38
|
-
expect(instance.value).toBe(42);
|
|
39
|
-
});
|
|
40
|
-
|
|
41
|
-
it("should use same instance with same constructor arguments", () => {
|
|
42
|
-
const factory = (value: number) => ({ value });
|
|
43
|
-
const singleton = new Singleton(factory);
|
|
44
|
-
|
|
45
|
-
const instance1 = singleton.get(42);
|
|
46
|
-
const instance2 = singleton.get(42);
|
|
47
|
-
|
|
48
|
-
expect(instance1).toBe(instance2);
|
|
49
|
-
expect(instance1.value).toBe(42);
|
|
50
|
-
});
|
|
51
|
-
|
|
52
|
-
it("should return same instance even with different constructor arguments", () => {
|
|
53
|
-
const factory = (value: number) => ({ value });
|
|
54
|
-
const singleton = new Singleton(factory);
|
|
55
|
-
|
|
56
|
-
const instance1 = singleton.get(42);
|
|
57
|
-
const instance2 = singleton.get(24);
|
|
58
|
-
|
|
59
|
-
expect(instance1).toBe(instance2);
|
|
60
|
-
expect(instance1.value).toBe(42);
|
|
61
|
-
expect(instance2.value).toBe(42);
|
|
62
|
-
});
|
|
63
|
-
});
|
|
64
|
-
|
|
65
|
-
describe("complex object types", () => {
|
|
66
|
-
it("should handle complex objects", () => {
|
|
67
|
-
interface ComplexObject {
|
|
68
|
-
id: number;
|
|
69
|
-
data: { name: string; value: number };
|
|
70
|
-
timestamp: Date;
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
const factory = (id: number, name: string, value: number) => ({
|
|
74
|
-
id,
|
|
75
|
-
data: { name, value },
|
|
76
|
-
timestamp: new Date(),
|
|
77
|
-
});
|
|
78
|
-
|
|
79
|
-
const singleton = new Singleton<ComplexObject, [number, string, number]>(factory);
|
|
80
|
-
const instance = singleton.get(1, "test", 42);
|
|
81
|
-
|
|
82
|
-
expect(instance.id).toBe(1);
|
|
83
|
-
expect(instance.data.name).toBe("test");
|
|
84
|
-
expect(instance.data.value).toBe(42);
|
|
85
|
-
expect(instance.timestamp).toBeInstanceOf(Date);
|
|
86
|
-
});
|
|
87
|
-
|
|
88
|
-
it("should maintain same complex object instance", () => {
|
|
89
|
-
const factory = () => ({
|
|
90
|
-
data: new Map<string, number>([["key", 42]]),
|
|
91
|
-
array: [1, 2, 3],
|
|
92
|
-
});
|
|
93
|
-
|
|
94
|
-
const singleton = new Singleton(factory);
|
|
95
|
-
const instance1 = singleton.get();
|
|
96
|
-
const instance2 = singleton.get();
|
|
97
|
-
|
|
98
|
-
expect(instance1).toBe(instance2);
|
|
99
|
-
expect(instance1.data).toBe(instance2.data);
|
|
100
|
-
expect(instance1.array).toBe(instance2.array);
|
|
101
|
-
});
|
|
102
|
-
});
|
|
103
|
-
|
|
104
|
-
describe("edge cases", () => {
|
|
105
|
-
it("should handle null factory return", () => {
|
|
106
|
-
const factory = () => null;
|
|
107
|
-
const singleton = new Singleton(factory);
|
|
108
|
-
|
|
109
|
-
const instance = singleton.get();
|
|
110
|
-
expect(instance).toBeNull();
|
|
111
|
-
});
|
|
112
|
-
|
|
113
|
-
it("should handle undefined factory return", () => {
|
|
114
|
-
const factory = () => undefined;
|
|
115
|
-
const singleton = new Singleton(factory);
|
|
116
|
-
|
|
117
|
-
const instance = singleton.get();
|
|
118
|
-
expect(instance).toBeUndefined();
|
|
119
|
-
});
|
|
120
|
-
|
|
121
|
-
it("should handle primitive values", () => {
|
|
122
|
-
const factory = () => 42;
|
|
123
|
-
const singleton = new Singleton(factory);
|
|
124
|
-
|
|
125
|
-
const instance = singleton.get();
|
|
126
|
-
expect(instance).toBe(42);
|
|
127
|
-
});
|
|
128
|
-
|
|
129
|
-
it("should handle factory throwing error", () => {
|
|
130
|
-
const factory = () => {
|
|
131
|
-
throw new Error("Factory error");
|
|
132
|
-
};
|
|
133
|
-
const singleton = new Singleton(factory);
|
|
134
|
-
|
|
135
|
-
expect(() => singleton.get()).toThrow("Factory error");
|
|
136
|
-
});
|
|
137
|
-
});
|
|
138
|
-
});
|
|
1
|
+
import { Singleton } from "./Singleton";
|
|
2
|
+
|
|
3
|
+
describe("singleton", () => {
|
|
4
|
+
describe("basic functionality", () => {
|
|
5
|
+
it("should create instance only once", () => {
|
|
6
|
+
const factory = () => ({ value: 42 });
|
|
7
|
+
const singleton = new Singleton(factory);
|
|
8
|
+
|
|
9
|
+
const instance1 = singleton.get();
|
|
10
|
+
const instance2 = singleton.get();
|
|
11
|
+
|
|
12
|
+
expect(instance1).toBe(instance2);
|
|
13
|
+
expect(instance1.value).toBe(42);
|
|
14
|
+
});
|
|
15
|
+
|
|
16
|
+
it("should create different instances for different singletons", () => {
|
|
17
|
+
const factory1 = () => ({ value: 42 });
|
|
18
|
+
const factory2 = () => ({ value: 24 });
|
|
19
|
+
|
|
20
|
+
const singleton1 = new Singleton(factory1);
|
|
21
|
+
const singleton2 = new Singleton(factory2);
|
|
22
|
+
|
|
23
|
+
const instance1 = singleton1.get();
|
|
24
|
+
const instance2 = singleton2.get();
|
|
25
|
+
|
|
26
|
+
expect(instance1).not.toBe(instance2);
|
|
27
|
+
expect(instance1.value).toBe(42);
|
|
28
|
+
expect(instance2.value).toBe(24);
|
|
29
|
+
});
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
describe("constructor arguments", () => {
|
|
33
|
+
it("should pass constructor arguments to factory", () => {
|
|
34
|
+
const factory = (value: number) => ({ value });
|
|
35
|
+
const singleton = new Singleton(factory);
|
|
36
|
+
|
|
37
|
+
const instance = singleton.get(42);
|
|
38
|
+
expect(instance.value).toBe(42);
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
it("should use same instance with same constructor arguments", () => {
|
|
42
|
+
const factory = (value: number) => ({ value });
|
|
43
|
+
const singleton = new Singleton(factory);
|
|
44
|
+
|
|
45
|
+
const instance1 = singleton.get(42);
|
|
46
|
+
const instance2 = singleton.get(42);
|
|
47
|
+
|
|
48
|
+
expect(instance1).toBe(instance2);
|
|
49
|
+
expect(instance1.value).toBe(42);
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
it("should return same instance even with different constructor arguments", () => {
|
|
53
|
+
const factory = (value: number) => ({ value });
|
|
54
|
+
const singleton = new Singleton(factory);
|
|
55
|
+
|
|
56
|
+
const instance1 = singleton.get(42);
|
|
57
|
+
const instance2 = singleton.get(24);
|
|
58
|
+
|
|
59
|
+
expect(instance1).toBe(instance2);
|
|
60
|
+
expect(instance1.value).toBe(42);
|
|
61
|
+
expect(instance2.value).toBe(42);
|
|
62
|
+
});
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
describe("complex object types", () => {
|
|
66
|
+
it("should handle complex objects", () => {
|
|
67
|
+
interface ComplexObject {
|
|
68
|
+
id: number;
|
|
69
|
+
data: { name: string; value: number };
|
|
70
|
+
timestamp: Date;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
const factory = (id: number, name: string, value: number) => ({
|
|
74
|
+
id,
|
|
75
|
+
data: { name, value },
|
|
76
|
+
timestamp: new Date(),
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
const singleton = new Singleton<ComplexObject, [number, string, number]>(factory);
|
|
80
|
+
const instance = singleton.get(1, "test", 42);
|
|
81
|
+
|
|
82
|
+
expect(instance.id).toBe(1);
|
|
83
|
+
expect(instance.data.name).toBe("test");
|
|
84
|
+
expect(instance.data.value).toBe(42);
|
|
85
|
+
expect(instance.timestamp).toBeInstanceOf(Date);
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
it("should maintain same complex object instance", () => {
|
|
89
|
+
const factory = () => ({
|
|
90
|
+
data: new Map<string, number>([["key", 42]]),
|
|
91
|
+
array: [1, 2, 3],
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
const singleton = new Singleton(factory);
|
|
95
|
+
const instance1 = singleton.get();
|
|
96
|
+
const instance2 = singleton.get();
|
|
97
|
+
|
|
98
|
+
expect(instance1).toBe(instance2);
|
|
99
|
+
expect(instance1.data).toBe(instance2.data);
|
|
100
|
+
expect(instance1.array).toBe(instance2.array);
|
|
101
|
+
});
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
describe("edge cases", () => {
|
|
105
|
+
it("should handle null factory return", () => {
|
|
106
|
+
const factory = () => null;
|
|
107
|
+
const singleton = new Singleton(factory);
|
|
108
|
+
|
|
109
|
+
const instance = singleton.get();
|
|
110
|
+
expect(instance).toBeNull();
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
it("should handle undefined factory return", () => {
|
|
114
|
+
const factory = () => undefined;
|
|
115
|
+
const singleton = new Singleton(factory);
|
|
116
|
+
|
|
117
|
+
const instance = singleton.get();
|
|
118
|
+
expect(instance).toBeUndefined();
|
|
119
|
+
});
|
|
120
|
+
|
|
121
|
+
it("should handle primitive values", () => {
|
|
122
|
+
const factory = () => 42;
|
|
123
|
+
const singleton = new Singleton(factory);
|
|
124
|
+
|
|
125
|
+
const instance = singleton.get();
|
|
126
|
+
expect(instance).toBe(42);
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
it("should handle factory throwing error", () => {
|
|
130
|
+
const factory = () => {
|
|
131
|
+
throw new Error("Factory error");
|
|
132
|
+
};
|
|
133
|
+
const singleton = new Singleton(factory);
|
|
134
|
+
|
|
135
|
+
expect(() => singleton.get()).toThrow("Factory error");
|
|
136
|
+
});
|
|
137
|
+
});
|
|
138
|
+
});
|
package/src/utils/Singleton.ts
CHANGED
|
@@ -1,42 +1,42 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @internal
|
|
3
|
-
*/
|
|
4
|
-
const NOT_MOUNTED_YET = {};
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* @internal
|
|
8
|
-
*
|
|
9
|
-
* @description
|
|
10
|
-
* A singleton class that creates a single instance of a class.
|
|
11
|
-
*
|
|
12
|
-
* @example
|
|
13
|
-
* ```ts
|
|
14
|
-
* const singleton = new Singleton((name: string) => new SomeClass(name));
|
|
15
|
-
* const instance = singleton.get("test");
|
|
16
|
-
* ```
|
|
17
|
-
*
|
|
18
|
-
* but next case is not work
|
|
19
|
-
* ```ts
|
|
20
|
-
* const singleton = new Singleton((name: string) => new SomeClass(name));
|
|
21
|
-
* const instance = singleton.get("test");
|
|
22
|
-
* const instance2 = singleton.get("test2");
|
|
23
|
-
*
|
|
24
|
-
* expect(instance).toBe(instance2); // true
|
|
25
|
-
* ```
|
|
26
|
-
*/
|
|
27
|
-
export class Singleton<T, Args extends any[] = []> {
|
|
28
|
-
private readonly closure_: (...args: Args) => T;
|
|
29
|
-
private value_: T | object;
|
|
30
|
-
|
|
31
|
-
public constructor(closure: (...args: Args) => T) {
|
|
32
|
-
this.closure_ = closure;
|
|
33
|
-
this.value_ = NOT_MOUNTED_YET;
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
public get(...args: Args): T {
|
|
37
|
-
if (this.value_ === NOT_MOUNTED_YET) {
|
|
38
|
-
this.value_ = this.closure_(...args);
|
|
39
|
-
}
|
|
40
|
-
return this.value_ as T;
|
|
41
|
-
}
|
|
42
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* @internal
|
|
3
|
+
*/
|
|
4
|
+
const NOT_MOUNTED_YET = {};
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* @internal
|
|
8
|
+
*
|
|
9
|
+
* @description
|
|
10
|
+
* A singleton class that creates a single instance of a class.
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```ts
|
|
14
|
+
* const singleton = new Singleton((name: string) => new SomeClass(name));
|
|
15
|
+
* const instance = singleton.get("test");
|
|
16
|
+
* ```
|
|
17
|
+
*
|
|
18
|
+
* but next case is not work
|
|
19
|
+
* ```ts
|
|
20
|
+
* const singleton = new Singleton((name: string) => new SomeClass(name));
|
|
21
|
+
* const instance = singleton.get("test");
|
|
22
|
+
* const instance2 = singleton.get("test2");
|
|
23
|
+
*
|
|
24
|
+
* expect(instance).toBe(instance2); // true
|
|
25
|
+
* ```
|
|
26
|
+
*/
|
|
27
|
+
export class Singleton<T, Args extends any[] = []> {
|
|
28
|
+
private readonly closure_: (...args: Args) => T;
|
|
29
|
+
private value_: T | object;
|
|
30
|
+
|
|
31
|
+
public constructor(closure: (...args: Args) => T) {
|
|
32
|
+
this.closure_ = closure;
|
|
33
|
+
this.value_ = NOT_MOUNTED_YET;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
public get(...args: Args): T {
|
|
37
|
+
if (this.value_ === NOT_MOUNTED_YET) {
|
|
38
|
+
this.value_ = this.closure_(...args);
|
|
39
|
+
}
|
|
40
|
+
return this.value_ as T;
|
|
41
|
+
}
|
|
42
|
+
}
|