@claude-flow/cli 3.0.0-alpha.32 → 3.0.0-alpha.34
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/.claude/agents/core/architect.yaml +11 -0
- package/.claude/agents/core/coder.yaml +11 -0
- package/.claude/agents/core/reviewer.yaml +10 -0
- package/.claude/agents/core/security-architect.yaml +10 -0
- package/.claude/agents/core/tester.yaml +10 -0
- package/.claude/commands/agents/README.md +10 -0
- package/.claude/commands/agents/agent-capabilities.md +21 -0
- package/.claude/commands/agents/agent-coordination.md +28 -0
- package/.claude/commands/agents/agent-spawning.md +28 -0
- package/.claude/commands/agents/agent-types.md +26 -0
- package/.claude/commands/analysis/COMMAND_COMPLIANCE_REPORT.md +54 -0
- package/.claude/commands/analysis/README.md +9 -0
- package/.claude/commands/analysis/bottleneck-detect.md +162 -0
- package/.claude/commands/analysis/performance-bottlenecks.md +59 -0
- package/.claude/commands/analysis/performance-report.md +25 -0
- package/.claude/commands/analysis/token-efficiency.md +45 -0
- package/.claude/commands/analysis/token-usage.md +25 -0
- package/.claude/commands/automation/README.md +9 -0
- package/.claude/commands/automation/auto-agent.md +122 -0
- package/.claude/commands/automation/self-healing.md +106 -0
- package/.claude/commands/automation/session-memory.md +90 -0
- package/.claude/commands/automation/smart-agents.md +73 -0
- package/.claude/commands/automation/smart-spawn.md +25 -0
- package/.claude/commands/automation/workflow-select.md +25 -0
- package/.claude/commands/claude-flow-help.md +103 -0
- package/.claude/commands/claude-flow-memory.md +107 -0
- package/.claude/commands/claude-flow-swarm.md +205 -0
- package/.claude/commands/coordination/README.md +9 -0
- package/.claude/commands/coordination/agent-spawn.md +25 -0
- package/.claude/commands/coordination/init.md +44 -0
- package/.claude/commands/coordination/orchestrate.md +43 -0
- package/.claude/commands/coordination/spawn.md +45 -0
- package/.claude/commands/coordination/swarm-init.md +85 -0
- package/.claude/commands/coordination/task-orchestrate.md +25 -0
- package/.claude/commands/flow-nexus/app-store.md +124 -0
- package/.claude/commands/flow-nexus/challenges.md +120 -0
- package/.claude/commands/flow-nexus/login-registration.md +65 -0
- package/.claude/commands/flow-nexus/neural-network.md +134 -0
- package/.claude/commands/flow-nexus/payments.md +116 -0
- package/.claude/commands/flow-nexus/sandbox.md +83 -0
- package/.claude/commands/flow-nexus/swarm.md +87 -0
- package/.claude/commands/flow-nexus/user-tools.md +152 -0
- package/.claude/commands/flow-nexus/workflow.md +115 -0
- package/.claude/commands/github/README.md +11 -0
- package/.claude/commands/github/code-review-swarm.md +514 -0
- package/.claude/commands/github/code-review.md +25 -0
- package/.claude/commands/github/github-modes.md +147 -0
- package/.claude/commands/github/github-swarm.md +121 -0
- package/.claude/commands/github/issue-tracker.md +292 -0
- package/.claude/commands/github/issue-triage.md +25 -0
- package/.claude/commands/github/multi-repo-swarm.md +519 -0
- package/.claude/commands/github/pr-enhance.md +26 -0
- package/.claude/commands/github/pr-manager.md +170 -0
- package/.claude/commands/github/project-board-sync.md +471 -0
- package/.claude/commands/github/release-manager.md +338 -0
- package/.claude/commands/github/release-swarm.md +544 -0
- package/.claude/commands/github/repo-analyze.md +25 -0
- package/.claude/commands/github/repo-architect.md +367 -0
- package/.claude/commands/github/swarm-issue.md +482 -0
- package/.claude/commands/github/swarm-pr.md +285 -0
- package/.claude/commands/github/sync-coordinator.md +301 -0
- package/.claude/commands/github/workflow-automation.md +442 -0
- package/.claude/commands/hive-mind/README.md +17 -0
- package/.claude/commands/hive-mind/hive-mind-consensus.md +8 -0
- package/.claude/commands/hive-mind/hive-mind-init.md +18 -0
- package/.claude/commands/hive-mind/hive-mind-memory.md +8 -0
- package/.claude/commands/hive-mind/hive-mind-metrics.md +8 -0
- package/.claude/commands/hive-mind/hive-mind-resume.md +8 -0
- package/.claude/commands/hive-mind/hive-mind-sessions.md +8 -0
- package/.claude/commands/hive-mind/hive-mind-spawn.md +21 -0
- package/.claude/commands/hive-mind/hive-mind-status.md +8 -0
- package/.claude/commands/hive-mind/hive-mind-stop.md +8 -0
- package/.claude/commands/hive-mind/hive-mind-wizard.md +8 -0
- package/.claude/commands/hive-mind/hive-mind.md +27 -0
- package/.claude/commands/hooks/README.md +11 -0
- package/.claude/commands/hooks/overview.md +58 -0
- package/.claude/commands/hooks/post-edit.md +117 -0
- package/.claude/commands/hooks/post-task.md +112 -0
- package/.claude/commands/hooks/pre-edit.md +113 -0
- package/.claude/commands/hooks/pre-task.md +111 -0
- package/.claude/commands/hooks/session-end.md +118 -0
- package/.claude/commands/hooks/setup.md +103 -0
- package/.claude/commands/memory/README.md +9 -0
- package/.claude/commands/memory/memory-persist.md +25 -0
- package/.claude/commands/memory/memory-search.md +25 -0
- package/.claude/commands/memory/memory-usage.md +25 -0
- package/.claude/commands/memory/neural.md +47 -0
- package/.claude/commands/monitoring/README.md +9 -0
- package/.claude/commands/monitoring/agent-metrics.md +25 -0
- package/.claude/commands/monitoring/agents.md +44 -0
- package/.claude/commands/monitoring/real-time-view.md +25 -0
- package/.claude/commands/monitoring/status.md +46 -0
- package/.claude/commands/monitoring/swarm-monitor.md +25 -0
- package/.claude/commands/optimization/README.md +9 -0
- package/.claude/commands/optimization/auto-topology.md +62 -0
- package/.claude/commands/optimization/cache-manage.md +25 -0
- package/.claude/commands/optimization/parallel-execute.md +25 -0
- package/.claude/commands/optimization/parallel-execution.md +50 -0
- package/.claude/commands/optimization/topology-optimize.md +25 -0
- package/.claude/commands/pair/README.md +261 -0
- package/.claude/commands/pair/commands.md +546 -0
- package/.claude/commands/pair/config.md +510 -0
- package/.claude/commands/pair/examples.md +512 -0
- package/.claude/commands/pair/modes.md +348 -0
- package/.claude/commands/pair/session.md +407 -0
- package/.claude/commands/pair/start.md +209 -0
- package/.claude/commands/sparc/analyzer.md +52 -0
- package/.claude/commands/sparc/architect.md +53 -0
- package/.claude/commands/sparc/ask.md +97 -0
- package/.claude/commands/sparc/batch-executor.md +54 -0
- package/.claude/commands/sparc/code.md +89 -0
- package/.claude/commands/sparc/coder.md +54 -0
- package/.claude/commands/sparc/debug.md +83 -0
- package/.claude/commands/sparc/debugger.md +54 -0
- package/.claude/commands/sparc/designer.md +53 -0
- package/.claude/commands/sparc/devops.md +109 -0
- package/.claude/commands/sparc/docs-writer.md +80 -0
- package/.claude/commands/sparc/documenter.md +54 -0
- package/.claude/commands/sparc/innovator.md +54 -0
- package/.claude/commands/sparc/integration.md +83 -0
- package/.claude/commands/sparc/mcp.md +117 -0
- package/.claude/commands/sparc/memory-manager.md +54 -0
- package/.claude/commands/sparc/optimizer.md +54 -0
- package/.claude/commands/sparc/orchestrator.md +132 -0
- package/.claude/commands/sparc/post-deployment-monitoring-mode.md +83 -0
- package/.claude/commands/sparc/refinement-optimization-mode.md +83 -0
- package/.claude/commands/sparc/researcher.md +54 -0
- package/.claude/commands/sparc/reviewer.md +54 -0
- package/.claude/commands/sparc/security-review.md +80 -0
- package/.claude/commands/sparc/sparc-modes.md +174 -0
- package/.claude/commands/sparc/sparc.md +111 -0
- package/.claude/commands/sparc/spec-pseudocode.md +80 -0
- package/.claude/commands/sparc/supabase-admin.md +348 -0
- package/.claude/commands/sparc/swarm-coordinator.md +54 -0
- package/.claude/commands/sparc/tdd.md +54 -0
- package/.claude/commands/sparc/tester.md +54 -0
- package/.claude/commands/sparc/tutorial.md +79 -0
- package/.claude/commands/sparc/workflow-manager.md +54 -0
- package/.claude/commands/sparc.md +166 -0
- package/.claude/commands/stream-chain/pipeline.md +121 -0
- package/.claude/commands/stream-chain/run.md +70 -0
- package/.claude/commands/swarm/README.md +15 -0
- package/.claude/commands/swarm/analysis.md +95 -0
- package/.claude/commands/swarm/development.md +96 -0
- package/.claude/commands/swarm/examples.md +168 -0
- package/.claude/commands/swarm/maintenance.md +102 -0
- package/.claude/commands/swarm/optimization.md +117 -0
- package/.claude/commands/swarm/research.md +136 -0
- package/.claude/commands/swarm/swarm-analysis.md +8 -0
- package/.claude/commands/swarm/swarm-background.md +8 -0
- package/.claude/commands/swarm/swarm-init.md +19 -0
- package/.claude/commands/swarm/swarm-modes.md +8 -0
- package/.claude/commands/swarm/swarm-monitor.md +8 -0
- package/.claude/commands/swarm/swarm-spawn.md +19 -0
- package/.claude/commands/swarm/swarm-status.md +8 -0
- package/.claude/commands/swarm/swarm-strategies.md +8 -0
- package/.claude/commands/swarm/swarm.md +27 -0
- package/.claude/commands/swarm/testing.md +131 -0
- package/.claude/commands/training/README.md +9 -0
- package/.claude/commands/training/model-update.md +25 -0
- package/.claude/commands/training/neural-patterns.md +74 -0
- package/.claude/commands/training/neural-train.md +25 -0
- package/.claude/commands/training/pattern-learn.md +25 -0
- package/.claude/commands/training/specialization.md +63 -0
- package/.claude/commands/truth/start.md +143 -0
- package/.claude/commands/verify/check.md +50 -0
- package/.claude/commands/verify/start.md +128 -0
- package/.claude/commands/workflows/README.md +9 -0
- package/.claude/commands/workflows/development.md +78 -0
- package/.claude/commands/workflows/research.md +63 -0
- package/.claude/commands/workflows/workflow-create.md +25 -0
- package/.claude/commands/workflows/workflow-execute.md +25 -0
- package/.claude/commands/workflows/workflow-export.md +25 -0
- package/.claude/helpers/memory.js +83 -0
- package/.claude/helpers/post-commit +16 -0
- package/.claude/helpers/pre-commit +26 -0
- package/.claude/helpers/router.js +66 -0
- package/.claude/helpers/session.js +127 -0
- package/.claude/helpers/statusline-hook.sh +21 -0
- package/.claude/helpers/statusline.js +227 -0
- package/.claude/settings.json +212 -0
- package/.claude/skills/agentdb-advanced/SKILL.md +550 -0
- package/.claude/skills/agentdb-learning/SKILL.md +545 -0
- package/.claude/skills/agentdb-memory-patterns/SKILL.md +339 -0
- package/.claude/skills/agentdb-optimization/SKILL.md +509 -0
- package/.claude/skills/agentdb-vector-search/SKILL.md +339 -0
- package/.claude/skills/agentic-jujutsu/SKILL.md +645 -0
- package/.claude/skills/flow-nexus-neural/SKILL.md +738 -0
- package/.claude/skills/flow-nexus-platform/SKILL.md +1157 -0
- package/.claude/skills/flow-nexus-swarm/SKILL.md +610 -0
- package/.claude/skills/github-code-review/SKILL.md +1140 -0
- package/.claude/skills/github-multi-repo/SKILL.md +874 -0
- package/.claude/skills/github-project-management/SKILL.md +1277 -0
- package/.claude/skills/github-release-management/SKILL.md +1081 -0
- package/.claude/skills/github-workflow-automation/SKILL.md +1065 -0
- package/.claude/skills/hive-mind-advanced/SKILL.md +712 -0
- package/.claude/skills/hooks-automation/SKILL.md +1201 -0
- package/.claude/skills/pair-programming/SKILL.md +1202 -0
- package/.claude/skills/performance-analysis/SKILL.md +563 -0
- package/.claude/skills/reasoningbank-agentdb/SKILL.md +446 -0
- package/.claude/skills/reasoningbank-intelligence/SKILL.md +201 -0
- package/.claude/skills/skill-builder/.claude-flow/metrics/agent-metrics.json +1 -0
- package/.claude/skills/skill-builder/.claude-flow/metrics/performance.json +87 -0
- package/.claude/skills/skill-builder/.claude-flow/metrics/task-metrics.json +10 -0
- package/.claude/skills/skill-builder/SKILL.md +910 -0
- package/.claude/skills/sparc-methodology/SKILL.md +1115 -0
- package/.claude/skills/stream-chain/SKILL.md +563 -0
- package/.claude/skills/swarm-advanced/SKILL.md +973 -0
- package/.claude/skills/swarm-orchestration/SKILL.md +179 -0
- package/.claude/skills/v3-cli-modernization/SKILL.md +872 -0
- package/.claude/skills/v3-core-implementation/SKILL.md +797 -0
- package/.claude/skills/v3-ddd-architecture/SKILL.md +442 -0
- package/.claude/skills/v3-integration-deep/SKILL.md +241 -0
- package/.claude/skills/v3-mcp-optimization/SKILL.md +777 -0
- package/.claude/skills/v3-memory-unification/SKILL.md +174 -0
- package/.claude/skills/v3-performance-optimization/SKILL.md +390 -0
- package/.claude/skills/v3-security-overhaul/SKILL.md +82 -0
- package/.claude/skills/v3-swarm-coordination/SKILL.md +340 -0
- package/.claude/skills/verification-quality/SKILL.md +649 -0
- package/.claude/skills/worker-benchmarks/skill.md +135 -0
- package/.claude/skills/worker-integration/skill.md +154 -0
- package/dist/src/init/executor.d.ts.map +1 -1
- package/dist/src/init/executor.js +13 -4
- package/dist/src/init/executor.js.map +1 -1
- package/dist/src/init/settings-generator.js +9 -9
- package/dist/src/init/settings-generator.js.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +2 -1
|
@@ -0,0 +1,797 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: "V3 Core Implementation"
|
|
3
|
+
description: "Core module implementation for claude-flow v3. Implements DDD domains, clean architecture patterns, dependency injection, and modular TypeScript codebase with comprehensive testing."
|
|
4
|
+
---
|
|
5
|
+
|
|
6
|
+
# V3 Core Implementation
|
|
7
|
+
|
|
8
|
+
## What This Skill Does
|
|
9
|
+
|
|
10
|
+
Implements the core TypeScript modules for claude-flow v3 following Domain-Driven Design principles, clean architecture patterns, and modern TypeScript best practices with comprehensive test coverage.
|
|
11
|
+
|
|
12
|
+
## Quick Start
|
|
13
|
+
|
|
14
|
+
```bash
|
|
15
|
+
# Initialize core implementation
|
|
16
|
+
Task("Core foundation", "Set up DDD domain structure and base classes", "core-implementer")
|
|
17
|
+
|
|
18
|
+
# Domain implementation (parallel)
|
|
19
|
+
Task("Task domain", "Implement task management domain with entities and services", "core-implementer")
|
|
20
|
+
Task("Session domain", "Implement session management domain", "core-implementer")
|
|
21
|
+
Task("Health domain", "Implement health monitoring domain", "core-implementer")
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
## Core Implementation Architecture
|
|
25
|
+
|
|
26
|
+
### Domain Structure
|
|
27
|
+
```
|
|
28
|
+
src/
|
|
29
|
+
├── core/
|
|
30
|
+
│ ├── kernel/ # Microkernel pattern
|
|
31
|
+
│ │ ├── claude-flow-kernel.ts
|
|
32
|
+
│ │ ├── domain-registry.ts
|
|
33
|
+
│ │ └── plugin-loader.ts
|
|
34
|
+
│ │
|
|
35
|
+
│ ├── domains/ # DDD Bounded Contexts
|
|
36
|
+
│ │ ├── task-management/
|
|
37
|
+
│ │ │ ├── entities/
|
|
38
|
+
│ │ │ ├── value-objects/
|
|
39
|
+
│ │ │ ├── services/
|
|
40
|
+
│ │ │ ├── repositories/
|
|
41
|
+
│ │ │ └── events/
|
|
42
|
+
│ │ │
|
|
43
|
+
│ │ ├── session-management/
|
|
44
|
+
│ │ ├── health-monitoring/
|
|
45
|
+
│ │ ├── lifecycle-management/
|
|
46
|
+
│ │ └── event-coordination/
|
|
47
|
+
│ │
|
|
48
|
+
│ ├── shared/ # Shared kernel
|
|
49
|
+
│ │ ├── domain/
|
|
50
|
+
│ │ │ ├── entity.ts
|
|
51
|
+
│ │ │ ├── value-object.ts
|
|
52
|
+
│ │ │ ├── domain-event.ts
|
|
53
|
+
│ │ │ └── aggregate-root.ts
|
|
54
|
+
│ │ │
|
|
55
|
+
│ │ ├── infrastructure/
|
|
56
|
+
│ │ │ ├── event-bus.ts
|
|
57
|
+
│ │ │ ├── dependency-container.ts
|
|
58
|
+
│ │ │ └── logger.ts
|
|
59
|
+
│ │ │
|
|
60
|
+
│ │ └── types/
|
|
61
|
+
│ │ ├── common.ts
|
|
62
|
+
│ │ ├── errors.ts
|
|
63
|
+
│ │ └── interfaces.ts
|
|
64
|
+
│ │
|
|
65
|
+
│ └── application/ # Application services
|
|
66
|
+
│ ├── use-cases/
|
|
67
|
+
│ ├── commands/
|
|
68
|
+
│ ├── queries/
|
|
69
|
+
│ └── handlers/
|
|
70
|
+
```
|
|
71
|
+
|
|
72
|
+
## Base Domain Classes
|
|
73
|
+
|
|
74
|
+
### Entity Base Class
|
|
75
|
+
```typescript
|
|
76
|
+
// src/core/shared/domain/entity.ts
|
|
77
|
+
export abstract class Entity<T> {
|
|
78
|
+
protected readonly _id: T;
|
|
79
|
+
private _domainEvents: DomainEvent[] = [];
|
|
80
|
+
|
|
81
|
+
constructor(id: T) {
|
|
82
|
+
this._id = id;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
get id(): T {
|
|
86
|
+
return this._id;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
public equals(object?: Entity<T>): boolean {
|
|
90
|
+
if (object == null || object == undefined) {
|
|
91
|
+
return false;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
if (this === object) {
|
|
95
|
+
return true;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
if (!(object instanceof Entity)) {
|
|
99
|
+
return false;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
return this._id === object._id;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
protected addDomainEvent(domainEvent: DomainEvent): void {
|
|
106
|
+
this._domainEvents.push(domainEvent);
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
public getUncommittedEvents(): DomainEvent[] {
|
|
110
|
+
return this._domainEvents;
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
public markEventsAsCommitted(): void {
|
|
114
|
+
this._domainEvents = [];
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
```
|
|
118
|
+
|
|
119
|
+
### Value Object Base Class
|
|
120
|
+
```typescript
|
|
121
|
+
// src/core/shared/domain/value-object.ts
|
|
122
|
+
export abstract class ValueObject<T> {
|
|
123
|
+
protected readonly props: T;
|
|
124
|
+
|
|
125
|
+
constructor(props: T) {
|
|
126
|
+
this.props = Object.freeze(props);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
public equals(object?: ValueObject<T>): boolean {
|
|
130
|
+
if (object == null || object == undefined) {
|
|
131
|
+
return false;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
if (this === object) {
|
|
135
|
+
return true;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
return JSON.stringify(this.props) === JSON.stringify(object.props);
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
get value(): T {
|
|
142
|
+
return this.props;
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
```
|
|
146
|
+
|
|
147
|
+
### Aggregate Root
|
|
148
|
+
```typescript
|
|
149
|
+
// src/core/shared/domain/aggregate-root.ts
|
|
150
|
+
export abstract class AggregateRoot<T> extends Entity<T> {
|
|
151
|
+
private _version: number = 0;
|
|
152
|
+
|
|
153
|
+
get version(): number {
|
|
154
|
+
return this._version;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
protected incrementVersion(): void {
|
|
158
|
+
this._version++;
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
public applyEvent(event: DomainEvent): void {
|
|
162
|
+
this.addDomainEvent(event);
|
|
163
|
+
this.incrementVersion();
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
```
|
|
167
|
+
|
|
168
|
+
## Task Management Domain Implementation
|
|
169
|
+
|
|
170
|
+
### Task Entity
|
|
171
|
+
```typescript
|
|
172
|
+
// src/core/domains/task-management/entities/task.entity.ts
|
|
173
|
+
import { AggregateRoot } from '../../../shared/domain/aggregate-root';
|
|
174
|
+
import { TaskId } from '../value-objects/task-id.vo';
|
|
175
|
+
import { TaskStatus } from '../value-objects/task-status.vo';
|
|
176
|
+
import { Priority } from '../value-objects/priority.vo';
|
|
177
|
+
import { TaskAssignedEvent } from '../events/task-assigned.event';
|
|
178
|
+
|
|
179
|
+
interface TaskProps {
|
|
180
|
+
id: TaskId;
|
|
181
|
+
description: string;
|
|
182
|
+
priority: Priority;
|
|
183
|
+
status: TaskStatus;
|
|
184
|
+
assignedAgentId?: string;
|
|
185
|
+
createdAt: Date;
|
|
186
|
+
updatedAt: Date;
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
export class Task extends AggregateRoot<TaskId> {
|
|
190
|
+
private props: TaskProps;
|
|
191
|
+
|
|
192
|
+
private constructor(props: TaskProps) {
|
|
193
|
+
super(props.id);
|
|
194
|
+
this.props = props;
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
static create(description: string, priority: Priority): Task {
|
|
198
|
+
const task = new Task({
|
|
199
|
+
id: TaskId.create(),
|
|
200
|
+
description,
|
|
201
|
+
priority,
|
|
202
|
+
status: TaskStatus.pending(),
|
|
203
|
+
createdAt: new Date(),
|
|
204
|
+
updatedAt: new Date()
|
|
205
|
+
});
|
|
206
|
+
|
|
207
|
+
return task;
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
static reconstitute(props: TaskProps): Task {
|
|
211
|
+
return new Task(props);
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
public assignTo(agentId: string): void {
|
|
215
|
+
if (this.props.status.equals(TaskStatus.completed())) {
|
|
216
|
+
throw new Error('Cannot assign completed task');
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
this.props.assignedAgentId = agentId;
|
|
220
|
+
this.props.status = TaskStatus.assigned();
|
|
221
|
+
this.props.updatedAt = new Date();
|
|
222
|
+
|
|
223
|
+
this.applyEvent(new TaskAssignedEvent(
|
|
224
|
+
this.id.value,
|
|
225
|
+
agentId,
|
|
226
|
+
this.props.priority
|
|
227
|
+
));
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
public complete(result: TaskResult): void {
|
|
231
|
+
if (!this.props.assignedAgentId) {
|
|
232
|
+
throw new Error('Cannot complete unassigned task');
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
this.props.status = TaskStatus.completed();
|
|
236
|
+
this.props.updatedAt = new Date();
|
|
237
|
+
|
|
238
|
+
this.applyEvent(new TaskCompletedEvent(
|
|
239
|
+
this.id.value,
|
|
240
|
+
result,
|
|
241
|
+
this.calculateDuration()
|
|
242
|
+
));
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
// Getters
|
|
246
|
+
get description(): string { return this.props.description; }
|
|
247
|
+
get priority(): Priority { return this.props.priority; }
|
|
248
|
+
get status(): TaskStatus { return this.props.status; }
|
|
249
|
+
get assignedAgentId(): string | undefined { return this.props.assignedAgentId; }
|
|
250
|
+
get createdAt(): Date { return this.props.createdAt; }
|
|
251
|
+
get updatedAt(): Date { return this.props.updatedAt; }
|
|
252
|
+
|
|
253
|
+
private calculateDuration(): number {
|
|
254
|
+
return this.props.updatedAt.getTime() - this.props.createdAt.getTime();
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
```
|
|
258
|
+
|
|
259
|
+
### Task Value Objects
|
|
260
|
+
```typescript
|
|
261
|
+
// src/core/domains/task-management/value-objects/task-id.vo.ts
|
|
262
|
+
export class TaskId extends ValueObject<string> {
|
|
263
|
+
private constructor(value: string) {
|
|
264
|
+
super({ value });
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
static create(): TaskId {
|
|
268
|
+
return new TaskId(crypto.randomUUID());
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
static fromString(id: string): TaskId {
|
|
272
|
+
if (!id || id.length === 0) {
|
|
273
|
+
throw new Error('TaskId cannot be empty');
|
|
274
|
+
}
|
|
275
|
+
return new TaskId(id);
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
get value(): string {
|
|
279
|
+
return this.props.value;
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
// src/core/domains/task-management/value-objects/task-status.vo.ts
|
|
284
|
+
type TaskStatusType = 'pending' | 'assigned' | 'in_progress' | 'completed' | 'failed';
|
|
285
|
+
|
|
286
|
+
export class TaskStatus extends ValueObject<TaskStatusType> {
|
|
287
|
+
private constructor(status: TaskStatusType) {
|
|
288
|
+
super({ value: status });
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
static pending(): TaskStatus { return new TaskStatus('pending'); }
|
|
292
|
+
static assigned(): TaskStatus { return new TaskStatus('assigned'); }
|
|
293
|
+
static inProgress(): TaskStatus { return new TaskStatus('in_progress'); }
|
|
294
|
+
static completed(): TaskStatus { return new TaskStatus('completed'); }
|
|
295
|
+
static failed(): TaskStatus { return new TaskStatus('failed'); }
|
|
296
|
+
|
|
297
|
+
get value(): TaskStatusType {
|
|
298
|
+
return this.props.value;
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
public isPending(): boolean { return this.value === 'pending'; }
|
|
302
|
+
public isAssigned(): boolean { return this.value === 'assigned'; }
|
|
303
|
+
public isInProgress(): boolean { return this.value === 'in_progress'; }
|
|
304
|
+
public isCompleted(): boolean { return this.value === 'completed'; }
|
|
305
|
+
public isFailed(): boolean { return this.value === 'failed'; }
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
// src/core/domains/task-management/value-objects/priority.vo.ts
|
|
309
|
+
type PriorityLevel = 'low' | 'medium' | 'high' | 'critical';
|
|
310
|
+
|
|
311
|
+
export class Priority extends ValueObject<PriorityLevel> {
|
|
312
|
+
private constructor(level: PriorityLevel) {
|
|
313
|
+
super({ value: level });
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
static low(): Priority { return new Priority('low'); }
|
|
317
|
+
static medium(): Priority { return new Priority('medium'); }
|
|
318
|
+
static high(): Priority { return new Priority('high'); }
|
|
319
|
+
static critical(): Priority { return new Priority('critical'); }
|
|
320
|
+
|
|
321
|
+
get value(): PriorityLevel {
|
|
322
|
+
return this.props.value;
|
|
323
|
+
}
|
|
324
|
+
|
|
325
|
+
public getNumericValue(): number {
|
|
326
|
+
const priorities = { low: 1, medium: 2, high: 3, critical: 4 };
|
|
327
|
+
return priorities[this.value];
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
```
|
|
331
|
+
|
|
332
|
+
## Domain Services
|
|
333
|
+
|
|
334
|
+
### Task Scheduling Service
|
|
335
|
+
```typescript
|
|
336
|
+
// src/core/domains/task-management/services/task-scheduling.service.ts
|
|
337
|
+
import { Injectable } from '../../../shared/infrastructure/dependency-container';
|
|
338
|
+
import { Task } from '../entities/task.entity';
|
|
339
|
+
import { Priority } from '../value-objects/priority.vo';
|
|
340
|
+
|
|
341
|
+
@Injectable()
|
|
342
|
+
export class TaskSchedulingService {
|
|
343
|
+
public prioritizeTasks(tasks: Task[]): Task[] {
|
|
344
|
+
return tasks.sort((a, b) =>
|
|
345
|
+
b.priority.getNumericValue() - a.priority.getNumericValue()
|
|
346
|
+
);
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
public canSchedule(task: Task, agentCapacity: number): boolean {
|
|
350
|
+
if (agentCapacity <= 0) return false;
|
|
351
|
+
|
|
352
|
+
// Critical tasks always schedulable
|
|
353
|
+
if (task.priority.equals(Priority.critical())) return true;
|
|
354
|
+
|
|
355
|
+
// Other logic based on capacity
|
|
356
|
+
return true;
|
|
357
|
+
}
|
|
358
|
+
|
|
359
|
+
public calculateEstimatedDuration(task: Task): number {
|
|
360
|
+
// Simple heuristic - would use ML in real implementation
|
|
361
|
+
const baseTime = 300000; // 5 minutes
|
|
362
|
+
const priorityMultiplier = {
|
|
363
|
+
low: 0.5,
|
|
364
|
+
medium: 1.0,
|
|
365
|
+
high: 1.5,
|
|
366
|
+
critical: 2.0
|
|
367
|
+
};
|
|
368
|
+
|
|
369
|
+
return baseTime * priorityMultiplier[task.priority.value];
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
```
|
|
373
|
+
|
|
374
|
+
## Repository Interfaces & Implementations
|
|
375
|
+
|
|
376
|
+
### Task Repository Interface
|
|
377
|
+
```typescript
|
|
378
|
+
// src/core/domains/task-management/repositories/task.repository.ts
|
|
379
|
+
export interface ITaskRepository {
|
|
380
|
+
save(task: Task): Promise<void>;
|
|
381
|
+
findById(id: TaskId): Promise<Task | null>;
|
|
382
|
+
findByAgentId(agentId: string): Promise<Task[]>;
|
|
383
|
+
findByStatus(status: TaskStatus): Promise<Task[]>;
|
|
384
|
+
findPendingTasks(): Promise<Task[]>;
|
|
385
|
+
delete(id: TaskId): Promise<void>;
|
|
386
|
+
}
|
|
387
|
+
```
|
|
388
|
+
|
|
389
|
+
### SQLite Implementation
|
|
390
|
+
```typescript
|
|
391
|
+
// src/core/domains/task-management/repositories/sqlite-task.repository.ts
|
|
392
|
+
@Injectable()
|
|
393
|
+
export class SqliteTaskRepository implements ITaskRepository {
|
|
394
|
+
constructor(
|
|
395
|
+
@Inject('Database') private db: Database,
|
|
396
|
+
@Inject('Logger') private logger: ILogger
|
|
397
|
+
) {}
|
|
398
|
+
|
|
399
|
+
async save(task: Task): Promise<void> {
|
|
400
|
+
const sql = `
|
|
401
|
+
INSERT OR REPLACE INTO tasks (
|
|
402
|
+
id, description, priority, status, assigned_agent_id, created_at, updated_at
|
|
403
|
+
) VALUES (?, ?, ?, ?, ?, ?, ?)
|
|
404
|
+
`;
|
|
405
|
+
|
|
406
|
+
await this.db.run(sql, [
|
|
407
|
+
task.id.value,
|
|
408
|
+
task.description,
|
|
409
|
+
task.priority.value,
|
|
410
|
+
task.status.value,
|
|
411
|
+
task.assignedAgentId,
|
|
412
|
+
task.createdAt.toISOString(),
|
|
413
|
+
task.updatedAt.toISOString()
|
|
414
|
+
]);
|
|
415
|
+
|
|
416
|
+
this.logger.debug(`Task saved: ${task.id.value}`);
|
|
417
|
+
}
|
|
418
|
+
|
|
419
|
+
async findById(id: TaskId): Promise<Task | null> {
|
|
420
|
+
const sql = 'SELECT * FROM tasks WHERE id = ?';
|
|
421
|
+
const row = await this.db.get(sql, [id.value]);
|
|
422
|
+
|
|
423
|
+
return row ? this.mapRowToTask(row) : null;
|
|
424
|
+
}
|
|
425
|
+
|
|
426
|
+
async findPendingTasks(): Promise<Task[]> {
|
|
427
|
+
const sql = 'SELECT * FROM tasks WHERE status = ? ORDER BY priority DESC, created_at ASC';
|
|
428
|
+
const rows = await this.db.all(sql, ['pending']);
|
|
429
|
+
|
|
430
|
+
return rows.map(row => this.mapRowToTask(row));
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
private mapRowToTask(row: any): Task {
|
|
434
|
+
return Task.reconstitute({
|
|
435
|
+
id: TaskId.fromString(row.id),
|
|
436
|
+
description: row.description,
|
|
437
|
+
priority: Priority.fromString(row.priority),
|
|
438
|
+
status: TaskStatus.fromString(row.status),
|
|
439
|
+
assignedAgentId: row.assigned_agent_id,
|
|
440
|
+
createdAt: new Date(row.created_at),
|
|
441
|
+
updatedAt: new Date(row.updated_at)
|
|
442
|
+
});
|
|
443
|
+
}
|
|
444
|
+
}
|
|
445
|
+
```
|
|
446
|
+
|
|
447
|
+
## Application Layer
|
|
448
|
+
|
|
449
|
+
### Use Case Implementation
|
|
450
|
+
```typescript
|
|
451
|
+
// src/core/application/use-cases/assign-task.use-case.ts
|
|
452
|
+
@Injectable()
|
|
453
|
+
export class AssignTaskUseCase {
|
|
454
|
+
constructor(
|
|
455
|
+
@Inject('TaskRepository') private taskRepository: ITaskRepository,
|
|
456
|
+
@Inject('AgentRepository') private agentRepository: IAgentRepository,
|
|
457
|
+
@Inject('DomainEventBus') private eventBus: DomainEventBus,
|
|
458
|
+
@Inject('Logger') private logger: ILogger
|
|
459
|
+
) {}
|
|
460
|
+
|
|
461
|
+
async execute(command: AssignTaskCommand): Promise<AssignTaskResult> {
|
|
462
|
+
try {
|
|
463
|
+
// 1. Validate command
|
|
464
|
+
await this.validateCommand(command);
|
|
465
|
+
|
|
466
|
+
// 2. Load aggregates
|
|
467
|
+
const task = await this.taskRepository.findById(command.taskId);
|
|
468
|
+
if (!task) {
|
|
469
|
+
throw new TaskNotFoundError(command.taskId);
|
|
470
|
+
}
|
|
471
|
+
|
|
472
|
+
const agent = await this.agentRepository.findById(command.agentId);
|
|
473
|
+
if (!agent) {
|
|
474
|
+
throw new AgentNotFoundError(command.agentId);
|
|
475
|
+
}
|
|
476
|
+
|
|
477
|
+
// 3. Business logic
|
|
478
|
+
if (!agent.canAcceptTask(task)) {
|
|
479
|
+
throw new AgentCannotAcceptTaskError(command.agentId, command.taskId);
|
|
480
|
+
}
|
|
481
|
+
|
|
482
|
+
task.assignTo(command.agentId);
|
|
483
|
+
agent.acceptTask(task.id);
|
|
484
|
+
|
|
485
|
+
// 4. Persist changes
|
|
486
|
+
await Promise.all([
|
|
487
|
+
this.taskRepository.save(task),
|
|
488
|
+
this.agentRepository.save(agent)
|
|
489
|
+
]);
|
|
490
|
+
|
|
491
|
+
// 5. Publish domain events
|
|
492
|
+
const events = [
|
|
493
|
+
...task.getUncommittedEvents(),
|
|
494
|
+
...agent.getUncommittedEvents()
|
|
495
|
+
];
|
|
496
|
+
|
|
497
|
+
for (const event of events) {
|
|
498
|
+
await this.eventBus.publish(event);
|
|
499
|
+
}
|
|
500
|
+
|
|
501
|
+
task.markEventsAsCommitted();
|
|
502
|
+
agent.markEventsAsCommitted();
|
|
503
|
+
|
|
504
|
+
// 6. Return result
|
|
505
|
+
this.logger.info(`Task ${command.taskId.value} assigned to agent ${command.agentId}`);
|
|
506
|
+
|
|
507
|
+
return AssignTaskResult.success({
|
|
508
|
+
taskId: task.id,
|
|
509
|
+
agentId: command.agentId,
|
|
510
|
+
assignedAt: new Date()
|
|
511
|
+
});
|
|
512
|
+
|
|
513
|
+
} catch (error) {
|
|
514
|
+
this.logger.error(`Failed to assign task ${command.taskId.value}:`, error);
|
|
515
|
+
return AssignTaskResult.failure(error);
|
|
516
|
+
}
|
|
517
|
+
}
|
|
518
|
+
|
|
519
|
+
private async validateCommand(command: AssignTaskCommand): Promise<void> {
|
|
520
|
+
if (!command.taskId) {
|
|
521
|
+
throw new ValidationError('Task ID is required');
|
|
522
|
+
}
|
|
523
|
+
if (!command.agentId) {
|
|
524
|
+
throw new ValidationError('Agent ID is required');
|
|
525
|
+
}
|
|
526
|
+
}
|
|
527
|
+
}
|
|
528
|
+
```
|
|
529
|
+
|
|
530
|
+
## Dependency Injection Setup
|
|
531
|
+
|
|
532
|
+
### Container Configuration
|
|
533
|
+
```typescript
|
|
534
|
+
// src/core/shared/infrastructure/dependency-container.ts
|
|
535
|
+
import { Container } from 'inversify';
|
|
536
|
+
import { TYPES } from './types';
|
|
537
|
+
|
|
538
|
+
export class DependencyContainer {
|
|
539
|
+
private container: Container;
|
|
540
|
+
|
|
541
|
+
constructor() {
|
|
542
|
+
this.container = new Container();
|
|
543
|
+
this.setupBindings();
|
|
544
|
+
}
|
|
545
|
+
|
|
546
|
+
private setupBindings(): void {
|
|
547
|
+
// Repositories
|
|
548
|
+
this.container.bind<ITaskRepository>(TYPES.TaskRepository)
|
|
549
|
+
.to(SqliteTaskRepository)
|
|
550
|
+
.inSingletonScope();
|
|
551
|
+
|
|
552
|
+
this.container.bind<IAgentRepository>(TYPES.AgentRepository)
|
|
553
|
+
.to(SqliteAgentRepository)
|
|
554
|
+
.inSingletonScope();
|
|
555
|
+
|
|
556
|
+
// Services
|
|
557
|
+
this.container.bind<TaskSchedulingService>(TYPES.TaskSchedulingService)
|
|
558
|
+
.to(TaskSchedulingService)
|
|
559
|
+
.inSingletonScope();
|
|
560
|
+
|
|
561
|
+
// Use Cases
|
|
562
|
+
this.container.bind<AssignTaskUseCase>(TYPES.AssignTaskUseCase)
|
|
563
|
+
.to(AssignTaskUseCase)
|
|
564
|
+
.inSingletonScope();
|
|
565
|
+
|
|
566
|
+
// Infrastructure
|
|
567
|
+
this.container.bind<ILogger>(TYPES.Logger)
|
|
568
|
+
.to(ConsoleLogger)
|
|
569
|
+
.inSingletonScope();
|
|
570
|
+
|
|
571
|
+
this.container.bind<DomainEventBus>(TYPES.DomainEventBus)
|
|
572
|
+
.to(InMemoryDomainEventBus)
|
|
573
|
+
.inSingletonScope();
|
|
574
|
+
}
|
|
575
|
+
|
|
576
|
+
get<T>(serviceIdentifier: symbol): T {
|
|
577
|
+
return this.container.get<T>(serviceIdentifier);
|
|
578
|
+
}
|
|
579
|
+
|
|
580
|
+
bind<T>(serviceIdentifier: symbol): BindingToSyntax<T> {
|
|
581
|
+
return this.container.bind<T>(serviceIdentifier);
|
|
582
|
+
}
|
|
583
|
+
}
|
|
584
|
+
```
|
|
585
|
+
|
|
586
|
+
## Modern TypeScript Configuration
|
|
587
|
+
|
|
588
|
+
### Strict TypeScript Setup
|
|
589
|
+
```json
|
|
590
|
+
// tsconfig.json
|
|
591
|
+
{
|
|
592
|
+
"compilerOptions": {
|
|
593
|
+
"target": "ES2022",
|
|
594
|
+
"lib": ["ES2022"],
|
|
595
|
+
"module": "NodeNext",
|
|
596
|
+
"moduleResolution": "NodeNext",
|
|
597
|
+
"declaration": true,
|
|
598
|
+
"outDir": "./dist",
|
|
599
|
+
"strict": true,
|
|
600
|
+
"exactOptionalPropertyTypes": true,
|
|
601
|
+
"noImplicitReturns": true,
|
|
602
|
+
"noFallthroughCasesInSwitch": true,
|
|
603
|
+
"noUncheckedIndexedAccess": true,
|
|
604
|
+
"noImplicitOverride": true,
|
|
605
|
+
"experimentalDecorators": true,
|
|
606
|
+
"emitDecoratorMetadata": true,
|
|
607
|
+
"skipLibCheck": true,
|
|
608
|
+
"forceConsistentCasingInFileNames": true,
|
|
609
|
+
"resolveJsonModule": true,
|
|
610
|
+
"esModuleInterop": true,
|
|
611
|
+
"allowSyntheticDefaultImports": true,
|
|
612
|
+
"baseUrl": ".",
|
|
613
|
+
"paths": {
|
|
614
|
+
"@/*": ["src/*"],
|
|
615
|
+
"@core/*": ["src/core/*"],
|
|
616
|
+
"@shared/*": ["src/core/shared/*"],
|
|
617
|
+
"@domains/*": ["src/core/domains/*"]
|
|
618
|
+
}
|
|
619
|
+
},
|
|
620
|
+
"include": ["src/**/*"],
|
|
621
|
+
"exclude": ["node_modules", "dist", "**/*.test.ts", "**/*.spec.ts"]
|
|
622
|
+
}
|
|
623
|
+
```
|
|
624
|
+
|
|
625
|
+
## Testing Implementation
|
|
626
|
+
|
|
627
|
+
### Domain Unit Tests
|
|
628
|
+
```typescript
|
|
629
|
+
// src/core/domains/task-management/__tests__/entities/task.entity.test.ts
|
|
630
|
+
describe('Task Entity', () => {
|
|
631
|
+
let task: Task;
|
|
632
|
+
|
|
633
|
+
beforeEach(() => {
|
|
634
|
+
task = Task.create('Test task', Priority.medium());
|
|
635
|
+
});
|
|
636
|
+
|
|
637
|
+
describe('creation', () => {
|
|
638
|
+
it('should create task with pending status', () => {
|
|
639
|
+
expect(task.status.isPending()).toBe(true);
|
|
640
|
+
expect(task.description).toBe('Test task');
|
|
641
|
+
expect(task.priority.equals(Priority.medium())).toBe(true);
|
|
642
|
+
});
|
|
643
|
+
|
|
644
|
+
it('should generate unique ID', () => {
|
|
645
|
+
const task1 = Task.create('Task 1', Priority.low());
|
|
646
|
+
const task2 = Task.create('Task 2', Priority.low());
|
|
647
|
+
|
|
648
|
+
expect(task1.id.equals(task2.id)).toBe(false);
|
|
649
|
+
});
|
|
650
|
+
});
|
|
651
|
+
|
|
652
|
+
describe('assignment', () => {
|
|
653
|
+
it('should assign to agent and change status', () => {
|
|
654
|
+
const agentId = 'agent-123';
|
|
655
|
+
|
|
656
|
+
task.assignTo(agentId);
|
|
657
|
+
|
|
658
|
+
expect(task.assignedAgentId).toBe(agentId);
|
|
659
|
+
expect(task.status.isAssigned()).toBe(true);
|
|
660
|
+
});
|
|
661
|
+
|
|
662
|
+
it('should emit TaskAssignedEvent when assigned', () => {
|
|
663
|
+
const agentId = 'agent-123';
|
|
664
|
+
|
|
665
|
+
task.assignTo(agentId);
|
|
666
|
+
|
|
667
|
+
const events = task.getUncommittedEvents();
|
|
668
|
+
expect(events).toHaveLength(1);
|
|
669
|
+
expect(events[0]).toBeInstanceOf(TaskAssignedEvent);
|
|
670
|
+
});
|
|
671
|
+
|
|
672
|
+
it('should not allow assignment of completed task', () => {
|
|
673
|
+
task.assignTo('agent-123');
|
|
674
|
+
task.complete(TaskResult.success('done'));
|
|
675
|
+
|
|
676
|
+
expect(() => task.assignTo('agent-456'))
|
|
677
|
+
.toThrow('Cannot assign completed task');
|
|
678
|
+
});
|
|
679
|
+
});
|
|
680
|
+
});
|
|
681
|
+
```
|
|
682
|
+
|
|
683
|
+
### Integration Tests
|
|
684
|
+
```typescript
|
|
685
|
+
// src/core/domains/task-management/__tests__/integration/task-repository.integration.test.ts
|
|
686
|
+
describe('TaskRepository Integration', () => {
|
|
687
|
+
let repository: SqliteTaskRepository;
|
|
688
|
+
let db: Database;
|
|
689
|
+
|
|
690
|
+
beforeEach(async () => {
|
|
691
|
+
db = new Database(':memory:');
|
|
692
|
+
await setupTasksTable(db);
|
|
693
|
+
repository = new SqliteTaskRepository(db, new ConsoleLogger());
|
|
694
|
+
});
|
|
695
|
+
|
|
696
|
+
afterEach(async () => {
|
|
697
|
+
await db.close();
|
|
698
|
+
});
|
|
699
|
+
|
|
700
|
+
it('should save and retrieve task', async () => {
|
|
701
|
+
const task = Task.create('Test task', Priority.high());
|
|
702
|
+
|
|
703
|
+
await repository.save(task);
|
|
704
|
+
const retrieved = await repository.findById(task.id);
|
|
705
|
+
|
|
706
|
+
expect(retrieved).toBeDefined();
|
|
707
|
+
expect(retrieved!.id.equals(task.id)).toBe(true);
|
|
708
|
+
expect(retrieved!.description).toBe('Test task');
|
|
709
|
+
expect(retrieved!.priority.equals(Priority.high())).toBe(true);
|
|
710
|
+
});
|
|
711
|
+
|
|
712
|
+
it('should find pending tasks ordered by priority', async () => {
|
|
713
|
+
const lowTask = Task.create('Low priority', Priority.low());
|
|
714
|
+
const highTask = Task.create('High priority', Priority.high());
|
|
715
|
+
|
|
716
|
+
await repository.save(lowTask);
|
|
717
|
+
await repository.save(highTask);
|
|
718
|
+
|
|
719
|
+
const pending = await repository.findPendingTasks();
|
|
720
|
+
|
|
721
|
+
expect(pending).toHaveLength(2);
|
|
722
|
+
expect(pending[0].id.equals(highTask.id)).toBe(true); // High priority first
|
|
723
|
+
expect(pending[1].id.equals(lowTask.id)).toBe(true);
|
|
724
|
+
});
|
|
725
|
+
});
|
|
726
|
+
```
|
|
727
|
+
|
|
728
|
+
## Performance Optimizations
|
|
729
|
+
|
|
730
|
+
### Entity Caching
|
|
731
|
+
```typescript
|
|
732
|
+
// src/core/shared/infrastructure/entity-cache.ts
|
|
733
|
+
@Injectable()
|
|
734
|
+
export class EntityCache<T extends Entity<any>> {
|
|
735
|
+
private cache = new Map<string, { entity: T; timestamp: number }>();
|
|
736
|
+
private readonly ttl: number = 300000; // 5 minutes
|
|
737
|
+
|
|
738
|
+
set(id: string, entity: T): void {
|
|
739
|
+
this.cache.set(id, { entity, timestamp: Date.now() });
|
|
740
|
+
}
|
|
741
|
+
|
|
742
|
+
get(id: string): T | null {
|
|
743
|
+
const cached = this.cache.get(id);
|
|
744
|
+
if (!cached) return null;
|
|
745
|
+
|
|
746
|
+
// Check TTL
|
|
747
|
+
if (Date.now() - cached.timestamp > this.ttl) {
|
|
748
|
+
this.cache.delete(id);
|
|
749
|
+
return null;
|
|
750
|
+
}
|
|
751
|
+
|
|
752
|
+
return cached.entity;
|
|
753
|
+
}
|
|
754
|
+
|
|
755
|
+
invalidate(id: string): void {
|
|
756
|
+
this.cache.delete(id);
|
|
757
|
+
}
|
|
758
|
+
|
|
759
|
+
clear(): void {
|
|
760
|
+
this.cache.clear();
|
|
761
|
+
}
|
|
762
|
+
}
|
|
763
|
+
```
|
|
764
|
+
|
|
765
|
+
## Success Metrics
|
|
766
|
+
|
|
767
|
+
- [ ] **Domain Isolation**: 100% clean dependency boundaries
|
|
768
|
+
- [ ] **Test Coverage**: >90% unit test coverage for domain logic
|
|
769
|
+
- [ ] **Type Safety**: Strict TypeScript compilation with zero any types
|
|
770
|
+
- [ ] **Performance**: <50ms average use case execution time
|
|
771
|
+
- [ ] **Memory Efficiency**: <100MB heap usage for core domains
|
|
772
|
+
- [ ] **Plugin Architecture**: Modular domain loading capability
|
|
773
|
+
|
|
774
|
+
## Related V3 Skills
|
|
775
|
+
|
|
776
|
+
- `v3-ddd-architecture` - DDD architectural design
|
|
777
|
+
- `v3-mcp-optimization` - MCP server integration
|
|
778
|
+
- `v3-memory-unification` - AgentDB repository integration
|
|
779
|
+
- `v3-swarm-coordination` - Swarm domain implementation
|
|
780
|
+
|
|
781
|
+
## Usage Examples
|
|
782
|
+
|
|
783
|
+
### Complete Core Implementation
|
|
784
|
+
```bash
|
|
785
|
+
# Full core module implementation
|
|
786
|
+
Task("Core implementation",
|
|
787
|
+
"Implement all core domains with DDD patterns and comprehensive testing",
|
|
788
|
+
"core-implementer")
|
|
789
|
+
```
|
|
790
|
+
|
|
791
|
+
### Domain-Specific Implementation
|
|
792
|
+
```bash
|
|
793
|
+
# Single domain implementation
|
|
794
|
+
Task("Task domain implementation",
|
|
795
|
+
"Implement task management domain with entities, services, and repositories",
|
|
796
|
+
"core-implementer")
|
|
797
|
+
```
|