Fixed ontime calculation #166
This commit is contained in:
parent
c443d108dc
commit
706b6732eb
@ -4,6 +4,8 @@ from typing import Callable
|
|||||||
from cpl_query.extension import List
|
from cpl_query.extension import List
|
||||||
from discord.ext.commands import Context
|
from discord.ext.commands import Context
|
||||||
|
|
||||||
|
from bot_data.model.user import User
|
||||||
|
|
||||||
|
|
||||||
class ClientUtilsServiceABC(ABC):
|
class ClientUtilsServiceABC(ABC):
|
||||||
|
|
||||||
@ -33,3 +35,6 @@ class ClientUtilsServiceABC(ABC):
|
|||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def get_auto_complete_list(self, _l: List, current: str, select: Callable = None) -> List: pass
|
def get_auto_complete_list(self, _l: List, current: str, select: Callable = None) -> List: pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def get_ontime_for_user(self, user: User) -> float: pass
|
||||||
|
@ -17,6 +17,8 @@ from bot_core.configuration.feature_flags_enum import FeatureFlagsEnum
|
|||||||
from bot_core.configuration.feature_flags_settings import FeatureFlagsSettings
|
from bot_core.configuration.feature_flags_settings import FeatureFlagsSettings
|
||||||
from bot_data.abc.client_repository_abc import ClientRepositoryABC
|
from bot_data.abc.client_repository_abc import ClientRepositoryABC
|
||||||
from bot_data.abc.server_repository_abc import ServerRepositoryABC
|
from bot_data.abc.server_repository_abc import ServerRepositoryABC
|
||||||
|
from bot_data.abc.user_joined_voice_channel_abc import UserJoinedVoiceChannelRepositoryABC
|
||||||
|
from bot_data.model.user import User
|
||||||
|
|
||||||
|
|
||||||
class ClientUtilsService(ClientUtilsServiceABC):
|
class ClientUtilsService(ClientUtilsServiceABC):
|
||||||
@ -28,6 +30,7 @@ class ClientUtilsService(ClientUtilsServiceABC):
|
|||||||
bot: DiscordBotServiceABC,
|
bot: DiscordBotServiceABC,
|
||||||
servers: ServerRepositoryABC,
|
servers: ServerRepositoryABC,
|
||||||
clients: ClientRepositoryABC,
|
clients: ClientRepositoryABC,
|
||||||
|
user_joined_vc: UserJoinedVoiceChannelRepositoryABC,
|
||||||
message_service: MessageServiceABC,
|
message_service: MessageServiceABC,
|
||||||
db: DatabaseContextABC,
|
db: DatabaseContextABC,
|
||||||
t: TranslatePipe,
|
t: TranslatePipe,
|
||||||
@ -39,6 +42,7 @@ class ClientUtilsService(ClientUtilsServiceABC):
|
|||||||
self._bot = bot
|
self._bot = bot
|
||||||
self._servers = servers
|
self._servers = servers
|
||||||
self._clients = clients
|
self._clients = clients
|
||||||
|
self._user_joined_voice_channel = user_joined_vc
|
||||||
self._message_service = message_service
|
self._message_service = message_service
|
||||||
self._db = db
|
self._db = db
|
||||||
self._t = t
|
self._t = t
|
||||||
@ -80,7 +84,8 @@ class ClientUtilsService(ClientUtilsServiceABC):
|
|||||||
async def check_if_bot_is_ready_yet_and_respond(self, ctx: Context) -> bool:
|
async def check_if_bot_is_ready_yet_and_respond(self, ctx: Context) -> bool:
|
||||||
result = await self.check_if_bot_is_ready_yet()
|
result = await self.check_if_bot_is_ready_yet()
|
||||||
if not result:
|
if not result:
|
||||||
await self._message_service.send_ctx_msg(ctx, self._t.transform('common.errors.bot_not_ready_yet'), without_tracking=True)
|
await self._message_service.send_ctx_msg(ctx, self._t.transform('common.errors.bot_not_ready_yet'),
|
||||||
|
without_tracking=True)
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
@ -106,3 +111,8 @@ class ClientUtilsService(ClientUtilsServiceABC):
|
|||||||
_l = _l.where(lambda x: x.name in sl)
|
_l = _l.where(lambda x: x.name in sl)
|
||||||
|
|
||||||
return _l.take(25)
|
return _l.take(25)
|
||||||
|
|
||||||
|
def get_ontime_for_user(self, user: User) -> float:
|
||||||
|
return self._user_joined_voice_channel.get_user_joined_voice_channels_by_user_id(user.user_id) \
|
||||||
|
.where(lambda x: x.leaved_on is not None and x.joined_on is not None) \
|
||||||
|
.sum(lambda join: round((join.leaved_on - join.joined_on).total_seconds() / 3600, 2))
|
||||||
|
@ -98,10 +98,6 @@ class UserGroup(DiscordCommandABC):
|
|||||||
color=int('ef9d0d', 16)
|
color=int('ef9d0d', 16)
|
||||||
)
|
)
|
||||||
|
|
||||||
ontime = self._user_joined_voice_channel.get_user_joined_voice_channels_by_user_id(user.user_id)\
|
|
||||||
.where(lambda x: x.leaved_on is not None and x.joined_on is not None)\
|
|
||||||
.sum(lambda join: round((join.leaved_on - join.joined_on).total_seconds() / 3600, 2))
|
|
||||||
|
|
||||||
embed.add_field(name=self._t.transform('modules.base.user.atr.id'), value=member.id)
|
embed.add_field(name=self._t.transform('modules.base.user.atr.id'), value=member.id)
|
||||||
embed.add_field(name=self._t.transform('modules.base.user.atr.name'), value=member.name)
|
embed.add_field(name=self._t.transform('modules.base.user.atr.name'), value=member.name)
|
||||||
embed.add_field(name=self._t.transform('modules.base.user.atr.discord_join'),
|
embed.add_field(name=self._t.transform('modules.base.user.atr.discord_join'),
|
||||||
@ -109,7 +105,8 @@ class UserGroup(DiscordCommandABC):
|
|||||||
embed.add_field(name=self._t.transform('modules.base.user.atr.last_join'),
|
embed.add_field(name=self._t.transform('modules.base.user.atr.last_join'),
|
||||||
value=self._date.transform(member.joined_at), inline=False)
|
value=self._date.transform(member.joined_at), inline=False)
|
||||||
embed.add_field(name=self._t.transform('modules.base.user.atr.xp'), value=str(user.xp))
|
embed.add_field(name=self._t.transform('modules.base.user.atr.xp'), value=str(user.xp))
|
||||||
embed.add_field(name=self._t.transform('modules.base.user.atr.ontime'), value=str(ontime))
|
embed.add_field(name=self._t.transform('modules.base.user.atr.ontime'),
|
||||||
|
value=str(self._client_utils.get_ontime_for_user(user)))
|
||||||
|
|
||||||
roles = ''
|
roles = ''
|
||||||
for role in member.roles:
|
for role in member.roles:
|
||||||
@ -169,7 +166,8 @@ class UserGroup(DiscordCommandABC):
|
|||||||
))
|
))
|
||||||
|
|
||||||
else:
|
else:
|
||||||
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform('modules.base.user.error.atr_not_found').format(atr))
|
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform(
|
||||||
|
'modules.base.user.error.atr_not_found').format(atr))
|
||||||
return
|
return
|
||||||
|
|
||||||
await self._message_service.send_interaction_msg(
|
await self._message_service.send_interaction_msg(
|
||||||
@ -196,14 +194,16 @@ class UserGroup(DiscordCommandABC):
|
|||||||
|
|
||||||
if atr == 'xp':
|
if atr == 'xp':
|
||||||
if not value.isnumeric():
|
if not value.isnumeric():
|
||||||
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform('modules.base.user.set.error.value_type_not_numeric'))
|
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform(
|
||||||
|
'modules.base.user.set.error.value_type_not_numeric'))
|
||||||
return
|
return
|
||||||
|
|
||||||
try:
|
try:
|
||||||
user.xp = int(value)
|
user.xp = int(value)
|
||||||
except TypeError as te:
|
except TypeError as te:
|
||||||
self._logger.error(__name__, f'String value couldn\'t be converted to int', te)
|
self._logger.error(__name__, f'String value couldn\'t be converted to int', te)
|
||||||
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform('modules.base.user.set.error.type_error'))
|
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform(
|
||||||
|
'modules.base.user.set.error.type_error'))
|
||||||
return
|
return
|
||||||
else:
|
else:
|
||||||
self._users.update_user(user)
|
self._users.update_user(user)
|
||||||
@ -211,10 +211,12 @@ class UserGroup(DiscordCommandABC):
|
|||||||
await self._level.check_level(member)
|
await self._level.check_level(member)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform('modules.base.user.error.atr_not_found').format(atr))
|
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform(
|
||||||
|
'modules.base.user.error.atr_not_found').format(atr))
|
||||||
return
|
return
|
||||||
|
|
||||||
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform(f'modules.base.user.set.{atr.lower()}').format(member.mention, value))
|
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform(
|
||||||
|
f'modules.base.user.set.{atr.lower()}').format(member.mention, value))
|
||||||
|
|
||||||
@set.autocomplete('atr')
|
@set.autocomplete('atr')
|
||||||
async def set_autocomplete(self, interaction: discord.Interaction, current: str) -> List[app_commands.Choice[str]]:
|
async def set_autocomplete(self, interaction: discord.Interaction, current: str) -> List[app_commands.Choice[str]]:
|
||||||
@ -245,7 +247,8 @@ class UserGroup(DiscordCommandABC):
|
|||||||
self._db.save_changes()
|
self._db.save_changes()
|
||||||
|
|
||||||
else:
|
else:
|
||||||
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform('modules.base.user.error.atr_not_found').format(atr))
|
await self._message_service.send_interaction_msg(ctx.interaction, self._t.transform(
|
||||||
|
'modules.base.user.error.atr_not_found').format(atr))
|
||||||
return
|
return
|
||||||
|
|
||||||
await self._message_service.send_interaction_msg(
|
await self._message_service.send_interaction_msg(
|
||||||
@ -254,5 +257,6 @@ class UserGroup(DiscordCommandABC):
|
|||||||
)
|
)
|
||||||
|
|
||||||
@remove.autocomplete('atr')
|
@remove.autocomplete('atr')
|
||||||
async def remove_autocomplete(self, interaction: discord.Interaction, current: str) -> List[app_commands.Choice[str]]:
|
async def remove_autocomplete(self, interaction: discord.Interaction, current: str) -> List[
|
||||||
|
app_commands.Choice[str]]:
|
||||||
return [app_commands.Choice(name=value, value=key) for key, value in self._atr_list]
|
return [app_commands.Choice(name=value, value=key) for key, value in self._atr_list]
|
||||||
|
@ -7,6 +7,7 @@ from cpl_core.database.context import DatabaseContextABC
|
|||||||
from cpl_core.logging import LoggerABC
|
from cpl_core.logging import LoggerABC
|
||||||
from cpl_discord.events import OnVoiceStateUpdateABC
|
from cpl_discord.events import OnVoiceStateUpdateABC
|
||||||
|
|
||||||
|
from bot_core.abc.client_utils_service_abc import ClientUtilsServiceABC
|
||||||
from bot_core.helper.event_checks import EventChecks
|
from bot_core.helper.event_checks import EventChecks
|
||||||
from bot_data.abc.known_user_repository_abc import KnownUserRepositoryABC
|
from bot_data.abc.known_user_repository_abc import KnownUserRepositoryABC
|
||||||
from bot_data.abc.server_repository_abc import ServerRepositoryABC
|
from bot_data.abc.server_repository_abc import ServerRepositoryABC
|
||||||
@ -32,6 +33,7 @@ class BaseOnVoiceStateUpdateEvent(OnVoiceStateUpdateABC):
|
|||||||
users: UserRepositoryABC,
|
users: UserRepositoryABC,
|
||||||
user_joins: UserJoinedServerRepositoryABC,
|
user_joins: UserJoinedServerRepositoryABC,
|
||||||
user_joins_vc: UserJoinedVoiceChannelRepositoryABC,
|
user_joins_vc: UserJoinedVoiceChannelRepositoryABC,
|
||||||
|
client_utils: ClientUtilsServiceABC,
|
||||||
db: DatabaseContextABC,
|
db: DatabaseContextABC,
|
||||||
):
|
):
|
||||||
OnVoiceStateUpdateABC.__init__(self)
|
OnVoiceStateUpdateABC.__init__(self)
|
||||||
@ -43,6 +45,7 @@ class BaseOnVoiceStateUpdateEvent(OnVoiceStateUpdateABC):
|
|||||||
self._users = users
|
self._users = users
|
||||||
self._user_joins = user_joins
|
self._user_joins = user_joins
|
||||||
self._user_joins_vc = user_joins_vc
|
self._user_joins_vc = user_joins_vc
|
||||||
|
self._client_utils = client_utils
|
||||||
self._db = db
|
self._db = db
|
||||||
|
|
||||||
self._logger.info(__name__, f'Module {type(self)} loaded')
|
self._logger.info(__name__, f'Module {type(self)} loaded')
|
||||||
@ -72,7 +75,7 @@ class BaseOnVoiceStateUpdateEvent(OnVoiceStateUpdateABC):
|
|||||||
join.leaved_on = datetime.now()
|
join.leaved_on = datetime.now()
|
||||||
|
|
||||||
# ontime as hours
|
# ontime as hours
|
||||||
ontime = round((join.leaved_on - join.joined_on).total_seconds() / 3600, 2)
|
ontime = self._client_utils.get_ontime_for_user(user)
|
||||||
old_xp = user.xp
|
old_xp = user.xp
|
||||||
user.xp += round(ontime * settings.xp_per_ontime_hour)
|
user.xp += round(ontime * settings.xp_per_ontime_hour)
|
||||||
|
|
||||||
@ -80,12 +83,14 @@ class BaseOnVoiceStateUpdateEvent(OnVoiceStateUpdateABC):
|
|||||||
self._users.update_user(user)
|
self._users.update_user(user)
|
||||||
self._db.save_changes()
|
self._db.save_changes()
|
||||||
|
|
||||||
self._logger.debug(__name__, f'User {user} leaved_on {join.leaved_on}. Ontime: {ontime}h | xp: from {old_xp} to {user.xp}')
|
self._logger.debug(__name__,
|
||||||
|
f'User {user} leaved_on {join.leaved_on}. Ontime: {ontime}h | xp: from {old_xp} to {user.xp}')
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self._logger.error(__name__, f'Ontime validation failed', e)
|
self._logger.error(__name__, f'Ontime validation failed', e)
|
||||||
|
|
||||||
@EventChecks.check_is_ready()
|
@EventChecks.check_is_ready()
|
||||||
async def on_voice_state_update(self, member: discord.Member, before: discord.VoiceState, after: discord.VoiceState):
|
async def on_voice_state_update(self, member: discord.Member, before: discord.VoiceState,
|
||||||
|
after: discord.VoiceState):
|
||||||
self._logger.debug(__name__, f'Module {type(self)} started')
|
self._logger.debug(__name__, f'Module {type(self)} started')
|
||||||
self._logger.trace(__name__, f'Detected on_voice_state_update {member.id} from {before} to {after}')
|
self._logger.trace(__name__, f'Detected on_voice_state_update {member.id} from {before} to {after}')
|
||||||
settings: BaseServerSettings = self._base_helper.get_config(member.guild.id)
|
settings: BaseServerSettings = self._base_helper.get_config(member.guild.id)
|
||||||
|
Loading…
Reference in New Issue
Block a user