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.
Files changed (34) hide show
  1. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/PKG-INFO +1 -1
  2. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/__init__.py +1 -1
  3. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/knowledge.py +63 -0
  4. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/domain/knowledge.py +5 -0
  5. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/infrastructure/fenix_api/client.py +3 -0
  6. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/PKG-INFO +1 -1
  7. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/README.md +0 -0
  8. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/presenters.py +0 -0
  9. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tool_base.py +0 -0
  10. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tool_registry.py +0 -0
  11. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/__init__.py +0 -0
  12. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/health.py +0 -0
  13. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/initialize.py +0 -0
  14. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/intelligence.py +0 -0
  15. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/productivity.py +0 -0
  16. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/application/tools/user_config.py +0 -0
  17. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/domain/initialization.py +0 -0
  18. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/domain/intelligence.py +0 -0
  19. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/domain/productivity.py +0 -0
  20. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/domain/user_config.py +0 -0
  21. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/infrastructure/config.py +0 -0
  22. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/infrastructure/context.py +0 -0
  23. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/infrastructure/http_client.py +0 -0
  24. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/infrastructure/logging.py +0 -0
  25. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/interface/mcp_server.py +0 -0
  26. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/interface/transports.py +0 -0
  27. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp/main.py +0 -0
  28. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/SOURCES.txt +0 -0
  29. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/dependency_links.txt +0 -0
  30. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/entry_points.txt +0 -0
  31. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/requires.txt +0 -0
  32. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/fenix_mcp.egg-info/top_level.txt +0 -0
  33. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/pyproject.toml +0 -0
  34. {fenix_mcp-1.6.0 → fenix_mcp-1.8.0}/setup.cfg +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: fenix-mcp
3
- Version: 1.6.0
3
+ Version: 1.8.0
4
4
  Summary: Fênix Cloud MCP server implemented in Python
5
5
  Author: Fenix Inc
6
6
  Requires-Python: >=3.10
@@ -8,4 +8,4 @@ Fênix Cloud MCP Server (Python edition).
8
8
  __all__ = ["__version__"]
9
9
 
10
10
 
11
- __version__ = "1.6.0"
11
+ __version__ = "1.8.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
  # ------------------------------------------------------------------
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: fenix-mcp
3
- Version: 1.6.0
3
+ Version: 1.8.0
4
4
  Summary: Fênix Cloud MCP server implemented in Python
5
5
  Author: Fenix Inc
6
6
  Requires-Python: >=3.10
File without changes
File without changes
File without changes
File without changes