Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 15 additions & 1 deletion bot/exts/backend/sync/_cog.py
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
import asyncio
import datetime
from typing import Any, Dict

from botcore.site_api import ResponseCodeError
from discord import Member, Role, User
from discord.ext import commands
from discord.ext.commands import Cog, Context
from discord.ext.commands import Cog, Context, errors

from bot import constants
from bot.bot import Bot
Expand All @@ -27,6 +29,18 @@ async def cog_load(self) -> None:
if guild is None:
return

log.info("Waiting for guild to be chunked to start syncers.")
end = datetime.datetime.now() + datetime.timedelta(minutes=30)
while not guild.chunked:
await asyncio.sleep(10)
if datetime.datetime.now() > end:
# More than 30 minutes have passed while trying, abort
raise errors.ExtensionFailed(
self.__class__.__name__,
RuntimeError("The guild was not chunked in time, not loading sync cog.")
)

log.info("Starting syncers.")
for syncer in (_syncers.RoleSyncer, _syncers.UserSyncer):
await syncer.sync(guild)

Expand Down
13 changes: 11 additions & 2 deletions bot/exts/backend/sync/_syncers.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,14 +2,14 @@
import typing as t
from collections import namedtuple

import discord.errors
from botcore.site_api import ResponseCodeError
from discord import Guild
from discord.ext.commands import Context
from more_itertools import chunked

import bot
from bot.log import get_logger
from bot.utils.members import get_or_fetch_member

log = get_logger(__name__)

Expand Down Expand Up @@ -157,7 +157,16 @@ def maybe_update(db_field: str, guild_value: t.Union[str, int]) -> None:
if db_user[db_field] != guild_value:
updated_fields[db_field] = guild_value

if guild_user := await get_or_fetch_member(guild, db_user["id"]):
guild_user = guild.get_member(db_user["id"])
if not guild_user and db_user["in_guild"]:
# The member was in the guild during the last sync.
# We try to fetch them to verify cache integrity.
try:
guild_user = await guild.fetch_member(db_user["id"])
except discord.errors.NotFound:
guild_user = None

if guild_user:
seen_guild_users.add(guild_user.id)

maybe_update("name", guild_user.name)
Expand Down
2 changes: 1 addition & 1 deletion tests/helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -171,7 +171,7 @@ class MockGuild(CustomMockMixin, unittest.mock.Mock, HashableMixin):
spec_set = guild_instance

def __init__(self, roles: Optional[Iterable[MockRole]] = None, **kwargs) -> None:
default_kwargs = {'id': next(self.discord_id), 'members': []}
default_kwargs = {'id': next(self.discord_id), 'members': [], "chunked": True}
super().__init__(**collections.ChainMap(kwargs, default_kwargs))

self.roles = [MockRole(name="@everyone", position=1, id=0)]
Expand Down