-
Notifications
You must be signed in to change notification settings - Fork 88
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: Add data visualization for Anthropic #432
Changes from 3 commits
daf2d54
b132ff4
7b0bfe9
cf4737b
55e9d64
047948a
3ba98b2
ad07bae
a0e3686
61118dd
e71f852
933d21a
aef7822
be0b486
662faeb
4a87df9
4ae8eb9
1af24dc
b2355de
5b31638
f19b58a
5aaf420
72b503f
7a4a0f4
0783625
bbd51e8
cb7b2d3
fe47253
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -10,6 +10,8 @@ | |||||||||||||||||||||||||||||||||||||||||||
from cognee.shared.data_models import KnowledgeGraph | ||||||||||||||||||||||||||||||||||||||||||||
from mcp.server import NotificationOptions, Server | ||||||||||||||||||||||||||||||||||||||||||||
from mcp.server.models import InitializationOptions | ||||||||||||||||||||||||||||||||||||||||||||
from PIL import Image | ||||||||||||||||||||||||||||||||||||||||||||
from PIL import Image as PILImage | ||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Double import here |
||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
server = Server("cognee-mcp") | ||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
|
@@ -87,9 +89,45 @@ async def handle_list_tools() -> list[types.Tool]: | |||||||||||||||||||||||||||||||||||||||||||
}, | ||||||||||||||||||||||||||||||||||||||||||||
}, | ||||||||||||||||||||||||||||||||||||||||||||
), | ||||||||||||||||||||||||||||||||||||||||||||
types.Tool( | ||||||||||||||||||||||||||||||||||||||||||||
name="visualize", | ||||||||||||||||||||||||||||||||||||||||||||
description="Visualize the knowledge graph.", | ||||||||||||||||||||||||||||||||||||||||||||
inputSchema={ | ||||||||||||||||||||||||||||||||||||||||||||
"type": "object", | ||||||||||||||||||||||||||||||||||||||||||||
"properties": { | ||||||||||||||||||||||||||||||||||||||||||||
"query": {"type": "string"}, | ||||||||||||||||||||||||||||||||||||||||||||
}, | ||||||||||||||||||||||||||||||||||||||||||||
}, | ||||||||||||||||||||||||||||||||||||||||||||
), | ||||||||||||||||||||||||||||||||||||||||||||
Comment on lines
+94
to
+103
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🛠️ Refactor suggestion Clarify query parameter requirement in visualization tool schema The tool's schema defines a "query" property but doesn't mark it as required. Either:
types.Tool(
name="visualize",
description="Visualize the knowledge graph.",
inputSchema={
"type": "object",
"properties": {
"query": {"type": "string"},
},
+ "required": ["query"], # Add this if query is required
},
), 📝 Committable suggestion
Suggested change
🧰 Tools🪛 GitHub Actions: ruff format[warning] File requires formatting. Code does not conform to Ruff's formatting standards. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is a good suggestion from coderabbit There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @borisarzentar Thank you for confirming! I'm glad the suggestion was helpful. ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ |
||||||||||||||||||||||||||||||||||||||||||||
] | ||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
def get_freshest_png(directory: str) -> Image.Image: | ||||||||||||||||||||||||||||||||||||||||||||
if not os.path.exists(directory): | ||||||||||||||||||||||||||||||||||||||||||||
raise FileNotFoundError(f"Directory {directory} does not exist") | ||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
# List all files in 'directory' that end with .png | ||||||||||||||||||||||||||||||||||||||||||||
files = [f for f in os.listdir(directory) if f.endswith(".png")] | ||||||||||||||||||||||||||||||||||||||||||||
if not files: | ||||||||||||||||||||||||||||||||||||||||||||
raise FileNotFoundError("No PNG files found in the given directory.") | ||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
# Sort by integer value of the filename (minus the '.png') | ||||||||||||||||||||||||||||||||||||||||||||
# Example filename: 1673185134.png -> integer 1673185134 | ||||||||||||||||||||||||||||||||||||||||||||
try: | ||||||||||||||||||||||||||||||||||||||||||||
files_sorted = sorted(files, key=lambda x: int(x.replace(".png", ""))) | ||||||||||||||||||||||||||||||||||||||||||||
except ValueError as e: | ||||||||||||||||||||||||||||||||||||||||||||
raise ValueError("Invalid PNG filename format. Expected timestamp format.") from e | ||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
# The "freshest" file has the largest timestamp | ||||||||||||||||||||||||||||||||||||||||||||
freshest_filename = files_sorted[-1] | ||||||||||||||||||||||||||||||||||||||||||||
freshest_path = os.path.join(directory, freshest_filename) | ||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
# Open the image with PIL and return the PIL Image object | ||||||||||||||||||||||||||||||||||||||||||||
try: | ||||||||||||||||||||||||||||||||||||||||||||
return Image.open(freshest_path) | ||||||||||||||||||||||||||||||||||||||||||||
except (IOError, OSError) as e: | ||||||||||||||||||||||||||||||||||||||||||||
raise IOError(f"Failed to open PNG file {freshest_path}") from e | ||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
@server.call_tool() | ||||||||||||||||||||||||||||||||||||||||||||
async def handle_call_tool( | ||||||||||||||||||||||||||||||||||||||||||||
name: str, arguments: dict | None | ||||||||||||||||||||||||||||||||||||||||||||
|
@@ -154,6 +192,16 @@ async def handle_call_tool( | |||||||||||||||||||||||||||||||||||||||||||
text="Pruned", | ||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||
] | ||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
elif name == "visualize": | ||||||||||||||||||||||||||||||||||||||||||||
with open(os.devnull, "w") as fnull: | ||||||||||||||||||||||||||||||||||||||||||||
with redirect_stdout(fnull), redirect_stderr(fnull): | ||||||||||||||||||||||||||||||||||||||||||||
try: | ||||||||||||||||||||||||||||||||||||||||||||
await cognee.visualize | ||||||||||||||||||||||||||||||||||||||||||||
Vasilije1990 marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||||||||||||||
img = get_freshest_png(".") | ||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🛠️ Refactor suggestion Use configured output directory instead of current directory Using "." as the directory path is brittle and depends on the current working directory when the server starts. Consider using a configured output directory. - img = get_freshest_png(".")
+ img = get_freshest_png(cognee.config.visualization_output_dir) You'll need to ensure the output directory is properly configured in the Cognee configuration.
🧰 Tools🪛 GitHub Actions: ruff format[warning] File requires formatting. Code does not conform to Ruff's formatting standards. |
||||||||||||||||||||||||||||||||||||||||||||
return types.Image(data=img.tobytes(), format="png") | ||||||||||||||||||||||||||||||||||||||||||||
except (FileNotFoundError, IOError, ValueError) as e: | ||||||||||||||||||||||||||||||||||||||||||||
raise ValueError(f"Failed to create visualization: {str(e)}") | ||||||||||||||||||||||||||||||||||||||||||||
else: | ||||||||||||||||||||||||||||||||||||||||||||
raise ValueError(f"Unknown tool: {name}") | ||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -13,7 +13,7 @@ | |
import tiktoken | ||
import nltk | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Remove duplicate nltk import The import nltk
import base64
import time
...
- import nltk
from cognee.shared.exceptions import IngestionError Also applies to: 26-26 🧰 Tools🪛 GitHub Actions: ruff format[warning] File needs formatting according to Ruff standards |
||
import base64 | ||
|
||
import time | ||
|
||
import logging | ||
import sys | ||
|
@@ -396,6 +396,7 @@ async def create_cognee_style_network_with_logo( | |
|
||
from bokeh.embed import file_html | ||
from bokeh.resources import CDN | ||
from bokeh.io import export_png | ||
|
||
logging.info("Converting graph to serializable format...") | ||
G = await convert_to_serializable_graph(G) | ||
|
@@ -443,6 +444,14 @@ async def create_cognee_style_network_with_logo( | |
) | ||
p.add_tools(hover_tool) | ||
|
||
# Get the latest Unix timestamp as an integer | ||
timestamp = int(time.time()) | ||
|
||
# Construct your filename | ||
filename = f"{timestamp}.png" | ||
|
||
export_png(p, filename=filename) | ||
|
||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🛠️ Refactor suggestion Add error handling for PNG export The PNG export functionality lacks error handling and cleanup of old files. # Get the latest Unix timestamp as an integer
timestamp = int(time.time())
# Construct your filename
filename = f"{timestamp}.png"
- export_png(p, filename=filename)
+ try:
+ # Cleanup old PNG files to prevent disk space issues
+ cleanup_old_pngs(directory=".", keep_latest=5)
+
+ # Export the new PNG
+ export_png(p, filename=filename)
+ except Exception as e:
+ logging.error(f"Failed to export PNG: {str(e)}")
+ raise Consider adding a helper function to cleanup old PNG files: def cleanup_old_pngs(directory: str, keep_latest: int = 5):
"""Cleanup old PNG files, keeping only the N latest files."""
png_files = [f for f in os.listdir(directory) if f.endswith('.png')]
if len(png_files) <= keep_latest:
return
# Sort by timestamp in filename
sorted_files = sorted(png_files, key=lambda x: int(x.replace(".png", "")))
# Remove older files
for f in sorted_files[:-keep_latest]:
try:
os.remove(os.path.join(directory, f))
except OSError as e:
logging.warning(f"Failed to remove old PNG file {f}: {str(e)}") |
||
logging.info(f"Saving visualization to {output_filename}...") | ||
html_content = file_html(p, CDN, title) | ||
with open(output_filename, "w") as f: | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Remove duplicate PIL Image import
The Image module from PIL is imported twice with different aliases. This is redundant and could cause confusion.
📝 Committable suggestion
🧰 Tools
🪛 GitHub Actions: ruff format
[warning] File requires formatting. Code does not conform to Ruff formatting standards.