Skip to content

Commit ca7dcf2

Browse files
committed
[Bot] update extension loading
1 parent a3c63c4 commit ca7dcf2

3 files changed

Lines changed: 59 additions & 1 deletion

File tree

capy_discord/bot.py

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
from discord.ext.commands import AutoShardedBot
44

5-
from capy_discord.utils.extensions import EXTENSIONS
5+
from capy_discord.utils import EXTENSIONS
66

77

88
class Bot(AutoShardedBot):

capy_discord/utils/__init__.py

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
from capy_discord.utils.extensions import EXTENSIONS

capy_discord/utils/extensions.py

Lines changed: 57 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
# MIT License
2+
#
3+
# Copyright (c) 2018 Python Discord
4+
#
5+
# Permission is hereby granted, free of charge, to any person obtaining a copy
6+
# of this software and associated documentation files (the "Software"), to deal
7+
# in the Software without restriction, including without limitation the rights
8+
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
# copies of the Software, and to permit persons to whom the Software is
10+
# furnished to do so, subject to the following conditions:
11+
#
12+
# The above copyright notice and this permission notice shall be included in all
13+
# copies or substantial portions of the Software.
14+
#
15+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21+
# SOFTWARE.
22+
23+
import importlib
24+
import inspect
25+
import pkgutil
26+
from collections.abc import Iterator
27+
from typing import NoReturn
28+
29+
from capy_discord import exts
30+
31+
32+
def unqualify(name: str) -> str:
33+
"""Return an unqualified name given a qualified module/package `name`."""
34+
return name.rsplit(".", maxsplit=1)[-1]
35+
36+
37+
def walk_extensions() -> Iterator[str]:
38+
"""Yield extension names from the bot.exts subpackage."""
39+
40+
def on_error(name: str) -> NoReturn:
41+
raise ImportError(name=name) # pragma: no cover
42+
43+
for module in pkgutil.walk_packages(exts.__path__, f"{exts.__name__}.", onerror=on_error):
44+
if unqualify(module.name).startswith("_"):
45+
# Ignore module/package names starting with an underscore.
46+
continue
47+
48+
if module.ispkg:
49+
imported = importlib.import_module(module.name)
50+
if not inspect.isfunction(getattr(imported, "setup", None)):
51+
# If it lacks a setup function, it's not an extension.
52+
continue
53+
54+
yield module.name
55+
56+
57+
EXTENSIONS = frozenset(walk_extensions())

0 commit comments

Comments
 (0)