World constitution engine for norm-governed multi-agent games
Project description
normsync
World constitution engine for norm-governed multi-agent games.
Quick Start · How It Works · CLI Reference · GitHub Action · vs. Alternatives · Contributing
Why
Multi-agent simulations and games need rules. But rules encoded in agent logic become invisible, hard to audit, and impossible to update without redeploying every agent.
normsync solves this by providing a world constitution engine: a centralized, content-addressed registry of normative rules that any agent can query. When an agent takes an action, normsync checks it against the active constitution and emits violations. Rules can be added, modified, or repealed at runtime without touching agent code.
normsync check agent1 attack safe_zone # Fails if "attack in safe_zone" is prohibited
This is especially powerful in:
- AI safety research: enforce behavioral constraints in multi-agent simulations
- Game design: codify win conditions, prohibited actions, and faction rules
- LLM agent governance: define and monitor behavioral policies for AI agents
- Compliance testing: record which agent violated which rule and when
How It Works
flowchart LR
A[Define WorldNorm\nname · condition · prohibited] --> B[NormStore\nSQLite backend]
B --> C[NormMonitor\nloads active norms]
C --> D[AgentAction\nagent_id · action · location]
D --> E{Condition matches\nprohibited matches?}
E -->|Yes| F[NormViolation\nstored + emitted]
E -->|No| G[Action allowed]
F --> H[Report\nJSON · Markdown · Rich table]
Core primitives:
- WorldNorm — a rule with a
condition(when it applies) and aprohibited(what is forbidden). ID = SHA-256[:16] ofname|condition|prohibited. Two agents defining the same rule always get the same ID. - AgentAction — a timestamped action taken by an agent with
agent_id,action,location,target, andfaction. - NormViolation — emitted when an action matches both the condition and prohibited token of an active norm.
- NormRevision — records when a norm is created, modified, or repealed.
Matching is token-based and case-insensitive: condition tokens must appear in the action's fields, and the prohibited token must match the action verb.
Features
| Feature | Details |
|---|---|
| Content-addressed norms | SHA-256[:16] of name|condition|prohibited — same rule always same ID |
| Token-based matching | Case-insensitive, split on whitespace — no regex needed |
| Norm lifecycle | Add, repeal, and query active norms at runtime |
| SQLite persistence | Single file, no server required |
| In-memory mode | NormStore(":memory:") for testing and ephemeral sessions |
| REST API | /norm, /norms, /check, /violations, /health endpoints |
| MCP server | Model Context Protocol tools for Claude and other agents |
| CLI | normsync add, check, violations, revisions, status |
| JSON output | Machine-readable reports for downstream automation |
| Markdown output | Ready-to-paste GitHub PR comments |
| 118 tests | Comprehensive test suite covering all layers |
Quick Start
pip install normsync
from normsync import NormMonitor, WorldNorm, AgentAction, print_violations
# Define world norms
monitor = NormMonitor()
monitor.add_norm(WorldNorm(
name="no-attack-in-safe-zone",
description="Attacking is prohibited in safe zones",
condition="safe_zone",
prohibited="attack",
))
# Check agent actions
action = AgentAction("hero", "attack", "safe_zone")
violations = monitor.check(action)
print_violations(violations)
# → Norm Violations table: hero | no-attack-in-safe-zone | ...
# Repeal a norm at runtime
monitor.repeal_norm(monitor.active_norms()[0].id)
With SQLite persistence — use NormStore to persist norms and violations across sessions, and NormMonitor to check actions against the stored constitution:
from normsync import NormStore, NormMonitor, NormRevision, WorldNorm, AgentAction, print_violations
import time
# Persist norms to SQLite (single file, shareable between agents)
store = NormStore(".normsync/norms.db")
norm = WorldNorm(
name="no-attack-in-safe-zone",
description="Attacking is prohibited in safe zones",
condition="safe_zone",
prohibited="attack",
)
store.save_norm(norm)
store.save_revision(NormRevision(norm_id=norm.id, revision_type="add", timestamp=time.time()))
# Load active norms into monitor and check actions
monitor = NormMonitor(store.get_norms(active_only=True))
action = AgentAction("hero", "attack", "safe_zone")
violations = monitor.check(action)
for v in violations:
store.save_violation(v)
print_violations(violations)
# → Norm Violations table: hero | no-attack-in-safe-zone | ...
store.close()
Track full norm history with NormVersionStore:
from normsync import NormStore, NormVersionStore, WorldNorm
store = NormStore(".normsync/norms.db")
version_store = NormVersionStore(store)
norm = WorldNorm(
name="no-friendly-fire",
description="Friendly fire is prohibited",
condition="friendly",
prohibited="attack",
)
store.save_norm(norm)
v1 = version_store.record_change(norm, changed_by="admin", reason="initial rule")
# Retrieve full version history
history = version_store.get_history("no-friendly-fire")
print(f"Version {history[0].version} by {history[0].changed_by}: {history[0].change_reason}")
store.close()
CLI Reference
normsync [--db PATH] COMMAND [OPTIONS]
| Command | Description | Key options |
|---|---|---|
add NAME DESC CONDITION PROHIBITED |
Add a norm to the constitution | --scope, --priority, --db |
check AGENT_ID ACTION [LOCATION] |
Check an action against active norms | --target, --faction, --db |
violations |
List all recorded violations | --format {table,json,markdown}, --db |
revisions |
List norm revision history | --db |
status |
Show constitution summary | --db |
Examples:
# Add a norm
normsync add no-attack "No attacking in safe zones" safe_zone attack
# Check an action
normsync check hero attack safe_zone
# Export violations as JSON
normsync violations --format json
# Check constitution status
normsync status
GitHub Action
Add normsync norm checks to your CI pipeline:
# .github/workflows/normsync.yml
name: normsync constitution check
on: [push, pull_request]
jobs:
norm-check:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: sandeep-alluru/normsync@main
with:
db: .normsync/norms.db
fail-on-violation: "true"
See docs/github-action.md for full documentation.
vs. Alternatives
| normsync | OpenAI moderation | Constitutional AI | LangChain guardrails | Guardrails AI | |
|---|---|---|---|---|---|
| Norm-as-code | Yes — version-controlled, content-addressed | No | No | No | No |
| Runtime repeal | Yes — deactivate without redeploying | No | No | No | No |
| Multi-agent | Yes — shared SQLite constitution | No | No | Limited | No |
| Offline / local | Yes — single SQLite file | No (API call) | No (training-time) | Partial | Partial |
| CI exit code | Yes — normsync status --db |
No | No | No | No |
| Primary purpose | Agent norm enforcement | Content moderation | Model alignment | LLM output validation | Output validation |
| Open source | MIT | Closed | Closed | MIT | Apache 2.0 |
normsync is not a content moderation system. It is specifically designed to answer: "Given these world norms, did this agent action violate any of them?"
Claude / MCP integration
normsync ships a Model Context Protocol server that lets Claude and other MCP-compatible agents define and check norms directly:
# Start the MCP server
python -m normsync.mcp_server
# In your Claude Code project's .claude/settings.json:
{
"mcpServers": {
"normsync": {
"command": "python",
"args": ["-m", "normsync.mcp_server"]
}
}
}
Once connected, Claude can call normsync/add_norm, normsync/check_action, and normsync/list_violations as tools. See docs/mcp.md for the full tool schema.
OpenAI integration
normsync exposes a FastAPI REST server compatible with OpenAI's function-calling format. The tool definitions are in tools/openai-tools.json and the full API spec is in openapi.yaml.
# Install with REST API dependencies
pip install 'normsync[api]'
# Start the REST server
uvicorn normsync.api:app --reload
# Pass to Codex CLI or any OpenAI-compatible agent
codex --tools tools/openai-tools.json "Check which agent actions violated the world constitution"
Endpoints: GET /health, POST /norm, GET /norms, POST /check, GET /violations. See docs/openai.md for details.
Repository structure
normsync/
├── src/
│ └── normsync/
│ ├── norm.py # WorldNorm, AgentAction, NormViolation, NormRevision dataclasses
│ ├── monitor.py # NormMonitor — token-based norm checking
│ ├── store.py # NormStore — SQLite persistence
│ ├── report.py # print_violations(), to_json(), to_markdown()
│ ├── cli.py # Click CLI (add, check, violations, revisions, status)
│ ├── api.py # FastAPI REST server
│ └── mcp_server.py # MCP server
├── tests/
│ ├── test_norm.py # WorldNorm, AgentAction, NormViolation, NormRevision tests
│ ├── test_monitor.py # NormMonitor unit tests
│ ├── test_store.py # NormStore SQLite tests
│ ├── test_report.py # Report formatter tests
│ ├── test_cli_runner.py # CLI integration tests
│ └── test_api.py # FastAPI endpoint tests
├── examples/
│ └── demo.py # Standalone demo script
├── docs/ # MkDocs documentation
├── tools/
│ └── openai-tools.json # OpenAI function-calling tool definitions
├── assets/
│ ├── hero.png # README hero image
│ └── logo.png # Project logo
├── action.yml # GitHub Action
├── openapi.yaml # OpenAPI 3.1 spec
├── pyproject.toml # Package metadata + dependencies
└── CONTRIBUTING.md # Contribution guide
GitHub Topics
Suggested topics for discoverability:
ai-agents governance norms ai-alignment sqlite mcp llmops multi-agent simulation norm-enforcement world-constitution python
Smithery
normsync is available as an MCP server on Smithery. Search for normsync to install it directly into your Claude Desktop or other MCP-compatible client.
Case Studies
See how teams are using normsync in production:
- Real-Time Behavioral Governance for a 12-Agent AI Fleet — Apex Capital achieves 100% norm violation surfacing and a regulatorily defensible audit trail
- Instant Norm Propagation for 200 Autonomous NPC Agents — Horizon Games reduces faction norm update propagation from 20 minutes to <100ms
Stay Updated
Subscribe to The Silence Layer — weekly dispatches on production AI infrastructure, new releases, and the failure modes that production AI systems don't surface until it's too late.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file normsync-0.1.0.tar.gz.
File metadata
- Download URL: normsync-0.1.0.tar.gz
- Upload date:
- Size: 3.0 MB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
17ede1ef95d2415ccad73523bb0b518bf6815f0e4e7b867a8f184c5c3941c021
|
|
| MD5 |
c38667399437f75cd11111ea99f870cc
|
|
| BLAKE2b-256 |
f4937770f25d1744ca3d49f6b3638f53cc171e6a9fd29dc1d9455690b40a6e3b
|
File details
Details for the file normsync-0.1.0-py3-none-any.whl.
File metadata
- Download URL: normsync-0.1.0-py3-none-any.whl
- Upload date:
- Size: 22.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
46d4d5e9ce888759ea648305b5c99e73af81e1b421c9dcad81dee17daaf37454
|
|
| MD5 |
04cf5dbeadf1eba446ba51d9c1c02e6b
|
|
| BLAKE2b-256 |
f3aa86cbe378a25c26f7f6878bf16a4d5aae3f8397f9cf7f50183cca4bc0e366
|