- Delete Models/AgentWorkspaceState.php (legacy port, no backing table) - Rewrite Models/WorkspaceState.php as the single canonical state model backed by agent_workspace_states table with array value cast, type helpers, scopeForPlan/scopeOfType, static getValue/setValue, and toMcpContext() for MCP tool output - Update AgentPlan::states() relation and setState() return type - Update StateSet MCP tool import - Update SecurityTest to use WorkspaceState - Add WorkspaceStateTest covering table, casts, type helpers, scopes, static helpers, toMcpContext, and AgentPlan integration - Mark CQ-001 done in TODO.md Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
140 lines
3.3 KiB
PHP
140 lines
3.3 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Core\Mod\Agentic\Models;
|
|
|
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
|
use Illuminate\Database\Eloquent\Model;
|
|
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
|
|
|
/**
|
|
* Workspace State Model
|
|
*
|
|
* Persistent key-value state storage for agent plans.
|
|
* Stores typed values shared across agent sessions within a plan,
|
|
* enabling context sharing and state recovery.
|
|
*
|
|
* @property int $id
|
|
* @property int $agent_plan_id
|
|
* @property string $key
|
|
* @property array $value
|
|
* @property string $type
|
|
* @property string|null $description
|
|
* @property \Carbon\Carbon|null $created_at
|
|
* @property \Carbon\Carbon|null $updated_at
|
|
*/
|
|
class WorkspaceState extends Model
|
|
{
|
|
use HasFactory;
|
|
|
|
protected $table = 'agent_workspace_states';
|
|
|
|
public const TYPE_JSON = 'json';
|
|
|
|
public const TYPE_MARKDOWN = 'markdown';
|
|
|
|
public const TYPE_CODE = 'code';
|
|
|
|
public const TYPE_REFERENCE = 'reference';
|
|
|
|
protected $fillable = [
|
|
'agent_plan_id',
|
|
'key',
|
|
'value',
|
|
'type',
|
|
'description',
|
|
];
|
|
|
|
protected $casts = [
|
|
'value' => 'array',
|
|
];
|
|
|
|
// Relationships
|
|
|
|
public function plan(): BelongsTo
|
|
{
|
|
return $this->belongsTo(AgentPlan::class, 'agent_plan_id');
|
|
}
|
|
|
|
// Scopes
|
|
|
|
public function scopeForPlan($query, AgentPlan|int $plan): mixed
|
|
{
|
|
$planId = $plan instanceof AgentPlan ? $plan->id : $plan;
|
|
|
|
return $query->where('agent_plan_id', $planId);
|
|
}
|
|
|
|
public function scopeOfType($query, string $type): mixed
|
|
{
|
|
return $query->where('type', $type);
|
|
}
|
|
|
|
// Type helpers
|
|
|
|
public function isJson(): bool
|
|
{
|
|
return $this->type === self::TYPE_JSON;
|
|
}
|
|
|
|
public function isMarkdown(): bool
|
|
{
|
|
return $this->type === self::TYPE_MARKDOWN;
|
|
}
|
|
|
|
public function isCode(): bool
|
|
{
|
|
return $this->type === self::TYPE_CODE;
|
|
}
|
|
|
|
public function isReference(): bool
|
|
{
|
|
return $this->type === self::TYPE_REFERENCE;
|
|
}
|
|
|
|
public function getFormattedValue(): string
|
|
{
|
|
if ($this->isMarkdown() || $this->isCode()) {
|
|
return is_string($this->value) ? $this->value : json_encode($this->value, JSON_PRETTY_PRINT);
|
|
}
|
|
|
|
return json_encode($this->value, JSON_PRETTY_PRINT);
|
|
}
|
|
|
|
// Static helpers
|
|
|
|
/**
|
|
* Get a state value for a plan, returning $default if not set.
|
|
*/
|
|
public static function getValue(AgentPlan $plan, string $key, mixed $default = null): mixed
|
|
{
|
|
$state = static::where('agent_plan_id', $plan->id)->where('key', $key)->first();
|
|
|
|
return $state !== null ? $state->value : $default;
|
|
}
|
|
|
|
/**
|
|
* Set (upsert) a state value for a plan.
|
|
*/
|
|
public static function setValue(AgentPlan $plan, string $key, mixed $value, string $type = self::TYPE_JSON): self
|
|
{
|
|
return static::updateOrCreate(
|
|
['agent_plan_id' => $plan->id, 'key' => $key],
|
|
['value' => $value, 'type' => $type]
|
|
);
|
|
}
|
|
|
|
// MCP output
|
|
|
|
public function toMcpContext(): array
|
|
{
|
|
return [
|
|
'key' => $this->key,
|
|
'type' => $this->type,
|
|
'description' => $this->description,
|
|
'value' => $this->value,
|
|
'updated_at' => $this->updated_at?->toIso8601String(),
|
|
];
|
|
}
|
|
}
|