1
Fork 0
mirror of https://github.com/RYGhub/royalnet.git synced 2024-11-24 03:54:20 +00:00
royalnet/royalpack/commands/leagueoflegends.py

226 lines
11 KiB
Python
Raw Normal View History

2020-05-10 22:46:12 +00:00
from typing import *
2019-11-11 08:56:08 +00:00
import riotwatcher
import logging
import asyncio
import sentry_sdk
2020-05-10 22:46:12 +00:00
import royalnet.commands as rc
import royalnet.utils as ru
import royalnet.serf.telegram as rst
from ..tables import LeagueOfLegends, FiorygiTransaction
2020-03-19 15:26:11 +00:00
from ..types import LeagueLeague
2019-11-11 08:56:08 +00:00
log = logging.getLogger(__name__)
2020-05-10 22:46:12 +00:00
class LeagueoflegendsCommand(rc.Command):
2019-11-11 08:56:08 +00:00
name: str = "leagueoflegends"
aliases = ["lol", "league"]
description: str = "Connetti un account di League of Legends a un account Royalnet, e visualizzane le statistiche."
syntax = "[nomeevocatore]"
2020-05-10 22:46:12 +00:00
def __init__(self, interface: rc.CommandInterface):
2019-11-11 08:56:08 +00:00
super().__init__(interface)
2020-07-08 00:55:15 +00:00
self._riotwatcher: Optional[riotwatcher.RiotWatcher] = None
if self.interface.name == "telegram" and self.config["Lol"]["updater"]["enabled"]:
self._riotwatcher = riotwatcher.RiotWatcher(api_key=self.config["Lol"]["token"])
self.loop.create_task(self._updater(int(self.config["Lol"]["updater"]["delay"])))
2019-11-11 08:56:08 +00:00
async def _send(self, message):
2019-11-29 14:55:52 +00:00
client = self.serf.client
await self.serf.api_call(client.send_message,
chat_id=self.config["Telegram"]["main_group_id"],
2020-05-10 22:46:12 +00:00
text=rst.escape(message),
2019-11-29 14:55:52 +00:00
parse_mode="HTML",
disable_webpage_preview=True)
2019-11-11 08:56:08 +00:00
async def _notify(self,
obj: LeagueOfLegends,
attribute_name: str,
2020-05-10 22:46:12 +00:00
old_value: Any,
new_value: Any):
2020-01-24 00:07:11 +00:00
if isinstance(old_value, LeagueLeague):
# This is a rank change!
# Don't send messages for every rank change, send messages just if the TIER or RANK changes!
if old_value.tier == new_value.tier and old_value.rank == new_value.rank:
return
# Find the queue
queue_names = {
"rank_soloq": "Solo/Duo",
"rank_flexq": "Flex",
"rank_twtrq": "3v3",
"rank_tftq": "TFT"
}
# Prepare the message
if new_value > old_value:
message = f"📈 [b]{obj.user}[/b] è salito a {new_value} su League of Legends " \
f"({queue_names[attribute_name]})! Congratulazioni!"
else:
message = f"📉 [b]{obj.user}[/b] è sceso a {new_value} su League of Legends " \
f"({queue_names[attribute_name]})."
# Send the message
await self._send(message)
# Level up!
elif attribute_name == "summoner_level":
if new_value == 30 or (new_value >= 50 and (new_value % 25 == 0)):
await self._send(f"🆙 [b]{obj.user}[/b] è salito al livello [b]{new_value}[/b] su League of Legends!")
2019-11-11 08:56:08 +00:00
@staticmethod
async def _change(obj: LeagueOfLegends,
attribute_name: str,
2020-05-10 22:46:12 +00:00
new_value: Any,
callback: Callable[
[LeagueOfLegends, str, Any, Any], Awaitable[None]]):
2019-11-11 08:56:08 +00:00
old_value = obj.__getattribute__(attribute_name)
if old_value != new_value:
await callback(obj, attribute_name, old_value, new_value)
obj.__setattr__(attribute_name, new_value)
async def _update(self, lol: LeagueOfLegends):
log.info(f"Updating: {lol}")
log.debug(f"Getting summoner data: {lol}")
2020-05-10 22:46:12 +00:00
summoner = await ru.asyncify(self._riotwatcher.summoner.by_id, region=self.config["Lol"]["region"],
encrypted_summoner_id=lol.summoner_id)
2019-11-11 08:56:08 +00:00
await self._change(lol, "profile_icon_id", summoner["profileIconId"], self._notify)
await self._change(lol, "summoner_name", summoner["name"], self._notify)
await self._change(lol, "puuid", summoner["puuid"], self._notify)
await self._change(lol, "summoner_level", summoner["summonerLevel"], self._notify)
await self._change(lol, "summoner_id", summoner["id"], self._notify)
await self._change(lol, "account_id", summoner["accountId"], self._notify)
log.debug(f"Getting leagues data: {lol}")
2020-05-10 22:46:12 +00:00
leagues = await ru.asyncify(self._riotwatcher.league.by_summoner, region=self.config["Lol"]["region"],
encrypted_summoner_id=lol.summoner_id)
2019-11-11 08:56:08 +00:00
soloq = LeagueLeague()
flexq = LeagueLeague()
twtrq = LeagueLeague()
tftq = LeagueLeague()
for league in leagues:
if league["queueType"] == "RANKED_SOLO_5x5":
soloq = LeagueLeague.from_dict(league)
if league["queueType"] == "RANKED_FLEX_SR":
flexq = LeagueLeague.from_dict(league)
if league["queueType"] == "RANKED_FLEX_TT":
twtrq = LeagueLeague.from_dict(league)
if league["queueType"] == "RANKED_TFT":
tftq = LeagueLeague.from_dict(league)
await self._change(lol, "rank_soloq", soloq, self._notify)
await self._change(lol, "rank_flexq", flexq, self._notify)
await self._change(lol, "rank_twtrq", twtrq, self._notify)
await self._change(lol, "rank_tftq", tftq, self._notify)
log.debug(f"Getting mastery data: {lol}")
2020-05-10 22:46:12 +00:00
mastery = await ru.asyncify(self._riotwatcher.champion_mastery.scores_by_summoner,
region=self.config["Lol"]["region"],
encrypted_summoner_id=lol.summoner_id)
2019-11-11 08:56:08 +00:00
await self._change(lol, "mastery_score", mastery, self._notify)
async def _updater(self, period: int):
log.info(f"Started updater with {period}s period")
while True:
log.info(f"Updating...")
session = self.alchemy.Session()
log.info("")
2019-11-30 13:49:23 +00:00
lols = session.query(self.alchemy.get(LeagueOfLegends)).all()
2019-11-11 08:56:08 +00:00
for lol in lols:
try:
await self._update(lol)
except Exception as e:
sentry_sdk.capture_exception(e)
log.error(f"Error while updating {lol.user.username}: {e}")
await asyncio.sleep(1)
2020-05-10 22:46:12 +00:00
await ru.asyncify(session.commit)
2019-11-11 08:56:08 +00:00
session.close()
log.info(f"Sleeping for {period}s")
await asyncio.sleep(period)
2019-11-29 14:55:52 +00:00
@staticmethod
def _display(lol: LeagueOfLegends) -> str:
2019-11-11 08:56:08 +00:00
string = f" [b]{lol.summoner_name}[/b]\n" \
f"Lv. {lol.summoner_level}\n" \
f"Mastery score: {lol.mastery_score}\n" \
f"\n"
if lol.rank_soloq:
string += f"Solo: {lol.rank_soloq}\n"
if lol.rank_flexq:
string += f"Flex: {lol.rank_flexq}\n"
if lol.rank_twtrq:
string += f"3v3: {lol.rank_twtrq}\n"
if lol.rank_tftq:
string += f"TFT: {lol.rank_tftq}\n"
return string
2020-05-10 22:46:12 +00:00
async def run(self, args: rc.CommandArgs, data: rc.CommandData) -> None:
2019-11-11 08:56:08 +00:00
author = await data.get_author(error_if_none=True)
name = args.joined()
if name:
# Connect a new League of Legends account to Royalnet
log.debug(f"Searching for: {name}")
2019-11-29 14:55:52 +00:00
summoner = self._riotwatcher.summoner.by_name(region=self.config["Lol"]["region"], summoner_name=name)
2019-11-11 08:56:08 +00:00
# Ensure the account isn't already connected to something else
2020-05-10 22:46:12 +00:00
leagueoflegends = await ru.asyncify(
2019-11-30 13:49:23 +00:00
data.session.query(self.alchemy.get(LeagueOfLegends)).filter_by(summoner_id=summoner["id"]).one_or_none)
2019-11-11 08:56:08 +00:00
if leagueoflegends:
2020-05-10 22:46:12 +00:00
raise rc.CommandError(f"L'account {leagueoflegends} è già registrato su Royalnet.")
2019-11-11 08:56:08 +00:00
# Get rank information
log.debug(f"Getting leagues data: {name}")
2019-11-29 14:55:52 +00:00
leagues = self._riotwatcher.league.by_summoner(region=self.config["Lol"]["region"],
encrypted_summoner_id=summoner["id"])
2019-11-11 08:56:08 +00:00
soloq = LeagueLeague()
flexq = LeagueLeague()
twtrq = LeagueLeague()
tftq = LeagueLeague()
for league in leagues:
if league["queueType"] == "RANKED_SOLO_5x5":
soloq = LeagueLeague.from_dict(league)
if league["queueType"] == "RANKED_FLEX_SR":
flexq = LeagueLeague.from_dict(league)
if league["queueType"] == "RANKED_FLEX_TT":
twtrq = LeagueLeague.from_dict(league)
if league["queueType"] == "RANKED_TFT":
tftq = LeagueLeague.from_dict(league)
# Get mastery score
log.debug(f"Getting mastery data: {name}")
2019-11-29 14:55:52 +00:00
mastery = self._riotwatcher.champion_mastery.scores_by_summoner(region=self.config["Lol"]["region"],
2019-11-11 08:56:08 +00:00
encrypted_summoner_id=summoner["id"])
# Create database row
2019-11-30 13:49:23 +00:00
leagueoflegends = self.alchemy.get(LeagueOfLegends)(
2019-11-29 14:55:52 +00:00
region=self.config["Lol"]["region"],
2019-11-11 08:56:08 +00:00
user=author,
profile_icon_id=summoner["profileIconId"],
summoner_name=summoner["name"],
puuid=summoner["puuid"],
summoner_level=summoner["summonerLevel"],
summoner_id=summoner["id"],
account_id=summoner["accountId"],
rank_soloq=soloq,
rank_flexq=flexq,
rank_twtrq=twtrq,
rank_tftq=tftq,
mastery_score=mastery
)
log.debug(f"Saving to the DB: {name}")
data.session.add(leagueoflegends)
await data.session_commit()
await data.reply(f"↔️ Account {leagueoflegends} connesso a {author}!")
await FiorygiTransaction.spawn_fiorygi(data, author, 1,
"aver connesso il proprio account di League of Legends a Royalnet")
2019-11-11 08:56:08 +00:00
else:
# Update and display the League of Legends stats for the current account
if len(author.leagueoflegends) == 0:
2020-05-10 22:46:12 +00:00
raise rc.UserError("Nessun account di League of Legends trovato.")
2019-11-11 08:56:08 +00:00
message = ""
for account in author.leagueoflegends:
try:
await self._update(account)
message += self._display(account)
except riotwatcher.ApiError as e:
message += f"⚠️ [b]{account.summoner_name}[/b]\n" \
f"{e}"
message += "\n"
await data.session_commit()
await data.reply(message)