2015-10-07 08:49:14 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
2016-02-01 16:31:03 +00:00
|
|
|
import time
|
|
|
|
import filemanager
|
2015-10-03 16:41:34 +00:00
|
|
|
import telegram
|
|
|
|
import steam
|
2015-10-08 16:39:56 +00:00
|
|
|
import random
|
2015-11-16 21:55:47 +00:00
|
|
|
import osu
|
2015-12-20 19:05:13 +00:00
|
|
|
import hearthstone
|
|
|
|
import sys
|
2016-01-28 17:55:42 +00:00
|
|
|
import mumbleboxes
|
2015-10-08 16:39:56 +00:00
|
|
|
|
2016-01-21 21:16:47 +00:00
|
|
|
|
|
|
|
# Check per la modalità votazione del bot, corrisponde al numero della chat in cui è attiva la votazione
|
|
|
|
# 0 per disattivare la votazione
|
|
|
|
class Votazione:
|
|
|
|
chat = int()
|
|
|
|
domanda = str()
|
|
|
|
# 0: non votato
|
|
|
|
# 1: sì
|
|
|
|
# 2: no
|
|
|
|
# 3: astenuto
|
|
|
|
voto = {
|
|
|
|
'steffo': int(0),
|
|
|
|
'alby1': int(0),
|
|
|
|
'boni3099': int(0),
|
|
|
|
'maxsensei': int(0),
|
|
|
|
'cosimo03': int(0),
|
|
|
|
'frankrekt': int(0),
|
|
|
|
'heisendoc': int(0),
|
|
|
|
'acterryg': int(0),
|
|
|
|
'adry99': int(0),
|
|
|
|
'alleanderl': int(0),
|
|
|
|
'thevagginadestroyer': int(0),
|
|
|
|
'tiztiztiz': int(0),
|
|
|
|
'fultz': int(0),
|
|
|
|
'gotob': int(0),
|
|
|
|
'enribenassati': int(0),
|
|
|
|
'iemax': int(0),
|
|
|
|
'peraemela99': int(0),
|
|
|
|
'ilgattopardo': int(0),
|
|
|
|
'mrdima98': int(0),
|
|
|
|
'ruozir': int(0),
|
|
|
|
'supersmurf': int(0),
|
|
|
|
'tauei': int(0),
|
|
|
|
'voltaggio': int(0),
|
2016-01-26 17:54:05 +00:00
|
|
|
'gibait': int(0),
|
2016-01-21 21:16:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def __init__(self, question, askin):
|
|
|
|
self.domanda = question
|
|
|
|
self.chat = askin
|
|
|
|
|
|
|
|
def ask(self):
|
|
|
|
telegram.sendmessage(self.domanda, self.chat)
|
|
|
|
|
|
|
|
def register(self, utente, voto):
|
|
|
|
self.voto[utente] = voto
|
|
|
|
|
|
|
|
def showresults(self):
|
|
|
|
lista = str()
|
|
|
|
si = 0
|
|
|
|
no = 0
|
|
|
|
astenuti = 0
|
|
|
|
for membro in self.voto:
|
|
|
|
if self.voto[membro] == 0:
|
|
|
|
lista += chr(9898)
|
|
|
|
elif self.voto[membro] == 1:
|
|
|
|
si += 1
|
|
|
|
lista += chr(128309)
|
|
|
|
elif self.voto[membro] == 2:
|
|
|
|
no += 1
|
|
|
|
lista += chr(128308)
|
|
|
|
elif self.voto[membro] == 3:
|
|
|
|
astenuti += 1
|
|
|
|
lista += chr(9899)
|
|
|
|
lista += " @" + membro + "\n"
|
|
|
|
telegram.sendmessage(self.domanda + "\n"
|
|
|
|
"*Risultati:*\n"
|
|
|
|
"Sì: " + str(si) + " (" + str(round(si / (si + no + astenuti) * 100, 2)) + "%)\n"
|
|
|
|
"No: " + str(no) + " (" + str(round(no / (si + no + astenuti) * 100, 2)) + "%)\n"
|
|
|
|
"Astenuti: " + str(astenuti) + "\n\n" + lista, self.chat)
|
|
|
|
|
|
|
|
# Votazione in corso
|
|
|
|
incorso = None
|
|
|
|
|
2015-11-13 17:15:27 +00:00
|
|
|
# Playlist di /rage, si riempie quando è vuota
|
2015-10-08 16:39:56 +00:00
|
|
|
rage = []
|
2015-10-03 16:41:34 +00:00
|
|
|
|
2016-01-14 18:37:37 +00:00
|
|
|
# TODO: Rimettere gli audio di Wololo
|
2016-01-21 21:16:47 +00:00
|
|
|
# wololo = []
|
2015-11-20 15:31:46 +00:00
|
|
|
|
2015-12-15 18:23:12 +00:00
|
|
|
# Dizionario con i nomi utenti di osu!
|
|
|
|
# Se qualcuno cambia nome utente di Telegram, lo cambi anche QUI.
|
|
|
|
osunames = {
|
|
|
|
'steffo': 'SteffoRYG',
|
|
|
|
'evilbalu': 'NemesisRYG',
|
|
|
|
'fultz': 'ftz99',
|
|
|
|
'ilgattopardo': 'gattopardo',
|
2016-01-20 22:37:05 +00:00
|
|
|
'frankrekt': 'FrankezRYG',
|
2016-01-20 20:54:25 +00:00
|
|
|
'tiztiztiz': 'fedececco',
|
2015-12-15 18:23:12 +00:00
|
|
|
'acterryg': 'Acter1',
|
|
|
|
'maxsensei': 'MaxSensei',
|
|
|
|
'heisendoc': 'ImHeisenberg',
|
|
|
|
'thevagginadestroyer': 'barboll',
|
|
|
|
'cosimo03': 'Cosimo03',
|
|
|
|
'albertino04': 'Alby1',
|
|
|
|
'voltaggio': 'voltaggio',
|
|
|
|
'tauei': 'tauei',
|
|
|
|
'boni3099': 'boni3099',
|
|
|
|
'mrdima98': 'MRdima98',
|
|
|
|
}
|
|
|
|
|
2016-02-01 16:31:03 +00:00
|
|
|
# Elenco di username dei membri della RYG
|
|
|
|
telegramnames = ['steffo', 'alby1', 'boni3099', 'maxsensei', 'cosimo03', 'frankrekt', 'heisendoc', 'acterryg', 'adry99',
|
|
|
|
'alleanderl', 'thevagginadestroyer', 'tiztiztiz', 'fultz', 'gotob', 'enribenassati', 'iemax',
|
|
|
|
'peraemela99', 'ilgattopardo', 'mrdima98', 'ruozir', 'supersmurf', 'tauei', 'voltaggio', 'gibait']
|
|
|
|
|
2016-01-21 19:47:25 +00:00
|
|
|
# Dizionario con gli steamID
|
|
|
|
# Vedi sopra
|
|
|
|
steamids = {
|
|
|
|
'steffo': 76561198034314260,
|
|
|
|
'alby1': 76561198071383448,
|
|
|
|
'boni3099': 76561198131868211,
|
|
|
|
'maxsensei': 76561198121094516,
|
|
|
|
'cosimo03': 76561198062778224,
|
|
|
|
'frankrekt': 76561198071099951,
|
|
|
|
'heisendoc': 76561198080377213,
|
|
|
|
'acterryg': 76561198146704979,
|
|
|
|
'adry99': 76561198230034568,
|
|
|
|
'alleanderl': 76561198154175301,
|
|
|
|
'thevagginadestroyer': 76561198128738388,
|
|
|
|
'tiztiztiz': 76561198109189938,
|
|
|
|
'fultz': 76561198035547490,
|
|
|
|
'gattino02': 76561198071069550,
|
|
|
|
'gotob': 76561198096658890,
|
|
|
|
'enribenassati': 76561198123018368,
|
|
|
|
'iemax': 76561198149695151,
|
|
|
|
'peraemela99': 76561198161867082,
|
|
|
|
'ilgattopardo': 76561198111021344,
|
|
|
|
'mrdima98': 76561198140863530,
|
|
|
|
'ruozir': 76561198117708290,
|
|
|
|
'supersmurf': 76561198115852550,
|
|
|
|
'tauei': 76561198104305298,
|
|
|
|
'voltaggio': 76561198147601821,
|
2016-01-26 17:54:05 +00:00
|
|
|
'gibait': 76561198157721704,
|
2016-01-21 19:47:25 +00:00
|
|
|
}
|
|
|
|
|
2015-11-23 16:18:08 +00:00
|
|
|
random.seed()
|
|
|
|
|
2016-01-14 18:37:37 +00:00
|
|
|
# Ciclo principale del bot
|
2015-10-04 17:48:45 +00:00
|
|
|
print("Bot avviato!")
|
2015-11-13 17:15:27 +00:00
|
|
|
while True:
|
|
|
|
# Guarda il comando ricevuto.
|
2015-11-15 15:58:07 +00:00
|
|
|
msg = telegram.getupdates()
|
2016-01-14 18:37:37 +00:00
|
|
|
# Se il messaggio non è una notifica di servizio...
|
2015-11-20 15:34:54 +00:00
|
|
|
if 'text' in msg:
|
2016-01-14 18:37:37 +00:00
|
|
|
# Salvatelo in una stringa
|
|
|
|
text = msg['text']
|
2015-12-15 18:23:12 +00:00
|
|
|
# Guarda l'ID della chat in cui è stato inviato
|
2016-01-14 18:37:37 +00:00
|
|
|
sentin = msg['chat']['id']
|
2016-01-28 19:00:37 +00:00
|
|
|
# ID del messaggio ricevuto
|
|
|
|
source = msg['message_id']
|
2015-12-15 18:23:12 +00:00
|
|
|
# Nome da visualizzare nella console per capire chi accidenti è che invia messaggi strani
|
|
|
|
if 'username' in msg['from']:
|
2016-01-14 18:37:37 +00:00
|
|
|
# Salva l'username se esiste
|
|
|
|
username = msg['from']['username']
|
2015-12-15 18:23:12 +00:00
|
|
|
else:
|
2016-01-14 18:37:37 +00:00
|
|
|
# Altrimenti, salva l'userID
|
|
|
|
username = str(msg['from']['id'])
|
2016-02-01 16:31:03 +00:00
|
|
|
# Se sei un membro della Royal Games
|
2016-02-01 16:33:17 +00:00
|
|
|
if username.lower() in telegramnames:
|
2016-02-01 16:31:03 +00:00
|
|
|
# Riconosci il comando.
|
|
|
|
# Viene usato startswith perchè il comando potrebbe anche essere inviato in forma /ciao@RoyalBot.
|
|
|
|
if text.startswith('/ahnonlosoio'):
|
|
|
|
print("@" + username + ": /ahnonlosoio")
|
2016-02-01 16:58:09 +00:00
|
|
|
# Rispondi con Ah, non lo so nemmeno io.
|
2016-02-01 16:31:03 +00:00
|
|
|
telegram.sendmessage("Ah, non lo so nemmeno io!", sentin, source)
|
|
|
|
elif text.startswith('/ehoh'):
|
|
|
|
print("@" + username + ": /ehoh")
|
2016-02-01 16:58:09 +00:00
|
|
|
# Rispondi con Eh, oh. Sono cose che capitano.
|
2016-02-01 16:31:03 +00:00
|
|
|
telegram.sendmessage("Eh, oh. Sono cose che capitano.", sentin, source)
|
|
|
|
elif text.startswith('/playing'):
|
|
|
|
print("@" + username + ": /playing")
|
2016-02-01 16:58:09 +00:00
|
|
|
# Informa Telegram che il messaggio è stato ricevuto e visualizza Royal Bot sta scrivendo.
|
2016-02-01 16:31:03 +00:00
|
|
|
telegram.sendchataction(sentin)
|
|
|
|
cmd = text.split(" ")
|
|
|
|
# Se è stato specificato un AppID...
|
|
|
|
if len(cmd) >= 2:
|
|
|
|
n = steam.getnumberofcurrentplayers(cmd[1])
|
|
|
|
# Se viene ricevuta una risposta...
|
|
|
|
if n is None:
|
|
|
|
telegram.sendmessage(chr(9888) + " L'app specificata non esiste!", sentin, source)
|
|
|
|
else:
|
|
|
|
telegram.sendmessage('In questo momento, ' + str(n) + ' persone stanno giocando a [' + cmd[1] +
|
|
|
|
'](https://steamdb.info/app/' + cmd[1] + '/graphs/)', sentin, source)
|
2015-11-20 15:34:54 +00:00
|
|
|
else:
|
2016-02-01 16:31:03 +00:00
|
|
|
telegram.sendmessage(chr(9888) + ' Non hai specificato un AppID!\n'
|
|
|
|
'La sintassi corretta è /playing <AppID>.', sentin, source)
|
|
|
|
elif text.startswith('/saldi'):
|
|
|
|
print("@" + username + ": /saldi")
|
2016-02-01 16:58:09 +00:00
|
|
|
# Visualizza il link di isthereanydeal con i saldi di un gioco.
|
|
|
|
# Informa Telegram che il messaggio è stato ricevuto e visualizza Royal Bot sta scrivendo.
|
2016-02-01 16:31:03 +00:00
|
|
|
telegram.sendchataction(sentin)
|
|
|
|
cmd = text.split(" ", 1)
|
|
|
|
if len(cmd) == 2:
|
|
|
|
telegram.sendmessage(
|
|
|
|
'Visualizza le offerte di '
|
|
|
|
'[' + cmd[1] + '](https://isthereanydeal.com/#/search:' + cmd[1].replace(' ', '%20') +
|
|
|
|
";/scroll:%23gamelist).", sentin, source)
|
2015-11-20 15:34:54 +00:00
|
|
|
else:
|
2016-02-01 16:31:03 +00:00
|
|
|
telegram.sendmessage(chr(9888) +
|
|
|
|
" Non hai specificato un gioco!"
|
|
|
|
"[Visualizza tutte le offerte]"
|
|
|
|
"(https://isthereanydeal.com/#/search:.;/scroll:%23gamelist).",
|
|
|
|
sentin, source)
|
2016-02-02 14:50:24 +00:00
|
|
|
elif text.startswith('/audio'):
|
|
|
|
# Se qualcuno ne ha voglia, qui si potrebbe aggiungere la selezione degli audio come argomento,
|
|
|
|
# invece che fare una playlist casuale...
|
|
|
|
# Se non ci sono rage nella playlist, riempila e mescolala!
|
|
|
|
if len(rage) <= 0:
|
|
|
|
# TODO: Rimettere gli audio di /audio
|
|
|
|
rage = ['BQADAgADMwIAAh8GgAFQaq1JNk1ZtwI', #ma dinuovo
|
|
|
|
'BQADAgADrAIAAh8GgAHTdcu8cG-LbAI', #sallati
|
|
|
|
'BQADAgADEAIAAh8GgAE4O2578G1EagI', #giummy per sempre
|
|
|
|
'BQADAgAD3wEAAh8GgAEUqoKiAaPP9wI', #non è wollac è gion sina
|
|
|
|
'BQADAgADqwIAAh8GgAE62csQVNai8QI', #tette crystal maiden
|
|
|
|
'BQADAgAD6wEAAh8GgAGe6IDqRVSAhwI', #la barba che apeggia
|
|
|
|
'BQADAgAD4AEAAh8GgAFRi-UD1VvyLwI', #gion cina original
|
|
|
|
'BQADAgADEwIAAh8GgAE-iNm-4V6pZAI', #ma porco 3
|
|
|
|
'BQADAgADrgIAAh8GgAGKOIASQZevMwI', #5 anni tette grosse
|
|
|
|
'BQADAgAD5gEAAh8GgAFsphnhj_xOnAI', #infilatevi un dito nel..
|
|
|
|
'BQADAgADqAIAAh8GgAEDx7kiV1MdAwI', #ma siete nvidiosi?
|
|
|
|
'BQADAgADqQIAAh8GgAHhGzfuq1LGXAI', #mi sale il cazzo
|
|
|
|
'BQADAgADpgIAAh8GgAFoIX9f88R-vAI', #mamma di mari e le parolacce..
|
|
|
|
]
|
|
|
|
random.shuffle(rage)
|
|
|
|
# Estrai un audio a caso tra quelli nella playlist e rimuovilo.
|
|
|
|
ragesend = rage.pop()
|
|
|
|
print("@" + username + ": /audio")
|
|
|
|
telegram.senddocument(ragesend, sentin)
|
2016-02-01 16:31:03 +00:00
|
|
|
elif text.startswith('/sbam'):
|
|
|
|
print("@" + username + ": /sbam")
|
2016-02-01 16:58:09 +00:00
|
|
|
# Manda l'audio contenente gli sbam di tutti i membri Royal Games.
|
2016-02-01 16:31:03 +00:00
|
|
|
telegram.senddocument('BQADAgADBwMAAh8GgAGSsR4rwmk_LwI', sentin)
|
|
|
|
elif text.startswith('/osu'):
|
|
|
|
print("@" + username + ": /osu")
|
2016-02-01 16:58:09 +00:00
|
|
|
# Visualizza il punteggio più recente di osu!
|
2016-02-01 16:31:03 +00:00
|
|
|
# Informa Telegram che il messaggio è stato ricevuto.
|
|
|
|
telegram.sendchataction(sentin)
|
2016-02-01 16:58:09 +00:00
|
|
|
# Trova il nome utente specificato
|
2016-02-01 16:31:03 +00:00
|
|
|
cmd = text.split(' ', 1)
|
2016-02-01 16:58:09 +00:00
|
|
|
# Se è stato specificato un nome utente
|
2016-02-01 16:31:03 +00:00
|
|
|
if len(cmd) >= 2:
|
|
|
|
# Trova la modalità
|
2016-02-01 16:58:09 +00:00
|
|
|
# 0 = osu!
|
|
|
|
# 1 = osu!taiko
|
|
|
|
# 2 = osu!catch
|
|
|
|
# 3 = osu!mania
|
2016-02-01 16:31:03 +00:00
|
|
|
cmd = text.split(' ', 2)
|
2016-02-01 16:58:09 +00:00
|
|
|
# Se è stata specificata una modalità
|
2016-02-01 16:31:03 +00:00
|
|
|
if len(cmd) >= 3:
|
|
|
|
# Modalità specificata
|
|
|
|
mode = int(cmd[2])
|
|
|
|
else:
|
2016-02-01 16:58:09 +00:00
|
|
|
# Imposta la modalità a osu!
|
2016-02-01 16:31:03 +00:00
|
|
|
mode = 0
|
2016-02-01 16:58:09 +00:00
|
|
|
# Prova a mandare una richiesta ai server di osu per l'ultima canzone giocata
|
2016-02-01 16:31:03 +00:00
|
|
|
try:
|
|
|
|
r = osu.getuserrecent(cmd[1], mode)
|
2016-02-01 16:58:09 +00:00
|
|
|
# Se la funzione restituisce un errore, riferisci su Telegram l'errore e previeni il crash.
|
2016-02-01 16:31:03 +00:00
|
|
|
except NameError:
|
|
|
|
telegram.sendmessage(chr(9888) + " Errore nella richiesta ai server di Osu!", sentin, source)
|
2016-02-01 16:58:09 +00:00
|
|
|
# Se tutto va bene, continua!
|
2016-01-28 17:55:42 +00:00
|
|
|
else:
|
2016-02-01 16:58:09 +00:00
|
|
|
# Se ci sono delle mod attive...
|
2016-02-01 16:31:03 +00:00
|
|
|
if "enabled_mods" in r:
|
|
|
|
mods = "*Mod:*"
|
2016-02-01 16:58:09 +00:00
|
|
|
# Dividi in bit l'ID delle mod selezionate usando un bitwise and
|
|
|
|
# Forse si potrebbe rifare usando la forma esadecimale...?
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x1:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " NoFail"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x2:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " Easy"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x4:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " NoVideo (?)"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x8:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " Hidden"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x10:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " HardRock"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x20:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " SuddenDeath"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x40:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " DoubleTime"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x80:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " Relax"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x100:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " HalfTime"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x200:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " Nightcore"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x400:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " Flashlight"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x800:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " Autoplay"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x1000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " SpunOut"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x2000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " Autopilot"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x4000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " Perfect"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x8000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " 4K"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x10000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " 5K"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x20000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " 6K"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x40000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " 7K"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x80000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " 8K"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x100000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " FadeIn"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x200000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " Random"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x400000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " 9K"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x800000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " 10K"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x1000000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " 1K"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x2000000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " 3K"
|
2016-02-02 14:50:24 +00:00
|
|
|
if int(r['enabled_mods']) & 0x4000000:
|
2016-02-01 16:31:03 +00:00
|
|
|
mods += " 2K"
|
|
|
|
mods += '\n'
|
|
|
|
else:
|
2016-02-01 16:58:09 +00:00
|
|
|
# Lascia la riga delle mod vuota.
|
2016-02-01 16:31:03 +00:00
|
|
|
mods = '\n'
|
|
|
|
if mode == 0:
|
2016-02-01 16:58:09 +00:00
|
|
|
# Visualizza le informazioni relative alla modalità osu!
|
|
|
|
telegram.sendmessage("*osu!*\n"
|
2016-02-01 16:31:03 +00:00
|
|
|
"[Beatmap " + r['beatmap_id'] + "](" + 'https://osu.ppy.sh/b/' + r[
|
|
|
|
'beatmap_id'] +
|
|
|
|
")\n*" + r['rank'] + "*\n" + mods +
|
|
|
|
"*Punti*: " + r['score'] + "\n"
|
|
|
|
"*Combo* x" + r['maxcombo'] + "\n"
|
|
|
|
"*300*: " + r['count300'] + "\n"
|
|
|
|
"*100*: " + r['count100'] + "\n"
|
|
|
|
"*50*: " + r['count50'] + "\n"
|
|
|
|
"*Awesome*: " + r['countkatu'] + "\n"
|
|
|
|
"*Good*: " + r['countgeki'] + "\n"
|
|
|
|
"*Miss*: " + r['countmiss'], sentin, source)
|
|
|
|
elif mode == 1:
|
2016-02-01 16:58:09 +00:00
|
|
|
# Visualizza le informazioni relative alla modalità osu!taiko
|
|
|
|
telegram.sendmessage("*osu!taiko*\n"
|
2016-02-01 16:31:03 +00:00
|
|
|
"[Beatmap " + r['beatmap_id'] + "](" + 'https://osu.ppy.sh/b/' + r[
|
|
|
|
'beatmap_id'] +
|
|
|
|
")\n*" + r['rank'] + "*\n" + mods +
|
|
|
|
"\n*Punti*: " + r['score'] + "\n"
|
|
|
|
"*Combo* x" + r['maxcombo'] + "\n"
|
|
|
|
"*Great*: " + r['count300'] + "\n"
|
|
|
|
"*Good*: " + r['count100'] + "\n"
|
|
|
|
"_Large_ *Great*: " + r['countkatu'] + "\n"
|
|
|
|
"_Large_ *Good*: " + r['countgeki'] + "\n"
|
|
|
|
"*Bad*: " + r['countmiss'], sentin, source)
|
|
|
|
elif mode == 2:
|
2016-02-01 16:58:09 +00:00
|
|
|
# Visualizza le informazioni relative alla modalità osu!catch
|
|
|
|
telegram.sendmessage("*osu!catch*\n"
|
2016-02-01 16:31:03 +00:00
|
|
|
"[Beatmap " + r['beatmap_id'] + "](" + 'https://osu.ppy.sh/b/' + r[
|
|
|
|
'beatmap_id'] +
|
|
|
|
")\n*" + r['rank'] + "*\n" + mods +
|
|
|
|
"\n*Punti*: " + r['score'] + "\n"
|
|
|
|
"*Combo* x" + r['maxcombo'] + "\n"
|
|
|
|
"*Fruit*: " + r['count300'] + "\n"
|
|
|
|
"*Droplet* _tick_: " + r['count100'] + "\n"
|
|
|
|
"*Droplet* _trail_: " + r['count50'] + "\n"
|
|
|
|
"*Miss*: " + r['countmiss'], sentin, source)
|
|
|
|
elif mode == 3:
|
2016-02-01 16:58:09 +00:00
|
|
|
# Visualizza le informazioni relative alla modalità osu!mania
|
|
|
|
telegram.sendmessage("*osu!mania*\n" +
|
2016-02-01 16:31:03 +00:00
|
|
|
"[Beatmap " + r['beatmap_id'] + "](" + 'https://osu.ppy.sh/b/' + r[
|
|
|
|
'beatmap_id'] + ")\n*" + r['rank'] + "*\n" + mods +
|
|
|
|
"\n*Punti*: " + r['score'] + "\n"
|
|
|
|
"*Combo* x" + r['maxcombo'] + "\n"
|
|
|
|
"_Rainbow_ *300*: " + r['countgeki'] + "\n"
|
|
|
|
"*300*: " + r['count300'] + "\n"
|
|
|
|
"*100*: " + r['count100'] + "\n"
|
|
|
|
"*200*: " + r['countkatu'] + "\n"
|
|
|
|
"*50*: " + r['count50'] + "\n"
|
|
|
|
"*Miss*: " + r['countmiss'], sentin, source)
|
|
|
|
else:
|
|
|
|
# TODO: Mettere a posto sto schifo.
|
|
|
|
if username.lower() in osunames:
|
|
|
|
r = osu.getuserrecent(osunames[username.lower()], 0)
|
|
|
|
if "enabled_mods" in r:
|
|
|
|
mods = "*Mod:*"
|
|
|
|
# Dividi in bit l'ID delle mod selezionate
|
|
|
|
if int(r['enabled_mods']) & 0b1:
|
|
|
|
mods += " NoFail"
|
|
|
|
if int(r['enabled_mods']) & 0b10:
|
|
|
|
mods += " Easy"
|
|
|
|
if int(r['enabled_mods']) & 0b100:
|
|
|
|
mods += " NoVideo (?)"
|
|
|
|
if int(r['enabled_mods']) & 0b1000:
|
|
|
|
mods += " Hidden"
|
|
|
|
if int(r['enabled_mods']) & 0b10000:
|
|
|
|
mods += " HardRock"
|
|
|
|
if int(r['enabled_mods']) & 0b100000:
|
|
|
|
mods += " SuddenDeath"
|
|
|
|
if int(r['enabled_mods']) & 0b1000000:
|
|
|
|
mods += " DoubleTime"
|
|
|
|
if int(r['enabled_mods']) & 0b10000000:
|
|
|
|
mods += " Relax"
|
|
|
|
if int(r['enabled_mods']) & 0b100000000:
|
|
|
|
mods += " HalfTime"
|
|
|
|
if int(r['enabled_mods']) & 0b1000000000:
|
|
|
|
mods += " Nightcore"
|
|
|
|
if int(r['enabled_mods']) & 0b10000000000:
|
|
|
|
mods += " Flashlight"
|
|
|
|
if int(r['enabled_mods']) & 0b100000000000:
|
|
|
|
mods += " Autoplay"
|
|
|
|
if int(r['enabled_mods']) & 0b1000000000000:
|
|
|
|
mods += " SpunOut"
|
|
|
|
if int(r['enabled_mods']) & 0b10000000000000:
|
|
|
|
mods += " Autopilot"
|
|
|
|
if int(r['enabled_mods']) & 0b100000000000000:
|
|
|
|
mods += " Perfect"
|
|
|
|
if int(r['enabled_mods']) & 0b1000000000000000:
|
|
|
|
mods += " 4K"
|
|
|
|
if int(r['enabled_mods']) & 0b10000000000000000:
|
|
|
|
mods += " 5K"
|
|
|
|
if int(r['enabled_mods']) & 0b100000000000000000:
|
|
|
|
mods += " 6K"
|
|
|
|
if int(r['enabled_mods']) & 0b1000000000000000000:
|
|
|
|
mods += " 7K"
|
|
|
|
if int(r['enabled_mods']) & 0b10000000000000000000:
|
|
|
|
mods += " 8K"
|
|
|
|
if int(r['enabled_mods']) & 0b100000000000000000000:
|
|
|
|
mods += " FadeIn"
|
|
|
|
if int(r['enabled_mods']) & 0b1000000000000000000000:
|
|
|
|
mods += " Random"
|
|
|
|
if int(r['enabled_mods']) & 0b1000000000000000000000000:
|
|
|
|
mods += " 9K"
|
|
|
|
if int(r['enabled_mods']) & 0b10000000000000000000000000:
|
|
|
|
mods += " 10K"
|
|
|
|
if int(r['enabled_mods']) & 0b100000000000000000000000000:
|
|
|
|
mods += " 1K"
|
|
|
|
if int(r['enabled_mods']) & 0b1000000000000000000000000000:
|
|
|
|
mods += " 3K"
|
|
|
|
if int(r['enabled_mods']) & 0b10000000000000000000000000000:
|
|
|
|
mods += " 2K"
|
|
|
|
mods += '\n'
|
|
|
|
else:
|
|
|
|
mods = '\n'
|
2016-01-26 18:04:56 +00:00
|
|
|
telegram.sendmessage("*Osu!*\n"
|
|
|
|
"[Beatmap " + r['beatmap_id'] + "](" + 'https://osu.ppy.sh/b/' + r[
|
|
|
|
'beatmap_id'] +
|
2016-01-28 17:55:42 +00:00
|
|
|
")\n*" + r['rank'] + "*\n" + mods +
|
2016-02-01 16:31:03 +00:00
|
|
|
"\n*Punti*: " + r['score'] + "\n"
|
2016-01-26 18:04:56 +00:00
|
|
|
"*Combo* x" + r['maxcombo'] + "\n"
|
|
|
|
"*300*: " + r['count300'] + "\n"
|
|
|
|
"*100*: " + r['count100'] + "\n"
|
|
|
|
"*50*: " + r['count50'] + "\n"
|
|
|
|
"*Awesome*: " + r['countkatu'] + "\n"
|
|
|
|
"*Good*: " + r['countgeki'] + "\n"
|
2016-01-28 19:00:37 +00:00
|
|
|
"*Miss*: " + r['countmiss'], sentin, source)
|
2016-02-01 16:31:03 +00:00
|
|
|
elif text.startswith('/roll'):
|
|
|
|
print("@" + username + ": /roll")
|
|
|
|
cmd = text.split(' ', 1)
|
2016-02-01 16:58:09 +00:00
|
|
|
# Se è stato specificato un numero
|
2016-02-01 16:31:03 +00:00
|
|
|
if len(cmd) >= 2:
|
2016-02-01 16:58:09 +00:00
|
|
|
# Controlla che sia convertibile in un intero.
|
2016-02-01 16:31:03 +00:00
|
|
|
try:
|
|
|
|
m = int(cmd[1])
|
|
|
|
except ValueError:
|
|
|
|
telegram.sendmessage(chr(9888) + " Il numero specificato non è un intero.", sentin, source)
|
|
|
|
else:
|
2016-02-01 16:58:09 +00:00
|
|
|
# Imposta il numero massimo a 100.
|
2016-02-01 16:31:03 +00:00
|
|
|
m = 100
|
2016-02-01 16:58:09 +00:00
|
|
|
# Prova a generare un numero casuale.
|
2016-01-27 14:31:45 +00:00
|
|
|
try:
|
2016-02-01 16:31:03 +00:00
|
|
|
n = random.randrange(m) + 1
|
2016-01-27 14:31:45 +00:00
|
|
|
except ValueError:
|
2016-02-01 16:58:09 +00:00
|
|
|
telegram.sendmessage(chr(9888) + " Il numero specificato non è maggiore o uguale a 0.", sentin,
|
|
|
|
source)
|
|
|
|
# Se tutto va bene visualizza il numero generato
|
2016-01-27 14:31:45 +00:00
|
|
|
else:
|
2016-02-01 16:31:03 +00:00
|
|
|
telegram.sendmessage("Numero casuale da 1 a " + str(m) + ":\n*" + str(n) + "*", sentin, source)
|
|
|
|
elif text.startswith('/automah'):
|
|
|
|
print("@" + username + ": /automah")
|
2016-02-01 16:58:09 +00:00
|
|
|
# Invia il messaggio.
|
2016-02-01 16:31:03 +00:00
|
|
|
telegram.sendmessage("Automaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa! Devi funzionare, cavolo!", sentin,
|
|
|
|
source)
|
|
|
|
elif text.startswith('/hs'):
|
|
|
|
print("@" + username + ": /hs")
|
|
|
|
# Informa Telegram che il messaggio è stato ricevuto.
|
|
|
|
telegram.sendchataction(sentin)
|
|
|
|
cmd = text.split(" ", 1)
|
2016-02-01 16:58:09 +00:00
|
|
|
# Se è stata specificata una carta...
|
2016-02-01 16:31:03 +00:00
|
|
|
if len(cmd) >= 2:
|
2016-02-01 16:58:09 +00:00
|
|
|
# Controlla che la carta specificata esista.
|
2016-02-01 16:31:03 +00:00
|
|
|
try:
|
|
|
|
r = hearthstone.card(cmd[1])
|
|
|
|
# Se ci sono più carte, prendine una a caso!
|
|
|
|
r = r[random.randrange(len(r))]
|
|
|
|
except ValueError:
|
|
|
|
telegram.sendmessage(chr(9888) + " La carta specificata non esiste!", sentin, source)
|
2016-02-01 16:58:09 +00:00
|
|
|
# Se tutto va bene, elabora e visualizza le informazioni sulla carta.
|
2016-02-01 16:31:03 +00:00
|
|
|
else:
|
|
|
|
# Si trova nelle bustine
|
|
|
|
if 'howToGet' not in r:
|
|
|
|
if 'cardSet' in r:
|
|
|
|
r['howToGet'] = "Trovala in " + r['cardSet'] + "."
|
|
|
|
else:
|
|
|
|
r['howToGet'] = "Inottenibile."
|
|
|
|
# Nessuna classe
|
|
|
|
if 'playerClass' not in r:
|
|
|
|
r['playerClass'] = "Neutral"
|
|
|
|
# Nessun effetto
|
|
|
|
if 'text' not in r:
|
|
|
|
r['text'] = ""
|
2016-02-01 16:58:09 +00:00
|
|
|
# Converti l'HTML nella descrizione in Markdown. Circa.
|
2016-02-01 16:31:03 +00:00
|
|
|
r['text'] = r['text'].replace("<b>", "*")
|
|
|
|
r['text'] = r['text'].replace("</b>", "*")
|
|
|
|
r['text'] = r['text'].replace("<i>", "_")
|
|
|
|
r['text'] = r['text'].replace("</i>", "_")
|
2016-02-01 16:58:09 +00:00
|
|
|
# Togli il $, che indica che il numero di danni può essere modificato dallo Spell Damage
|
2016-02-01 16:31:03 +00:00
|
|
|
r['text'] = r['text'].replace("$", "")
|
|
|
|
# Nessuna rarità
|
|
|
|
if 'rarity' not in r:
|
|
|
|
r['rarity'] = "None"
|
|
|
|
# Nessuna descrizione
|
|
|
|
if 'flavor' not in r:
|
|
|
|
r['flavor'] = ""
|
|
|
|
# Testo principale
|
|
|
|
# Magie
|
|
|
|
if r['type'] == "Spell":
|
|
|
|
text = str("[" + r['name'] + "](" + r['img'] + ") "
|
|
|
|
"(" + r['rarity'] + ")\n" +
|
|
|
|
r['playerClass'] + "\n" +
|
|
|
|
str(r['cost']) + " mana\n" +
|
|
|
|
r['text'] + "\n" +
|
|
|
|
r['howToGet'] + "\n\n_" +
|
|
|
|
r['flavor'] + "_\n")
|
|
|
|
# Servitori
|
|
|
|
elif r['type'] == "Minion":
|
|
|
|
text = str("[" + r['name'] + "](" + r['img'] + ") "
|
|
|
|
"(" + r['rarity'] + ")\n" +
|
|
|
|
r['playerClass'] + "\n" +
|
|
|
|
str(r['cost']) + " mana\n" +
|
|
|
|
str(r['attack']) + " attacco\n" +
|
|
|
|
str(r['health']) + " salute\n" +
|
|
|
|
r['text'] + "\n" +
|
|
|
|
r['howToGet'] + "\n\n_" +
|
|
|
|
r['flavor'] + "_\n")
|
|
|
|
# Armi
|
|
|
|
elif r['type'] == "Weapon":
|
|
|
|
text = str("[" + r['name'] + "](" + r['img'] + ") "
|
|
|
|
"(" + r['rarity'] + ")\n" +
|
|
|
|
r['playerClass'] + "\n" +
|
|
|
|
str(r['cost']) + " mana\n" +
|
|
|
|
str(r['attack']) + " attacco\n" +
|
|
|
|
str(r['durability']) + " integrita'\n" +
|
|
|
|
r['text'] + "\n" +
|
|
|
|
r['howToGet'] + "\n\n_" +
|
|
|
|
r['flavor'] + "_\n")
|
|
|
|
# Potere Eroe
|
|
|
|
elif r['type'] == "Hero Power":
|
|
|
|
text = str("[" + r['name'] + "](" + r['img'] + ")\n" +
|
|
|
|
r['playerClass'] + "\n" +
|
|
|
|
str(r['cost']) + " mana\n" +
|
|
|
|
r['text'] + "\n")
|
|
|
|
# Eroe
|
|
|
|
elif r['type'] == "Hero":
|
|
|
|
text = str("[" + r['name'] + "](" + r['img'] + ")\n" +
|
|
|
|
"*Eroe*\n" +
|
|
|
|
str(r['health']) + " salute\n")
|
|
|
|
telegram.sendmessage(text, sentin, source)
|
|
|
|
else:
|
|
|
|
telegram.sendmessage(chr(9888) + " Non hai specificato nessuna carta!\n"
|
|
|
|
"La sintassi corretta è _/hs nomecarta_ .", sentin, source)
|
|
|
|
elif text.startswith('/online'):
|
|
|
|
# Elenco di tutte le persone online su Steam
|
|
|
|
print("@" + username + ": /online ")
|
|
|
|
# Informa Telegram che il messaggio è stato ricevuto.
|
|
|
|
telegram.sendchataction(sentin)
|
|
|
|
cmd = text.split(" ")
|
|
|
|
if len(cmd) >= 2:
|
|
|
|
if cmd[1].lower() == "help":
|
|
|
|
telegram.sendmessage(chr(128309) + " Online\n" +
|
|
|
|
chr(128308) + " In gioco | Occupato\n" +
|
|
|
|
chr(9899) + " Assente | Inattivo\n" +
|
|
|
|
chr(128310) + " Disponibile per scambiare\n" +
|
|
|
|
chr(128311) + " Disponibile per giocare", sentin, source)
|
2016-01-28 17:55:42 +00:00
|
|
|
else:
|
2016-02-01 16:31:03 +00:00
|
|
|
# Stringa utilizzata per ottenere informazioni su tutti gli utenti in una sola richiesta a steam
|
|
|
|
userids = str()
|
|
|
|
for nome in steamids:
|
|
|
|
userids += str(steamids[nome]) + ','
|
|
|
|
tosend = "*Online ora:*\n"
|
|
|
|
r = steam.getplayersummaries(userids)
|
|
|
|
for player in r:
|
|
|
|
# In gioco
|
|
|
|
if 'gameextrainfo' in player:
|
|
|
|
tosend += chr(128308) + " _" + player['gameextrainfo'] + "_ |"
|
|
|
|
elif 'gameid' in player:
|
|
|
|
tosend += chr(128308) + " _" + player['gameid'] + "_ |"
|
|
|
|
# Online
|
|
|
|
elif player['personastate'] == 1:
|
|
|
|
tosend += chr(128309)
|
|
|
|
# Occupato
|
|
|
|
elif player['personastate'] == 2:
|
|
|
|
tosend += chr(128308)
|
|
|
|
# Assente o Inattivo
|
|
|
|
elif player['personastate'] == 3 or player['personastate'] == 4:
|
|
|
|
tosend += chr(9899)
|
|
|
|
# Disponibile per scambiare
|
|
|
|
elif player['personastate'] == 5:
|
|
|
|
tosend += chr(128310)
|
|
|
|
# Disponibile per giocare
|
|
|
|
elif player['personastate'] == 6:
|
|
|
|
tosend += chr(128311)
|
|
|
|
if player['personastate'] != 0:
|
|
|
|
tosend += " " + player['personaname'] + "\n"
|
2016-01-28 17:55:42 +00:00
|
|
|
else:
|
2016-02-01 16:31:03 +00:00
|
|
|
telegram.sendmessage(tosend, sentin, source)
|
|
|
|
elif text.startswith('/shrekt'):
|
|
|
|
print("@" + username + ": /shrekt ")
|
|
|
|
telegram.senddocument("BQADBAADsQADiBjiAqYN-EBXASyhAg", sentin)
|
|
|
|
elif text.startswith('/restart') and username == "Steffo":
|
|
|
|
print("@" + username + ": /restart ")
|
|
|
|
telegram.sendmessage("Riavvio accettato.", sentin, source)
|
|
|
|
sys.exit(0)
|
|
|
|
elif text.startswith('/nuovavotazione') and username == "Steffo":
|
|
|
|
print("@" + username + ": /nuovavotazione ")
|
|
|
|
cmd = text.split(" ", 1)
|
|
|
|
incorso = Votazione(cmd[1], sentin)
|
|
|
|
elif text.startswith('/si') and incorso is not None:
|
|
|
|
if incorso.chat == sentin:
|
|
|
|
print("@" + username + ": /si ")
|
|
|
|
incorso.register(username.lower(), 1)
|
|
|
|
telegram.sendmessage("Votazione registrata!", sentin, source)
|
|
|
|
elif text.startswith('/no') and incorso is not None:
|
|
|
|
if incorso.chat == sentin:
|
|
|
|
print("@" + username + ": /no ")
|
|
|
|
incorso.register(username.lower(), 2)
|
|
|
|
telegram.sendmessage("Votazione registrata!", sentin, source)
|
|
|
|
elif text.startswith('/astieniti') and incorso is not None:
|
|
|
|
if incorso.chat == sentin:
|
|
|
|
print("@" + username + ": /astieniti ")
|
|
|
|
incorso.register(username.lower(), 3)
|
|
|
|
telegram.sendmessage("Votazione registrata!", sentin, source)
|
|
|
|
elif text.startswith('/domanda') and incorso is not None:
|
|
|
|
if incorso.chat == sentin:
|
|
|
|
print("@" + username + ": /domanda ")
|
|
|
|
incorso.ask()
|
|
|
|
elif text.startswith('/risultati') and incorso is not None:
|
|
|
|
if incorso.chat == sentin:
|
|
|
|
print("@" + username + ": /risultati ")
|
|
|
|
incorso.showresults()
|
|
|
|
elif text.startswith('/cv'):
|
|
|
|
print("@" + username + ": /cv ")
|
|
|
|
# Informa Telegram che il messaggio è stato ricevuto.
|
|
|
|
telegram.sendchataction(sentin)
|
|
|
|
r = mumbleboxes.getserverstatus("https://www.mumbleboxes.com/servers/5454/cvp.json").json()
|
|
|
|
tosend = "Utenti online: " + str(len(r['root']['users'])) + " / 15\n"
|
|
|
|
for u in r['root']['users']:
|
|
|
|
if not u['mute']:
|
|
|
|
if u['selfDeaf']:
|
|
|
|
tosend += chr(128263) + " "
|
|
|
|
elif u['selfMute']:
|
|
|
|
tosend += chr(128264) + " "
|
|
|
|
else:
|
|
|
|
tosend += chr(128266) + " "
|
|
|
|
tosend += u['name'] + "\n"
|
|
|
|
telegram.sendmessage(tosend, sentin, source)
|
|
|
|
elif text.startswith('/diario'):
|
2016-02-01 16:32:42 +00:00
|
|
|
print("@" + username + ": /diario ")
|
2016-02-01 16:31:03 +00:00
|
|
|
cmd = text.split(" ", 1)
|
|
|
|
d = filemanager.readfile("diario.txt")
|
|
|
|
d += str(time.time()) + " | " + cmd[1] + "\n"
|
|
|
|
filemanager.writefile("diario.txt", d)
|
|
|
|
telegram.sendmessage("Aggiunto al diario RYG.", sentin, source)
|
2016-02-02 15:08:49 +00:00
|
|
|
elif text.startswith('/leggi'):
|
|
|
|
print("@" + username + ": /leggi")
|
|
|
|
cmd = text.split(" ", 1)
|
|
|
|
d = filemanager.readfile("diario.txt")
|
|
|
|
d = d.split('\n')
|
|
|
|
text = str()
|
|
|
|
# L'ultimo numero è escluso.
|
|
|
|
for n in range(int(cmd[1]) + 1, 1, -1):
|
|
|
|
text += d[len(d) - n] + "\n"
|
|
|
|
telegram.sendmessage(text, sentin, source)
|
2016-02-01 16:32:42 +00:00
|
|
|
else:
|
|
|
|
print("@" + username + " bloccato.")
|
2016-02-01 16:31:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
|