Files
Syntrel/cogs/utilities/codepreview.py

529 lines
20 KiB
Python
Raw Permalink Normal View History

2025-10-01 00:14:04 -04:00
import discord
from discord import app_commands
from discord.ext import commands
from discord.ext.commands import Context
import aiohttp
import re
import json
def codepreview_command():
LANGUAGE_MAP = {
2025-11-02 23:32:52 -05:00
".py": "python",
".js": "js",
".ts": "ts",
".java": "java",
".cpp": "cpp",
".c": "c",
".cs": "cs",
".go": "go",
".rs": "rust",
".rb": "ruby",
".php": "php",
".html": "html",
".css": "css",
".json": "json",
".xml": "xml",
".yaml": "yaml",
".yml": "yaml",
".ini": "ini",
".toml": "toml",
".lua": "lua",
".sh": "bash",
".md": "markdown",
".sql": "sql",
".diff": "diff",
".txt": "",
2025-10-01 00:14:04 -04:00
}
2025-11-02 23:32:52 -05:00
async def send_embed(
context, embed: discord.Embed, *, ephemeral: bool = False
) -> None:
2025-10-01 00:14:04 -04:00
interaction = getattr(context, "interaction", None)
if interaction is not None:
if interaction.response.is_done():
await interaction.followup.send(embed=embed, ephemeral=ephemeral)
else:
2025-11-02 23:32:52 -05:00
await interaction.response.send_message(
embed=embed, ephemeral=ephemeral
)
2025-10-01 00:14:04 -04:00
else:
await context.send(embed=embed)
def get_language_from_filename(filename):
for ext, lang in LANGUAGE_MAP.items():
if filename.endswith(ext):
return lang
2025-11-02 23:32:52 -05:00
return ""
2025-10-01 00:14:04 -04:00
async def fetch_github_content(url):
try:
async with aiohttp.ClientSession() as session:
async with session.get(url) as response:
if response.status == 200:
return await response.text()
except Exception:
pass
return None
2025-10-08 23:28:30 -04:00
async def fetch_pr_diff(owner, repo, pr_number):
try:
2025-11-02 23:32:52 -05:00
api_url = f"https://api.github.com/repos/{owner}/{repo}/pulls/{pr_number}"
2025-10-08 23:28:30 -04:00
headers = {
2025-11-02 23:32:52 -05:00
"Accept": "application/vnd.github.v3.diff",
"User-Agent": "Discord-Bot",
2025-10-08 23:28:30 -04:00
}
2025-11-02 23:32:52 -05:00
2025-10-08 23:28:30 -04:00
async with aiohttp.ClientSession() as session:
async with session.get(api_url, headers=headers) as response:
if response.status == 200:
diff_content = await response.text()
return diff_content
except Exception:
pass
return None
async def fetch_pr_info(owner, repo, pr_number):
try:
2025-11-02 23:32:52 -05:00
api_url = f"https://api.github.com/repos/{owner}/{repo}/pulls/{pr_number}"
2025-10-08 23:28:30 -04:00
headers = {
2025-11-02 23:32:52 -05:00
"Accept": "application/vnd.github.v3+json",
"User-Agent": "Discord-Bot",
2025-10-08 23:28:30 -04:00
}
2025-11-02 23:32:52 -05:00
2025-10-08 23:28:30 -04:00
async with aiohttp.ClientSession() as session:
async with session.get(api_url, headers=headers) as response:
if response.status == 200:
pr_data = await response.json()
return {
2025-11-02 23:32:52 -05:00
"title": pr_data.get("title", ""),
"number": pr_data.get("number", pr_number),
"state": pr_data.get("state", ""),
"merged": pr_data.get("merged", False),
"additions": pr_data.get("additions", 0),
"deletions": pr_data.get("deletions", 0),
"changed_files": pr_data.get("changed_files", 0),
"user": pr_data.get("user", {}).get("login", ""),
"base_branch": pr_data.get("base", {}).get("ref", ""),
"head_branch": pr_data.get("head", {}).get("ref", ""),
2025-10-08 23:28:30 -04:00
}
except Exception:
pass
return None
2025-10-01 00:14:04 -04:00
def parse_github_url(url):
2025-11-02 23:32:52 -05:00
pr_pattern = r"https://github\.com/([^/]+)/([^/]+)/pull/(\d+)(?:/files)?"
2025-10-08 23:28:30 -04:00
pr_match = re.match(pr_pattern, url)
2025-11-02 23:32:52 -05:00
2025-10-08 23:28:30 -04:00
if pr_match:
owner, repo, pr_number = pr_match.groups()
2025-11-02 23:32:52 -05:00
return {"type": "pr", "owner": owner, "repo": repo, "pr_number": pr_number}
raw_pattern = (
r"https://raw\.githubusercontent\.com/([^/]+)/([^/]+)/([^/]+)/(.+?)$"
)
2025-10-01 00:14:04 -04:00
raw_match = re.match(raw_pattern, url)
2025-11-02 23:32:52 -05:00
2025-10-01 00:14:04 -04:00
if raw_match:
owner, repo, branch, filepath = raw_match.groups()
2025-11-02 23:32:52 -05:00
2025-10-01 00:14:04 -04:00
return {
2025-11-02 23:32:52 -05:00
"type": "file",
"owner": owner,
"repo": repo,
"branch": branch,
"filepath": filepath,
"raw_url": url,
"start_line": None,
"end_line": None,
2025-10-01 00:14:04 -04:00
}
2025-11-02 23:32:52 -05:00
pattern = r"https://github\.com/([^/]+)/([^/]+)/blob/([^/]+)/(.+?)(?:#L(\d+)(?:-L(\d+))?)?$"
2025-10-01 00:14:04 -04:00
match = re.match(pattern, url)
2025-11-02 23:32:52 -05:00
2025-10-01 00:14:04 -04:00
if match:
owner, repo, branch, filepath, start_line, end_line = match.groups()
2025-11-02 23:32:52 -05:00
raw_url = (
f"https://raw.githubusercontent.com/{owner}/{repo}/{branch}/{filepath}"
)
2025-10-01 00:14:04 -04:00
return {
2025-11-02 23:32:52 -05:00
"type": "file",
"owner": owner,
"repo": repo,
"branch": branch,
"filepath": filepath,
"raw_url": raw_url,
"start_line": int(start_line) if start_line else None,
"end_line": int(end_line) if end_line else None,
2025-10-01 00:14:04 -04:00
}
return None
def extract_lines(content, start_line, end_line):
2025-11-02 23:32:52 -05:00
lines = content.split("\n")
2025-10-01 00:14:04 -04:00
if start_line and end_line:
2025-11-02 23:32:52 -05:00
return "\n".join(lines[start_line - 1 : end_line])
2025-10-01 00:14:04 -04:00
elif start_line:
2025-11-02 23:32:52 -05:00
return lines[start_line - 1] if start_line <= len(lines) else content
2025-10-01 00:14:04 -04:00
return content
@commands.hybrid_command(
name="codepreview",
description="Preview code from GitHub URLs",
)
2025-11-02 23:32:52 -05:00
@app_commands.describe(url="GitHub URL to preview code from")
2025-10-01 00:14:04 -04:00
async def codepreview(self, context, url: str = None):
if isinstance(context.channel, discord.DMChannel):
2025-10-18 00:19:04 -04:00
embed = discord.Embed(
title="Error",
description="This command can only be used in servers.",
2025-10-18 00:19:04 -04:00
color=0xE02B2B,
)
2025-11-02 23:32:52 -05:00
embed.set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
interaction = getattr(context, "interaction", None)
if interaction is not None:
if not interaction.response.is_done():
await interaction.response.send_message(embed=embed, ephemeral=True)
else:
await interaction.followup.send(embed=embed, ephemeral=True)
else:
await context.send(embed=embed, ephemeral=True)
return
if isinstance(context.channel, discord.PartialMessageable):
embed = discord.Embed(
title="Error",
description="The bot needs the `send messages` permission in this channel.",
color=0xE02B2B,
)
2025-11-02 23:32:52 -05:00
embed.set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
interaction = getattr(context, "interaction", None)
if interaction is not None:
if not interaction.response.is_done():
await interaction.response.send_message(embed=embed, ephemeral=True)
else:
await interaction.followup.send(embed=embed, ephemeral=True)
else:
await context.send(embed=embed, ephemeral=True)
2025-10-18 00:19:04 -04:00
return
2025-11-02 23:32:52 -05:00
2025-10-01 00:14:04 -04:00
if not url or not url.strip():
2025-11-02 23:32:52 -05:00
if (
context.message
and context.message.reference
and context.message.reference.resolved
):
2025-10-01 00:14:04 -04:00
replied_message = context.message.reference.resolved
2025-11-02 23:32:52 -05:00
if hasattr(replied_message, "content") and replied_message.content:
github_pattern = (
r"https://(?:github\.com|raw\.githubusercontent\.com)/[^\s]+"
)
2025-10-01 00:14:04 -04:00
urls = re.findall(github_pattern, replied_message.content)
if urls:
url = urls[0]
else:
embed = discord.Embed(
title="Error",
description="No GitHub URL found in the replied message.",
color=0xE02B2B,
2025-11-02 23:32:52 -05:00
).set_author(
name="Utility",
icon_url="https://yes.nighty.works/raw/8VLDcg.webp",
)
2025-10-01 00:14:04 -04:00
await send_embed(context, embed, ephemeral=True)
return
else:
embed = discord.Embed(
title="Error",
description="The replied message has no content to extract GitHub URL from.",
color=0xE02B2B,
2025-11-02 23:32:52 -05:00
).set_author(
name="Utility",
icon_url="https://yes.nighty.works/raw/8VLDcg.webp",
)
2025-10-01 00:14:04 -04:00
await send_embed(context, embed, ephemeral=True)
return
else:
embed = discord.Embed(
title="Error",
description="Please provide a GitHub URL or reply to a message containing a GitHub URL.",
color=0xE02B2B,
2025-11-02 23:32:52 -05:00
).set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
2025-10-01 00:14:04 -04:00
await send_embed(context, embed, ephemeral=True)
return
2025-11-02 23:32:52 -05:00
if not url.startswith("https://github.com/") and not url.startswith(
"https://raw.githubusercontent.com/"
):
2025-10-01 00:14:04 -04:00
embed = discord.Embed(
title="Error",
description="Please provide a valid GitHub URL.",
color=0xE02B2B,
2025-11-02 23:32:52 -05:00
).set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
2025-10-01 00:14:04 -04:00
await send_embed(context, embed, ephemeral=True)
return
# Check if bot has send messages permission before starting processing
try:
test_embed = discord.Embed(title="Testing permissions...", color=0x7289DA)
2025-11-02 23:32:52 -05:00
test_embed.set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
await context.channel.send(embed=test_embed, delete_after=0.1)
except discord.Forbidden:
embed = discord.Embed(
title="Permission Error",
description="The bot needs the `send messages` permission to execute this command.",
color=0xE02B2B,
)
2025-11-02 23:32:52 -05:00
embed.set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
interaction = getattr(context, "interaction", None)
if interaction is not None:
if not interaction.response.is_done():
await interaction.response.send_message(embed=embed, ephemeral=True)
else:
await interaction.followup.send(embed=embed, ephemeral=True)
else:
await context.send(embed=embed, ephemeral=True)
return
2025-11-02 23:32:52 -05:00
2025-10-01 00:14:04 -04:00
parsed = parse_github_url(url)
2025-11-02 23:32:52 -05:00
2025-10-01 00:14:04 -04:00
if not parsed:
embed = discord.Embed(
title="Error",
2025-10-08 23:28:30 -04:00
description="Invalid GitHub URL format. Please provide a valid GitHub blob URL or PR URL.",
2025-10-01 00:14:04 -04:00
color=0xE02B2B,
2025-11-02 23:32:52 -05:00
).set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
2025-10-01 00:14:04 -04:00
await send_embed(context, embed, ephemeral=True)
return
2025-11-02 23:32:52 -05:00
if parsed.get("type") == "pr":
pr_info = await fetch_pr_info(
parsed["owner"], parsed["repo"], parsed["pr_number"]
)
diff_content = await fetch_pr_diff(
parsed["owner"], parsed["repo"], parsed["pr_number"]
)
2025-10-08 23:28:30 -04:00
if not pr_info or not diff_content:
embed = discord.Embed(
title="Error",
description="Failed to fetch pull request information. The PR might not exist or be accessible.",
color=0xE02B2B,
2025-11-02 23:32:52 -05:00
).set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
2025-10-08 23:28:30 -04:00
await send_embed(context, embed, ephemeral=True)
return
2025-11-02 23:32:52 -05:00
2025-10-08 23:28:30 -04:00
pr_url = f"https://github.com/{parsed['owner']}/{parsed['repo']}/pull/{parsed['pr_number']}"
2025-11-02 23:32:52 -05:00
if pr_info["merged"]:
pr_color = 0x6F42C1
pr_status = "Merged"
2025-11-02 23:32:52 -05:00
elif pr_info["state"] == "open":
pr_color = 0x57F287
pr_status = "Open"
else:
pr_color = 0xE02B2B
pr_status = "Closed"
2025-11-02 23:32:52 -05:00
2025-10-08 23:28:30 -04:00
embed = discord.Embed(
title=f"Pull Request #{pr_info['number']}: {pr_info['title'][:100]}",
description=f"**Repository:** [{parsed['owner']}/{parsed['repo']}]({pr_url})\n**Author:** {pr_info['user']}\n**Status:** {pr_status}",
color=pr_color,
2025-10-08 23:28:30 -04:00
)
2025-11-02 23:32:52 -05:00
embed.set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
embed.add_field(
name="Changes",
value=f"**+{pr_info['additions']}** / **-{pr_info['deletions']}**",
inline=True,
)
embed.add_field(
name="Files Changed", value=f"{pr_info['changed_files']}", inline=True
)
embed.add_field(
name="Branches",
value=f"`{pr_info['base_branch']}` ← `{pr_info['head_branch']}`",
inline=False,
)
embed.set_footer(
text=f"Requested by {context.author.name}",
icon_url=context.author.display_avatar.url,
)
2025-10-08 23:28:30 -04:00
interaction = getattr(context, "interaction", None)
if interaction is not None and not interaction.response.is_done():
await interaction.response.defer(ephemeral=True)
2025-11-02 23:32:52 -05:00
2025-10-08 23:28:30 -04:00
await context.channel.send(embed=embed)
2025-11-02 23:32:52 -05:00
2025-10-08 23:28:30 -04:00
max_diff_length = 1900
max_lines = 100
2025-11-02 23:32:52 -05:00
diff_lines = diff_content.split("\n")
2025-10-08 23:28:30 -04:00
if len(diff_lines) > max_lines:
diff_lines = diff_lines[:max_lines]
2025-11-02 23:32:52 -05:00
diff_lines.append(
f"\n... ({len(diff_content.split(chr(10))) - max_lines} more lines omitted)"
)
2025-10-08 23:28:30 -04:00
current_chunk = ""
2025-11-02 23:32:52 -05:00
2025-10-08 23:28:30 -04:00
for line in diff_lines:
2025-11-02 23:32:52 -05:00
test_chunk = current_chunk + line + "\n"
2025-10-08 23:28:30 -04:00
if len(test_chunk) + 10 > max_diff_length:
if current_chunk.strip():
2025-11-02 23:32:52 -05:00
remaining_lines = len(diff_lines) - len(
current_chunk.split("\n")
)
2025-10-08 23:28:30 -04:00
if remaining_lines > 0:
2025-11-02 23:32:52 -05:00
current_chunk += (
f"\n... ({remaining_lines} more lines omitted)"
)
await context.channel.send(
f"```diff\n{current_chunk.rstrip()}\n```"
)
2025-10-08 23:28:30 -04:00
break
else:
current_chunk = test_chunk
else:
if current_chunk.strip():
2025-11-02 23:32:52 -05:00
await context.channel.send(
f"```diff\n{current_chunk.rstrip()}\n```"
)
2025-10-08 23:28:30 -04:00
if interaction is not None:
try:
await interaction.delete_original_response()
except:
pass
return
2025-11-02 23:32:52 -05:00
content = await fetch_github_content(parsed["raw_url"])
2025-10-01 00:14:04 -04:00
if not content:
embed = discord.Embed(
title="Error",
description="Failed to fetch content from GitHub. The file might not exist or be accessible.",
color=0xE02B2B,
2025-11-02 23:32:52 -05:00
).set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
2025-10-01 00:14:04 -04:00
await send_embed(context, embed, ephemeral=True)
return
2025-11-02 23:32:52 -05:00
if parsed["start_line"]:
code = extract_lines(content, parsed["start_line"], parsed["end_line"])
2025-10-01 00:14:04 -04:00
line_info = f" (Lines {parsed['start_line']}"
2025-11-02 23:32:52 -05:00
if parsed["end_line"]:
2025-10-01 00:14:04 -04:00
line_info += f"-{parsed['end_line']}"
line_info += ")"
else:
code = content
line_info = ""
2025-11-02 23:32:52 -05:00
code_lines = code.split("\n")
2025-10-08 23:28:30 -04:00
if len(code_lines) > 100:
2025-11-02 23:32:52 -05:00
code = "\n".join(code_lines[:100])
2025-10-08 23:28:30 -04:00
code += f"\n\n... ({len(code_lines) - 100} more lines omitted)"
2025-11-02 23:32:52 -05:00
filename = parsed["filepath"].split("/")[-1]
2025-10-01 00:14:04 -04:00
language = get_language_from_filename(filename)
2025-11-02 23:32:52 -05:00
2025-10-01 00:14:04 -04:00
embed = discord.Embed(
title="Code Preview",
description=f"**Repository URL:** [{parsed['owner']}/{parsed['repo']}]({url})",
color=0x7289DA,
)
2025-11-02 23:32:52 -05:00
embed.set_author(
name="Utility", icon_url="https://yes.nighty.works/raw/8VLDcg.webp"
)
embed.add_field(
name="File", value=f"`{parsed['filepath']}`{line_info}", inline=True
)
2025-10-01 00:14:04 -04:00
embed.add_field(name="Branch", value=f"`{parsed['branch']}`", inline=True)
2025-11-02 23:32:52 -05:00
embed.set_footer(
text=f"Requested by {context.author.name}",
icon_url=context.author.display_avatar.url,
)
2025-10-01 00:14:04 -04:00
code_block = f"```{language}\n{code}\n```"
2025-11-02 23:32:52 -05:00
2025-10-01 00:14:04 -04:00
if len(code_block) > 1990:
interaction = getattr(context, "interaction", None)
if interaction is not None:
if not interaction.response.is_done():
await interaction.response.defer(ephemeral=True)
await context.channel.send(embed=embed)
else:
await context.channel.send(embed=embed)
2025-11-02 23:32:52 -05:00
2025-10-01 00:14:04 -04:00
max_code_length = 1980 - len(language) - 8
2025-11-02 23:32:52 -05:00
code_lines = code.split("\n")
2025-10-01 00:14:04 -04:00
current_chunk = []
current_length = 0
2025-11-02 23:32:52 -05:00
2025-10-01 00:14:04 -04:00
for line in code_lines:
line_length = len(line) + 1
if current_length + line_length > max_code_length:
2025-10-08 23:28:30 -04:00
remaining_lines = len(code_lines) - len(current_chunk)
if remaining_lines > 0:
2025-11-02 23:32:52 -05:00
current_chunk.append(
f"\n... ({remaining_lines} more lines omitted)"
)
chunk_text = "\n".join(current_chunk)
2025-10-08 23:28:30 -04:00
await context.channel.send(f"```{language}\n{chunk_text}\n```")
break
2025-10-01 00:14:04 -04:00
else:
current_chunk.append(line)
current_length += line_length
2025-10-08 23:28:30 -04:00
else:
if current_chunk:
2025-11-02 23:32:52 -05:00
chunk_text = "\n".join(current_chunk)
2025-10-01 00:14:04 -04:00
await context.channel.send(f"```{language}\n{chunk_text}\n```")
2025-11-02 23:32:52 -05:00
2025-10-08 23:28:30 -04:00
if interaction is not None:
try:
await interaction.delete_original_response()
except:
pass
2025-10-01 00:14:04 -04:00
else:
interaction = getattr(context, "interaction", None)
if interaction is not None:
if not interaction.response.is_done():
await interaction.response.defer(ephemeral=True)
await context.channel.send(embed=embed)
await context.channel.send(code_block)
try:
await interaction.delete_original_response()
except:
pass
else:
await context.channel.send(embed=embed)
await context.channel.send(code_block)
2025-11-02 23:32:52 -05:00
return codepreview