feat: structured skills system with meta skill runtime injection
Replace agent.skills TEXT field with structured skill/skill_file/agent_skill tables. Skills are workspace-level entities with supporting files, reusable across agents via many-to-many bindings. Backend: migration 008, sqlc queries, CRUD handler, agent-skill junction, structured skill loading in task context snapshot. Daemon: meta skill injection via runtime-native config (.claude/CLAUDE.md for Claude, AGENTS.md for Codex) so agents discover .agent_context/ skills through their native mechanism. Lean prompt without inlined skill content. Frontend: Skills management page, agent Skills tab picker, SDK methods, TypeScript types, workspace store integration. Also removes auto-creation of init issues when creating agents. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
parent
63df5dccda
commit
02df33803a
29 changed files with 2320 additions and 192 deletions
382
server/pkg/db/generated/skill.sql.go
Normal file
382
server/pkg/db/generated/skill.sql.go
Normal file
|
|
@ -0,0 +1,382 @@
|
|||
// Code generated by sqlc. DO NOT EDIT.
|
||||
// versions:
|
||||
// sqlc v1.30.0
|
||||
// source: skill.sql
|
||||
|
||||
package db
|
||||
|
||||
import (
|
||||
"context"
|
||||
|
||||
"github.com/jackc/pgx/v5/pgtype"
|
||||
)
|
||||
|
||||
const addAgentSkill = `-- name: AddAgentSkill :exec
|
||||
INSERT INTO agent_skill (agent_id, skill_id)
|
||||
VALUES ($1, $2)
|
||||
ON CONFLICT DO NOTHING
|
||||
`
|
||||
|
||||
type AddAgentSkillParams struct {
|
||||
AgentID pgtype.UUID `json:"agent_id"`
|
||||
SkillID pgtype.UUID `json:"skill_id"`
|
||||
}
|
||||
|
||||
func (q *Queries) AddAgentSkill(ctx context.Context, arg AddAgentSkillParams) error {
|
||||
_, err := q.db.Exec(ctx, addAgentSkill, arg.AgentID, arg.SkillID)
|
||||
return err
|
||||
}
|
||||
|
||||
const createSkill = `-- name: CreateSkill :one
|
||||
INSERT INTO skill (workspace_id, name, description, content, config, created_by)
|
||||
VALUES ($1, $2, $3, $4, $5, $6)
|
||||
RETURNING id, workspace_id, name, description, content, config, created_by, created_at, updated_at
|
||||
`
|
||||
|
||||
type CreateSkillParams struct {
|
||||
WorkspaceID pgtype.UUID `json:"workspace_id"`
|
||||
Name string `json:"name"`
|
||||
Description string `json:"description"`
|
||||
Content string `json:"content"`
|
||||
Config []byte `json:"config"`
|
||||
CreatedBy pgtype.UUID `json:"created_by"`
|
||||
}
|
||||
|
||||
func (q *Queries) CreateSkill(ctx context.Context, arg CreateSkillParams) (Skill, error) {
|
||||
row := q.db.QueryRow(ctx, createSkill,
|
||||
arg.WorkspaceID,
|
||||
arg.Name,
|
||||
arg.Description,
|
||||
arg.Content,
|
||||
arg.Config,
|
||||
arg.CreatedBy,
|
||||
)
|
||||
var i Skill
|
||||
err := row.Scan(
|
||||
&i.ID,
|
||||
&i.WorkspaceID,
|
||||
&i.Name,
|
||||
&i.Description,
|
||||
&i.Content,
|
||||
&i.Config,
|
||||
&i.CreatedBy,
|
||||
&i.CreatedAt,
|
||||
&i.UpdatedAt,
|
||||
)
|
||||
return i, err
|
||||
}
|
||||
|
||||
const deleteSkill = `-- name: DeleteSkill :exec
|
||||
DELETE FROM skill WHERE id = $1
|
||||
`
|
||||
|
||||
func (q *Queries) DeleteSkill(ctx context.Context, id pgtype.UUID) error {
|
||||
_, err := q.db.Exec(ctx, deleteSkill, id)
|
||||
return err
|
||||
}
|
||||
|
||||
const deleteSkillFile = `-- name: DeleteSkillFile :exec
|
||||
DELETE FROM skill_file WHERE id = $1
|
||||
`
|
||||
|
||||
func (q *Queries) DeleteSkillFile(ctx context.Context, id pgtype.UUID) error {
|
||||
_, err := q.db.Exec(ctx, deleteSkillFile, id)
|
||||
return err
|
||||
}
|
||||
|
||||
const deleteSkillFilesBySkill = `-- name: DeleteSkillFilesBySkill :exec
|
||||
DELETE FROM skill_file WHERE skill_id = $1
|
||||
`
|
||||
|
||||
func (q *Queries) DeleteSkillFilesBySkill(ctx context.Context, skillID pgtype.UUID) error {
|
||||
_, err := q.db.Exec(ctx, deleteSkillFilesBySkill, skillID)
|
||||
return err
|
||||
}
|
||||
|
||||
const getSkill = `-- name: GetSkill :one
|
||||
SELECT id, workspace_id, name, description, content, config, created_by, created_at, updated_at FROM skill
|
||||
WHERE id = $1
|
||||
`
|
||||
|
||||
func (q *Queries) GetSkill(ctx context.Context, id pgtype.UUID) (Skill, error) {
|
||||
row := q.db.QueryRow(ctx, getSkill, id)
|
||||
var i Skill
|
||||
err := row.Scan(
|
||||
&i.ID,
|
||||
&i.WorkspaceID,
|
||||
&i.Name,
|
||||
&i.Description,
|
||||
&i.Content,
|
||||
&i.Config,
|
||||
&i.CreatedBy,
|
||||
&i.CreatedAt,
|
||||
&i.UpdatedAt,
|
||||
)
|
||||
return i, err
|
||||
}
|
||||
|
||||
const getSkillFile = `-- name: GetSkillFile :one
|
||||
SELECT id, skill_id, path, content, created_at, updated_at FROM skill_file
|
||||
WHERE id = $1
|
||||
`
|
||||
|
||||
func (q *Queries) GetSkillFile(ctx context.Context, id pgtype.UUID) (SkillFile, error) {
|
||||
row := q.db.QueryRow(ctx, getSkillFile, id)
|
||||
var i SkillFile
|
||||
err := row.Scan(
|
||||
&i.ID,
|
||||
&i.SkillID,
|
||||
&i.Path,
|
||||
&i.Content,
|
||||
&i.CreatedAt,
|
||||
&i.UpdatedAt,
|
||||
)
|
||||
return i, err
|
||||
}
|
||||
|
||||
const listAgentSkills = `-- name: ListAgentSkills :many
|
||||
|
||||
SELECT s.id, s.workspace_id, s.name, s.description, s.content, s.config, s.created_by, s.created_at, s.updated_at FROM skill s
|
||||
JOIN agent_skill ask ON ask.skill_id = s.id
|
||||
WHERE ask.agent_id = $1
|
||||
ORDER BY s.name ASC
|
||||
`
|
||||
|
||||
// Agent-Skill junction
|
||||
func (q *Queries) ListAgentSkills(ctx context.Context, agentID pgtype.UUID) ([]Skill, error) {
|
||||
rows, err := q.db.Query(ctx, listAgentSkills, agentID)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer rows.Close()
|
||||
items := []Skill{}
|
||||
for rows.Next() {
|
||||
var i Skill
|
||||
if err := rows.Scan(
|
||||
&i.ID,
|
||||
&i.WorkspaceID,
|
||||
&i.Name,
|
||||
&i.Description,
|
||||
&i.Content,
|
||||
&i.Config,
|
||||
&i.CreatedBy,
|
||||
&i.CreatedAt,
|
||||
&i.UpdatedAt,
|
||||
); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
items = append(items, i)
|
||||
}
|
||||
if err := rows.Err(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return items, nil
|
||||
}
|
||||
|
||||
const listAgentSkillsByWorkspace = `-- name: ListAgentSkillsByWorkspace :many
|
||||
SELECT ask.agent_id, s.id, s.name, s.description
|
||||
FROM agent_skill ask
|
||||
JOIN skill s ON s.id = ask.skill_id
|
||||
WHERE s.workspace_id = $1
|
||||
ORDER BY s.name ASC
|
||||
`
|
||||
|
||||
type ListAgentSkillsByWorkspaceRow struct {
|
||||
AgentID pgtype.UUID `json:"agent_id"`
|
||||
ID pgtype.UUID `json:"id"`
|
||||
Name string `json:"name"`
|
||||
Description string `json:"description"`
|
||||
}
|
||||
|
||||
func (q *Queries) ListAgentSkillsByWorkspace(ctx context.Context, workspaceID pgtype.UUID) ([]ListAgentSkillsByWorkspaceRow, error) {
|
||||
rows, err := q.db.Query(ctx, listAgentSkillsByWorkspace, workspaceID)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer rows.Close()
|
||||
items := []ListAgentSkillsByWorkspaceRow{}
|
||||
for rows.Next() {
|
||||
var i ListAgentSkillsByWorkspaceRow
|
||||
if err := rows.Scan(
|
||||
&i.AgentID,
|
||||
&i.ID,
|
||||
&i.Name,
|
||||
&i.Description,
|
||||
); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
items = append(items, i)
|
||||
}
|
||||
if err := rows.Err(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return items, nil
|
||||
}
|
||||
|
||||
const listSkillFiles = `-- name: ListSkillFiles :many
|
||||
|
||||
SELECT id, skill_id, path, content, created_at, updated_at FROM skill_file
|
||||
WHERE skill_id = $1
|
||||
ORDER BY path ASC
|
||||
`
|
||||
|
||||
// Skill File CRUD
|
||||
func (q *Queries) ListSkillFiles(ctx context.Context, skillID pgtype.UUID) ([]SkillFile, error) {
|
||||
rows, err := q.db.Query(ctx, listSkillFiles, skillID)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer rows.Close()
|
||||
items := []SkillFile{}
|
||||
for rows.Next() {
|
||||
var i SkillFile
|
||||
if err := rows.Scan(
|
||||
&i.ID,
|
||||
&i.SkillID,
|
||||
&i.Path,
|
||||
&i.Content,
|
||||
&i.CreatedAt,
|
||||
&i.UpdatedAt,
|
||||
); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
items = append(items, i)
|
||||
}
|
||||
if err := rows.Err(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return items, nil
|
||||
}
|
||||
|
||||
const listSkillsByWorkspace = `-- name: ListSkillsByWorkspace :many
|
||||
|
||||
SELECT id, workspace_id, name, description, content, config, created_by, created_at, updated_at FROM skill
|
||||
WHERE workspace_id = $1
|
||||
ORDER BY name ASC
|
||||
`
|
||||
|
||||
// Skill CRUD
|
||||
func (q *Queries) ListSkillsByWorkspace(ctx context.Context, workspaceID pgtype.UUID) ([]Skill, error) {
|
||||
rows, err := q.db.Query(ctx, listSkillsByWorkspace, workspaceID)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer rows.Close()
|
||||
items := []Skill{}
|
||||
for rows.Next() {
|
||||
var i Skill
|
||||
if err := rows.Scan(
|
||||
&i.ID,
|
||||
&i.WorkspaceID,
|
||||
&i.Name,
|
||||
&i.Description,
|
||||
&i.Content,
|
||||
&i.Config,
|
||||
&i.CreatedBy,
|
||||
&i.CreatedAt,
|
||||
&i.UpdatedAt,
|
||||
); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
items = append(items, i)
|
||||
}
|
||||
if err := rows.Err(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return items, nil
|
||||
}
|
||||
|
||||
const removeAgentSkill = `-- name: RemoveAgentSkill :exec
|
||||
DELETE FROM agent_skill
|
||||
WHERE agent_id = $1 AND skill_id = $2
|
||||
`
|
||||
|
||||
type RemoveAgentSkillParams struct {
|
||||
AgentID pgtype.UUID `json:"agent_id"`
|
||||
SkillID pgtype.UUID `json:"skill_id"`
|
||||
}
|
||||
|
||||
func (q *Queries) RemoveAgentSkill(ctx context.Context, arg RemoveAgentSkillParams) error {
|
||||
_, err := q.db.Exec(ctx, removeAgentSkill, arg.AgentID, arg.SkillID)
|
||||
return err
|
||||
}
|
||||
|
||||
const removeAllAgentSkills = `-- name: RemoveAllAgentSkills :exec
|
||||
DELETE FROM agent_skill WHERE agent_id = $1
|
||||
`
|
||||
|
||||
func (q *Queries) RemoveAllAgentSkills(ctx context.Context, agentID pgtype.UUID) error {
|
||||
_, err := q.db.Exec(ctx, removeAllAgentSkills, agentID)
|
||||
return err
|
||||
}
|
||||
|
||||
const updateSkill = `-- name: UpdateSkill :one
|
||||
UPDATE skill SET
|
||||
name = COALESCE($2, name),
|
||||
description = COALESCE($3, description),
|
||||
content = COALESCE($4, content),
|
||||
config = COALESCE($5, config),
|
||||
updated_at = now()
|
||||
WHERE id = $1
|
||||
RETURNING id, workspace_id, name, description, content, config, created_by, created_at, updated_at
|
||||
`
|
||||
|
||||
type UpdateSkillParams struct {
|
||||
ID pgtype.UUID `json:"id"`
|
||||
Name pgtype.Text `json:"name"`
|
||||
Description pgtype.Text `json:"description"`
|
||||
Content pgtype.Text `json:"content"`
|
||||
Config []byte `json:"config"`
|
||||
}
|
||||
|
||||
func (q *Queries) UpdateSkill(ctx context.Context, arg UpdateSkillParams) (Skill, error) {
|
||||
row := q.db.QueryRow(ctx, updateSkill,
|
||||
arg.ID,
|
||||
arg.Name,
|
||||
arg.Description,
|
||||
arg.Content,
|
||||
arg.Config,
|
||||
)
|
||||
var i Skill
|
||||
err := row.Scan(
|
||||
&i.ID,
|
||||
&i.WorkspaceID,
|
||||
&i.Name,
|
||||
&i.Description,
|
||||
&i.Content,
|
||||
&i.Config,
|
||||
&i.CreatedBy,
|
||||
&i.CreatedAt,
|
||||
&i.UpdatedAt,
|
||||
)
|
||||
return i, err
|
||||
}
|
||||
|
||||
const upsertSkillFile = `-- name: UpsertSkillFile :one
|
||||
INSERT INTO skill_file (skill_id, path, content)
|
||||
VALUES ($1, $2, $3)
|
||||
ON CONFLICT (skill_id, path) DO UPDATE SET
|
||||
content = EXCLUDED.content,
|
||||
updated_at = now()
|
||||
RETURNING id, skill_id, path, content, created_at, updated_at
|
||||
`
|
||||
|
||||
type UpsertSkillFileParams struct {
|
||||
SkillID pgtype.UUID `json:"skill_id"`
|
||||
Path string `json:"path"`
|
||||
Content string `json:"content"`
|
||||
}
|
||||
|
||||
func (q *Queries) UpsertSkillFile(ctx context.Context, arg UpsertSkillFileParams) (SkillFile, error) {
|
||||
row := q.db.QueryRow(ctx, upsertSkillFile, arg.SkillID, arg.Path, arg.Content)
|
||||
var i SkillFile
|
||||
err := row.Scan(
|
||||
&i.ID,
|
||||
&i.SkillID,
|
||||
&i.Path,
|
||||
&i.Content,
|
||||
&i.CreatedAt,
|
||||
&i.UpdatedAt,
|
||||
)
|
||||
return i, err
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue