@smythos/sre 1.6.8 → 1.6.10
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/CHANGELOG +111 -111
- package/LICENSE +18 -18
- package/README.md +135 -135
- package/dist/bundle-analysis-lazy.html +4949 -0
- package/dist/bundle-analysis.html +4949 -0
- package/dist/index.js +2 -2
- package/dist/index.js.map +1 -1
- package/dist/types/Components/Triggers/Gmail.trigger.d.ts +58 -0
- package/dist/types/Components/Triggers/GmailTrigger.class.d.ts +44 -0
- package/dist/types/Components/Triggers/Trigger.class.d.ts +21 -0
- package/dist/types/Components/Triggers/WhatsApp.trigger.d.ts +22 -0
- package/dist/types/helpers/AIPerformanceAnalyzer.helper.d.ts +45 -0
- package/dist/types/helpers/AIPerformanceCollector.helper.d.ts +111 -0
- package/dist/types/subsystems/IO/Storage.service/connectors/AzureBlobStorage.class.d.ts +211 -0
- package/dist/types/subsystems/IO/VectorDB.service/connectors/WeaviateVectorDB.class.d.ts +187 -0
- package/dist/types/subsystems/PerformanceManager/Performance.service/PerformanceConnector.d.ts +102 -0
- package/dist/types/subsystems/PerformanceManager/Performance.service/connectors/LocalPerformanceConnector.class.d.ts +100 -0
- package/dist/types/subsystems/PerformanceManager/Performance.service/index.d.ts +22 -0
- package/dist/types/subsystems/Security/Credentials/Credentials.class.d.ts +2 -0
- package/dist/types/subsystems/Security/Credentials/ManagedOAuth2Credentials.class.d.ts +18 -0
- package/dist/types/subsystems/Security/Credentials/OAuth2Credentials.class.d.ts +14 -0
- package/dist/types/types/Performance.types.d.ts +468 -0
- package/dist/types/utils/package-manager.utils.d.ts +26 -0
- package/package.json +1 -1
- package/src/Components/APICall/APICall.class.ts +161 -161
- package/src/Components/APICall/AccessTokenManager.ts +166 -166
- package/src/Components/APICall/ArrayBufferResponse.helper.ts +58 -58
- package/src/Components/APICall/OAuth.helper.ts +447 -447
- package/src/Components/APICall/mimeTypeCategories.ts +46 -46
- package/src/Components/APICall/parseData.ts +167 -167
- package/src/Components/APICall/parseHeaders.ts +41 -41
- package/src/Components/APICall/parseProxy.ts +68 -68
- package/src/Components/APICall/parseUrl.ts +91 -91
- package/src/Components/APIEndpoint.class.ts +234 -234
- package/src/Components/APIOutput.class.ts +58 -58
- package/src/Components/AgentPlugin.class.ts +102 -102
- package/src/Components/Async.class.ts +155 -155
- package/src/Components/Await.class.ts +90 -90
- package/src/Components/Classifier.class.ts +158 -158
- package/src/Components/Component.class.ts +147 -147
- package/src/Components/ComponentHost.class.ts +38 -38
- package/src/Components/DataSourceCleaner.class.ts +92 -92
- package/src/Components/DataSourceIndexer.class.ts +181 -181
- package/src/Components/DataSourceLookup.class.ts +161 -161
- package/src/Components/ECMASandbox.class.ts +72 -72
- package/src/Components/FEncDec.class.ts +29 -29
- package/src/Components/FHash.class.ts +33 -33
- package/src/Components/FSign.class.ts +80 -80
- package/src/Components/FSleep.class.ts +25 -25
- package/src/Components/FTimestamp.class.ts +66 -66
- package/src/Components/FileStore.class.ts +78 -78
- package/src/Components/ForEach.class.ts +97 -97
- package/src/Components/GPTPlugin.class.ts +70 -70
- package/src/Components/GenAILLM.class.ts +586 -586
- package/src/Components/HuggingFace.class.ts +313 -313
- package/src/Components/Image/imageSettings.config.ts +70 -70
- package/src/Components/ImageGenerator.class.ts +483 -483
- package/src/Components/JSONFilter.class.ts +54 -54
- package/src/Components/LLMAssistant.class.ts +213 -213
- package/src/Components/LogicAND.class.ts +28 -28
- package/src/Components/LogicAtLeast.class.ts +85 -85
- package/src/Components/LogicAtMost.class.ts +86 -86
- package/src/Components/LogicOR.class.ts +29 -29
- package/src/Components/LogicXOR.class.ts +34 -34
- package/src/Components/MCPClient.class.ts +137 -137
- package/src/Components/MemoryDeleteKeyVal.class.ts +70 -70
- package/src/Components/MemoryReadKeyVal.class.ts +67 -67
- package/src/Components/MemoryWriteKeyVal.class.ts +62 -62
- package/src/Components/MemoryWriteObject.class.ts +97 -97
- package/src/Components/MultimodalLLM.class.ts +128 -128
- package/src/Components/OpenAPI.class.ts +72 -72
- package/src/Components/PromptGenerator.class.ts +122 -122
- package/src/Components/ScrapflyWebScrape.class.ts +183 -183
- package/src/Components/ServerlessCode.class.ts +123 -123
- package/src/Components/TavilyWebSearch.class.ts +103 -103
- package/src/Components/VisionLLM.class.ts +104 -104
- package/src/Components/ZapierAction.class.ts +127 -127
- package/src/Components/index.ts +97 -97
- package/src/Core/AgentProcess.helper.ts +240 -240
- package/src/Core/Connector.class.ts +123 -123
- package/src/Core/ConnectorsService.ts +197 -197
- package/src/Core/DummyConnector.ts +49 -49
- package/src/Core/HookService.ts +105 -105
- package/src/Core/SmythRuntime.class.ts +241 -241
- package/src/Core/SystemEvents.ts +16 -16
- package/src/Core/boot.ts +56 -56
- package/src/config.ts +15 -15
- package/src/constants.ts +126 -126
- package/src/data/hugging-face.params.json +579 -579
- package/src/helpers/AWSLambdaCode.helper.ts +624 -624
- package/src/helpers/BinaryInput.helper.ts +331 -331
- package/src/helpers/Conversation.helper.ts +1157 -1157
- package/src/helpers/ECMASandbox.helper.ts +64 -64
- package/src/helpers/JsonContent.helper.ts +97 -97
- package/src/helpers/LocalCache.helper.ts +97 -97
- package/src/helpers/Log.helper.ts +274 -274
- package/src/helpers/OpenApiParser.helper.ts +150 -150
- package/src/helpers/S3Cache.helper.ts +147 -147
- package/src/helpers/SmythURI.helper.ts +5 -5
- package/src/helpers/Sysconfig.helper.ts +95 -95
- package/src/helpers/TemplateString.helper.ts +243 -243
- package/src/helpers/TypeChecker.helper.ts +329 -329
- package/src/index.ts +3 -3
- package/src/index.ts.bak +3 -3
- package/src/subsystems/AgentManager/Agent.class.ts +1114 -1114
- package/src/subsystems/AgentManager/Agent.helper.ts +3 -3
- package/src/subsystems/AgentManager/AgentData.service/AgentDataConnector.ts +230 -230
- package/src/subsystems/AgentManager/AgentData.service/connectors/CLIAgentDataConnector.class.ts +66 -66
- package/src/subsystems/AgentManager/AgentData.service/connectors/LocalAgentDataConnector.class.ts +145 -145
- package/src/subsystems/AgentManager/AgentData.service/connectors/NullAgentData.class.ts +39 -39
- package/src/subsystems/AgentManager/AgentData.service/index.ts +18 -18
- package/src/subsystems/AgentManager/AgentLogger.class.ts +301 -301
- package/src/subsystems/AgentManager/AgentRequest.class.ts +51 -51
- package/src/subsystems/AgentManager/AgentRuntime.class.ts +557 -557
- package/src/subsystems/AgentManager/AgentSSE.class.ts +101 -101
- package/src/subsystems/AgentManager/AgentSettings.class.ts +52 -52
- package/src/subsystems/AgentManager/Component.service/ComponentConnector.ts +32 -32
- package/src/subsystems/AgentManager/Component.service/connectors/LocalComponentConnector.class.ts +60 -60
- package/src/subsystems/AgentManager/Component.service/index.ts +11 -11
- package/src/subsystems/AgentManager/EmbodimentSettings.class.ts +47 -47
- package/src/subsystems/AgentManager/ForkedAgent.class.ts +154 -154
- package/src/subsystems/AgentManager/OSResourceMonitor.ts +77 -77
- package/src/subsystems/ComputeManager/Code.service/CodeConnector.ts +98 -98
- package/src/subsystems/ComputeManager/Code.service/connectors/AWSLambdaCode.class.ts +171 -171
- package/src/subsystems/ComputeManager/Code.service/connectors/ECMASandbox.class.ts +131 -131
- package/src/subsystems/ComputeManager/Code.service/index.ts +13 -13
- package/src/subsystems/IO/CLI.service/CLIConnector.ts +47 -47
- package/src/subsystems/IO/CLI.service/index.ts +9 -9
- package/src/subsystems/IO/Log.service/LogConnector.ts +32 -32
- package/src/subsystems/IO/Log.service/connectors/ConsoleLog.class.ts +28 -28
- package/src/subsystems/IO/Log.service/index.ts +13 -13
- package/src/subsystems/IO/NKV.service/NKVConnector.ts +43 -43
- package/src/subsystems/IO/NKV.service/connectors/NKVLocalStorage.class.ts +234 -234
- package/src/subsystems/IO/NKV.service/connectors/NKVRAM.class.ts +204 -204
- package/src/subsystems/IO/NKV.service/connectors/NKVRedis.class.ts +182 -182
- package/src/subsystems/IO/NKV.service/index.ts +14 -14
- package/src/subsystems/IO/Router.service/RouterConnector.ts +21 -21
- package/src/subsystems/IO/Router.service/connectors/ExpressRouter.class.ts +48 -48
- package/src/subsystems/IO/Router.service/connectors/NullRouter.class.ts +40 -40
- package/src/subsystems/IO/Router.service/index.ts +11 -11
- package/src/subsystems/IO/Storage.service/SmythFS.class.ts +488 -488
- package/src/subsystems/IO/Storage.service/StorageConnector.ts +66 -66
- package/src/subsystems/IO/Storage.service/connectors/LocalStorage.class.ts +327 -327
- package/src/subsystems/IO/Storage.service/connectors/S3Storage.class.ts +482 -482
- package/src/subsystems/IO/Storage.service/index.ts +13 -13
- package/src/subsystems/IO/VectorDB.service/VectorDBConnector.ts +108 -108
- package/src/subsystems/IO/VectorDB.service/connectors/MilvusVectorDB.class.ts +465 -465
- package/src/subsystems/IO/VectorDB.service/connectors/PineconeVectorDB.class.ts +387 -387
- package/src/subsystems/IO/VectorDB.service/connectors/RAMVecrtorDB.class.ts +408 -408
- package/src/subsystems/IO/VectorDB.service/embed/BaseEmbedding.ts +107 -107
- package/src/subsystems/IO/VectorDB.service/embed/GoogleEmbedding.ts +118 -118
- package/src/subsystems/IO/VectorDB.service/embed/OpenAIEmbedding.ts +109 -109
- package/src/subsystems/IO/VectorDB.service/embed/index.ts +26 -26
- package/src/subsystems/IO/VectorDB.service/index.ts +14 -14
- package/src/subsystems/LLMManager/LLM.helper.ts +251 -251
- package/src/subsystems/LLMManager/LLM.inference.ts +345 -345
- package/src/subsystems/LLMManager/LLM.service/LLMConnector.ts +492 -492
- package/src/subsystems/LLMManager/LLM.service/LLMCredentials.helper.ts +171 -171
- package/src/subsystems/LLMManager/LLM.service/connectors/Anthropic.class.ts +666 -666
- package/src/subsystems/LLMManager/LLM.service/connectors/Bedrock.class.ts +407 -407
- package/src/subsystems/LLMManager/LLM.service/connectors/Echo.class.ts +92 -92
- package/src/subsystems/LLMManager/LLM.service/connectors/GoogleAI.class.ts +983 -983
- package/src/subsystems/LLMManager/LLM.service/connectors/Groq.class.ts +319 -319
- package/src/subsystems/LLMManager/LLM.service/connectors/Ollama.class.ts +361 -361
- package/src/subsystems/LLMManager/LLM.service/connectors/Perplexity.class.ts +257 -257
- package/src/subsystems/LLMManager/LLM.service/connectors/VertexAI.class.ts +430 -430
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/OpenAIConnector.class.ts +503 -503
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/ChatCompletionsApiInterface.ts +524 -524
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/OpenAIApiInterface.ts +100 -100
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/OpenAIApiInterfaceFactory.ts +81 -81
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/ResponsesApiInterface.ts +1145 -1145
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/constants.ts +13 -13
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/index.ts +4 -4
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/utils.ts +11 -11
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/types.ts +32 -32
- package/src/subsystems/LLMManager/LLM.service/connectors/xAI.class.ts +478 -478
- package/src/subsystems/LLMManager/LLM.service/index.ts +47 -47
- package/src/subsystems/LLMManager/ModelsProvider.service/ModelsProviderConnector.ts +303 -303
- package/src/subsystems/LLMManager/ModelsProvider.service/connectors/JSONModelsProvider.class.ts +280 -271
- package/src/subsystems/LLMManager/ModelsProvider.service/index.ts +11 -11
- package/src/subsystems/LLMManager/custom-models.ts +854 -854
- package/src/subsystems/LLMManager/models.ts +2540 -2540
- package/src/subsystems/LLMManager/paramMappings.ts +69 -69
- package/src/subsystems/MemoryManager/Cache.service/CacheConnector.ts +86 -86
- package/src/subsystems/MemoryManager/Cache.service/connectors/LocalStorageCache.class.ts +297 -297
- package/src/subsystems/MemoryManager/Cache.service/connectors/RAMCache.class.ts +214 -214
- package/src/subsystems/MemoryManager/Cache.service/connectors/RedisCache.class.ts +252 -252
- package/src/subsystems/MemoryManager/Cache.service/connectors/S3Cache.class.ts +373 -373
- package/src/subsystems/MemoryManager/Cache.service/index.ts +15 -15
- package/src/subsystems/MemoryManager/LLMCache.ts +72 -72
- package/src/subsystems/MemoryManager/LLMContext.ts +124 -124
- package/src/subsystems/MemoryManager/LLMMemory.service/LLMMemoryConnector.ts +26 -26
- package/src/subsystems/MemoryManager/RuntimeContext.ts +277 -277
- package/src/subsystems/Security/AccessControl/ACL.class.ts +208 -208
- package/src/subsystems/Security/AccessControl/AccessCandidate.class.ts +82 -82
- package/src/subsystems/Security/AccessControl/AccessRequest.class.ts +52 -52
- package/src/subsystems/Security/Account.service/AccountConnector.ts +44 -44
- package/src/subsystems/Security/Account.service/connectors/DummyAccount.class.ts +130 -130
- package/src/subsystems/Security/Account.service/connectors/JSONFileAccount.class.ts +170 -170
- package/src/subsystems/Security/Account.service/connectors/MySQLAccount.class.ts +76 -76
- package/src/subsystems/Security/Account.service/index.ts +14 -14
- package/src/subsystems/Security/Credentials.helper.ts +62 -62
- package/src/subsystems/Security/ManagedVault.service/ManagedVaultConnector.ts +38 -38
- package/src/subsystems/Security/ManagedVault.service/connectors/NullManagedVault.class.ts +53 -53
- package/src/subsystems/Security/ManagedVault.service/connectors/SecretManagerManagedVault.ts +154 -154
- package/src/subsystems/Security/ManagedVault.service/index.ts +12 -12
- package/src/subsystems/Security/SecureConnector.class.ts +110 -110
- package/src/subsystems/Security/Vault.service/Vault.helper.ts +30 -30
- package/src/subsystems/Security/Vault.service/VaultConnector.ts +29 -29
- package/src/subsystems/Security/Vault.service/connectors/HashicorpVault.class.ts +46 -46
- package/src/subsystems/Security/Vault.service/connectors/JSONFileVault.class.ts +221 -221
- package/src/subsystems/Security/Vault.service/connectors/NullVault.class.ts +54 -54
- package/src/subsystems/Security/Vault.service/connectors/SecretsManager.class.ts +140 -140
- package/src/subsystems/Security/Vault.service/index.ts +12 -12
- package/src/types/ACL.types.ts +104 -104
- package/src/types/AWS.types.ts +10 -10
- package/src/types/Agent.types.ts +61 -61
- package/src/types/AgentLogger.types.ts +17 -17
- package/src/types/Cache.types.ts +1 -1
- package/src/types/Common.types.ts +2 -2
- package/src/types/LLM.types.ts +520 -520
- package/src/types/Redis.types.ts +8 -8
- package/src/types/SRE.types.ts +64 -64
- package/src/types/Security.types.ts +14 -14
- package/src/types/Storage.types.ts +5 -5
- package/src/types/VectorDB.types.ts +86 -86
- package/src/utils/base64.utils.ts +275 -275
- package/src/utils/cli.utils.ts +68 -68
- package/src/utils/data.utils.ts +322 -322
- package/src/utils/date-time.utils.ts +22 -22
- package/src/utils/general.utils.ts +238 -238
- package/src/utils/index.ts +12 -12
- package/src/utils/lazy-client.ts +261 -261
- package/src/utils/numbers.utils.ts +13 -13
- package/src/utils/oauth.utils.ts +35 -35
- package/src/utils/string.utils.ts +414 -414
- package/src/utils/url.utils.ts +19 -19
- package/src/utils/validation.utils.ts +74 -74
- package/dist/types/subsystems/LLMManager/ModelsProvider.service/connectors/SmythModelsProvider.class.d.ts +0 -39
|
@@ -1,38 +1,38 @@
|
|
|
1
|
-
import { ACL } from '@sre/Security/AccessControl/ACL.class';
|
|
2
|
-
import { AccessCandidate } from '@sre/Security/AccessControl/AccessCandidate.class';
|
|
3
|
-
import { AccessRequest } from '@sre/Security/AccessControl/AccessRequest.class';
|
|
4
|
-
import { SecureConnector } from '@sre/Security/SecureConnector.class';
|
|
5
|
-
import { IAccessCandidate, IACL } from '@sre/types/ACL.types';
|
|
6
|
-
|
|
7
|
-
/**
|
|
8
|
-
* The managed vault is a vault that is managed by the SRE, its keys are not visible to the user.
|
|
9
|
-
* it's used to store generated tokens at runtime, like OAuth tokens
|
|
10
|
-
*/
|
|
11
|
-
|
|
12
|
-
export interface IManagedVaultRequest {
|
|
13
|
-
get(keyId: string): Promise<string>;
|
|
14
|
-
set(keyId: string, value: string): Promise<void>;
|
|
15
|
-
delete(keyId: string): Promise<void>;
|
|
16
|
-
exists(keyId: string): Promise<boolean>;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
export abstract class ManagedVaultConnector extends SecureConnector {
|
|
20
|
-
constructor(protected _settings?: any) {
|
|
21
|
-
super(_settings);
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
requester(candidate: AccessCandidate): IManagedVaultRequest {
|
|
25
|
-
return {
|
|
26
|
-
get: async (keyId: string) => this.get(candidate.readRequest, keyId),
|
|
27
|
-
set: async (keyId: string, value: string) => this.set(candidate.writeRequest, keyId, value),
|
|
28
|
-
delete: async (keyId: string) => this.delete(candidate.writeRequest, keyId),
|
|
29
|
-
exists: async (keyId: string) => this.exists(candidate.readRequest, keyId),
|
|
30
|
-
};
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
public abstract getResourceACL(resourceId: string, candidate: IAccessCandidate): Promise<ACL>;
|
|
34
|
-
protected abstract get(acRequest: AccessRequest, keyId: string): Promise<string>;
|
|
35
|
-
protected abstract set(acRequest: AccessRequest, keyId: string, value: string): Promise<void>;
|
|
36
|
-
protected abstract delete(acRequest: AccessRequest, keyId: string): Promise<void>;
|
|
37
|
-
protected abstract exists(acRequest: AccessRequest, keyId: string): Promise<boolean>;
|
|
38
|
-
}
|
|
1
|
+
import { ACL } from '@sre/Security/AccessControl/ACL.class';
|
|
2
|
+
import { AccessCandidate } from '@sre/Security/AccessControl/AccessCandidate.class';
|
|
3
|
+
import { AccessRequest } from '@sre/Security/AccessControl/AccessRequest.class';
|
|
4
|
+
import { SecureConnector } from '@sre/Security/SecureConnector.class';
|
|
5
|
+
import { IAccessCandidate, IACL } from '@sre/types/ACL.types';
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* The managed vault is a vault that is managed by the SRE, its keys are not visible to the user.
|
|
9
|
+
* it's used to store generated tokens at runtime, like OAuth tokens
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
export interface IManagedVaultRequest {
|
|
13
|
+
get(keyId: string): Promise<string>;
|
|
14
|
+
set(keyId: string, value: string): Promise<void>;
|
|
15
|
+
delete(keyId: string): Promise<void>;
|
|
16
|
+
exists(keyId: string): Promise<boolean>;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export abstract class ManagedVaultConnector extends SecureConnector {
|
|
20
|
+
constructor(protected _settings?: any) {
|
|
21
|
+
super(_settings);
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
requester(candidate: AccessCandidate): IManagedVaultRequest {
|
|
25
|
+
return {
|
|
26
|
+
get: async (keyId: string) => this.get(candidate.readRequest, keyId),
|
|
27
|
+
set: async (keyId: string, value: string) => this.set(candidate.writeRequest, keyId, value),
|
|
28
|
+
delete: async (keyId: string) => this.delete(candidate.writeRequest, keyId),
|
|
29
|
+
exists: async (keyId: string) => this.exists(candidate.readRequest, keyId),
|
|
30
|
+
};
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
public abstract getResourceACL(resourceId: string, candidate: IAccessCandidate): Promise<ACL>;
|
|
34
|
+
protected abstract get(acRequest: AccessRequest, keyId: string): Promise<string>;
|
|
35
|
+
protected abstract set(acRequest: AccessRequest, keyId: string, value: string): Promise<void>;
|
|
36
|
+
protected abstract delete(acRequest: AccessRequest, keyId: string): Promise<void>;
|
|
37
|
+
protected abstract exists(acRequest: AccessRequest, keyId: string): Promise<boolean>;
|
|
38
|
+
}
|
|
@@ -1,53 +1,53 @@
|
|
|
1
|
-
import { ConnectorService } from '@sre/Core/ConnectorsService';
|
|
2
|
-
import { Logger } from '@sre/helpers/Log.helper';
|
|
3
|
-
//import { SmythRuntime } from '@sre/Core/SmythRuntime.class';
|
|
4
|
-
import { AccessRequest } from '@sre/Security/AccessControl/AccessRequest.class';
|
|
5
|
-
import { ACL } from '@sre/Security/AccessControl/ACL.class';
|
|
6
|
-
import { SecureConnector } from '@sre/Security/SecureConnector.class';
|
|
7
|
-
import { IAccessCandidate, TAccessLevel } from '@sre/types/ACL.types';
|
|
8
|
-
|
|
9
|
-
import { ManagedVaultConnector } from '../ManagedVaultConnector';
|
|
10
|
-
|
|
11
|
-
const console = Logger('NullManagedVault');
|
|
12
|
-
export class NullManagedVault extends ManagedVaultConnector {
|
|
13
|
-
public name: string = 'NullManagedVault';
|
|
14
|
-
|
|
15
|
-
constructor(protected _settings: any) {
|
|
16
|
-
super(_settings);
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
@SecureConnector.AccessControl
|
|
20
|
-
protected async get(acRequest: AccessRequest, keyId: string) {
|
|
21
|
-
console.debug(`Ignored operation:NullManagedVault.get: ${keyId}`);
|
|
22
|
-
return undefined;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
@SecureConnector.AccessControl
|
|
26
|
-
protected async set(acRequest: AccessRequest, keyId: string, value: string) {
|
|
27
|
-
console.debug(`Ignored operation:NullManagedVault.set: ${keyId} = ${value}`);
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
@SecureConnector.AccessControl
|
|
31
|
-
protected async delete(acRequest: AccessRequest, keyId: string) {
|
|
32
|
-
console.debug(`Ignored operation:NullManagedVault.delete: ${keyId}`);
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
@SecureConnector.AccessControl
|
|
36
|
-
protected async exists(acRequest: AccessRequest, keyId: string) {
|
|
37
|
-
console.debug(`Ignored operation:NullManagedVault.exists: ${keyId}`);
|
|
38
|
-
return false;
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
public async getResourceACL(resourceId: string, candidate: IAccessCandidate) {
|
|
42
|
-
const accountConnector = ConnectorService.getAccountConnector();
|
|
43
|
-
const teamId = await accountConnector.getCandidateTeam(candidate);
|
|
44
|
-
|
|
45
|
-
const acl = new ACL();
|
|
46
|
-
|
|
47
|
-
//give just read access by default
|
|
48
|
-
//Cannot write to null vault
|
|
49
|
-
acl.addAccess(candidate.role, candidate.id, TAccessLevel.Read);
|
|
50
|
-
|
|
51
|
-
return acl;
|
|
52
|
-
}
|
|
53
|
-
}
|
|
1
|
+
import { ConnectorService } from '@sre/Core/ConnectorsService';
|
|
2
|
+
import { Logger } from '@sre/helpers/Log.helper';
|
|
3
|
+
//import { SmythRuntime } from '@sre/Core/SmythRuntime.class';
|
|
4
|
+
import { AccessRequest } from '@sre/Security/AccessControl/AccessRequest.class';
|
|
5
|
+
import { ACL } from '@sre/Security/AccessControl/ACL.class';
|
|
6
|
+
import { SecureConnector } from '@sre/Security/SecureConnector.class';
|
|
7
|
+
import { IAccessCandidate, TAccessLevel } from '@sre/types/ACL.types';
|
|
8
|
+
|
|
9
|
+
import { ManagedVaultConnector } from '../ManagedVaultConnector';
|
|
10
|
+
|
|
11
|
+
const console = Logger('NullManagedVault');
|
|
12
|
+
export class NullManagedVault extends ManagedVaultConnector {
|
|
13
|
+
public name: string = 'NullManagedVault';
|
|
14
|
+
|
|
15
|
+
constructor(protected _settings: any) {
|
|
16
|
+
super(_settings);
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
@SecureConnector.AccessControl
|
|
20
|
+
protected async get(acRequest: AccessRequest, keyId: string) {
|
|
21
|
+
console.debug(`Ignored operation:NullManagedVault.get: ${keyId}`);
|
|
22
|
+
return undefined;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
@SecureConnector.AccessControl
|
|
26
|
+
protected async set(acRequest: AccessRequest, keyId: string, value: string) {
|
|
27
|
+
console.debug(`Ignored operation:NullManagedVault.set: ${keyId} = ${value}`);
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
@SecureConnector.AccessControl
|
|
31
|
+
protected async delete(acRequest: AccessRequest, keyId: string) {
|
|
32
|
+
console.debug(`Ignored operation:NullManagedVault.delete: ${keyId}`);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
@SecureConnector.AccessControl
|
|
36
|
+
protected async exists(acRequest: AccessRequest, keyId: string) {
|
|
37
|
+
console.debug(`Ignored operation:NullManagedVault.exists: ${keyId}`);
|
|
38
|
+
return false;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
public async getResourceACL(resourceId: string, candidate: IAccessCandidate) {
|
|
42
|
+
const accountConnector = ConnectorService.getAccountConnector();
|
|
43
|
+
const teamId = await accountConnector.getCandidateTeam(candidate);
|
|
44
|
+
|
|
45
|
+
const acl = new ACL();
|
|
46
|
+
|
|
47
|
+
//give just read access by default
|
|
48
|
+
//Cannot write to null vault
|
|
49
|
+
acl.addAccess(candidate.role, candidate.id, TAccessLevel.Read);
|
|
50
|
+
|
|
51
|
+
return acl;
|
|
52
|
+
}
|
|
53
|
+
}
|
package/src/subsystems/Security/ManagedVault.service/connectors/SecretManagerManagedVault.ts
CHANGED
|
@@ -1,154 +1,154 @@
|
|
|
1
|
-
import { ConnectorService } from '@sre/Core/ConnectorsService';
|
|
2
|
-
import { Logger } from '@sre/helpers/Log.helper';
|
|
3
|
-
//import { SmythRuntime } from '@sre/Core/SmythRuntime.class';
|
|
4
|
-
import { AccessRequest } from '@sre/Security/AccessControl/AccessRequest.class';
|
|
5
|
-
import { ACL } from '@sre/Security/AccessControl/ACL.class';
|
|
6
|
-
import { SecureConnector } from '@sre/Security/SecureConnector.class';
|
|
7
|
-
import { IAccessCandidate, TAccessLevel, TAccessRole } from '@sre/types/ACL.types';
|
|
8
|
-
|
|
9
|
-
import {
|
|
10
|
-
CreateSecretCommand,
|
|
11
|
-
DeleteSecretCommand,
|
|
12
|
-
GetSecretValueCommand,
|
|
13
|
-
GetSecretValueCommandOutput,
|
|
14
|
-
ListSecretsCommand,
|
|
15
|
-
ListSecretsCommandOutput,
|
|
16
|
-
PutSecretValueCommand,
|
|
17
|
-
SecretsManagerClient,
|
|
18
|
-
} from '@aws-sdk/client-secrets-manager';
|
|
19
|
-
import { randomUUID } from 'crypto';
|
|
20
|
-
import { ManagedVaultConnector } from '../ManagedVaultConnector';
|
|
21
|
-
import { SecretsManagerConfig } from '../../Vault.service/connectors/SecretsManager.class';
|
|
22
|
-
|
|
23
|
-
const console = Logger('SecretManagerManagedVault');
|
|
24
|
-
|
|
25
|
-
export class SecretManagerManagedVault extends ManagedVaultConnector {
|
|
26
|
-
public name: string = 'SecretManagerManagedVault';
|
|
27
|
-
public scope: string = 'smyth-managed-vault';
|
|
28
|
-
private secretsManager: SecretsManagerClient;
|
|
29
|
-
|
|
30
|
-
constructor(protected _settings: SecretsManagerConfig & { vaultName: string }) {
|
|
31
|
-
super(_settings);
|
|
32
|
-
//if (!SmythRuntime.Instance) throw new Error('SRE not initialized');
|
|
33
|
-
|
|
34
|
-
this.secretsManager = new SecretsManagerClient({
|
|
35
|
-
region: _settings.region,
|
|
36
|
-
...(_settings.awsAccessKeyId && _settings.awsSecretAccessKey
|
|
37
|
-
? {
|
|
38
|
-
accessKeyId: _settings.awsAccessKeyId,
|
|
39
|
-
secretAccessKey: _settings.awsSecretAccessKey,
|
|
40
|
-
}
|
|
41
|
-
: {}),
|
|
42
|
-
});
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
@SecureConnector.AccessControl
|
|
46
|
-
protected async get(acRequest: AccessRequest, secretName: string) {
|
|
47
|
-
const secret = await this.getSecretByName(secretName);
|
|
48
|
-
return secret?.SecretString;
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
@SecureConnector.AccessControl
|
|
52
|
-
protected async set(acRequest: AccessRequest, secretName: string, value: string) {
|
|
53
|
-
const secret = await this.getSecretByName(secretName);
|
|
54
|
-
if (secret) {
|
|
55
|
-
await this.secretsManager.send(new PutSecretValueCommand({ SecretId: secret.ARN, SecretString: value }));
|
|
56
|
-
} else {
|
|
57
|
-
await this.secretsManager.send(
|
|
58
|
-
new CreateSecretCommand({
|
|
59
|
-
Name: `smyth/${randomUUID()}`,
|
|
60
|
-
SecretString: JSON.stringify({ [secretName]: value }),
|
|
61
|
-
Tags: [{ Key: this.scope, Value: 'true' }],
|
|
62
|
-
})
|
|
63
|
-
);
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
@SecureConnector.AccessControl
|
|
68
|
-
protected async delete(acRequest: AccessRequest, secretName: string) {
|
|
69
|
-
const secret = await this.getSecretByName(secretName);
|
|
70
|
-
if (secret) {
|
|
71
|
-
await this.secretsManager.send(new DeleteSecretCommand({ SecretId: secret.ARN }));
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
|
|
75
|
-
@SecureConnector.AccessControl
|
|
76
|
-
protected async exists(acRequest: AccessRequest, secretName: string) {
|
|
77
|
-
const secret = await this.get(acRequest, secretName);
|
|
78
|
-
return !!secret;
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
public async getResourceACL(resourceId: string, candidate: IAccessCandidate) {
|
|
82
|
-
const accountConnector = ConnectorService.getAccountConnector();
|
|
83
|
-
const teamId = await accountConnector.getCandidateTeam(candidate);
|
|
84
|
-
|
|
85
|
-
const acl = new ACL();
|
|
86
|
-
|
|
87
|
-
acl.addAccess(TAccessRole.Team, teamId, TAccessLevel.Owner)
|
|
88
|
-
.addAccess(TAccessRole.Team, teamId, TAccessLevel.Read)
|
|
89
|
-
.addAccess(TAccessRole.Team, teamId, TAccessLevel.Write);
|
|
90
|
-
|
|
91
|
-
return acl;
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
private async getSecretByName(secretName: string) {
|
|
95
|
-
try {
|
|
96
|
-
const secrets = [];
|
|
97
|
-
let nextToken: string | undefined;
|
|
98
|
-
do {
|
|
99
|
-
const listResponse: ListSecretsCommandOutput = await this.secretsManager.send(
|
|
100
|
-
new ListSecretsCommand({ NextToken: nextToken, Filters: [{ Key: 'tag-key', Values: [this.scope] }] })
|
|
101
|
-
);
|
|
102
|
-
if (listResponse.SecretList) {
|
|
103
|
-
for (const secret of listResponse.SecretList) {
|
|
104
|
-
if (secret.Name) {
|
|
105
|
-
secrets.push({
|
|
106
|
-
ARN: secret.ARN,
|
|
107
|
-
Name: secret.Name,
|
|
108
|
-
CreatedDate: secret.CreatedDate,
|
|
109
|
-
});
|
|
110
|
-
}
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
nextToken = listResponse.NextToken;
|
|
114
|
-
} while (nextToken);
|
|
115
|
-
|
|
116
|
-
const formattedSecrets = [];
|
|
117
|
-
const $promises = [];
|
|
118
|
-
for (const secret of secrets) {
|
|
119
|
-
$promises.push(getSpecificSecret(secret, this.secretsManager));
|
|
120
|
-
}
|
|
121
|
-
const results = await Promise.all($promises);
|
|
122
|
-
for (const result of results) {
|
|
123
|
-
formattedSecrets.push(result);
|
|
124
|
-
}
|
|
125
|
-
const secret = formattedSecrets.find((s) => s.Name === secretName);
|
|
126
|
-
return secret;
|
|
127
|
-
} catch (error) {
|
|
128
|
-
console.error(error);
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
async function getSpecificSecret(secret, secretsManager: SecretsManagerClient) {
|
|
132
|
-
const data: GetSecretValueCommandOutput = await secretsManager.send(new GetSecretValueCommand({ SecretId: secret.ARN }));
|
|
133
|
-
let secretString = data.SecretString;
|
|
134
|
-
let secretName = secret.Name;
|
|
135
|
-
|
|
136
|
-
if (secretString) {
|
|
137
|
-
try {
|
|
138
|
-
let parsedSecret = JSON.parse(secretString);
|
|
139
|
-
if (Object.keys(parsedSecret).length === 1) {
|
|
140
|
-
secretName = Object.keys(parsedSecret)[0];
|
|
141
|
-
secretString = parsedSecret[secretName];
|
|
142
|
-
}
|
|
143
|
-
} catch (error) {}
|
|
144
|
-
}
|
|
145
|
-
return {
|
|
146
|
-
Name: secretName,
|
|
147
|
-
ARN: secret.ARN,
|
|
148
|
-
CreatedDate: secret.CreatedDate,
|
|
149
|
-
SecretId: secret.Name,
|
|
150
|
-
SecretString: secretString,
|
|
151
|
-
};
|
|
152
|
-
}
|
|
153
|
-
}
|
|
154
|
-
}
|
|
1
|
+
import { ConnectorService } from '@sre/Core/ConnectorsService';
|
|
2
|
+
import { Logger } from '@sre/helpers/Log.helper';
|
|
3
|
+
//import { SmythRuntime } from '@sre/Core/SmythRuntime.class';
|
|
4
|
+
import { AccessRequest } from '@sre/Security/AccessControl/AccessRequest.class';
|
|
5
|
+
import { ACL } from '@sre/Security/AccessControl/ACL.class';
|
|
6
|
+
import { SecureConnector } from '@sre/Security/SecureConnector.class';
|
|
7
|
+
import { IAccessCandidate, TAccessLevel, TAccessRole } from '@sre/types/ACL.types';
|
|
8
|
+
|
|
9
|
+
import {
|
|
10
|
+
CreateSecretCommand,
|
|
11
|
+
DeleteSecretCommand,
|
|
12
|
+
GetSecretValueCommand,
|
|
13
|
+
GetSecretValueCommandOutput,
|
|
14
|
+
ListSecretsCommand,
|
|
15
|
+
ListSecretsCommandOutput,
|
|
16
|
+
PutSecretValueCommand,
|
|
17
|
+
SecretsManagerClient,
|
|
18
|
+
} from '@aws-sdk/client-secrets-manager';
|
|
19
|
+
import { randomUUID } from 'crypto';
|
|
20
|
+
import { ManagedVaultConnector } from '../ManagedVaultConnector';
|
|
21
|
+
import { SecretsManagerConfig } from '../../Vault.service/connectors/SecretsManager.class';
|
|
22
|
+
|
|
23
|
+
const console = Logger('SecretManagerManagedVault');
|
|
24
|
+
|
|
25
|
+
export class SecretManagerManagedVault extends ManagedVaultConnector {
|
|
26
|
+
public name: string = 'SecretManagerManagedVault';
|
|
27
|
+
public scope: string = 'smyth-managed-vault';
|
|
28
|
+
private secretsManager: SecretsManagerClient;
|
|
29
|
+
|
|
30
|
+
constructor(protected _settings: SecretsManagerConfig & { vaultName: string }) {
|
|
31
|
+
super(_settings);
|
|
32
|
+
//if (!SmythRuntime.Instance) throw new Error('SRE not initialized');
|
|
33
|
+
|
|
34
|
+
this.secretsManager = new SecretsManagerClient({
|
|
35
|
+
region: _settings.region,
|
|
36
|
+
...(_settings.awsAccessKeyId && _settings.awsSecretAccessKey
|
|
37
|
+
? {
|
|
38
|
+
accessKeyId: _settings.awsAccessKeyId,
|
|
39
|
+
secretAccessKey: _settings.awsSecretAccessKey,
|
|
40
|
+
}
|
|
41
|
+
: {}),
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
@SecureConnector.AccessControl
|
|
46
|
+
protected async get(acRequest: AccessRequest, secretName: string) {
|
|
47
|
+
const secret = await this.getSecretByName(secretName);
|
|
48
|
+
return secret?.SecretString;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
@SecureConnector.AccessControl
|
|
52
|
+
protected async set(acRequest: AccessRequest, secretName: string, value: string) {
|
|
53
|
+
const secret = await this.getSecretByName(secretName);
|
|
54
|
+
if (secret) {
|
|
55
|
+
await this.secretsManager.send(new PutSecretValueCommand({ SecretId: secret.ARN, SecretString: value }));
|
|
56
|
+
} else {
|
|
57
|
+
await this.secretsManager.send(
|
|
58
|
+
new CreateSecretCommand({
|
|
59
|
+
Name: `smyth/${randomUUID()}`,
|
|
60
|
+
SecretString: JSON.stringify({ [secretName]: value }),
|
|
61
|
+
Tags: [{ Key: this.scope, Value: 'true' }],
|
|
62
|
+
})
|
|
63
|
+
);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
@SecureConnector.AccessControl
|
|
68
|
+
protected async delete(acRequest: AccessRequest, secretName: string) {
|
|
69
|
+
const secret = await this.getSecretByName(secretName);
|
|
70
|
+
if (secret) {
|
|
71
|
+
await this.secretsManager.send(new DeleteSecretCommand({ SecretId: secret.ARN }));
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
@SecureConnector.AccessControl
|
|
76
|
+
protected async exists(acRequest: AccessRequest, secretName: string) {
|
|
77
|
+
const secret = await this.get(acRequest, secretName);
|
|
78
|
+
return !!secret;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
public async getResourceACL(resourceId: string, candidate: IAccessCandidate) {
|
|
82
|
+
const accountConnector = ConnectorService.getAccountConnector();
|
|
83
|
+
const teamId = await accountConnector.getCandidateTeam(candidate);
|
|
84
|
+
|
|
85
|
+
const acl = new ACL();
|
|
86
|
+
|
|
87
|
+
acl.addAccess(TAccessRole.Team, teamId, TAccessLevel.Owner)
|
|
88
|
+
.addAccess(TAccessRole.Team, teamId, TAccessLevel.Read)
|
|
89
|
+
.addAccess(TAccessRole.Team, teamId, TAccessLevel.Write);
|
|
90
|
+
|
|
91
|
+
return acl;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
private async getSecretByName(secretName: string) {
|
|
95
|
+
try {
|
|
96
|
+
const secrets = [];
|
|
97
|
+
let nextToken: string | undefined;
|
|
98
|
+
do {
|
|
99
|
+
const listResponse: ListSecretsCommandOutput = await this.secretsManager.send(
|
|
100
|
+
new ListSecretsCommand({ NextToken: nextToken, Filters: [{ Key: 'tag-key', Values: [this.scope] }] })
|
|
101
|
+
);
|
|
102
|
+
if (listResponse.SecretList) {
|
|
103
|
+
for (const secret of listResponse.SecretList) {
|
|
104
|
+
if (secret.Name) {
|
|
105
|
+
secrets.push({
|
|
106
|
+
ARN: secret.ARN,
|
|
107
|
+
Name: secret.Name,
|
|
108
|
+
CreatedDate: secret.CreatedDate,
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
nextToken = listResponse.NextToken;
|
|
114
|
+
} while (nextToken);
|
|
115
|
+
|
|
116
|
+
const formattedSecrets = [];
|
|
117
|
+
const $promises = [];
|
|
118
|
+
for (const secret of secrets) {
|
|
119
|
+
$promises.push(getSpecificSecret(secret, this.secretsManager));
|
|
120
|
+
}
|
|
121
|
+
const results = await Promise.all($promises);
|
|
122
|
+
for (const result of results) {
|
|
123
|
+
formattedSecrets.push(result);
|
|
124
|
+
}
|
|
125
|
+
const secret = formattedSecrets.find((s) => s.Name === secretName);
|
|
126
|
+
return secret;
|
|
127
|
+
} catch (error) {
|
|
128
|
+
console.error(error);
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
async function getSpecificSecret(secret, secretsManager: SecretsManagerClient) {
|
|
132
|
+
const data: GetSecretValueCommandOutput = await secretsManager.send(new GetSecretValueCommand({ SecretId: secret.ARN }));
|
|
133
|
+
let secretString = data.SecretString;
|
|
134
|
+
let secretName = secret.Name;
|
|
135
|
+
|
|
136
|
+
if (secretString) {
|
|
137
|
+
try {
|
|
138
|
+
let parsedSecret = JSON.parse(secretString);
|
|
139
|
+
if (Object.keys(parsedSecret).length === 1) {
|
|
140
|
+
secretName = Object.keys(parsedSecret)[0];
|
|
141
|
+
secretString = parsedSecret[secretName];
|
|
142
|
+
}
|
|
143
|
+
} catch (error) {}
|
|
144
|
+
}
|
|
145
|
+
return {
|
|
146
|
+
Name: secretName,
|
|
147
|
+
ARN: secret.ARN,
|
|
148
|
+
CreatedDate: secret.CreatedDate,
|
|
149
|
+
SecretId: secret.Name,
|
|
150
|
+
SecretString: secretString,
|
|
151
|
+
};
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
}
|
|
@@ -1,12 +1,12 @@
|
|
|
1
|
-
import { ConnectorService, ConnectorServiceProvider } from '@sre/Core/ConnectorsService';
|
|
2
|
-
import { TConnectorService } from '@sre/types/SRE.types';
|
|
3
|
-
|
|
4
|
-
import { SecretManagerManagedVault } from './connectors/SecretManagerManagedVault';
|
|
5
|
-
import { NullManagedVault } from './connectors/NullManagedVault.class';
|
|
6
|
-
|
|
7
|
-
export class ManagedVaultService extends ConnectorServiceProvider {
|
|
8
|
-
public register() {
|
|
9
|
-
ConnectorService.register(TConnectorService.ManagedVault, 'SecretManagerManagedVault', SecretManagerManagedVault);
|
|
10
|
-
ConnectorService.register(TConnectorService.ManagedVault, 'NullManagedVault', NullManagedVault);
|
|
11
|
-
}
|
|
12
|
-
}
|
|
1
|
+
import { ConnectorService, ConnectorServiceProvider } from '@sre/Core/ConnectorsService';
|
|
2
|
+
import { TConnectorService } from '@sre/types/SRE.types';
|
|
3
|
+
|
|
4
|
+
import { SecretManagerManagedVault } from './connectors/SecretManagerManagedVault';
|
|
5
|
+
import { NullManagedVault } from './connectors/NullManagedVault.class';
|
|
6
|
+
|
|
7
|
+
export class ManagedVaultService extends ConnectorServiceProvider {
|
|
8
|
+
public register() {
|
|
9
|
+
ConnectorService.register(TConnectorService.ManagedVault, 'SecretManagerManagedVault', SecretManagerManagedVault);
|
|
10
|
+
ConnectorService.register(TConnectorService.ManagedVault, 'NullManagedVault', NullManagedVault);
|
|
11
|
+
}
|
|
12
|
+
}
|