Implement intelligent agent learning from Knowledge Graph execution history with per-task-type expertise tracking, recency bias, and learning curves. ## Phase 5.3 Implementation ### Learning Infrastructure (✅ Complete) - LearningProfileService with per-task-type expertise metrics - TaskTypeExpertise model tracking success_rate, confidence, learning curves - Recency bias weighting: recent 7 days weighted 3x higher (exponential decay) - Confidence scoring prevents overfitting: min(1.0, executions / 20) - Learning curves computed from daily execution windows ### Agent Scoring Service (✅ Complete) - Unified AgentScore combining SwarmCoordinator + learning profiles - Scoring formula: 0.3*base + 0.5*expertise + 0.2*confidence - Rank agents by combined score for intelligent assignment - Support for recency-biased scoring (recent_success_rate) - Methods: rank_agents, select_best, rank_agents_with_recency ### KG Integration (✅ Complete) - KGPersistence::get_executions_for_task_type() - query by agent + task type - KGPersistence::get_agent_executions() - all executions for agent - Coordinator::load_learning_profile_from_kg() - core KG→Learning integration - Coordinator::load_all_learning_profiles() - batch load for multiple agents - Convert PersistedExecution → ExecutionData for learning calculations ### Agent Assignment Integration (✅ Complete) - AgentCoordinator uses learning profiles for task assignment - extract_task_type() infers task type from title/description - assign_task() scores candidates using AgentScoringService - Fallback to load-based selection if no learning data available - Learning profiles stored in coordinator.learning_profiles RwLock ### Profile Adapter Enhancements (✅ Complete) - create_learning_profile() - initialize empty profiles - add_task_type_expertise() - set task-type expertise - update_profile_with_learning() - update swarm profiles from learning ## Files Modified ### vapora-knowledge-graph/src/persistence.rs (+30 lines) - get_executions_for_task_type(agent_id, task_type, limit) - get_agent_executions(agent_id, limit) ### vapora-agents/src/coordinator.rs (+100 lines) - load_learning_profile_from_kg() - core KG integration method - load_all_learning_profiles() - batch loading for agents - assign_task() already uses learning-based scoring via AgentScoringService ### Existing Complete Implementation - vapora-knowledge-graph/src/learning.rs - calculation functions - vapora-agents/src/learning_profile.rs - data structures and expertise - vapora-agents/src/scoring.rs - unified scoring service - vapora-agents/src/profile_adapter.rs - adapter methods ## Tests Passing - learning_profile: 7 tests ✅ - scoring: 5 tests ✅ - profile_adapter: 6 tests ✅ - coordinator: learning-specific tests ✅ ## Data Flow 1. Task arrives → AgentCoordinator::assign_task() 2. Extract task_type from description 3. Query KG for task-type executions (load_learning_profile_from_kg) 4. Calculate expertise with recency bias 5. Score candidates (SwarmCoordinator + learning) 6. Assign to top-scored agent 7. Execution result → KG → Update learning profiles ## Key Design Decisions ✅ Recency bias: 7-day half-life with 3x weight for recent performance ✅ Confidence scoring: min(1.0, total_executions / 20) prevents overfitting ✅ Hierarchical scoring: 30% base load, 50% expertise, 20% confidence ✅ KG query limit: 100 recent executions per task-type for performance ✅ Async loading: load_learning_profile_from_kg supports concurrent loads ## Next: Phase 5.4 - Cost Optimization Ready to implement budget enforcement and cost-aware provider selection.
215 lines
8.3 KiB
XML
215 lines
8.3 KiB
XML
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 600 250" width="600" height="250">
|
|
<defs>
|
|
<!-- Gradiente principal cyan -> purple -> pink -->
|
|
<linearGradient id="vaporGrad" x1="0%" y1="0%" x2="100%" y2="100%">
|
|
<stop offset="0%" style="stop-color:#22d3ee;stop-opacity:1" />
|
|
<stop offset="50%" style="stop-color:#a855f7;stop-opacity:1" />
|
|
<stop offset="100%" style="stop-color:#ec4899;stop-opacity:1" />
|
|
</linearGradient>
|
|
|
|
<!-- Gradiente vertical para ondas -->
|
|
<linearGradient id="waveGrad" x1="0%" y1="100%" x2="0%" y2="0%">
|
|
<stop offset="0%" style="stop-color:#22d3ee;stop-opacity:0.9" />
|
|
<stop offset="50%" style="stop-color:#a855f7;stop-opacity:0.6" />
|
|
<stop offset="100%" style="stop-color:#ec4899;stop-opacity:0.2" />
|
|
</linearGradient>
|
|
|
|
<!-- Gradiente radial para el núcleo -->
|
|
<radialGradient id="coreGrad">
|
|
<stop offset="0%" style="stop-color:#22d3ee;stop-opacity:1" />
|
|
<stop offset="50%" style="stop-color:#a855f7;stop-opacity:0.8" />
|
|
<stop offset="100%" style="stop-color:#ec4899;stop-opacity:0.4" />
|
|
</radialGradient>
|
|
|
|
<!-- Filtro glow intenso -->
|
|
<filter id="strongGlow">
|
|
<feGaussianBlur stdDeviation="4" result="coloredBlur"/>
|
|
<feMerge>
|
|
<feMergeNode in="coloredBlur"/>
|
|
<feMergeNode in="coloredBlur"/>
|
|
<feMergeNode in="SourceGraphic"/>
|
|
</feMerge>
|
|
</filter>
|
|
|
|
<!-- Filtro glass -->
|
|
<filter id="glassBlur">
|
|
<feGaussianBlur in="SourceGraphic" stdDeviation="0.8" result="blur" />
|
|
<feColorMatrix in="blur" type="matrix" values="1 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 18 -7" result="goo" />
|
|
<feBlend in="SourceGraphic" in2="goo" />
|
|
</filter>
|
|
</defs>
|
|
|
|
<!-- Fondo -->
|
|
<rect width="600" height="250" fill="#000000"/>
|
|
|
|
<!-- Símbolo de vapor: núcleo brillante + ondas ascendentes -->
|
|
<g transform="translate(100, 125)">
|
|
<!-- Núcleo central brillante -->
|
|
<circle cx="0" cy="0" r="12" fill="url(#coreGrad)" filter="url(#strongGlow)" opacity="0.9">
|
|
<animate attributeName="r" values="12;14;12" dur="2s" repeatCount="indefinite"/>
|
|
<animate attributeName="opacity" values="0.9;1;0.9" dur="2s" repeatCount="indefinite"/>
|
|
</circle>
|
|
|
|
<circle cx="0" cy="0" r="8" fill="#22d3ee" opacity="0.6">
|
|
<animate attributeName="r" values="8;10;8" dur="2s" repeatCount="indefinite"/>
|
|
</circle>
|
|
|
|
<!-- Ondas de vapor 1 - izquierda -->
|
|
<path d="M -25 0 Q -30 -15, -25 -30 Q -20 -45, -25 -60"
|
|
stroke="url(#waveGrad)"
|
|
stroke-width="4"
|
|
fill="none"
|
|
stroke-linecap="round"
|
|
opacity="0.8">
|
|
<animate attributeName="d"
|
|
values="M -25 0 Q -30 -15, -25 -30 Q -20 -45, -25 -60;
|
|
M -25 0 Q -32 -15, -25 -30 Q -18 -45, -25 -60;
|
|
M -25 0 Q -30 -15, -25 -30 Q -20 -45, -25 -60"
|
|
dur="3s"
|
|
repeatCount="indefinite"/>
|
|
<animate attributeName="opacity" values="0.8;0.5;0.8" dur="3s" repeatCount="indefinite"/>
|
|
</path>
|
|
|
|
<!-- Ondas de vapor 2 - izquierda interna -->
|
|
<path d="M -12 0 Q -15 -12, -12 -24 Q -9 -36, -12 -48"
|
|
stroke="url(#waveGrad)"
|
|
stroke-width="3.5"
|
|
fill="none"
|
|
stroke-linecap="round"
|
|
opacity="0.7">
|
|
<animate attributeName="d"
|
|
values="M -12 0 Q -15 -12, -12 -24 Q -9 -36, -12 -48;
|
|
M -12 0 Q -16 -12, -12 -24 Q -8 -36, -12 -48;
|
|
M -12 0 Q -15 -12, -12 -24 Q -9 -36, -12 -48"
|
|
dur="2.5s"
|
|
repeatCount="indefinite"/>
|
|
</path>
|
|
|
|
<!-- Ondas de vapor 3 - derecha -->
|
|
<path d="M 25 0 Q 30 -15, 25 -30 Q 20 -45, 25 -60"
|
|
stroke="url(#waveGrad)"
|
|
stroke-width="4"
|
|
fill="none"
|
|
stroke-linecap="round"
|
|
opacity="0.8">
|
|
<animate attributeName="d"
|
|
values="M 25 0 Q 30 -15, 25 -30 Q 20 -45, 25 -60;
|
|
M 25 0 Q 28 -15, 25 -30 Q 22 -45, 25 -60;
|
|
M 25 0 Q 30 -15, 25 -30 Q 20 -45, 25 -60"
|
|
dur="3.2s"
|
|
repeatCount="indefinite"/>
|
|
<animate attributeName="opacity" values="0.8;0.5;0.8" dur="3.2s" repeatCount="indefinite"/>
|
|
</path>
|
|
|
|
<!-- Ondas de vapor 4 - derecha interna -->
|
|
<path d="M 12 0 Q 15 -12, 12 -24 Q 9 -36, 12 -48"
|
|
stroke="url(#waveGrad)"
|
|
stroke-width="3.5"
|
|
fill="none"
|
|
stroke-linecap="round"
|
|
opacity="0.7">
|
|
<animate attributeName="d"
|
|
values="M 12 0 Q 15 -12, 12 -24 Q 9 -36, 12 -48;
|
|
M 12 0 Q 16 -12, 12 -24 Q 8 -36, 12 -48;
|
|
M 12 0 Q 15 -12, 12 -24 Q 9 -36, 12 -48"
|
|
dur="2.8s"
|
|
repeatCount="indefinite"/>
|
|
</path>
|
|
|
|
<!-- Onda central -->
|
|
<path d="M 0 0 Q 0 -20, 0 -40 Q 0 -50, 0 -65"
|
|
stroke="url(#waveGrad)"
|
|
stroke-width="3"
|
|
fill="none"
|
|
stroke-linecap="round"
|
|
opacity="0.6">
|
|
<animate attributeName="d"
|
|
values="M 0 0 Q 0 -20, 0 -40 Q 0 -50, 0 -65;
|
|
M 0 0 Q -2 -20, 0 -40 Q 2 -50, 0 -65;
|
|
M 0 0 Q 0 -20, 0 -40 Q 0 -50, 0 -65"
|
|
dur="2.2s"
|
|
repeatCount="indefinite"/>
|
|
</path>
|
|
|
|
<!-- Partículas ascendentes -->
|
|
<circle cx="-18" cy="-20" r="2.5" fill="#22d3ee" opacity="0.8">
|
|
<animate attributeName="cy" values="-20;-70;-20" dur="4s" repeatCount="indefinite"/>
|
|
<animate attributeName="opacity" values="0;0.8;0" dur="4s" repeatCount="indefinite"/>
|
|
<animate attributeName="r" values="2.5;1.5;2.5" dur="4s" repeatCount="indefinite"/>
|
|
</circle>
|
|
|
|
<circle cx="18" cy="-25" r="2" fill="#ec4899" opacity="0.7">
|
|
<animate attributeName="cy" values="-25;-75;-25" dur="4.5s" repeatCount="indefinite"/>
|
|
<animate attributeName="opacity" values="0;0.7;0" dur="4.5s" repeatCount="indefinite"/>
|
|
<animate attributeName="r" values="2;1;2" dur="4.5s" repeatCount="indefinite"/>
|
|
</circle>
|
|
|
|
<circle cx="-8" cy="-15" r="1.8" fill="#a855f7" opacity="0.6">
|
|
<animate attributeName="cy" values="-15;-65;-15" dur="3.8s" repeatCount="indefinite"/>
|
|
<animate attributeName="opacity" values="0;0.6;0" dur="3.8s" repeatCount="indefinite"/>
|
|
</circle>
|
|
|
|
<circle cx="8" cy="-18" r="1.8" fill="#22d3ee" opacity="0.6">
|
|
<animate attributeName="cy" values="-18;-68;-18" dur="4.2s" repeatCount="indefinite"/>
|
|
<animate attributeName="opacity" values="0;0.6;0" dur="4.2s" repeatCount="indefinite"/>
|
|
</circle>
|
|
|
|
<circle cx="0" cy="-22" r="1.5" fill="#a855f7" opacity="0.5">
|
|
<animate attributeName="cy" values="-22;-72;-22" dur="3.5s" repeatCount="indefinite"/>
|
|
<animate attributeName="opacity" values="0;0.5;0" dur="3.5s" repeatCount="indefinite"/>
|
|
</circle>
|
|
</g>
|
|
|
|
<!-- Texto VAPORA con glassmorphism -->
|
|
<g filter="url(#glassBlur)">
|
|
<text x="230" y="135"
|
|
font-family="'JetBrains Mono', 'Fira Code', monospace"
|
|
font-size="56"
|
|
font-weight="800"
|
|
fill="url(#vaporGrad)"
|
|
letter-spacing="3">
|
|
VAPORA
|
|
</text>
|
|
|
|
<!-- Capa de brillo -->
|
|
<text x="230" y="135"
|
|
font-family="'JetBrains Mono', 'Fira Code', monospace"
|
|
font-size="56"
|
|
font-weight="800"
|
|
fill="none"
|
|
stroke="rgba(255,255,255,0.2)"
|
|
stroke-width="0.5"
|
|
letter-spacing="3">
|
|
VAPORA
|
|
</text>
|
|
</g>
|
|
|
|
<!-- Glow adicional al texto -->
|
|
<text x="230" y="135"
|
|
font-family="'JetBrains Mono', 'Fira Code', monospace"
|
|
font-size="56"
|
|
font-weight="800"
|
|
fill="url(#vaporGrad)"
|
|
letter-spacing="3"
|
|
filter="url(#strongGlow)"
|
|
opacity="0.3">
|
|
VAPORA
|
|
</text>
|
|
|
|
<!-- Tagline -->
|
|
<text x="230" y="160"
|
|
font-family="'Inter', sans-serif"
|
|
font-size="13"
|
|
fill="#a855f7"
|
|
opacity="0.75"
|
|
letter-spacing="1.5">
|
|
Evaporate complexity. Build in the cloud.
|
|
</text>
|
|
|
|
<!-- Punto decorativo animado -->
|
|
<circle cx="215" cy="125" r="3" fill="#22d3ee" opacity="0.6">
|
|
<animate attributeName="r" values="3;4.5;3" dur="2.5s" repeatCount="indefinite"/>
|
|
<animate attributeName="opacity" values="0.6;1;0.6" dur="2.5s" repeatCount="indefinite"/>
|
|
</circle>
|
|
</svg>
|