fenix-mcp 1.6.0__tar.gz → 1.8.0__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/PKG-INFO +1 -1
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/__init__.py +1 -1
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/knowledge.py +63 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/domain/knowledge.py +5 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/infrastructure/fenix_api/client.py +3 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/PKG-INFO +1 -1
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/README.md +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/presenters.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tool_base.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tool_registry.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/__init__.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/health.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/initialize.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/intelligence.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/productivity.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/user_config.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/domain/initialization.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/domain/intelligence.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/domain/productivity.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/domain/user_config.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/infrastructure/config.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/infrastructure/context.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/infrastructure/http_client.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/infrastructure/logging.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/interface/mcp_server.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/interface/transports.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/main.py +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/SOURCES.txt +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/dependency_links.txt +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/entry_points.txt +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/requires.txt +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/top_level.txt +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/pyproject.toml +0 -0
- {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/setup.cfg +0 -0
|
@@ -78,6 +78,10 @@ class KnowledgeAction(str, Enum):
|
|
|
78
78
|
"work_assign_to_me",
|
|
79
79
|
"Assigns a work item to the current user.",
|
|
80
80
|
)
|
|
81
|
+
WORK_BULK_CREATE = (
|
|
82
|
+
"work_bulk_create",
|
|
83
|
+
"Creates multiple work items atomically with hierarchy. Use temp_id as temporary identifier and parent_temp_id to reference parent in the same batch, or parent_key to reference an existing work item (e.g., TEMA-0056). Cannot use both parent_temp_id and parent_key on the same item. Example: [{temp_id:'epic-1', title:'My Epic', item_type:'epic', work_category:'backend'}, {temp_id:'task-1', parent_temp_id:'epic-1', title:'My Task', item_type:'task', work_category:'backend'}] or [{temp_id:'task-1', parent_key:'TEMA-0056', title:'My Task', item_type:'task', work_category:'backend'}]",
|
|
84
|
+
)
|
|
81
85
|
|
|
82
86
|
# Boards
|
|
83
87
|
BOARD_LIST = ("board_list", "Lists available boards with optional filters.")
|
|
@@ -260,6 +264,19 @@ class KnowledgeRequest(ToolRequest):
|
|
|
260
264
|
default=None,
|
|
261
265
|
description="List of work item IDs for batch operations (UUIDs).",
|
|
262
266
|
)
|
|
267
|
+
work_items: Optional[List[Dict[str, Any]]] = Field(
|
|
268
|
+
default=None,
|
|
269
|
+
description=(
|
|
270
|
+
"List of work items for bulk create (max 50). Each item requires: "
|
|
271
|
+
"temp_id (your temporary ID like 'epic-1', 'task-a'), title, item_type, work_category. "
|
|
272
|
+
"Use parent_temp_id to set parent from same batch (e.g., parent_temp_id:'epic-1'), "
|
|
273
|
+
"OR parent_key to set parent from existing work item (e.g., parent_key:'TEMA-0056'). "
|
|
274
|
+
"Cannot use both parent_temp_id and parent_key on the same item. "
|
|
275
|
+
"Optional: description, priority (low/medium/high/critical), story_points, tags, due_date. "
|
|
276
|
+
'Example: [{"temp_id":"e1", "title":"Epic", "item_type":"epic", "work_category":"backend"}, '
|
|
277
|
+
'{"temp_id":"t1", "parent_temp_id":"e1", "title":"Task", "item_type":"task", "work_category":"backend"}]'
|
|
278
|
+
),
|
|
279
|
+
)
|
|
263
280
|
|
|
264
281
|
# Mode fields
|
|
265
282
|
mode_id: Optional[UUIDStr] = Field(
|
|
@@ -579,6 +596,52 @@ class KnowledgeTool(Tool):
|
|
|
579
596
|
count = len(payload.work_item_ids)
|
|
580
597
|
return text(f"✅ {count} work item(s) assigned to sprint.")
|
|
581
598
|
|
|
599
|
+
if action is KnowledgeAction.WORK_BULK_CREATE:
|
|
600
|
+
if not payload.work_items:
|
|
601
|
+
return text(
|
|
602
|
+
"❌ Provide work_items array. Each item requires: temp_id, title, item_type, work_category. "
|
|
603
|
+
"Optional: parent_temp_id OR parent_key (not both), description, priority, story_points, tags, due_date."
|
|
604
|
+
)
|
|
605
|
+
if len(payload.work_items) > 50:
|
|
606
|
+
return text("❌ Maximum 50 items per bulk create.")
|
|
607
|
+
|
|
608
|
+
# Validate required fields
|
|
609
|
+
for i, item in enumerate(payload.work_items):
|
|
610
|
+
if not item.get("temp_id"):
|
|
611
|
+
return text(f"❌ Item {i}: missing temp_id.")
|
|
612
|
+
if not item.get("title"):
|
|
613
|
+
return text(f"❌ Item {i}: missing title.")
|
|
614
|
+
if not item.get("item_type"):
|
|
615
|
+
return text(f"❌ Item {i}: missing item_type.")
|
|
616
|
+
if not item.get("work_category"):
|
|
617
|
+
return text(f"❌ Item {i}: missing work_category.")
|
|
618
|
+
if item.get("parent_temp_id") and item.get("parent_key"):
|
|
619
|
+
return text(
|
|
620
|
+
f"❌ Item {i}: cannot have both parent_temp_id and parent_key. Use one or the other."
|
|
621
|
+
)
|
|
622
|
+
|
|
623
|
+
items = await self._service.work_bulk_create({"items": payload.work_items})
|
|
624
|
+
|
|
625
|
+
# Format response with summary
|
|
626
|
+
type_counts: Dict[str, int] = {}
|
|
627
|
+
for item in items:
|
|
628
|
+
item_type = item.get("item_type", "unknown")
|
|
629
|
+
type_counts[item_type] = type_counts.get(item_type, 0) + 1
|
|
630
|
+
|
|
631
|
+
lines = [f"✅ **{len(items)} work items created**", ""]
|
|
632
|
+
for item in items:
|
|
633
|
+
key = item.get("key", "")
|
|
634
|
+
title = item.get("title", "Untitled")
|
|
635
|
+
item_type = item.get("item_type", "unknown")
|
|
636
|
+
lines.append(f"- [{key}] {title} ({item_type})")
|
|
637
|
+
|
|
638
|
+
lines.append("")
|
|
639
|
+
lines.append("**Summary by type:**")
|
|
640
|
+
for item_type, count in sorted(type_counts.items()):
|
|
641
|
+
lines.append(f"- {item_type}: {count}")
|
|
642
|
+
|
|
643
|
+
return text("\n".join(lines))
|
|
644
|
+
|
|
582
645
|
return text(
|
|
583
646
|
"❌ Unsupported work item action.\n\nChoose one of the values:\n"
|
|
584
647
|
+ "\n".join(
|
|
@@ -181,6 +181,11 @@ class KnowledgeService:
|
|
|
181
181
|
async def work_bulk_update(self, payload: Dict[str, Any]) -> Dict[str, Any]:
|
|
182
182
|
return await self._call(self.api.bulk_update_work_items, _strip_none(payload))
|
|
183
183
|
|
|
184
|
+
async def work_bulk_create(self, payload: Dict[str, Any]) -> List[Dict[str, Any]]:
|
|
185
|
+
return await self._call_list(
|
|
186
|
+
self.api.bulk_create_work_items, _strip_none(payload)
|
|
187
|
+
)
|
|
188
|
+
|
|
184
189
|
# ------------------------------------------------------------------
|
|
185
190
|
# Work boards
|
|
186
191
|
# ------------------------------------------------------------------
|
|
@@ -573,6 +573,9 @@ class FenixApiClient:
|
|
|
573
573
|
def bulk_update_work_items(self, payload: Mapping[str, Any]) -> Any:
|
|
574
574
|
return self._request("PATCH", "/api/work-items/bulk-update", json=payload)
|
|
575
575
|
|
|
576
|
+
def bulk_create_work_items(self, payload: Mapping[str, Any]) -> Any:
|
|
577
|
+
return self._request("POST", "/api/work-items/bulk-create", json=payload)
|
|
578
|
+
|
|
576
579
|
# ------------------------------------------------------------------
|
|
577
580
|
# Knowledge: boards
|
|
578
581
|
# ------------------------------------------------------------------
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|