Vapora/assets/wrks/vapora-logo-hybrid.svg
Jesús Pérez d14150da75 feat: Phase 5.3 - Multi-Agent Learning Infrastructure
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.
2026-01-11 13:03:53 +00:00

225 lines
8.4 KiB
XML

<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 600 250" width="600" height="250">
<defs>
<!-- Gradiente principal -->
<linearGradient id="mainGrad" 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 para ondas suaves -->
<linearGradient id="smoothWave" x1="0%" y1="100%" x2="0%" y2="0%">
<stop offset="0%" style="stop-color:#22d3ee;stop-opacity:0.8" />
<stop offset="50%" style="stop-color:#a855f7;stop-opacity:0.5" />
<stop offset="100%" style="stop-color:#ec4899;stop-opacity:0.2" />
</linearGradient>
<!-- Gradiente radial para núcleo -->
<radialGradient id="coreGlow">
<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.3" />
</radialGradient>
<!-- Filtro glow suave -->
<filter id="softGlow">
<feGaussianBlur stdDeviation="3" result="coloredBlur"/>
<feMerge>
<feMergeNode in="coloredBlur"/>
<feMergeNode in="SourceGraphic"/>
</feMerge>
</filter>
<!-- Filtro glow intenso -->
<filter id="intensGlow">
<feGaussianBlur stdDeviation="5" result="coloredBlur"/>
<feMerge>
<feMergeNode in="coloredBlur"/>
<feMergeNode in="coloredBlur"/>
<feMergeNode in="SourceGraphic"/>
</feMerge>
</filter>
<!-- Filtro glassmorphism -->
<filter id="glass">
<feGaussianBlur in="SourceGraphic" stdDeviation="0.5" 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 híbrido -->
<g transform="translate(100, 140)">
<!-- Núcleo central brillante (de v3) -->
<circle cx="0" cy="0" r="10" fill="url(#coreGlow)" filter="url(#intensGlow)" opacity="0.95">
<animate attributeName="r" values="10;12;10" dur="2s" repeatCount="indefinite"/>
<animate attributeName="opacity" values="0.95;1;0.95" dur="2s" repeatCount="indefinite"/>
</circle>
<!-- Ondas suaves estilo v1 pero mejoradas -->
<!-- Onda curva izquierda -->
<path d="M -20 0 Q -25 -20, -20 -40 T -20 -60"
stroke="url(#smoothWave)"
stroke-width="3.5"
fill="none"
stroke-linecap="round"
filter="url(#softGlow)"
opacity="0.8">
<animate attributeName="d"
values="M -20 0 Q -25 -20, -20 -40 T -20 -60;
M -20 0 Q -28 -20, -20 -40 T -20 -60;
M -20 0 Q -25 -20, -20 -40 T -20 -60"
dur="3s"
repeatCount="indefinite"/>
</path>
<!-- Onda curva derecha -->
<path d="M 20 0 Q 25 -20, 20 -40 T 20 -60"
stroke="url(#smoothWave)"
stroke-width="3.5"
fill="none"
stroke-linecap="round"
filter="url(#softGlow)"
opacity="0.8">
<animate attributeName="d"
values="M 20 0 Q 25 -20, 20 -40 T 20 -60;
M 20 0 Q 28 -20, 20 -40 T 20 -60;
M 20 0 Q 25 -20, 20 -40 T 20 -60"
dur="3.2s"
repeatCount="indefinite"/>
</path>
<!-- Onda central sutil -->
<path d="M 0 -5 Q -5 -25, 0 -45 T 0 -65"
stroke="url(#smoothWave)"
stroke-width="2.5"
fill="none"
stroke-linecap="round"
opacity="0.6">
<animate attributeName="d"
values="M 0 -5 Q -5 -25, 0 -45 T 0 -65;
M 0 -5 Q 5 -25, 0 -45 T 0 -65;
M 0 -5 Q -5 -25, 0 -45 T 0 -65"
dur="2.5s"
repeatCount="indefinite"/>
</path>
<!-- Forma abstracta adicional (estilo v1) -->
<path d="M -10 -5 Q -15 -15, -10 -25 Q -5 -35, -10 -45"
stroke="url(#smoothWave)"
stroke-width="2"
fill="none"
stroke-linecap="round"
opacity="0.5">
<animate attributeName="d"
values="M -10 -5 Q -15 -15, -10 -25 Q -5 -35, -10 -45;
M -10 -5 Q -13 -15, -10 -25 Q -7 -35, -10 -45;
M -10 -5 Q -15 -15, -10 -25 Q -5 -35, -10 -45"
dur="2.8s"
repeatCount="indefinite"/>
</path>
<path d="M 10 -5 Q 15 -15, 10 -25 Q 5 -35, 10 -45"
stroke="url(#smoothWave)"
stroke-width="2"
fill="none"
stroke-linecap="round"
opacity="0.5">
<animate attributeName="d"
values="M 10 -5 Q 15 -15, 10 -25 Q 5 -35, 10 -45;
M 10 -5 Q 13 -15, 10 -25 Q 7 -35, 10 -45;
M 10 -5 Q 15 -15, 10 -25 Q 5 -35, 10 -45"
dur="3.3s"
repeatCount="indefinite"/>
</path>
<!-- Partículas flotantes mejoradas -->
<circle cx="-15" 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="15" 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="-5" 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="-30" r="1.5" fill="#a855f7" opacity="0.5">
<animate attributeName="cy" values="-30;-80;-30" dur="3.5s" repeatCount="indefinite"/>
<animate attributeName="opacity" values="0;0.5;0" dur="3.5s" repeatCount="indefinite"/>
</circle>
<circle cx="-8" cy="-35" r="1.2" fill="#ec4899" opacity="0.4">
<animate attributeName="cy" values="-35;-75;-35" dur="4.8s" repeatCount="indefinite"/>
<animate attributeName="opacity" values="0;0.4;0" dur="4.8s" repeatCount="indefinite"/>
</circle>
</g>
<!-- Texto VAPORA con glassmorphism -->
<g filter="url(#glass)">
<text x="220" y="135"
font-family="'JetBrains Mono', 'Fira Code', monospace"
font-size="58"
font-weight="800"
fill="url(#mainGrad)"
letter-spacing="3">
VAPORA
</text>
<!-- Capa de brillo glassmorphism -->
<text x="220" y="135"
font-family="'JetBrains Mono', 'Fira Code', monospace"
font-size="58"
font-weight="800"
fill="none"
stroke="rgba(255,255,255,0.25)"
stroke-width="0.5"
letter-spacing="3">
VAPORA
</text>
</g>
<!-- Glow sutil adicional -->
<text x="220" y="135"
font-family="'JetBrains Mono', 'Fira Code', monospace"
font-size="58"
font-weight="800"
fill="url(#mainGrad)"
letter-spacing="3"
filter="url(#softGlow)"
opacity="0.25">
VAPORA
</text>
<!-- Tagline -->
<text x="220" y="162"
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 pulsante -->
<circle cx="205" 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>