battle4suisse/bot.py

256 lines
9.8 KiB
Python
Executable File

#!/usr/bin/env python3
from collections import namedtuple
import copy
from functools import partial
import json
from pathlib import Path
import random
from typing import Literal
from xml.dom import minidom
import cairosvg
import discord
from discord.ext import commands
from config import *
CANTONS = {
"AG": "Argovie",
"AI": "Appenzell Rhodes-Intérieures",
"AR": "Appenzell Rhodes-Extérieures",
"BE": "Berne",
"BL": "Bâle-Campagne",
"BS": "Bâle-Ville",
"FR": "Fribourg",
"GE": "Genève",
"GL": "Glaris",
"GR": "Grisons",
"JU": "Jura",
"LU": "Lucerne",
"NE": "Neuchâtel",
"NW": "Nidwald",
"OW": "Obwald",
"SG": "Saint-Gall",
"SH": "Schaffhouse",
"SO": "Soleure",
"SZ": "Schwytz",
"TH": "Thurgovie",
"TI": "Tessin",
"UR": "Uri",
"VD": "Vaud",
"VS": "Valais",
"ZG": "Zoug",
"ZH": "Zurich",
}
CodeCanton = Literal["AG", "AI", "AR", "BE", "BL", "BS", "FR", "GE", "GL", "GR", "JU", "LU", "NE",
"NW", "OW", "SG", "SH", "SO", "SZ", "TH", "TI", "UR", "VD", "VS", "ZG", "ZH"]
Couleur = Literal["rouge", "vert"]
intents = discord.Intents.default()
intents.message_content = True
bot = commands.Bot(command_prefix='$', intents=intents)
DATA_FILE = Path(__file__).parent / "data.json"
if DATA_FILE.exists():
with DATA_FILE.open() as data_file:
data = json.load(data_file)
else:
data = {
'equipes': {'rouge': [], 'vert': []},
'cantons': {code_canton: {'capture': None, 'verrouille': False} for code_canton in CANTONS.keys()},
'defis': {
'mains': {'rouge': [], 'vert': []},
'tires_capture': [],
'tires_competition': [],
}
}
with DATA_FILE.open('w') as data_file:
json.dump(data, data_file, indent=2)
DEFIS_FILE = Path(__file__).parent / "defis.json"
with DEFIS_FILE.open() as defis_file:
DEFIS = json.load(defis_file)
def generer_carte():
doc = minidom.parse("map_blank.svg")
for code_canton, data_canton in data['cantons'].items():
if data_canton['capture']:
path = next(e for e in doc.getElementsByTagName('path') if e.getAttribute('id') == code_canton)
couleur = data_canton['capture']
if data_canton['verrouille']:
path.setAttribute('fill', f"url(#verrouille-{couleur})")
else:
path.setAttribute('class', f"capture-{couleur}")
with open('map.svg', 'w') as f:
doc.writexml(f)
cairosvg.svg2png(url='map.svg', write_to='map.png')
@bot.command()
async def carte(ctx: commands.Context):
rouges = list(canton_code for canton_code, canton in data['cantons'].items()
if canton['capture'] == "rouge")
rouges_verrouilles = list(canton_code for canton_code, canton in data['cantons'].items()
if canton['capture'] == "rouge" and canton['verrouille'])
noms_rouges = ", ".join(code_canton + (":lock:" if code_canton in rouges_verrouilles else "") for code_canton in rouges)
verts = list(canton_code for canton_code, canton in data['cantons'].items()
if canton['capture'] == "vert")
verts_verrouilles = list(canton_code for canton_code, canton in data['cantons'].items()
if canton['capture'] == "vert" and canton['verrouille'])
noms_verts = ", ".join(code_canton + (":lock:" if code_canton in verts_verrouilles else "") for code_canton in verts)
libres = list(canton_code for canton_code, canton in data['cantons'].items()
if canton['capture'] is None)
message = f""":red_circle: Équipe rouge : **{len(rouges)} canton{"s" if len(rouges) > 1 else ""}** (dont **{len(rouges_verrouilles)} verrouillé{"s" if len(rouges_verrouilles) > 1 else ""}**) : {noms_rouges}
:green_circle: Équipe verte : **{len(verts)} canton{"s" if len(verts) > 1 else ""}** (dont **{len(verts_verrouilles)} verrouillé{"s" if len(verts_verrouilles) > 1 else ""}**) : {noms_verts}
:white_circle: **{len(libres)} canton{"s" if len(libres) > 1 else ""}** libre{"s" if len(libres) > 1 else ""} : {", ".join(libres)}"""
generer_carte()
with open('map.png', 'rb') as f:
await ctx.send(message, file=discord.File(f, filename="battle4suisse.png"))
@bot.command()
async def capturer(ctx: commands.Context, canton: CodeCanton, *, couleur: Couleur | None = None):
if couleur is None:
author_id = ctx.author.id
for couleur, membres_equipe in data['equipes'].items():
if author_id in membres_equipe:
break
else:
raise commands.BadArgument("Vous n'appartez à aucune équipe. Merci de faire `$equipe [rouge|vert]`.")
data['cantons'][canton]['capture'] = couleur
with DATA_FILE.open('w') as data_file:
json.dump(data, data_file, indent=2)
await ctx.send(f"@everyone L'équipe {couleur} a capturé le canton de **{CANTONS[canton]}** !")
return await carte(ctx)
@capturer.error
async def capture_error(ctx, error):
if isinstance(error, commands.BadLiteralArgument):
await ctx.send(f"Canton inconnu : {error.argument}, valeurs possibles : {", ".join(error.literals)}")
else:
await ctx.send(str(error))
@bot.command()
async def verrouiller(ctx: commands.Context, canton: CodeCanton, *, couleur: Couleur | None = None):
if couleur is None:
author_id = ctx.author.id
for couleur, membres_equipe in data['equipes'].items():
if author_id in membres_equipe:
break
else:
raise commands.BadArgument("Vous n'appartez à aucune équipe. Merci de faire `$equipe [rouge|vert]`.")
data['cantons'][canton]['capture'] = couleur
data['cantons'][canton]['verrouille'] = True
with DATA_FILE.open('w') as data_file:
json.dump(data, data_file, indent=2)
generer_carte()
await ctx.send(f"@everyone L'équipe {couleur} a capturé le canton de **{CANTONS[canton]}** !")
return await carte(ctx)
@bot.command()
async def reset(ctx: commands.Context, canton: CodeCanton):
data['cantons'][canton]['capture'] = None
data['cantons'][canton]['verrouille'] = False
with DATA_FILE.open('w') as data_file:
json.dump(data, data_file, indent=2)
generer_carte()
return await carte(ctx)
@bot.command()
async def equipe(ctx: commands.Context, couleur: Couleur):
author_id = ctx.author.id
for membres_equipe in data['equipes'].values():
if author_id in membres_equipe:
membres_equipe.remove(author_id)
data['equipes'][couleur].append(author_id)
with DATA_FILE.open('w') as data_file:
json.dump(data, data_file, indent=2)
await ctx.send(f"Équipe {couleur} rejointe")
@bot.command()
async def defis(ctx: commands.Context, *, type_defi: Literal['capture', 'competition'] = "capture"):
await ctx.send(f"Liste des défis de {type_defi} :\n" + "\n".join(f"* {defi['id']} : {defi['nom']}" for defi in DEFIS[type_defi]))
@bot.command()
async def description(ctx: commands.Context, type_defi: Literal['capture', 'competition'] = "capture"):
defis = DEFIS[type_defi]
embeds = []
for page in range((len(defis) - 1) // 25 + 1):
defis_page = defis[page * 25:(page + 1) * 25]
embed = discord.Embed(title=f"Description des défis", colour=discord.Colour.gold())
embed.set_footer(f"Page {page}/{(len(defis) - 1) // 25 + 1}")
for defi in defis_page:
embed.add_field(name=f"{defi['nom']} (n°{defi['id']})", value=defi['description'], inline=False)
embeds.append(embed)
await ctx.send(embeds=embeds)
@bot.command()
async def tirage(ctx: commands.Context, nb_defis: int = 7):
if data['defis']['mains']['rouge'] or data['defis']['mains']['vert']:
raise commands.BadArgument("Les mains sont déjà initialisées")
defis_libres = copy.deepcopy(DEFIS['capture'])
for equipe in ('rouge', 'vert'):
for _i in range(nb_defis):
defi = random.choice(defis_libres)
defis_libres.remove(defi)
data['defis']['mains'][equipe].append(defi['id'])
data['defis']['tires_capture'].append(defi['id'])
main = data['defis']['mains'][equipe]
embeds = []
colour = discord.Color.red() if equipe == "rouge" else discord.Color.green()
for id_defi in main:
defi = next(defi for defi in DEFIS['capture'] if defi['id'] == id_defi)
embed = discord.Embed(title=defi['nom'], description=defi['description'], colour=colour)
embed.set_footer(text=f"Défi n°{defi['id']}")
embeds.append(embed)
for member_id in data['equipes'][equipe]:
channel_dm = await bot.create_dm(namedtuple("User", "id")(member_id))
await channel_dm.send("Vos défis en main :", embeds=embeds)
with DATA_FILE.open('w') as data_file:
json.dump(data, data_file, indent=2)
await ctx.send("Les mains de départ ont bien été tirées ! Le contenu vous a été envoyé en MP.")
@bot.command()
async def main(ctx: commands.Context):
author_id = ctx.author.id
for couleur, membres_equipe in data['equipes'].items():
if author_id in membres_equipe:
break
else:
raise commands.BadArgument("Vous n'appartez à aucune équipe. Merci de faire `$equipe [rouge|vert]`.")
main = data['defis']['mains'][couleur]
embeds = []
colour = discord.Color.red() if couleur == "rouge" else discord.Color.green()
for id_defi in main:
defi = next(defi for defi in DEFIS['capture'] if defi['id'] == id_defi)
embed = discord.Embed(title=defi['nom'], description=defi['description'], colour=colour)
embed.set_footer(text=f"Défi n°{defi['id']}")
embeds.append(embed)
channel_dm = await bot.create_dm(ctx.author)
await channel_dm.send("Vos défis en main :", embeds=embeds)
@equipe.error
async def equipe_error(ctx, error):
await ctx.send(str(error))
bot.run(DISCORD_TOKEN)