Compare commits
11 Commits
v5.9.0
...
c4dd4ee25d
| Author | SHA1 | Date | |
|---|---|---|---|
| c4dd4ee25d | |||
| 184ab48933 | |||
| f1732f07c1 | |||
| b0e6d738fa | |||
| 9044fe28ec | |||
| c37107fc42 | |||
| da0be51946 | |||
| d9d80d77cb | |||
| a005610a37 | |||
| 19ba80191f | |||
| e35e22cffb |
63
CHANGELOG.md
63
CHANGELOG.md
@@ -6,6 +6,39 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).
|
|||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
#### NetBox MCP Server: Module-Based Tool Filtering
|
||||||
|
|
||||||
|
Environment-variable-driven module filtering to reduce token consumption:
|
||||||
|
|
||||||
|
- **New config option**: `NETBOX_ENABLED_MODULES` in `~/.config/claude/netbox.env`
|
||||||
|
- **Token savings**: ~15,000 tokens (from ~19,810 to ~4,500) with recommended config
|
||||||
|
- **Default behavior**: All modules enabled if env var unset (backward compatible)
|
||||||
|
- **Startup logging**: Shows enabled modules and tool count on initialization
|
||||||
|
- **Routing guard**: Clear error message when calling disabled module's tools
|
||||||
|
|
||||||
|
**Recommended configuration for cmdb-assistant users:**
|
||||||
|
```bash
|
||||||
|
NETBOX_ENABLED_MODULES=dcim,ipam,virtualization,extras
|
||||||
|
```
|
||||||
|
|
||||||
|
This enables ~43 tools covering all cmdb-assistant commands while staying well below the 25K token warning threshold.
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
#### cmdb-assistant Documentation: Incorrect Tool Names
|
||||||
|
|
||||||
|
Fixed documentation referencing non-existent `virtualization_*` tool names:
|
||||||
|
|
||||||
|
| File | Wrong | Correct |
|
||||||
|
|------|-------|---------|
|
||||||
|
| `claude-md-integration.md` | `virtualization_list_virtual_machines` | `virt_list_vms` |
|
||||||
|
| `claude-md-integration.md` | `virtualization_create_virtual_machine` | `virt_create_vm` |
|
||||||
|
| `cmdb-search.md` | `virtualization_list_virtual_machines` | `virt_list_vms` |
|
||||||
|
|
||||||
|
Also fixed NetBox README.md tool name references for virtualization, wireless, and circuits modules.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## [5.9.0] - 2026-02-03
|
## [5.9.0] - 2026-02-03
|
||||||
@@ -87,6 +120,36 @@ Per-agent model selection using Claude Code's now-supported `model` frontmatter
|
|||||||
- Added missing frontmatter to 13 agents across pr-review, viz-platform, contract-validator, clarity-assist, git-flow, doc-guardian, code-sentinel, cmdb-assistant, and data-platform
|
- Added missing frontmatter to 13 agents across pr-review, viz-platform, contract-validator, clarity-assist, git-flow, doc-guardian, code-sentinel, cmdb-assistant, and data-platform
|
||||||
- All 25 agents now have consistent `name`, `description`, and `model` fields
|
- All 25 agents now have consistent `name`, `description`, and `model` fields
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
#### Agent Frontmatter Hardening v3
|
||||||
|
|
||||||
|
Comprehensive agent-level configuration using Claude Code's supported frontmatter fields.
|
||||||
|
|
||||||
|
**permissionMode added to all 25 agents:**
|
||||||
|
- `bypassPermissions` (1): Executor — full autonomy with code-sentinel + Code Reviewer safety nets
|
||||||
|
- `acceptEdits` (7): Orchestrator, Data Ingestion, Theme Setup, Refactor Advisor, Doc Analyzer, Git Assistant, Maintainer
|
||||||
|
- `default` (7): Planner, Code Reviewer, Data Advisor, Layout Builder, Full Validation, Clarity Coach, CMDB Assistant
|
||||||
|
- `plan` (10): All pr-review agents (5), Data Analysis, Design Reviewer, Component Check, Agent Check, Security Reviewer (code-sentinel)
|
||||||
|
|
||||||
|
**disallowedTools added to 12 agents:**
|
||||||
|
- All `plan`-mode agents (10) + Code Reviewer + Clarity Coach receive `disallowedTools: Write, Edit, MultiEdit`
|
||||||
|
- Enforces read-only contracts at platform level (defense-in-depth with `permissionMode`)
|
||||||
|
|
||||||
|
**Model promotions:**
|
||||||
|
- Planner: `sonnet` → `opus` (architectural reasoning benefits from deeper analysis)
|
||||||
|
- Code Reviewer: `sonnet` → `opus` (quality gate benefits from thorough review)
|
||||||
|
|
||||||
|
**skills frontmatter on 3 agents:**
|
||||||
|
- Executor: 7 safety-critical skills auto-injected (branch-security, runaway-detection, etc.)
|
||||||
|
- Code Reviewer: 4 review skills auto-injected
|
||||||
|
- Maintainer: 2 config skills auto-injected
|
||||||
|
- Body text `## Skills to Load` removed for these agents to avoid duplication
|
||||||
|
|
||||||
|
**Documentation:**
|
||||||
|
- `CLAUDE.md` and `docs/CONFIGURATION.md` updated with complete agent configuration matrix
|
||||||
|
- New subsections: permissionMode Guide, disallowedTools Guide, skills Frontmatter Guide
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## [5.8.0] - 2026-02-02
|
## [5.8.0] - 2026-02-02
|
||||||
|
|||||||
81
CLAUDE.md
81
CLAUDE.md
@@ -271,39 +271,60 @@ leo-claude-mktplace/
|
|||||||
| **Executor** | Implementation-focused | Code implementation, branch management, MR creation |
|
| **Executor** | Implementation-focused | Code implementation, branch management, MR creation |
|
||||||
| **Code Reviewer** | Thorough, practical | Pre-close quality review, security scan, test verification |
|
| **Code Reviewer** | Thorough, practical | Pre-close quality review, security scan, test verification |
|
||||||
|
|
||||||
### Agent Model Selection
|
### Agent Frontmatter Configuration
|
||||||
|
|
||||||
Agents specify their model in frontmatter using Claude Code's `model` field. Supported values: `sonnet` (default), `opus`, `haiku`, `inherit`.
|
Agents specify their configuration in frontmatter using Claude Code's supported fields. Reference: https://code.claude.com/docs/en/sub-agents
|
||||||
|
|
||||||
| Plugin | Agent | Model | Rationale |
|
**Supported frontmatter fields:**
|
||||||
|--------|-------|-------|-----------|
|
|
||||||
| projman | Planner | sonnet | Architectural analysis, sprint planning |
|
|
||||||
| projman | Orchestrator | sonnet | Coordination and tool dispatch |
|
|
||||||
| projman | Executor | sonnet | Code generation and implementation |
|
|
||||||
| projman | Code Reviewer | sonnet | Quality gate, pattern detection |
|
|
||||||
| pr-review | Coordinator | sonnet | Orchestrates sub-agents, aggregates findings |
|
|
||||||
| pr-review | Security Reviewer | sonnet | Security analysis |
|
|
||||||
| pr-review | Performance Analyst | sonnet | Performance pattern detection |
|
|
||||||
| pr-review | Maintainability Auditor | haiku | Pattern matching (complexity, duplication) |
|
|
||||||
| pr-review | Test Validator | haiku | Coverage gap detection |
|
|
||||||
| data-platform | Data Advisor | sonnet | Schema validation, dbt orchestration |
|
|
||||||
| data-platform | Data Analysis | sonnet | Data exploration and profiling |
|
|
||||||
| data-platform | Data Ingestion | haiku | Data loading operations |
|
|
||||||
| viz-platform | Design Reviewer | sonnet | DMC validation + accessibility |
|
|
||||||
| viz-platform | Layout Builder | sonnet | Dashboard design guidance |
|
|
||||||
| viz-platform | Component Check | haiku | Quick component validation |
|
|
||||||
| viz-platform | Theme Setup | haiku | Theme configuration |
|
|
||||||
| contract-validator | Agent Check | haiku | Reference checking |
|
|
||||||
| contract-validator | Full Validation | sonnet | Marketplace sweep |
|
|
||||||
| code-sentinel | Security Reviewer | sonnet | Security analysis |
|
|
||||||
| code-sentinel | Refactor Advisor | sonnet | Code refactoring advice |
|
|
||||||
| doc-guardian | Doc Analyzer | sonnet | Documentation drift detection |
|
|
||||||
| clarity-assist | Clarity Coach | sonnet | Conversational coaching |
|
|
||||||
| git-flow | Git Assistant | haiku | Git operations |
|
|
||||||
| claude-config-maintainer | Maintainer | sonnet | CLAUDE.md optimization |
|
|
||||||
| cmdb-assistant | CMDB Assistant | sonnet | NetBox operations |
|
|
||||||
|
|
||||||
Override by editing the `model:` field in `plugins/{plugin}/agents/{agent}.md`.
|
| Field | Required | Default | Description |
|
||||||
|
|-------|----------|---------|-------------|
|
||||||
|
| `name` | Yes | — | Unique identifier, lowercase + hyphens |
|
||||||
|
| `description` | Yes | — | When Claude should delegate to this subagent |
|
||||||
|
| `model` | No | `inherit` | `sonnet`, `opus`, `haiku`, or `inherit` |
|
||||||
|
| `permissionMode` | No | `default` | Controls permission prompts: `default`, `acceptEdits`, `dontAsk`, `bypassPermissions`, `plan` |
|
||||||
|
| `disallowedTools` | No | none | Comma-separated tools to remove from agent's toolset |
|
||||||
|
| `skills` | No | none | Comma-separated skills auto-injected into context at startup |
|
||||||
|
| `hooks` | No | none | Lifecycle hooks scoped to this subagent |
|
||||||
|
|
||||||
|
**Complete agent matrix:**
|
||||||
|
|
||||||
|
| Plugin | Agent | `model` | `permissionMode` | `disallowedTools` | `skills` |
|
||||||
|
|--------|-------|---------|-------------------|--------------------|----------|
|
||||||
|
| projman | planner | opus | default | — | body text (14) |
|
||||||
|
| projman | orchestrator | sonnet | acceptEdits | — | body text (12) |
|
||||||
|
| projman | executor | sonnet | bypassPermissions | — | frontmatter (7) |
|
||||||
|
| projman | code-reviewer | opus | default | Write, Edit, MultiEdit | frontmatter (4) |
|
||||||
|
| pr-review | coordinator | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| pr-review | security-reviewer | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| pr-review | performance-analyst | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| pr-review | maintainability-auditor | haiku | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| pr-review | test-validator | haiku | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| data-platform | data-advisor | sonnet | default | — | — |
|
||||||
|
| data-platform | data-analysis | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| data-platform | data-ingestion | haiku | acceptEdits | — | — |
|
||||||
|
| viz-platform | design-reviewer | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| viz-platform | layout-builder | sonnet | default | — | — |
|
||||||
|
| viz-platform | component-check | haiku | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| viz-platform | theme-setup | haiku | acceptEdits | — | — |
|
||||||
|
| contract-validator | full-validation | sonnet | default | — | — |
|
||||||
|
| contract-validator | agent-check | haiku | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| code-sentinel | security-reviewer | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| code-sentinel | refactor-advisor | sonnet | acceptEdits | — | — |
|
||||||
|
| doc-guardian | doc-analyzer | sonnet | acceptEdits | — | — |
|
||||||
|
| clarity-assist | clarity-coach | sonnet | default | Write, Edit, MultiEdit | — |
|
||||||
|
| git-flow | git-assistant | haiku | acceptEdits | — | — |
|
||||||
|
| claude-config-maintainer | maintainer | sonnet | acceptEdits | — | frontmatter (2) |
|
||||||
|
| cmdb-assistant | cmdb-assistant | sonnet | default | — | — |
|
||||||
|
|
||||||
|
**Design principles:**
|
||||||
|
- `bypassPermissions` is granted to exactly ONE agent (Executor) which has code-sentinel PreToolUse hook + Code Reviewer downstream as safety nets.
|
||||||
|
- `plan` mode is assigned to all pure analysis agents (pr-review, read-only validators).
|
||||||
|
- `disallowedTools: Write, Edit, MultiEdit` provides defense-in-depth on agents that should never write files.
|
||||||
|
- `skills` frontmatter is used for agents with ≤7 skills where guaranteed loading is safety-critical. Agents with 8+ skills use body text `## Skills to Load` for selective loading.
|
||||||
|
- `hooks` (agent-scoped) is reserved for future use (v6.0+).
|
||||||
|
|
||||||
|
Override any field by editing the agent's `.md` file in `plugins/{plugin}/agents/`.
|
||||||
|
|
||||||
### MCP Server Tools (Gitea)
|
### MCP Server Tools (Gitea)
|
||||||
|
|
||||||
|
|||||||
@@ -496,60 +496,100 @@ Not all plugins have MCP servers. The install script handles this automatically:
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## Agent Model Selection
|
## Agent Frontmatter Configuration
|
||||||
|
|
||||||
Marketplace agents specify their preferred model using Claude Code's `model` frontmatter field. This allows cost/performance optimization per agent.
|
Agents specify their configuration in frontmatter using Claude Code's supported fields. Reference: https://code.claude.com/docs/en/sub-agents
|
||||||
|
|
||||||
### Supported Values
|
### Supported Frontmatter Fields
|
||||||
|
|
||||||
| Value | Description |
|
| Field | Required | Default | Description |
|
||||||
|-------|-------------|
|
|-------|----------|---------|-------------|
|
||||||
| `sonnet` | Default. Balanced performance and cost. |
|
| `name` | Yes | — | Unique identifier, lowercase + hyphens |
|
||||||
| `opus` | Higher reasoning depth. Use for complex analysis. |
|
| `description` | Yes | — | When Claude should delegate to this subagent |
|
||||||
| `haiku` | Faster, lower cost. Use for mechanical tasks. |
|
| `model` | No | `inherit` | `sonnet`, `opus`, `haiku`, or `inherit` |
|
||||||
| `inherit` | Use session's current model setting. |
|
| `permissionMode` | No | `default` | Controls permission prompts: `default`, `acceptEdits`, `dontAsk`, `bypassPermissions`, `plan` |
|
||||||
|
| `disallowedTools` | No | none | Comma-separated tools to remove from agent's toolset |
|
||||||
|
| `skills` | No | none | Comma-separated skills auto-injected into context at startup |
|
||||||
|
| `hooks` | No | none | Lifecycle hooks scoped to this subagent |
|
||||||
|
|
||||||
### How It Works
|
### Complete Agent Matrix
|
||||||
|
|
||||||
Each agent in `plugins/{plugin}/agents/{agent}.md` has frontmatter like:
|
| Plugin | Agent | `model` | `permissionMode` | `disallowedTools` | `skills` |
|
||||||
|
|--------|-------|---------|-------------------|--------------------|----------|
|
||||||
|
| projman | planner | opus | default | — | body text (14) |
|
||||||
|
| projman | orchestrator | sonnet | acceptEdits | — | body text (12) |
|
||||||
|
| projman | executor | sonnet | bypassPermissions | — | frontmatter (7) |
|
||||||
|
| projman | code-reviewer | opus | default | Write, Edit, MultiEdit | frontmatter (4) |
|
||||||
|
| pr-review | coordinator | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| pr-review | security-reviewer | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| pr-review | performance-analyst | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| pr-review | maintainability-auditor | haiku | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| pr-review | test-validator | haiku | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| data-platform | data-advisor | sonnet | default | — | — |
|
||||||
|
| data-platform | data-analysis | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| data-platform | data-ingestion | haiku | acceptEdits | — | — |
|
||||||
|
| viz-platform | design-reviewer | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| viz-platform | layout-builder | sonnet | default | — | — |
|
||||||
|
| viz-platform | component-check | haiku | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| viz-platform | theme-setup | haiku | acceptEdits | — | — |
|
||||||
|
| contract-validator | full-validation | sonnet | default | — | — |
|
||||||
|
| contract-validator | agent-check | haiku | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| code-sentinel | security-reviewer | sonnet | plan | Write, Edit, MultiEdit | — |
|
||||||
|
| code-sentinel | refactor-advisor | sonnet | acceptEdits | — | — |
|
||||||
|
| doc-guardian | doc-analyzer | sonnet | acceptEdits | — | — |
|
||||||
|
| clarity-assist | clarity-coach | sonnet | default | Write, Edit, MultiEdit | — |
|
||||||
|
| git-flow | git-assistant | haiku | acceptEdits | — | — |
|
||||||
|
| claude-config-maintainer | maintainer | sonnet | acceptEdits | — | frontmatter (2) |
|
||||||
|
| cmdb-assistant | cmdb-assistant | sonnet | default | — | — |
|
||||||
|
|
||||||
```yaml
|
### Design Principles
|
||||||
---
|
|
||||||
name: planner
|
|
||||||
description: Sprint planning agent - thoughtful architecture analysis
|
|
||||||
model: sonnet
|
|
||||||
---
|
|
||||||
```
|
|
||||||
|
|
||||||
Claude Code reads this field when invoking the agent as a subagent.
|
- `bypassPermissions` is granted to exactly ONE agent (Executor) which has code-sentinel PreToolUse hook + Code Reviewer downstream as safety nets.
|
||||||
|
- `plan` mode is assigned to all pure analysis agents (pr-review, read-only validators).
|
||||||
|
- `disallowedTools: Write, Edit, MultiEdit` provides defense-in-depth on agents that should never write files.
|
||||||
|
- `skills` frontmatter is used for agents with ≤7 skills where guaranteed loading is safety-critical. Agents with 8+ skills use body text `## Skills to Load` for selective loading.
|
||||||
|
- `hooks` (agent-scoped) is reserved for future use (v6.0+).
|
||||||
|
|
||||||
### Model Assignments
|
Override any field by editing the agent's `.md` file in `plugins/{plugin}/agents/`.
|
||||||
|
|
||||||
Agents are assigned models based on their task complexity:
|
### permissionMode Guide
|
||||||
|
|
||||||
| Model | Agents | Rationale |
|
| Value | Prompts for file ops? | Prompts for Bash? | Prompts for MCP? | Use when |
|
||||||
|-------|--------|-----------|
|
|-------|-----------------------|-------------------|-------------------|----------|
|
||||||
| **sonnet** | Planner, Orchestrator, Executor, Code Reviewer, Coordinator, Security Reviewers, Performance Analyst, Data Advisor, Data Analysis, Design Reviewer, Layout Builder, Full Validation, Doc Analyzer, Clarity Coach, Maintainer, CMDB Assistant, Refactor Advisor | Standard reasoning, tool orchestration, code generation |
|
| `default` | Yes | Yes | No (MCP bypasses permissions) | You want full visibility |
|
||||||
| **haiku** | Maintainability Auditor, Test Validator, Component Check, Theme Setup, Agent Check, Data Ingestion, Git Assistant | Pattern matching, quick validation, mechanical tasks |
|
| `acceptEdits` | No | Yes | No | Core job is file read/write, Bash visibility useful |
|
||||||
|
| `dontAsk` | No | No (most) | No | Even Bash prompts are friction |
|
||||||
|
| `bypassPermissions` | No | No | No | Agent has downstream safety layers |
|
||||||
|
| `plan` | N/A (read-only) | N/A (read-only) | No | Pure analysis, no modifications |
|
||||||
|
|
||||||
### Overriding Model Selection
|
### disallowedTools Guide
|
||||||
|
|
||||||
**Per-agent override:** Edit the `model:` field in the agent file:
|
Use `disallowedTools` to remove specific tools from an agent's toolset. This is a blacklist — the agent inherits all tools from the main thread, then the listed tools are removed.
|
||||||
|
|
||||||
```bash
|
Prefer `disallowedTools` over `tools` (whitelist) because:
|
||||||
# Change executor to use opus for heavy implementation work
|
- New MCP servers are automatically available without updating every agent.
|
||||||
nano plugins/projman/agents/executor.md
|
- Less configuration to maintain.
|
||||||
# Change model: sonnet to model: opus
|
- Easier to audit — you only list what's blocked.
|
||||||
```
|
|
||||||
|
|
||||||
**Session-level:** Users on Opus subscription can change the agent's model to `inherit` to use whatever model the session is using.
|
Common patterns:
|
||||||
|
- `disallowedTools: Write, Edit, MultiEdit` — read-only agent, cannot modify files.
|
||||||
|
- `disallowedTools: Bash` — no shell access (rare, most agents need at least read-only Bash).
|
||||||
|
|
||||||
### Best Practices
|
### skills Frontmatter Guide
|
||||||
|
|
||||||
1. **Default to sonnet** - Good balance for most tasks
|
The `skills` field auto-injects skill file contents into the agent's context window at startup. The agent does NOT need to read the files — they are already present.
|
||||||
2. **Use haiku for speed-sensitive agents** - Sub-agents dispatched in parallel, read-only tasks
|
|
||||||
3. **Reserve opus for heavy analysis** - Only when sonnet's reasoning isn't sufficient
|
**When to use frontmatter `skills`:**
|
||||||
4. **Use inherit sparingly** - Only when you want session-level control
|
- Agent has ≤7 skills.
|
||||||
|
- Skills are safety-critical (e.g., `branch-security`, `runaway-detection`).
|
||||||
|
- You need guaranteed loading — no risk of the agent skipping a skill.
|
||||||
|
|
||||||
|
**When to keep body text `## Skills to Load`:**
|
||||||
|
- Agent has 8+ skills (context window cost too high for full injection).
|
||||||
|
- Skills are situational — not all needed for every invocation.
|
||||||
|
- Agent benefits from selective loading based on the specific task.
|
||||||
|
|
||||||
|
Skill names in frontmatter are resolved relative to the plugin's `skills/` directory. Use the filename without the `.md` extension.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
|||||||
6
mcp-servers/gitea/.doc-guardian-queue
Normal file
6
mcp-servers/gitea/.doc-guardian-queue
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
2026-02-03T14:09:25 | mcp-servers | /home/lmiranda/claude-plugins-work/mcp-servers/gitea/tests/test_config.py | docs/COMMANDS-CHEATSHEET.md CLAUDE.md
|
||||||
|
2026-02-03T14:09:33 | mcp-servers | /home/lmiranda/claude-plugins-work/mcp-servers/gitea/tests/test_gitea_client.py | docs/COMMANDS-CHEATSHEET.md CLAUDE.md
|
||||||
|
2026-02-03T14:10:22 | mcp-servers | /home/lmiranda/claude-plugins-work/mcp-servers/gitea/tests/test_issues.py | docs/COMMANDS-CHEATSHEET.md CLAUDE.md
|
||||||
|
2026-02-03T14:17:12 | mcp-servers | /home/lmiranda/claude-plugins-work/mcp-servers/gitea/README.md | docs/COMMANDS-CHEATSHEET.md CLAUDE.md
|
||||||
|
2026-02-03T14:18:27 | mcp-servers | /home/lmiranda/claude-plugins-work/mcp-servers/gitea/CHANGELOG.md | docs/COMMANDS-CHEATSHEET.md CLAUDE.md
|
||||||
|
2026-02-03T14:18:41 | mcp-servers | /home/lmiranda/claude-plugins-work/mcp-servers/gitea/TESTING.md | docs/COMMANDS-CHEATSHEET.md CLAUDE.md
|
||||||
92
mcp-servers/gitea/CHANGELOG.md
Normal file
92
mcp-servers/gitea/CHANGELOG.md
Normal file
@@ -0,0 +1,92 @@
|
|||||||
|
# Changelog
|
||||||
|
|
||||||
|
All notable changes to the Gitea MCP Server will be documented in this file.
|
||||||
|
|
||||||
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## [Unreleased]
|
||||||
|
|
||||||
|
## [1.3.0] - 2026-02-03
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- Pull request tools (7 tools):
|
||||||
|
- `list_pull_requests` - List PRs from repository
|
||||||
|
- `get_pull_request` - Get specific PR details
|
||||||
|
- `get_pr_diff` - Get PR diff content
|
||||||
|
- `get_pr_comments` - Get comments on a PR
|
||||||
|
- `create_pr_review` - Create PR review (approve/request changes/comment)
|
||||||
|
- `add_pr_comment` - Add comment to PR
|
||||||
|
- `create_pull_request` - Create new pull request
|
||||||
|
- Label creation tools (3 tools):
|
||||||
|
- `create_label` - Create repo-level label
|
||||||
|
- `create_org_label` - Create organization-level label
|
||||||
|
- `create_label_smart` - Auto-detect org vs repo for label creation
|
||||||
|
- Validation tools (2 tools):
|
||||||
|
- `validate_repo_org` - Check if repo belongs to organization
|
||||||
|
- `get_branch_protection` - Get branch protection rules
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
- Total tools increased from 20 to 36
|
||||||
|
- Updated test suite to 64 tests (was 42)
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
- Test fixtures updated to use `owner/repo` format
|
||||||
|
- Fixed aggregate_issues tests to pass required `org` argument
|
||||||
|
|
||||||
|
## [1.2.0] - 2026-01-28
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- Milestone management tools (5 tools):
|
||||||
|
- `list_milestones` - List all milestones
|
||||||
|
- `get_milestone` - Get specific milestone
|
||||||
|
- `create_milestone` - Create new milestone
|
||||||
|
- `update_milestone` - Update existing milestone
|
||||||
|
- `delete_milestone` - Delete a milestone
|
||||||
|
- Issue dependency tools (4 tools):
|
||||||
|
- `list_issue_dependencies` - List blocking issues
|
||||||
|
- `create_issue_dependency` - Create dependency between issues
|
||||||
|
- `remove_issue_dependency` - Remove dependency
|
||||||
|
- `get_execution_order` - Calculate parallelizable execution order
|
||||||
|
|
||||||
|
## [1.1.0] - 2026-01-21
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- Wiki and lessons learned tools (7 tools):
|
||||||
|
- `list_wiki_pages` - List all wiki pages
|
||||||
|
- `get_wiki_page` - Get specific wiki page content
|
||||||
|
- `create_wiki_page` - Create new wiki page
|
||||||
|
- `update_wiki_page` - Update existing wiki page
|
||||||
|
- `create_lesson` - Create lessons learned entry
|
||||||
|
- `search_lessons` - Search lessons by query/tags
|
||||||
|
- `allocate_rfc_number` - Get next available RFC number
|
||||||
|
- Automatic git remote URL detection for repository configuration
|
||||||
|
- Support for SSH, HTTPS, and HTTP git URL formats
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
- Configuration now uses `owner/repo` format exclusively
|
||||||
|
- Removed separate `GITEA_OWNER` configuration (now derived from repo path)
|
||||||
|
|
||||||
|
## [1.0.0] - 2025-01-06
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- Initial release with 8 core tools:
|
||||||
|
- `list_issues` - List issues from repository
|
||||||
|
- `get_issue` - Get specific issue details
|
||||||
|
- `create_issue` - Create new issue with labels
|
||||||
|
- `update_issue` - Update existing issue
|
||||||
|
- `add_comment` - Add comment to issue
|
||||||
|
- `get_labels` - Get all labels (org + repo)
|
||||||
|
- `suggest_labels` - Intelligent label suggestion
|
||||||
|
- `aggregate_issues` - Cross-repository issue aggregation (PMO mode)
|
||||||
|
- Hybrid configuration system (system + project level)
|
||||||
|
- Branch-aware security model
|
||||||
|
- Mode detection (project vs company/PMO)
|
||||||
|
- 42 unit tests with mocks
|
||||||
|
- Comprehensive documentation
|
||||||
|
|
||||||
|
[Unreleased]: https://github.com/owner/repo/compare/v1.3.0...HEAD
|
||||||
|
[1.3.0]: https://github.com/owner/repo/compare/v1.2.0...v1.3.0
|
||||||
|
[1.2.0]: https://github.com/owner/repo/compare/v1.1.0...v1.2.0
|
||||||
|
[1.1.0]: https://github.com/owner/repo/compare/v1.0.0...v1.1.0
|
||||||
|
[1.0.0]: https://github.com/owner/repo/releases/tag/v1.0.0
|
||||||
@@ -19,8 +19,9 @@ The Gitea MCP Server provides Claude Code with direct access to Gitea for issue
|
|||||||
- **Hybrid Configuration**: System-level credentials + project-level paths
|
- **Hybrid Configuration**: System-level credentials + project-level paths
|
||||||
- **PMO Support**: Multi-repository aggregation for organization-wide views
|
- **PMO Support**: Multi-repository aggregation for organization-wide views
|
||||||
|
|
||||||
### Tools Provided
|
### Tools Provided (36 total)
|
||||||
|
|
||||||
|
#### Issue Management (6 tools)
|
||||||
| Tool | Description | Mode |
|
| Tool | Description | Mode |
|
||||||
|------|-------------|------|
|
|------|-------------|------|
|
||||||
| `list_issues` | List issues from repository | Both |
|
| `list_issues` | List issues from repository | Both |
|
||||||
@@ -28,9 +29,61 @@ The Gitea MCP Server provides Claude Code with direct access to Gitea for issue
|
|||||||
| `create_issue` | Create new issue with labels | Both |
|
| `create_issue` | Create new issue with labels | Both |
|
||||||
| `update_issue` | Update existing issue | Both |
|
| `update_issue` | Update existing issue | Both |
|
||||||
| `add_comment` | Add comment to issue | Both |
|
| `add_comment` | Add comment to issue | Both |
|
||||||
|
| `aggregate_issues` | Cross-repository issue aggregation | PMO Only |
|
||||||
|
|
||||||
|
#### Label Management (5 tools)
|
||||||
|
| Tool | Description | Mode |
|
||||||
|
|------|-------------|------|
|
||||||
| `get_labels` | Get all labels (org + repo) | Both |
|
| `get_labels` | Get all labels (org + repo) | Both |
|
||||||
| `suggest_labels` | Intelligent label suggestion | Both |
|
| `suggest_labels` | Intelligent label suggestion | Both |
|
||||||
| `aggregate_issues` | Cross-repository issue aggregation | PMO Only |
|
| `create_label` | Create repo-level label | Both |
|
||||||
|
| `create_org_label` | Create organization-level label | Both |
|
||||||
|
| `create_label_smart` | Auto-detect org vs repo for label creation | Both |
|
||||||
|
|
||||||
|
#### Wiki & Lessons Learned (7 tools)
|
||||||
|
| Tool | Description | Mode |
|
||||||
|
|------|-------------|------|
|
||||||
|
| `list_wiki_pages` | List all wiki pages | Both |
|
||||||
|
| `get_wiki_page` | Get specific wiki page content | Both |
|
||||||
|
| `create_wiki_page` | Create new wiki page | Both |
|
||||||
|
| `update_wiki_page` | Update existing wiki page | Both |
|
||||||
|
| `create_lesson` | Create lessons learned entry | Both |
|
||||||
|
| `search_lessons` | Search lessons by query/tags | Both |
|
||||||
|
| `allocate_rfc_number` | Get next available RFC number | Both |
|
||||||
|
|
||||||
|
#### Milestone Management (5 tools)
|
||||||
|
| Tool | Description | Mode |
|
||||||
|
|------|-------------|------|
|
||||||
|
| `list_milestones` | List all milestones | Both |
|
||||||
|
| `get_milestone` | Get specific milestone | Both |
|
||||||
|
| `create_milestone` | Create new milestone | Both |
|
||||||
|
| `update_milestone` | Update existing milestone | Both |
|
||||||
|
| `delete_milestone` | Delete a milestone | Both |
|
||||||
|
|
||||||
|
#### Issue Dependencies (4 tools)
|
||||||
|
| Tool | Description | Mode |
|
||||||
|
|------|-------------|------|
|
||||||
|
| `list_issue_dependencies` | List blocking issues | Both |
|
||||||
|
| `create_issue_dependency` | Create dependency between issues | Both |
|
||||||
|
| `remove_issue_dependency` | Remove dependency | Both |
|
||||||
|
| `get_execution_order` | Calculate parallelizable execution order | Both |
|
||||||
|
|
||||||
|
#### Pull Request Tools (7 tools)
|
||||||
|
| Tool | Description | Mode |
|
||||||
|
|------|-------------|------|
|
||||||
|
| `list_pull_requests` | List PRs from repository | Both |
|
||||||
|
| `get_pull_request` | Get specific PR details | Both |
|
||||||
|
| `get_pr_diff` | Get PR diff content | Both |
|
||||||
|
| `get_pr_comments` | Get comments on a PR | Both |
|
||||||
|
| `create_pr_review` | Create PR review (approve/request changes) | Both |
|
||||||
|
| `add_pr_comment` | Add comment to PR | Both |
|
||||||
|
| `create_pull_request` | Create new pull request | Both |
|
||||||
|
|
||||||
|
#### Validation Tools (2 tools)
|
||||||
|
| Tool | Description | Mode |
|
||||||
|
|------|-------------|------|
|
||||||
|
| `validate_repo_org` | Check if repo belongs to organization | Both |
|
||||||
|
| `get_branch_protection` | Get branch protection rules | Both |
|
||||||
|
|
||||||
## Architecture
|
## Architecture
|
||||||
|
|
||||||
@@ -40,15 +93,20 @@ The Gitea MCP Server provides Claude Code with direct access to Gitea for issue
|
|||||||
mcp-servers/gitea/
|
mcp-servers/gitea/
|
||||||
├── .venv/ # Python virtual environment
|
├── .venv/ # Python virtual environment
|
||||||
├── requirements.txt # Python dependencies
|
├── requirements.txt # Python dependencies
|
||||||
|
├── run.sh # Entry point script
|
||||||
├── mcp_server/
|
├── mcp_server/
|
||||||
│ ├── __init__.py
|
│ ├── __init__.py
|
||||||
│ ├── server.py # MCP server entry point
|
│ ├── server.py # MCP server entry point (36 tools)
|
||||||
│ ├── config.py # Configuration loader
|
│ ├── config.py # Configuration loader with auto-detection
|
||||||
│ ├── gitea_client.py # Gitea API client
|
│ ├── gitea_client.py # Gitea API client
|
||||||
│ └── tools/
|
│ └── tools/
|
||||||
│ ├── __init__.py
|
│ ├── __init__.py
|
||||||
│ ├── issues.py # Issue tools
|
│ ├── issues.py # Issue management tools
|
||||||
│ └── labels.py # Label tools
|
│ ├── labels.py # Label management tools
|
||||||
|
│ ├── wiki.py # Wiki & lessons learned tools
|
||||||
|
│ ├── milestones.py # Milestone management tools
|
||||||
|
│ ├── dependencies.py # Issue dependency tools
|
||||||
|
│ └── pull_requests.py # Pull request tools
|
||||||
├── tests/
|
├── tests/
|
||||||
│ ├── __init__.py
|
│ ├── __init__.py
|
||||||
│ ├── test_config.py
|
│ ├── test_config.py
|
||||||
@@ -56,7 +114,8 @@ mcp-servers/gitea/
|
|||||||
│ ├── test_issues.py
|
│ ├── test_issues.py
|
||||||
│ └── test_labels.py
|
│ └── test_labels.py
|
||||||
├── README.md # This file
|
├── README.md # This file
|
||||||
└── TESTING.md # Testing instructions
|
├── TESTING.md # Testing instructions
|
||||||
|
└── CHANGELOG.md # Version history
|
||||||
```
|
```
|
||||||
|
|
||||||
### Mode Detection
|
### Mode Detection
|
||||||
@@ -111,7 +170,6 @@ mkdir -p ~/.config/claude
|
|||||||
cat > ~/.config/claude/gitea.env << EOF
|
cat > ~/.config/claude/gitea.env << EOF
|
||||||
GITEA_API_URL=https://gitea.example.com/api/v1
|
GITEA_API_URL=https://gitea.example.com/api/v1
|
||||||
GITEA_API_TOKEN=your_gitea_token_here
|
GITEA_API_TOKEN=your_gitea_token_here
|
||||||
GITEA_OWNER=bandit
|
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
chmod 600 ~/.config/claude/gitea.env
|
chmod 600 ~/.config/claude/gitea.env
|
||||||
@@ -137,14 +195,34 @@ For company/PMO mode, omit the `.env` file or don't set `GITEA_REPO`.
|
|||||||
**Required Variables**:
|
**Required Variables**:
|
||||||
- `GITEA_API_URL` - Gitea API endpoint (e.g., `https://gitea.example.com/api/v1`)
|
- `GITEA_API_URL` - Gitea API endpoint (e.g., `https://gitea.example.com/api/v1`)
|
||||||
- `GITEA_API_TOKEN` - Personal access token with repo permissions
|
- `GITEA_API_TOKEN` - Personal access token with repo permissions
|
||||||
- `GITEA_OWNER` - Organization or user name (e.g., `bandit`)
|
|
||||||
|
|
||||||
### Project-Level Configuration
|
### Project-Level Configuration
|
||||||
|
|
||||||
**File**: `<project-root>/.env`
|
**File**: `<project-root>/.env`
|
||||||
|
|
||||||
**Optional Variables**:
|
**Optional Variables**:
|
||||||
- `GITEA_REPO` - Repository name (enables project mode)
|
- `GITEA_REPO` - Repository in `owner/repo` format (enables project mode)
|
||||||
|
|
||||||
|
### Automatic Repository Detection
|
||||||
|
|
||||||
|
If `GITEA_REPO` is not set, the server auto-detects the repository from your git remote:
|
||||||
|
|
||||||
|
**Supported URL Formats**:
|
||||||
|
- SSH: `ssh://git@gitea.example.com:22/owner/repo.git`
|
||||||
|
- SSH short: `git@gitea.example.com:owner/repo.git`
|
||||||
|
- HTTPS: `https://gitea.example.com/owner/repo.git`
|
||||||
|
- HTTP: `http://gitea.example.com/owner/repo.git`
|
||||||
|
|
||||||
|
The repository is extracted as `owner/repo` format automatically.
|
||||||
|
|
||||||
|
### Project Directory Detection
|
||||||
|
|
||||||
|
The server finds your project directory using these strategies (in order):
|
||||||
|
|
||||||
|
1. `CLAUDE_PROJECT_DIR` environment variable (highest priority)
|
||||||
|
2. `PWD` environment variable (if `.git` or `.env` present)
|
||||||
|
3. Current working directory (if `.git` or `.env` present)
|
||||||
|
4. Falls back to company/PMO mode if no project found
|
||||||
|
|
||||||
### Generating Gitea API Token
|
### Generating Gitea API Token
|
||||||
|
|
||||||
@@ -220,13 +298,13 @@ suggestions = await label_tools.suggest_labels(context)
|
|||||||
|
|
||||||
### Unit Tests
|
### Unit Tests
|
||||||
|
|
||||||
Run all 42 unit tests with mocks:
|
Run all 64 unit tests with mocks:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
pytest tests/ -v
|
pytest tests/ -v
|
||||||
```
|
```
|
||||||
|
|
||||||
Expected: `42 passed in 0.57s`
|
Expected: `64 passed`
|
||||||
|
|
||||||
### Integration Tests
|
### Integration Tests
|
||||||
|
|
||||||
@@ -327,11 +405,15 @@ See [TESTING.md](./TESTING.md#troubleshooting) for more details.
|
|||||||
|
|
||||||
### Project Structure
|
### Project Structure
|
||||||
|
|
||||||
- `config.py` - Hybrid configuration loader with mode detection
|
- `config.py` - Hybrid configuration loader with auto-detection
|
||||||
- `gitea_client.py` - Synchronous Gitea API client using requests
|
- `gitea_client.py` - Synchronous Gitea API client using requests
|
||||||
- `tools/issues.py` - Async wrappers with branch detection
|
- `tools/issues.py` - Issue management with branch detection
|
||||||
- `tools/labels.py` - Label management and suggestion
|
- `tools/labels.py` - Label management and intelligent suggestions
|
||||||
- `server.py` - MCP server with JSON-RPC 2.0 over stdio
|
- `tools/wiki.py` - Wiki pages and lessons learned
|
||||||
|
- `tools/milestones.py` - Milestone CRUD operations
|
||||||
|
- `tools/dependencies.py` - Issue dependency tracking
|
||||||
|
- `tools/pull_requests.py` - PR review and management
|
||||||
|
- `server.py` - MCP server with 36 tools over JSON-RPC 2.0 stdio
|
||||||
|
|
||||||
### Adding New Tools
|
### Adding New Tools
|
||||||
|
|
||||||
@@ -374,18 +456,14 @@ def list_issues(self, state='open', labels=None, repo=None):
|
|||||||
|
|
||||||
## Changelog
|
## Changelog
|
||||||
|
|
||||||
### v1.0.0 (2025-01-06) - Phase 1 Complete
|
See [CHANGELOG.md](./CHANGELOG.md) for full version history.
|
||||||
|
|
||||||
✅ Initial implementation:
|
### Recent Updates
|
||||||
- Configuration management (hybrid system + project)
|
|
||||||
- Gitea API client with all CRUD operations
|
- **v1.3.0** - Pull request tools (7 tools), label creation tools (3)
|
||||||
- MCP server with 8 tools
|
- **v1.2.0** - Milestone management (5 tools), issue dependencies (4 tools)
|
||||||
- Issue tools with branch detection
|
- **v1.1.0** - Wiki & lessons learned system (7 tools)
|
||||||
- Label tools with intelligent suggestions
|
- **v1.0.0** - Initial release with core issue/label tools (8 tools)
|
||||||
- Mode detection (project vs company)
|
|
||||||
- Branch-aware security model
|
|
||||||
- 42 unit tests (100% passing)
|
|
||||||
- Comprehensive documentation
|
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
@@ -407,6 +485,6 @@ For issues or questions:
|
|||||||
---
|
---
|
||||||
|
|
||||||
**Built for**: Leo Claude Marketplace - Project Management Plugins
|
**Built for**: Leo Claude Marketplace - Project Management Plugins
|
||||||
**Phase**: 1 (Complete)
|
**Tools**: 36
|
||||||
**Status**: ✅ Production Ready
|
**Status**: ✅ Production Ready
|
||||||
**Last Updated**: 2025-01-06
|
**Last Updated**: 2026-02-03
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ source .venv/bin/activate # Linux/Mac
|
|||||||
|
|
||||||
### Running All Tests
|
### Running All Tests
|
||||||
|
|
||||||
Run all 42 unit tests:
|
Run all 64 unit tests:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
pytest tests/ -v
|
pytest tests/ -v
|
||||||
@@ -36,7 +36,7 @@ pytest tests/ -v
|
|||||||
|
|
||||||
Expected output:
|
Expected output:
|
||||||
```
|
```
|
||||||
============================== 42 passed in 0.57s ==============================
|
============================== 64 passed ==============================
|
||||||
```
|
```
|
||||||
|
|
||||||
### Running Specific Test Files
|
### Running Specific Test Files
|
||||||
@@ -532,7 +532,7 @@ python -m mcp_server.server
|
|||||||
|
|
||||||
After completing all tests, verify:
|
After completing all tests, verify:
|
||||||
|
|
||||||
- ✅ All 42 unit tests pass
|
- ✅ All 64 unit tests pass
|
||||||
- ✅ MCP server starts without errors
|
- ✅ MCP server starts without errors
|
||||||
- ✅ Configuration loads correctly
|
- ✅ Configuration loads correctly
|
||||||
- ✅ Gitea API client connects successfully
|
- ✅ Gitea API client connects successfully
|
||||||
@@ -548,7 +548,7 @@ After completing all tests, verify:
|
|||||||
|
|
||||||
Phase 1 is complete when:
|
Phase 1 is complete when:
|
||||||
|
|
||||||
1. **All unit tests pass** (42/42)
|
1. **All unit tests pass** (64/64)
|
||||||
2. **MCP server starts without errors**
|
2. **MCP server starts without errors**
|
||||||
3. **Can list issues from Gitea**
|
3. **Can list issues from Gitea**
|
||||||
4. **Can create issues with labels** (in development mode)
|
4. **Can create issues with labels** (in development mode)
|
||||||
|
|||||||
@@ -0,0 +1,30 @@
|
|||||||
|
"""
|
||||||
|
Gitea MCP Server package.
|
||||||
|
|
||||||
|
Provides MCP tools for Gitea integration via JSON-RPC 2.0.
|
||||||
|
|
||||||
|
For external consumers (e.g., HTTP transport), use:
|
||||||
|
from mcp_server import get_tool_definitions, create_tool_dispatcher, GiteaClient
|
||||||
|
|
||||||
|
# Get tool schemas
|
||||||
|
tools = get_tool_definitions()
|
||||||
|
|
||||||
|
# Create dispatcher bound to a client
|
||||||
|
client = GiteaClient()
|
||||||
|
dispatch = create_tool_dispatcher(client)
|
||||||
|
result = await dispatch("list_issues", {"state": "open"})
|
||||||
|
"""
|
||||||
|
|
||||||
|
__version__ = "1.0.0"
|
||||||
|
|
||||||
|
from .tool_registry import get_tool_definitions, create_tool_dispatcher
|
||||||
|
from .gitea_client import GiteaClient
|
||||||
|
from .config import GiteaConfig
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"__version__",
|
||||||
|
"get_tool_definitions",
|
||||||
|
"create_tool_dispatcher",
|
||||||
|
"GiteaClient",
|
||||||
|
"GiteaConfig",
|
||||||
|
]
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
1098
mcp-servers/gitea/mcp_server/tool_registry.py
Normal file
1098
mcp-servers/gitea/mcp_server/tool_registry.py
Normal file
File diff suppressed because it is too large
Load Diff
43
mcp-servers/gitea/pyproject.toml
Normal file
43
mcp-servers/gitea/pyproject.toml
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
[build-system]
|
||||||
|
requires = ["setuptools>=61.0", "wheel"]
|
||||||
|
build-backend = "setuptools.build_meta"
|
||||||
|
|
||||||
|
[project]
|
||||||
|
name = "gitea-mcp-server"
|
||||||
|
version = "1.0.0"
|
||||||
|
description = "MCP Server for Gitea integration - provides issue, label, wiki, milestone, dependency, and PR tools"
|
||||||
|
readme = "README.md"
|
||||||
|
requires-python = ">=3.10"
|
||||||
|
license = {text = "MIT"}
|
||||||
|
authors = [
|
||||||
|
{ name = "Leo Miranda" }
|
||||||
|
]
|
||||||
|
keywords = ["mcp", "gitea", "claude", "tools"]
|
||||||
|
classifiers = [
|
||||||
|
"Development Status :: 4 - Beta",
|
||||||
|
"Intended Audience :: Developers",
|
||||||
|
"Programming Language :: Python :: 3",
|
||||||
|
"Programming Language :: Python :: 3.10",
|
||||||
|
"Programming Language :: Python :: 3.11",
|
||||||
|
"Programming Language :: Python :: 3.12",
|
||||||
|
]
|
||||||
|
dependencies = [
|
||||||
|
"mcp>=0.9.0",
|
||||||
|
"python-dotenv>=1.0.0",
|
||||||
|
"requests>=2.31.0",
|
||||||
|
"pydantic>=2.5.0",
|
||||||
|
]
|
||||||
|
|
||||||
|
[project.optional-dependencies]
|
||||||
|
test = [
|
||||||
|
"pytest>=7.4.3",
|
||||||
|
"pytest-asyncio>=0.23.0",
|
||||||
|
]
|
||||||
|
|
||||||
|
[tool.setuptools.packages.find]
|
||||||
|
where = ["."]
|
||||||
|
include = ["mcp_server*"]
|
||||||
|
|
||||||
|
[tool.pytest.ini_options]
|
||||||
|
asyncio_mode = "auto"
|
||||||
|
testpaths = ["tests"]
|
||||||
@@ -28,7 +28,6 @@ def test_load_system_config(tmp_path, monkeypatch):
|
|||||||
|
|
||||||
assert result['api_url'] == 'https://test.com/api/v1'
|
assert result['api_url'] == 'https://test.com/api/v1'
|
||||||
assert result['api_token'] == 'test_token'
|
assert result['api_token'] == 'test_token'
|
||||||
assert result['owner'] == 'test_owner'
|
|
||||||
assert result['mode'] == 'company' # No repo specified
|
assert result['mode'] == 'company' # No repo specified
|
||||||
assert result['repo'] is None
|
assert result['repo'] is None
|
||||||
|
|
||||||
|
|||||||
@@ -14,8 +14,7 @@ def mock_config():
|
|||||||
mock_instance.load.return_value = {
|
mock_instance.load.return_value = {
|
||||||
'api_url': 'https://test.com/api/v1',
|
'api_url': 'https://test.com/api/v1',
|
||||||
'api_token': 'test_token',
|
'api_token': 'test_token',
|
||||||
'owner': 'test_owner',
|
'repo': 'test_owner/test_repo', # Combined owner/repo format
|
||||||
'repo': 'test_repo',
|
|
||||||
'mode': 'project'
|
'mode': 'project'
|
||||||
}
|
}
|
||||||
yield mock_cfg
|
yield mock_cfg
|
||||||
@@ -31,8 +30,7 @@ def test_client_initialization(gitea_client):
|
|||||||
"""Test client initializes with correct configuration"""
|
"""Test client initializes with correct configuration"""
|
||||||
assert gitea_client.base_url == 'https://test.com/api/v1'
|
assert gitea_client.base_url == 'https://test.com/api/v1'
|
||||||
assert gitea_client.token == 'test_token'
|
assert gitea_client.token == 'test_token'
|
||||||
assert gitea_client.owner == 'test_owner'
|
assert gitea_client.repo == 'test_owner/test_repo' # Combined format
|
||||||
assert gitea_client.repo == 'test_repo'
|
|
||||||
assert gitea_client.mode == 'project'
|
assert gitea_client.mode == 'project'
|
||||||
assert 'Authorization' in gitea_client.session.headers
|
assert 'Authorization' in gitea_client.session.headers
|
||||||
assert gitea_client.session.headers['Authorization'] == 'token test_token'
|
assert gitea_client.session.headers['Authorization'] == 'token test_token'
|
||||||
@@ -92,15 +90,20 @@ def test_create_issue(gitea_client):
|
|||||||
}
|
}
|
||||||
mock_response.raise_for_status = Mock()
|
mock_response.raise_for_status = Mock()
|
||||||
|
|
||||||
with patch.object(gitea_client.session, 'post', return_value=mock_response):
|
# Mock is_org_repo to avoid network call during label resolution
|
||||||
issue = gitea_client.create_issue(
|
with patch.object(gitea_client, 'is_org_repo', return_value=True):
|
||||||
title='New Issue',
|
# Mock get_org_labels and get_labels for label resolution
|
||||||
body='Issue body',
|
with patch.object(gitea_client, 'get_org_labels', return_value=[{'name': 'Type/Bug', 'id': 1}]):
|
||||||
labels=['Type/Bug']
|
with patch.object(gitea_client, 'get_labels', return_value=[]):
|
||||||
)
|
with patch.object(gitea_client.session, 'post', return_value=mock_response):
|
||||||
|
issue = gitea_client.create_issue(
|
||||||
|
title='New Issue',
|
||||||
|
body='Issue body',
|
||||||
|
labels=['Type/Bug']
|
||||||
|
)
|
||||||
|
|
||||||
assert issue['title'] == 'New Issue'
|
assert issue['title'] == 'New Issue'
|
||||||
gitea_client.session.post.assert_called_once()
|
gitea_client.session.post.assert_called_once()
|
||||||
|
|
||||||
|
|
||||||
def test_update_issue(gitea_client):
|
def test_update_issue(gitea_client):
|
||||||
@@ -161,7 +164,7 @@ def test_get_org_labels(gitea_client):
|
|||||||
mock_response.raise_for_status = Mock()
|
mock_response.raise_for_status = Mock()
|
||||||
|
|
||||||
with patch.object(gitea_client.session, 'get', return_value=mock_response):
|
with patch.object(gitea_client.session, 'get', return_value=mock_response):
|
||||||
labels = gitea_client.get_org_labels()
|
labels = gitea_client.get_org_labels(org='test_owner')
|
||||||
|
|
||||||
assert len(labels) == 2
|
assert len(labels) == 2
|
||||||
|
|
||||||
@@ -176,7 +179,7 @@ def test_list_repos(gitea_client):
|
|||||||
mock_response.raise_for_status = Mock()
|
mock_response.raise_for_status = Mock()
|
||||||
|
|
||||||
with patch.object(gitea_client.session, 'get', return_value=mock_response):
|
with patch.object(gitea_client.session, 'get', return_value=mock_response):
|
||||||
repos = gitea_client.list_repos()
|
repos = gitea_client.list_repos(org='test_owner')
|
||||||
|
|
||||||
assert len(repos) == 2
|
assert len(repos) == 2
|
||||||
assert repos[0]['name'] == 'repo1'
|
assert repos[0]['name'] == 'repo1'
|
||||||
@@ -196,7 +199,7 @@ def test_aggregate_issues(gitea_client):
|
|||||||
[{'number': 2, 'title': 'Issue 2'}] # repo2
|
[{'number': 2, 'title': 'Issue 2'}] # repo2
|
||||||
])
|
])
|
||||||
|
|
||||||
aggregated = gitea_client.aggregate_issues(state='open')
|
aggregated = gitea_client.aggregate_issues(org='test_owner', state='open')
|
||||||
|
|
||||||
assert 'repo1' in aggregated
|
assert 'repo1' in aggregated
|
||||||
assert 'repo2' in aggregated
|
assert 'repo2' in aggregated
|
||||||
@@ -205,14 +208,13 @@ def test_aggregate_issues(gitea_client):
|
|||||||
|
|
||||||
|
|
||||||
def test_no_repo_specified_error(gitea_client):
|
def test_no_repo_specified_error(gitea_client):
|
||||||
"""Test error when repository not specified"""
|
"""Test error when repository not specified or invalid format"""
|
||||||
# Create client without repo
|
# Create client without repo
|
||||||
with patch('mcp_server.gitea_client.GiteaConfig') as mock_cfg:
|
with patch('mcp_server.gitea_client.GiteaConfig') as mock_cfg:
|
||||||
mock_instance = mock_cfg.return_value
|
mock_instance = mock_cfg.return_value
|
||||||
mock_instance.load.return_value = {
|
mock_instance.load.return_value = {
|
||||||
'api_url': 'https://test.com/api/v1',
|
'api_url': 'https://test.com/api/v1',
|
||||||
'api_token': 'test_token',
|
'api_token': 'test_token',
|
||||||
'owner': 'test_owner',
|
|
||||||
'repo': None, # No repo
|
'repo': None, # No repo
|
||||||
'mode': 'company'
|
'mode': 'company'
|
||||||
}
|
}
|
||||||
@@ -221,7 +223,7 @@ def test_no_repo_specified_error(gitea_client):
|
|||||||
with pytest.raises(ValueError) as exc_info:
|
with pytest.raises(ValueError) as exc_info:
|
||||||
client.list_issues()
|
client.list_issues()
|
||||||
|
|
||||||
assert "Repository not specified" in str(exc_info.value)
|
assert "Use 'owner/repo' format" in str(exc_info.value)
|
||||||
|
|
||||||
|
|
||||||
# ========================================
|
# ========================================
|
||||||
|
|||||||
@@ -119,22 +119,26 @@ async def test_aggregate_issues_company_mode(issue_tools):
|
|||||||
'repo2': [{'number': 2}]
|
'repo2': [{'number': 2}]
|
||||||
})
|
})
|
||||||
|
|
||||||
aggregated = await issue_tools.aggregate_issues()
|
aggregated = await issue_tools.aggregate_issues(org='test_owner')
|
||||||
|
|
||||||
assert 'repo1' in aggregated
|
assert 'repo1' in aggregated
|
||||||
assert 'repo2' in aggregated
|
assert 'repo2' in aggregated
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.asyncio
|
@pytest.mark.asyncio
|
||||||
async def test_aggregate_issues_project_mode_error(issue_tools):
|
async def test_aggregate_issues_project_mode(issue_tools):
|
||||||
"""Test that aggregate_issues fails in project mode"""
|
"""Test that aggregate_issues works in project mode with org argument"""
|
||||||
issue_tools.gitea.mode = 'project'
|
issue_tools.gitea.mode = 'project'
|
||||||
|
|
||||||
with patch.object(issue_tools, '_get_current_branch', return_value='development'):
|
with patch.object(issue_tools, '_get_current_branch', return_value='development'):
|
||||||
with pytest.raises(ValueError) as exc_info:
|
issue_tools.gitea.aggregate_issues = Mock(return_value={
|
||||||
await issue_tools.aggregate_issues()
|
'repo1': [{'number': 1}]
|
||||||
|
})
|
||||||
|
|
||||||
assert "only available in company mode" in str(exc_info.value)
|
# aggregate_issues now works in any mode when org is provided
|
||||||
|
aggregated = await issue_tools.aggregate_issues(org='test_owner')
|
||||||
|
|
||||||
|
assert 'repo1' in aggregated
|
||||||
|
|
||||||
|
|
||||||
def test_branch_detection():
|
def test_branch_detection():
|
||||||
|
|||||||
@@ -79,6 +79,69 @@ Add to your Claude Code MCP configuration (`~/.config/claude/mcp.json` or projec
|
|||||||
1. **System-level** (`~/.config/claude/netbox.env`): Credentials and defaults
|
1. **System-level** (`~/.config/claude/netbox.env`): Credentials and defaults
|
||||||
2. **Project-level** (`.env` in current directory): Optional overrides
|
2. **Project-level** (`.env` in current directory): Optional overrides
|
||||||
|
|
||||||
|
## Module Filtering (Token Optimization)
|
||||||
|
|
||||||
|
By default, the NetBox MCP server registers all 182 tools across 8 modules, consuming ~19,810 tokens of context. For most workflows, you only need a subset of modules.
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
Add `NETBOX_ENABLED_MODULES` to your `~/.config/claude/netbox.env`:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Enable only specific modules (comma-separated)
|
||||||
|
NETBOX_ENABLED_MODULES=dcim,ipam,virtualization,extras
|
||||||
|
```
|
||||||
|
|
||||||
|
If unset, all modules are enabled (backward compatible).
|
||||||
|
|
||||||
|
### Available Modules
|
||||||
|
|
||||||
|
| Module | Tool Count | Description | cmdb-assistant Commands |
|
||||||
|
|--------|------------|-------------|------------------------|
|
||||||
|
| `dcim` | ~60 | Sites, devices, racks, interfaces, cables | `/cmdb-device`, `/cmdb-site`, `/cmdb-search`, `/cmdb-topology` |
|
||||||
|
| `ipam` | ~40 | IP addresses, prefixes, VLANs, VRFs | `/cmdb-ip`, `/ip-conflicts`, `/cmdb-search` |
|
||||||
|
| `virtualization` | ~20 | Clusters, VMs, VM interfaces | `/cmdb-search`, `/cmdb-audit`, `/cmdb-register` |
|
||||||
|
| `extras` | ~12 | Tags, journal entries, audit log | `/change-audit`, `/cmdb-register` |
|
||||||
|
| `circuits` | ~15 | Providers, circuits, terminations | — |
|
||||||
|
| `tenancy` | ~12 | Tenants, contacts | — |
|
||||||
|
| `vpn` | ~15 | Tunnels, IKE/IPSec policies, L2VPN | — |
|
||||||
|
| `wireless` | ~8 | Wireless LANs, links, groups | — |
|
||||||
|
|
||||||
|
### Recommended Configurations
|
||||||
|
|
||||||
|
**For cmdb-assistant users** (~43 tools, ~4,500 tokens):
|
||||||
|
```bash
|
||||||
|
NETBOX_ENABLED_MODULES=dcim,ipam,virtualization,extras
|
||||||
|
```
|
||||||
|
|
||||||
|
**Basic infrastructure** (~100 tools):
|
||||||
|
```bash
|
||||||
|
NETBOX_ENABLED_MODULES=dcim,ipam
|
||||||
|
```
|
||||||
|
|
||||||
|
**Full CMDB** (all modules, ~182 tools):
|
||||||
|
```bash
|
||||||
|
# Omit NETBOX_ENABLED_MODULES or set to all modules
|
||||||
|
NETBOX_ENABLED_MODULES=dcim,ipam,circuits,virtualization,tenancy,vpn,wireless,extras
|
||||||
|
```
|
||||||
|
|
||||||
|
### Startup Logging
|
||||||
|
|
||||||
|
On startup, the server logs enabled modules and tool count:
|
||||||
|
|
||||||
|
```
|
||||||
|
NetBox MCP Server initialized: 43 tools registered (modules: dcim, extras, ipam, virtualization)
|
||||||
|
```
|
||||||
|
|
||||||
|
### Disabled Tool Behavior
|
||||||
|
|
||||||
|
Calling a tool from a disabled module returns a clear error:
|
||||||
|
|
||||||
|
```
|
||||||
|
Tool 'circuits_list_circuits' is not available (module 'circuits' not enabled).
|
||||||
|
Enabled modules: dcim, extras, ipam, virtualization
|
||||||
|
```
|
||||||
|
|
||||||
## Available Tools
|
## Available Tools
|
||||||
|
|
||||||
### DCIM (Data Center Infrastructure Management)
|
### DCIM (Data Center Infrastructure Management)
|
||||||
@@ -128,18 +191,18 @@ Add to your Claude Code MCP configuration (`~/.config/claude/mcp.json` or projec
|
|||||||
| `circuits_create_provider` | Create a provider |
|
| `circuits_create_provider` | Create a provider |
|
||||||
| `circuits_list_circuits` | List circuits |
|
| `circuits_list_circuits` | List circuits |
|
||||||
| `circuits_create_circuit` | Create a circuit |
|
| `circuits_create_circuit` | Create a circuit |
|
||||||
| `circuits_list_circuit_terminations` | List terminations |
|
| `circ_list_terminations` | List terminations |
|
||||||
| ... and more |
|
| ... and more |
|
||||||
|
|
||||||
### Virtualization
|
### Virtualization
|
||||||
|
|
||||||
| Tool | Description |
|
| Tool | Description |
|
||||||
|------|-------------|
|
|------|-------------|
|
||||||
| `virtualization_list_clusters` | List clusters |
|
| `virt_list_clusters` | List clusters |
|
||||||
| `virtualization_create_cluster` | Create a cluster |
|
| `virt_create_cluster` | Create a cluster |
|
||||||
| `virtualization_list_virtual_machines` | List VMs |
|
| `virt_list_vms` | List VMs |
|
||||||
| `virtualization_create_virtual_machine` | Create a VM |
|
| `virt_create_vm` | Create a VM |
|
||||||
| `virtualization_list_vm_interfaces` | List VM interfaces |
|
| `virt_list_vm_ifaces` | List VM interfaces |
|
||||||
| ... and more |
|
| ... and more |
|
||||||
|
|
||||||
### Tenancy
|
### Tenancy
|
||||||
@@ -167,9 +230,9 @@ Add to your Claude Code MCP configuration (`~/.config/claude/mcp.json` or projec
|
|||||||
|
|
||||||
| Tool | Description |
|
| Tool | Description |
|
||||||
|------|-------------|
|
|------|-------------|
|
||||||
| `wireless_list_wireless_lans` | List wireless LANs |
|
| `wlan_list_lans` | List wireless LANs |
|
||||||
| `wireless_create_wireless_lan` | Create a WLAN |
|
| `wlan_create_lan` | Create a WLAN |
|
||||||
| `wireless_list_wireless_links` | List wireless links |
|
| `wlan_list_links` | List wireless links |
|
||||||
| ... and more |
|
| ... and more |
|
||||||
|
|
||||||
### Extras
|
### Extras
|
||||||
|
|||||||
@@ -9,11 +9,17 @@ from pathlib import Path
|
|||||||
from dotenv import load_dotenv
|
from dotenv import load_dotenv
|
||||||
import os
|
import os
|
||||||
import logging
|
import logging
|
||||||
from typing import Dict, Optional
|
from typing import Dict, List, Optional, Set
|
||||||
|
|
||||||
logging.basicConfig(level=logging.INFO)
|
logging.basicConfig(level=logging.INFO)
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
# All available NetBox modules
|
||||||
|
ALL_MODULES = frozenset([
|
||||||
|
'dcim', 'ipam', 'circuits', 'virtualization',
|
||||||
|
'tenancy', 'vpn', 'wireless', 'extras'
|
||||||
|
])
|
||||||
|
|
||||||
|
|
||||||
class NetBoxConfig:
|
class NetBoxConfig:
|
||||||
"""Configuration loader for NetBox MCP Server"""
|
"""Configuration loader for NetBox MCP Server"""
|
||||||
@@ -23,6 +29,7 @@ class NetBoxConfig:
|
|||||||
self.api_token: Optional[str] = None
|
self.api_token: Optional[str] = None
|
||||||
self.verify_ssl: bool = True
|
self.verify_ssl: bool = True
|
||||||
self.timeout: int = 30
|
self.timeout: int = 30
|
||||||
|
self.enabled_modules: Set[str] = set(ALL_MODULES)
|
||||||
|
|
||||||
def load(self) -> Dict[str, any]:
|
def load(self) -> Dict[str, any]:
|
||||||
"""
|
"""
|
||||||
@@ -73,6 +80,9 @@ class NetBoxConfig:
|
|||||||
self.timeout = 30
|
self.timeout = 30
|
||||||
logger.warning(f"Invalid NETBOX_TIMEOUT value '{timeout_str}', using default 30")
|
logger.warning(f"Invalid NETBOX_TIMEOUT value '{timeout_str}', using default 30")
|
||||||
|
|
||||||
|
# Module filtering
|
||||||
|
self.enabled_modules = self._load_enabled_modules()
|
||||||
|
|
||||||
# Validate required variables
|
# Validate required variables
|
||||||
self._validate()
|
self._validate()
|
||||||
|
|
||||||
@@ -84,7 +94,8 @@ class NetBoxConfig:
|
|||||||
'api_url': self.api_url,
|
'api_url': self.api_url,
|
||||||
'api_token': self.api_token,
|
'api_token': self.api_token,
|
||||||
'verify_ssl': self.verify_ssl,
|
'verify_ssl': self.verify_ssl,
|
||||||
'timeout': self.timeout
|
'timeout': self.timeout,
|
||||||
|
'enabled_modules': self.enabled_modules
|
||||||
}
|
}
|
||||||
|
|
||||||
def _validate(self) -> None:
|
def _validate(self) -> None:
|
||||||
@@ -106,3 +117,40 @@ class NetBoxConfig:
|
|||||||
f"Missing required configuration: {', '.join(missing)}\n"
|
f"Missing required configuration: {', '.join(missing)}\n"
|
||||||
"Check your ~/.config/claude/netbox.env file"
|
"Check your ~/.config/claude/netbox.env file"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def _load_enabled_modules(self) -> Set[str]:
|
||||||
|
"""
|
||||||
|
Load enabled modules from NETBOX_ENABLED_MODULES environment variable.
|
||||||
|
|
||||||
|
Format: Comma-separated list of module names.
|
||||||
|
Example: NETBOX_ENABLED_MODULES=dcim,ipam,virtualization,extras
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Set of enabled module names. If env var is unset/empty, returns all modules.
|
||||||
|
"""
|
||||||
|
modules_str = os.getenv('NETBOX_ENABLED_MODULES', '').strip()
|
||||||
|
|
||||||
|
if not modules_str:
|
||||||
|
logger.info("NETBOX_ENABLED_MODULES not set, all modules enabled (default)")
|
||||||
|
return set(ALL_MODULES)
|
||||||
|
|
||||||
|
# Parse comma-separated list, strip whitespace
|
||||||
|
requested = {m.strip().lower() for m in modules_str.split(',') if m.strip()}
|
||||||
|
|
||||||
|
# Validate module names
|
||||||
|
invalid = requested - ALL_MODULES
|
||||||
|
if invalid:
|
||||||
|
logger.warning(
|
||||||
|
f"Unknown modules in NETBOX_ENABLED_MODULES: {', '.join(sorted(invalid))}. "
|
||||||
|
f"Valid modules: {', '.join(sorted(ALL_MODULES))}"
|
||||||
|
)
|
||||||
|
|
||||||
|
# Return only valid modules
|
||||||
|
enabled = requested & ALL_MODULES
|
||||||
|
|
||||||
|
if not enabled:
|
||||||
|
logger.warning("No valid modules enabled, falling back to all modules")
|
||||||
|
return set(ALL_MODULES)
|
||||||
|
|
||||||
|
logger.info(f"Enabled modules: {', '.join(sorted(enabled))}")
|
||||||
|
return enabled
|
||||||
|
|||||||
@@ -8,11 +8,12 @@ Tenancy, VPN, Wireless, and Extras.
|
|||||||
import asyncio
|
import asyncio
|
||||||
import logging
|
import logging
|
||||||
import json
|
import json
|
||||||
|
from typing import Optional, Set
|
||||||
from mcp.server import Server
|
from mcp.server import Server
|
||||||
from mcp.server.stdio import stdio_server
|
from mcp.server.stdio import stdio_server
|
||||||
from mcp.types import Tool, TextContent
|
from mcp.types import Tool, TextContent
|
||||||
|
|
||||||
from .config import NetBoxConfig
|
from .config import NetBoxConfig, ALL_MODULES
|
||||||
from .netbox_client import NetBoxClient
|
from .netbox_client import NetBoxClient
|
||||||
from .tools.dcim import DCIMTools
|
from .tools.dcim import DCIMTools
|
||||||
from .tools.ipam import IPAMTools
|
from .tools.ipam import IPAMTools
|
||||||
@@ -1453,6 +1454,49 @@ TOOL_NAME_MAP = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# Map tool name prefixes to module names.
|
||||||
|
# This handles both full prefixes and shortened prefixes used in TOOL_NAME_MAP.
|
||||||
|
PREFIX_TO_MODULE = {
|
||||||
|
'dcim': 'dcim',
|
||||||
|
'ipam': 'ipam',
|
||||||
|
'circuits': 'circuits',
|
||||||
|
'circ': 'circuits', # Shortened prefix
|
||||||
|
'virtualization': 'virtualization',
|
||||||
|
'virt': 'virtualization', # Shortened prefix
|
||||||
|
'tenancy': 'tenancy',
|
||||||
|
'vpn': 'vpn',
|
||||||
|
'wireless': 'wireless',
|
||||||
|
'wlan': 'wireless', # Shortened prefix
|
||||||
|
'extras': 'extras',
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def _get_tool_module(tool_name: str) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
Determine which module a tool belongs to.
|
||||||
|
|
||||||
|
Checks TOOL_NAME_MAP first for shortened names, then falls back to prefix extraction.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
tool_name: The tool name (e.g., 'dcim_list_devices', 'virt_list_vms')
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Module name (e.g., 'dcim', 'virtualization') or None if unknown
|
||||||
|
"""
|
||||||
|
# Check mapped short names first
|
||||||
|
if tool_name in TOOL_NAME_MAP:
|
||||||
|
category, _ = TOOL_NAME_MAP[tool_name]
|
||||||
|
return category
|
||||||
|
|
||||||
|
# Fall back to prefix extraction
|
||||||
|
parts = tool_name.split('_', 1)
|
||||||
|
if len(parts) < 2:
|
||||||
|
return None
|
||||||
|
|
||||||
|
prefix = parts[0]
|
||||||
|
return PREFIX_TO_MODULE.get(prefix)
|
||||||
|
|
||||||
|
|
||||||
class NetBoxMCPServer:
|
class NetBoxMCPServer:
|
||||||
"""MCP Server for NetBox integration"""
|
"""MCP Server for NetBox integration"""
|
||||||
|
|
||||||
@@ -1460,6 +1504,8 @@ class NetBoxMCPServer:
|
|||||||
self.server = Server("netbox-mcp")
|
self.server = Server("netbox-mcp")
|
||||||
self.config = None
|
self.config = None
|
||||||
self.client = None
|
self.client = None
|
||||||
|
self.enabled_modules: Set[str] = set(ALL_MODULES)
|
||||||
|
# Tool instances - only instantiated for enabled modules
|
||||||
self.dcim_tools = None
|
self.dcim_tools = None
|
||||||
self.ipam_tools = None
|
self.ipam_tools = None
|
||||||
self.circuits_tools = None
|
self.circuits_tools = None
|
||||||
@@ -1474,18 +1520,39 @@ class NetBoxMCPServer:
|
|||||||
try:
|
try:
|
||||||
config_loader = NetBoxConfig()
|
config_loader = NetBoxConfig()
|
||||||
self.config = config_loader.load()
|
self.config = config_loader.load()
|
||||||
|
self.enabled_modules = self.config['enabled_modules']
|
||||||
|
|
||||||
self.client = NetBoxClient()
|
self.client = NetBoxClient()
|
||||||
self.dcim_tools = DCIMTools(self.client)
|
|
||||||
self.ipam_tools = IPAMTools(self.client)
|
|
||||||
self.circuits_tools = CircuitsTools(self.client)
|
|
||||||
self.virtualization_tools = VirtualizationTools(self.client)
|
|
||||||
self.tenancy_tools = TenancyTools(self.client)
|
|
||||||
self.vpn_tools = VPNTools(self.client)
|
|
||||||
self.wireless_tools = WirelessTools(self.client)
|
|
||||||
self.extras_tools = ExtrasTools(self.client)
|
|
||||||
|
|
||||||
logger.info(f"NetBox MCP Server initialized for {self.config['api_url']}")
|
# Conditionally instantiate tool classes for enabled modules only
|
||||||
|
if 'dcim' in self.enabled_modules:
|
||||||
|
self.dcim_tools = DCIMTools(self.client)
|
||||||
|
if 'ipam' in self.enabled_modules:
|
||||||
|
self.ipam_tools = IPAMTools(self.client)
|
||||||
|
if 'circuits' in self.enabled_modules:
|
||||||
|
self.circuits_tools = CircuitsTools(self.client)
|
||||||
|
if 'virtualization' in self.enabled_modules:
|
||||||
|
self.virtualization_tools = VirtualizationTools(self.client)
|
||||||
|
if 'tenancy' in self.enabled_modules:
|
||||||
|
self.tenancy_tools = TenancyTools(self.client)
|
||||||
|
if 'vpn' in self.enabled_modules:
|
||||||
|
self.vpn_tools = VPNTools(self.client)
|
||||||
|
if 'wireless' in self.enabled_modules:
|
||||||
|
self.wireless_tools = WirelessTools(self.client)
|
||||||
|
if 'extras' in self.enabled_modules:
|
||||||
|
self.extras_tools = ExtrasTools(self.client)
|
||||||
|
|
||||||
|
# Count tools that will be registered
|
||||||
|
tool_count = sum(
|
||||||
|
1 for name in TOOL_DEFINITIONS
|
||||||
|
if _get_tool_module(name) in self.enabled_modules
|
||||||
|
)
|
||||||
|
|
||||||
|
modules_str = ', '.join(sorted(self.enabled_modules))
|
||||||
|
logger.info(
|
||||||
|
f"NetBox MCP Server initialized: {tool_count} tools registered "
|
||||||
|
f"(modules: {modules_str})"
|
||||||
|
)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.error(f"Failed to initialize: {e}")
|
logger.error(f"Failed to initialize: {e}")
|
||||||
raise
|
raise
|
||||||
@@ -1495,9 +1562,14 @@ class NetBoxMCPServer:
|
|||||||
|
|
||||||
@self.server.list_tools()
|
@self.server.list_tools()
|
||||||
async def list_tools() -> list[Tool]:
|
async def list_tools() -> list[Tool]:
|
||||||
"""Return list of available tools"""
|
"""Return list of available tools, filtered by enabled modules"""
|
||||||
tools = []
|
tools = []
|
||||||
for name, definition in TOOL_DEFINITIONS.items():
|
for name, definition in TOOL_DEFINITIONS.items():
|
||||||
|
# Filter tools by enabled modules
|
||||||
|
module = _get_tool_module(name)
|
||||||
|
if module not in self.enabled_modules:
|
||||||
|
continue
|
||||||
|
|
||||||
tools.append(Tool(
|
tools.append(Tool(
|
||||||
name=name,
|
name=name,
|
||||||
description=definition['description'],
|
description=definition['description'],
|
||||||
@@ -1532,6 +1604,14 @@ class NetBoxMCPServer:
|
|||||||
'virtualization_list_virtual_machines') to meet the 28-character
|
'virtualization_list_virtual_machines') to meet the 28-character
|
||||||
limit. TOOL_NAME_MAP handles the translation to actual method names.
|
limit. TOOL_NAME_MAP handles the translation to actual method names.
|
||||||
"""
|
"""
|
||||||
|
# Check module is enabled (routing guard)
|
||||||
|
module = _get_tool_module(name)
|
||||||
|
if module and module not in self.enabled_modules:
|
||||||
|
raise ValueError(
|
||||||
|
f"Tool '{name}' is not available (module '{module}' not enabled). "
|
||||||
|
f"Enabled modules: {', '.join(sorted(self.enabled_modules))}"
|
||||||
|
)
|
||||||
|
|
||||||
# Check if this is a mapped short name
|
# Check if this is a mapped short name
|
||||||
if name in TOOL_NAME_MAP:
|
if name in TOOL_NAME_MAP:
|
||||||
category, method_name = TOOL_NAME_MAP[name]
|
category, method_name = TOOL_NAME_MAP[name]
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: clarity-coach
|
name: clarity-coach
|
||||||
description: Patient, structured coach helping users articulate requirements clearly. Uses neurodivergent-friendly communication patterns.
|
description: Patient, structured coach helping users articulate requirements clearly. Uses neurodivergent-friendly communication patterns.
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: default
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Clarity Coach Agent
|
# Clarity Coach Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: maintainer
|
name: maintainer
|
||||||
description: CLAUDE.md optimization and maintenance agent
|
description: CLAUDE.md optimization and maintenance agent
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: acceptEdits
|
||||||
|
skills: visual-header, settings-optimization
|
||||||
---
|
---
|
||||||
|
|
||||||
# CLAUDE.md Maintainer Agent
|
# CLAUDE.md Maintainer Agent
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
name: cmdb-assistant
|
name: cmdb-assistant
|
||||||
description: Infrastructure management assistant specialized in NetBox CMDB operations. Use for device management, IP addressing, and infrastructure queries.
|
description: Infrastructure management assistant specialized in NetBox CMDB operations. Use for device management, IP addressing, and infrastructure queries.
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: default
|
||||||
---
|
---
|
||||||
|
|
||||||
# CMDB Assistant Agent
|
# CMDB Assistant Agent
|
||||||
|
|||||||
@@ -32,9 +32,9 @@ The following NetBox MCP tools are available for infrastructure management:
|
|||||||
- `ipam_list_available_ips`, `ipam_create_available_ip` - IP allocation
|
- `ipam_list_available_ips`, `ipam_create_available_ip` - IP allocation
|
||||||
|
|
||||||
**Virtualization:**
|
**Virtualization:**
|
||||||
- `virtualization_list_virtual_machines`, `virtualization_create_virtual_machine` - VM management
|
- `virt_list_vms`, `virt_create_vm`, `virt_update_vm`, `virt_delete_vm` - VM management
|
||||||
- `virtualization_list_clusters`, `virtualization_create_cluster` - Cluster management
|
- `virt_list_clusters`, `virt_create_cluster`, `virt_update_cluster`, `virt_delete_cluster` - Cluster management
|
||||||
- `virtualization_list_vm_interfaces` - VM interface management
|
- `virt_list_vm_ifaces`, `virt_create_vm_iface` - VM interface management
|
||||||
|
|
||||||
**Circuits:**
|
**Circuits:**
|
||||||
- `circuits_list_circuits`, `circuits_create_circuit` - Circuit management
|
- `circuits_list_circuits`, `circuits_create_circuit` - Circuit management
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ When the user provides a search query, determine the best approach:
|
|||||||
3. **Site search**: Use `dcim_list_sites` with name filter
|
3. **Site search**: Use `dcim_list_sites` with name filter
|
||||||
4. **Prefix search**: Use `ipam_list_prefixes` with prefix or within filter
|
4. **Prefix search**: Use `ipam_list_prefixes` with prefix or within filter
|
||||||
5. **VLAN search**: Use `ipam_list_vlans` with vid or name filter
|
5. **VLAN search**: Use `ipam_list_vlans` with vid or name filter
|
||||||
6. **VM search**: Use `virtualization_list_virtual_machines` with name filter
|
6. **VM search**: Use `virt_list_vms` with name filter
|
||||||
|
|
||||||
For broad searches, query multiple endpoints and consolidate results.
|
For broad searches, query multiple endpoints and consolidate results.
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
name: refactor-advisor
|
name: refactor-advisor
|
||||||
description: Code structure and refactoring specialist. Use when analyzing code quality, design patterns, or planning refactoring work.
|
description: Code structure and refactoring specialist. Use when analyzing code quality, design patterns, or planning refactoring work.
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: acceptEdits
|
||||||
---
|
---
|
||||||
|
|
||||||
# Refactor Advisor Agent
|
# Refactor Advisor Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: security-reviewer
|
name: security-reviewer
|
||||||
description: Security-focused code review agent
|
description: Security-focused code review agent
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: plan
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Security Reviewer Agent
|
# Security Reviewer Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: agent-check
|
name: agent-check
|
||||||
description: Agent definition validator for quick verification
|
description: Agent definition validator for quick verification
|
||||||
model: haiku
|
model: haiku
|
||||||
|
permissionMode: plan
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Agent Check Agent
|
# Agent Check Agent
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
name: full-validation
|
name: full-validation
|
||||||
description: Contract validation specialist for comprehensive cross-plugin compatibility validation of the entire marketplace.
|
description: Contract validation specialist for comprehensive cross-plugin compatibility validation of the entire marketplace.
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: default
|
||||||
---
|
---
|
||||||
|
|
||||||
# Full Validation Agent
|
# Full Validation Agent
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
name: data-advisor
|
name: data-advisor
|
||||||
description: Reviews code for data integrity, schema validity, and dbt compliance using data-platform MCP tools. Use when validating database operations or data pipelines.
|
description: Reviews code for data integrity, schema validity, and dbt compliance using data-platform MCP tools. Use when validating database operations or data pipelines.
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: default
|
||||||
---
|
---
|
||||||
|
|
||||||
# Data Advisor Agent
|
# Data Advisor Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: data-analysis
|
name: data-analysis
|
||||||
description: Data analysis specialist for exploration and profiling
|
description: Data analysis specialist for exploration and profiling
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: plan
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Data Analysis Agent
|
# Data Analysis Agent
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
name: data-ingestion
|
name: data-ingestion
|
||||||
description: Data ingestion specialist for loading, transforming, and preparing data for analysis.
|
description: Data ingestion specialist for loading, transforming, and preparing data for analysis.
|
||||||
model: haiku
|
model: haiku
|
||||||
|
permissionMode: acceptEdits
|
||||||
---
|
---
|
||||||
|
|
||||||
# Data Ingestion Agent
|
# Data Ingestion Agent
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
name: doc-analyzer
|
name: doc-analyzer
|
||||||
description: Specialized agent for documentation analysis and drift detection. Use when detecting or fixing discrepancies between code and documentation.
|
description: Specialized agent for documentation analysis and drift detection. Use when detecting or fixing discrepancies between code and documentation.
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: acceptEdits
|
||||||
---
|
---
|
||||||
|
|
||||||
# Documentation Analyzer Agent
|
# Documentation Analyzer Agent
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
name: git-assistant
|
name: git-assistant
|
||||||
description: Git workflow assistant for complex git operations, conflict resolution, and repository history management.
|
description: Git workflow assistant for complex git operations, conflict resolution, and repository history management.
|
||||||
model: haiku
|
model: haiku
|
||||||
|
permissionMode: acceptEdits
|
||||||
---
|
---
|
||||||
|
|
||||||
# Git Assistant Agent
|
# Git Assistant Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: coordinator
|
name: coordinator
|
||||||
description: Review coordinator that orchestrates the multi-agent PR review process. Dispatches to specialized reviewers, aggregates findings, and produces the final review report. Use proactively after code changes.
|
description: Review coordinator that orchestrates the multi-agent PR review process. Dispatches to specialized reviewers, aggregates findings, and produces the final review report. Use proactively after code changes.
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: plan
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Coordinator Agent
|
# Coordinator Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: maintainability-auditor
|
name: maintainability-auditor
|
||||||
description: Identifies code complexity, duplication, naming issues, and architecture concerns in PR changes.
|
description: Identifies code complexity, duplication, naming issues, and architecture concerns in PR changes.
|
||||||
model: haiku
|
model: haiku
|
||||||
|
permissionMode: plan
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Maintainability Auditor Agent
|
# Maintainability Auditor Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: performance-analyst
|
name: performance-analyst
|
||||||
description: Performance-focused code reviewer that identifies performance issues, inefficiencies, and optimization opportunities.
|
description: Performance-focused code reviewer that identifies performance issues, inefficiencies, and optimization opportunities.
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: plan
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Performance Analyst Agent
|
# Performance Analyst Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: security-reviewer
|
name: security-reviewer
|
||||||
description: Security-focused code reviewer for PR analysis
|
description: Security-focused code reviewer for PR analysis
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: plan
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Security Reviewer Agent
|
# Security Reviewer Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: test-validator
|
name: test-validator
|
||||||
description: Test quality reviewer that validates test coverage, test quality, and testing practices in PR changes.
|
description: Test quality reviewer that validates test coverage, test quality, and testing practices in PR changes.
|
||||||
model: haiku
|
model: haiku
|
||||||
|
permissionMode: plan
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Test Validator Agent
|
# Test Validator Agent
|
||||||
|
|||||||
@@ -1,20 +1,16 @@
|
|||||||
---
|
---
|
||||||
name: code-reviewer
|
name: code-reviewer
|
||||||
description: Pre-sprint code quality review agent
|
description: Pre-sprint code quality review agent
|
||||||
model: sonnet
|
model: opus
|
||||||
|
permissionMode: default
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
|
skills: review-checklist, test-standards, sprint-lifecycle, visual-output
|
||||||
---
|
---
|
||||||
|
|
||||||
# Code Reviewer Agent
|
# Code Reviewer Agent
|
||||||
|
|
||||||
You are the **Code Reviewer Agent** - a thorough, practical reviewer who ensures code quality before sprint close.
|
You are the **Code Reviewer Agent** - a thorough, practical reviewer who ensures code quality before sprint close.
|
||||||
|
|
||||||
## Skills to Load
|
|
||||||
|
|
||||||
- skills/review-checklist.md
|
|
||||||
- skills/test-standards.md
|
|
||||||
- skills/sprint-lifecycle.md
|
|
||||||
- skills/visual-output.md
|
|
||||||
|
|
||||||
## Your Personality
|
## Your Personality
|
||||||
|
|
||||||
**Thorough but Practical:**
|
**Thorough but Practical:**
|
||||||
|
|||||||
@@ -2,22 +2,14 @@
|
|||||||
name: executor
|
name: executor
|
||||||
description: Implementation executor agent - precise implementation guidance and code quality
|
description: Implementation executor agent - precise implementation guidance and code quality
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: bypassPermissions
|
||||||
|
skills: mcp-tools-reference, branch-security, git-workflow, progress-tracking, runaway-detection, lessons-learned, visual-output
|
||||||
---
|
---
|
||||||
|
|
||||||
# Implementation Executor Agent
|
# Implementation Executor Agent
|
||||||
|
|
||||||
You are the **Executor Agent** - an implementation-focused specialist who writes clean code and ensures quality.
|
You are the **Executor Agent** - an implementation-focused specialist who writes clean code and ensures quality.
|
||||||
|
|
||||||
## Skills to Load
|
|
||||||
|
|
||||||
- skills/mcp-tools-reference.md
|
|
||||||
- skills/branch-security.md
|
|
||||||
- skills/git-workflow.md
|
|
||||||
- skills/progress-tracking.md
|
|
||||||
- skills/runaway-detection.md
|
|
||||||
- skills/lessons-learned.md
|
|
||||||
- skills/visual-output.md
|
|
||||||
|
|
||||||
## Your Personality
|
## Your Personality
|
||||||
|
|
||||||
**Implementation-Focused:**
|
**Implementation-Focused:**
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
name: orchestrator
|
name: orchestrator
|
||||||
description: Sprint orchestration agent - coordinates execution and tracks progress
|
description: Sprint orchestration agent - coordinates execution and tracks progress
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: acceptEdits
|
||||||
---
|
---
|
||||||
|
|
||||||
# Sprint Orchestration Agent
|
# Sprint Orchestration Agent
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
---
|
---
|
||||||
name: planner
|
name: planner
|
||||||
description: Sprint planning agent - thoughtful architecture analysis and issue creation
|
description: Sprint planning agent - thoughtful architecture analysis and issue creation
|
||||||
model: sonnet
|
model: opus
|
||||||
|
permissionMode: default
|
||||||
---
|
---
|
||||||
|
|
||||||
# Sprint Planning Agent
|
# Sprint Planning Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: component-check
|
name: component-check
|
||||||
description: DMC component validation specialist
|
description: DMC component validation specialist
|
||||||
model: haiku
|
model: haiku
|
||||||
|
permissionMode: plan
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Component Check Agent
|
# Component Check Agent
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
name: design-reviewer
|
name: design-reviewer
|
||||||
description: Reviews code for design system compliance using viz-platform MCP tools. Use when validating DMC components, theme tokens, or accessibility standards.
|
description: Reviews code for design system compliance using viz-platform MCP tools. Use when validating DMC components, theme tokens, or accessibility standards.
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: plan
|
||||||
|
disallowedTools: Write, Edit, MultiEdit
|
||||||
---
|
---
|
||||||
|
|
||||||
# Design Reviewer Agent
|
# Design Reviewer Agent
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
name: layout-builder
|
name: layout-builder
|
||||||
description: Practical dashboard layout specialist for creating well-structured layouts with filtering, grid systems, and responsive design.
|
description: Practical dashboard layout specialist for creating well-structured layouts with filtering, grid systems, and responsive design.
|
||||||
model: sonnet
|
model: sonnet
|
||||||
|
permissionMode: default
|
||||||
---
|
---
|
||||||
|
|
||||||
# Layout Builder Agent
|
# Layout Builder Agent
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
name: theme-setup
|
name: theme-setup
|
||||||
description: Design-focused theme setup specialist for creating consistent, brand-aligned themes for Dash Mantine Components applications.
|
description: Design-focused theme setup specialist for creating consistent, brand-aligned themes for Dash Mantine Components applications.
|
||||||
model: haiku
|
model: haiku
|
||||||
|
permissionMode: acceptEdits
|
||||||
---
|
---
|
||||||
|
|
||||||
# Theme Setup Agent
|
# Theme Setup Agent
|
||||||
|
|||||||
Reference in New Issue
Block a user