structured.sh
Self-hosted MCP server for persistent agent memory. Define typed schemas, write records, and query with SQL via DuckDB against Parquet files on disk.
■ structured
schema-native memory for AI agents
structured.sh · Quick Start · Connect to Claude · MCP Tools · API Reference · Deploy
Define schemas. Write structured data. Query with SQL. All as Parquet files you own.
docker compose up
What is this?
Structured gives AI agents (and humans) persistent, queryable memory:
- Define a memory — name + typed schema (like a table)
- Write records — buffered and auto-flushed to Parquet files
- Query with SQL — DuckDB runs directly against the Parquet files
- Own your data — everything is local files on disk, no vendor lock-in
Works with Claude Desktop, Cursor, Windsurf, Cline, or any MCP-compatible client.
Architecture
┌──────────────────────────────────────────────────────┐
│ docker compose up │
│ │
│ ┌──────────┐ ┌──────────────┐ ┌────────────┐ │
│ │Dashboard │ │ REST API │ │ MCP Server │ │
│ │ :3000 │───▶│ :3001 │◀───│ stdio │ │
│ │ │ │ │ │ │ │
│ │Vite+React│ │ Hono + WASM │ │ 9 tools │ │
│ └──────────┘ │ │ └────────────┘ │
│ │ SQLite (sql.js) │
│ │ DuckDB (wasm) │
│ │ Parquet (tiny-parquet) │
│ └──────┬───────┘ │
│ │ │
│ ./data/ │
│ ├── structured.db ← metadata │
│ └── parquet/ ← your data │
│ ├── user_prefs/ │
│ │ └── 2026/04/09/...parquet │
│ └── campaign_data/ │
│ └── 2026/04/09/...parquet │
└──────────────────────────────────────────────────────┘
Zero native dependencies. Everything runs in WASM — no C++ builds, no platform issues.
Quick Start
1. Clone and configure
git clone https://github.com/structured-sh/structured.git
cd structured
Edit docker-compose.yml and set your credentials:
environment:
- API_KEY=your-secret-api-key # Used by MCP clients & scripts
- DASHBOARD_PASSWORD=your-password # Protects the dashboard UI
Two separate credentials:
API_KEY— machine auth for MCP clients, scripts, and analytics ingestionDASHBOARD_PASSWORD— human auth for the web dashboard. Leave unset to disable login (local-only use).
2. Start the stack
docker compose up -d
| Service | URL |
|---|---|
| Dashboard | http://localhost:3000 |
| API | http://localhost:3001 |
| MCP | stdio (auto-connected) |
3. Create a memory
curl -X POST http://localhost:3001/memories \
-H "Authorization: Bearer your-secret-api-key" \
-H "Content-Type: application/json" \
-d '{
"name": "user_preferences",
"fields": [
{ "name": "key", "type": "string" },
{ "name": "value", "type": "string" },
{ "name": "priority", "type": "int32" }
],
"description": "User preference settings"
}'
4. Write data
curl -X POST http://localhost:3001/memories/user_preferences/write \
-H "Authorization: Bearer your-secret-api-key" \
-H "Content-Type: application/json" \
-d '{
"data": [
{ "key": "theme", "value": "dark", "priority": 1 },
{ "key": "language", "value": "en", "priority": 2 }
]
}'
5. Query with SQL
curl -X POST http://localhost:3001/query \
-H "Authorization: Bearer your-secret-api-key" \
-H "Content-Type: application/json" \
-d '{ "sql": "SELECT * FROM user_preferences ORDER BY priority" }'
Memory names work as table names — DuckDB resolves them to Parquet files automatically.
6. Access raw files
# Files on disk
ls ./data/parquet/user_preferences/
# DuckDB CLI
duckdb -c "SELECT * FROM read_parquet('./data/parquet/user_preferences/**/*.parquet')"
# Python
import duckdb
duckdb.sql("SELECT * FROM './data/parquet/user_preferences/**/*.parquet'").show()
Connect to Claude / Cursor
Local (Docker)
Add to ~/Library/Application Support/Claude/claude_desktop_config.json:
{
"mcpServers": {
"structured": {
"command": "docker",
"args": ["exec", "-i", "structured-mcp", "node", "index.js"]
}
}
}
For Cursor, add to Settings → Features → MCP Servers:
{
"structured": {
"command": "docker",
"args": ["exec", "-i", "structured-mcp", "node", "index.js"]
}
}
Cloud (structured.sh)
Coming soon — hosted version at structured.sh
{
"mcpServers": {
"structured": {
"command": "npx",
"args": ["-y", "@anthropic-ai/mcp-proxy", "https://mcp.structured.sh"]
}
}
}
MCP Tools
Once connected, just talk naturally. The AI picks the right tool.
| Tool | What to say |
|---|---|
create_memory | "Create a memory for tracking daily sales with fields: date, revenue, units" |
list_memories | "What memories do I have?" |
describe_memory | "Show me the schema for daily_sales" |
write_memory | "Save today's sales: date=2026-04-09, revenue=1250.50, units=42" |
query_memory | "What's the total revenue this month?" |
store_document | "Remember this config for later" |
get_document | "Get the document abc-123" |
flush_memory | "Flush all pending data to disk" |
delete_memory | "Delete the test_data memory" |
Ingesting Data from Your Apps
Use the templates in templates/ to send events from external systems:
| Template | Use case |
|---|---|
templates/ingest-app-analytics.js | Mobile/web app events (installs, actions, purchases) |
templates/ingest-webhook.js | Stripe, GitHub, Shopify webhooks |
templates/query-report.js | Generate SQL reports → terminal, Markdown, or Slack |
// Example: track an install from your iOS app
import { track } from './templates/ingest-app-analytics.js';
await track('install', 'user_abc', { platform: 'ios', app_version: '1.0.0' });
Then query across all your events:
SELECT event, COUNT(*) as n, COUNT(DISTINCT user_id) as users
FROM app_events
WHERE timestamp > now() - INTERVAL '30 days'
GROUP BY event ORDER BY n DESC
Dead Letter Queue
Records rejected by strict or evolve schema modes are not dropped — they're automatically written to _dlq_{memory_name} for inspection:
-- See what was rejected and why
SELECT _reason, _payload, _rejected_at
FROM _dlq_app_events
ORDER BY _rejected_at DESC
LIMIT 20
API Key Rotation
Rotate your API key at any time from the dashboard Connect page without restarting:
- Go to Connect → API Key section
- Click Rotate Key
- Copy the new key (shown once)
- Update your MCP config and any scripts
The old key is invalidated immediately.
API Reference
Auth (Dashboard)
| Method | Path | Description |
|---|---|---|
GET | /auth/status | Check if dashboard auth is enabled |
POST | /auth/login | Login with { password }, returns session token |
GET | /auth/me | Validate current session (X-Session-Token header) |
POST | /auth/logout | Logout (client discards token) |
Memories
| Method | Path | Description |
|---|---|---|
GET | /memories | List all memories |
POST | /memories | Create a memory |
GET | /memories/:name | Get memory details |
PUT | /memories/:name | Update memory |
DELETE | /memories/:name | Delete memory |
POST | /memories/:name/write | Write records |
POST | /memories/:name/flush | Flush to Parquet |
GET | /memories/:name/preview | Preview data |
GET | /memories/:name/files | List Parquet files |
Query
| Method | Path | Description |
|---|---|---|
POST | /query | Execute DuckDB SQL |
Store (Documents)
| Method | Path | Description |
|---|---|---|
POST | /documents | Store a JSON document |
GET | /documents/collections | List collections |
GET | /documents/:collection | List documents |
DELETE | /documents/:collection/:id | Delete document |
Settings
| Method | Path | Description |
|---|---|---|
GET | /settings/api-key | Get masked current API key |
POST | /settings/rotate-key | Generate new API key |
Files
| Method | Path | Description |
|---|---|---|
GET | /files/* | Download raw Parquet file |
MCP (SSE)
| Method | Path | Description |
|---|---|---|
GET | /sse | SSE stream for MCP clients |
POST | /messages | JSON-RPC messages |
Schema Modes
| Mode | Behavior |
|---|---|
flex | Accept all fields, no validation (default) |
evolve | Accept all, detect and log schema drift |
strict | Reject records with mismatched fields → DLQ |
Field Types
| Type | Parquet Type | Notes |
|---|---|---|
string | BYTE_ARRAY (UTF-8) | Default |
int32 | INT32 | |
int64 | INT64 | |
float32 | FLOAT | |
float64 | DOUBLE | |
boolean | BOOLEAN | |
timestamp | INT64 (millis) | Unix ms, UTC |
Stack
Only 7 npm packages. No native addons — everything runs in WASM or pure JS.
| Package | What it does |
|---|---|
hono + @hono/node-server | HTTP router — fast, Web-standard API |
@duckdb/duckdb-wasm | SQL query engine, runs fully in WASM |
sql.js | SQLite in WASM — stores schema metadata |
tiny-parquet | Pure-JS Parquet writer, zero native deps |
@modelcontextprotocol/sdk | MCP server/tool registration |
zod | Schema validation for API inputs |
Runtime: Node.js ≥ 20
Because everything runs in WASM, there are no C++ builds, no node-gyp, no platform-specific binaries. The Docker image works on any architecture Docker supports.
Development
# Local dev (no Docker)
cd api && npm install && node index.js
cd dashboard && npm install && npm run dev
# Full stack
docker compose up --build
License
Apache 2.0
Built with tiny-parquet · Powered by DuckDB · structured.sh
Servidores relacionados
MCP Memory-mesh
An MCP server that gives Claude Code persistent memory across sessions (open source, SQLite)
BrainFlow - Shared Team Memory - MCP
Your company's brain, connected to Claude, ChatGPT, Gemini, Cursor, and VS Code. Turn your team's email history into shared memory that any AI assistant can query with natural language.
Adaptive Recall
Adaptive MCP memory system for AI applications. Learns which retrieval strategies work for your data, scores results using cognitive science models, builds a knowledge graph automatically, and validates every parameter change against real query history before adopting it. Patent pending.
MemPalace Cloud
Hosted MCP memory across Claude Code, Cursor, ChatGPT and any MCP client. Community-hosted instance of the MIT-licensed MemPalace engine. EU-hosted, GDPR-compliant. Free tier with 200 memories.
Obsidian MCP Server
Self-hosted MCP server for Obsidian: semantic + full-text search, wikilink graph, note CRUD, OAuth, and a self-describing vault guide.
OpenMemBrain
OpenMemBrain is the intelligent membrane for AI coding memory. It autonomously reads and learns from your coding sessions — you never have to tell it what to save. It selectively absorbs project knowledge, blocks secrets, filters noise, resolves conflicts, and persists only what matters.
Agentmemory MCP
Paid remote MCP endpoint for persistent AI agent memory, site analytics, pricing, checkout status, and search-readiness operations.
Rekindle
A local MCP continuity engine that helps Claude Code pick up the thread across sessions.
Vilix
Persistent memory for AI assistants and coding agents across ChatGPT, Claude, Cursor, and other MCP-compatible tools.
Mnemory
A self-hosted, secure, feature-rich memory system for AI agents and assistants. Provides intelligent fact extraction and deduplication, with an artifact store for detailed content.