Compare commits
28 Commits
master
...
cogguide_d
Author | SHA1 | Date | |
---|---|---|---|
![]() |
f351b7fbc9 | ||
![]() |
65e5991ceb | ||
![]() |
11f17e3eed | ||
![]() |
bd45f353a8 | ||
![]() |
18c0fd6fc2 | ||
![]() |
86b211c76b | ||
![]() |
8ed622b1a9 | ||
![]() |
7c171b75fb | ||
![]() |
fe3c52c2fa | ||
![]() |
bcf2def56f | ||
![]() |
e0289d505e | ||
![]() |
81dab81eb2 | ||
![]() |
610310ed62 | ||
![]() |
df2ddeee24 | ||
![]() |
59c6d303f7 | ||
![]() |
b53ff659d3 | ||
![]() |
5507ffbfcf | ||
![]() |
21b1408a5b | ||
![]() |
ef8bceba6b | ||
![]() |
f6d3f2c4ea | ||
![]() |
295ec32fa2 | ||
![]() |
c98da991f3 | ||
![]() |
a3af978caa | ||
![]() |
1c8a0c9cd0 | ||
![]() |
42832391fa | ||
![]() |
746414e4d4 | ||
![]() |
bd10d8cfb1 | ||
![]() |
a91bf42155 |
5
cogguide/__init__.py
Normal file
5
cogguide/__init__.py
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
from .cogguide import CogGuide
|
||||||
|
|
||||||
|
|
||||||
|
async def setup(bot):
|
||||||
|
bot.add_cog(CogGuide(bot))
|
299
cogguide/cogguide.py
Normal file
299
cogguide/cogguide.py
Normal file
@ -0,0 +1,299 @@
|
|||||||
|
import logging
|
||||||
|
import pathlib
|
||||||
|
import re
|
||||||
|
from typing import Optional, Union
|
||||||
|
|
||||||
|
from redbot.core import Config, checks, commands
|
||||||
|
from redbot.core.bot import Red
|
||||||
|
from redbot.core.commands import Cog, PrivilegeLevel
|
||||||
|
from redbot.core.data_manager import cog_data_path
|
||||||
|
|
||||||
|
log = logging.getLogger("red.fox-v3.cogguide")
|
||||||
|
|
||||||
|
EXCLUDED_LIST = ["reinstallreqs"]
|
||||||
|
|
||||||
|
# Matches [link](https://website.com) but not `[link](https://web.com)`
|
||||||
|
LINK_PATTERN = re.compile(r"(?<!`)\[(.+?)]\s?\((https?:\/\/[\w\d.\/?=#-@]+)\)(?!`)")
|
||||||
|
|
||||||
|
BOTNAME_PATTERN = re.compile(r"\[botname]")
|
||||||
|
|
||||||
|
|
||||||
|
# Adding more as necessary.
|
||||||
|
# Options:
|
||||||
|
# [X] attention
|
||||||
|
# [ ] caution
|
||||||
|
# [ ] danger
|
||||||
|
# [ ] error
|
||||||
|
# [X] hint
|
||||||
|
# [X] important
|
||||||
|
# [X] note
|
||||||
|
# [X] tip
|
||||||
|
# [X] warning
|
||||||
|
NOTICE_PATTERNS = re.compile(r"^(Warning|Note|Attention|Hint|Important|Tip): (.*)", re.MULTILINE)
|
||||||
|
|
||||||
|
ZWSP_PATTERN = re.compile(r" ")
|
||||||
|
ZWSP_LINE_PATTERN = re.compile(r"(?: )+")
|
||||||
|
|
||||||
|
|
||||||
|
def get_parent_tree(command: commands.Command):
|
||||||
|
out = f"{command.name}"
|
||||||
|
if command.parent:
|
||||||
|
# out = f"{get_parent_tree(command.parent)}-" + out
|
||||||
|
out = f"{'-'.join(command.full_parent_name.split())}-" + out
|
||||||
|
return out
|
||||||
|
|
||||||
|
|
||||||
|
def markdown_link_replace(starts_with_text=None):
|
||||||
|
"""
|
||||||
|
Converts a markdown match to restructuredtext match
|
||||||
|
|
||||||
|
re.sub needs a command with one argument, so this puts it in.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def handle_starts_with_markdown_link_replace(match):
|
||||||
|
text: str = match.group(1)
|
||||||
|
url: str = match.group(2)
|
||||||
|
if starts_with_text and url.startswith(starts_with_text):
|
||||||
|
i = len(starts_with_text)
|
||||||
|
url = url[i:]
|
||||||
|
i = url.find(".")
|
||||||
|
url = url[:i]
|
||||||
|
i = url.find("#")
|
||||||
|
if i > 0:
|
||||||
|
url = url[:i]
|
||||||
|
return f"`{text}<{url}>`" # Attempt :any: match
|
||||||
|
return f":doc:`{text}<../{url}>`"
|
||||||
|
|
||||||
|
return f"`{text} <{url}>`_"
|
||||||
|
|
||||||
|
return handle_starts_with_markdown_link_replace
|
||||||
|
|
||||||
|
|
||||||
|
def format_text(text):
|
||||||
|
"""Borrowed from commands.CogCommandMixin.format_text_for_context"""
|
||||||
|
|
||||||
|
# formatting_pattern = re.compile(r"\[p\]|\[botname\]")
|
||||||
|
def replace_botname(m: re.Match) -> str:
|
||||||
|
s = m.group(0)
|
||||||
|
# if s == "[p]":
|
||||||
|
# return ctx.clean_prefix
|
||||||
|
if s == "[botname]":
|
||||||
|
return "Red"
|
||||||
|
|
||||||
|
# We shouldn't get here:
|
||||||
|
return s
|
||||||
|
|
||||||
|
def replace_notice(m: re.Match) -> str:
|
||||||
|
return f".. {m.group(1)}:: {m.group(2)}\n"
|
||||||
|
|
||||||
|
def newline_zwsp(m: re.match) -> str:
|
||||||
|
s = m.group(0)
|
||||||
|
return f"\n{s}" # Add a newline in front of ZWSP lines
|
||||||
|
|
||||||
|
def replace_zwsp(m: re.Match) -> str:
|
||||||
|
return " " # Could be /t, but we use spaces not tabs
|
||||||
|
|
||||||
|
out = BOTNAME_PATTERN.sub(replace_botname, text)
|
||||||
|
out = NOTICE_PATTERNS.sub(replace_notice, out)
|
||||||
|
out = ZWSP_LINE_PATTERN.sub(newline_zwsp, out)
|
||||||
|
out = ZWSP_PATTERN.sub(replace_zwsp, out)
|
||||||
|
return out
|
||||||
|
|
||||||
|
|
||||||
|
def prepare_description(comm_or_cog: Union[commands.Command, Cog], markdown_command):
|
||||||
|
description = comm_or_cog.description or comm_or_cog.help
|
||||||
|
description = format_text(description)
|
||||||
|
description = description.replace("`", "``")
|
||||||
|
description = LINK_PATTERN.sub(markdown_command, description)
|
||||||
|
return description
|
||||||
|
|
||||||
|
|
||||||
|
def produce_aliases(command: commands.Command):
|
||||||
|
return ", ".join(
|
||||||
|
f"``{command.full_parent_name + ' ' if command.full_parent_name else ''}{alias}``"
|
||||||
|
for alias in command.aliases
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class CogGuide(commands.Cog):
|
||||||
|
"""
|
||||||
|
Cog to create cog guides
|
||||||
|
|
||||||
|
Dunno if this is a good idea but I did it. Sue me.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, bot: Red):
|
||||||
|
super().__init__()
|
||||||
|
self.bot = bot
|
||||||
|
self.config = Config.get_conf(
|
||||||
|
self, identifier=6711110371117105100101, force_registration=True
|
||||||
|
)
|
||||||
|
|
||||||
|
default_global = {"starts_with": None}
|
||||||
|
self.config.register_global(**default_global)
|
||||||
|
|
||||||
|
async def red_delete_data_for_user(self, **kwargs):
|
||||||
|
"""Nothing to delete"""
|
||||||
|
return
|
||||||
|
|
||||||
|
@checks.is_owner()
|
||||||
|
@commands.group()
|
||||||
|
async def cogguideset(self, ctx: commands.Context):
|
||||||
|
"""Base command for configuring cogguide"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@cogguideset.command(name="list")
|
||||||
|
async def cogguideset_list(self, ctx: commands.Context):
|
||||||
|
"""List current cogguide settings"""
|
||||||
|
await ctx.maybe_send_embed(
|
||||||
|
f"""**Current Settings**
|
||||||
|
ReadTheDocs URL: {await self.config.starts_with()}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
@cogguideset.command(name="url")
|
||||||
|
async def cogguideset_url(self, ctx: commands.Context, url: str = None):
|
||||||
|
"""Sets the url of your ReadTheDocs for creating references
|
||||||
|
|
||||||
|
For example: 'https://docs.discord.red/en/stable/' for Red docs
|
||||||
|
"""
|
||||||
|
if not url:
|
||||||
|
await self.config.starts_with.clear()
|
||||||
|
else:
|
||||||
|
await self.config.starts_with.set(url)
|
||||||
|
|
||||||
|
await ctx.tick()
|
||||||
|
|
||||||
|
@checks.is_owner()
|
||||||
|
@commands.command()
|
||||||
|
async def allcogguides(self, ctx: commands.Context):
|
||||||
|
"""
|
||||||
|
Create a ReStructuredText file for all loaded cogs.
|
||||||
|
|
||||||
|
Results can be found in the cog data folder.
|
||||||
|
|
||||||
|
Returns: tick()
|
||||||
|
|
||||||
|
"""
|
||||||
|
for cog_name, cog in self.bot.cogs.items():
|
||||||
|
await self.create_cog_guide(cog_name, cog)
|
||||||
|
await ctx.tick()
|
||||||
|
|
||||||
|
@checks.is_owner()
|
||||||
|
@commands.command()
|
||||||
|
async def cogguide(self, ctx: commands.Context, camel_cog_name: str):
|
||||||
|
"""
|
||||||
|
Create a ReStructuredText file for a given loaded cog.
|
||||||
|
|
||||||
|
Result can be found in the cog data folder.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
camel_cog_name:
|
||||||
|
|
||||||
|
Returns: tick
|
||||||
|
|
||||||
|
"""
|
||||||
|
cog: Optional[Cog] = self.bot.get_cog(camel_cog_name)
|
||||||
|
if cog is None:
|
||||||
|
await ctx.send("No cog found with that name")
|
||||||
|
return
|
||||||
|
await self.create_cog_guide(camel_cog_name, cog)
|
||||||
|
|
||||||
|
await ctx.tick()
|
||||||
|
|
||||||
|
async def create_cog_guide(self, camel_cog_name, cog):
|
||||||
|
path: pathlib.Path = cog_data_path(self)
|
||||||
|
lower_cog_name = f"{camel_cog_name.lower()}"
|
||||||
|
reference = f"_{camel_cog_name.lower()}"
|
||||||
|
filename = f"{lower_cog_name}.rst"
|
||||||
|
filepath = path / filename
|
||||||
|
privilege_levels = {
|
||||||
|
PrivilegeLevel.MOD: "|mod-lock|",
|
||||||
|
PrivilegeLevel.ADMIN: "|admin-lock|",
|
||||||
|
PrivilegeLevel.GUILD_OWNER: "|guildowner-lock|",
|
||||||
|
PrivilegeLevel.BOT_OWNER: "|owner-lock|",
|
||||||
|
}
|
||||||
|
url_command = markdown_link_replace(await self.config.starts_with())
|
||||||
|
intro = f""".. {reference}:
|
||||||
|
|
||||||
|
{'=' * len(camel_cog_name)}
|
||||||
|
{camel_cog_name}
|
||||||
|
{'=' * len(camel_cog_name)}
|
||||||
|
|
||||||
|
This is the cog guide for the {lower_cog_name} cog. You will
|
||||||
|
find detailed docs about usage and commands.
|
||||||
|
|
||||||
|
``[p]`` is considered as your prefix.
|
||||||
|
"""
|
||||||
|
if camel_cog_name != "Core":
|
||||||
|
intro += f"""
|
||||||
|
.. note:: To use this cog, load it by typing this::
|
||||||
|
|
||||||
|
[p]load {'customcom' if lower_cog_name == 'customcommands' else lower_cog_name}
|
||||||
|
"""
|
||||||
|
intro += f"""
|
||||||
|
.. _{lower_cog_name}-usage:
|
||||||
|
|
||||||
|
-----
|
||||||
|
Usage
|
||||||
|
-----
|
||||||
|
|
||||||
|
{prepare_description(cog, url_command)}
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
cog_commands_intro = f"""
|
||||||
|
.. {reference}-commands:
|
||||||
|
|
||||||
|
--------
|
||||||
|
Commands
|
||||||
|
--------
|
||||||
|
"""
|
||||||
|
|
||||||
|
def get_command_rst(command: commands.Command):
|
||||||
|
description = prepare_description(command, url_command)
|
||||||
|
|
||||||
|
cog_command = f"""
|
||||||
|
.. {reference}-command-{get_parent_tree(command)}:
|
||||||
|
|
||||||
|
{'^' * len(command.qualified_name) if not command.parent else '"' * len(command.qualified_name)}
|
||||||
|
{command.qualified_name}
|
||||||
|
{'^' * len(command.qualified_name) if not command.parent else '"' * len(command.qualified_name)}
|
||||||
|
"""
|
||||||
|
if command.requires.privilege_level in privilege_levels:
|
||||||
|
cog_command += f"""
|
||||||
|
.. note:: {privilege_levels[command.requires.privilege_level]}
|
||||||
|
"""
|
||||||
|
cog_command += f"""
|
||||||
|
**Syntax**
|
||||||
|
|
||||||
|
.. code-block:: none
|
||||||
|
|
||||||
|
[p]{command.qualified_name} {command.signature}
|
||||||
|
"""
|
||||||
|
if command.aliases:
|
||||||
|
cog_command += f"""
|
||||||
|
.. tip:: Alias{'es' if len(command.aliases) > 1 else ''}: {produce_aliases(command)}
|
||||||
|
"""
|
||||||
|
cog_command += f"""
|
||||||
|
**Description**
|
||||||
|
|
||||||
|
{description}
|
||||||
|
"""
|
||||||
|
return cog_command
|
||||||
|
|
||||||
|
cog_commands_list = []
|
||||||
|
com_list = [com for com in cog.walk_commands()]
|
||||||
|
com_list.sort(key=lambda x: x.qualified_name)
|
||||||
|
for com in com_list:
|
||||||
|
com: commands.Command
|
||||||
|
if com.name in EXCLUDED_LIST:
|
||||||
|
continue
|
||||||
|
if not com.hidden and com.enabled:
|
||||||
|
cog_commands_list.append(get_command_rst(com))
|
||||||
|
# for com in cog.walk_commands():
|
||||||
|
# cog_commands_list.append(get_command_rst(com))
|
||||||
|
with filepath.open("w", encoding="utf-8") as f:
|
||||||
|
f.write(intro)
|
||||||
|
f.write(cog_commands_intro)
|
||||||
|
f.writelines(cog_commands_list)
|
18
cogguide/info.json
Normal file
18
cogguide/info.json
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
{
|
||||||
|
"author": [
|
||||||
|
"Bobloy"
|
||||||
|
],
|
||||||
|
"min_bot_version": "3.4.0",
|
||||||
|
"description": "Create a template ReStructuredText file for a given loaded cog",
|
||||||
|
"hidden": true,
|
||||||
|
"install_msg": "Thank you for installing CogGuide! Get started with `[p]load cogguide` and `[p]help CogGuide`",
|
||||||
|
"requirements": [
|
||||||
|
],
|
||||||
|
"short": "Create a template ReStructuredText file for a given loaded cog",
|
||||||
|
"end_user_data_statement": "This cog does not store any End User Data",
|
||||||
|
"tags": [
|
||||||
|
"bobloy",
|
||||||
|
"utils",
|
||||||
|
"tools"
|
||||||
|
]
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user