Compare commits
1 Commits
main
..
4bb3e1af58
| Author | SHA1 | Date | |
|---|---|---|---|
| 4bb3e1af58 |
@@ -31,9 +31,6 @@ jobs:
|
||||
- name: Download dependencies
|
||||
run: go mod download
|
||||
|
||||
- name: Tidy modules
|
||||
run: go mod tidy
|
||||
|
||||
- name: Run tests
|
||||
run: go test ./...
|
||||
|
||||
|
||||
+1
-2
@@ -33,6 +33,5 @@ bin/
|
||||
OB1/
|
||||
ui/node_modules/
|
||||
ui/.svelte-kit/
|
||||
internal/app/ui/dist/*
|
||||
!internal/app/ui/dist/placeholder.txt
|
||||
internal/app/ui/dist/
|
||||
.codex
|
||||
|
||||
@@ -1,101 +0,0 @@
|
||||
## What This Project Is
|
||||
|
||||
AMCS (Avalon Memory Control Service) is an MCP server written in Go that provides memory, knowledge, and task management for AI agents. It exposes 56+ tools via HTTP/SSE MCP transport and includes a Svelte 5 admin UI embedded in the binary.
|
||||
|
||||
# Agent Rules
|
||||
|
||||
Keep your answers short. Question everything, never assume or guess. Ask the user if you are unsure about anything.
|
||||
You must use the AMCS MCP system tools. Set Origin as the active project and always capture summaries of what was done as thoughts using the capture_thought tool. If amcs is not available write files to doc/llm/log/yyyymmdd_hh.md
|
||||
When working on postgresql use the postgresql skill. On go/api/backend the go-skill and for the frontend the "svelte-architect" skill.
|
||||
|
||||
## Commands
|
||||
|
||||
### Build & Run
|
||||
```bash
|
||||
make build # Build server binary (outputs to bin/amcs-server), embeds UI
|
||||
make build-cli # Build CLI binary (bin/amcs-cli)
|
||||
make clean # Remove bin/
|
||||
|
||||
# Run locally (builds UI then starts server)
|
||||
./scripts/run-local.sh configs/dev.yaml
|
||||
|
||||
# UI hot-reload dev server (proxy to backend on :8080)
|
||||
make ui-dev # http://localhost:5173
|
||||
```
|
||||
|
||||
### Testing
|
||||
```bash
|
||||
make test # All tests (svelte-check + go test ./...)
|
||||
go test ./internal/tools -run TestFunctionName -v # Single test
|
||||
go test ./internal/tools -v # Package tests
|
||||
```
|
||||
|
||||
### Database
|
||||
```bash
|
||||
make migrate # Apply SQL migrations
|
||||
make check-schema-drift # Verify no drift between DBML and SQL
|
||||
```
|
||||
|
||||
## Architecture
|
||||
|
||||
### Key Pattern: Store → Tool → MCP Handler
|
||||
|
||||
Every tool domain follows the same three-layer pattern:
|
||||
|
||||
1. **Store** (`internal/store/`) — database access via BUN ORM + pgx. One file per domain (e.g., `thoughts.go`, `plans.go`).
|
||||
2. **Tool** (`internal/tools/`) — MCP tool handler structs with a `Handle(ctx, req, input)` method. Input/output types are plain Go structs; JSON schemas are auto-generated from struct tags.
|
||||
3. **Registration** (`internal/mcpserver/handlers.go`) — each domain has a `registerXxxTools(server, logger, ts)` function called from `NewHandlers`.
|
||||
|
||||
### Adding a New Tool Domain
|
||||
|
||||
Follow the checklist in `internal/tools/` (see `project_conventions.md` in memory, or look at any existing domain like `skills.go` as a reference):
|
||||
- Define input/output structs with `jsonschema` tags
|
||||
- Implement a struct with `Handle` method
|
||||
- Add the struct to `ToolSet` in `mcpserver/handlers.go`
|
||||
- Wire the store in `internal/store/db.go`
|
||||
- Add a `registerXxxTools` call in `NewHandlers`
|
||||
|
||||
### AI Provider Architecture
|
||||
|
||||
Configured in YAML under `ai.providers`. Each role (embeddings, metadata extraction) supports a primary provider and a fallback chain. Providers are named references (litellm, ollama, openrouter) that resolve to OpenAI-compatible endpoints. Health tracking per provider with cooldown on transient failures.
|
||||
|
||||
### Error Handling
|
||||
|
||||
All tool errors return structured `mcperrors.RPCError` with:
|
||||
- `data.type` — machine-readable category (`invalid_input`, `project_required`, `entity_not_found`, etc.)
|
||||
- `data.field` / `data.detail` / `data.hint` — structured validation data
|
||||
|
||||
Use helpers in `internal/mcperrors/` rather than raw errors.
|
||||
|
||||
### Authentication
|
||||
|
||||
- API key via `x-brain-key` header (dev/simple deployments)
|
||||
- OAuth 2.0 client credentials for production
|
||||
- Session store tracks active project per client connection
|
||||
|
||||
### Database Schema
|
||||
|
||||
DBML files in `schema/` are the source of truth for the database schema. Never edit the generated SQL or Go models directly — always edit the DBML, then regenerate:
|
||||
|
||||
```bash
|
||||
make generate-migrations # DBML → SQL migration files
|
||||
make generate-models # DBML → Go model structs
|
||||
make migrate # Apply pending migrations to the database
|
||||
```
|
||||
|
||||
All primary keys are `bigserial int64` (migrated from UUID in migration 015).
|
||||
|
||||
### Frontend
|
||||
|
||||
Svelte 5 + Tailwind + Skeleton UI in `ui/`. Built via pnpm and embedded into the Go binary as a static filesystem. The Vite dev server proxies API calls to the backend on `:8080`.
|
||||
|
||||
The UI uses **resolvespec-js** for admin CRUD views. The backend API for the UI follows the **ResolveSpec** convention — resource endpoints, filtering, and pagination are structured according to that spec.
|
||||
|
||||
## Configuration
|
||||
|
||||
YAML config with schema `version: 2`. Key env var overrides:
|
||||
- `DATABASE_URL` — PostgreSQL connection string
|
||||
- `AMCS_LITELLM_API_KEY`, `AMCS_OPENROUTER_API_KEY` — AI provider keys
|
||||
- `AMCS_SERVER_PORT` — defaults to 8080
|
||||
|
||||
See `configs/config.example.yaml` for the full schema. Config auto-migrates v1→v2 on startup.
|
||||
@@ -4,19 +4,14 @@ SERVER_BIN := $(BIN_DIR)/amcs-server
|
||||
CMD_SERVER := ./cmd/amcs-server
|
||||
BUILDINFO_PKG := git.warky.dev/wdevs/amcs/internal/buildinfo
|
||||
UI_DIR := $(CURDIR)/ui
|
||||
AMCS_UI_BACKEND ?= http://127.0.0.1:8080
|
||||
PATCH_INCREMENT ?= 1
|
||||
RELEASE_VERSION ?=
|
||||
RELEASE_REMOTE ?= origin
|
||||
VERSION_TAG ?= $(shell git describe --tags --exact-match 2>/dev/null || echo dev)
|
||||
COMMIT_SHA ?= $(shell git rev-parse --short HEAD 2>/dev/null || echo unknown)
|
||||
BUILD_DATE ?= $(shell date -u +%Y-%m-%dT%H:%M:%SZ)
|
||||
RELSPEC ?= $(shell command -v relspec 2>/dev/null || echo $(HOME)/go/bin/relspec)
|
||||
#RELSPEC = /mnt/vault/relspecgo/build/relspec
|
||||
SCHEMA_FILES := $(sort $(wildcard schema/*.dbml))
|
||||
MERGE_TARGET_TMP := $(CURDIR)/.cache/schema.merge-target.dbml
|
||||
GENERATED_SCHEMA_MIGRATION := migrations/020_generated_schema.sql
|
||||
GENERATED_MODELS_DIR := internal/generatedmodels
|
||||
PNPM ?= pnpm
|
||||
LDFLAGS := -s -w \
|
||||
-X $(BUILDINFO_PKG).Version=$(VERSION_TAG) \
|
||||
@@ -24,27 +19,10 @@ LDFLAGS := -s -w \
|
||||
-X $(BUILDINFO_PKG).Commit=$(COMMIT_SHA) \
|
||||
-X $(BUILDINFO_PKG).BuildDate=$(BUILD_DATE)
|
||||
|
||||
.PHONY: all build clean migrate release-version release-build test generate-migrations generate-models check-schema-drift build-cli ui-install ui-build ui-dev ui-check help
|
||||
.PHONY: all build clean migrate release-version test generate-migrations check-schema-drift build-cli ui-install ui-build ui-dev ui-check
|
||||
|
||||
all: build
|
||||
|
||||
help:
|
||||
@echo "Available targets:"
|
||||
@echo " build Build server binary (includes UI build)"
|
||||
@echo " build-cli Build CLI binary"
|
||||
@echo " test Run all tests (includes UI check)"
|
||||
@echo " clean Remove build artifacts"
|
||||
@echo " migrate Run database migrations"
|
||||
@echo " release-version Tag and push a release (auto patch bump or RELEASE_VERSION=vX.Y.Z)"
|
||||
@echo " release-build Build with a specific release tag (RELEASE_VERSION=vX.Y.Z)"
|
||||
@echo " generate-migrations Generate SQL migration from DBML schema files"
|
||||
@echo " generate-models Generate Go models from DBML schema"
|
||||
@echo " check-schema-drift Verify generated migration matches current schema"
|
||||
@echo " ui-install Install UI dependencies"
|
||||
@echo " ui-build Build UI assets"
|
||||
@echo " ui-dev Start UI dev server with local API proxy"
|
||||
@echo " ui-check Run UI type checks"
|
||||
|
||||
build: ui-build
|
||||
@mkdir -p $(BIN_DIR)
|
||||
go build -ldflags "$(LDFLAGS)" -o $(SERVER_BIN) $(CMD_SERVER)
|
||||
@@ -56,7 +34,7 @@ ui-build: ui-install
|
||||
cd $(UI_DIR) && $(PNPM) run build
|
||||
|
||||
ui-dev: ui-install
|
||||
cd $(UI_DIR) && VITE_API_URL=/api AMCS_UI_BACKEND=$(AMCS_UI_BACKEND) $(PNPM) run dev
|
||||
cd $(UI_DIR) && $(PNPM) run dev
|
||||
|
||||
ui-check: ui-install
|
||||
cd $(UI_DIR) && $(PNPM) run check
|
||||
@@ -69,41 +47,22 @@ release-version:
|
||||
@case "$(PATCH_INCREMENT)" in \
|
||||
''|*[!0-9]*|0) echo "PATCH_INCREMENT must be a positive integer" >&2; exit 1 ;; \
|
||||
esac
|
||||
@if ! git diff --quiet || ! git diff --cached --quiet; then \
|
||||
echo "Refusing to release from a dirty working tree. Commit or stash changes first." >&2; \
|
||||
exit 1; \
|
||||
fi
|
||||
@next_tag="$(RELEASE_VERSION)"; \
|
||||
if [ -z "$$next_tag" ]; then \
|
||||
latest=$$(git tag --list 'v[0-9]*.[0-9]*.[0-9]*' --sort=-v:refname | head -n 1); \
|
||||
if [ -z "$$latest" ]; then latest="v0.0.0"; fi; \
|
||||
version=$${latest#v}; \
|
||||
major=$${version%%.*}; \
|
||||
rest=$${version#*.}; \
|
||||
minor=$${rest%%.*}; \
|
||||
patch=$${rest##*.}; \
|
||||
next_patch=$$((patch + $(PATCH_INCREMENT))); \
|
||||
next_tag="v$$major.$$minor.$$next_patch"; \
|
||||
fi; \
|
||||
case "$$next_tag" in \
|
||||
v[0-9]*.[0-9]*.[0-9]*) ;; \
|
||||
*) echo "RELEASE_VERSION must look like vX.Y.Z (got '$$next_tag')" >&2; exit 1 ;; \
|
||||
esac; \
|
||||
@latest=$$(git tag --list 'v[0-9]*.[0-9]*.[0-9]*' --sort=-v:refname | head -n 1); \
|
||||
if [ -z "$$latest" ]; then latest="v0.0.0"; fi; \
|
||||
version=$${latest#v}; \
|
||||
major=$${version%%.*}; \
|
||||
rest=$${version#*.}; \
|
||||
minor=$${rest%%.*}; \
|
||||
patch=$${rest##*.}; \
|
||||
next_patch=$$((patch + $(PATCH_INCREMENT))); \
|
||||
next_tag="v$$major.$$minor.$$next_patch"; \
|
||||
if git rev-parse -q --verify "refs/tags/$$next_tag" >/dev/null; then \
|
||||
echo "$$next_tag already exists" >&2; \
|
||||
exit 1; \
|
||||
fi; \
|
||||
git tag -a "$$next_tag" -m "Release $$next_tag"; \
|
||||
git push $(RELEASE_REMOTE) "$$next_tag"; \
|
||||
$(MAKE) release-build RELEASE_VERSION="$$next_tag"; \
|
||||
echo "Released $$next_tag"
|
||||
|
||||
release-build:
|
||||
@case "$(RELEASE_VERSION)" in \
|
||||
v[0-9]*.[0-9]*.[0-9]*) ;; \
|
||||
*) echo "RELEASE_VERSION must look like vX.Y.Z" >&2; exit 1 ;; \
|
||||
esac
|
||||
@$(MAKE) build build-cli VERSION_TAG="$(RELEASE_VERSION)"
|
||||
git push origin "$$next_tag"; \
|
||||
echo "$$next_tag"
|
||||
|
||||
migrate:
|
||||
./scripts/migrate.sh
|
||||
@@ -119,10 +78,6 @@ generate-migrations:
|
||||
@schema_list=$$(printf '%s\n' $(SCHEMA_FILES) | paste -sd, -); \
|
||||
$(RELSPEC) merge --target dbml --target-path $(MERGE_TARGET_TMP) --source dbml --from-list "$$schema_list" --output pgsql --output-path $(GENERATED_SCHEMA_MIGRATION)
|
||||
|
||||
generate-models:
|
||||
@test -n "$(SCHEMA_FILES)" || (echo "No DBML schema files found in schema/" >&2; exit 1)
|
||||
@./scripts/generate-models.sh
|
||||
|
||||
check-schema-drift:
|
||||
@test -f $(GENERATED_SCHEMA_MIGRATION) || (echo "$(GENERATED_SCHEMA_MIGRATION) is missing; run make generate-migrations" >&2; exit 1)
|
||||
@command -v $(RELSPEC) >/dev/null 2>&1 || (echo "relspec not found; install git.warky.dev/wdevs/relspecgo/cmd/relspec@latest" >&2; exit 1)
|
||||
|
||||
@@ -1,18 +1,26 @@
|
||||
# AMCS Directory
|
||||
# Avalon Memory Crystal Server (amcs)
|
||||
|
||||
This is the AMCS (Avalon Memory Control Service) directory.
|
||||

|
||||
|
||||
## Purpose
|
||||
A Go MCP server for capturing and retrieving thoughts, memory, and project context. Exposes tools over Streamable HTTP, backed by Postgres with pgvector for semantic search.
|
||||
|
||||
The AMCS directory is used to store configuration and code for the Avalon Memory Control Service, which handles...
|
||||
The structured learnings feature adds a separate record type for distilled, reusable knowledge with provenance, verification, and actionability fields, while leaving thoughts as the original captured notes.
|
||||
|
||||
## Structure
|
||||
## What it does
|
||||
|
||||
- `configs/` - Configuration files
|
||||
- `scripts/` - Scripts for managing the system
|
||||
- `assets/` - Asset files
|
||||
- **Capture** thoughts with automatic embedding and metadata extraction
|
||||
- **Search** thoughts semantically via vector similarity
|
||||
- **Organise** thoughts into projects and retrieve full project context
|
||||
- **Summarise** and recall memory across topics and time windows
|
||||
- **Link** related thoughts and traverse relationships
|
||||
|
||||
## Next Steps
|
||||
## Stack
|
||||
|
||||
- Go — MCP server over Streamable HTTP
|
||||
- Postgres + pgvector — storage and vector search
|
||||
- LiteLLM — primary hosted AI provider (embeddings + metadata extraction)
|
||||
- OpenRouter — default upstream behind LiteLLM
|
||||
- Ollama — supported local or self-hosted OpenAI-compatible provider
|
||||
|
||||
## Tools
|
||||
|
||||
@@ -31,9 +39,6 @@ The AMCS directory is used to store configuration and code for the Avalon Memory
|
||||
| `get_project_context` | Recent + semantic context for a project; uses explicit `project` or the active session project |
|
||||
| `set_active_project` | Set session project scope; requires a stateful MCP session |
|
||||
| `get_active_project` | Get current session project |
|
||||
| `add_learning` | Create a curated learning record distinct from raw thoughts |
|
||||
| `get_learning` | Retrieve a structured learning by ID |
|
||||
| `list_learnings` | List structured learnings by project/category/area/status/priority/tag/query |
|
||||
| `summarize_thoughts` | LLM prose summary over a filtered set |
|
||||
| `recall_context` | Semantic + recency context block for injection |
|
||||
| `link_thoughts` | Create a typed relationship between thoughts |
|
||||
@@ -69,17 +74,6 @@ The AMCS directory is used to store configuration and code for the Avalon Memory
|
||||
| `describe_tools` | List all available MCP tools with names, descriptions, categories, and model-authored usage notes; call this at the start of a session to orient yourself |
|
||||
| `annotate_tool` | Persist your own usage notes for a specific tool; notes are returned by `describe_tools` in future sessions |
|
||||
|
||||
## Learnings
|
||||
|
||||
Learnings are curated, structured memory records for durable insights you want to keep distinct from raw thoughts. Use them for normalized lessons, decisions, and evidence-backed findings that should be easy to retrieve and review over time.
|
||||
|
||||
Compared with `capture_thought`, learnings are more explicit and reviewable: they include a required `summary`, optional `details`, and structured fields like `category`, `area`, `status`, `priority`, `confidence`, and `tags`, plus optional links to a `project`, `related_thought_id`, or `related_skill_id`.
|
||||
|
||||
Use:
|
||||
- `add_learning` to create a curated learning.
|
||||
- `get_learning` to fetch one by ID.
|
||||
- `list_learnings` to filter curated learnings across project and status dimensions.
|
||||
|
||||
## Self-Documenting Tools
|
||||
|
||||
AMCS includes a built-in tool directory that models can read and annotate.
|
||||
@@ -612,26 +606,6 @@ Run the SQL migrations against a local database with:
|
||||
|
||||
The web UI now lives in the top-level `ui/` module and is embedded into the Go binary at build time with `go:embed`.
|
||||
|
||||
### Admin UI deployment model
|
||||
|
||||
AMCS uses a **lightweight embedded SPA panel** model:
|
||||
|
||||
- the Svelte admin app is compiled to static assets
|
||||
- assets are embedded in the server binary and served from `/`
|
||||
- backend APIs (`/api/status`, `/api/rs/*`, admin action routes, OAuth endpoints) stay on the same origin
|
||||
- auth is enforced server-side for all sensitive API routes
|
||||
|
||||
This keeps deployment simple (single binary/container) while preserving SPA ergonomics for operator workflows.
|
||||
|
||||
### UI stack baseline
|
||||
|
||||
The admin frontend baseline is:
|
||||
|
||||
- Svelte 5 for the app shell and pages
|
||||
- ResolveSpec-backed APIs for data access
|
||||
- `@warkypublic/svelix` for admin UX components (including `GridlerFull` and form controllers)
|
||||
- `@warkypublic/artemis-kit` as the default JavaScript tooling dependency baseline in `ui/package.json`
|
||||
|
||||
**Use `pnpm` for all UI work in this repo.**
|
||||
|
||||
- `make build` — runs the real UI build first, then compiles the Go server
|
||||
|
||||
Binary file not shown.
|
Before Width: | Height: | Size: 13 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 2.1 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 11 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 6.3 KiB |
+2
-2
@@ -25,8 +25,8 @@ auth:
|
||||
oauth:
|
||||
clients:
|
||||
- id: "oauth-client"
|
||||
client_id: "test_aab32200464910ab697efbd760e7ed2c"
|
||||
client_secret: "test_135369559a422b4b93fcb534a4aed2c9"
|
||||
client_id: ""
|
||||
client_secret: ""
|
||||
description: "used when auth.mode=oauth_client_credentials"
|
||||
|
||||
database:
|
||||
|
||||
@@ -1,20 +0,0 @@
|
||||
Completed personas UI reachability and fixed the missing backend ResolveSpec registrations that blocked it.
|
||||
|
||||
- Added Personas navigation/page wiring in the Svelte admin shell.
|
||||
- Added a personas overview page with tabs for personas, parts, and traits.
|
||||
- Expanded the persona inspector to load linked parts, traits, skills, guardrails, and arc state.
|
||||
- Found that `/api/rs/public/agent_parts` and related persona routes were missing because `internal/app/resolvespec_admin.go` manually whitelists ResolveSpec models.
|
||||
- Registered persona-related ResolveSpec models: `agent_personas`, `agent_parts`, `agent_traits`, persona join tables, arc tables, and `persona_arc`.
|
||||
- Allowed ResolveSpec mutations for `agent_personas`, `agent_parts`, and `agent_traits`.
|
||||
- Verified the `internal/app` package still compiles with `env GOCACHE=/tmp/amcs-go-cache go test -run '^$' ./internal/app`.
|
||||
|
||||
Follow-up:
|
||||
|
||||
- Automated ResolveSpec model registration generation with `relspec templ`.
|
||||
- Added `scripts/templates/resolvespec_models.tmpl`.
|
||||
- Updated `scripts/generate-models.sh` to generate `internal/app/resolvespec_models_generated.go`.
|
||||
- Removed the handwritten `resolveSpecModels()` from `internal/app/resolvespec_admin.go`.
|
||||
- Extended `scripts/patch-generated-models.sh` to fix current relspec output quirks:
|
||||
- incorrect `persona_arc` primary-key cast
|
||||
- unused `resolvespec_common` imports in join-table models
|
||||
- Added focused tests covering persona entity presence and persona mutation allowlisting.
|
||||
@@ -1,15 +0,0 @@
|
||||
Fixed the Gitea build break caused by `go:embed` requiring `internal/app/ui/dist` to exist in a clean checkout.
|
||||
|
||||
Changes made:
|
||||
- Added `internal/app/ui/dist/placeholder.txt` so the embedded UI directory is always present in source control.
|
||||
- Updated `internal/mcpserver/server_test.go` to derive expected tool names from `BuildToolCatalog()` instead of a stale hard-coded list.
|
||||
- Removed stale maintenance tool entries from `internal/mcpserver/server.go` because those tools are not currently registered.
|
||||
|
||||
Verification:
|
||||
- `env GOCACHE=/tmp/go-build go test ./internal/mcpserver -run TestNewListsAllRegisteredTools -v` passes.
|
||||
- A broader `go test ./internal/app ./cmd/amcs-server` compile check was started, but it did not finish before this log entry was written.
|
||||
|
||||
Migration follow-up:
|
||||
- Fixed `migrations/020_generated_schema.sql` after PostgreSQL failed with `operator does not exist: name[] = text[]`.
|
||||
- Root cause: `pg_attribute.attname` is type `name`, so `ARRAY(SELECT a.attname ...)` produced `name[]`, which was compared against `text[]` literals.
|
||||
- Updated each repeated primary-key introspection block to use `SELECT a.attname::text`, keeping the existing `ARRAY[]::text[]` and `ARRAY['id']` / `ARRAY['persona_id']` comparisons valid.
|
||||
@@ -3,85 +3,29 @@ module git.warky.dev/wdevs/amcs
|
||||
go 1.26.1
|
||||
|
||||
require (
|
||||
github.com/bitechdev/ResolveSpec v1.0.87
|
||||
github.com/google/jsonschema-go v0.4.2
|
||||
github.com/google/uuid v1.6.0
|
||||
github.com/jackc/pgx/v5 v5.9.1
|
||||
github.com/modelcontextprotocol/go-sdk v1.4.1
|
||||
github.com/pgvector/pgvector-go v0.3.0
|
||||
github.com/spf13/cobra v1.10.2
|
||||
github.com/uptrace/bun v1.2.16
|
||||
github.com/uptrace/bun/dialect/pgdialect v1.2.16
|
||||
github.com/uptrace/bun/driver/pgdriver v1.1.12
|
||||
github.com/uptrace/bunrouter v1.0.23
|
||||
golang.org/x/sync v0.19.0
|
||||
golang.org/x/sync v0.17.0
|
||||
gopkg.in/yaml.v3 v3.0.1
|
||||
)
|
||||
|
||||
require (
|
||||
github.com/beorn7/perks v1.0.1 // indirect
|
||||
github.com/bradfitz/gomemcache v0.0.0-20250403215159-8d39553ac7cf // indirect
|
||||
github.com/cespare/xxhash/v2 v2.3.0 // indirect
|
||||
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect
|
||||
github.com/fsnotify/fsnotify v1.9.0 // indirect
|
||||
github.com/getsentry/sentry-go v0.40.0 // indirect
|
||||
github.com/go-viper/mapstructure/v2 v2.4.0 // indirect
|
||||
github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9 // indirect
|
||||
github.com/golang-sql/sqlexp v0.1.0 // indirect
|
||||
github.com/gorilla/mux v1.8.1 // indirect
|
||||
github.com/inconshreveable/mousetrap v1.1.0 // indirect
|
||||
github.com/jackc/pgpassfile v1.0.0 // indirect
|
||||
github.com/jackc/pgservicefile v0.0.0-20240606120523-5a60cdf6a761 // indirect
|
||||
github.com/jackc/puddle/v2 v2.2.2 // indirect
|
||||
github.com/jinzhu/inflection v1.0.0 // indirect
|
||||
github.com/jinzhu/now v1.1.5 // indirect
|
||||
github.com/mattn/go-sqlite3 v1.14.33 // indirect
|
||||
github.com/microsoft/go-mssqldb v1.9.5 // indirect
|
||||
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect
|
||||
github.com/pelletier/go-toml/v2 v2.2.4 // indirect
|
||||
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||
github.com/prometheus/client_golang v1.23.2 // indirect
|
||||
github.com/prometheus/client_model v0.6.2 // indirect
|
||||
github.com/prometheus/common v0.67.4 // indirect
|
||||
github.com/prometheus/procfs v0.19.2 // indirect
|
||||
github.com/puzpuzpuz/xsync/v3 v3.5.1 // indirect
|
||||
github.com/redis/go-redis/v9 v9.17.2 // indirect
|
||||
github.com/kr/text v0.2.0 // indirect
|
||||
github.com/rogpeppe/go-internal v1.14.1 // indirect
|
||||
github.com/sagikazarmark/locafero v0.12.0 // indirect
|
||||
github.com/segmentio/asm v1.1.3 // indirect
|
||||
github.com/segmentio/encoding v0.5.4 // indirect
|
||||
github.com/shopspring/decimal v1.4.0 // indirect
|
||||
github.com/spf13/afero v1.15.0 // indirect
|
||||
github.com/spf13/cast v1.10.0 // indirect
|
||||
github.com/spf13/pflag v1.0.10 // indirect
|
||||
github.com/spf13/viper v1.21.0 // indirect
|
||||
github.com/stretchr/testify v1.11.1 // indirect
|
||||
github.com/subosito/gotenv v1.6.0 // indirect
|
||||
github.com/tidwall/gjson v1.18.0 // indirect
|
||||
github.com/tidwall/match v1.2.0 // indirect
|
||||
github.com/tidwall/pretty v1.2.1 // indirect
|
||||
github.com/tidwall/sjson v1.2.5 // indirect
|
||||
github.com/tmthrgd/go-hex v0.0.0-20190904060850-447a3041c3bc // indirect
|
||||
github.com/uptrace/bun/dialect/mssqldialect v1.2.16 // indirect
|
||||
github.com/uptrace/bun/dialect/sqlitedialect v1.2.16 // indirect
|
||||
github.com/vmihailenco/msgpack/v5 v5.4.1 // indirect
|
||||
github.com/vmihailenco/tagparser/v2 v2.0.0 // indirect
|
||||
github.com/spf13/pflag v1.0.9 // indirect
|
||||
github.com/x448/float16 v0.8.4 // indirect
|
||||
github.com/yosida95/uritemplate/v3 v3.0.2 // indirect
|
||||
go.uber.org/multierr v1.11.0 // indirect
|
||||
go.uber.org/zap v1.27.1 // indirect
|
||||
go.yaml.in/yaml/v2 v2.4.3 // indirect
|
||||
go.yaml.in/yaml/v3 v3.0.4 // indirect
|
||||
golang.org/x/crypto v0.46.0 // indirect
|
||||
golang.org/x/mod v0.31.0 // indirect
|
||||
golang.org/x/oauth2 v0.34.0 // indirect
|
||||
golang.org/x/sys v0.40.0 // indirect
|
||||
golang.org/x/text v0.32.0 // indirect
|
||||
google.golang.org/protobuf v1.36.11 // indirect
|
||||
gorm.io/driver/postgres v1.6.0 // indirect
|
||||
gorm.io/driver/sqlite v1.6.0 // indirect
|
||||
gorm.io/driver/sqlserver v1.6.3 // indirect
|
||||
gorm.io/gorm v1.31.1 // indirect
|
||||
mellium.im/sasl v0.3.1 // indirect
|
||||
golang.org/x/text v0.29.0 // indirect
|
||||
)
|
||||
|
||||
@@ -1,129 +1,22 @@
|
||||
dario.cat/mergo v1.0.2 h1:85+piFYR1tMbRrLcDwR18y4UKJ3aH1Tbzi24VRW1TK8=
|
||||
dario.cat/mergo v1.0.2/go.mod h1:E/hbnu0NxMFBjpMIE34DRGLWqDy0g5FuKDhCb31ngxA=
|
||||
entgo.io/ent v0.14.3 h1:wokAV/kIlH9TeklJWGGS7AYJdVckr0DloWjIcO9iIIQ=
|
||||
entgo.io/ent v0.14.3/go.mod h1:aDPE/OziPEu8+OWbzy4UlvWmD2/kbRuWfK2A40hcxJM=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.7.0/go.mod h1:bjGvMhVMb+EEm3VRNQawDMUyMMjo+S5ewNjflkep/0Q=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.7.1/go.mod h1:bjGvMhVMb+EEm3VRNQawDMUyMMjo+S5ewNjflkep/0Q=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.11.1/go.mod h1:a6xsAQUZg+VsS3TJ05SRp524Hs4pZ/AeFSr5ENf0Yjo=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.18.0 h1:Gt0j3wceWMwPmiazCa8MzMA0MfhmPIz0Qp0FJ6qcM0U=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.18.0/go.mod h1:Ot/6aikWnKWi4l9QB7qVSwa8iMphQNqkWALMoNT3rzM=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.3.1/go.mod h1:uE9zaUfEQT/nbQjVi2IblCG9iaLtZsuYZ8ne+PuQ02M=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.6.0/go.mod h1:9kIvujWAA58nmPmWB1m23fyWic1kYZMxD9CxaWn4Qpg=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.10.1 h1:B+blDbyVIG3WaikNxPnhPiJ1MThR03b3vKGtER95TP4=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.10.1/go.mod h1:JdM5psgjfBf5fo2uWOZhflPWyDBZ/O/CNAH9CtsuZE4=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/internal v1.3.0/go.mod h1:okt5dMMTOFjX/aovMlrjvvXoPMBVSPzk9185BT0+eZM=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/internal v1.5.2/go.mod h1:yInRyqWXAuaPrgI7p70+lDDgh3mlBohis29jGMISnmc=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/internal v1.8.0/go.mod h1:4OG6tQ9EOP/MT0NMjDlRzWoVFxfu9rN9B2X+tlSVktg=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.1 h1:FPKJS1T+clwv+OLGt13a8UjqeRuh0O4SJ3lUriThc+4=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.1/go.mod h1:j2chePtV91HrC22tGoRX3sGY42uF13WzmmV80/OdVAA=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/azkeys v1.0.1/go.mod h1:GpPjLhVR9dnUoJMyHWSPy71xY9/lcmpzIPZXmF0FCVY=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/azkeys v1.3.1 h1:Wgf5rZba3YZqeTNJPtvqZoBu1sBN/L4sry+u2U3Y75w=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/azkeys v1.3.1/go.mod h1:xxCBG/f/4Vbmh2XQJBsOmNdxWUY5j/s27jujKPbQf14=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/internal v1.0.0/go.mod h1:bTSOgj05NGRuHHhQwAdPnYr9TOdNmKlZTgGLL6nyAdI=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/internal v1.1.1 h1:bFWuoEKg+gImo7pvkiQEFAc8ocibADgXeiLAxWhWmkI=
|
||||
github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/internal v1.1.1/go.mod h1:Vih/3yc6yac2JzU4hzpaDupBJP0Flaia9rXXrU8xyww=
|
||||
github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1 h1:UQHMgLO+TxOElx5B5HZ4hJQsoJ/PvUvKRhJHDQXO8P8=
|
||||
github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E=
|
||||
github.com/AzureAD/microsoft-authentication-library-for-go v1.1.1/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI=
|
||||
github.com/AzureAD/microsoft-authentication-library-for-go v1.2.2/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI=
|
||||
github.com/AzureAD/microsoft-authentication-library-for-go v1.4.2 h1:oygO0locgZJe7PpYPXT5A29ZkwJaPqcva7BVeemZOZs=
|
||||
github.com/AzureAD/microsoft-authentication-library-for-go v1.4.2/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI=
|
||||
github.com/DATA-DOG/go-sqlmock v1.5.2 h1:OcvFkGmslmlZibjAjaHm3L//6LiuBgolP7OputlJIzU=
|
||||
github.com/DATA-DOG/go-sqlmock v1.5.2/go.mod h1:88MAG/4G7SMwSE3CeA0ZKzrT5CiOU3OJ+JlNzwDqpNU=
|
||||
github.com/Microsoft/go-winio v0.6.2 h1:F2VQgta7ecxGYO8k3ZZz3RS8fVIXVxONVUPlNERoyfY=
|
||||
github.com/Microsoft/go-winio v0.6.2/go.mod h1:yd8OoFMLzJbo9gZq8j5qaps8bJ9aShtEA8Ipt1oGCvU=
|
||||
github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM=
|
||||
github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw=
|
||||
github.com/bitechdev/ResolveSpec v1.0.86 h1:a4yFMMDizrmvDOV61cj/+kD+mEtKL/5EIHY2GcP3uJU=
|
||||
github.com/bitechdev/ResolveSpec v1.0.86/go.mod h1:YZOY2YCD0Kmb+pjAMhOqPh4q82Hij57F/CLlCMkzT78=
|
||||
github.com/bitechdev/ResolveSpec v1.0.87 h1:zLiHynLK8LLpXIfCZOjL5Iy1COBS6YZcWE1BHKfYqbA=
|
||||
github.com/bitechdev/ResolveSpec v1.0.87/go.mod h1:YZOY2YCD0Kmb+pjAMhOqPh4q82Hij57F/CLlCMkzT78=
|
||||
github.com/bradfitz/gomemcache v0.0.0-20250403215159-8d39553ac7cf h1:TqhNAT4zKbTdLa62d2HDBFdvgSbIGB3eJE8HqhgiL9I=
|
||||
github.com/bradfitz/gomemcache v0.0.0-20250403215159-8d39553ac7cf/go.mod h1:r5xuitiExdLAJ09PR7vBVENGvp4ZuTBeWTGtxuX3K+c=
|
||||
github.com/bsm/ginkgo/v2 v2.12.0 h1:Ny8MWAHyOepLGlLKYmXG4IEkioBysk6GpaRTLC8zwWs=
|
||||
github.com/bsm/ginkgo/v2 v2.12.0/go.mod h1:SwYbGRRDovPVboqFv0tPTcG1sN61LM1Z4ARdbAV9g4c=
|
||||
github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA=
|
||||
github.com/bsm/gomega v1.27.10/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0=
|
||||
github.com/cenkalti/backoff/v4 v4.3.0 h1:MyRJ/UdXutAwSAT+s3wNd7MfTIcy71VQueUuFK343L8=
|
||||
github.com/cenkalti/backoff/v4 v4.3.0/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE=
|
||||
github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs=
|
||||
github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
|
||||
github.com/containerd/errdefs v1.0.0 h1:tg5yIfIlQIrxYtu9ajqY42W3lpS19XqdxRQeEwYG8PI=
|
||||
github.com/containerd/errdefs v1.0.0/go.mod h1:+YBYIdtsnF4Iw6nWZhJcqGSg/dwvV7tyJ/kCkyJ2k+M=
|
||||
github.com/containerd/errdefs/pkg v0.3.0 h1:9IKJ06FvyNlexW690DXuQNx2KA2cUJXx151Xdx3ZPPE=
|
||||
github.com/containerd/errdefs/pkg v0.3.0/go.mod h1:NJw6s9HwNuRhnjJhM7pylWwMyAkmCQvQ4GpJHEqRLVk=
|
||||
github.com/containerd/log v0.1.0 h1:TCJt7ioM2cr/tfR8GPbGf9/VRAX8D2B4PjzCpfX540I=
|
||||
github.com/containerd/log v0.1.0/go.mod h1:VRRf09a7mHDIRezVKTRCrOq78v577GXq3bSa3EhrzVo=
|
||||
github.com/containerd/platforms v0.2.1 h1:zvwtM3rz2YHPQsF2CHYM8+KtB5dvhISiXh5ZpSBQv6A=
|
||||
github.com/containerd/platforms v0.2.1/go.mod h1:XHCb+2/hzowdiut9rkudds9bE5yJ7npe7dG/wG+uFPw=
|
||||
github.com/cpuguy83/dockercfg v0.3.2 h1:DlJTyZGBDlXqUZ2Dk2Q3xHs/FtnooJJVaad2S9GKorA=
|
||||
github.com/cpuguy83/dockercfg v0.3.2/go.mod h1:sugsbF4//dDlL/i+S+rtpIWp+5h0BHJHfjj5/jFyUJc=
|
||||
github.com/cpuguy83/go-md2man/v2 v2.0.6/go.mod h1:oOW0eioCTA6cOiMLiUPZOpcVxMig6NIQQ7OS05n1F4g=
|
||||
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
|
||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78=
|
||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc=
|
||||
github.com/distribution/reference v0.6.0 h1:0IXCQ5g4/QMHHkarYzh5l+u8T3t73zM5QvfrDyIgxBk=
|
||||
github.com/distribution/reference v0.6.0/go.mod h1:BbU0aIcezP1/5jX/8MP0YiH4SdvB5Y4f/wlDRiLyi3E=
|
||||
github.com/dnaeon/go-vcr v1.1.0/go.mod h1:M7tiix8f0r6mKKJ3Yq/kqU1OYf3MnfmBWVbPx/yU9ko=
|
||||
github.com/dnaeon/go-vcr v1.2.0/go.mod h1:R4UdLID7HZT3taECzJs4YgbbH6PIGXB6W/sc5OLb6RQ=
|
||||
github.com/docker/docker v28.5.1+incompatible h1:Bm8DchhSD2J6PsFzxC35TZo4TLGR2PdW/E69rU45NhM=
|
||||
github.com/docker/docker v28.5.1+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk=
|
||||
github.com/docker/go-connections v0.6.0 h1:LlMG9azAe1TqfR7sO+NJttz1gy6KO7VJBh+pMmjSD94=
|
||||
github.com/docker/go-connections v0.6.0/go.mod h1:AahvXYshr6JgfUJGdDCs2b5EZG/vmaMAntpSFH5BFKE=
|
||||
github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4=
|
||||
github.com/docker/go-units v0.5.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
||||
github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY=
|
||||
github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto=
|
||||
github.com/ebitengine/purego v0.8.4 h1:CF7LEKg5FFOsASUj0+QwaXf8Ht6TlFxg09+S9wz0omw=
|
||||
github.com/ebitengine/purego v0.8.4/go.mod h1:iIjxzd6CiRiOG0UyXP+V1+jWqUXVjPKLAI0mRfJZTmQ=
|
||||
github.com/felixge/httpsnoop v1.0.4 h1:NFTV2Zj1bL4mc9sqWACXbQFVBBg2W3GPvqp8/ESS2Wg=
|
||||
github.com/felixge/httpsnoop v1.0.4/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U=
|
||||
github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8=
|
||||
github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
|
||||
github.com/fsnotify/fsnotify v1.9.0 h1:2Ml+OJNzbYCTzsxtv8vKSFD9PbJjmhYF14k/jKC7S9k=
|
||||
github.com/fsnotify/fsnotify v1.9.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0=
|
||||
github.com/getsentry/sentry-go v0.40.0 h1:VTJMN9zbTvqDqPwheRVLcp0qcUcM+8eFivvGocAaSbo=
|
||||
github.com/getsentry/sentry-go v0.40.0/go.mod h1:eRXCoh3uvmjQLY6qu63BjUZnaBu5L5WhMV1RwYO8W5s=
|
||||
github.com/go-errors/errors v1.4.2 h1:J6MZopCL4uSllY1OfXM374weqZFFItUbrImctkmUxIA=
|
||||
github.com/go-errors/errors v1.4.2/go.mod h1:sIVyrIiJhuEF+Pj9Ebtd6P/rEYROXFi3BopGUQ5a5Og=
|
||||
github.com/go-logr/logr v1.4.3 h1:CjnDlHq8ikf6E492q6eKboGOC0T8CDaOvkHCIg8idEI=
|
||||
github.com/go-logr/logr v1.4.3/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
|
||||
github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag=
|
||||
github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE=
|
||||
github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY=
|
||||
github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0=
|
||||
github.com/go-pg/pg/v10 v10.11.0 h1:CMKJqLgTrfpE/aOVeLdybezR2om071Vh38OLZjsyMI0=
|
||||
github.com/go-pg/pg/v10 v10.11.0/go.mod h1:4BpHRoxE61y4Onpof3x1a2SQvi9c+q1dJnrNdMjsroA=
|
||||
github.com/go-pg/zerochecker v0.2.0 h1:pp7f72c3DobMWOb2ErtZsnrPaSvHd2W4o9//8HtF4mU=
|
||||
github.com/go-pg/zerochecker v0.2.0/go.mod h1:NJZ4wKL0NmTtz0GKCoJ8kym6Xn/EQzXRl2OnAe7MmDo=
|
||||
github.com/go-viper/mapstructure/v2 v2.4.0 h1:EBsztssimR/CONLSZZ04E8qAkxNYq4Qp9LvH92wZUgs=
|
||||
github.com/go-viper/mapstructure/v2 v2.4.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM=
|
||||
github.com/golang-jwt/jwt/v5 v5.0.0/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk=
|
||||
github.com/golang-jwt/jwt/v5 v5.2.1/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk=
|
||||
github.com/golang-jwt/jwt/v5 v5.2.2/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk=
|
||||
github.com/golang-jwt/jwt/v5 v5.3.0 h1:pv4AsKCKKZuqlgs5sUmn4x8UlGa0kEVt/puTpKx9vvo=
|
||||
github.com/golang-jwt/jwt/v5 v5.3.0/go.mod h1:fxCRLWMO43lRc8nhHWY6LGqRcf+1gQWArsqaEUEa5bE=
|
||||
github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9 h1:au07oEsX2xN0ktxqI+Sida1w446QrXBRJ0nee3SNZlA=
|
||||
github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9/go.mod h1:8vg3r2VgvsThLBIFL93Qb5yWzgyZWhEmBwUJWevAkK0=
|
||||
github.com/golang-sql/sqlexp v0.1.0 h1:ZCD6MBpcuOVfGVqsEmY5/4FtYiKz6tSyUv9LPEDei6A=
|
||||
github.com/golang-sql/sqlexp v0.1.0/go.mod h1:J4ad9Vo8ZCWQ2GMrC4UCQy1JpCbwU9m3EOqtpKwwwHI=
|
||||
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
||||
github.com/google/go-cmp v0.7.0 h1:wk8382ETsv4JYUZwIsn6YpYiWiBsYLSJiTsyBybVuN8=
|
||||
github.com/google/go-cmp v0.7.0/go.mod h1:pXiqmnSA92OHEEa9HXL2W4E7lf9JzCmGVUdgjX3N/iU=
|
||||
github.com/google/jsonschema-go v0.4.2 h1:tmrUohrwoLZZS/P3x7ex0WAVknEkBZM46iALbcqoRA8=
|
||||
github.com/google/jsonschema-go v0.4.2/go.mod h1:r5quNTdLOYEz95Ru18zA0ydNbBuYoo9tgaYcxEYhJVE=
|
||||
github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
|
||||
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||
github.com/gorilla/mux v1.8.1 h1:TuBL49tXwgrFYWhqrNgrUNEY92u81SPhu7sTdzQEiWY=
|
||||
github.com/gorilla/mux v1.8.1/go.mod h1:AKf9I4AEqPTmMytcMc0KkNouC66V3BtZ4qD5fmWSiMQ=
|
||||
github.com/gorilla/securecookie v1.1.1/go.mod h1:ra0sb63/xPlUeL+yeDciTfxMRAA+MP+HVt/4epWDjd4=
|
||||
github.com/gorilla/sessions v1.2.1/go.mod h1:dk2InVEVJ0sfLlnXv9EAgkf6ecYs/i80K/zI+bUmuGM=
|
||||
github.com/hashicorp/go-uuid v1.0.2/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
|
||||
github.com/hashicorp/go-uuid v1.0.3/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
|
||||
github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8=
|
||||
github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
|
||||
github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM=
|
||||
@@ -134,181 +27,52 @@ github.com/jackc/pgx/v5 v5.9.1 h1:uwrxJXBnx76nyISkhr33kQLlUqjv7et7b9FjCen/tdc=
|
||||
github.com/jackc/pgx/v5 v5.9.1/go.mod h1:mal1tBGAFfLHvZzaYh77YS/eC6IX9OWbRV1QIIM0Jn4=
|
||||
github.com/jackc/puddle/v2 v2.2.2 h1:PR8nw+E/1w0GLuRFSmiioY6UooMp6KJv0/61nB7icHo=
|
||||
github.com/jackc/puddle/v2 v2.2.2/go.mod h1:vriiEXHvEE654aYKXXjOvZM39qJ0q+azkZFrfEOc3H4=
|
||||
github.com/jcmturner/aescts/v2 v2.0.0/go.mod h1:AiaICIRyfYg35RUkr8yESTqvSy7csK90qZ5xfvvsoNs=
|
||||
github.com/jcmturner/dnsutils/v2 v2.0.0/go.mod h1:b0TnjGOvI/n42bZa+hmXL+kFJZsFT7G4t3HTlQ184QM=
|
||||
github.com/jcmturner/gofork v1.7.6/go.mod h1:1622LH6i/EZqLloHfE7IeZ0uEJwMSUyQ/nDd82IeqRo=
|
||||
github.com/jcmturner/goidentity/v6 v6.0.1/go.mod h1:X1YW3bgtvwAXju7V3LCIMpY0Gbxyjn/mY9zx4tFonSg=
|
||||
github.com/jcmturner/gokrb5/v8 v8.4.4/go.mod h1:1btQEpgT6k+unzCwX1KdWMEwPPkkgBtP+F6aCACiMrs=
|
||||
github.com/jcmturner/rpc/v2 v2.0.3/go.mod h1:VUJYCIDm3PVOEHw8sgt091/20OJjskO/YJki3ELg/Hc=
|
||||
github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E=
|
||||
github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc=
|
||||
github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ=
|
||||
github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8=
|
||||
github.com/jmoiron/sqlx v1.3.5 h1:vFFPA71p1o5gAeqtEAwLU4dnX2napprKtHr7PYIcN3g=
|
||||
github.com/jmoiron/sqlx v1.3.5/go.mod h1:nRVWtLre0KfCLJvgxzCsLVMogSvQ1zNJtpYr2Ccp0mQ=
|
||||
github.com/klauspost/compress v1.18.2 h1:iiPHWW0YrcFgpBYhsA6D1+fqHssJscY/Tm/y2Uqnapk=
|
||||
github.com/klauspost/compress v1.18.2/go.mod h1:R0h/fSBs8DE4ENlcrlib3PsXS61voFxhIs2DeRhCvJ4=
|
||||
github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
|
||||
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
|
||||
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
|
||||
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
|
||||
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
|
||||
github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0=
|
||||
github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk=
|
||||
github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
|
||||
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
|
||||
github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc=
|
||||
github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw=
|
||||
github.com/lib/pq v1.10.9 h1:YXG7RB+JIjhP29X+OtkiDnYaXQwpS4JEWq7dtCCRUEw=
|
||||
github.com/lib/pq v1.10.9/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o=
|
||||
github.com/lufia/plan9stats v0.0.0-20211012122336-39d0f177ccd0 h1:6E+4a0GO5zZEnZ81pIr0yLvtUWk2if982qA3F3QD6H4=
|
||||
github.com/lufia/plan9stats v0.0.0-20211012122336-39d0f177ccd0/go.mod h1:zJYVVT2jmtg6P3p1VtQj7WsuWi/y4VnjVBn7F8KPB3I=
|
||||
github.com/magiconair/properties v1.8.10 h1:s31yESBquKXCV9a/ScB3ESkOjUYYv+X0rg8SYxI99mE=
|
||||
github.com/magiconair/properties v1.8.10/go.mod h1:Dhd985XPs7jluiymwWYZ0G4Z61jb3vdS329zhj2hYo0=
|
||||
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
|
||||
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
|
||||
github.com/mattn/go-sqlite3 v1.14.33 h1:A5blZ5ulQo2AtayQ9/limgHEkFreKj1Dv226a1K73s0=
|
||||
github.com/mattn/go-sqlite3 v1.14.33/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
|
||||
github.com/microsoft/go-mssqldb v1.8.2/go.mod h1:vp38dT33FGfVotRiTmDo3bFyaHq+p3LektQrjTULowo=
|
||||
github.com/microsoft/go-mssqldb v1.9.5 h1:orwya0X/5bsL1o+KasupTkk2eNTNFkTQG0BEe/HxCn0=
|
||||
github.com/microsoft/go-mssqldb v1.9.5/go.mod h1:VCP2a0KEZZtGLRHd1PsLavLFYy/3xX2yJUPycv3Sr2Q=
|
||||
github.com/moby/docker-image-spec v1.3.1 h1:jMKff3w6PgbfSa69GfNg+zN/XLhfXJGnEx3Nl2EsFP0=
|
||||
github.com/moby/docker-image-spec v1.3.1/go.mod h1:eKmb5VW8vQEh/BAr2yvVNvuiJuY6UIocYsFu/DxxRpo=
|
||||
github.com/moby/go-archive v0.1.0 h1:Kk/5rdW/g+H8NHdJW2gsXyZ7UnzvJNOy6VKJqueWdcQ=
|
||||
github.com/moby/go-archive v0.1.0/go.mod h1:G9B+YoujNohJmrIYFBpSd54GTUB4lt9S+xVQvsJyFuo=
|
||||
github.com/moby/patternmatcher v0.6.0 h1:GmP9lR19aU5GqSSFko+5pRqHi+Ohk1O69aFiKkVGiPk=
|
||||
github.com/moby/patternmatcher v0.6.0/go.mod h1:hDPoyOpDY7OrrMDLaYoY3hf52gNCR/YOUYxkhApJIxc=
|
||||
github.com/moby/sys/sequential v0.6.0 h1:qrx7XFUd/5DxtqcoH1h438hF5TmOvzC/lspjy7zgvCU=
|
||||
github.com/moby/sys/sequential v0.6.0/go.mod h1:uyv8EUTrca5PnDsdMGXhZe6CCe8U/UiTWd+lL+7b/Ko=
|
||||
github.com/moby/sys/user v0.4.0 h1:jhcMKit7SA80hivmFJcbB1vqmw//wU61Zdui2eQXuMs=
|
||||
github.com/moby/sys/user v0.4.0/go.mod h1:bG+tYYYJgaMtRKgEmuueC0hJEAZWwtIbZTB+85uoHjs=
|
||||
github.com/moby/sys/userns v0.1.0 h1:tVLXkFOxVu9A64/yh59slHVv9ahO9UIev4JZusOLG/g=
|
||||
github.com/moby/sys/userns v0.1.0/go.mod h1:IHUYgu/kao6N8YZlp9Cf444ySSvCmDlmzUcYfDHOl28=
|
||||
github.com/moby/term v0.5.0 h1:xt8Q1nalod/v7BqbG21f8mQPqH+xAaC9C3N3wfWbVP0=
|
||||
github.com/moby/term v0.5.0/go.mod h1:8FzsFHVUBGZdbDsJw/ot+X+d5HLUbvklYLJ9uGfcI3Y=
|
||||
github.com/modelcontextprotocol/go-sdk v1.4.1 h1:M4x9GyIPj+HoIlHNGpK2hq5o3BFhC+78PkEaldQRphc=
|
||||
github.com/modelcontextprotocol/go-sdk v1.4.1/go.mod h1:Bo/mS87hPQqHSRkMv4dQq1XCu6zv4INdXnFZabkNU6s=
|
||||
github.com/modocache/gover v0.0.0-20171022184752-b58185e213c5/go.mod h1:caMODM3PzxT8aQXRPkAt8xlV/e7d7w8GM5g0fa5F0D8=
|
||||
github.com/montanaflynn/stats v0.7.0/go.mod h1:etXPPgVO6n31NxCd9KQUMvCM+ve0ruNzt6R8Bnaayow=
|
||||
github.com/morikuni/aec v1.0.0 h1:nP9CBfwrvYnBRgY6qfDQkygYDmYwOilePFkwzv4dU8A=
|
||||
github.com/morikuni/aec v1.0.0/go.mod h1:BbKIizmSmc5MMPqRYbxO4ZU0S0+P200+tUnFx7PXmsc=
|
||||
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 h1:C3w9PqII01/Oq1c1nUAm88MOHcQC9l5mIlSMApZMrHA=
|
||||
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ=
|
||||
github.com/ncruces/go-strftime v1.0.0 h1:HMFp8mLCTPp341M/ZnA4qaf7ZlsbTc+miZjCLOFAw7w=
|
||||
github.com/ncruces/go-strftime v1.0.0/go.mod h1:Fwc5htZGVVkseilnfgOVb9mKy6w1naJmn9CehxcKcls=
|
||||
github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8Oi/yOhh5U=
|
||||
github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM=
|
||||
github.com/opencontainers/image-spec v1.1.1 h1:y0fUlFfIZhPF1W537XOLg0/fcx6zcHCJwooC2xJA040=
|
||||
github.com/opencontainers/image-spec v1.1.1/go.mod h1:qpqAh3Dmcf36wStyyWU+kCeDgrGnAve2nCC8+7h8Q0M=
|
||||
github.com/pelletier/go-toml/v2 v2.2.4 h1:mye9XuhQ6gvn5h28+VilKrrPoQVanw5PMw/TB0t5Ec4=
|
||||
github.com/pelletier/go-toml/v2 v2.2.4/go.mod h1:2gIqNv+qfxSVS7cM2xJQKtLSTLUE9V8t9Stt+h56mCY=
|
||||
github.com/pgvector/pgvector-go v0.3.0 h1:Ij+Yt78R//uYqs3Zk35evZFvr+G0blW0OUN+Q2D1RWc=
|
||||
github.com/pgvector/pgvector-go v0.3.0/go.mod h1:duFy+PXWfW7QQd5ibqutBO4GxLsUZ9RVXhFZGIBsWSA=
|
||||
github.com/pingcap/errors v0.11.4 h1:lFuQV/oaUMGcD2tqt+01ROSmJs75VG1ToEOkZIZ4nE4=
|
||||
github.com/pingcap/errors v0.11.4/go.mod h1:Oi8TUi2kEtXXLMJk9l1cGmz20kV3TaQ0usTwv5KuLY8=
|
||||
github.com/pkg/browser v0.0.0-20210911075715-681adbf594b8/go.mod h1:HKlIX3XHQyzLZPlr7++PzdhaXEj94dEiJgZDTsxEqUI=
|
||||
github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c h1:+mdjkGKdHQG3305AYmdv1U2eRNDiU2ErMBj1gwrq8eQ=
|
||||
github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c/go.mod h1:7rwL4CYBLnjLxUqIJNnCWiEdr3bn6IUYi15bNlnbCCU=
|
||||
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA=
|
||||
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||
github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c h1:ncq/mPwQF4JjgDlrVEn3C11VoGHZN7m8qihwgMEtzYw=
|
||||
github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c/go.mod h1:OmDBASR4679mdNQnz2pUhc2G8CO2JrUAVFDRBDP/hJE=
|
||||
github.com/prometheus/client_golang v1.23.2 h1:Je96obch5RDVy3FDMndoUsjAhG5Edi49h0RJWRi/o0o=
|
||||
github.com/prometheus/client_golang v1.23.2/go.mod h1:Tb1a6LWHB3/SPIzCoaDXI4I8UHKeFTEQ1YCr+0Gyqmg=
|
||||
github.com/prometheus/client_model v0.6.2 h1:oBsgwpGs7iVziMvrGhE53c/GrLUsZdHnqNwqPLxwZyk=
|
||||
github.com/prometheus/client_model v0.6.2/go.mod h1:y3m2F6Gdpfy6Ut/GBsUqTWZqCUvMVzSfMLjcu6wAwpE=
|
||||
github.com/prometheus/common v0.67.4 h1:yR3NqWO1/UyO1w2PhUvXlGQs/PtFmoveVO0KZ4+Lvsc=
|
||||
github.com/prometheus/common v0.67.4/go.mod h1:gP0fq6YjjNCLssJCQp0yk4M8W6ikLURwkdd/YKtTbyI=
|
||||
github.com/prometheus/procfs v0.19.2 h1:zUMhqEW66Ex7OXIiDkll3tl9a1ZdilUOd/F6ZXw4Vws=
|
||||
github.com/prometheus/procfs v0.19.2/go.mod h1:M0aotyiemPhBCM0z5w87kL22CxfcH05ZpYlu+b4J7mw=
|
||||
github.com/puzpuzpuz/xsync/v3 v3.5.1 h1:GJYJZwO6IdxN/IKbneznS6yPkVC+c3zyY/j19c++5Fg=
|
||||
github.com/puzpuzpuz/xsync/v3 v3.5.1/go.mod h1:VjzYrABPabuM4KyBh1Ftq6u8nhwY5tBPKP9jpmh0nnA=
|
||||
github.com/redis/go-redis/v9 v9.17.2 h1:P2EGsA4qVIM3Pp+aPocCJ7DguDHhqrXNhVcEp4ViluI=
|
||||
github.com/redis/go-redis/v9 v9.17.2/go.mod h1:u410H11HMLoB+TP67dz8rL9s6QW2j76l0//kSOd3370=
|
||||
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE=
|
||||
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo=
|
||||
github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs=
|
||||
github.com/rogpeppe/go-internal v1.12.0/go.mod h1:E+RYuTGaKKdloAfM02xzb0FW3Paa99yedzYV+kq4uf4=
|
||||
github.com/rogpeppe/go-internal v1.14.1 h1:UQB4HGPB6osV0SQTLymcB4TgvyWu6ZyliaW0tI/otEQ=
|
||||
github.com/rogpeppe/go-internal v1.14.1/go.mod h1:MaRKkUm5W0goXpeCfT7UZI6fk/L7L7so1lCWt35ZSgc=
|
||||
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
|
||||
github.com/sagikazarmark/locafero v0.12.0 h1:/NQhBAkUb4+fH1jivKHWusDYFjMOOKU88eegjfxfHb4=
|
||||
github.com/sagikazarmark/locafero v0.12.0/go.mod h1:sZh36u/YSZ918v0Io+U9ogLYQJ9tLLBmM4eneO6WwsI=
|
||||
github.com/segmentio/asm v1.1.3 h1:WM03sfUOENvvKexOLp+pCqgb/WDjsi7EK8gIsICtzhc=
|
||||
github.com/segmentio/asm v1.1.3/go.mod h1:Ld3L4ZXGNcSLRg4JBsZ3//1+f/TjYl0Mzen/DQy1EJg=
|
||||
github.com/segmentio/encoding v0.5.4 h1:OW1VRern8Nw6ITAtwSZ7Idrl3MXCFwXHPgqESYfvNt0=
|
||||
github.com/segmentio/encoding v0.5.4/go.mod h1:HS1ZKa3kSN32ZHVZ7ZLPLXWvOVIiZtyJnO1gPH1sKt0=
|
||||
github.com/shirou/gopsutil/v4 v4.25.6 h1:kLysI2JsKorfaFPcYmcJqbzROzsBWEOAtw6A7dIfqXs=
|
||||
github.com/shirou/gopsutil/v4 v4.25.6/go.mod h1:PfybzyydfZcN+JMMjkF6Zb8Mq1A/VcogFFg7hj50W9c=
|
||||
github.com/shopspring/decimal v1.4.0 h1:bxl37RwXBklmTi0C79JfXCEBD1cqqHt0bbgBAGFp81k=
|
||||
github.com/shopspring/decimal v1.4.0/go.mod h1:gawqmDU56v4yIKSwfBSFip1HdCCXN8/+DMd9qYNcwME=
|
||||
github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ=
|
||||
github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
|
||||
github.com/spf13/afero v1.15.0 h1:b/YBCLWAJdFWJTN9cLhiXXcD7mzKn9Dm86dNnfyQw1I=
|
||||
github.com/spf13/afero v1.15.0/go.mod h1:NC2ByUVxtQs4b3sIUphxK0NioZnmxgyCrfzeuq8lxMg=
|
||||
github.com/spf13/cast v1.10.0 h1:h2x0u2shc1QuLHfxi+cTJvs30+ZAHOGRic8uyGTDWxY=
|
||||
github.com/spf13/cast v1.10.0/go.mod h1:jNfB8QC9IA6ZuY2ZjDp0KtFO2LZZlg4S/7bzP6qqeHo=
|
||||
github.com/spf13/cobra v1.10.2 h1:DMTTonx5m65Ic0GOoRY2c16WCbHxOOw6xxezuLaBpcU=
|
||||
github.com/spf13/cobra v1.10.2/go.mod h1:7C1pvHqHw5A4vrJfjNwvOdzYu0Gml16OCs2GRiTUUS4=
|
||||
github.com/spf13/pflag v1.0.9 h1:9exaQaMOCwffKiiiYk6/BndUBv+iRViNW+4lEMi0PvY=
|
||||
github.com/spf13/pflag v1.0.9/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
||||
github.com/spf13/pflag v1.0.10 h1:4EBh2KAYBwaONj6b2Ye1GiHfwjqyROoF4RwYO+vPwFk=
|
||||
github.com/spf13/pflag v1.0.10/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
||||
github.com/spf13/viper v1.21.0 h1:x5S+0EU27Lbphp4UKm1C+1oQO+rKx36vfCoaVebLFSU=
|
||||
github.com/spf13/viper v1.21.0/go.mod h1:P0lhsswPGWD/1lZJ9ny3fYnVqxiegrlNrEmgLjbTCAY=
|
||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
|
||||
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
|
||||
github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA=
|
||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
||||
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
||||
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||
github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
|
||||
github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
||||
github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U=
|
||||
github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U=
|
||||
github.com/subosito/gotenv v1.6.0 h1:9NlTDc1FTs4qu0DDq7AEtTPNw6SVm7uBMsUCUjABIf8=
|
||||
github.com/subosito/gotenv v1.6.0/go.mod h1:Dk4QP5c2W3ibzajGcXpNraDfq2IrhjMIvMSWPKKo0FU=
|
||||
github.com/testcontainers/testcontainers-go v0.40.0 h1:pSdJYLOVgLE8YdUY2FHQ1Fxu+aMnb6JfVz1mxk7OeMU=
|
||||
github.com/testcontainers/testcontainers-go v0.40.0/go.mod h1:FSXV5KQtX2HAMlm7U3APNyLkkap35zNLxukw9oBi/MY=
|
||||
github.com/tidwall/gjson v1.14.2/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk=
|
||||
github.com/tidwall/gjson v1.18.0 h1:FIDeeyB800efLX89e5a8Y0BNH+LOngJyGrIWxG2FKQY=
|
||||
github.com/tidwall/gjson v1.18.0/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk=
|
||||
github.com/tidwall/match v1.1.1/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM=
|
||||
github.com/tidwall/match v1.2.0 h1:0pt8FlkOwjN2fPt4bIl4BoNxb98gGHN2ObFEDkrfZnM=
|
||||
github.com/tidwall/match v1.2.0/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM=
|
||||
github.com/tidwall/pretty v1.2.0/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU=
|
||||
github.com/tidwall/pretty v1.2.1 h1:qjsOFOWWQl+N3RsoF5/ssm1pHmJJwhjlSbZ51I6wMl4=
|
||||
github.com/tidwall/pretty v1.2.1/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU=
|
||||
github.com/tidwall/sjson v1.2.5 h1:kLy8mja+1c9jlljvWTlSazM7cKDRfJuR/bOJhcY5NcY=
|
||||
github.com/tidwall/sjson v1.2.5/go.mod h1:Fvgq9kS/6ociJEDnK0Fk1cpYF4FIW6ZF7LAe+6jwd28=
|
||||
github.com/tklauser/go-sysconf v0.3.12 h1:0QaGUFOdQaIVdPgfITYzaTegZvdCjmYO52cSFAEVmqU=
|
||||
github.com/tklauser/go-sysconf v0.3.12/go.mod h1:Ho14jnntGE1fpdOqQEEaiKRpvIavV0hSfmBq8nJbHYI=
|
||||
github.com/tklauser/numcpus v0.6.1 h1:ng9scYS7az0Bk4OZLvrNXNSAO2Pxr1XXRAPyjhIx+Fk=
|
||||
github.com/tklauser/numcpus v0.6.1/go.mod h1:1XfjsgE2zo8GVw7POkMbHENHzVg3GzmoZ9fESEdAacY=
|
||||
github.com/tmthrgd/go-hex v0.0.0-20190904060850-447a3041c3bc h1:9lRDQMhESg+zvGYmW5DyG0UqvY96Bu5QYsTLvCHdrgo=
|
||||
github.com/tmthrgd/go-hex v0.0.0-20190904060850-447a3041c3bc/go.mod h1:bciPuU6GHm1iF1pBvUfxfsH0Wmnc2VbpgvbI9ZWuIRs=
|
||||
github.com/uptrace/bun v1.2.16 h1:QlObi6ZIK5Ao7kAALnh91HWYNZUBbVwye52fmlQM9kc=
|
||||
github.com/uptrace/bun v1.2.16/go.mod h1:jMoNg2n56ckaawi/O/J92BHaECmrz6IRjuMWqlMaMTM=
|
||||
github.com/uptrace/bun/dialect/mssqldialect v1.2.16 h1:rKv0cKPNBviXadB/+2Y/UedA/c1JnwGzUWZkdN5FdSQ=
|
||||
github.com/uptrace/bun/dialect/mssqldialect v1.2.16/go.mod h1:J5U7tGKWDsx2Q7MwDZF2417jCdpD6yD/ZMFJcCR80bk=
|
||||
github.com/uptrace/bun/dialect/pgdialect v1.2.16 h1:KFNZ0LxAyczKNfK/IJWMyaleO6eI9/Z5tUv3DE1NVL4=
|
||||
github.com/uptrace/bun/dialect/pgdialect v1.2.16/go.mod h1:IJdMeV4sLfh0LDUZl7TIxLI0LipF1vwTK3hBC7p5qLo=
|
||||
github.com/uptrace/bun/dialect/sqlitedialect v1.2.16 h1:6wVAiYLj1pMibRthGwy4wDLa3D5AQo32Y8rvwPd8CQ0=
|
||||
github.com/uptrace/bun/dialect/sqlitedialect v1.2.16/go.mod h1:Z7+5qK8CGZkDQiPMu+LSdVuDuR1I5jcwtkB1Pi3F82E=
|
||||
github.com/uptrace/bun v1.1.12 h1:sOjDVHxNTuM6dNGaba0wUuz7KvDE1BmNu9Gqs2gJSXQ=
|
||||
github.com/uptrace/bun v1.1.12/go.mod h1:NPG6JGULBeQ9IU6yHp7YGELRa5Agmd7ATZdz4tGZ6z0=
|
||||
github.com/uptrace/bun/dialect/pgdialect v1.1.12 h1:m/CM1UfOkoBTglGO5CUTKnIKKOApOYxkcP2qn0F9tJk=
|
||||
github.com/uptrace/bun/dialect/pgdialect v1.1.12/go.mod h1:Ij6WIxQILxLlL2frUBxUBOZJtLElD2QQNDcu/PWDHTc=
|
||||
github.com/uptrace/bun/driver/pgdriver v1.1.12 h1:3rRWB1GK0psTJrHwxzNfEij2MLibggiLdTqjTtfHc1w=
|
||||
github.com/uptrace/bun/driver/pgdriver v1.1.12/go.mod h1:ssYUP+qwSEgeDDS1xm2XBip9el1y9Mi5mTAvLoiADLM=
|
||||
github.com/uptrace/bun/driver/sqliteshim v1.2.16 h1:M6Dh5kkDWFbUWBrOsIE1g1zdZ5JbSytTD4piFRBOUAI=
|
||||
github.com/uptrace/bun/driver/sqliteshim v1.2.16/go.mod h1:iKdJ06P3XS+pwKcONjSIK07bbhksH3lWsw3mpfr0+bY=
|
||||
github.com/uptrace/bunrouter v1.0.23 h1:Bi7NKw3uCQkcA/GUCtDNPq5LE5UdR9pe+UyWbjHB/wU=
|
||||
github.com/uptrace/bunrouter v1.0.23/go.mod h1:O3jAcl+5qgnF+ejhgkmbceEk0E/mqaK+ADOocdNpY8M=
|
||||
github.com/vmihailenco/bufpool v0.1.11 h1:gOq2WmBrq0i2yW5QJ16ykccQ4wH9UyEsgLm6czKAd94=
|
||||
github.com/vmihailenco/bufpool v0.1.11/go.mod h1:AFf/MOy3l2CFTKbxwt0mp2MwnqjNEs5H/UxrkA5jxTQ=
|
||||
github.com/vmihailenco/msgpack/v5 v5.4.1 h1:cQriyiUvjTwOHg8QZaPihLWeRAAVoCpE00IUPn0Bjt8=
|
||||
github.com/vmihailenco/msgpack/v5 v5.4.1/go.mod h1:GaZTsDaehaPpQVyxrf5mtQlH+pc21PIudVV/E3rRQok=
|
||||
github.com/vmihailenco/msgpack/v5 v5.3.5 h1:5gO0H1iULLWGhs2H5tbAHIZTV8/cYafcFOr9znI5mJU=
|
||||
github.com/vmihailenco/msgpack/v5 v5.3.5/go.mod h1:7xyJ9e+0+9SaZT0Wt1RGleJXzli6Q/V5KbhBonMG9jc=
|
||||
github.com/vmihailenco/tagparser v0.1.2 h1:gnjoVuB/kljJ5wICEEOpx98oXMWPLj22G67Vbd1qPqc=
|
||||
github.com/vmihailenco/tagparser v0.1.2/go.mod h1:OeAg3pn3UbLjkWt+rN9oFYB6u/cQgqMEUPoW2WPyhdI=
|
||||
github.com/vmihailenco/tagparser/v2 v2.0.0 h1:y09buUbR+b5aycVFQs/g70pqKVZNBmxwAhO7/IwNM9g=
|
||||
@@ -317,171 +81,28 @@ github.com/x448/float16 v0.8.4 h1:qLwI1I70+NjRFUR3zs1JPUCgaCXSh3SW62uAKT1mSBM=
|
||||
github.com/x448/float16 v0.8.4/go.mod h1:14CWIYCyZA/cWjXOioeEpHeN/83MdbZDRQHoFcYsOfg=
|
||||
github.com/yosida95/uritemplate/v3 v3.0.2 h1:Ed3Oyj9yrmi9087+NczuL5BwkIc4wvTb5zIM+UJPGz4=
|
||||
github.com/yosida95/uritemplate/v3 v3.0.2/go.mod h1:ILOh0sOhIJR3+L/8afwt/kE++YT040gmv5BQTMR2HP4=
|
||||
github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
|
||||
github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo0=
|
||||
github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0=
|
||||
go.opentelemetry.io/auto/sdk v1.1.0 h1:cH53jehLUN6UFLY71z+NDOiNJqDdPRaXzTel0sJySYA=
|
||||
go.opentelemetry.io/auto/sdk v1.1.0/go.mod h1:3wSPjt5PWp2RhlCcmmOial7AvC4DQqZb7a7wCow3W8A=
|
||||
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0 h1:jq9TW8u3so/bN+JPT166wjOI6/vQPF6Xe7nMNIltagk=
|
||||
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0/go.mod h1:p8pYQP+m5XfbZm9fxtSKAbM6oIllS7s2AfxrChvc7iw=
|
||||
go.opentelemetry.io/otel v1.38.0 h1:RkfdswUDRimDg0m2Az18RKOsnI8UDzppJAtj01/Ymk8=
|
||||
go.opentelemetry.io/otel v1.38.0/go.mod h1:zcmtmQ1+YmQM9wrNsTGV/q/uyusom3P8RxwExxkZhjM=
|
||||
go.opentelemetry.io/otel/metric v1.38.0 h1:Kl6lzIYGAh5M159u9NgiRkmoMKjvbsKtYRwgfrA6WpA=
|
||||
go.opentelemetry.io/otel/metric v1.38.0/go.mod h1:kB5n/QoRM8YwmUahxvI3bO34eVtQf2i4utNVLr9gEmI=
|
||||
go.opentelemetry.io/otel/trace v1.38.0 h1:Fxk5bKrDZJUH+AMyyIXGcFAPah0oRcT+LuNtJrmcNLE=
|
||||
go.opentelemetry.io/otel/trace v1.38.0/go.mod h1:j1P9ivuFsTceSWe1oY+EeW3sc+Pp42sO++GHkg4wwhs=
|
||||
go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
|
||||
go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
|
||||
go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
|
||||
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
|
||||
go.uber.org/zap v1.27.1 h1:08RqriUEv8+ArZRYSTXy1LeBScaMpVSTBhCeaZYfMYc=
|
||||
go.uber.org/zap v1.27.1/go.mod h1:GB2qFLM7cTU87MWRP2mPIjqfIDnGu+VIO4V/SdhGo2E=
|
||||
go.yaml.in/yaml/v2 v2.4.3 h1:6gvOSjQoTB3vt1l+CU+tSyi/HOjfOjRLJ4YwYZGwRO0=
|
||||
go.yaml.in/yaml/v2 v2.4.3/go.mod h1:zSxWcmIDjOzPXpjlTTbAsKokqkDNAVtZO0WOMiT90s8=
|
||||
go.yaml.in/yaml/v3 v3.0.4 h1:tfq32ie2Jv2UxXFdLJdh3jXuOzWiL1fo0bu/FbuKpbc=
|
||||
go.yaml.in/yaml/v3 v3.0.4/go.mod h1:DhzuOOF2ATzADvBadXxruRBLzYTpT36CKvDb3+aBEFg=
|
||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
|
||||
golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58=
|
||||
golang.org/x/crypto v0.11.0/go.mod h1:xgJhtzW8F9jGdVFWZESrid1U1bjeNy4zgy5cRr/CIio=
|
||||
golang.org/x/crypto v0.12.0/go.mod h1:NF0Gs7EO5K4qLn+Ylc+fih8BSTeIjAP05siRnAh98yw=
|
||||
golang.org/x/crypto v0.13.0/go.mod h1:y6Z2r+Rw4iayiXXAIxJIDAJ1zMW4yaTpebo8fPOliYc=
|
||||
golang.org/x/crypto v0.18.0/go.mod h1:R0j02AL6hcrfOiy9T4ZYp/rcWeMxM3L6QYxlOuEG1mg=
|
||||
golang.org/x/crypto v0.19.0/go.mod h1:Iy9bg/ha4yyC70EfRS8jz+B6ybOBKMaSxLj6P6oBDfU=
|
||||
golang.org/x/crypto v0.21.0/go.mod h1:0BP7YvVV9gBbVKyeTG0Gyn+gZm94bibOW5BjDEYAOMs=
|
||||
golang.org/x/crypto v0.22.0/go.mod h1:vr6Su+7cTlO45qkww3VDJlzDn0ctJvRgYbC2NvXHt+M=
|
||||
golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8=
|
||||
golang.org/x/crypto v0.24.0/go.mod h1:Z1PMYSOR5nyMcyAVAIQSKCDwalqy85Aqn1x3Ws4L5DM=
|
||||
golang.org/x/crypto v0.46.0 h1:cKRW/pmt1pKAfetfu+RCEvjvZkA9RimPbh7bhFjGVBU=
|
||||
golang.org/x/crypto v0.46.0/go.mod h1:Evb/oLKmMraqjZ2iQTwDwvCtJkczlDuTmdJXoZVzqU0=
|
||||
golang.org/x/exp v0.0.0-20251219203646-944ab1f22d93 h1:fQsdNF2N+/YewlRZiricy4P1iimyPKZ/xwniHj8Q2a0=
|
||||
golang.org/x/exp v0.0.0-20251219203646-944ab1f22d93/go.mod h1:EPRbTFwzwjXj9NpYyyrvenVh9Y+GFeEvMNh7Xuz7xgU=
|
||||
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
|
||||
golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
|
||||
golang.org/x/mod v0.9.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
|
||||
golang.org/x/mod v0.12.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
|
||||
golang.org/x/mod v0.15.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
|
||||
golang.org/x/mod v0.17.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
|
||||
golang.org/x/mod v0.31.0 h1:HaW9xtz0+kOcWKwli0ZXy79Ix+UW/vOfmWI5QVd2tgI=
|
||||
golang.org/x/mod v0.31.0/go.mod h1:43JraMp9cGx1Rx3AqioxrbrhNsLl2l/iNAvuBkrezpg=
|
||||
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
||||
golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
||||
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
|
||||
golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
|
||||
golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
|
||||
golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
|
||||
golang.org/x/net v0.8.0/go.mod h1:QVkue5JL9kW//ek3r6jTKnTFis1tRmNAW2P1shuFdJc=
|
||||
golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg=
|
||||
golang.org/x/net v0.13.0/go.mod h1:zEVYFnQC7m/vmpQFELhcD1EWkZlX69l4oqgmer6hfKA=
|
||||
golang.org/x/net v0.14.0/go.mod h1:PpSgVXXLK0OxS0F31C1/tv6XNguvCrnXIDrFMspZIUI=
|
||||
golang.org/x/net v0.15.0/go.mod h1:idbUs1IY1+zTqbi8yxTbhexhEEk5ur9LInksu6HrEpk=
|
||||
golang.org/x/net v0.20.0/go.mod h1:z8BVo6PvndSri0LbOE3hAn0apkU+1YvI6E70E9jsnvY=
|
||||
golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44=
|
||||
golang.org/x/net v0.22.0/go.mod h1:JKghWKKOSdJwpW2GEx0Ja7fmaKnMsbu+MWVZTokSYmg=
|
||||
golang.org/x/net v0.24.0/go.mod h1:2Q7sJY5mzlzWjKtYUEXSlBWCdyaioyXzRB2RtU8KVE8=
|
||||
golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM=
|
||||
golang.org/x/net v0.26.0/go.mod h1:5YKkiSynbBIh3p6iOc/vibscux0x38BZDkn8sCUPxHE=
|
||||
golang.org/x/net v0.48.0 h1:zyQRTTrjc33Lhh0fBgT/H3oZq9WuvRR5gPC70xpDiQU=
|
||||
golang.org/x/net v0.48.0/go.mod h1:+ndRgGjkh8FGtu1w1FGbEC31if4VrNVMuKTgcAAnQRY=
|
||||
golang.org/x/crypto v0.36.0 h1:AnAEvhDddvBdpY+uR+MyHmuZzzNqXSe/GvuDeob5L34=
|
||||
golang.org/x/crypto v0.36.0/go.mod h1:Y4J0ReaxCR1IMaabaSMugxJES1EpwhBHhv2bDHklZvc=
|
||||
golang.org/x/oauth2 v0.34.0 h1:hqK/t4AKgbqWkdkcAeI8XLmbK+4m4G5YeQRrmiotGlw=
|
||||
golang.org/x/oauth2 v0.34.0/go.mod h1:lzm5WQJQwKZ3nwavOZ3IS5Aulzxi68dUSgRHujetwEA=
|
||||
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.3.0/go.mod h1:FU7BRWz2tNW+3quACPkgCx/L+uEAv1htQ0V83Z9Rj+Y=
|
||||
golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
|
||||
golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
|
||||
golang.org/x/sync v0.9.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
|
||||
golang.org/x/sync v0.19.0 h1:vV+1eWNmZ5geRlYjzm2adRgW2/mcpevXNg50YZtPCE4=
|
||||
golang.org/x/sync v0.19.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
|
||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.0.0-20210616045830-e2b7044e8c71/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.10.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.16.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||
golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||
golang.org/x/sys v0.18.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||
golang.org/x/sys v0.19.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||
golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||
golang.org/x/sys v0.21.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||
golang.org/x/sync v0.17.0 h1:l60nONMj9l5drqw6jlhIELNv9I0A4OFgRsG9k2oT9Ug=
|
||||
golang.org/x/sync v0.17.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
|
||||
golang.org/x/sys v0.40.0 h1:DBZZqJ2Rkml6QMQsZywtnjnnGvHza6BTfYFWY9kjEWQ=
|
||||
golang.org/x/sys v0.40.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
|
||||
golang.org/x/telemetry v0.0.0-20240228155512-f48c80bd79b2/go.mod h1:TeRTkGYfJXctD9OcfyVLyj2J3IxLnKwHJR8f4D8a3YE=
|
||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
|
||||
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
|
||||
golang.org/x/term v0.6.0/go.mod h1:m6U89DPEgQRMq3DNkDClhWw02AUbt2daBVO4cn4Hv9U=
|
||||
golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo=
|
||||
golang.org/x/term v0.10.0/go.mod h1:lpqdcUyK/oCiQxvxVrppt5ggO2KCZ5QblwqPnfZ6d5o=
|
||||
golang.org/x/term v0.11.0/go.mod h1:zC9APTIj3jG3FdV/Ons+XE1riIZXG4aZ4GTHiPZJPIU=
|
||||
golang.org/x/term v0.12.0/go.mod h1:owVbMEjm3cBLCHdkQu9b1opXd4ETQWc3BhuQGKgXgvU=
|
||||
golang.org/x/term v0.16.0/go.mod h1:yn7UURbUtPyrVJPGPq404EukNFxcm/foM+bV/bfcDsY=
|
||||
golang.org/x/term v0.17.0/go.mod h1:lLRBjIVuehSbZlaOtGMbcMncT+aqLLLmKrsjNrUguwk=
|
||||
golang.org/x/term v0.18.0/go.mod h1:ILwASektA3OnRv7amZ1xhE/KTR+u50pbXfZ03+6Nx58=
|
||||
golang.org/x/term v0.19.0/go.mod h1:2CuTdWZ7KHSQwUzKva0cbMg6q2DMI3Mmxp+gKJbskEk=
|
||||
golang.org/x/term v0.20.0/go.mod h1:8UkIAJTvZgivsXaD6/pH6U9ecQzZ45awqEOzuCvwpFY=
|
||||
golang.org/x/term v0.21.0/go.mod h1:ooXLefLobQVslOqselCNF4SxFAaoS6KujMbsGzSDmX0=
|
||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
|
||||
golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
|
||||
golang.org/x/text v0.8.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=
|
||||
golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=
|
||||
golang.org/x/text v0.11.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
|
||||
golang.org/x/text v0.12.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
|
||||
golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
|
||||
golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
|
||||
golang.org/x/text v0.15.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
|
||||
golang.org/x/text v0.16.0/go.mod h1:GhwF1Be+LQoKShO3cGOHzqOgRrGaYc9AvblQOmPVHnI=
|
||||
golang.org/x/text v0.20.0/go.mod h1:D4IsuqiFMhST5bX19pQ9ikHC2GsaKyk/oF+pn3ducp4=
|
||||
golang.org/x/text v0.32.0 h1:ZD01bjUt1FQ9WJ0ClOL5vxgxOI/sVCNgX1YtKwcY0mU=
|
||||
golang.org/x/text v0.32.0/go.mod h1:o/rUWzghvpD5TXrTIBuJU77MTaN0ljMWE47kxGJQ7jY=
|
||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
|
||||
golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
|
||||
golang.org/x/tools v0.13.0/go.mod h1:HvlwmtVNQAhOuCjW7xxvovg8wbNq7LwfXh/k7wXUl58=
|
||||
golang.org/x/tools v0.21.1-0.20240508182429-e35e4ccd0d2d/go.mod h1:aiJjzUbINMkxbQROHiO6hDPo2LHcIPhhQsa9DLh0yGk=
|
||||
golang.org/x/text v0.29.0 h1:1neNs90w9YzJ9BocxfsQNHKuAT4pkghyXc4nhZ6sJvk=
|
||||
golang.org/x/text v0.29.0/go.mod h1:7MhJOA9CD2qZyOKYazxdYMF85OwPdEr9jTtBpO7ydH4=
|
||||
golang.org/x/tools v0.41.0 h1:a9b8iMweWG+S0OBnlU36rzLp20z1Rp10w+IY2czHTQc=
|
||||
golang.org/x/tools v0.41.0/go.mod h1:XSY6eDqxVNiYgezAVqqCeihT4j1U2CCsqvH3WhQpnlg=
|
||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||
google.golang.org/protobuf v1.36.11 h1:fV6ZwhNocDyBLK0dj+fg8ektcVegBBuEolpbTQyBNVE=
|
||||
google.golang.org/protobuf v1.36.11/go.mod h1:HTf+CrKn2C3g5S8VImy6tdcUvCska2kB7j23XfzDpco=
|
||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
|
||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
|
||||
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
|
||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||
gorm.io/driver/postgres v1.6.0 h1:2dxzU8xJ+ivvqTRph34QX+WrRaJlmfyPqXmoGVjMBa4=
|
||||
gorm.io/driver/postgres v1.6.0/go.mod h1:vUw0mrGgrTK+uPHEhAdV4sfFELrByKVGnaVRkXDhtWo=
|
||||
gorm.io/driver/sqlite v1.6.0 h1:WHRRrIiulaPiPFmDcod6prc4l2VGVWHz80KspNsxSfQ=
|
||||
gorm.io/driver/sqlite v1.6.0/go.mod h1:AO9V1qIQddBESngQUKWL9yoH93HIeA1X6V633rBwyT8=
|
||||
gorm.io/driver/sqlserver v1.6.3 h1:UR+nWCuphPnq7UxnL57PSrlYjuvs+sf1N59GgFX7uAI=
|
||||
gorm.io/driver/sqlserver v1.6.3/go.mod h1:VZeNn7hqX1aXoN5TPAFGWvxWG90xtA8erGn2gQmpc6U=
|
||||
gorm.io/gorm v1.30.0/go.mod h1:8Z33v652h4//uMA76KjeDH8mJXPm1QNCYrMeatR0DOE=
|
||||
gorm.io/gorm v1.31.1 h1:7CA8FTFz/gRfgqgpeKIBcervUn3xSyPUmr6B2WXJ7kg=
|
||||
gorm.io/gorm v1.31.1/go.mod h1:XyQVbO2k6YkOis7C2437jSit3SsDK72s7n7rsSHd+Gs=
|
||||
gorm.io/driver/postgres v1.5.4 h1:Iyrp9Meh3GmbSuyIAGyjkN+n9K+GHX9b9MqsTL4EJCo=
|
||||
gorm.io/driver/postgres v1.5.4/go.mod h1:Bgo89+h0CRcdA33Y6frlaHHVuTdOf87pmyzwW9C/BH0=
|
||||
gorm.io/gorm v1.25.5 h1:zR9lOiiYf09VNh5Q1gphfyia1JpiClIWG9hQaxB/mls=
|
||||
gorm.io/gorm v1.25.5/go.mod h1:hbnx/Oo0ChWMn1BIhpy1oYozzpM15i4YPuHDmfYtwg8=
|
||||
mellium.im/sasl v0.3.1 h1:wE0LW6g7U83vhvxjC1IY8DnXM+EU095yeo8XClvCdfo=
|
||||
mellium.im/sasl v0.3.1/go.mod h1:xm59PUYpZHhgQ9ZqoJ5QaCqzWMi8IeS49dhp6plPCzw=
|
||||
modernc.org/libc v1.67.4 h1:zZGmCMUVPORtKv95c2ReQN5VDjvkoRm9GWPTEPuvlWg=
|
||||
modernc.org/libc v1.67.4/go.mod h1:QvvnnJ5P7aitu0ReNpVIEyesuhmDLQ8kaEoyMjIFZJA=
|
||||
modernc.org/mathutil v1.7.1 h1:GCZVGXdaN8gTqB1Mf/usp1Y/hSqgI2vAGGP4jZMCxOU=
|
||||
modernc.org/mathutil v1.7.1/go.mod h1:4p5IwJITfppl0G4sUEDtCr4DthTaT47/N3aT6MhfgJg=
|
||||
modernc.org/memory v1.11.0 h1:o4QC8aMQzmcwCK3t3Ux/ZHmwFPzE6hf2Y5LbkRs+hbI=
|
||||
modernc.org/memory v1.11.0/go.mod h1:/JP4VbVC+K5sU2wZi9bHoq2MAkCnrt2r98UGeSK7Mjw=
|
||||
modernc.org/sqlite v1.42.2 h1:7hkZUNJvJFN2PgfUdjni9Kbvd4ef4mNLOu0B9FGxM74=
|
||||
modernc.org/sqlite v1.42.2/go.mod h1:+VkC6v3pLOAE0A0uVucQEcbVW0I5nHCeDaBf+DpsQT8=
|
||||
|
||||
@@ -1,79 +0,0 @@
|
||||
package app
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"log/slog"
|
||||
"net/http"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/tools"
|
||||
)
|
||||
|
||||
type adminActions struct {
|
||||
backfill *tools.BackfillTool
|
||||
retry *tools.EnrichmentRetryer
|
||||
logger *slog.Logger
|
||||
}
|
||||
|
||||
func newAdminActions(backfill *tools.BackfillTool, retry *tools.EnrichmentRetryer, logger *slog.Logger) *adminActions {
|
||||
return &adminActions{
|
||||
backfill: backfill,
|
||||
retry: retry,
|
||||
logger: logger,
|
||||
}
|
||||
}
|
||||
|
||||
func (a *adminActions) backfillHandler() http.Handler {
|
||||
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||
if r.Method != http.MethodPost {
|
||||
w.Header().Set("Allow", http.MethodPost)
|
||||
http.Error(w, "method not allowed", http.StatusMethodNotAllowed)
|
||||
return
|
||||
}
|
||||
|
||||
var in tools.BackfillInput
|
||||
if err := json.NewDecoder(r.Body).Decode(&in); err != nil {
|
||||
http.Error(w, "invalid request body", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
_, out, err := a.backfill.Handle(r.Context(), nil, in)
|
||||
if err != nil {
|
||||
if a.logger != nil {
|
||||
a.logger.Warn("admin backfill failed", slog.String("error", err.Error()))
|
||||
}
|
||||
http.Error(w, err.Error(), http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
w.Header().Set("Content-Type", "application/json")
|
||||
_ = json.NewEncoder(w).Encode(out)
|
||||
})
|
||||
}
|
||||
|
||||
func (a *adminActions) retryMetadataHandler() http.Handler {
|
||||
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||
if r.Method != http.MethodPost {
|
||||
w.Header().Set("Allow", http.MethodPost)
|
||||
http.Error(w, "method not allowed", http.StatusMethodNotAllowed)
|
||||
return
|
||||
}
|
||||
|
||||
var in tools.RetryEnrichmentInput
|
||||
if err := json.NewDecoder(r.Body).Decode(&in); err != nil {
|
||||
http.Error(w, "invalid request body", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
_, out, err := a.retry.Handle(r.Context(), nil, in)
|
||||
if err != nil {
|
||||
if a.logger != nil {
|
||||
a.logger.Warn("admin metadata retry failed", slog.String("error", err.Error()))
|
||||
}
|
||||
http.Error(w, err.Error(), http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
w.Header().Set("Content-Type", "application/json")
|
||||
_ = json.NewEncoder(w).Encode(out)
|
||||
})
|
||||
}
|
||||
@@ -1,268 +0,0 @@
|
||||
package app
|
||||
// Legacy admin handlers retired in favor of ResolveSpec-backed routes.
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"log/slog"
|
||||
"net/http"
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/store"
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
|
||||
"github.com/google/uuid"
|
||||
)
|
||||
|
||||
type adminHandlers struct {
|
||||
db *store.DB
|
||||
logger *slog.Logger
|
||||
}
|
||||
|
||||
func newAdminHandlers(db *store.DB, logger *slog.Logger) *adminHandlers {
|
||||
return &adminHandlers{db: db, logger: logger}
|
||||
}
|
||||
|
||||
func (h *adminHandlers) register(mux *http.ServeMux, middleware func(http.Handler) http.Handler) {
|
||||
handle := func(pattern string, fn http.HandlerFunc) {
|
||||
mux.Handle(pattern, middleware(fn))
|
||||
}
|
||||
|
||||
handle("GET /api/admin/projects", h.listProjects)
|
||||
handle("POST /api/admin/projects", h.createProject)
|
||||
handle("GET /api/admin/thoughts", h.listThoughts)
|
||||
handle("GET /api/admin/thoughts/{id}", h.getThought)
|
||||
handle("DELETE /api/admin/thoughts/{id}", h.deleteThought)
|
||||
handle("POST /api/admin/thoughts/{id}/archive", h.archiveThought)
|
||||
handle("GET /api/admin/skills", h.listSkills)
|
||||
handle("DELETE /api/admin/skills/{id}", h.deleteSkill)
|
||||
handle("GET /api/admin/guardrails", h.listGuardrails)
|
||||
handle("DELETE /api/admin/guardrails/{id}", h.deleteGuardrail)
|
||||
handle("GET /api/admin/files", h.listFiles)
|
||||
handle("GET /api/admin/stats", h.stats)
|
||||
}
|
||||
|
||||
// --- Projects ---
|
||||
|
||||
func (h *adminHandlers) listProjects(w http.ResponseWriter, r *http.Request) {
|
||||
projects, err := h.db.ListProjects(r.Context())
|
||||
if err != nil {
|
||||
h.internalError(w, "list projects", err)
|
||||
return
|
||||
}
|
||||
writeJSON(w, projects)
|
||||
}
|
||||
|
||||
func (h *adminHandlers) createProject(w http.ResponseWriter, r *http.Request) {
|
||||
var body struct {
|
||||
Name string `json:"name"`
|
||||
Description string `json:"description"`
|
||||
}
|
||||
if err := json.NewDecoder(r.Body).Decode(&body); err != nil {
|
||||
http.Error(w, "invalid request body", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
if strings.TrimSpace(body.Name) == "" {
|
||||
http.Error(w, "name is required", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
project, err := h.db.CreateProject(r.Context(), body.Name, body.Description)
|
||||
if err != nil {
|
||||
h.internalError(w, "create project", err)
|
||||
return
|
||||
}
|
||||
w.WriteHeader(http.StatusCreated)
|
||||
writeJSON(w, project)
|
||||
}
|
||||
|
||||
// --- Thoughts ---
|
||||
|
||||
func (h *adminHandlers) listThoughts(w http.ResponseWriter, r *http.Request) {
|
||||
q := r.URL.Query()
|
||||
limit := 50
|
||||
if l := q.Get("limit"); l != "" {
|
||||
if n, err := strconv.Atoi(l); err == nil && n > 0 {
|
||||
limit = min(n, 200)
|
||||
}
|
||||
}
|
||||
|
||||
query := strings.TrimSpace(q.Get("q"))
|
||||
includeArchived := q.Get("include_archived") == "true"
|
||||
|
||||
var projectID *uuid.UUID
|
||||
if pid := q.Get("project_id"); pid != "" {
|
||||
if id, err := uuid.Parse(pid); err == nil {
|
||||
projectID = &id
|
||||
}
|
||||
}
|
||||
|
||||
if query != "" {
|
||||
results, err := h.db.SearchThoughtsText(r.Context(), query, limit, projectID, nil)
|
||||
if err != nil {
|
||||
h.internalError(w, "search thoughts", err)
|
||||
return
|
||||
}
|
||||
writeJSON(w, results)
|
||||
return
|
||||
}
|
||||
|
||||
thoughts, err := h.db.ListThoughts(r.Context(), ext.ListFilter{
|
||||
Limit: limit,
|
||||
ProjectID: projectID,
|
||||
IncludeArchived: includeArchived,
|
||||
})
|
||||
if err != nil {
|
||||
h.internalError(w, "list thoughts", err)
|
||||
return
|
||||
}
|
||||
writeJSON(w, thoughts)
|
||||
}
|
||||
|
||||
func (h *adminHandlers) getThought(w http.ResponseWriter, r *http.Request) {
|
||||
id, ok := parseUUID(w, r.PathValue("id"))
|
||||
if !ok {
|
||||
return
|
||||
}
|
||||
thought, err := h.db.GetThought(r.Context(), id)
|
||||
if err != nil {
|
||||
h.internalError(w, "get thought", err)
|
||||
return
|
||||
}
|
||||
writeJSON(w, thought)
|
||||
}
|
||||
|
||||
func (h *adminHandlers) deleteThought(w http.ResponseWriter, r *http.Request) {
|
||||
id, ok := parseUUID(w, r.PathValue("id"))
|
||||
if !ok {
|
||||
return
|
||||
}
|
||||
if err := h.db.DeleteThought(r.Context(), id); err != nil {
|
||||
h.internalError(w, "delete thought", err)
|
||||
return
|
||||
}
|
||||
w.WriteHeader(http.StatusNoContent)
|
||||
}
|
||||
|
||||
func (h *adminHandlers) archiveThought(w http.ResponseWriter, r *http.Request) {
|
||||
id, ok := parseUUID(w, r.PathValue("id"))
|
||||
if !ok {
|
||||
return
|
||||
}
|
||||
if err := h.db.ArchiveThought(r.Context(), id); err != nil {
|
||||
h.internalError(w, "archive thought", err)
|
||||
return
|
||||
}
|
||||
w.WriteHeader(http.StatusNoContent)
|
||||
}
|
||||
|
||||
// --- Skills ---
|
||||
|
||||
func (h *adminHandlers) listSkills(w http.ResponseWriter, r *http.Request) {
|
||||
tag := r.URL.Query().Get("tag")
|
||||
skills, err := h.db.ListSkills(r.Context(), tag)
|
||||
if err != nil {
|
||||
h.internalError(w, "list skills", err)
|
||||
return
|
||||
}
|
||||
writeJSON(w, skills)
|
||||
}
|
||||
|
||||
func (h *adminHandlers) deleteSkill(w http.ResponseWriter, r *http.Request) {
|
||||
id, ok := parseUUID(w, r.PathValue("id"))
|
||||
if !ok {
|
||||
return
|
||||
}
|
||||
if err := h.db.RemoveSkill(r.Context(), id); err != nil {
|
||||
h.internalError(w, "delete skill", err)
|
||||
return
|
||||
}
|
||||
w.WriteHeader(http.StatusNoContent)
|
||||
}
|
||||
|
||||
// --- Guardrails ---
|
||||
|
||||
func (h *adminHandlers) listGuardrails(w http.ResponseWriter, r *http.Request) {
|
||||
q := r.URL.Query()
|
||||
guardrails, err := h.db.ListGuardrails(r.Context(), q.Get("tag"), q.Get("severity"))
|
||||
if err != nil {
|
||||
h.internalError(w, "list guardrails", err)
|
||||
return
|
||||
}
|
||||
writeJSON(w, guardrails)
|
||||
}
|
||||
|
||||
func (h *adminHandlers) deleteGuardrail(w http.ResponseWriter, r *http.Request) {
|
||||
id, ok := parseUUID(w, r.PathValue("id"))
|
||||
if !ok {
|
||||
return
|
||||
}
|
||||
if err := h.db.RemoveGuardrail(r.Context(), id); err != nil {
|
||||
h.internalError(w, "delete guardrail", err)
|
||||
return
|
||||
}
|
||||
w.WriteHeader(http.StatusNoContent)
|
||||
}
|
||||
|
||||
// --- Files ---
|
||||
|
||||
func (h *adminHandlers) listFiles(w http.ResponseWriter, r *http.Request) {
|
||||
q := r.URL.Query()
|
||||
limit := 100
|
||||
if l := q.Get("limit"); l != "" {
|
||||
if n, err := strconv.Atoi(l); err == nil && n > 0 {
|
||||
limit = min(n, 500)
|
||||
}
|
||||
}
|
||||
|
||||
filter := ext.StoredFileFilter{Limit: limit}
|
||||
if pid := q.Get("project_id"); pid != "" {
|
||||
if id, err := uuid.Parse(pid); err == nil {
|
||||
filter.ProjectID = &id
|
||||
}
|
||||
}
|
||||
if tid := q.Get("thought_id"); tid != "" {
|
||||
if id, err := uuid.Parse(tid); err == nil {
|
||||
filter.ThoughtID = &id
|
||||
}
|
||||
}
|
||||
filter.Kind = q.Get("kind")
|
||||
|
||||
files, err := h.db.ListStoredFiles(r.Context(), filter)
|
||||
if err != nil {
|
||||
h.internalError(w, "list files", err)
|
||||
return
|
||||
}
|
||||
writeJSON(w, files)
|
||||
}
|
||||
|
||||
// --- Stats ---
|
||||
|
||||
func (h *adminHandlers) stats(w http.ResponseWriter, r *http.Request) {
|
||||
stats, err := h.db.Stats(r.Context())
|
||||
if err != nil {
|
||||
h.internalError(w, "stats", err)
|
||||
return
|
||||
}
|
||||
writeJSON(w, stats)
|
||||
}
|
||||
|
||||
// --- Helpers ---
|
||||
|
||||
func (h *adminHandlers) internalError(w http.ResponseWriter, op string, err error) {
|
||||
h.logger.Error("admin handler error", slog.String("op", op), slog.String("error", err.Error()))
|
||||
http.Error(w, "internal server error", http.StatusInternalServerError)
|
||||
}
|
||||
|
||||
func writeJSON(w http.ResponseWriter, v any) {
|
||||
w.Header().Set("Content-Type", "application/json")
|
||||
_ = json.NewEncoder(w).Encode(v)
|
||||
}
|
||||
|
||||
func parseUUID(w http.ResponseWriter, s string) (uuid.UUID, bool) {
|
||||
id, err := uuid.Parse(s)
|
||||
if err != nil {
|
||||
http.Error(w, "invalid id", http.StatusBadRequest)
|
||||
return uuid.UUID{}, false
|
||||
}
|
||||
return id, true
|
||||
}
|
||||
+17
-26
@@ -92,15 +92,15 @@ func Run(ctx context.Context, configPath string) error {
|
||||
return err
|
||||
}
|
||||
}
|
||||
tokenStore = auth.NewTokenStore(0)
|
||||
if len(cfg.Auth.OAuth.Clients) > 0 {
|
||||
oauthRegistry, err = auth.NewOAuthRegistry(cfg.Auth.OAuth.Clients)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
tokenStore = auth.NewTokenStore(0)
|
||||
}
|
||||
authCodes := auth.NewAuthCodeStore()
|
||||
dynClients := auth.NewPostgresClientStore(db.Pool())
|
||||
dynClients := auth.NewDynamicClientStore()
|
||||
activeProjects := session.NewActiveProjects()
|
||||
|
||||
logger.Info("ai providers initialised",
|
||||
@@ -183,15 +183,14 @@ func Run(ctx context.Context, configPath string) error {
|
||||
}
|
||||
}
|
||||
|
||||
func routes(logger *slog.Logger, cfg *config.Config, info buildinfo.Info, db *store.DB, embeddings *ai.EmbeddingRunner, metadata *ai.MetadataRunner, bgEmbeddings *ai.EmbeddingRunner, bgMetadata *ai.MetadataRunner, keyring *auth.Keyring, oauthRegistry *auth.OAuthRegistry, tokenStore *auth.TokenStore, authCodes *auth.AuthCodeStore, dynClients auth.ClientStore, activeProjects *session.ActiveProjects) (http.Handler, error) {
|
||||
func routes(logger *slog.Logger, cfg *config.Config, info buildinfo.Info, db *store.DB, embeddings *ai.EmbeddingRunner, metadata *ai.MetadataRunner, bgEmbeddings *ai.EmbeddingRunner, bgMetadata *ai.MetadataRunner, keyring *auth.Keyring, oauthRegistry *auth.OAuthRegistry, tokenStore *auth.TokenStore, authCodes *auth.AuthCodeStore, dynClients *auth.DynamicClientStore, activeProjects *session.ActiveProjects) (http.Handler, error) {
|
||||
mux := http.NewServeMux()
|
||||
accessTracker := auth.NewAccessTracker()
|
||||
oauthEnabled := oauthRegistry != nil
|
||||
oauthEnabled := oauthRegistry != nil && tokenStore != nil
|
||||
authMiddleware := auth.Middleware(cfg.Auth, keyring, oauthRegistry, tokenStore, accessTracker, logger)
|
||||
filesTool := tools.NewFilesTool(db, activeProjects)
|
||||
enrichmentRetryer := tools.NewEnrichmentRetryer(context.Background(), db, bgMetadata, cfg.Capture, cfg.AI.Metadata.Timeout, activeProjects, logger)
|
||||
backfillTool := tools.NewBackfillTool(db, bgEmbeddings, activeProjects, logger)
|
||||
adminActions := newAdminActions(backfillTool, enrichmentRetryer, logger)
|
||||
|
||||
toolSet := mcpserver.ToolSet{
|
||||
Capture: tools.NewCaptureTool(db, embeddings, cfg.Capture, activeProjects, enrichmentRetryer, backfillTool),
|
||||
@@ -204,8 +203,6 @@ func routes(logger *slog.Logger, cfg *config.Config, info buildinfo.Info, db *st
|
||||
Archive: tools.NewArchiveTool(db),
|
||||
Projects: tools.NewProjectsTool(db, activeProjects),
|
||||
Version: tools.NewVersionTool(cfg.MCP.ServerName, info),
|
||||
Learnings: tools.NewLearningsTool(db, activeProjects, cfg.Search),
|
||||
Plans: tools.NewPlansTool(db, activeProjects, cfg.Search),
|
||||
Context: tools.NewContextTool(db, embeddings, cfg.Search, activeProjects),
|
||||
Recall: tools.NewRecallTool(db, embeddings, cfg.Search, activeProjects),
|
||||
Summarize: tools.NewSummarizeTool(db, embeddings, metadata, cfg.Search, activeProjects),
|
||||
@@ -214,11 +211,10 @@ func routes(logger *slog.Logger, cfg *config.Config, info buildinfo.Info, db *st
|
||||
Backfill: backfillTool,
|
||||
Reparse: tools.NewReparseMetadataTool(db, bgMetadata, cfg.Capture, activeProjects, logger),
|
||||
RetryMetadata: tools.NewRetryEnrichmentTool(enrichmentRetryer),
|
||||
//Maintenance: tools.NewMaintenanceTool(db),
|
||||
Skills: tools.NewSkillsTool(db, activeProjects),
|
||||
Personas: tools.NewAgentPersonasTool(db),
|
||||
ChatHistory: tools.NewChatHistoryTool(db, activeProjects),
|
||||
Describe: tools.NewDescribeTool(db, mcpserver.BuildToolCatalog()),
|
||||
Maintenance: tools.NewMaintenanceTool(db),
|
||||
Skills: tools.NewSkillsTool(db, activeProjects),
|
||||
ChatHistory: tools.NewChatHistoryTool(db, activeProjects),
|
||||
Describe: tools.NewDescribeTool(db, mcpserver.BuildToolCatalog()),
|
||||
}
|
||||
|
||||
mcpHandlers, err := mcpserver.NewHandlers(cfg.MCP, logger, toolSet, activeProjects.Clear)
|
||||
@@ -230,26 +226,21 @@ func routes(logger *slog.Logger, cfg *config.Config, info buildinfo.Info, db *st
|
||||
mux.Handle(cfg.MCP.SSEPath, authMiddleware(mcpHandlers.SSE))
|
||||
logger.Info("SSE transport enabled", slog.String("sse_path", cfg.MCP.SSEPath))
|
||||
}
|
||||
if err := registerResolveSpecAdminRoutes(mux, db, authMiddleware, logger); err != nil {
|
||||
return nil, fmt.Errorf("setup resolvespec admin routes: %w", err)
|
||||
}
|
||||
mux.Handle("/files", authMiddleware(fileHandler(filesTool)))
|
||||
mux.Handle("/files/{id}", authMiddleware(fileHandler(filesTool)))
|
||||
mux.HandleFunc("/.well-known/oauth-authorization-server", oauthMetadataHandler())
|
||||
mux.HandleFunc("/api/oauth/register", oauthRegisterHandler(dynClients, logger))
|
||||
mux.HandleFunc("/api/oauth/authorize", oauthAuthorizeHandler(dynClients, authCodes, logger))
|
||||
mux.HandleFunc("/api/oauth/token", oauthTokenHandler(oauthRegistry, tokenStore, authCodes, logger))
|
||||
mux.Handle("/api/admin/actions/backfill", authMiddleware(adminActions.backfillHandler()))
|
||||
mux.Handle("/api/admin/actions/retry-metadata", authMiddleware(adminActions.retryMetadataHandler()))
|
||||
if oauthEnabled {
|
||||
mux.HandleFunc("/.well-known/oauth-authorization-server", oauthMetadataHandler())
|
||||
mux.HandleFunc("/oauth-authorization-server", oauthMetadataHandler())
|
||||
mux.HandleFunc("/oauth/register", oauthRegisterHandler(dynClients, logger))
|
||||
mux.HandleFunc("/authorize", oauthAuthorizeHandler(dynClients, authCodes, logger))
|
||||
mux.HandleFunc("/oauth/authorize", oauthAuthorizeHandler(dynClients, authCodes, logger))
|
||||
mux.HandleFunc("/oauth/token", oauthTokenHandler(oauthRegistry, tokenStore, authCodes, logger))
|
||||
}
|
||||
mux.HandleFunc("/favicon.ico", serveFavicon)
|
||||
mux.HandleFunc("/images/project.jpg", serveHomeImage)
|
||||
mux.HandleFunc("/images/icon.png", serveIcon)
|
||||
mux.HandleFunc("/llm", serveLLMInstructions)
|
||||
mux.HandleFunc("/llms.txt", serveLLMSTXT)
|
||||
mux.HandleFunc("/.well-known/llms.txt", serveLLMSTXT)
|
||||
mux.HandleFunc("/robots.txt", serveRobotsTXT)
|
||||
mux.Handle("/api/status", authMiddleware(statusAPIHandler(info, accessTracker, oauthEnabled)))
|
||||
mux.HandleFunc("/status", publicStatusHandler(accessTracker))
|
||||
mux.HandleFunc("/api/status", statusAPIHandler(info, accessTracker, oauthEnabled))
|
||||
|
||||
mux.HandleFunc("/healthz", func(w http.ResponseWriter, r *http.Request) {
|
||||
w.WriteHeader(http.StatusOK)
|
||||
|
||||
@@ -1,9 +1,7 @@
|
||||
package app
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/http"
|
||||
"strings"
|
||||
|
||||
amcsllm "git.warky.dev/wdevs/amcs/llm"
|
||||
)
|
||||
@@ -22,74 +20,3 @@ func serveLLMInstructions(w http.ResponseWriter, r *http.Request) {
|
||||
}
|
||||
_, _ = w.Write(amcsllm.MemoryInstructions)
|
||||
}
|
||||
|
||||
func serveRobotsTXT(w http.ResponseWriter, r *http.Request) {
|
||||
if r.URL.Path != "/robots.txt" {
|
||||
http.NotFound(w, r)
|
||||
return
|
||||
}
|
||||
if r.Method != http.MethodGet && r.Method != http.MethodHead {
|
||||
w.Header().Set("Allow", "GET, HEAD")
|
||||
http.Error(w, "method not allowed", http.StatusMethodNotAllowed)
|
||||
return
|
||||
}
|
||||
|
||||
w.Header().Set("Content-Type", "text/plain; charset=utf-8")
|
||||
w.Header().Set("Cache-Control", "public, max-age=300")
|
||||
w.WriteHeader(http.StatusOK)
|
||||
if r.Method == http.MethodHead {
|
||||
return
|
||||
}
|
||||
|
||||
body := fmt.Sprintf("User-agent: *\nAllow: /\n\n# LLM-friendly docs\nLLM: %s/llm\nLLMS: %s/llms.txt\n", requestBaseURL(r), requestBaseURL(r))
|
||||
_, _ = w.Write([]byte(body))
|
||||
}
|
||||
|
||||
func serveLLMSTXT(w http.ResponseWriter, r *http.Request) {
|
||||
if r.URL.Path != "/llms.txt" && r.URL.Path != "/.well-known/llms.txt" {
|
||||
http.NotFound(w, r)
|
||||
return
|
||||
}
|
||||
if r.Method != http.MethodGet && r.Method != http.MethodHead {
|
||||
w.Header().Set("Allow", "GET, HEAD")
|
||||
http.Error(w, "method not allowed", http.StatusMethodNotAllowed)
|
||||
return
|
||||
}
|
||||
|
||||
w.Header().Set("Content-Type", "text/plain; charset=utf-8")
|
||||
w.Header().Set("Cache-Control", "public, max-age=300")
|
||||
w.WriteHeader(http.StatusOK)
|
||||
if r.Method == http.MethodHead {
|
||||
return
|
||||
}
|
||||
|
||||
base := requestBaseURL(r)
|
||||
body := fmt.Sprintf(
|
||||
"# AMCS\n\n> A memory server for AI assistants (MCP tools, semantic retrieval, and structured project memory).\n\n## Endpoints\n- %s/llm\n- %s/status\n- %s/mcp\n- %s/.well-known/oauth-authorization-server\n",
|
||||
base,
|
||||
base,
|
||||
base,
|
||||
base,
|
||||
)
|
||||
_, _ = w.Write([]byte(body))
|
||||
}
|
||||
|
||||
func requestBaseURL(r *http.Request) string {
|
||||
scheme := "http"
|
||||
if r != nil && r.TLS != nil {
|
||||
scheme = "https"
|
||||
}
|
||||
if r != nil {
|
||||
if proto := strings.TrimSpace(r.Header.Get("X-Forwarded-Proto")); proto != "" {
|
||||
scheme = proto
|
||||
}
|
||||
}
|
||||
|
||||
host := "localhost"
|
||||
if r != nil {
|
||||
if v := strings.TrimSpace(r.Host); v != "" {
|
||||
host = v
|
||||
}
|
||||
}
|
||||
return scheme + "://" + host
|
||||
}
|
||||
|
||||
@@ -3,7 +3,6 @@ package app
|
||||
import (
|
||||
"net/http"
|
||||
"net/http/httptest"
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
amcsllm "git.warky.dev/wdevs/amcs/llm"
|
||||
@@ -30,70 +29,3 @@ func TestServeLLMInstructions(t *testing.T) {
|
||||
t.Fatalf("body = %q, want embedded instructions", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestServeRobotsTXT(t *testing.T) {
|
||||
req := httptest.NewRequest(http.MethodGet, "/robots.txt", nil)
|
||||
req.Host = "amcs.example.com"
|
||||
req.Header.Set("X-Forwarded-Proto", "https")
|
||||
rec := httptest.NewRecorder()
|
||||
|
||||
serveRobotsTXT(rec, req)
|
||||
|
||||
res := rec.Result()
|
||||
defer func() {
|
||||
_ = res.Body.Close()
|
||||
}()
|
||||
|
||||
if res.StatusCode != http.StatusOK {
|
||||
t.Fatalf("status = %d, want %d", res.StatusCode, http.StatusOK)
|
||||
}
|
||||
if got := res.Header.Get("Content-Type"); got != "text/plain; charset=utf-8" {
|
||||
t.Fatalf("content-type = %q, want %q", got, "text/plain; charset=utf-8")
|
||||
}
|
||||
body := rec.Body.String()
|
||||
if !strings.Contains(body, "LLM: https://amcs.example.com/llm") {
|
||||
t.Fatalf("body = %q, want LLM link", body)
|
||||
}
|
||||
if !strings.Contains(body, "LLMS: https://amcs.example.com/llms.txt") {
|
||||
t.Fatalf("body = %q, want LLMS link", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestServeLLMSTXT(t *testing.T) {
|
||||
req := httptest.NewRequest(http.MethodGet, "/llms.txt", nil)
|
||||
req.Host = "amcs.example.com"
|
||||
req.Header.Set("X-Forwarded-Proto", "https")
|
||||
rec := httptest.NewRecorder()
|
||||
|
||||
serveLLMSTXT(rec, req)
|
||||
|
||||
res := rec.Result()
|
||||
defer func() {
|
||||
_ = res.Body.Close()
|
||||
}()
|
||||
|
||||
if res.StatusCode != http.StatusOK {
|
||||
t.Fatalf("status = %d, want %d", res.StatusCode, http.StatusOK)
|
||||
}
|
||||
if got := res.Header.Get("Content-Type"); got != "text/plain; charset=utf-8" {
|
||||
t.Fatalf("content-type = %q, want %q", got, "text/plain; charset=utf-8")
|
||||
}
|
||||
body := rec.Body.String()
|
||||
if !strings.Contains(body, "https://amcs.example.com/llm") {
|
||||
t.Fatalf("body = %q, want /llm link", body)
|
||||
}
|
||||
if !strings.Contains(body, "https://amcs.example.com/.well-known/oauth-authorization-server") {
|
||||
t.Fatalf("body = %q, want oauth discovery link", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestServeLLMSTXTWellKnownPath(t *testing.T) {
|
||||
req := httptest.NewRequest(http.MethodGet, "/.well-known/llms.txt", nil)
|
||||
rec := httptest.NewRecorder()
|
||||
|
||||
serveLLMSTXT(rec, req)
|
||||
|
||||
if rec.Code != http.StatusOK {
|
||||
t.Fatalf("status = %d, want %d", rec.Code, http.StatusOK)
|
||||
}
|
||||
}
|
||||
|
||||
+8
-12
@@ -67,9 +67,9 @@ func oauthMetadataHandler() http.HandlerFunc {
|
||||
base := serverBaseURL(r)
|
||||
meta := oauthServerMetadata{
|
||||
Issuer: base,
|
||||
AuthorizationEndpoint: base + "/api/oauth/authorize",
|
||||
TokenEndpoint: base + "/api/oauth/token",
|
||||
RegistrationEndpoint: base + "/api/oauth/register",
|
||||
AuthorizationEndpoint: base + "/authorize",
|
||||
TokenEndpoint: base + "/oauth/token",
|
||||
RegistrationEndpoint: base + "/oauth/register",
|
||||
ScopesSupported: []string{"mcp"},
|
||||
ResponseTypesSupported: []string{"code"},
|
||||
GrantTypesSupported: []string{"authorization_code", "client_credentials"},
|
||||
@@ -83,7 +83,7 @@ func oauthMetadataHandler() http.HandlerFunc {
|
||||
|
||||
// oauthRegisterHandler serves POST /oauth/register per RFC 7591
|
||||
// (OAuth 2.0 Dynamic Client Registration).
|
||||
func oauthRegisterHandler(dynClients auth.ClientStore, log *slog.Logger) http.HandlerFunc {
|
||||
func oauthRegisterHandler(dynClients *auth.DynamicClientStore, log *slog.Logger) http.HandlerFunc {
|
||||
return func(w http.ResponseWriter, r *http.Request) {
|
||||
if r.Method != http.MethodPost {
|
||||
w.Header().Set("Allow", http.MethodPost)
|
||||
@@ -130,7 +130,7 @@ func oauthRegisterHandler(dynClients auth.ClientStore, log *slog.Logger) http.Ha
|
||||
|
||||
// oauthAuthorizeHandler serves GET and POST /oauth/authorize.
|
||||
// GET shows an approval page; POST processes the user's approve/deny action.
|
||||
func oauthAuthorizeHandler(dynClients auth.ClientStore, authCodes *auth.AuthCodeStore, log *slog.Logger) http.HandlerFunc {
|
||||
func oauthAuthorizeHandler(dynClients *auth.DynamicClientStore, authCodes *auth.AuthCodeStore, log *slog.Logger) http.HandlerFunc {
|
||||
return func(w http.ResponseWriter, r *http.Request) {
|
||||
switch r.Method {
|
||||
case http.MethodGet:
|
||||
@@ -144,7 +144,7 @@ func oauthAuthorizeHandler(dynClients auth.ClientStore, authCodes *auth.AuthCode
|
||||
}
|
||||
}
|
||||
|
||||
func handleAuthorizeGET(w http.ResponseWriter, r *http.Request, dynClients auth.ClientStore) {
|
||||
func handleAuthorizeGET(w http.ResponseWriter, r *http.Request, dynClients *auth.DynamicClientStore) {
|
||||
q := r.URL.Query()
|
||||
clientID := q.Get("client_id")
|
||||
redirectURI := q.Get("redirect_uri")
|
||||
@@ -178,7 +178,7 @@ func handleAuthorizeGET(w http.ResponseWriter, r *http.Request, dynClients auth.
|
||||
serveAuthorizePage(w, client.ClientName, clientID, redirectURI, state, codeChallenge, codeChallengeMethod, scope)
|
||||
}
|
||||
|
||||
func handleAuthorizePOST(w http.ResponseWriter, r *http.Request, dynClients auth.ClientStore, authCodes *auth.AuthCodeStore, log *slog.Logger) {
|
||||
func handleAuthorizePOST(w http.ResponseWriter, r *http.Request, dynClients *auth.DynamicClientStore, authCodes *auth.AuthCodeStore, log *slog.Logger) {
|
||||
if err := r.ParseForm(); err != nil {
|
||||
http.Error(w, "invalid form", http.StatusBadRequest)
|
||||
return
|
||||
@@ -244,10 +244,6 @@ func oauthTokenHandler(oauthRegistry *auth.OAuthRegistry, tokenStore *auth.Token
|
||||
|
||||
switch r.FormValue("grant_type") {
|
||||
case "client_credentials":
|
||||
if oauthRegistry == nil {
|
||||
writeTokenError(w, "unsupported_grant_type", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
handleClientCredentials(w, r, oauthRegistry, tokenStore, log)
|
||||
case "authorization_code":
|
||||
handleAuthorizationCode(w, r, authCodes, tokenStore, log)
|
||||
@@ -338,7 +334,7 @@ button{padding:.5rem 1.2rem;margin-right:.5rem;cursor:pointer;font-size:1rem}
|
||||
<body>
|
||||
<h2>Authorize Access</h2>
|
||||
<p><strong>%s</strong> is requesting access to this AMCS server.</p>
|
||||
<form method=POST action=/api/oauth/authorize>
|
||||
<form method=POST action=/oauth/authorize>
|
||||
<input type=hidden name=client_id value="%s">
|
||||
<input type=hidden name=redirect_uri value="%s">
|
||||
<input type=hidden name=state value="%s">
|
||||
|
||||
@@ -1,149 +0,0 @@
|
||||
package app
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"log/slog"
|
||||
"net/http"
|
||||
"strings"
|
||||
|
||||
"github.com/bitechdev/ResolveSpec/pkg/resolvespec"
|
||||
"github.com/uptrace/bunrouter"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/store"
|
||||
)
|
||||
|
||||
func registerResolveSpecAdminRoutes(mux *http.ServeMux, db *store.DB, middleware func(http.Handler) http.Handler, logger *slog.Logger) error {
|
||||
rs := resolvespec.NewHandlerWithBun(db.Bun())
|
||||
registerResolveSpecGuards(rs)
|
||||
for _, model := range resolveSpecModels() {
|
||||
if err := rs.RegisterModel(model.schema, model.entity, model.model); err != nil {
|
||||
return fmt.Errorf("register resolvespec model %s.%s: %w", model.schema, model.entity, err)
|
||||
}
|
||||
}
|
||||
|
||||
rsRouter := bunrouter.New()
|
||||
resolvespec.SetupBunRouterRoutes(rsRouter, rs, nil)
|
||||
|
||||
rsMount := http.StripPrefix("/api/rs", rsRouter)
|
||||
protectedRSMount := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||
if r.URL.Path != "/" && strings.HasSuffix(r.URL.Path, "/") {
|
||||
trimmed := strings.TrimRight(r.URL.Path, "/")
|
||||
if trimmed == "" {
|
||||
trimmed = "/"
|
||||
}
|
||||
clone := r.Clone(r.Context())
|
||||
clone.URL.Path = trimmed
|
||||
if clone.URL.RawPath != "" {
|
||||
clone.URL.RawPath = strings.TrimRight(clone.URL.RawPath, "/")
|
||||
if clone.URL.RawPath == "" {
|
||||
clone.URL.RawPath = "/"
|
||||
}
|
||||
}
|
||||
r = clone
|
||||
}
|
||||
if r.Method == http.MethodOptions {
|
||||
rsMount.ServeHTTP(w, r)
|
||||
return
|
||||
}
|
||||
middleware(rsMount).ServeHTTP(w, r)
|
||||
})
|
||||
|
||||
mux.Handle("/api/rs/", protectedRSMount)
|
||||
mux.Handle("/api/rs", http.RedirectHandler("/api/rs/openapi", http.StatusTemporaryRedirect))
|
||||
|
||||
if logger != nil {
|
||||
logger.Info("resolvespec admin api enabled",
|
||||
slog.String("prefix", "/api/rs"),
|
||||
slog.Int("models", len(resolveSpecModels())),
|
||||
)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func registerResolveSpecGuards(rs *resolvespec.Handler) {
|
||||
mutableByEntity := map[string]map[string]struct{}{
|
||||
"projects": {
|
||||
"create": {},
|
||||
"update": {},
|
||||
"delete": {},
|
||||
},
|
||||
"thoughts": {
|
||||
"create": {},
|
||||
"update": {},
|
||||
"delete": {},
|
||||
},
|
||||
"plans": {
|
||||
"create": {},
|
||||
"update": {},
|
||||
"delete": {},
|
||||
},
|
||||
"learnings": {
|
||||
"create": {},
|
||||
"update": {},
|
||||
"delete": {},
|
||||
},
|
||||
"agent_personas": {
|
||||
"create": {},
|
||||
"update": {},
|
||||
"delete": {},
|
||||
},
|
||||
"agent_parts": {
|
||||
"create": {},
|
||||
"update": {},
|
||||
"delete": {},
|
||||
},
|
||||
"agent_traits": {
|
||||
"create": {},
|
||||
"update": {},
|
||||
"delete": {},
|
||||
},
|
||||
"agent_skills": {
|
||||
"create": {},
|
||||
"update": {},
|
||||
"delete": {},
|
||||
},
|
||||
"agent_guardrails": {
|
||||
"create": {},
|
||||
"update": {},
|
||||
"delete": {},
|
||||
},
|
||||
"stored_files": {
|
||||
"update": {},
|
||||
"delete": {},
|
||||
},
|
||||
}
|
||||
|
||||
rs.Hooks().Register(resolvespec.BeforeHandle, func(hookCtx *resolvespec.HookContext) error {
|
||||
switch hookCtx.Operation {
|
||||
case "read", "meta":
|
||||
return nil
|
||||
case "create", "update", "delete":
|
||||
allowedOps, ok := mutableByEntity[hookCtx.Entity]
|
||||
if !ok {
|
||||
hookCtx.Abort = true
|
||||
hookCtx.AbortCode = http.StatusForbidden
|
||||
hookCtx.AbortMessage = fmt.Sprintf("operation %q is not allowed for %s.%s", hookCtx.Operation, hookCtx.Schema, hookCtx.Entity)
|
||||
return fmt.Errorf("forbidden operation")
|
||||
}
|
||||
if _, ok := allowedOps[hookCtx.Operation]; !ok {
|
||||
hookCtx.Abort = true
|
||||
hookCtx.AbortCode = http.StatusForbidden
|
||||
hookCtx.AbortMessage = fmt.Sprintf("operation %q is not allowed for %s.%s", hookCtx.Operation, hookCtx.Schema, hookCtx.Entity)
|
||||
return fmt.Errorf("forbidden operation")
|
||||
}
|
||||
return nil
|
||||
default:
|
||||
hookCtx.Abort = true
|
||||
hookCtx.AbortCode = http.StatusBadRequest
|
||||
hookCtx.AbortMessage = fmt.Sprintf("unsupported operation %q", hookCtx.Operation)
|
||||
return fmt.Errorf("unsupported operation")
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
type resolveSpecModel struct {
|
||||
schema string
|
||||
entity string
|
||||
model any
|
||||
}
|
||||
@@ -1,206 +0,0 @@
|
||||
package app
|
||||
|
||||
import (
|
||||
"io"
|
||||
"log/slog"
|
||||
"net/http"
|
||||
"net/http/httptest"
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/auth"
|
||||
"git.warky.dev/wdevs/amcs/internal/config"
|
||||
|
||||
"github.com/bitechdev/ResolveSpec/pkg/resolvespec"
|
||||
)
|
||||
|
||||
func TestResolveSpecAuthRequiresValidCredentials(t *testing.T) {
|
||||
keyring, err := auth.NewKeyring([]config.APIKey{{ID: "operator", Value: "secret"}})
|
||||
if err != nil {
|
||||
t.Fatalf("NewKeyring() error = %v", err)
|
||||
}
|
||||
|
||||
logger := slog.New(slog.NewTextHandler(io.Discard, nil))
|
||||
protected := auth.Middleware(config.AuthConfig{}, keyring, nil, nil, nil, logger)(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||
if r.URL.Path != "/api/rs/public/projects" {
|
||||
t.Fatalf("path = %q, want /api/rs/public/projects", r.URL.Path)
|
||||
}
|
||||
w.WriteHeader(http.StatusNoContent)
|
||||
}))
|
||||
|
||||
t.Run("missing credentials are rejected", func(t *testing.T) {
|
||||
req := httptest.NewRequest(http.MethodPost, "/api/rs/public/projects", strings.NewReader(`{"operation":"read"}`))
|
||||
rec := httptest.NewRecorder()
|
||||
|
||||
protected.ServeHTTP(rec, req)
|
||||
|
||||
if rec.Code != http.StatusUnauthorized {
|
||||
t.Fatalf("status = %d, want %d", rec.Code, http.StatusUnauthorized)
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("valid API key is accepted", func(t *testing.T) {
|
||||
req := httptest.NewRequest(http.MethodPost, "/api/rs/public/projects", strings.NewReader(`{"operation":"read"}`))
|
||||
req.Header.Set("x-brain-key", "secret")
|
||||
rec := httptest.NewRecorder()
|
||||
|
||||
protected.ServeHTTP(rec, req)
|
||||
|
||||
if rec.Code != http.StatusNoContent {
|
||||
t.Fatalf("status = %d, want %d", rec.Code, http.StatusNoContent)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func TestResolveSpecGuardAllowsSupportedMutations(t *testing.T) {
|
||||
rs := resolvespec.NewHandler(nil, nil)
|
||||
registerResolveSpecGuards(rs)
|
||||
|
||||
cases := []struct {
|
||||
name string
|
||||
entity string
|
||||
operation string
|
||||
}{
|
||||
{name: "learnings read", entity: "learnings", operation: "read"},
|
||||
{name: "projects create", entity: "projects", operation: "create"},
|
||||
{name: "projects update", entity: "projects", operation: "update"},
|
||||
{name: "projects delete", entity: "projects", operation: "delete"},
|
||||
{name: "plans create", entity: "plans", operation: "create"},
|
||||
{name: "plans update", entity: "plans", operation: "update"},
|
||||
{name: "plans delete", entity: "plans", operation: "delete"},
|
||||
{name: "learnings create", entity: "learnings", operation: "create"},
|
||||
{name: "learnings update", entity: "learnings", operation: "update"},
|
||||
{name: "learnings delete", entity: "learnings", operation: "delete"},
|
||||
{name: "thoughts create", entity: "thoughts", operation: "create"},
|
||||
{name: "thoughts update", entity: "thoughts", operation: "update"},
|
||||
{name: "thoughts delete", entity: "thoughts", operation: "delete"},
|
||||
{name: "agent_skills create", entity: "agent_skills", operation: "create"},
|
||||
{name: "agent_skills update", entity: "agent_skills", operation: "update"},
|
||||
{name: "agent_skills delete", entity: "agent_skills", operation: "delete"},
|
||||
{name: "agent_guardrails create", entity: "agent_guardrails", operation: "create"},
|
||||
{name: "agent_guardrails update", entity: "agent_guardrails", operation: "update"},
|
||||
{name: "agent_guardrails delete", entity: "agent_guardrails", operation: "delete"},
|
||||
{name: "agent_personas create", entity: "agent_personas", operation: "create"},
|
||||
{name: "agent_personas update", entity: "agent_personas", operation: "update"},
|
||||
{name: "agent_personas delete", entity: "agent_personas", operation: "delete"},
|
||||
{name: "agent_parts create", entity: "agent_parts", operation: "create"},
|
||||
{name: "agent_parts update", entity: "agent_parts", operation: "update"},
|
||||
{name: "agent_parts delete", entity: "agent_parts", operation: "delete"},
|
||||
{name: "agent_traits create", entity: "agent_traits", operation: "create"},
|
||||
{name: "agent_traits update", entity: "agent_traits", operation: "update"},
|
||||
{name: "agent_traits delete", entity: "agent_traits", operation: "delete"},
|
||||
{name: "stored_files update", entity: "stored_files", operation: "update"},
|
||||
{name: "stored_files delete", entity: "stored_files", operation: "delete"},
|
||||
}
|
||||
|
||||
for _, tc := range cases {
|
||||
t.Run(tc.name, func(t *testing.T) {
|
||||
hookCtx := &resolvespec.HookContext{
|
||||
Schema: "public",
|
||||
Entity: tc.entity,
|
||||
Operation: tc.operation,
|
||||
}
|
||||
|
||||
err := rs.Hooks().Execute(resolvespec.BeforeHandle, hookCtx)
|
||||
if err != nil {
|
||||
t.Fatalf("Execute() error = %v, want nil", err)
|
||||
}
|
||||
if hookCtx.Abort {
|
||||
t.Fatalf("Abort = true, want false (code=%d message=%q)", hookCtx.AbortCode, hookCtx.AbortMessage)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestResolveSpecGuardBlocksUnsupportedMutations(t *testing.T) {
|
||||
rs := resolvespec.NewHandler(nil, nil)
|
||||
registerResolveSpecGuards(rs)
|
||||
|
||||
cases := []struct {
|
||||
name string
|
||||
entity string
|
||||
operation string
|
||||
wantCode int
|
||||
wantMessageIn string
|
||||
}{
|
||||
{
|
||||
name: "mutations blocked for non-allowlisted operation",
|
||||
entity: "stored_files",
|
||||
operation: "create",
|
||||
wantCode: http.StatusForbidden,
|
||||
wantMessageIn: `operation "create" is not allowed for public.stored_files`,
|
||||
},
|
||||
{
|
||||
name: "mutations blocked for non-allowlisted entity",
|
||||
entity: "maintenance_logs",
|
||||
operation: "delete",
|
||||
wantCode: http.StatusForbidden,
|
||||
wantMessageIn: `operation "delete" is not allowed for public.maintenance_logs`,
|
||||
},
|
||||
{
|
||||
name: "unknown operation is rejected",
|
||||
entity: "projects",
|
||||
operation: "scan",
|
||||
wantCode: http.StatusBadRequest,
|
||||
wantMessageIn: `unsupported operation "scan"`,
|
||||
},
|
||||
}
|
||||
|
||||
for _, tc := range cases {
|
||||
t.Run(tc.name, func(t *testing.T) {
|
||||
hookCtx := &resolvespec.HookContext{
|
||||
Schema: "public",
|
||||
Entity: tc.entity,
|
||||
Operation: tc.operation,
|
||||
}
|
||||
|
||||
err := rs.Hooks().Execute(resolvespec.BeforeHandle, hookCtx)
|
||||
if err == nil {
|
||||
t.Fatal("Execute() error = nil, want non-nil")
|
||||
}
|
||||
if !hookCtx.Abort {
|
||||
t.Fatal("Abort = false, want true")
|
||||
}
|
||||
if hookCtx.AbortCode != tc.wantCode {
|
||||
t.Fatalf("AbortCode = %d, want %d", hookCtx.AbortCode, tc.wantCode)
|
||||
}
|
||||
if !strings.Contains(hookCtx.AbortMessage, tc.wantMessageIn) {
|
||||
t.Fatalf("AbortMessage = %q, want substring %q", hookCtx.AbortMessage, tc.wantMessageIn)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestResolveSpecModelsIncludeLearnings(t *testing.T) {
|
||||
models := resolveSpecModels()
|
||||
for _, model := range models {
|
||||
if model.schema == "public" && model.entity == "learnings" {
|
||||
return
|
||||
}
|
||||
}
|
||||
t.Fatal("resolveSpecModels() missing public.learnings")
|
||||
}
|
||||
|
||||
func TestResolveSpecModelsIncludePersonaEntities(t *testing.T) {
|
||||
models := resolveSpecModels()
|
||||
required := map[string]bool{
|
||||
"agent_personas": false,
|
||||
"agent_parts": false,
|
||||
"agent_traits": false,
|
||||
}
|
||||
|
||||
for _, model := range models {
|
||||
if model.schema != "public" {
|
||||
continue
|
||||
}
|
||||
if _, ok := required[model.entity]; ok {
|
||||
required[model.entity] = true
|
||||
}
|
||||
}
|
||||
|
||||
for entity, found := range required {
|
||||
if !found {
|
||||
t.Fatalf("resolveSpecModels() missing public.%s", entity)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,38 +0,0 @@
|
||||
// Code generated by relspec templ. DO NOT EDIT.
|
||||
package app
|
||||
|
||||
import "git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
|
||||
func resolveSpecModels() []resolveSpecModel {
|
||||
return []resolveSpecModel{
|
||||
{schema: "public", entity: "agent_guardrails", model: generatedmodels.ModelPublicAgentGuardrails{}},
|
||||
{schema: "public", entity: "agent_parts", model: generatedmodels.ModelPublicAgentParts{}},
|
||||
{schema: "public", entity: "agent_persona_guardrails", model: generatedmodels.ModelPublicAgentPersonaGuardrails{}},
|
||||
{schema: "public", entity: "agent_persona_parts", model: generatedmodels.ModelPublicAgentPersonaParts{}},
|
||||
{schema: "public", entity: "agent_persona_skills", model: generatedmodels.ModelPublicAgentPersonaSkills{}},
|
||||
{schema: "public", entity: "agent_persona_traits", model: generatedmodels.ModelPublicAgentPersonaTraits{}},
|
||||
{schema: "public", entity: "agent_personas", model: generatedmodels.ModelPublicAgentPersonas{}},
|
||||
{schema: "public", entity: "agent_skills", model: generatedmodels.ModelPublicAgentSkills{}},
|
||||
{schema: "public", entity: "agent_traits", model: generatedmodels.ModelPublicAgentTraits{}},
|
||||
{schema: "public", entity: "arc_stage_parts", model: generatedmodels.ModelPublicArcStageParts{}},
|
||||
{schema: "public", entity: "arc_stages", model: generatedmodels.ModelPublicArcStages{}},
|
||||
{schema: "public", entity: "character_arcs", model: generatedmodels.ModelPublicCharacterArcs{}},
|
||||
{schema: "public", entity: "chat_histories", model: generatedmodels.ModelPublicChatHistories{}},
|
||||
{schema: "public", entity: "embeddings", model: generatedmodels.ModelPublicEmbeddings{}},
|
||||
{schema: "public", entity: "learnings", model: generatedmodels.ModelPublicLearnings{}},
|
||||
{schema: "public", entity: "oauth_clients", model: generatedmodels.ModelPublicOauthClients{}},
|
||||
{schema: "public", entity: "persona_arc", model: generatedmodels.ModelPublicPersonaArc{}},
|
||||
{schema: "public", entity: "plan_dependencies", model: generatedmodels.ModelPublicPlanDependencies{}},
|
||||
{schema: "public", entity: "plan_guardrails", model: generatedmodels.ModelPublicPlanGuardrails{}},
|
||||
{schema: "public", entity: "plan_related_plans", model: generatedmodels.ModelPublicPlanRelatedPlans{}},
|
||||
{schema: "public", entity: "plan_skills", model: generatedmodels.ModelPublicPlanSkills{}},
|
||||
{schema: "public", entity: "plans", model: generatedmodels.ModelPublicPlans{}},
|
||||
{schema: "public", entity: "project_guardrails", model: generatedmodels.ModelPublicProjectGuardrails{}},
|
||||
{schema: "public", entity: "project_skills", model: generatedmodels.ModelPublicProjectSkills{}},
|
||||
{schema: "public", entity: "projects", model: generatedmodels.ModelPublicProjects{}},
|
||||
{schema: "public", entity: "stored_files", model: generatedmodels.ModelPublicStoredFiles{}},
|
||||
{schema: "public", entity: "thought_links", model: generatedmodels.ModelPublicThoughtLinks{}},
|
||||
{schema: "public", entity: "thoughts", model: generatedmodels.ModelPublicThoughts{}},
|
||||
{schema: "public", entity: "tool_annotations", model: generatedmodels.ModelPublicToolAnnotations{}},
|
||||
}
|
||||
}
|
||||
+3
-57
@@ -25,25 +25,11 @@ type statusAPIResponse struct {
|
||||
TotalKnown int `json:"total_known"`
|
||||
ConnectedWindow string `json:"connected_window"`
|
||||
Entries []auth.AccessSnapshot `json:"entries"`
|
||||
Metrics auth.AccessMetrics `json:"metrics"`
|
||||
OAuthEnabled bool `json:"oauth_enabled"`
|
||||
}
|
||||
|
||||
type publicClientStatus struct {
|
||||
KeyID string `json:"key_id"`
|
||||
RequestCount int `json:"request_count"`
|
||||
LastAccessedAt time.Time `json:"last_accessed_at"`
|
||||
}
|
||||
|
||||
type publicStatusResponse struct {
|
||||
ConnectedCount int `json:"connected_count"`
|
||||
ConnectedWindow string `json:"connected_window"`
|
||||
Entries []publicClientStatus `json:"entries"`
|
||||
}
|
||||
|
||||
func statusSnapshot(info buildinfo.Info, tracker *auth.AccessTracker, oauthEnabled bool, now time.Time) statusAPIResponse {
|
||||
entries := tracker.Snapshot()
|
||||
metrics := tracker.Metrics(20)
|
||||
return statusAPIResponse{
|
||||
Title: "Avelon Memory Crystal Server (AMCS)",
|
||||
Description: "AMCS is a memory server that captures, links, and retrieves structured project thoughts for AI assistants using semantic search, summaries, and MCP tools.",
|
||||
@@ -53,8 +39,7 @@ func statusSnapshot(info buildinfo.Info, tracker *auth.AccessTracker, oauthEnabl
|
||||
ConnectedCount: tracker.ConnectedCount(now, connectedWindow),
|
||||
TotalKnown: len(entries),
|
||||
ConnectedWindow: "last 10 minutes",
|
||||
Entries: nil,
|
||||
Metrics: metrics,
|
||||
Entries: entries,
|
||||
OAuthEnabled: oauthEnabled,
|
||||
}
|
||||
}
|
||||
@@ -88,47 +73,6 @@ func statusAPIHandler(info buildinfo.Info, tracker *auth.AccessTracker, oauthEna
|
||||
}
|
||||
}
|
||||
|
||||
func publicStatusHandler(tracker *auth.AccessTracker) http.HandlerFunc {
|
||||
return func(w http.ResponseWriter, r *http.Request) {
|
||||
if r.URL.Path != "/status" {
|
||||
http.NotFound(w, r)
|
||||
return
|
||||
}
|
||||
if r.Method != http.MethodGet && r.Method != http.MethodHead {
|
||||
w.Header().Set("Allow", "GET, HEAD")
|
||||
http.Error(w, "method not allowed", http.StatusMethodNotAllowed)
|
||||
return
|
||||
}
|
||||
|
||||
now := time.Now()
|
||||
cutoff := now.UTC().Add(-connectedWindow)
|
||||
snapshot := tracker.Snapshot()
|
||||
entries := make([]publicClientStatus, 0, len(snapshot))
|
||||
for _, item := range snapshot {
|
||||
if item.LastAccessedAt.Before(cutoff) {
|
||||
continue
|
||||
}
|
||||
entries = append(entries, publicClientStatus{
|
||||
KeyID: item.KeyID,
|
||||
RequestCount: item.RequestCount,
|
||||
LastAccessedAt: item.LastAccessedAt,
|
||||
})
|
||||
}
|
||||
|
||||
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
||||
w.WriteHeader(http.StatusOK)
|
||||
if r.Method == http.MethodHead {
|
||||
return
|
||||
}
|
||||
|
||||
_ = json.NewEncoder(w).Encode(publicStatusResponse{
|
||||
ConnectedCount: len(entries),
|
||||
ConnectedWindow: "last 10 minutes",
|
||||
Entries: entries,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func homeHandler(_ buildinfo.Info, _ *auth.AccessTracker, _ bool) http.HandlerFunc {
|
||||
return func(w http.ResponseWriter, r *http.Request) {
|
||||
if r.Method != http.MethodGet && r.Method != http.MethodHead {
|
||||
@@ -146,6 +90,8 @@ func homeHandler(_ buildinfo.Info, _ *auth.AccessTracker, _ bool) http.HandlerFu
|
||||
if serveUIAsset(w, r, requestPath) {
|
||||
return
|
||||
}
|
||||
http.NotFound(w, r)
|
||||
return
|
||||
}
|
||||
|
||||
serveUIIndex(w, r)
|
||||
|
||||
@@ -33,7 +33,7 @@ func TestStatusSnapshotHidesOAuthLinkWhenDisabled(t *testing.T) {
|
||||
func TestStatusSnapshotShowsTrackedAccess(t *testing.T) {
|
||||
tracker := auth.NewAccessTracker()
|
||||
now := time.Date(2026, 4, 4, 12, 0, 0, 0, time.UTC)
|
||||
tracker.Record("client-a", "/files", "127.0.0.1:1234", "tester", "list_projects", now)
|
||||
tracker.Record("client-a", "/files", "127.0.0.1:1234", "tester", now)
|
||||
|
||||
snapshot := statusSnapshot(buildinfo.Info{Version: "v1.2.3"}, tracker, true, now)
|
||||
|
||||
@@ -43,29 +43,11 @@ func TestStatusSnapshotShowsTrackedAccess(t *testing.T) {
|
||||
if snapshot.ConnectedCount != 1 {
|
||||
t.Fatalf("ConnectedCount = %d, want 1", snapshot.ConnectedCount)
|
||||
}
|
||||
if len(snapshot.Entries) != 0 {
|
||||
t.Fatalf("len(Entries) = %d, want 0 for counts-only status", len(snapshot.Entries))
|
||||
if len(snapshot.Entries) != 1 {
|
||||
t.Fatalf("len(Entries) = %d, want 1", len(snapshot.Entries))
|
||||
}
|
||||
if snapshot.Metrics.TotalRequests != 1 {
|
||||
t.Fatalf("Metrics.TotalRequests = %d, want 1", snapshot.Metrics.TotalRequests)
|
||||
}
|
||||
if snapshot.Metrics.UniqueIPs != 1 {
|
||||
t.Fatalf("Metrics.UniqueIPs = %d, want 1", snapshot.Metrics.UniqueIPs)
|
||||
}
|
||||
if snapshot.Metrics.UniqueAgents != 1 {
|
||||
t.Fatalf("Metrics.UniqueAgents = %d, want 1", snapshot.Metrics.UniqueAgents)
|
||||
}
|
||||
if snapshot.Metrics.UniqueTools != 1 {
|
||||
t.Fatalf("Metrics.UniqueTools = %d, want 1", snapshot.Metrics.UniqueTools)
|
||||
}
|
||||
if len(snapshot.Metrics.TopIPs) != 1 || len(snapshot.Metrics.TopAgents) != 1 || len(snapshot.Metrics.TopTools) != 1 {
|
||||
t.Fatalf("Top breakdowns not populated: %+v", snapshot.Metrics)
|
||||
}
|
||||
if len(snapshot.Metrics.RecentLog) != 1 {
|
||||
t.Fatalf("RecentLog len = %d, want 1", len(snapshot.Metrics.RecentLog))
|
||||
}
|
||||
if snapshot.Metrics.RecentLog[0].Tool != "list_projects" {
|
||||
t.Fatalf("RecentLog[0].Tool = %q, want %q", snapshot.Metrics.RecentLog[0].Tool, "list_projects")
|
||||
if snapshot.Entries[0].KeyID != "client-a" || snapshot.Entries[0].LastPath != "/files" {
|
||||
t.Fatalf("entry = %+v, want keyID client-a and path /files", snapshot.Entries[0])
|
||||
}
|
||||
}
|
||||
|
||||
@@ -92,52 +74,6 @@ func TestStatusAPIHandlerReturnsJSON(t *testing.T) {
|
||||
}
|
||||
}
|
||||
|
||||
func TestStatusAPIHandlerRejectsStatusPath(t *testing.T) {
|
||||
handler := statusAPIHandler(buildinfo.Info{Version: "v1"}, auth.NewAccessTracker(), true)
|
||||
req := httptest.NewRequest(http.MethodGet, "/status", nil)
|
||||
rec := httptest.NewRecorder()
|
||||
|
||||
handler.ServeHTTP(rec, req)
|
||||
|
||||
if rec.Code != http.StatusNotFound {
|
||||
t.Fatalf("status = %d, want %d", rec.Code, http.StatusNotFound)
|
||||
}
|
||||
}
|
||||
|
||||
func TestPublicStatusHandlerReturnsConnectedClientsOnly(t *testing.T) {
|
||||
tracker := auth.NewAccessTracker()
|
||||
now := time.Now().UTC()
|
||||
tracker.Record("recent-client", "/mcp", "127.0.0.1:1234", "tester", "list_projects", now.Add(-2*time.Minute))
|
||||
tracker.Record("stale-client", "/mcp", "127.0.0.1:9999", "tester", "list_projects", now.Add(-30*time.Minute))
|
||||
|
||||
handler := publicStatusHandler(tracker)
|
||||
req := httptest.NewRequest(http.MethodGet, "/status", nil)
|
||||
rec := httptest.NewRecorder()
|
||||
|
||||
handler.ServeHTTP(rec, req)
|
||||
|
||||
if rec.Code != http.StatusOK {
|
||||
t.Fatalf("status = %d, want %d", rec.Code, http.StatusOK)
|
||||
}
|
||||
|
||||
var payload publicStatusResponse
|
||||
if err := json.Unmarshal(rec.Body.Bytes(), &payload); err != nil {
|
||||
t.Fatalf("json.Unmarshal() error = %v", err)
|
||||
}
|
||||
if payload.ConnectedCount != 1 {
|
||||
t.Fatalf("ConnectedCount = %d, want 1", payload.ConnectedCount)
|
||||
}
|
||||
if len(payload.Entries) != 1 {
|
||||
t.Fatalf("len(Entries) = %d, want 1", len(payload.Entries))
|
||||
}
|
||||
if payload.Entries[0].KeyID != "recent-client" {
|
||||
t.Fatalf("Entries[0].KeyID = %q, want %q", payload.Entries[0].KeyID, "recent-client")
|
||||
}
|
||||
if payload.Entries[0].LastAccessedAt.Before(now.Add(-11 * time.Minute)) {
|
||||
t.Fatalf("Entries[0].LastAccessedAt = %v, expected recent timestamp", payload.Entries[0].LastAccessedAt)
|
||||
}
|
||||
}
|
||||
|
||||
func TestHomeHandlerAllowsHead(t *testing.T) {
|
||||
handler := homeHandler(buildinfo.Info{Version: "v1"}, auth.NewAccessTracker(), false)
|
||||
req := httptest.NewRequest(http.MethodHead, "/", nil)
|
||||
|
||||
Vendored
-2
@@ -1,2 +0,0 @@
|
||||
This placeholder keeps internal/app/ui/dist present in clean source checkouts.
|
||||
The real UI bundle is generated by the frontend build into this directory.
|
||||
@@ -1,9 +1,7 @@
|
||||
package auth
|
||||
|
||||
import (
|
||||
"net"
|
||||
"sort"
|
||||
"strings"
|
||||
"sync"
|
||||
"time"
|
||||
)
|
||||
@@ -17,38 +15,16 @@ type AccessSnapshot struct {
|
||||
LastAccessedAt time.Time `json:"last_accessed_at"`
|
||||
}
|
||||
|
||||
const maxRecentLog = 100
|
||||
|
||||
type AccessLogEntry struct {
|
||||
Timestamp time.Time `json:"timestamp"`
|
||||
KeyID string `json:"key_id"`
|
||||
IP string `json:"ip"`
|
||||
UserAgent string `json:"user_agent"`
|
||||
Tool string `json:"tool"`
|
||||
Path string `json:"path"`
|
||||
}
|
||||
|
||||
type AccessTracker struct {
|
||||
mu sync.RWMutex
|
||||
entries map[string]AccessSnapshot
|
||||
ipCounts map[string]int
|
||||
agentCounts map[string]int
|
||||
toolCounts map[string]int
|
||||
recentLog []AccessLogEntry
|
||||
totalRequests int
|
||||
mu sync.RWMutex
|
||||
entries map[string]AccessSnapshot
|
||||
}
|
||||
|
||||
func NewAccessTracker() *AccessTracker {
|
||||
return &AccessTracker{
|
||||
entries: make(map[string]AccessSnapshot),
|
||||
ipCounts: make(map[string]int),
|
||||
agentCounts: make(map[string]int),
|
||||
toolCounts: make(map[string]int),
|
||||
recentLog: make([]AccessLogEntry, 0, maxRecentLog),
|
||||
}
|
||||
return &AccessTracker{entries: make(map[string]AccessSnapshot)}
|
||||
}
|
||||
|
||||
func (t *AccessTracker) Record(keyID, path, remoteAddr, userAgent, toolName string, now time.Time) {
|
||||
func (t *AccessTracker) Record(keyID, path, remoteAddr, userAgent string, now time.Time) {
|
||||
if t == nil || keyID == "" {
|
||||
return
|
||||
}
|
||||
@@ -56,52 +32,14 @@ func (t *AccessTracker) Record(keyID, path, remoteAddr, userAgent, toolName stri
|
||||
t.mu.Lock()
|
||||
defer t.mu.Unlock()
|
||||
|
||||
normalizedRemoteAddr := normalizeRemoteAddr(remoteAddr)
|
||||
|
||||
entry := t.entries[keyID]
|
||||
entry.KeyID = keyID
|
||||
entry.LastPath = path
|
||||
entry.RemoteAddr = normalizedRemoteAddr
|
||||
entry.RemoteAddr = remoteAddr
|
||||
entry.UserAgent = userAgent
|
||||
entry.LastAccessedAt = now.UTC()
|
||||
entry.RequestCount++
|
||||
t.entries[keyID] = entry
|
||||
t.totalRequests++
|
||||
|
||||
if normalizedRemoteAddr != "" {
|
||||
t.ipCounts[normalizedRemoteAddr]++
|
||||
}
|
||||
if userAgent != "" {
|
||||
t.agentCounts[userAgent]++
|
||||
}
|
||||
if tool := strings.TrimSpace(toolName); tool != "" {
|
||||
t.toolCounts[tool]++
|
||||
}
|
||||
|
||||
logEntry := AccessLogEntry{
|
||||
Timestamp: now.UTC(),
|
||||
KeyID: keyID,
|
||||
IP: normalizedRemoteAddr,
|
||||
UserAgent: userAgent,
|
||||
Tool: strings.TrimSpace(toolName),
|
||||
Path: path,
|
||||
}
|
||||
t.recentLog = append([]AccessLogEntry{logEntry}, t.recentLog...)
|
||||
if len(t.recentLog) > maxRecentLog {
|
||||
t.recentLog = t.recentLog[:maxRecentLog]
|
||||
}
|
||||
}
|
||||
|
||||
func normalizeRemoteAddr(value string) string {
|
||||
trimmed := strings.TrimSpace(value)
|
||||
if trimmed == "" {
|
||||
return ""
|
||||
}
|
||||
host, _, err := net.SplitHostPort(trimmed)
|
||||
if err == nil {
|
||||
return host
|
||||
}
|
||||
return trimmed
|
||||
}
|
||||
|
||||
func (t *AccessTracker) Snapshot() []AccessSnapshot {
|
||||
@@ -141,64 +79,3 @@ func (t *AccessTracker) ConnectedCount(now time.Time, window time.Duration) int
|
||||
}
|
||||
return count
|
||||
}
|
||||
|
||||
type RequestAggregate struct {
|
||||
Key string `json:"key"`
|
||||
RequestCount int `json:"request_count"`
|
||||
}
|
||||
|
||||
type AccessMetrics struct {
|
||||
TotalRequests int `json:"total_requests"`
|
||||
UniquePrincipals int `json:"unique_principals"`
|
||||
UniqueIPs int `json:"unique_ips"`
|
||||
UniqueAgents int `json:"unique_agents"`
|
||||
UniqueTools int `json:"unique_tools"`
|
||||
TopIPs []RequestAggregate `json:"top_ips"`
|
||||
TopAgents []RequestAggregate `json:"top_agents"`
|
||||
TopTools []RequestAggregate `json:"top_tools"`
|
||||
RecentLog []AccessLogEntry `json:"recent_log"`
|
||||
}
|
||||
|
||||
func (t *AccessTracker) Metrics(topN int) AccessMetrics {
|
||||
if t == nil {
|
||||
return AccessMetrics{}
|
||||
}
|
||||
if topN <= 0 {
|
||||
topN = 10
|
||||
}
|
||||
|
||||
t.mu.RLock()
|
||||
defer t.mu.RUnlock()
|
||||
|
||||
log := make([]AccessLogEntry, len(t.recentLog))
|
||||
copy(log, t.recentLog)
|
||||
|
||||
return AccessMetrics{
|
||||
TotalRequests: t.totalRequests,
|
||||
UniquePrincipals: len(t.entries),
|
||||
UniqueIPs: len(t.ipCounts),
|
||||
UniqueAgents: len(t.agentCounts),
|
||||
UniqueTools: len(t.toolCounts),
|
||||
TopIPs: topAggregates(t.ipCounts, topN),
|
||||
TopAgents: topAggregates(t.agentCounts, topN),
|
||||
TopTools: topAggregates(t.toolCounts, topN),
|
||||
RecentLog: log,
|
||||
}
|
||||
}
|
||||
|
||||
func topAggregates(items map[string]int, topN int) []RequestAggregate {
|
||||
out := make([]RequestAggregate, 0, len(items))
|
||||
for key, count := range items {
|
||||
out = append(out, RequestAggregate{Key: key, RequestCount: count})
|
||||
}
|
||||
sort.Slice(out, func(i, j int) bool {
|
||||
if out[i].RequestCount == out[j].RequestCount {
|
||||
return out[i].Key < out[j].Key
|
||||
}
|
||||
return out[i].RequestCount > out[j].RequestCount
|
||||
})
|
||||
if len(out) > topN {
|
||||
out = out[:topN]
|
||||
}
|
||||
return out
|
||||
}
|
||||
|
||||
@@ -10,9 +10,9 @@ func TestAccessTrackerRecordAndSnapshot(t *testing.T) {
|
||||
older := time.Date(2026, 4, 4, 10, 0, 0, 0, time.UTC)
|
||||
newer := older.Add(2 * time.Minute)
|
||||
|
||||
tracker.Record("client-a", "/files", "10.0.0.1:1234", "agent-a", "", older)
|
||||
tracker.Record("client-b", "/mcp", "10.0.0.2:1234", "agent-b", "list_projects", newer)
|
||||
tracker.Record("client-a", "/files/1", "10.0.0.1:1234", "agent-a2", "", newer.Add(30*time.Second))
|
||||
tracker.Record("client-a", "/files", "10.0.0.1:1234", "agent-a", older)
|
||||
tracker.Record("client-b", "/mcp", "10.0.0.2:1234", "agent-b", newer)
|
||||
tracker.Record("client-a", "/files/1", "10.0.0.1:1234", "agent-a2", newer.Add(30*time.Second))
|
||||
|
||||
snap := tracker.Snapshot()
|
||||
if len(snap) != 2 {
|
||||
@@ -30,67 +30,16 @@ func TestAccessTrackerRecordAndSnapshot(t *testing.T) {
|
||||
if snap[0].UserAgent != "agent-a2" {
|
||||
t.Fatalf("snapshot[0].UserAgent = %q, want agent-a2", snap[0].UserAgent)
|
||||
}
|
||||
if snap[0].RemoteAddr != "10.0.0.1" {
|
||||
t.Fatalf("snapshot[0].RemoteAddr = %q, want 10.0.0.1", snap[0].RemoteAddr)
|
||||
}
|
||||
}
|
||||
|
||||
func TestAccessTrackerConnectedCount(t *testing.T) {
|
||||
tracker := NewAccessTracker()
|
||||
now := time.Date(2026, 4, 4, 12, 0, 0, 0, time.UTC)
|
||||
|
||||
tracker.Record("recent", "/mcp", "", "", "", now.Add(-2*time.Minute))
|
||||
tracker.Record("stale", "/mcp", "", "", "", now.Add(-11*time.Minute))
|
||||
tracker.Record("recent", "/mcp", "", "", now.Add(-2*time.Minute))
|
||||
tracker.Record("stale", "/mcp", "", "", now.Add(-11*time.Minute))
|
||||
|
||||
if got := tracker.ConnectedCount(now, 10*time.Minute); got != 1 {
|
||||
t.Fatalf("ConnectedCount() = %d, want 1", got)
|
||||
}
|
||||
}
|
||||
|
||||
func TestAccessTrackerMetrics(t *testing.T) {
|
||||
tracker := NewAccessTracker()
|
||||
now := time.Date(2026, 4, 4, 12, 0, 0, 0, time.UTC)
|
||||
|
||||
tracker.Record("client-a", "/mcp", "10.0.0.1:1234", "agent-a", "list_projects", now)
|
||||
tracker.Record("client-a", "/mcp", "10.0.0.1:1234", "agent-a", "list_projects", now.Add(1*time.Second))
|
||||
tracker.Record("client-b", "/files", "10.0.0.2:5678", "agent-b", "", now.Add(2*time.Second))
|
||||
tracker.Record("client-c", "/files", "10.0.0.2:5678", "agent-b", "search_thoughts", now.Add(3*time.Second))
|
||||
|
||||
metrics := tracker.Metrics(5)
|
||||
if metrics.TotalRequests != 4 {
|
||||
t.Fatalf("TotalRequests = %d, want 4", metrics.TotalRequests)
|
||||
}
|
||||
if metrics.UniquePrincipals != 3 {
|
||||
t.Fatalf("UniquePrincipals = %d, want 3", metrics.UniquePrincipals)
|
||||
}
|
||||
if metrics.UniqueIPs != 2 {
|
||||
t.Fatalf("UniqueIPs = %d, want 2", metrics.UniqueIPs)
|
||||
}
|
||||
if metrics.UniqueAgents != 2 {
|
||||
t.Fatalf("UniqueAgents = %d, want 2", metrics.UniqueAgents)
|
||||
}
|
||||
if metrics.UniqueTools != 2 {
|
||||
t.Fatalf("UniqueTools = %d, want 2", metrics.UniqueTools)
|
||||
}
|
||||
if len(metrics.TopIPs) != 2 {
|
||||
t.Fatalf("len(TopIPs) = %d, want 2", len(metrics.TopIPs))
|
||||
}
|
||||
if metrics.TopIPs[0].RequestCount != 2 || metrics.TopIPs[1].RequestCount != 2 {
|
||||
t.Fatalf("TopIPs counts = %+v, want both counts to be 2", metrics.TopIPs)
|
||||
}
|
||||
if metrics.TopIPs[0].Key != "10.0.0.1" && metrics.TopIPs[0].Key != "10.0.0.2" {
|
||||
t.Fatalf("TopIPs[0].Key = %q, want normalized IP", metrics.TopIPs[0].Key)
|
||||
}
|
||||
if len(metrics.TopAgents) != 2 {
|
||||
t.Fatalf("len(TopAgents) = %d, want 2", len(metrics.TopAgents))
|
||||
}
|
||||
if metrics.TopAgents[0].RequestCount != 2 || metrics.TopAgents[1].RequestCount != 2 {
|
||||
t.Fatalf("TopAgents counts = %+v, want both counts to be 2", metrics.TopAgents)
|
||||
}
|
||||
if len(metrics.TopTools) != 2 {
|
||||
t.Fatalf("len(TopTools) = %d, want 2", len(metrics.TopTools))
|
||||
}
|
||||
if metrics.TopTools[0].Key != "list_projects" || metrics.TopTools[0].RequestCount != 2 {
|
||||
t.Fatalf("TopTools[0] = %+v, want list_projects with count 2", metrics.TopTools[0])
|
||||
}
|
||||
}
|
||||
|
||||
@@ -26,12 +26,6 @@ func (c *DynamicClient) HasRedirectURI(uri string) bool {
|
||||
return false
|
||||
}
|
||||
|
||||
// ClientStore is the interface implemented by both DynamicClientStore and PostgresClientStore.
|
||||
type ClientStore interface {
|
||||
Register(name string, redirectURIs []string) (DynamicClient, error)
|
||||
Lookup(clientID string) (DynamicClient, bool)
|
||||
}
|
||||
|
||||
// DynamicClientStore holds dynamically registered OAuth clients in memory.
|
||||
type DynamicClientStore struct {
|
||||
mu sync.RWMutex
|
||||
|
||||
@@ -1,8 +1,6 @@
|
||||
package auth
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/json"
|
||||
"io"
|
||||
"log/slog"
|
||||
"net/http"
|
||||
@@ -10,7 +8,6 @@ import (
|
||||
"testing"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/config"
|
||||
"git.warky.dev/wdevs/amcs/internal/observability"
|
||||
)
|
||||
|
||||
func testLogger() *slog.Logger {
|
||||
@@ -191,50 +188,3 @@ func TestMiddlewareRecordsForwardedRemoteAddr(t *testing.T) {
|
||||
t.Fatalf("snapshot remote_addr = %q, want %q", snap[0].RemoteAddr, "203.0.113.99")
|
||||
}
|
||||
}
|
||||
|
||||
func TestMiddlewareRecordsMCPToolUsage(t *testing.T) {
|
||||
keyring, err := NewKeyring([]config.APIKey{{ID: "client-a", Value: "secret"}})
|
||||
if err != nil {
|
||||
t.Fatalf("NewKeyring() error = %v", err)
|
||||
}
|
||||
tracker := NewAccessTracker()
|
||||
logger := testLogger()
|
||||
|
||||
authenticated := Middleware(config.AuthConfig{HeaderName: "x-brain-key"}, keyring, nil, nil, tracker, logger)(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||
w.WriteHeader(http.StatusNoContent)
|
||||
}))
|
||||
handler := observability.AccessLog(logger)(authenticated)
|
||||
|
||||
payload := map[string]any{
|
||||
"jsonrpc": "2.0",
|
||||
"id": "1",
|
||||
"method": "tools/call",
|
||||
"params": map[string]any{
|
||||
"name": "list_projects",
|
||||
},
|
||||
}
|
||||
body, err := json.Marshal(payload)
|
||||
if err != nil {
|
||||
t.Fatalf("json.Marshal() error = %v", err)
|
||||
}
|
||||
|
||||
req := httptest.NewRequest(http.MethodPost, "/mcp", bytes.NewReader(body))
|
||||
req.Header.Set("x-brain-key", "secret")
|
||||
rec := httptest.NewRecorder()
|
||||
handler.ServeHTTP(rec, req)
|
||||
|
||||
if rec.Code != http.StatusNoContent {
|
||||
t.Fatalf("status = %d, want %d", rec.Code, http.StatusNoContent)
|
||||
}
|
||||
|
||||
metrics := tracker.Metrics(10)
|
||||
if metrics.UniqueTools != 1 {
|
||||
t.Fatalf("UniqueTools = %d, want 1", metrics.UniqueTools)
|
||||
}
|
||||
if len(metrics.TopTools) != 1 {
|
||||
t.Fatalf("len(TopTools) = %d, want 1", len(metrics.TopTools))
|
||||
}
|
||||
if metrics.TopTools[0].Key != "list_projects" || metrics.TopTools[0].RequestCount != 1 {
|
||||
t.Fatalf("TopTools[0] = %+v, want list_projects with count 1", metrics.TopTools[0])
|
||||
}
|
||||
}
|
||||
|
||||
@@ -9,7 +9,6 @@ import (
|
||||
"time"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/config"
|
||||
"git.warky.dev/wdevs/amcs/internal/observability"
|
||||
"git.warky.dev/wdevs/amcs/internal/requestip"
|
||||
)
|
||||
|
||||
@@ -17,22 +16,6 @@ type contextKey string
|
||||
|
||||
const keyIDContextKey contextKey = "auth.key_id"
|
||||
|
||||
// wwwAuthenticate returns the value for a WWW-Authenticate header.
|
||||
// It advertises Bearer and, when a public URL is known, the OAuth metadata URL per RFC 9728.
|
||||
func wwwAuthenticate(r *http.Request, publicURL string) string {
|
||||
base := publicURL
|
||||
if base == "" {
|
||||
scheme := "https"
|
||||
if proto := r.Header.Get("X-Forwarded-Proto"); proto != "" {
|
||||
scheme = strings.ToLower(proto)
|
||||
} else if r.TLS == nil {
|
||||
scheme = "http"
|
||||
}
|
||||
base = scheme + "://" + r.Host
|
||||
}
|
||||
return `Bearer resource_metadata="` + base + `/.well-known/oauth-authorization-server"`
|
||||
}
|
||||
|
||||
func Middleware(cfg config.AuthConfig, keyring *Keyring, oauthRegistry *OAuthRegistry, tokenStore *TokenStore, tracker *AccessTracker, log *slog.Logger) func(http.Handler) http.Handler {
|
||||
headerName := cfg.HeaderName
|
||||
if headerName == "" {
|
||||
@@ -40,14 +23,7 @@ func Middleware(cfg config.AuthConfig, keyring *Keyring, oauthRegistry *OAuthReg
|
||||
}
|
||||
recordAccess := func(r *http.Request, keyID string) {
|
||||
if tracker != nil {
|
||||
tracker.Record(
|
||||
keyID,
|
||||
r.URL.Path,
|
||||
requestip.FromRequest(r),
|
||||
r.UserAgent(),
|
||||
observability.MCPToolFromContext(r.Context()),
|
||||
time.Now(),
|
||||
)
|
||||
tracker.Record(keyID, r.URL.Path, requestip.FromRequest(r), r.UserAgent(), time.Now())
|
||||
}
|
||||
}
|
||||
return func(next http.Handler) http.Handler {
|
||||
@@ -85,7 +61,6 @@ func Middleware(cfg config.AuthConfig, keyring *Keyring, oauthRegistry *OAuthReg
|
||||
}
|
||||
}
|
||||
log.Warn("bearer token rejected", slog.String("remote_addr", remoteAddr))
|
||||
w.Header().Set("WWW-Authenticate", wwwAuthenticate(r, "")+`, error="invalid_token"`)
|
||||
http.Error(w, "invalid token or API key", http.StatusUnauthorized)
|
||||
return
|
||||
}
|
||||
@@ -122,7 +97,6 @@ func Middleware(cfg config.AuthConfig, keyring *Keyring, oauthRegistry *OAuthReg
|
||||
}
|
||||
}
|
||||
|
||||
w.Header().Set("WWW-Authenticate", wwwAuthenticate(r, ""))
|
||||
http.Error(w, "authentication required", http.StatusUnauthorized)
|
||||
})
|
||||
}
|
||||
|
||||
@@ -1,50 +0,0 @@
|
||||
package auth
|
||||
|
||||
import (
|
||||
"context"
|
||||
"crypto/rand"
|
||||
"fmt"
|
||||
|
||||
"github.com/jackc/pgx/v5/pgxpool"
|
||||
)
|
||||
|
||||
// PostgresClientStore persists dynamically registered OAuth clients (RFC 7591) in PostgreSQL.
|
||||
type PostgresClientStore struct {
|
||||
pool *pgxpool.Pool
|
||||
}
|
||||
|
||||
func NewPostgresClientStore(pool *pgxpool.Pool) *PostgresClientStore {
|
||||
return &PostgresClientStore{pool: pool}
|
||||
}
|
||||
|
||||
func (s *PostgresClientStore) Register(name string, redirectURIs []string) (DynamicClient, error) {
|
||||
b := make([]byte, 16)
|
||||
if _, err := rand.Read(b); err != nil {
|
||||
return DynamicClient{}, err
|
||||
}
|
||||
clientID := fmt.Sprintf("%x-%x-%x-%x-%x", b[0:4], b[4:6], b[6:8], b[8:10], b[10:])
|
||||
|
||||
var client DynamicClient
|
||||
row := s.pool.QueryRow(context.Background(), `
|
||||
insert into oauth_clients (client_id, client_name, redirect_uris)
|
||||
values ($1, $2, $3)
|
||||
returning client_id, client_name, redirect_uris, created_at
|
||||
`, clientID, name, redirectURIs)
|
||||
if err := row.Scan(&client.ClientID, &client.ClientName, &client.RedirectURIs, &client.CreatedAt); err != nil {
|
||||
return DynamicClient{}, fmt.Errorf("register oauth client: %w", err)
|
||||
}
|
||||
return client, nil
|
||||
}
|
||||
|
||||
func (s *PostgresClientStore) Lookup(clientID string) (DynamicClient, bool) {
|
||||
var client DynamicClient
|
||||
row := s.pool.QueryRow(context.Background(), `
|
||||
select client_id, client_name, redirect_uris, created_at
|
||||
from oauth_clients
|
||||
where client_id = $1
|
||||
`, clientID)
|
||||
if err := row.Scan(&client.ClientID, &client.ClientName, &client.RedirectURIs, &client.CreatedAt); err != nil {
|
||||
return DynamicClient{}, false
|
||||
}
|
||||
return client, true
|
||||
}
|
||||
@@ -1,69 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicAgentGuardrails struct {
|
||||
bun.BaseModel `bun:"table:public.agent_guardrails,alias:agent_guardrails"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
Content resolvespec_common.SqlString `bun:"content,type:text,notnull," json:"content"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Description resolvespec_common.SqlString `bun:"description,type:text,default:'',notnull," json:"description"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
Name resolvespec_common.SqlString `bun:"name,type:text,notnull," json:"name"`
|
||||
Severity resolvespec_common.SqlString `bun:"severity,type:text,default:'medium',notnull," json:"severity"`
|
||||
Tags resolvespec_common.SqlStringArray `bun:"tags,type:text,default:'{}',notnull," json:"tags"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelGuardrailIDPublicAgentPersonaGuardrails []*ModelPublicAgentPersonaGuardrails `bun:"rel:has-many,join:id=guardrail_id" json:"relguardrailidpublicagentpersonaguardrails,omitempty"` // Has many ModelPublicAgentPersonaGuardrails
|
||||
RelGuardrailIDPublicPlanGuardrails []*ModelPublicPlanGuardrails `bun:"rel:has-many,join:id=guardrail_id" json:"relguardrailidpublicplanguardrails,omitempty"` // Has many ModelPublicPlanGuardrails
|
||||
RelGuardrailIDPublicProjectGuardrails []*ModelPublicProjectGuardrails `bun:"rel:has-many,join:id=guardrail_id" json:"relguardrailidpublicprojectguardrails,omitempty"` // Has many ModelPublicProjectGuardrails
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicAgentGuardrails
|
||||
func (m ModelPublicAgentGuardrails) TableName() string {
|
||||
return "public.agent_guardrails"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicAgentGuardrails
|
||||
func (m ModelPublicAgentGuardrails) TableNameOnly() string {
|
||||
return "agent_guardrails"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicAgentGuardrails
|
||||
func (m ModelPublicAgentGuardrails) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicAgentGuardrails) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicAgentGuardrails) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicAgentGuardrails) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicAgentGuardrails) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicAgentGuardrails) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicAgentGuardrails) GetPrefix() string {
|
||||
return "AGG"
|
||||
}
|
||||
@@ -1,69 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicAgentParts struct {
|
||||
bun.BaseModel `bun:"table:public.agent_parts,alias:agent_parts"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
Content resolvespec_common.SqlString `bun:"content,type:text,default:'',notnull," json:"content"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Description resolvespec_common.SqlString `bun:"description,type:text,default:'',notnull," json:"description"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
Name resolvespec_common.SqlString `bun:"name,type:text,notnull," json:"name"`
|
||||
PartType resolvespec_common.SqlString `bun:"part_type,type:text,notnull," json:"part_type"`
|
||||
Summary resolvespec_common.SqlString `bun:"summary,type:text,notnull," json:"summary"`
|
||||
Tags resolvespec_common.SqlStringArray `bun:"tags,type:text,default:'{}',notnull," json:"tags"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelPartIDPublicAgentPersonaParts []*ModelPublicAgentPersonaParts `bun:"rel:has-many,join:id=part_id" json:"relpartidpublicagentpersonaparts,omitempty"` // Has many ModelPublicAgentPersonaParts
|
||||
RelPartIDPublicArcStageParts []*ModelPublicArcStageParts `bun:"rel:has-many,join:id=part_id" json:"relpartidpublicarcstageparts,omitempty"` // Has many ModelPublicArcStageParts
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicAgentParts
|
||||
func (m ModelPublicAgentParts) TableName() string {
|
||||
return "public.agent_parts"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicAgentParts
|
||||
func (m ModelPublicAgentParts) TableNameOnly() string {
|
||||
return "agent_parts"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicAgentParts
|
||||
func (m ModelPublicAgentParts) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicAgentParts) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicAgentParts) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicAgentParts) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicAgentParts) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicAgentParts) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicAgentParts) GetPrefix() string {
|
||||
return "APG"
|
||||
}
|
||||
@@ -1,62 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicAgentPersonaGuardrails struct {
|
||||
bun.BaseModel `bun:"table:public.agent_persona_guardrails,alias:agent_persona_guardrails"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
GuardrailID int64 `bun:"guardrail_id,type:bigint,notnull," json:"guardrail_id"`
|
||||
PersonaID int64 `bun:"persona_id,type:bigint,notnull," json:"persona_id"`
|
||||
RelGuardrailID *ModelPublicAgentGuardrails `bun:"rel:has-one,join:guardrail_id=id" json:"relguardrailid,omitempty"` // Has one ModelPublicAgentGuardrails
|
||||
RelPersonaID *ModelPublicAgentPersonas `bun:"rel:has-one,join:persona_id=id" json:"relpersonaid,omitempty"` // Has one ModelPublicAgentPersonas
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicAgentPersonaGuardrails
|
||||
func (m ModelPublicAgentPersonaGuardrails) TableName() string {
|
||||
return "public.agent_persona_guardrails"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicAgentPersonaGuardrails
|
||||
func (m ModelPublicAgentPersonaGuardrails) TableNameOnly() string {
|
||||
return "agent_persona_guardrails"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicAgentPersonaGuardrails
|
||||
func (m ModelPublicAgentPersonaGuardrails) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicAgentPersonaGuardrails) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicAgentPersonaGuardrails) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicAgentPersonaGuardrails) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicAgentPersonaGuardrails) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicAgentPersonaGuardrails) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicAgentPersonaGuardrails) GetPrefix() string {
|
||||
return "APG"
|
||||
}
|
||||
@@ -1,64 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicAgentPersonaParts struct {
|
||||
bun.BaseModel `bun:"table:public.agent_persona_parts,alias:agent_persona_parts"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
PartID int64 `bun:"part_id,type:bigint,notnull," json:"part_id"`
|
||||
PartOrder int32 `bun:"part_order,type:int,default:0,notnull," json:"part_order"`
|
||||
PersonaID int64 `bun:"persona_id,type:bigint,notnull," json:"persona_id"`
|
||||
Priority int32 `bun:"priority,type:int,default:0,notnull," json:"priority"`
|
||||
RelPartID *ModelPublicAgentParts `bun:"rel:has-one,join:part_id=id" json:"relpartid,omitempty"` // Has one ModelPublicAgentParts
|
||||
RelPersonaID *ModelPublicAgentPersonas `bun:"rel:has-one,join:persona_id=id" json:"relpersonaid,omitempty"` // Has one ModelPublicAgentPersonas
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicAgentPersonaParts
|
||||
func (m ModelPublicAgentPersonaParts) TableName() string {
|
||||
return "public.agent_persona_parts"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicAgentPersonaParts
|
||||
func (m ModelPublicAgentPersonaParts) TableNameOnly() string {
|
||||
return "agent_persona_parts"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicAgentPersonaParts
|
||||
func (m ModelPublicAgentPersonaParts) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicAgentPersonaParts) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicAgentPersonaParts) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicAgentPersonaParts) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicAgentPersonaParts) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicAgentPersonaParts) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicAgentPersonaParts) GetPrefix() string {
|
||||
return "APP"
|
||||
}
|
||||
@@ -1,62 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicAgentPersonaSkills struct {
|
||||
bun.BaseModel `bun:"table:public.agent_persona_skills,alias:agent_persona_skills"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
PersonaID int64 `bun:"persona_id,type:bigint,notnull," json:"persona_id"`
|
||||
SkillID int64 `bun:"skill_id,type:bigint,notnull," json:"skill_id"`
|
||||
RelPersonaID *ModelPublicAgentPersonas `bun:"rel:has-one,join:persona_id=id" json:"relpersonaid,omitempty"` // Has one ModelPublicAgentPersonas
|
||||
RelSkillID *ModelPublicAgentSkills `bun:"rel:has-one,join:skill_id=id" json:"relskillid,omitempty"` // Has one ModelPublicAgentSkills
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicAgentPersonaSkills
|
||||
func (m ModelPublicAgentPersonaSkills) TableName() string {
|
||||
return "public.agent_persona_skills"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicAgentPersonaSkills
|
||||
func (m ModelPublicAgentPersonaSkills) TableNameOnly() string {
|
||||
return "agent_persona_skills"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicAgentPersonaSkills
|
||||
func (m ModelPublicAgentPersonaSkills) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicAgentPersonaSkills) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicAgentPersonaSkills) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicAgentPersonaSkills) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicAgentPersonaSkills) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicAgentPersonaSkills) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicAgentPersonaSkills) GetPrefix() string {
|
||||
return "APS"
|
||||
}
|
||||
@@ -1,62 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicAgentPersonaTraits struct {
|
||||
bun.BaseModel `bun:"table:public.agent_persona_traits,alias:agent_persona_traits"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
PersonaID int64 `bun:"persona_id,type:bigint,notnull," json:"persona_id"`
|
||||
TraitID int64 `bun:"trait_id,type:bigint,notnull," json:"trait_id"`
|
||||
RelPersonaID *ModelPublicAgentPersonas `bun:"rel:has-one,join:persona_id=id" json:"relpersonaid,omitempty"` // Has one ModelPublicAgentPersonas
|
||||
RelTraitID *ModelPublicAgentTraits `bun:"rel:has-one,join:trait_id=id" json:"reltraitid,omitempty"` // Has one ModelPublicAgentTraits
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicAgentPersonaTraits
|
||||
func (m ModelPublicAgentPersonaTraits) TableName() string {
|
||||
return "public.agent_persona_traits"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicAgentPersonaTraits
|
||||
func (m ModelPublicAgentPersonaTraits) TableNameOnly() string {
|
||||
return "agent_persona_traits"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicAgentPersonaTraits
|
||||
func (m ModelPublicAgentPersonaTraits) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicAgentPersonaTraits) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicAgentPersonaTraits) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicAgentPersonaTraits) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicAgentPersonaTraits) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicAgentPersonaTraits) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicAgentPersonaTraits) GetPrefix() string {
|
||||
return "APT"
|
||||
}
|
||||
@@ -1,74 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicAgentPersonas struct {
|
||||
bun.BaseModel `bun:"table:public.agent_personas,alias:agent_personas"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CompiledAt resolvespec_common.SqlTimeStamp `bun:"compiled_at,type:timestamptz,nullzero," json:"compiled_at"`
|
||||
CompiledDetail resolvespec_common.SqlString `bun:"compiled_detail,type:text,default:'',notnull," json:"compiled_detail"`
|
||||
CompiledSummary resolvespec_common.SqlString `bun:"compiled_summary,type:text,default:'',notnull," json:"compiled_summary"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Description resolvespec_common.SqlString `bun:"description,type:text,default:'',notnull," json:"description"`
|
||||
Detail resolvespec_common.SqlString `bun:"detail,type:text,default:'',notnull," json:"detail"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
Name resolvespec_common.SqlString `bun:"name,type:text,notnull," json:"name"`
|
||||
Summary resolvespec_common.SqlString `bun:"summary,type:text,notnull," json:"summary"`
|
||||
Tags resolvespec_common.SqlStringArray `bun:"tags,type:text,default:'{}',notnull," json:"tags"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelPersonaIDPublicAgentPersonaParts []*ModelPublicAgentPersonaParts `bun:"rel:has-many,join:id=persona_id" json:"relpersonaidpublicagentpersonaparts,omitempty"` // Has many ModelPublicAgentPersonaParts
|
||||
RelPersonaIDPublicAgentPersonaSkills []*ModelPublicAgentPersonaSkills `bun:"rel:has-many,join:id=persona_id" json:"relpersonaidpublicagentpersonaskills,omitempty"` // Has many ModelPublicAgentPersonaSkills
|
||||
RelPersonaIDPublicAgentPersonaGuardrails []*ModelPublicAgentPersonaGuardrails `bun:"rel:has-many,join:id=persona_id" json:"relpersonaidpublicagentpersonaguardrails,omitempty"` // Has many ModelPublicAgentPersonaGuardrails
|
||||
RelPersonaIDPublicAgentPersonaTraits []*ModelPublicAgentPersonaTraits `bun:"rel:has-many,join:id=persona_id" json:"relpersonaidpublicagentpersonatraits,omitempty"` // Has many ModelPublicAgentPersonaTraits
|
||||
RelPersonaIDPublicPersonaArcs []*ModelPublicPersonaArc `bun:"rel:has-many,join:id=persona_id" json:"relpersonaidpublicpersonaarcs,omitempty"` // Has many ModelPublicPersonaArc
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicAgentPersonas
|
||||
func (m ModelPublicAgentPersonas) TableName() string {
|
||||
return "public.agent_personas"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicAgentPersonas
|
||||
func (m ModelPublicAgentPersonas) TableNameOnly() string {
|
||||
return "agent_personas"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicAgentPersonas
|
||||
func (m ModelPublicAgentPersonas) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicAgentPersonas) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicAgentPersonas) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicAgentPersonas) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicAgentPersonas) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicAgentPersonas) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicAgentPersonas) GetPrefix() string {
|
||||
return "APG"
|
||||
}
|
||||
@@ -1,73 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicAgentSkills struct {
|
||||
bun.BaseModel `bun:"table:public.agent_skills,alias:agent_skills"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
Content resolvespec_common.SqlString `bun:"content,type:text,notnull," json:"content"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Description resolvespec_common.SqlString `bun:"description,type:text,default:'',notnull," json:"description"`
|
||||
DomainTags resolvespec_common.SqlStringArray `bun:"domain_tags,type:text,default:'{}',notnull," json:"domain_tags"`
|
||||
FrameworkTags resolvespec_common.SqlStringArray `bun:"framework_tags,type:text,default:'{}',notnull," json:"framework_tags"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
LanguageTags resolvespec_common.SqlStringArray `bun:"language_tags,type:text,default:'{}',notnull," json:"language_tags"`
|
||||
LibraryTags resolvespec_common.SqlStringArray `bun:"library_tags,type:text,default:'{}',notnull," json:"library_tags"`
|
||||
Name resolvespec_common.SqlString `bun:"name,type:text,notnull," json:"name"`
|
||||
Tags resolvespec_common.SqlStringArray `bun:"tags,type:text,default:'{}',notnull," json:"tags"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelSkillIDPublicAgentPersonaSkills []*ModelPublicAgentPersonaSkills `bun:"rel:has-many,join:id=skill_id" json:"relskillidpublicagentpersonaskills,omitempty"` // Has many ModelPublicAgentPersonaSkills
|
||||
RelRelatedSkillIDPublicLearnings []*ModelPublicLearnings `bun:"rel:has-many,join:id=related_skill_id" json:"relrelatedskillidpubliclearnings,omitempty"` // Has many ModelPublicLearnings
|
||||
RelSkillIDPublicPlanSkills []*ModelPublicPlanSkills `bun:"rel:has-many,join:id=skill_id" json:"relskillidpublicplanskills,omitempty"` // Has many ModelPublicPlanSkills
|
||||
RelSkillIDPublicProjectSkills []*ModelPublicProjectSkills `bun:"rel:has-many,join:id=skill_id" json:"relskillidpublicprojectskills,omitempty"` // Has many ModelPublicProjectSkills
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicAgentSkills
|
||||
func (m ModelPublicAgentSkills) TableName() string {
|
||||
return "public.agent_skills"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicAgentSkills
|
||||
func (m ModelPublicAgentSkills) TableNameOnly() string {
|
||||
return "agent_skills"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicAgentSkills
|
||||
func (m ModelPublicAgentSkills) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicAgentSkills) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicAgentSkills) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicAgentSkills) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicAgentSkills) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicAgentSkills) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicAgentSkills) GetPrefix() string {
|
||||
return "ASG"
|
||||
}
|
||||
@@ -1,67 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicAgentTraits struct {
|
||||
bun.BaseModel `bun:"table:public.agent_traits,alias:agent_traits"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Description resolvespec_common.SqlString `bun:"description,type:text,default:'',notnull," json:"description"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
Instruction resolvespec_common.SqlString `bun:"instruction,type:text,default:'',notnull," json:"instruction"`
|
||||
Name resolvespec_common.SqlString `bun:"name,type:text,notnull," json:"name"`
|
||||
Tags resolvespec_common.SqlStringArray `bun:"tags,type:text,default:'{}',notnull," json:"tags"`
|
||||
TraitType resolvespec_common.SqlString `bun:"trait_type,type:text,notnull," json:"trait_type"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelTraitIDPublicAgentPersonaTraits []*ModelPublicAgentPersonaTraits `bun:"rel:has-many,join:id=trait_id" json:"reltraitidpublicagentpersonatraits,omitempty"` // Has many ModelPublicAgentPersonaTraits
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicAgentTraits
|
||||
func (m ModelPublicAgentTraits) TableName() string {
|
||||
return "public.agent_traits"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicAgentTraits
|
||||
func (m ModelPublicAgentTraits) TableNameOnly() string {
|
||||
return "agent_traits"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicAgentTraits
|
||||
func (m ModelPublicAgentTraits) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicAgentTraits) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicAgentTraits) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicAgentTraits) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicAgentTraits) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicAgentTraits) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicAgentTraits) GetPrefix() string {
|
||||
return "ATG"
|
||||
}
|
||||
@@ -1,62 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicArcStageParts struct {
|
||||
bun.BaseModel `bun:"table:public.arc_stage_parts,alias:arc_stage_parts"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
PartID int64 `bun:"part_id,type:bigint,notnull," json:"part_id"`
|
||||
StageID int64 `bun:"stage_id,type:bigint,notnull," json:"stage_id"`
|
||||
RelPartID *ModelPublicAgentParts `bun:"rel:has-one,join:part_id=id" json:"relpartid,omitempty"` // Has one ModelPublicAgentParts
|
||||
RelStageID *ModelPublicArcStages `bun:"rel:has-one,join:stage_id=id" json:"relstageid,omitempty"` // Has one ModelPublicArcStages
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicArcStageParts
|
||||
func (m ModelPublicArcStageParts) TableName() string {
|
||||
return "public.arc_stage_parts"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicArcStageParts
|
||||
func (m ModelPublicArcStageParts) TableNameOnly() string {
|
||||
return "arc_stage_parts"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicArcStageParts
|
||||
func (m ModelPublicArcStageParts) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicArcStageParts) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicArcStageParts) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicArcStageParts) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicArcStageParts) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicArcStageParts) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicArcStageParts) GetPrefix() string {
|
||||
return "ASP"
|
||||
}
|
||||
@@ -1,67 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicArcStages struct {
|
||||
bun.BaseModel `bun:"table:public.arc_stages,alias:arc_stages"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
ArcID int64 `bun:"arc_id,type:bigint,notnull," json:"arc_id"`
|
||||
Condition resolvespec_common.SqlString `bun:"condition,type:text,default:'',notnull," json:"condition"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Description resolvespec_common.SqlString `bun:"description,type:text,default:'',notnull," json:"description"`
|
||||
Name resolvespec_common.SqlString `bun:"name,type:text,notnull," json:"name"`
|
||||
StageOrder int32 `bun:"stage_order,type:int,default:0,notnull," json:"stage_order"`
|
||||
RelArcID *ModelPublicCharacterArcs `bun:"rel:has-one,join:arc_id=id" json:"relarcid,omitempty"` // Has one ModelPublicCharacterArcs
|
||||
RelStageIDPublicArcStageParts []*ModelPublicArcStageParts `bun:"rel:has-many,join:id=stage_id" json:"relstageidpublicarcstageparts,omitempty"` // Has many ModelPublicArcStageParts
|
||||
RelCurrentStageIDPublicPersonaArcs []*ModelPublicPersonaArc `bun:"rel:has-many,join:id=current_stage_id" json:"relcurrentstageidpublicpersonaarcs,omitempty"` // Has many ModelPublicPersonaArc
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicArcStages
|
||||
func (m ModelPublicArcStages) TableName() string {
|
||||
return "public.arc_stages"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicArcStages
|
||||
func (m ModelPublicArcStages) TableNameOnly() string {
|
||||
return "arc_stages"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicArcStages
|
||||
func (m ModelPublicArcStages) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicArcStages) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicArcStages) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicArcStages) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicArcStages) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicArcStages) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicArcStages) GetPrefix() string {
|
||||
return "ASR"
|
||||
}
|
||||
@@ -1,65 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicCharacterArcs struct {
|
||||
bun.BaseModel `bun:"table:public.character_arcs,alias:character_arcs"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Description resolvespec_common.SqlString `bun:"description,type:text,default:'',notnull," json:"description"`
|
||||
Name resolvespec_common.SqlString `bun:"name,type:text,notnull," json:"name"`
|
||||
Summary resolvespec_common.SqlString `bun:"summary,type:text,default:'',notnull," json:"summary"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelArcIDPublicArcStages []*ModelPublicArcStages `bun:"rel:has-many,join:id=arc_id" json:"relarcidpublicarcstages,omitempty"` // Has many ModelPublicArcStages
|
||||
RelArcIDPublicPersonaArcs []*ModelPublicPersonaArc `bun:"rel:has-many,join:id=arc_id" json:"relarcidpublicpersonaarcs,omitempty"` // Has many ModelPublicPersonaArc
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicCharacterArcs
|
||||
func (m ModelPublicCharacterArcs) TableName() string {
|
||||
return "public.character_arcs"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicCharacterArcs
|
||||
func (m ModelPublicCharacterArcs) TableNameOnly() string {
|
||||
return "character_arcs"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicCharacterArcs
|
||||
func (m ModelPublicCharacterArcs) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicCharacterArcs) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicCharacterArcs) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicCharacterArcs) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicCharacterArcs) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicCharacterArcs) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicCharacterArcs) GetPrefix() string {
|
||||
return "CAH"
|
||||
}
|
||||
@@ -1,70 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicChatHistories struct {
|
||||
bun.BaseModel `bun:"table:public.chat_histories,alias:chat_histories"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
AgentID resolvespec_common.SqlString `bun:"agent_id,type:text,nullzero," json:"agent_id"`
|
||||
Channel resolvespec_common.SqlString `bun:"channel,type:text,nullzero," json:"channel"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
Messages resolvespec_common.SqlJSONB `bun:"messages,type:jsonb,default:'',notnull," json:"messages"`
|
||||
Metadata resolvespec_common.SqlJSONB `bun:"metadata,type:jsonb,default:'{}',notnull," json:"metadata"`
|
||||
ProjectID resolvespec_common.SqlInt64 `bun:"project_id,type:bigint,nullzero," json:"project_id"`
|
||||
SessionID resolvespec_common.SqlString `bun:"session_id,type:text,notnull," json:"session_id"`
|
||||
Summary resolvespec_common.SqlString `bun:"summary,type:text,nullzero," json:"summary"`
|
||||
Title resolvespec_common.SqlString `bun:"title,type:text,nullzero," json:"title"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelProjectID *ModelPublicProjects `bun:"rel:has-one,join:project_id=id" json:"relprojectid,omitempty"` // Has one ModelPublicProjects
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicChatHistories
|
||||
func (m ModelPublicChatHistories) TableName() string {
|
||||
return "public.chat_histories"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicChatHistories
|
||||
func (m ModelPublicChatHistories) TableNameOnly() string {
|
||||
return "chat_histories"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicChatHistories
|
||||
func (m ModelPublicChatHistories) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicChatHistories) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicChatHistories) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicChatHistories) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicChatHistories) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicChatHistories) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicChatHistories) GetPrefix() string {
|
||||
return "CHH"
|
||||
}
|
||||
@@ -1,66 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicEmbeddings struct {
|
||||
bun.BaseModel `bun:"table:public.embeddings,alias:embeddings"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),nullzero," json:"created_at"`
|
||||
Dim int32 `bun:"dim,type:int,notnull," json:"dim"`
|
||||
Embedding resolvespec_common.SqlVector `bun:"embedding,type:vector,notnull," json:"embedding"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
Model resolvespec_common.SqlString `bun:"model,type:text,notnull,unique:uidx_embeddings_thought_id_model," json:"model"`
|
||||
ThoughtID int64 `bun:"thought_id,type:bigint,notnull,unique:uidx_embeddings_thought_id_model," json:"thought_id"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),nullzero," json:"updated_at"`
|
||||
RelThoughtID *ModelPublicThoughts `bun:"rel:has-one,join:thought_id=id" json:"relthoughtid,omitempty"` // Has one ModelPublicThoughts
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicEmbeddings
|
||||
func (m ModelPublicEmbeddings) TableName() string {
|
||||
return "public.embeddings"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicEmbeddings
|
||||
func (m ModelPublicEmbeddings) TableNameOnly() string {
|
||||
return "embeddings"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicEmbeddings
|
||||
func (m ModelPublicEmbeddings) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicEmbeddings) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicEmbeddings) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicEmbeddings) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicEmbeddings) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicEmbeddings) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicEmbeddings) GetPrefix() string {
|
||||
return "EMB"
|
||||
}
|
||||
@@ -1,84 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicLearnings struct {
|
||||
bun.BaseModel `bun:"table:public.learnings,alias:learnings"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
ActionRequired bool `bun:"action_required,type:boolean,default:false,notnull," json:"action_required"`
|
||||
Area resolvespec_common.SqlString `bun:"area,type:text,default:'other',notnull," json:"area"`
|
||||
Category resolvespec_common.SqlString `bun:"category,type:text,default:'insight',notnull," json:"category"`
|
||||
Confidence resolvespec_common.SqlString `bun:"confidence,type:text,default:'hypothesis',notnull," json:"confidence"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Details resolvespec_common.SqlString `bun:"details,type:text,default:'',notnull," json:"details"`
|
||||
DuplicateOfLearningID resolvespec_common.SqlInt64 `bun:"duplicate_of_learning_id,type:bigint,nullzero," json:"duplicate_of_learning_id"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
Priority resolvespec_common.SqlString `bun:"priority,type:text,default:'medium',notnull," json:"priority"`
|
||||
ProjectID resolvespec_common.SqlInt64 `bun:"project_id,type:bigint,nullzero," json:"project_id"`
|
||||
RelatedSkillID resolvespec_common.SqlInt64 `bun:"related_skill_id,type:bigint,nullzero," json:"related_skill_id"`
|
||||
RelatedThoughtID resolvespec_common.SqlInt64 `bun:"related_thought_id,type:bigint,nullzero," json:"related_thought_id"`
|
||||
ReviewedAt resolvespec_common.SqlTimeStamp `bun:"reviewed_at,type:timestamptz,nullzero," json:"reviewed_at"`
|
||||
ReviewedBy resolvespec_common.SqlString `bun:"reviewed_by,type:text,nullzero," json:"reviewed_by"`
|
||||
SourceRef resolvespec_common.SqlString `bun:"source_ref,type:text,nullzero," json:"source_ref"`
|
||||
SourceType resolvespec_common.SqlString `bun:"source_type,type:text,nullzero," json:"source_type"`
|
||||
Status resolvespec_common.SqlString `bun:"status,type:text,default:'pending',notnull," json:"status"`
|
||||
Summary resolvespec_common.SqlString `bun:"summary,type:text,notnull," json:"summary"`
|
||||
SupersedesLearningID resolvespec_common.SqlInt64 `bun:"supersedes_learning_id,type:bigint,nullzero," json:"supersedes_learning_id"`
|
||||
Tags resolvespec_common.SqlStringArray `bun:"tags,type:text,default:'{}',notnull," json:"tags"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelDuplicateOfLearningID *ModelPublicLearnings `bun:"rel:has-one,join:duplicate_of_learning_id=id" json:"relduplicateoflearningid,omitempty"` // Has one ModelPublicLearnings
|
||||
RelProjectID *ModelPublicProjects `bun:"rel:has-one,join:project_id=id" json:"relprojectid,omitempty"` // Has one ModelPublicProjects
|
||||
RelRelatedSkillID *ModelPublicAgentSkills `bun:"rel:has-one,join:related_skill_id=id" json:"relrelatedskillid,omitempty"` // Has one ModelPublicAgentSkills
|
||||
RelRelatedThoughtID *ModelPublicThoughts `bun:"rel:has-one,join:related_thought_id=id" json:"relrelatedthoughtid,omitempty"` // Has one ModelPublicThoughts
|
||||
RelSupersedesLearningID *ModelPublicLearnings `bun:"rel:has-one,join:supersedes_learning_id=id" json:"relsupersedeslearningid,omitempty"` // Has one ModelPublicLearnings
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicLearnings
|
||||
func (m ModelPublicLearnings) TableName() string {
|
||||
return "public.learnings"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicLearnings
|
||||
func (m ModelPublicLearnings) TableNameOnly() string {
|
||||
return "learnings"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicLearnings
|
||||
func (m ModelPublicLearnings) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicLearnings) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicLearnings) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicLearnings) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicLearnings) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicLearnings) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicLearnings) GetPrefix() string {
|
||||
return "LEA"
|
||||
}
|
||||
@@ -1,62 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicOauthClients struct {
|
||||
bun.BaseModel `bun:"table:public.oauth_clients,alias:oauth_clients"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
ClientID resolvespec_common.SqlString `bun:"client_id,type:text,notnull," json:"client_id"`
|
||||
ClientName resolvespec_common.SqlString `bun:"client_name,type:text,default:'',notnull," json:"client_name"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
RedirectUris resolvespec_common.SqlStringArray `bun:"redirect_uris,type:text,default:'{}',notnull," json:"redirect_uris"`
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicOauthClients
|
||||
func (m ModelPublicOauthClients) TableName() string {
|
||||
return "public.oauth_clients"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicOauthClients
|
||||
func (m ModelPublicOauthClients) TableNameOnly() string {
|
||||
return "oauth_clients"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicOauthClients
|
||||
func (m ModelPublicOauthClients) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicOauthClients) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicOauthClients) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicOauthClients) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicOauthClients) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicOauthClients) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicOauthClients) GetPrefix() string {
|
||||
return "OCA"
|
||||
}
|
||||
@@ -1,65 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicPersonaArc struct {
|
||||
bun.BaseModel `bun:"table:public.persona_arc,alias:persona_arc"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
PersonaID int64 `bun:"persona_id,type:bigint,pk," json:"persona_id"`
|
||||
ArcID int64 `bun:"arc_id,type:bigint,notnull," json:"arc_id"`
|
||||
CurrentStageID int64 `bun:"current_stage_id,type:bigint,notnull," json:"current_stage_id"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelArcID *ModelPublicCharacterArcs `bun:"rel:has-one,join:arc_id=id" json:"relarcid,omitempty"` // Has one ModelPublicCharacterArcs
|
||||
RelCurrentStageID *ModelPublicArcStages `bun:"rel:has-one,join:current_stage_id=id" json:"relcurrentstageid,omitempty"` // Has one ModelPublicArcStages
|
||||
RelPersonaID *ModelPublicAgentPersonas `bun:"rel:has-one,join:persona_id=id" json:"relpersonaid,omitempty"` // Has one ModelPublicAgentPersonas
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicPersonaArc
|
||||
func (m ModelPublicPersonaArc) TableName() string {
|
||||
return "public.persona_arc"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicPersonaArc
|
||||
func (m ModelPublicPersonaArc) TableNameOnly() string {
|
||||
return "persona_arc"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicPersonaArc
|
||||
func (m ModelPublicPersonaArc) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicPersonaArc) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicPersonaArc) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicPersonaArc) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicPersonaArc) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicPersonaArc) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicPersonaArc) GetPrefix() string {
|
||||
return "PAE"
|
||||
}
|
||||
@@ -1,63 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicPlanDependencies struct {
|
||||
bun.BaseModel `bun:"table:public.plan_dependencies,alias:plan_dependencies"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
DependsOnPlanID int64 `bun:"depends_on_plan_id,type:bigint,notnull,unique:uidx_plan_dependencies_plan_id_depends_on_plan_id," json:"depends_on_plan_id"`
|
||||
PlanID int64 `bun:"plan_id,type:bigint,notnull,unique:uidx_plan_dependencies_plan_id_depends_on_plan_id," json:"plan_id"`
|
||||
RelDependsOnPlanID *ModelPublicPlans `bun:"rel:has-one,join:depends_on_plan_id=id" json:"reldependsonplanid,omitempty"` // Has one ModelPublicPlans
|
||||
RelPlanID *ModelPublicPlans `bun:"rel:has-one,join:plan_id=id" json:"relplanid,omitempty"` // Has one ModelPublicPlans
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicPlanDependencies
|
||||
func (m ModelPublicPlanDependencies) TableName() string {
|
||||
return "public.plan_dependencies"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicPlanDependencies
|
||||
func (m ModelPublicPlanDependencies) TableNameOnly() string {
|
||||
return "plan_dependencies"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicPlanDependencies
|
||||
func (m ModelPublicPlanDependencies) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicPlanDependencies) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicPlanDependencies) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicPlanDependencies) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicPlanDependencies) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicPlanDependencies) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicPlanDependencies) GetPrefix() string {
|
||||
return "PDL"
|
||||
}
|
||||
@@ -1,63 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicPlanGuardrails struct {
|
||||
bun.BaseModel `bun:"table:public.plan_guardrails,alias:plan_guardrails"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
GuardrailID int64 `bun:"guardrail_id,type:bigint,notnull,unique:uidx_plan_guardrails_plan_id_guardrail_id," json:"guardrail_id"`
|
||||
PlanID int64 `bun:"plan_id,type:bigint,notnull,unique:uidx_plan_guardrails_plan_id_guardrail_id," json:"plan_id"`
|
||||
RelGuardrailID *ModelPublicAgentGuardrails `bun:"rel:has-one,join:guardrail_id=id" json:"relguardrailid,omitempty"` // Has one ModelPublicAgentGuardrails
|
||||
RelPlanID *ModelPublicPlans `bun:"rel:has-one,join:plan_id=id" json:"relplanid,omitempty"` // Has one ModelPublicPlans
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicPlanGuardrails
|
||||
func (m ModelPublicPlanGuardrails) TableName() string {
|
||||
return "public.plan_guardrails"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicPlanGuardrails
|
||||
func (m ModelPublicPlanGuardrails) TableNameOnly() string {
|
||||
return "plan_guardrails"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicPlanGuardrails
|
||||
func (m ModelPublicPlanGuardrails) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicPlanGuardrails) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicPlanGuardrails) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicPlanGuardrails) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicPlanGuardrails) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicPlanGuardrails) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicPlanGuardrails) GetPrefix() string {
|
||||
return "PGL"
|
||||
}
|
||||
@@ -1,63 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicPlanRelatedPlans struct {
|
||||
bun.BaseModel `bun:"table:public.plan_related_plans,alias:plan_related_plans"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
PlanAID int64 `bun:"plan_a_id,type:bigint,notnull,unique:uidx_plan_related_plans_plan_a_id_plan_b_id," json:"plan_a_id"`
|
||||
PlanBID int64 `bun:"plan_b_id,type:bigint,notnull,unique:uidx_plan_related_plans_plan_a_id_plan_b_id," json:"plan_b_id"`
|
||||
RelPlanAID *ModelPublicPlans `bun:"rel:has-one,join:plan_a_id=id" json:"relplanaid,omitempty"` // Has one ModelPublicPlans
|
||||
RelPlanBID *ModelPublicPlans `bun:"rel:has-one,join:plan_b_id=id" json:"relplanbid,omitempty"` // Has one ModelPublicPlans
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicPlanRelatedPlans
|
||||
func (m ModelPublicPlanRelatedPlans) TableName() string {
|
||||
return "public.plan_related_plans"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicPlanRelatedPlans
|
||||
func (m ModelPublicPlanRelatedPlans) TableNameOnly() string {
|
||||
return "plan_related_plans"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicPlanRelatedPlans
|
||||
func (m ModelPublicPlanRelatedPlans) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicPlanRelatedPlans) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicPlanRelatedPlans) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicPlanRelatedPlans) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicPlanRelatedPlans) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicPlanRelatedPlans) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicPlanRelatedPlans) GetPrefix() string {
|
||||
return "PRP"
|
||||
}
|
||||
@@ -1,63 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicPlanSkills struct {
|
||||
bun.BaseModel `bun:"table:public.plan_skills,alias:plan_skills"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
PlanID int64 `bun:"plan_id,type:bigint,notnull,unique:uidx_plan_skills_plan_id_skill_id," json:"plan_id"`
|
||||
SkillID int64 `bun:"skill_id,type:bigint,notnull,unique:uidx_plan_skills_plan_id_skill_id," json:"skill_id"`
|
||||
RelPlanID *ModelPublicPlans `bun:"rel:has-one,join:plan_id=id" json:"relplanid,omitempty"` // Has one ModelPublicPlans
|
||||
RelSkillID *ModelPublicAgentSkills `bun:"rel:has-one,join:skill_id=id" json:"relskillid,omitempty"` // Has one ModelPublicAgentSkills
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicPlanSkills
|
||||
func (m ModelPublicPlanSkills) TableName() string {
|
||||
return "public.plan_skills"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicPlanSkills
|
||||
func (m ModelPublicPlanSkills) TableNameOnly() string {
|
||||
return "plan_skills"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicPlanSkills
|
||||
func (m ModelPublicPlanSkills) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicPlanSkills) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicPlanSkills) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicPlanSkills) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicPlanSkills) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicPlanSkills) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicPlanSkills) GetPrefix() string {
|
||||
return "PSL"
|
||||
}
|
||||
@@ -1,81 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicPlans struct {
|
||||
bun.BaseModel `bun:"table:public.plans,alias:plans"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CompletedAt resolvespec_common.SqlTimeStamp `bun:"completed_at,type:timestamptz,nullzero," json:"completed_at"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Description resolvespec_common.SqlString `bun:"description,type:text,default:'',notnull," json:"description"`
|
||||
DueDate resolvespec_common.SqlTimeStamp `bun:"due_date,type:timestamptz,nullzero," json:"due_date"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
LastReviewedAt resolvespec_common.SqlTimeStamp `bun:"last_reviewed_at,type:timestamptz,nullzero," json:"last_reviewed_at"`
|
||||
Owner resolvespec_common.SqlString `bun:"owner,type:text,nullzero," json:"owner"`
|
||||
Priority resolvespec_common.SqlString `bun:"priority,type:text,default:'medium',notnull," json:"priority"` // low, medium, high, critical
|
||||
ProjectID resolvespec_common.SqlInt64 `bun:"project_id,type:bigint,nullzero," json:"project_id"`
|
||||
ReviewedBy resolvespec_common.SqlString `bun:"reviewed_by,type:text,nullzero," json:"reviewed_by"`
|
||||
Status resolvespec_common.SqlString `bun:"status,type:text,default:'draft',notnull," json:"status"` // draft, active, blocked, completed, cancelled, superseded
|
||||
SupersedesPlanID resolvespec_common.SqlInt64 `bun:"supersedes_plan_id,type:bigint,nullzero," json:"supersedes_plan_id"`
|
||||
Tags resolvespec_common.SqlStringArray `bun:"tags,type:text,default:'{}',notnull," json:"tags"`
|
||||
Title resolvespec_common.SqlString `bun:"title,type:text,notnull," json:"title"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelProjectID *ModelPublicProjects `bun:"rel:has-one,join:project_id=id" json:"relprojectid,omitempty"` // Has one ModelPublicProjects
|
||||
RelSupersedesPlanID *ModelPublicPlans `bun:"rel:has-one,join:supersedes_plan_id=id" json:"relsupersedesplanid,omitempty"` // Has one ModelPublicPlans
|
||||
RelDependsOnPlanIDPublicPlanDependencies []*ModelPublicPlanDependencies `bun:"rel:has-many,join:id=depends_on_plan_id" json:"reldependsonplanidpublicplandependencies,omitempty"` // Has many ModelPublicPlanDependencies
|
||||
RelPlanIDPublicPlanDependencies []*ModelPublicPlanDependencies `bun:"rel:has-many,join:id=plan_id" json:"relplanidpublicplandependencies,omitempty"` // Has many ModelPublicPlanDependencies
|
||||
RelPlanAIDPublicPlanRelatedPlans []*ModelPublicPlanRelatedPlans `bun:"rel:has-many,join:id=plan_a_id" json:"relplanaidpublicplanrelatedplans,omitempty"` // Has many ModelPublicPlanRelatedPlans
|
||||
RelPlanBIDPublicPlanRelatedPlans []*ModelPublicPlanRelatedPlans `bun:"rel:has-many,join:id=plan_b_id" json:"relplanbidpublicplanrelatedplans,omitempty"` // Has many ModelPublicPlanRelatedPlans
|
||||
RelPlanIDPublicPlanSkills []*ModelPublicPlanSkills `bun:"rel:has-many,join:id=plan_id" json:"relplanidpublicplanskills,omitempty"` // Has many ModelPublicPlanSkills
|
||||
RelPlanIDPublicPlanGuardrails []*ModelPublicPlanGuardrails `bun:"rel:has-many,join:id=plan_id" json:"relplanidpublicplanguardrails,omitempty"` // Has many ModelPublicPlanGuardrails
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicPlans
|
||||
func (m ModelPublicPlans) TableName() string {
|
||||
return "public.plans"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicPlans
|
||||
func (m ModelPublicPlans) TableNameOnly() string {
|
||||
return "plans"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicPlans
|
||||
func (m ModelPublicPlans) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicPlans) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicPlans) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicPlans) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicPlans) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicPlans) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicPlans) GetPrefix() string {
|
||||
return "PLA"
|
||||
}
|
||||
@@ -1,63 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicProjectGuardrails struct {
|
||||
bun.BaseModel `bun:"table:public.project_guardrails,alias:project_guardrails"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
GuardrailID int64 `bun:"guardrail_id,type:bigint,notnull," json:"guardrail_id"`
|
||||
ProjectID int64 `bun:"project_id,type:bigint,notnull," json:"project_id"`
|
||||
RelGuardrailID *ModelPublicAgentGuardrails `bun:"rel:has-one,join:guardrail_id=id" json:"relguardrailid,omitempty"` // Has one ModelPublicAgentGuardrails
|
||||
RelProjectID *ModelPublicProjects `bun:"rel:has-one,join:project_id=id" json:"relprojectid,omitempty"` // Has one ModelPublicProjects
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicProjectGuardrails
|
||||
func (m ModelPublicProjectGuardrails) TableName() string {
|
||||
return "public.project_guardrails"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicProjectGuardrails
|
||||
func (m ModelPublicProjectGuardrails) TableNameOnly() string {
|
||||
return "project_guardrails"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicProjectGuardrails
|
||||
func (m ModelPublicProjectGuardrails) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicProjectGuardrails) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicProjectGuardrails) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicProjectGuardrails) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicProjectGuardrails) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicProjectGuardrails) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicProjectGuardrails) GetPrefix() string {
|
||||
return "PGR"
|
||||
}
|
||||
@@ -1,63 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicProjectSkills struct {
|
||||
bun.BaseModel `bun:"table:public.project_skills,alias:project_skills"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
ProjectID int64 `bun:"project_id,type:bigint,notnull," json:"project_id"`
|
||||
SkillID int64 `bun:"skill_id,type:bigint,notnull," json:"skill_id"`
|
||||
RelProjectID *ModelPublicProjects `bun:"rel:has-one,join:project_id=id" json:"relprojectid,omitempty"` // Has one ModelPublicProjects
|
||||
RelSkillID *ModelPublicAgentSkills `bun:"rel:has-one,join:skill_id=id" json:"relskillid,omitempty"` // Has one ModelPublicAgentSkills
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicProjectSkills
|
||||
func (m ModelPublicProjectSkills) TableName() string {
|
||||
return "public.project_skills"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicProjectSkills
|
||||
func (m ModelPublicProjectSkills) TableNameOnly() string {
|
||||
return "project_skills"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicProjectSkills
|
||||
func (m ModelPublicProjectSkills) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicProjectSkills) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicProjectSkills) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicProjectSkills) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicProjectSkills) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicProjectSkills) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicProjectSkills) GetPrefix() string {
|
||||
return "PSR"
|
||||
}
|
||||
@@ -1,71 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicProjects struct {
|
||||
bun.BaseModel `bun:"table:public.projects,alias:projects"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),nullzero," json:"created_at"`
|
||||
Description resolvespec_common.SqlString `bun:"description,type:text,nullzero," json:"description"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
LastActiveAt resolvespec_common.SqlTimeStamp `bun:"last_active_at,type:timestamptz,default:now(),nullzero," json:"last_active_at"`
|
||||
Name resolvespec_common.SqlString `bun:"name,type:text,notnull," json:"name"`
|
||||
ThoughtCount resolvespec_common.SqlInt64 `bun:"thought_count,scanonly" json:"thought_count"`
|
||||
RelProjectIDPublicThoughts []*ModelPublicThoughts `bun:"rel:has-many,join:id=project_id" json:"relprojectidpublicthoughts,omitempty"` // Has many ModelPublicThoughts
|
||||
RelProjectIDPublicStoredFiles []*ModelPublicStoredFiles `bun:"rel:has-many,join:id=project_id" json:"relprojectidpublicstoredfiles,omitempty"` // Has many ModelPublicStoredFiles
|
||||
RelProjectIDPublicChatHistories []*ModelPublicChatHistories `bun:"rel:has-many,join:id=project_id" json:"relprojectidpublicchathistories,omitempty"` // Has many ModelPublicChatHistories
|
||||
RelProjectIDPublicLearnings []*ModelPublicLearnings `bun:"rel:has-many,join:id=project_id" json:"relprojectidpubliclearnings,omitempty"` // Has many ModelPublicLearnings
|
||||
RelProjectIDPublicPlans []*ModelPublicPlans `bun:"rel:has-many,join:id=project_id" json:"relprojectidpublicplans,omitempty"` // Has many ModelPublicPlans
|
||||
RelProjectIDPublicProjectSkills []*ModelPublicProjectSkills `bun:"rel:has-many,join:id=project_id" json:"relprojectidpublicprojectskills,omitempty"` // Has many ModelPublicProjectSkills
|
||||
RelProjectIDPublicProjectGuardrails []*ModelPublicProjectGuardrails `bun:"rel:has-many,join:id=project_id" json:"relprojectidpublicprojectguardrails,omitempty"` // Has many ModelPublicProjectGuardrails
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicProjects
|
||||
func (m ModelPublicProjects) TableName() string {
|
||||
return "public.projects"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicProjects
|
||||
func (m ModelPublicProjects) TableNameOnly() string {
|
||||
return "projects"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicProjects
|
||||
func (m ModelPublicProjects) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicProjects) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicProjects) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicProjects) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicProjects) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicProjects) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicProjects) GetPrefix() string {
|
||||
return "PRO"
|
||||
}
|
||||
@@ -1,72 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicStoredFiles struct {
|
||||
bun.BaseModel `bun:"table:public.stored_files,alias:stored_files"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
Content []byte `bun:"content,type:bytea,notnull," json:"content"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Encoding resolvespec_common.SqlString `bun:"encoding,type:text,default:'base64',notnull," json:"encoding"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
Kind resolvespec_common.SqlString `bun:"kind,type:text,default:'file',notnull," json:"kind"`
|
||||
MediaType resolvespec_common.SqlString `bun:"media_type,type:text,notnull," json:"media_type"`
|
||||
Name resolvespec_common.SqlString `bun:"name,type:text,notnull," json:"name"`
|
||||
ProjectID resolvespec_common.SqlInt64 `bun:"project_id,type:bigint,nullzero," json:"project_id"`
|
||||
Sha256 resolvespec_common.SqlString `bun:"sha256,type:text,notnull," json:"sha256"`
|
||||
SizeBytes int64 `bun:"size_bytes,type:bigint,notnull," json:"size_bytes"`
|
||||
ThoughtID resolvespec_common.SqlInt64 `bun:"thought_id,type:bigint,nullzero," json:"thought_id"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
RelProjectID *ModelPublicProjects `bun:"rel:has-one,join:project_id=id" json:"relprojectid,omitempty"` // Has one ModelPublicProjects
|
||||
RelThoughtID *ModelPublicThoughts `bun:"rel:has-one,join:thought_id=id" json:"relthoughtid,omitempty"` // Has one ModelPublicThoughts
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicStoredFiles
|
||||
func (m ModelPublicStoredFiles) TableName() string {
|
||||
return "public.stored_files"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicStoredFiles
|
||||
func (m ModelPublicStoredFiles) TableNameOnly() string {
|
||||
return "stored_files"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicStoredFiles
|
||||
func (m ModelPublicStoredFiles) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicStoredFiles) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicStoredFiles) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicStoredFiles) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicStoredFiles) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicStoredFiles) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicStoredFiles) GetPrefix() string {
|
||||
return "SFT"
|
||||
}
|
||||
@@ -1,64 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicThoughtLinks struct {
|
||||
bun.BaseModel `bun:"table:public.thought_links,alias:thought_links"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),nullzero," json:"created_at"`
|
||||
FromID int64 `bun:"from_id,type:bigint,notnull," json:"from_id"`
|
||||
Relation resolvespec_common.SqlString `bun:"relation,type:text,notnull," json:"relation"`
|
||||
ToID int64 `bun:"to_id,type:bigint,notnull," json:"to_id"`
|
||||
RelFromID *ModelPublicThoughts `bun:"rel:has-one,join:from_id=id" json:"relfromid,omitempty"` // Has one ModelPublicThoughts
|
||||
RelToID *ModelPublicThoughts `bun:"rel:has-one,join:to_id=id" json:"reltoid,omitempty"` // Has one ModelPublicThoughts
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicThoughtLinks
|
||||
func (m ModelPublicThoughtLinks) TableName() string {
|
||||
return "public.thought_links"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicThoughtLinks
|
||||
func (m ModelPublicThoughtLinks) TableNameOnly() string {
|
||||
return "thought_links"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicThoughtLinks
|
||||
func (m ModelPublicThoughtLinks) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicThoughtLinks) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicThoughtLinks) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicThoughtLinks) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicThoughtLinks) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicThoughtLinks) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicThoughtLinks) GetPrefix() string {
|
||||
return "TLH"
|
||||
}
|
||||
@@ -1,71 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicThoughts struct {
|
||||
bun.BaseModel `bun:"table:public.thoughts,alias:thoughts"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
ArchivedAt resolvespec_common.SqlTimeStamp `bun:"archived_at,type:timestamptz,nullzero," json:"archived_at"`
|
||||
Content resolvespec_common.SqlString `bun:"content,type:text,notnull," json:"content"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),nullzero," json:"created_at"`
|
||||
GUID resolvespec_common.SqlUUID `bun:"guid,type:uuid,default:gen_random_uuid(),notnull," json:"guid"`
|
||||
Metadata resolvespec_common.SqlJSONB `bun:"metadata,type:jsonb,default:{}::jsonb,nullzero," json:"metadata"`
|
||||
ProjectID resolvespec_common.SqlInt64 `bun:"project_id,type:bigint,nullzero," json:"project_id"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),nullzero," json:"updated_at"`
|
||||
RelProjectID *ModelPublicProjects `bun:"rel:has-one,join:project_id=id" json:"relprojectid,omitempty"` // Has one ModelPublicProjects
|
||||
RelFromIDPublicThoughtLinks []*ModelPublicThoughtLinks `bun:"rel:has-many,join:id=from_id" json:"relfromidpublicthoughtlinks,omitempty"` // Has many ModelPublicThoughtLinks
|
||||
RelToIDPublicThoughtLinks []*ModelPublicThoughtLinks `bun:"rel:has-many,join:id=to_id" json:"reltoidpublicthoughtlinks,omitempty"` // Has many ModelPublicThoughtLinks
|
||||
RelThoughtIDPublicEmbeddings []*ModelPublicEmbeddings `bun:"rel:has-many,join:id=thought_id" json:"relthoughtidpublicembeddings,omitempty"` // Has many ModelPublicEmbeddings
|
||||
RelThoughtIDPublicStoredFiles []*ModelPublicStoredFiles `bun:"rel:has-many,join:id=thought_id" json:"relthoughtidpublicstoredfiles,omitempty"` // Has many ModelPublicStoredFiles
|
||||
RelRelatedThoughtIDPublicLearnings []*ModelPublicLearnings `bun:"rel:has-many,join:id=related_thought_id" json:"relrelatedthoughtidpubliclearnings,omitempty"` // Has many ModelPublicLearnings
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicThoughts
|
||||
func (m ModelPublicThoughts) TableName() string {
|
||||
return "public.thoughts"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicThoughts
|
||||
func (m ModelPublicThoughts) TableNameOnly() string {
|
||||
return "thoughts"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicThoughts
|
||||
func (m ModelPublicThoughts) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicThoughts) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicThoughts) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicThoughts) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicThoughts) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicThoughts) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicThoughts) GetPrefix() string {
|
||||
return "THO"
|
||||
}
|
||||
@@ -1,62 +0,0 @@
|
||||
// Code generated by relspecgo. DO NOT EDIT.
|
||||
package generatedmodels
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
resolvespec_common "github.com/bitechdev/ResolveSpec/pkg/spectypes"
|
||||
"github.com/uptrace/bun"
|
||||
)
|
||||
|
||||
type ModelPublicToolAnnotations struct {
|
||||
bun.BaseModel `bun:"table:public.tool_annotations,alias:tool_annotations"`
|
||||
ID resolvespec_common.SqlInt64 `bun:"id,type:bigserial,pk,autoincrement," json:"id"`
|
||||
CreatedAt resolvespec_common.SqlTimeStamp `bun:"created_at,type:timestamptz,default:now(),notnull," json:"created_at"`
|
||||
Notes resolvespec_common.SqlString `bun:"notes,type:text,default:'',notnull," json:"notes"`
|
||||
ToolName resolvespec_common.SqlString `bun:"tool_name,type:text,notnull," json:"tool_name"`
|
||||
UpdatedAt resolvespec_common.SqlTimeStamp `bun:"updated_at,type:timestamptz,default:now(),notnull," json:"updated_at"`
|
||||
}
|
||||
|
||||
// TableName returns the table name for ModelPublicToolAnnotations
|
||||
func (m ModelPublicToolAnnotations) TableName() string {
|
||||
return "public.tool_annotations"
|
||||
}
|
||||
|
||||
// TableNameOnly returns the table name without schema for ModelPublicToolAnnotations
|
||||
func (m ModelPublicToolAnnotations) TableNameOnly() string {
|
||||
return "tool_annotations"
|
||||
}
|
||||
|
||||
// SchemaName returns the schema name for ModelPublicToolAnnotations
|
||||
func (m ModelPublicToolAnnotations) SchemaName() string {
|
||||
return "public"
|
||||
}
|
||||
|
||||
// GetID returns the primary key value
|
||||
func (m ModelPublicToolAnnotations) GetID() int64 {
|
||||
return m.ID.Int64()
|
||||
}
|
||||
|
||||
// GetIDStr returns the primary key as a string
|
||||
func (m ModelPublicToolAnnotations) GetIDStr() string {
|
||||
return fmt.Sprintf("%v", m.ID)
|
||||
}
|
||||
|
||||
// SetID sets the primary key value
|
||||
func (m ModelPublicToolAnnotations) SetID(newid int64) {
|
||||
m.UpdateID(newid)
|
||||
}
|
||||
|
||||
// UpdateID updates the primary key value
|
||||
func (m *ModelPublicToolAnnotations) UpdateID(newid int64) {
|
||||
m.ID.FromString(fmt.Sprintf("%d", newid))
|
||||
}
|
||||
|
||||
// GetIDName returns the name of the primary key column
|
||||
func (m ModelPublicToolAnnotations) GetIDName() string {
|
||||
return "id"
|
||||
}
|
||||
|
||||
// GetPrefix returns the table prefix
|
||||
func (m ModelPublicToolAnnotations) GetPrefix() string {
|
||||
return "TAO"
|
||||
}
|
||||
@@ -31,7 +31,7 @@ func TestSetToolSchemasAddsEmptyPropertiesForNoArgInput(t *testing.T) {
|
||||
}
|
||||
}
|
||||
|
||||
func TestSetToolSchemasUsesIntegerIDsInListOutput(t *testing.T) {
|
||||
func TestSetToolSchemasUsesStringUUIDsInListOutput(t *testing.T) {
|
||||
tool := &mcp.Tool{Name: "list_thoughts"}
|
||||
|
||||
if err := setToolSchemas[tools.ListInput, tools.ListOutput](tool); err != nil {
|
||||
@@ -55,8 +55,11 @@ func TestSetToolSchemasUsesIntegerIDsInListOutput(t *testing.T) {
|
||||
if idSchema == nil {
|
||||
t.Fatal("missing id schema")
|
||||
}
|
||||
if idSchema.Type != "integer" {
|
||||
t.Fatalf("id schema type = %q, want %q", idSchema.Type, "integer")
|
||||
if idSchema.Type != "string" {
|
||||
t.Fatalf("id schema type = %q, want %q", idSchema.Type, "string")
|
||||
}
|
||||
if idSchema.Format != "uuid" {
|
||||
t.Fatalf("id schema format = %q, want %q", idSchema.Format, "uuid")
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
+39
-330
@@ -36,13 +36,10 @@ type ToolSet struct {
|
||||
Backfill *tools.BackfillTool
|
||||
Reparse *tools.ReparseMetadataTool
|
||||
RetryMetadata *tools.RetryEnrichmentTool
|
||||
//Maintenance *tools.MaintenanceTool
|
||||
Skills *tools.SkillsTool
|
||||
Personas *tools.AgentPersonasTool
|
||||
ChatHistory *tools.ChatHistoryTool
|
||||
Describe *tools.DescribeTool
|
||||
Learnings *tools.LearningsTool
|
||||
Plans *tools.PlansTool
|
||||
Maintenance *tools.MaintenanceTool
|
||||
Skills *tools.SkillsTool
|
||||
ChatHistory *tools.ChatHistoryTool
|
||||
Describe *tools.DescribeTool
|
||||
}
|
||||
|
||||
// Handlers groups the HTTP handlers produced for an MCP server instance.
|
||||
@@ -86,12 +83,9 @@ func NewHandlers(cfg config.MCPConfig, logger *slog.Logger, toolSet ToolSet, onS
|
||||
registerSystemTools,
|
||||
registerThoughtTools,
|
||||
registerProjectTools,
|
||||
registerLearningTools,
|
||||
registerPlanTools,
|
||||
registerFileTools,
|
||||
registerMaintenanceTools,
|
||||
registerSkillTools,
|
||||
registerPersonaTools,
|
||||
registerChatHistoryTools,
|
||||
registerDescribeTools,
|
||||
} {
|
||||
@@ -255,122 +249,6 @@ func registerProjectTools(server *mcp.Server, logger *slog.Logger, toolSet ToolS
|
||||
return nil
|
||||
}
|
||||
|
||||
func registerLearningTools(server *mcp.Server, logger *slog.Logger, toolSet ToolSet) error {
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "add_learning",
|
||||
Description: "Create a curated learning record distinct from raw thoughts.",
|
||||
}, toolSet.Learnings.Add); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "get_learning",
|
||||
Description: "Retrieve a structured learning by id.",
|
||||
}, toolSet.Learnings.Get); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "list_learnings",
|
||||
Description: "List structured learnings with optional project, status, priority, tag, and text filters.",
|
||||
}, toolSet.Learnings.List); err != nil {
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func registerPlanTools(server *mcp.Server, logger *slog.Logger, toolSet ToolSet) error {
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "create_plan",
|
||||
Description: "Create a structured plan linked to a project.",
|
||||
}, toolSet.Plans.Create); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "get_plan",
|
||||
Description: "Retrieve a plan with its dependencies, related plans, skills, and guardrails.",
|
||||
}, toolSet.Plans.Get); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "update_plan",
|
||||
Description: "Update plan fields; only provided fields are changed.",
|
||||
}, toolSet.Plans.Update); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "delete_plan",
|
||||
Description: "Hard-delete a plan by id.",
|
||||
}, toolSet.Plans.Delete); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "list_plans",
|
||||
Description: "List plans with optional project, status, priority, owner, tag, and text filters.",
|
||||
}, toolSet.Plans.List); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "add_plan_dependency",
|
||||
Description: "Mark plan_id as depending on depends_on_plan_id (must complete first).",
|
||||
}, toolSet.Plans.AddDependency); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "remove_plan_dependency",
|
||||
Description: "Remove a dependency between two plans.",
|
||||
}, toolSet.Plans.RemoveDependency); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "add_related_plan",
|
||||
Description: "Link two plans as thematically related (bidirectional).",
|
||||
}, toolSet.Plans.AddRelated); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "remove_related_plan",
|
||||
Description: "Unlink two related plans.",
|
||||
}, toolSet.Plans.RemoveRelated); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "add_plan_skill",
|
||||
Description: "Link an agent skill to a plan.",
|
||||
}, toolSet.Plans.AddSkill); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "remove_plan_skill",
|
||||
Description: "Unlink an agent skill from a plan.",
|
||||
}, toolSet.Plans.RemoveSkill); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "list_plan_skills",
|
||||
Description: "List skills linked to a plan.",
|
||||
}, toolSet.Plans.ListSkills); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "add_plan_guardrail",
|
||||
Description: "Link an agent guardrail to a plan.",
|
||||
}, toolSet.Plans.AddGuardrail); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "remove_plan_guardrail",
|
||||
Description: "Unlink an agent guardrail from a plan.",
|
||||
}, toolSet.Plans.RemoveGuardrail); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "list_plan_guardrails",
|
||||
Description: "List guardrails linked to a plan.",
|
||||
}, toolSet.Plans.ListGuardrails); err != nil {
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func registerFileTools(server *mcp.Server, logger *slog.Logger, toolSet ToolSet) error {
|
||||
server.AddResourceTemplate(&mcp.ResourceTemplate{
|
||||
Name: "stored_file",
|
||||
@@ -424,30 +302,30 @@ func registerMaintenanceTools(server *mcp.Server, logger *slog.Logger, toolSet T
|
||||
}, toolSet.RetryMetadata.Handle); err != nil {
|
||||
return err
|
||||
}
|
||||
// if err := addTool(server, logger, &mcp.Tool{
|
||||
// Name: "add_maintenance_task",
|
||||
// Description: "Create a recurring or one-time home maintenance task.",
|
||||
// }, toolSet.Maintenance.AddTask); err != nil {
|
||||
// return err
|
||||
// }
|
||||
// if err := addTool(server, logger, &mcp.Tool{
|
||||
// Name: "log_maintenance",
|
||||
// Description: "Log completed maintenance; updates next due date.",
|
||||
// }, toolSet.Maintenance.LogWork); err != nil {
|
||||
// return err
|
||||
// }
|
||||
// if err := addTool(server, logger, &mcp.Tool{
|
||||
// Name: "get_upcoming_maintenance",
|
||||
// Description: "List maintenance tasks due within the next N days.",
|
||||
// }, toolSet.Maintenance.GetUpcoming); err != nil {
|
||||
// return err
|
||||
// }
|
||||
// if err := addTool(server, logger, &mcp.Tool{
|
||||
// Name: "search_maintenance_history",
|
||||
// Description: "Search the maintenance log by task name, category, or date range.",
|
||||
// }, toolSet.Maintenance.SearchHistory); err != nil {
|
||||
// return err
|
||||
// }
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "add_maintenance_task",
|
||||
Description: "Create a recurring or one-time home maintenance task.",
|
||||
}, toolSet.Maintenance.AddTask); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "log_maintenance",
|
||||
Description: "Log completed maintenance; updates next due date.",
|
||||
}, toolSet.Maintenance.LogWork); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "get_upcoming_maintenance",
|
||||
Description: "List maintenance tasks due within the next N days.",
|
||||
}, toolSet.Maintenance.GetUpcoming); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "search_maintenance_history",
|
||||
Description: "Search the maintenance log by task name, category, or date range.",
|
||||
}, toolSet.Maintenance.SearchHistory); err != nil {
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -466,16 +344,10 @@ func registerSkillTools(server *mcp.Server, logger *slog.Logger, toolSet ToolSet
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "list_skills",
|
||||
Description: "List agent skills (metadata only by default). Set include_content=true to get full content, or use get_skill for a single skill.",
|
||||
Description: "List all agent skills, optionally filtered by tag.",
|
||||
}, toolSet.Skills.ListSkills); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "get_skill",
|
||||
Description: "Fetch a single agent skill with full content by id or name.",
|
||||
}, toolSet.Skills.GetSkill); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "add_guardrail",
|
||||
Description: "Store an agent guardrail (constraint or safety rule).",
|
||||
@@ -494,12 +366,6 @@ func registerSkillTools(server *mcp.Server, logger *slog.Logger, toolSet ToolSet
|
||||
}, toolSet.Skills.ListGuardrails); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "get_guardrail",
|
||||
Description: "Fetch a single agent guardrail with full content by id or name.",
|
||||
}, toolSet.Skills.GetGuardrail); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "add_project_skill",
|
||||
Description: "Link a skill to a project. Pass project if client is stateless.",
|
||||
@@ -567,114 +433,10 @@ func registerChatHistoryTools(server *mcp.Server, logger *slog.Logger, toolSet T
|
||||
return nil
|
||||
}
|
||||
|
||||
func registerPersonaTools(server *mcp.Server, logger *slog.Logger, toolSet ToolSet) error {
|
||||
p := toolSet.Personas
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "create_agent_persona", Description: "Create a named, loadable agent persona."}, p.CreatePersona); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "update_agent_persona", Description: "Update persona fields (name, description, summary, detail, tags)."}, p.UpdatePersona); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "delete_agent_persona", Description: "Delete a persona by name."}, p.DeletePersona); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "list_agent_personas", Description: "List all personas, optionally filtered by tag."}, p.ListPersonas); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "get_agent_persona", Description: "Load a persona with assembled parts. detail=true returns full content. overrides replaces parts per type at runtime."}, p.GetPersona); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "get_persona_manifest", Description: "Lightweight structure-only view: parts, traits, skills, guardrails — no content. Includes on_demand_tools hints."}, p.GetPersonaManifest); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "compile_persona", Description: "Regenerate compiled_summary and compiled_detail from current parts and arc stage. Use compiled_summary for agents with tight context budgets."}, p.CompilePersona); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "create_agent_part", Description: "Create a reusable behaviour building block. Parts compose personas and can be overridden at load time by name."}, p.CreatePart); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "update_agent_part", Description: "Update part fields (name, part_type, description, summary, content, tags)."}, p.UpdatePart); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "delete_agent_part", Description: "Delete a part by name."}, p.DeletePart); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "list_agent_parts", Description: "List parts, optionally filtered by part_type or tag."}, p.ListParts); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "get_agent_part", Description: "Fetch a single part with full content by name."}, p.GetPart); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "add_persona_part", Description: "Link a part to a persona with optional order and priority."}, p.AddPersonaPart); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "remove_persona_part", Description: "Unlink a part from a persona."}, p.RemovePersonaPart); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "add_persona_skill", Description: "Link an agent_skill to a persona."}, p.AddPersonaSkill); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "remove_persona_skill", Description: "Unlink an agent_skill from a persona."}, p.RemovePersonaSkill); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "add_persona_guardrail", Description: "Link an agent_guardrail to a persona."}, p.AddPersonaGuardrail); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "remove_persona_guardrail", Description: "Unlink an agent_guardrail from a persona."}, p.RemovePersonaGuardrail); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "create_agent_trait", Description: "Create an atomic personality trait (personality, cognitive, emotional, social, behavioral)."}, p.CreateTrait); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "update_agent_trait", Description: "Update trait fields."}, p.UpdateTrait); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "delete_agent_trait", Description: "Delete a trait by name."}, p.DeleteTrait); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "list_agent_traits", Description: "List traits, optionally filtered by trait_type or tag."}, p.ListTraits); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "get_agent_trait", Description: "Fetch a single trait with instruction by name."}, p.GetTrait); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "add_persona_trait", Description: "Link a trait to a persona."}, p.AddPersonaTrait); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "remove_persona_trait", Description: "Unlink a trait from a persona."}, p.RemovePersonaTrait); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "create_character_arc", Description: "Define a named character progression arc."}, p.CreateCharacterArc); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "list_character_arcs", Description: "List all character arcs."}, p.ListCharacterArcs); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "add_arc_stage", Description: "Add an ordered stage to a character arc."}, p.AddArcStage); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "add_stage_part", Description: "Link a part to an arc stage — active when the persona is at that stage."}, p.AddStagePart); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "remove_stage_part", Description: "Unlink a part from an arc stage."}, p.RemoveStagePart); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "assign_persona_arc", Description: "Attach an arc to a persona and set the starting stage."}, p.AssignPersonaArc); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "advance_persona_stage", Description: "Move persona to the next stage in its arc."}, p.AdvancePersonaStage); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := addTool(server, logger, &mcp.Tool{Name: "reset_persona_stage", Description: "Reset persona to the first stage of its arc."}, p.ResetPersonaStage); err != nil {
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func registerDescribeTools(server *mcp.Server, logger *slog.Logger, toolSet ToolSet) error {
|
||||
if err := addTool(server, logger, &mcp.Tool{
|
||||
Name: "describe_tools",
|
||||
Description: "Call first each session. All tools with categories and usage notes. Categories: system, thoughts, projects, files, admin, maintenance, skills, personas, plans, chat, meta.",
|
||||
Description: "Call first each session. All tools with categories and usage notes. Categories: system, thoughts, projects, files, admin, maintenance, skills, chat, meta.",
|
||||
}, toolSet.Describe.Describe); err != nil {
|
||||
return err
|
||||
}
|
||||
@@ -715,28 +477,6 @@ func BuildToolCatalog() []tools.ToolEntry {
|
||||
{Name: "get_active_project", Description: "Return the active project for the current MCP session. If your client does not preserve MCP sessions, pass project explicitly to project-scoped tools instead of relying on this.", Category: "projects"},
|
||||
{Name: "get_project_context", Description: "Get recent and semantic context for a project. Uses the explicit project when provided, otherwise the active MCP session project. Falls back to full-text search when no embeddings exist.", Category: "projects"},
|
||||
|
||||
// learnings
|
||||
{Name: "add_learning", Description: "Create a curated learning record distinct from raw thoughts.", Category: "projects"},
|
||||
{Name: "get_learning", Description: "Retrieve a structured learning by id.", Category: "projects"},
|
||||
{Name: "list_learnings", Description: "List structured learnings with optional project, category, area, status, priority, tag, and text filters.", Category: "projects"},
|
||||
|
||||
// plans
|
||||
{Name: "create_plan", Description: "Create a structured plan with status, priority, owner, due date, and optional project link.", Category: "plans"},
|
||||
{Name: "get_plan", Description: "Retrieve a full plan including dependencies (depends_on/blocks), related plans, linked skills, and guardrails.", Category: "plans"},
|
||||
{Name: "update_plan", Description: "Partially update a plan; only provided fields are changed. Use mark_reviewed to stamp last_reviewed_at.", Category: "plans"},
|
||||
{Name: "delete_plan", Description: "Hard-delete a plan by id.", Category: "plans"},
|
||||
{Name: "list_plans", Description: "List plans with optional filters: project, status, priority, owner, tag, and full-text query.", Category: "plans"},
|
||||
{Name: "add_plan_dependency", Description: "Declare that plan_id cannot proceed until depends_on_plan_id is complete.", Category: "plans"},
|
||||
{Name: "remove_plan_dependency", Description: "Remove a directional dependency between two plans.", Category: "plans"},
|
||||
{Name: "add_related_plan", Description: "Link two plans as thematically related (bidirectional, order-independent).", Category: "plans"},
|
||||
{Name: "remove_related_plan", Description: "Unlink two related plans.", Category: "plans"},
|
||||
{Name: "add_plan_skill", Description: "Link an agent skill to a plan so it is loaded with the plan's context.", Category: "plans"},
|
||||
{Name: "remove_plan_skill", Description: "Unlink an agent skill from a plan.", Category: "plans"},
|
||||
{Name: "list_plan_skills", Description: "List all skills linked to a plan.", Category: "plans"},
|
||||
{Name: "add_plan_guardrail", Description: "Link an agent guardrail to a plan so it applies during plan execution.", Category: "plans"},
|
||||
{Name: "remove_plan_guardrail", Description: "Unlink an agent guardrail from a plan.", Category: "plans"},
|
||||
{Name: "list_plan_guardrails", Description: "List all guardrails linked to a plan.", Category: "plans"},
|
||||
|
||||
// files
|
||||
{Name: "upload_file", Description: "Stage a file and get an amcs://files/{id} resource URI. Use content_path (absolute server-side path, no size limit) for large or binary files, or content_base64 (≤10 MB) for small files. Pass thought_id/project to link immediately, or omit and pass the URI to save_file later.", Category: "files"},
|
||||
{Name: "save_file", Description: "Store a file and optionally link it to a thought. Use content_base64 (≤10 MB) for small files, or content_uri (amcs://files/{id} from a prior upload_file) for previously staged files. For files larger than 10 MB, use upload_file with content_path first. If the goal is to retain the artifact, store the file directly instead of reading or summarising it first.", Category: "files"},
|
||||
@@ -748,15 +488,19 @@ func BuildToolCatalog() []tools.ToolEntry {
|
||||
{Name: "reparse_thought_metadata", Description: "Re-extract and normalize metadata for stored thoughts from their content.", Category: "admin"},
|
||||
{Name: "retry_failed_metadata", Description: "Retry metadata extraction for thoughts still marked pending or failed.", Category: "admin"},
|
||||
|
||||
// maintenance
|
||||
{Name: "add_maintenance_task", Description: "Create a recurring or one-time home maintenance task.", Category: "maintenance"},
|
||||
{Name: "log_maintenance", Description: "Log completed maintenance work; automatically updates the task's next due date.", Category: "maintenance"},
|
||||
{Name: "get_upcoming_maintenance", Description: "List maintenance tasks due within the next N days.", Category: "maintenance"},
|
||||
{Name: "search_maintenance_history", Description: "Search the maintenance log by task name, category, or date range.", Category: "maintenance"},
|
||||
|
||||
// skills
|
||||
{Name: "add_skill", Description: "Store a reusable agent skill. Supports language_tags, library_tags, framework_tags, and domain_tags for precise retrieval.", Category: "skills"},
|
||||
{Name: "add_skill", Description: "Store a reusable agent skill (behavioural instruction or capability prompt).", Category: "skills"},
|
||||
{Name: "remove_skill", Description: "Delete an agent skill by id.", Category: "skills"},
|
||||
{Name: "list_skills", Description: "List agent skills (metadata only by default). Filter by tag (searches all tag fields). Set include_content=true for full bodies, or use get_skill to load one.", Category: "skills"},
|
||||
{Name: "get_skill", Description: "Fetch a single agent skill with full content by id or name. Prefer this over list_skills when you know which skill you need.", Category: "skills"},
|
||||
{Name: "list_skills", Description: "List all agent skills, optionally filtered by tag.", Category: "skills"},
|
||||
{Name: "add_guardrail", Description: "Store a reusable agent guardrail (constraint or safety rule).", Category: "skills"},
|
||||
{Name: "remove_guardrail", Description: "Delete an agent guardrail by id.", Category: "skills"},
|
||||
{Name: "list_guardrails", Description: "List all agent guardrails, optionally filtered by tag or severity.", Category: "skills"},
|
||||
{Name: "get_guardrail", Description: "Fetch a single agent guardrail with full content by id or name.", Category: "skills"},
|
||||
{Name: "add_project_skill", Description: "Link an agent skill to a project. Pass project explicitly when your client does not preserve MCP sessions.", Category: "skills"},
|
||||
{Name: "remove_project_skill", Description: "Unlink an agent skill from a project. Pass project explicitly when your client does not preserve MCP sessions.", Category: "skills"},
|
||||
{Name: "list_project_skills", Description: "List all skills linked to a project. Call this at the start of every project session to load agent behaviour instructions before generating new ones. Only create new skills if none are returned. Pass project explicitly when your client does not preserve MCP sessions.", Category: "skills"},
|
||||
@@ -764,41 +508,6 @@ func BuildToolCatalog() []tools.ToolEntry {
|
||||
{Name: "remove_project_guardrail", Description: "Unlink an agent guardrail from a project. Pass project explicitly when your client does not preserve MCP sessions.", Category: "skills"},
|
||||
{Name: "list_project_guardrails", Description: "List all guardrails linked to a project. Call this at the start of every project session to load agent constraints before generating new ones. Only create new guardrails if none are returned. Pass project explicitly when your client does not preserve MCP sessions.", Category: "skills"},
|
||||
|
||||
// personas
|
||||
{Name: "create_agent_persona", Description: "Create a named, loadable agent persona that composes parts, traits, skills, and guardrails.", Category: "personas"},
|
||||
{Name: "update_agent_persona", Description: "Update persona fields.", Category: "personas"},
|
||||
{Name: "delete_agent_persona", Description: "Delete a persona by name.", Category: "personas"},
|
||||
{Name: "list_agent_personas", Description: "List all personas, optionally filtered by tag.", Category: "personas"},
|
||||
{Name: "get_agent_persona", Description: "Load a persona with all assembled parts. detail=true returns full content. overrides replaces parts per type at runtime without modifying the persona.", Category: "personas"},
|
||||
{Name: "get_persona_manifest", Description: "Lightweight discovery: returns persona structure (parts, traits, skills, guardrails) with no content, plus on_demand_tools hints. Call this first for unknown personas.", Category: "personas"},
|
||||
{Name: "compile_persona", Description: "Regenerate compiled_summary and compiled_detail from current parts and arc stage. Agents with tight context budgets can use compiled_summary directly.", Category: "personas"},
|
||||
{Name: "create_agent_part", Description: "Create a reusable behaviour building block. Part types: system, agent, soul, identity, skill, specialization, tone, goal, context, protocol, backstory, motivation, voice, archetype, flaw, relationship.", Category: "personas"},
|
||||
{Name: "update_agent_part", Description: "Update part fields.", Category: "personas"},
|
||||
{Name: "delete_agent_part", Description: "Delete a part by name.", Category: "personas"},
|
||||
{Name: "list_agent_parts", Description: "List parts, optionally filtered by part_type or tag.", Category: "personas"},
|
||||
{Name: "get_agent_part", Description: "Fetch a single part with full content by name.", Category: "personas"},
|
||||
{Name: "add_persona_part", Description: "Link a part to a persona with optional order and priority.", Category: "personas"},
|
||||
{Name: "remove_persona_part", Description: "Unlink a part from a persona.", Category: "personas"},
|
||||
{Name: "add_persona_skill", Description: "Link an agent_skill to a persona.", Category: "personas"},
|
||||
{Name: "remove_persona_skill", Description: "Unlink an agent_skill from a persona.", Category: "personas"},
|
||||
{Name: "add_persona_guardrail", Description: "Link an agent_guardrail to a persona.", Category: "personas"},
|
||||
{Name: "remove_persona_guardrail", Description: "Unlink an agent_guardrail from a persona.", Category: "personas"},
|
||||
{Name: "create_agent_trait", Description: "Create an atomic personality trait. Trait types: personality, cognitive, emotional, social, behavioral.", Category: "personas"},
|
||||
{Name: "update_agent_trait", Description: "Update trait fields.", Category: "personas"},
|
||||
{Name: "delete_agent_trait", Description: "Delete a trait by name.", Category: "personas"},
|
||||
{Name: "list_agent_traits", Description: "List traits, optionally filtered by trait_type or tag.", Category: "personas"},
|
||||
{Name: "get_agent_trait", Description: "Fetch a single trait with instruction by name. Use this for on-demand trait loading.", Category: "personas"},
|
||||
{Name: "add_persona_trait", Description: "Link a trait to a persona.", Category: "personas"},
|
||||
{Name: "remove_persona_trait", Description: "Unlink a trait from a persona.", Category: "personas"},
|
||||
{Name: "create_character_arc", Description: "Define a named character progression arc with ordered stages.", Category: "personas"},
|
||||
{Name: "list_character_arcs", Description: "List all character arcs.", Category: "personas"},
|
||||
{Name: "add_arc_stage", Description: "Add an ordered stage to a character arc.", Category: "personas"},
|
||||
{Name: "add_stage_part", Description: "Link a part to an arc stage — overrides matching persona parts when that stage is active.", Category: "personas"},
|
||||
{Name: "remove_stage_part", Description: "Unlink a part from an arc stage.", Category: "personas"},
|
||||
{Name: "assign_persona_arc", Description: "Attach a character arc to a persona and set the starting stage.", Category: "personas"},
|
||||
{Name: "advance_persona_stage", Description: "Move persona to the next stage in its arc.", Category: "personas"},
|
||||
{Name: "reset_persona_stage", Description: "Reset persona to the first stage of its arc.", Category: "personas"},
|
||||
|
||||
// chat
|
||||
{Name: "save_chat_history", Description: "Save a chat session's message history for later retrieval. Stores messages with optional title, summary, channel, agent, and project metadata.", Category: "chat"},
|
||||
{Name: "get_chat_history", Description: "Retrieve a saved chat history by its UUID or session_id. Returns the full message list.", Category: "chat"},
|
||||
@@ -806,7 +515,7 @@ func BuildToolCatalog() []tools.ToolEntry {
|
||||
{Name: "delete_chat_history", Description: "Permanently delete a saved chat history by id.", Category: "chat"},
|
||||
|
||||
// meta
|
||||
{Name: "describe_tools", Description: "Call this first in every session. Returns all available MCP tools with names, descriptions, categories, and your accumulated usage notes. Filter by category to narrow results. Available categories: system, thoughts, projects, files, admin, household, maintenance, calendar, meals, crm, skills, personas, plans, chat, meta.", Category: "meta"},
|
||||
{Name: "describe_tools", Description: "Call this first in every session. Returns all available MCP tools with names, descriptions, categories, and your accumulated usage notes. Filter by category to narrow results. Available categories: system, thoughts, projects, files, admin, household, maintenance, calendar, meals, crm, skills, chat, meta.", Category: "meta"},
|
||||
{Name: "annotate_tool", Description: "Persist usage notes, gotchas, or workflow patterns for a specific tool. Notes survive across sessions and are returned by describe_tools. Call this whenever you discover something non-obvious about a tool's behaviour. Pass an empty string to clear notes.", Category: "meta"},
|
||||
}
|
||||
}
|
||||
|
||||
@@ -27,10 +27,54 @@ func TestNewListsAllRegisteredTools(t *testing.T) {
|
||||
}
|
||||
sort.Strings(got)
|
||||
|
||||
catalog := BuildToolCatalog()
|
||||
want := make([]string, 0, len(catalog))
|
||||
for _, tool := range catalog {
|
||||
want = append(want, tool.Name)
|
||||
want := []string{
|
||||
"add_guardrail",
|
||||
"add_maintenance_task",
|
||||
"add_project_guardrail",
|
||||
"add_project_skill",
|
||||
"add_skill",
|
||||
"annotate_tool",
|
||||
"archive_thought",
|
||||
"backfill_embeddings",
|
||||
"capture_thought",
|
||||
"create_project",
|
||||
"delete_chat_history",
|
||||
"delete_thought",
|
||||
"describe_tools",
|
||||
"get_active_project",
|
||||
"get_chat_history",
|
||||
"get_project_context",
|
||||
"get_thought",
|
||||
"get_upcoming_maintenance",
|
||||
"get_version_info",
|
||||
"link_thoughts",
|
||||
"list_chat_histories",
|
||||
"list_files",
|
||||
"list_guardrails",
|
||||
"list_project_guardrails",
|
||||
"list_project_skills",
|
||||
"list_projects",
|
||||
"list_skills",
|
||||
"list_thoughts",
|
||||
"load_file",
|
||||
"log_maintenance",
|
||||
"recall_context",
|
||||
"related_thoughts",
|
||||
"remove_guardrail",
|
||||
"remove_project_guardrail",
|
||||
"remove_project_skill",
|
||||
"remove_skill",
|
||||
"reparse_thought_metadata",
|
||||
"retry_failed_metadata",
|
||||
"save_chat_history",
|
||||
"save_file",
|
||||
"search_maintenance_history",
|
||||
"search_thoughts",
|
||||
"set_active_project",
|
||||
"summarize_thoughts",
|
||||
"thought_stats",
|
||||
"update_thought",
|
||||
"upload_file",
|
||||
}
|
||||
sort.Strings(want)
|
||||
|
||||
|
||||
@@ -105,86 +105,6 @@ func TestStreamableHTTPReturnsStructuredToolErrors(t *testing.T) {
|
||||
t.Fatalf("build_date = %#v, want %q", got["build_date"], "2026-03-31T00:00:00Z")
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("add_learning_requires_summary", func(t *testing.T) {
|
||||
_, err := cs.CallTool(context.Background(), &mcp.CallToolParams{
|
||||
Name: "add_learning",
|
||||
Arguments: map[string]any{},
|
||||
})
|
||||
if err == nil {
|
||||
t.Fatal("CallTool(add_learning) error = nil, want error")
|
||||
}
|
||||
|
||||
rpcErr, data := requireWireError(t, err)
|
||||
if rpcErr.Code != jsonrpc.CodeInvalidParams {
|
||||
t.Fatalf("add_learning code = %d, want %d", rpcErr.Code, jsonrpc.CodeInvalidParams)
|
||||
}
|
||||
if data.Type != mcperrors.TypeInvalidArguments {
|
||||
t.Fatalf("add_learning data.type = %q, want %q", data.Type, mcperrors.TypeInvalidArguments)
|
||||
}
|
||||
if data.Field != "summary" {
|
||||
t.Fatalf("add_learning data.field = %q, want %q", data.Field, "summary")
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("get_learning_requires_id", func(t *testing.T) {
|
||||
_, err := cs.CallTool(context.Background(), &mcp.CallToolParams{
|
||||
Name: "get_learning",
|
||||
Arguments: map[string]any{},
|
||||
})
|
||||
if err == nil {
|
||||
t.Fatal("CallTool(get_learning) error = nil, want error")
|
||||
}
|
||||
|
||||
rpcErr, data := requireWireError(t, err)
|
||||
if rpcErr.Code != jsonrpc.CodeInvalidParams {
|
||||
t.Fatalf("get_learning code = %d, want %d", rpcErr.Code, jsonrpc.CodeInvalidParams)
|
||||
}
|
||||
if data.Type != mcperrors.TypeInvalidArguments {
|
||||
t.Fatalf("get_learning data.type = %q, want %q", data.Type, mcperrors.TypeInvalidArguments)
|
||||
}
|
||||
if data.Field != "id" {
|
||||
t.Fatalf("get_learning data.field = %q, want %q", data.Field, "id")
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("add_learning_unconfigured_returns_structured_error", func(t *testing.T) {
|
||||
_, err := cs.CallTool(context.Background(), &mcp.CallToolParams{
|
||||
Name: "add_learning",
|
||||
Arguments: map[string]any{
|
||||
"summary": "Learning with configured check",
|
||||
},
|
||||
})
|
||||
if err == nil {
|
||||
t.Fatal("CallTool(add_learning) error = nil, want error")
|
||||
}
|
||||
|
||||
rpcErr, data := requireWireError(t, err)
|
||||
if rpcErr.Code != jsonrpc.CodeInvalidParams {
|
||||
t.Fatalf("add_learning code = %d, want %d", rpcErr.Code, jsonrpc.CodeInvalidParams)
|
||||
}
|
||||
if data.Type != mcperrors.TypeInvalidInput {
|
||||
t.Fatalf("add_learning data.type = %q, want %q", data.Type, mcperrors.TypeInvalidInput)
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("list_learnings_unconfigured_returns_structured_error", func(t *testing.T) {
|
||||
_, err := cs.CallTool(context.Background(), &mcp.CallToolParams{
|
||||
Name: "list_learnings",
|
||||
Arguments: map[string]any{},
|
||||
})
|
||||
if err == nil {
|
||||
t.Fatal("CallTool(list_learnings) error = nil, want error")
|
||||
}
|
||||
|
||||
rpcErr, data := requireWireError(t, err)
|
||||
if rpcErr.Code != jsonrpc.CodeInvalidParams {
|
||||
t.Fatalf("list_learnings code = %d, want %d", rpcErr.Code, jsonrpc.CodeInvalidParams)
|
||||
}
|
||||
if data.Type != mcperrors.TypeInvalidInput {
|
||||
t.Fatalf("list_learnings data.type = %q, want %q", data.Type, mcperrors.TypeInvalidInput)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func streamableTestToolSet() ToolSet {
|
||||
@@ -207,10 +127,7 @@ func streamableTestToolSet() ToolSet {
|
||||
Backfill: new(tools.BackfillTool),
|
||||
Reparse: new(tools.ReparseMetadataTool),
|
||||
RetryMetadata: new(tools.RetryEnrichmentTool),
|
||||
Maintenance: new(tools.MaintenanceTool),
|
||||
Skills: new(tools.SkillsTool),
|
||||
ChatHistory: new(tools.ChatHistoryTool),
|
||||
Describe: new(tools.DescribeTool),
|
||||
Learnings: new(tools.LearningsTool),
|
||||
Plans: new(tools.PlansTool),
|
||||
}
|
||||
}
|
||||
|
||||
@@ -106,11 +106,6 @@ func RequestIDFromContext(ctx context.Context) string {
|
||||
return value
|
||||
}
|
||||
|
||||
func MCPToolFromContext(ctx context.Context) string {
|
||||
value, _ := ctx.Value(mcpToolContextKey).(string)
|
||||
return strings.TrimSpace(value)
|
||||
}
|
||||
|
||||
type statusRecorder struct {
|
||||
http.ResponseWriter
|
||||
status int
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
+186
-182
@@ -1,206 +1,210 @@
|
||||
package store
|
||||
|
||||
// import (
|
||||
// "context"
|
||||
// "fmt"
|
||||
// "strings"
|
||||
// "time"
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
// "github.com/google/uuid"
|
||||
"github.com/google/uuid"
|
||||
|
||||
// "git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
// ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
// )
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
// func (db *DB) AddFamilyMember(ctx context.Context, m ext.FamilyMember) (ext.FamilyMember, error) {
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into family_members (name, relationship, birth_date, notes)
|
||||
// values ($1, $2, $3, $4)
|
||||
// returning id, created_at
|
||||
// `, m.Name, nullStr(m.Relationship), m.BirthDate, nullStr(m.Notes))
|
||||
func (db *DB) AddFamilyMember(ctx context.Context, m ext.FamilyMember) (ext.FamilyMember, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into family_members (name, relationship, birth_date, notes)
|
||||
values ($1, $2, $3, $4)
|
||||
returning id, created_at
|
||||
`, m.Name, nullStr(m.Relationship), m.BirthDate, nullStr(m.Notes))
|
||||
|
||||
// created := m
|
||||
// var model generatedmodels.ModelPublicFamilyMembers
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt); err != nil {
|
||||
// return ext.FamilyMember{}, fmt.Errorf("insert family member: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// created.CreatedAt = model.CreatedAt.Time()
|
||||
// return created, nil
|
||||
// }
|
||||
created := m
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt); err != nil {
|
||||
return ext.FamilyMember{}, fmt.Errorf("insert family member: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) ListFamilyMembers(ctx context.Context) ([]ext.FamilyMember, error) {
|
||||
// rows, err := db.pool.Query(ctx, `select id, name, relationship, birth_date, notes, created_at from family_members order by name`)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("list family members: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
func (db *DB) ListFamilyMembers(ctx context.Context) ([]ext.FamilyMember, error) {
|
||||
rows, err := db.pool.Query(ctx, `select id, name, relationship, birth_date, notes, created_at from family_members order by name`)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("list family members: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// var members []ext.FamilyMember
|
||||
// for rows.Next() {
|
||||
// var model generatedmodels.ModelPublicFamilyMembers
|
||||
// if err := rows.Scan(&model.ID, &model.Name, &model.Relationship, &model.BirthDate, &model.Notes, &model.CreatedAt); err != nil {
|
||||
// return nil, fmt.Errorf("scan family member: %w", err)
|
||||
// }
|
||||
// members = append(members, familyMemberFromModel(model))
|
||||
// }
|
||||
// return members, rows.Err()
|
||||
// }
|
||||
var members []ext.FamilyMember
|
||||
for rows.Next() {
|
||||
var m ext.FamilyMember
|
||||
var relationship, notes *string
|
||||
if err := rows.Scan(&m.ID, &m.Name, &relationship, &m.BirthDate, ¬es, &m.CreatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan family member: %w", err)
|
||||
}
|
||||
m.Relationship = strVal(relationship)
|
||||
m.Notes = strVal(notes)
|
||||
members = append(members, m)
|
||||
}
|
||||
return members, rows.Err()
|
||||
}
|
||||
|
||||
// func (db *DB) AddActivity(ctx context.Context, a ext.Activity) (ext.Activity, error) {
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into activities (family_member_id, title, activity_type, day_of_week, start_time, end_time, start_date, end_date, location, notes)
|
||||
// values ($1, $2, $3, $4, $5::time, $6::time, $7, $8, $9, $10)
|
||||
// returning id, created_at
|
||||
// `, a.FamilyMemberID, a.Title, nullStr(a.ActivityType), nullStr(a.DayOfWeek),
|
||||
// nullStr(a.StartTime), nullStr(a.EndTime), a.StartDate, a.EndDate,
|
||||
// nullStr(a.Location), nullStr(a.Notes))
|
||||
func (db *DB) AddActivity(ctx context.Context, a ext.Activity) (ext.Activity, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into activities (family_member_id, title, activity_type, day_of_week, start_time, end_time, start_date, end_date, location, notes)
|
||||
values ($1, $2, $3, $4, $5::time, $6::time, $7, $8, $9, $10)
|
||||
returning id, created_at
|
||||
`, a.FamilyMemberID, a.Title, nullStr(a.ActivityType), nullStr(a.DayOfWeek),
|
||||
nullStr(a.StartTime), nullStr(a.EndTime), a.StartDate, a.EndDate,
|
||||
nullStr(a.Location), nullStr(a.Notes))
|
||||
|
||||
// created := a
|
||||
// var model generatedmodels.ModelPublicActivities
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt); err != nil {
|
||||
// return ext.Activity{}, fmt.Errorf("insert activity: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// created.CreatedAt = model.CreatedAt.Time()
|
||||
// return created, nil
|
||||
// }
|
||||
created := a
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt); err != nil {
|
||||
return ext.Activity{}, fmt.Errorf("insert activity: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) GetWeekSchedule(ctx context.Context, weekStart time.Time) ([]ext.Activity, error) {
|
||||
// weekEnd := weekStart.AddDate(0, 0, 7)
|
||||
func (db *DB) GetWeekSchedule(ctx context.Context, weekStart time.Time) ([]ext.Activity, error) {
|
||||
weekEnd := weekStart.AddDate(0, 0, 7)
|
||||
|
||||
// rows, err := db.pool.Query(ctx, `
|
||||
// select a.id, a.family_member_id, fm.name, a.title, a.activity_type,
|
||||
// a.day_of_week, a.start_time::text, a.end_time::text,
|
||||
// a.start_date, a.end_date, a.location, a.notes, a.created_at
|
||||
// from activities a
|
||||
// left join family_members fm on fm.id = a.family_member_id
|
||||
// where (a.start_date >= $1 and a.start_date < $2)
|
||||
// or (a.day_of_week is not null and (a.end_date is null or a.end_date >= $1))
|
||||
// order by a.start_date, a.start_time
|
||||
// `, weekStart, weekEnd)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("get week schedule: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select a.id, a.family_member_id, fm.name, a.title, a.activity_type,
|
||||
a.day_of_week, a.start_time::text, a.end_time::text,
|
||||
a.start_date, a.end_date, a.location, a.notes, a.created_at
|
||||
from activities a
|
||||
left join family_members fm on fm.id = a.family_member_id
|
||||
where (a.start_date >= $1 and a.start_date < $2)
|
||||
or (a.day_of_week is not null and (a.end_date is null or a.end_date >= $1))
|
||||
order by a.start_date, a.start_time
|
||||
`, weekStart, weekEnd)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("get week schedule: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// return scanActivities(rows)
|
||||
// }
|
||||
return scanActivities(rows)
|
||||
}
|
||||
|
||||
// func (db *DB) SearchActivities(ctx context.Context, query, activityType string, memberID *uuid.UUID) ([]ext.Activity, error) {
|
||||
// args := []any{}
|
||||
// conditions := []string{}
|
||||
func (db *DB) SearchActivities(ctx context.Context, query, activityType string, memberID *uuid.UUID) ([]ext.Activity, error) {
|
||||
args := []any{}
|
||||
conditions := []string{}
|
||||
|
||||
// if q := strings.TrimSpace(query); q != "" {
|
||||
// args = append(args, "%"+q+"%")
|
||||
// conditions = append(conditions, fmt.Sprintf("(a.title ILIKE $%d OR a.notes ILIKE $%d)", len(args), len(args)))
|
||||
// }
|
||||
// if t := strings.TrimSpace(activityType); t != "" {
|
||||
// args = append(args, t)
|
||||
// conditions = append(conditions, fmt.Sprintf("a.activity_type = $%d", len(args)))
|
||||
// }
|
||||
// if memberID != nil {
|
||||
// args = append(args, *memberID)
|
||||
// conditions = append(conditions, fmt.Sprintf("a.family_member_id = $%d", len(args)))
|
||||
// }
|
||||
if q := strings.TrimSpace(query); q != "" {
|
||||
args = append(args, "%"+q+"%")
|
||||
conditions = append(conditions, fmt.Sprintf("(a.title ILIKE $%d OR a.notes ILIKE $%d)", len(args), len(args)))
|
||||
}
|
||||
if t := strings.TrimSpace(activityType); t != "" {
|
||||
args = append(args, t)
|
||||
conditions = append(conditions, fmt.Sprintf("a.activity_type = $%d", len(args)))
|
||||
}
|
||||
if memberID != nil {
|
||||
args = append(args, *memberID)
|
||||
conditions = append(conditions, fmt.Sprintf("a.family_member_id = $%d", len(args)))
|
||||
}
|
||||
|
||||
// q := `
|
||||
// select a.id, a.family_member_id, fm.name, a.title, a.activity_type,
|
||||
// a.day_of_week, a.start_time::text, a.end_time::text,
|
||||
// a.start_date, a.end_date, a.location, a.notes, a.created_at
|
||||
// from activities a
|
||||
// left join family_members fm on fm.id = a.family_member_id
|
||||
// `
|
||||
// if len(conditions) > 0 {
|
||||
// q += " where " + strings.Join(conditions, " and ")
|
||||
// }
|
||||
// q += " order by a.start_date, a.start_time"
|
||||
q := `
|
||||
select a.id, a.family_member_id, fm.name, a.title, a.activity_type,
|
||||
a.day_of_week, a.start_time::text, a.end_time::text,
|
||||
a.start_date, a.end_date, a.location, a.notes, a.created_at
|
||||
from activities a
|
||||
left join family_members fm on fm.id = a.family_member_id
|
||||
`
|
||||
if len(conditions) > 0 {
|
||||
q += " where " + strings.Join(conditions, " and ")
|
||||
}
|
||||
q += " order by a.start_date, a.start_time"
|
||||
|
||||
// rows, err := db.pool.Query(ctx, q, args...)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("search activities: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
rows, err := db.pool.Query(ctx, q, args...)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("search activities: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// return scanActivities(rows)
|
||||
// }
|
||||
return scanActivities(rows)
|
||||
}
|
||||
|
||||
// func (db *DB) AddImportantDate(ctx context.Context, d ext.ImportantDate) (ext.ImportantDate, error) {
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into important_dates (family_member_id, title, date_value, recurring_yearly, reminder_days_before, notes)
|
||||
// values ($1, $2, $3, $4, $5, $6)
|
||||
// returning id, created_at
|
||||
// `, d.FamilyMemberID, d.Title, d.DateValue, d.RecurringYearly, d.ReminderDaysBefore, nullStr(d.Notes))
|
||||
func (db *DB) AddImportantDate(ctx context.Context, d ext.ImportantDate) (ext.ImportantDate, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into important_dates (family_member_id, title, date_value, recurring_yearly, reminder_days_before, notes)
|
||||
values ($1, $2, $3, $4, $5, $6)
|
||||
returning id, created_at
|
||||
`, d.FamilyMemberID, d.Title, d.DateValue, d.RecurringYearly, d.ReminderDaysBefore, nullStr(d.Notes))
|
||||
|
||||
// created := d
|
||||
// var model generatedmodels.ModelPublicImportantDates
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt); err != nil {
|
||||
// return ext.ImportantDate{}, fmt.Errorf("insert important date: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// created.CreatedAt = model.CreatedAt.Time()
|
||||
// return created, nil
|
||||
// }
|
||||
created := d
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt); err != nil {
|
||||
return ext.ImportantDate{}, fmt.Errorf("insert important date: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) GetUpcomingDates(ctx context.Context, daysAhead int) ([]ext.ImportantDate, error) {
|
||||
// if daysAhead <= 0 {
|
||||
// daysAhead = 30
|
||||
// }
|
||||
// now := time.Now()
|
||||
// cutoff := now.AddDate(0, 0, daysAhead)
|
||||
func (db *DB) GetUpcomingDates(ctx context.Context, daysAhead int) ([]ext.ImportantDate, error) {
|
||||
if daysAhead <= 0 {
|
||||
daysAhead = 30
|
||||
}
|
||||
now := time.Now()
|
||||
cutoff := now.AddDate(0, 0, daysAhead)
|
||||
|
||||
// // For yearly recurring events, check if this year's occurrence falls in range
|
||||
// rows, err := db.pool.Query(ctx, `
|
||||
// select d.id, d.family_member_id, fm.name, d.title, d.date_value,
|
||||
// d.recurring_yearly, d.reminder_days_before, d.notes, d.created_at
|
||||
// from important_dates d
|
||||
// left join family_members fm on fm.id = d.family_member_id
|
||||
// where (
|
||||
// (d.recurring_yearly = false and d.date_value between $1 and $2)
|
||||
// or
|
||||
// (d.recurring_yearly = true and
|
||||
// make_date(extract(year from now())::int, extract(month from d.date_value)::int, extract(day from d.date_value)::int)
|
||||
// between $1 and $2)
|
||||
// )
|
||||
// order by d.date_value
|
||||
// `, now, cutoff)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("get upcoming dates: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
// For yearly recurring events, check if this year's occurrence falls in range
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select d.id, d.family_member_id, fm.name, d.title, d.date_value,
|
||||
d.recurring_yearly, d.reminder_days_before, d.notes, d.created_at
|
||||
from important_dates d
|
||||
left join family_members fm on fm.id = d.family_member_id
|
||||
where (
|
||||
(d.recurring_yearly = false and d.date_value between $1 and $2)
|
||||
or
|
||||
(d.recurring_yearly = true and
|
||||
make_date(extract(year from now())::int, extract(month from d.date_value)::int, extract(day from d.date_value)::int)
|
||||
between $1 and $2)
|
||||
)
|
||||
order by d.date_value
|
||||
`, now, cutoff)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("get upcoming dates: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// var dates []ext.ImportantDate
|
||||
// for rows.Next() {
|
||||
// var model generatedmodels.ModelPublicImportantDates
|
||||
// var memberName *string
|
||||
// if err := rows.Scan(&model.ID, &model.FamilyMemberID, &memberName, &model.Title, &model.DateValue,
|
||||
// &model.RecurringYearly, &model.ReminderDaysBefore, &model.Notes, &model.CreatedAt); err != nil {
|
||||
// return nil, fmt.Errorf("scan important date: %w", err)
|
||||
// }
|
||||
// dates = append(dates, importantDateFromModel(model, strVal(memberName)))
|
||||
// }
|
||||
// return dates, rows.Err()
|
||||
// }
|
||||
var dates []ext.ImportantDate
|
||||
for rows.Next() {
|
||||
var d ext.ImportantDate
|
||||
var memberID *uuid.UUID
|
||||
var memberName, notes *string
|
||||
if err := rows.Scan(&d.ID, &memberID, &memberName, &d.Title, &d.DateValue,
|
||||
&d.RecurringYearly, &d.ReminderDaysBefore, ¬es, &d.CreatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan important date: %w", err)
|
||||
}
|
||||
d.FamilyMemberID = memberID
|
||||
d.MemberName = strVal(memberName)
|
||||
d.Notes = strVal(notes)
|
||||
dates = append(dates, d)
|
||||
}
|
||||
return dates, rows.Err()
|
||||
}
|
||||
|
||||
// func scanActivities(rows interface {
|
||||
// Next() bool
|
||||
// Scan(...any) error
|
||||
// Err() error
|
||||
// Close()
|
||||
// }) ([]ext.Activity, error) {
|
||||
// defer rows.Close()
|
||||
// var activities []ext.Activity
|
||||
// for rows.Next() {
|
||||
// var model generatedmodels.ModelPublicActivities
|
||||
// var memberName *string
|
||||
// if err := rows.Scan(
|
||||
// &model.ID, &model.FamilyMemberID, &memberName, &model.Title, &model.ActivityType,
|
||||
// &model.DayOfWeek, &model.StartTime, &model.EndTime,
|
||||
// &model.StartDate, &model.EndDate, &model.Location, &model.Notes, &model.CreatedAt,
|
||||
// ); err != nil {
|
||||
// return nil, fmt.Errorf("scan activity: %w", err)
|
||||
// }
|
||||
// activities = append(activities, activityFromModel(model, strVal(memberName)))
|
||||
// }
|
||||
// return activities, rows.Err()
|
||||
// }
|
||||
func scanActivities(rows interface {
|
||||
Next() bool
|
||||
Scan(...any) error
|
||||
Err() error
|
||||
Close()
|
||||
}) ([]ext.Activity, error) {
|
||||
defer rows.Close()
|
||||
var activities []ext.Activity
|
||||
for rows.Next() {
|
||||
var a ext.Activity
|
||||
var memberName, activityType, dayOfWeek, startTime, endTime, location, notes *string
|
||||
if err := rows.Scan(
|
||||
&a.ID, &a.FamilyMemberID, &memberName, &a.Title, &activityType,
|
||||
&dayOfWeek, &startTime, &endTime,
|
||||
&a.StartDate, &a.EndDate, &location, ¬es, &a.CreatedAt,
|
||||
); err != nil {
|
||||
return nil, fmt.Errorf("scan activity: %w", err)
|
||||
}
|
||||
a.MemberName = strVal(memberName)
|
||||
a.ActivityType = strVal(activityType)
|
||||
a.DayOfWeek = strVal(dayOfWeek)
|
||||
a.StartTime = strVal(startTime)
|
||||
a.EndTime = strVal(endTime)
|
||||
a.Location = strVal(location)
|
||||
a.Notes = strVal(notes)
|
||||
activities = append(activities, a)
|
||||
}
|
||||
return activities, rows.Err()
|
||||
}
|
||||
|
||||
@@ -7,9 +7,9 @@ import (
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/google/uuid"
|
||||
"github.com/jackc/pgx/v5"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
@@ -27,25 +27,20 @@ func (db *DB) SaveChatHistory(ctx context.Context, h ext.ChatHistory) (ext.ChatH
|
||||
insert into chat_histories
|
||||
(session_id, title, channel, agent_id, project_id, messages, summary, metadata)
|
||||
values ($1, $2, $3, $4, $5, $6, $7, $8)
|
||||
returning id, guid, created_at, updated_at
|
||||
returning id, created_at, updated_at
|
||||
`,
|
||||
h.SessionID, nullStr(h.Title), nullStr(h.Channel), nullStr(h.AgentID),
|
||||
h.ProjectID, messages, nullStr(h.Summary), meta,
|
||||
)
|
||||
|
||||
created := h
|
||||
var model generatedmodels.ModelPublicChatHistories
|
||||
if err := row.Scan(&model.ID, &model.GUID, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt, &created.UpdatedAt); err != nil {
|
||||
return ext.ChatHistory{}, fmt.Errorf("insert chat history: %w", err)
|
||||
}
|
||||
created.ID = model.ID.Int64()
|
||||
created.GUID = model.GUID.UUID()
|
||||
created.CreatedAt = model.CreatedAt.Time()
|
||||
created.UpdatedAt = model.UpdatedAt.Time()
|
||||
return created, nil
|
||||
}
|
||||
|
||||
func (db *DB) GetChatHistory(ctx context.Context, id int64) (ext.ChatHistory, bool, error) {
|
||||
func (db *DB) GetChatHistory(ctx context.Context, id uuid.UUID) (ext.ChatHistory, bool, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, session_id, title, channel, agent_id, project_id,
|
||||
messages, summary, metadata, created_at, updated_at
|
||||
@@ -81,7 +76,7 @@ func (db *DB) GetChatHistoryBySessionID(ctx context.Context, sessionID string) (
|
||||
}
|
||||
|
||||
type ListChatHistoriesFilter struct {
|
||||
ProjectID *int64
|
||||
ProjectID *uuid.UUID
|
||||
Channel string
|
||||
AgentID string
|
||||
SessionID string
|
||||
@@ -149,7 +144,7 @@ func (db *DB) ListChatHistories(ctx context.Context, f ListChatHistoriesFilter)
|
||||
return result, rows.Err()
|
||||
}
|
||||
|
||||
func (db *DB) DeleteChatHistory(ctx context.Context, id int64) (bool, error) {
|
||||
func (db *DB) DeleteChatHistory(ctx context.Context, id uuid.UUID) (bool, error) {
|
||||
tag, err := db.pool.Exec(ctx, `delete from chat_histories where id = $1`, id)
|
||||
if err != nil {
|
||||
return false, fmt.Errorf("delete chat history: %w", err)
|
||||
@@ -162,34 +157,26 @@ type rowScanner interface {
|
||||
}
|
||||
|
||||
func scanChatHistory(row rowScanner) (ext.ChatHistory, error) {
|
||||
var model generatedmodels.ModelPublicChatHistories
|
||||
var h ext.ChatHistory
|
||||
var title, channel, agentID, summary *string
|
||||
var messagesJSON, metaJSON []byte
|
||||
|
||||
if err := row.Scan(
|
||||
&model.ID, &model.SessionID, &model.Title, &model.Channel, &model.AgentID, &model.ProjectID,
|
||||
&model.Messages, &model.Summary, &model.Metadata, &model.CreatedAt, &model.UpdatedAt,
|
||||
&h.ID, &h.SessionID, &title, &channel, &agentID, &h.ProjectID,
|
||||
&messagesJSON, &summary, &metaJSON, &h.CreatedAt, &h.UpdatedAt,
|
||||
); err != nil {
|
||||
return ext.ChatHistory{}, err
|
||||
}
|
||||
|
||||
h := ext.ChatHistory{
|
||||
ID: model.ID.Int64(),
|
||||
GUID: model.GUID.UUID(),
|
||||
SessionID: model.SessionID.String(),
|
||||
Title: model.Title.String(),
|
||||
Channel: model.Channel.String(),
|
||||
AgentID: model.AgentID.String(),
|
||||
Summary: model.Summary.String(),
|
||||
CreatedAt: model.CreatedAt.Time(),
|
||||
UpdatedAt: model.UpdatedAt.Time(),
|
||||
}
|
||||
if model.ProjectID.Valid {
|
||||
id := model.ProjectID.Int64()
|
||||
h.ProjectID = &id
|
||||
}
|
||||
h.Title = strVal(title)
|
||||
h.Channel = strVal(channel)
|
||||
h.AgentID = strVal(agentID)
|
||||
h.Summary = strVal(summary)
|
||||
|
||||
if err := json.Unmarshal(model.Messages, &h.Messages); err != nil {
|
||||
if err := json.Unmarshal(messagesJSON, &h.Messages); err != nil {
|
||||
return ext.ChatHistory{}, fmt.Errorf("unmarshal messages: %w", err)
|
||||
}
|
||||
if err := json.Unmarshal(model.Metadata, &h.Metadata); err != nil {
|
||||
if err := json.Unmarshal(metaJSON, &h.Metadata); err != nil {
|
||||
return ext.ChatHistory{}, fmt.Errorf("unmarshal metadata: %w", err)
|
||||
}
|
||||
if h.Messages == nil {
|
||||
|
||||
+217
-205
@@ -1,235 +1,247 @@
|
||||
package store
|
||||
|
||||
// import (
|
||||
// "context"
|
||||
// "fmt"
|
||||
// "strings"
|
||||
// "time"
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
// "github.com/google/uuid"
|
||||
"github.com/google/uuid"
|
||||
|
||||
// "git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
// ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
// )
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
// func (db *DB) AddProfessionalContact(ctx context.Context, c ext.ProfessionalContact) (ext.ProfessionalContact, error) {
|
||||
// if c.Tags == nil {
|
||||
// c.Tags = []string{}
|
||||
// }
|
||||
func (db *DB) AddProfessionalContact(ctx context.Context, c ext.ProfessionalContact) (ext.ProfessionalContact, error) {
|
||||
if c.Tags == nil {
|
||||
c.Tags = []string{}
|
||||
}
|
||||
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into professional_contacts (name, company, title, email, phone, linkedin_url, how_we_met, tags, notes, follow_up_date)
|
||||
// values ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10)
|
||||
// returning id, created_at, updated_at
|
||||
// `, c.Name, nullStr(c.Company), nullStr(c.Title), nullStr(c.Email), nullStr(c.Phone),
|
||||
// nullStr(c.LinkedInURL), nullStr(c.HowWeMet), c.Tags, nullStr(c.Notes), c.FollowUpDate)
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into professional_contacts (name, company, title, email, phone, linkedin_url, how_we_met, tags, notes, follow_up_date)
|
||||
values ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10)
|
||||
returning id, created_at, updated_at
|
||||
`, c.Name, nullStr(c.Company), nullStr(c.Title), nullStr(c.Email), nullStr(c.Phone),
|
||||
nullStr(c.LinkedInURL), nullStr(c.HowWeMet), c.Tags, nullStr(c.Notes), c.FollowUpDate)
|
||||
|
||||
// created := c
|
||||
// var model generatedmodels.ModelPublicProfessionalContacts
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.ProfessionalContact{}, fmt.Errorf("insert contact: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// created.CreatedAt = model.CreatedAt.Time()
|
||||
// created.UpdatedAt = model.UpdatedAt.Time()
|
||||
// return created, nil
|
||||
// }
|
||||
created := c
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt, &created.UpdatedAt); err != nil {
|
||||
return ext.ProfessionalContact{}, fmt.Errorf("insert contact: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) SearchContacts(ctx context.Context, query string, tags []string) ([]ext.ProfessionalContact, error) {
|
||||
// args := []any{}
|
||||
// conditions := []string{}
|
||||
func (db *DB) SearchContacts(ctx context.Context, query string, tags []string) ([]ext.ProfessionalContact, error) {
|
||||
args := []any{}
|
||||
conditions := []string{}
|
||||
|
||||
// if q := strings.TrimSpace(query); q != "" {
|
||||
// args = append(args, "%"+q+"%")
|
||||
// idx := len(args)
|
||||
// conditions = append(conditions, fmt.Sprintf(
|
||||
// "(name ILIKE $%[1]d OR company ILIKE $%[1]d OR title ILIKE $%[1]d OR notes ILIKE $%[1]d)", idx))
|
||||
// }
|
||||
// if len(tags) > 0 {
|
||||
// args = append(args, tags)
|
||||
// conditions = append(conditions, fmt.Sprintf("tags @> $%d", len(args)))
|
||||
// }
|
||||
if q := strings.TrimSpace(query); q != "" {
|
||||
args = append(args, "%"+q+"%")
|
||||
idx := len(args)
|
||||
conditions = append(conditions, fmt.Sprintf(
|
||||
"(name ILIKE $%[1]d OR company ILIKE $%[1]d OR title ILIKE $%[1]d OR notes ILIKE $%[1]d)", idx))
|
||||
}
|
||||
if len(tags) > 0 {
|
||||
args = append(args, tags)
|
||||
conditions = append(conditions, fmt.Sprintf("tags @> $%d", len(args)))
|
||||
}
|
||||
|
||||
// q := `select id, name, company, title, email, phone, linkedin_url, how_we_met, tags::text[], notes, last_contacted, follow_up_date, created_at, updated_at from professional_contacts`
|
||||
// if len(conditions) > 0 {
|
||||
// q += " where " + strings.Join(conditions, " and ")
|
||||
// }
|
||||
// q += " order by name"
|
||||
q := `select id, name, company, title, email, phone, linkedin_url, how_we_met, tags, notes, last_contacted, follow_up_date, created_at, updated_at from professional_contacts`
|
||||
if len(conditions) > 0 {
|
||||
q += " where " + strings.Join(conditions, " and ")
|
||||
}
|
||||
q += " order by name"
|
||||
|
||||
// rows, err := db.pool.Query(ctx, q, args...)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("search contacts: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
rows, err := db.pool.Query(ctx, q, args...)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("search contacts: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// return scanContacts(rows)
|
||||
// }
|
||||
return scanContacts(rows)
|
||||
}
|
||||
|
||||
// func (db *DB) GetContact(ctx context.Context, id uuid.UUID) (ext.ProfessionalContact, error) {
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// select id, name, company, title, email, phone, linkedin_url, how_we_met, tags::text[], notes, last_contacted, follow_up_date, created_at, updated_at
|
||||
// from professional_contacts where id = $1
|
||||
// `, id)
|
||||
func (db *DB) GetContact(ctx context.Context, id uuid.UUID) (ext.ProfessionalContact, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, name, company, title, email, phone, linkedin_url, how_we_met, tags, notes, last_contacted, follow_up_date, created_at, updated_at
|
||||
from professional_contacts where id = $1
|
||||
`, id)
|
||||
|
||||
// var model generatedmodels.ModelPublicProfessionalContacts
|
||||
// var tags []string
|
||||
// if err := row.Scan(&model.ID, &model.Name, &model.Company, &model.Title, &model.Email, &model.Phone,
|
||||
// &model.LinkedinURL, &model.HowWeMet, &tags, &model.Notes, &model.LastContacted, &model.FollowUpDate,
|
||||
// &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.ProfessionalContact{}, fmt.Errorf("get contact: %w", err)
|
||||
// }
|
||||
// c := professionalContactFromModel(model, tags)
|
||||
// return c, nil
|
||||
// }
|
||||
var c ext.ProfessionalContact
|
||||
var company, title, email, phone, linkedInURL, howWeMet, notes *string
|
||||
if err := row.Scan(&c.ID, &c.Name, &company, &title, &email, &phone,
|
||||
&linkedInURL, &howWeMet, &c.Tags, ¬es, &c.LastContacted, &c.FollowUpDate,
|
||||
&c.CreatedAt, &c.UpdatedAt); err != nil {
|
||||
return ext.ProfessionalContact{}, fmt.Errorf("get contact: %w", err)
|
||||
}
|
||||
c.Company = strVal(company)
|
||||
c.Title = strVal(title)
|
||||
c.Email = strVal(email)
|
||||
c.Phone = strVal(phone)
|
||||
c.LinkedInURL = strVal(linkedInURL)
|
||||
c.HowWeMet = strVal(howWeMet)
|
||||
c.Notes = strVal(notes)
|
||||
if c.Tags == nil {
|
||||
c.Tags = []string{}
|
||||
}
|
||||
return c, nil
|
||||
}
|
||||
|
||||
// func (db *DB) LogInteraction(ctx context.Context, interaction ext.ContactInteraction) (ext.ContactInteraction, error) {
|
||||
// occurredAt := interaction.OccurredAt
|
||||
// if occurredAt.IsZero() {
|
||||
// occurredAt = time.Now()
|
||||
// }
|
||||
func (db *DB) LogInteraction(ctx context.Context, interaction ext.ContactInteraction) (ext.ContactInteraction, error) {
|
||||
occurredAt := interaction.OccurredAt
|
||||
if occurredAt.IsZero() {
|
||||
occurredAt = time.Now()
|
||||
}
|
||||
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into contact_interactions (contact_id, interaction_type, occurred_at, summary, follow_up_needed, follow_up_notes)
|
||||
// values ($1, $2, $3, $4, $5, $6)
|
||||
// returning id, created_at
|
||||
// `, interaction.ContactID, interaction.InteractionType, occurredAt, interaction.Summary,
|
||||
// interaction.FollowUpNeeded, nullStr(interaction.FollowUpNotes))
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into contact_interactions (contact_id, interaction_type, occurred_at, summary, follow_up_needed, follow_up_notes)
|
||||
values ($1, $2, $3, $4, $5, $6)
|
||||
returning id, created_at
|
||||
`, interaction.ContactID, interaction.InteractionType, occurredAt, interaction.Summary,
|
||||
interaction.FollowUpNeeded, nullStr(interaction.FollowUpNotes))
|
||||
|
||||
// created := interaction
|
||||
// created.OccurredAt = occurredAt
|
||||
// var model generatedmodels.ModelPublicContactInteractions
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt); err != nil {
|
||||
// return ext.ContactInteraction{}, fmt.Errorf("insert interaction: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// created.CreatedAt = model.CreatedAt.Time()
|
||||
// return created, nil
|
||||
// }
|
||||
created := interaction
|
||||
created.OccurredAt = occurredAt
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt); err != nil {
|
||||
return ext.ContactInteraction{}, fmt.Errorf("insert interaction: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) GetContactHistory(ctx context.Context, contactID uuid.UUID) (ext.ContactHistory, error) {
|
||||
// contact, err := db.GetContact(ctx, contactID)
|
||||
// if err != nil {
|
||||
// return ext.ContactHistory{}, err
|
||||
// }
|
||||
func (db *DB) GetContactHistory(ctx context.Context, contactID uuid.UUID) (ext.ContactHistory, error) {
|
||||
contact, err := db.GetContact(ctx, contactID)
|
||||
if err != nil {
|
||||
return ext.ContactHistory{}, err
|
||||
}
|
||||
|
||||
// rows, err := db.pool.Query(ctx, `
|
||||
// select id, contact_id, interaction_type, occurred_at, summary, follow_up_needed, follow_up_notes, created_at
|
||||
// from contact_interactions where contact_id = $1 order by occurred_at desc
|
||||
// `, contactID)
|
||||
// if err != nil {
|
||||
// return ext.ContactHistory{}, fmt.Errorf("get interactions: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select id, contact_id, interaction_type, occurred_at, summary, follow_up_needed, follow_up_notes, created_at
|
||||
from contact_interactions where contact_id = $1 order by occurred_at desc
|
||||
`, contactID)
|
||||
if err != nil {
|
||||
return ext.ContactHistory{}, fmt.Errorf("get interactions: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// var interactions []ext.ContactInteraction
|
||||
// for rows.Next() {
|
||||
// var model generatedmodels.ModelPublicContactInteractions
|
||||
// if err := rows.Scan(&model.ID, &model.ContactID, &model.InteractionType, &model.OccurredAt, &model.Summary,
|
||||
// &model.FollowUpNeeded, &model.FollowUpNotes, &model.CreatedAt); err != nil {
|
||||
// return ext.ContactHistory{}, fmt.Errorf("scan interaction: %w", err)
|
||||
// }
|
||||
// interactions = append(interactions, contactInteractionFromModel(model))
|
||||
// }
|
||||
// if err := rows.Err(); err != nil {
|
||||
// return ext.ContactHistory{}, err
|
||||
// }
|
||||
var interactions []ext.ContactInteraction
|
||||
for rows.Next() {
|
||||
var i ext.ContactInteraction
|
||||
var followUpNotes *string
|
||||
if err := rows.Scan(&i.ID, &i.ContactID, &i.InteractionType, &i.OccurredAt, &i.Summary,
|
||||
&i.FollowUpNeeded, &followUpNotes, &i.CreatedAt); err != nil {
|
||||
return ext.ContactHistory{}, fmt.Errorf("scan interaction: %w", err)
|
||||
}
|
||||
i.FollowUpNotes = strVal(followUpNotes)
|
||||
interactions = append(interactions, i)
|
||||
}
|
||||
if err := rows.Err(); err != nil {
|
||||
return ext.ContactHistory{}, err
|
||||
}
|
||||
|
||||
// oppRows, err := db.pool.Query(ctx, `
|
||||
// select id, contact_id, title, description, stage, value, expected_close_date, notes, created_at, updated_at
|
||||
// from opportunities where contact_id = $1 order by created_at desc
|
||||
// `, contactID)
|
||||
// if err != nil {
|
||||
// return ext.ContactHistory{}, fmt.Errorf("get opportunities: %w", err)
|
||||
// }
|
||||
// defer oppRows.Close()
|
||||
oppRows, err := db.pool.Query(ctx, `
|
||||
select id, contact_id, title, description, stage, value, expected_close_date, notes, created_at, updated_at
|
||||
from opportunities where contact_id = $1 order by created_at desc
|
||||
`, contactID)
|
||||
if err != nil {
|
||||
return ext.ContactHistory{}, fmt.Errorf("get opportunities: %w", err)
|
||||
}
|
||||
defer oppRows.Close()
|
||||
|
||||
// var opportunities []ext.Opportunity
|
||||
// for oppRows.Next() {
|
||||
// var model generatedmodels.ModelPublicOpportunities
|
||||
// if err := oppRows.Scan(&model.ID, &model.ContactID, &model.Title, &model.Description, &model.Stage, &model.Value,
|
||||
// &model.ExpectedCloseDate, &model.Notes, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.ContactHistory{}, fmt.Errorf("scan opportunity: %w", err)
|
||||
// }
|
||||
// opportunities = append(opportunities, opportunityFromModel(model))
|
||||
// }
|
||||
// if err := oppRows.Err(); err != nil {
|
||||
// return ext.ContactHistory{}, err
|
||||
// }
|
||||
var opportunities []ext.Opportunity
|
||||
for oppRows.Next() {
|
||||
var o ext.Opportunity
|
||||
var description, notes *string
|
||||
if err := oppRows.Scan(&o.ID, &o.ContactID, &o.Title, &description, &o.Stage, &o.Value,
|
||||
&o.ExpectedCloseDate, ¬es, &o.CreatedAt, &o.UpdatedAt); err != nil {
|
||||
return ext.ContactHistory{}, fmt.Errorf("scan opportunity: %w", err)
|
||||
}
|
||||
o.Description = strVal(description)
|
||||
o.Notes = strVal(notes)
|
||||
opportunities = append(opportunities, o)
|
||||
}
|
||||
if err := oppRows.Err(); err != nil {
|
||||
return ext.ContactHistory{}, err
|
||||
}
|
||||
|
||||
// return ext.ContactHistory{
|
||||
// Contact: contact,
|
||||
// Interactions: interactions,
|
||||
// Opportunities: opportunities,
|
||||
// }, nil
|
||||
// }
|
||||
return ext.ContactHistory{
|
||||
Contact: contact,
|
||||
Interactions: interactions,
|
||||
Opportunities: opportunities,
|
||||
}, nil
|
||||
}
|
||||
|
||||
// func (db *DB) CreateOpportunity(ctx context.Context, o ext.Opportunity) (ext.Opportunity, error) {
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into opportunities (contact_id, title, description, stage, value, expected_close_date, notes)
|
||||
// values ($1, $2, $3, $4, $5, $6, $7)
|
||||
// returning id, created_at, updated_at
|
||||
// `, o.ContactID, o.Title, nullStr(o.Description), o.Stage, o.Value, o.ExpectedCloseDate, nullStr(o.Notes))
|
||||
func (db *DB) CreateOpportunity(ctx context.Context, o ext.Opportunity) (ext.Opportunity, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into opportunities (contact_id, title, description, stage, value, expected_close_date, notes)
|
||||
values ($1, $2, $3, $4, $5, $6, $7)
|
||||
returning id, created_at, updated_at
|
||||
`, o.ContactID, o.Title, nullStr(o.Description), o.Stage, o.Value, o.ExpectedCloseDate, nullStr(o.Notes))
|
||||
|
||||
// created := o
|
||||
// var model generatedmodels.ModelPublicOpportunities
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.Opportunity{}, fmt.Errorf("insert opportunity: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// created.CreatedAt = model.CreatedAt.Time()
|
||||
// created.UpdatedAt = model.UpdatedAt.Time()
|
||||
// return created, nil
|
||||
// }
|
||||
created := o
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt, &created.UpdatedAt); err != nil {
|
||||
return ext.Opportunity{}, fmt.Errorf("insert opportunity: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) GetFollowUpsDue(ctx context.Context, daysAhead int) ([]ext.ProfessionalContact, error) {
|
||||
// if daysAhead <= 0 {
|
||||
// daysAhead = 7
|
||||
// }
|
||||
// cutoff := time.Now().AddDate(0, 0, daysAhead)
|
||||
func (db *DB) GetFollowUpsDue(ctx context.Context, daysAhead int) ([]ext.ProfessionalContact, error) {
|
||||
if daysAhead <= 0 {
|
||||
daysAhead = 7
|
||||
}
|
||||
cutoff := time.Now().AddDate(0, 0, daysAhead)
|
||||
|
||||
// rows, err := db.pool.Query(ctx, `
|
||||
// select id, name, company, title, email, phone, linkedin_url, how_we_met, tags::text[], notes, last_contacted, follow_up_date, created_at, updated_at
|
||||
// from professional_contacts
|
||||
// where follow_up_date <= $1
|
||||
// order by follow_up_date asc
|
||||
// `, cutoff)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("get follow-ups: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select id, name, company, title, email, phone, linkedin_url, how_we_met, tags, notes, last_contacted, follow_up_date, created_at, updated_at
|
||||
from professional_contacts
|
||||
where follow_up_date <= $1
|
||||
order by follow_up_date asc
|
||||
`, cutoff)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("get follow-ups: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// return scanContacts(rows)
|
||||
// }
|
||||
return scanContacts(rows)
|
||||
}
|
||||
|
||||
// func (db *DB) AppendThoughtToContactNotes(ctx context.Context, contactID uuid.UUID, thoughtContent string) error {
|
||||
// _, err := db.pool.Exec(ctx, `
|
||||
// update professional_contacts
|
||||
// set notes = coalesce(notes, '') || $2
|
||||
// where id = $1
|
||||
// `, contactID, thoughtContent)
|
||||
// if err != nil {
|
||||
// return fmt.Errorf("append thought to contact: %w", err)
|
||||
// }
|
||||
// return nil
|
||||
// }
|
||||
func (db *DB) AppendThoughtToContactNotes(ctx context.Context, contactID uuid.UUID, thoughtContent string) error {
|
||||
_, err := db.pool.Exec(ctx, `
|
||||
update professional_contacts
|
||||
set notes = coalesce(notes, '') || $2
|
||||
where id = $1
|
||||
`, contactID, thoughtContent)
|
||||
if err != nil {
|
||||
return fmt.Errorf("append thought to contact: %w", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// func scanContacts(rows interface {
|
||||
// Next() bool
|
||||
// Scan(...any) error
|
||||
// Err() error
|
||||
// Close()
|
||||
// }) ([]ext.ProfessionalContact, error) {
|
||||
// defer rows.Close()
|
||||
// var contacts []ext.ProfessionalContact
|
||||
// for rows.Next() {
|
||||
// var model generatedmodels.ModelPublicProfessionalContacts
|
||||
// var tags []string
|
||||
// if err := rows.Scan(&model.ID, &model.Name, &model.Company, &model.Title, &model.Email, &model.Phone,
|
||||
// &model.LinkedinURL, &model.HowWeMet, &tags, &model.Notes, &model.LastContacted, &model.FollowUpDate,
|
||||
// &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return nil, fmt.Errorf("scan contact: %w", err)
|
||||
// }
|
||||
// contacts = append(contacts, professionalContactFromModel(model, tags))
|
||||
// }
|
||||
// return contacts, rows.Err()
|
||||
// }
|
||||
func scanContacts(rows interface {
|
||||
Next() bool
|
||||
Scan(...any) error
|
||||
Err() error
|
||||
Close()
|
||||
}) ([]ext.ProfessionalContact, error) {
|
||||
defer rows.Close()
|
||||
var contacts []ext.ProfessionalContact
|
||||
for rows.Next() {
|
||||
var c ext.ProfessionalContact
|
||||
var company, title, email, phone, linkedInURL, howWeMet, notes *string
|
||||
if err := rows.Scan(&c.ID, &c.Name, &company, &title, &email, &phone,
|
||||
&linkedInURL, &howWeMet, &c.Tags, ¬es, &c.LastContacted, &c.FollowUpDate,
|
||||
&c.CreatedAt, &c.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan contact: %w", err)
|
||||
}
|
||||
c.Company = strVal(company)
|
||||
c.Title = strVal(title)
|
||||
c.Email = strVal(email)
|
||||
c.Phone = strVal(phone)
|
||||
c.LinkedInURL = strVal(linkedInURL)
|
||||
c.HowWeMet = strVal(howWeMet)
|
||||
c.Notes = strVal(notes)
|
||||
if c.Tags == nil {
|
||||
c.Tags = []string{}
|
||||
}
|
||||
contacts = append(contacts, c)
|
||||
}
|
||||
return contacts, rows.Err()
|
||||
}
|
||||
|
||||
+3
-39
@@ -2,23 +2,18 @@ package store
|
||||
|
||||
import (
|
||||
"context"
|
||||
"database/sql"
|
||||
"fmt"
|
||||
"time"
|
||||
|
||||
"github.com/jackc/pgx/v5"
|
||||
"github.com/jackc/pgx/v5/pgxpool"
|
||||
pgxvec "github.com/pgvector/pgvector-go/pgx"
|
||||
"github.com/uptrace/bun"
|
||||
"github.com/uptrace/bun/dialect/pgdialect"
|
||||
"github.com/uptrace/bun/driver/pgdriver"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/config"
|
||||
)
|
||||
|
||||
type DB struct {
|
||||
pool *pgxpool.Pool
|
||||
bun *bun.DB
|
||||
}
|
||||
|
||||
func New(ctx context.Context, cfg config.DatabaseConfig) (*DB, error) {
|
||||
@@ -40,20 +35,8 @@ func New(ctx context.Context, cfg config.DatabaseConfig) (*DB, error) {
|
||||
return nil, fmt.Errorf("create database pool: %w", err)
|
||||
}
|
||||
|
||||
bunSQLDB := sql.OpenDB(pgdriver.NewConnector(pgdriver.WithDSN(cfg.URL)))
|
||||
bunSQLDB.SetMaxOpenConns(int(cfg.MaxConns))
|
||||
bunSQLDB.SetMaxIdleConns(int(cfg.MinConns))
|
||||
bunSQLDB.SetConnMaxLifetime(cfg.MaxConnLifetime)
|
||||
bunSQLDB.SetConnMaxIdleTime(cfg.MaxConnIdleTime)
|
||||
|
||||
db := &DB{
|
||||
pool: pool,
|
||||
bun: bun.NewDB(bunSQLDB, pgdialect.New()),
|
||||
}
|
||||
db := &DB{pool: pool}
|
||||
if err := db.Ping(ctx); err != nil {
|
||||
if db.bun != nil {
|
||||
_ = db.bun.Close()
|
||||
}
|
||||
pool.Close()
|
||||
return nil, err
|
||||
}
|
||||
@@ -62,16 +45,11 @@ func New(ctx context.Context, cfg config.DatabaseConfig) (*DB, error) {
|
||||
}
|
||||
|
||||
func (db *DB) Close() {
|
||||
if db == nil {
|
||||
if db == nil || db.pool == nil {
|
||||
return
|
||||
}
|
||||
|
||||
if db.bun != nil {
|
||||
_ = db.bun.Close()
|
||||
}
|
||||
if db.pool != nil {
|
||||
db.pool.Close()
|
||||
}
|
||||
db.pool.Close()
|
||||
}
|
||||
|
||||
func (db *DB) Ping(ctx context.Context) error {
|
||||
@@ -124,17 +102,3 @@ func (db *DB) VerifyRequirements(ctx context.Context) error {
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) Bun() *bun.DB {
|
||||
if db == nil {
|
||||
return nil
|
||||
}
|
||||
return db.bun
|
||||
}
|
||||
|
||||
func (db *DB) Pool() *pgxpool.Pool {
|
||||
if db == nil {
|
||||
return nil
|
||||
}
|
||||
return db.pool
|
||||
}
|
||||
|
||||
+46
-50
@@ -9,7 +9,6 @@ import (
|
||||
"github.com/google/uuid"
|
||||
"github.com/jackc/pgx/v5"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
thoughttypes "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
@@ -17,52 +16,50 @@ func (db *DB) InsertStoredFile(ctx context.Context, file thoughttypes.StoredFile
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into stored_files (thought_id, project_id, name, media_type, kind, encoding, size_bytes, sha256, content)
|
||||
values ($1, $2, $3, $4, $5, $6, $7, $8, $9)
|
||||
returning id, guid, thought_id, project_id, name, media_type, kind, encoding, size_bytes, sha256, created_at, updated_at
|
||||
returning guid, thought_id, project_id, name, media_type, kind, encoding, size_bytes, sha256, created_at, updated_at
|
||||
`, file.ThoughtID, file.ProjectID, file.Name, file.MediaType, file.Kind, file.Encoding, file.SizeBytes, file.SHA256, file.Content)
|
||||
|
||||
var model generatedmodels.ModelPublicStoredFiles
|
||||
var created thoughttypes.StoredFile
|
||||
if err := row.Scan(
|
||||
&model.ID,
|
||||
&model.GUID,
|
||||
&model.ThoughtID,
|
||||
&model.ProjectID,
|
||||
&model.Name,
|
||||
&model.MediaType,
|
||||
&model.Kind,
|
||||
&model.Encoding,
|
||||
&model.SizeBytes,
|
||||
&model.Sha256,
|
||||
&model.CreatedAt,
|
||||
&model.UpdatedAt,
|
||||
&created.ID,
|
||||
&created.ThoughtID,
|
||||
&created.ProjectID,
|
||||
&created.Name,
|
||||
&created.MediaType,
|
||||
&created.Kind,
|
||||
&created.Encoding,
|
||||
&created.SizeBytes,
|
||||
&created.SHA256,
|
||||
&created.CreatedAt,
|
||||
&created.UpdatedAt,
|
||||
); err != nil {
|
||||
return thoughttypes.StoredFile{}, fmt.Errorf("insert stored file: %w", err)
|
||||
}
|
||||
|
||||
return storedFileFromModel(model), nil
|
||||
return created, nil
|
||||
}
|
||||
|
||||
func (db *DB) GetStoredFile(ctx context.Context, id uuid.UUID) (thoughttypes.StoredFile, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, guid, thought_id, project_id, name, media_type, kind, encoding, size_bytes, sha256, content, created_at, updated_at
|
||||
select guid, thought_id, project_id, name, media_type, kind, encoding, size_bytes, sha256, content, created_at, updated_at
|
||||
from stored_files
|
||||
where guid = $1
|
||||
`, id)
|
||||
|
||||
var model generatedmodels.ModelPublicStoredFiles
|
||||
var file thoughttypes.StoredFile
|
||||
if err := row.Scan(
|
||||
&model.ID,
|
||||
&model.GUID,
|
||||
&model.ThoughtID,
|
||||
&model.ProjectID,
|
||||
&model.Name,
|
||||
&model.MediaType,
|
||||
&model.Kind,
|
||||
&model.Encoding,
|
||||
&model.SizeBytes,
|
||||
&model.Sha256,
|
||||
&model.Content,
|
||||
&model.CreatedAt,
|
||||
&model.UpdatedAt,
|
||||
&file.ID,
|
||||
&file.ThoughtID,
|
||||
&file.ProjectID,
|
||||
&file.Name,
|
||||
&file.MediaType,
|
||||
&file.Kind,
|
||||
&file.Encoding,
|
||||
&file.SizeBytes,
|
||||
&file.SHA256,
|
||||
&file.Content,
|
||||
&file.CreatedAt,
|
||||
&file.UpdatedAt,
|
||||
); err != nil {
|
||||
if err == pgx.ErrNoRows {
|
||||
return thoughttypes.StoredFile{}, err
|
||||
@@ -70,7 +67,7 @@ func (db *DB) GetStoredFile(ctx context.Context, id uuid.UUID) (thoughttypes.Sto
|
||||
return thoughttypes.StoredFile{}, fmt.Errorf("get stored file: %w", err)
|
||||
}
|
||||
|
||||
return storedFileFromModel(model), nil
|
||||
return file, nil
|
||||
}
|
||||
|
||||
func (db *DB) ListStoredFiles(ctx context.Context, filter thoughttypes.StoredFileFilter) ([]thoughttypes.StoredFile, error) {
|
||||
@@ -91,7 +88,7 @@ func (db *DB) ListStoredFiles(ctx context.Context, filter thoughttypes.StoredFil
|
||||
}
|
||||
|
||||
query := `
|
||||
select id, guid, thought_id, project_id, name, media_type, kind, encoding, size_bytes, sha256, created_at, updated_at
|
||||
select guid, thought_id, project_id, name, media_type, kind, encoding, size_bytes, sha256, created_at, updated_at
|
||||
from stored_files
|
||||
`
|
||||
if len(conditions) > 0 {
|
||||
@@ -109,24 +106,23 @@ func (db *DB) ListStoredFiles(ctx context.Context, filter thoughttypes.StoredFil
|
||||
|
||||
files := make([]thoughttypes.StoredFile, 0, filter.Limit)
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicStoredFiles
|
||||
var file thoughttypes.StoredFile
|
||||
if err := rows.Scan(
|
||||
&model.ID,
|
||||
&model.GUID,
|
||||
&model.ThoughtID,
|
||||
&model.ProjectID,
|
||||
&model.Name,
|
||||
&model.MediaType,
|
||||
&model.Kind,
|
||||
&model.Encoding,
|
||||
&model.SizeBytes,
|
||||
&model.Sha256,
|
||||
&model.CreatedAt,
|
||||
&model.UpdatedAt,
|
||||
&file.ID,
|
||||
&file.ThoughtID,
|
||||
&file.ProjectID,
|
||||
&file.Name,
|
||||
&file.MediaType,
|
||||
&file.Kind,
|
||||
&file.Encoding,
|
||||
&file.SizeBytes,
|
||||
&file.SHA256,
|
||||
&file.CreatedAt,
|
||||
&file.UpdatedAt,
|
||||
); err != nil {
|
||||
return nil, fmt.Errorf("scan stored file: %w", err)
|
||||
}
|
||||
files = append(files, storedFileFromModel(model))
|
||||
files = append(files, file)
|
||||
}
|
||||
if err := rows.Err(); err != nil {
|
||||
return nil, fmt.Errorf("iterate stored files: %w", err)
|
||||
@@ -135,7 +131,7 @@ func (db *DB) ListStoredFiles(ctx context.Context, filter thoughttypes.StoredFil
|
||||
return files, nil
|
||||
}
|
||||
|
||||
func (db *DB) AddThoughtAttachment(ctx context.Context, thoughtID int64, attachment thoughttypes.ThoughtAttachment) error {
|
||||
func (db *DB) AddThoughtAttachment(ctx context.Context, thoughtID uuid.UUID, attachment thoughttypes.ThoughtAttachment) error {
|
||||
tx, err := db.pool.Begin(ctx)
|
||||
if err != nil {
|
||||
return fmt.Errorf("begin transaction: %w", err)
|
||||
@@ -145,7 +141,7 @@ func (db *DB) AddThoughtAttachment(ctx context.Context, thoughtID int64, attachm
|
||||
}()
|
||||
|
||||
var metadataBytes []byte
|
||||
if err := tx.QueryRow(ctx, `select metadata from thoughts where id = $1 for update`, thoughtID).Scan(&metadataBytes); err != nil {
|
||||
if err := tx.QueryRow(ctx, `select metadata from thoughts where guid = $1 for update`, thoughtID).Scan(&metadataBytes); err != nil {
|
||||
if err == pgx.ErrNoRows {
|
||||
return err
|
||||
}
|
||||
@@ -180,7 +176,7 @@ func (db *DB) AddThoughtAttachment(ctx context.Context, thoughtID int64, attachm
|
||||
update thoughts
|
||||
set metadata = $2::jsonb,
|
||||
updated_at = now()
|
||||
where id = $1
|
||||
where guid = $1
|
||||
`, thoughtID, updatedMetadata)
|
||||
if err != nil {
|
||||
return fmt.Errorf("update thought attachments: %w", err)
|
||||
|
||||
+131
-114
@@ -1,133 +1,150 @@
|
||||
package store
|
||||
|
||||
// import (
|
||||
// "context"
|
||||
// "encoding/json"
|
||||
// "fmt"
|
||||
// "strings"
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
// "github.com/google/uuid"
|
||||
"github.com/google/uuid"
|
||||
|
||||
// "git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
// ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
// )
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
// func (db *DB) AddHouseholdItem(ctx context.Context, item ext.HouseholdItem) (ext.HouseholdItem, error) {
|
||||
// details, err := json.Marshal(item.Details)
|
||||
// if err != nil {
|
||||
// return ext.HouseholdItem{}, fmt.Errorf("marshal details: %w", err)
|
||||
// }
|
||||
func (db *DB) AddHouseholdItem(ctx context.Context, item ext.HouseholdItem) (ext.HouseholdItem, error) {
|
||||
details, err := json.Marshal(item.Details)
|
||||
if err != nil {
|
||||
return ext.HouseholdItem{}, fmt.Errorf("marshal details: %w", err)
|
||||
}
|
||||
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into household_items (name, category, location, details, notes)
|
||||
// values ($1, $2, $3, $4::jsonb, $5)
|
||||
// returning id, created_at, updated_at
|
||||
// `, item.Name, nullStr(item.Category), nullStr(item.Location), details, nullStr(item.Notes))
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into household_items (name, category, location, details, notes)
|
||||
values ($1, $2, $3, $4::jsonb, $5)
|
||||
returning id, created_at, updated_at
|
||||
`, item.Name, nullStr(item.Category), nullStr(item.Location), details, nullStr(item.Notes))
|
||||
|
||||
// created := item
|
||||
// var model generatedmodels.ModelPublicHouseholdItems
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.HouseholdItem{}, fmt.Errorf("insert household item: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// created.CreatedAt = model.CreatedAt.Time()
|
||||
// created.UpdatedAt = model.UpdatedAt.Time()
|
||||
// return created, nil
|
||||
// }
|
||||
created := item
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt, &created.UpdatedAt); err != nil {
|
||||
return ext.HouseholdItem{}, fmt.Errorf("insert household item: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) SearchHouseholdItems(ctx context.Context, query, category, location string) ([]ext.HouseholdItem, error) {
|
||||
// args := []any{}
|
||||
// conditions := []string{}
|
||||
func (db *DB) SearchHouseholdItems(ctx context.Context, query, category, location string) ([]ext.HouseholdItem, error) {
|
||||
args := []any{}
|
||||
conditions := []string{}
|
||||
|
||||
// if q := strings.TrimSpace(query); q != "" {
|
||||
// args = append(args, "%"+q+"%")
|
||||
// conditions = append(conditions, fmt.Sprintf("(name ILIKE $%d OR notes ILIKE $%d)", len(args), len(args)))
|
||||
// }
|
||||
// if c := strings.TrimSpace(category); c != "" {
|
||||
// args = append(args, c)
|
||||
// conditions = append(conditions, fmt.Sprintf("category = $%d", len(args)))
|
||||
// }
|
||||
// if l := strings.TrimSpace(location); l != "" {
|
||||
// args = append(args, "%"+l+"%")
|
||||
// conditions = append(conditions, fmt.Sprintf("location ILIKE $%d", len(args)))
|
||||
// }
|
||||
if q := strings.TrimSpace(query); q != "" {
|
||||
args = append(args, "%"+q+"%")
|
||||
conditions = append(conditions, fmt.Sprintf("(name ILIKE $%d OR notes ILIKE $%d)", len(args), len(args)))
|
||||
}
|
||||
if c := strings.TrimSpace(category); c != "" {
|
||||
args = append(args, c)
|
||||
conditions = append(conditions, fmt.Sprintf("category = $%d", len(args)))
|
||||
}
|
||||
if l := strings.TrimSpace(location); l != "" {
|
||||
args = append(args, "%"+l+"%")
|
||||
conditions = append(conditions, fmt.Sprintf("location ILIKE $%d", len(args)))
|
||||
}
|
||||
|
||||
// q := `select id, name, category, location, details, notes, created_at, updated_at from household_items`
|
||||
// if len(conditions) > 0 {
|
||||
// q += " where " + strings.Join(conditions, " and ")
|
||||
// }
|
||||
// q += " order by name"
|
||||
q := `select id, name, category, location, details, notes, created_at, updated_at from household_items`
|
||||
if len(conditions) > 0 {
|
||||
q += " where " + strings.Join(conditions, " and ")
|
||||
}
|
||||
q += " order by name"
|
||||
|
||||
// rows, err := db.pool.Query(ctx, q, args...)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("search household items: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
rows, err := db.pool.Query(ctx, q, args...)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("search household items: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// var items []ext.HouseholdItem
|
||||
// for rows.Next() {
|
||||
// var model generatedmodels.ModelPublicHouseholdItems
|
||||
// if err := rows.Scan(&model.ID, &model.Name, &model.Category, &model.Location, &model.Details, &model.Notes, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return nil, fmt.Errorf("scan household item: %w", err)
|
||||
// }
|
||||
// items = append(items, householdItemFromModel(model))
|
||||
// }
|
||||
// return items, rows.Err()
|
||||
// }
|
||||
var items []ext.HouseholdItem
|
||||
for rows.Next() {
|
||||
var item ext.HouseholdItem
|
||||
var detailsBytes []byte
|
||||
var category, location, notes *string
|
||||
if err := rows.Scan(&item.ID, &item.Name, &category, &location, &detailsBytes, ¬es, &item.CreatedAt, &item.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan household item: %w", err)
|
||||
}
|
||||
item.Category = strVal(category)
|
||||
item.Location = strVal(location)
|
||||
item.Notes = strVal(notes)
|
||||
if err := json.Unmarshal(detailsBytes, &item.Details); err != nil {
|
||||
item.Details = map[string]any{}
|
||||
}
|
||||
items = append(items, item)
|
||||
}
|
||||
return items, rows.Err()
|
||||
}
|
||||
|
||||
// func (db *DB) GetHouseholdItem(ctx context.Context, id uuid.UUID) (ext.HouseholdItem, error) {
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// select id, name, category, location, details, notes, created_at, updated_at
|
||||
// from household_items where id = $1
|
||||
// `, id)
|
||||
func (db *DB) GetHouseholdItem(ctx context.Context, id uuid.UUID) (ext.HouseholdItem, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, name, category, location, details, notes, created_at, updated_at
|
||||
from household_items where id = $1
|
||||
`, id)
|
||||
|
||||
// var model generatedmodels.ModelPublicHouseholdItems
|
||||
// if err := row.Scan(&model.ID, &model.Name, &model.Category, &model.Location, &model.Details, &model.Notes, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.HouseholdItem{}, fmt.Errorf("get household item: %w", err)
|
||||
// }
|
||||
// return householdItemFromModel(model), nil
|
||||
// }
|
||||
var item ext.HouseholdItem
|
||||
var detailsBytes []byte
|
||||
var category, location, notes *string
|
||||
if err := row.Scan(&item.ID, &item.Name, &category, &location, &detailsBytes, ¬es, &item.CreatedAt, &item.UpdatedAt); err != nil {
|
||||
return ext.HouseholdItem{}, fmt.Errorf("get household item: %w", err)
|
||||
}
|
||||
item.Category = strVal(category)
|
||||
item.Location = strVal(location)
|
||||
item.Notes = strVal(notes)
|
||||
if err := json.Unmarshal(detailsBytes, &item.Details); err != nil {
|
||||
item.Details = map[string]any{}
|
||||
}
|
||||
return item, nil
|
||||
}
|
||||
|
||||
// func (db *DB) AddVendor(ctx context.Context, v ext.HouseholdVendor) (ext.HouseholdVendor, error) {
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into household_vendors (name, service_type, phone, email, website, notes, rating, last_used)
|
||||
// values ($1, $2, $3, $4, $5, $6, $7, $8)
|
||||
// returning id, created_at
|
||||
// `, v.Name, nullStr(v.ServiceType), nullStr(v.Phone), nullStr(v.Email),
|
||||
// nullStr(v.Website), nullStr(v.Notes), v.Rating, v.LastUsed)
|
||||
func (db *DB) AddVendor(ctx context.Context, v ext.HouseholdVendor) (ext.HouseholdVendor, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into household_vendors (name, service_type, phone, email, website, notes, rating, last_used)
|
||||
values ($1, $2, $3, $4, $5, $6, $7, $8)
|
||||
returning id, created_at
|
||||
`, v.Name, nullStr(v.ServiceType), nullStr(v.Phone), nullStr(v.Email),
|
||||
nullStr(v.Website), nullStr(v.Notes), v.Rating, v.LastUsed)
|
||||
|
||||
// created := v
|
||||
// var model generatedmodels.ModelPublicHouseholdVendors
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt); err != nil {
|
||||
// return ext.HouseholdVendor{}, fmt.Errorf("insert vendor: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// created.CreatedAt = model.CreatedAt.Time()
|
||||
// return created, nil
|
||||
// }
|
||||
created := v
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt); err != nil {
|
||||
return ext.HouseholdVendor{}, fmt.Errorf("insert vendor: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) ListVendors(ctx context.Context, serviceType string) ([]ext.HouseholdVendor, error) {
|
||||
// args := []any{}
|
||||
// q := `select id, name, service_type, phone, email, website, notes, rating, last_used, created_at from household_vendors`
|
||||
// if st := strings.TrimSpace(serviceType); st != "" {
|
||||
// args = append(args, st)
|
||||
// q += " where service_type = $1"
|
||||
// }
|
||||
// q += " order by name"
|
||||
func (db *DB) ListVendors(ctx context.Context, serviceType string) ([]ext.HouseholdVendor, error) {
|
||||
args := []any{}
|
||||
q := `select id, name, service_type, phone, email, website, notes, rating, last_used, created_at from household_vendors`
|
||||
if st := strings.TrimSpace(serviceType); st != "" {
|
||||
args = append(args, st)
|
||||
q += " where service_type = $1"
|
||||
}
|
||||
q += " order by name"
|
||||
|
||||
// rows, err := db.pool.Query(ctx, q, args...)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("list vendors: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
rows, err := db.pool.Query(ctx, q, args...)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("list vendors: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// var vendors []ext.HouseholdVendor
|
||||
// for rows.Next() {
|
||||
// var model generatedmodels.ModelPublicHouseholdVendors
|
||||
// if err := rows.Scan(&model.ID, &model.Name, &model.ServiceType, &model.Phone, &model.Email, &model.Website, &model.Notes, &model.Rating, &model.LastUsed, &model.CreatedAt); err != nil {
|
||||
// return nil, fmt.Errorf("scan vendor: %w", err)
|
||||
// }
|
||||
// vendors = append(vendors, householdVendorFromModel(model))
|
||||
// }
|
||||
// return vendors, rows.Err()
|
||||
// }
|
||||
var vendors []ext.HouseholdVendor
|
||||
for rows.Next() {
|
||||
var v ext.HouseholdVendor
|
||||
var serviceType, phone, email, website, notes *string
|
||||
var lastUsed *time.Time
|
||||
if err := rows.Scan(&v.ID, &v.Name, &serviceType, &phone, &email, &website, ¬es, &v.Rating, &lastUsed, &v.CreatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan vendor: %w", err)
|
||||
}
|
||||
v.ServiceType = strVal(serviceType)
|
||||
v.Phone = strVal(phone)
|
||||
v.Email = strVal(email)
|
||||
v.Website = strVal(website)
|
||||
v.Notes = strVal(notes)
|
||||
v.LastUsed = lastUsed
|
||||
vendors = append(vendors, v)
|
||||
}
|
||||
return vendors, rows.Err()
|
||||
}
|
||||
|
||||
+128
-169
@@ -2,207 +2,166 @@ package store
|
||||
|
||||
import (
|
||||
"context"
|
||||
"database/sql"
|
||||
"fmt"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
"github.com/jackc/pgx/v5"
|
||||
|
||||
thoughttypes "git.warky.dev/wdevs/amcs/internal/types"
|
||||
"github.com/google/uuid"
|
||||
"github.com/lib/pq"
|
||||
"amcs/internal/types"
|
||||
)
|
||||
|
||||
func (db *DB) CreateLearning(ctx context.Context, learning thoughttypes.Learning) (thoughttypes.Learning, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into learnings (
|
||||
type LearningStore struct {
|
||||
db *sql.DB
|
||||
}
|
||||
|
||||
func NewLearningStore(db *sql.DB) *LearningStore {
|
||||
return &LearningStore{db: db}
|
||||
}
|
||||
|
||||
func (s *LearningStore) Create(ctx context.Context, l *types.Learning) error {
|
||||
query := `
|
||||
INSERT INTO learnings (
|
||||
summary, details, category, area, status, priority, confidence,
|
||||
action_required, source_type, source_ref, project_id, related_thought_id,
|
||||
related_skill_id, reviewed_by, reviewed_at, duplicate_of_learning_id,
|
||||
supersedes_learning_id, tags
|
||||
) values (
|
||||
$1, $2, $3, $4, $5, $6, $7,
|
||||
$8, $9, $10, $11, $12,
|
||||
$13, $14, $15, $16,
|
||||
$17, $18
|
||||
)
|
||||
returning id, guid, created_at, updated_at
|
||||
`,
|
||||
strings.TrimSpace(learning.Summary),
|
||||
strings.TrimSpace(learning.Details),
|
||||
strings.TrimSpace(learning.Category),
|
||||
strings.TrimSpace(learning.Area),
|
||||
string(learning.Status),
|
||||
string(learning.Priority),
|
||||
string(learning.Confidence),
|
||||
learning.ActionRequired,
|
||||
nullableText(learning.SourceType),
|
||||
nullableText(learning.SourceRef),
|
||||
learning.ProjectID,
|
||||
learning.RelatedThoughtID,
|
||||
learning.RelatedSkillID,
|
||||
nullableTextPtr(learning.ReviewedBy),
|
||||
learning.ReviewedAt,
|
||||
learning.DuplicateOfLearningID,
|
||||
learning.SupersedesLearningID,
|
||||
learning.Tags,
|
||||
)
|
||||
related_skill_id, reviewed_by, reviewed_at, duplicate_of, supersedes, tags
|
||||
) VALUES (
|
||||
$1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12, $13, $14, $15, $16, $17, $18
|
||||
) RETURNING guid, created_at, updated_at`
|
||||
|
||||
created := learning
|
||||
var model generatedmodels.ModelPublicLearnings
|
||||
if err := row.Scan(&model.ID, &model.GUID, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
return thoughttypes.Learning{}, fmt.Errorf("create learning: %w", err)
|
||||
}
|
||||
created.ID = model.ID.Int64()
|
||||
created.GUID = model.GUID.UUID()
|
||||
created.CreatedAt = model.CreatedAt.Time()
|
||||
created.UpdatedAt = model.UpdatedAt.Time()
|
||||
return created, nil
|
||||
}
|
||||
var id uuid.UUID
|
||||
var createdAt, updatedAt time.Time
|
||||
|
||||
func (db *DB) GetLearning(ctx context.Context, id int64) (thoughttypes.Learning, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, summary, details, category, area, status, priority, confidence,
|
||||
action_required, source_type, source_ref, project_id, related_thought_id,
|
||||
related_skill_id, reviewed_by, reviewed_at, duplicate_of_learning_id,
|
||||
supersedes_learning_id, tags::text[], created_at, updated_at
|
||||
from learnings
|
||||
where id = $1
|
||||
`, id)
|
||||
err := s.db.QueryRowContext(ctx, query,
|
||||
l.Summary, l.Details, l.Category, l.Area, l.Status, l.Priority, l.Confidence,
|
||||
l.ActionRequired, l.SourceType, l.SourceRef, l.ProjectID, l.RelatedThoughtID,
|
||||
l.RelatedSkillID, l.ReviewedBy, l.ReviewedAt, l.DuplicateOf, l.Supersedes,
|
||||
pq.Array(l.Tags),
|
||||
).Scan(&id, &createdAt, &updatedAt)
|
||||
|
||||
learning, err := scanLearning(row)
|
||||
if err != nil {
|
||||
if err == pgx.ErrNoRows {
|
||||
return thoughttypes.Learning{}, fmt.Errorf("learning not found: %d", id)
|
||||
}
|
||||
return thoughttypes.Learning{}, fmt.Errorf("get learning: %w", err)
|
||||
return fmt.Errorf("failed to create learning: %w", err)
|
||||
}
|
||||
return learning, nil
|
||||
|
||||
l.ID = id
|
||||
l.CreatedAt = createdAt
|
||||
l.UpdatedAt = updatedAt
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) ListLearnings(ctx context.Context, filter thoughttypes.LearningFilter) ([]thoughttypes.Learning, error) {
|
||||
args := make([]any, 0, 8)
|
||||
conditions := make([]string, 0, 8)
|
||||
func (s *LearningStore) Get(ctx context.Context, id uuid.UUID) (*types.Learning, error) {
|
||||
query := `SELECT * FROM learnings WHERE guid = $1`
|
||||
|
||||
l := &types.Learning{}
|
||||
err := s.db.QueryRowContext(ctx, query, id).Scan(
|
||||
&l.ID, &l.Summary, &l.Details, &l.Category, &l.Area, &l.Status, &l.Priority,
|
||||
&l.Confidence, &l.ActionRequired, &l.SourceType, &l.SourceRef, &l.ProjectID,
|
||||
&l.RelatedThoughtID, &l.RelatedSkillID, &l.ReviewedBy, &l.ReviewedAt,
|
||||
&l.DuplicateOf, &l.Supersedes, &l.Tags, &l.CreatedAt, &l.UpdatedAt,
|
||||
)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to get learning: %w", err)
|
||||
}
|
||||
return l, nil
|
||||
}
|
||||
|
||||
func (s *LearningStore) Update(ctx context.Context, l *types.Learning) error {
|
||||
query := `
|
||||
UPDATE learnings SET
|
||||
summary=$1, details=$2, category=$3, area=$4, status=$5, priority=$6,
|
||||
confidence=$7, action_required=$8, source_type=$9, source_ref=$10,
|
||||
project_id=$11, related_thought_id=$12, related_skill_id=$13,
|
||||
reviewed_by=$14, reviewed_at=$15, duplicate_of=$16, supersedes=$17,
|
||||
tags=$18, updated_at=now()
|
||||
WHERE guid=$19`
|
||||
|
||||
_, err := s.db.ExecContext(ctx, query,
|
||||
l.Summary, l.Details, l.Category, l.Area, l.Status, l.Priority, l.Confidence,
|
||||
l.ActionRequired, l.SourceType, l.SourceRef, l.ProjectID, l.RelatedThoughtID,
|
||||
l.RelatedSkillID, l.ReviewedBy, l.ReviewedAt, l.DuplicateOf, l.Supersedes,
|
||||
pq.Array(l.Tags), l.ID,
|
||||
)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to update learning: %w", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *LearningStore) Delete(ctx context.Context, id uuid.UUID) error {
|
||||
query := `DELETE FROM learnings WHERE guid = $1`
|
||||
_, err := s.db.ExecContext(ctx, query, id)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to delete learning: %w", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *LearningStore) List(ctx context.Context, filter types.LearningFilter) ([]types.Learning, error) {
|
||||
query := `SELECT * FROM learnings WHERE 1=1`
|
||||
args := []interface{}{}
|
||||
argCount := 1
|
||||
|
||||
if filter.ProjectID != nil {
|
||||
query += fmt.Sprintf(` AND project_id = $%d`, argCount)
|
||||
args = append(args, *filter.ProjectID)
|
||||
conditions = append(conditions, fmt.Sprintf("project_id = $%d", len(args)))
|
||||
argCount++
|
||||
}
|
||||
if value := strings.TrimSpace(filter.Category); value != "" {
|
||||
args = append(args, value)
|
||||
conditions = append(conditions, fmt.Sprintf("category = $%d", len(args)))
|
||||
if filter.Category != "" {
|
||||
query += fmt.Sprintf(` AND category = $%d`, argCount)
|
||||
args = append(args, filter.Category)
|
||||
argCount++
|
||||
}
|
||||
if value := strings.TrimSpace(filter.Area); value != "" {
|
||||
args = append(args, value)
|
||||
conditions = append(conditions, fmt.Sprintf("area = $%d", len(args)))
|
||||
if filter.Area != "" {
|
||||
query += fmt.Sprintf(` AND area = $%d`, argCount)
|
||||
args = append(args, filter.Area)
|
||||
argCount++
|
||||
}
|
||||
if value := strings.TrimSpace(filter.Status); value != "" {
|
||||
args = append(args, value)
|
||||
conditions = append(conditions, fmt.Sprintf("status = $%d", len(args)))
|
||||
if filter.Status != "" {
|
||||
query += fmt.Sprintf(` AND status = $%d`, argCount)
|
||||
args = append(args, filter.Status)
|
||||
argCount++
|
||||
}
|
||||
if value := strings.TrimSpace(filter.Priority); value != "" {
|
||||
args = append(args, value)
|
||||
conditions = append(conditions, fmt.Sprintf("priority = $%d", len(args)))
|
||||
if filter.Priority != "" {
|
||||
query += fmt.Sprintf(` AND priority = $%d`, argCount)
|
||||
args = append(args, filter.Priority)
|
||||
argCount++
|
||||
}
|
||||
if value := strings.TrimSpace(filter.Tag); value != "" {
|
||||
args = append(args, value)
|
||||
conditions = append(conditions, fmt.Sprintf("$%d = any(tags)", len(args)))
|
||||
if filter.Tag != "" {
|
||||
query += fmt.Sprintf(` AND %d = ANY(tags)`, argCount) // Wait, tags is array. Correct is:
|
||||
query = fmt.Sprintf("%s AND $%d = ANY(tags)", query, argCount)
|
||||
args = append(args, filter.Tag)
|
||||
argCount++
|
||||
}
|
||||
if value := strings.TrimSpace(filter.Query); value != "" {
|
||||
args = append(args, value)
|
||||
conditions = append(conditions, fmt.Sprintf("to_tsvector('simple', summary || ' ' || coalesce(details, '')) @@ websearch_to_tsquery('simple', $%d)", len(args)))
|
||||
if filter.Query != "" {
|
||||
query += fmt.Sprintf(` AND to_tsvector('simple', summary || ' ' || coalesce(details, '')) @@ websearch_to_tsquery('simple', $%d)`, argCount)
|
||||
args = append(args, filter.Query)
|
||||
argCount++
|
||||
}
|
||||
|
||||
query := `
|
||||
select id, summary, details, category, area, status, priority, confidence,
|
||||
action_required, source_type, source_ref, project_id, related_thought_id,
|
||||
related_skill_id, reviewed_by, reviewed_at, duplicate_of_learning_id,
|
||||
supersedes_learning_id, tags::text[], created_at, updated_at
|
||||
from learnings
|
||||
`
|
||||
if len(conditions) > 0 {
|
||||
query += " where " + strings.Join(conditions, " and ")
|
||||
}
|
||||
query += " order by updated_at desc"
|
||||
if filter.Limit > 0 {
|
||||
args = append(args, filter.Limit)
|
||||
query += fmt.Sprintf(" limit $%d", len(args))
|
||||
query += fmt.Sprintf(` ORDER BY created_at DESC LIMIT %d`, filter.Limit)
|
||||
if filter.Limit == 0 {
|
||||
query = query[:len(query)-10] // remove LIMIT 0
|
||||
}
|
||||
|
||||
rows, err := db.pool.Query(ctx, query, args...)
|
||||
rows, err := s.db.QueryContext(ctx, query, args...)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("list learnings: %w", err)
|
||||
return nil, err
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
items := make([]thoughttypes.Learning, 0)
|
||||
var learnings []types.Learning
|
||||
for rows.Next() {
|
||||
item, err := scanLearning(rows)
|
||||
l := types.Learning{}
|
||||
err := rows.Scan(
|
||||
&l.ID, &l.Summary, &l.Details, &l.Category, &l.Area, &l.Status, &l.Priority,
|
||||
&l.Confidence, &l.ActionRequired, &l.SourceType, &l.SourceRef, &l.ProjectID,
|
||||
&l.RelatedThoughtID, &l.RelatedSkillID, &l.ReviewedBy, &l.ReviewedAt,
|
||||
&l.DuplicateOf, &l.Supersedes, &l.Tags, &l.CreatedAt, &l.UpdatedAt,
|
||||
)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("scan learning: %w", err)
|
||||
return nil, err
|
||||
}
|
||||
items = append(items, item)
|
||||
learnings = append(learnings, l)
|
||||
}
|
||||
if err := rows.Err(); err != nil {
|
||||
return nil, fmt.Errorf("iterate learnings: %w", err)
|
||||
}
|
||||
return items, nil
|
||||
}
|
||||
|
||||
type learningScanner interface {
|
||||
Scan(dest ...any) error
|
||||
}
|
||||
|
||||
func scanLearning(row learningScanner) (thoughttypes.Learning, error) {
|
||||
var model generatedmodels.ModelPublicLearnings
|
||||
var tags []string
|
||||
|
||||
err := row.Scan(
|
||||
&model.ID,
|
||||
&model.Summary,
|
||||
&model.Details,
|
||||
&model.Category,
|
||||
&model.Area,
|
||||
&model.Status,
|
||||
&model.Priority,
|
||||
&model.Confidence,
|
||||
&model.ActionRequired,
|
||||
&model.SourceType,
|
||||
&model.SourceRef,
|
||||
&model.ProjectID,
|
||||
&model.RelatedThoughtID,
|
||||
&model.RelatedSkillID,
|
||||
&model.ReviewedBy,
|
||||
&model.ReviewedAt,
|
||||
&model.DuplicateOfLearningID,
|
||||
&model.SupersedesLearningID,
|
||||
&tags,
|
||||
&model.CreatedAt,
|
||||
&model.UpdatedAt,
|
||||
)
|
||||
if err != nil {
|
||||
return thoughttypes.Learning{}, err
|
||||
}
|
||||
|
||||
if tags == nil {
|
||||
tags = []string{}
|
||||
}
|
||||
return learningFromModel(model, tags), nil
|
||||
}
|
||||
|
||||
func nullableText(value string) *string {
|
||||
trimmed := strings.TrimSpace(value)
|
||||
if trimmed == "" {
|
||||
return nil
|
||||
}
|
||||
return &trimmed
|
||||
}
|
||||
|
||||
func nullableTextPtr(value *string) *string {
|
||||
if value == nil {
|
||||
return nil
|
||||
}
|
||||
trimmed := strings.TrimSpace(*value)
|
||||
if trimmed == "" {
|
||||
return nil
|
||||
}
|
||||
return &trimmed
|
||||
return learnings, nil
|
||||
}
|
||||
|
||||
+13
-16
@@ -2,11 +2,11 @@ package store
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
|
||||
"github.com/google/uuid"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
thoughttypes "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
@@ -25,12 +25,12 @@ func (db *DB) InsertLink(ctx context.Context, link thoughttypes.ThoughtLink) err
|
||||
|
||||
func (db *DB) LinkedThoughts(ctx context.Context, thoughtID uuid.UUID) ([]thoughttypes.LinkedThought, error) {
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select t.id, t.guid, t.content, t.metadata, t.project_id, t.archived_at, t.created_at, t.updated_at, l.relation, 'outgoing' as direction, l.created_at
|
||||
select t.guid, t.content, t.metadata, t.project_id, t.archived_at, t.created_at, t.updated_at, l.relation, 'outgoing' as direction, l.created_at
|
||||
from thought_links l
|
||||
join thoughts t on t.id = l.to_id
|
||||
where l.from_id = (select id from thoughts where guid = $1)
|
||||
union all
|
||||
select t.id, t.guid, t.content, t.metadata, t.project_id, t.archived_at, t.created_at, t.updated_at, l.relation, 'incoming' as direction, l.created_at
|
||||
select t.guid, t.content, t.metadata, t.project_id, t.archived_at, t.created_at, t.updated_at, l.relation, 'incoming' as direction, l.created_at
|
||||
from thought_links l
|
||||
join thoughts t on t.id = l.from_id
|
||||
where l.to_id = (select id from thoughts where guid = $1)
|
||||
@@ -44,27 +44,24 @@ func (db *DB) LinkedThoughts(ctx context.Context, thoughtID uuid.UUID) ([]though
|
||||
links := make([]thoughttypes.LinkedThought, 0)
|
||||
for rows.Next() {
|
||||
var linked thoughttypes.LinkedThought
|
||||
var model generatedmodels.ModelPublicThoughts
|
||||
var metadataBytes []byte
|
||||
if err := rows.Scan(
|
||||
&model.ID,
|
||||
&model.GUID,
|
||||
&model.Content,
|
||||
&model.Metadata,
|
||||
&model.ProjectID,
|
||||
&model.ArchivedAt,
|
||||
&model.CreatedAt,
|
||||
&model.UpdatedAt,
|
||||
&linked.Thought.ID,
|
||||
&linked.Thought.Content,
|
||||
&metadataBytes,
|
||||
&linked.Thought.ProjectID,
|
||||
&linked.Thought.ArchivedAt,
|
||||
&linked.Thought.CreatedAt,
|
||||
&linked.Thought.UpdatedAt,
|
||||
&linked.Relation,
|
||||
&linked.Direction,
|
||||
&linked.CreatedAt,
|
||||
); err != nil {
|
||||
return nil, fmt.Errorf("scan linked thought: %w", err)
|
||||
}
|
||||
thought, err := thoughtFromModel(model)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("map linked thought: %w", err)
|
||||
if err := json.Unmarshal(metadataBytes, &linked.Thought.Metadata); err != nil {
|
||||
return nil, fmt.Errorf("decode linked thought metadata: %w", err)
|
||||
}
|
||||
linked.Thought = thought
|
||||
links = append(links, linked)
|
||||
}
|
||||
if err := rows.Err(); err != nil {
|
||||
|
||||
+126
-121
@@ -1,137 +1,142 @@
|
||||
package store
|
||||
|
||||
// import (
|
||||
// "context"
|
||||
// "fmt"
|
||||
// "strings"
|
||||
// "time"
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
// "git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
// ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
// )
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
// func (db *DB) AddMaintenanceTask(ctx context.Context, t ext.MaintenanceTask) (ext.MaintenanceTask, error) {
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into maintenance_tasks (name, category, frequency_days, next_due, priority, notes)
|
||||
// values ($1, $2, $3, $4, $5, $6)
|
||||
// returning id, created_at, updated_at
|
||||
// `, t.Name, nullStr(t.Category), t.FrequencyDays, t.NextDue, t.Priority, nullStr(t.Notes))
|
||||
func (db *DB) AddMaintenanceTask(ctx context.Context, t ext.MaintenanceTask) (ext.MaintenanceTask, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into maintenance_tasks (name, category, frequency_days, next_due, priority, notes)
|
||||
values ($1, $2, $3, $4, $5, $6)
|
||||
returning id, created_at, updated_at
|
||||
`, t.Name, nullStr(t.Category), t.FrequencyDays, t.NextDue, t.Priority, nullStr(t.Notes))
|
||||
|
||||
// created := t
|
||||
// var model generatedmodels.ModelPublicMaintenanceTasks
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.MaintenanceTask{}, fmt.Errorf("insert maintenance task: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// created.CreatedAt = model.CreatedAt.Time()
|
||||
// created.UpdatedAt = model.UpdatedAt.Time()
|
||||
// return created, nil
|
||||
// }
|
||||
created := t
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt, &created.UpdatedAt); err != nil {
|
||||
return ext.MaintenanceTask{}, fmt.Errorf("insert maintenance task: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) LogMaintenance(ctx context.Context, log ext.MaintenanceLog) (ext.MaintenanceLog, error) {
|
||||
// completedAt := log.CompletedAt
|
||||
// if completedAt.IsZero() {
|
||||
// completedAt = time.Now()
|
||||
// }
|
||||
func (db *DB) LogMaintenance(ctx context.Context, log ext.MaintenanceLog) (ext.MaintenanceLog, error) {
|
||||
completedAt := log.CompletedAt
|
||||
if completedAt.IsZero() {
|
||||
completedAt = time.Now()
|
||||
}
|
||||
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into maintenance_logs (task_id, completed_at, performed_by, cost, notes, next_action)
|
||||
// values ($1, $2, $3, $4, $5, $6)
|
||||
// returning id
|
||||
// `, log.TaskID, completedAt, nullStr(log.PerformedBy), log.Cost, nullStr(log.Notes), nullStr(log.NextAction))
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into maintenance_logs (task_id, completed_at, performed_by, cost, notes, next_action)
|
||||
values ($1, $2, $3, $4, $5, $6)
|
||||
returning id
|
||||
`, log.TaskID, completedAt, nullStr(log.PerformedBy), log.Cost, nullStr(log.Notes), nullStr(log.NextAction))
|
||||
|
||||
// created := log
|
||||
// created.CompletedAt = completedAt
|
||||
// var model generatedmodels.ModelPublicMaintenanceLogs
|
||||
// if err := row.Scan(&model.ID); err != nil {
|
||||
// return ext.MaintenanceLog{}, fmt.Errorf("insert maintenance log: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// return created, nil
|
||||
// }
|
||||
created := log
|
||||
created.CompletedAt = completedAt
|
||||
if err := row.Scan(&created.ID); err != nil {
|
||||
return ext.MaintenanceLog{}, fmt.Errorf("insert maintenance log: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) GetUpcomingMaintenance(ctx context.Context, daysAhead int) ([]ext.MaintenanceTask, error) {
|
||||
// if daysAhead <= 0 {
|
||||
// daysAhead = 30
|
||||
// }
|
||||
// cutoff := time.Now().Add(time.Duration(daysAhead) * 24 * time.Hour)
|
||||
func (db *DB) GetUpcomingMaintenance(ctx context.Context, daysAhead int) ([]ext.MaintenanceTask, error) {
|
||||
if daysAhead <= 0 {
|
||||
daysAhead = 30
|
||||
}
|
||||
cutoff := time.Now().Add(time.Duration(daysAhead) * 24 * time.Hour)
|
||||
|
||||
// rows, err := db.pool.Query(ctx, `
|
||||
// select id, name, category, frequency_days, last_completed, next_due, priority, notes, created_at, updated_at
|
||||
// from maintenance_tasks
|
||||
// where next_due <= $1 or next_due is null
|
||||
// order by next_due asc nulls last, priority desc
|
||||
// `, cutoff)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("get upcoming maintenance: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select id, name, category, frequency_days, last_completed, next_due, priority, notes, created_at, updated_at
|
||||
from maintenance_tasks
|
||||
where next_due <= $1 or next_due is null
|
||||
order by next_due asc nulls last, priority desc
|
||||
`, cutoff)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("get upcoming maintenance: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// tasks := make([]ext.MaintenanceTask, 0)
|
||||
// for rows.Next() {
|
||||
// var model generatedmodels.ModelPublicMaintenanceTasks
|
||||
// if err := rows.Scan(&model.ID, &model.Name, &model.Category, &model.FrequencyDays, &model.LastCompleted, &model.NextDue, &model.Priority, &model.Notes, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return nil, fmt.Errorf("scan maintenance task: %w", err)
|
||||
// }
|
||||
// tasks = append(tasks, maintenanceTaskFromModel(model))
|
||||
// }
|
||||
// return tasks, rows.Err()
|
||||
// }
|
||||
return scanMaintenanceTasks(rows)
|
||||
}
|
||||
|
||||
// func (db *DB) SearchMaintenanceHistory(ctx context.Context, query, category string, start, end *time.Time) ([]ext.MaintenanceLogWithTask, error) {
|
||||
// args := []any{}
|
||||
// conditions := []string{}
|
||||
func (db *DB) SearchMaintenanceHistory(ctx context.Context, query, category string, start, end *time.Time) ([]ext.MaintenanceLogWithTask, error) {
|
||||
args := []any{}
|
||||
conditions := []string{}
|
||||
|
||||
// if q := strings.TrimSpace(query); q != "" {
|
||||
// args = append(args, "%"+q+"%")
|
||||
// conditions = append(conditions, fmt.Sprintf("(mt.name ILIKE $%d OR ml.notes ILIKE $%d)", len(args), len(args)))
|
||||
// }
|
||||
// if c := strings.TrimSpace(category); c != "" {
|
||||
// args = append(args, c)
|
||||
// conditions = append(conditions, fmt.Sprintf("mt.category = $%d", len(args)))
|
||||
// }
|
||||
// if start != nil {
|
||||
// args = append(args, *start)
|
||||
// conditions = append(conditions, fmt.Sprintf("ml.completed_at >= $%d", len(args)))
|
||||
// }
|
||||
// if end != nil {
|
||||
// args = append(args, *end)
|
||||
// conditions = append(conditions, fmt.Sprintf("ml.completed_at <= $%d", len(args)))
|
||||
// }
|
||||
if q := strings.TrimSpace(query); q != "" {
|
||||
args = append(args, "%"+q+"%")
|
||||
conditions = append(conditions, fmt.Sprintf("(mt.name ILIKE $%d OR ml.notes ILIKE $%d)", len(args), len(args)))
|
||||
}
|
||||
if c := strings.TrimSpace(category); c != "" {
|
||||
args = append(args, c)
|
||||
conditions = append(conditions, fmt.Sprintf("mt.category = $%d", len(args)))
|
||||
}
|
||||
if start != nil {
|
||||
args = append(args, *start)
|
||||
conditions = append(conditions, fmt.Sprintf("ml.completed_at >= $%d", len(args)))
|
||||
}
|
||||
if end != nil {
|
||||
args = append(args, *end)
|
||||
conditions = append(conditions, fmt.Sprintf("ml.completed_at <= $%d", len(args)))
|
||||
}
|
||||
|
||||
// q := `
|
||||
// select ml.id, ml.task_id, ml.completed_at, ml.performed_by, ml.cost, ml.notes, ml.next_action,
|
||||
// mt.name, mt.category
|
||||
// from maintenance_logs ml
|
||||
// join maintenance_tasks mt on mt.id = ml.task_id
|
||||
// `
|
||||
// if len(conditions) > 0 {
|
||||
// q += " where " + strings.Join(conditions, " and ")
|
||||
// }
|
||||
// q += " order by ml.completed_at desc"
|
||||
q := `
|
||||
select ml.id, ml.task_id, ml.completed_at, ml.performed_by, ml.cost, ml.notes, ml.next_action,
|
||||
mt.name, mt.category
|
||||
from maintenance_logs ml
|
||||
join maintenance_tasks mt on mt.id = ml.task_id
|
||||
`
|
||||
if len(conditions) > 0 {
|
||||
q += " where " + strings.Join(conditions, " and ")
|
||||
}
|
||||
q += " order by ml.completed_at desc"
|
||||
|
||||
// rows, err := db.pool.Query(ctx, q, args...)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("search maintenance history: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
rows, err := db.pool.Query(ctx, q, args...)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("search maintenance history: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// var logs []ext.MaintenanceLogWithTask
|
||||
// for rows.Next() {
|
||||
// var model generatedmodels.ModelPublicMaintenanceLogs
|
||||
// var taskName, taskCategory string
|
||||
// if err := rows.Scan(
|
||||
// &model.ID, &model.TaskID, &model.CompletedAt, &model.PerformedBy, &model.Cost, &model.Notes, &model.NextAction,
|
||||
// &taskName, &taskCategory,
|
||||
// ); err != nil {
|
||||
// return nil, fmt.Errorf("scan maintenance log: %w", err)
|
||||
// }
|
||||
// l := ext.MaintenanceLogWithTask{
|
||||
// MaintenanceLog: maintenanceLogFromModel(model),
|
||||
// TaskName: taskName,
|
||||
// TaskCategory: taskCategory,
|
||||
// }
|
||||
// logs = append(logs, l)
|
||||
// }
|
||||
// return logs, rows.Err()
|
||||
// }
|
||||
var logs []ext.MaintenanceLogWithTask
|
||||
for rows.Next() {
|
||||
var l ext.MaintenanceLogWithTask
|
||||
var performedBy, notes, nextAction, taskCategory *string
|
||||
if err := rows.Scan(
|
||||
&l.ID, &l.TaskID, &l.CompletedAt, &performedBy, &l.Cost, ¬es, &nextAction,
|
||||
&l.TaskName, &taskCategory,
|
||||
); err != nil {
|
||||
return nil, fmt.Errorf("scan maintenance log: %w", err)
|
||||
}
|
||||
l.PerformedBy = strVal(performedBy)
|
||||
l.Notes = strVal(notes)
|
||||
l.NextAction = strVal(nextAction)
|
||||
l.TaskCategory = strVal(taskCategory)
|
||||
logs = append(logs, l)
|
||||
}
|
||||
return logs, rows.Err()
|
||||
}
|
||||
|
||||
func scanMaintenanceTasks(rows interface {
|
||||
Next() bool
|
||||
Scan(...any) error
|
||||
Err() error
|
||||
Close()
|
||||
}) ([]ext.MaintenanceTask, error) {
|
||||
defer rows.Close()
|
||||
var tasks []ext.MaintenanceTask
|
||||
for rows.Next() {
|
||||
var t ext.MaintenanceTask
|
||||
var category, notes *string
|
||||
if err := rows.Scan(&t.ID, &t.Name, &category, &t.FrequencyDays, &t.LastCompleted, &t.NextDue, &t.Priority, ¬es, &t.CreatedAt, &t.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan maintenance task: %w", err)
|
||||
}
|
||||
t.Category = strVal(category)
|
||||
t.Notes = strVal(notes)
|
||||
tasks = append(tasks, t)
|
||||
}
|
||||
return tasks, rows.Err()
|
||||
}
|
||||
|
||||
+260
-251
@@ -1,280 +1,289 @@
|
||||
package store
|
||||
|
||||
// import (
|
||||
// "context"
|
||||
// "encoding/json"
|
||||
// "fmt"
|
||||
// "strings"
|
||||
// "time"
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
// "github.com/google/uuid"
|
||||
"github.com/google/uuid"
|
||||
|
||||
// "git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
// ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
// )
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
// func (db *DB) AddRecipe(ctx context.Context, r ext.Recipe) (ext.Recipe, error) {
|
||||
// ingredients, err := json.Marshal(r.Ingredients)
|
||||
// if err != nil {
|
||||
// return ext.Recipe{}, fmt.Errorf("marshal ingredients: %w", err)
|
||||
// }
|
||||
// instructions, err := json.Marshal(r.Instructions)
|
||||
// if err != nil {
|
||||
// return ext.Recipe{}, fmt.Errorf("marshal instructions: %w", err)
|
||||
// }
|
||||
// if r.Tags == nil {
|
||||
// r.Tags = []string{}
|
||||
// }
|
||||
func (db *DB) AddRecipe(ctx context.Context, r ext.Recipe) (ext.Recipe, error) {
|
||||
ingredients, err := json.Marshal(r.Ingredients)
|
||||
if err != nil {
|
||||
return ext.Recipe{}, fmt.Errorf("marshal ingredients: %w", err)
|
||||
}
|
||||
instructions, err := json.Marshal(r.Instructions)
|
||||
if err != nil {
|
||||
return ext.Recipe{}, fmt.Errorf("marshal instructions: %w", err)
|
||||
}
|
||||
if r.Tags == nil {
|
||||
r.Tags = []string{}
|
||||
}
|
||||
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into recipes (name, cuisine, prep_time_minutes, cook_time_minutes, servings, ingredients, instructions, tags, rating, notes)
|
||||
// values ($1, $2, $3, $4, $5, $6::jsonb, $7::jsonb, $8, $9, $10)
|
||||
// returning id, created_at, updated_at
|
||||
// `, r.Name, nullStr(r.Cuisine), r.PrepTimeMinutes, r.CookTimeMinutes, r.Servings,
|
||||
// ingredients, instructions, r.Tags, r.Rating, nullStr(r.Notes))
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into recipes (name, cuisine, prep_time_minutes, cook_time_minutes, servings, ingredients, instructions, tags, rating, notes)
|
||||
values ($1, $2, $3, $4, $5, $6::jsonb, $7::jsonb, $8, $9, $10)
|
||||
returning id, created_at, updated_at
|
||||
`, r.Name, nullStr(r.Cuisine), r.PrepTimeMinutes, r.CookTimeMinutes, r.Servings,
|
||||
ingredients, instructions, r.Tags, r.Rating, nullStr(r.Notes))
|
||||
|
||||
// created := r
|
||||
// var model generatedmodels.ModelPublicRecipes
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.Recipe{}, fmt.Errorf("insert recipe: %w", err)
|
||||
// }
|
||||
// created.ID = model.ID.UUID()
|
||||
// created.CreatedAt = model.CreatedAt.Time()
|
||||
// created.UpdatedAt = model.UpdatedAt.Time()
|
||||
// return created, nil
|
||||
// }
|
||||
created := r
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt, &created.UpdatedAt); err != nil {
|
||||
return ext.Recipe{}, fmt.Errorf("insert recipe: %w", err)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
// func (db *DB) SearchRecipes(ctx context.Context, query, cuisine string, tags []string, ingredient string) ([]ext.Recipe, error) {
|
||||
// args := []any{}
|
||||
// conditions := []string{}
|
||||
func (db *DB) SearchRecipes(ctx context.Context, query, cuisine string, tags []string, ingredient string) ([]ext.Recipe, error) {
|
||||
args := []any{}
|
||||
conditions := []string{}
|
||||
|
||||
// if q := strings.TrimSpace(query); q != "" {
|
||||
// args = append(args, "%"+q+"%")
|
||||
// conditions = append(conditions, fmt.Sprintf("name ILIKE $%d", len(args)))
|
||||
// }
|
||||
// if c := strings.TrimSpace(cuisine); c != "" {
|
||||
// args = append(args, c)
|
||||
// conditions = append(conditions, fmt.Sprintf("cuisine = $%d", len(args)))
|
||||
// }
|
||||
// if len(tags) > 0 {
|
||||
// args = append(args, tags)
|
||||
// conditions = append(conditions, fmt.Sprintf("tags @> $%d", len(args)))
|
||||
// }
|
||||
// if ing := strings.TrimSpace(ingredient); ing != "" {
|
||||
// args = append(args, "%"+ing+"%")
|
||||
// conditions = append(conditions, fmt.Sprintf("ingredients::text ILIKE $%d", len(args)))
|
||||
// }
|
||||
if q := strings.TrimSpace(query); q != "" {
|
||||
args = append(args, "%"+q+"%")
|
||||
conditions = append(conditions, fmt.Sprintf("name ILIKE $%d", len(args)))
|
||||
}
|
||||
if c := strings.TrimSpace(cuisine); c != "" {
|
||||
args = append(args, c)
|
||||
conditions = append(conditions, fmt.Sprintf("cuisine = $%d", len(args)))
|
||||
}
|
||||
if len(tags) > 0 {
|
||||
args = append(args, tags)
|
||||
conditions = append(conditions, fmt.Sprintf("tags @> $%d", len(args)))
|
||||
}
|
||||
if ing := strings.TrimSpace(ingredient); ing != "" {
|
||||
args = append(args, "%"+ing+"%")
|
||||
conditions = append(conditions, fmt.Sprintf("ingredients::text ILIKE $%d", len(args)))
|
||||
}
|
||||
|
||||
// q := `select id, name, cuisine, prep_time_minutes, cook_time_minutes, servings, ingredients, instructions, tags::text[], rating, notes, created_at, updated_at from recipes`
|
||||
// if len(conditions) > 0 {
|
||||
// q += " where " + strings.Join(conditions, " and ")
|
||||
// }
|
||||
// q += " order by name"
|
||||
q := `select id, name, cuisine, prep_time_minutes, cook_time_minutes, servings, ingredients, instructions, tags, rating, notes, created_at, updated_at from recipes`
|
||||
if len(conditions) > 0 {
|
||||
q += " where " + strings.Join(conditions, " and ")
|
||||
}
|
||||
q += " order by name"
|
||||
|
||||
// rows, err := db.pool.Query(ctx, q, args...)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("search recipes: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
rows, err := db.pool.Query(ctx, q, args...)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("search recipes: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// var recipes []ext.Recipe
|
||||
// for rows.Next() {
|
||||
// r, err := scanRecipeRow(rows)
|
||||
// if err != nil {
|
||||
// return nil, err
|
||||
// }
|
||||
// recipes = append(recipes, r)
|
||||
// }
|
||||
// return recipes, rows.Err()
|
||||
// }
|
||||
var recipes []ext.Recipe
|
||||
for rows.Next() {
|
||||
r, err := scanRecipeRow(rows)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
recipes = append(recipes, r)
|
||||
}
|
||||
return recipes, rows.Err()
|
||||
}
|
||||
|
||||
// func (db *DB) GetRecipe(ctx context.Context, id uuid.UUID) (ext.Recipe, error) {
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// select id, name, cuisine, prep_time_minutes, cook_time_minutes, servings, ingredients, instructions, tags::text[], rating, notes, created_at, updated_at
|
||||
// from recipes where id = $1
|
||||
// `, id)
|
||||
func (db *DB) GetRecipe(ctx context.Context, id uuid.UUID) (ext.Recipe, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, name, cuisine, prep_time_minutes, cook_time_minutes, servings, ingredients, instructions, tags, rating, notes, created_at, updated_at
|
||||
from recipes where id = $1
|
||||
`, id)
|
||||
|
||||
// var model generatedmodels.ModelPublicRecipes
|
||||
// var tags []string
|
||||
// if err := row.Scan(&model.ID, &model.Name, &model.Cuisine, &model.PrepTimeMinutes, &model.CookTimeMinutes, &model.Servings,
|
||||
// &model.Ingredients, &model.Instructions, &tags, &model.Rating, &model.Notes, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.Recipe{}, fmt.Errorf("get recipe: %w", err)
|
||||
// }
|
||||
// if tags == nil {
|
||||
// tags = []string{}
|
||||
// }
|
||||
// return recipeFromModel(model, tags), nil
|
||||
// }
|
||||
var r ext.Recipe
|
||||
var cuisine, notes *string
|
||||
var ingredientsBytes, instructionsBytes []byte
|
||||
if err := row.Scan(&r.ID, &r.Name, &cuisine, &r.PrepTimeMinutes, &r.CookTimeMinutes, &r.Servings,
|
||||
&ingredientsBytes, &instructionsBytes, &r.Tags, &r.Rating, ¬es, &r.CreatedAt, &r.UpdatedAt); err != nil {
|
||||
return ext.Recipe{}, fmt.Errorf("get recipe: %w", err)
|
||||
}
|
||||
r.Cuisine = strVal(cuisine)
|
||||
r.Notes = strVal(notes)
|
||||
if r.Tags == nil {
|
||||
r.Tags = []string{}
|
||||
}
|
||||
if err := json.Unmarshal(ingredientsBytes, &r.Ingredients); err != nil {
|
||||
r.Ingredients = []ext.Ingredient{}
|
||||
}
|
||||
if err := json.Unmarshal(instructionsBytes, &r.Instructions); err != nil {
|
||||
r.Instructions = []string{}
|
||||
}
|
||||
return r, nil
|
||||
}
|
||||
|
||||
// func (db *DB) UpdateRecipe(ctx context.Context, id uuid.UUID, r ext.Recipe) (ext.Recipe, error) {
|
||||
// ingredients, err := json.Marshal(r.Ingredients)
|
||||
// if err != nil {
|
||||
// return ext.Recipe{}, fmt.Errorf("marshal ingredients: %w", err)
|
||||
// }
|
||||
// instructions, err := json.Marshal(r.Instructions)
|
||||
// if err != nil {
|
||||
// return ext.Recipe{}, fmt.Errorf("marshal instructions: %w", err)
|
||||
// }
|
||||
// if r.Tags == nil {
|
||||
// r.Tags = []string{}
|
||||
// }
|
||||
func (db *DB) UpdateRecipe(ctx context.Context, id uuid.UUID, r ext.Recipe) (ext.Recipe, error) {
|
||||
ingredients, err := json.Marshal(r.Ingredients)
|
||||
if err != nil {
|
||||
return ext.Recipe{}, fmt.Errorf("marshal ingredients: %w", err)
|
||||
}
|
||||
instructions, err := json.Marshal(r.Instructions)
|
||||
if err != nil {
|
||||
return ext.Recipe{}, fmt.Errorf("marshal instructions: %w", err)
|
||||
}
|
||||
if r.Tags == nil {
|
||||
r.Tags = []string{}
|
||||
}
|
||||
|
||||
// _, err = db.pool.Exec(ctx, `
|
||||
// update recipes set
|
||||
// name = $2, cuisine = $3, prep_time_minutes = $4, cook_time_minutes = $5,
|
||||
// servings = $6, ingredients = $7::jsonb, instructions = $8::jsonb,
|
||||
// tags = $9, rating = $10, notes = $11
|
||||
// where id = $1
|
||||
// `, id, r.Name, nullStr(r.Cuisine), r.PrepTimeMinutes, r.CookTimeMinutes, r.Servings,
|
||||
// ingredients, instructions, r.Tags, r.Rating, nullStr(r.Notes))
|
||||
// if err != nil {
|
||||
// return ext.Recipe{}, fmt.Errorf("update recipe: %w", err)
|
||||
// }
|
||||
// return db.GetRecipe(ctx, id)
|
||||
// }
|
||||
_, err = db.pool.Exec(ctx, `
|
||||
update recipes set
|
||||
name = $2, cuisine = $3, prep_time_minutes = $4, cook_time_minutes = $5,
|
||||
servings = $6, ingredients = $7::jsonb, instructions = $8::jsonb,
|
||||
tags = $9, rating = $10, notes = $11
|
||||
where id = $1
|
||||
`, id, r.Name, nullStr(r.Cuisine), r.PrepTimeMinutes, r.CookTimeMinutes, r.Servings,
|
||||
ingredients, instructions, r.Tags, r.Rating, nullStr(r.Notes))
|
||||
if err != nil {
|
||||
return ext.Recipe{}, fmt.Errorf("update recipe: %w", err)
|
||||
}
|
||||
return db.GetRecipe(ctx, id)
|
||||
}
|
||||
|
||||
// func (db *DB) CreateMealPlan(ctx context.Context, weekStart time.Time, entries []ext.MealPlanInput) ([]ext.MealPlanEntry, error) {
|
||||
// if _, err := db.pool.Exec(ctx, `delete from meal_plans where week_start = $1`, weekStart); err != nil {
|
||||
// return nil, fmt.Errorf("clear meal plan: %w", err)
|
||||
// }
|
||||
func (db *DB) CreateMealPlan(ctx context.Context, weekStart time.Time, entries []ext.MealPlanInput) ([]ext.MealPlanEntry, error) {
|
||||
if _, err := db.pool.Exec(ctx, `delete from meal_plans where week_start = $1`, weekStart); err != nil {
|
||||
return nil, fmt.Errorf("clear meal plan: %w", err)
|
||||
}
|
||||
|
||||
// var results []ext.MealPlanEntry
|
||||
// for _, e := range entries {
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into meal_plans (week_start, day_of_week, meal_type, recipe_id, custom_meal, servings, notes)
|
||||
// values ($1, $2, $3, $4, $5, $6, $7)
|
||||
// returning id, created_at
|
||||
// `, weekStart, e.DayOfWeek, e.MealType, e.RecipeID, nullStr(e.CustomMeal), e.Servings, nullStr(e.Notes))
|
||||
var results []ext.MealPlanEntry
|
||||
for _, e := range entries {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into meal_plans (week_start, day_of_week, meal_type, recipe_id, custom_meal, servings, notes)
|
||||
values ($1, $2, $3, $4, $5, $6, $7)
|
||||
returning id, created_at
|
||||
`, weekStart, e.DayOfWeek, e.MealType, e.RecipeID, nullStr(e.CustomMeal), e.Servings, nullStr(e.Notes))
|
||||
|
||||
// entry := ext.MealPlanEntry{
|
||||
// WeekStart: weekStart,
|
||||
// DayOfWeek: e.DayOfWeek,
|
||||
// MealType: e.MealType,
|
||||
// RecipeID: e.RecipeID,
|
||||
// CustomMeal: e.CustomMeal,
|
||||
// Servings: e.Servings,
|
||||
// Notes: e.Notes,
|
||||
// }
|
||||
// var model generatedmodels.ModelPublicMealPlans
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt); err != nil {
|
||||
// return nil, fmt.Errorf("insert meal plan entry: %w", err)
|
||||
// }
|
||||
// entry.ID = model.ID.UUID()
|
||||
// entry.CreatedAt = model.CreatedAt.Time()
|
||||
// results = append(results, entry)
|
||||
// }
|
||||
// return results, nil
|
||||
// }
|
||||
entry := ext.MealPlanEntry{
|
||||
WeekStart: weekStart,
|
||||
DayOfWeek: e.DayOfWeek,
|
||||
MealType: e.MealType,
|
||||
RecipeID: e.RecipeID,
|
||||
CustomMeal: e.CustomMeal,
|
||||
Servings: e.Servings,
|
||||
Notes: e.Notes,
|
||||
}
|
||||
if err := row.Scan(&entry.ID, &entry.CreatedAt); err != nil {
|
||||
return nil, fmt.Errorf("insert meal plan entry: %w", err)
|
||||
}
|
||||
results = append(results, entry)
|
||||
}
|
||||
return results, nil
|
||||
}
|
||||
|
||||
// func (db *DB) GetMealPlan(ctx context.Context, weekStart time.Time) ([]ext.MealPlanEntry, error) {
|
||||
// rows, err := db.pool.Query(ctx, `
|
||||
// select mp.id, mp.week_start, mp.day_of_week, mp.meal_type, mp.recipe_id, r.name, mp.custom_meal, mp.servings, mp.notes, mp.created_at
|
||||
// from meal_plans mp
|
||||
// left join recipes r on r.id = mp.recipe_id
|
||||
// where mp.week_start = $1
|
||||
// order by
|
||||
// case mp.day_of_week
|
||||
// when 'monday' then 1 when 'tuesday' then 2 when 'wednesday' then 3
|
||||
// when 'thursday' then 4 when 'friday' then 5 when 'saturday' then 6
|
||||
// when 'sunday' then 7 else 8
|
||||
// end,
|
||||
// case mp.meal_type
|
||||
// when 'breakfast' then 1 when 'lunch' then 2 when 'dinner' then 3
|
||||
// when 'snack' then 4 else 5
|
||||
// end
|
||||
// `, weekStart)
|
||||
// if err != nil {
|
||||
// return nil, fmt.Errorf("get meal plan: %w", err)
|
||||
// }
|
||||
// defer rows.Close()
|
||||
func (db *DB) GetMealPlan(ctx context.Context, weekStart time.Time) ([]ext.MealPlanEntry, error) {
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select mp.id, mp.week_start, mp.day_of_week, mp.meal_type, mp.recipe_id, r.name, mp.custom_meal, mp.servings, mp.notes, mp.created_at
|
||||
from meal_plans mp
|
||||
left join recipes r on r.id = mp.recipe_id
|
||||
where mp.week_start = $1
|
||||
order by
|
||||
case mp.day_of_week
|
||||
when 'monday' then 1 when 'tuesday' then 2 when 'wednesday' then 3
|
||||
when 'thursday' then 4 when 'friday' then 5 when 'saturday' then 6
|
||||
when 'sunday' then 7 else 8
|
||||
end,
|
||||
case mp.meal_type
|
||||
when 'breakfast' then 1 when 'lunch' then 2 when 'dinner' then 3
|
||||
when 'snack' then 4 else 5
|
||||
end
|
||||
`, weekStart)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("get meal plan: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
// var entries []ext.MealPlanEntry
|
||||
// for rows.Next() {
|
||||
// var model generatedmodels.ModelPublicMealPlans
|
||||
// var recipeName *string
|
||||
// if err := rows.Scan(&model.ID, &model.WeekStart, &model.DayOfWeek, &model.MealType, &model.RecipeID, &recipeName, &model.CustomMeal, &model.Servings, &model.Notes, &model.CreatedAt); err != nil {
|
||||
// return nil, fmt.Errorf("scan meal plan entry: %w", err)
|
||||
// }
|
||||
// entries = append(entries, mealPlanEntryFromModel(model, strVal(recipeName)))
|
||||
// }
|
||||
// return entries, rows.Err()
|
||||
// }
|
||||
var entries []ext.MealPlanEntry
|
||||
for rows.Next() {
|
||||
var e ext.MealPlanEntry
|
||||
var recipeName, customMeal, notes *string
|
||||
if err := rows.Scan(&e.ID, &e.WeekStart, &e.DayOfWeek, &e.MealType, &e.RecipeID, &recipeName, &customMeal, &e.Servings, ¬es, &e.CreatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan meal plan entry: %w", err)
|
||||
}
|
||||
e.RecipeName = strVal(recipeName)
|
||||
e.CustomMeal = strVal(customMeal)
|
||||
e.Notes = strVal(notes)
|
||||
entries = append(entries, e)
|
||||
}
|
||||
return entries, rows.Err()
|
||||
}
|
||||
|
||||
// func (db *DB) GenerateShoppingList(ctx context.Context, weekStart time.Time) (ext.ShoppingList, error) {
|
||||
// entries, err := db.GetMealPlan(ctx, weekStart)
|
||||
// if err != nil {
|
||||
// return ext.ShoppingList{}, err
|
||||
// }
|
||||
func (db *DB) GenerateShoppingList(ctx context.Context, weekStart time.Time) (ext.ShoppingList, error) {
|
||||
entries, err := db.GetMealPlan(ctx, weekStart)
|
||||
if err != nil {
|
||||
return ext.ShoppingList{}, err
|
||||
}
|
||||
|
||||
// recipeIDs := map[uuid.UUID]bool{}
|
||||
// for _, e := range entries {
|
||||
// if e.RecipeID != nil {
|
||||
// recipeIDs[*e.RecipeID] = true
|
||||
// }
|
||||
// }
|
||||
recipeIDs := map[uuid.UUID]bool{}
|
||||
for _, e := range entries {
|
||||
if e.RecipeID != nil {
|
||||
recipeIDs[*e.RecipeID] = true
|
||||
}
|
||||
}
|
||||
|
||||
// aggregated := map[string]*ext.ShoppingItem{}
|
||||
// for id := range recipeIDs {
|
||||
// recipe, err := db.GetRecipe(ctx, id)
|
||||
// if err != nil {
|
||||
// continue
|
||||
// }
|
||||
// for _, ing := range recipe.Ingredients {
|
||||
// key := strings.ToLower(ing.Name)
|
||||
// if existing, ok := aggregated[key]; ok {
|
||||
// if ing.Quantity != "" {
|
||||
// existing.Quantity += "+" + ing.Quantity
|
||||
// }
|
||||
// } else {
|
||||
// recipeIDCopy := id
|
||||
// aggregated[key] = &ext.ShoppingItem{
|
||||
// Name: ing.Name,
|
||||
// Quantity: ing.Quantity,
|
||||
// Unit: ing.Unit,
|
||||
// Purchased: false,
|
||||
// RecipeID: &recipeIDCopy,
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
aggregated := map[string]*ext.ShoppingItem{}
|
||||
for id := range recipeIDs {
|
||||
recipe, err := db.GetRecipe(ctx, id)
|
||||
if err != nil {
|
||||
continue
|
||||
}
|
||||
for _, ing := range recipe.Ingredients {
|
||||
key := strings.ToLower(ing.Name)
|
||||
if existing, ok := aggregated[key]; ok {
|
||||
if ing.Quantity != "" {
|
||||
existing.Quantity += "+" + ing.Quantity
|
||||
}
|
||||
} else {
|
||||
recipeIDCopy := id
|
||||
aggregated[key] = &ext.ShoppingItem{
|
||||
Name: ing.Name,
|
||||
Quantity: ing.Quantity,
|
||||
Unit: ing.Unit,
|
||||
Purchased: false,
|
||||
RecipeID: &recipeIDCopy,
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// items := make([]ext.ShoppingItem, 0, len(aggregated))
|
||||
// for _, item := range aggregated {
|
||||
// items = append(items, *item)
|
||||
// }
|
||||
items := make([]ext.ShoppingItem, 0, len(aggregated))
|
||||
for _, item := range aggregated {
|
||||
items = append(items, *item)
|
||||
}
|
||||
|
||||
// itemsJSON, err := json.Marshal(items)
|
||||
// if err != nil {
|
||||
// return ext.ShoppingList{}, fmt.Errorf("marshal shopping items: %w", err)
|
||||
// }
|
||||
itemsJSON, err := json.Marshal(items)
|
||||
if err != nil {
|
||||
return ext.ShoppingList{}, fmt.Errorf("marshal shopping items: %w", err)
|
||||
}
|
||||
|
||||
// row := db.pool.QueryRow(ctx, `
|
||||
// insert into shopping_lists (week_start, items)
|
||||
// values ($1, $2::jsonb)
|
||||
// on conflict (week_start) do update set items = excluded.items, updated_at = now()
|
||||
// returning id, created_at, updated_at
|
||||
// `, weekStart, itemsJSON)
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into shopping_lists (week_start, items)
|
||||
values ($1, $2::jsonb)
|
||||
on conflict (week_start) do update set items = excluded.items, updated_at = now()
|
||||
returning id, created_at, updated_at
|
||||
`, weekStart, itemsJSON)
|
||||
|
||||
// var model generatedmodels.ModelPublicShoppingLists
|
||||
// list := ext.ShoppingList{WeekStart: weekStart, Items: items}
|
||||
// if err := row.Scan(&model.ID, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.ShoppingList{}, fmt.Errorf("upsert shopping list: %w", err)
|
||||
// }
|
||||
// list.ID = model.ID.UUID()
|
||||
// list.CreatedAt = model.CreatedAt.Time()
|
||||
// list.UpdatedAt = model.UpdatedAt.Time()
|
||||
// return list, nil
|
||||
// }
|
||||
list := ext.ShoppingList{WeekStart: weekStart, Items: items}
|
||||
if err := row.Scan(&list.ID, &list.CreatedAt, &list.UpdatedAt); err != nil {
|
||||
return ext.ShoppingList{}, fmt.Errorf("upsert shopping list: %w", err)
|
||||
}
|
||||
return list, nil
|
||||
}
|
||||
|
||||
// func scanRecipeRow(rows interface{ Scan(...any) error }) (ext.Recipe, error) {
|
||||
// var model generatedmodels.ModelPublicRecipes
|
||||
// var tags []string
|
||||
// if err := rows.Scan(&model.ID, &model.Name, &model.Cuisine, &model.PrepTimeMinutes, &model.CookTimeMinutes, &model.Servings,
|
||||
// &model.Ingredients, &model.Instructions, &tags, &model.Rating, &model.Notes, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
// return ext.Recipe{}, fmt.Errorf("scan recipe: %w", err)
|
||||
// }
|
||||
// if tags == nil {
|
||||
// tags = []string{}
|
||||
// }
|
||||
// return recipeFromModel(model, tags), nil
|
||||
// }
|
||||
func scanRecipeRow(rows interface{ Scan(...any) error }) (ext.Recipe, error) {
|
||||
var r ext.Recipe
|
||||
var cuisine, notes *string
|
||||
var ingredientsBytes, instructionsBytes []byte
|
||||
if err := rows.Scan(&r.ID, &r.Name, &cuisine, &r.PrepTimeMinutes, &r.CookTimeMinutes, &r.Servings,
|
||||
&ingredientsBytes, &instructionsBytes, &r.Tags, &r.Rating, ¬es, &r.CreatedAt, &r.UpdatedAt); err != nil {
|
||||
return ext.Recipe{}, fmt.Errorf("scan recipe: %w", err)
|
||||
}
|
||||
r.Cuisine = strVal(cuisine)
|
||||
r.Notes = strVal(notes)
|
||||
if r.Tags == nil {
|
||||
r.Tags = []string{}
|
||||
}
|
||||
if err := json.Unmarshal(ingredientsBytes, &r.Ingredients); err != nil {
|
||||
r.Ingredients = []ext.Ingredient{}
|
||||
}
|
||||
if err := json.Unmarshal(instructionsBytes, &r.Instructions); err != nil {
|
||||
r.Instructions = []string{}
|
||||
}
|
||||
return r, nil
|
||||
}
|
||||
|
||||
@@ -1,540 +0,0 @@
|
||||
package store
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"time"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
func projectFromModel(m generatedmodels.ModelPublicProjects) ext.Project {
|
||||
return ext.Project{
|
||||
ID: m.GUID.UUID(),
|
||||
NumericID: m.ID.Int64(),
|
||||
Name: m.Name.String(),
|
||||
Description: m.Description.String(),
|
||||
CreatedAt: m.CreatedAt.Time(),
|
||||
LastActiveAt: m.LastActiveAt.Time(),
|
||||
}
|
||||
}
|
||||
|
||||
func thoughtFromModel(m generatedmodels.ModelPublicThoughts) (ext.Thought, error) {
|
||||
var metadata ext.ThoughtMetadata
|
||||
if len(m.Metadata) > 0 {
|
||||
if err := json.Unmarshal(m.Metadata, &metadata); err != nil {
|
||||
return ext.Thought{}, fmt.Errorf("decode thought metadata: %w", err)
|
||||
}
|
||||
}
|
||||
|
||||
var projectID *int64
|
||||
if m.ProjectID.Valid {
|
||||
id := m.ProjectID.Int64()
|
||||
projectID = &id
|
||||
}
|
||||
|
||||
var archivedAt *time.Time
|
||||
if m.ArchivedAt.Valid {
|
||||
t := m.ArchivedAt.Time()
|
||||
archivedAt = &t
|
||||
}
|
||||
|
||||
return ext.Thought{
|
||||
ID: m.ID.Int64(),
|
||||
GUID: m.GUID.UUID(),
|
||||
Content: m.Content.String(),
|
||||
Metadata: metadata,
|
||||
ProjectID: projectID,
|
||||
ArchivedAt: archivedAt,
|
||||
CreatedAt: m.CreatedAt.Time(),
|
||||
UpdatedAt: m.UpdatedAt.Time(),
|
||||
}, nil
|
||||
}
|
||||
|
||||
func storedFileFromModel(m generatedmodels.ModelPublicStoredFiles) ext.StoredFile {
|
||||
var thoughtID *int64
|
||||
if m.ThoughtID.Valid {
|
||||
id := m.ThoughtID.Int64()
|
||||
thoughtID = &id
|
||||
}
|
||||
|
||||
var projectID *int64
|
||||
if m.ProjectID.Valid {
|
||||
id := m.ProjectID.Int64()
|
||||
projectID = &id
|
||||
}
|
||||
|
||||
return ext.StoredFile{
|
||||
ID: m.ID.Int64(),
|
||||
GUID: m.GUID.UUID(),
|
||||
ThoughtID: thoughtID,
|
||||
ProjectID: projectID,
|
||||
Name: m.Name.String(),
|
||||
MediaType: m.MediaType.String(),
|
||||
Kind: m.Kind.String(),
|
||||
Encoding: m.Encoding.String(),
|
||||
SizeBytes: m.SizeBytes,
|
||||
SHA256: m.Sha256.String(),
|
||||
Content: m.Content,
|
||||
CreatedAt: m.CreatedAt.Time(),
|
||||
UpdatedAt: m.UpdatedAt.Time(),
|
||||
}
|
||||
}
|
||||
|
||||
// func maintenanceTaskFromModel(m generatedmodels.ModelPublicMaintenanceTasks) ext.MaintenanceTask {
|
||||
// var frequencyDays *int
|
||||
// if m.FrequencyDays.Valid {
|
||||
// n := int(m.FrequencyDays.Int64())
|
||||
// frequencyDays = &n
|
||||
// }
|
||||
|
||||
// var lastCompleted *time.Time
|
||||
// if m.LastCompleted.Valid {
|
||||
// t := m.LastCompleted.Time()
|
||||
// lastCompleted = &t
|
||||
// }
|
||||
|
||||
// var nextDue *time.Time
|
||||
// if m.NextDue.Valid {
|
||||
// t := m.NextDue.Time()
|
||||
// nextDue = &t
|
||||
// }
|
||||
|
||||
// return ext.MaintenanceTask{
|
||||
// ID: m.ID.UUID(),
|
||||
// Name: m.Name.String(),
|
||||
// Category: m.Category.String(),
|
||||
// FrequencyDays: frequencyDays,
|
||||
// LastCompleted: lastCompleted,
|
||||
// NextDue: nextDue,
|
||||
// Priority: m.Priority.String(),
|
||||
// Notes: m.Notes.String(),
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// UpdatedAt: m.UpdatedAt.Time(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func maintenanceLogFromModel(m generatedmodels.ModelPublicMaintenanceLogs) ext.MaintenanceLog {
|
||||
// var cost *float64
|
||||
// if m.Cost.Valid {
|
||||
// v := m.Cost.Float64()
|
||||
// cost = &v
|
||||
// }
|
||||
|
||||
// return ext.MaintenanceLog{
|
||||
// ID: m.ID.UUID(),
|
||||
// TaskID: m.TaskID.UUID(),
|
||||
// CompletedAt: m.CompletedAt.Time(),
|
||||
// PerformedBy: m.PerformedBy.String(),
|
||||
// Cost: cost,
|
||||
// Notes: m.Notes.String(),
|
||||
// NextAction: m.NextAction.String(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func householdItemFromModel(m generatedmodels.ModelPublicHouseholdItems) ext.HouseholdItem {
|
||||
// details := map[string]any{}
|
||||
// if len(m.Details) > 0 {
|
||||
// if err := json.Unmarshal(m.Details, &details); err != nil {
|
||||
// details = map[string]any{}
|
||||
// }
|
||||
// }
|
||||
|
||||
// return ext.HouseholdItem{
|
||||
// ID: m.ID.UUID(),
|
||||
// Name: m.Name.String(),
|
||||
// Category: m.Category.String(),
|
||||
// Location: m.Location.String(),
|
||||
// Details: details,
|
||||
// Notes: m.Notes.String(),
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// UpdatedAt: m.UpdatedAt.Time(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func householdVendorFromModel(m generatedmodels.ModelPublicHouseholdVendors) ext.HouseholdVendor {
|
||||
// var rating *int
|
||||
// if m.Rating.Valid {
|
||||
// v := int(m.Rating.Int64())
|
||||
// rating = &v
|
||||
// }
|
||||
|
||||
// var lastUsed *time.Time
|
||||
// if m.LastUsed.Valid {
|
||||
// t := m.LastUsed.Time()
|
||||
// lastUsed = &t
|
||||
// }
|
||||
|
||||
// return ext.HouseholdVendor{
|
||||
// ID: m.ID.UUID(),
|
||||
// Name: m.Name.String(),
|
||||
// ServiceType: m.ServiceType.String(),
|
||||
// Phone: m.Phone.String(),
|
||||
// Email: m.Email.String(),
|
||||
// Website: m.Website.String(),
|
||||
// Notes: m.Notes.String(),
|
||||
// Rating: rating,
|
||||
// LastUsed: lastUsed,
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func familyMemberFromModel(m generatedmodels.ModelPublicFamilyMembers) ext.FamilyMember {
|
||||
// var birthDate *time.Time
|
||||
// if m.BirthDate.Valid {
|
||||
// t := m.BirthDate.Time()
|
||||
// birthDate = &t
|
||||
// }
|
||||
|
||||
// return ext.FamilyMember{
|
||||
// ID: m.ID.UUID(),
|
||||
// Name: m.Name.String(),
|
||||
// Relationship: m.Relationship.String(),
|
||||
// BirthDate: birthDate,
|
||||
// Notes: m.Notes.String(),
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func activityFromModel(m generatedmodels.ModelPublicActivities, memberName string) ext.Activity {
|
||||
// var familyMemberID *uuid.UUID
|
||||
// if m.FamilyMemberID.Valid {
|
||||
// id := m.FamilyMemberID.UUID()
|
||||
// familyMemberID = &id
|
||||
// }
|
||||
|
||||
// var startDate *time.Time
|
||||
// if m.StartDate.Valid {
|
||||
// t := m.StartDate.Time()
|
||||
// startDate = &t
|
||||
// }
|
||||
|
||||
// var endDate *time.Time
|
||||
// if m.EndDate.Valid {
|
||||
// t := m.EndDate.Time()
|
||||
// endDate = &t
|
||||
// }
|
||||
|
||||
// return ext.Activity{
|
||||
// ID: m.ID.UUID(),
|
||||
// FamilyMemberID: familyMemberID,
|
||||
// MemberName: memberName,
|
||||
// Title: m.Title.String(),
|
||||
// ActivityType: m.ActivityType.String(),
|
||||
// DayOfWeek: m.DayOfWeek.String(),
|
||||
// StartTime: m.StartTime.String(),
|
||||
// EndTime: m.EndTime.String(),
|
||||
// StartDate: startDate,
|
||||
// EndDate: endDate,
|
||||
// Location: m.Location.String(),
|
||||
// Notes: m.Notes.String(),
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func importantDateFromModel(m generatedmodels.ModelPublicImportantDates, memberName string) ext.ImportantDate {
|
||||
// var familyMemberID *uuid.UUID
|
||||
// if m.FamilyMemberID.Valid {
|
||||
// id := m.FamilyMemberID.UUID()
|
||||
// familyMemberID = &id
|
||||
// }
|
||||
|
||||
// return ext.ImportantDate{
|
||||
// ID: m.ID.UUID(),
|
||||
// FamilyMemberID: familyMemberID,
|
||||
// MemberName: memberName,
|
||||
// Title: m.Title.String(),
|
||||
// DateValue: m.DateValue.Time(),
|
||||
// RecurringYearly: m.RecurringYearly,
|
||||
// ReminderDaysBefore: int(m.ReminderDaysBefore),
|
||||
// Notes: m.Notes.String(),
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func professionalContactFromModel(m generatedmodels.ModelPublicProfessionalContacts, tags []string) ext.ProfessionalContact {
|
||||
// var lastContacted *time.Time
|
||||
// if m.LastContacted.Valid {
|
||||
// t := m.LastContacted.Time()
|
||||
// lastContacted = &t
|
||||
// }
|
||||
|
||||
// var followUpDate *time.Time
|
||||
// if m.FollowUpDate.Valid {
|
||||
// t := m.FollowUpDate.Time()
|
||||
// followUpDate = &t
|
||||
// }
|
||||
|
||||
// return ext.ProfessionalContact{
|
||||
// ID: m.ID.UUID(),
|
||||
// Name: m.Name.String(),
|
||||
// Company: m.Company.String(),
|
||||
// Title: m.Title.String(),
|
||||
// Email: m.Email.String(),
|
||||
// Phone: m.Phone.String(),
|
||||
// LinkedInURL: m.LinkedinURL.String(),
|
||||
// HowWeMet: m.HowWeMet.String(),
|
||||
// Tags: tags,
|
||||
// Notes: m.Notes.String(),
|
||||
// LastContacted: lastContacted,
|
||||
// FollowUpDate: followUpDate,
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// UpdatedAt: m.UpdatedAt.Time(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func contactInteractionFromModel(m generatedmodels.ModelPublicContactInteractions) ext.ContactInteraction {
|
||||
// return ext.ContactInteraction{
|
||||
// ID: m.ID.UUID(),
|
||||
// ContactID: m.ContactID.UUID(),
|
||||
// InteractionType: m.InteractionType.String(),
|
||||
// OccurredAt: m.OccurredAt.Time(),
|
||||
// Summary: m.Summary.String(),
|
||||
// FollowUpNeeded: m.FollowUpNeeded,
|
||||
// FollowUpNotes: m.FollowUpNotes.String(),
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func opportunityFromModel(m generatedmodels.ModelPublicOpportunities) ext.Opportunity {
|
||||
// var contactID *uuid.UUID
|
||||
// if m.ContactID.Valid {
|
||||
// id := m.ContactID.UUID()
|
||||
// contactID = &id
|
||||
// }
|
||||
|
||||
// var value *float64
|
||||
// if m.Value.Valid {
|
||||
// v := m.Value.Float64()
|
||||
// value = &v
|
||||
// }
|
||||
|
||||
// var expectedCloseDate *time.Time
|
||||
// if m.ExpectedCloseDate.Valid {
|
||||
// t := m.ExpectedCloseDate.Time()
|
||||
// expectedCloseDate = &t
|
||||
// }
|
||||
|
||||
// return ext.Opportunity{
|
||||
// ID: m.ID.UUID(),
|
||||
// ContactID: contactID,
|
||||
// Title: m.Title.String(),
|
||||
// Description: m.Description.String(),
|
||||
// Stage: m.Stage.String(),
|
||||
// Value: value,
|
||||
// ExpectedCloseDate: expectedCloseDate,
|
||||
// Notes: m.Notes.String(),
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// UpdatedAt: m.UpdatedAt.Time(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func recipeFromModel(m generatedmodels.ModelPublicRecipes, tags []string) ext.Recipe {
|
||||
// var prepTimeMinutes *int
|
||||
// if m.PrepTimeMinutes.Valid {
|
||||
// v := int(m.PrepTimeMinutes.Int64())
|
||||
// prepTimeMinutes = &v
|
||||
// }
|
||||
|
||||
// var cookTimeMinutes *int
|
||||
// if m.CookTimeMinutes.Valid {
|
||||
// v := int(m.CookTimeMinutes.Int64())
|
||||
// cookTimeMinutes = &v
|
||||
// }
|
||||
|
||||
// var servings *int
|
||||
// if m.Servings.Valid {
|
||||
// v := int(m.Servings.Int64())
|
||||
// servings = &v
|
||||
// }
|
||||
|
||||
// var rating *int
|
||||
// if m.Rating.Valid {
|
||||
// v := int(m.Rating.Int64())
|
||||
// rating = &v
|
||||
// }
|
||||
|
||||
// recipe := ext.Recipe{
|
||||
// ID: m.ID.UUID(),
|
||||
// Name: m.Name.String(),
|
||||
// Cuisine: m.Cuisine.String(),
|
||||
// PrepTimeMinutes: prepTimeMinutes,
|
||||
// CookTimeMinutes: cookTimeMinutes,
|
||||
// Servings: servings,
|
||||
// Tags: tags,
|
||||
// Rating: rating,
|
||||
// Notes: m.Notes.String(),
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// UpdatedAt: m.UpdatedAt.Time(),
|
||||
// }
|
||||
|
||||
// if err := json.Unmarshal(m.Ingredients, &recipe.Ingredients); err != nil {
|
||||
// recipe.Ingredients = []ext.Ingredient{}
|
||||
// }
|
||||
// if err := json.Unmarshal(m.Instructions, &recipe.Instructions); err != nil {
|
||||
// recipe.Instructions = []string{}
|
||||
// }
|
||||
// return recipe
|
||||
// }
|
||||
|
||||
// func mealPlanEntryFromModel(m generatedmodels.ModelPublicMealPlans, recipeName string) ext.MealPlanEntry {
|
||||
// var recipeID *uuid.UUID
|
||||
// if m.RecipeID.Valid {
|
||||
// id := m.RecipeID.UUID()
|
||||
// recipeID = &id
|
||||
// }
|
||||
|
||||
// var servings *int
|
||||
// if m.Servings.Valid {
|
||||
// v := int(m.Servings.Int64())
|
||||
// servings = &v
|
||||
// }
|
||||
|
||||
// return ext.MealPlanEntry{
|
||||
// ID: m.ID.UUID(),
|
||||
// WeekStart: m.WeekStart.Time(),
|
||||
// DayOfWeek: m.DayOfWeek.String(),
|
||||
// MealType: m.MealType.String(),
|
||||
// RecipeID: recipeID,
|
||||
// RecipeName: recipeName,
|
||||
// CustomMeal: m.CustomMeal.String(),
|
||||
// Servings: servings,
|
||||
// Notes: m.Notes.String(),
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// }
|
||||
// }
|
||||
|
||||
// func shoppingListFromModel(m generatedmodels.ModelPublicShoppingLists) ext.ShoppingList {
|
||||
// list := ext.ShoppingList{
|
||||
// ID: m.ID.UUID(),
|
||||
// WeekStart: m.WeekStart.Time(),
|
||||
// Notes: m.Notes.String(),
|
||||
// CreatedAt: m.CreatedAt.Time(),
|
||||
// UpdatedAt: m.UpdatedAt.Time(),
|
||||
// }
|
||||
// if err := json.Unmarshal(m.Items, &list.Items); err != nil {
|
||||
// list.Items = []ext.ShoppingItem{}
|
||||
// }
|
||||
// return list
|
||||
// }
|
||||
|
||||
func planFromModel(m generatedmodels.ModelPublicPlans, tags []string) ext.Plan {
|
||||
var projectID *int64
|
||||
if m.ProjectID.Valid {
|
||||
id := m.ProjectID.Int64()
|
||||
projectID = &id
|
||||
}
|
||||
|
||||
var dueDate *time.Time
|
||||
if m.DueDate.Valid {
|
||||
t := m.DueDate.Time()
|
||||
dueDate = &t
|
||||
}
|
||||
|
||||
var completedAt *time.Time
|
||||
if m.CompletedAt.Valid {
|
||||
t := m.CompletedAt.Time()
|
||||
completedAt = &t
|
||||
}
|
||||
|
||||
var lastReviewedAt *time.Time
|
||||
if m.LastReviewedAt.Valid {
|
||||
t := m.LastReviewedAt.Time()
|
||||
lastReviewedAt = &t
|
||||
}
|
||||
|
||||
var supersedesPlanID *int64
|
||||
if m.SupersedesPlanID.Valid {
|
||||
id := m.SupersedesPlanID.Int64()
|
||||
supersedesPlanID = &id
|
||||
}
|
||||
|
||||
return ext.Plan{
|
||||
ID: m.ID.Int64(),
|
||||
GUID: m.GUID.UUID(),
|
||||
Title: m.Title.String(),
|
||||
Description: m.Description.String(),
|
||||
Status: ext.PlanStatus(m.Status.String()),
|
||||
Priority: ext.PlanPriority(m.Priority.String()),
|
||||
ProjectID: projectID,
|
||||
Owner: m.Owner.String(),
|
||||
DueDate: dueDate,
|
||||
CompletedAt: completedAt,
|
||||
ReviewedBy: m.ReviewedBy.String(),
|
||||
LastReviewedAt: lastReviewedAt,
|
||||
SupersedesPlanID: supersedesPlanID,
|
||||
Tags: tags,
|
||||
CreatedAt: m.CreatedAt.Time(),
|
||||
UpdatedAt: m.UpdatedAt.Time(),
|
||||
}
|
||||
}
|
||||
|
||||
func learningFromModel(m generatedmodels.ModelPublicLearnings, tags []string) ext.Learning {
|
||||
var projectID *int64
|
||||
if m.ProjectID.Valid {
|
||||
id := m.ProjectID.Int64()
|
||||
projectID = &id
|
||||
}
|
||||
|
||||
var relatedThoughtID *int64
|
||||
if m.RelatedThoughtID.Valid {
|
||||
id := m.RelatedThoughtID.Int64()
|
||||
relatedThoughtID = &id
|
||||
}
|
||||
|
||||
var relatedSkillID *int64
|
||||
if m.RelatedSkillID.Valid {
|
||||
id := m.RelatedSkillID.Int64()
|
||||
relatedSkillID = &id
|
||||
}
|
||||
|
||||
var duplicateOfLearningID *int64
|
||||
if m.DuplicateOfLearningID.Valid {
|
||||
id := m.DuplicateOfLearningID.Int64()
|
||||
duplicateOfLearningID = &id
|
||||
}
|
||||
|
||||
var supersedesLearningID *int64
|
||||
if m.SupersedesLearningID.Valid {
|
||||
id := m.SupersedesLearningID.Int64()
|
||||
supersedesLearningID = &id
|
||||
}
|
||||
|
||||
var reviewedBy *string
|
||||
if m.ReviewedBy.Valid {
|
||||
value := m.ReviewedBy.String()
|
||||
reviewedBy = &value
|
||||
}
|
||||
|
||||
var reviewedAt *time.Time
|
||||
if m.ReviewedAt.Valid {
|
||||
t := m.ReviewedAt.Time()
|
||||
reviewedAt = &t
|
||||
}
|
||||
|
||||
return ext.Learning{
|
||||
ID: m.ID.Int64(),
|
||||
GUID: m.GUID.UUID(),
|
||||
Summary: m.Summary.String(),
|
||||
Details: m.Details.String(),
|
||||
Category: m.Category.String(),
|
||||
Area: m.Area.String(),
|
||||
Status: ext.LearningStatus(m.Status.String()),
|
||||
Priority: ext.LearningPriority(m.Priority.String()),
|
||||
Confidence: ext.LearningEvidenceLevel(m.Confidence.String()),
|
||||
ActionRequired: m.ActionRequired,
|
||||
SourceType: m.SourceType.String(),
|
||||
SourceRef: m.SourceRef.String(),
|
||||
ProjectID: projectID,
|
||||
RelatedThoughtID: relatedThoughtID,
|
||||
RelatedSkillID: relatedSkillID,
|
||||
ReviewedBy: reviewedBy,
|
||||
ReviewedAt: reviewedAt,
|
||||
DuplicateOfLearningID: duplicateOfLearningID,
|
||||
SupersedesLearningID: supersedesLearningID,
|
||||
Tags: tags,
|
||||
CreatedAt: m.CreatedAt.Time(),
|
||||
UpdatedAt: m.UpdatedAt.Time(),
|
||||
}
|
||||
}
|
||||
@@ -1,478 +0,0 @@
|
||||
package store
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"strings"
|
||||
|
||||
"github.com/jackc/pgx/v5"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
const planColumns = `
|
||||
id, guid, title, description, status, priority, project_id, owner, due_date,
|
||||
completed_at, reviewed_by, last_reviewed_at, supersedes_plan_id, tags::text[], created_at, updated_at`
|
||||
|
||||
func (db *DB) CreatePlan(ctx context.Context, plan ext.Plan) (ext.Plan, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into plans (title, description, status, priority, project_id, owner, due_date,
|
||||
completed_at, reviewed_by, last_reviewed_at, supersedes_plan_id, tags)
|
||||
values ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12)
|
||||
returning`+planColumns,
|
||||
strings.TrimSpace(plan.Title),
|
||||
strings.TrimSpace(plan.Description),
|
||||
string(plan.Status),
|
||||
string(plan.Priority),
|
||||
plan.ProjectID,
|
||||
nullableText(plan.Owner),
|
||||
plan.DueDate,
|
||||
plan.CompletedAt,
|
||||
nullableText(plan.ReviewedBy),
|
||||
plan.LastReviewedAt,
|
||||
plan.SupersedesPlanID,
|
||||
plan.Tags,
|
||||
)
|
||||
return scanPlan(row)
|
||||
}
|
||||
|
||||
func (db *DB) GetPlan(ctx context.Context, id int64) (ext.Plan, error) {
|
||||
row := db.pool.QueryRow(ctx, `select`+planColumns+` from plans where id = $1`, id)
|
||||
plan, err := scanPlan(row)
|
||||
if err != nil {
|
||||
if err == pgx.ErrNoRows {
|
||||
return ext.Plan{}, fmt.Errorf("plan not found: %d", id)
|
||||
}
|
||||
return ext.Plan{}, fmt.Errorf("get plan: %w", err)
|
||||
}
|
||||
return plan, nil
|
||||
}
|
||||
|
||||
func (db *DB) GetPlanDetail(ctx context.Context, id int64) (ext.PlanDetail, error) {
|
||||
plan, err := db.GetPlan(ctx, id)
|
||||
if err != nil {
|
||||
return ext.PlanDetail{}, err
|
||||
}
|
||||
|
||||
dependsOn, err := db.listPlansByQuery(ctx, `
|
||||
select`+planColumns+`
|
||||
from plans p
|
||||
join plan_dependencies pd on pd.depends_on_plan_id = p.id
|
||||
where pd.plan_id = $1 order by p.title`, id)
|
||||
if err != nil {
|
||||
return ext.PlanDetail{}, fmt.Errorf("get plan depends_on: %w", err)
|
||||
}
|
||||
|
||||
blocks, err := db.listPlansByQuery(ctx, `
|
||||
select`+planColumns+`
|
||||
from plans p
|
||||
join plan_dependencies pd on pd.plan_id = p.id
|
||||
where pd.depends_on_plan_id = $1 order by p.title`, id)
|
||||
if err != nil {
|
||||
return ext.PlanDetail{}, fmt.Errorf("get plan blocks: %w", err)
|
||||
}
|
||||
|
||||
related, err := db.listPlansByQuery(ctx, `
|
||||
select`+planColumns+`
|
||||
from plans p
|
||||
where p.id in (
|
||||
select plan_b_id from plan_related_plans where plan_a_id = $1
|
||||
union
|
||||
select plan_a_id from plan_related_plans where plan_b_id = $1
|
||||
) order by p.title`, id)
|
||||
if err != nil {
|
||||
return ext.PlanDetail{}, fmt.Errorf("get plan related: %w", err)
|
||||
}
|
||||
|
||||
skills, err := db.ListPlanSkills(ctx, id)
|
||||
if err != nil {
|
||||
return ext.PlanDetail{}, fmt.Errorf("get plan skills: %w", err)
|
||||
}
|
||||
|
||||
guardrails, err := db.ListPlanGuardrails(ctx, id)
|
||||
if err != nil {
|
||||
return ext.PlanDetail{}, fmt.Errorf("get plan guardrails: %w", err)
|
||||
}
|
||||
|
||||
return ext.PlanDetail{
|
||||
Plan: plan,
|
||||
DependsOn: dependsOn,
|
||||
Blocks: blocks,
|
||||
RelatedPlans: related,
|
||||
Skills: skills,
|
||||
Guardrails: guardrails,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func (db *DB) UpdatePlan(ctx context.Context, id int64, u ext.PlanUpdate) (ext.Plan, error) {
|
||||
sets := []string{"updated_at = now()"}
|
||||
args := []any{}
|
||||
|
||||
if u.Title != nil {
|
||||
args = append(args, strings.TrimSpace(*u.Title))
|
||||
sets = append(sets, fmt.Sprintf("title = $%d", len(args)))
|
||||
}
|
||||
if u.Description != nil {
|
||||
args = append(args, strings.TrimSpace(*u.Description))
|
||||
sets = append(sets, fmt.Sprintf("description = $%d", len(args)))
|
||||
}
|
||||
if u.Status != nil {
|
||||
args = append(args, strings.TrimSpace(*u.Status))
|
||||
sets = append(sets, fmt.Sprintf("status = $%d", len(args)))
|
||||
}
|
||||
if u.Priority != nil {
|
||||
args = append(args, strings.TrimSpace(*u.Priority))
|
||||
sets = append(sets, fmt.Sprintf("priority = $%d", len(args)))
|
||||
}
|
||||
if u.Owner != nil {
|
||||
args = append(args, nullableText(*u.Owner))
|
||||
sets = append(sets, fmt.Sprintf("owner = $%d", len(args)))
|
||||
}
|
||||
if u.ClearDueDate {
|
||||
sets = append(sets, "due_date = null")
|
||||
} else if u.DueDate != nil {
|
||||
args = append(args, *u.DueDate)
|
||||
sets = append(sets, fmt.Sprintf("due_date = $%d", len(args)))
|
||||
}
|
||||
if u.ClearCompletedAt {
|
||||
sets = append(sets, "completed_at = null")
|
||||
} else if u.CompletedAt != nil {
|
||||
args = append(args, *u.CompletedAt)
|
||||
sets = append(sets, fmt.Sprintf("completed_at = $%d", len(args)))
|
||||
}
|
||||
if u.ReviewedBy != nil {
|
||||
args = append(args, nullableText(*u.ReviewedBy))
|
||||
sets = append(sets, fmt.Sprintf("reviewed_by = $%d", len(args)))
|
||||
}
|
||||
if u.MarkReviewed {
|
||||
sets = append(sets, "last_reviewed_at = now()")
|
||||
}
|
||||
if u.ClearSupersedesPlanID {
|
||||
sets = append(sets, "supersedes_plan_id = null")
|
||||
} else if u.SupersedesPlanID != nil {
|
||||
args = append(args, *u.SupersedesPlanID)
|
||||
sets = append(sets, fmt.Sprintf("supersedes_plan_id = $%d", len(args)))
|
||||
}
|
||||
if u.Tags != nil {
|
||||
args = append(args, *u.Tags)
|
||||
sets = append(sets, fmt.Sprintf("tags = $%d", len(args)))
|
||||
}
|
||||
|
||||
args = append(args, id)
|
||||
query := fmt.Sprintf(
|
||||
"update plans set %s where id = $%d returning%s",
|
||||
strings.Join(sets, ", "), len(args), planColumns,
|
||||
)
|
||||
|
||||
row := db.pool.QueryRow(ctx, query, args...)
|
||||
plan, err := scanPlan(row)
|
||||
if err != nil {
|
||||
if err == pgx.ErrNoRows {
|
||||
return ext.Plan{}, fmt.Errorf("plan not found: %d", id)
|
||||
}
|
||||
return ext.Plan{}, fmt.Errorf("update plan: %w", err)
|
||||
}
|
||||
return plan, nil
|
||||
}
|
||||
|
||||
func (db *DB) DeletePlan(ctx context.Context, id int64) error {
|
||||
tag, err := db.pool.Exec(ctx, `delete from plans where id = $1`, id)
|
||||
if err != nil {
|
||||
return fmt.Errorf("delete plan: %w", err)
|
||||
}
|
||||
if tag.RowsAffected() == 0 {
|
||||
return fmt.Errorf("plan not found")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) ListPlans(ctx context.Context, filter ext.PlanFilter) ([]ext.Plan, error) {
|
||||
args := make([]any, 0, 8)
|
||||
conditions := make([]string, 0, 8)
|
||||
|
||||
if filter.ProjectID != nil {
|
||||
args = append(args, *filter.ProjectID)
|
||||
conditions = append(conditions, fmt.Sprintf("project_id = $%d", len(args)))
|
||||
}
|
||||
if v := strings.TrimSpace(filter.Status); v != "" {
|
||||
args = append(args, v)
|
||||
conditions = append(conditions, fmt.Sprintf("status = $%d", len(args)))
|
||||
}
|
||||
if v := strings.TrimSpace(filter.Priority); v != "" {
|
||||
args = append(args, v)
|
||||
conditions = append(conditions, fmt.Sprintf("priority = $%d", len(args)))
|
||||
}
|
||||
if v := strings.TrimSpace(filter.Owner); v != "" {
|
||||
args = append(args, v)
|
||||
conditions = append(conditions, fmt.Sprintf("owner = $%d", len(args)))
|
||||
}
|
||||
if v := strings.TrimSpace(filter.Tag); v != "" {
|
||||
args = append(args, v)
|
||||
conditions = append(conditions, fmt.Sprintf("$%d = any(tags)", len(args)))
|
||||
}
|
||||
if v := strings.TrimSpace(filter.Query); v != "" {
|
||||
args = append(args, v)
|
||||
conditions = append(conditions, fmt.Sprintf(
|
||||
"to_tsvector('simple', title || ' ' || coalesce(description, '')) @@ websearch_to_tsquery('simple', $%d)", len(args)))
|
||||
}
|
||||
|
||||
query := "select" + planColumns + " from plans"
|
||||
if len(conditions) > 0 {
|
||||
query += " where " + strings.Join(conditions, " and ")
|
||||
}
|
||||
query += " order by updated_at desc"
|
||||
if filter.Limit > 0 {
|
||||
args = append(args, filter.Limit)
|
||||
query += fmt.Sprintf(" limit $%d", len(args))
|
||||
}
|
||||
|
||||
return db.listPlansByQuery(ctx, query, args...)
|
||||
}
|
||||
|
||||
// Dependencies
|
||||
|
||||
func (db *DB) AddPlanDependency(ctx context.Context, planID, dependsOnPlanID int64) error {
|
||||
_, err := db.pool.Exec(ctx, `
|
||||
insert into plan_dependencies (plan_id, depends_on_plan_id)
|
||||
values ($1, $2)
|
||||
on conflict do nothing
|
||||
`, planID, dependsOnPlanID)
|
||||
if err != nil {
|
||||
return fmt.Errorf("add plan dependency: %w", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) RemovePlanDependency(ctx context.Context, planID, dependsOnPlanID int64) error {
|
||||
tag, err := db.pool.Exec(ctx, `
|
||||
delete from plan_dependencies where plan_id = $1 and depends_on_plan_id = $2
|
||||
`, planID, dependsOnPlanID)
|
||||
if err != nil {
|
||||
return fmt.Errorf("remove plan dependency: %w", err)
|
||||
}
|
||||
if tag.RowsAffected() == 0 {
|
||||
return fmt.Errorf("plan dependency not found")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Related Plans
|
||||
|
||||
func (db *DB) AddRelatedPlan(ctx context.Context, planAID, planBID int64) error {
|
||||
a, b := canonicalPlanPair(planAID, planBID)
|
||||
_, err := db.pool.Exec(ctx, `
|
||||
insert into plan_related_plans (plan_a_id, plan_b_id)
|
||||
values ($1, $2)
|
||||
on conflict do nothing
|
||||
`, a, b)
|
||||
if err != nil {
|
||||
return fmt.Errorf("add related plan: %w", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) RemoveRelatedPlan(ctx context.Context, planAID, planBID int64) error {
|
||||
a, b := canonicalPlanPair(planAID, planBID)
|
||||
tag, err := db.pool.Exec(ctx, `
|
||||
delete from plan_related_plans where plan_a_id = $1 and plan_b_id = $2
|
||||
`, a, b)
|
||||
if err != nil {
|
||||
return fmt.Errorf("remove related plan: %w", err)
|
||||
}
|
||||
if tag.RowsAffected() == 0 {
|
||||
return fmt.Errorf("related plan link not found")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Plan Skills
|
||||
|
||||
func (db *DB) AddPlanSkill(ctx context.Context, planID, skillID int64) error {
|
||||
_, err := db.pool.Exec(ctx, `
|
||||
insert into plan_skills (plan_id, skill_id) values ($1, $2) on conflict do nothing
|
||||
`, planID, skillID)
|
||||
if err != nil {
|
||||
return fmt.Errorf("add plan skill: %w", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) RemovePlanSkill(ctx context.Context, planID, skillID int64) error {
|
||||
tag, err := db.pool.Exec(ctx, `
|
||||
delete from plan_skills where plan_id = $1 and skill_id = $2
|
||||
`, planID, skillID)
|
||||
if err != nil {
|
||||
return fmt.Errorf("remove plan skill: %w", err)
|
||||
}
|
||||
if tag.RowsAffected() == 0 {
|
||||
return fmt.Errorf("plan skill link not found")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) ListPlanSkills(ctx context.Context, planID int64) ([]ext.AgentSkill, error) {
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select s.id, s.name, s.description, s.content, s.tags::text[], s.created_at, s.updated_at
|
||||
from agent_skills s
|
||||
join plan_skills ps on ps.skill_id = s.id
|
||||
where ps.plan_id = $1
|
||||
order by s.name
|
||||
`, planID)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("list plan skills: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
var skills []ext.AgentSkill
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicAgentSkills
|
||||
var tags []string
|
||||
if err := rows.Scan(&model.ID, &model.Name, &model.Description, &model.Content, &tags, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan plan skill: %w", err)
|
||||
}
|
||||
s := ext.AgentSkill{
|
||||
ID: model.ID.Int64(),
|
||||
Name: model.Name.String(),
|
||||
Description: model.Description.String(),
|
||||
Content: model.Content.String(),
|
||||
Tags: tags,
|
||||
CreatedAt: model.CreatedAt.Time(),
|
||||
UpdatedAt: model.UpdatedAt.Time(),
|
||||
}
|
||||
if s.Tags == nil {
|
||||
s.Tags = []string{}
|
||||
}
|
||||
skills = append(skills, s)
|
||||
}
|
||||
return skills, rows.Err()
|
||||
}
|
||||
|
||||
// Plan Guardrails
|
||||
|
||||
func (db *DB) AddPlanGuardrail(ctx context.Context, planID, guardrailID int64) error {
|
||||
_, err := db.pool.Exec(ctx, `
|
||||
insert into plan_guardrails (plan_id, guardrail_id) values ($1, $2) on conflict do nothing
|
||||
`, planID, guardrailID)
|
||||
if err != nil {
|
||||
return fmt.Errorf("add plan guardrail: %w", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) RemovePlanGuardrail(ctx context.Context, planID, guardrailID int64) error {
|
||||
tag, err := db.pool.Exec(ctx, `
|
||||
delete from plan_guardrails where plan_id = $1 and guardrail_id = $2
|
||||
`, planID, guardrailID)
|
||||
if err != nil {
|
||||
return fmt.Errorf("remove plan guardrail: %w", err)
|
||||
}
|
||||
if tag.RowsAffected() == 0 {
|
||||
return fmt.Errorf("plan guardrail link not found")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) ListPlanGuardrails(ctx context.Context, planID int64) ([]ext.AgentGuardrail, error) {
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select g.id, g.name, g.description, g.content, g.severity, g.tags::text[], g.created_at, g.updated_at
|
||||
from agent_guardrails g
|
||||
join plan_guardrails pg on pg.guardrail_id = g.id
|
||||
where pg.plan_id = $1
|
||||
order by g.name
|
||||
`, planID)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("list plan guardrails: %w", err)
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
var guardrails []ext.AgentGuardrail
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicAgentGuardrails
|
||||
var tags []string
|
||||
if err := rows.Scan(&model.ID, &model.Name, &model.Description, &model.Content, &model.Severity, &tags, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan plan guardrail: %w", err)
|
||||
}
|
||||
g := ext.AgentGuardrail{
|
||||
ID: model.ID.Int64(),
|
||||
Name: model.Name.String(),
|
||||
Description: model.Description.String(),
|
||||
Content: model.Content.String(),
|
||||
Severity: model.Severity.String(),
|
||||
Tags: tags,
|
||||
CreatedAt: model.CreatedAt.Time(),
|
||||
UpdatedAt: model.UpdatedAt.Time(),
|
||||
}
|
||||
if g.Tags == nil {
|
||||
g.Tags = []string{}
|
||||
}
|
||||
guardrails = append(guardrails, g)
|
||||
}
|
||||
return guardrails, rows.Err()
|
||||
}
|
||||
|
||||
// helpers
|
||||
|
||||
type planScanner interface {
|
||||
Scan(dest ...any) error
|
||||
}
|
||||
|
||||
func scanPlan(row planScanner) (ext.Plan, error) {
|
||||
var model generatedmodels.ModelPublicPlans
|
||||
var tags []string
|
||||
err := row.Scan(
|
||||
&model.ID,
|
||||
&model.GUID,
|
||||
&model.Title,
|
||||
&model.Description,
|
||||
&model.Status,
|
||||
&model.Priority,
|
||||
&model.ProjectID,
|
||||
&model.Owner,
|
||||
&model.DueDate,
|
||||
&model.CompletedAt,
|
||||
&model.ReviewedBy,
|
||||
&model.LastReviewedAt,
|
||||
&model.SupersedesPlanID,
|
||||
&tags,
|
||||
&model.CreatedAt,
|
||||
&model.UpdatedAt,
|
||||
)
|
||||
if err != nil {
|
||||
return ext.Plan{}, err
|
||||
}
|
||||
if tags == nil {
|
||||
tags = []string{}
|
||||
}
|
||||
return planFromModel(model, tags), nil
|
||||
}
|
||||
|
||||
func (db *DB) listPlansByQuery(ctx context.Context, query string, args ...any) ([]ext.Plan, error) {
|
||||
rows, err := db.pool.Query(ctx, query, args...)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer rows.Close()
|
||||
|
||||
plans := make([]ext.Plan, 0)
|
||||
for rows.Next() {
|
||||
plan, err := scanPlan(rows)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("scan plan: %w", err)
|
||||
}
|
||||
plans = append(plans, plan)
|
||||
}
|
||||
if err := rows.Err(); err != nil {
|
||||
return nil, fmt.Errorf("iterate plans: %w", err)
|
||||
}
|
||||
return plans, nil
|
||||
}
|
||||
|
||||
// canonicalPlanPair ensures the smaller ID is always plan_a_id to prevent duplicates.
|
||||
func canonicalPlanPair(a, b int64) (int64, int64) {
|
||||
if a <= b {
|
||||
return a, b
|
||||
}
|
||||
return b, a
|
||||
}
|
||||
|
||||
+17
-22
@@ -8,7 +8,6 @@ import (
|
||||
"github.com/google/uuid"
|
||||
"github.com/jackc/pgx/v5"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
thoughttypes "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
@@ -16,14 +15,14 @@ func (db *DB) CreateProject(ctx context.Context, name, description string) (thou
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into projects (name, description)
|
||||
values ($1, $2)
|
||||
returning id, guid, name, description, created_at, last_active_at
|
||||
returning guid, name, description, created_at, last_active_at
|
||||
`, name, description)
|
||||
|
||||
var model generatedmodels.ModelPublicProjects
|
||||
if err := row.Scan(&model.ID, &model.GUID, &model.Name, &model.Description, &model.CreatedAt, &model.LastActiveAt); err != nil {
|
||||
var project thoughttypes.Project
|
||||
if err := row.Scan(&project.ID, &project.Name, &project.Description, &project.CreatedAt, &project.LastActiveAt); err != nil {
|
||||
return thoughttypes.Project{}, fmt.Errorf("create project: %w", err)
|
||||
}
|
||||
return projectFromModel(model), nil
|
||||
return project, nil
|
||||
}
|
||||
|
||||
func (db *DB) GetProject(ctx context.Context, nameOrID string) (thoughttypes.Project, error) {
|
||||
@@ -46,7 +45,7 @@ func (db *DB) GetProject(ctx context.Context, nameOrID string) (thoughttypes.Pro
|
||||
|
||||
func (db *DB) getProjectByGUID(ctx context.Context, id uuid.UUID) (thoughttypes.Project, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, guid, name, description, created_at, last_active_at
|
||||
select guid, name, description, created_at, last_active_at
|
||||
from projects
|
||||
where guid = $1
|
||||
`, id)
|
||||
@@ -55,7 +54,7 @@ func (db *DB) getProjectByGUID(ctx context.Context, id uuid.UUID) (thoughttypes.
|
||||
|
||||
func (db *DB) getProjectByName(ctx context.Context, name string) (thoughttypes.Project, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, guid, name, description, created_at, last_active_at
|
||||
select guid, name, description, created_at, last_active_at
|
||||
from projects
|
||||
where name = $1
|
||||
`, name)
|
||||
@@ -63,22 +62,22 @@ func (db *DB) getProjectByName(ctx context.Context, name string) (thoughttypes.P
|
||||
}
|
||||
|
||||
func scanProject(row pgx.Row) (thoughttypes.Project, error) {
|
||||
var model generatedmodels.ModelPublicProjects
|
||||
if err := row.Scan(&model.ID, &model.GUID, &model.Name, &model.Description, &model.CreatedAt, &model.LastActiveAt); err != nil {
|
||||
var project thoughttypes.Project
|
||||
if err := row.Scan(&project.ID, &project.Name, &project.Description, &project.CreatedAt, &project.LastActiveAt); err != nil {
|
||||
if err == pgx.ErrNoRows {
|
||||
return thoughttypes.Project{}, err
|
||||
}
|
||||
return thoughttypes.Project{}, fmt.Errorf("get project: %w", err)
|
||||
}
|
||||
return projectFromModel(model), nil
|
||||
return project, nil
|
||||
}
|
||||
|
||||
func (db *DB) ListProjects(ctx context.Context) ([]thoughttypes.ProjectSummary, error) {
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select p.id, p.guid, p.name, p.description, p.created_at, p.last_active_at, count(t.id) as thought_count
|
||||
select p.guid, p.name, p.description, p.created_at, p.last_active_at, count(t.id) as thought_count
|
||||
from projects p
|
||||
left join thoughts t on t.project_id = p.id and t.archived_at is null
|
||||
group by p.id, p.guid, p.name, p.description, p.created_at, p.last_active_at
|
||||
left join thoughts t on t.project_id = p.guid and t.archived_at is null
|
||||
group by p.guid, p.name, p.description, p.created_at, p.last_active_at
|
||||
order by p.last_active_at desc, p.created_at desc
|
||||
`)
|
||||
if err != nil {
|
||||
@@ -88,15 +87,11 @@ func (db *DB) ListProjects(ctx context.Context) ([]thoughttypes.ProjectSummary,
|
||||
|
||||
projects := make([]thoughttypes.ProjectSummary, 0)
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicProjects
|
||||
var thoughtCount int
|
||||
if err := rows.Scan(&model.ID, &model.GUID, &model.Name, &model.Description, &model.CreatedAt, &model.LastActiveAt, &thoughtCount); err != nil {
|
||||
var project thoughttypes.ProjectSummary
|
||||
if err := rows.Scan(&project.ID, &project.Name, &project.Description, &project.CreatedAt, &project.LastActiveAt, &project.ThoughtCount); err != nil {
|
||||
return nil, fmt.Errorf("scan project summary: %w", err)
|
||||
}
|
||||
projects = append(projects, thoughttypes.ProjectSummary{
|
||||
Project: projectFromModel(model),
|
||||
ThoughtCount: thoughtCount,
|
||||
})
|
||||
projects = append(projects, project)
|
||||
}
|
||||
if err := rows.Err(); err != nil {
|
||||
return nil, fmt.Errorf("iterate projects: %w", err)
|
||||
@@ -104,8 +99,8 @@ func (db *DB) ListProjects(ctx context.Context) ([]thoughttypes.ProjectSummary,
|
||||
return projects, nil
|
||||
}
|
||||
|
||||
func (db *DB) TouchProject(ctx context.Context, id int64) error {
|
||||
tag, err := db.pool.Exec(ctx, `update projects set last_active_at = now() where id = $1`, id)
|
||||
func (db *DB) TouchProject(ctx context.Context, id uuid.UUID) error {
|
||||
tag, err := db.pool.Exec(ctx, `update projects set last_active_at = now() where guid = $1`, id)
|
||||
if err != nil {
|
||||
return fmt.Errorf("touch project: %w", err)
|
||||
}
|
||||
|
||||
+61
-160
@@ -5,7 +5,8 @@ import (
|
||||
"fmt"
|
||||
"strings"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
"github.com/google/uuid"
|
||||
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
@@ -15,38 +16,20 @@ func (db *DB) AddSkill(ctx context.Context, skill ext.AgentSkill) (ext.AgentSkil
|
||||
if skill.Tags == nil {
|
||||
skill.Tags = []string{}
|
||||
}
|
||||
if skill.LanguageTags == nil {
|
||||
skill.LanguageTags = []string{}
|
||||
}
|
||||
if skill.LibraryTags == nil {
|
||||
skill.LibraryTags = []string{}
|
||||
}
|
||||
if skill.FrameworkTags == nil {
|
||||
skill.FrameworkTags = []string{}
|
||||
}
|
||||
if skill.DomainTags == nil {
|
||||
skill.DomainTags = []string{}
|
||||
}
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into agent_skills (name, description, content, tags, language_tags, library_tags, framework_tags, domain_tags)
|
||||
values ($1, $2, $3, $4, $5, $6, $7, $8)
|
||||
returning id, guid, created_at, updated_at
|
||||
`, skill.Name, skill.Description, skill.Content, skill.Tags,
|
||||
skill.LanguageTags, skill.LibraryTags, skill.FrameworkTags, skill.DomainTags)
|
||||
insert into agent_skills (name, description, content, tags)
|
||||
values ($1, $2, $3, $4)
|
||||
returning id, created_at, updated_at
|
||||
`, skill.Name, skill.Description, skill.Content, skill.Tags)
|
||||
|
||||
created := skill
|
||||
var model generatedmodels.ModelPublicAgentSkills
|
||||
if err := row.Scan(&model.ID, &model.GUID, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt, &created.UpdatedAt); err != nil {
|
||||
return ext.AgentSkill{}, fmt.Errorf("insert agent skill: %w", err)
|
||||
}
|
||||
created.ID = model.ID.Int64()
|
||||
created.GUID = model.GUID.UUID()
|
||||
created.CreatedAt = model.CreatedAt.Time()
|
||||
created.UpdatedAt = model.UpdatedAt.Time()
|
||||
return created, nil
|
||||
}
|
||||
|
||||
func (db *DB) RemoveSkill(ctx context.Context, id int64) error {
|
||||
func (db *DB) RemoveSkill(ctx context.Context, id uuid.UUID) error {
|
||||
tag, err := db.pool.Exec(ctx, `delete from agent_skills where id = $1`, id)
|
||||
if err != nil {
|
||||
return fmt.Errorf("delete agent skill: %w", err)
|
||||
@@ -58,11 +41,11 @@ func (db *DB) RemoveSkill(ctx context.Context, id int64) error {
|
||||
}
|
||||
|
||||
func (db *DB) ListSkills(ctx context.Context, tag string) ([]ext.AgentSkill, error) {
|
||||
q := `select id, name, description, content, tags::text[], language_tags::text[], library_tags::text[], framework_tags::text[], domain_tags::text[], created_at, updated_at from agent_skills`
|
||||
q := `select id, name, description, content, tags, created_at, updated_at from agent_skills`
|
||||
args := []any{}
|
||||
if t := strings.TrimSpace(tag); t != "" {
|
||||
args = append(args, t)
|
||||
q += fmt.Sprintf(" where $%d = any(tags) or $%d = any(language_tags) or $%d = any(library_tags) or $%d = any(framework_tags) or $%d = any(domain_tags)", len(args), len(args), len(args), len(args), len(args))
|
||||
q += fmt.Sprintf(" where $%d = any(tags)", len(args))
|
||||
}
|
||||
q += " order by name"
|
||||
|
||||
@@ -74,93 +57,38 @@ func (db *DB) ListSkills(ctx context.Context, tag string) ([]ext.AgentSkill, err
|
||||
|
||||
var skills []ext.AgentSkill
|
||||
for rows.Next() {
|
||||
s, err := scanSkill(rows)
|
||||
if err != nil {
|
||||
var s ext.AgentSkill
|
||||
var desc *string
|
||||
if err := rows.Scan(&s.ID, &s.Name, &desc, &s.Content, &s.Tags, &s.CreatedAt, &s.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan agent skill: %w", err)
|
||||
}
|
||||
s.Description = strVal(desc)
|
||||
if s.Tags == nil {
|
||||
s.Tags = []string{}
|
||||
}
|
||||
skills = append(skills, s)
|
||||
}
|
||||
return skills, rows.Err()
|
||||
}
|
||||
|
||||
const skillSelectCols = `id, name, description, content, tags::text[], language_tags::text[], library_tags::text[], framework_tags::text[], domain_tags::text[], created_at, updated_at`
|
||||
func (db *DB) GetSkill(ctx context.Context, id uuid.UUID) (ext.AgentSkill, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, name, description, content, tags, created_at, updated_at
|
||||
from agent_skills where id = $1
|
||||
`, id)
|
||||
|
||||
type skillScanner interface {
|
||||
Scan(dest ...any) error
|
||||
}
|
||||
|
||||
func scanSkill(row skillScanner) (ext.AgentSkill, error) {
|
||||
var model generatedmodels.ModelPublicAgentSkills
|
||||
var tags, langTags, libTags, fwTags, domTags []string
|
||||
if err := row.Scan(&model.ID, &model.Name, &model.Description, &model.Content, &tags, &langTags, &libTags, &fwTags, &domTags, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
return ext.AgentSkill{}, err
|
||||
}
|
||||
nilToEmpty := func(s []string) []string {
|
||||
if s == nil {
|
||||
return []string{}
|
||||
}
|
||||
return s
|
||||
}
|
||||
return ext.AgentSkill{
|
||||
ID: model.ID.Int64(),
|
||||
Name: model.Name.String(),
|
||||
Description: model.Description.String(),
|
||||
Content: model.Content.String(),
|
||||
Tags: nilToEmpty(tags),
|
||||
LanguageTags: nilToEmpty(langTags),
|
||||
LibraryTags: nilToEmpty(libTags),
|
||||
FrameworkTags: nilToEmpty(fwTags),
|
||||
DomainTags: nilToEmpty(domTags),
|
||||
CreatedAt: model.CreatedAt.Time(),
|
||||
UpdatedAt: model.UpdatedAt.Time(),
|
||||
}, nil
|
||||
}
|
||||
|
||||
func (db *DB) GetSkill(ctx context.Context, id int64) (ext.AgentSkill, error) {
|
||||
row := db.pool.QueryRow(ctx, `select `+skillSelectCols+` from agent_skills where id = $1`, id)
|
||||
s, err := scanSkill(row)
|
||||
if err != nil {
|
||||
var s ext.AgentSkill
|
||||
var desc *string
|
||||
if err := row.Scan(&s.ID, &s.Name, &desc, &s.Content, &s.Tags, &s.CreatedAt, &s.UpdatedAt); err != nil {
|
||||
return ext.AgentSkill{}, fmt.Errorf("get agent skill: %w", err)
|
||||
}
|
||||
return s, nil
|
||||
}
|
||||
|
||||
func (db *DB) GetSkillByName(ctx context.Context, name string) (ext.AgentSkill, error) {
|
||||
row := db.pool.QueryRow(ctx, `select `+skillSelectCols+` from agent_skills where name = $1`, name)
|
||||
s, err := scanSkill(row)
|
||||
if err != nil {
|
||||
return ext.AgentSkill{}, fmt.Errorf("get agent skill by name: %w", err)
|
||||
s.Description = strVal(desc)
|
||||
if s.Tags == nil {
|
||||
s.Tags = []string{}
|
||||
}
|
||||
return s, nil
|
||||
}
|
||||
|
||||
func (db *DB) GetGuardrailByName(ctx context.Context, name string) (ext.AgentGuardrail, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, name, description, content, severity, tags::text[], created_at, updated_at
|
||||
from agent_guardrails where name = $1
|
||||
`, name)
|
||||
|
||||
var model generatedmodels.ModelPublicAgentGuardrails
|
||||
var tags []string
|
||||
if err := row.Scan(&model.ID, &model.Name, &model.Description, &model.Content, &model.Severity, &tags, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
return ext.AgentGuardrail{}, fmt.Errorf("get agent guardrail by name: %w", err)
|
||||
}
|
||||
g := ext.AgentGuardrail{
|
||||
ID: model.ID.Int64(),
|
||||
Name: model.Name.String(),
|
||||
Description: model.Description.String(),
|
||||
Content: model.Content.String(),
|
||||
Severity: model.Severity.String(),
|
||||
Tags: tags,
|
||||
CreatedAt: model.CreatedAt.Time(),
|
||||
UpdatedAt: model.UpdatedAt.Time(),
|
||||
}
|
||||
if g.Tags == nil {
|
||||
g.Tags = []string{}
|
||||
}
|
||||
return g, nil
|
||||
}
|
||||
|
||||
// Agent Guardrails
|
||||
|
||||
func (db *DB) AddGuardrail(ctx context.Context, g ext.AgentGuardrail) (ext.AgentGuardrail, error) {
|
||||
@@ -173,22 +101,17 @@ func (db *DB) AddGuardrail(ctx context.Context, g ext.AgentGuardrail) (ext.Agent
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
insert into agent_guardrails (name, description, content, severity, tags)
|
||||
values ($1, $2, $3, $4, $5)
|
||||
returning id, guid, created_at, updated_at
|
||||
returning id, created_at, updated_at
|
||||
`, g.Name, g.Description, g.Content, g.Severity, g.Tags)
|
||||
|
||||
created := g
|
||||
var model generatedmodels.ModelPublicAgentGuardrails
|
||||
if err := row.Scan(&model.ID, &model.GUID, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt, &created.UpdatedAt); err != nil {
|
||||
return ext.AgentGuardrail{}, fmt.Errorf("insert agent guardrail: %w", err)
|
||||
}
|
||||
created.ID = model.ID.Int64()
|
||||
created.GUID = model.GUID.UUID()
|
||||
created.CreatedAt = model.CreatedAt.Time()
|
||||
created.UpdatedAt = model.UpdatedAt.Time()
|
||||
return created, nil
|
||||
}
|
||||
|
||||
func (db *DB) RemoveGuardrail(ctx context.Context, id int64) error {
|
||||
func (db *DB) RemoveGuardrail(ctx context.Context, id uuid.UUID) error {
|
||||
tag, err := db.pool.Exec(ctx, `delete from agent_guardrails where id = $1`, id)
|
||||
if err != nil {
|
||||
return fmt.Errorf("delete agent guardrail: %w", err)
|
||||
@@ -212,7 +135,7 @@ func (db *DB) ListGuardrails(ctx context.Context, tag, severity string) ([]ext.A
|
||||
conditions = append(conditions, fmt.Sprintf("severity = $%d", len(args)))
|
||||
}
|
||||
|
||||
q := `select id, name, description, content, severity, tags::text[], created_at, updated_at from agent_guardrails`
|
||||
q := `select id, name, description, content, severity, tags, created_at, updated_at from agent_guardrails`
|
||||
if len(conditions) > 0 {
|
||||
q += " where " + strings.Join(conditions, " and ")
|
||||
}
|
||||
@@ -226,21 +149,12 @@ func (db *DB) ListGuardrails(ctx context.Context, tag, severity string) ([]ext.A
|
||||
|
||||
var guardrails []ext.AgentGuardrail
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicAgentGuardrails
|
||||
var tags []string
|
||||
if err := rows.Scan(&model.ID, &model.Name, &model.Description, &model.Content, &model.Severity, &tags, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
var g ext.AgentGuardrail
|
||||
var desc *string
|
||||
if err := rows.Scan(&g.ID, &g.Name, &desc, &g.Content, &g.Severity, &g.Tags, &g.CreatedAt, &g.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan agent guardrail: %w", err)
|
||||
}
|
||||
g := ext.AgentGuardrail{
|
||||
ID: model.ID.Int64(),
|
||||
Name: model.Name.String(),
|
||||
Description: model.Description.String(),
|
||||
Content: model.Content.String(),
|
||||
Severity: model.Severity.String(),
|
||||
Tags: tags,
|
||||
CreatedAt: model.CreatedAt.Time(),
|
||||
UpdatedAt: model.UpdatedAt.Time(),
|
||||
}
|
||||
g.Description = strVal(desc)
|
||||
if g.Tags == nil {
|
||||
g.Tags = []string{}
|
||||
}
|
||||
@@ -249,27 +163,18 @@ func (db *DB) ListGuardrails(ctx context.Context, tag, severity string) ([]ext.A
|
||||
return guardrails, rows.Err()
|
||||
}
|
||||
|
||||
func (db *DB) GetGuardrail(ctx context.Context, id int64) (ext.AgentGuardrail, error) {
|
||||
func (db *DB) GetGuardrail(ctx context.Context, id uuid.UUID) (ext.AgentGuardrail, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, name, description, content, severity, tags::text[], created_at, updated_at
|
||||
select id, name, description, content, severity, tags, created_at, updated_at
|
||||
from agent_guardrails where id = $1
|
||||
`, id)
|
||||
|
||||
var model generatedmodels.ModelPublicAgentGuardrails
|
||||
var tags []string
|
||||
if err := row.Scan(&model.ID, &model.Name, &model.Description, &model.Content, &model.Severity, &tags, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
var g ext.AgentGuardrail
|
||||
var desc *string
|
||||
if err := row.Scan(&g.ID, &g.Name, &desc, &g.Content, &g.Severity, &g.Tags, &g.CreatedAt, &g.UpdatedAt); err != nil {
|
||||
return ext.AgentGuardrail{}, fmt.Errorf("get agent guardrail: %w", err)
|
||||
}
|
||||
g := ext.AgentGuardrail{
|
||||
ID: model.ID.Int64(),
|
||||
Name: model.Name.String(),
|
||||
Description: model.Description.String(),
|
||||
Content: model.Content.String(),
|
||||
Severity: model.Severity.String(),
|
||||
Tags: tags,
|
||||
CreatedAt: model.CreatedAt.Time(),
|
||||
UpdatedAt: model.UpdatedAt.Time(),
|
||||
}
|
||||
g.Description = strVal(desc)
|
||||
if g.Tags == nil {
|
||||
g.Tags = []string{}
|
||||
}
|
||||
@@ -278,7 +183,7 @@ func (db *DB) GetGuardrail(ctx context.Context, id int64) (ext.AgentGuardrail, e
|
||||
|
||||
// Project Skills
|
||||
|
||||
func (db *DB) AddProjectSkill(ctx context.Context, projectID, skillID int64) error {
|
||||
func (db *DB) AddProjectSkill(ctx context.Context, projectID, skillID uuid.UUID) error {
|
||||
_, err := db.pool.Exec(ctx, `
|
||||
insert into project_skills (project_id, skill_id)
|
||||
values ($1, $2)
|
||||
@@ -290,7 +195,7 @@ func (db *DB) AddProjectSkill(ctx context.Context, projectID, skillID int64) err
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) RemoveProjectSkill(ctx context.Context, projectID, skillID int64) error {
|
||||
func (db *DB) RemoveProjectSkill(ctx context.Context, projectID, skillID uuid.UUID) error {
|
||||
tag, err := db.pool.Exec(ctx, `
|
||||
delete from project_skills where project_id = $1 and skill_id = $2
|
||||
`, projectID, skillID)
|
||||
@@ -303,9 +208,9 @@ func (db *DB) RemoveProjectSkill(ctx context.Context, projectID, skillID int64)
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) ListProjectSkills(ctx context.Context, projectID int64) ([]ext.AgentSkill, error) {
|
||||
func (db *DB) ListProjectSkills(ctx context.Context, projectID uuid.UUID) ([]ext.AgentSkill, error) {
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select s.`+skillSelectCols+`
|
||||
select s.id, s.name, s.description, s.content, s.tags, s.created_at, s.updated_at
|
||||
from agent_skills s
|
||||
join project_skills ps on ps.skill_id = s.id
|
||||
where ps.project_id = $1
|
||||
@@ -318,10 +223,15 @@ func (db *DB) ListProjectSkills(ctx context.Context, projectID int64) ([]ext.Age
|
||||
|
||||
var skills []ext.AgentSkill
|
||||
for rows.Next() {
|
||||
s, err := scanSkill(rows)
|
||||
if err != nil {
|
||||
var s ext.AgentSkill
|
||||
var desc *string
|
||||
if err := rows.Scan(&s.ID, &s.Name, &desc, &s.Content, &s.Tags, &s.CreatedAt, &s.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan project skill: %w", err)
|
||||
}
|
||||
s.Description = strVal(desc)
|
||||
if s.Tags == nil {
|
||||
s.Tags = []string{}
|
||||
}
|
||||
skills = append(skills, s)
|
||||
}
|
||||
return skills, rows.Err()
|
||||
@@ -329,7 +239,7 @@ func (db *DB) ListProjectSkills(ctx context.Context, projectID int64) ([]ext.Age
|
||||
|
||||
// Project Guardrails
|
||||
|
||||
func (db *DB) AddProjectGuardrail(ctx context.Context, projectID, guardrailID int64) error {
|
||||
func (db *DB) AddProjectGuardrail(ctx context.Context, projectID, guardrailID uuid.UUID) error {
|
||||
_, err := db.pool.Exec(ctx, `
|
||||
insert into project_guardrails (project_id, guardrail_id)
|
||||
values ($1, $2)
|
||||
@@ -341,7 +251,7 @@ func (db *DB) AddProjectGuardrail(ctx context.Context, projectID, guardrailID in
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) RemoveProjectGuardrail(ctx context.Context, projectID, guardrailID int64) error {
|
||||
func (db *DB) RemoveProjectGuardrail(ctx context.Context, projectID, guardrailID uuid.UUID) error {
|
||||
tag, err := db.pool.Exec(ctx, `
|
||||
delete from project_guardrails where project_id = $1 and guardrail_id = $2
|
||||
`, projectID, guardrailID)
|
||||
@@ -354,9 +264,9 @@ func (db *DB) RemoveProjectGuardrail(ctx context.Context, projectID, guardrailID
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) ListProjectGuardrails(ctx context.Context, projectID int64) ([]ext.AgentGuardrail, error) {
|
||||
func (db *DB) ListProjectGuardrails(ctx context.Context, projectID uuid.UUID) ([]ext.AgentGuardrail, error) {
|
||||
rows, err := db.pool.Query(ctx, `
|
||||
select g.id, g.name, g.description, g.content, g.severity, g.tags::text[], g.created_at, g.updated_at
|
||||
select g.id, g.name, g.description, g.content, g.severity, g.tags, g.created_at, g.updated_at
|
||||
from agent_guardrails g
|
||||
join project_guardrails pg on pg.guardrail_id = g.id
|
||||
where pg.project_id = $1
|
||||
@@ -369,21 +279,12 @@ func (db *DB) ListProjectGuardrails(ctx context.Context, projectID int64) ([]ext
|
||||
|
||||
var guardrails []ext.AgentGuardrail
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicAgentGuardrails
|
||||
var tags []string
|
||||
if err := rows.Scan(&model.ID, &model.Name, &model.Description, &model.Content, &model.Severity, &tags, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
var g ext.AgentGuardrail
|
||||
var desc *string
|
||||
if err := rows.Scan(&g.ID, &g.Name, &desc, &g.Content, &g.Severity, &g.Tags, &g.CreatedAt, &g.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan project guardrail: %w", err)
|
||||
}
|
||||
g := ext.AgentGuardrail{
|
||||
ID: model.ID.Int64(),
|
||||
Name: model.Name.String(),
|
||||
Description: model.Description.String(),
|
||||
Content: model.Content.String(),
|
||||
Severity: model.Severity.String(),
|
||||
Tags: tags,
|
||||
CreatedAt: model.CreatedAt.Time(),
|
||||
UpdatedAt: model.UpdatedAt.Time(),
|
||||
}
|
||||
g.Description = strVal(desc)
|
||||
if g.Tags == nil {
|
||||
g.Tags = []string{}
|
||||
}
|
||||
|
||||
+48
-72
@@ -12,7 +12,6 @@ import (
|
||||
"github.com/jackc/pgx/v5"
|
||||
"github.com/pgvector/pgvector-go"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
thoughttypes "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
@@ -33,12 +32,12 @@ func (db *DB) InsertThought(ctx context.Context, thought thoughttypes.Thought, e
|
||||
row := tx.QueryRow(ctx, `
|
||||
insert into thoughts (content, metadata, project_id)
|
||||
values ($1, $2::jsonb, $3)
|
||||
returning id, guid, created_at, updated_at
|
||||
returning guid, created_at, updated_at
|
||||
`, thought.Content, metadata, thought.ProjectID)
|
||||
|
||||
created := thought
|
||||
created.Embedding = nil
|
||||
if err := row.Scan(&created.ID, &created.GUID, &created.CreatedAt, &created.UpdatedAt); err != nil {
|
||||
if err := row.Scan(&created.ID, &created.CreatedAt, &created.UpdatedAt); err != nil {
|
||||
return thoughttypes.Thought{}, fmt.Errorf("insert thought: %w", err)
|
||||
}
|
||||
|
||||
@@ -132,7 +131,7 @@ func (db *DB) ListThoughts(ctx context.Context, filter thoughttypes.ListFilter)
|
||||
}
|
||||
|
||||
query := `
|
||||
select id, guid, content, metadata, project_id, archived_at, created_at, updated_at
|
||||
select guid, content, metadata, project_id, archived_at, created_at, updated_at
|
||||
from thoughts
|
||||
`
|
||||
if len(conditions) > 0 {
|
||||
@@ -150,13 +149,13 @@ func (db *DB) ListThoughts(ctx context.Context, filter thoughttypes.ListFilter)
|
||||
|
||||
thoughts := make([]thoughttypes.Thought, 0, filter.Limit)
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicThoughts
|
||||
if err := rows.Scan(&model.ID, &model.GUID, &model.Content, &model.Metadata, &model.ProjectID, &model.ArchivedAt, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
var thought thoughttypes.Thought
|
||||
var metadataBytes []byte
|
||||
if err := rows.Scan(&thought.ID, &thought.Content, &metadataBytes, &thought.ProjectID, &thought.ArchivedAt, &thought.CreatedAt, &thought.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan listed thought: %w", err)
|
||||
}
|
||||
thought, err := thoughtFromModel(model)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("map listed thought: %w", err)
|
||||
if err := json.Unmarshal(metadataBytes, &thought.Metadata); err != nil {
|
||||
return nil, fmt.Errorf("decode listed metadata: %w", err)
|
||||
}
|
||||
thoughts = append(thoughts, thought)
|
||||
}
|
||||
@@ -218,51 +217,28 @@ func (db *DB) Stats(ctx context.Context) (thoughttypes.ThoughtStats, error) {
|
||||
|
||||
func (db *DB) GetThought(ctx context.Context, id uuid.UUID) (thoughttypes.Thought, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, guid, content, metadata, project_id, archived_at, created_at, updated_at
|
||||
select guid, content, metadata, project_id, archived_at, created_at, updated_at
|
||||
from thoughts
|
||||
where guid = $1
|
||||
`, id)
|
||||
|
||||
var model generatedmodels.ModelPublicThoughts
|
||||
if err := row.Scan(&model.ID, &model.GUID, &model.Content, &model.Metadata, &model.ProjectID, &model.ArchivedAt, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
var thought thoughttypes.Thought
|
||||
var metadataBytes []byte
|
||||
if err := row.Scan(&thought.ID, &thought.Content, &metadataBytes, &thought.ProjectID, &thought.ArchivedAt, &thought.CreatedAt, &thought.UpdatedAt); err != nil {
|
||||
if err == pgx.ErrNoRows {
|
||||
return thoughttypes.Thought{}, err
|
||||
}
|
||||
return thoughttypes.Thought{}, fmt.Errorf("get thought: %w", err)
|
||||
}
|
||||
|
||||
thought, err := thoughtFromModel(model)
|
||||
if err != nil {
|
||||
return thoughttypes.Thought{}, fmt.Errorf("map thought: %w", err)
|
||||
if err := json.Unmarshal(metadataBytes, &thought.Metadata); err != nil {
|
||||
return thoughttypes.Thought{}, fmt.Errorf("decode thought metadata: %w", err)
|
||||
}
|
||||
|
||||
return thought, nil
|
||||
}
|
||||
|
||||
func (db *DB) GetThoughtByID(ctx context.Context, id int64) (thoughttypes.Thought, error) {
|
||||
row := db.pool.QueryRow(ctx, `
|
||||
select id, guid, content, metadata, project_id, archived_at, created_at, updated_at
|
||||
from thoughts
|
||||
where id = $1
|
||||
`, id)
|
||||
|
||||
var model generatedmodels.ModelPublicThoughts
|
||||
if err := row.Scan(&model.ID, &model.GUID, &model.Content, &model.Metadata, &model.ProjectID, &model.ArchivedAt, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
if err == pgx.ErrNoRows {
|
||||
return thoughttypes.Thought{}, err
|
||||
}
|
||||
return thoughttypes.Thought{}, fmt.Errorf("get thought by id: %w", err)
|
||||
}
|
||||
|
||||
thought, err := thoughtFromModel(model)
|
||||
if err != nil {
|
||||
return thoughttypes.Thought{}, fmt.Errorf("map thought: %w", err)
|
||||
}
|
||||
|
||||
return thought, nil
|
||||
}
|
||||
|
||||
func (db *DB) UpdateThought(ctx context.Context, id uuid.UUID, content string, embedding []float32, embeddingModel string, metadata thoughttypes.ThoughtMetadata, projectID *int64) (thoughttypes.Thought, error) {
|
||||
func (db *DB) UpdateThought(ctx context.Context, id uuid.UUID, content string, embedding []float32, embeddingModel string, metadata thoughttypes.ThoughtMetadata, projectID *uuid.UUID) (thoughttypes.Thought, error) {
|
||||
metadataBytes, err := json.Marshal(metadata)
|
||||
if err != nil {
|
||||
return thoughttypes.Thought{}, fmt.Errorf("marshal updated metadata: %w", err)
|
||||
@@ -294,7 +270,7 @@ func (db *DB) UpdateThought(ctx context.Context, id uuid.UUID, content string, e
|
||||
if len(embedding) > 0 && embeddingModel != "" {
|
||||
if _, err := tx.Exec(ctx, `
|
||||
insert into embeddings (thought_id, model, dim, embedding)
|
||||
select id, $2, $3, $4 from thoughts where guid = $1
|
||||
values ($1, $2, $3, $4)
|
||||
on conflict (thought_id, model) do update
|
||||
set embedding = excluded.embedding,
|
||||
dim = excluded.dim,
|
||||
@@ -311,7 +287,7 @@ func (db *DB) UpdateThought(ctx context.Context, id uuid.UUID, content string, e
|
||||
return db.GetThought(ctx, id)
|
||||
}
|
||||
|
||||
func (db *DB) UpdateThoughtMetadata(ctx context.Context, id int64, metadata thoughttypes.ThoughtMetadata) (thoughttypes.Thought, error) {
|
||||
func (db *DB) UpdateThoughtMetadata(ctx context.Context, id uuid.UUID, metadata thoughttypes.ThoughtMetadata) (thoughttypes.Thought, error) {
|
||||
metadataBytes, err := json.Marshal(metadata)
|
||||
if err != nil {
|
||||
return thoughttypes.Thought{}, fmt.Errorf("marshal updated metadata: %w", err)
|
||||
@@ -321,7 +297,7 @@ func (db *DB) UpdateThoughtMetadata(ctx context.Context, id int64, metadata thou
|
||||
update thoughts
|
||||
set metadata = $2::jsonb,
|
||||
updated_at = now()
|
||||
where id = $1
|
||||
where guid = $1
|
||||
`, id, metadataBytes)
|
||||
if err != nil {
|
||||
return thoughttypes.Thought{}, fmt.Errorf("update thought metadata: %w", err)
|
||||
@@ -330,7 +306,7 @@ func (db *DB) UpdateThoughtMetadata(ctx context.Context, id int64, metadata thou
|
||||
return thoughttypes.Thought{}, pgx.ErrNoRows
|
||||
}
|
||||
|
||||
return db.GetThoughtByID(ctx, id)
|
||||
return db.GetThought(ctx, id)
|
||||
}
|
||||
|
||||
func (db *DB) DeleteThought(ctx context.Context, id uuid.UUID) error {
|
||||
@@ -355,7 +331,7 @@ func (db *DB) ArchiveThought(ctx context.Context, id uuid.UUID) error {
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) RecentThoughts(ctx context.Context, projectID *int64, limit int, days int) ([]thoughttypes.Thought, error) {
|
||||
func (db *DB) RecentThoughts(ctx context.Context, projectID *uuid.UUID, limit int, days int) ([]thoughttypes.Thought, error) {
|
||||
filter := thoughttypes.ListFilter{
|
||||
Limit: limit,
|
||||
ProjectID: projectID,
|
||||
@@ -365,7 +341,7 @@ func (db *DB) RecentThoughts(ctx context.Context, projectID *int64, limit int, d
|
||||
return db.ListThoughts(ctx, filter)
|
||||
}
|
||||
|
||||
func (db *DB) ListThoughtsPendingMetadataRetry(ctx context.Context, limit int, projectID *int64, includeArchived bool, olderThanDays int) ([]thoughttypes.Thought, error) {
|
||||
func (db *DB) ListThoughtsPendingMetadataRetry(ctx context.Context, limit int, projectID *uuid.UUID, includeArchived bool, olderThanDays int) ([]thoughttypes.Thought, error) {
|
||||
args := make([]any, 0, 4)
|
||||
conditions := []string{
|
||||
"(metadata->>'metadata_status' = 'pending' or metadata->>'metadata_status' = 'failed')",
|
||||
@@ -384,7 +360,7 @@ func (db *DB) ListThoughtsPendingMetadataRetry(ctx context.Context, limit int, p
|
||||
}
|
||||
|
||||
query := `
|
||||
select id, guid, content, metadata, project_id, archived_at, created_at, updated_at
|
||||
select guid, content, metadata, project_id, archived_at, created_at, updated_at
|
||||
from thoughts
|
||||
where ` + strings.Join(conditions, " and ")
|
||||
|
||||
@@ -399,12 +375,12 @@ func (db *DB) ListThoughtsPendingMetadataRetry(ctx context.Context, limit int, p
|
||||
|
||||
thoughts := make([]thoughttypes.Thought, 0, limit)
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicThoughts
|
||||
if err := rows.Scan(&model.ID, &model.GUID, &model.Content, &model.Metadata, &model.ProjectID, &model.ArchivedAt, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
var thought thoughttypes.Thought
|
||||
var metadataBytes []byte
|
||||
if err := rows.Scan(&thought.ID, &thought.Content, &metadataBytes, &thought.ProjectID, &thought.ArchivedAt, &thought.CreatedAt, &thought.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan pending metadata retry thought: %w", err)
|
||||
}
|
||||
thought, err := thoughtFromModel(model)
|
||||
if err != nil {
|
||||
if err := json.Unmarshal(metadataBytes, &thought.Metadata); err != nil {
|
||||
return nil, fmt.Errorf("decode pending metadata retry thought: %w", err)
|
||||
}
|
||||
thoughts = append(thoughts, thought)
|
||||
@@ -417,7 +393,7 @@ func (db *DB) ListThoughtsPendingMetadataRetry(ctx context.Context, limit int, p
|
||||
return thoughts, nil
|
||||
}
|
||||
|
||||
func (db *DB) SearchSimilarThoughts(ctx context.Context, embedding []float32, embeddingModel string, threshold float64, limit int, projectID *int64, excludeID *uuid.UUID) ([]thoughttypes.SearchResult, error) {
|
||||
func (db *DB) SearchSimilarThoughts(ctx context.Context, embedding []float32, embeddingModel string, threshold float64, limit int, projectID *uuid.UUID, excludeID *uuid.UUID) ([]thoughttypes.SearchResult, error) {
|
||||
args := []any{pgvector.NewVector(embedding), threshold, embeddingModel}
|
||||
conditions := []string{
|
||||
"t.archived_at is null",
|
||||
@@ -435,9 +411,9 @@ func (db *DB) SearchSimilarThoughts(ctx context.Context, embedding []float32, em
|
||||
args = append(args, limit)
|
||||
|
||||
query := `
|
||||
select t.id, t.content, t.metadata, 1 - (e.embedding <=> $1) as similarity, t.created_at
|
||||
select t.guid, t.content, t.metadata, 1 - (e.embedding <=> $1) as similarity, t.created_at
|
||||
from thoughts t
|
||||
join embeddings e on e.thought_id = t.id
|
||||
join embeddings e on e.thought_id = t.guid
|
||||
where ` + strings.Join(conditions, " and ") + fmt.Sprintf(`
|
||||
order by e.embedding <=> $1
|
||||
limit $%d`, len(args))
|
||||
@@ -466,7 +442,7 @@ func (db *DB) SearchSimilarThoughts(ctx context.Context, embedding []float32, em
|
||||
return results, nil
|
||||
}
|
||||
|
||||
func (db *DB) HasEmbeddingsForModel(ctx context.Context, model string, projectID *int64) (bool, error) {
|
||||
func (db *DB) HasEmbeddingsForModel(ctx context.Context, model string, projectID *uuid.UUID) (bool, error) {
|
||||
args := []any{model}
|
||||
conditions := []string{
|
||||
"e.model = $1",
|
||||
@@ -477,7 +453,7 @@ func (db *DB) HasEmbeddingsForModel(ctx context.Context, model string, projectID
|
||||
conditions = append(conditions, fmt.Sprintf("t.project_id = $%d", len(args)))
|
||||
}
|
||||
|
||||
query := `select exists(select 1 from embeddings e join thoughts t on t.id = e.thought_id where ` +
|
||||
query := `select exists(select 1 from embeddings e join thoughts t on t.guid = e.thought_id where ` +
|
||||
strings.Join(conditions, " and ") + `)`
|
||||
|
||||
var exists bool
|
||||
@@ -487,7 +463,7 @@ func (db *DB) HasEmbeddingsForModel(ctx context.Context, model string, projectID
|
||||
return exists, nil
|
||||
}
|
||||
|
||||
func (db *DB) ListThoughtsMissingEmbedding(ctx context.Context, model string, limit int, projectID *int64, includeArchived bool, olderThanDays int) ([]thoughttypes.Thought, error) {
|
||||
func (db *DB) ListThoughtsMissingEmbedding(ctx context.Context, model string, limit int, projectID *uuid.UUID, includeArchived bool, olderThanDays int) ([]thoughttypes.Thought, error) {
|
||||
args := []any{model}
|
||||
conditions := []string{"e.id is null"}
|
||||
|
||||
@@ -505,9 +481,9 @@ func (db *DB) ListThoughtsMissingEmbedding(ctx context.Context, model string, li
|
||||
args = append(args, limit)
|
||||
|
||||
query := `
|
||||
select t.id, t.guid, t.content, t.metadata, t.project_id, t.archived_at, t.created_at, t.updated_at
|
||||
select t.guid, t.content, t.metadata, t.project_id, t.archived_at, t.created_at, t.updated_at
|
||||
from thoughts t
|
||||
left join embeddings e on e.thought_id = t.id and e.model = $1
|
||||
left join embeddings e on e.thought_id = t.guid and e.model = $1
|
||||
where ` + strings.Join(conditions, " and ") + `
|
||||
order by t.created_at asc
|
||||
limit $` + fmt.Sprintf("%d", len(args))
|
||||
@@ -520,12 +496,12 @@ func (db *DB) ListThoughtsMissingEmbedding(ctx context.Context, model string, li
|
||||
|
||||
thoughts := make([]thoughttypes.Thought, 0, limit)
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicThoughts
|
||||
if err := rows.Scan(&model.ID, &model.GUID, &model.Content, &model.Metadata, &model.ProjectID, &model.ArchivedAt, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
var thought thoughttypes.Thought
|
||||
var metadataBytes []byte
|
||||
if err := rows.Scan(&thought.ID, &thought.Content, &metadataBytes, &thought.ProjectID, &thought.ArchivedAt, &thought.CreatedAt, &thought.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan missing-embedding thought: %w", err)
|
||||
}
|
||||
thought, err := thoughtFromModel(model)
|
||||
if err != nil {
|
||||
if err := json.Unmarshal(metadataBytes, &thought.Metadata); err != nil {
|
||||
return nil, fmt.Errorf("decode missing-embedding metadata: %w", err)
|
||||
}
|
||||
thoughts = append(thoughts, thought)
|
||||
@@ -536,7 +512,7 @@ func (db *DB) ListThoughtsMissingEmbedding(ctx context.Context, model string, li
|
||||
return thoughts, nil
|
||||
}
|
||||
|
||||
func (db *DB) ListThoughtsForMetadataReparse(ctx context.Context, limit int, projectID *int64, includeArchived bool, olderThanDays int) ([]thoughttypes.Thought, error) {
|
||||
func (db *DB) ListThoughtsForMetadataReparse(ctx context.Context, limit int, projectID *uuid.UUID, includeArchived bool, olderThanDays int) ([]thoughttypes.Thought, error) {
|
||||
args := make([]any, 0, 3)
|
||||
conditions := make([]string, 0, 4)
|
||||
|
||||
@@ -554,7 +530,7 @@ func (db *DB) ListThoughtsForMetadataReparse(ctx context.Context, limit int, pro
|
||||
args = append(args, limit)
|
||||
|
||||
query := `
|
||||
select id, guid, content, metadata, project_id, archived_at, created_at, updated_at
|
||||
select guid, content, metadata, project_id, archived_at, created_at, updated_at
|
||||
from thoughts
|
||||
`
|
||||
if len(conditions) > 0 {
|
||||
@@ -570,12 +546,12 @@ func (db *DB) ListThoughtsForMetadataReparse(ctx context.Context, limit int, pro
|
||||
|
||||
thoughts := make([]thoughttypes.Thought, 0, limit)
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicThoughts
|
||||
if err := rows.Scan(&model.ID, &model.GUID, &model.Content, &model.Metadata, &model.ProjectID, &model.ArchivedAt, &model.CreatedAt, &model.UpdatedAt); err != nil {
|
||||
var thought thoughttypes.Thought
|
||||
var metadataBytes []byte
|
||||
if err := rows.Scan(&thought.ID, &thought.Content, &metadataBytes, &thought.ProjectID, &thought.ArchivedAt, &thought.CreatedAt, &thought.UpdatedAt); err != nil {
|
||||
return nil, fmt.Errorf("scan metadata-reparse thought: %w", err)
|
||||
}
|
||||
thought, err := thoughtFromModel(model)
|
||||
if err != nil {
|
||||
if err := json.Unmarshal(metadataBytes, &thought.Metadata); err != nil {
|
||||
return nil, fmt.Errorf("decode metadata-reparse thought metadata: %w", err)
|
||||
}
|
||||
thoughts = append(thoughts, thought)
|
||||
@@ -587,7 +563,7 @@ func (db *DB) ListThoughtsForMetadataReparse(ctx context.Context, limit int, pro
|
||||
return thoughts, nil
|
||||
}
|
||||
|
||||
func (db *DB) UpsertEmbedding(ctx context.Context, thoughtID int64, model string, embedding []float32) error {
|
||||
func (db *DB) UpsertEmbedding(ctx context.Context, thoughtID uuid.UUID, model string, embedding []float32) error {
|
||||
_, err := db.pool.Exec(ctx, `
|
||||
insert into embeddings (thought_id, model, dim, embedding)
|
||||
values ($1, $2, $3, $4)
|
||||
@@ -602,7 +578,7 @@ func (db *DB) UpsertEmbedding(ctx context.Context, thoughtID int64, model string
|
||||
return nil
|
||||
}
|
||||
|
||||
func (db *DB) SearchThoughtsText(ctx context.Context, query string, limit int, projectID *int64, excludeID *uuid.UUID) ([]thoughttypes.SearchResult, error) {
|
||||
func (db *DB) SearchThoughtsText(ctx context.Context, query string, limit int, projectID *uuid.UUID, excludeID *uuid.UUID) ([]thoughttypes.SearchResult, error) {
|
||||
args := []any{query}
|
||||
conditions := []string{
|
||||
"t.archived_at is null",
|
||||
@@ -619,11 +595,11 @@ func (db *DB) SearchThoughtsText(ctx context.Context, query string, limit int, p
|
||||
args = append(args, limit)
|
||||
|
||||
q := `
|
||||
select t.id, t.content, t.metadata,
|
||||
select t.guid, t.content, t.metadata,
|
||||
ts_rank_cd(to_tsvector('simple', t.content) || to_tsvector('simple', coalesce(p.name, '')), websearch_to_tsquery('simple', $1)) as similarity,
|
||||
t.created_at
|
||||
from thoughts t
|
||||
left join projects p on t.project_id = p.id
|
||||
left join projects p on t.project_id = p.guid
|
||||
where ` + strings.Join(conditions, " and ") + `
|
||||
order by similarity desc
|
||||
limit $` + fmt.Sprintf("%d", len(args))
|
||||
|
||||
@@ -3,8 +3,6 @@ package store
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/generatedmodels"
|
||||
)
|
||||
|
||||
func (db *DB) UpsertToolAnnotation(ctx context.Context, toolName, notes string) error {
|
||||
@@ -30,11 +28,11 @@ func (db *DB) GetToolAnnotations(ctx context.Context) (map[string]string, error)
|
||||
|
||||
annotations := make(map[string]string)
|
||||
for rows.Next() {
|
||||
var model generatedmodels.ModelPublicToolAnnotations
|
||||
if err := rows.Scan(&model.ToolName, &model.Notes); err != nil {
|
||||
var toolName, notes string
|
||||
if err := rows.Scan(&toolName, ¬es); err != nil {
|
||||
return nil, fmt.Errorf("scan tool annotation: %w", err)
|
||||
}
|
||||
annotations[model.ToolName.String()] = model.Notes.String()
|
||||
annotations[toolName] = notes
|
||||
}
|
||||
return annotations, rows.Err()
|
||||
}
|
||||
|
||||
@@ -1,873 +0,0 @@
|
||||
package tools
|
||||
|
||||
import (
|
||||
"context"
|
||||
"strings"
|
||||
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/store"
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
type AgentPersonasTool struct {
|
||||
store *store.DB
|
||||
}
|
||||
|
||||
func NewAgentPersonasTool(db *store.DB) *AgentPersonasTool {
|
||||
return &AgentPersonasTool{store: db}
|
||||
}
|
||||
|
||||
// ──────────────────────────────────────────────
|
||||
// Personas
|
||||
// ──────────────────────────────────────────────
|
||||
|
||||
// create_agent_persona
|
||||
|
||||
type CreatePersonaInput struct {
|
||||
Name string `json:"name" jsonschema:"unique persona name — used as the load key"`
|
||||
Description string `json:"description,omitempty" jsonschema:"short description of the persona"`
|
||||
Summary string `json:"summary" jsonschema:"concise behaviour summary, returned by default on load"`
|
||||
Detail string `json:"detail,omitempty" jsonschema:"full behaviour detail, returned only when detail=true"`
|
||||
Tags []string `json:"tags,omitempty" jsonschema:"optional tags for grouping or filtering"`
|
||||
}
|
||||
|
||||
type CreatePersonaOutput struct {
|
||||
Persona ext.Persona `json:"persona"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) CreatePersona(ctx context.Context, _ *mcp.CallToolRequest, in CreatePersonaInput) (*mcp.CallToolResult, CreatePersonaOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, CreatePersonaOutput{}, errRequiredField("name")
|
||||
}
|
||||
if strings.TrimSpace(in.Summary) == "" {
|
||||
return nil, CreatePersonaOutput{}, errRequiredField("summary")
|
||||
}
|
||||
if in.Tags == nil {
|
||||
in.Tags = []string{}
|
||||
}
|
||||
persona, err := t.store.CreatePersona(ctx, ext.Persona{
|
||||
Name: strings.TrimSpace(in.Name),
|
||||
Description: strings.TrimSpace(in.Description),
|
||||
Summary: strings.TrimSpace(in.Summary),
|
||||
Detail: strings.TrimSpace(in.Detail),
|
||||
Tags: in.Tags,
|
||||
})
|
||||
if err != nil {
|
||||
return nil, CreatePersonaOutput{}, err
|
||||
}
|
||||
return nil, CreatePersonaOutput{Persona: persona}, nil
|
||||
}
|
||||
|
||||
// update_agent_persona
|
||||
|
||||
type UpdatePersonaInput struct {
|
||||
Name string `json:"name" jsonschema:"name of the persona to update"`
|
||||
NewName string `json:"new_name,omitempty" jsonschema:"rename the persona"`
|
||||
Description *string `json:"description,omitempty" jsonschema:"update description"`
|
||||
Summary *string `json:"summary,omitempty" jsonschema:"update summary"`
|
||||
Detail *string `json:"detail,omitempty" jsonschema:"update detail"`
|
||||
Tags []string `json:"tags,omitempty" jsonschema:"replace tags"`
|
||||
}
|
||||
|
||||
type UpdatePersonaOutput struct {
|
||||
Persona ext.Persona `json:"persona"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) UpdatePersona(ctx context.Context, _ *mcp.CallToolRequest, in UpdatePersonaInput) (*mcp.CallToolResult, UpdatePersonaOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, UpdatePersonaOutput{}, errRequiredField("name")
|
||||
}
|
||||
updates := map[string]any{}
|
||||
if n := strings.TrimSpace(in.NewName); n != "" {
|
||||
updates["name"] = n
|
||||
}
|
||||
if in.Description != nil {
|
||||
updates["description"] = strings.TrimSpace(*in.Description)
|
||||
}
|
||||
if in.Summary != nil {
|
||||
updates["summary"] = strings.TrimSpace(*in.Summary)
|
||||
}
|
||||
if in.Detail != nil {
|
||||
updates["detail"] = strings.TrimSpace(*in.Detail)
|
||||
}
|
||||
if in.Tags != nil {
|
||||
updates["tags"] = in.Tags
|
||||
}
|
||||
persona, err := t.store.UpdatePersona(ctx, strings.TrimSpace(in.Name), updates)
|
||||
if err != nil {
|
||||
return nil, UpdatePersonaOutput{}, err
|
||||
}
|
||||
return nil, UpdatePersonaOutput{Persona: persona}, nil
|
||||
}
|
||||
|
||||
// delete_agent_persona
|
||||
|
||||
type DeletePersonaInput struct {
|
||||
Name string `json:"name" jsonschema:"name of the persona to delete"`
|
||||
}
|
||||
|
||||
type DeletePersonaOutput struct {
|
||||
Deleted bool `json:"deleted"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) DeletePersona(ctx context.Context, _ *mcp.CallToolRequest, in DeletePersonaInput) (*mcp.CallToolResult, DeletePersonaOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, DeletePersonaOutput{}, errRequiredField("name")
|
||||
}
|
||||
if err := t.store.DeletePersona(ctx, strings.TrimSpace(in.Name)); err != nil {
|
||||
return nil, DeletePersonaOutput{}, err
|
||||
}
|
||||
return nil, DeletePersonaOutput{Deleted: true}, nil
|
||||
}
|
||||
|
||||
// list_agent_personas
|
||||
|
||||
type ListPersonasInput struct {
|
||||
Tag string `json:"tag,omitempty" jsonschema:"filter by tag"`
|
||||
}
|
||||
|
||||
type ListPersonasOutput struct {
|
||||
Personas []ext.Persona `json:"personas"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) ListPersonas(ctx context.Context, _ *mcp.CallToolRequest, in ListPersonasInput) (*mcp.CallToolResult, ListPersonasOutput, error) {
|
||||
personas, err := t.store.ListPersonas(ctx, in.Tag)
|
||||
if err != nil {
|
||||
return nil, ListPersonasOutput{}, err
|
||||
}
|
||||
if personas == nil {
|
||||
personas = []ext.Persona{}
|
||||
}
|
||||
return nil, ListPersonasOutput{Personas: personas}, nil
|
||||
}
|
||||
|
||||
// get_agent_persona
|
||||
|
||||
type GetPersonaInput struct {
|
||||
Name string `json:"name" jsonschema:"persona name to load"`
|
||||
Detail bool `json:"detail,omitempty" jsonschema:"when true, return full part content and persona detail instead of summaries"`
|
||||
Overrides map[string]string `json:"overrides,omitempty" jsonschema:"runtime part substitutions: {part_type: part_name}. Replaces all parts of that type without modifying the persona."`
|
||||
}
|
||||
|
||||
type GetPersonaOutput struct {
|
||||
Persona ext.PersonaFull `json:"persona"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) GetPersona(ctx context.Context, _ *mcp.CallToolRequest, in GetPersonaInput) (*mcp.CallToolResult, GetPersonaOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, GetPersonaOutput{}, errRequiredField("name")
|
||||
}
|
||||
persona, err := t.store.GetPersona(ctx, strings.TrimSpace(in.Name), in.Detail, in.Overrides)
|
||||
if err != nil {
|
||||
return nil, GetPersonaOutput{}, err
|
||||
}
|
||||
return nil, GetPersonaOutput{Persona: persona}, nil
|
||||
}
|
||||
|
||||
// get_persona_manifest
|
||||
|
||||
type GetPersonaManifestInput struct {
|
||||
Name string `json:"name" jsonschema:"persona name"`
|
||||
}
|
||||
|
||||
type GetPersonaManifestOutput struct {
|
||||
Manifest ext.PersonaManifest `json:"manifest"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) GetPersonaManifest(ctx context.Context, _ *mcp.CallToolRequest, in GetPersonaManifestInput) (*mcp.CallToolResult, GetPersonaManifestOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, GetPersonaManifestOutput{}, errRequiredField("name")
|
||||
}
|
||||
manifest, err := t.store.GetPersonaManifest(ctx, strings.TrimSpace(in.Name))
|
||||
if err != nil {
|
||||
return nil, GetPersonaManifestOutput{}, err
|
||||
}
|
||||
return nil, GetPersonaManifestOutput{Manifest: manifest}, nil
|
||||
}
|
||||
|
||||
// compile_persona
|
||||
|
||||
type CompilePersonaInput struct {
|
||||
Name string `json:"name" jsonschema:"persona name to compile"`
|
||||
}
|
||||
|
||||
type CompilePersonaOutput struct {
|
||||
Persona ext.Persona `json:"persona"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) CompilePersona(ctx context.Context, _ *mcp.CallToolRequest, in CompilePersonaInput) (*mcp.CallToolResult, CompilePersonaOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, CompilePersonaOutput{}, errRequiredField("name")
|
||||
}
|
||||
persona, err := t.store.CompilePersona(ctx, strings.TrimSpace(in.Name))
|
||||
if err != nil {
|
||||
return nil, CompilePersonaOutput{}, err
|
||||
}
|
||||
return nil, CompilePersonaOutput{Persona: persona}, nil
|
||||
}
|
||||
|
||||
// ──────────────────────────────────────────────
|
||||
// Parts
|
||||
// ──────────────────────────────────────────────
|
||||
|
||||
// create_agent_part
|
||||
|
||||
type CreatePartInput struct {
|
||||
Name string `json:"name" jsonschema:"globally unique part name — used as the override key"`
|
||||
PartType string `json:"part_type" jsonschema:"one of: system, agent, soul, identity, skill, specialization, tone, goal, context, protocol, backstory, motivation, voice, archetype, flaw, relationship"`
|
||||
Description string `json:"description,omitempty" jsonschema:"short description of what this part does"`
|
||||
Summary string `json:"summary" jsonschema:"concise version, used in summary mode"`
|
||||
Content string `json:"content,omitempty" jsonschema:"full content, used in detail mode"`
|
||||
Tags []string `json:"tags,omitempty" jsonschema:"optional tags"`
|
||||
}
|
||||
|
||||
type CreatePartOutput struct {
|
||||
Part ext.Part `json:"part"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) CreatePart(ctx context.Context, _ *mcp.CallToolRequest, in CreatePartInput) (*mcp.CallToolResult, CreatePartOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, CreatePartOutput{}, errRequiredField("name")
|
||||
}
|
||||
if strings.TrimSpace(in.PartType) == "" {
|
||||
return nil, CreatePartOutput{}, errRequiredField("part_type")
|
||||
}
|
||||
if strings.TrimSpace(in.Summary) == "" {
|
||||
return nil, CreatePartOutput{}, errRequiredField("summary")
|
||||
}
|
||||
if in.Tags == nil {
|
||||
in.Tags = []string{}
|
||||
}
|
||||
part, err := t.store.CreatePart(ctx, ext.Part{
|
||||
Name: strings.TrimSpace(in.Name),
|
||||
PartType: strings.TrimSpace(in.PartType),
|
||||
Description: strings.TrimSpace(in.Description),
|
||||
Summary: strings.TrimSpace(in.Summary),
|
||||
Content: strings.TrimSpace(in.Content),
|
||||
Tags: in.Tags,
|
||||
})
|
||||
if err != nil {
|
||||
return nil, CreatePartOutput{}, err
|
||||
}
|
||||
return nil, CreatePartOutput{Part: part}, nil
|
||||
}
|
||||
|
||||
// update_agent_part
|
||||
|
||||
type UpdatePartInput struct {
|
||||
Name string `json:"name" jsonschema:"name of the part to update"`
|
||||
NewName string `json:"new_name,omitempty" jsonschema:"rename the part"`
|
||||
PartType *string `json:"part_type,omitempty" jsonschema:"update part type"`
|
||||
Description *string `json:"description,omitempty" jsonschema:"update description"`
|
||||
Summary *string `json:"summary,omitempty" jsonschema:"update summary"`
|
||||
Content *string `json:"content,omitempty" jsonschema:"update content"`
|
||||
Tags []string `json:"tags,omitempty" jsonschema:"replace tags"`
|
||||
}
|
||||
|
||||
type UpdatePartOutput struct {
|
||||
Part ext.Part `json:"part"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) UpdatePart(ctx context.Context, _ *mcp.CallToolRequest, in UpdatePartInput) (*mcp.CallToolResult, UpdatePartOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, UpdatePartOutput{}, errRequiredField("name")
|
||||
}
|
||||
updates := map[string]any{}
|
||||
if n := strings.TrimSpace(in.NewName); n != "" {
|
||||
updates["name"] = n
|
||||
}
|
||||
if in.PartType != nil {
|
||||
updates["part_type"] = strings.TrimSpace(*in.PartType)
|
||||
}
|
||||
if in.Description != nil {
|
||||
updates["description"] = strings.TrimSpace(*in.Description)
|
||||
}
|
||||
if in.Summary != nil {
|
||||
updates["summary"] = strings.TrimSpace(*in.Summary)
|
||||
}
|
||||
if in.Content != nil {
|
||||
updates["content"] = strings.TrimSpace(*in.Content)
|
||||
}
|
||||
if in.Tags != nil {
|
||||
updates["tags"] = in.Tags
|
||||
}
|
||||
part, err := t.store.UpdatePart(ctx, strings.TrimSpace(in.Name), updates)
|
||||
if err != nil {
|
||||
return nil, UpdatePartOutput{}, err
|
||||
}
|
||||
return nil, UpdatePartOutput{Part: part}, nil
|
||||
}
|
||||
|
||||
// delete_agent_part
|
||||
|
||||
type DeletePartInput struct {
|
||||
Name string `json:"name" jsonschema:"name of the part to delete"`
|
||||
}
|
||||
|
||||
type DeletePartOutput struct {
|
||||
Deleted bool `json:"deleted"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) DeletePart(ctx context.Context, _ *mcp.CallToolRequest, in DeletePartInput) (*mcp.CallToolResult, DeletePartOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, DeletePartOutput{}, errRequiredField("name")
|
||||
}
|
||||
if err := t.store.DeletePart(ctx, strings.TrimSpace(in.Name)); err != nil {
|
||||
return nil, DeletePartOutput{}, err
|
||||
}
|
||||
return nil, DeletePartOutput{Deleted: true}, nil
|
||||
}
|
||||
|
||||
// list_agent_parts
|
||||
|
||||
type ListPartsInput struct {
|
||||
PartType string `json:"part_type,omitempty" jsonschema:"filter by part type"`
|
||||
Tag string `json:"tag,omitempty" jsonschema:"filter by tag"`
|
||||
}
|
||||
|
||||
type ListPartsOutput struct {
|
||||
Parts []ext.Part `json:"parts"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) ListParts(ctx context.Context, _ *mcp.CallToolRequest, in ListPartsInput) (*mcp.CallToolResult, ListPartsOutput, error) {
|
||||
parts, err := t.store.ListParts(ctx, in.PartType, in.Tag)
|
||||
if err != nil {
|
||||
return nil, ListPartsOutput{}, err
|
||||
}
|
||||
if parts == nil {
|
||||
parts = []ext.Part{}
|
||||
}
|
||||
return nil, ListPartsOutput{Parts: parts}, nil
|
||||
}
|
||||
|
||||
// get_agent_part
|
||||
|
||||
type GetPartInput struct {
|
||||
Name string `json:"name" jsonschema:"part name"`
|
||||
}
|
||||
|
||||
type GetPartOutput struct {
|
||||
Part ext.Part `json:"part"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) GetPart(ctx context.Context, _ *mcp.CallToolRequest, in GetPartInput) (*mcp.CallToolResult, GetPartOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, GetPartOutput{}, errRequiredField("name")
|
||||
}
|
||||
part, err := t.store.GetPartByName(ctx, strings.TrimSpace(in.Name))
|
||||
if err != nil {
|
||||
return nil, GetPartOutput{}, err
|
||||
}
|
||||
return nil, GetPartOutput{Part: part}, nil
|
||||
}
|
||||
|
||||
// ──────────────────────────────────────────────
|
||||
// Persona-Part / Skill / Guardrail links
|
||||
// ──────────────────────────────────────────────
|
||||
|
||||
// add_persona_part
|
||||
|
||||
type AddPersonaPartInput struct {
|
||||
Persona string `json:"persona" jsonschema:"persona name"`
|
||||
Part string `json:"part" jsonschema:"part name to link"`
|
||||
Order int `json:"order,omitempty" jsonschema:"assembly order (lower first, default 0)"`
|
||||
Priority int `json:"priority,omitempty" jsonschema:"context-budget priority (higher loads first when trimming)"`
|
||||
}
|
||||
|
||||
type AddPersonaPartOutput struct {
|
||||
Persona string `json:"persona"`
|
||||
Part string `json:"part"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) AddPersonaPart(ctx context.Context, _ *mcp.CallToolRequest, in AddPersonaPartInput) (*mcp.CallToolResult, AddPersonaPartOutput, error) {
|
||||
if strings.TrimSpace(in.Persona) == "" {
|
||||
return nil, AddPersonaPartOutput{}, errRequiredField("persona")
|
||||
}
|
||||
if strings.TrimSpace(in.Part) == "" {
|
||||
return nil, AddPersonaPartOutput{}, errRequiredField("part")
|
||||
}
|
||||
if err := t.store.AddPersonaPart(ctx, strings.TrimSpace(in.Persona), strings.TrimSpace(in.Part), in.Order, in.Priority); err != nil {
|
||||
return nil, AddPersonaPartOutput{}, err
|
||||
}
|
||||
return nil, AddPersonaPartOutput{Persona: in.Persona, Part: in.Part}, nil
|
||||
}
|
||||
|
||||
// remove_persona_part
|
||||
|
||||
type RemovePersonaPartInput struct {
|
||||
Persona string `json:"persona" jsonschema:"persona name"`
|
||||
Part string `json:"part" jsonschema:"part name to unlink"`
|
||||
}
|
||||
|
||||
type RemovePersonaPartOutput struct {
|
||||
Removed bool `json:"removed"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) RemovePersonaPart(ctx context.Context, _ *mcp.CallToolRequest, in RemovePersonaPartInput) (*mcp.CallToolResult, RemovePersonaPartOutput, error) {
|
||||
if strings.TrimSpace(in.Persona) == "" {
|
||||
return nil, RemovePersonaPartOutput{}, errRequiredField("persona")
|
||||
}
|
||||
if strings.TrimSpace(in.Part) == "" {
|
||||
return nil, RemovePersonaPartOutput{}, errRequiredField("part")
|
||||
}
|
||||
if err := t.store.RemovePersonaPart(ctx, strings.TrimSpace(in.Persona), strings.TrimSpace(in.Part)); err != nil {
|
||||
return nil, RemovePersonaPartOutput{}, err
|
||||
}
|
||||
return nil, RemovePersonaPartOutput{Removed: true}, nil
|
||||
}
|
||||
|
||||
// add_persona_skill
|
||||
|
||||
type AddPersonaSkillInput struct {
|
||||
PersonaID int64 `json:"persona_id" jsonschema:"persona id"`
|
||||
SkillID int64 `json:"skill_id" jsonschema:"agent skill id to link"`
|
||||
}
|
||||
|
||||
type AddPersonaSkillOutput struct {
|
||||
PersonaID int64 `json:"persona_id"`
|
||||
SkillID int64 `json:"skill_id"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) AddPersonaSkill(ctx context.Context, _ *mcp.CallToolRequest, in AddPersonaSkillInput) (*mcp.CallToolResult, AddPersonaSkillOutput, error) {
|
||||
if err := t.store.AddPersonaSkill(ctx, in.PersonaID, in.SkillID); err != nil {
|
||||
return nil, AddPersonaSkillOutput{}, err
|
||||
}
|
||||
return nil, AddPersonaSkillOutput{PersonaID: in.PersonaID, SkillID: in.SkillID}, nil
|
||||
}
|
||||
|
||||
// remove_persona_skill
|
||||
|
||||
type RemovePersonaSkillInput struct {
|
||||
PersonaID int64 `json:"persona_id" jsonschema:"persona id"`
|
||||
SkillID int64 `json:"skill_id" jsonschema:"agent skill id to unlink"`
|
||||
}
|
||||
|
||||
type RemovePersonaSkillOutput struct {
|
||||
Removed bool `json:"removed"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) RemovePersonaSkill(ctx context.Context, _ *mcp.CallToolRequest, in RemovePersonaSkillInput) (*mcp.CallToolResult, RemovePersonaSkillOutput, error) {
|
||||
if err := t.store.RemovePersonaSkill(ctx, in.PersonaID, in.SkillID); err != nil {
|
||||
return nil, RemovePersonaSkillOutput{}, err
|
||||
}
|
||||
return nil, RemovePersonaSkillOutput{Removed: true}, nil
|
||||
}
|
||||
|
||||
// add_persona_guardrail
|
||||
|
||||
type AddPersonaGuardrailInput struct {
|
||||
PersonaID int64 `json:"persona_id" jsonschema:"persona id"`
|
||||
GuardrailID int64 `json:"guardrail_id" jsonschema:"agent guardrail id to link"`
|
||||
}
|
||||
|
||||
type AddPersonaGuardrailOutput struct {
|
||||
PersonaID int64 `json:"persona_id"`
|
||||
GuardrailID int64 `json:"guardrail_id"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) AddPersonaGuardrail(ctx context.Context, _ *mcp.CallToolRequest, in AddPersonaGuardrailInput) (*mcp.CallToolResult, AddPersonaGuardrailOutput, error) {
|
||||
if err := t.store.AddPersonaGuardrail(ctx, in.PersonaID, in.GuardrailID); err != nil {
|
||||
return nil, AddPersonaGuardrailOutput{}, err
|
||||
}
|
||||
return nil, AddPersonaGuardrailOutput{PersonaID: in.PersonaID, GuardrailID: in.GuardrailID}, nil
|
||||
}
|
||||
|
||||
// remove_persona_guardrail
|
||||
|
||||
type RemovePersonaGuardrailInput struct {
|
||||
PersonaID int64 `json:"persona_id" jsonschema:"persona id"`
|
||||
GuardrailID int64 `json:"guardrail_id" jsonschema:"agent guardrail id to unlink"`
|
||||
}
|
||||
|
||||
type RemovePersonaGuardrailOutput struct {
|
||||
Removed bool `json:"removed"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) RemovePersonaGuardrail(ctx context.Context, _ *mcp.CallToolRequest, in RemovePersonaGuardrailInput) (*mcp.CallToolResult, RemovePersonaGuardrailOutput, error) {
|
||||
if err := t.store.RemovePersonaGuardrail(ctx, in.PersonaID, in.GuardrailID); err != nil {
|
||||
return nil, RemovePersonaGuardrailOutput{}, err
|
||||
}
|
||||
return nil, RemovePersonaGuardrailOutput{Removed: true}, nil
|
||||
}
|
||||
|
||||
// ──────────────────────────────────────────────
|
||||
// Traits
|
||||
// ──────────────────────────────────────────────
|
||||
|
||||
// create_agent_trait
|
||||
|
||||
type CreateTraitInput struct {
|
||||
Name string `json:"name" jsonschema:"globally unique trait name"`
|
||||
TraitType string `json:"trait_type" jsonschema:"one of: personality, cognitive, emotional, social, behavioral"`
|
||||
Description string `json:"description,omitempty" jsonschema:"short description of this trait"`
|
||||
Instruction string `json:"instruction,omitempty" jsonschema:"how to apply this trait in practice"`
|
||||
Tags []string `json:"tags,omitempty" jsonschema:"optional tags"`
|
||||
}
|
||||
|
||||
type CreateTraitOutput struct {
|
||||
Trait ext.Trait `json:"trait"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) CreateTrait(ctx context.Context, _ *mcp.CallToolRequest, in CreateTraitInput) (*mcp.CallToolResult, CreateTraitOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, CreateTraitOutput{}, errRequiredField("name")
|
||||
}
|
||||
if strings.TrimSpace(in.TraitType) == "" {
|
||||
return nil, CreateTraitOutput{}, errRequiredField("trait_type")
|
||||
}
|
||||
if in.Tags == nil {
|
||||
in.Tags = []string{}
|
||||
}
|
||||
trait, err := t.store.CreateTrait(ctx, ext.Trait{
|
||||
Name: strings.TrimSpace(in.Name),
|
||||
TraitType: strings.TrimSpace(in.TraitType),
|
||||
Description: strings.TrimSpace(in.Description),
|
||||
Instruction: strings.TrimSpace(in.Instruction),
|
||||
Tags: in.Tags,
|
||||
})
|
||||
if err != nil {
|
||||
return nil, CreateTraitOutput{}, err
|
||||
}
|
||||
return nil, CreateTraitOutput{Trait: trait}, nil
|
||||
}
|
||||
|
||||
// update_agent_trait
|
||||
|
||||
type UpdateTraitInput struct {
|
||||
Name string `json:"name" jsonschema:"name of the trait to update"`
|
||||
NewName string `json:"new_name,omitempty" jsonschema:"rename the trait"`
|
||||
TraitType *string `json:"trait_type,omitempty" jsonschema:"update trait type"`
|
||||
Description *string `json:"description,omitempty" jsonschema:"update description"`
|
||||
Instruction *string `json:"instruction,omitempty" jsonschema:"update instruction"`
|
||||
Tags []string `json:"tags,omitempty" jsonschema:"replace tags"`
|
||||
}
|
||||
|
||||
type UpdateTraitOutput struct {
|
||||
Trait ext.Trait `json:"trait"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) UpdateTrait(ctx context.Context, _ *mcp.CallToolRequest, in UpdateTraitInput) (*mcp.CallToolResult, UpdateTraitOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, UpdateTraitOutput{}, errRequiredField("name")
|
||||
}
|
||||
updates := map[string]any{}
|
||||
if n := strings.TrimSpace(in.NewName); n != "" {
|
||||
updates["name"] = n
|
||||
}
|
||||
if in.TraitType != nil {
|
||||
updates["trait_type"] = strings.TrimSpace(*in.TraitType)
|
||||
}
|
||||
if in.Description != nil {
|
||||
updates["description"] = strings.TrimSpace(*in.Description)
|
||||
}
|
||||
if in.Instruction != nil {
|
||||
updates["instruction"] = strings.TrimSpace(*in.Instruction)
|
||||
}
|
||||
if in.Tags != nil {
|
||||
updates["tags"] = in.Tags
|
||||
}
|
||||
trait, err := t.store.UpdateTrait(ctx, strings.TrimSpace(in.Name), updates)
|
||||
if err != nil {
|
||||
return nil, UpdateTraitOutput{}, err
|
||||
}
|
||||
return nil, UpdateTraitOutput{Trait: trait}, nil
|
||||
}
|
||||
|
||||
// delete_agent_trait
|
||||
|
||||
type DeleteTraitInput struct {
|
||||
Name string `json:"name" jsonschema:"name of the trait to delete"`
|
||||
}
|
||||
|
||||
type DeleteTraitOutput struct {
|
||||
Deleted bool `json:"deleted"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) DeleteTrait(ctx context.Context, _ *mcp.CallToolRequest, in DeleteTraitInput) (*mcp.CallToolResult, DeleteTraitOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, DeleteTraitOutput{}, errRequiredField("name")
|
||||
}
|
||||
if err := t.store.DeleteTrait(ctx, strings.TrimSpace(in.Name)); err != nil {
|
||||
return nil, DeleteTraitOutput{}, err
|
||||
}
|
||||
return nil, DeleteTraitOutput{Deleted: true}, nil
|
||||
}
|
||||
|
||||
// list_agent_traits
|
||||
|
||||
type ListTraitsInput struct {
|
||||
TraitType string `json:"trait_type,omitempty" jsonschema:"filter by trait type"`
|
||||
Tag string `json:"tag,omitempty" jsonschema:"filter by tag"`
|
||||
}
|
||||
|
||||
type ListTraitsOutput struct {
|
||||
Traits []ext.Trait `json:"traits"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) ListTraits(ctx context.Context, _ *mcp.CallToolRequest, in ListTraitsInput) (*mcp.CallToolResult, ListTraitsOutput, error) {
|
||||
traits, err := t.store.ListTraits(ctx, in.TraitType, in.Tag)
|
||||
if err != nil {
|
||||
return nil, ListTraitsOutput{}, err
|
||||
}
|
||||
if traits == nil {
|
||||
traits = []ext.Trait{}
|
||||
}
|
||||
return nil, ListTraitsOutput{Traits: traits}, nil
|
||||
}
|
||||
|
||||
// get_agent_trait
|
||||
|
||||
type GetTraitInput struct {
|
||||
Name string `json:"name" jsonschema:"trait name"`
|
||||
}
|
||||
|
||||
type GetTraitOutput struct {
|
||||
Trait ext.Trait `json:"trait"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) GetTrait(ctx context.Context, _ *mcp.CallToolRequest, in GetTraitInput) (*mcp.CallToolResult, GetTraitOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, GetTraitOutput{}, errRequiredField("name")
|
||||
}
|
||||
trait, err := t.store.GetTraitByName(ctx, strings.TrimSpace(in.Name))
|
||||
if err != nil {
|
||||
return nil, GetTraitOutput{}, err
|
||||
}
|
||||
return nil, GetTraitOutput{Trait: trait}, nil
|
||||
}
|
||||
|
||||
// add_persona_trait
|
||||
|
||||
type AddPersonaTraitInput struct {
|
||||
PersonaID int64 `json:"persona_id" jsonschema:"persona id"`
|
||||
TraitID int64 `json:"trait_id" jsonschema:"agent trait id to link"`
|
||||
}
|
||||
|
||||
type AddPersonaTraitOutput struct {
|
||||
PersonaID int64 `json:"persona_id"`
|
||||
TraitID int64 `json:"trait_id"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) AddPersonaTrait(ctx context.Context, _ *mcp.CallToolRequest, in AddPersonaTraitInput) (*mcp.CallToolResult, AddPersonaTraitOutput, error) {
|
||||
if err := t.store.AddPersonaTrait(ctx, in.PersonaID, in.TraitID); err != nil {
|
||||
return nil, AddPersonaTraitOutput{}, err
|
||||
}
|
||||
return nil, AddPersonaTraitOutput{PersonaID: in.PersonaID, TraitID: in.TraitID}, nil
|
||||
}
|
||||
|
||||
// remove_persona_trait
|
||||
|
||||
type RemovePersonaTraitInput struct {
|
||||
PersonaID int64 `json:"persona_id" jsonschema:"persona id"`
|
||||
TraitID int64 `json:"trait_id" jsonschema:"agent trait id to unlink"`
|
||||
}
|
||||
|
||||
type RemovePersonaTraitOutput struct {
|
||||
Removed bool `json:"removed"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) RemovePersonaTrait(ctx context.Context, _ *mcp.CallToolRequest, in RemovePersonaTraitInput) (*mcp.CallToolResult, RemovePersonaTraitOutput, error) {
|
||||
if err := t.store.RemovePersonaTrait(ctx, in.PersonaID, in.TraitID); err != nil {
|
||||
return nil, RemovePersonaTraitOutput{}, err
|
||||
}
|
||||
return nil, RemovePersonaTraitOutput{Removed: true}, nil
|
||||
}
|
||||
|
||||
// ──────────────────────────────────────────────
|
||||
// Character Arcs
|
||||
// ──────────────────────────────────────────────
|
||||
|
||||
// create_character_arc
|
||||
|
||||
type CreateCharacterArcInput struct {
|
||||
Name string `json:"name" jsonschema:"unique arc name"`
|
||||
Description string `json:"description,omitempty" jsonschema:"description of the arc"`
|
||||
Summary string `json:"summary,omitempty" jsonschema:"brief arc summary"`
|
||||
}
|
||||
|
||||
type CreateCharacterArcOutput struct {
|
||||
Arc ext.CharacterArc `json:"arc"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) CreateCharacterArc(ctx context.Context, _ *mcp.CallToolRequest, in CreateCharacterArcInput) (*mcp.CallToolResult, CreateCharacterArcOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, CreateCharacterArcOutput{}, errRequiredField("name")
|
||||
}
|
||||
arc, err := t.store.CreateCharacterArc(ctx, ext.CharacterArc{
|
||||
Name: strings.TrimSpace(in.Name),
|
||||
Description: strings.TrimSpace(in.Description),
|
||||
Summary: strings.TrimSpace(in.Summary),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, CreateCharacterArcOutput{}, err
|
||||
}
|
||||
return nil, CreateCharacterArcOutput{Arc: arc}, nil
|
||||
}
|
||||
|
||||
// list_character_arcs
|
||||
|
||||
type ListCharacterArcsInput struct{}
|
||||
|
||||
type ListCharacterArcsOutput struct {
|
||||
Arcs []ext.CharacterArc `json:"arcs"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) ListCharacterArcs(ctx context.Context, _ *mcp.CallToolRequest, _ ListCharacterArcsInput) (*mcp.CallToolResult, ListCharacterArcsOutput, error) {
|
||||
arcs, err := t.store.ListCharacterArcs(ctx)
|
||||
if err != nil {
|
||||
return nil, ListCharacterArcsOutput{}, err
|
||||
}
|
||||
if arcs == nil {
|
||||
arcs = []ext.CharacterArc{}
|
||||
}
|
||||
return nil, ListCharacterArcsOutput{Arcs: arcs}, nil
|
||||
}
|
||||
|
||||
// add_arc_stage
|
||||
|
||||
type AddArcStageInput struct {
|
||||
Arc string `json:"arc" jsonschema:"arc name"`
|
||||
Name string `json:"name" jsonschema:"stage name"`
|
||||
StageOrder int `json:"stage_order,omitempty" jsonschema:"position in arc sequence (lower first)"`
|
||||
Description string `json:"description,omitempty" jsonschema:"what happens at this stage"`
|
||||
Condition string `json:"condition,omitempty" jsonschema:"trigger condition description (evaluated externally)"`
|
||||
}
|
||||
|
||||
type AddArcStageOutput struct {
|
||||
Stage ext.ArcStage `json:"stage"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) AddArcStage(ctx context.Context, _ *mcp.CallToolRequest, in AddArcStageInput) (*mcp.CallToolResult, AddArcStageOutput, error) {
|
||||
if strings.TrimSpace(in.Arc) == "" {
|
||||
return nil, AddArcStageOutput{}, errRequiredField("arc")
|
||||
}
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, AddArcStageOutput{}, errRequiredField("name")
|
||||
}
|
||||
stage, err := t.store.AddArcStage(ctx, strings.TrimSpace(in.Arc), ext.ArcStage{
|
||||
Name: strings.TrimSpace(in.Name),
|
||||
StageOrder: in.StageOrder,
|
||||
Description: strings.TrimSpace(in.Description),
|
||||
Condition: strings.TrimSpace(in.Condition),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, AddArcStageOutput{}, err
|
||||
}
|
||||
return nil, AddArcStageOutput{Stage: stage}, nil
|
||||
}
|
||||
|
||||
// add_stage_part
|
||||
|
||||
type AddStagePartInput struct {
|
||||
StageID int64 `json:"stage_id" jsonschema:"arc stage id"`
|
||||
PartName string `json:"part_name" jsonschema:"part name to link"`
|
||||
}
|
||||
|
||||
type AddStagePartOutput struct {
|
||||
StageID int64 `json:"stage_id"`
|
||||
PartName string `json:"part_name"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) AddStagePart(ctx context.Context, _ *mcp.CallToolRequest, in AddStagePartInput) (*mcp.CallToolResult, AddStagePartOutput, error) {
|
||||
if strings.TrimSpace(in.PartName) == "" {
|
||||
return nil, AddStagePartOutput{}, errRequiredField("part_name")
|
||||
}
|
||||
if err := t.store.AddStagePart(ctx, in.StageID, strings.TrimSpace(in.PartName)); err != nil {
|
||||
return nil, AddStagePartOutput{}, err
|
||||
}
|
||||
return nil, AddStagePartOutput{StageID: in.StageID, PartName: in.PartName}, nil
|
||||
}
|
||||
|
||||
// remove_stage_part
|
||||
|
||||
type RemoveStagePartInput struct {
|
||||
StageID int64 `json:"stage_id" jsonschema:"arc stage id"`
|
||||
PartName string `json:"part_name" jsonschema:"part name to unlink"`
|
||||
}
|
||||
|
||||
type RemoveStagePartOutput struct {
|
||||
Removed bool `json:"removed"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) RemoveStagePart(ctx context.Context, _ *mcp.CallToolRequest, in RemoveStagePartInput) (*mcp.CallToolResult, RemoveStagePartOutput, error) {
|
||||
if strings.TrimSpace(in.PartName) == "" {
|
||||
return nil, RemoveStagePartOutput{}, errRequiredField("part_name")
|
||||
}
|
||||
if err := t.store.RemoveStagePart(ctx, in.StageID, strings.TrimSpace(in.PartName)); err != nil {
|
||||
return nil, RemoveStagePartOutput{}, err
|
||||
}
|
||||
return nil, RemoveStagePartOutput{Removed: true}, nil
|
||||
}
|
||||
|
||||
// assign_persona_arc
|
||||
|
||||
type AssignPersonaArcInput struct {
|
||||
Persona string `json:"persona" jsonschema:"persona name"`
|
||||
Arc string `json:"arc" jsonschema:"arc name to assign"`
|
||||
StartStage string `json:"start_stage" jsonschema:"name of the starting stage"`
|
||||
}
|
||||
|
||||
type AssignPersonaArcOutput struct {
|
||||
Persona string `json:"persona"`
|
||||
Arc string `json:"arc"`
|
||||
Stage string `json:"stage"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) AssignPersonaArc(ctx context.Context, _ *mcp.CallToolRequest, in AssignPersonaArcInput) (*mcp.CallToolResult, AssignPersonaArcOutput, error) {
|
||||
if strings.TrimSpace(in.Persona) == "" {
|
||||
return nil, AssignPersonaArcOutput{}, errRequiredField("persona")
|
||||
}
|
||||
if strings.TrimSpace(in.Arc) == "" {
|
||||
return nil, AssignPersonaArcOutput{}, errRequiredField("arc")
|
||||
}
|
||||
if strings.TrimSpace(in.StartStage) == "" {
|
||||
return nil, AssignPersonaArcOutput{}, errRequiredField("start_stage")
|
||||
}
|
||||
if err := t.store.AssignPersonaArc(ctx, strings.TrimSpace(in.Persona), strings.TrimSpace(in.Arc), strings.TrimSpace(in.StartStage)); err != nil {
|
||||
return nil, AssignPersonaArcOutput{}, err
|
||||
}
|
||||
return nil, AssignPersonaArcOutput{Persona: in.Persona, Arc: in.Arc, Stage: in.StartStage}, nil
|
||||
}
|
||||
|
||||
// advance_persona_stage
|
||||
|
||||
type AdvancePersonaStageInput struct {
|
||||
Persona string `json:"persona" jsonschema:"persona name"`
|
||||
}
|
||||
|
||||
type AdvancePersonaStageOutput struct {
|
||||
Stage ext.ArcStage `json:"stage"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) AdvancePersonaStage(ctx context.Context, _ *mcp.CallToolRequest, in AdvancePersonaStageInput) (*mcp.CallToolResult, AdvancePersonaStageOutput, error) {
|
||||
if strings.TrimSpace(in.Persona) == "" {
|
||||
return nil, AdvancePersonaStageOutput{}, errRequiredField("persona")
|
||||
}
|
||||
stage, err := t.store.AdvancePersonaStage(ctx, strings.TrimSpace(in.Persona))
|
||||
if err != nil {
|
||||
return nil, AdvancePersonaStageOutput{}, err
|
||||
}
|
||||
return nil, AdvancePersonaStageOutput{Stage: stage}, nil
|
||||
}
|
||||
|
||||
// reset_persona_stage
|
||||
|
||||
type ResetPersonaStageInput struct {
|
||||
Persona string `json:"persona" jsonschema:"persona name"`
|
||||
}
|
||||
|
||||
type ResetPersonaStageOutput struct {
|
||||
Stage ext.ArcStage `json:"stage"`
|
||||
}
|
||||
|
||||
func (t *AgentPersonasTool) ResetPersonaStage(ctx context.Context, _ *mcp.CallToolRequest, in ResetPersonaStageInput) (*mcp.CallToolResult, ResetPersonaStageOutput, error) {
|
||||
if strings.TrimSpace(in.Persona) == "" {
|
||||
return nil, ResetPersonaStageOutput{}, errRequiredField("persona")
|
||||
}
|
||||
stage, err := t.store.ResetPersonaStage(ctx, strings.TrimSpace(in.Persona))
|
||||
if err != nil {
|
||||
return nil, ResetPersonaStageOutput{}, err
|
||||
}
|
||||
return nil, ResetPersonaStageOutput{Stage: stage}, nil
|
||||
}
|
||||
+13
-15
@@ -2,11 +2,11 @@ package tools
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"log/slog"
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
"github.com/google/uuid"
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
"golang.org/x/sync/semaphore"
|
||||
|
||||
@@ -53,12 +53,11 @@ func NewBackfillTool(db *store.DB, embeddings *ai.EmbeddingRunner, sessions *ses
|
||||
|
||||
// QueueThought queues a single thought for background embedding generation.
|
||||
// It is used by capture when the embedding provider is temporarily unavailable.
|
||||
func (t *BackfillTool) QueueThought(ctx context.Context, id int64, content string) {
|
||||
func (t *BackfillTool) QueueThought(ctx context.Context, id uuid.UUID, content string) {
|
||||
go func() {
|
||||
started := time.Now()
|
||||
idStr := fmt.Sprint(id)
|
||||
t.logger.Info("background embedding started",
|
||||
slog.String("thought_id", idStr),
|
||||
slog.String("thought_id", id.String()),
|
||||
slog.String("provider", t.embeddings.PrimaryProvider()),
|
||||
slog.String("model", t.embeddings.PrimaryModel()),
|
||||
)
|
||||
@@ -66,7 +65,7 @@ func (t *BackfillTool) QueueThought(ctx context.Context, id int64, content strin
|
||||
result, err := t.embeddings.Embed(ctx, content)
|
||||
if err != nil {
|
||||
t.logger.Warn("background embedding error",
|
||||
slog.String("thought_id", idStr),
|
||||
slog.String("thought_id", id.String()),
|
||||
slog.String("provider", t.embeddings.PrimaryProvider()),
|
||||
slog.String("model", t.embeddings.PrimaryModel()),
|
||||
slog.String("stage", "embed"),
|
||||
@@ -77,7 +76,7 @@ func (t *BackfillTool) QueueThought(ctx context.Context, id int64, content strin
|
||||
}
|
||||
if err := t.store.UpsertEmbedding(ctx, id, result.Model, result.Vector); err != nil {
|
||||
t.logger.Warn("background embedding error",
|
||||
slog.String("thought_id", idStr),
|
||||
slog.String("thought_id", id.String()),
|
||||
slog.String("provider", t.embeddings.PrimaryProvider()),
|
||||
slog.String("model", result.Model),
|
||||
slog.String("stage", "upsert"),
|
||||
@@ -87,7 +86,7 @@ func (t *BackfillTool) QueueThought(ctx context.Context, id int64, content strin
|
||||
return
|
||||
}
|
||||
t.logger.Info("background embedding complete",
|
||||
slog.String("thought_id", idStr),
|
||||
slog.String("thought_id", id.String()),
|
||||
slog.String("provider", t.embeddings.PrimaryProvider()),
|
||||
slog.String("model", result.Model),
|
||||
slog.Duration("duration", time.Since(started)),
|
||||
@@ -106,9 +105,9 @@ func (t *BackfillTool) Handle(ctx context.Context, req *mcp.CallToolRequest, in
|
||||
return nil, BackfillOutput{}, err
|
||||
}
|
||||
|
||||
var projectID *int64
|
||||
var projectID *uuid.UUID
|
||||
if project != nil {
|
||||
projectID = &project.NumericID
|
||||
projectID = &project.ID
|
||||
}
|
||||
|
||||
primaryModel := t.embeddings.PrimaryModel()
|
||||
@@ -141,25 +140,24 @@ func (t *BackfillTool) Handle(ctx context.Context, req *mcp.CallToolRequest, in
|
||||
break
|
||||
}
|
||||
wg.Add(1)
|
||||
go func(id int64, content string) {
|
||||
go func(id uuid.UUID, content string) {
|
||||
defer wg.Done()
|
||||
defer sem.Release(1)
|
||||
|
||||
idStr := fmt.Sprint(id)
|
||||
result, embedErr := t.embeddings.Embed(ctx, content)
|
||||
if embedErr != nil {
|
||||
mu.Lock()
|
||||
out.Failures = append(out.Failures, BackfillFailure{ID: idStr, Error: embedErr.Error()})
|
||||
out.Failures = append(out.Failures, BackfillFailure{ID: id.String(), Error: embedErr.Error()})
|
||||
mu.Unlock()
|
||||
t.logger.Warn("backfill embed failed", slog.String("thought_id", idStr), slog.String("error", embedErr.Error()))
|
||||
t.logger.Warn("backfill embed failed", slog.String("thought_id", id.String()), slog.String("error", embedErr.Error()))
|
||||
return
|
||||
}
|
||||
|
||||
if upsertErr := t.store.UpsertEmbedding(ctx, id, result.Model, result.Vector); upsertErr != nil {
|
||||
mu.Lock()
|
||||
out.Failures = append(out.Failures, BackfillFailure{ID: idStr, Error: upsertErr.Error()})
|
||||
out.Failures = append(out.Failures, BackfillFailure{ID: id.String(), Error: upsertErr.Error()})
|
||||
mu.Unlock()
|
||||
t.logger.Warn("backfill upsert failed", slog.String("thought_id", idStr), slog.String("error", upsertErr.Error()))
|
||||
t.logger.Warn("backfill upsert failed", slog.String("thought_id", id.String()), slog.String("error", upsertErr.Error()))
|
||||
return
|
||||
}
|
||||
|
||||
|
||||
+178
-178
@@ -1,212 +1,212 @@
|
||||
package tools
|
||||
|
||||
// import (
|
||||
// "context"
|
||||
// "strings"
|
||||
// "time"
|
||||
import (
|
||||
"context"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
// "github.com/google/uuid"
|
||||
// "github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
"github.com/google/uuid"
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
|
||||
// "git.warky.dev/wdevs/amcs/internal/store"
|
||||
// ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
// )
|
||||
"git.warky.dev/wdevs/amcs/internal/store"
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
// type CalendarTool struct {
|
||||
// store *store.DB
|
||||
// }
|
||||
type CalendarTool struct {
|
||||
store *store.DB
|
||||
}
|
||||
|
||||
// func NewCalendarTool(db *store.DB) *CalendarTool {
|
||||
// return &CalendarTool{store: db}
|
||||
// }
|
||||
func NewCalendarTool(db *store.DB) *CalendarTool {
|
||||
return &CalendarTool{store: db}
|
||||
}
|
||||
|
||||
// // add_family_member
|
||||
// add_family_member
|
||||
|
||||
// type AddFamilyMemberInput struct {
|
||||
// Name string `json:"name" jsonschema:"person's name"`
|
||||
// Relationship string `json:"relationship,omitempty" jsonschema:"e.g. self, spouse, child, parent"`
|
||||
// BirthDate *time.Time `json:"birth_date,omitempty"`
|
||||
// Notes string `json:"notes,omitempty"`
|
||||
// }
|
||||
type AddFamilyMemberInput struct {
|
||||
Name string `json:"name" jsonschema:"person's name"`
|
||||
Relationship string `json:"relationship,omitempty" jsonschema:"e.g. self, spouse, child, parent"`
|
||||
BirthDate *time.Time `json:"birth_date,omitempty"`
|
||||
Notes string `json:"notes,omitempty"`
|
||||
}
|
||||
|
||||
// type AddFamilyMemberOutput struct {
|
||||
// Member ext.FamilyMember `json:"member"`
|
||||
// }
|
||||
type AddFamilyMemberOutput struct {
|
||||
Member ext.FamilyMember `json:"member"`
|
||||
}
|
||||
|
||||
// func (t *CalendarTool) AddMember(ctx context.Context, _ *mcp.CallToolRequest, in AddFamilyMemberInput) (*mcp.CallToolResult, AddFamilyMemberOutput, error) {
|
||||
// if strings.TrimSpace(in.Name) == "" {
|
||||
// return nil, AddFamilyMemberOutput{}, errRequiredField("name")
|
||||
// }
|
||||
// member, err := t.store.AddFamilyMember(ctx, ext.FamilyMember{
|
||||
// Name: strings.TrimSpace(in.Name),
|
||||
// Relationship: strings.TrimSpace(in.Relationship),
|
||||
// BirthDate: in.BirthDate,
|
||||
// Notes: strings.TrimSpace(in.Notes),
|
||||
// })
|
||||
// if err != nil {
|
||||
// return nil, AddFamilyMemberOutput{}, err
|
||||
// }
|
||||
// return nil, AddFamilyMemberOutput{Member: member}, nil
|
||||
// }
|
||||
func (t *CalendarTool) AddMember(ctx context.Context, _ *mcp.CallToolRequest, in AddFamilyMemberInput) (*mcp.CallToolResult, AddFamilyMemberOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, AddFamilyMemberOutput{}, errRequiredField("name")
|
||||
}
|
||||
member, err := t.store.AddFamilyMember(ctx, ext.FamilyMember{
|
||||
Name: strings.TrimSpace(in.Name),
|
||||
Relationship: strings.TrimSpace(in.Relationship),
|
||||
BirthDate: in.BirthDate,
|
||||
Notes: strings.TrimSpace(in.Notes),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, AddFamilyMemberOutput{}, err
|
||||
}
|
||||
return nil, AddFamilyMemberOutput{Member: member}, nil
|
||||
}
|
||||
|
||||
// // list_family_members
|
||||
// list_family_members
|
||||
|
||||
// type ListFamilyMembersInput struct{}
|
||||
type ListFamilyMembersInput struct{}
|
||||
|
||||
// type ListFamilyMembersOutput struct {
|
||||
// Members []ext.FamilyMember `json:"members"`
|
||||
// }
|
||||
type ListFamilyMembersOutput struct {
|
||||
Members []ext.FamilyMember `json:"members"`
|
||||
}
|
||||
|
||||
// func (t *CalendarTool) ListMembers(ctx context.Context, _ *mcp.CallToolRequest, _ ListFamilyMembersInput) (*mcp.CallToolResult, ListFamilyMembersOutput, error) {
|
||||
// members, err := t.store.ListFamilyMembers(ctx)
|
||||
// if err != nil {
|
||||
// return nil, ListFamilyMembersOutput{}, err
|
||||
// }
|
||||
// if members == nil {
|
||||
// members = []ext.FamilyMember{}
|
||||
// }
|
||||
// return nil, ListFamilyMembersOutput{Members: members}, nil
|
||||
// }
|
||||
func (t *CalendarTool) ListMembers(ctx context.Context, _ *mcp.CallToolRequest, _ ListFamilyMembersInput) (*mcp.CallToolResult, ListFamilyMembersOutput, error) {
|
||||
members, err := t.store.ListFamilyMembers(ctx)
|
||||
if err != nil {
|
||||
return nil, ListFamilyMembersOutput{}, err
|
||||
}
|
||||
if members == nil {
|
||||
members = []ext.FamilyMember{}
|
||||
}
|
||||
return nil, ListFamilyMembersOutput{Members: members}, nil
|
||||
}
|
||||
|
||||
// // add_activity
|
||||
// add_activity
|
||||
|
||||
// type AddActivityInput struct {
|
||||
// Title string `json:"title" jsonschema:"activity title"`
|
||||
// ActivityType string `json:"activity_type,omitempty" jsonschema:"e.g. sports, medical, school, social"`
|
||||
// FamilyMemberID *uuid.UUID `json:"family_member_id,omitempty" jsonschema:"leave empty for whole-family activities"`
|
||||
// DayOfWeek string `json:"day_of_week,omitempty" jsonschema:"for recurring: monday, tuesday, etc."`
|
||||
// StartTime string `json:"start_time,omitempty" jsonschema:"HH:MM format"`
|
||||
// EndTime string `json:"end_time,omitempty" jsonschema:"HH:MM format"`
|
||||
// StartDate *time.Time `json:"start_date,omitempty"`
|
||||
// EndDate *time.Time `json:"end_date,omitempty" jsonschema:"for recurring activities, when they end"`
|
||||
// Location string `json:"location,omitempty"`
|
||||
// Notes string `json:"notes,omitempty"`
|
||||
// }
|
||||
type AddActivityInput struct {
|
||||
Title string `json:"title" jsonschema:"activity title"`
|
||||
ActivityType string `json:"activity_type,omitempty" jsonschema:"e.g. sports, medical, school, social"`
|
||||
FamilyMemberID *uuid.UUID `json:"family_member_id,omitempty" jsonschema:"leave empty for whole-family activities"`
|
||||
DayOfWeek string `json:"day_of_week,omitempty" jsonschema:"for recurring: monday, tuesday, etc."`
|
||||
StartTime string `json:"start_time,omitempty" jsonschema:"HH:MM format"`
|
||||
EndTime string `json:"end_time,omitempty" jsonschema:"HH:MM format"`
|
||||
StartDate *time.Time `json:"start_date,omitempty"`
|
||||
EndDate *time.Time `json:"end_date,omitempty" jsonschema:"for recurring activities, when they end"`
|
||||
Location string `json:"location,omitempty"`
|
||||
Notes string `json:"notes,omitempty"`
|
||||
}
|
||||
|
||||
// type AddActivityOutput struct {
|
||||
// Activity ext.Activity `json:"activity"`
|
||||
// }
|
||||
type AddActivityOutput struct {
|
||||
Activity ext.Activity `json:"activity"`
|
||||
}
|
||||
|
||||
// func (t *CalendarTool) AddActivity(ctx context.Context, _ *mcp.CallToolRequest, in AddActivityInput) (*mcp.CallToolResult, AddActivityOutput, error) {
|
||||
// if strings.TrimSpace(in.Title) == "" {
|
||||
// return nil, AddActivityOutput{}, errRequiredField("title")
|
||||
// }
|
||||
// activity, err := t.store.AddActivity(ctx, ext.Activity{
|
||||
// FamilyMemberID: in.FamilyMemberID,
|
||||
// Title: strings.TrimSpace(in.Title),
|
||||
// ActivityType: strings.TrimSpace(in.ActivityType),
|
||||
// DayOfWeek: strings.ToLower(strings.TrimSpace(in.DayOfWeek)),
|
||||
// StartTime: strings.TrimSpace(in.StartTime),
|
||||
// EndTime: strings.TrimSpace(in.EndTime),
|
||||
// StartDate: in.StartDate,
|
||||
// EndDate: in.EndDate,
|
||||
// Location: strings.TrimSpace(in.Location),
|
||||
// Notes: strings.TrimSpace(in.Notes),
|
||||
// })
|
||||
// if err != nil {
|
||||
// return nil, AddActivityOutput{}, err
|
||||
// }
|
||||
// return nil, AddActivityOutput{Activity: activity}, nil
|
||||
// }
|
||||
func (t *CalendarTool) AddActivity(ctx context.Context, _ *mcp.CallToolRequest, in AddActivityInput) (*mcp.CallToolResult, AddActivityOutput, error) {
|
||||
if strings.TrimSpace(in.Title) == "" {
|
||||
return nil, AddActivityOutput{}, errRequiredField("title")
|
||||
}
|
||||
activity, err := t.store.AddActivity(ctx, ext.Activity{
|
||||
FamilyMemberID: in.FamilyMemberID,
|
||||
Title: strings.TrimSpace(in.Title),
|
||||
ActivityType: strings.TrimSpace(in.ActivityType),
|
||||
DayOfWeek: strings.ToLower(strings.TrimSpace(in.DayOfWeek)),
|
||||
StartTime: strings.TrimSpace(in.StartTime),
|
||||
EndTime: strings.TrimSpace(in.EndTime),
|
||||
StartDate: in.StartDate,
|
||||
EndDate: in.EndDate,
|
||||
Location: strings.TrimSpace(in.Location),
|
||||
Notes: strings.TrimSpace(in.Notes),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, AddActivityOutput{}, err
|
||||
}
|
||||
return nil, AddActivityOutput{Activity: activity}, nil
|
||||
}
|
||||
|
||||
// // get_week_schedule
|
||||
// get_week_schedule
|
||||
|
||||
// type GetWeekScheduleInput struct {
|
||||
// WeekStart time.Time `json:"week_start" jsonschema:"start of the week (Monday) to retrieve"`
|
||||
// }
|
||||
type GetWeekScheduleInput struct {
|
||||
WeekStart time.Time `json:"week_start" jsonschema:"start of the week (Monday) to retrieve"`
|
||||
}
|
||||
|
||||
// type GetWeekScheduleOutput struct {
|
||||
// Activities []ext.Activity `json:"activities"`
|
||||
// }
|
||||
type GetWeekScheduleOutput struct {
|
||||
Activities []ext.Activity `json:"activities"`
|
||||
}
|
||||
|
||||
// func (t *CalendarTool) GetWeekSchedule(ctx context.Context, _ *mcp.CallToolRequest, in GetWeekScheduleInput) (*mcp.CallToolResult, GetWeekScheduleOutput, error) {
|
||||
// activities, err := t.store.GetWeekSchedule(ctx, in.WeekStart)
|
||||
// if err != nil {
|
||||
// return nil, GetWeekScheduleOutput{}, err
|
||||
// }
|
||||
// if activities == nil {
|
||||
// activities = []ext.Activity{}
|
||||
// }
|
||||
// return nil, GetWeekScheduleOutput{Activities: activities}, nil
|
||||
// }
|
||||
func (t *CalendarTool) GetWeekSchedule(ctx context.Context, _ *mcp.CallToolRequest, in GetWeekScheduleInput) (*mcp.CallToolResult, GetWeekScheduleOutput, error) {
|
||||
activities, err := t.store.GetWeekSchedule(ctx, in.WeekStart)
|
||||
if err != nil {
|
||||
return nil, GetWeekScheduleOutput{}, err
|
||||
}
|
||||
if activities == nil {
|
||||
activities = []ext.Activity{}
|
||||
}
|
||||
return nil, GetWeekScheduleOutput{Activities: activities}, nil
|
||||
}
|
||||
|
||||
// // search_activities
|
||||
// search_activities
|
||||
|
||||
// type SearchActivitiesInput struct {
|
||||
// Query string `json:"query,omitempty" jsonschema:"search text matching title or notes"`
|
||||
// ActivityType string `json:"activity_type,omitempty" jsonschema:"filter by type"`
|
||||
// FamilyMemberID *uuid.UUID `json:"family_member_id,omitempty" jsonschema:"filter by family member"`
|
||||
// }
|
||||
type SearchActivitiesInput struct {
|
||||
Query string `json:"query,omitempty" jsonschema:"search text matching title or notes"`
|
||||
ActivityType string `json:"activity_type,omitempty" jsonschema:"filter by type"`
|
||||
FamilyMemberID *uuid.UUID `json:"family_member_id,omitempty" jsonschema:"filter by family member"`
|
||||
}
|
||||
|
||||
// type SearchActivitiesOutput struct {
|
||||
// Activities []ext.Activity `json:"activities"`
|
||||
// }
|
||||
type SearchActivitiesOutput struct {
|
||||
Activities []ext.Activity `json:"activities"`
|
||||
}
|
||||
|
||||
// func (t *CalendarTool) SearchActivities(ctx context.Context, _ *mcp.CallToolRequest, in SearchActivitiesInput) (*mcp.CallToolResult, SearchActivitiesOutput, error) {
|
||||
// activities, err := t.store.SearchActivities(ctx, in.Query, in.ActivityType, in.FamilyMemberID)
|
||||
// if err != nil {
|
||||
// return nil, SearchActivitiesOutput{}, err
|
||||
// }
|
||||
// if activities == nil {
|
||||
// activities = []ext.Activity{}
|
||||
// }
|
||||
// return nil, SearchActivitiesOutput{Activities: activities}, nil
|
||||
// }
|
||||
func (t *CalendarTool) SearchActivities(ctx context.Context, _ *mcp.CallToolRequest, in SearchActivitiesInput) (*mcp.CallToolResult, SearchActivitiesOutput, error) {
|
||||
activities, err := t.store.SearchActivities(ctx, in.Query, in.ActivityType, in.FamilyMemberID)
|
||||
if err != nil {
|
||||
return nil, SearchActivitiesOutput{}, err
|
||||
}
|
||||
if activities == nil {
|
||||
activities = []ext.Activity{}
|
||||
}
|
||||
return nil, SearchActivitiesOutput{Activities: activities}, nil
|
||||
}
|
||||
|
||||
// // add_important_date
|
||||
// add_important_date
|
||||
|
||||
// type AddImportantDateInput struct {
|
||||
// Title string `json:"title" jsonschema:"description of the date"`
|
||||
// DateValue time.Time `json:"date_value" jsonschema:"the date"`
|
||||
// FamilyMemberID *uuid.UUID `json:"family_member_id,omitempty"`
|
||||
// RecurringYearly bool `json:"recurring_yearly,omitempty" jsonschema:"if true, reminds every year"`
|
||||
// ReminderDaysBefore int `json:"reminder_days_before,omitempty" jsonschema:"how many days before to remind (default: 7)"`
|
||||
// Notes string `json:"notes,omitempty"`
|
||||
// }
|
||||
type AddImportantDateInput struct {
|
||||
Title string `json:"title" jsonschema:"description of the date"`
|
||||
DateValue time.Time `json:"date_value" jsonschema:"the date"`
|
||||
FamilyMemberID *uuid.UUID `json:"family_member_id,omitempty"`
|
||||
RecurringYearly bool `json:"recurring_yearly,omitempty" jsonschema:"if true, reminds every year"`
|
||||
ReminderDaysBefore int `json:"reminder_days_before,omitempty" jsonschema:"how many days before to remind (default: 7)"`
|
||||
Notes string `json:"notes,omitempty"`
|
||||
}
|
||||
|
||||
// type AddImportantDateOutput struct {
|
||||
// Date ext.ImportantDate `json:"date"`
|
||||
// }
|
||||
type AddImportantDateOutput struct {
|
||||
Date ext.ImportantDate `json:"date"`
|
||||
}
|
||||
|
||||
// func (t *CalendarTool) AddImportantDate(ctx context.Context, _ *mcp.CallToolRequest, in AddImportantDateInput) (*mcp.CallToolResult, AddImportantDateOutput, error) {
|
||||
// if strings.TrimSpace(in.Title) == "" {
|
||||
// return nil, AddImportantDateOutput{}, errRequiredField("title")
|
||||
// }
|
||||
// reminder := in.ReminderDaysBefore
|
||||
// if reminder <= 0 {
|
||||
// reminder = 7
|
||||
// }
|
||||
// d, err := t.store.AddImportantDate(ctx, ext.ImportantDate{
|
||||
// FamilyMemberID: in.FamilyMemberID,
|
||||
// Title: strings.TrimSpace(in.Title),
|
||||
// DateValue: in.DateValue,
|
||||
// RecurringYearly: in.RecurringYearly,
|
||||
// ReminderDaysBefore: reminder,
|
||||
// Notes: strings.TrimSpace(in.Notes),
|
||||
// })
|
||||
// if err != nil {
|
||||
// return nil, AddImportantDateOutput{}, err
|
||||
// }
|
||||
// return nil, AddImportantDateOutput{Date: d}, nil
|
||||
// }
|
||||
func (t *CalendarTool) AddImportantDate(ctx context.Context, _ *mcp.CallToolRequest, in AddImportantDateInput) (*mcp.CallToolResult, AddImportantDateOutput, error) {
|
||||
if strings.TrimSpace(in.Title) == "" {
|
||||
return nil, AddImportantDateOutput{}, errRequiredField("title")
|
||||
}
|
||||
reminder := in.ReminderDaysBefore
|
||||
if reminder <= 0 {
|
||||
reminder = 7
|
||||
}
|
||||
d, err := t.store.AddImportantDate(ctx, ext.ImportantDate{
|
||||
FamilyMemberID: in.FamilyMemberID,
|
||||
Title: strings.TrimSpace(in.Title),
|
||||
DateValue: in.DateValue,
|
||||
RecurringYearly: in.RecurringYearly,
|
||||
ReminderDaysBefore: reminder,
|
||||
Notes: strings.TrimSpace(in.Notes),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, AddImportantDateOutput{}, err
|
||||
}
|
||||
return nil, AddImportantDateOutput{Date: d}, nil
|
||||
}
|
||||
|
||||
// // get_upcoming_dates
|
||||
// get_upcoming_dates
|
||||
|
||||
// type GetUpcomingDatesInput struct {
|
||||
// DaysAhead int `json:"days_ahead,omitempty" jsonschema:"how many days to look ahead (default: 30)"`
|
||||
// }
|
||||
type GetUpcomingDatesInput struct {
|
||||
DaysAhead int `json:"days_ahead,omitempty" jsonschema:"how many days to look ahead (default: 30)"`
|
||||
}
|
||||
|
||||
// type GetUpcomingDatesOutput struct {
|
||||
// Dates []ext.ImportantDate `json:"dates"`
|
||||
// }
|
||||
type GetUpcomingDatesOutput struct {
|
||||
Dates []ext.ImportantDate `json:"dates"`
|
||||
}
|
||||
|
||||
// func (t *CalendarTool) GetUpcomingDates(ctx context.Context, _ *mcp.CallToolRequest, in GetUpcomingDatesInput) (*mcp.CallToolResult, GetUpcomingDatesOutput, error) {
|
||||
// dates, err := t.store.GetUpcomingDates(ctx, in.DaysAhead)
|
||||
// if err != nil {
|
||||
// return nil, GetUpcomingDatesOutput{}, err
|
||||
// }
|
||||
// if dates == nil {
|
||||
// dates = []ext.ImportantDate{}
|
||||
// }
|
||||
// return nil, GetUpcomingDatesOutput{Dates: dates}, nil
|
||||
// }
|
||||
func (t *CalendarTool) GetUpcomingDates(ctx context.Context, _ *mcp.CallToolRequest, in GetUpcomingDatesInput) (*mcp.CallToolResult, GetUpcomingDatesOutput, error) {
|
||||
dates, err := t.store.GetUpcomingDates(ctx, in.DaysAhead)
|
||||
if err != nil {
|
||||
return nil, GetUpcomingDatesOutput{}, err
|
||||
}
|
||||
if dates == nil {
|
||||
dates = []ext.ImportantDate{}
|
||||
}
|
||||
return nil, GetUpcomingDatesOutput{Dates: dates}, nil
|
||||
}
|
||||
|
||||
@@ -4,6 +4,7 @@ import (
|
||||
"context"
|
||||
"strings"
|
||||
|
||||
"github.com/google/uuid"
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/ai"
|
||||
@@ -16,13 +17,13 @@ import (
|
||||
|
||||
// EmbeddingQueuer queues a thought for background embedding generation.
|
||||
type EmbeddingQueuer interface {
|
||||
QueueThought(ctx context.Context, id int64, content string)
|
||||
QueueThought(ctx context.Context, id uuid.UUID, content string)
|
||||
}
|
||||
|
||||
// MetadataQueuer queues a thought for background metadata retry. Both
|
||||
// MetadataRetryer and EnrichmentRetryer satisfy this.
|
||||
type MetadataQueuer interface {
|
||||
QueueThought(id int64)
|
||||
QueueThought(id uuid.UUID)
|
||||
}
|
||||
|
||||
type CaptureTool struct {
|
||||
@@ -65,7 +66,7 @@ func (t *CaptureTool) Handle(ctx context.Context, req *mcp.CallToolRequest, in C
|
||||
Metadata: rawMetadata,
|
||||
}
|
||||
if project != nil {
|
||||
thought.ProjectID = &project.NumericID
|
||||
thought.ProjectID = &project.ID
|
||||
}
|
||||
|
||||
created, err := t.store.InsertThought(ctx, thought, t.embeddings.PrimaryModel())
|
||||
@@ -73,7 +74,7 @@ func (t *CaptureTool) Handle(ctx context.Context, req *mcp.CallToolRequest, in C
|
||||
return nil, CaptureOutput{}, err
|
||||
}
|
||||
if project != nil {
|
||||
_ = t.store.TouchProject(ctx, project.NumericID)
|
||||
_ = t.store.TouchProject(ctx, project.ID)
|
||||
}
|
||||
|
||||
if t.retryer != nil {
|
||||
|
||||
@@ -4,6 +4,7 @@ import (
|
||||
"context"
|
||||
"strings"
|
||||
|
||||
"github.com/google/uuid"
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/session"
|
||||
@@ -63,7 +64,7 @@ func (t *ChatHistoryTool) SaveChatHistory(ctx context.Context, req *mcp.CallTool
|
||||
h.Metadata = map[string]any{}
|
||||
}
|
||||
if project != nil {
|
||||
h.ProjectID = &project.NumericID
|
||||
h.ProjectID = &project.ID
|
||||
}
|
||||
|
||||
saved, err := t.store.SaveChatHistory(ctx, h)
|
||||
@@ -76,7 +77,7 @@ func (t *ChatHistoryTool) SaveChatHistory(ctx context.Context, req *mcp.CallTool
|
||||
// get_chat_history
|
||||
|
||||
type GetChatHistoryInput struct {
|
||||
ID string `json:"id,omitempty" jsonschema:"numeric id of the saved chat history"`
|
||||
ID string `json:"id,omitempty" jsonschema:"UUID of the saved chat history"`
|
||||
SessionID string `json:"session_id,omitempty" jsonschema:"original session_id — returns the most recent history for that session"`
|
||||
}
|
||||
|
||||
@@ -90,9 +91,9 @@ func (t *ChatHistoryTool) GetChatHistory(ctx context.Context, _ *mcp.CallToolReq
|
||||
}
|
||||
|
||||
if in.ID != "" {
|
||||
id, err := parseID(in.ID)
|
||||
id, err := uuid.Parse(in.ID)
|
||||
if err != nil {
|
||||
return nil, GetChatHistoryOutput{}, errInvalidField("id", "invalid id", "must be a valid numeric id")
|
||||
return nil, GetChatHistoryOutput{}, errInvalidField("id", "invalid id", "must be a valid UUID")
|
||||
}
|
||||
h, found, err := t.store.GetChatHistory(ctx, id)
|
||||
if err != nil {
|
||||
@@ -144,7 +145,7 @@ func (t *ChatHistoryTool) ListChatHistories(ctx context.Context, req *mcp.CallTo
|
||||
return nil, ListChatHistoriesOutput{}, err
|
||||
}
|
||||
if project != nil {
|
||||
filter.ProjectID = &project.NumericID
|
||||
filter.ProjectID = &project.ID
|
||||
}
|
||||
}
|
||||
|
||||
@@ -161,7 +162,7 @@ func (t *ChatHistoryTool) ListChatHistories(ctx context.Context, req *mcp.CallTo
|
||||
// delete_chat_history
|
||||
|
||||
type DeleteChatHistoryInput struct {
|
||||
ID string `json:"id" jsonschema:"numeric id of the chat history to delete"`
|
||||
ID uuid.UUID `json:"id" jsonschema:"UUID of the chat history to delete"`
|
||||
}
|
||||
|
||||
type DeleteChatHistoryOutput struct {
|
||||
@@ -169,11 +170,7 @@ type DeleteChatHistoryOutput struct {
|
||||
}
|
||||
|
||||
func (t *ChatHistoryTool) DeleteChatHistory(ctx context.Context, _ *mcp.CallToolRequest, in DeleteChatHistoryInput) (*mcp.CallToolResult, DeleteChatHistoryOutput, error) {
|
||||
id, err := parseID(in.ID)
|
||||
if err != nil {
|
||||
return nil, DeleteChatHistoryOutput{}, errInvalidField("id", "invalid id", "must be a valid numeric id")
|
||||
}
|
||||
deleted, err := t.store.DeleteChatHistory(ctx, id)
|
||||
deleted, err := t.store.DeleteChatHistory(ctx, in.ID)
|
||||
if err != nil {
|
||||
return nil, DeleteChatHistoryOutput{}, err
|
||||
}
|
||||
|
||||
@@ -52,7 +52,7 @@ func (t *ContextTool) Handle(ctx context.Context, req *mcp.CallToolRequest, in P
|
||||
}
|
||||
|
||||
limit := normalizeLimit(in.Limit, t.search)
|
||||
recent, err := t.store.RecentThoughts(ctx, &project.NumericID, limit, 0)
|
||||
recent, err := t.store.RecentThoughts(ctx, &project.ID, limit, 0)
|
||||
if err != nil {
|
||||
return nil, ProjectContextOutput{}, err
|
||||
}
|
||||
@@ -60,7 +60,7 @@ func (t *ContextTool) Handle(ctx context.Context, req *mcp.CallToolRequest, in P
|
||||
items := make([]ContextItem, 0, limit*2)
|
||||
seen := map[string]struct{}{}
|
||||
for _, thought := range recent {
|
||||
key := fmt.Sprint(thought.ID)
|
||||
key := thought.ID.String()
|
||||
seen[key] = struct{}{}
|
||||
items = append(items, ContextItem{
|
||||
ID: key,
|
||||
@@ -72,12 +72,12 @@ func (t *ContextTool) Handle(ctx context.Context, req *mcp.CallToolRequest, in P
|
||||
|
||||
query := strings.TrimSpace(in.Query)
|
||||
if query != "" {
|
||||
semantic, err := semanticSearch(ctx, t.store, t.embeddings, t.search, query, limit, t.search.DefaultThreshold, &project.NumericID, nil)
|
||||
semantic, err := semanticSearch(ctx, t.store, t.embeddings, t.search, query, limit, t.search.DefaultThreshold, &project.ID, nil)
|
||||
if err != nil {
|
||||
return nil, ProjectContextOutput{}, err
|
||||
}
|
||||
for _, result := range semantic {
|
||||
key := fmt.Sprint(result.ID)
|
||||
key := result.ID.String()
|
||||
if _, ok := seen[key]; ok {
|
||||
continue
|
||||
}
|
||||
@@ -97,7 +97,7 @@ func (t *ContextTool) Handle(ctx context.Context, req *mcp.CallToolRequest, in P
|
||||
lines = append(lines, thoughtContextLine(i, item.Content, item.Metadata, item.Similarity))
|
||||
}
|
||||
contextBlock := formatContextBlock(fmt.Sprintf("Project context for %s", project.Name), lines)
|
||||
_ = t.store.TouchProject(ctx, project.NumericID)
|
||||
_ = t.store.TouchProject(ctx, project.ID)
|
||||
|
||||
return nil, ProjectContextOutput{
|
||||
Project: *project,
|
||||
|
||||
+204
-204
@@ -1,240 +1,240 @@
|
||||
package tools
|
||||
|
||||
// import (
|
||||
// "context"
|
||||
// "errors"
|
||||
// "fmt"
|
||||
// "strings"
|
||||
// "time"
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
// "github.com/google/uuid"
|
||||
// "github.com/jackc/pgx/v5"
|
||||
// "github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
"github.com/google/uuid"
|
||||
"github.com/jackc/pgx/v5"
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
|
||||
// "git.warky.dev/wdevs/amcs/internal/store"
|
||||
// ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
// )
|
||||
"git.warky.dev/wdevs/amcs/internal/store"
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
// type CRMTool struct {
|
||||
// store *store.DB
|
||||
// }
|
||||
type CRMTool struct {
|
||||
store *store.DB
|
||||
}
|
||||
|
||||
// func NewCRMTool(db *store.DB) *CRMTool {
|
||||
// return &CRMTool{store: db}
|
||||
// }
|
||||
func NewCRMTool(db *store.DB) *CRMTool {
|
||||
return &CRMTool{store: db}
|
||||
}
|
||||
|
||||
// // add_professional_contact
|
||||
// add_professional_contact
|
||||
|
||||
// type AddContactInput struct {
|
||||
// Name string `json:"name" jsonschema:"contact's full name"`
|
||||
// Company string `json:"company,omitempty"`
|
||||
// Title string `json:"title,omitempty" jsonschema:"job title"`
|
||||
// Email string `json:"email,omitempty"`
|
||||
// Phone string `json:"phone,omitempty"`
|
||||
// LinkedInURL string `json:"linkedin_url,omitempty"`
|
||||
// HowWeMet string `json:"how_we_met,omitempty"`
|
||||
// Tags []string `json:"tags,omitempty"`
|
||||
// Notes string `json:"notes,omitempty"`
|
||||
// FollowUpDate *time.Time `json:"follow_up_date,omitempty"`
|
||||
// }
|
||||
type AddContactInput struct {
|
||||
Name string `json:"name" jsonschema:"contact's full name"`
|
||||
Company string `json:"company,omitempty"`
|
||||
Title string `json:"title,omitempty" jsonschema:"job title"`
|
||||
Email string `json:"email,omitempty"`
|
||||
Phone string `json:"phone,omitempty"`
|
||||
LinkedInURL string `json:"linkedin_url,omitempty"`
|
||||
HowWeMet string `json:"how_we_met,omitempty"`
|
||||
Tags []string `json:"tags,omitempty"`
|
||||
Notes string `json:"notes,omitempty"`
|
||||
FollowUpDate *time.Time `json:"follow_up_date,omitempty"`
|
||||
}
|
||||
|
||||
// type AddContactOutput struct {
|
||||
// Contact ext.ProfessionalContact `json:"contact"`
|
||||
// }
|
||||
type AddContactOutput struct {
|
||||
Contact ext.ProfessionalContact `json:"contact"`
|
||||
}
|
||||
|
||||
// func (t *CRMTool) AddContact(ctx context.Context, _ *mcp.CallToolRequest, in AddContactInput) (*mcp.CallToolResult, AddContactOutput, error) {
|
||||
// if strings.TrimSpace(in.Name) == "" {
|
||||
// return nil, AddContactOutput{}, errRequiredField("name")
|
||||
// }
|
||||
// if in.Tags == nil {
|
||||
// in.Tags = []string{}
|
||||
// }
|
||||
// contact, err := t.store.AddProfessionalContact(ctx, ext.ProfessionalContact{
|
||||
// Name: strings.TrimSpace(in.Name),
|
||||
// Company: strings.TrimSpace(in.Company),
|
||||
// Title: strings.TrimSpace(in.Title),
|
||||
// Email: strings.TrimSpace(in.Email),
|
||||
// Phone: strings.TrimSpace(in.Phone),
|
||||
// LinkedInURL: strings.TrimSpace(in.LinkedInURL),
|
||||
// HowWeMet: strings.TrimSpace(in.HowWeMet),
|
||||
// Tags: in.Tags,
|
||||
// Notes: strings.TrimSpace(in.Notes),
|
||||
// FollowUpDate: in.FollowUpDate,
|
||||
// })
|
||||
// if err != nil {
|
||||
// return nil, AddContactOutput{}, err
|
||||
// }
|
||||
// return nil, AddContactOutput{Contact: contact}, nil
|
||||
// }
|
||||
func (t *CRMTool) AddContact(ctx context.Context, _ *mcp.CallToolRequest, in AddContactInput) (*mcp.CallToolResult, AddContactOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, AddContactOutput{}, errRequiredField("name")
|
||||
}
|
||||
if in.Tags == nil {
|
||||
in.Tags = []string{}
|
||||
}
|
||||
contact, err := t.store.AddProfessionalContact(ctx, ext.ProfessionalContact{
|
||||
Name: strings.TrimSpace(in.Name),
|
||||
Company: strings.TrimSpace(in.Company),
|
||||
Title: strings.TrimSpace(in.Title),
|
||||
Email: strings.TrimSpace(in.Email),
|
||||
Phone: strings.TrimSpace(in.Phone),
|
||||
LinkedInURL: strings.TrimSpace(in.LinkedInURL),
|
||||
HowWeMet: strings.TrimSpace(in.HowWeMet),
|
||||
Tags: in.Tags,
|
||||
Notes: strings.TrimSpace(in.Notes),
|
||||
FollowUpDate: in.FollowUpDate,
|
||||
})
|
||||
if err != nil {
|
||||
return nil, AddContactOutput{}, err
|
||||
}
|
||||
return nil, AddContactOutput{Contact: contact}, nil
|
||||
}
|
||||
|
||||
// // search_contacts
|
||||
// search_contacts
|
||||
|
||||
// type SearchContactsInput struct {
|
||||
// Query string `json:"query,omitempty" jsonschema:"search text matching name, company, title, or notes"`
|
||||
// Tags []string `json:"tags,omitempty" jsonschema:"filter by tags (all must match)"`
|
||||
// }
|
||||
type SearchContactsInput struct {
|
||||
Query string `json:"query,omitempty" jsonschema:"search text matching name, company, title, or notes"`
|
||||
Tags []string `json:"tags,omitempty" jsonschema:"filter by tags (all must match)"`
|
||||
}
|
||||
|
||||
// type SearchContactsOutput struct {
|
||||
// Contacts []ext.ProfessionalContact `json:"contacts"`
|
||||
// }
|
||||
type SearchContactsOutput struct {
|
||||
Contacts []ext.ProfessionalContact `json:"contacts"`
|
||||
}
|
||||
|
||||
// func (t *CRMTool) SearchContacts(ctx context.Context, _ *mcp.CallToolRequest, in SearchContactsInput) (*mcp.CallToolResult, SearchContactsOutput, error) {
|
||||
// contacts, err := t.store.SearchContacts(ctx, in.Query, in.Tags)
|
||||
// if err != nil {
|
||||
// return nil, SearchContactsOutput{}, err
|
||||
// }
|
||||
// if contacts == nil {
|
||||
// contacts = []ext.ProfessionalContact{}
|
||||
// }
|
||||
// return nil, SearchContactsOutput{Contacts: contacts}, nil
|
||||
// }
|
||||
func (t *CRMTool) SearchContacts(ctx context.Context, _ *mcp.CallToolRequest, in SearchContactsInput) (*mcp.CallToolResult, SearchContactsOutput, error) {
|
||||
contacts, err := t.store.SearchContacts(ctx, in.Query, in.Tags)
|
||||
if err != nil {
|
||||
return nil, SearchContactsOutput{}, err
|
||||
}
|
||||
if contacts == nil {
|
||||
contacts = []ext.ProfessionalContact{}
|
||||
}
|
||||
return nil, SearchContactsOutput{Contacts: contacts}, nil
|
||||
}
|
||||
|
||||
// // log_interaction
|
||||
// log_interaction
|
||||
|
||||
// type LogInteractionInput struct {
|
||||
// ContactID uuid.UUID `json:"contact_id" jsonschema:"id of the contact"`
|
||||
// InteractionType string `json:"interaction_type" jsonschema:"one of: meeting, email, call, coffee, event, linkedin, other"`
|
||||
// OccurredAt *time.Time `json:"occurred_at,omitempty" jsonschema:"when it happened (defaults to now)"`
|
||||
// Summary string `json:"summary" jsonschema:"summary of the interaction"`
|
||||
// FollowUpNeeded bool `json:"follow_up_needed,omitempty"`
|
||||
// FollowUpNotes string `json:"follow_up_notes,omitempty"`
|
||||
// }
|
||||
type LogInteractionInput struct {
|
||||
ContactID uuid.UUID `json:"contact_id" jsonschema:"id of the contact"`
|
||||
InteractionType string `json:"interaction_type" jsonschema:"one of: meeting, email, call, coffee, event, linkedin, other"`
|
||||
OccurredAt *time.Time `json:"occurred_at,omitempty" jsonschema:"when it happened (defaults to now)"`
|
||||
Summary string `json:"summary" jsonschema:"summary of the interaction"`
|
||||
FollowUpNeeded bool `json:"follow_up_needed,omitempty"`
|
||||
FollowUpNotes string `json:"follow_up_notes,omitempty"`
|
||||
}
|
||||
|
||||
// type LogInteractionOutput struct {
|
||||
// Interaction ext.ContactInteraction `json:"interaction"`
|
||||
// }
|
||||
type LogInteractionOutput struct {
|
||||
Interaction ext.ContactInteraction `json:"interaction"`
|
||||
}
|
||||
|
||||
// func (t *CRMTool) LogInteraction(ctx context.Context, _ *mcp.CallToolRequest, in LogInteractionInput) (*mcp.CallToolResult, LogInteractionOutput, error) {
|
||||
// if strings.TrimSpace(in.Summary) == "" {
|
||||
// return nil, LogInteractionOutput{}, errRequiredField("summary")
|
||||
// }
|
||||
// occurredAt := time.Now()
|
||||
// if in.OccurredAt != nil {
|
||||
// occurredAt = *in.OccurredAt
|
||||
// }
|
||||
// interaction, err := t.store.LogInteraction(ctx, ext.ContactInteraction{
|
||||
// ContactID: in.ContactID,
|
||||
// InteractionType: in.InteractionType,
|
||||
// OccurredAt: occurredAt,
|
||||
// Summary: strings.TrimSpace(in.Summary),
|
||||
// FollowUpNeeded: in.FollowUpNeeded,
|
||||
// FollowUpNotes: strings.TrimSpace(in.FollowUpNotes),
|
||||
// })
|
||||
// if err != nil {
|
||||
// return nil, LogInteractionOutput{}, err
|
||||
// }
|
||||
// return nil, LogInteractionOutput{Interaction: interaction}, nil
|
||||
// }
|
||||
func (t *CRMTool) LogInteraction(ctx context.Context, _ *mcp.CallToolRequest, in LogInteractionInput) (*mcp.CallToolResult, LogInteractionOutput, error) {
|
||||
if strings.TrimSpace(in.Summary) == "" {
|
||||
return nil, LogInteractionOutput{}, errRequiredField("summary")
|
||||
}
|
||||
occurredAt := time.Now()
|
||||
if in.OccurredAt != nil {
|
||||
occurredAt = *in.OccurredAt
|
||||
}
|
||||
interaction, err := t.store.LogInteraction(ctx, ext.ContactInteraction{
|
||||
ContactID: in.ContactID,
|
||||
InteractionType: in.InteractionType,
|
||||
OccurredAt: occurredAt,
|
||||
Summary: strings.TrimSpace(in.Summary),
|
||||
FollowUpNeeded: in.FollowUpNeeded,
|
||||
FollowUpNotes: strings.TrimSpace(in.FollowUpNotes),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, LogInteractionOutput{}, err
|
||||
}
|
||||
return nil, LogInteractionOutput{Interaction: interaction}, nil
|
||||
}
|
||||
|
||||
// // get_contact_history
|
||||
// get_contact_history
|
||||
|
||||
// type GetContactHistoryInput struct {
|
||||
// ContactID uuid.UUID `json:"contact_id" jsonschema:"id of the contact"`
|
||||
// }
|
||||
type GetContactHistoryInput struct {
|
||||
ContactID uuid.UUID `json:"contact_id" jsonschema:"id of the contact"`
|
||||
}
|
||||
|
||||
// type GetContactHistoryOutput struct {
|
||||
// History ext.ContactHistory `json:"history"`
|
||||
// }
|
||||
type GetContactHistoryOutput struct {
|
||||
History ext.ContactHistory `json:"history"`
|
||||
}
|
||||
|
||||
// func (t *CRMTool) GetHistory(ctx context.Context, _ *mcp.CallToolRequest, in GetContactHistoryInput) (*mcp.CallToolResult, GetContactHistoryOutput, error) {
|
||||
// history, err := t.store.GetContactHistory(ctx, in.ContactID)
|
||||
// if err != nil {
|
||||
// return nil, GetContactHistoryOutput{}, err
|
||||
// }
|
||||
// return nil, GetContactHistoryOutput{History: history}, nil
|
||||
// }
|
||||
func (t *CRMTool) GetHistory(ctx context.Context, _ *mcp.CallToolRequest, in GetContactHistoryInput) (*mcp.CallToolResult, GetContactHistoryOutput, error) {
|
||||
history, err := t.store.GetContactHistory(ctx, in.ContactID)
|
||||
if err != nil {
|
||||
return nil, GetContactHistoryOutput{}, err
|
||||
}
|
||||
return nil, GetContactHistoryOutput{History: history}, nil
|
||||
}
|
||||
|
||||
// // create_opportunity
|
||||
// create_opportunity
|
||||
|
||||
// type CreateOpportunityInput struct {
|
||||
// ContactID *uuid.UUID `json:"contact_id,omitempty"`
|
||||
// Title string `json:"title" jsonschema:"opportunity title"`
|
||||
// Description string `json:"description,omitempty"`
|
||||
// Stage string `json:"stage,omitempty" jsonschema:"one of: identified, in_conversation, proposal, negotiation, won, lost (default: identified)"`
|
||||
// Value *float64 `json:"value,omitempty" jsonschema:"monetary value"`
|
||||
// ExpectedCloseDate *time.Time `json:"expected_close_date,omitempty"`
|
||||
// Notes string `json:"notes,omitempty"`
|
||||
// }
|
||||
type CreateOpportunityInput struct {
|
||||
ContactID *uuid.UUID `json:"contact_id,omitempty"`
|
||||
Title string `json:"title" jsonschema:"opportunity title"`
|
||||
Description string `json:"description,omitempty"`
|
||||
Stage string `json:"stage,omitempty" jsonschema:"one of: identified, in_conversation, proposal, negotiation, won, lost (default: identified)"`
|
||||
Value *float64 `json:"value,omitempty" jsonschema:"monetary value"`
|
||||
ExpectedCloseDate *time.Time `json:"expected_close_date,omitempty"`
|
||||
Notes string `json:"notes,omitempty"`
|
||||
}
|
||||
|
||||
// type CreateOpportunityOutput struct {
|
||||
// Opportunity ext.Opportunity `json:"opportunity"`
|
||||
// }
|
||||
type CreateOpportunityOutput struct {
|
||||
Opportunity ext.Opportunity `json:"opportunity"`
|
||||
}
|
||||
|
||||
// func (t *CRMTool) CreateOpportunity(ctx context.Context, _ *mcp.CallToolRequest, in CreateOpportunityInput) (*mcp.CallToolResult, CreateOpportunityOutput, error) {
|
||||
// if strings.TrimSpace(in.Title) == "" {
|
||||
// return nil, CreateOpportunityOutput{}, errRequiredField("title")
|
||||
// }
|
||||
// stage := strings.TrimSpace(in.Stage)
|
||||
// if stage == "" {
|
||||
// stage = "identified"
|
||||
// }
|
||||
// opp, err := t.store.CreateOpportunity(ctx, ext.Opportunity{
|
||||
// ContactID: in.ContactID,
|
||||
// Title: strings.TrimSpace(in.Title),
|
||||
// Description: strings.TrimSpace(in.Description),
|
||||
// Stage: stage,
|
||||
// Value: in.Value,
|
||||
// ExpectedCloseDate: in.ExpectedCloseDate,
|
||||
// Notes: strings.TrimSpace(in.Notes),
|
||||
// })
|
||||
// if err != nil {
|
||||
// return nil, CreateOpportunityOutput{}, err
|
||||
// }
|
||||
// return nil, CreateOpportunityOutput{Opportunity: opp}, nil
|
||||
// }
|
||||
func (t *CRMTool) CreateOpportunity(ctx context.Context, _ *mcp.CallToolRequest, in CreateOpportunityInput) (*mcp.CallToolResult, CreateOpportunityOutput, error) {
|
||||
if strings.TrimSpace(in.Title) == "" {
|
||||
return nil, CreateOpportunityOutput{}, errRequiredField("title")
|
||||
}
|
||||
stage := strings.TrimSpace(in.Stage)
|
||||
if stage == "" {
|
||||
stage = "identified"
|
||||
}
|
||||
opp, err := t.store.CreateOpportunity(ctx, ext.Opportunity{
|
||||
ContactID: in.ContactID,
|
||||
Title: strings.TrimSpace(in.Title),
|
||||
Description: strings.TrimSpace(in.Description),
|
||||
Stage: stage,
|
||||
Value: in.Value,
|
||||
ExpectedCloseDate: in.ExpectedCloseDate,
|
||||
Notes: strings.TrimSpace(in.Notes),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, CreateOpportunityOutput{}, err
|
||||
}
|
||||
return nil, CreateOpportunityOutput{Opportunity: opp}, nil
|
||||
}
|
||||
|
||||
// // get_follow_ups_due
|
||||
// get_follow_ups_due
|
||||
|
||||
// type GetFollowUpsDueInput struct {
|
||||
// DaysAhead int `json:"days_ahead,omitempty" jsonschema:"look ahead window in days (default: 7)"`
|
||||
// }
|
||||
type GetFollowUpsDueInput struct {
|
||||
DaysAhead int `json:"days_ahead,omitempty" jsonschema:"look ahead window in days (default: 7)"`
|
||||
}
|
||||
|
||||
// type GetFollowUpsDueOutput struct {
|
||||
// Contacts []ext.ProfessionalContact `json:"contacts"`
|
||||
// }
|
||||
type GetFollowUpsDueOutput struct {
|
||||
Contacts []ext.ProfessionalContact `json:"contacts"`
|
||||
}
|
||||
|
||||
// func (t *CRMTool) GetFollowUpsDue(ctx context.Context, _ *mcp.CallToolRequest, in GetFollowUpsDueInput) (*mcp.CallToolResult, GetFollowUpsDueOutput, error) {
|
||||
// contacts, err := t.store.GetFollowUpsDue(ctx, in.DaysAhead)
|
||||
// if err != nil {
|
||||
// return nil, GetFollowUpsDueOutput{}, err
|
||||
// }
|
||||
// if contacts == nil {
|
||||
// contacts = []ext.ProfessionalContact{}
|
||||
// }
|
||||
// return nil, GetFollowUpsDueOutput{Contacts: contacts}, nil
|
||||
// }
|
||||
func (t *CRMTool) GetFollowUpsDue(ctx context.Context, _ *mcp.CallToolRequest, in GetFollowUpsDueInput) (*mcp.CallToolResult, GetFollowUpsDueOutput, error) {
|
||||
contacts, err := t.store.GetFollowUpsDue(ctx, in.DaysAhead)
|
||||
if err != nil {
|
||||
return nil, GetFollowUpsDueOutput{}, err
|
||||
}
|
||||
if contacts == nil {
|
||||
contacts = []ext.ProfessionalContact{}
|
||||
}
|
||||
return nil, GetFollowUpsDueOutput{Contacts: contacts}, nil
|
||||
}
|
||||
|
||||
// // link_thought_to_contact
|
||||
// link_thought_to_contact
|
||||
|
||||
// type LinkThoughtToContactInput struct {
|
||||
// ContactID uuid.UUID `json:"contact_id" jsonschema:"id of the contact"`
|
||||
// ThoughtID uuid.UUID `json:"thought_id" jsonschema:"id of the thought to link"`
|
||||
// }
|
||||
type LinkThoughtToContactInput struct {
|
||||
ContactID uuid.UUID `json:"contact_id" jsonschema:"id of the contact"`
|
||||
ThoughtID uuid.UUID `json:"thought_id" jsonschema:"id of the thought to link"`
|
||||
}
|
||||
|
||||
// type LinkThoughtToContactOutput struct {
|
||||
// Contact ext.ProfessionalContact `json:"contact"`
|
||||
// }
|
||||
type LinkThoughtToContactOutput struct {
|
||||
Contact ext.ProfessionalContact `json:"contact"`
|
||||
}
|
||||
|
||||
// func (t *CRMTool) LinkThought(ctx context.Context, _ *mcp.CallToolRequest, in LinkThoughtToContactInput) (*mcp.CallToolResult, LinkThoughtToContactOutput, error) {
|
||||
// thought, err := t.store.GetThought(ctx, in.ThoughtID)
|
||||
// if err != nil {
|
||||
// if errors.Is(err, pgx.ErrNoRows) {
|
||||
// return nil, LinkThoughtToContactOutput{}, errEntityNotFound("thought", "thought_id", in.ThoughtID.String())
|
||||
// }
|
||||
// return nil, LinkThoughtToContactOutput{}, err
|
||||
// }
|
||||
func (t *CRMTool) LinkThought(ctx context.Context, _ *mcp.CallToolRequest, in LinkThoughtToContactInput) (*mcp.CallToolResult, LinkThoughtToContactOutput, error) {
|
||||
thought, err := t.store.GetThought(ctx, in.ThoughtID)
|
||||
if err != nil {
|
||||
if errors.Is(err, pgx.ErrNoRows) {
|
||||
return nil, LinkThoughtToContactOutput{}, errEntityNotFound("thought", "thought_id", in.ThoughtID.String())
|
||||
}
|
||||
return nil, LinkThoughtToContactOutput{}, err
|
||||
}
|
||||
|
||||
// appendText := fmt.Sprintf("\n\n[Linked thought %s]: %s", thought.ID, thought.Content)
|
||||
// if err := t.store.AppendThoughtToContactNotes(ctx, in.ContactID, appendText); err != nil {
|
||||
// return nil, LinkThoughtToContactOutput{}, err
|
||||
// }
|
||||
appendText := fmt.Sprintf("\n\n[Linked thought %s]: %s", thought.ID, thought.Content)
|
||||
if err := t.store.AppendThoughtToContactNotes(ctx, in.ContactID, appendText); err != nil {
|
||||
return nil, LinkThoughtToContactOutput{}, err
|
||||
}
|
||||
|
||||
// contact, err := t.store.GetContact(ctx, in.ContactID)
|
||||
// if err != nil {
|
||||
// if errors.Is(err, pgx.ErrNoRows) {
|
||||
// return nil, LinkThoughtToContactOutput{}, errEntityNotFound("contact", "contact_id", in.ContactID.String())
|
||||
// }
|
||||
// return nil, LinkThoughtToContactOutput{}, err
|
||||
// }
|
||||
// return nil, LinkThoughtToContactOutput{Contact: contact}, nil
|
||||
// }
|
||||
contact, err := t.store.GetContact(ctx, in.ContactID)
|
||||
if err != nil {
|
||||
if errors.Is(err, pgx.ErrNoRows) {
|
||||
return nil, LinkThoughtToContactOutput{}, errEntityNotFound("contact", "contact_id", in.ContactID.String())
|
||||
}
|
||||
return nil, LinkThoughtToContactOutput{}, err
|
||||
}
|
||||
return nil, LinkThoughtToContactOutput{Contact: contact}, nil
|
||||
}
|
||||
|
||||
@@ -2,11 +2,11 @@ package tools
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"log/slog"
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
"github.com/google/uuid"
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
"golang.org/x/sync/semaphore"
|
||||
|
||||
@@ -89,19 +89,18 @@ func (t *RetryEnrichmentTool) Handle(ctx context.Context, req *mcp.CallToolReque
|
||||
return t.retryer.Handle(ctx, req, in)
|
||||
}
|
||||
|
||||
func (r *EnrichmentRetryer) QueueThought(id int64) {
|
||||
func (r *EnrichmentRetryer) QueueThought(id uuid.UUID) {
|
||||
go func() {
|
||||
started := time.Now()
|
||||
idStr := fmt.Sprint(id)
|
||||
r.logger.Info("background metadata started",
|
||||
slog.String("thought_id", idStr),
|
||||
slog.String("thought_id", id.String()),
|
||||
slog.String("provider", r.metadata.PrimaryProvider()),
|
||||
slog.String("model", r.metadata.PrimaryModel()),
|
||||
)
|
||||
updated, err := r.retryOne(r.backgroundCtx, id)
|
||||
if err != nil {
|
||||
r.logger.Warn("background metadata error",
|
||||
slog.String("thought_id", idStr),
|
||||
slog.String("thought_id", id.String()),
|
||||
slog.String("provider", r.metadata.PrimaryProvider()),
|
||||
slog.String("model", r.metadata.PrimaryModel()),
|
||||
slog.Duration("duration", time.Since(started)),
|
||||
@@ -110,7 +109,7 @@ func (r *EnrichmentRetryer) QueueThought(id int64) {
|
||||
return
|
||||
}
|
||||
r.logger.Info("background metadata complete",
|
||||
slog.String("thought_id", idStr),
|
||||
slog.String("thought_id", id.String()),
|
||||
slog.String("provider", r.metadata.PrimaryProvider()),
|
||||
slog.String("model", r.metadata.PrimaryModel()),
|
||||
slog.Bool("updated", updated),
|
||||
@@ -130,9 +129,9 @@ func (r *EnrichmentRetryer) Handle(ctx context.Context, req *mcp.CallToolRequest
|
||||
return nil, RetryEnrichmentOutput{}, err
|
||||
}
|
||||
|
||||
var projectID *int64
|
||||
var projectID *uuid.UUID
|
||||
if project != nil {
|
||||
projectID = &project.NumericID
|
||||
projectID = &project.ID
|
||||
}
|
||||
|
||||
thoughts, err := r.store.ListThoughtsPendingMetadataRetry(ctx, limit, projectID, in.IncludeArchived, in.OlderThanDays)
|
||||
@@ -169,7 +168,7 @@ func (r *EnrichmentRetryer) Handle(ctx context.Context, req *mcp.CallToolRequest
|
||||
updated, err := r.retryOne(ctx, thought.ID)
|
||||
if err != nil {
|
||||
mu.Lock()
|
||||
out.Failures = append(out.Failures, RetryEnrichmentFailure{ID: fmt.Sprint(thought.ID), Error: err.Error()})
|
||||
out.Failures = append(out.Failures, RetryEnrichmentFailure{ID: thought.ID.String(), Error: err.Error()})
|
||||
mu.Unlock()
|
||||
return
|
||||
}
|
||||
@@ -192,8 +191,8 @@ func (r *EnrichmentRetryer) Handle(ctx context.Context, req *mcp.CallToolRequest
|
||||
return nil, out, nil
|
||||
}
|
||||
|
||||
func (r *EnrichmentRetryer) retryOne(ctx context.Context, id int64) (bool, error) {
|
||||
thought, err := r.store.GetThoughtByID(ctx, id)
|
||||
func (r *EnrichmentRetryer) retryOne(ctx context.Context, id uuid.UUID) (bool, error) {
|
||||
thought, err := r.store.GetThought(ctx, id)
|
||||
if err != nil {
|
||||
return false, err
|
||||
}
|
||||
|
||||
+16
-15
@@ -5,7 +5,6 @@ import (
|
||||
"crypto/sha256"
|
||||
"encoding/base64"
|
||||
"encoding/hex"
|
||||
"fmt"
|
||||
"net/http"
|
||||
"os"
|
||||
"path/filepath"
|
||||
@@ -149,7 +148,7 @@ func (t *FilesTool) Upload(ctx context.Context, req *mcp.CallToolRequest, in Upl
|
||||
return nil, UploadFileOutput{}, err
|
||||
}
|
||||
|
||||
uri := fileURIPrefix + fmt.Sprint(out.File.ID)
|
||||
uri := fileURIPrefix + out.File.ID.String()
|
||||
return nil, UploadFileOutput{File: out.File, URI: uri}, nil
|
||||
}
|
||||
|
||||
@@ -248,7 +247,7 @@ func (t *FilesTool) Load(ctx context.Context, _ *mcp.CallToolRequest, in LoadFil
|
||||
return nil, LoadFileOutput{}, err
|
||||
}
|
||||
|
||||
uri := fileURIPrefix + fmt.Sprint(file.ID)
|
||||
uri := fileURIPrefix + file.ID.String()
|
||||
result := &mcp.CallToolResult{
|
||||
Content: []mcp.Content{
|
||||
&mcp.EmbeddedResource{
|
||||
@@ -295,7 +294,7 @@ func (t *FilesTool) List(ctx context.Context, req *mcp.CallToolRequest, in ListF
|
||||
return nil, ListFilesOutput{}, err
|
||||
}
|
||||
|
||||
var thoughtID *int64
|
||||
var thoughtID *uuid.UUID
|
||||
if rawThoughtID := strings.TrimSpace(in.ThoughtID); rawThoughtID != "" {
|
||||
parsedThoughtID, err := parseUUID(rawThoughtID)
|
||||
if err != nil {
|
||||
@@ -305,12 +304,12 @@ func (t *FilesTool) List(ctx context.Context, req *mcp.CallToolRequest, in ListF
|
||||
if err != nil {
|
||||
return nil, ListFilesOutput{}, err
|
||||
}
|
||||
thoughtID = &thought.ID
|
||||
if project != nil && thought.ProjectID != nil && *thought.ProjectID != project.NumericID {
|
||||
thoughtID = &parsedThoughtID
|
||||
if project != nil && thought.ProjectID != nil && *thought.ProjectID != project.ID {
|
||||
return nil, ListFilesOutput{}, errInvalidInput("project does not match the linked thought's project")
|
||||
}
|
||||
if project == nil && thought.ProjectID != nil {
|
||||
project = &thoughttypes.Project{NumericID: *thought.ProjectID}
|
||||
project = &thoughttypes.Project{ID: *thought.ProjectID}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -324,7 +323,7 @@ func (t *FilesTool) List(ctx context.Context, req *mcp.CallToolRequest, in ListF
|
||||
return nil, ListFilesOutput{}, err
|
||||
}
|
||||
if project != nil {
|
||||
_ = t.store.TouchProject(ctx, project.NumericID)
|
||||
_ = t.store.TouchProject(ctx, project.ID)
|
||||
}
|
||||
|
||||
return nil, ListFilesOutput{Files: files}, nil
|
||||
@@ -344,7 +343,7 @@ func (t *FilesTool) SaveDecoded(ctx context.Context, req *mcp.CallToolRequest, i
|
||||
return SaveFileOutput{}, err
|
||||
}
|
||||
|
||||
var thoughtNumericID *int64
|
||||
var thoughtID *uuid.UUID
|
||||
var projectID = projectIDPtr(project)
|
||||
if rawThoughtID := strings.TrimSpace(in.ThoughtID); rawThoughtID != "" {
|
||||
parsedThoughtID, err := parseUUID(rawThoughtID)
|
||||
@@ -355,9 +354,9 @@ func (t *FilesTool) SaveDecoded(ctx context.Context, req *mcp.CallToolRequest, i
|
||||
if err != nil {
|
||||
return SaveFileOutput{}, err
|
||||
}
|
||||
thoughtNumericID = &thought.ID
|
||||
thoughtID = &parsedThoughtID
|
||||
projectID = thought.ProjectID
|
||||
if project != nil && thought.ProjectID != nil && *thought.ProjectID != project.NumericID {
|
||||
if project != nil && thought.ProjectID != nil && *thought.ProjectID != project.ID {
|
||||
return SaveFileOutput{}, errInvalidInput("project does not match the linked thought's project")
|
||||
}
|
||||
}
|
||||
@@ -375,7 +374,9 @@ func (t *FilesTool) SaveDecoded(ctx context.Context, req *mcp.CallToolRequest, i
|
||||
SHA256: hex.EncodeToString(sum[:]),
|
||||
Content: in.Content,
|
||||
ProjectID: projectID,
|
||||
ThoughtID: thoughtNumericID,
|
||||
}
|
||||
if thoughtID != nil {
|
||||
file.ThoughtID = thoughtID
|
||||
}
|
||||
|
||||
created, err := t.store.InsertStoredFile(ctx, file)
|
||||
@@ -397,7 +398,7 @@ func (t *FilesTool) SaveDecoded(ctx context.Context, req *mcp.CallToolRequest, i
|
||||
|
||||
func thoughtAttachmentFromFile(file thoughttypes.StoredFile) thoughttypes.ThoughtAttachment {
|
||||
return thoughttypes.ThoughtAttachment{
|
||||
FileID: file.GUID,
|
||||
FileID: file.ID,
|
||||
Name: file.Name,
|
||||
MediaType: file.MediaType,
|
||||
Kind: file.Kind,
|
||||
@@ -497,11 +498,11 @@ func normalizeFileKind(explicit string, mediaType string) string {
|
||||
}
|
||||
}
|
||||
|
||||
func projectIDPtr(project *thoughttypes.Project) *int64 {
|
||||
func projectIDPtr(project *thoughttypes.Project) *uuid.UUID {
|
||||
if project == nil {
|
||||
return nil
|
||||
}
|
||||
return &project.NumericID
|
||||
return &project.ID
|
||||
}
|
||||
|
||||
func normalizeFileLimit(limit int) int {
|
||||
|
||||
@@ -3,7 +3,6 @@ package tools
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
"github.com/google/uuid"
|
||||
@@ -35,14 +34,6 @@ func parseUUID(id string) (uuid.UUID, error) {
|
||||
return parsed, nil
|
||||
}
|
||||
|
||||
func parseID(id string) (int64, error) {
|
||||
n, err := strconv.ParseInt(strings.TrimSpace(id), 10, 64)
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf("invalid numeric id %q", id)
|
||||
}
|
||||
return n, nil
|
||||
}
|
||||
|
||||
func sessionID(req *mcp.CallToolRequest) (string, error) {
|
||||
if req == nil || req.Session == nil || req.Session.ID() == "" {
|
||||
return "", newMCPError(
|
||||
|
||||
+125
-125
@@ -1,151 +1,151 @@
|
||||
package tools
|
||||
|
||||
// import (
|
||||
// "context"
|
||||
// "strings"
|
||||
import (
|
||||
"context"
|
||||
"strings"
|
||||
|
||||
// "github.com/google/uuid"
|
||||
// "github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
"github.com/google/uuid"
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
|
||||
// "git.warky.dev/wdevs/amcs/internal/store"
|
||||
// ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
// )
|
||||
"git.warky.dev/wdevs/amcs/internal/store"
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
// type HouseholdTool struct {
|
||||
// store *store.DB
|
||||
// }
|
||||
type HouseholdTool struct {
|
||||
store *store.DB
|
||||
}
|
||||
|
||||
// func NewHouseholdTool(db *store.DB) *HouseholdTool {
|
||||
// return &HouseholdTool{store: db}
|
||||
// }
|
||||
func NewHouseholdTool(db *store.DB) *HouseholdTool {
|
||||
return &HouseholdTool{store: db}
|
||||
}
|
||||
|
||||
// // add_household_item
|
||||
// add_household_item
|
||||
|
||||
// type AddHouseholdItemInput struct {
|
||||
// Name string `json:"name" jsonschema:"name of the item"`
|
||||
// Category string `json:"category,omitempty" jsonschema:"category (e.g. paint, appliance, measurement, document)"`
|
||||
// Location string `json:"location,omitempty" jsonschema:"where in the home this item is"`
|
||||
// Details map[string]any `json:"details,omitempty" jsonschema:"flexible metadata (model numbers, colors, specs, etc.)"`
|
||||
// Notes string `json:"notes,omitempty"`
|
||||
// }
|
||||
type AddHouseholdItemInput struct {
|
||||
Name string `json:"name" jsonschema:"name of the item"`
|
||||
Category string `json:"category,omitempty" jsonschema:"category (e.g. paint, appliance, measurement, document)"`
|
||||
Location string `json:"location,omitempty" jsonschema:"where in the home this item is"`
|
||||
Details map[string]any `json:"details,omitempty" jsonschema:"flexible metadata (model numbers, colors, specs, etc.)"`
|
||||
Notes string `json:"notes,omitempty"`
|
||||
}
|
||||
|
||||
// type AddHouseholdItemOutput struct {
|
||||
// Item ext.HouseholdItem `json:"item"`
|
||||
// }
|
||||
type AddHouseholdItemOutput struct {
|
||||
Item ext.HouseholdItem `json:"item"`
|
||||
}
|
||||
|
||||
// func (t *HouseholdTool) AddItem(ctx context.Context, _ *mcp.CallToolRequest, in AddHouseholdItemInput) (*mcp.CallToolResult, AddHouseholdItemOutput, error) {
|
||||
// if strings.TrimSpace(in.Name) == "" {
|
||||
// return nil, AddHouseholdItemOutput{}, errRequiredField("name")
|
||||
// }
|
||||
// if in.Details == nil {
|
||||
// in.Details = map[string]any{}
|
||||
// }
|
||||
// item, err := t.store.AddHouseholdItem(ctx, ext.HouseholdItem{
|
||||
// Name: strings.TrimSpace(in.Name),
|
||||
// Category: strings.TrimSpace(in.Category),
|
||||
// Location: strings.TrimSpace(in.Location),
|
||||
// Details: in.Details,
|
||||
// Notes: strings.TrimSpace(in.Notes),
|
||||
// })
|
||||
// if err != nil {
|
||||
// return nil, AddHouseholdItemOutput{}, err
|
||||
// }
|
||||
// return nil, AddHouseholdItemOutput{Item: item}, nil
|
||||
// }
|
||||
func (t *HouseholdTool) AddItem(ctx context.Context, _ *mcp.CallToolRequest, in AddHouseholdItemInput) (*mcp.CallToolResult, AddHouseholdItemOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, AddHouseholdItemOutput{}, errRequiredField("name")
|
||||
}
|
||||
if in.Details == nil {
|
||||
in.Details = map[string]any{}
|
||||
}
|
||||
item, err := t.store.AddHouseholdItem(ctx, ext.HouseholdItem{
|
||||
Name: strings.TrimSpace(in.Name),
|
||||
Category: strings.TrimSpace(in.Category),
|
||||
Location: strings.TrimSpace(in.Location),
|
||||
Details: in.Details,
|
||||
Notes: strings.TrimSpace(in.Notes),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, AddHouseholdItemOutput{}, err
|
||||
}
|
||||
return nil, AddHouseholdItemOutput{Item: item}, nil
|
||||
}
|
||||
|
||||
// // search_household_items
|
||||
// search_household_items
|
||||
|
||||
// type SearchHouseholdItemsInput struct {
|
||||
// Query string `json:"query,omitempty" jsonschema:"search text matching name or notes"`
|
||||
// Category string `json:"category,omitempty" jsonschema:"filter by category"`
|
||||
// Location string `json:"location,omitempty" jsonschema:"filter by location"`
|
||||
// }
|
||||
type SearchHouseholdItemsInput struct {
|
||||
Query string `json:"query,omitempty" jsonschema:"search text matching name or notes"`
|
||||
Category string `json:"category,omitempty" jsonschema:"filter by category"`
|
||||
Location string `json:"location,omitempty" jsonschema:"filter by location"`
|
||||
}
|
||||
|
||||
// type SearchHouseholdItemsOutput struct {
|
||||
// Items []ext.HouseholdItem `json:"items"`
|
||||
// }
|
||||
type SearchHouseholdItemsOutput struct {
|
||||
Items []ext.HouseholdItem `json:"items"`
|
||||
}
|
||||
|
||||
// func (t *HouseholdTool) SearchItems(ctx context.Context, _ *mcp.CallToolRequest, in SearchHouseholdItemsInput) (*mcp.CallToolResult, SearchHouseholdItemsOutput, error) {
|
||||
// items, err := t.store.SearchHouseholdItems(ctx, in.Query, in.Category, in.Location)
|
||||
// if err != nil {
|
||||
// return nil, SearchHouseholdItemsOutput{}, err
|
||||
// }
|
||||
// if items == nil {
|
||||
// items = []ext.HouseholdItem{}
|
||||
// }
|
||||
// return nil, SearchHouseholdItemsOutput{Items: items}, nil
|
||||
// }
|
||||
func (t *HouseholdTool) SearchItems(ctx context.Context, _ *mcp.CallToolRequest, in SearchHouseholdItemsInput) (*mcp.CallToolResult, SearchHouseholdItemsOutput, error) {
|
||||
items, err := t.store.SearchHouseholdItems(ctx, in.Query, in.Category, in.Location)
|
||||
if err != nil {
|
||||
return nil, SearchHouseholdItemsOutput{}, err
|
||||
}
|
||||
if items == nil {
|
||||
items = []ext.HouseholdItem{}
|
||||
}
|
||||
return nil, SearchHouseholdItemsOutput{Items: items}, nil
|
||||
}
|
||||
|
||||
// // get_household_item
|
||||
// get_household_item
|
||||
|
||||
// type GetHouseholdItemInput struct {
|
||||
// ID uuid.UUID `json:"id" jsonschema:"item id"`
|
||||
// }
|
||||
type GetHouseholdItemInput struct {
|
||||
ID uuid.UUID `json:"id" jsonschema:"item id"`
|
||||
}
|
||||
|
||||
// type GetHouseholdItemOutput struct {
|
||||
// Item ext.HouseholdItem `json:"item"`
|
||||
// }
|
||||
type GetHouseholdItemOutput struct {
|
||||
Item ext.HouseholdItem `json:"item"`
|
||||
}
|
||||
|
||||
// func (t *HouseholdTool) GetItem(ctx context.Context, _ *mcp.CallToolRequest, in GetHouseholdItemInput) (*mcp.CallToolResult, GetHouseholdItemOutput, error) {
|
||||
// item, err := t.store.GetHouseholdItem(ctx, in.ID)
|
||||
// if err != nil {
|
||||
// return nil, GetHouseholdItemOutput{}, err
|
||||
// }
|
||||
// return nil, GetHouseholdItemOutput{Item: item}, nil
|
||||
// }
|
||||
func (t *HouseholdTool) GetItem(ctx context.Context, _ *mcp.CallToolRequest, in GetHouseholdItemInput) (*mcp.CallToolResult, GetHouseholdItemOutput, error) {
|
||||
item, err := t.store.GetHouseholdItem(ctx, in.ID)
|
||||
if err != nil {
|
||||
return nil, GetHouseholdItemOutput{}, err
|
||||
}
|
||||
return nil, GetHouseholdItemOutput{Item: item}, nil
|
||||
}
|
||||
|
||||
// // add_vendor
|
||||
// add_vendor
|
||||
|
||||
// type AddVendorInput struct {
|
||||
// Name string `json:"name" jsonschema:"vendor name"`
|
||||
// ServiceType string `json:"service_type,omitempty" jsonschema:"type of service (e.g. plumber, electrician, landscaper)"`
|
||||
// Phone string `json:"phone,omitempty"`
|
||||
// Email string `json:"email,omitempty"`
|
||||
// Website string `json:"website,omitempty"`
|
||||
// Notes string `json:"notes,omitempty"`
|
||||
// Rating *int `json:"rating,omitempty" jsonschema:"1-5 rating"`
|
||||
// }
|
||||
type AddVendorInput struct {
|
||||
Name string `json:"name" jsonschema:"vendor name"`
|
||||
ServiceType string `json:"service_type,omitempty" jsonschema:"type of service (e.g. plumber, electrician, landscaper)"`
|
||||
Phone string `json:"phone,omitempty"`
|
||||
Email string `json:"email,omitempty"`
|
||||
Website string `json:"website,omitempty"`
|
||||
Notes string `json:"notes,omitempty"`
|
||||
Rating *int `json:"rating,omitempty" jsonschema:"1-5 rating"`
|
||||
}
|
||||
|
||||
// type AddVendorOutput struct {
|
||||
// Vendor ext.HouseholdVendor `json:"vendor"`
|
||||
// }
|
||||
type AddVendorOutput struct {
|
||||
Vendor ext.HouseholdVendor `json:"vendor"`
|
||||
}
|
||||
|
||||
// func (t *HouseholdTool) AddVendor(ctx context.Context, _ *mcp.CallToolRequest, in AddVendorInput) (*mcp.CallToolResult, AddVendorOutput, error) {
|
||||
// if strings.TrimSpace(in.Name) == "" {
|
||||
// return nil, AddVendorOutput{}, errRequiredField("name")
|
||||
// }
|
||||
// vendor, err := t.store.AddVendor(ctx, ext.HouseholdVendor{
|
||||
// Name: strings.TrimSpace(in.Name),
|
||||
// ServiceType: strings.TrimSpace(in.ServiceType),
|
||||
// Phone: strings.TrimSpace(in.Phone),
|
||||
// Email: strings.TrimSpace(in.Email),
|
||||
// Website: strings.TrimSpace(in.Website),
|
||||
// Notes: strings.TrimSpace(in.Notes),
|
||||
// Rating: in.Rating,
|
||||
// })
|
||||
// if err != nil {
|
||||
// return nil, AddVendorOutput{}, err
|
||||
// }
|
||||
// return nil, AddVendorOutput{Vendor: vendor}, nil
|
||||
// }
|
||||
func (t *HouseholdTool) AddVendor(ctx context.Context, _ *mcp.CallToolRequest, in AddVendorInput) (*mcp.CallToolResult, AddVendorOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, AddVendorOutput{}, errRequiredField("name")
|
||||
}
|
||||
vendor, err := t.store.AddVendor(ctx, ext.HouseholdVendor{
|
||||
Name: strings.TrimSpace(in.Name),
|
||||
ServiceType: strings.TrimSpace(in.ServiceType),
|
||||
Phone: strings.TrimSpace(in.Phone),
|
||||
Email: strings.TrimSpace(in.Email),
|
||||
Website: strings.TrimSpace(in.Website),
|
||||
Notes: strings.TrimSpace(in.Notes),
|
||||
Rating: in.Rating,
|
||||
})
|
||||
if err != nil {
|
||||
return nil, AddVendorOutput{}, err
|
||||
}
|
||||
return nil, AddVendorOutput{Vendor: vendor}, nil
|
||||
}
|
||||
|
||||
// // list_vendors
|
||||
// list_vendors
|
||||
|
||||
// type ListVendorsInput struct {
|
||||
// ServiceType string `json:"service_type,omitempty" jsonschema:"filter by service type"`
|
||||
// }
|
||||
type ListVendorsInput struct {
|
||||
ServiceType string `json:"service_type,omitempty" jsonschema:"filter by service type"`
|
||||
}
|
||||
|
||||
// type ListVendorsOutput struct {
|
||||
// Vendors []ext.HouseholdVendor `json:"vendors"`
|
||||
// }
|
||||
type ListVendorsOutput struct {
|
||||
Vendors []ext.HouseholdVendor `json:"vendors"`
|
||||
}
|
||||
|
||||
// func (t *HouseholdTool) ListVendors(ctx context.Context, _ *mcp.CallToolRequest, in ListVendorsInput) (*mcp.CallToolResult, ListVendorsOutput, error) {
|
||||
// vendors, err := t.store.ListVendors(ctx, in.ServiceType)
|
||||
// if err != nil {
|
||||
// return nil, ListVendorsOutput{}, err
|
||||
// }
|
||||
// if vendors == nil {
|
||||
// vendors = []ext.HouseholdVendor{}
|
||||
// }
|
||||
// return nil, ListVendorsOutput{Vendors: vendors}, nil
|
||||
// }
|
||||
func (t *HouseholdTool) ListVendors(ctx context.Context, _ *mcp.CallToolRequest, in ListVendorsInput) (*mcp.CallToolResult, ListVendorsOutput, error) {
|
||||
vendors, err := t.store.ListVendors(ctx, in.ServiceType)
|
||||
if err != nil {
|
||||
return nil, ListVendorsOutput{}, err
|
||||
}
|
||||
if vendors == nil {
|
||||
vendors = []ext.HouseholdVendor{}
|
||||
}
|
||||
return nil, ListVendorsOutput{Vendors: vendors}, nil
|
||||
}
|
||||
|
||||
@@ -1,191 +0,0 @@
|
||||
package tools
|
||||
|
||||
import (
|
||||
"context"
|
||||
"strings"
|
||||
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/config"
|
||||
"git.warky.dev/wdevs/amcs/internal/session"
|
||||
"git.warky.dev/wdevs/amcs/internal/store"
|
||||
thoughttypes "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
type LearningsTool struct {
|
||||
store *store.DB
|
||||
sessions *session.ActiveProjects
|
||||
cfg config.SearchConfig
|
||||
}
|
||||
|
||||
type AddLearningInput struct {
|
||||
Summary string `json:"summary" jsonschema:"short curated learning summary"`
|
||||
Details string `json:"details,omitempty" jsonschema:"optional detailed learning body"`
|
||||
Category string `json:"category,omitempty"`
|
||||
Area string `json:"area,omitempty"`
|
||||
Status string `json:"status,omitempty"`
|
||||
Priority string `json:"priority,omitempty"`
|
||||
Confidence string `json:"confidence,omitempty"`
|
||||
ActionRequired *bool `json:"action_required,omitempty"`
|
||||
SourceType string `json:"source_type,omitempty"`
|
||||
SourceRef string `json:"source_ref,omitempty"`
|
||||
Project string `json:"project,omitempty" jsonschema:"project name or id; falls back to active session project"`
|
||||
RelatedThoughtID *int64 `json:"related_thought_id,omitempty"`
|
||||
RelatedSkillID *int64 `json:"related_skill_id,omitempty"`
|
||||
ReviewedBy *string `json:"reviewed_by,omitempty"`
|
||||
DuplicateOfLearningID *int64 `json:"duplicate_of_learning_id,omitempty"`
|
||||
SupersedesLearningID *int64 `json:"supersedes_learning_id,omitempty"`
|
||||
Tags []string `json:"tags,omitempty"`
|
||||
}
|
||||
|
||||
type AddLearningOutput struct {
|
||||
Learning thoughttypes.Learning `json:"learning"`
|
||||
}
|
||||
|
||||
type GetLearningInput struct {
|
||||
ID int64 `json:"id" jsonschema:"learning id"`
|
||||
}
|
||||
|
||||
type GetLearningOutput struct {
|
||||
Learning thoughttypes.Learning `json:"learning"`
|
||||
}
|
||||
|
||||
type ListLearningsInput struct {
|
||||
Limit int `json:"limit,omitempty"`
|
||||
Project string `json:"project,omitempty" jsonschema:"project name or id; falls back to active session project"`
|
||||
Category string `json:"category,omitempty"`
|
||||
Area string `json:"area,omitempty"`
|
||||
Status string `json:"status,omitempty"`
|
||||
Priority string `json:"priority,omitempty"`
|
||||
Tag string `json:"tag,omitempty"`
|
||||
Query string `json:"query,omitempty"`
|
||||
}
|
||||
|
||||
type ListLearningsOutput struct {
|
||||
Learnings []thoughttypes.Learning `json:"learnings"`
|
||||
}
|
||||
|
||||
func NewLearningsTool(db *store.DB, sessions *session.ActiveProjects, cfg config.SearchConfig) *LearningsTool {
|
||||
return &LearningsTool{store: db, sessions: sessions, cfg: cfg}
|
||||
}
|
||||
|
||||
func (t *LearningsTool) Add(ctx context.Context, req *mcp.CallToolRequest, in AddLearningInput) (*mcp.CallToolResult, AddLearningOutput, error) {
|
||||
summary := strings.TrimSpace(in.Summary)
|
||||
if summary == "" {
|
||||
return nil, AddLearningOutput{}, errRequiredField("summary")
|
||||
}
|
||||
if err := t.ensureConfigured(); err != nil {
|
||||
return nil, AddLearningOutput{}, err
|
||||
}
|
||||
|
||||
project, err := resolveProject(ctx, t.store, t.sessions, req, in.Project, false)
|
||||
if err != nil {
|
||||
return nil, AddLearningOutput{}, err
|
||||
}
|
||||
|
||||
learning := thoughttypes.Learning{
|
||||
Summary: summary,
|
||||
Details: strings.TrimSpace(in.Details),
|
||||
Category: defaultString(strings.TrimSpace(in.Category), "insight"),
|
||||
Area: defaultString(strings.TrimSpace(in.Area), "other"),
|
||||
Status: thoughttypes.LearningStatus(defaultString(strings.TrimSpace(in.Status), string(thoughttypes.LearningStatusPending))),
|
||||
Priority: thoughttypes.LearningPriority(defaultString(strings.TrimSpace(in.Priority), string(thoughttypes.LearningPriorityMedium))),
|
||||
Confidence: thoughttypes.LearningEvidenceLevel(defaultString(strings.TrimSpace(in.Confidence), string(thoughttypes.LearningEvidenceHypothesis))),
|
||||
SourceType: strings.TrimSpace(in.SourceType),
|
||||
SourceRef: strings.TrimSpace(in.SourceRef),
|
||||
RelatedThoughtID: in.RelatedThoughtID,
|
||||
RelatedSkillID: in.RelatedSkillID,
|
||||
ReviewedBy: in.ReviewedBy,
|
||||
DuplicateOfLearningID: in.DuplicateOfLearningID,
|
||||
SupersedesLearningID: in.SupersedesLearningID,
|
||||
Tags: normalizeStringSlice(in.Tags),
|
||||
}
|
||||
if in.ActionRequired != nil {
|
||||
learning.ActionRequired = *in.ActionRequired
|
||||
}
|
||||
if project != nil {
|
||||
learning.ProjectID = &project.NumericID
|
||||
}
|
||||
|
||||
created, err := t.store.CreateLearning(ctx, learning)
|
||||
if err != nil {
|
||||
return nil, AddLearningOutput{}, err
|
||||
}
|
||||
return nil, AddLearningOutput{Learning: created}, nil
|
||||
}
|
||||
|
||||
func (t *LearningsTool) Get(ctx context.Context, _ *mcp.CallToolRequest, in GetLearningInput) (*mcp.CallToolResult, GetLearningOutput, error) {
|
||||
if err := t.ensureConfigured(); err != nil {
|
||||
return nil, GetLearningOutput{}, err
|
||||
}
|
||||
|
||||
learning, err := t.store.GetLearning(ctx, in.ID)
|
||||
if err != nil {
|
||||
return nil, GetLearningOutput{}, err
|
||||
}
|
||||
return nil, GetLearningOutput{Learning: learning}, nil
|
||||
}
|
||||
|
||||
func (t *LearningsTool) List(ctx context.Context, req *mcp.CallToolRequest, in ListLearningsInput) (*mcp.CallToolResult, ListLearningsOutput, error) {
|
||||
if err := t.ensureConfigured(); err != nil {
|
||||
return nil, ListLearningsOutput{}, err
|
||||
}
|
||||
|
||||
project, err := resolveProject(ctx, t.store, t.sessions, req, in.Project, false)
|
||||
if err != nil {
|
||||
return nil, ListLearningsOutput{}, err
|
||||
}
|
||||
|
||||
filter := thoughttypes.LearningFilter{
|
||||
Limit: normalizeLimit(in.Limit, t.cfg),
|
||||
Category: strings.TrimSpace(in.Category),
|
||||
Area: strings.TrimSpace(in.Area),
|
||||
Status: strings.TrimSpace(in.Status),
|
||||
Priority: strings.TrimSpace(in.Priority),
|
||||
Tag: strings.TrimSpace(in.Tag),
|
||||
Query: strings.TrimSpace(in.Query),
|
||||
}
|
||||
if project != nil {
|
||||
filter.ProjectID = &project.NumericID
|
||||
}
|
||||
|
||||
items, err := t.store.ListLearnings(ctx, filter)
|
||||
if err != nil {
|
||||
return nil, ListLearningsOutput{}, err
|
||||
}
|
||||
return nil, ListLearningsOutput{Learnings: items}, nil
|
||||
}
|
||||
|
||||
func (t *LearningsTool) ensureConfigured() error {
|
||||
if t == nil || t.store == nil {
|
||||
return errInvalidInput("learnings tool is not configured")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func defaultString(value string, fallback string) string {
|
||||
if value == "" {
|
||||
return fallback
|
||||
}
|
||||
return value
|
||||
}
|
||||
|
||||
func normalizeStringSlice(values []string) []string {
|
||||
if len(values) == 0 {
|
||||
return []string{}
|
||||
}
|
||||
out := make([]string, 0, len(values))
|
||||
seen := map[string]struct{}{}
|
||||
for _, value := range values {
|
||||
trimmed := strings.TrimSpace(value)
|
||||
if trimmed == "" {
|
||||
continue
|
||||
}
|
||||
if _, ok := seen[trimmed]; ok {
|
||||
continue
|
||||
}
|
||||
seen[trimmed] = struct{}{}
|
||||
out = append(out, trimmed)
|
||||
}
|
||||
return out
|
||||
}
|
||||
@@ -1,69 +0,0 @@
|
||||
package tools
|
||||
|
||||
import (
|
||||
"context"
|
||||
"testing"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/mcperrors"
|
||||
)
|
||||
|
||||
func TestLearningsAddRequiresSummary(t *testing.T) {
|
||||
tool := &LearningsTool{}
|
||||
|
||||
_, _, err := tool.Add(context.Background(), nil, AddLearningInput{})
|
||||
if err == nil {
|
||||
t.Fatal("Add() error = nil, want error")
|
||||
}
|
||||
|
||||
_, data := requireRPCError(t, err)
|
||||
if data.Field != "summary" {
|
||||
t.Fatalf("Add() error field = %q, want %q", data.Field, "summary")
|
||||
}
|
||||
}
|
||||
|
||||
func TestLearningsMethodsRequireConfiguredStore(t *testing.T) {
|
||||
tool := &LearningsTool{}
|
||||
|
||||
t.Run("add", func(t *testing.T) {
|
||||
_, _, err := tool.Add(context.Background(), nil, AddLearningInput{Summary: "Keep this"})
|
||||
if err == nil {
|
||||
t.Fatal("Add() error = nil, want error")
|
||||
}
|
||||
_, data := requireRPCError(t, err)
|
||||
if data.Type != mcperrors.TypeInvalidInput {
|
||||
t.Fatalf("Add() data.type = %q, want %q", data.Type, mcperrors.TypeInvalidInput)
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("get", func(t *testing.T) {
|
||||
_, _, err := tool.Get(context.Background(), nil, GetLearningInput{ID: 0})
|
||||
if err == nil {
|
||||
t.Fatal("Get() error = nil, want error")
|
||||
}
|
||||
_, data := requireRPCError(t, err)
|
||||
if data.Type != mcperrors.TypeInvalidInput {
|
||||
t.Fatalf("Get() data.type = %q, want %q", data.Type, mcperrors.TypeInvalidInput)
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("list", func(t *testing.T) {
|
||||
_, _, err := tool.List(context.Background(), nil, ListLearningsInput{})
|
||||
if err == nil {
|
||||
t.Fatal("List() error = nil, want error")
|
||||
}
|
||||
_, data := requireRPCError(t, err)
|
||||
if data.Type != mcperrors.TypeInvalidInput {
|
||||
t.Fatalf("List() data.type = %q, want %q", data.Type, mcperrors.TypeInvalidInput)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func TestNormalizeStringSliceTrimsDedupesAndDropsEmpties(t *testing.T) {
|
||||
got := normalizeStringSlice([]string{" alpha ", "beta", "", "beta", "alpha"})
|
||||
if len(got) != 2 {
|
||||
t.Fatalf("normalizeStringSlice() len = %d, want 2", len(got))
|
||||
}
|
||||
if got[0] != "alpha" || got[1] != "beta" {
|
||||
t.Fatalf("normalizeStringSlice() = %#v, want [alpha beta]", got)
|
||||
}
|
||||
}
|
||||
@@ -2,7 +2,6 @@ package tools
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"strings"
|
||||
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
@@ -98,9 +97,9 @@ func (t *LinksTool) Related(ctx context.Context, _ *mcp.CallToolRequest, in Rela
|
||||
}
|
||||
|
||||
related := make([]RelatedThought, 0, len(linked)+t.search.DefaultLimit)
|
||||
seen := map[string]struct{}{fmt.Sprint(thought.ID): {}}
|
||||
seen := map[string]struct{}{thought.ID.String(): {}}
|
||||
for _, item := range linked {
|
||||
key := fmt.Sprint(item.Thought.ID)
|
||||
key := item.Thought.ID.String()
|
||||
seen[key] = struct{}{}
|
||||
related = append(related, RelatedThought{
|
||||
ID: key,
|
||||
@@ -118,12 +117,12 @@ func (t *LinksTool) Related(ctx context.Context, _ *mcp.CallToolRequest, in Rela
|
||||
}
|
||||
|
||||
if includeSemantic {
|
||||
semantic, err := semanticSearch(ctx, t.store, t.embeddings, t.search, thought.Content, t.search.DefaultLimit, t.search.DefaultThreshold, thought.ProjectID, &thought.GUID)
|
||||
semantic, err := semanticSearch(ctx, t.store, t.embeddings, t.search, thought.Content, t.search.DefaultLimit, t.search.DefaultThreshold, thought.ProjectID, &thought.ID)
|
||||
if err != nil {
|
||||
return nil, RelatedOutput{}, err
|
||||
}
|
||||
for _, item := range semantic {
|
||||
key := fmt.Sprint(item.ID)
|
||||
key := item.ID.String()
|
||||
if _, ok := seen[key]; ok {
|
||||
continue
|
||||
}
|
||||
|
||||
@@ -4,6 +4,7 @@ import (
|
||||
"context"
|
||||
"strings"
|
||||
|
||||
"github.com/google/uuid"
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
|
||||
"git.warky.dev/wdevs/amcs/internal/config"
|
||||
@@ -42,9 +43,9 @@ func (t *ListTool) Handle(ctx context.Context, req *mcp.CallToolRequest, in List
|
||||
return nil, ListOutput{}, err
|
||||
}
|
||||
|
||||
var projectID *int64
|
||||
var projectID *uuid.UUID
|
||||
if project != nil {
|
||||
projectID = &project.NumericID
|
||||
projectID = &project.ID
|
||||
}
|
||||
|
||||
thoughts, err := t.store.ListThoughts(ctx, thoughttypes.ListFilter{
|
||||
@@ -60,7 +61,7 @@ func (t *ListTool) Handle(ctx context.Context, req *mcp.CallToolRequest, in List
|
||||
return nil, ListOutput{}, err
|
||||
}
|
||||
if project != nil {
|
||||
_ = t.store.TouchProject(ctx, project.NumericID)
|
||||
_ = t.store.TouchProject(ctx, project.ID)
|
||||
}
|
||||
|
||||
return nil, ListOutput{Thoughts: thoughts}, nil
|
||||
|
||||
+115
-115
@@ -1,137 +1,137 @@
|
||||
package tools
|
||||
|
||||
// import (
|
||||
// "context"
|
||||
// "strings"
|
||||
// "time"
|
||||
import (
|
||||
"context"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
// "github.com/google/uuid"
|
||||
// "github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
"github.com/google/uuid"
|
||||
"github.com/modelcontextprotocol/go-sdk/mcp"
|
||||
|
||||
// "git.warky.dev/wdevs/amcs/internal/store"
|
||||
// ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
// )
|
||||
"git.warky.dev/wdevs/amcs/internal/store"
|
||||
ext "git.warky.dev/wdevs/amcs/internal/types"
|
||||
)
|
||||
|
||||
// type MaintenanceTool struct {
|
||||
// store *store.DB
|
||||
// }
|
||||
type MaintenanceTool struct {
|
||||
store *store.DB
|
||||
}
|
||||
|
||||
// func NewMaintenanceTool(db *store.DB) *MaintenanceTool {
|
||||
// return &MaintenanceTool{store: db}
|
||||
// }
|
||||
func NewMaintenanceTool(db *store.DB) *MaintenanceTool {
|
||||
return &MaintenanceTool{store: db}
|
||||
}
|
||||
|
||||
// // add_maintenance_task
|
||||
// add_maintenance_task
|
||||
|
||||
// type AddMaintenanceTaskInput struct {
|
||||
// Name string `json:"name" jsonschema:"task name"`
|
||||
// Category string `json:"category,omitempty" jsonschema:"e.g. hvac, plumbing, exterior, appliance, landscaping"`
|
||||
// FrequencyDays *int `json:"frequency_days,omitempty" jsonschema:"recurrence interval in days; omit for one-time tasks"`
|
||||
// NextDue *time.Time `json:"next_due,omitempty" jsonschema:"when the task is next due"`
|
||||
// Priority string `json:"priority,omitempty" jsonschema:"low, medium, high, or urgent (default: medium)"`
|
||||
// Notes string `json:"notes,omitempty"`
|
||||
// }
|
||||
type AddMaintenanceTaskInput struct {
|
||||
Name string `json:"name" jsonschema:"task name"`
|
||||
Category string `json:"category,omitempty" jsonschema:"e.g. hvac, plumbing, exterior, appliance, landscaping"`
|
||||
FrequencyDays *int `json:"frequency_days,omitempty" jsonschema:"recurrence interval in days; omit for one-time tasks"`
|
||||
NextDue *time.Time `json:"next_due,omitempty" jsonschema:"when the task is next due"`
|
||||
Priority string `json:"priority,omitempty" jsonschema:"low, medium, high, or urgent (default: medium)"`
|
||||
Notes string `json:"notes,omitempty"`
|
||||
}
|
||||
|
||||
// type AddMaintenanceTaskOutput struct {
|
||||
// Task ext.MaintenanceTask `json:"task"`
|
||||
// }
|
||||
type AddMaintenanceTaskOutput struct {
|
||||
Task ext.MaintenanceTask `json:"task"`
|
||||
}
|
||||
|
||||
// func (t *MaintenanceTool) AddTask(ctx context.Context, _ *mcp.CallToolRequest, in AddMaintenanceTaskInput) (*mcp.CallToolResult, AddMaintenanceTaskOutput, error) {
|
||||
// if strings.TrimSpace(in.Name) == "" {
|
||||
// return nil, AddMaintenanceTaskOutput{}, errRequiredField("name")
|
||||
// }
|
||||
// priority := strings.TrimSpace(in.Priority)
|
||||
// if priority == "" {
|
||||
// priority = "medium"
|
||||
// }
|
||||
// task, err := t.store.AddMaintenanceTask(ctx, ext.MaintenanceTask{
|
||||
// Name: strings.TrimSpace(in.Name),
|
||||
// Category: strings.TrimSpace(in.Category),
|
||||
// FrequencyDays: in.FrequencyDays,
|
||||
// NextDue: in.NextDue,
|
||||
// Priority: priority,
|
||||
// Notes: strings.TrimSpace(in.Notes),
|
||||
// })
|
||||
// if err != nil {
|
||||
// return nil, AddMaintenanceTaskOutput{}, err
|
||||
// }
|
||||
// return nil, AddMaintenanceTaskOutput{Task: task}, nil
|
||||
// }
|
||||
func (t *MaintenanceTool) AddTask(ctx context.Context, _ *mcp.CallToolRequest, in AddMaintenanceTaskInput) (*mcp.CallToolResult, AddMaintenanceTaskOutput, error) {
|
||||
if strings.TrimSpace(in.Name) == "" {
|
||||
return nil, AddMaintenanceTaskOutput{}, errRequiredField("name")
|
||||
}
|
||||
priority := strings.TrimSpace(in.Priority)
|
||||
if priority == "" {
|
||||
priority = "medium"
|
||||
}
|
||||
task, err := t.store.AddMaintenanceTask(ctx, ext.MaintenanceTask{
|
||||
Name: strings.TrimSpace(in.Name),
|
||||
Category: strings.TrimSpace(in.Category),
|
||||
FrequencyDays: in.FrequencyDays,
|
||||
NextDue: in.NextDue,
|
||||
Priority: priority,
|
||||
Notes: strings.TrimSpace(in.Notes),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, AddMaintenanceTaskOutput{}, err
|
||||
}
|
||||
return nil, AddMaintenanceTaskOutput{Task: task}, nil
|
||||
}
|
||||
|
||||
// // log_maintenance
|
||||
// log_maintenance
|
||||
|
||||
// type LogMaintenanceInput struct {
|
||||
// TaskID uuid.UUID `json:"task_id" jsonschema:"id of the maintenance task"`
|
||||
// CompletedAt *time.Time `json:"completed_at,omitempty" jsonschema:"when the work was done (defaults to now)"`
|
||||
// PerformedBy string `json:"performed_by,omitempty" jsonschema:"who did the work (self, vendor name, etc.)"`
|
||||
// Cost *float64 `json:"cost,omitempty" jsonschema:"cost of the work"`
|
||||
// Notes string `json:"notes,omitempty"`
|
||||
// NextAction string `json:"next_action,omitempty" jsonschema:"recommended follow-up"`
|
||||
// }
|
||||
type LogMaintenanceInput struct {
|
||||
TaskID uuid.UUID `json:"task_id" jsonschema:"id of the maintenance task"`
|
||||
CompletedAt *time.Time `json:"completed_at,omitempty" jsonschema:"when the work was done (defaults to now)"`
|
||||
PerformedBy string `json:"performed_by,omitempty" jsonschema:"who did the work (self, vendor name, etc.)"`
|
||||
Cost *float64 `json:"cost,omitempty" jsonschema:"cost of the work"`
|
||||
Notes string `json:"notes,omitempty"`
|
||||
NextAction string `json:"next_action,omitempty" jsonschema:"recommended follow-up"`
|
||||
}
|
||||
|
||||
// type LogMaintenanceOutput struct {
|
||||
// Log ext.MaintenanceLog `json:"log"`
|
||||
// }
|
||||
type LogMaintenanceOutput struct {
|
||||
Log ext.MaintenanceLog `json:"log"`
|
||||
}
|
||||
|
||||
// func (t *MaintenanceTool) LogWork(ctx context.Context, _ *mcp.CallToolRequest, in LogMaintenanceInput) (*mcp.CallToolResult, LogMaintenanceOutput, error) {
|
||||
// completedAt := time.Now()
|
||||
// if in.CompletedAt != nil {
|
||||
// completedAt = *in.CompletedAt
|
||||
// }
|
||||
// log, err := t.store.LogMaintenance(ctx, ext.MaintenanceLog{
|
||||
// TaskID: in.TaskID,
|
||||
// CompletedAt: completedAt,
|
||||
// PerformedBy: strings.TrimSpace(in.PerformedBy),
|
||||
// Cost: in.Cost,
|
||||
// Notes: strings.TrimSpace(in.Notes),
|
||||
// NextAction: strings.TrimSpace(in.NextAction),
|
||||
// })
|
||||
// if err != nil {
|
||||
// return nil, LogMaintenanceOutput{}, err
|
||||
// }
|
||||
// return nil, LogMaintenanceOutput{Log: log}, nil
|
||||
// }
|
||||
func (t *MaintenanceTool) LogWork(ctx context.Context, _ *mcp.CallToolRequest, in LogMaintenanceInput) (*mcp.CallToolResult, LogMaintenanceOutput, error) {
|
||||
completedAt := time.Now()
|
||||
if in.CompletedAt != nil {
|
||||
completedAt = *in.CompletedAt
|
||||
}
|
||||
log, err := t.store.LogMaintenance(ctx, ext.MaintenanceLog{
|
||||
TaskID: in.TaskID,
|
||||
CompletedAt: completedAt,
|
||||
PerformedBy: strings.TrimSpace(in.PerformedBy),
|
||||
Cost: in.Cost,
|
||||
Notes: strings.TrimSpace(in.Notes),
|
||||
NextAction: strings.TrimSpace(in.NextAction),
|
||||
})
|
||||
if err != nil {
|
||||
return nil, LogMaintenanceOutput{}, err
|
||||
}
|
||||
return nil, LogMaintenanceOutput{Log: log}, nil
|
||||
}
|
||||
|
||||
// // get_upcoming_maintenance
|
||||
// get_upcoming_maintenance
|
||||
|
||||
// type GetUpcomingMaintenanceInput struct {
|
||||
// DaysAhead int `json:"days_ahead,omitempty" jsonschema:"how many days to look ahead (default: 30)"`
|
||||
// }
|
||||
type GetUpcomingMaintenanceInput struct {
|
||||
DaysAhead int `json:"days_ahead,omitempty" jsonschema:"how many days to look ahead (default: 30)"`
|
||||
}
|
||||
|
||||
// type GetUpcomingMaintenanceOutput struct {
|
||||
// Tasks []ext.MaintenanceTask `json:"tasks"`
|
||||
// }
|
||||
type GetUpcomingMaintenanceOutput struct {
|
||||
Tasks []ext.MaintenanceTask `json:"tasks"`
|
||||
}
|
||||
|
||||
// func (t *MaintenanceTool) GetUpcoming(ctx context.Context, _ *mcp.CallToolRequest, in GetUpcomingMaintenanceInput) (*mcp.CallToolResult, GetUpcomingMaintenanceOutput, error) {
|
||||
// tasks, err := t.store.GetUpcomingMaintenance(ctx, in.DaysAhead)
|
||||
// if err != nil {
|
||||
// return nil, GetUpcomingMaintenanceOutput{}, err
|
||||
// }
|
||||
// if tasks == nil {
|
||||
// tasks = []ext.MaintenanceTask{}
|
||||
// }
|
||||
// return nil, GetUpcomingMaintenanceOutput{Tasks: tasks}, nil
|
||||
// }
|
||||
func (t *MaintenanceTool) GetUpcoming(ctx context.Context, _ *mcp.CallToolRequest, in GetUpcomingMaintenanceInput) (*mcp.CallToolResult, GetUpcomingMaintenanceOutput, error) {
|
||||
tasks, err := t.store.GetUpcomingMaintenance(ctx, in.DaysAhead)
|
||||
if err != nil {
|
||||
return nil, GetUpcomingMaintenanceOutput{}, err
|
||||
}
|
||||
if tasks == nil {
|
||||
tasks = []ext.MaintenanceTask{}
|
||||
}
|
||||
return nil, GetUpcomingMaintenanceOutput{Tasks: tasks}, nil
|
||||
}
|
||||
|
||||
// // search_maintenance_history
|
||||
// search_maintenance_history
|
||||
|
||||
// type SearchMaintenanceHistoryInput struct {
|
||||
// Query string `json:"query,omitempty" jsonschema:"search text matching task name or notes"`
|
||||
// Category string `json:"category,omitempty" jsonschema:"filter by task category"`
|
||||
// Start *time.Time `json:"start,omitempty" jsonschema:"filter logs completed on or after this date"`
|
||||
// End *time.Time `json:"end,omitempty" jsonschema:"filter logs completed on or before this date"`
|
||||
// }
|
||||
type SearchMaintenanceHistoryInput struct {
|
||||
Query string `json:"query,omitempty" jsonschema:"search text matching task name or notes"`
|
||||
Category string `json:"category,omitempty" jsonschema:"filter by task category"`
|
||||
Start *time.Time `json:"start,omitempty" jsonschema:"filter logs completed on or after this date"`
|
||||
End *time.Time `json:"end,omitempty" jsonschema:"filter logs completed on or before this date"`
|
||||
}
|
||||
|
||||
// type SearchMaintenanceHistoryOutput struct {
|
||||
// Logs []ext.MaintenanceLogWithTask `json:"logs"`
|
||||
// }
|
||||
type SearchMaintenanceHistoryOutput struct {
|
||||
Logs []ext.MaintenanceLogWithTask `json:"logs"`
|
||||
}
|
||||
|
||||
// func (t *MaintenanceTool) SearchHistory(ctx context.Context, _ *mcp.CallToolRequest, in SearchMaintenanceHistoryInput) (*mcp.CallToolResult, SearchMaintenanceHistoryOutput, error) {
|
||||
// logs, err := t.store.SearchMaintenanceHistory(ctx, in.Query, in.Category, in.Start, in.End)
|
||||
// if err != nil {
|
||||
// return nil, SearchMaintenanceHistoryOutput{}, err
|
||||
// }
|
||||
// if logs == nil {
|
||||
// logs = []ext.MaintenanceLogWithTask{}
|
||||
// }
|
||||
// return nil, SearchMaintenanceHistoryOutput{Logs: logs}, nil
|
||||
// }
|
||||
func (t *MaintenanceTool) SearchHistory(ctx context.Context, _ *mcp.CallToolRequest, in SearchMaintenanceHistoryInput) (*mcp.CallToolResult, SearchMaintenanceHistoryOutput, error) {
|
||||
logs, err := t.store.SearchMaintenanceHistory(ctx, in.Query, in.Category, in.Start, in.End)
|
||||
if err != nil {
|
||||
return nil, SearchMaintenanceHistoryOutput{}, err
|
||||
}
|
||||
if logs == nil {
|
||||
logs = []ext.MaintenanceLogWithTask{}
|
||||
}
|
||||
return nil, SearchMaintenanceHistoryOutput{Logs: logs}, nil
|
||||
}
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user