nia-mcp-server 1.0.11__tar.gz → 1.0.12__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.

Potentially problematic release.


This version of nia-mcp-server might be problematic. Click here for more details.

Files changed (20) hide show
  1. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/PKG-INFO +1 -1
  2. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/pyproject.toml +1 -1
  3. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/__init__.py +1 -1
  4. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/api_client.py +29 -3
  5. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/server.py +225 -30
  6. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/.gitignore +0 -0
  7. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/ARCHITECTURE.md +0 -0
  8. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/LICENSE +0 -0
  9. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/README.md +0 -0
  10. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/nia_analytics.log +0 -0
  11. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/nia_mcp_server.log +0 -0
  12. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/__main__.py +0 -0
  13. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/assets/rules/claude_rules.md +0 -0
  14. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/assets/rules/cursor_rules.md +0 -0
  15. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/assets/rules/nia_rules.md +0 -0
  16. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/assets/rules/vscode_rules.md +0 -0
  17. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/assets/rules/windsurf_rules.md +0 -0
  18. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/profiles.py +0 -0
  19. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/project_init.py +0 -0
  20. {nia_mcp_server-1.0.11 → nia_mcp_server-1.0.12}/src/nia_mcp_server/rule_transformer.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: nia-mcp-server
3
- Version: 1.0.11
3
+ Version: 1.0.12
4
4
  Summary: Nia Knowledge Agent
5
5
  Project-URL: Homepage, https://trynia.ai
6
6
  Project-URL: Documentation, https://docs.trynia.ai
@@ -4,7 +4,7 @@ build-backend = "hatchling.build"
4
4
 
5
5
  [project]
6
6
  name = "nia-mcp-server"
7
- version = "1.0.11"
7
+ version = "1.0.12"
8
8
  description = "Nia Knowledge Agent"
9
9
  readme = "README.md"
10
10
  requires-python = ">=3.8"
@@ -2,4 +2,4 @@
2
2
  NIA MCP Server - Proxy server for NIA Knowledge Agent
3
3
  """
4
4
 
5
- __version__ = "1.0.11"
5
+ __version__ = "1.0.12"
@@ -75,7 +75,7 @@ class NIAApiClient:
75
75
  "lifetime limit",
76
76
  "no chat credits",
77
77
  "free api requests",
78
- "25 free",
78
+ "5 free",
79
79
  "usage limit",
80
80
  ]
81
81
  ):
@@ -99,7 +99,7 @@ class NIAApiClient:
99
99
  for phrase in [
100
100
  "lifetime limit",
101
101
  "free api requests",
102
- "25 free",
102
+ "5 free",
103
103
  "usage limit",
104
104
  ]
105
105
  ):
@@ -623,4 +623,30 @@ class NIAApiClient:
623
623
  except httpx.HTTPStatusError as e:
624
624
  raise self._handle_api_error(e)
625
625
  except Exception as e:
626
- raise APIError(f"Deep research failed: {str(e)}")
626
+ raise APIError(f"Deep research failed: {str(e)}")
627
+
628
+ async def get_source_content(
629
+ self,
630
+ source_type: str,
631
+ source_identifier: str,
632
+ metadata: Dict[str, Any] = None
633
+ ) -> Dict[str, Any]:
634
+ """Get full content of a specific source file or document."""
635
+ try:
636
+ payload = {
637
+ "source_type": source_type,
638
+ "source_identifier": source_identifier,
639
+ "metadata": metadata or {}
640
+ }
641
+
642
+ response = await self.client.post(
643
+ f"{self.base_url}/v2/sources/content",
644
+ json=payload
645
+ )
646
+ response.raise_for_status()
647
+ return response.json()
648
+
649
+ except httpx.HTTPStatusError as e:
650
+ raise self._handle_api_error(e)
651
+ except Exception as e:
652
+ raise APIError(f"Failed to get source content: {str(e)}")
@@ -25,7 +25,7 @@ load_dotenv(env_path)
25
25
 
26
26
  # Configure logging
27
27
  logging.basicConfig(
28
- level=logging.INFO,
28
+ level=logging.INFO, # Changed from INFO to DEBUG for troubleshooting
29
29
  format='%(asctime)s [%(levelname)s] %(name)s: %(message)s',
30
30
  )
31
31
  logger = logging.getLogger(__name__)
@@ -122,7 +122,7 @@ async def index_repository(
122
122
  except APIError as e:
123
123
  logger.error(f"API Error indexing repository: {e} (status_code={e.status_code}, detail={e.detail})")
124
124
  if e.status_code == 403 or "free tier limit" in str(e).lower() or "free api requests" in str(e).lower():
125
- if e.detail and "25 free API requests" in e.detail:
125
+ if e.detail and "5 free API requests" in e.detail:
126
126
  return [TextContent(
127
127
  type="text",
128
128
  text=f"❌ {e.detail}\n\n💡 Tip: Upgrade to Pro at https://trynia.ai/billing for unlimited API access."
@@ -222,9 +222,11 @@ async def search_codebase(
222
222
  response_parts.append(data["content"])
223
223
 
224
224
  if "sources" in data and data["sources"]:
225
+ logger.debug(f"Received sources data: {type(data['sources'])}, count: {len(data['sources'])}")
225
226
  sources_parts.extend(data["sources"])
226
227
 
227
- except json.JSONDecodeError:
228
+ except json.JSONDecodeError as e:
229
+ logger.warning(f"Failed to parse JSON chunk: {chunk}, error: {e}")
228
230
  continue
229
231
 
230
232
  # Format the response
@@ -232,21 +234,53 @@ async def search_codebase(
232
234
 
233
235
  if sources_parts and include_sources:
234
236
  response_text += "\n\n## Sources\n\n"
235
- for i, source in enumerate(sources_parts[:5], 1): # Limit to 5 sources
237
+ for i, source in enumerate(sources_parts[:10], 1): # Limit to 10 sources (matches backend)
236
238
  response_text += f"### Source {i}\n"
237
- if "repository" in source:
238
- response_text += f"**Repository:** {source['repository']}\n"
239
- if "file" in source:
240
- response_text += f"**File:** `{source['file']}`\n"
241
- if "preview" in source:
242
- response_text += f"```\n{source['preview']}\n```\n\n"
239
+
240
+ # Handle both string sources (file paths) and dictionary sources
241
+ if isinstance(source, str):
242
+ # Source is just a file path string
243
+ response_text += f"**File:** `{source}`\n\n"
244
+ continue
245
+ elif not isinstance(source, dict):
246
+ logger.warning(f"Expected source to be dict or str, got {type(source)}: {source}")
247
+ response_text += f"**Source:** {str(source)}\n\n"
248
+ continue
249
+
250
+ # Handle dictionary sources with metadata
251
+ metadata = source.get("metadata", {})
252
+
253
+ # Repository name
254
+ repository = source.get("repository") or metadata.get("source_name") or metadata.get("repository")
255
+ if repository:
256
+ response_text += f"**Repository:** {repository}\n"
257
+
258
+ # File path
259
+ file_path = source.get("file") or source.get("file_path") or metadata.get("file_path")
260
+ if file_path:
261
+ response_text += f"**File:** `{file_path}`\n"
262
+
263
+ # Content/preview
264
+ content = source.get("preview") or source.get("content")
265
+ if content:
266
+ # Truncate very long content
267
+ if len(content) > 500:
268
+ content = content[:500] + "..."
269
+ response_text += f"```\n{content}\n```\n\n"
270
+ else:
271
+ # If no content, at least show that this is a valid source
272
+ response_text += f"*Referenced source*\n\n"
273
+
274
+ # Add helpful text about read_source_content tool
275
+ response_text += "\n💡 **Need more details from a source?**\n\n"
276
+ response_text += "If you need more information from the source links provided above, use the `read_source_content` tool from the available tools provided by Nia to get full context about that particular source.\n"
243
277
 
244
278
  return [TextContent(type="text", text=response_text)]
245
279
 
246
280
  except APIError as e:
247
281
  logger.error(f"API Error searching codebase: {e} (status_code={e.status_code}, detail={e.detail})")
248
282
  if e.status_code == 403 or "free tier limit" in str(e).lower() or "free api requests" in str(e).lower():
249
- if e.detail and "25 free API requests" in e.detail:
283
+ if e.detail and "5 free API requests" in e.detail:
250
284
  return [TextContent(
251
285
  type="text",
252
286
  text=f"❌ {e.detail}\n\n💡 Tip: Upgrade to Pro at https://trynia.ai/billing for unlimited API access."
@@ -287,6 +321,9 @@ async def search_documentation(
287
321
 
288
322
  Returns:
289
323
  Search results with relevant documentation excerpts
324
+
325
+ Important:
326
+ - Always use Source ID. If you don't have it, use `list_documentation` tool to get it.
290
327
  """
291
328
  try:
292
329
  client = await ensure_api_client()
@@ -316,7 +353,7 @@ async def search_documentation(
316
353
  messages=messages,
317
354
  repositories=[], # No repositories
318
355
  data_sources=sources,
319
- search_mode="unified", # Use unified for full answers with documentation
356
+ search_mode="unified", # Use unified mode for intelligent LLM processing
320
357
  stream=True,
321
358
  include_sources=include_sources
322
359
  ):
@@ -327,9 +364,11 @@ async def search_documentation(
327
364
  response_parts.append(data["content"])
328
365
 
329
366
  if "sources" in data and data["sources"]:
367
+ logger.debug(f"Received doc sources data: {type(data['sources'])}, count: {len(data['sources'])}")
330
368
  sources_parts.extend(data["sources"])
331
369
 
332
- except json.JSONDecodeError:
370
+ except json.JSONDecodeError as e:
371
+ logger.warning(f"Failed to parse JSON chunk in documentation search: {chunk}, error: {e}")
333
372
  continue
334
373
 
335
374
  # Format the response
@@ -337,14 +376,42 @@ async def search_documentation(
337
376
 
338
377
  if sources_parts and include_sources:
339
378
  response_text += "\n\n## Sources\n\n"
340
- for i, source in enumerate(sources_parts[:5], 1): # Limit to 5 sources
379
+ for i, source in enumerate(sources_parts[:10], 1): # Limit to 10 sources (matches backend)
341
380
  response_text += f"### Source {i}\n"
342
- if "url" in source:
343
- response_text += f"**URL:** {source['url']}\n"
344
- elif "file" in source:
345
- response_text += f"**Page:** {source['file']}\n"
346
- if "preview" in source:
347
- response_text += f"```\n{source['preview']}\n```\n\n"
381
+
382
+ # Handle both string sources and dictionary sources
383
+ if isinstance(source, str):
384
+ # Source is just a URL or file path string
385
+ response_text += f"**Document:** {source}\n\n"
386
+ continue
387
+ elif not isinstance(source, dict):
388
+ logger.warning(f"Expected source to be dict or str, got {type(source)}: {source}")
389
+ response_text += f"**Source:** {str(source)}\n\n"
390
+ continue
391
+
392
+ # Handle dictionary sources with metadata
393
+ metadata = source.get("metadata", {})
394
+
395
+ # URL or file
396
+ url = source.get("url") or metadata.get("url") or metadata.get("source") or metadata.get("sourceURL")
397
+ file_path = source.get("file") or source.get("file_path") or metadata.get("file_path") or metadata.get("document_name")
398
+
399
+ if url:
400
+ response_text += f"**URL:** {url}\n"
401
+ elif file_path:
402
+ response_text += f"**Document:** {file_path}\n"
403
+
404
+ # Title if available
405
+ title = source.get("title") or metadata.get("title")
406
+ if title:
407
+ response_text += f"**Title:** {title}\n"
408
+
409
+ # Add spacing after each source
410
+ response_text += "\n"
411
+
412
+ # Add helpful text about read_source_content tool
413
+ response_text += "\n💡 **Need more details from a source?**\n\n"
414
+ response_text += "If you need more information from the source links provided above, use the `read_source_content` tool from the available tools provided by Nia to get full context about that particular source.\n"
348
415
 
349
416
  return [TextContent(type="text", text=response_text)]
350
417
 
@@ -352,7 +419,7 @@ async def search_documentation(
352
419
  logger.error(f"API Error searching documentation: {e}")
353
420
  error_msg = f"❌ {str(e)}"
354
421
  if e.status_code == 403 and "lifetime limit" in str(e).lower():
355
- error_msg += "\n\n💡 Tip: You've reached the free tier limit of 25 API requests. Upgrade to Pro for unlimited access."
422
+ error_msg += "\n\n💡 Tip: You've reached the free tier limit of 5 API requests. Upgrade to Pro for unlimited access."
356
423
  return [TextContent(type="text", text=error_msg)]
357
424
  except Exception as e:
358
425
  logger.error(f"Error searching documentation: {e}")
@@ -411,7 +478,7 @@ async def list_repositories() -> List[TextContent]:
411
478
  # Check for free tier limit errors
412
479
  if e.status_code == 403 or "free tier limit" in str(e).lower() or "free api requests" in str(e).lower():
413
480
  # Extract the specific limit message
414
- if e.detail and "25 free API requests" in e.detail:
481
+ if e.detail and "5 free API requests" in e.detail:
415
482
  return [TextContent(
416
483
  type="text",
417
484
  text=f"❌ {e.detail}\n\n💡 Tip: Upgrade to Pro at https://trynia.ai/billing for unlimited API access."
@@ -491,7 +558,7 @@ async def check_repository_status(repository: str) -> List[TextContent]:
491
558
  logger.error(f"API Error checking repository status: {e}")
492
559
  error_msg = f"❌ {str(e)}"
493
560
  if e.status_code == 403 and "lifetime limit" in str(e).lower():
494
- error_msg += "\n\n💡 Tip: You've reached the free tier limit of 25 API requests. Upgrade to Pro for unlimited access."
561
+ error_msg += "\n\n💡 Tip: You've reached the free tier limit of 5 API requests. Upgrade to Pro for unlimited access."
495
562
  return [TextContent(type="text", text=error_msg)]
496
563
  except Exception as e:
497
564
  logger.error(f"Error checking repository status: {e}")
@@ -578,7 +645,7 @@ async def index_documentation(
578
645
  logger.error(f"API Error indexing documentation: {e}")
579
646
  error_msg = f"❌ {str(e)}"
580
647
  if e.status_code == 403 and "lifetime limit" in str(e).lower():
581
- error_msg += "\n\n💡 Tip: You've reached the free tier limit of 25 API requests. Upgrade to Pro for unlimited access."
648
+ error_msg += "\n\n💡 Tip: You've reached the free tier limit of 5 API requests. Upgrade to Pro for unlimited access."
582
649
  return [TextContent(type="text", text=error_msg)]
583
650
  except Exception as e:
584
651
  logger.error(f"Error indexing documentation: {e}")
@@ -637,7 +704,7 @@ async def list_documentation() -> List[TextContent]:
637
704
  logger.error(f"API Error listing documentation: {e}")
638
705
  error_msg = f"❌ {str(e)}"
639
706
  if e.status_code == 403 and "lifetime limit" in str(e).lower():
640
- error_msg += "\n\n💡 Tip: You've reached the free tier limit of 25 API requests. Upgrade to Pro for unlimited access."
707
+ error_msg += "\n\n💡 Tip: You've reached the free tier limit of 5 API requests. Upgrade to Pro for unlimited access."
641
708
  return [TextContent(type="text", text=error_msg)]
642
709
  except Exception as e:
643
710
  logger.error(f"Error listing documentation: {e}")
@@ -703,7 +770,7 @@ async def check_documentation_status(source_id: str) -> List[TextContent]:
703
770
  logger.error(f"API Error checking documentation status: {e}")
704
771
  error_msg = f"❌ {str(e)}"
705
772
  if e.status_code == 403 and "lifetime limit" in str(e).lower():
706
- error_msg += "\n\n💡 Tip: You've reached the free tier limit of 25 API requests. Upgrade to Pro for unlimited access."
773
+ error_msg += "\n\n💡 Tip: You've reached the free tier limit of 5 API requests. Upgrade to Pro for unlimited access."
707
774
  return [TextContent(type="text", text=error_msg)]
708
775
  except Exception as e:
709
776
  logger.error(f"Error checking documentation status: {e}")
@@ -742,7 +809,7 @@ async def delete_documentation(source_id: str) -> List[TextContent]:
742
809
  logger.error(f"API Error deleting documentation: {e}")
743
810
  error_msg = f"❌ {str(e)}"
744
811
  if e.status_code == 403 and "lifetime limit" in str(e).lower():
745
- error_msg += "\n\n💡 Tip: You've reached the free tier limit of 25 API requests. Upgrade to Pro for unlimited access."
812
+ error_msg += "\n\n💡 Tip: You've reached the free tier limit of 5 API requests. Upgrade to Pro for unlimited access."
746
813
  return [TextContent(type="text", text=error_msg)]
747
814
  except Exception as e:
748
815
  logger.error(f"Error deleting documentation: {e}")
@@ -781,7 +848,7 @@ async def delete_repository(repository: str) -> List[TextContent]:
781
848
  logger.error(f"API Error deleting repository: {e}")
782
849
  error_msg = f"❌ {str(e)}"
783
850
  if e.status_code == 403 and "lifetime limit" in str(e).lower():
784
- error_msg += "\n\n💡 Tip: You've reached the free tier limit of 25 API requests. Upgrade to Pro for unlimited access."
851
+ error_msg += "\n\n💡 Tip: You've reached the free tier limit of 5 API requests. Upgrade to Pro for unlimited access."
785
852
  return [TextContent(type="text", text=error_msg)]
786
853
  except Exception as e:
787
854
  logger.error(f"Error deleting repository: {e}")
@@ -1033,7 +1100,7 @@ async def nia_web_search(
1033
1100
  except APIError as e:
1034
1101
  logger.error(f"API Error in web search: {e}")
1035
1102
  if e.status_code == 403 or "free tier limit" in str(e).lower() or "free api requests" in str(e).lower():
1036
- if e.detail and "25 free API requests" in e.detail:
1103
+ if e.detail and "5 free API requests" in e.detail:
1037
1104
  return [TextContent(
1038
1105
  type="text",
1039
1106
  text=f"❌ {e.detail}\n\n💡 Tip: Upgrade to Pro at https://trynia.ai/billing for unlimited API access."
@@ -1212,7 +1279,7 @@ async def nia_deep_research_agent(
1212
1279
  except APIError as e:
1213
1280
  logger.error(f"API Error in deep research: {e}")
1214
1281
  if e.status_code == 403 or "free tier limit" in str(e).lower() or "free api requests" in str(e).lower():
1215
- if e.detail and "25 free API requests" in e.detail:
1282
+ if e.detail and "5 free API requests" in e.detail:
1216
1283
  return [TextContent(
1217
1284
  type="text",
1218
1285
  text=f"❌ {e.detail}\n\n💡 Tip: Upgrade to Pro at https://trynia.ai/billing for unlimited API access."
@@ -1388,6 +1455,134 @@ async def initialize_project(
1388
1455
  "- The NIA MCP server is properly installed"
1389
1456
  )]
1390
1457
 
1458
+ @mcp.tool()
1459
+ async def read_source_content(
1460
+ source_type: str,
1461
+ source_identifier: str,
1462
+ metadata: Optional[Dict[str, Any]] = None
1463
+ ) -> List[TextContent]:
1464
+ """
1465
+ Read the full content of a specific source file or document.
1466
+
1467
+ This tool allows AI to fetch complete content from sources identified during search,
1468
+ enabling deeper analysis when the truncated search results are insufficient.
1469
+
1470
+ Args:
1471
+ source_type: Type of source - "repository" or "documentation"
1472
+ source_identifier:
1473
+ - For repository: "owner/repo:path/to/file.py" (e.g., "facebook/react:src/React.js")
1474
+ - For documentation: The source URL or document ID
1475
+ metadata: Optional metadata from search results to help locate the source
1476
+
1477
+ Returns:
1478
+ Full content of the requested source with metadata
1479
+
1480
+ Examples:
1481
+ - read_source_content("repository", "langchain-ai/langchain:libs/core/langchain_core/runnables/base.py")
1482
+ - read_source_content("documentation", "https://docs.python.org/3/library/asyncio.html")
1483
+ """
1484
+ try:
1485
+ client = await ensure_api_client()
1486
+
1487
+ logger.info(f"Reading source content - type: {source_type}, identifier: {source_identifier}")
1488
+
1489
+ # Call the API to get source content
1490
+ result = await client.get_source_content(
1491
+ source_type=source_type,
1492
+ source_identifier=source_identifier,
1493
+ metadata=metadata or {}
1494
+ )
1495
+
1496
+ if not result or not result.get("success"):
1497
+ error_msg = result.get("error", "Unknown error") if result else "Failed to fetch source content"
1498
+ return [TextContent(
1499
+ type="text",
1500
+ text=f"❌ Error reading source: {error_msg}"
1501
+ )]
1502
+
1503
+ # Format the response
1504
+ content = result.get("content", "")
1505
+ source_metadata = result.get("metadata", {})
1506
+
1507
+ # Build response with metadata header
1508
+ response_lines = []
1509
+
1510
+ if source_type == "repository":
1511
+ repo_name = source_metadata.get("repository", "Unknown")
1512
+ file_path = source_metadata.get("file_path", source_identifier.split(":", 1)[-1] if ":" in source_identifier else "Unknown")
1513
+ branch = source_metadata.get("branch", "main")
1514
+
1515
+ response_lines.extend([
1516
+ f"# Source: {repo_name}",
1517
+ f"**File:** `{file_path}`",
1518
+ f"**Branch:** {branch}",
1519
+ ""
1520
+ ])
1521
+
1522
+ if source_metadata.get("url"):
1523
+ response_lines.append(f"**GitHub URL:** {source_metadata['url']}")
1524
+ response_lines.append("")
1525
+
1526
+ # Add file info if available
1527
+ if source_metadata.get("size"):
1528
+ response_lines.append(f"**Size:** {source_metadata['size']} bytes")
1529
+ if source_metadata.get("language"):
1530
+ response_lines.append(f"**Language:** {source_metadata['language']}")
1531
+
1532
+ response_lines.extend(["", "## Content", ""])
1533
+
1534
+ # Add code block with language hint
1535
+ language = source_metadata.get("language", "").lower() or "text"
1536
+ response_lines.append(f"```{language}")
1537
+ response_lines.append(content)
1538
+ response_lines.append("```")
1539
+
1540
+ elif source_type == "documentation":
1541
+ url = source_metadata.get("url", source_identifier)
1542
+ title = source_metadata.get("title", "Documentation")
1543
+
1544
+ response_lines.extend([
1545
+ f"# Documentation: {title}",
1546
+ f"**URL:** {url}",
1547
+ ""
1548
+ ])
1549
+
1550
+ if source_metadata.get("last_updated"):
1551
+ response_lines.append(f"**Last Updated:** {source_metadata['last_updated']}")
1552
+ response_lines.append("")
1553
+
1554
+ response_lines.extend(["## Content", "", content])
1555
+
1556
+ else:
1557
+ # Generic format for unknown source types
1558
+ response_lines.extend([
1559
+ f"# Source Content",
1560
+ f"**Type:** {source_type}",
1561
+ f"**Identifier:** {source_identifier}",
1562
+ "",
1563
+ "## Content",
1564
+ "",
1565
+ content
1566
+ ])
1567
+
1568
+ return [TextContent(type="text", text="\n".join(response_lines))]
1569
+
1570
+ except APIError as e:
1571
+ logger.error(f"API Error reading source content: {e}")
1572
+ if e.status_code == 403 or "free tier limit" in str(e).lower():
1573
+ return [TextContent(
1574
+ type="text",
1575
+ text=f"❌ {str(e)}\n\n💡 Tip: Upgrade to Pro at https://trynia.ai/billing for unlimited access."
1576
+ )]
1577
+ else:
1578
+ return [TextContent(type="text", text=f"❌ {str(e)}")]
1579
+ except Exception as e:
1580
+ logger.error(f"Error reading source content: {e}")
1581
+ return [TextContent(
1582
+ type="text",
1583
+ text=f"❌ Error reading source content: {str(e)}"
1584
+ )]
1585
+
1391
1586
  @mcp.tool()
1392
1587
  async def visualize_codebase(
1393
1588
  repository: str
File without changes