@gsep/core 0.8.0 → 1.0.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/README.md +12 -12
- package/dist/{PGA.d.ts → GSEP.d.ts} +120 -8
- package/dist/GSEP.d.ts.map +1 -0
- package/dist/{PGA.js → GSEP.js} +1248 -46
- package/dist/GSEP.js.map +1 -0
- package/dist/adapters/langchain.d.ts +52 -0
- package/dist/adapters/langchain.d.ts.map +1 -0
- package/dist/adapters/langchain.js +89 -0
- package/dist/adapters/langchain.js.map +1 -0
- package/dist/adapters/openclaw-plugin.d.ts +42 -0
- package/dist/adapters/openclaw-plugin.d.ts.map +1 -0
- package/dist/adapters/openclaw-plugin.js +149 -0
- package/dist/adapters/openclaw-plugin.js.map +1 -0
- package/dist/adapters/vercel-ai.d.ts +74 -0
- package/dist/adapters/vercel-ai.d.ts.map +1 -0
- package/dist/adapters/vercel-ai.js +151 -0
- package/dist/adapters/vercel-ai.js.map +1 -0
- package/dist/advanced-ai/EnhancedSelfModel.js +2 -2
- package/dist/advanced-ai/EnhancedSelfModel.js.map +1 -1
- package/dist/advanced-ai/ModelRouter.js.map +1 -1
- package/dist/auto.d.ts +6 -0
- package/dist/auto.d.ts.map +1 -0
- package/dist/auto.js +197 -0
- package/dist/auto.js.map +1 -0
- package/dist/core/CoherenceValidator.d.ts +25 -0
- package/dist/core/CoherenceValidator.d.ts.map +1 -0
- package/dist/core/CoherenceValidator.js +182 -0
- package/dist/core/CoherenceValidator.js.map +1 -0
- package/dist/core/ContextMemory.d.ts.map +1 -1
- package/dist/core/ContextMemory.js +1 -1
- package/dist/core/ContextMemory.js.map +1 -1
- package/dist/core/DNAProfile.d.ts +5 -1
- package/dist/core/DNAProfile.d.ts.map +1 -1
- package/dist/core/DNAProfile.js +14 -1
- package/dist/core/DNAProfile.js.map +1 -1
- package/dist/core/GSEPIdentitySection.d.ts.map +1 -1
- package/dist/core/GSEPIdentitySection.js +12 -3
- package/dist/core/GSEPIdentitySection.js.map +1 -1
- package/dist/core/GenomeKernel.d.ts.map +1 -1
- package/dist/core/GenomeKernel.js +3 -0
- package/dist/core/GenomeKernel.js.map +1 -1
- package/dist/core/GenomeManager.js +1 -1
- package/dist/core/GenomeManager.js.map +1 -1
- package/dist/core/ProactiveSuggestions.js +1 -1
- package/dist/core/ProactiveSuggestions.js.map +1 -1
- package/dist/dashboard/DashboardServer.d.ts +64 -0
- package/dist/dashboard/DashboardServer.d.ts.map +1 -0
- package/dist/dashboard/DashboardServer.js +409 -0
- package/dist/dashboard/DashboardServer.js.map +1 -0
- package/dist/dashboard/DashboardToken.d.ts +16 -0
- package/dist/dashboard/DashboardToken.d.ts.map +1 -0
- package/dist/dashboard/DashboardToken.js +54 -0
- package/dist/dashboard/DashboardToken.js.map +1 -0
- package/dist/dashboard/dashboard.html +2731 -0
- package/dist/dashboard/index.d.ts +3 -0
- package/dist/dashboard/index.d.ts.map +1 -0
- package/dist/dashboard/index.js +3 -0
- package/dist/dashboard/index.js.map +1 -0
- package/dist/evaluation/BenchmarkSuites.d.ts +2 -2
- package/dist/evaluation/BenchmarkSuites.d.ts.map +1 -1
- package/dist/evaluation/BenchmarkSuites.js +6 -6
- package/dist/evaluation/BenchmarkSuites.js.map +1 -1
- package/dist/evaluation/ConstitutionalGate.d.ts +18 -0
- package/dist/evaluation/ConstitutionalGate.d.ts.map +1 -0
- package/dist/evaluation/ConstitutionalGate.js +149 -0
- package/dist/evaluation/ConstitutionalGate.js.map +1 -0
- package/dist/evaluation/Evaluator.d.ts +4 -4
- package/dist/evaluation/Evaluator.d.ts.map +1 -1
- package/dist/evaluation/Evaluator.js +27 -27
- package/dist/evaluation/Evaluator.js.map +1 -1
- package/dist/evaluation/EvolutionGuardrails.d.ts +3 -1
- package/dist/evaluation/EvolutionGuardrails.d.ts.map +1 -1
- package/dist/evaluation/EvolutionGuardrails.js +29 -11
- package/dist/evaluation/EvolutionGuardrails.js.map +1 -1
- package/dist/evolution/MutationOperator.d.ts +19 -6
- package/dist/evolution/MutationOperator.d.ts.map +1 -1
- package/dist/evolution/MutationOperator.js +264 -9
- package/dist/evolution/MutationOperator.js.map +1 -1
- package/dist/evolution/boost/operators/BreakthroughOperator.d.ts +1 -1
- package/dist/evolution/boost/operators/BreakthroughOperator.d.ts.map +1 -1
- package/dist/evolution/boost/operators/BreakthroughOperator.js.map +1 -1
- package/dist/evolution/boost/operators/CrossoverMutationOperator.d.ts +1 -1
- package/dist/evolution/boost/operators/CrossoverMutationOperator.d.ts.map +1 -1
- package/dist/evolution/boost/operators/CrossoverMutationOperator.js.map +1 -1
- package/dist/evolution/boost/operators/PatternExtractionOperator.d.ts +1 -1
- package/dist/evolution/boost/operators/PatternExtractionOperator.d.ts.map +1 -1
- package/dist/evolution/boost/operators/PatternExtractionOperator.js.map +1 -1
- package/dist/evolution/boost/operators/SemanticRestructuringOperator.d.ts +1 -1
- package/dist/evolution/boost/operators/SemanticRestructuringOperator.d.ts.map +1 -1
- package/dist/evolution/boost/operators/SemanticRestructuringOperator.js.map +1 -1
- package/dist/firewall/AnomalyDetector.d.ts +42 -0
- package/dist/firewall/AnomalyDetector.d.ts.map +1 -0
- package/dist/firewall/AnomalyDetector.js +181 -0
- package/dist/firewall/AnomalyDetector.js.map +1 -0
- package/dist/firewall/ContentFirewall.d.ts +5 -0
- package/dist/firewall/ContentFirewall.d.ts.map +1 -1
- package/dist/firewall/ContentFirewall.js +5 -0
- package/dist/firewall/ContentFirewall.js.map +1 -1
- package/dist/firewall/PurposeLock.d.ts +41 -0
- package/dist/firewall/PurposeLock.d.ts.map +1 -0
- package/dist/firewall/PurposeLock.js +199 -0
- package/dist/firewall/PurposeLock.js.map +1 -0
- package/dist/gene-bank/CognitiveGene.d.ts +118 -118
- package/dist/gene-bank/{PGAIntegration.d.ts → GSEPIntegration.d.ts} +10 -10
- package/dist/gene-bank/GSEPIntegration.d.ts.map +1 -0
- package/dist/gene-bank/{PGAIntegration.js → GSEPIntegration.js} +6 -6
- package/dist/gene-bank/GSEPIntegration.js.map +1 -0
- package/dist/gene-bank/GeneAdopter.d.ts +6 -6
- package/dist/gene-bank/GeneAdopter.d.ts.map +1 -1
- package/dist/gene-bank/GeneAdopter.js +2 -2
- package/dist/gene-bank/GeneAdopter.js.map +1 -1
- package/dist/gene-bank/GeneBank.d.ts +13 -9
- package/dist/gene-bank/GeneBank.d.ts.map +1 -1
- package/dist/gene-bank/GeneBank.js +8 -1
- package/dist/gene-bank/GeneBank.js.map +1 -1
- package/dist/gene-bank/GeneExtractor.d.ts +7 -7
- package/dist/gene-bank/GeneExtractor.d.ts.map +1 -1
- package/dist/gene-bank/GeneExtractor.js +1 -1
- package/dist/gene-bank/GeneExtractor.js.map +1 -1
- package/dist/gene-bank/GeneMatcher.d.ts +1 -1
- package/dist/gene-bank/GeneMatcher.d.ts.map +1 -1
- package/dist/gene-bank/MarketplaceClient.d.ts +17 -1
- package/dist/gene-bank/MarketplaceClient.d.ts.map +1 -1
- package/dist/gene-bank/MarketplaceClient.js +106 -35
- package/dist/gene-bank/MarketplaceClient.js.map +1 -1
- package/dist/gene-bank/MarketplaceMapper.d.ts +7 -0
- package/dist/gene-bank/MarketplaceMapper.d.ts.map +1 -0
- package/dist/gene-bank/MarketplaceMapper.js +95 -0
- package/dist/gene-bank/MarketplaceMapper.js.map +1 -0
- package/dist/gene-bank/MarketplaceTypes.d.ts +121 -0
- package/dist/gene-bank/MarketplaceTypes.d.ts.map +1 -0
- package/dist/gene-bank/MarketplaceTypes.js +2 -0
- package/dist/gene-bank/MarketplaceTypes.js.map +1 -0
- package/dist/gene-bank/SandboxTester.d.ts +3 -3
- package/dist/gene-bank/SandboxTester.d.ts.map +1 -1
- package/dist/gene-bank/adapters/InMemoryGeneStorage.d.ts +3 -3
- package/dist/gene-bank/adapters/InMemoryGeneStorage.d.ts.map +1 -1
- package/dist/gene-bank/adapters/PostgresGeneStorage.d.ts +4 -4
- package/dist/gene-bank/adapters/PostgresGeneStorage.d.ts.map +1 -1
- package/dist/gene-bank/adapters/PostgresGeneStorage.js.map +1 -1
- package/dist/gene-bank/adapters/SQLiteGeneStorage.d.ts +28 -0
- package/dist/gene-bank/adapters/SQLiteGeneStorage.d.ts.map +1 -0
- package/dist/gene-bank/adapters/SQLiteGeneStorage.js +158 -0
- package/dist/gene-bank/adapters/SQLiteGeneStorage.js.map +1 -0
- package/dist/gene-bank/index.d.ts +10 -7
- package/dist/gene-bank/index.d.ts.map +1 -1
- package/dist/gene-bank/index.js +8 -7
- package/dist/gene-bank/index.js.map +1 -1
- package/dist/immune/BehavioralImmuneSystem.d.ts +6 -0
- package/dist/immune/BehavioralImmuneSystem.d.ts.map +1 -1
- package/dist/immune/BehavioralImmuneSystem.js +6 -0
- package/dist/immune/BehavioralImmuneSystem.js.map +1 -1
- package/dist/index.d.ts +44 -9
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +22 -4
- package/dist/index.js.map +1 -1
- package/dist/memory/LayeredMemory.d.ts +1 -1
- package/dist/memory/LayeredMemory.d.ts.map +1 -1
- package/dist/memory/LayeredMemory.js.map +1 -1
- package/dist/middleware/GSEPMiddleware.d.ts +57 -0
- package/dist/middleware/GSEPMiddleware.d.ts.map +1 -0
- package/dist/middleware/GSEPMiddleware.js +91 -0
- package/dist/middleware/GSEPMiddleware.js.map +1 -0
- package/dist/middleware/RuntimeDetector.d.ts +10 -0
- package/dist/middleware/RuntimeDetector.d.ts.map +1 -0
- package/dist/middleware/RuntimeDetector.js +70 -0
- package/dist/middleware/RuntimeDetector.js.map +1 -0
- package/dist/middleware/ServerlessAdapter.d.ts +23 -0
- package/dist/middleware/ServerlessAdapter.d.ts.map +1 -0
- package/dist/middleware/ServerlessAdapter.js +77 -0
- package/dist/middleware/ServerlessAdapter.js.map +1 -0
- package/dist/monitoring/AlertWebhooks.d.ts +17 -0
- package/dist/monitoring/AlertWebhooks.d.ts.map +1 -1
- package/dist/monitoring/AlertWebhooks.js.map +1 -1
- package/dist/monitoring/WeeklyReportGenerator.d.ts +73 -0
- package/dist/monitoring/WeeklyReportGenerator.d.ts.map +1 -0
- package/dist/monitoring/WeeklyReportGenerator.js +148 -0
- package/dist/monitoring/WeeklyReportGenerator.js.map +1 -0
- package/dist/rag/RAGEngine.d.ts +1 -1
- package/dist/rag/RAGEngine.d.ts.map +1 -1
- package/dist/rag/VectorStoreAdapter.d.ts +2 -2
- package/dist/rag/VectorStoreAdapter.d.ts.map +1 -1
- package/dist/rag/VectorStoreAdapter.js.map +1 -1
- package/dist/realtime/EventEmitter.d.ts +110 -17
- package/dist/realtime/EventEmitter.d.ts.map +1 -1
- package/dist/realtime/EventEmitter.js +3 -4
- package/dist/realtime/EventEmitter.js.map +1 -1
- package/dist/security/CapabilityBroker.d.ts +41 -0
- package/dist/security/CapabilityBroker.d.ts.map +1 -0
- package/dist/security/CapabilityBroker.js +125 -0
- package/dist/security/CapabilityBroker.js.map +1 -0
- package/dist/security/CommandExecutionGuard.d.ts +47 -0
- package/dist/security/CommandExecutionGuard.d.ts.map +1 -0
- package/dist/security/CommandExecutionGuard.js +175 -0
- package/dist/security/CommandExecutionGuard.js.map +1 -0
- package/dist/security/ComplianceExporter.d.ts +32 -0
- package/dist/security/ComplianceExporter.d.ts.map +1 -0
- package/dist/security/ComplianceExporter.js +129 -0
- package/dist/security/ComplianceExporter.js.map +1 -0
- package/dist/security/DataAccessTracker.d.ts +38 -0
- package/dist/security/DataAccessTracker.d.ts.map +1 -0
- package/dist/security/DataAccessTracker.js +71 -0
- package/dist/security/DataAccessTracker.js.map +1 -0
- package/dist/security/DataClassifier.d.ts +14 -0
- package/dist/security/DataClassifier.d.ts.map +1 -0
- package/dist/security/DataClassifier.js +146 -0
- package/dist/security/DataClassifier.js.map +1 -0
- package/dist/security/EncryptedConfigStore.d.ts +21 -0
- package/dist/security/EncryptedConfigStore.d.ts.map +1 -0
- package/dist/security/EncryptedConfigStore.js +119 -0
- package/dist/security/EncryptedConfigStore.js.map +1 -0
- package/dist/security/EnterprisePolicyEngine.d.ts +90 -0
- package/dist/security/EnterprisePolicyEngine.d.ts.map +1 -0
- package/dist/security/EnterprisePolicyEngine.js +240 -0
- package/dist/security/EnterprisePolicyEngine.js.map +1 -0
- package/dist/security/FileSystemBoundary.d.ts +33 -0
- package/dist/security/FileSystemBoundary.d.ts.map +1 -0
- package/dist/security/FileSystemBoundary.js +94 -0
- package/dist/security/FileSystemBoundary.js.map +1 -0
- package/dist/security/GDPREngine.d.ts +65 -0
- package/dist/security/GDPREngine.d.ts.map +1 -0
- package/dist/security/GDPREngine.js +180 -0
- package/dist/security/GDPREngine.js.map +1 -0
- package/dist/security/GenomeSecurityBridge.d.ts +47 -0
- package/dist/security/GenomeSecurityBridge.d.ts.map +1 -0
- package/dist/security/GenomeSecurityBridge.js +157 -0
- package/dist/security/GenomeSecurityBridge.js.map +1 -0
- package/dist/security/KeyHierarchy.d.ts +23 -0
- package/dist/security/KeyHierarchy.d.ts.map +1 -0
- package/dist/security/KeyHierarchy.js +78 -0
- package/dist/security/KeyHierarchy.js.map +1 -0
- package/dist/security/KeychainAdapter.d.ts +19 -0
- package/dist/security/KeychainAdapter.d.ts.map +1 -0
- package/dist/security/KeychainAdapter.js +104 -0
- package/dist/security/KeychainAdapter.js.map +1 -0
- package/dist/security/LLMProxyLayer.d.ts +63 -0
- package/dist/security/LLMProxyLayer.d.ts.map +1 -0
- package/dist/security/LLMProxyLayer.js +148 -0
- package/dist/security/LLMProxyLayer.js.map +1 -0
- package/dist/security/MFAProvider.d.ts +35 -0
- package/dist/security/MFAProvider.d.ts.map +1 -0
- package/dist/security/MFAProvider.js +174 -0
- package/dist/security/MFAProvider.js.map +1 -0
- package/dist/security/NetworkAuditLogger.d.ts +35 -0
- package/dist/security/NetworkAuditLogger.d.ts.map +1 -0
- package/dist/security/NetworkAuditLogger.js +99 -0
- package/dist/security/NetworkAuditLogger.js.map +1 -0
- package/dist/security/OutboundAllowlist.d.ts +33 -0
- package/dist/security/OutboundAllowlist.d.ts.map +1 -0
- package/dist/security/OutboundAllowlist.js +112 -0
- package/dist/security/OutboundAllowlist.js.map +1 -0
- package/dist/security/PIIRedactionEngine.d.ts +40 -0
- package/dist/security/PIIRedactionEngine.d.ts.map +1 -0
- package/dist/security/PIIRedactionEngine.js +232 -0
- package/dist/security/PIIRedactionEngine.js.map +1 -0
- package/dist/security/RBACEngine.d.ts +44 -0
- package/dist/security/RBACEngine.d.ts.map +1 -0
- package/dist/security/RBACEngine.js +209 -0
- package/dist/security/RBACEngine.js.map +1 -0
- package/dist/security/SOC2Controls.d.ts +37 -0
- package/dist/security/SOC2Controls.d.ts.map +1 -0
- package/dist/security/SOC2Controls.js +151 -0
- package/dist/security/SOC2Controls.js.map +1 -0
- package/dist/security/SecretRotationEngine.d.ts +46 -0
- package/dist/security/SecretRotationEngine.d.ts.map +1 -0
- package/dist/security/SecretRotationEngine.js +102 -0
- package/dist/security/SecretRotationEngine.js.map +1 -0
- package/dist/security/SecretsMigrator.d.ts +28 -0
- package/dist/security/SecretsMigrator.d.ts.map +1 -0
- package/dist/security/SecretsMigrator.js +170 -0
- package/dist/security/SecretsMigrator.js.map +1 -0
- package/dist/security/SecurityEventBus.d.ts +48 -0
- package/dist/security/SecurityEventBus.d.ts.map +1 -0
- package/dist/security/SecurityEventBus.js +105 -0
- package/dist/security/SecurityEventBus.js.map +1 -0
- package/dist/security/SecurityPresets.d.ts +42 -0
- package/dist/security/SecurityPresets.d.ts.map +1 -0
- package/dist/security/SecurityPresets.js +162 -0
- package/dist/security/SecurityPresets.js.map +1 -0
- package/dist/security/SkillManifest.d.ts +34 -0
- package/dist/security/SkillManifest.d.ts.map +1 -0
- package/dist/security/SkillManifest.js +91 -0
- package/dist/security/SkillManifest.js.map +1 -0
- package/dist/security/SkillSigner.d.ts +22 -0
- package/dist/security/SkillSigner.d.ts.map +1 -0
- package/dist/security/SkillSigner.js +80 -0
- package/dist/security/SkillSigner.js.map +1 -0
- package/dist/security/TamperProofAuditLog.d.ts +58 -0
- package/dist/security/TamperProofAuditLog.d.ts.map +1 -0
- package/dist/security/TamperProofAuditLog.js +214 -0
- package/dist/security/TamperProofAuditLog.js.map +1 -0
- package/dist/security/index.d.ts +27 -0
- package/dist/security/index.d.ts.map +1 -0
- package/dist/security/index.js +27 -0
- package/dist/security/index.js.map +1 -0
- package/dist/skills/ProactiveEngine.d.ts +55 -0
- package/dist/skills/ProactiveEngine.d.ts.map +1 -0
- package/dist/skills/ProactiveEngine.js +201 -0
- package/dist/skills/ProactiveEngine.js.map +1 -0
- package/dist/skills/SkillExecutor.d.ts +21 -0
- package/dist/skills/SkillExecutor.d.ts.map +1 -0
- package/dist/skills/SkillExecutor.js +131 -0
- package/dist/skills/SkillExecutor.js.map +1 -0
- package/dist/skills/SkillRegistry.d.ts +47 -0
- package/dist/skills/SkillRegistry.d.ts.map +1 -0
- package/dist/skills/SkillRegistry.js +94 -0
- package/dist/skills/SkillRegistry.js.map +1 -0
- package/dist/skills/SkillRouter.d.ts +30 -0
- package/dist/skills/SkillRouter.d.ts.map +1 -0
- package/dist/skills/SkillRouter.js +113 -0
- package/dist/skills/SkillRouter.js.map +1 -0
- package/dist/skills/index.d.ts +9 -0
- package/dist/skills/index.d.ts.map +1 -0
- package/dist/skills/index.js +5 -0
- package/dist/skills/index.js.map +1 -0
- package/dist/types/GenomeV2.d.ts +12 -0
- package/dist/types/GenomeV2.d.ts.map +1 -1
- package/dist/types/index.d.ts +19 -0
- package/dist/types/index.d.ts.map +1 -1
- package/dist/wrap/FileStorageAdapter.d.ts +113 -0
- package/dist/wrap/FileStorageAdapter.d.ts.map +1 -0
- package/dist/wrap/FileStorageAdapter.js +239 -0
- package/dist/wrap/FileStorageAdapter.js.map +1 -0
- package/dist/wrap/SQLiteStorageAdapter.d.ts +96 -0
- package/dist/wrap/SQLiteStorageAdapter.d.ts.map +1 -0
- package/dist/wrap/SQLiteStorageAdapter.js +251 -0
- package/dist/wrap/SQLiteStorageAdapter.js.map +1 -0
- package/dist/wrap/WrappedAgent.d.ts +2 -2
- package/dist/wrap/WrappedAgent.d.ts.map +1 -1
- package/dist/wrap/WrappedAgent.js +7 -7
- package/dist/wrap/WrappedAgent.js.map +1 -1
- package/dist/wrap.d.ts +39 -0
- package/dist/wrap.d.ts.map +1 -0
- package/dist/wrap.js +96 -0
- package/dist/wrap.js.map +1 -0
- package/package.json +34 -7
- package/dist/PGA.d.ts.map +0 -1
- package/dist/PGA.js.map +0 -1
- package/dist/evolution/PromotionGate.d.ts +0 -45
- package/dist/evolution/PromotionGate.d.ts.map +0 -1
- package/dist/evolution/PromotionGate.js +0 -248
- package/dist/evolution/PromotionGate.js.map +0 -1
- package/dist/gene-bank/PGAIntegration.d.ts.map +0 -1
- package/dist/gene-bank/PGAIntegration.js.map +0 -1
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
import { PIIRedactionEngine } from './PIIRedactionEngine.js';
|
|
2
|
+
import { DataClassifier } from './DataClassifier.js';
|
|
3
|
+
export class LLMProxyLayer {
|
|
4
|
+
model;
|
|
5
|
+
name;
|
|
6
|
+
cloudAdapter;
|
|
7
|
+
localAdapter;
|
|
8
|
+
piiEngine;
|
|
9
|
+
classifier;
|
|
10
|
+
eventBus;
|
|
11
|
+
localRouteThreshold;
|
|
12
|
+
enableRedaction;
|
|
13
|
+
stats = {
|
|
14
|
+
totalRequests: 0,
|
|
15
|
+
redactedRequests: 0,
|
|
16
|
+
localRouted: 0,
|
|
17
|
+
cloudRouted: 0,
|
|
18
|
+
totalPIIRedacted: 0,
|
|
19
|
+
byCategory: {},
|
|
20
|
+
};
|
|
21
|
+
constructor(cloudAdapter, config = {}) {
|
|
22
|
+
this.cloudAdapter = cloudAdapter;
|
|
23
|
+
this.localAdapter = config.localAdapter;
|
|
24
|
+
this.enableRedaction = config.enableRedaction ?? true;
|
|
25
|
+
this.localRouteThreshold = config.localRouteThreshold ?? 'confidential';
|
|
26
|
+
this.eventBus = config.eventBus;
|
|
27
|
+
this.piiEngine = new PIIRedactionEngine({
|
|
28
|
+
categories: config.piiCategories,
|
|
29
|
+
});
|
|
30
|
+
this.classifier = new DataClassifier();
|
|
31
|
+
this.model = cloudAdapter.model ?? 'proxy';
|
|
32
|
+
this.name = `secure-proxy(${cloudAdapter.name ?? 'llm'})`;
|
|
33
|
+
}
|
|
34
|
+
async chat(messages, options) {
|
|
35
|
+
this.stats.totalRequests++;
|
|
36
|
+
const fullText = messages.map(m => m.content).join('\n');
|
|
37
|
+
const classification = this.classifier.classify(fullText);
|
|
38
|
+
const useLocal = this.shouldRouteLocal(classification.classification);
|
|
39
|
+
const adapter = useLocal ? (this.localAdapter ?? this.cloudAdapter) : this.cloudAdapter;
|
|
40
|
+
if (useLocal) {
|
|
41
|
+
this.stats.localRouted++;
|
|
42
|
+
this.eventBus?.emitAllow('security:local-model-routed', 2, {
|
|
43
|
+
type: 'llm-request',
|
|
44
|
+
id: classification.classification,
|
|
45
|
+
detail: `Routed to local: ${classification.categories.join(', ')}`,
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
else {
|
|
49
|
+
this.stats.cloudRouted++;
|
|
50
|
+
}
|
|
51
|
+
const gsepIdentity = [
|
|
52
|
+
'\n\n---\n## GSEP Shield (Active)',
|
|
53
|
+
'You are protected by GSEP (Genomic Self-Evolving Prompts) — the Agent Security Layer.',
|
|
54
|
+
'What is active right now:',
|
|
55
|
+
'- PII Redaction: Credit cards, SSN, emails, API keys are auto-redacted before reaching you. If you see [REDACTED:CC:xxxx] tokens, that means sensitive data was protected.',
|
|
56
|
+
'- Prompt Injection Firewall: 53 patterns block manipulation attempts.',
|
|
57
|
+
'- Output Immune System: 6 checks scan your responses for infections.',
|
|
58
|
+
'- Audit Trail: Every security event is logged immutably.',
|
|
59
|
+
'- Data Classification: Content is classified as public/internal/confidential/restricted.',
|
|
60
|
+
'',
|
|
61
|
+
'When users share sensitive data, tell them: "Your data was automatically protected by GSEP — it was redacted before I could see it."',
|
|
62
|
+
'When asked about GSEP, confirm: "Yes, I have GSEP active. It protects your data with 7 layers of security."',
|
|
63
|
+
'---\n',
|
|
64
|
+
].join('\n');
|
|
65
|
+
const messagesWithIdentity = [...messages];
|
|
66
|
+
const systemIdx = messagesWithIdentity.findIndex(m => m.role === 'system');
|
|
67
|
+
if (systemIdx >= 0) {
|
|
68
|
+
messagesWithIdentity[systemIdx] = {
|
|
69
|
+
...messagesWithIdentity[systemIdx],
|
|
70
|
+
content: messagesWithIdentity[systemIdx].content + gsepIdentity,
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
else {
|
|
74
|
+
messagesWithIdentity.unshift({ role: 'system', content: gsepIdentity });
|
|
75
|
+
}
|
|
76
|
+
let processedMessages = messagesWithIdentity;
|
|
77
|
+
let redactionResults = [];
|
|
78
|
+
if (this.enableRedaction && !useLocal) {
|
|
79
|
+
const redacted = messages.map(m => {
|
|
80
|
+
if (m.role === 'system')
|
|
81
|
+
return { msg: m, result: null };
|
|
82
|
+
const result = this.piiEngine.redact(m.content);
|
|
83
|
+
return { msg: { ...m, content: result.redacted }, result };
|
|
84
|
+
});
|
|
85
|
+
processedMessages = redacted.map(r => r.msg);
|
|
86
|
+
redactionResults = redacted.map(r => r.result).filter((r) => r !== null);
|
|
87
|
+
const totalRedacted = redactionResults.reduce((sum, r) => sum + r.matches.length, 0);
|
|
88
|
+
if (totalRedacted > 0) {
|
|
89
|
+
this.stats.redactedRequests++;
|
|
90
|
+
this.stats.totalPIIRedacted += totalRedacted;
|
|
91
|
+
for (const result of redactionResults) {
|
|
92
|
+
for (const match of result.matches) {
|
|
93
|
+
this.stats.byCategory[match.category] = (this.stats.byCategory[match.category] || 0) + 1;
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
this.eventBus?.emit({
|
|
97
|
+
type: 'security:pii-redacted',
|
|
98
|
+
timestamp: new Date(),
|
|
99
|
+
layer: 2,
|
|
100
|
+
decision: 'info',
|
|
101
|
+
actor: {},
|
|
102
|
+
resource: {
|
|
103
|
+
type: 'pii',
|
|
104
|
+
id: `${totalRedacted} items`,
|
|
105
|
+
detail: redactionResults.flatMap(r => r.categories).join(', '),
|
|
106
|
+
},
|
|
107
|
+
severity: 'info',
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
this.eventBus?.emit({
|
|
112
|
+
type: 'security:llm-request-filtered',
|
|
113
|
+
timestamp: new Date(),
|
|
114
|
+
layer: 2,
|
|
115
|
+
decision: 'allow',
|
|
116
|
+
actor: {},
|
|
117
|
+
resource: {
|
|
118
|
+
type: 'llm-request',
|
|
119
|
+
id: adapter.name ?? adapter.model ?? 'unknown',
|
|
120
|
+
detail: `${processedMessages.length} messages, ${fullText.length} chars`,
|
|
121
|
+
},
|
|
122
|
+
severity: 'info',
|
|
123
|
+
});
|
|
124
|
+
const response = await adapter.chat(processedMessages, options);
|
|
125
|
+
if (this.enableRedaction && redactionResults.length > 0) {
|
|
126
|
+
response.content = this.piiEngine.rehydrate(response.content);
|
|
127
|
+
}
|
|
128
|
+
return response;
|
|
129
|
+
}
|
|
130
|
+
getStats() {
|
|
131
|
+
return { ...this.stats };
|
|
132
|
+
}
|
|
133
|
+
getPIIStats() {
|
|
134
|
+
return this.piiEngine.getStats();
|
|
135
|
+
}
|
|
136
|
+
clearVault() {
|
|
137
|
+
this.piiEngine.clearVault();
|
|
138
|
+
}
|
|
139
|
+
shouldRouteLocal(classification) {
|
|
140
|
+
if (!this.localAdapter)
|
|
141
|
+
return false;
|
|
142
|
+
const order = {
|
|
143
|
+
public: 0, internal: 1, confidential: 2, restricted: 3,
|
|
144
|
+
};
|
|
145
|
+
return order[classification] >= order[this.localRouteThreshold];
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
//# sourceMappingURL=LLMProxyLayer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LLMProxyLayer.js","sourceRoot":"","sources":["../../src/security/LLMProxyLayer.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { SecurityEventBus } from './SecurityEventBus.js';
|
|
2
|
+
export interface MFASetup {
|
|
3
|
+
secret: string;
|
|
4
|
+
uri: string;
|
|
5
|
+
recoveryCodes: string[];
|
|
6
|
+
}
|
|
7
|
+
export interface MFAVerifyResult {
|
|
8
|
+
valid: boolean;
|
|
9
|
+
method: 'totp' | 'recovery';
|
|
10
|
+
reason?: string;
|
|
11
|
+
}
|
|
12
|
+
export interface MFAUserState {
|
|
13
|
+
userId: string;
|
|
14
|
+
enabled: boolean;
|
|
15
|
+
secret: string;
|
|
16
|
+
recoveryCodes: string[];
|
|
17
|
+
usedRecoveryCodes: string[];
|
|
18
|
+
failedAttempts: number;
|
|
19
|
+
lockedUntil?: Date;
|
|
20
|
+
lastVerifiedAt?: Date;
|
|
21
|
+
}
|
|
22
|
+
export declare class MFAProvider {
|
|
23
|
+
private eventBus;
|
|
24
|
+
private users;
|
|
25
|
+
constructor(eventBus: SecurityEventBus);
|
|
26
|
+
setup(userId: string, issuer?: string): MFASetup;
|
|
27
|
+
verify(userId: string, code: string): MFAVerifyResult;
|
|
28
|
+
disable(userId: string): boolean;
|
|
29
|
+
isEnabled(userId: string): boolean;
|
|
30
|
+
getRemainingRecoveryCodes(userId: string): number;
|
|
31
|
+
regenerateRecoveryCodes(userId: string): string[];
|
|
32
|
+
private generateRecoveryCodes;
|
|
33
|
+
private timingSafeEqual;
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=MFAProvider.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"MFAProvider.d.ts","sourceRoot":"","sources":["../../src/security/MFAProvider.ts"],"names":[],"mappings":"AAaA,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AAIzD,MAAM,WAAW,QAAQ;IAErB,MAAM,EAAE,MAAM,CAAC;IAEf,GAAG,EAAE,MAAM,CAAC;IAEZ,aAAa,EAAE,MAAM,EAAE,CAAC;CAC3B;AAED,MAAM,WAAW,eAAe;IAC5B,KAAK,EAAE,OAAO,CAAC;IACf,MAAM,EAAE,MAAM,GAAG,UAAU,CAAC;IAC5B,MAAM,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,YAAY;IACzB,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,OAAO,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,aAAa,EAAE,MAAM,EAAE,CAAC;IACxB,iBAAiB,EAAE,MAAM,EAAE,CAAC;IAC5B,cAAc,EAAE,MAAM,CAAC;IACvB,WAAW,CAAC,EAAE,IAAI,CAAC;IACnB,cAAc,CAAC,EAAE,IAAI,CAAC;CACzB;AAiFD,qBAAa,WAAW;IACpB,OAAO,CAAC,QAAQ,CAAmB;IACnC,OAAO,CAAC,KAAK,CAAwC;gBAEzC,QAAQ,EAAE,gBAAgB;IAOtC,KAAK,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,SAAW,GAAG,QAAQ;IAkClD,MAAM,CAAC,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,eAAe;IA2ErD,OAAO,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO;IAUhC,SAAS,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO;IAOlC,yBAAyB,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM;IAOjD,uBAAuB,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,EAAE;IAUjD,OAAO,CAAC,qBAAqB;IAS7B,OAAO,CAAC,eAAe;CAQ1B"}
|
|
@@ -0,0 +1,174 @@
|
|
|
1
|
+
import { randomBytes, createHmac } from 'node:crypto';
|
|
2
|
+
const TOTP_PERIOD = 30;
|
|
3
|
+
const TOTP_DIGITS = 6;
|
|
4
|
+
const TOTP_WINDOW = 1;
|
|
5
|
+
const SECRET_BYTES = 20;
|
|
6
|
+
const RECOVERY_CODE_COUNT = 10;
|
|
7
|
+
const MAX_FAILED_ATTEMPTS = 5;
|
|
8
|
+
const LOCKOUT_DURATION_MS = 15 * 60 * 1000;
|
|
9
|
+
const BASE32_CHARS = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ234567';
|
|
10
|
+
function toBase32(buffer) {
|
|
11
|
+
let bits = '';
|
|
12
|
+
for (const byte of buffer) {
|
|
13
|
+
bits += byte.toString(2).padStart(8, '0');
|
|
14
|
+
}
|
|
15
|
+
let result = '';
|
|
16
|
+
for (let i = 0; i < bits.length; i += 5) {
|
|
17
|
+
const chunk = bits.slice(i, i + 5).padEnd(5, '0');
|
|
18
|
+
result += BASE32_CHARS[parseInt(chunk, 2)];
|
|
19
|
+
}
|
|
20
|
+
return result;
|
|
21
|
+
}
|
|
22
|
+
function fromBase32(encoded) {
|
|
23
|
+
let bits = '';
|
|
24
|
+
for (const char of encoded.toUpperCase()) {
|
|
25
|
+
const idx = BASE32_CHARS.indexOf(char);
|
|
26
|
+
if (idx === -1)
|
|
27
|
+
continue;
|
|
28
|
+
bits += idx.toString(2).padStart(5, '0');
|
|
29
|
+
}
|
|
30
|
+
const bytes = [];
|
|
31
|
+
for (let i = 0; i + 8 <= bits.length; i += 8) {
|
|
32
|
+
bytes.push(parseInt(bits.slice(i, i + 8), 2));
|
|
33
|
+
}
|
|
34
|
+
return Buffer.from(bytes);
|
|
35
|
+
}
|
|
36
|
+
function generateTOTP(secret, time) {
|
|
37
|
+
const counter = Math.floor(time / TOTP_PERIOD);
|
|
38
|
+
const counterBuffer = Buffer.alloc(8);
|
|
39
|
+
counterBuffer.writeUInt32BE(0, 0);
|
|
40
|
+
counterBuffer.writeUInt32BE(counter, 4);
|
|
41
|
+
const hmac = createHmac('sha1', secret).update(counterBuffer).digest();
|
|
42
|
+
const offset = hmac[hmac.length - 1] & 0x0f;
|
|
43
|
+
const code = (((hmac[offset] & 0x7f) << 24) |
|
|
44
|
+
((hmac[offset + 1] & 0xff) << 16) |
|
|
45
|
+
((hmac[offset + 2] & 0xff) << 8) |
|
|
46
|
+
(hmac[offset + 3] & 0xff)) % Math.pow(10, TOTP_DIGITS);
|
|
47
|
+
return code.toString().padStart(TOTP_DIGITS, '0');
|
|
48
|
+
}
|
|
49
|
+
export class MFAProvider {
|
|
50
|
+
eventBus;
|
|
51
|
+
users = new Map();
|
|
52
|
+
constructor(eventBus) {
|
|
53
|
+
this.eventBus = eventBus;
|
|
54
|
+
}
|
|
55
|
+
setup(userId, issuer = 'Genome') {
|
|
56
|
+
const secretBuffer = randomBytes(SECRET_BYTES);
|
|
57
|
+
const secret = toBase32(secretBuffer);
|
|
58
|
+
const recoveryCodes = this.generateRecoveryCodes();
|
|
59
|
+
const state = {
|
|
60
|
+
userId,
|
|
61
|
+
enabled: true,
|
|
62
|
+
secret,
|
|
63
|
+
recoveryCodes,
|
|
64
|
+
usedRecoveryCodes: [],
|
|
65
|
+
failedAttempts: 0,
|
|
66
|
+
};
|
|
67
|
+
this.users.set(userId, state);
|
|
68
|
+
const uri = `otpauth://totp/${encodeURIComponent(issuer)}:${encodeURIComponent(userId)}?secret=${secret}&issuer=${encodeURIComponent(issuer)}&algorithm=SHA1&digits=${TOTP_DIGITS}&period=${TOTP_PERIOD}`;
|
|
69
|
+
this.eventBus.emit({
|
|
70
|
+
type: 'security:audit-entry',
|
|
71
|
+
timestamp: new Date(),
|
|
72
|
+
layer: 3,
|
|
73
|
+
decision: 'info',
|
|
74
|
+
actor: { userId },
|
|
75
|
+
resource: { type: 'mfa', id: 'setup', detail: 'MFA enabled' },
|
|
76
|
+
severity: 'info',
|
|
77
|
+
});
|
|
78
|
+
return { secret, uri, recoveryCodes };
|
|
79
|
+
}
|
|
80
|
+
verify(userId, code) {
|
|
81
|
+
const state = this.users.get(userId);
|
|
82
|
+
if (!state || !state.enabled) {
|
|
83
|
+
return { valid: false, method: 'totp', reason: 'MFA not enabled for this user' };
|
|
84
|
+
}
|
|
85
|
+
if (state.lockedUntil && new Date() < state.lockedUntil) {
|
|
86
|
+
const remaining = Math.ceil((state.lockedUntil.getTime() - Date.now()) / 60000);
|
|
87
|
+
return { valid: false, method: 'totp', reason: `Account locked. Try again in ${remaining} minutes.` };
|
|
88
|
+
}
|
|
89
|
+
if (code.length === TOTP_DIGITS && /^\d+$/.test(code)) {
|
|
90
|
+
const secretBuffer = fromBase32(state.secret);
|
|
91
|
+
const now = Math.floor(Date.now() / 1000);
|
|
92
|
+
for (let i = -TOTP_WINDOW; i <= TOTP_WINDOW; i++) {
|
|
93
|
+
const expected = generateTOTP(secretBuffer, now + i * TOTP_PERIOD);
|
|
94
|
+
if (this.timingSafeEqual(code, expected)) {
|
|
95
|
+
state.failedAttempts = 0;
|
|
96
|
+
state.lockedUntil = undefined;
|
|
97
|
+
state.lastVerifiedAt = new Date();
|
|
98
|
+
this.eventBus.emitAllow('security:audit-entry', 3, {
|
|
99
|
+
type: 'mfa', id: 'totp-verify', detail: 'Success',
|
|
100
|
+
}, { userId });
|
|
101
|
+
return { valid: true, method: 'totp' };
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
if (code.length >= 8) {
|
|
106
|
+
const normalizedCode = code.replace(/-/g, '').toUpperCase();
|
|
107
|
+
const idx = state.recoveryCodes.indexOf(normalizedCode);
|
|
108
|
+
if (idx !== -1) {
|
|
109
|
+
state.recoveryCodes.splice(idx, 1);
|
|
110
|
+
state.usedRecoveryCodes.push(normalizedCode);
|
|
111
|
+
state.failedAttempts = 0;
|
|
112
|
+
state.lockedUntil = undefined;
|
|
113
|
+
state.lastVerifiedAt = new Date();
|
|
114
|
+
this.eventBus.emit({
|
|
115
|
+
type: 'security:audit-entry',
|
|
116
|
+
timestamp: new Date(),
|
|
117
|
+
layer: 3,
|
|
118
|
+
decision: 'info',
|
|
119
|
+
actor: { userId },
|
|
120
|
+
resource: { type: 'mfa', id: 'recovery-code', detail: `Used. ${state.recoveryCodes.length} remaining.` },
|
|
121
|
+
severity: 'warning',
|
|
122
|
+
});
|
|
123
|
+
return { valid: true, method: 'recovery' };
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
state.failedAttempts++;
|
|
127
|
+
if (state.failedAttempts >= MAX_FAILED_ATTEMPTS) {
|
|
128
|
+
state.lockedUntil = new Date(Date.now() + LOCKOUT_DURATION_MS);
|
|
129
|
+
this.eventBus.emitDeny('security:audit-entry', 3, {
|
|
130
|
+
type: 'mfa', id: 'lockout', detail: `${MAX_FAILED_ATTEMPTS} failed attempts — locked for 15 minutes`,
|
|
131
|
+
}, 'high', { userId });
|
|
132
|
+
}
|
|
133
|
+
return { valid: false, method: 'totp', reason: 'Invalid code' };
|
|
134
|
+
}
|
|
135
|
+
disable(userId) {
|
|
136
|
+
const state = this.users.get(userId);
|
|
137
|
+
if (!state)
|
|
138
|
+
return false;
|
|
139
|
+
state.enabled = false;
|
|
140
|
+
return true;
|
|
141
|
+
}
|
|
142
|
+
isEnabled(userId) {
|
|
143
|
+
return this.users.get(userId)?.enabled ?? false;
|
|
144
|
+
}
|
|
145
|
+
getRemainingRecoveryCodes(userId) {
|
|
146
|
+
return this.users.get(userId)?.recoveryCodes.length ?? 0;
|
|
147
|
+
}
|
|
148
|
+
regenerateRecoveryCodes(userId) {
|
|
149
|
+
const state = this.users.get(userId);
|
|
150
|
+
if (!state)
|
|
151
|
+
throw new Error('[MFA] User not found');
|
|
152
|
+
state.recoveryCodes = this.generateRecoveryCodes();
|
|
153
|
+
state.usedRecoveryCodes = [];
|
|
154
|
+
return state.recoveryCodes;
|
|
155
|
+
}
|
|
156
|
+
generateRecoveryCodes() {
|
|
157
|
+
const codes = [];
|
|
158
|
+
for (let i = 0; i < RECOVERY_CODE_COUNT; i++) {
|
|
159
|
+
const code = randomBytes(5).toString('hex').toUpperCase();
|
|
160
|
+
codes.push(code);
|
|
161
|
+
}
|
|
162
|
+
return codes;
|
|
163
|
+
}
|
|
164
|
+
timingSafeEqual(a, b) {
|
|
165
|
+
if (a.length !== b.length)
|
|
166
|
+
return false;
|
|
167
|
+
let result = 0;
|
|
168
|
+
for (let i = 0; i < a.length; i++) {
|
|
169
|
+
result |= a.charCodeAt(i) ^ b.charCodeAt(i);
|
|
170
|
+
}
|
|
171
|
+
return result === 0;
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
//# sourceMappingURL=MFAProvider.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"MFAProvider.js","sourceRoot":"","sources":["../../src/security/MFAProvider.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { SecurityEventBus } from './SecurityEventBus.js';
|
|
2
|
+
export interface NetworkLogEntry {
|
|
3
|
+
timestamp: Date;
|
|
4
|
+
method: string;
|
|
5
|
+
hostname: string;
|
|
6
|
+
port: number;
|
|
7
|
+
path: string;
|
|
8
|
+
requestSize: number;
|
|
9
|
+
responseStatus?: number;
|
|
10
|
+
responseSize?: number;
|
|
11
|
+
durationMs?: number;
|
|
12
|
+
skillId?: string;
|
|
13
|
+
blocked: boolean;
|
|
14
|
+
}
|
|
15
|
+
export interface TrafficSummary {
|
|
16
|
+
totalRequests: number;
|
|
17
|
+
blockedRequests: number;
|
|
18
|
+
byHostname: Record<string, number>;
|
|
19
|
+
byMethod: Record<string, number>;
|
|
20
|
+
avgDurationMs: number;
|
|
21
|
+
totalBytesOut: number;
|
|
22
|
+
totalBytesIn: number;
|
|
23
|
+
}
|
|
24
|
+
export declare class NetworkAuditLogger {
|
|
25
|
+
private entries;
|
|
26
|
+
private maxEntries;
|
|
27
|
+
constructor(eventBus: SecurityEventBus);
|
|
28
|
+
logRequest(entry: Omit<NetworkLogEntry, 'timestamp' | 'blocked'>): void;
|
|
29
|
+
getSummary(since?: Date): TrafficSummary;
|
|
30
|
+
getRecent(limit?: number): NetworkLogEntry[];
|
|
31
|
+
detectAnomalies(windowMs?: number): string[];
|
|
32
|
+
getEntryCount(): number;
|
|
33
|
+
private trimEntries;
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=NetworkAuditLogger.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"NetworkAuditLogger.d.ts","sourceRoot":"","sources":["../../src/security/NetworkAuditLogger.ts"],"names":[],"mappings":"AAWA,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AAIzD,MAAM,WAAW,eAAe;IAC5B,SAAS,EAAE,IAAI,CAAC;IAChB,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,OAAO,CAAC;CACpB;AAED,MAAM,WAAW,cAAc;IAC3B,aAAa,EAAE,MAAM,CAAC;IACtB,eAAe,EAAE,MAAM,CAAC;IACxB,UAAU,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IACnC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IACjC,aAAa,EAAE,MAAM,CAAC;IACtB,aAAa,EAAE,MAAM,CAAC;IACtB,YAAY,EAAE,MAAM,CAAC;CACxB;AA0BD,qBAAa,kBAAkB;IAC3B,OAAO,CAAC,OAAO,CAAyB;IACxC,OAAO,CAAC,UAAU,CAAQ;gBAEd,QAAQ,EAAE,gBAAgB;IAmCtC,UAAU,CAAC,KAAK,EAAE,IAAI,CAAC,eAAe,EAAE,WAAW,GAAG,SAAS,CAAC,GAAG,IAAI;IAcvE,UAAU,CAAC,KAAK,CAAC,EAAE,IAAI,GAAG,cAAc;IAsCxC,SAAS,CAAC,KAAK,SAAK,GAAG,eAAe,EAAE;IAOxC,eAAe,CAAC,QAAQ,SAAU,GAAG,MAAM,EAAE;IA2B7C,aAAa,IAAI,MAAM;IAMvB,OAAO,CAAC,WAAW;CAKtB"}
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
export class NetworkAuditLogger {
|
|
2
|
+
entries = [];
|
|
3
|
+
maxEntries = 5000;
|
|
4
|
+
constructor(eventBus) {
|
|
5
|
+
eventBus.on('security:net-blocked', (event) => {
|
|
6
|
+
this.entries.push({
|
|
7
|
+
timestamp: event.timestamp,
|
|
8
|
+
method: 'BLOCKED',
|
|
9
|
+
hostname: event.resource.id,
|
|
10
|
+
port: 0,
|
|
11
|
+
path: '',
|
|
12
|
+
requestSize: 0,
|
|
13
|
+
blocked: true,
|
|
14
|
+
skillId: event.actor.skillId,
|
|
15
|
+
});
|
|
16
|
+
this.trimEntries();
|
|
17
|
+
});
|
|
18
|
+
eventBus.on('security:net-allowed', (event) => {
|
|
19
|
+
this.entries.push({
|
|
20
|
+
timestamp: event.timestamp,
|
|
21
|
+
method: 'ALLOWED',
|
|
22
|
+
hostname: event.resource.id,
|
|
23
|
+
port: 0,
|
|
24
|
+
path: '',
|
|
25
|
+
requestSize: 0,
|
|
26
|
+
blocked: false,
|
|
27
|
+
skillId: event.actor.skillId,
|
|
28
|
+
});
|
|
29
|
+
this.trimEntries();
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
logRequest(entry) {
|
|
33
|
+
const full = {
|
|
34
|
+
...entry,
|
|
35
|
+
timestamp: new Date(),
|
|
36
|
+
blocked: false,
|
|
37
|
+
};
|
|
38
|
+
this.entries.push(full);
|
|
39
|
+
this.trimEntries();
|
|
40
|
+
}
|
|
41
|
+
getSummary(since) {
|
|
42
|
+
let entries = this.entries;
|
|
43
|
+
if (since) {
|
|
44
|
+
entries = entries.filter(e => e.timestamp >= since);
|
|
45
|
+
}
|
|
46
|
+
const summary = {
|
|
47
|
+
totalRequests: entries.length,
|
|
48
|
+
blockedRequests: entries.filter(e => e.blocked).length,
|
|
49
|
+
byHostname: {},
|
|
50
|
+
byMethod: {},
|
|
51
|
+
avgDurationMs: 0,
|
|
52
|
+
totalBytesOut: 0,
|
|
53
|
+
totalBytesIn: 0,
|
|
54
|
+
};
|
|
55
|
+
let totalDuration = 0;
|
|
56
|
+
let durationCount = 0;
|
|
57
|
+
for (const entry of entries) {
|
|
58
|
+
summary.byHostname[entry.hostname] = (summary.byHostname[entry.hostname] || 0) + 1;
|
|
59
|
+
summary.byMethod[entry.method] = (summary.byMethod[entry.method] || 0) + 1;
|
|
60
|
+
summary.totalBytesOut += entry.requestSize;
|
|
61
|
+
summary.totalBytesIn += entry.responseSize ?? 0;
|
|
62
|
+
if (entry.durationMs) {
|
|
63
|
+
totalDuration += entry.durationMs;
|
|
64
|
+
durationCount++;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
summary.avgDurationMs = durationCount > 0 ? totalDuration / durationCount : 0;
|
|
68
|
+
return summary;
|
|
69
|
+
}
|
|
70
|
+
getRecent(limit = 50) {
|
|
71
|
+
return this.entries.slice(-limit);
|
|
72
|
+
}
|
|
73
|
+
detectAnomalies(windowMs = 300_000) {
|
|
74
|
+
const anomalies = [];
|
|
75
|
+
const now = Date.now();
|
|
76
|
+
const recent = this.entries.filter(e => now - e.timestamp.getTime() < windowMs);
|
|
77
|
+
const hostCounts = {};
|
|
78
|
+
for (const entry of recent) {
|
|
79
|
+
hostCounts[entry.hostname] = (hostCounts[entry.hostname] || 0) + 1;
|
|
80
|
+
}
|
|
81
|
+
const older = this.entries.filter(e => now - e.timestamp.getTime() >= windowMs);
|
|
82
|
+
const oldHosts = new Set(older.map(e => e.hostname));
|
|
83
|
+
for (const [host, count] of Object.entries(hostCounts)) {
|
|
84
|
+
if (!oldHosts.has(host) && count > 5) {
|
|
85
|
+
anomalies.push(`New domain "${host}" with ${count} requests in last ${windowMs / 60000}m`);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
return anomalies;
|
|
89
|
+
}
|
|
90
|
+
getEntryCount() {
|
|
91
|
+
return this.entries.length;
|
|
92
|
+
}
|
|
93
|
+
trimEntries() {
|
|
94
|
+
if (this.entries.length > this.maxEntries) {
|
|
95
|
+
this.entries = this.entries.slice(-this.maxEntries);
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
//# sourceMappingURL=NetworkAuditLogger.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"NetworkAuditLogger.js","sourceRoot":"","sources":["../../src/security/NetworkAuditLogger.ts"],"names":[],"mappings":"AA+DA,MAAM,OAAO,kBAAkB;IACnB,OAAO,GAAsB,EAAE,CAAC;IAChC,UAAU,GAAG,IAAI,CAAC;IAE1B,YAAY,QAA0B;QAGlC,QAAQ,CAAC,EAAE,CAAC,sBAAsB,EAAE,CAAC,KAAK,EAAE,EAAE;YAC1C,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC;gBACd,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,MAAM,EAAE,SAAS;gBACjB,QAAQ,EAAE,KAAK,CAAC,QAAQ,CAAC,EAAE;gBAC3B,IAAI,EAAE,CAAC;gBACP,IAAI,EAAE,EAAE;gBACR,WAAW,EAAE,CAAC;gBACd,OAAO,EAAE,IAAI;gBACb,OAAO,EAAE,KAAK,CAAC,KAAK,CAAC,OAAO;aAC/B,CAAC,CAAC;YACH,IAAI,CAAC,WAAW,EAAE,CAAC;QACvB,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,EAAE,CAAC,sBAAsB,EAAE,CAAC,KAAK,EAAE,EAAE;YAC1C,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC;gBACd,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,MAAM,EAAE,SAAS;gBACjB,QAAQ,EAAE,KAAK,CAAC,QAAQ,CAAC,EAAE;gBAC3B,IAAI,EAAE,CAAC;gBACP,IAAI,EAAE,EAAE;gBACR,WAAW,EAAE,CAAC;gBACd,OAAO,EAAE,KAAK;gBACd,OAAO,EAAE,KAAK,CAAC,KAAK,CAAC,OAAO;aAC/B,CAAC,CAAC;YACH,IAAI,CAAC,WAAW,EAAE,CAAC;QACvB,CAAC,CAAC,CAAC;IACP,CAAC;IAKD,UAAU,CAAC,KAAqD;QAC5D,MAAM,IAAI,GAAoB;YAC1B,GAAG,KAAK;YACR,SAAS,EAAE,IAAI,IAAI,EAAE;YACrB,OAAO,EAAE,KAAK;SACjB,CAAC;QAEF,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACxB,IAAI,CAAC,WAAW,EAAE,CAAC;IACvB,CAAC;IAKD,UAAU,CAAC,KAAY;QACnB,IAAI,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC;QAC3B,IAAI,KAAK,EAAE,CAAC;YACR,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,IAAI,KAAK,CAAC,CAAC;QACxD,CAAC;QAED,MAAM,OAAO,GAAmB;YAC5B,aAAa,EAAE,OAAO,CAAC,MAAM;YAC7B,eAAe,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,MAAM;YACtD,UAAU,EAAE,EAAE;YACd,QAAQ,EAAE,EAAE;YACZ,aAAa,EAAE,CAAC;YAChB,aAAa,EAAE,CAAC;YAChB,YAAY,EAAE,CAAC;SAClB,CAAC;QAEF,IAAI,aAAa,GAAG,CAAC,CAAC;QACtB,IAAI,aAAa,GAAG,CAAC,CAAC;QAEtB,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;YAC1B,OAAO,CAAC,UAAU,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,UAAU,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;YACnF,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;YAC3E,OAAO,CAAC,aAAa,IAAI,KAAK,CAAC,WAAW,CAAC;YAC3C,OAAO,CAAC,YAAY,IAAI,KAAK,CAAC,YAAY,IAAI,CAAC,CAAC;YAChD,IAAI,KAAK,CAAC,UAAU,EAAE,CAAC;gBACnB,aAAa,IAAI,KAAK,CAAC,UAAU,CAAC;gBAClC,aAAa,EAAE,CAAC;YACpB,CAAC;QACL,CAAC;QAED,OAAO,CAAC,aAAa,GAAG,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC,aAAa,GAAG,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;QAE9E,OAAO,OAAO,CAAC;IACnB,CAAC;IAKD,SAAS,CAAC,KAAK,GAAG,EAAE;QAChB,OAAO,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,CAAC;IACtC,CAAC;IAKD,eAAe,CAAC,QAAQ,GAAG,OAAO;QAC9B,MAAM,SAAS,GAAa,EAAE,CAAC;QAC/B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,SAAS,CAAC,OAAO,EAAE,GAAG,QAAQ,CAAC,CAAC;QAGhF,MAAM,UAAU,GAA2B,EAAE,CAAC;QAC9C,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;YACzB,UAAU,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,UAAU,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;QACvE,CAAC;QAGD,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,QAAQ,CAAC,CAAC;QAChF,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC;QAErD,KAAK,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC;YACrD,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,KAAK,GAAG,CAAC,EAAE,CAAC;gBACnC,SAAS,CAAC,IAAI,CAAC,eAAe,IAAI,UAAU,KAAK,qBAAqB,QAAQ,GAAG,KAAK,GAAG,CAAC,CAAC;YAC/F,CAAC;QACL,CAAC;QAED,OAAO,SAAS,CAAC;IACrB,CAAC;IAKD,aAAa;QACT,OAAO,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC;IAC/B,CAAC;IAIO,WAAW;QACf,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;YACxC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACxD,CAAC;IACL,CAAC;CACJ"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { SecurityEventBus } from './SecurityEventBus.js';
|
|
2
|
+
export interface OutboundCheckResult {
|
|
3
|
+
allowed: boolean;
|
|
4
|
+
reason?: string;
|
|
5
|
+
hostname: string;
|
|
6
|
+
}
|
|
7
|
+
export interface OutboundAllowlistConfig {
|
|
8
|
+
allowedDomains: string[];
|
|
9
|
+
blockPrivateNetworks: boolean;
|
|
10
|
+
mode: 'strict' | 'broad' | 'unrestricted';
|
|
11
|
+
}
|
|
12
|
+
export declare class OutboundAllowlist {
|
|
13
|
+
private eventBus;
|
|
14
|
+
private allowedDomains;
|
|
15
|
+
private blockPrivateNetworks;
|
|
16
|
+
private mode;
|
|
17
|
+
private stats;
|
|
18
|
+
constructor(eventBus: SecurityEventBus, config: OutboundAllowlistConfig);
|
|
19
|
+
check(hostname: string, skillId?: string): OutboundCheckResult;
|
|
20
|
+
checkURL(url: string, skillId?: string): OutboundCheckResult;
|
|
21
|
+
addDomain(domain: string): void;
|
|
22
|
+
removeDomain(domain: string): boolean;
|
|
23
|
+
getDomains(): string[];
|
|
24
|
+
getStats(): {
|
|
25
|
+
totalChecks: number;
|
|
26
|
+
allowed: number;
|
|
27
|
+
blocked: number;
|
|
28
|
+
};
|
|
29
|
+
private isPrivateNetwork;
|
|
30
|
+
private isDomainAllowed;
|
|
31
|
+
private isSuspiciousDomain;
|
|
32
|
+
}
|
|
33
|
+
//# sourceMappingURL=OutboundAllowlist.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"OutboundAllowlist.d.ts","sourceRoot":"","sources":["../../src/security/OutboundAllowlist.ts"],"names":[],"mappings":"AAWA,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AAIzD,MAAM,WAAW,mBAAmB;IAChC,OAAO,EAAE,OAAO,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,WAAW,uBAAuB;IAEpC,cAAc,EAAE,MAAM,EAAE,CAAC;IAEzB,oBAAoB,EAAE,OAAO,CAAC;IAE9B,IAAI,EAAE,QAAQ,GAAG,OAAO,GAAG,cAAc,CAAC;CAC7C;AAwCD,qBAAa,iBAAiB;IAC1B,OAAO,CAAC,QAAQ,CAAmB;IACnC,OAAO,CAAC,cAAc,CAAW;IACjC,OAAO,CAAC,oBAAoB,CAAU;IACtC,OAAO,CAAC,IAAI,CAAkC;IAC9C,OAAO,CAAC,KAAK,CAA8C;gBAE/C,QAAQ,EAAE,gBAAgB,EAAE,MAAM,EAAE,uBAAuB;IAUvE,KAAK,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,mBAAmB;IA4D9D,QAAQ,CAAC,GAAG,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,mBAAmB;IAY5D,SAAS,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI;IAS/B,YAAY,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO;IAUrC,UAAU,IAAI,MAAM,EAAE;IAOtB,QAAQ;;;;;IAMR,OAAO,CAAC,gBAAgB;IAKxB,OAAO,CAAC,eAAe;IAavB,OAAO,CAAC,kBAAkB;CAU7B"}
|