Skip to content

Commit 8a527d9

Browse files
Upgrade tooling to use ruff+black over isort/black (#59)
* Add recommended extensions for tooling * update deps for ruff * run ruff * add git blame ignore revs * fix gh action to use ruff
1 parent 8d3b308 commit 8a527d9

32 files changed

+416
-375
lines changed

.git-blame-ignore-revs

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
# run ruff
2+
174ab9e3d7bc643805ac97acf71a225dd9523977

.github/workflows/coverage_and_lint.yml

+2-10
Original file line numberDiff line numberDiff line change
@@ -71,13 +71,5 @@ jobs:
7171
no-comments: ${{ matrix.python-version != '3.x' }}
7272

7373
- name: Lint
74-
uses: github/super-linter/slim@v4
75-
env:
76-
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
77-
DEFAULT_BRANCH: main
78-
VALIDATE_ALL_CODEBASE: false
79-
VALIDATE_PYTHON_BLACK: true
80-
VALIDATE_PYTHON_ISORT: true
81-
LINTER_RULES_PATH: /
82-
PYTHON_ISORT_CONFIG_FILE: pyproject.toml
83-
PYTHON_BLACK_CONFIG_FILE: pyproject.toml
74+
if: ${{ always() && steps.install-deps.outcome == 'success' }}
75+
uses: chartboost/ruff-action@v1

.vscode/extensions.json

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"recommendations": [
3+
"ms-python.python",
4+
"ms-python.vscode-pylance",
5+
"charliermarsh.ruff",
6+
"ms-python.black-formatter"
7+
]
8+
}

README.md

+5
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,11 @@ which will enable all services and snekbox.
4343

4444
This also means that it will use internal docker networking to resolve the database and snekbox names. By default these will be `database` and `snekbox`, but these will be the **service names** in `docker-compose.yml` if you change them. Please keep this in mind when editing your config file.
4545

46+
We also provide a file for ignoring git blame references. You can enable this with:-
47+
```sh
48+
git config blame.ignoreRevsFile .git-blame-ignore-revs
49+
```
50+
4651
### Support
4752

4853
As this bot is not really designed for use outside of our Guild, support provided will be **very** limited.

constants/__init__.py

-1
Original file line numberDiff line numberDiff line change
@@ -22,5 +22,4 @@
2222
"""
2323
from .constants import *
2424

25-
2625
GUILD_ID: int = 490948346773635102

constants/_meta.py

-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,6 @@
2626

2727
import toml
2828

29-
3029
if TYPE_CHECKING:
3130
from typing_extensions import Self
3231

constants/constants.py

-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,6 @@
2222
"""
2323
from ._meta import CONSTANTS
2424

25-
2625
__all__ = (
2726
"Roles",
2827
"Colours",

core/bot.py

+6-6
Original file line numberDiff line numberDiff line change
@@ -31,24 +31,24 @@
3131
from collections import deque
3232
from typing import TYPE_CHECKING, Any
3333

34-
import aiohttp
35-
import asyncpg
3634
import discord
3735
from discord.ext import commands
38-
from discord.ext.commands.cog import Cog # type: ignore # stubs
3936

4037
from constants import GUILD_ID
4138

4239
from .context import Context
4340
from .core import CONFIG
44-
from .utils import LogHandler
45-
4641

4742
if TYPE_CHECKING:
4843
from asyncio import Queue
4944
from logging import LogRecord
5045

46+
import aiohttp
47+
import asyncpg
5148
import mystbin
49+
from discord.ext.commands.cog import Cog # type: ignore # stubs
50+
51+
from .utils import LogHandler
5252

5353

5454
class Bot(commands.Bot):
@@ -67,7 +67,7 @@ class Bot(commands.Bot):
6767
"_previous_websocket_events",
6868
)
6969

70-
def __init__(self):
70+
def __init__(self) -> None:
7171
super().__init__(
7272
command_prefix=commands.when_mentioned_or(CONFIG["prefix"]),
7373
intents=discord.Intents.all(),

core/checks.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -27,12 +27,12 @@
2727
from discord.ext import commands
2828

2929
import constants
30-
from core.context import GuildContext
31-
3230

3331
if TYPE_CHECKING:
3432
from discord.ext.commands._types import Check # type: ignore # why would this need stubs
3533

34+
from core.context import GuildContext
35+
3636

3737
def is_role_or_higher(role_id: int) -> Check[Any]:
3838
def predicate(ctx: GuildContext) -> bool:

core/context.py

-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,6 @@
77

88
from constants import GUILD_ID, Roles
99

10-
1110
if TYPE_CHECKING:
1211
from .bot import Bot
1312

core/converters.py

-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,6 @@
2727

2828
from core.context import Context
2929

30-
3130
__all__ = ("Codeblock", "CodeblockConverter")
3231

3332

core/core.py

-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,6 @@
2727
import toml
2828
from discord.ext import commands
2929

30-
3130
if TYPE_CHECKING:
3231
from types_.config import Config
3332

core/enums.py

-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
from discord.enums import Enum
22

3-
43
__all__ = ("DiscordPyModerationEvent",)
54

65

core/errors.py

-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,6 @@
2020
"""
2121
from discord.ext import commands
2222

23-
2423
__all__ = ("InvalidEval",)
2524

2625

core/utils/formatters.py

+1-2
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,6 @@
2525
from discord import Colour
2626
from discord.utils import escape_markdown
2727

28-
2928
__all__ = (
3029
"to_codeblock",
3130
"random_pastel_colour",
@@ -38,7 +37,7 @@ def to_codeblock(
3837
replace_existing: bool = True,
3938
escape_md: bool = True,
4039
new: str = "'''",
41-
):
40+
) -> str:
4241
"""
4342
Quick function to put our content into a Discord accepted codeblock.
4443
"""

core/utils/logging.py

-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,6 @@
99

1010
import core
1111

12-
1312
if TYPE_CHECKING:
1413
from typing_extensions import Self
1514

core/utils/paginator.py

+10-11
Original file line numberDiff line numberDiff line change
@@ -30,13 +30,12 @@
3030
from discord.ext.commands import CommandError, Paginator as _Paginator # type: ignore # why does this need a stub file?
3131
from discord.utils import MISSING
3232

33-
from core import Bot, Context
34-
35-
3633
if TYPE_CHECKING:
3734
from discord.abc import MessageableChannel
3835
from typing_extensions import Self
3936

37+
from core import Bot, Context
38+
4039

4140
__all__ = ("CannotPaginate", "Pager", "KVPager", "TextPager")
4241

@@ -63,7 +62,7 @@ def __init__(
6362
author_url: str | None = None,
6463
stop: bool = False,
6564
reply_author_takes_paginator: bool = False,
66-
):
65+
) -> None:
6766
super().__init__()
6867
self.bot: Bot = ctx.bot
6968
self.stoppable: bool = stop
@@ -117,7 +116,7 @@ def __init__(
117116
def setup_buttons(self) -> None:
118117
self.clear_items()
119118
for emoji, button in self.reaction_emojis:
120-
btn = ui.Button[Self](emoji=emoji)
119+
btn = ui.Button["Self"](emoji=emoji)
121120
btn.callback = button # type: ignore
122121
self.add_item(btn)
123122

@@ -175,7 +174,7 @@ async def show_page(
175174

176175
self.message = await self.channel.send(content=content, embed=embed, view=self)
177176

178-
async def checked_show_page(self, page: int):
177+
async def checked_show_page(self, page: int) -> None:
179178
if page != 0 and page <= self.maximum_pages:
180179
await self.show_page(page)
181180

@@ -226,7 +225,7 @@ def message_check(m: discord.Message) -> bool:
226225
except (AttributeError, discord.HTTPException):
227226
pass
228227

229-
async def stop_pages(self, interaction: discord.Interaction | None = None):
228+
async def stop_pages(self, interaction: discord.Interaction | None = None) -> None:
230229
"""stops the interactive pagination session"""
231230
if self.delete_after and self.message:
232231
await self.message.delete()
@@ -235,7 +234,7 @@ async def stop_pages(self, interaction: discord.Interaction | None = None):
235234

236235
stop = stop_pages # type: ignore
237236

238-
def _check(self, interaction: discord.Interaction):
237+
def _check(self, interaction: discord.Interaction) -> bool:
239238
if interaction.user.id != self.author.id:
240239
return False
241240

@@ -249,7 +248,7 @@ async def interaction_check(self, interaction: discord.Interaction) -> bool:
249248

250249
return resp
251250

252-
async def paginate(self, msg_kwargs: dict[str, Any] | None = None):
251+
async def paginate(self, msg_kwargs: dict[str, Any] | None = None) -> None:
253252
if self.maximum_pages > 1:
254253
self.paginating = True
255254

@@ -276,7 +275,7 @@ def __init__(
276275
title: str | None = None,
277276
embed_color: discord.Colour = discord.Colour.blurple(),
278277
**kwargs: Any,
279-
):
278+
) -> None:
280279
super().__init__(
281280
ctx,
282281
entries=entries,
@@ -288,7 +287,7 @@ def __init__(
288287
)
289288
self.description = description
290289

291-
def prepare_embed(self, entries: list[Any], page: int, *, first: bool = False):
290+
def prepare_embed(self, entries: list[Any], page: int, *, first: bool = False) -> None:
292291
self.embed.clear_fields()
293292
self.embed.description = self.description or MISSING
294293
self.embed.title = self.title or MISSING

modules/__init__.py

-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
import pathlib
22
from pkgutil import ModuleInfo, iter_modules
33

4-
54
_ext: list[ModuleInfo] = []
65
_ext.extend(
76
[

modules/admin.py

-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,6 @@
2828
from constants import GUILD_ID
2929
from core.context import Context
3030

31-
3231
LOGGER = logging.getLogger(__name__)
3332

3433

modules/api.py

+9-10
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,6 @@
3737
PAPIWebsocketSubscriptions,
3838
)
3939

40-
4140
LOGGER = logging.getLogger(__name__)
4241

4342

@@ -69,7 +68,7 @@ async def cog_unload(self) -> None:
6968
try:
7069
self.connection_task.cancel()
7170
except Exception as e:
72-
LOGGER.error(f'Unable to cancel Pythonista API connection_task in "cog_unload": {e}')
71+
LOGGER.error('Unable to cancel Pythonista API connection_task in "cog_unload": %s', e)
7372

7473
if self.is_connected():
7574
assert self.websocket
@@ -79,7 +78,7 @@ async def cog_unload(self) -> None:
7978
try:
8079
self.keep_alive_task.cancel()
8180
except Exception as e:
82-
LOGGER.error(f'Unable to cancel Pythonista API keep_alive_task in "cog_unload": {e}')
81+
LOGGER.error('Unable to cancel Pythonista API keep_alive_task in "cog_unload": %s', e)
8382

8483
def dispatch(self, *, data: dict[str, Any]) -> None:
8584
subscription: str = data["subscription"]
@@ -99,7 +98,7 @@ async def connect(self) -> None:
9998
try:
10099
self.keep_alive_task.cancel()
101100
except Exception as e:
102-
LOGGER.warning(f"Failed to cancel Pythonista API Websocket keep alive. This is likely not a problem: {e}")
101+
LOGGER.warning("Failed to cancel Pythonista API Websocket keep alive. This is likely not a problem: %s", e)
103102

104103
while True:
105104
try:
@@ -109,13 +108,13 @@ async def connect(self) -> None:
109108
LOGGER.critical("Unable to connect to Pythonista API Websocket, due to an incorrect token.")
110109
return
111110
else:
112-
LOGGER.error(f"Unable to connect to Pythonista API Websocket: {e}.")
111+
LOGGER.error("Unable to connect to Pythonista API Websocket: %s.", e)
113112

114113
if self.is_connected():
115114
break
116115
else:
117116
delay: float = self.backoff.delay() # type: ignore
118-
LOGGER.warning(f'Retrying Pythonista API Websocket connection in "{delay}" seconds.')
117+
LOGGER.warning("Retrying Pythonista API Websocket connection in '%s' seconds.", delay)
119118

120119
await asyncio.sleep(delay)
121120

@@ -149,7 +148,7 @@ async def keep_alive(self) -> None:
149148
op: int | None = data.get("op")
150149

151150
if op == PAPIWebsocketOPCodes.HELLO:
152-
LOGGER.debug(f'Received HELLO from Pythonista API: user={data["user_id"]}')
151+
LOGGER.debug("Received HELLO from Pythonista API: user=%s", data["user_id"])
153152

154153
elif op == PAPIWebsocketOPCodes.EVENT:
155154
self.dispatch(data=data)
@@ -159,12 +158,12 @@ async def keep_alive(self) -> None:
159158

160159
if type_ == PAPIWebsocketNotificationTypes.SUBSCRIPTION_ADDED:
161160
subscribed: str = ", ".join(data["subscriptions"])
162-
LOGGER.info(f"Pythonista API added our subscription, currently subscribed: `{subscribed}`")
161+
LOGGER.info("Pythonista API added our subscription, currently subscribed: `%s`", subscribed)
163162
elif type_ == PAPIWebsocketNotificationTypes.SUBSCRIPTION_REMOVED:
164163
subscribed: str = ", ".join(data["subscriptions"])
165-
LOGGER.info(f"Pythonista API removed our subscription, currently subscribed: `{subscribed}`")
164+
LOGGER.info("Pythonista API removed our subscription, currently subscribed: `%s`", subscribed)
166165
elif type_ == PAPIWebsocketNotificationTypes.UNKNOWN_OP:
167-
LOGGER.info(f'We sent an UNKNOWN OP to Pythonista API: `{data["received"]}`')
166+
LOGGER.info("We sent an UNKNOWN OP to Pythonista API: `%s`", data["received"])
168167

169168
else:
170169
LOGGER.info("Received an UNKNOWN OP from Pythonista API.")

modules/eval.py

-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,6 @@
3333
from core.errors import InvalidEval
3434
from core.utils import formatters
3535

36-
3736
LOGGER = logging.getLogger(__name__)
3837

3938

modules/github.py

+4-4
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,6 @@
3131
import constants
3232
import core
3333

34-
3534
GITHUB_ISSUE_URL = "https://github.com/{}/issues/{}"
3635
LIB_ISSUE_REGEX = re.compile(r"(?P<lib>[a-z]+)?(?P<pounds>#{2,})(?P<number>[0-9]+)", flags=re.IGNORECASE)
3736
GITHUB_CODE_REGION_REGEX = re.compile(
@@ -166,8 +165,9 @@ async def format_highlight_block(self, url: str, line_adjustment: int = 10) -> d
166165

167166
github_dict = {
168167
"path": file_path,
169-
"min": (_min_boundary if _min_boundary > 0 else highlighted_line - 1)
170-
+ 1, # Do not display negative numbers if <0
168+
"min": (
169+
_min_boundary if _min_boundary > 0 else highlighted_line - 1
170+
) + 1, # Do not display negative numbers if <0
171171
"max": _max_boundary + 1,
172172
"msg": msg,
173173
}
@@ -182,7 +182,7 @@ def _smart_guess_lib(self, msg: discord.Message) -> LibEnum | None:
182182
return None # there's not much hope here, stay quick
183183

184184
if isinstance(msg.channel, discord.Thread) and msg.channel.parent_id == constants.Channels.HELP_FORUM:
185-
tags = set(x.name for x in msg.channel.applied_tags)
185+
tags = {x.name for x in msg.channel.applied_tags}
186186

187187
if "twitchio-help" in tags:
188188
return LibEnum.twitchio

0 commit comments

Comments
 (0)