Compare commits
231 Commits
tourney-fi
...
master
Author | SHA1 | Date | |
---|---|---|---|
|
0550222eed | ||
|
a0053de311 | ||
|
848dacf621 | ||
|
5111b6b05f | ||
|
9ad8d5779e | ||
|
71f5a8c4d0 | ||
|
768971e240 | ||
|
a248b4032c | ||
|
cb6d828fd9 | ||
|
925d55bb16 | ||
|
4e97c717f5 | ||
|
f11708d463 | ||
|
4c3427fd76 | ||
|
a3ab53e108 | ||
|
74cfadb261 | ||
|
c09c7b5d96 | ||
|
56c1dcc48a | ||
|
1b9224fc79 | ||
|
887836045d | ||
|
cb2d1e74f7 | ||
|
7f283d9aa2 | ||
|
333cfca806 | ||
|
1dc37ecb77 | ||
|
a456508ed3 | ||
|
66acc12099 | ||
|
a9fc15d524 | ||
|
02b08d3024 | ||
|
d8c6fa4993 | ||
|
fe05aa7ace | ||
|
79ecdbcd44 | ||
|
fd47bc6e32 | ||
|
1a4e952e4f | ||
|
2cd69a9a63 | ||
|
ea4e2bd4fd | ||
|
508f6b507a | ||
|
e89fbe7604 | ||
|
05694c5d87 | ||
|
198bdb9997 | ||
|
473a2e1f2d | ||
|
a809008e55 | ||
|
201593ea02 | ||
|
3275d31861 | ||
|
05eead0e1a | ||
|
8bdc56faf6 | ||
|
745a833aab | ||
|
dcad5c5736 | ||
|
c1f8ca8ed3 | ||
|
e0d54f49d1 | ||
|
c2af1b9772 | ||
|
a99bf0c74b | ||
|
f0fa00b181 | ||
|
956aa8161f | ||
|
08b367812f | ||
|
0722c91018 | ||
|
f0e8223b5c | ||
|
27a5f9c000 | ||
|
29db61fd12 | ||
|
e63a85e4a4 | ||
|
050c1d5fe8 | ||
|
90cd4634fd | ||
|
e3f1bc05e9 | ||
|
0939ec972c | ||
|
2ef89daf4c | ||
|
bb8ccf8c85 | ||
|
dc90f506bd | ||
|
2f9179362c | ||
|
3dfb1228ee | ||
|
c4123eb636 | ||
|
e3e46a34ec | ||
|
3ed837dc96 | ||
|
48534bb551 | ||
|
36d701eac1 | ||
|
60fe2bc56e | ||
|
b8baddf694 | ||
|
018da5c0a2 | ||
|
0a2ca07198 | ||
|
34e7a332e6 | ||
|
8d97227965 | ||
|
176775f8f3 | ||
|
0329847477 | ||
|
b24b4ee88d | ||
|
e40acd335b | ||
|
ce75f5ee99 | ||
|
29359ad4fd | ||
|
def4891008 | ||
|
5d73682a71 | ||
|
0229fc4e65 | ||
|
05c4c89002 | ||
|
02b266f229 | ||
|
612e808702 | ||
|
466004f239 | ||
|
5f279da6cf | ||
|
aa1887e2c4 | ||
|
df2a9bb13d | ||
|
e2149d7d61 | ||
|
ffc84448a2 | ||
|
933c92e5f2 | ||
|
3bb1029832 | ||
|
ce889e608a | ||
|
aeecccdd13 | ||
|
9425043b1e | ||
|
f660a20f60 | ||
|
5adc7f4261 | ||
|
3653447761 | ||
|
17aab9551e | ||
|
87d6186993 | ||
|
5975e84f52 | ||
|
3309f2f8fd | ||
|
baa8ae4cc2 | ||
|
837df03f05 | ||
|
c2a2d9c97b | ||
|
fc989a2705 | ||
|
310bc1d3b3 | ||
|
9f647d5f9e | ||
|
1e6ee91685 | ||
|
86995feb34 | ||
|
af554c94d5 | ||
|
3373bc9581 | ||
|
2cf1cdf1fd | ||
|
8c3fc6842d | ||
|
eab8bee828 | ||
|
66061d5fb2 | ||
|
401dd5ecdb | ||
|
d439490029 | ||
|
f4d0a2424d | ||
|
a177e65fcf | ||
|
c14c86fe0d | ||
|
25df2228e3 | ||
|
941cf81877 | ||
|
76bb15f9f3 | ||
|
8043d686c5 | ||
|
cae82bd107 | ||
|
2c2c85b382 | ||
|
a0fdc6c292 | ||
|
487f583083 | ||
|
530d0c3b74 | ||
|
46ec4f3704 | ||
|
0aa0ab0475 | ||
|
1c2a29a88e | ||
|
92e57aff28 | ||
|
7e535d7ed5 | ||
|
0464f713f7 | ||
|
66776c60e0 | ||
|
4ef98b5fc0 | ||
|
96add06b44 | ||
|
31971d4a8b | ||
|
9175f9e7f2 | ||
|
784c4a11f1 | ||
|
2407ecc3bf | ||
|
9880c5004d | ||
|
d2f111fd7b | ||
|
e489221e39 | ||
|
84f1fb566c | ||
|
fc3736eba8 | ||
|
768913da59 | ||
|
c6417c31ed | ||
|
9055fcaf5e | ||
|
f85640ae39 | ||
|
18a7c47db6 | ||
|
22ae4c332b | ||
|
8f156a0702 | ||
|
62b67da9fb | ||
|
501130721d | ||
|
e6cdef4580 | ||
|
00c544b7c7 | ||
|
ebf0e1d458 | ||
|
fd23cf2b2c | ||
|
8a8a4968a3 | ||
|
2ae3c5f701 | ||
|
f4c099c809 | ||
|
f8cc0c738c | ||
|
4557b08df8 | ||
|
49f8bd8cf1 | ||
|
7ba5db62b4 | ||
|
7f534f0984 | ||
|
daf457fc5c | ||
|
b4d498c26c | ||
|
44545c3bcb | ||
|
c4a6c84cec | ||
|
8532731f19 | ||
|
b836f77446 | ||
|
e92cbe47bd | ||
|
6ca2016f7b | ||
|
2f54a56b7a | ||
|
cf9e506875 | ||
|
5c93d692ea | ||
|
a8a1dfb1bc | ||
|
9d562e7acd | ||
|
4f4253afce | ||
|
04898c24ae | ||
|
1b94936092 | ||
|
d4591b42a3 | ||
|
69508f9a0e | ||
|
5cf8c1bde8 | ||
|
20be60d9db | ||
|
61935f323c | ||
|
cecef18d13 | ||
|
5723c0e68f | ||
|
525235a27e | ||
|
3bc390e3e6 | ||
|
f6ae673401 | ||
|
aa32e8bea6 | ||
|
fb00063e0f | ||
|
e30893d66f | ||
|
8078596a0a | ||
|
eefec1f47b | ||
|
2992dc6190 | ||
|
b6e2319e8c | ||
|
a2ef03c887 | ||
|
abad698fe3 | ||
|
030d556b9c | ||
|
d51b304fbe | ||
|
1ecc73e0dc | ||
|
b1315815b2 | ||
|
38bcf3a735 | ||
|
a6292c7374 | ||
|
ef940771d8 | ||
|
b03d51abff | ||
|
8ef02faf36 | ||
|
d29dcd25f7 | ||
|
cd75d1ad8d | ||
|
c7c5528588 | ||
|
3b150d70cd | ||
|
d249dd593f | ||
|
78a6931805 | ||
|
555c9cca1f | ||
|
83c514b75e | ||
|
0e5471383e | ||
|
abc8e058a1 | ||
|
0a53b31e42 | ||
|
b61ac0e9b6 |
8
.gitignore
vendored
8
.gitignore
vendored
|
@ -1,8 +1,10 @@
|
||||||
**/__pycache__
|
**/__pycache__
|
||||||
|
**/build
|
||||||
config.ini
|
config.ini
|
||||||
filters.txt
|
filters.txt
|
||||||
.data
|
.data
|
||||||
.idea
|
.idea
|
||||||
common_funzia
|
redistest.py
|
||||||
common_refractor
|
*.c
|
||||||
common_memato
|
*.so
|
||||||
|
.pyenv
|
2
.gitmodules
vendored
2
.gitmodules
vendored
|
@ -1,3 +1,3 @@
|
||||||
[submodule "common"]
|
[submodule "common"]
|
||||||
path = common
|
path = common
|
||||||
url = git@git.zxq.co:ripple/ripple-python-common.git
|
url = https://github.com/osufx/ripple-python-common.git
|
||||||
|
|
7
.landscape.yaml
Normal file
7
.landscape.yaml
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
python-targets:
|
||||||
|
- 3
|
||||||
|
pep8:
|
||||||
|
none: true
|
||||||
|
pylint:
|
||||||
|
disable:
|
||||||
|
- cyclic-import
|
15
README.md
15
README.md
|
@ -1,4 +1,8 @@
|
||||||
## pep.py
|
## pep.py [![Code Health](https://landscape.io/github/osuripple/pep.py/master/landscape.svg?style=flat)](https://landscape.io/github/osuripple/pep.py/master)
|
||||||
|
|
||||||
|
- Origin: https://git.zxq.co/ripple/pep.py
|
||||||
|
- Mirror: https://github.com/osuripple/pep.py
|
||||||
|
|
||||||
This is Ripple's bancho server. It handles:
|
This is Ripple's bancho server. It handles:
|
||||||
- Client login
|
- Client login
|
||||||
- Online users listing and statuses
|
- Online users listing and statuses
|
||||||
|
@ -9,6 +13,8 @@ This is Ripple's bancho server. It handles:
|
||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
- Python 3.5
|
- Python 3.5
|
||||||
|
- Cython
|
||||||
|
- C compiler
|
||||||
- MySQLdb (`mysqlclient`)
|
- MySQLdb (`mysqlclient`)
|
||||||
- Tornado
|
- Tornado
|
||||||
- Bcrypt
|
- Bcrypt
|
||||||
|
@ -23,9 +29,14 @@ afterwards, install the required dependencies with pip
|
||||||
```
|
```
|
||||||
$ pip install -r requirements.txt
|
$ pip install -r requirements.txt
|
||||||
```
|
```
|
||||||
then, run pep.py once to create the default config file and edit it
|
then, compile all `*.pyx` files to `*.so` or `*.dll` files using `setup.py` (distutils file)
|
||||||
|
```
|
||||||
|
$ python3 setup.py build_ext --inplace
|
||||||
|
```
|
||||||
|
finally, run pep.py once to create the default config file and edit it
|
||||||
```
|
```
|
||||||
$ python3 pep.py
|
$ python3 pep.py
|
||||||
|
...
|
||||||
$ nano config.ini
|
$ nano config.ini
|
||||||
```
|
```
|
||||||
you can run pep.py by typing
|
you can run pep.py by typing
|
||||||
|
|
2
common
2
common
|
@ -1 +1 @@
|
||||||
Subproject commit a899c0be8912bc5e1dee9a5729a60377991890ec
|
Subproject commit 6103fe96a79cd8f5cbabe24b5fac9cf2a5cacb4a
|
|
@ -1,4 +1,3 @@
|
||||||
""" Contains functions used to read specific client packets from byte stream """
|
|
||||||
from constants import dataTypes
|
from constants import dataTypes
|
||||||
from helpers import packetHelper
|
from helpers import packetHelper
|
||||||
from constants import slotStatuses
|
from constants import slotStatuses
|
||||||
|
@ -100,7 +99,7 @@ def matchSettings(stream):
|
||||||
start += 2
|
start += 2
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
s = data[0]["slot{}Status".format(str(i))]
|
s = data[0]["slot{}Status".format(str(i))]
|
||||||
if s != slotStatuses.free and s != slotStatuses.locked:
|
if s != slotStatuses.FREE and s != slotStatuses.LOCKED:
|
||||||
start += 4
|
start += 4
|
||||||
|
|
||||||
# Other settings
|
# Other settings
|
||||||
|
@ -144,6 +143,26 @@ def transferHost(stream):
|
||||||
def matchInvite(stream):
|
def matchInvite(stream):
|
||||||
return packetHelper.readPacketData(stream, [["userID", dataTypes.UINT32]])
|
return packetHelper.readPacketData(stream, [["userID", dataTypes.UINT32]])
|
||||||
|
|
||||||
|
def matchFrames(stream):
|
||||||
|
return packetHelper.readPacketData(stream,
|
||||||
|
[
|
||||||
|
["time", dataTypes.SINT32],
|
||||||
|
["id", dataTypes.BYTE],
|
||||||
|
["count300", dataTypes.UINT16],
|
||||||
|
["count100", dataTypes.UINT16],
|
||||||
|
["count50", dataTypes.UINT16],
|
||||||
|
["countGeki", dataTypes.UINT16],
|
||||||
|
["countKatu", dataTypes.UINT16],
|
||||||
|
["countMiss", dataTypes.UINT16],
|
||||||
|
["totalScore", dataTypes.SINT32],
|
||||||
|
["maxCombo", dataTypes.UINT16],
|
||||||
|
["currentCombo", dataTypes.UINT16],
|
||||||
|
["perfect", dataTypes.BYTE],
|
||||||
|
["currentHp", dataTypes.BYTE],
|
||||||
|
["tagByte", dataTypes.BYTE],
|
||||||
|
["usingScoreV2", dataTypes.BYTE]
|
||||||
|
])
|
||||||
|
|
||||||
def tournamentMatchInfoRequest(stream):
|
def tournamentMatchInfoRequest(stream):
|
||||||
return packetHelper.readPacketData(stream, [["matchID", dataTypes.UINT32]])
|
return packetHelper.readPacketData(stream, [["matchID", dataTypes.UINT32]])
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,3 @@
|
||||||
"""Bancho exceptions"""
|
|
||||||
# TODO: Prints in exceptions
|
|
||||||
class loginFailedException(Exception):
|
class loginFailedException(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@ -92,3 +90,21 @@ class unknownStreamException(Exception):
|
||||||
|
|
||||||
class userTournamentException(Exception):
|
class userTournamentException(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
class userAlreadyInChannelException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
class userNotInChannelException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
class missingReportInfoException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
class invalidUserException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
class wrongChannelException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
class periodicLoopException(Exception):
|
||||||
|
pass
|
|
@ -1,27 +1,42 @@
|
||||||
import json
|
import json
|
||||||
import random
|
import random
|
||||||
|
import re
|
||||||
|
import threading
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
|
import time
|
||||||
|
|
||||||
from common import generalUtils
|
from common import generalUtils
|
||||||
from common.constants import mods
|
from common.constants import mods
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from common.ripple import userUtils
|
from common.ripple import userUtils
|
||||||
from constants import exceptions
|
from constants import exceptions, slotStatuses, matchModModes, matchTeams, matchTeamTypes, matchScoringTypes
|
||||||
from common.constants import gameModes
|
from common.constants import gameModes
|
||||||
from common.constants import privileges
|
from common.constants import privileges
|
||||||
from constants import serverPackets
|
from constants import serverPackets
|
||||||
from helpers import systemHelper
|
from helpers import systemHelper
|
||||||
from objects import fokabot
|
from objects import fokabot
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
from helpers import chatHelper as chat
|
||||||
|
from common.web import cheesegull
|
||||||
|
|
||||||
|
|
||||||
|
def bloodcatMessage(beatmapID):
|
||||||
|
beatmap = glob.db.fetch("SELECT song_name, beatmapset_id FROM beatmaps WHERE beatmap_id = %s LIMIT 1", [beatmapID])
|
||||||
|
if beatmap is None:
|
||||||
|
return "Sorry, I'm not able to provide a download link for this map :("
|
||||||
|
return "Download [https://bloodcat.com/osu/s/{} {}] from Bloodcat".format(
|
||||||
|
beatmap["beatmapset_id"],
|
||||||
|
beatmap["song_name"],
|
||||||
|
)
|
||||||
|
|
||||||
"""
|
"""
|
||||||
Commands callbacks
|
Commands callbacks
|
||||||
|
|
||||||
Must have fro, chan and messages as arguments
|
Must have fro, chan and messages as arguments
|
||||||
fro -- name of who triggered the command
|
:param fro: username of who triggered the command
|
||||||
chan -- channel where the message was sent
|
:param chan: channel"(or username, if PM) where the message was sent
|
||||||
message -- list containing arguments passed from the message
|
:param message: list containing arguments passed from the message
|
||||||
[0] = first argument
|
[0] = first argument
|
||||||
[1] = second argument
|
[1] = second argument
|
||||||
. . .
|
. . .
|
||||||
|
@ -31,37 +46,19 @@ TODO: Change False to None, because False doesn't make any sense
|
||||||
"""
|
"""
|
||||||
def instantRestart(fro, chan, message):
|
def instantRestart(fro, chan, message):
|
||||||
glob.streams.broadcast("main", serverPackets.notification("We are restarting Bancho. Be right back!"))
|
glob.streams.broadcast("main", serverPackets.notification("We are restarting Bancho. Be right back!"))
|
||||||
systemHelper.scheduleShutdown(0, True, delay=1)
|
systemHelper.scheduleShutdown(0, True, delay=5)
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def faq(fro, chan, message):
|
def faq(fro, chan, message):
|
||||||
if message[0] == "rules":
|
key = message[0].lower()
|
||||||
return "Please make sure to check (Ripple's rules)[http://ripple.moe/?p=23]."
|
if key not in glob.conf.extra["pep.py"]["faq"]:
|
||||||
elif message[0] == "swearing":
|
|
||||||
return "Please don't abuse swearing"
|
|
||||||
elif message[0] == "spam":
|
|
||||||
return "Please don't spam"
|
|
||||||
elif message[0] == "offend":
|
|
||||||
return "Please don't offend other players"
|
|
||||||
elif message[0] == "github":
|
|
||||||
return "(Ripple's Github page!)[https://github.com/osuripple/ripple]"
|
|
||||||
elif message[0] == "discord":
|
|
||||||
return "(Join Ripple's Discord!)[https://discord.gg/0rJcZruIsA6rXuIx]"
|
|
||||||
elif message[0] == "blog":
|
|
||||||
return "You can find the latest Ripple news on the (blog)[https://ripple.moe/blog/]!"
|
|
||||||
elif message[0] == "changelog":
|
|
||||||
return "Check the (changelog)[https://ripple.moe/index.php?p=17] !"
|
|
||||||
elif message[0] == "status":
|
|
||||||
return "Check the server status (here!)[https://ripple.moe/index.php?p=27]"
|
|
||||||
elif message[0] == "english":
|
|
||||||
return "Please keep this channel in english."
|
|
||||||
else:
|
|
||||||
return False
|
return False
|
||||||
|
return glob.conf.extra["pep.py"]["faq"][key]
|
||||||
|
|
||||||
def roll(fro, chan, message):
|
def roll(fro, chan, message):
|
||||||
maxPoints = 100
|
maxPoints = 100
|
||||||
if len(message) >= 1:
|
if len(message) >= 1:
|
||||||
if message[0].isdigit() == True and int(message[0]) > 0:
|
if message[0].isdigit() and int(message[0]) > 0:
|
||||||
maxPoints = int(message[0])
|
maxPoints = int(message[0])
|
||||||
|
|
||||||
points = random.randrange(0,maxPoints)
|
points = random.randrange(0,maxPoints)
|
||||||
|
@ -71,15 +68,20 @@ def roll(fro, chan, message):
|
||||||
# return random.choice(["yes", "no", "maybe"])
|
# return random.choice(["yes", "no", "maybe"])
|
||||||
|
|
||||||
def alert(fro, chan, message):
|
def alert(fro, chan, message):
|
||||||
glob.streams.broadcast("main", serverPackets.notification(' '.join(message[:])))
|
msg = ' '.join(message[:]).strip()
|
||||||
|
if not msg:
|
||||||
|
return False
|
||||||
|
glob.streams.broadcast("main", serverPackets.notification(msg))
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def alertUser(fro, chan, message):
|
def alertUser(fro, chan, message):
|
||||||
target = message[0].replace("_", " ")
|
target = message[0].lower()
|
||||||
|
targetToken = glob.tokens.getTokenFromUsername(userUtils.safeUsername(target), safe=True)
|
||||||
targetToken = glob.tokens.getTokenFromUsername(target)
|
|
||||||
if targetToken is not None:
|
if targetToken is not None:
|
||||||
targetToken.enqueue(serverPackets.notification(' '.join(message[1:])))
|
msg = ' '.join(message[1:]).strip()
|
||||||
|
if not msg:
|
||||||
|
return False
|
||||||
|
targetToken.enqueue(serverPackets.notification(msg))
|
||||||
return False
|
return False
|
||||||
else:
|
else:
|
||||||
return "User offline."
|
return "User offline."
|
||||||
|
@ -105,6 +107,7 @@ def moderated(fro, chan, message):
|
||||||
def kickAll(fro, chan, message):
|
def kickAll(fro, chan, message):
|
||||||
# Kick everyone but mods/admins
|
# Kick everyone but mods/admins
|
||||||
toKick = []
|
toKick = []
|
||||||
|
with glob.tokens:
|
||||||
for key, value in glob.tokens.tokens.items():
|
for key, value in glob.tokens.tokens.items():
|
||||||
if not value.admin:
|
if not value.admin:
|
||||||
toKick.append(key)
|
toKick.append(key)
|
||||||
|
@ -118,15 +121,18 @@ def kickAll(fro, chan, message):
|
||||||
|
|
||||||
def kick(fro, chan, message):
|
def kick(fro, chan, message):
|
||||||
# Get parameters
|
# Get parameters
|
||||||
target = message[0].replace("_", " ")
|
target = message[0].lower()
|
||||||
|
if target == glob.BOT_NAME.lower():
|
||||||
|
return "Nope."
|
||||||
|
|
||||||
# Get target token and make sure is connected
|
# Get target token and make sure is connected
|
||||||
targetToken = glob.tokens.getTokenFromUsername(target)
|
tokens = glob.tokens.getTokenFromUsername(userUtils.safeUsername(target), safe=True, _all=True)
|
||||||
if targetToken is None:
|
if len(tokens) == 0:
|
||||||
return "{} is not online".format(target)
|
return "{} is not online".format(target)
|
||||||
|
|
||||||
# Kick user
|
# Kick users
|
||||||
targetToken.kick()
|
for i in tokens:
|
||||||
|
i.kick()
|
||||||
|
|
||||||
# Bot response
|
# Bot response
|
||||||
return "{} has been kicked from the server.".format(target)
|
return "{} has been kicked from the server.".format(target)
|
||||||
|
@ -134,22 +140,25 @@ def kick(fro, chan, message):
|
||||||
def fokabotReconnect(fro, chan, message):
|
def fokabotReconnect(fro, chan, message):
|
||||||
# Check if fokabot is already connected
|
# Check if fokabot is already connected
|
||||||
if glob.tokens.getTokenFromUserID(999) is not None:
|
if glob.tokens.getTokenFromUserID(999) is not None:
|
||||||
return "Fokabot is already connected to Bancho"
|
return "{} is already connected to Bancho".format(glob.BOT_NAME)
|
||||||
|
|
||||||
# Fokabot is not connected, connect it
|
# Fokabot is not connected, connect it
|
||||||
fokabot.connect()
|
fokabot.connect()
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def silence(fro, chan, message):
|
def silence(fro, chan, message):
|
||||||
for i in message:
|
message = [x.lower() for x in message]
|
||||||
i = i.lower()
|
target = message[0]
|
||||||
target = message[0].replace("_", " ")
|
|
||||||
amount = message[1]
|
amount = message[1]
|
||||||
unit = message[2]
|
unit = message[2]
|
||||||
reason = ' '.join(message[3:])
|
reason = ' '.join(message[3:]).strip()
|
||||||
|
if not reason:
|
||||||
|
return "Please provide a valid reason."
|
||||||
|
if not amount.isdigit():
|
||||||
|
return "The amount must be a number."
|
||||||
|
|
||||||
# Get target user ID
|
# Get target user ID
|
||||||
targetUserID = userUtils.getID(target)
|
targetUserID = userUtils.getIDSafe(target)
|
||||||
userID = userUtils.getID(fro)
|
userID = userUtils.getID(fro)
|
||||||
|
|
||||||
# Make sure the user exists
|
# Make sure the user exists
|
||||||
|
@ -173,7 +182,7 @@ def silence(fro, chan, message):
|
||||||
return "Invalid silence time. Max silence time is 7 days."
|
return "Invalid silence time. Max silence time is 7 days."
|
||||||
|
|
||||||
# Send silence packet to target if he's connected
|
# Send silence packet to target if he's connected
|
||||||
targetToken = glob.tokens.getTokenFromUsername(target)
|
targetToken = glob.tokens.getTokenFromUsername(userUtils.safeUsername(target), safe=True)
|
||||||
if targetToken is not None:
|
if targetToken is not None:
|
||||||
# user online, silence both in db and with packet
|
# user online, silence both in db and with packet
|
||||||
targetToken.silence(silenceTime, reason, userID)
|
targetToken.silence(silenceTime, reason, userID)
|
||||||
|
@ -189,16 +198,16 @@ def removeSilence(fro, chan, message):
|
||||||
# Get parameters
|
# Get parameters
|
||||||
for i in message:
|
for i in message:
|
||||||
i = i.lower()
|
i = i.lower()
|
||||||
target = message[0].replace("_", " ")
|
target = message[0]
|
||||||
|
|
||||||
# Make sure the user exists
|
# Make sure the user exists
|
||||||
targetUserID = userUtils.getID(target)
|
targetUserID = userUtils.getIDSafe(target)
|
||||||
userID = userUtils.getID(fro)
|
userID = userUtils.getID(fro)
|
||||||
if not targetUserID:
|
if not targetUserID:
|
||||||
return "{}: user not found".format(target)
|
return "{}: user not found".format(target)
|
||||||
|
|
||||||
# Send new silence end packet to user if he's online
|
# Send new silence end packet to user if he's online
|
||||||
targetToken = glob.tokens.getTokenFromUsername(target)
|
targetToken = glob.tokens.getTokenFromUsername(userUtils.safeUsername(target), safe=True)
|
||||||
if targetToken is not None:
|
if targetToken is not None:
|
||||||
# User online, remove silence both in db and with packet
|
# User online, remove silence both in db and with packet
|
||||||
targetToken.silence(0, "", userID)
|
targetToken.silence(0, "", userID)
|
||||||
|
@ -212,10 +221,10 @@ def ban(fro, chan, message):
|
||||||
# Get parameters
|
# Get parameters
|
||||||
for i in message:
|
for i in message:
|
||||||
i = i.lower()
|
i = i.lower()
|
||||||
target = message[0].replace("_", " ")
|
target = message[0]
|
||||||
|
|
||||||
# Make sure the user exists
|
# Make sure the user exists
|
||||||
targetUserID = userUtils.getID(target)
|
targetUserID = userUtils.getIDSafe(target)
|
||||||
userID = userUtils.getID(fro)
|
userID = userUtils.getID(fro)
|
||||||
if not targetUserID:
|
if not targetUserID:
|
||||||
return "{}: user not found".format(target)
|
return "{}: user not found".format(target)
|
||||||
|
@ -224,7 +233,7 @@ def ban(fro, chan, message):
|
||||||
userUtils.ban(targetUserID)
|
userUtils.ban(targetUserID)
|
||||||
|
|
||||||
# Send ban packet to the user if he's online
|
# Send ban packet to the user if he's online
|
||||||
targetToken = glob.tokens.getTokenFromUsername(target)
|
targetToken = glob.tokens.getTokenFromUsername(userUtils.safeUsername(target), safe=True)
|
||||||
if targetToken is not None:
|
if targetToken is not None:
|
||||||
targetToken.enqueue(serverPackets.loginBanned())
|
targetToken.enqueue(serverPackets.loginBanned())
|
||||||
|
|
||||||
|
@ -235,10 +244,10 @@ def unban(fro, chan, message):
|
||||||
# Get parameters
|
# Get parameters
|
||||||
for i in message:
|
for i in message:
|
||||||
i = i.lower()
|
i = i.lower()
|
||||||
target = message[0].replace("_", " ")
|
target = message[0]
|
||||||
|
|
||||||
# Make sure the user exists
|
# Make sure the user exists
|
||||||
targetUserID = userUtils.getID(target)
|
targetUserID = userUtils.getIDSafe(target)
|
||||||
userID = userUtils.getID(fro)
|
userID = userUtils.getID(fro)
|
||||||
if not targetUserID:
|
if not targetUserID:
|
||||||
return "{}: user not found".format(target)
|
return "{}: user not found".format(target)
|
||||||
|
@ -253,10 +262,10 @@ def restrict(fro, chan, message):
|
||||||
# Get parameters
|
# Get parameters
|
||||||
for i in message:
|
for i in message:
|
||||||
i = i.lower()
|
i = i.lower()
|
||||||
target = message[0].replace("_", " ")
|
target = message[0]
|
||||||
|
|
||||||
# Make sure the user exists
|
# Make sure the user exists
|
||||||
targetUserID = userUtils.getID(target)
|
targetUserID = userUtils.getIDSafe(target)
|
||||||
userID = userUtils.getID(fro)
|
userID = userUtils.getID(fro)
|
||||||
if not targetUserID:
|
if not targetUserID:
|
||||||
return "{}: user not found".format(target)
|
return "{}: user not found".format(target)
|
||||||
|
@ -265,7 +274,7 @@ def restrict(fro, chan, message):
|
||||||
userUtils.restrict(targetUserID)
|
userUtils.restrict(targetUserID)
|
||||||
|
|
||||||
# Send restricted mode packet to this user if he's online
|
# Send restricted mode packet to this user if he's online
|
||||||
targetToken = glob.tokens.getTokenFromUsername(target)
|
targetToken = glob.tokens.getTokenFromUsername(userUtils.safeUsername(target), safe=True)
|
||||||
if targetToken is not None:
|
if targetToken is not None:
|
||||||
targetToken.setRestricted()
|
targetToken.setRestricted()
|
||||||
|
|
||||||
|
@ -276,10 +285,10 @@ def unrestrict(fro, chan, message):
|
||||||
# Get parameters
|
# Get parameters
|
||||||
for i in message:
|
for i in message:
|
||||||
i = i.lower()
|
i = i.lower()
|
||||||
target = message[0].replace("_", " ")
|
target = message[0]
|
||||||
|
|
||||||
# Make sure the user exists
|
# Make sure the user exists
|
||||||
targetUserID = userUtils.getID(target)
|
targetUserID = userUtils.getIDSafe(target)
|
||||||
userID = userUtils.getID(fro)
|
userID = userUtils.getID(fro)
|
||||||
if not targetUserID:
|
if not targetUserID:
|
||||||
return "{}: user not found".format(target)
|
return "{}: user not found".format(target)
|
||||||
|
@ -303,22 +312,7 @@ def systemShutdown(fro, chan, message):
|
||||||
return restartShutdown(False)
|
return restartShutdown(False)
|
||||||
|
|
||||||
def systemReload(fro, chan, message):
|
def systemReload(fro, chan, message):
|
||||||
# Reload settings from bancho_settings
|
glob.banchoConf.reload()
|
||||||
glob.banchoConf.loadSettings()
|
|
||||||
|
|
||||||
# Reload channels too
|
|
||||||
glob.channels.loadChannels()
|
|
||||||
|
|
||||||
# And chat filters
|
|
||||||
glob.chatFilters.loadFilters()
|
|
||||||
|
|
||||||
# Send new channels and new bottom icon to everyone
|
|
||||||
glob.streams.broadcast("main", serverPackets.mainMenuIcon(glob.banchoConf.config["menuIcon"]))
|
|
||||||
glob.streams.broadcast("main", serverPackets.channelInfoEnd())
|
|
||||||
for key, value in glob.channels.channels.items():
|
|
||||||
if value.publicRead == True and value.hidden == False:
|
|
||||||
glob.streams.broadcast("main", serverPackets.channelInfo(key))
|
|
||||||
|
|
||||||
return "Bancho settings reloaded!"
|
return "Bancho settings reloaded!"
|
||||||
|
|
||||||
def systemMaintenance(fro, chan, message):
|
def systemMaintenance(fro, chan, message):
|
||||||
|
@ -339,6 +333,7 @@ def systemMaintenance(fro, chan, message):
|
||||||
who = []
|
who = []
|
||||||
|
|
||||||
# Disconnect everyone but mod/admins
|
# Disconnect everyone but mod/admins
|
||||||
|
with glob.tokens:
|
||||||
for _, value in glob.tokens.tokens.items():
|
for _, value in glob.tokens.tokens.items():
|
||||||
if not value.admin:
|
if not value.admin:
|
||||||
who.append(value.userID)
|
who.append(value.userID)
|
||||||
|
@ -359,7 +354,13 @@ def systemStatus(fro, chan, message):
|
||||||
data = systemHelper.getSystemInfo()
|
data = systemHelper.getSystemInfo()
|
||||||
|
|
||||||
# Final message
|
# Final message
|
||||||
|
letsVersion = glob.redis.get("lets:version")
|
||||||
|
if letsVersion is None:
|
||||||
|
letsVersion = "\_(xd)_/"
|
||||||
|
else:
|
||||||
|
letsVersion = letsVersion.decode("utf-8")
|
||||||
msg = "pep.py bancho server v{}\n".format(glob.VERSION)
|
msg = "pep.py bancho server v{}\n".format(glob.VERSION)
|
||||||
|
msg += "LETS scores server v{}\n".format(letsVersion)
|
||||||
msg += "made by the Ripple team\n"
|
msg += "made by the Ripple team\n"
|
||||||
msg += "\n"
|
msg += "\n"
|
||||||
msg += "=== BANCHO STATS ===\n"
|
msg += "=== BANCHO STATS ===\n"
|
||||||
|
@ -388,7 +389,7 @@ def getPPMessage(userID, just_data = False):
|
||||||
currentAcc = token.tillerino[2]
|
currentAcc = token.tillerino[2]
|
||||||
|
|
||||||
# Send request to LETS api
|
# Send request to LETS api
|
||||||
resp = requests.get("http://127.0.0.1:5002/api/v1/pp?b={}&m={}".format(currentMap, currentMods, currentAcc), timeout=10).text
|
resp = requests.get("http://127.0.0.1:5002/api/v1/pp?b={}&m={}".format(currentMap, currentMods), timeout=10).text
|
||||||
data = json.loads(resp)
|
data = json.loads(resp)
|
||||||
|
|
||||||
# Make sure status is in response data
|
# Make sure status is in response data
|
||||||
|
@ -398,7 +399,7 @@ def getPPMessage(userID, just_data = False):
|
||||||
# Make sure status is 200
|
# Make sure status is 200
|
||||||
if data["status"] != 200:
|
if data["status"] != 200:
|
||||||
if "message" in data:
|
if "message" in data:
|
||||||
return "Error in LETS API call ({}). Please tell this to a dev.".format(data["message"])
|
return "Error in LETS API call ({}).".format(data["message"])
|
||||||
else:
|
else:
|
||||||
raise exceptions.apiException
|
raise exceptions.apiException
|
||||||
|
|
||||||
|
@ -434,7 +435,7 @@ def getPPMessage(userID, just_data = False):
|
||||||
return "API Timeout. Please try again in a few seconds."
|
return "API Timeout. Please try again in a few seconds."
|
||||||
except exceptions.apiException:
|
except exceptions.apiException:
|
||||||
# API error
|
# API error
|
||||||
return "Unknown error in LETS API call. Please tell this to a dev."
|
return "Unknown error in LETS API call."
|
||||||
#except:
|
#except:
|
||||||
# Unknown exception
|
# Unknown exception
|
||||||
# TODO: print exception
|
# TODO: print exception
|
||||||
|
@ -442,6 +443,14 @@ def getPPMessage(userID, just_data = False):
|
||||||
|
|
||||||
def tillerinoNp(fro, chan, message):
|
def tillerinoNp(fro, chan, message):
|
||||||
try:
|
try:
|
||||||
|
# Bloodcat trigger for #spect_
|
||||||
|
if chan.startswith("#spect_"):
|
||||||
|
spectatorHostUserID = getSpectatorHostUserIDFromChannel(chan)
|
||||||
|
spectatorHostToken = glob.tokens.getTokenFromUserID(spectatorHostUserID, ignoreIRC=True)
|
||||||
|
if spectatorHostToken is None:
|
||||||
|
return False
|
||||||
|
return bloodcatMessage(spectatorHostToken.beatmapID)
|
||||||
|
|
||||||
# Run the command in PM only
|
# Run the command in PM only
|
||||||
if chan.startswith("#"):
|
if chan.startswith("#"):
|
||||||
return False
|
return False
|
||||||
|
@ -572,6 +581,10 @@ def tillerinoAcc(fro, chan, message):
|
||||||
|
|
||||||
def tillerinoLast(fro, chan, message):
|
def tillerinoLast(fro, chan, message):
|
||||||
try:
|
try:
|
||||||
|
# Run the command in PM only
|
||||||
|
if chan.startswith("#"):
|
||||||
|
return False
|
||||||
|
|
||||||
data = glob.db.fetch("""SELECT beatmaps.song_name as sn, scores.*,
|
data = glob.db.fetch("""SELECT beatmaps.song_name as sn, scores.*,
|
||||||
beatmaps.beatmap_id as bid, beatmaps.difficulty_std, beatmaps.difficulty_taiko, beatmaps.difficulty_ctb, beatmaps.difficulty_mania, beatmaps.max_combo as fc
|
beatmaps.beatmap_id as bid, beatmaps.difficulty_std, beatmaps.difficulty_taiko, beatmaps.difficulty_ctb, beatmaps.difficulty_mania, beatmaps.max_combo as fc
|
||||||
FROM scores
|
FROM scores
|
||||||
|
@ -587,11 +600,11 @@ def tillerinoLast(fro, chan, message):
|
||||||
rank = generalUtils.getRank(data["play_mode"], data["mods"], data["accuracy"],
|
rank = generalUtils.getRank(data["play_mode"], data["mods"], data["accuracy"],
|
||||||
data["300_count"], data["100_count"], data["50_count"], data["misses_count"])
|
data["300_count"], data["100_count"], data["50_count"], data["misses_count"])
|
||||||
|
|
||||||
ifPlayer = "{0} | ".format(fro) if chan != "FokaBot" else ""
|
ifPlayer = "{0} | ".format(fro) if chan != glob.BOT_NAME else ""
|
||||||
ifFc = " (FC)" if data["max_combo"] == data["fc"] else " {0}x/{1}x".format(data["max_combo"], data["fc"])
|
ifFc = " (FC)" if data["max_combo"] == data["fc"] else " {0}x/{1}x".format(data["max_combo"], data["fc"])
|
||||||
beatmapLink = "[http://osu.ppy.sh/b/{1} {0}]".format(data["sn"], data["bid"])
|
beatmapLink = "[http://osu.ppy.sh/b/{1} {0}]".format(data["sn"], data["bid"])
|
||||||
|
|
||||||
hasPP = data["play_mode"] == gameModes.STD or data["play_mode"] == gameModes.MANIA
|
hasPP = data["play_mode"] != gameModes.CTB
|
||||||
|
|
||||||
msg = ifPlayer
|
msg = ifPlayer
|
||||||
msg += beatmapLink
|
msg += beatmapLink
|
||||||
|
@ -664,7 +677,7 @@ def pp(fro, chan, message):
|
||||||
pp = userUtils.getPP(token.userID, gameMode)
|
pp = userUtils.getPP(token.userID, gameMode)
|
||||||
return "You have {:,} pp".format(pp)
|
return "You have {:,} pp".format(pp)
|
||||||
|
|
||||||
def updateBeatmap(fro, chan, to):
|
def updateBeatmap(fro, chan, message):
|
||||||
try:
|
try:
|
||||||
# Run the command in PM only
|
# Run the command in PM only
|
||||||
if chan.startswith("#"):
|
if chan.startswith("#"):
|
||||||
|
@ -679,26 +692,515 @@ def updateBeatmap(fro, chan, to):
|
||||||
if token.tillerino[0] == 0:
|
if token.tillerino[0] == 0:
|
||||||
return "Please give me a beatmap first with /np command."
|
return "Please give me a beatmap first with /np command."
|
||||||
|
|
||||||
# Send request
|
# Send the request to cheesegull
|
||||||
beatmapData = glob.db.fetch("SELECT beatmapset_id, song_name FROM beatmaps WHERE beatmap_id = %s LIMIT 1", [token.tillerino[0]])
|
ok, message = cheesegull.updateBeatmap(token.tillerino[0])
|
||||||
if beatmapData is None:
|
if ok:
|
||||||
return "Couldn't find beatmap data in database. Please load the beatmap's leaderboard and try again."
|
return "An update request for that beatmap has been queued. Check back in a few minutes and the beatmap should be updated!"
|
||||||
|
|
||||||
response = requests.post("{}/api/v1/update_beatmap".format(glob.conf.config["mirror"]["apiurl"]), {
|
|
||||||
"beatmap_set_id": beatmapData["beatmapset_id"],
|
|
||||||
"beatmap_name": beatmapData["song_name"],
|
|
||||||
"username": token.username,
|
|
||||||
"key": glob.conf.config["mirror"]["apikey"]
|
|
||||||
})
|
|
||||||
if response.status_code == 200:
|
|
||||||
return "An update request for that beatmap has been queued. You'll receive a message once the beatmap has been updated on our mirror!"
|
|
||||||
elif response.status_code == 429:
|
|
||||||
return "You are sending too many beatmaps update requests. Wait a bit and retry later."
|
|
||||||
else:
|
else:
|
||||||
return "Error in beatmap mirror API request. Tell this to a dev: {}".format(response.text)
|
return "Error in beatmap mirror API request: {}".format(message)
|
||||||
except:
|
except:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
def report(fro, chan, message):
|
||||||
|
msg = ""
|
||||||
|
try:
|
||||||
|
# TODO: Rate limit
|
||||||
|
# Regex on message
|
||||||
|
reportRegex = re.compile("^(.+) \((.+)\)\:(?: )?(.+)?$")
|
||||||
|
result = reportRegex.search(" ".join(message))
|
||||||
|
|
||||||
|
# Make sure the message matches the regex
|
||||||
|
if result is None:
|
||||||
|
raise exceptions.invalidArgumentsException()
|
||||||
|
|
||||||
|
# Get username, report reason and report info
|
||||||
|
target, reason, additionalInfo = result.groups()
|
||||||
|
target = chat.fixUsernameForBancho(target)
|
||||||
|
|
||||||
|
# Make sure the target is not foka
|
||||||
|
if target.lower() == glob.BOT_NAME.lower():
|
||||||
|
raise exceptions.invalidUserException()
|
||||||
|
|
||||||
|
# Make sure the user exists
|
||||||
|
targetID = userUtils.getID(target)
|
||||||
|
if targetID == 0:
|
||||||
|
raise exceptions.userNotFoundException()
|
||||||
|
|
||||||
|
# Make sure that the user has specified additional info if report reason is 'Other'
|
||||||
|
if reason.lower() == "other" and additionalInfo is None:
|
||||||
|
raise exceptions.missingReportInfoException()
|
||||||
|
|
||||||
|
# Get the token if possible
|
||||||
|
chatlog = ""
|
||||||
|
token = glob.tokens.getTokenFromUsername(userUtils.safeUsername(target), safe=True)
|
||||||
|
if token is not None:
|
||||||
|
chatlog = token.getMessagesBufferString()
|
||||||
|
|
||||||
|
# Everything is fine, submit report
|
||||||
|
glob.db.execute("INSERT INTO reports (id, from_uid, to_uid, reason, chatlog, time) VALUES (NULL, %s, %s, %s, %s, %s)", [userUtils.getID(fro), targetID, "{reason} - ingame {info}".format(reason=reason, info="({})".format(additionalInfo) if additionalInfo is not None else ""), chatlog, int(time.time())])
|
||||||
|
msg = "You've reported {target} for {reason}{info}. A Community Manager will check your report as soon as possible. Every !report message you may see in chat wasn't sent to anyone, so nobody in chat, but admins, know about your report. Thank you for reporting!".format(target=target, reason=reason, info="" if additionalInfo is None else " (" + additionalInfo + ")")
|
||||||
|
adminMsg = "{user} has reported {target} for {reason} ({info})".format(user=fro, target=target, reason=reason, info=additionalInfo)
|
||||||
|
|
||||||
|
# Log report in #admin and on discord
|
||||||
|
chat.sendMessage(glob.BOT_NAME, "#admin", adminMsg)
|
||||||
|
log.warning(adminMsg, discord="cm")
|
||||||
|
except exceptions.invalidUserException:
|
||||||
|
msg = "Hello, {} here! You can't report me. I won't forget what you've tried to do. Watch out.".format(glob.BOT_NAME)
|
||||||
|
except exceptions.invalidArgumentsException:
|
||||||
|
msg = "Invalid report command syntax. To report an user, click on it and select 'Report user'."
|
||||||
|
except exceptions.userNotFoundException:
|
||||||
|
msg = "The user you've tried to report doesn't exist."
|
||||||
|
except exceptions.missingReportInfoException:
|
||||||
|
msg = "Please specify the reason of your report."
|
||||||
|
except:
|
||||||
|
raise
|
||||||
|
finally:
|
||||||
|
if msg != "":
|
||||||
|
token = glob.tokens.getTokenFromUsername(fro)
|
||||||
|
if token is not None:
|
||||||
|
if token.irc:
|
||||||
|
chat.sendMessage(glob.BOT_NAME, fro, msg)
|
||||||
|
else:
|
||||||
|
token.enqueue(serverPackets.notification(msg))
|
||||||
|
return False
|
||||||
|
|
||||||
|
def getMatchIDFromChannel(chan):
|
||||||
|
if not chan.lower().startswith("#multi_"):
|
||||||
|
raise exceptions.wrongChannelException()
|
||||||
|
parts = chan.lower().split("_")
|
||||||
|
if len(parts) < 2 or not parts[1].isdigit():
|
||||||
|
raise exceptions.wrongChannelException()
|
||||||
|
matchID = int(parts[1])
|
||||||
|
if matchID not in glob.matches.matches:
|
||||||
|
raise exceptions.matchNotFoundException()
|
||||||
|
return matchID
|
||||||
|
|
||||||
|
def getSpectatorHostUserIDFromChannel(chan):
|
||||||
|
if not chan.lower().startswith("#spect_"):
|
||||||
|
raise exceptions.wrongChannelException()
|
||||||
|
parts = chan.lower().split("_")
|
||||||
|
if len(parts) < 2 or not parts[1].isdigit():
|
||||||
|
raise exceptions.wrongChannelException()
|
||||||
|
userID = int(parts[1])
|
||||||
|
return userID
|
||||||
|
|
||||||
|
def multiplayer(fro, chan, message):
|
||||||
|
def mpMake():
|
||||||
|
if len(message) < 2:
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp make <name>")
|
||||||
|
matchName = " ".join(message[1:]).strip()
|
||||||
|
if not matchName:
|
||||||
|
raise exceptions.invalidArgumentsException("Match name must not be empty!")
|
||||||
|
matchID = glob.matches.createMatch(matchName, generalUtils.stringMd5(generalUtils.randomString(32)), 0, "Tournament", "", 0, -1, isTourney=True)
|
||||||
|
glob.matches.matches[matchID].sendUpdates()
|
||||||
|
return "Tourney match #{} created!".format(matchID)
|
||||||
|
|
||||||
|
def mpJoin():
|
||||||
|
if len(message) < 2 or not message[1].isdigit():
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp join <id>")
|
||||||
|
matchID = int(message[1])
|
||||||
|
userToken = glob.tokens.getTokenFromUsername(fro, ignoreIRC=True)
|
||||||
|
if userToken is None:
|
||||||
|
raise exceptions.invalidArgumentsException(
|
||||||
|
"No game clients found for {}, can't join the match. "
|
||||||
|
"If you're a referee and you want to join the chat "
|
||||||
|
"channel from IRC, use /join #multi_{} instead.".format(fro, matchID)
|
||||||
|
)
|
||||||
|
userToken.joinMatch(matchID)
|
||||||
|
return "Attempting to join match #{}!".format(matchID)
|
||||||
|
|
||||||
|
def mpClose():
|
||||||
|
matchID = getMatchIDFromChannel(chan)
|
||||||
|
glob.matches.disposeMatch(matchID)
|
||||||
|
return "Multiplayer match #{} disposed successfully".format(matchID)
|
||||||
|
|
||||||
|
def mpLock():
|
||||||
|
matchID = getMatchIDFromChannel(chan)
|
||||||
|
glob.matches.matches[matchID].isLocked = True
|
||||||
|
return "This match has been locked"
|
||||||
|
|
||||||
|
def mpUnlock():
|
||||||
|
matchID = getMatchIDFromChannel(chan)
|
||||||
|
glob.matches.matches[matchID].isLocked = False
|
||||||
|
return "This match has been unlocked"
|
||||||
|
|
||||||
|
def mpSize():
|
||||||
|
if len(message) < 2 or not message[1].isdigit() or int(message[1]) < 2 or int(message[1]) > 16:
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp size <slots(2-16)>")
|
||||||
|
matchSize = int(message[1])
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
_match.forceSize(matchSize)
|
||||||
|
return "Match size changed to {}".format(matchSize)
|
||||||
|
|
||||||
|
def mpMove():
|
||||||
|
if len(message) < 3 or not message[2].isdigit() or int(message[2]) < 0 or int(message[2]) > 16:
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp move <username> <slot>")
|
||||||
|
username = message[1]
|
||||||
|
newSlotID = int(message[2])
|
||||||
|
userID = userUtils.getIDSafe(username)
|
||||||
|
if userID is None:
|
||||||
|
raise exceptions.userNotFoundException("No such user")
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
success = _match.userChangeSlot(userID, newSlotID)
|
||||||
|
if success:
|
||||||
|
result = "Player {} moved to slot {}".format(username, newSlotID)
|
||||||
|
else:
|
||||||
|
result = "You can't use that slot: it's either already occupied by someone else or locked"
|
||||||
|
return result
|
||||||
|
|
||||||
|
def mpHost():
|
||||||
|
if len(message) < 2:
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp host <username>")
|
||||||
|
username = message[1].strip()
|
||||||
|
if not username:
|
||||||
|
raise exceptions.invalidArgumentsException("Please provide a username")
|
||||||
|
userID = userUtils.getIDSafe(username)
|
||||||
|
if userID is None:
|
||||||
|
raise exceptions.userNotFoundException("No such user")
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
success = _match.setHost(userID)
|
||||||
|
return "{} is now the host".format(username) if success else "Couldn't give host to {}".format(username)
|
||||||
|
|
||||||
|
def mpClearHost():
|
||||||
|
matchID = getMatchIDFromChannel(chan)
|
||||||
|
glob.matches.matches[matchID].removeHost()
|
||||||
|
return "Host has been removed from this match"
|
||||||
|
|
||||||
|
def mpStart():
|
||||||
|
def _start():
|
||||||
|
matchID = getMatchIDFromChannel(chan)
|
||||||
|
success = glob.matches.matches[matchID].start()
|
||||||
|
if not success:
|
||||||
|
chat.sendMessage(glob.BOT_NAME, chan, "Couldn't start match. Make sure there are enough players and "
|
||||||
|
"teams are valid. The match has been unlocked.")
|
||||||
|
else:
|
||||||
|
chat.sendMessage(glob.BOT_NAME, chan, "Have fun!")
|
||||||
|
|
||||||
|
|
||||||
|
def _decreaseTimer(t):
|
||||||
|
if t <= 0:
|
||||||
|
_start()
|
||||||
|
else:
|
||||||
|
if t % 10 == 0 or t <= 5:
|
||||||
|
chat.sendMessage(glob.BOT_NAME, chan, "Match starts in {} seconds.".format(t))
|
||||||
|
threading.Timer(1.00, _decreaseTimer, [t - 1]).start()
|
||||||
|
|
||||||
|
if len(message) < 2 or not message[1].isdigit():
|
||||||
|
startTime = 0
|
||||||
|
else:
|
||||||
|
startTime = int(message[1])
|
||||||
|
|
||||||
|
force = False if len(message) < 3 else message[2].lower() == "force"
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
|
||||||
|
# Force everyone to ready
|
||||||
|
someoneNotReady = False
|
||||||
|
for i, slot in enumerate(_match.slots):
|
||||||
|
if slot.status != slotStatuses.READY and slot.user is not None:
|
||||||
|
someoneNotReady = True
|
||||||
|
if force:
|
||||||
|
_match.toggleSlotReady(i)
|
||||||
|
|
||||||
|
if someoneNotReady and not force:
|
||||||
|
return "Some users aren't ready yet. Use '!mp start force' if you want to start the match, " \
|
||||||
|
"even with non-ready players."
|
||||||
|
|
||||||
|
if startTime == 0:
|
||||||
|
_start()
|
||||||
|
return "Starting match"
|
||||||
|
else:
|
||||||
|
_match.isStarting = True
|
||||||
|
threading.Timer(1.00, _decreaseTimer, [startTime - 1]).start()
|
||||||
|
return "Match starts in {} seconds. The match has been locked. " \
|
||||||
|
"Please don't leave the match during the countdown " \
|
||||||
|
"or you might receive a penalty.".format(startTime)
|
||||||
|
|
||||||
|
def mpInvite():
|
||||||
|
if len(message) < 2:
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp invite <username>")
|
||||||
|
username = message[1].strip()
|
||||||
|
if not username:
|
||||||
|
raise exceptions.invalidArgumentsException("Please provide a username")
|
||||||
|
userID = userUtils.getIDSafe(username)
|
||||||
|
if userID is None:
|
||||||
|
raise exceptions.userNotFoundException("No such user")
|
||||||
|
token = glob.tokens.getTokenFromUserID(userID, ignoreIRC=True)
|
||||||
|
if token is None:
|
||||||
|
raise exceptions.invalidUserException("That user is not connected to bancho right now.")
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
_match.invite(999, userID)
|
||||||
|
token.enqueue(serverPackets.notification("Please accept the invite you've just received from {} to "
|
||||||
|
"enter your tourney match.".format(glob.BOT_NAME)))
|
||||||
|
return "An invite to this match has been sent to {}".format(username)
|
||||||
|
|
||||||
|
def mpMap():
|
||||||
|
if len(message) < 2 or not message[1].isdigit() or (len(message) == 3 and not message[2].isdigit()):
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp map <beatmapid> [<gamemode>]")
|
||||||
|
beatmapID = int(message[1])
|
||||||
|
gameMode = int(message[2]) if len(message) == 3 else 0
|
||||||
|
if gameMode < 0 or gameMode > 3:
|
||||||
|
raise exceptions.invalidArgumentsException("Gamemode must be 0, 1, 2 or 3")
|
||||||
|
beatmapData = glob.db.fetch("SELECT * FROM beatmaps WHERE beatmap_id = %s LIMIT 1", [beatmapID])
|
||||||
|
if beatmapData is None:
|
||||||
|
raise exceptions.invalidArgumentsException("The beatmap you've selected couldn't be found in the database."
|
||||||
|
"If the beatmap id is valid, please load the scoreboard first in "
|
||||||
|
"order to cache it, then try again.")
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
_match.beatmapID = beatmapID
|
||||||
|
_match.beatmapName = beatmapData["song_name"]
|
||||||
|
_match.beatmapMD5 = beatmapData["beatmap_md5"]
|
||||||
|
_match.gameMode = gameMode
|
||||||
|
_match.resetReady()
|
||||||
|
_match.sendUpdates()
|
||||||
|
return "Match map has been updated"
|
||||||
|
|
||||||
|
def mpSet():
|
||||||
|
if len(message) < 2 or not message[1].isdigit() or \
|
||||||
|
(len(message) >= 3 and not message[2].isdigit()) or \
|
||||||
|
(len(message) >= 4 and not message[3].isdigit()):
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp set <teammode> [<scoremode>] [<size>]")
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
matchTeamType = int(message[1])
|
||||||
|
matchScoringType = int(message[2]) if len(message) >= 3 else _match.matchScoringType
|
||||||
|
if not 0 <= matchTeamType <= 3:
|
||||||
|
raise exceptions.invalidArgumentsException("Match team type must be between 0 and 3")
|
||||||
|
if not 0 <= matchScoringType <= 3:
|
||||||
|
raise exceptions.invalidArgumentsException("Match scoring type must be between 0 and 3")
|
||||||
|
oldMatchTeamType = _match.matchTeamType
|
||||||
|
_match.matchTeamType = matchTeamType
|
||||||
|
_match.matchScoringType = matchScoringType
|
||||||
|
if len(message) >= 4:
|
||||||
|
_match.forceSize(int(message[3]))
|
||||||
|
if _match.matchTeamType != oldMatchTeamType:
|
||||||
|
_match.initializeTeams()
|
||||||
|
if _match.matchTeamType == matchTeamTypes.TAG_COOP or _match.matchTeamType == matchTeamTypes.TAG_TEAM_VS:
|
||||||
|
_match.matchModMode = matchModModes.NORMAL
|
||||||
|
|
||||||
|
_match.sendUpdates()
|
||||||
|
return "Match settings have been updated!"
|
||||||
|
|
||||||
|
def mpAbort():
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
_match.abort()
|
||||||
|
return "Match aborted!"
|
||||||
|
|
||||||
|
def mpKick():
|
||||||
|
if len(message) < 2:
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp kick <username>")
|
||||||
|
username = message[1].strip()
|
||||||
|
if not username:
|
||||||
|
raise exceptions.invalidArgumentsException("Please provide a username")
|
||||||
|
userID = userUtils.getIDSafe(username)
|
||||||
|
if userID is None:
|
||||||
|
raise exceptions.userNotFoundException("No such user")
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
slotID = _match.getUserSlotID(userID)
|
||||||
|
if slotID is None:
|
||||||
|
raise exceptions.userNotFoundException("The specified user is not in this match")
|
||||||
|
for i in range(0, 2):
|
||||||
|
_match.toggleSlotLocked(slotID)
|
||||||
|
return "{} has been kicked from the match.".format(username)
|
||||||
|
|
||||||
|
def mpPassword():
|
||||||
|
password = "" if len(message) < 2 or not message[1].strip() else message[1]
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
_match.changePassword(password)
|
||||||
|
return "Match password has been changed!"
|
||||||
|
|
||||||
|
def mpRandomPassword():
|
||||||
|
password = generalUtils.stringMd5(generalUtils.randomString(32))
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
_match.changePassword(password)
|
||||||
|
return "Match password has been changed to a random one"
|
||||||
|
|
||||||
|
def mpMods():
|
||||||
|
if len(message) < 2:
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp <mod1> [<mod2>] ...")
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
newMods = 0
|
||||||
|
freeMod = False
|
||||||
|
for _mod in message[1:]:
|
||||||
|
if _mod.lower().strip() == "hd":
|
||||||
|
newMods |= mods.HIDDEN
|
||||||
|
elif _mod.lower().strip() == "hr":
|
||||||
|
newMods |= mods.HARDROCK
|
||||||
|
elif _mod.lower().strip() == "dt":
|
||||||
|
newMods |= mods.DOUBLETIME
|
||||||
|
elif _mod.lower().strip() == "fl":
|
||||||
|
newMods |= mods.FLASHLIGHT
|
||||||
|
elif _mod.lower().strip() == "fi":
|
||||||
|
newMods |= mods.FADEIN
|
||||||
|
elif _mod.lower().strip() == "ez":
|
||||||
|
newMods |= mods.EASY
|
||||||
|
if _mod.lower().strip() == "none":
|
||||||
|
newMods = 0
|
||||||
|
|
||||||
|
if _mod.lower().strip() == "freemod":
|
||||||
|
freeMod = True
|
||||||
|
|
||||||
|
_match.matchModMode = matchModModes.FREE_MOD if freeMod else matchModModes.NORMAL
|
||||||
|
_match.resetReady()
|
||||||
|
if _match.matchModMode == matchModModes.FREE_MOD:
|
||||||
|
_match.resetMods()
|
||||||
|
_match.changeMods(newMods)
|
||||||
|
return "Match mods have been updated!"
|
||||||
|
|
||||||
|
def mpTeam():
|
||||||
|
if len(message) < 3:
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp team <username> <colour>")
|
||||||
|
username = message[1].strip()
|
||||||
|
if not username:
|
||||||
|
raise exceptions.invalidArgumentsException("Please provide a username")
|
||||||
|
colour = message[2].lower().strip()
|
||||||
|
if colour not in ["red", "blue"]:
|
||||||
|
raise exceptions.invalidArgumentsException("Team colour must be red or blue")
|
||||||
|
userID = userUtils.getIDSafe(username)
|
||||||
|
if userID is None:
|
||||||
|
raise exceptions.userNotFoundException("No such user")
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
_match.changeTeam(userID, matchTeams.BLUE if colour == "blue" else matchTeams.RED)
|
||||||
|
return "{} is now in {} team".format(username, colour)
|
||||||
|
|
||||||
|
def mpSettings():
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
single = False if len(message) < 2 else message[1].strip().lower() == "single"
|
||||||
|
msg = "PLAYERS IN THIS MATCH "
|
||||||
|
if not single:
|
||||||
|
msg += "(use !mp settings single for a single-line version):"
|
||||||
|
msg += "\n"
|
||||||
|
else:
|
||||||
|
msg += ": "
|
||||||
|
empty = True
|
||||||
|
for slot in _match.slots:
|
||||||
|
if slot.user is None:
|
||||||
|
continue
|
||||||
|
readableStatuses = {
|
||||||
|
slotStatuses.READY: "ready",
|
||||||
|
slotStatuses.NOT_READY: "not ready",
|
||||||
|
slotStatuses.NO_MAP: "no map",
|
||||||
|
slotStatuses.PLAYING: "playing",
|
||||||
|
}
|
||||||
|
if slot.status not in readableStatuses:
|
||||||
|
readableStatus = "???"
|
||||||
|
else:
|
||||||
|
readableStatus = readableStatuses[slot.status]
|
||||||
|
empty = False
|
||||||
|
msg += "* [{team}] <{status}> ~ {username}{mods}{nl}".format(
|
||||||
|
team="red" if slot.team == matchTeams.RED else "blue" if slot.team == matchTeams.BLUE else "!! no team !!",
|
||||||
|
status=readableStatus,
|
||||||
|
username=glob.tokens.tokens[slot.user].username,
|
||||||
|
mods=" (+ {})".format(generalUtils.readableMods(slot.mods)) if slot.mods > 0 else "",
|
||||||
|
nl=" | " if single else "\n"
|
||||||
|
)
|
||||||
|
if empty:
|
||||||
|
msg += "Nobody.\n"
|
||||||
|
msg = msg.rstrip(" | " if single else "\n")
|
||||||
|
return msg
|
||||||
|
|
||||||
|
def mpScoreV():
|
||||||
|
if len(message) < 2 or message[1] not in ("1", "2"):
|
||||||
|
raise exceptions.invalidArgumentsException("Wrong syntax: !mp scorev <1|2>")
|
||||||
|
_match = glob.matches.matches[getMatchIDFromChannel(chan)]
|
||||||
|
_match.matchScoringType = matchScoringTypes.SCORE_V2 if message[1] == "2" else matchScoringTypes.SCORE
|
||||||
|
_match.sendUpdates()
|
||||||
|
return "Match scoring type set to scorev{}".format(message[1])
|
||||||
|
|
||||||
|
def mpHelp():
|
||||||
|
return "Supported subcommands: !mp <{}>".format("|".join(k for k in subcommands.keys()))
|
||||||
|
|
||||||
|
try:
|
||||||
|
subcommands = {
|
||||||
|
"make": mpMake,
|
||||||
|
"close": mpClose,
|
||||||
|
"join": mpJoin,
|
||||||
|
"lock": mpLock,
|
||||||
|
"unlock": mpUnlock,
|
||||||
|
"size": mpSize,
|
||||||
|
"move": mpMove,
|
||||||
|
"host": mpHost,
|
||||||
|
"clearhost": mpClearHost,
|
||||||
|
"start": mpStart,
|
||||||
|
"invite": mpInvite,
|
||||||
|
"map": mpMap,
|
||||||
|
"set": mpSet,
|
||||||
|
"abort": mpAbort,
|
||||||
|
"kick": mpKick,
|
||||||
|
"password": mpPassword,
|
||||||
|
"randompassword": mpRandomPassword,
|
||||||
|
"mods": mpMods,
|
||||||
|
"team": mpTeam,
|
||||||
|
"settings": mpSettings,
|
||||||
|
"scorev": mpScoreV,
|
||||||
|
"help": mpHelp
|
||||||
|
}
|
||||||
|
requestedSubcommand = message[0].lower().strip()
|
||||||
|
if requestedSubcommand not in subcommands:
|
||||||
|
raise exceptions.invalidArgumentsException("Invalid subcommand")
|
||||||
|
return subcommands[requestedSubcommand]()
|
||||||
|
except (exceptions.invalidArgumentsException, exceptions.userNotFoundException, exceptions.invalidUserException) as e:
|
||||||
|
return str(e)
|
||||||
|
except exceptions.wrongChannelException:
|
||||||
|
return "This command only works in multiplayer chat channels"
|
||||||
|
except exceptions.matchNotFoundException:
|
||||||
|
return "Match not found"
|
||||||
|
except:
|
||||||
|
raise
|
||||||
|
|
||||||
|
def switchServer(fro, chan, message):
|
||||||
|
# Get target user ID
|
||||||
|
target = message[0]
|
||||||
|
newServer = message[1].strip()
|
||||||
|
if not newServer:
|
||||||
|
return "Invalid server IP"
|
||||||
|
targetUserID = userUtils.getIDSafe(target)
|
||||||
|
userID = userUtils.getID(fro)
|
||||||
|
|
||||||
|
# Make sure the user exists
|
||||||
|
if not targetUserID:
|
||||||
|
return "{}: user not found".format(target)
|
||||||
|
|
||||||
|
# Connect the user to the end server
|
||||||
|
userToken = glob.tokens.getTokenFromUserID(userID, ignoreIRC=True, _all=False)
|
||||||
|
userToken.enqueue(serverPackets.switchServer(newServer))
|
||||||
|
|
||||||
|
# Disconnect the user from the origin server
|
||||||
|
# userToken.kick()
|
||||||
|
return "{} has been connected to {}".format(target, newServer)
|
||||||
|
|
||||||
|
def rtx(fro, chan, message):
|
||||||
|
target = message[0]
|
||||||
|
message = " ".join(message[1:]).strip()
|
||||||
|
if not message:
|
||||||
|
return "Invalid message"
|
||||||
|
targetUserID = userUtils.getIDSafe(target)
|
||||||
|
if not targetUserID:
|
||||||
|
return "{}: user not found".format(target)
|
||||||
|
userToken = glob.tokens.getTokenFromUserID(targetUserID, ignoreIRC=True, _all=False)
|
||||||
|
userToken.enqueue(serverPackets.rtx(message))
|
||||||
|
return ":ok_hand:"
|
||||||
|
|
||||||
|
def bloodcat(fro, chan, message):
|
||||||
|
try:
|
||||||
|
matchID = getMatchIDFromChannel(chan)
|
||||||
|
except exceptions.wrongChannelException:
|
||||||
|
matchID = None
|
||||||
|
try:
|
||||||
|
spectatorHostUserID = getSpectatorHostUserIDFromChannel(chan)
|
||||||
|
except exceptions.wrongChannelException:
|
||||||
|
spectatorHostUserID = None
|
||||||
|
|
||||||
|
if matchID is not None:
|
||||||
|
if matchID not in glob.matches.matches:
|
||||||
|
return "This match doesn't seem to exist... Or does it...?"
|
||||||
|
beatmapID = glob.matches.matches[matchID].beatmapID
|
||||||
|
else:
|
||||||
|
spectatorHostToken = glob.tokens.getTokenFromUserID(spectatorHostUserID, ignoreIRC=True)
|
||||||
|
if spectatorHostToken is None:
|
||||||
|
return "The spectator host is offline."
|
||||||
|
beatmapID = spectatorHostToken.beatmapID
|
||||||
|
return bloodcatMessage(beatmapID)
|
||||||
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
Commands list
|
Commands list
|
||||||
|
|
||||||
|
@ -707,11 +1209,6 @@ callback: function to call when the command is triggered. Optional.
|
||||||
response: text to return when the command is triggered. Optional.
|
response: text to return when the command is triggered. Optional.
|
||||||
syntax: command syntax. Arguments must be separated by spaces (eg: <arg1> <arg2>)
|
syntax: command syntax. Arguments must be separated by spaces (eg: <arg1> <arg2>)
|
||||||
privileges: privileges needed to execute the command. Optional.
|
privileges: privileges needed to execute the command. Optional.
|
||||||
|
|
||||||
NOTES:
|
|
||||||
- You CAN'T use both rank and minRank at the same time.
|
|
||||||
- If both rank and minrank are **not** present, everyone will be able to run that command.
|
|
||||||
- You MUST set trigger and callback/response, or the command won't work.
|
|
||||||
"""
|
"""
|
||||||
commands = [
|
commands = [
|
||||||
{
|
{
|
||||||
|
@ -723,10 +1220,10 @@ commands = [
|
||||||
"callback": faq
|
"callback": faq
|
||||||
}, {
|
}, {
|
||||||
"trigger": "!report",
|
"trigger": "!report",
|
||||||
"response": "Report command isn't here yet :c"
|
"callback": report
|
||||||
}, {
|
}, {
|
||||||
"trigger": "!help",
|
"trigger": "!help",
|
||||||
"response": "Click (here)[https://ripple.moe/index.php?p=16&id=4] for FokaBot's full command list"
|
"response": "Click (here)[https://ripple.moe/index.php?p=16&id=4] for the full command list"
|
||||||
}, #{
|
}, #{
|
||||||
#"trigger": "!ask",
|
#"trigger": "!ask",
|
||||||
#"syntax": "<question>",
|
#"syntax": "<question>",
|
||||||
|
@ -751,7 +1248,7 @@ commands = [
|
||||||
"callback": moderated
|
"callback": moderated
|
||||||
}, {
|
}, {
|
||||||
"trigger": "!kickall",
|
"trigger": "!kickall",
|
||||||
"privileges": privileges.ADMIN_KICK_USERS,
|
"privileges": privileges.ADMIN_MANAGE_SERVERS,
|
||||||
"callback": kickAll
|
"callback": kickAll
|
||||||
}, {
|
}, {
|
||||||
"trigger": "!kick",
|
"trigger": "!kick",
|
||||||
|
@ -759,7 +1256,7 @@ commands = [
|
||||||
"privileges": privileges.ADMIN_KICK_USERS,
|
"privileges": privileges.ADMIN_KICK_USERS,
|
||||||
"callback": kick
|
"callback": kick
|
||||||
}, {
|
}, {
|
||||||
"trigger": "!fokabot reconnect",
|
"trigger": "!bot reconnect",
|
||||||
"privileges": privileges.ADMIN_MANAGE_SERVERS,
|
"privileges": privileges.ADMIN_MANAGE_SERVERS,
|
||||||
"callback": fokabotReconnect
|
"callback": fokabotReconnect
|
||||||
}, {
|
}, {
|
||||||
|
@ -837,8 +1334,25 @@ commands = [
|
||||||
"callback": pp
|
"callback": pp
|
||||||
}, {
|
}, {
|
||||||
"trigger": "!update",
|
"trigger": "!update",
|
||||||
"callback": updateBeatmap,
|
"callback": updateBeatmap
|
||||||
"privileges": privileges.ADMIN_MANAGE_SERVERS, # TODO: Remove privileges for !update
|
}, {
|
||||||
|
"trigger": "!mp",
|
||||||
|
"privileges": privileges.USER_TOURNAMENT_STAFF,
|
||||||
|
"syntax": "<subcommand>",
|
||||||
|
"callback": multiplayer
|
||||||
|
}, {
|
||||||
|
"trigger": "!switchserver",
|
||||||
|
"privileges": privileges.ADMIN_MANAGE_SERVERS,
|
||||||
|
"syntax": "<username> <server_address>",
|
||||||
|
"callback": switchServer
|
||||||
|
}, {
|
||||||
|
"trigger": "!rtx",
|
||||||
|
"privileges": privileges.ADMIN_MANAGE_USERS,
|
||||||
|
"syntax": "<username> <message>",
|
||||||
|
"callback": rtx
|
||||||
|
}, {
|
||||||
|
"trigger": "!bloodcat",
|
||||||
|
"callback": bloodcat
|
||||||
}
|
}
|
||||||
#
|
#
|
||||||
# "trigger": "!acc",
|
# "trigger": "!acc",
|
||||||
|
|
|
@ -1,2 +1,2 @@
|
||||||
normal = 0
|
NORMAL = 0
|
||||||
freeMod = 1
|
FREE_MOD = 1
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
score = 0
|
SCORE = 0
|
||||||
accuracy = 1
|
ACCURACY = 1
|
||||||
combo = 2
|
COMBO = 2
|
||||||
|
SCORE_V2 = 3
|
|
@ -1,4 +1,4 @@
|
||||||
headToHead = 0
|
HEAD_TO_HEAD = 0
|
||||||
tagCoop = 1
|
TAG_COOP = 1
|
||||||
teamVs = 2
|
TEAM_VS = 2
|
||||||
tagTeamVs = 3
|
TAG_TEAM_VS = 3
|
||||||
|
|
|
@ -1,3 +1,3 @@
|
||||||
noTeam = 0
|
NO_TEAM = 0
|
||||||
blue = 1
|
BLUE = 1
|
||||||
red = 2
|
RED = 2
|
||||||
|
|
|
@ -78,5 +78,7 @@ server_userSilenced = 94
|
||||||
server_userPresenceBundle = 96
|
server_userPresenceBundle = 96
|
||||||
client_userPanelRequest = 97
|
client_userPanelRequest = 97
|
||||||
client_tournamentMatchInfoRequest = 93
|
client_tournamentMatchInfoRequest = 93
|
||||||
|
server_matchAbort = 106
|
||||||
|
server_switchServer = 107
|
||||||
client_tournamentJoinMatchChannel = 108
|
client_tournamentJoinMatchChannel = 108
|
||||||
client_tournamentLeaveMatchChannel = 109
|
client_tournamentLeaveMatchChannel = 109
|
|
@ -16,12 +16,12 @@ def forceUpdate():
|
||||||
|
|
||||||
def loginBanned():
|
def loginBanned():
|
||||||
packets = packetHelper.buildPacket(packetIDs.server_userID, [[-1, dataTypes.SINT32]])
|
packets = packetHelper.buildPacket(packetIDs.server_userID, [[-1, dataTypes.SINT32]])
|
||||||
packets += notification("You are banned. You can appeal after one month since your ban by sending an email to support@ripple.moe from the email address you've used to sign up.")
|
packets += notification("You are banned. You can appeal after one month since your ban by sending an email to {} from the email address you've used to sign up.".format(glob.conf.extra["pep.py"]["support-email"]))
|
||||||
return packets
|
return packets
|
||||||
|
|
||||||
def loginLocked():
|
def loginLocked():
|
||||||
packets = packetHelper.buildPacket(packetIDs.server_userID, [[-1, dataTypes.SINT32]])
|
packets = packetHelper.buildPacket(packetIDs.server_userID, [[-1, dataTypes.SINT32]])
|
||||||
packets += notification("Your account is locked. You can't log in, but your profile and scores are still visible from the website. If you want to unlock your account, send an email to support@ripple.moe from the email address you've used to sign up.")
|
packets += notification("Your account is locked. You can't log in, but your profile and scores are still visible from the website. If you want to unlock your account, send an email to {} from the email address you've used to sign up.".format(glob.conf.extra["pep.py"]["support-email"]))
|
||||||
return packets
|
return packets
|
||||||
|
|
||||||
def loginError():
|
def loginError():
|
||||||
|
@ -94,12 +94,12 @@ def userPanel(userID, force = False):
|
||||||
# Get username color according to rank
|
# Get username color according to rank
|
||||||
# Only admins and normal users are currently supported
|
# Only admins and normal users are currently supported
|
||||||
userRank = 0
|
userRank = 0
|
||||||
if username == "FokaBot":
|
if username == glob.BOT_NAME:
|
||||||
userRank |= userRanks.MOD
|
|
||||||
elif userUtils.isInPrivilegeGroup(userID, "community manager"):
|
|
||||||
userRank |= userRanks.MOD
|
userRank |= userRanks.MOD
|
||||||
elif userUtils.isInPrivilegeGroup(userID, "developer"):
|
elif userUtils.isInPrivilegeGroup(userID, "developer"):
|
||||||
userRank |= userRanks.ADMIN
|
userRank |= userRanks.ADMIN
|
||||||
|
elif userUtils.isInPrivilegeGroup(userID, "chat mod"):
|
||||||
|
userRank |= userRanks.MOD
|
||||||
elif (userToken.privileges & privileges.USER_DONOR) > 0:
|
elif (userToken.privileges & privileges.USER_DONOR) > 0:
|
||||||
userRank |= userRanks.SUPPORTER
|
userRank |= userRanks.SUPPORTER
|
||||||
else:
|
else:
|
||||||
|
@ -155,11 +155,13 @@ def channelJoinSuccess(userID, chan):
|
||||||
return packetHelper.buildPacket(packetIDs.server_channelJoinSuccess, [[chan, dataTypes.STRING]])
|
return packetHelper.buildPacket(packetIDs.server_channelJoinSuccess, [[chan, dataTypes.STRING]])
|
||||||
|
|
||||||
def channelInfo(chan):
|
def channelInfo(chan):
|
||||||
|
if chan not in glob.channels.channels:
|
||||||
|
return bytes()
|
||||||
channel = glob.channels.channels[chan]
|
channel = glob.channels.channels[chan]
|
||||||
return packetHelper.buildPacket(packetIDs.server_channelInfo, [
|
return packetHelper.buildPacket(packetIDs.server_channelInfo, [
|
||||||
[chan, dataTypes.STRING],
|
[channel.name, dataTypes.STRING],
|
||||||
[channel.description, dataTypes.STRING],
|
[channel.description, dataTypes.STRING],
|
||||||
[channel.getConnectedUsersCount(), dataTypes.UINT16]
|
[len(glob.streams.streams["chat/{}".format(chan)].clients), dataTypes.UINT16]
|
||||||
])
|
])
|
||||||
|
|
||||||
def channelInfoEnd():
|
def channelInfoEnd():
|
||||||
|
@ -200,17 +202,18 @@ def createMatch(matchID):
|
||||||
|
|
||||||
# Get match binary data and build packet
|
# Get match binary data and build packet
|
||||||
match = glob.matches.matches[matchID]
|
match = glob.matches.matches[matchID]
|
||||||
return packetHelper.buildPacket(packetIDs.server_newMatch, match.getMatchData())
|
matchData = match.getMatchData(censored=True)
|
||||||
|
return packetHelper.buildPacket(packetIDs.server_newMatch, matchData)
|
||||||
|
|
||||||
# TODO: Add match object argument to save some CPU
|
# TODO: Add match object argument to save some CPU
|
||||||
def updateMatch(matchID):
|
def updateMatch(matchID, censored = False):
|
||||||
# Make sure the match exists
|
# Make sure the match exists
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return bytes()
|
return bytes()
|
||||||
|
|
||||||
# Get match binary data and build packet
|
# Get match binary data and build packet
|
||||||
match = glob.matches.matches[matchID]
|
match = glob.matches.matches[matchID]
|
||||||
return packetHelper.buildPacket(packetIDs.server_updateMatch, match.getMatchData())
|
return packetHelper.buildPacket(packetIDs.server_updateMatch, match.getMatchData(censored=censored))
|
||||||
|
|
||||||
def matchStart(matchID):
|
def matchStart(matchID):
|
||||||
# Make sure the match exists
|
# Make sure the match exists
|
||||||
|
@ -261,6 +264,11 @@ def playerFailed(slotID):
|
||||||
def matchTransferHost():
|
def matchTransferHost():
|
||||||
return packetHelper.buildPacket(packetIDs.server_matchTransferHost)
|
return packetHelper.buildPacket(packetIDs.server_matchTransferHost)
|
||||||
|
|
||||||
|
def matchAbort():
|
||||||
|
return packetHelper.buildPacket(packetIDs.server_matchAbort)
|
||||||
|
|
||||||
|
def switchServer(address):
|
||||||
|
return packetHelper.buildPacket(packetIDs.server_switchServer, [[address, dataTypes.STRING]])
|
||||||
|
|
||||||
""" Other packets """
|
""" Other packets """
|
||||||
def notification(message):
|
def notification(message):
|
||||||
|
@ -268,3 +276,6 @@ def notification(message):
|
||||||
|
|
||||||
def banchoRestart(msUntilReconnection):
|
def banchoRestart(msUntilReconnection):
|
||||||
return packetHelper.buildPacket(packetIDs.server_restart, [[msUntilReconnection, dataTypes.UINT32]])
|
return packetHelper.buildPacket(packetIDs.server_restart, [[msUntilReconnection, dataTypes.UINT32]])
|
||||||
|
|
||||||
|
def rtx(message):
|
||||||
|
return packetHelper.buildPacket(0x69, [[message, dataTypes.STRING]])
|
|
@ -1,8 +1,8 @@
|
||||||
free = 1
|
FREE = 1
|
||||||
locked = 2
|
LOCKED = 2
|
||||||
notReady = 4
|
NOT_READY = 4
|
||||||
ready = 8
|
READY = 8
|
||||||
noMap = 16
|
NO_MAP = 16
|
||||||
playing = 32
|
PLAYING = 32
|
||||||
occupied = 124
|
OCCUPIED = 124
|
||||||
playingQuit = 128
|
PLAYING_QUIT = 128
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
"""Bancho user ranks"""
|
|
||||||
NORMAL = 0
|
NORMAL = 0
|
||||||
PLAYER = 1
|
PLAYER = 1
|
||||||
BAT = 2
|
BAT = 2
|
||||||
|
|
|
@ -1,25 +1,21 @@
|
||||||
from common.constants import actions
|
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from common.ripple import userUtils
|
|
||||||
from constants import clientPackets
|
from constants import clientPackets
|
||||||
from constants import serverPackets
|
from constants import serverPackets
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
|
|
||||||
def handle(userToken, packetData):
|
def handle(userToken, packetData):
|
||||||
# Get usertoken data
|
# Get usertoken data
|
||||||
userID = userToken.userID
|
userID = userToken.userID
|
||||||
username = userToken.username
|
username = userToken.username
|
||||||
|
|
||||||
# Make sure we are not banned
|
# Make sure we are not banned
|
||||||
if userUtils.isBanned(userID):
|
#if userUtils.isBanned(userID):
|
||||||
userToken.enqueue(serverPackets.loginBanned())
|
# userToken.enqueue(serverPackets.loginBanned())
|
||||||
return
|
# return
|
||||||
|
|
||||||
# Send restricted message if needed
|
# Send restricted message if needed
|
||||||
if not userToken.restricted:
|
#if userToken.restricted:
|
||||||
if userUtils.isRestricted(userID):
|
# userToken.checkRestricted(True)
|
||||||
userToken.setRestricted()
|
|
||||||
|
|
||||||
# Change action packet
|
# Change action packet
|
||||||
packetData = clientPackets.userActionChange(packetData)
|
packetData = clientPackets.userActionChange(packetData)
|
||||||
|
@ -35,8 +31,10 @@ if userToken.matchID != -1 and userToken.actionID != actions.MULTIPLAYING and us
|
||||||
'''
|
'''
|
||||||
|
|
||||||
# Update cached stats if our pp changed if we've just submitted a score or we've changed gameMode
|
# Update cached stats if our pp changed if we've just submitted a score or we've changed gameMode
|
||||||
if (userToken.actionID == actions.PLAYING or userToken.actionID == actions.MULTIPLAYING) or (userToken.pp != userUtils.getPP(userID, userToken.gameMode)) or (userToken.gameMode != packetData["gameMode"]):
|
#if (userToken.actionID == actions.PLAYING or userToken.actionID == actions.MULTIPLAYING) or (userToken.pp != userUtils.getPP(userID, userToken.gameMode)) or (userToken.gameMode != packetData["gameMode"]):
|
||||||
# Always update game mode, or we'll cache stats from the wrong game mode if we've changed it
|
|
||||||
|
# Update cached stats if we've changed gamemode
|
||||||
|
if userToken.gameMode != packetData["gameMode"]:
|
||||||
userToken.gameMode = packetData["gameMode"]
|
userToken.gameMode = packetData["gameMode"]
|
||||||
userToken.updateCachedStats()
|
userToken.updateCachedStats()
|
||||||
|
|
||||||
|
|
|
@ -15,10 +15,10 @@ def handle(userToken, packetData):
|
||||||
matchID = userToken.matchID
|
matchID = userToken.matchID
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Set slot or match mods according to modType
|
# Set slot or match mods according to modType
|
||||||
if match.matchModMode == matchModModes.freeMod:
|
with glob.matches.matches[matchID] as match:
|
||||||
|
if match.matchModMode == matchModModes.FREE_MOD:
|
||||||
# Freemod
|
# Freemod
|
||||||
# Host can set global DT/HT
|
# Host can set global DT/HT
|
||||||
if userID == match.hostUserID:
|
if userID == match.hostUserID:
|
||||||
|
|
|
@ -10,9 +10,7 @@ def handle(userToken, packetData):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Get our match
|
with glob.matches.matches[matchID] as match:
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Host check
|
# Host check
|
||||||
if userToken.userID != match.hostUserID:
|
if userToken.userID != match.hostUserID:
|
||||||
return
|
return
|
||||||
|
|
|
@ -21,10 +21,8 @@ def handle(userToken, packetData):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Get match object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Host check
|
# Host check
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
if userToken.userID != match.hostUserID:
|
if userToken.userID != match.hostUserID:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
@ -71,6 +69,7 @@ def handle(userToken, packetData):
|
||||||
|
|
||||||
oldBeatmapMD5 = match.beatmapMD5
|
oldBeatmapMD5 = match.beatmapMD5
|
||||||
oldMods = match.mods
|
oldMods = match.mods
|
||||||
|
oldMatchTeamType = match.matchTeamType
|
||||||
|
|
||||||
match.mods = packetData["mods"]
|
match.mods = packetData["mods"]
|
||||||
match.beatmapMD5 = packetData["beatmapMD5"]
|
match.beatmapMD5 = packetData["beatmapMD5"]
|
||||||
|
@ -80,35 +79,23 @@ def handle(userToken, packetData):
|
||||||
|
|
||||||
# Reset ready if needed
|
# Reset ready if needed
|
||||||
if oldMods != match.mods or oldBeatmapMD5 != match.beatmapMD5:
|
if oldMods != match.mods or oldBeatmapMD5 != match.beatmapMD5:
|
||||||
for i in range(0,16):
|
match.resetReady()
|
||||||
if match.slots[i].status == slotStatuses.ready:
|
|
||||||
match.slots[i].status = slotStatuses.notReady
|
|
||||||
|
|
||||||
# Reset mods if needed
|
# Reset mods if needed
|
||||||
if match.matchModMode == matchModModes.normal:
|
if match.matchModMode == matchModModes.NORMAL:
|
||||||
# Reset slot mods if not freeMods
|
# Reset slot mods if not freeMods
|
||||||
for i in range(0,16):
|
match.resetMods()
|
||||||
match.slots[i].mods = 0
|
|
||||||
else:
|
else:
|
||||||
# Reset match mods if freemod
|
# Reset match mods if freemod
|
||||||
match.mods = 0
|
match.mods = 0
|
||||||
|
|
||||||
# Set/reset teams
|
# Initialize teams if team type changed
|
||||||
if match.matchTeamType == matchTeamTypes.teamVs or match.matchTeamType == matchTeamTypes.tagTeamVs:
|
if match.matchTeamType != oldMatchTeamType:
|
||||||
# Set teams
|
match.initializeTeams()
|
||||||
c=0
|
|
||||||
for i in range(0,16):
|
|
||||||
if match.slots[i].team == matchTeams.noTeam:
|
|
||||||
match.slots[i].team = matchTeams.red if c % 2 == 0 else matchTeams.blue
|
|
||||||
c+=1
|
|
||||||
else:
|
|
||||||
# Reset teams
|
|
||||||
for i in range(0,16):
|
|
||||||
match.slots[i].team = matchTeams.noTeam
|
|
||||||
|
|
||||||
# Force no freemods if tag coop
|
# Force no freemods if tag coop
|
||||||
if match.matchTeamType == matchTeamTypes.tagCoop or match.matchTeamType == matchTeamTypes.tagTeamVs:
|
if match.matchTeamType == matchTeamTypes.TAG_COOP or match.matchTeamType == matchTeamTypes.TAG_TEAM_VS:
|
||||||
match.matchModMode = matchModModes.normal
|
match.matchModMode = matchModModes.NORMAL
|
||||||
|
|
||||||
# Send updated settings
|
# Send updated settings
|
||||||
match.sendUpdates()
|
match.sendUpdates()
|
||||||
|
|
|
@ -8,8 +8,6 @@ def handle(userToken, packetData):
|
||||||
# Read packet data
|
# Read packet data
|
||||||
packetData = clientPackets.changeSlot(packetData)
|
packetData = clientPackets.changeSlot(packetData)
|
||||||
|
|
||||||
# Get match
|
with glob.matches.matches[userToken.matchID] as match:
|
||||||
match = glob.matches.matches[userToken.matchID]
|
|
||||||
|
|
||||||
# Change slot
|
# Change slot
|
||||||
match.userChangeSlot(userID, packetData["slotID"])
|
match.userChangeSlot(userID, packetData["slotID"])
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from constants import clientPackets
|
from constants import clientPackets, serverPackets
|
||||||
from constants import exceptions
|
from constants import exceptions
|
||||||
from constants import serverPackets
|
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
|
|
||||||
|
@ -13,26 +12,32 @@ def handle(userToken, packetData):
|
||||||
# Read packet data
|
# Read packet data
|
||||||
packetData = clientPackets.createMatch(packetData)
|
packetData = clientPackets.createMatch(packetData)
|
||||||
|
|
||||||
|
# Make sure the name is valid
|
||||||
|
matchName = packetData["matchName"].strip()
|
||||||
|
if not matchName:
|
||||||
|
raise exceptions.matchCreateError()
|
||||||
|
|
||||||
# Create a match object
|
# Create a match object
|
||||||
# TODO: Player number check
|
# TODO: Player number check (Dirty hack below)
|
||||||
matchID = glob.matches.createMatch(packetData["matchName"], packetData["matchPassword"], packetData["beatmapID"], packetData["beatmapName"], packetData["beatmapMD5"], packetData["gameMode"], userID)
|
matchID = glob.matches.createMatch(matchName, packetData["matchPassword"].strip(), packetData["beatmapID"], packetData["beatmapName"], packetData["beatmapMD5"], packetData["gameMode"], userID)
|
||||||
|
|
||||||
# Make sure the match has been created
|
# Make sure the match has been created
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
raise exceptions.matchCreateError
|
raise exceptions.matchCreateError()
|
||||||
|
|
||||||
# Get match object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
# Join that match
|
# Join that match
|
||||||
userToken.joinMatch(matchID)
|
userToken.joinMatch(matchID)
|
||||||
|
|
||||||
|
# Disable slots (Dirty)
|
||||||
|
for i in range(0,16):
|
||||||
|
if match.slots[i].status is not 4:
|
||||||
|
match.slots[i].status = packetData["slot{}Status".format(i)]
|
||||||
|
|
||||||
# Give host to match creator
|
# Give host to match creator
|
||||||
match.setHost(userID)
|
match.setHost(userID)
|
||||||
match.sendUpdates()
|
match.sendUpdates()
|
||||||
match.changePassword(packetData["matchPassword"])
|
match.changePassword(packetData["matchPassword"])
|
||||||
|
|
||||||
# Console output
|
|
||||||
log.info("MPROOM{}: Room created!".format(matchID))
|
|
||||||
except exceptions.matchCreateError:
|
except exceptions.matchCreateError:
|
||||||
log.error("Error while creating match!")
|
log.error("Error while creating match!")
|
||||||
|
userToken.enqueue(serverPackets.matchJoinFail())
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
from common import generalUtils
|
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from constants import clientPackets
|
from constants import clientPackets
|
||||||
from constants import exceptions
|
from constants import exceptions
|
||||||
|
@ -18,18 +17,14 @@ def handle(userToken, packetData):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Match exists, get object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Hash password if needed
|
# Hash password if needed
|
||||||
# if password != "":
|
# if password != "":
|
||||||
# password = generalUtils.stringMd5(password)
|
# password = generalUtils.stringMd5(password)
|
||||||
|
|
||||||
# Check password
|
# Check password
|
||||||
# TODO: Admins can enter every match
|
with glob.matches.matches[matchID] as match:
|
||||||
if match.matchPassword != "":
|
if match.matchPassword != "" and match.matchPassword != password:
|
||||||
if match.matchPassword != password:
|
raise exceptions.matchWrongPasswordException()
|
||||||
raise exceptions.matchWrongPasswordException
|
|
||||||
|
|
||||||
# Password is correct, join match
|
# Password is correct, join match
|
||||||
userToken.joinMatch(matchID)
|
userToken.joinMatch(matchID)
|
||||||
|
|
|
@ -15,6 +15,7 @@ from objects import glob
|
||||||
|
|
||||||
def handle(tornadoRequest):
|
def handle(tornadoRequest):
|
||||||
# Data to return
|
# Data to return
|
||||||
|
responseToken = None
|
||||||
responseTokenString = "ayy"
|
responseTokenString = "ayy"
|
||||||
responseData = bytes()
|
responseData = bytes()
|
||||||
|
|
||||||
|
@ -29,9 +30,6 @@ def handle(tornadoRequest):
|
||||||
# 2:-3 thing is because requestData has some escape stuff that we don't need
|
# 2:-3 thing is because requestData has some escape stuff that we don't need
|
||||||
loginData = str(tornadoRequest.request.body)[2:-3].split("\\n")
|
loginData = str(tornadoRequest.request.body)[2:-3].split("\\n")
|
||||||
try:
|
try:
|
||||||
# If true, print error to console
|
|
||||||
err = False
|
|
||||||
|
|
||||||
# Make sure loginData is valid
|
# Make sure loginData is valid
|
||||||
if len(loginData) < 3:
|
if len(loginData) < 3:
|
||||||
raise exceptions.invalidArgumentsException()
|
raise exceptions.invalidArgumentsException()
|
||||||
|
@ -46,7 +44,6 @@ def handle(tornadoRequest):
|
||||||
splitData = loginData[2].split("|")
|
splitData = loginData[2].split("|")
|
||||||
osuVersion = splitData[0]
|
osuVersion = splitData[0]
|
||||||
timeOffset = int(splitData[1])
|
timeOffset = int(splitData[1])
|
||||||
print(str(timeOffset))
|
|
||||||
clientData = splitData[3].split(":")[:5]
|
clientData = splitData[3].split(":")[:5]
|
||||||
if len(clientData) < 4:
|
if len(clientData) < 4:
|
||||||
raise exceptions.forceUpdateException()
|
raise exceptions.forceUpdateException()
|
||||||
|
@ -64,9 +61,9 @@ def handle(tornadoRequest):
|
||||||
|
|
||||||
# Make sure we are not banned or locked
|
# Make sure we are not banned or locked
|
||||||
priv = userUtils.getPrivileges(userID)
|
priv = userUtils.getPrivileges(userID)
|
||||||
if userUtils.isBanned(userID) == True and priv & privileges.USER_PENDING_VERIFICATION == 0:
|
if userUtils.isBanned(userID) and priv & privileges.USER_PENDING_VERIFICATION == 0:
|
||||||
raise exceptions.loginBannedException()
|
raise exceptions.loginBannedException()
|
||||||
if userUtils.isLocked(userID) == True and priv & privileges.USER_PENDING_VERIFICATION == 0:
|
if userUtils.isLocked(userID) and priv & privileges.USER_PENDING_VERIFICATION == 0:
|
||||||
raise exceptions.loginLockedException()
|
raise exceptions.loginLockedException()
|
||||||
|
|
||||||
# 2FA check
|
# 2FA check
|
||||||
|
@ -78,7 +75,7 @@ def handle(tornadoRequest):
|
||||||
|
|
||||||
# Verify this user (if pending activation)
|
# Verify this user (if pending activation)
|
||||||
firstLogin = False
|
firstLogin = False
|
||||||
if priv & privileges.USER_PENDING_VERIFICATION > 0 or userUtils.hasVerifiedHardware(userID) == False:
|
if priv & privileges.USER_PENDING_VERIFICATION > 0 or not userUtils.hasVerifiedHardware(userID):
|
||||||
if userUtils.verifyUser(userID, clientData):
|
if userUtils.verifyUser(userID, clientData):
|
||||||
# Valid account
|
# Valid account
|
||||||
log.info("Account {} verified successfully!".format(userID))
|
log.info("Account {} verified successfully!".format(userID))
|
||||||
|
@ -121,6 +118,9 @@ def handle(tornadoRequest):
|
||||||
expireIn = "{} days".format(expireDays) if expireDays > 1 else "less than 24 hours"
|
expireIn = "{} days".format(expireDays) if expireDays > 1 else "less than 24 hours"
|
||||||
responseToken.enqueue(serverPackets.notification("Your donor tag expires in {}! When your donor tag expires, you won't have any of the donor privileges, like yellow username, custom badge and discord custom role and username color! If you wish to keep supporting Ripple and you don't want to lose your donor privileges, you can donate again by clicking on 'Support us' on Ripple's website.".format(expireIn)))
|
responseToken.enqueue(serverPackets.notification("Your donor tag expires in {}! When your donor tag expires, you won't have any of the donor privileges, like yellow username, custom badge and discord custom role and username color! If you wish to keep supporting Ripple and you don't want to lose your donor privileges, you can donate again by clicking on 'Support us' on Ripple's website.".format(expireIn)))
|
||||||
|
|
||||||
|
# Deprecate telegram 2fa and send alert
|
||||||
|
if userUtils.deprecateTelegram2Fa(userID):
|
||||||
|
responseToken.enqueue(serverPackets.notification("As stated on our blog, Telegram 2FA has been deprecated on 29th June 2018. Telegram 2FA has just been disabled from your account. If you want to keep your account secure with 2FA, please enable TOTP-based 2FA from our website https://ripple.moe. Thank you for your patience."))
|
||||||
|
|
||||||
# Set silence end UNIX time in token
|
# Set silence end UNIX time in token
|
||||||
responseToken.silenceEndTime = userUtils.getSilenceEnd(userID)
|
responseToken.silenceEndTime = userUtils.getSilenceEnd(userID)
|
||||||
|
@ -176,7 +176,7 @@ def handle(tornadoRequest):
|
||||||
|
|
||||||
# Output channels info
|
# Output channels info
|
||||||
for key, value in glob.channels.channels.items():
|
for key, value in glob.channels.channels.items():
|
||||||
if value.publicRead == True and value.hidden == False:
|
if value.publicRead and not value.hidden:
|
||||||
responseToken.enqueue(serverPackets.channelInfo(key))
|
responseToken.enqueue(serverPackets.channelInfo(key))
|
||||||
|
|
||||||
# Send friends list
|
# Send friends list
|
||||||
|
@ -186,25 +186,29 @@ def handle(tornadoRequest):
|
||||||
if glob.banchoConf.config["menuIcon"] != "":
|
if glob.banchoConf.config["menuIcon"] != "":
|
||||||
responseToken.enqueue(serverPackets.mainMenuIcon(glob.banchoConf.config["menuIcon"]))
|
responseToken.enqueue(serverPackets.mainMenuIcon(glob.banchoConf.config["menuIcon"]))
|
||||||
|
|
||||||
# Send online users IDs array
|
# Send online users' panels
|
||||||
responseToken.enqueue(serverPackets.onlineUsers())
|
with glob.tokens:
|
||||||
|
for _, token in glob.tokens.tokens.items():
|
||||||
|
if not token.restricted:
|
||||||
|
responseToken.enqueue(serverPackets.userPanel(token.userID))
|
||||||
|
|
||||||
# Get location and country from ip.zxq.co or database
|
# Get location and country from ip.zxq.co or database
|
||||||
if glob.localize:
|
if glob.localize:
|
||||||
# Get location and country from IP
|
# Get location and country from IP
|
||||||
location = locationHelper.getLocation(requestIP)
|
latitude, longitude = locationHelper.getLocation(requestIP)
|
||||||
countryLetters = locationHelper.getCountry(requestIP)
|
countryLetters = locationHelper.getCountry(requestIP)
|
||||||
country = countryHelper.getCountryID(countryLetters)
|
country = countryHelper.getCountryID(countryLetters)
|
||||||
else:
|
else:
|
||||||
# Set location to 0,0 and get country from db
|
# Set location to 0,0 and get country from db
|
||||||
log.warning("Location skipped")
|
log.warning("Location skipped")
|
||||||
location = [0,0]
|
latitude = 0
|
||||||
|
longitude = 0
|
||||||
countryLetters = "XX"
|
countryLetters = "XX"
|
||||||
country = countryHelper.getCountryID(userUtils.getCountry(userID))
|
country = countryHelper.getCountryID(userUtils.getCountry(userID))
|
||||||
|
|
||||||
# Set location and country
|
# Set location and country
|
||||||
responseToken.setLocation(location)
|
responseToken.setLocation(latitude, longitude)
|
||||||
responseToken.setCountry(country)
|
responseToken.country = country
|
||||||
|
|
||||||
# Set country in db if user has no country (first bancho login)
|
# Set country in db if user has no country (first bancho login)
|
||||||
if userUtils.getCountry(userID) == "XX":
|
if userUtils.getCountry(userID) == "XX":
|
||||||
|
@ -220,24 +224,22 @@ def handle(tornadoRequest):
|
||||||
except exceptions.loginFailedException:
|
except exceptions.loginFailedException:
|
||||||
# Login failed error packet
|
# Login failed error packet
|
||||||
# (we don't use enqueue because we don't have a token since login has failed)
|
# (we don't use enqueue because we don't have a token since login has failed)
|
||||||
err = True
|
|
||||||
responseData += serverPackets.loginFailed()
|
responseData += serverPackets.loginFailed()
|
||||||
except exceptions.invalidArgumentsException:
|
except exceptions.invalidArgumentsException:
|
||||||
# Invalid POST data
|
# Invalid POST data
|
||||||
# (we don't use enqueue because we don't have a token since login has failed)
|
# (we don't use enqueue because we don't have a token since login has failed)
|
||||||
err = True
|
|
||||||
responseData += serverPackets.loginFailed()
|
responseData += serverPackets.loginFailed()
|
||||||
responseData += serverPackets.notification("I see what you're doing...")
|
responseData += serverPackets.notification("I see what you're doing...")
|
||||||
except exceptions.loginBannedException:
|
except exceptions.loginBannedException:
|
||||||
# Login banned error packet
|
# Login banned error packet
|
||||||
err = True
|
|
||||||
responseData += serverPackets.loginBanned()
|
responseData += serverPackets.loginBanned()
|
||||||
except exceptions.loginLockedException:
|
except exceptions.loginLockedException:
|
||||||
# Login banned error packet
|
# Login banned error packet
|
||||||
err = True
|
|
||||||
responseData += serverPackets.loginLocked()
|
responseData += serverPackets.loginLocked()
|
||||||
except exceptions.banchoMaintenanceException:
|
except exceptions.banchoMaintenanceException:
|
||||||
# Bancho is in maintenance mode
|
# Bancho is in maintenance mode
|
||||||
|
responseData = bytes()
|
||||||
|
if responseToken is not None:
|
||||||
responseData = responseToken.queue
|
responseData = responseToken.queue
|
||||||
responseData += serverPackets.notification("Our bancho server is in maintenance mode. Please try to login again later.")
|
responseData += serverPackets.notification("Our bancho server is in maintenance mode. Please try to login again later.")
|
||||||
responseData += serverPackets.loginFailed()
|
responseData += serverPackets.loginFailed()
|
||||||
|
@ -251,7 +253,6 @@ def handle(tornadoRequest):
|
||||||
except exceptions.haxException:
|
except exceptions.haxException:
|
||||||
# Using oldoldold client, we don't have client data. Force update.
|
# Using oldoldold client, we don't have client data. Force update.
|
||||||
# (we don't use enqueue because we don't have a token since login has failed)
|
# (we don't use enqueue because we don't have a token since login has failed)
|
||||||
err = True
|
|
||||||
responseData += serverPackets.forceUpdate()
|
responseData += serverPackets.forceUpdate()
|
||||||
responseData += serverPackets.notification("Hory shitto, your client is TOO old! Nice prehistory! Please turn update it from the settings!")
|
responseData += serverPackets.notification("Hory shitto, your client is TOO old! Nice prehistory! Please turn update it from the settings!")
|
||||||
except:
|
except:
|
||||||
|
@ -259,10 +260,7 @@ def handle(tornadoRequest):
|
||||||
finally:
|
finally:
|
||||||
# Console and discord log
|
# Console and discord log
|
||||||
if len(loginData) < 3:
|
if len(loginData) < 3:
|
||||||
msg = "Invalid bancho login request from **{}** (insufficient POST data)".format(requestIP)
|
log.info("Invalid bancho login request from **{}** (insufficient POST data)".format(requestIP), "bunker")
|
||||||
else:
|
|
||||||
msg = "Bancho login request from **{}** for user **{}** ({})".format(requestIP, loginData[0], "failed" if err == True else "success")
|
|
||||||
log.info(msg, "bunker")
|
|
||||||
|
|
||||||
# Return token string and data
|
# Return token string and data
|
||||||
return responseTokenString, responseData
|
return responseTokenString, responseData
|
|
@ -1,4 +1,5 @@
|
||||||
import time
|
import time
|
||||||
|
import json
|
||||||
|
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from constants import serverPackets
|
from constants import serverPackets
|
||||||
|
@ -6,7 +7,7 @@ from helpers import chatHelper as chat
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
|
|
||||||
def handle(userToken, _=None):
|
def handle(userToken, _=None, deleteToken=True):
|
||||||
# get usertoken data
|
# get usertoken data
|
||||||
userID = userToken.userID
|
userID = userToken.userID
|
||||||
username = userToken.username
|
username = userToken.username
|
||||||
|
@ -16,7 +17,7 @@ def handle(userToken, _=None):
|
||||||
# the old logout packet will still be in the queue and will be sent to
|
# the old logout packet will still be in the queue and will be sent to
|
||||||
# the server, so we accept logout packets sent at least 5 seconds after login
|
# the server, so we accept logout packets sent at least 5 seconds after login
|
||||||
# if the user logs out before 5 seconds, he will be disconnected later with timeout check
|
# if the user logs out before 5 seconds, he will be disconnected later with timeout check
|
||||||
if (int(time.time()-userToken.loginTime) >= 5 or userToken.irc):
|
if int(time.time() - userToken.loginTime) >= 5 or userToken.irc:
|
||||||
# Stop spectating
|
# Stop spectating
|
||||||
userToken.stopSpectating()
|
userToken.stopSpectating()
|
||||||
|
|
||||||
|
@ -34,11 +35,23 @@ def handle(userToken, _=None):
|
||||||
glob.streams.broadcast("main", serverPackets.userLogout(userID))
|
glob.streams.broadcast("main", serverPackets.userLogout(userID))
|
||||||
|
|
||||||
# Disconnect from IRC if needed
|
# Disconnect from IRC if needed
|
||||||
if userToken.irc == True and glob.irc == True:
|
if userToken.irc and glob.irc:
|
||||||
glob.ircServer.forceDisconnection(userToken.username)
|
glob.ircServer.forceDisconnection(userToken.username)
|
||||||
|
|
||||||
# Delete token
|
# Delete token
|
||||||
|
if deleteToken:
|
||||||
glob.tokens.deleteToken(requestToken)
|
glob.tokens.deleteToken(requestToken)
|
||||||
|
else:
|
||||||
|
userToken.kicked = True
|
||||||
|
|
||||||
|
# Change username if needed
|
||||||
|
newUsername = glob.redis.get("ripple:change_username_pending:{}".format(userID))
|
||||||
|
if newUsername is not None:
|
||||||
|
log.debug("Sending username change request for user {}".format(userID))
|
||||||
|
glob.redis.publish("peppy:change_username", json.dumps({
|
||||||
|
"userID": userID,
|
||||||
|
"newUsername": newUsername.decode("utf-8")
|
||||||
|
}))
|
||||||
|
|
||||||
# Console output
|
# Console output
|
||||||
log.info("{} has been disconnected. (logout)".format(username))
|
log.info("{} has been disconnected. (logout)".format(username))
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
def handle(userToken, packetData, has):
|
def handle(userToken, _, has):
|
||||||
# Get usertoken data
|
# Get usertoken data
|
||||||
userID = userToken.userID
|
userID = userToken.userID
|
||||||
|
|
||||||
|
@ -15,8 +15,6 @@ def handle(userToken, packetData, has):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# The match exists, get object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Set has beatmap/no beatmap
|
# Set has beatmap/no beatmap
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
match.userHasBeatmap(userID, has)
|
match.userHasBeatmap(userID, has)
|
||||||
|
|
|
@ -15,8 +15,6 @@ def handle(userToken, _):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Get match object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Change team
|
# Change team
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
match.changeTeam(userID)
|
match.changeTeam(userID)
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
def handle(userToken, packetData):
|
def handle(userToken, _):
|
||||||
# Get usertoken data
|
# Get usertoken data
|
||||||
userID = userToken.userID
|
userID = userToken.userID
|
||||||
|
|
||||||
|
@ -15,8 +15,6 @@ def handle(userToken, packetData):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# The match exists, get object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Set our match complete
|
# Set our match complete
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
match.playerCompleted(userID)
|
match.playerCompleted(userID)
|
||||||
|
|
|
@ -15,8 +15,6 @@ def handle(userToken, _):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Match exists, get object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Fail user
|
# Fail user
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
match.playerFailed(userID)
|
match.playerFailed(userID)
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
from objects import glob
|
from objects import glob
|
||||||
from constants import slotStatuses
|
from constants import serverPackets, clientPackets
|
||||||
from constants import serverPackets
|
|
||||||
|
|
||||||
def handle(userToken, packetData):
|
def handle(userToken, packetData):
|
||||||
# Get usertoken data
|
# Get usertoken data
|
||||||
|
@ -17,11 +16,16 @@ def handle(userToken, packetData):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# The match exists, get object
|
# Parse the data
|
||||||
match = glob.matches.matches[matchID]
|
data = clientPackets.matchFrames(packetData)
|
||||||
|
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
# Change slot id in packetData
|
# Change slot id in packetData
|
||||||
slotID = match.getUserSlotID(userID)
|
slotID = match.getUserSlotID(userID)
|
||||||
|
|
||||||
|
# Update the score
|
||||||
|
match.updateScore(slotID, data["totalScore"])
|
||||||
|
match.updateHP(slotID, data["currentHp"])
|
||||||
|
|
||||||
# Enqueue frames to who's playing
|
# Enqueue frames to who's playing
|
||||||
glob.streams.broadcast(match.playingStreamName, serverPackets.matchFrames(slotID, packetData))
|
glob.streams.broadcast(match.playingStreamName, serverPackets.matchFrames(slotID, packetData))
|
|
@ -17,8 +17,6 @@ def handle(userToken, packetData):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Get match object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Send invite
|
# Send invite
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
match.invite(userID, packetData["userID"])
|
match.invite(userID, packetData["userID"])
|
||||||
|
|
|
@ -12,8 +12,8 @@ def handle(userToken, packetData):
|
||||||
matchID = userToken.matchID
|
matchID = userToken.matchID
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
# Host check
|
# Host check
|
||||||
if userID != match.hostUserID:
|
if userID != match.hostUserID:
|
||||||
return
|
return
|
||||||
|
@ -24,4 +24,4 @@ def handle(userToken, packetData):
|
||||||
return
|
return
|
||||||
|
|
||||||
# Lock/Unlock slot
|
# Lock/Unlock slot
|
||||||
match.toggleSlotLock(packetData["slotID"])
|
match.toggleSlotLocked(packetData["slotID"])
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
def handle(userToken, packetData):
|
def handle(userToken, _):
|
||||||
# Get userToken data
|
# Get userToken data
|
||||||
userID = userToken.userID
|
userID = userToken.userID
|
||||||
|
|
||||||
|
@ -15,8 +15,6 @@ def handle(userToken, packetData):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# The match exists, get object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Set our load status
|
# Set our load status
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
match.playerLoaded(userID)
|
match.playerLoaded(userID)
|
||||||
|
|
|
@ -8,9 +8,14 @@ def handle(userToken, _):
|
||||||
matchID = userToken.matchID
|
matchID = userToken.matchID
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
# Get our slotID and change ready status
|
# Get our slotID and change ready status
|
||||||
slotID = match.getUserSlotID(userID)
|
slotID = match.getUserSlotID(userID)
|
||||||
if slotID is not None:
|
if slotID is not None:
|
||||||
match.toggleSlotReady(slotID)
|
match.toggleSlotReady(slotID)
|
||||||
|
|
||||||
|
# If this is a tournament match, we should send the current status of ready
|
||||||
|
# players.
|
||||||
|
if match.isTourney:
|
||||||
|
match.sendReadyStatus()
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
def handle(userToken, packetData):
|
def handle(userToken, _):
|
||||||
# Get userToken data
|
# Get userToken data
|
||||||
userID = userToken.userID
|
userID = userToken.userID
|
||||||
|
|
||||||
|
@ -15,8 +15,6 @@ def handle(userToken, packetData):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# The match exists, get object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Skip
|
# Skip
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
match.playerSkip(userID)
|
match.playerSkip(userID)
|
||||||
|
|
|
@ -1,6 +1,4 @@
|
||||||
from objects import glob
|
from objects import glob
|
||||||
from constants import slotStatuses
|
|
||||||
from constants import serverPackets
|
|
||||||
|
|
||||||
def handle(userToken, _):
|
def handle(userToken, _):
|
||||||
|
|
||||||
|
@ -15,9 +13,7 @@ def handle(userToken, _):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# The match exists, get object
|
with glob.matches.matches[matchID] as match:
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Host check
|
# Host check
|
||||||
if userToken.userID != match.hostUserID:
|
if userToken.userID != match.hostUserID:
|
||||||
return
|
return
|
||||||
|
|
|
@ -16,10 +16,8 @@ def handle(userToken, packetData):
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Match exists, get object
|
|
||||||
match = glob.matches.matches[matchID]
|
|
||||||
|
|
||||||
# Host check
|
# Host check
|
||||||
|
with glob.matches.matches[matchID] as match:
|
||||||
if userToken.userID != match.hostUserID:
|
if userToken.userID != match.hostUserID:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
|
|
@ -1,17 +1,15 @@
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from helpers import chatHelper as chat
|
from helpers import chatHelper as chat
|
||||||
from objects import glob
|
|
||||||
|
|
||||||
|
|
||||||
def handle(userToken, _):
|
def handle(userToken, _):
|
||||||
# Get usertoken data
|
# Get usertoken data
|
||||||
userID = userToken.userID
|
|
||||||
username = userToken.username
|
username = userToken.username
|
||||||
|
|
||||||
# Remove user from users in lobby
|
# Remove user from users in lobby
|
||||||
userToken.leaveStream("lobby")
|
userToken.leaveStream("lobby")
|
||||||
|
|
||||||
# Part lobby channel
|
# Part lobby channel
|
||||||
|
# Done automatically by the client
|
||||||
chat.partChannel(channel="#lobby", token=userToken, kick=True)
|
chat.partChannel(channel="#lobby", token=userToken, kick=True)
|
||||||
|
|
||||||
# Console output
|
# Console output
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from constants import clientPackets
|
from constants import clientPackets
|
||||||
from constants import serverPackets
|
from constants import serverPackets
|
||||||
|
from objects import glob
|
||||||
|
|
||||||
|
|
||||||
def handle(userToken, packetData):
|
def handle(userToken, packetData):
|
||||||
|
@ -11,12 +12,12 @@ def handle(userToken, packetData):
|
||||||
packetData = clientPackets.setAwayMessage(packetData)
|
packetData = clientPackets.setAwayMessage(packetData)
|
||||||
|
|
||||||
# Set token away message
|
# Set token away message
|
||||||
userToken.setAwayMessage(packetData["awayMessage"])
|
userToken.awayMessage = packetData["awayMessage"]
|
||||||
|
|
||||||
# Send private message from fokabot
|
# Send private message from fokabot
|
||||||
if packetData["awayMessage"] == "":
|
if packetData["awayMessage"] == "":
|
||||||
fokaMessage = "Your away message has been reset"
|
fokaMessage = "Your away message has been reset"
|
||||||
else:
|
else:
|
||||||
fokaMessage = "Your away message is now: {}".format(packetData["awayMessage"])
|
fokaMessage = "Your away message is now: {}".format(packetData["awayMessage"])
|
||||||
userToken.enqueue(serverPackets.sendMessage("FokaBot", username, fokaMessage))
|
userToken.enqueue(serverPackets.sendMessage(glob.BOT_NAME, username, fokaMessage))
|
||||||
log.info("{} has changed their away message to: {}".format(username, packetData["awayMessage"]))
|
log.info("{} has changed their away message to: {}".format(username, packetData["awayMessage"]))
|
||||||
|
|
|
@ -1,31 +1,15 @@
|
||||||
from objects import glob
|
from objects import glob
|
||||||
from constants import serverPackets
|
from constants import serverPackets
|
||||||
from constants import exceptions
|
from common.log import logUtils as log
|
||||||
|
|
||||||
def handle(userToken, packetData):
|
def handle(userToken, packetData):
|
||||||
# get token data
|
# get token data
|
||||||
userID = userToken.userID
|
userID = userToken.userID
|
||||||
|
|
||||||
# Send spectator frames to every spectator
|
# Send spectator frames to every spectator
|
||||||
glob.streams.broadcast("spect/{}".format(userID), serverPackets.spectatorFrames(packetData[7:]))
|
streamName = "spect/{}".format(userID)
|
||||||
'''for i in userToken.spectators:
|
glob.streams.broadcast(streamName, serverPackets.spectatorFrames(packetData[7:]))
|
||||||
# Send to every user but host
|
log.debug("Broadcasting {}'s frames to {} clients".format(
|
||||||
if i != userID:
|
userID,
|
||||||
try:
|
len(glob.streams.streams[streamName].clients))
|
||||||
# Get spectator token object
|
)
|
||||||
spectatorToken = glob.tokens.getTokenFromUserID(i)
|
|
||||||
|
|
||||||
# Make sure the token exists
|
|
||||||
if spectatorToken is None:
|
|
||||||
raise exceptions.stopSpectating
|
|
||||||
|
|
||||||
# Make sure this user is spectating us
|
|
||||||
if spectatorToken.spectating != userID:
|
|
||||||
raise exceptions.stopSpectating
|
|
||||||
|
|
||||||
# Everything seems fine, send spectator frames to this spectator
|
|
||||||
spectatorToken.enqueue(serverPackets.spectatorFrames(packetData[7:]))
|
|
||||||
except exceptions.stopSpectating:
|
|
||||||
# Remove this user from spectators
|
|
||||||
userToken.removeSpectator(i)
|
|
||||||
userToken.enqueue(serverPackets.removeSpectator(i))'''
|
|
|
@ -8,6 +8,11 @@ def handle(userToken, packetData):
|
||||||
# Start spectating packet
|
# Start spectating packet
|
||||||
packetData = clientPackets.startSpectating(packetData)
|
packetData = clientPackets.startSpectating(packetData)
|
||||||
|
|
||||||
|
# If the user id is less than 0, treat this as a stop spectating packet
|
||||||
|
if packetData["userID"] < 0:
|
||||||
|
userToken.stopSpectating()
|
||||||
|
return
|
||||||
|
|
||||||
# Get host token
|
# Get host token
|
||||||
targetToken = glob.tokens.getTokenFromUserID(packetData["userID"])
|
targetToken = glob.tokens.getTokenFromUserID(packetData["userID"])
|
||||||
if targetToken is None:
|
if targetToken is None:
|
||||||
|
|
|
@ -5,7 +5,7 @@ from helpers import chatHelper as chat
|
||||||
def handle(userToken, packetData):
|
def handle(userToken, packetData):
|
||||||
packetData = clientPackets.tournamentJoinMatchChannel(packetData)
|
packetData = clientPackets.tournamentJoinMatchChannel(packetData)
|
||||||
matchID = packetData["matchID"]
|
matchID = packetData["matchID"]
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches or not userToken.tournament:
|
||||||
return
|
return
|
||||||
userToken.matchID = matchID
|
userToken.matchID = matchID
|
||||||
chat.joinChannel(token=userToken, channel="#multi_{}".format(matchID))
|
chat.joinChannel(token=userToken, channel="#multi_{}".format(matchID), force=True)
|
|
@ -5,7 +5,7 @@ from helpers import chatHelper as chat
|
||||||
def handle(userToken, packetData):
|
def handle(userToken, packetData):
|
||||||
packetData = clientPackets.tournamentLeaveMatchChannel(packetData)
|
packetData = clientPackets.tournamentLeaveMatchChannel(packetData)
|
||||||
matchID = packetData["matchID"]
|
matchID = packetData["matchID"]
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches or not userToken.tournament:
|
||||||
return
|
return
|
||||||
chat.partChannel(token=userToken, channel="#multi_{}".format(matchID))
|
chat.partChannel(token=userToken, channel="#multi_{}".format(matchID), force=True)
|
||||||
userToken.matchID = 0
|
userToken.matchID = 0
|
|
@ -1,10 +1,10 @@
|
||||||
from constants import clientPackets
|
from constants import clientPackets
|
||||||
from constants import serverPackets
|
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
def handle(userToken, packetData):
|
def handle(userToken, packetData):
|
||||||
packetData = clientPackets.tournamentMatchInfoRequest(packetData)
|
packetData = clientPackets.tournamentMatchInfoRequest(packetData)
|
||||||
matchID = packetData["matchID"]
|
matchID = packetData["matchID"]
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches or not userToken.tournament:
|
||||||
return
|
return
|
||||||
userToken.enqueue(glob.matches.matches[matchID].matchDataCache)
|
with glob.matches.matches[matchID] as m:
|
||||||
|
userToken.enqueue(m.matchDataCache)
|
|
@ -14,3 +14,10 @@ boobs=bob
|
||||||
tits=teeth
|
tits=teeth
|
||||||
cum=yogurt
|
cum=yogurt
|
||||||
cunt=count
|
cunt=count
|
||||||
|
nigger=flowers
|
||||||
|
ngger=flowers
|
||||||
|
niggers=flowers
|
||||||
|
weed=grass
|
||||||
|
AQN=meme
|
||||||
|
theaquila=meme
|
||||||
|
aquila=meme
|
4
full_build.sh
Normal file
4
full_build.sh
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
find . -name "*.c" -type f -delete
|
||||||
|
find . -name "*.o" -type f -delete
|
||||||
|
find . -name "*.so" -type f -delete
|
||||||
|
python3 setup.py build_ext --inplace
|
|
@ -1,6 +1,9 @@
|
||||||
import json
|
import json
|
||||||
|
|
||||||
from common.log import logUtils as log
|
import tornado.web
|
||||||
|
import tornado.gen
|
||||||
|
|
||||||
|
from common.sentry import sentry
|
||||||
from common.web import requestsManager
|
from common.web import requestsManager
|
||||||
from constants import exceptions
|
from constants import exceptions
|
||||||
from helpers import chatHelper
|
from helpers import chatHelper
|
||||||
|
@ -8,6 +11,9 @@ from objects import glob
|
||||||
|
|
||||||
|
|
||||||
class handler(requestsManager.asyncRequestHandler):
|
class handler(requestsManager.asyncRequestHandler):
|
||||||
|
@tornado.web.asynchronous
|
||||||
|
@tornado.gen.engine
|
||||||
|
@sentry.captureTornado
|
||||||
def asyncGet(self):
|
def asyncGet(self):
|
||||||
statusCode = 400
|
statusCode = 400
|
||||||
data = {"message": "unknown error"}
|
data = {"message": "unknown error"}
|
||||||
|
@ -21,8 +27,11 @@ class handler(requestsManager.asyncRequestHandler):
|
||||||
if key is None or key != glob.conf.config["server"]["cikey"]:
|
if key is None or key != glob.conf.config["server"]["cikey"]:
|
||||||
raise exceptions.invalidArgumentsException()
|
raise exceptions.invalidArgumentsException()
|
||||||
|
|
||||||
log.info("API REQUEST FOR FOKABOT MESSAGE AAAAAAA")
|
chatHelper.sendMessage(
|
||||||
chatHelper.sendMessage("FokaBot", self.get_argument("to"), self.get_argument("msg"))
|
glob.BOT_NAME,
|
||||||
|
self.get_argument("to").encode().decode("ASCII", "ignore"),
|
||||||
|
self.get_argument("msg").encode().decode("ASCII", "ignore")
|
||||||
|
)
|
||||||
|
|
||||||
# Status code and message
|
# Status code and message
|
||||||
statusCode = 200
|
statusCode = 200
|
||||||
|
@ -35,7 +44,5 @@ class handler(requestsManager.asyncRequestHandler):
|
||||||
data["status"] = statusCode
|
data["status"] = statusCode
|
||||||
|
|
||||||
# Send response
|
# Send response
|
||||||
#self.clear()
|
|
||||||
self.write(json.dumps(data))
|
self.write(json.dumps(data))
|
||||||
self.set_status(statusCode)
|
self.set_status(statusCode)
|
||||||
#self.finish(json.dumps(data))
|
|
||||||
|
|
|
@ -1,11 +1,19 @@
|
||||||
import json
|
import json
|
||||||
|
|
||||||
|
import tornado.web
|
||||||
|
import tornado.gen
|
||||||
|
|
||||||
|
from common.sentry import sentry
|
||||||
|
from common.ripple import userUtils
|
||||||
from common.web import requestsManager
|
from common.web import requestsManager
|
||||||
from constants import exceptions
|
from constants import exceptions
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
|
|
||||||
class handler(requestsManager.asyncRequestHandler):
|
class handler(requestsManager.asyncRequestHandler):
|
||||||
|
@tornado.web.asynchronous
|
||||||
|
@tornado.gen.engine
|
||||||
|
@sentry.captureTornado
|
||||||
def asyncGet(self):
|
def asyncGet(self):
|
||||||
statusCode = 400
|
statusCode = 400
|
||||||
data = {"message": "unknown error"}
|
data = {"message": "unknown error"}
|
||||||
|
@ -18,7 +26,8 @@ class handler(requestsManager.asyncRequestHandler):
|
||||||
username = None
|
username = None
|
||||||
userID = None
|
userID = None
|
||||||
if "u" in self.request.arguments:
|
if "u" in self.request.arguments:
|
||||||
username = self.get_argument("u")
|
#username = self.get_argument("u").lower().replace(" ", "_")
|
||||||
|
username = userUtils.safeUsername(self.get_argument("u"))
|
||||||
else:
|
else:
|
||||||
try:
|
try:
|
||||||
userID = int(self.get_argument("id"))
|
userID = int(self.get_argument("id"))
|
||||||
|
@ -29,7 +38,7 @@ class handler(requestsManager.asyncRequestHandler):
|
||||||
data["result"] = False
|
data["result"] = False
|
||||||
else:
|
else:
|
||||||
if username is not None:
|
if username is not None:
|
||||||
data["result"] = True if glob.tokens.getTokenFromUsername(username) is not None else False
|
data["result"] = True if glob.tokens.getTokenFromUsername(username, safe=True) is not None else False
|
||||||
else:
|
else:
|
||||||
data["result"] = True if glob.tokens.getTokenFromUserID(userID) is not None else False
|
data["result"] = True if glob.tokens.getTokenFromUserID(userID) is not None else False
|
||||||
|
|
||||||
|
@ -44,7 +53,5 @@ class handler(requestsManager.asyncRequestHandler):
|
||||||
data["status"] = statusCode
|
data["status"] = statusCode
|
||||||
|
|
||||||
# Send response
|
# Send response
|
||||||
#self.clear()
|
|
||||||
self.write(json.dumps(data))
|
self.write(json.dumps(data))
|
||||||
self.set_status(statusCode)
|
self.set_status(statusCode)
|
||||||
#self.finish(json.dumps(data))
|
|
||||||
|
|
|
@ -1,16 +1,23 @@
|
||||||
import json
|
import json
|
||||||
|
|
||||||
|
import tornado.web
|
||||||
|
import tornado.gen
|
||||||
|
|
||||||
|
from common.sentry import sentry
|
||||||
from common.web import requestsManager
|
from common.web import requestsManager
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
|
|
||||||
class handler(requestsManager.asyncRequestHandler):
|
class handler(requestsManager.asyncRequestHandler):
|
||||||
|
@tornado.web.asynchronous
|
||||||
|
@tornado.gen.engine
|
||||||
|
@sentry.captureTornado
|
||||||
def asyncGet(self):
|
def asyncGet(self):
|
||||||
statusCode = 400
|
statusCode = 400
|
||||||
data = {"message": "unknown error"}
|
data = {"message": "unknown error"}
|
||||||
try:
|
try:
|
||||||
# Get online users count
|
# Get online users count
|
||||||
data["result"] = len(glob.tokens.tokens)
|
data["result"] = int(glob.redis.get("ripple:online_users").decode("utf-8"))
|
||||||
|
|
||||||
# Status code and message
|
# Status code and message
|
||||||
statusCode = 200
|
statusCode = 200
|
||||||
|
@ -20,7 +27,5 @@ class handler(requestsManager.asyncRequestHandler):
|
||||||
data["status"] = statusCode
|
data["status"] = statusCode
|
||||||
|
|
||||||
# Send response
|
# Send response
|
||||||
#self.clear()
|
|
||||||
self.write(json.dumps(data))
|
self.write(json.dumps(data))
|
||||||
self.set_status(statusCode)
|
self.set_status(statusCode)
|
||||||
#self.finish(json.dumps(data))
|
|
||||||
|
|
|
@ -1,16 +1,23 @@
|
||||||
import json
|
import json
|
||||||
|
|
||||||
|
import tornado.web
|
||||||
|
import tornado.gen
|
||||||
|
|
||||||
|
from common.sentry import sentry
|
||||||
from common.web import requestsManager
|
from common.web import requestsManager
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
|
|
||||||
class handler(requestsManager.asyncRequestHandler):
|
class handler(requestsManager.asyncRequestHandler):
|
||||||
|
@tornado.web.asynchronous
|
||||||
|
@tornado.gen.engine
|
||||||
|
@sentry.captureTornado
|
||||||
def asyncGet(self):
|
def asyncGet(self):
|
||||||
statusCode = 400
|
statusCode = 400
|
||||||
data = {"message": "unknown error"}
|
data = {"message": "unknown error"}
|
||||||
try:
|
try:
|
||||||
# Get online users count
|
# Get online users count
|
||||||
data["result"] = -1 if glob.restarting == True else 1
|
data["result"] = -1 if glob.restarting else 1
|
||||||
|
|
||||||
# Status code and message
|
# Status code and message
|
||||||
statusCode = 200
|
statusCode = 200
|
||||||
|
@ -20,7 +27,5 @@ class handler(requestsManager.asyncRequestHandler):
|
||||||
data["status"] = statusCode
|
data["status"] = statusCode
|
||||||
|
|
||||||
# Send response
|
# Send response
|
||||||
#self.clear()
|
|
||||||
self.write(json.dumps(data))
|
self.write(json.dumps(data))
|
||||||
self.set_status(statusCode)
|
self.set_status(statusCode)
|
||||||
#self.finish(json.dumps(data))
|
|
||||||
|
|
|
@ -1,11 +1,18 @@
|
||||||
import json
|
import json
|
||||||
|
|
||||||
|
import tornado.web
|
||||||
|
import tornado.gen
|
||||||
|
|
||||||
|
from common.sentry import sentry
|
||||||
from common.web import requestsManager
|
from common.web import requestsManager
|
||||||
from constants import exceptions
|
from constants import exceptions
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
|
|
||||||
class handler(requestsManager.asyncRequestHandler):
|
class handler(requestsManager.asyncRequestHandler):
|
||||||
|
@tornado.web.asynchronous
|
||||||
|
@tornado.gen.engine
|
||||||
|
@sentry.captureTornado
|
||||||
def asyncGet(self):
|
def asyncGet(self):
|
||||||
statusCode = 400
|
statusCode = 400
|
||||||
data = {"message": "unknown error"}
|
data = {"message": "unknown error"}
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
import json
|
import json
|
||||||
|
|
||||||
|
import tornado.web
|
||||||
|
import tornado.gen
|
||||||
|
|
||||||
|
from common.sentry import sentry
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from common.web import requestsManager
|
from common.web import requestsManager
|
||||||
from constants import exceptions
|
from constants import exceptions
|
||||||
|
@ -8,6 +12,9 @@ from objects import glob
|
||||||
|
|
||||||
|
|
||||||
class handler(requestsManager.asyncRequestHandler):
|
class handler(requestsManager.asyncRequestHandler):
|
||||||
|
@tornado.web.asynchronous
|
||||||
|
@tornado.gen.engine
|
||||||
|
@sentry.captureTornado
|
||||||
def asyncGet(self):
|
def asyncGet(self):
|
||||||
statusCode = 400
|
statusCode = 400
|
||||||
data = {"message": "unknown error"}
|
data = {"message": "unknown error"}
|
||||||
|
|
|
@ -1,270 +0,0 @@
|
||||||
import datetime
|
|
||||||
import gzip
|
|
||||||
import sys
|
|
||||||
import traceback
|
|
||||||
|
|
||||||
import tornado.gen
|
|
||||||
import tornado.web
|
|
||||||
from raven.contrib.tornado import SentryMixin
|
|
||||||
|
|
||||||
from common.log import logUtils as log
|
|
||||||
from common.web import requestsManager
|
|
||||||
from constants import exceptions
|
|
||||||
from constants import packetIDs
|
|
||||||
from constants import serverPackets
|
|
||||||
from events import cantSpectateEvent
|
|
||||||
from events import changeActionEvent
|
|
||||||
from events import changeMatchModsEvent
|
|
||||||
from events import changeMatchPasswordEvent
|
|
||||||
from events import changeMatchSettingsEvent
|
|
||||||
from events import changeSlotEvent
|
|
||||||
from events import channelJoinEvent
|
|
||||||
from events import channelPartEvent
|
|
||||||
from events import createMatchEvent
|
|
||||||
from events import friendAddEvent
|
|
||||||
from events import friendRemoveEvent
|
|
||||||
from events import joinLobbyEvent
|
|
||||||
from events import joinMatchEvent
|
|
||||||
from events import loginEvent
|
|
||||||
from events import logoutEvent
|
|
||||||
from events import matchChangeTeamEvent
|
|
||||||
from events import matchCompleteEvent
|
|
||||||
from events import matchFailedEvent
|
|
||||||
from events import matchFramesEvent
|
|
||||||
from events import matchHasBeatmapEvent
|
|
||||||
from events import matchInviteEvent
|
|
||||||
from events import matchLockEvent
|
|
||||||
from events import matchNoBeatmapEvent
|
|
||||||
from events import matchPlayerLoadEvent
|
|
||||||
from events import matchReadyEvent
|
|
||||||
from events import matchSkipEvent
|
|
||||||
from events import matchStartEvent
|
|
||||||
from events import matchTransferHostEvent
|
|
||||||
from events import partLobbyEvent
|
|
||||||
from events import partMatchEvent
|
|
||||||
from events import requestStatusUpdateEvent
|
|
||||||
from events import sendPrivateMessageEvent
|
|
||||||
from events import sendPublicMessageEvent
|
|
||||||
from events import setAwayMessageEvent
|
|
||||||
from events import spectateFramesEvent
|
|
||||||
from events import startSpectatingEvent
|
|
||||||
from events import stopSpectatingEvent
|
|
||||||
from events import userPanelRequestEvent
|
|
||||||
from events import userStatsRequestEvent
|
|
||||||
from events import tournamentMatchInfoRequestEvent
|
|
||||||
from events import tournamentJoinMatchChannelEvent
|
|
||||||
from events import tournamentLeaveMatchChannelEvent
|
|
||||||
from helpers import packetHelper
|
|
||||||
from objects import glob
|
|
||||||
|
|
||||||
|
|
||||||
class handler(SentryMixin, requestsManager.asyncRequestHandler):
|
|
||||||
@tornado.web.asynchronous
|
|
||||||
@tornado.gen.engine
|
|
||||||
def asyncPost(self):
|
|
||||||
try:
|
|
||||||
# Track time if needed
|
|
||||||
if glob.outputRequestTime:
|
|
||||||
# Start time
|
|
||||||
st = datetime.datetime.now()
|
|
||||||
|
|
||||||
# Client's token string and request data
|
|
||||||
requestTokenString = self.request.headers.get("osu-token")
|
|
||||||
requestData = self.request.body
|
|
||||||
|
|
||||||
# Server's token string and request data
|
|
||||||
responseTokenString = "ayy"
|
|
||||||
responseData = bytes()
|
|
||||||
|
|
||||||
if requestTokenString is None:
|
|
||||||
# No token, first request. Handle login.
|
|
||||||
responseTokenString, responseData = loginEvent.handle(self)
|
|
||||||
else:
|
|
||||||
userToken = None # default value
|
|
||||||
try:
|
|
||||||
# This is not the first packet, send response based on client's request
|
|
||||||
# Packet start position, used to read stacked packets
|
|
||||||
pos = 0
|
|
||||||
|
|
||||||
# Make sure the token exists
|
|
||||||
if requestTokenString not in glob.tokens.tokens:
|
|
||||||
raise exceptions.tokenNotFoundException()
|
|
||||||
|
|
||||||
# Token exists, get its object and lock it
|
|
||||||
userToken = glob.tokens.tokens[requestTokenString]
|
|
||||||
userToken.lock.acquire()
|
|
||||||
|
|
||||||
# Keep reading packets until everything has been read
|
|
||||||
while pos < len(requestData):
|
|
||||||
# Get packet from stack starting from new packet
|
|
||||||
leftData = requestData[pos:]
|
|
||||||
|
|
||||||
# Get packet ID, data length and data
|
|
||||||
packetID = packetHelper.readPacketID(leftData)
|
|
||||||
dataLength = packetHelper.readPacketLength(leftData)
|
|
||||||
packetData = requestData[pos:(pos+dataLength+7)]
|
|
||||||
|
|
||||||
# Console output if needed
|
|
||||||
if glob.outputPackets == True and packetID != 4:
|
|
||||||
log.debug("Incoming packet ({})({}):\n\nPacket code: {}\nPacket length: {}\nSingle packet data: {}\n".format(requestTokenString, userToken.username, str(packetID), str(dataLength), str(packetData)))
|
|
||||||
|
|
||||||
# Event handler
|
|
||||||
def handleEvent(ev):
|
|
||||||
def wrapper():
|
|
||||||
ev.handle(userToken, packetData)
|
|
||||||
return wrapper
|
|
||||||
|
|
||||||
eventHandler = {
|
|
||||||
# TODO: Rename packets and events
|
|
||||||
# TODO: Host check for multi
|
|
||||||
packetIDs.client_changeAction: handleEvent(changeActionEvent),
|
|
||||||
packetIDs.client_logout: handleEvent(logoutEvent),
|
|
||||||
packetIDs.client_friendAdd: handleEvent(friendAddEvent),
|
|
||||||
packetIDs.client_friendRemove: handleEvent(friendRemoveEvent),
|
|
||||||
packetIDs.client_userStatsRequest: handleEvent(userStatsRequestEvent),
|
|
||||||
packetIDs.client_requestStatusUpdate: handleEvent(requestStatusUpdateEvent),
|
|
||||||
packetIDs.client_userPanelRequest: handleEvent(userPanelRequestEvent),
|
|
||||||
|
|
||||||
packetIDs.client_channelJoin: handleEvent(channelJoinEvent),
|
|
||||||
packetIDs.client_channelPart: handleEvent(channelPartEvent),
|
|
||||||
packetIDs.client_sendPublicMessage: handleEvent(sendPublicMessageEvent),
|
|
||||||
packetIDs.client_sendPrivateMessage: handleEvent(sendPrivateMessageEvent),
|
|
||||||
packetIDs.client_setAwayMessage: handleEvent(setAwayMessageEvent),
|
|
||||||
|
|
||||||
packetIDs.client_startSpectating: handleEvent(startSpectatingEvent),
|
|
||||||
packetIDs.client_stopSpectating: handleEvent(stopSpectatingEvent),
|
|
||||||
packetIDs.client_cantSpectate: handleEvent(cantSpectateEvent),
|
|
||||||
packetIDs.client_spectateFrames: handleEvent(spectateFramesEvent),
|
|
||||||
|
|
||||||
packetIDs.client_joinLobby: handleEvent(joinLobbyEvent),
|
|
||||||
packetIDs.client_partLobby: handleEvent(partLobbyEvent),
|
|
||||||
packetIDs.client_createMatch: handleEvent(createMatchEvent),
|
|
||||||
packetIDs.client_joinMatch: handleEvent(joinMatchEvent),
|
|
||||||
packetIDs.client_partMatch: handleEvent(partMatchEvent),
|
|
||||||
packetIDs.client_matchChangeSlot: handleEvent(changeSlotEvent),
|
|
||||||
packetIDs.client_matchChangeSettings: handleEvent(changeMatchSettingsEvent),
|
|
||||||
packetIDs.client_matchChangePassword: handleEvent(changeMatchPasswordEvent),
|
|
||||||
packetIDs.client_matchChangeMods: handleEvent(changeMatchModsEvent),
|
|
||||||
packetIDs.client_matchReady: handleEvent(matchReadyEvent),
|
|
||||||
packetIDs.client_matchNotReady: handleEvent(matchReadyEvent),
|
|
||||||
packetIDs.client_matchLock: handleEvent(matchLockEvent),
|
|
||||||
packetIDs.client_matchStart: handleEvent(matchStartEvent),
|
|
||||||
packetIDs.client_matchLoadComplete: handleEvent(matchPlayerLoadEvent),
|
|
||||||
packetIDs.client_matchSkipRequest: handleEvent(matchSkipEvent),
|
|
||||||
packetIDs.client_matchScoreUpdate: handleEvent(matchFramesEvent),
|
|
||||||
packetIDs.client_matchComplete: handleEvent(matchCompleteEvent),
|
|
||||||
packetIDs.client_matchNoBeatmap: handleEvent(matchNoBeatmapEvent),
|
|
||||||
packetIDs.client_matchHasBeatmap: handleEvent(matchHasBeatmapEvent),
|
|
||||||
packetIDs.client_matchTransferHost: handleEvent(matchTransferHostEvent),
|
|
||||||
packetIDs.client_matchFailed: handleEvent(matchFailedEvent),
|
|
||||||
packetIDs.client_matchChangeTeam: handleEvent(matchChangeTeamEvent),
|
|
||||||
packetIDs.client_invite: handleEvent(matchInviteEvent),
|
|
||||||
|
|
||||||
packetIDs.client_tournamentMatchInfoRequest: handleEvent(tournamentMatchInfoRequestEvent),
|
|
||||||
packetIDs.client_tournamentJoinMatchChannel: handleEvent(tournamentJoinMatchChannelEvent),
|
|
||||||
packetIDs.client_tournamentLeaveMatchChannel: handleEvent(tournamentLeaveMatchChannelEvent),
|
|
||||||
}
|
|
||||||
|
|
||||||
# Packets processed if in restricted mode.
|
|
||||||
# All other packets will be ignored if the user is in restricted mode
|
|
||||||
packetsRestricted = [
|
|
||||||
packetIDs.client_logout,
|
|
||||||
packetIDs.client_userStatsRequest,
|
|
||||||
packetIDs.client_requestStatusUpdate,
|
|
||||||
packetIDs.client_userPanelRequest,
|
|
||||||
packetIDs.client_changeAction,
|
|
||||||
packetIDs.client_channelJoin,
|
|
||||||
packetIDs.client_channelPart,
|
|
||||||
]
|
|
||||||
|
|
||||||
# Process/ignore packet
|
|
||||||
if packetID != 4:
|
|
||||||
if packetID in eventHandler:
|
|
||||||
if userToken.restricted == False or (userToken.restricted == True and packetID in packetsRestricted):
|
|
||||||
eventHandler[packetID]()
|
|
||||||
else:
|
|
||||||
log.warning("Ignored packet id from {} ({}) (user is restricted)".format(requestTokenString, packetID))
|
|
||||||
else:
|
|
||||||
log.warning("Unknown packet id from {} ({})".format(requestTokenString, packetID))
|
|
||||||
|
|
||||||
# Update pos so we can read the next stacked packet
|
|
||||||
# +7 because we add packet ID bytes, unused byte and data length bytes
|
|
||||||
pos += dataLength+7
|
|
||||||
|
|
||||||
# Token queue built, send it
|
|
||||||
responseTokenString = userToken.token
|
|
||||||
responseData = userToken.queue
|
|
||||||
userToken.resetQueue()
|
|
||||||
except exceptions.tokenNotFoundException:
|
|
||||||
# Token not found. Disconnect that user
|
|
||||||
responseData = serverPackets.loginError()
|
|
||||||
responseData += serverPackets.notification("Whoops! Something went wrong, please login again.")
|
|
||||||
log.warning("Received packet from unknown token ({}).".format(requestTokenString))
|
|
||||||
log.info("{} has been disconnected (invalid token)".format(requestTokenString))
|
|
||||||
finally:
|
|
||||||
# Unlock token
|
|
||||||
if userToken is not None:
|
|
||||||
# Update ping time for timeout
|
|
||||||
userToken.updatePingTime()
|
|
||||||
# Release token lock
|
|
||||||
userToken.lock.release()
|
|
||||||
|
|
||||||
if glob.outputRequestTime:
|
|
||||||
# End time
|
|
||||||
et = datetime.datetime.now()
|
|
||||||
|
|
||||||
# Total time:
|
|
||||||
tt = float((et.microsecond-st.microsecond)/1000)
|
|
||||||
log.debug("Request time: {}ms".format(tt))
|
|
||||||
|
|
||||||
# Send server's response to client
|
|
||||||
# We don't use token object because we might not have a token (failed login)
|
|
||||||
if glob.gzip:
|
|
||||||
# First, write the gzipped response
|
|
||||||
self.write(gzip.compress(responseData, int(glob.conf.config["server"]["gziplevel"])))
|
|
||||||
|
|
||||||
# Then, add gzip headers
|
|
||||||
self.add_header("Vary", "Accept-Encoding")
|
|
||||||
self.add_header("Content-Encoding", "gzip")
|
|
||||||
else:
|
|
||||||
# First, write the response
|
|
||||||
self.write(responseData)
|
|
||||||
|
|
||||||
# Add all the headers AFTER the response has been written
|
|
||||||
self.set_status(200)
|
|
||||||
self.add_header("cho-token", responseTokenString)
|
|
||||||
self.add_header("cho-protocol", "19")
|
|
||||||
self.add_header("Connection", "keep-alive")
|
|
||||||
self.add_header("Keep-Alive", "timeout=5, max=100")
|
|
||||||
self.add_header("Content-Type", "text/html; charset=UTF-8")
|
|
||||||
except:
|
|
||||||
log.error("Unknown error!\n```\n{}\n{}```".format(sys.exc_info(), traceback.format_exc()))
|
|
||||||
if glob.sentry:
|
|
||||||
yield tornado.gen.Task(self.captureException, exc_info=True)
|
|
||||||
#finally:
|
|
||||||
# self.finish()
|
|
||||||
|
|
||||||
@tornado.web.asynchronous
|
|
||||||
@tornado.gen.engine
|
|
||||||
def asyncGet(self):
|
|
||||||
html = "<html><head><title>MA MAURO ESISTE?</title><style type='text/css'>body{width:30%}</style></head><body><pre>"
|
|
||||||
html += " _ __<br>"
|
|
||||||
html += " (_) / /<br>"
|
|
||||||
html += " ______ __ ____ ____ / /____<br>"
|
|
||||||
html += " / ___/ / _ \\/ _ \\/ / _ \\<br>"
|
|
||||||
html += " / / / / /_) / /_) / / ____/<br>"
|
|
||||||
html += "/__/ /__/ .___/ .___/__/ \\_____/<br>"
|
|
||||||
html += " / / / /<br>"
|
|
||||||
html += " /__/ /__/<br>"
|
|
||||||
html += "<b>PYTHON > ALL VERSION</b><br><br>"
|
|
||||||
html += "<marquee style='white-space:pre;'><br>"
|
|
||||||
html += " .. o .<br>"
|
|
||||||
html += " o.o o . o<br>"
|
|
||||||
html += " oo...<br>"
|
|
||||||
html += " __[]__<br>"
|
|
||||||
html += " phwr--> _\\:D/_/o_o_o_|__ <span style=\"font-family: 'Comic Sans MS'; font-size: 8pt;\">u wot m8</span><br>"
|
|
||||||
html += " \\\"\"\"\"\"\"\"\"\"\"\"\"\"\"/<br>"
|
|
||||||
html += " \\ . .. .. . /<br>"
|
|
||||||
html += "^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^<br>"
|
|
||||||
html += "</marquee><br><strike>reverse engineering a protocol impossible to reverse engineer since always</strike><br>we are actually reverse engineering bancho successfully. for the third time.<br><br><i>© Ripple team, 2016</i></pre></body></html>"
|
|
||||||
self.write(html)
|
|
266
handlers/mainHandler.pyx
Normal file
266
handlers/mainHandler.pyx
Normal file
|
@ -0,0 +1,266 @@
|
||||||
|
import datetime
|
||||||
|
import gzip
|
||||||
|
import sys
|
||||||
|
import traceback
|
||||||
|
|
||||||
|
import tornado.gen
|
||||||
|
import tornado.web
|
||||||
|
from raven.contrib.tornado import SentryMixin
|
||||||
|
|
||||||
|
from common.log import logUtils as log
|
||||||
|
from common.web import requestsManager
|
||||||
|
from constants import exceptions
|
||||||
|
from constants import packetIDs
|
||||||
|
from constants import serverPackets
|
||||||
|
from events import cantSpectateEvent
|
||||||
|
from events import changeActionEvent
|
||||||
|
from events import changeMatchModsEvent
|
||||||
|
from events import changeMatchPasswordEvent
|
||||||
|
from events import changeMatchSettingsEvent
|
||||||
|
from events import changeSlotEvent
|
||||||
|
from events import channelJoinEvent
|
||||||
|
from events import channelPartEvent
|
||||||
|
from events import createMatchEvent
|
||||||
|
from events import friendAddEvent
|
||||||
|
from events import friendRemoveEvent
|
||||||
|
from events import joinLobbyEvent
|
||||||
|
from events import joinMatchEvent
|
||||||
|
from events import loginEvent
|
||||||
|
from events import logoutEvent
|
||||||
|
from events import matchChangeTeamEvent
|
||||||
|
from events import matchCompleteEvent
|
||||||
|
from events import matchFailedEvent
|
||||||
|
from events import matchFramesEvent
|
||||||
|
from events import matchHasBeatmapEvent
|
||||||
|
from events import matchInviteEvent
|
||||||
|
from events import matchLockEvent
|
||||||
|
from events import matchNoBeatmapEvent
|
||||||
|
from events import matchPlayerLoadEvent
|
||||||
|
from events import matchReadyEvent
|
||||||
|
from events import matchSkipEvent
|
||||||
|
from events import matchStartEvent
|
||||||
|
from events import matchTransferHostEvent
|
||||||
|
from events import partLobbyEvent
|
||||||
|
from events import partMatchEvent
|
||||||
|
from events import requestStatusUpdateEvent
|
||||||
|
from events import sendPrivateMessageEvent
|
||||||
|
from events import sendPublicMessageEvent
|
||||||
|
from events import setAwayMessageEvent
|
||||||
|
from events import spectateFramesEvent
|
||||||
|
from events import startSpectatingEvent
|
||||||
|
from events import stopSpectatingEvent
|
||||||
|
from events import userPanelRequestEvent
|
||||||
|
from events import userStatsRequestEvent
|
||||||
|
from events import tournamentMatchInfoRequestEvent
|
||||||
|
from events import tournamentJoinMatchChannelEvent
|
||||||
|
from events import tournamentLeaveMatchChannelEvent
|
||||||
|
from helpers import packetHelper
|
||||||
|
from objects import glob
|
||||||
|
from common.sentry import sentry
|
||||||
|
|
||||||
|
|
||||||
|
class handler(requestsManager.asyncRequestHandler):
|
||||||
|
@tornado.web.asynchronous
|
||||||
|
@tornado.gen.engine
|
||||||
|
@sentry.captureTornado
|
||||||
|
def asyncPost(self):
|
||||||
|
# Track time if needed
|
||||||
|
if glob.outputRequestTime:
|
||||||
|
# Start time
|
||||||
|
st = datetime.datetime.now()
|
||||||
|
|
||||||
|
# Client's token string and request data
|
||||||
|
requestTokenString = self.request.headers.get("osu-token")
|
||||||
|
requestData = self.request.body
|
||||||
|
|
||||||
|
# Server's token string and request data
|
||||||
|
responseTokenString = "ayy"
|
||||||
|
responseData = bytes()
|
||||||
|
|
||||||
|
if requestTokenString is None:
|
||||||
|
# No token, first request. Handle login.
|
||||||
|
responseTokenString, responseData = loginEvent.handle(self)
|
||||||
|
else:
|
||||||
|
userToken = None # default value
|
||||||
|
try:
|
||||||
|
# This is not the first packet, send response based on client's request
|
||||||
|
# Packet start position, used to read stacked packets
|
||||||
|
pos = 0
|
||||||
|
|
||||||
|
# Make sure the token exists
|
||||||
|
if requestTokenString not in glob.tokens.tokens:
|
||||||
|
raise exceptions.tokenNotFoundException()
|
||||||
|
|
||||||
|
# Token exists, get its object and lock it
|
||||||
|
userToken = glob.tokens.tokens[requestTokenString]
|
||||||
|
userToken.processingLock.acquire()
|
||||||
|
|
||||||
|
# Keep reading packets until everything has been read
|
||||||
|
while pos < len(requestData):
|
||||||
|
# Get packet from stack starting from new packet
|
||||||
|
leftData = requestData[pos:]
|
||||||
|
|
||||||
|
# Get packet ID, data length and data
|
||||||
|
packetID = packetHelper.readPacketID(leftData)
|
||||||
|
dataLength = packetHelper.readPacketLength(leftData)
|
||||||
|
packetData = requestData[pos:(pos+dataLength+7)]
|
||||||
|
|
||||||
|
# Console output if needed
|
||||||
|
if glob.outputPackets and packetID != 4:
|
||||||
|
log.debug("Incoming packet ({})({}):\n\nPacket code: {}\nPacket length: {}\nSingle packet data: {}\n".format(requestTokenString, userToken.username, str(packetID), str(dataLength), str(packetData)))
|
||||||
|
|
||||||
|
# Event handler
|
||||||
|
def handleEvent(ev):
|
||||||
|
def wrapper():
|
||||||
|
ev.handle(userToken, packetData)
|
||||||
|
return wrapper
|
||||||
|
|
||||||
|
eventHandler = {
|
||||||
|
packetIDs.client_changeAction: handleEvent(changeActionEvent),
|
||||||
|
packetIDs.client_logout: handleEvent(logoutEvent),
|
||||||
|
packetIDs.client_friendAdd: handleEvent(friendAddEvent),
|
||||||
|
packetIDs.client_friendRemove: handleEvent(friendRemoveEvent),
|
||||||
|
packetIDs.client_userStatsRequest: handleEvent(userStatsRequestEvent),
|
||||||
|
packetIDs.client_requestStatusUpdate: handleEvent(requestStatusUpdateEvent),
|
||||||
|
packetIDs.client_userPanelRequest: handleEvent(userPanelRequestEvent),
|
||||||
|
|
||||||
|
packetIDs.client_channelJoin: handleEvent(channelJoinEvent),
|
||||||
|
packetIDs.client_channelPart: handleEvent(channelPartEvent),
|
||||||
|
packetIDs.client_sendPublicMessage: handleEvent(sendPublicMessageEvent),
|
||||||
|
packetIDs.client_sendPrivateMessage: handleEvent(sendPrivateMessageEvent),
|
||||||
|
packetIDs.client_setAwayMessage: handleEvent(setAwayMessageEvent),
|
||||||
|
|
||||||
|
packetIDs.client_startSpectating: handleEvent(startSpectatingEvent),
|
||||||
|
packetIDs.client_stopSpectating: handleEvent(stopSpectatingEvent),
|
||||||
|
packetIDs.client_cantSpectate: handleEvent(cantSpectateEvent),
|
||||||
|
packetIDs.client_spectateFrames: handleEvent(spectateFramesEvent),
|
||||||
|
|
||||||
|
packetIDs.client_joinLobby: handleEvent(joinLobbyEvent),
|
||||||
|
packetIDs.client_partLobby: handleEvent(partLobbyEvent),
|
||||||
|
packetIDs.client_createMatch: handleEvent(createMatchEvent),
|
||||||
|
packetIDs.client_joinMatch: handleEvent(joinMatchEvent),
|
||||||
|
packetIDs.client_partMatch: handleEvent(partMatchEvent),
|
||||||
|
packetIDs.client_matchChangeSlot: handleEvent(changeSlotEvent),
|
||||||
|
packetIDs.client_matchChangeSettings: handleEvent(changeMatchSettingsEvent),
|
||||||
|
packetIDs.client_matchChangePassword: handleEvent(changeMatchPasswordEvent),
|
||||||
|
packetIDs.client_matchChangeMods: handleEvent(changeMatchModsEvent),
|
||||||
|
packetIDs.client_matchReady: handleEvent(matchReadyEvent),
|
||||||
|
packetIDs.client_matchNotReady: handleEvent(matchReadyEvent),
|
||||||
|
packetIDs.client_matchLock: handleEvent(matchLockEvent),
|
||||||
|
packetIDs.client_matchStart: handleEvent(matchStartEvent),
|
||||||
|
packetIDs.client_matchLoadComplete: handleEvent(matchPlayerLoadEvent),
|
||||||
|
packetIDs.client_matchSkipRequest: handleEvent(matchSkipEvent),
|
||||||
|
packetIDs.client_matchScoreUpdate: handleEvent(matchFramesEvent),
|
||||||
|
packetIDs.client_matchComplete: handleEvent(matchCompleteEvent),
|
||||||
|
packetIDs.client_matchNoBeatmap: handleEvent(matchNoBeatmapEvent),
|
||||||
|
packetIDs.client_matchHasBeatmap: handleEvent(matchHasBeatmapEvent),
|
||||||
|
packetIDs.client_matchTransferHost: handleEvent(matchTransferHostEvent),
|
||||||
|
packetIDs.client_matchFailed: handleEvent(matchFailedEvent),
|
||||||
|
packetIDs.client_matchChangeTeam: handleEvent(matchChangeTeamEvent),
|
||||||
|
packetIDs.client_invite: handleEvent(matchInviteEvent),
|
||||||
|
|
||||||
|
packetIDs.client_tournamentMatchInfoRequest: handleEvent(tournamentMatchInfoRequestEvent),
|
||||||
|
packetIDs.client_tournamentJoinMatchChannel: handleEvent(tournamentJoinMatchChannelEvent),
|
||||||
|
packetIDs.client_tournamentLeaveMatchChannel: handleEvent(tournamentLeaveMatchChannelEvent),
|
||||||
|
}
|
||||||
|
|
||||||
|
# Packets processed if in restricted mode.
|
||||||
|
# All other packets will be ignored if the user is in restricted mode
|
||||||
|
packetsRestricted = [
|
||||||
|
packetIDs.client_logout,
|
||||||
|
packetIDs.client_userStatsRequest,
|
||||||
|
packetIDs.client_requestStatusUpdate,
|
||||||
|
packetIDs.client_userPanelRequest,
|
||||||
|
packetIDs.client_changeAction,
|
||||||
|
packetIDs.client_channelJoin,
|
||||||
|
packetIDs.client_channelPart,
|
||||||
|
]
|
||||||
|
|
||||||
|
# Process/ignore packet
|
||||||
|
if packetID != 4:
|
||||||
|
if packetID in eventHandler:
|
||||||
|
if not userToken.restricted or (userToken.restricted and packetID in packetsRestricted):
|
||||||
|
eventHandler[packetID]()
|
||||||
|
else:
|
||||||
|
log.warning("Ignored packet id from {} ({}) (user is restricted)".format(requestTokenString, packetID))
|
||||||
|
else:
|
||||||
|
log.warning("Unknown packet id from {} ({})".format(requestTokenString, packetID))
|
||||||
|
|
||||||
|
# Update pos so we can read the next stacked packet
|
||||||
|
# +7 because we add packet ID bytes, unused byte and data length bytes
|
||||||
|
pos += dataLength+7
|
||||||
|
|
||||||
|
# Token queue built, send it
|
||||||
|
responseTokenString = userToken.token
|
||||||
|
responseData = userToken.queue
|
||||||
|
userToken.resetQueue()
|
||||||
|
except exceptions.tokenNotFoundException:
|
||||||
|
# Token not found. Disconnect that user
|
||||||
|
responseData = serverPackets.loginError()
|
||||||
|
responseData += serverPackets.notification("Whoops! Something went wrong, please login again.")
|
||||||
|
log.warning("Received packet from unknown token ({}).".format(requestTokenString))
|
||||||
|
log.info("{} has been disconnected (invalid token)".format(requestTokenString))
|
||||||
|
finally:
|
||||||
|
# Unlock token
|
||||||
|
if userToken is not None:
|
||||||
|
# Update ping time for timeout
|
||||||
|
userToken.updatePingTime()
|
||||||
|
# Release processing lock
|
||||||
|
userToken.processingLock.release()
|
||||||
|
# Delete token if kicked
|
||||||
|
if userToken.kicked:
|
||||||
|
glob.tokens.deleteToken(userToken)
|
||||||
|
|
||||||
|
if glob.outputRequestTime:
|
||||||
|
# End time
|
||||||
|
et = datetime.datetime.now()
|
||||||
|
|
||||||
|
# Total time:
|
||||||
|
tt = float((et.microsecond-st.microsecond)/1000)
|
||||||
|
log.debug("Request time: {}ms".format(tt))
|
||||||
|
|
||||||
|
# Send server's response to client
|
||||||
|
# We don't use token object because we might not have a token (failed login)
|
||||||
|
if glob.gzip:
|
||||||
|
# First, write the gzipped response
|
||||||
|
self.write(gzip.compress(responseData, int(glob.conf.config["server"]["gziplevel"])))
|
||||||
|
|
||||||
|
# Then, add gzip headers
|
||||||
|
self.add_header("Vary", "Accept-Encoding")
|
||||||
|
self.add_header("Content-Encoding", "gzip")
|
||||||
|
else:
|
||||||
|
# First, write the response
|
||||||
|
self.write(responseData)
|
||||||
|
|
||||||
|
# Add all the headers AFTER the response has been written
|
||||||
|
self.set_status(200)
|
||||||
|
self.add_header("cho-token", responseTokenString)
|
||||||
|
self.add_header("cho-protocol", "19")
|
||||||
|
self.add_header("Connection", "keep-alive")
|
||||||
|
self.add_header("Keep-Alive", "timeout=5, max=100")
|
||||||
|
self.add_header("Content-Type", "text/html; charset=UTF-8")
|
||||||
|
|
||||||
|
@tornado.web.asynchronous
|
||||||
|
@tornado.gen.engine
|
||||||
|
def asyncGet(self):
|
||||||
|
html = "<html><head><title>MA MAURO ESISTE?</title><style type='text/css'>body{width:30%;background:#222;color:#fff;}</style></head><body><pre>"
|
||||||
|
html += " _ __<br>"
|
||||||
|
html += " (_) / /<br>"
|
||||||
|
html += " ______ __ ____ ____ / /____<br>"
|
||||||
|
html += " / ___/ / _ \\/ _ \\/ / _ \\<br>"
|
||||||
|
html += " / / / / /_) / /_) / / ____/<br>"
|
||||||
|
html += "/__/ /__/ .___/ .___/__/ \\_____/<br>"
|
||||||
|
html += " / / / /<br>"
|
||||||
|
html += " /__/ /__/<br>"
|
||||||
|
html += "<b>PYTHON > ALL VERSION</b><br><br>"
|
||||||
|
html += "<marquee style='white-space:pre;'><br>"
|
||||||
|
html += " .. o .<br>"
|
||||||
|
html += " o.o o . o<br>"
|
||||||
|
html += " oo...<br>"
|
||||||
|
html += " __[]__<br>"
|
||||||
|
html += " phwr--> _\\:D/_/o_o_o_|__ <span style=\"font-family: 'Comic Sans MS'; font-size: 8pt;\">u wot m8</span><br>"
|
||||||
|
html += " \\\"\"\"\"\"\"\"\"\"\"\"\"\"\"/<br>"
|
||||||
|
html += " \\ . .. .. . /<br>"
|
||||||
|
html += "^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^<br>"
|
||||||
|
html += "</marquee><br><strike>reverse engineering a protocol impossible to reverse engineer since always</strike><br>we are actually reverse engineering bancho successfully. for the third time.<br><br>Running osufx branch.<br><i>© Ripple team, 2016</i></pre></body></html>"
|
||||||
|
self.write(html)
|
|
@ -8,18 +8,16 @@ from objects import fokabot
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
|
|
||||||
def joinChannel(userID = 0, channel = "", token = None, toIRC = True):
|
def joinChannel(userID = 0, channel = "", token = None, toIRC = True, force=False):
|
||||||
"""
|
"""
|
||||||
Join a channel
|
Join a channel
|
||||||
|
|
||||||
userID -- user ID of the user that joins the channel. Optional.
|
:param userID: user ID of the user that joins the channel. Optional. token can be used instead.
|
||||||
token can be used instead.
|
:param token: user token object of user that joins the channel. Optional. userID can be used instead.
|
||||||
token -- user token object of user that joins the channel. Optional.
|
:param channel: channel name
|
||||||
userID can be used instead.
|
:param toIRC: if True, send this channel join event to IRC. Must be true if joining from bancho. Default: True
|
||||||
channel -- name of channe
|
:param force: whether to allow game clients to join #spect_ and #multi_ channels
|
||||||
toIRC -- if True, send this channel join event to IRC. Must be true if joining from bancho.
|
:return: 0 if joined or other IRC code in case of error. Needed only on IRC-side
|
||||||
Optional. Defaukt: True
|
|
||||||
return -- returns 0 if joined or other IRC code in case of error. Needed only on IRC-side
|
|
||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
# Get token if not defined
|
# Get token if not defined
|
||||||
|
@ -30,76 +28,67 @@ def joinChannel(userID = 0, channel = "", token = None, toIRC = True):
|
||||||
raise exceptions.userNotFoundException
|
raise exceptions.userNotFoundException
|
||||||
else:
|
else:
|
||||||
token = token
|
token = token
|
||||||
userID = token.userID
|
|
||||||
|
|
||||||
# Get usertoken data
|
|
||||||
username = token.username
|
|
||||||
|
|
||||||
# Normal channel, do check stuff
|
# Normal channel, do check stuff
|
||||||
# Make sure the channel exists
|
# Make sure the channel exists
|
||||||
if channel not in glob.channels.channels:
|
if channel not in glob.channels.channels:
|
||||||
raise exceptions.channelUnknownException
|
raise exceptions.channelUnknownException()
|
||||||
|
|
||||||
# Check channel permissions
|
# Make sure a game client is not trying to join a #multi_ or #spect_ channel manually
|
||||||
channelObject = glob.channels.channels[channel]
|
channelObject = glob.channels.channels[channel]
|
||||||
if channelObject.publicRead == False and token.admin == False:
|
if channelObject.isSpecial and not token.irc and not force:
|
||||||
raise exceptions.channelNoPermissionsException
|
raise exceptions.channelUnknownException()
|
||||||
|
|
||||||
# Add our userID to users in that channel
|
|
||||||
channelObject.userJoin(userID)
|
|
||||||
|
|
||||||
# Add the channel to our joined channel
|
# Add the channel to our joined channel
|
||||||
token.joinChannel(channel)
|
token.joinChannel(channelObject)
|
||||||
|
|
||||||
# Send channel joined (bancho). We use clientName here because of #multiplayer and #spectator channels
|
|
||||||
token.enqueue(serverPackets.channelJoinSuccess(userID, channelObject.clientName))
|
|
||||||
|
|
||||||
# Send channel joined (IRC)
|
# Send channel joined (IRC)
|
||||||
if glob.irc == True and toIRC == True:
|
if glob.irc and not toIRC:
|
||||||
glob.ircServer.banchoJoinChannel(username, channel)
|
glob.ircServer.banchoJoinChannel(token.username, channel)
|
||||||
|
|
||||||
# Console output
|
# Console output
|
||||||
log.info("{} joined channel {}".format(username, channel))
|
log.info("{} joined channel {}".format(token.username, channel))
|
||||||
|
|
||||||
# IRC code return
|
# IRC code return
|
||||||
return 0
|
return 0
|
||||||
except exceptions.channelNoPermissionsException:
|
except exceptions.channelNoPermissionsException:
|
||||||
log.warning("{} attempted to join channel {}, but they have no read permissions".format(username, channel))
|
log.warning("{} attempted to join channel {}, but they have no read permissions".format(token.username, channel))
|
||||||
return 403
|
return 403
|
||||||
except exceptions.channelUnknownException:
|
except exceptions.channelUnknownException:
|
||||||
log.warning("{} attempted to join an unknown channel ({})".format(username, channel))
|
log.warning("{} attempted to join an unknown channel ({})".format(token.username, channel))
|
||||||
|
return 403
|
||||||
|
except exceptions.userAlreadyInChannelException:
|
||||||
|
log.warning("User {} already in channel {}".format(token.username, channel))
|
||||||
return 403
|
return 403
|
||||||
except exceptions.userNotFoundException:
|
except exceptions.userNotFoundException:
|
||||||
log.warning("User not connected to IRC/Bancho")
|
log.warning("User not connected to IRC/Bancho")
|
||||||
return 403 # idk
|
return 403 # idk
|
||||||
|
|
||||||
def partChannel(userID = 0, channel = "", token = None, toIRC = True, kick = False):
|
def partChannel(userID = 0, channel = "", token = None, toIRC = True, kick = False, force=False):
|
||||||
"""
|
"""
|
||||||
Part a channel
|
Part a channel
|
||||||
|
|
||||||
userID -- user ID of the user that parts the channel. Optional.
|
:param userID: user ID of the user that parts the channel. Optional. token can be used instead.
|
||||||
token can be used instead.
|
:param token: user token object of user that parts the channel. Optional. userID can be used instead.
|
||||||
token -- user token object of user that parts the channel. Optional.
|
:param channel: channel name
|
||||||
userID can be used instead.
|
:param toIRC: if True, send this channel join event to IRC. Must be true if joining from bancho. Optional. Default: True
|
||||||
channel -- name of channel
|
:param kick: if True, channel tab will be closed on client. Used when leaving lobby. Optional. Default: False
|
||||||
toIRC -- if True, send this channel join event to IRC. Must be true if joining from bancho.
|
:param force: whether to allow game clients to part #spect_ and #multi_ channels
|
||||||
Optional. Defaukt: True
|
:return: 0 if joined or other IRC code in case of error. Needed only on IRC-side
|
||||||
kick -- if True, channel tab will be closed on client. Used when leaving lobby. Optional. Default: False
|
|
||||||
return -- returns 0 if joined or other IRC code in case of error. Needed only on IRC-side
|
|
||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
|
# Make sure the client is not drunk and sends partChannel when closing a PM tab
|
||||||
|
if not channel.startswith("#"):
|
||||||
|
return
|
||||||
|
|
||||||
# Get token if not defined
|
# Get token if not defined
|
||||||
if token is None:
|
if token is None:
|
||||||
token = glob.tokens.getTokenFromUserID(userID)
|
token = glob.tokens.getTokenFromUserID(userID)
|
||||||
# Make sure the token exists
|
# Make sure the token exists
|
||||||
if token is None:
|
if token is None:
|
||||||
raise exceptions.userNotFoundException
|
raise exceptions.userNotFoundException()
|
||||||
else:
|
else:
|
||||||
token = token
|
token = token
|
||||||
userID = token.userID
|
|
||||||
|
|
||||||
# Get usertoken data
|
|
||||||
username = token.username
|
|
||||||
|
|
||||||
# Determine internal/client name if needed
|
# Determine internal/client name if needed
|
||||||
# (toclient is used clientwise for #multiplayer and #spectator channels)
|
# (toclient is used clientwise for #multiplayer and #spectator channels)
|
||||||
|
@ -119,12 +108,24 @@ def partChannel(userID = 0, channel = "", token = None, toIRC = True, kick = Fal
|
||||||
|
|
||||||
# Make sure the channel exists
|
# Make sure the channel exists
|
||||||
if channel not in glob.channels.channels:
|
if channel not in glob.channels.channels:
|
||||||
raise exceptions.channelUnknownException
|
raise exceptions.channelUnknownException()
|
||||||
|
|
||||||
|
# Make sure a game client is not trying to join a #multi_ or #spect_ channel manually
|
||||||
|
channelObject = glob.channels.channels[channel]
|
||||||
|
if channelObject.isSpecial and not token.irc and not force:
|
||||||
|
raise exceptions.channelUnknownException()
|
||||||
|
|
||||||
|
# Make sure the user is in the channel
|
||||||
|
if channel not in token.joinedChannels:
|
||||||
|
raise exceptions.userNotInChannelException()
|
||||||
|
|
||||||
# Part channel (token-side and channel-side)
|
# Part channel (token-side and channel-side)
|
||||||
channelObject = glob.channels.channels[channel]
|
token.partChannel(channelObject)
|
||||||
token.partChannel(channel)
|
|
||||||
channelObject.userPart(userID)
|
# Delete temporary channel if everyone left
|
||||||
|
if "chat/{}".format(channelObject.name) in glob.streams.streams:
|
||||||
|
if channelObject.temp and len(glob.streams.streams["chat/{}".format(channelObject.name)].clients) - 1 == 0:
|
||||||
|
glob.channels.removeChannel(channelObject.name)
|
||||||
|
|
||||||
# Force close tab if needed
|
# Force close tab if needed
|
||||||
# NOTE: Maybe always needed, will check later
|
# NOTE: Maybe always needed, will check later
|
||||||
|
@ -132,17 +133,20 @@ def partChannel(userID = 0, channel = "", token = None, toIRC = True, kick = Fal
|
||||||
token.enqueue(serverPackets.channelKicked(channelClient))
|
token.enqueue(serverPackets.channelKicked(channelClient))
|
||||||
|
|
||||||
# IRC part
|
# IRC part
|
||||||
if glob.irc == True and toIRC == True:
|
if glob.irc and toIRC:
|
||||||
glob.ircServer.banchoPartChannel(username, channel)
|
glob.ircServer.banchoPartChannel(token.username, channel)
|
||||||
|
|
||||||
# Console output
|
# Console output
|
||||||
log.info("{} parted channel {} ({})".format(username, channel, channelClient))
|
log.info("{} parted channel {} ({})".format(token.username, channel, channelClient))
|
||||||
|
|
||||||
# Return IRC code
|
# Return IRC code
|
||||||
return 0
|
return 0
|
||||||
except exceptions.channelUnknownException:
|
except exceptions.channelUnknownException:
|
||||||
log.warning("{} attempted to part an unknown channel ({})".format(username, channel))
|
log.warning("{} attempted to part an unknown channel ({})".format(token.username, channel))
|
||||||
return 403
|
return 403
|
||||||
|
except exceptions.userNotInChannelException:
|
||||||
|
log.warning("{} attempted to part {}, but he's not in that channel".format(token.username, channel))
|
||||||
|
return 442
|
||||||
except exceptions.userNotFoundException:
|
except exceptions.userNotFoundException:
|
||||||
log.warning("User not connected to IRC/Bancho")
|
log.warning("User not connected to IRC/Bancho")
|
||||||
return 442 # idk
|
return 442 # idk
|
||||||
|
@ -151,35 +155,28 @@ def sendMessage(fro = "", to = "", message = "", token = None, toIRC = True):
|
||||||
"""
|
"""
|
||||||
Send a message to osu!bancho and IRC server
|
Send a message to osu!bancho and IRC server
|
||||||
|
|
||||||
fro -- sender username. Optional.
|
:param fro: sender username. Optional. token can be used instead
|
||||||
You can use token instead of this if you wish.
|
:param to: receiver channel (if starts with #) or username
|
||||||
to -- receiver channel (if starts with #) or username
|
:param message: text of the message
|
||||||
message -- text of the message
|
:param token: sender token object. Optional. fro can be used instead
|
||||||
token -- sender token object.
|
:param toIRC: if True, send the message to IRC. If False, send it to Bancho only. Default: True
|
||||||
You can use this instead of fro if you are sending messages from bancho.
|
:return: 0 if joined or other IRC code in case of error. Needed only on IRC-side
|
||||||
Optional.
|
|
||||||
toIRC -- if True, send the message to IRC. If False, send it to Bancho only.
|
|
||||||
Optional. Default: True
|
|
||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
tokenString = ""
|
#tokenString = ""
|
||||||
# Get token object if not passed
|
# Get token object if not passed
|
||||||
if token is None:
|
if token is None:
|
||||||
token = glob.tokens.getTokenFromUsername(fro)
|
token = glob.tokens.getTokenFromUsername(fro)
|
||||||
if token is None:
|
if token is None:
|
||||||
raise exceptions.userNotFoundException
|
raise exceptions.userNotFoundException()
|
||||||
else:
|
else:
|
||||||
# token object alredy passed, get its string and its username (fro)
|
# token object alredy passed, get its string and its username (fro)
|
||||||
fro = token.username
|
fro = token.username
|
||||||
tokenString = token.token
|
#tokenString = token.token
|
||||||
|
|
||||||
# Set some variables
|
|
||||||
userID = token.userID
|
|
||||||
username = token.username
|
|
||||||
|
|
||||||
# Make sure this is not a tournament client
|
# Make sure this is not a tournament client
|
||||||
if token.tournament:
|
# if token.tournament:
|
||||||
raise exceptions.userTournamentException()
|
# raise exceptions.userTournamentException()
|
||||||
|
|
||||||
# Make sure the user is not in restricted mode
|
# Make sure the user is not in restricted mode
|
||||||
if token.restricted:
|
if token.restricted:
|
||||||
|
@ -189,12 +186,16 @@ def sendMessage(fro = "", to = "", message = "", token = None, toIRC = True):
|
||||||
if token.isSilenced():
|
if token.isSilenced():
|
||||||
raise exceptions.userSilencedException()
|
raise exceptions.userSilencedException()
|
||||||
|
|
||||||
|
# Redirect !report to FokaBot
|
||||||
|
if message.startswith("!report"):
|
||||||
|
to = glob.BOT_NAME
|
||||||
|
|
||||||
# Determine internal name if needed
|
# Determine internal name if needed
|
||||||
# (toclient is used clientwise for #multiplayer and #spectator channels)
|
# (toclient is used clientwise for #multiplayer and #spectator channels)
|
||||||
toClient = to
|
toClient = to
|
||||||
if to == "#spectator":
|
if to == "#spectator":
|
||||||
if token.spectating is None:
|
if token.spectating is None:
|
||||||
s = userID
|
s = token.userID
|
||||||
else:
|
else:
|
||||||
s = token.spectatingUserID
|
s = token.spectatingUserID
|
||||||
to = "#spect_{}".format(s)
|
to = "#spect_{}".format(s)
|
||||||
|
@ -205,6 +206,10 @@ def sendMessage(fro = "", to = "", message = "", token = None, toIRC = True):
|
||||||
elif to.startswith("#multi_"):
|
elif to.startswith("#multi_"):
|
||||||
toClient = "#multiplayer"
|
toClient = "#multiplayer"
|
||||||
|
|
||||||
|
# Make sure the message is valid
|
||||||
|
if not message.strip():
|
||||||
|
raise exceptions.invalidArgumentsException()
|
||||||
|
|
||||||
# Truncate message if > 2048 characters
|
# Truncate message if > 2048 characters
|
||||||
message = message[:2048]+"..." if len(message) > 2048 else message
|
message = message[:2048]+"..." if len(message) > 2048 else message
|
||||||
|
|
||||||
|
@ -212,7 +217,7 @@ def sendMessage(fro = "", to = "", message = "", token = None, toIRC = True):
|
||||||
message = glob.chatFilters.filterMessage(message)
|
message = glob.chatFilters.filterMessage(message)
|
||||||
|
|
||||||
# Build packet bytes
|
# Build packet bytes
|
||||||
packet = serverPackets.sendMessage(username, toClient, message)
|
packet = serverPackets.sendMessage(token.username, toClient, message)
|
||||||
|
|
||||||
# Send the message
|
# Send the message
|
||||||
isChannel = to.startswith("#")
|
isChannel = to.startswith("#")
|
||||||
|
@ -220,94 +225,113 @@ def sendMessage(fro = "", to = "", message = "", token = None, toIRC = True):
|
||||||
# CHANNEL
|
# CHANNEL
|
||||||
# Make sure the channel exists
|
# Make sure the channel exists
|
||||||
if to not in glob.channels.channels:
|
if to not in glob.channels.channels:
|
||||||
raise exceptions.channelUnknownException
|
raise exceptions.channelUnknownException()
|
||||||
|
|
||||||
# Make sure the channel is not in moderated mode
|
# Make sure the channel is not in moderated mode
|
||||||
if glob.channels.channels[to].moderated == True and token.admin == False:
|
if glob.channels.channels[to].moderated and not token.admin:
|
||||||
raise exceptions.channelModeratedException
|
raise exceptions.channelModeratedException()
|
||||||
|
|
||||||
|
# Make sure we are in the channel
|
||||||
|
if to not in token.joinedChannels:
|
||||||
|
# I'm too lazy to put and test the correct IRC error code here...
|
||||||
|
# but IRC is not strict at all so who cares
|
||||||
|
raise exceptions.channelNoPermissionsException()
|
||||||
|
|
||||||
# Make sure we have write permissions
|
# Make sure we have write permissions
|
||||||
if glob.channels.channels[to].publicWrite == False and token.admin == False:
|
if not glob.channels.channels[to].publicWrite and not token.admin:
|
||||||
raise exceptions.channelNoPermissionsException
|
raise exceptions.channelNoPermissionsException()
|
||||||
|
|
||||||
|
# Add message in buffer
|
||||||
|
token.addMessageInBuffer(to, message)
|
||||||
|
|
||||||
# Everything seems fine, build recipients list and send packet
|
# Everything seems fine, build recipients list and send packet
|
||||||
recipients = glob.channels.channels[to].getConnectedUsers()[:]
|
glob.streams.broadcast("chat/{}".format(to), packet, but=[token.token])
|
||||||
for key, value in glob.tokens.tokens.items():
|
|
||||||
# Skip our client and irc clients
|
|
||||||
if key == tokenString or value.irc == True:
|
|
||||||
continue
|
|
||||||
# Send to this client if it's connected to the channel
|
|
||||||
if value.userID in recipients:
|
|
||||||
value.enqueue(packet)
|
|
||||||
else:
|
else:
|
||||||
# USER
|
# USER
|
||||||
# Make sure recipient user is connected
|
# Make sure recipient user is connected
|
||||||
recipientToken = glob.tokens.getTokenFromUsername(to)
|
recipientToken = glob.tokens.getTokenFromUsername(to)
|
||||||
if recipientToken is None:
|
if recipientToken is None:
|
||||||
raise exceptions.userNotFoundException
|
raise exceptions.userNotFoundException()
|
||||||
|
|
||||||
# Make sure the recipient is not a tournament client
|
# Make sure the recipient is not a tournament client
|
||||||
if recipientToken.tournament:
|
#if recipientToken.tournament:
|
||||||
raise exceptions.userTournamentException()
|
# raise exceptions.userTournamentException()
|
||||||
|
|
||||||
# Make sure the recipient is not restricted or we are FokaBot
|
# Make sure the recipient is not restricted or we are FokaBot
|
||||||
if recipientToken.restricted == True and fro.lower() != "fokabot":
|
if recipientToken.restricted and fro.lower() != glob.BOT_NAME.lower():
|
||||||
raise exceptions.userRestrictedException()
|
raise exceptions.userRestrictedException()
|
||||||
|
|
||||||
# TODO: Make sure the recipient has not disabled PMs for non-friends or he's our friend
|
# TODO: Make sure the recipient has not disabled PMs for non-friends or he's our friend
|
||||||
|
|
||||||
|
# Away check
|
||||||
|
if recipientToken.awayCheck(token.userID):
|
||||||
|
sendMessage(to, fro, "\x01ACTION is away: {}\x01".format(recipientToken.awayMessage))
|
||||||
|
|
||||||
# Check message templates (mods/admins only)
|
# Check message templates (mods/admins only)
|
||||||
if message in messageTemplates.templates and token.admin == True:
|
if message in messageTemplates.templates and token.admin:
|
||||||
sendMessage(fro, to, messageTemplates.templates[message])
|
sendMessage(fro, to, messageTemplates.templates[message])
|
||||||
|
|
||||||
# Everything seems fine, send packet
|
# Everything seems fine, send packet
|
||||||
recipientToken.enqueue(packet)
|
recipientToken.enqueue(packet)
|
||||||
|
|
||||||
# Send the message to IRC
|
# Send the message to IRC
|
||||||
if glob.irc == True and toIRC == True:
|
if glob.irc and toIRC:
|
||||||
glob.ircServer.banchoMessage(fro, to, message)
|
messageSplitInLines = message.encode("latin-1").decode("utf-8").split("\n")
|
||||||
|
for line in messageSplitInLines:
|
||||||
|
if line == messageSplitInLines[:1] and line == "":
|
||||||
|
continue
|
||||||
|
glob.ircServer.banchoMessage(fro, to, line)
|
||||||
|
|
||||||
# Spam protection (ignore FokaBot)
|
# Spam protection (ignore FokaBot)
|
||||||
if userID > 999:
|
if token.userID > 999:
|
||||||
token.spamProtection()
|
token.spamProtection()
|
||||||
|
|
||||||
# Fokabot message
|
# Fokabot message
|
||||||
if isChannel == True or to.lower() == "fokabot":
|
if isChannel or to.lower() == glob.BOT_NAME.lower():
|
||||||
fokaMessage = fokabot.fokabotResponse(username, to, message)
|
fokaMessage = fokabot.fokabotResponse(token.username, to, message)
|
||||||
if fokaMessage:
|
if fokaMessage:
|
||||||
sendMessage("FokaBot", to if isChannel else fro, fokaMessage)
|
sendMessage(glob.BOT_NAME, to if isChannel else fro, fokaMessage)
|
||||||
|
|
||||||
# File and discord logs (public chat only)
|
# File and discord logs (public chat only)
|
||||||
if to.startswith("#"):
|
if to.startswith("#") and not (message.startswith("\x01ACTION is playing") and to.startswith("#spect_")):
|
||||||
log.chat("{fro} @ {to}: {message}".format(fro=username, to=to, message=str(message.encode("utf-8"))))
|
log.chat("{fro} @ {to}: {message}".format(fro=token.username, to=to, message=message.encode("latin-1").decode("utf-8")))
|
||||||
glob.schiavo.sendChatlog("**{fro} @ {to}:** {message}".format(fro=username, to=to, message=str(message.encode("utf-8"))[2:-1]))
|
glob.schiavo.sendChatlog("**{fro} @ {to}:** {message}".format(fro=token.username, to=to, message=message.encode("latin-1").decode("utf-8")))
|
||||||
return 0
|
return 0
|
||||||
except exceptions.userSilencedException:
|
except exceptions.userSilencedException:
|
||||||
token.enqueue(serverPackets.silenceEndTime(token.getSilenceSecondsLeft()))
|
token.enqueue(serverPackets.silenceEndTime(token.getSilenceSecondsLeft()))
|
||||||
log.warning("{} tried to send a message during silence".format(username))
|
log.warning("{} tried to send a message during silence".format(token.username))
|
||||||
return 404
|
return 404
|
||||||
except exceptions.channelModeratedException:
|
except exceptions.channelModeratedException:
|
||||||
log.warning("{} tried to send a message to a channel that is in moderated mode ({})".format(username, to))
|
log.warning("{} tried to send a message to a channel that is in moderated mode ({})".format(token.username, to))
|
||||||
return 404
|
return 404
|
||||||
except exceptions.channelUnknownException:
|
except exceptions.channelUnknownException:
|
||||||
log.warning("{} tried to send a message to an unknown channel ({})".format(username, to))
|
log.warning("{} tried to send a message to an unknown channel ({})".format(token.username, to))
|
||||||
return 403
|
return 403
|
||||||
except exceptions.channelNoPermissionsException:
|
except exceptions.channelNoPermissionsException:
|
||||||
log.warning("{} tried to send a message to channel {}, but they have no write permissions".format(username, to))
|
log.warning("{} tried to send a message to channel {}, but they have no write permissions".format(token.username, to))
|
||||||
return 404
|
return 404
|
||||||
except exceptions.userRestrictedException:
|
except exceptions.userRestrictedException:
|
||||||
log.warning("{} tried to send a message {}, but the recipient is in restricted mode".format(username, to))
|
log.warning("{} tried to send a message {}, but the recipient is in restricted mode".format(token.username, to))
|
||||||
return 404
|
return 404
|
||||||
except exceptions.userTournamentException:
|
except exceptions.userTournamentException:
|
||||||
log.warning("{} tried to send a message {}, but the recipient is a tournament client".format(username, to))
|
log.warning("{} tried to send a message {}, but the recipient is a tournament client".format(token.username, to))
|
||||||
return 404
|
return 404
|
||||||
except exceptions.userNotFoundException:
|
except exceptions.userNotFoundException:
|
||||||
log.warning("User not connected to IRC/Bancho")
|
log.warning("User not connected to IRC/Bancho")
|
||||||
return 401
|
return 401
|
||||||
|
except exceptions.invalidArgumentsException:
|
||||||
|
log.warning("{} tried to send an invalid message to {}".format(token.username, to))
|
||||||
|
return 404
|
||||||
|
|
||||||
|
|
||||||
""" IRC-Bancho Connect/Disconnect/Join/Part interfaces"""
|
""" IRC-Bancho Connect/Disconnect/Join/Part interfaces"""
|
||||||
def fixUsernameForBancho(username):
|
def fixUsernameForBancho(username):
|
||||||
|
"""
|
||||||
|
Convert username from IRC format (without spaces) to Bancho format (with spaces)
|
||||||
|
|
||||||
|
:param username: username to convert
|
||||||
|
:return: converted username
|
||||||
|
"""
|
||||||
# If there are no spaces or underscores in the name
|
# If there are no spaces or underscores in the name
|
||||||
# return it
|
# return it
|
||||||
if " " not in username and "_" not in username:
|
if " " not in username and "_" not in username:
|
||||||
|
@ -322,9 +346,22 @@ def fixUsernameForBancho(username):
|
||||||
return username.replace("_", " ")
|
return username.replace("_", " ")
|
||||||
|
|
||||||
def fixUsernameForIRC(username):
|
def fixUsernameForIRC(username):
|
||||||
|
"""
|
||||||
|
Convert an username from Bancho format to IRC format (underscores instead of spaces)
|
||||||
|
|
||||||
|
:param username: username to convert
|
||||||
|
:return: converted username
|
||||||
|
"""
|
||||||
return username.replace(" ", "_")
|
return username.replace(" ", "_")
|
||||||
|
|
||||||
def IRCConnect(username):
|
def IRCConnect(username):
|
||||||
|
"""
|
||||||
|
Handle IRC login bancho-side.
|
||||||
|
Add token and broadcast login packet.
|
||||||
|
|
||||||
|
:param username: username
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
userID = userUtils.getID(username)
|
userID = userUtils.getID(username)
|
||||||
if not userID:
|
if not userID:
|
||||||
log.warning("{} doesn't exist".format(username))
|
log.warning("{} doesn't exist".format(username))
|
||||||
|
@ -335,6 +372,13 @@ def IRCConnect(username):
|
||||||
log.info("{} logged in from IRC".format(username))
|
log.info("{} logged in from IRC".format(username))
|
||||||
|
|
||||||
def IRCDisconnect(username):
|
def IRCDisconnect(username):
|
||||||
|
"""
|
||||||
|
Handle IRC logout bancho-side.
|
||||||
|
Remove token and broadcast logout packet.
|
||||||
|
|
||||||
|
:param username: username
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
token = glob.tokens.getTokenFromUsername(username)
|
token = glob.tokens.getTokenFromUsername(username)
|
||||||
if token is None:
|
if token is None:
|
||||||
log.warning("{} doesn't exist".format(username))
|
log.warning("{} doesn't exist".format(username))
|
||||||
|
@ -343,6 +387,13 @@ def IRCDisconnect(username):
|
||||||
log.info("{} disconnected from IRC".format(username))
|
log.info("{} disconnected from IRC".format(username))
|
||||||
|
|
||||||
def IRCJoinChannel(username, channel):
|
def IRCJoinChannel(username, channel):
|
||||||
|
"""
|
||||||
|
Handle IRC channel join bancho-side.
|
||||||
|
|
||||||
|
:param username: username
|
||||||
|
:param channel: channel name
|
||||||
|
:return: IRC return code
|
||||||
|
"""
|
||||||
userID = userUtils.getID(username)
|
userID = userUtils.getID(username)
|
||||||
if not userID:
|
if not userID:
|
||||||
log.warning("{} doesn't exist".format(username))
|
log.warning("{} doesn't exist".format(username))
|
||||||
|
@ -353,8 +404,30 @@ def IRCJoinChannel(username, channel):
|
||||||
return joinChannel(userID, channel)
|
return joinChannel(userID, channel)
|
||||||
|
|
||||||
def IRCPartChannel(username, channel):
|
def IRCPartChannel(username, channel):
|
||||||
|
"""
|
||||||
|
Handle IRC channel part bancho-side.
|
||||||
|
|
||||||
|
:param username: username
|
||||||
|
:param channel: channel name
|
||||||
|
:return: IRC return code
|
||||||
|
"""
|
||||||
userID = userUtils.getID(username)
|
userID = userUtils.getID(username)
|
||||||
if not userID:
|
if not userID:
|
||||||
log.warning("{} doesn't exist".format(username))
|
log.warning("{} doesn't exist".format(username))
|
||||||
return
|
return
|
||||||
return partChannel(userID, channel)
|
return partChannel(userID, channel)
|
||||||
|
|
||||||
|
def IRCAway(username, message):
|
||||||
|
"""
|
||||||
|
Handle IRC away command bancho-side.
|
||||||
|
|
||||||
|
:param username:
|
||||||
|
:param message: away message
|
||||||
|
:return: IRC return code
|
||||||
|
"""
|
||||||
|
userID = userUtils.getID(username)
|
||||||
|
if not userID:
|
||||||
|
log.warning("{} doesn't exist".format(username))
|
||||||
|
return
|
||||||
|
glob.tokens.getTokenFromUserID(userID).awayMessage = message
|
||||||
|
return 305 if message == "" else 306
|
|
@ -5,9 +5,9 @@ class config:
|
||||||
# Check if config.ini exists and load/generate it
|
# Check if config.ini exists and load/generate it
|
||||||
def __init__(self, file):
|
def __init__(self, file):
|
||||||
"""
|
"""
|
||||||
Initialize a config object
|
Initialize a config file object
|
||||||
|
|
||||||
file -- filename
|
:param file: file name
|
||||||
"""
|
"""
|
||||||
self.config = configparser.ConfigParser()
|
self.config = configparser.ConfigParser()
|
||||||
self.default = True
|
self.default = True
|
||||||
|
@ -25,9 +25,9 @@ class config:
|
||||||
# Check if config.ini has all needed the keys
|
# Check if config.ini has all needed the keys
|
||||||
def checkConfig(self):
|
def checkConfig(self):
|
||||||
"""
|
"""
|
||||||
Check if this config has the required keys
|
Check is the config file has all required keys
|
||||||
|
|
||||||
return -- True if valid, False if not
|
:return: True if valid, False if not valid
|
||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
# Try to get all the required keys
|
# Try to get all the required keys
|
||||||
|
@ -37,23 +37,27 @@ class config:
|
||||||
self.config.get("db","database")
|
self.config.get("db","database")
|
||||||
self.config.get("db","workers")
|
self.config.get("db","workers")
|
||||||
|
|
||||||
|
self.config.get("redis","host")
|
||||||
|
self.config.get("redis","port")
|
||||||
|
self.config.get("redis","database")
|
||||||
|
self.config.get("redis","password")
|
||||||
|
|
||||||
self.config.get("server","port")
|
self.config.get("server","port")
|
||||||
self.config.get("server","threads")
|
self.config.get("server","threads")
|
||||||
self.config.get("server","gzip")
|
self.config.get("server","gzip")
|
||||||
self.config.get("server","gziplevel")
|
self.config.get("server","gziplevel")
|
||||||
self.config.get("server","cikey")
|
self.config.get("server","cikey")
|
||||||
self.config.get("server","cloudflare")
|
|
||||||
|
|
||||||
self.config.get("mirror","apiurl")
|
self.config.get("cheesegull", "apiurl")
|
||||||
self.config.get("mirror","apikey")
|
self.config.get("cheesegull", "apikey")
|
||||||
|
|
||||||
self.config.get("debug","enable")
|
self.config.get("debug","enable")
|
||||||
self.config.get("debug","packets")
|
self.config.get("debug","packets")
|
||||||
self.config.get("debug","time")
|
self.config.get("debug","time")
|
||||||
|
|
||||||
self.config.get("sentry","enable")
|
self.config.get("sentry","enable")
|
||||||
self.config.get("sentry","banchodns")
|
self.config.get("sentry","banchodsn")
|
||||||
self.config.get("sentry","ircdns")
|
self.config.get("sentry","ircdsn")
|
||||||
|
|
||||||
self.config.get("discord","enable")
|
self.config.get("discord","enable")
|
||||||
self.config.get("discord","boturl")
|
self.config.get("discord","boturl")
|
||||||
|
@ -69,13 +73,17 @@ class config:
|
||||||
|
|
||||||
self.config.get("localize","enable")
|
self.config.get("localize","enable")
|
||||||
self.config.get("localize","ipapiurl")
|
self.config.get("localize","ipapiurl")
|
||||||
|
|
||||||
|
self.config.get("custom", "config")
|
||||||
return True
|
return True
|
||||||
except:
|
except configparser.Error:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def generateDefaultConfig(self):
|
def generateDefaultConfig(self):
|
||||||
"""
|
"""
|
||||||
Open and set default keys for that config file
|
Write a default config file to disk
|
||||||
|
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Open config.ini in write mode
|
# Open config.ini in write mode
|
||||||
f = open(self.fileName, "w")
|
f = open(self.fileName, "w")
|
||||||
|
@ -88,17 +96,22 @@ class config:
|
||||||
self.config.set("db", "database", "ripple")
|
self.config.set("db", "database", "ripple")
|
||||||
self.config.set("db", "workers", "4")
|
self.config.set("db", "workers", "4")
|
||||||
|
|
||||||
|
self.config.add_section("redis")
|
||||||
|
self.config.set("redis", "host", "localhost")
|
||||||
|
self.config.set("redis", "port", "6379")
|
||||||
|
self.config.set("redis", "database", "0")
|
||||||
|
self.config.set("redis", "password", "")
|
||||||
|
|
||||||
self.config.add_section("server")
|
self.config.add_section("server")
|
||||||
self.config.set("server", "port", "5001")
|
self.config.set("server", "port", "5001")
|
||||||
self.config.set("server", "threads", "16")
|
self.config.set("server", "threads", "16")
|
||||||
self.config.set("server", "gzip", "1")
|
self.config.set("server", "gzip", "1")
|
||||||
self.config.set("server", "gziplevel", "6")
|
self.config.set("server", "gziplevel", "6")
|
||||||
self.config.set("server", "cikey", "changeme")
|
self.config.set("server", "cikey", "changeme")
|
||||||
self.config.set("server", "cloudflare", "0")
|
|
||||||
|
|
||||||
self.config.add_section("mirror")
|
self.config.add_section("cheesegull")
|
||||||
self.config.set("mirror", "apiurl", "http://storage.ripple.moe")
|
self.config.set("cheesegull", "apiurl", "http://cheesegu.ll/api")
|
||||||
self.config.set("mirror", "apikey", "anotherkey")
|
self.config.set("cheesegull", "apikey", "")
|
||||||
|
|
||||||
self.config.add_section("debug")
|
self.config.add_section("debug")
|
||||||
self.config.set("debug", "enable", "0")
|
self.config.set("debug", "enable", "0")
|
||||||
|
@ -107,8 +120,8 @@ class config:
|
||||||
|
|
||||||
self.config.add_section("sentry")
|
self.config.add_section("sentry")
|
||||||
self.config.set("sentry", "enable", "0")
|
self.config.set("sentry", "enable", "0")
|
||||||
self.config.set("sentry", "banchodns", "")
|
self.config.set("sentry", "banchodsn", "")
|
||||||
self.config.set("sentry", "ircdns", "")
|
self.config.set("sentry", "ircdsn", "")
|
||||||
|
|
||||||
self.config.add_section("discord")
|
self.config.add_section("discord")
|
||||||
self.config.set("discord", "enable", "0")
|
self.config.set("discord", "enable", "0")
|
||||||
|
@ -116,9 +129,9 @@ class config:
|
||||||
self.config.set("discord", "devgroup", "")
|
self.config.set("discord", "devgroup", "")
|
||||||
|
|
||||||
self.config.add_section("datadog")
|
self.config.add_section("datadog")
|
||||||
self.config.set("datadog", "enable")
|
self.config.set("datadog", "enable", "0")
|
||||||
self.config.set("datadog", "apikey")
|
self.config.set("datadog", "apikey", "")
|
||||||
self.config.set("datadog", "appkey")
|
self.config.set("datadog", "appkey", "")
|
||||||
|
|
||||||
self.config.add_section("irc")
|
self.config.add_section("irc")
|
||||||
self.config.set("irc", "enable", "1")
|
self.config.set("irc", "enable", "1")
|
||||||
|
@ -129,6 +142,9 @@ class config:
|
||||||
self.config.set("localize", "enable", "1")
|
self.config.set("localize", "enable", "1")
|
||||||
self.config.set("localize", "ipapiurl", "http://ip.zxq.co")
|
self.config.set("localize", "ipapiurl", "http://ip.zxq.co")
|
||||||
|
|
||||||
|
self.config.add_section("custom")
|
||||||
|
self.config.set("custom", "config", "common/config.json")
|
||||||
|
|
||||||
# Write ini to file and close
|
# Write ini to file and close
|
||||||
self.config.write(f)
|
self.config.write(f)
|
||||||
f.close()
|
f.close()
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
from common.constants import bcolors
|
from common.constants import bcolors
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
def printServerStartHeader(asciiArt):
|
def printServerStartHeader(asciiArt=True):
|
||||||
"""
|
"""
|
||||||
Print server start header with optional ascii art
|
Print server start message
|
||||||
|
|
||||||
asciiArt -- if True, will print ascii art too
|
:param asciiArt: print BanchoBoat ascii art. Default: True
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
if asciiArt:
|
if asciiArt:
|
||||||
print("{} _ __".format(bcolors.GREEN))
|
print("{} _ __".format(bcolors.GREEN))
|
||||||
|
@ -26,41 +27,52 @@ def printServerStartHeader(asciiArt):
|
||||||
print("^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^{}".format(bcolors.ENDC))
|
print("^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^{}".format(bcolors.ENDC))
|
||||||
|
|
||||||
printColored("> Welcome to pep.py osu!bancho server v{}".format(glob.VERSION), bcolors.GREEN)
|
printColored("> Welcome to pep.py osu!bancho server v{}".format(glob.VERSION), bcolors.GREEN)
|
||||||
|
printColored("> Common submodule v{}".format(glob.COMMON_VERSION), bcolors.GREEN)
|
||||||
printColored("> Made by the Ripple team", bcolors.GREEN)
|
printColored("> Made by the Ripple team", bcolors.GREEN)
|
||||||
printColored("> {}https://git.zxq.co/ripple/pep.py".format(bcolors.UNDERLINE), bcolors.GREEN)
|
printColored("> {}https://zxq.co/ripple/pep.py".format(bcolors.UNDERLINE), bcolors.GREEN)
|
||||||
|
printColored("> Custom branch by the osufx team (just Sunpy)", bcolors.GREEN)
|
||||||
|
printColored("> {}https://github.com/osufx/pep.py".format(bcolors.UNDERLINE), bcolors.GREEN)
|
||||||
printColored("> Press CTRL+C to exit\n", bcolors.GREEN)
|
printColored("> Press CTRL+C to exit\n", bcolors.GREEN)
|
||||||
|
|
||||||
def printNoNl(string):
|
def printNoNl(string):
|
||||||
"""
|
"""
|
||||||
Print string without new line at the end
|
Print a string without \n at the end
|
||||||
|
|
||||||
string -- string to print
|
:param string: string to print
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
print(string, end="")
|
print(string, end="")
|
||||||
|
|
||||||
def printColored(string, color):
|
def printColored(string, color):
|
||||||
"""
|
"""
|
||||||
Print colored string
|
Print a colored string
|
||||||
|
|
||||||
string -- string to print
|
:param string: string to print
|
||||||
color -- see bcolors.py
|
:param color: ANSI color code
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
print("{}{}{}".format(color, string, bcolors.ENDC))
|
print("{}{}{}".format(color, string, bcolors.ENDC))
|
||||||
|
|
||||||
def printError():
|
def printError():
|
||||||
"""
|
"""
|
||||||
Print error text FOR LOADING
|
Print a red "Error"
|
||||||
|
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
printColored("Error", bcolors.RED)
|
printColored("Error", bcolors.RED)
|
||||||
|
|
||||||
def printDone():
|
def printDone():
|
||||||
"""
|
"""
|
||||||
Print error text FOR LOADING
|
Print a green "Done"
|
||||||
|
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
printColored("Done", bcolors.GREEN)
|
printColored("Done", bcolors.GREEN)
|
||||||
|
|
||||||
def printWarning():
|
def printWarning():
|
||||||
"""
|
"""
|
||||||
Print error text FOR LOADING
|
Print a yellow "Warning"
|
||||||
|
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
printColored("Warning", bcolors.YELLOW)
|
printColored("Warning", bcolors.YELLOW)
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
"""Contains all country codes with their osu numeric code"""
|
# TODO: Update countries list
|
||||||
|
|
||||||
countryCodes = {
|
countryCodes = {
|
||||||
"LV": 132,
|
"LV": 132,
|
||||||
"AD": 3,
|
"AD": 3,
|
||||||
|
@ -255,12 +254,11 @@ countryCodes = {
|
||||||
|
|
||||||
def getCountryID(code):
|
def getCountryID(code):
|
||||||
"""
|
"""
|
||||||
Get country ID for osu client
|
Get osu country ID from country letters
|
||||||
|
|
||||||
code -- country name abbreviation (eg: US)
|
:param code: country letters (eg: US)
|
||||||
return -- country code int
|
:return: country osu code
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if code in countryCodes:
|
if code in countryCodes:
|
||||||
return countryCodes[code]
|
return countryCodes[code]
|
||||||
else:
|
else:
|
||||||
|
@ -270,10 +268,9 @@ def getCountryLetters(code):
|
||||||
"""
|
"""
|
||||||
Get country letters from osu country ID
|
Get country letters from osu country ID
|
||||||
|
|
||||||
code -- country code int
|
:param code: osu country ID
|
||||||
return -- country name (2 letters) (XX if code not found)
|
:return: country letters (XX if not found)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
for key, value in countryCodes.items():
|
for key, value in countryCodes.items():
|
||||||
if value == code:
|
if value == code:
|
||||||
return key
|
return key
|
||||||
|
|
|
@ -7,10 +7,10 @@ from objects import glob
|
||||||
|
|
||||||
def getCountry(ip):
|
def getCountry(ip):
|
||||||
"""
|
"""
|
||||||
Get country from IP address
|
Get country from IP address using geoip api
|
||||||
|
|
||||||
ip -- IP Address
|
:param ip: IP address
|
||||||
return -- Country code (2 letters)
|
:return: country code. XX if invalid.
|
||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
# Try to get country from Pikolo Aul's Go-Sanic ip API
|
# Try to get country from Pikolo Aul's Go-Sanic ip API
|
||||||
|
@ -22,15 +22,15 @@ def getCountry(ip):
|
||||||
|
|
||||||
def getLocation(ip):
|
def getLocation(ip):
|
||||||
"""
|
"""
|
||||||
Get latitude and longitude from IP address
|
Get latitude and longitude from IP address using geoip api
|
||||||
|
|
||||||
ip -- IP address
|
:param ip: IP address
|
||||||
return -- [latitude, longitude]
|
:return: (latitude, longitude)
|
||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
# Try to get position from Pikolo Aul's Go-Sanic ip API
|
# Try to get position from Pikolo Aul's Go-Sanic ip API
|
||||||
result = json.loads(urllib.request.urlopen("{}/{}".format(glob.conf.config["localize"]["ipapiurl"], ip), timeout=3).read().decode())["loc"].split(",")
|
result = json.loads(urllib.request.urlopen("{}/{}".format(glob.conf.config["localize"]["ipapiurl"], ip), timeout=3).read().decode())["loc"].split(",")
|
||||||
return [float(result[0]), float(result[1])]
|
return float(result[0]), float(result[1])
|
||||||
except:
|
except:
|
||||||
log.error("Error in get position")
|
log.error("Error in get position")
|
||||||
return [0,0]
|
return 0, 0
|
||||||
|
|
|
@ -1,15 +1,15 @@
|
||||||
import struct
|
import struct
|
||||||
from constants import dataTypes
|
from constants import dataTypes
|
||||||
|
|
||||||
def uleb128Encode(num):
|
cpdef bytearray uleb128Encode(int num):
|
||||||
"""
|
"""
|
||||||
Encode int -> uleb128
|
Encode an int to uleb128
|
||||||
|
|
||||||
num -- int to encode
|
:param num: int to encode
|
||||||
return -- bytearray with encoded number
|
:return: bytearray with encoded number
|
||||||
"""
|
"""
|
||||||
arr = bytearray()
|
cdef bytearray arr = bytearray()
|
||||||
length = 0
|
cdef int length = 0
|
||||||
|
|
||||||
if num == 0:
|
if num == 0:
|
||||||
return bytearray(b"\x00")
|
return bytearray(b"\x00")
|
||||||
|
@ -23,15 +23,16 @@ def uleb128Encode(num):
|
||||||
|
|
||||||
return arr
|
return arr
|
||||||
|
|
||||||
def uleb128Decode(num):
|
cpdef list uleb128Decode(bytes num):
|
||||||
"""
|
"""
|
||||||
Decode uleb128 -> int
|
Decode a uleb128 to int
|
||||||
|
|
||||||
num -- encoded uleb128
|
:param num: encoded uleb128 int
|
||||||
return -- list. [total, length]
|
:return: (total, length)
|
||||||
"""
|
"""
|
||||||
shift = 0
|
cdef int shift = 0
|
||||||
arr = [0,0] #total, length
|
cdef list arr = [0,0] #total, length
|
||||||
|
cdef int b
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
b = num[arr[1]]
|
b = num[arr[1]]
|
||||||
|
@ -43,16 +44,14 @@ def uleb128Decode(num):
|
||||||
|
|
||||||
return arr
|
return arr
|
||||||
|
|
||||||
def unpackData(data, dataType):
|
cpdef unpackData(bytes data, int dataType):
|
||||||
"""
|
"""
|
||||||
Unpacks data according to dataType
|
Unpacks a single section of a packet.
|
||||||
|
|
||||||
data -- bytes array to unpack
|
:param data: bytes to unpack
|
||||||
dataType -- data type. See dataTypes.py
|
:param dataType: data type
|
||||||
|
:return: unpacked bytes
|
||||||
return -- unpacked bytes
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Get right pack Type
|
# Get right pack Type
|
||||||
if dataType == dataTypes.UINT16:
|
if dataType == dataTypes.UINT16:
|
||||||
unpackType = "<H"
|
unpackType = "<H"
|
||||||
|
@ -76,18 +75,17 @@ def unpackData(data, dataType):
|
||||||
# Unpack
|
# Unpack
|
||||||
return struct.unpack(unpackType, bytes(data))[0]
|
return struct.unpack(unpackType, bytes(data))[0]
|
||||||
|
|
||||||
def packData(__data, dataType):
|
cpdef bytes packData(__data, int dataType):
|
||||||
"""
|
"""
|
||||||
Packs data according to dataType
|
Packs a single section of a packet.
|
||||||
|
|
||||||
data -- bytes to pack
|
:param __data: data to pack
|
||||||
dataType -- data type. See dataTypes.py
|
:param dataType: data type
|
||||||
|
:return: packed bytes
|
||||||
return -- packed bytes
|
|
||||||
"""
|
"""
|
||||||
|
cdef bytes data = bytes() # data to return
|
||||||
data = bytes() # data to return
|
cdef bint pack = True # if True, use pack. False only with strings
|
||||||
pack = True # if True, use pack. False only with strings
|
cdef str packType
|
||||||
|
|
||||||
# Get right pack Type
|
# Get right pack Type
|
||||||
if dataType == dataTypes.BBYTES:
|
if dataType == dataTypes.BBYTES:
|
||||||
|
@ -138,23 +136,24 @@ def packData(__data, dataType):
|
||||||
|
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def buildPacket(__packet, __packetData=None):
|
cpdef bytes buildPacket(int __packet, list __packetData = None):
|
||||||
"""
|
"""
|
||||||
Build a packet
|
Builds a packet
|
||||||
|
|
||||||
packet -- packet id (int)
|
:param __packet: packet ID
|
||||||
packetData -- list [[data, dataType], [data, dataType], ...]
|
:param __packetData: packet structure [[data, dataType], [data, dataType], ...]
|
||||||
|
:return: packet bytes
|
||||||
return -- packet bytes
|
|
||||||
"""
|
"""
|
||||||
# Set some variables
|
# Default argument
|
||||||
if __packetData is None:
|
if __packetData is None:
|
||||||
__packetData = []
|
__packetData = []
|
||||||
packetData = bytes()
|
# Set some variables
|
||||||
packetLength = 0
|
cdef bytes packetData = bytes()
|
||||||
packetBytes = bytes()
|
cdef int packetLength = 0
|
||||||
|
cdef bytes packetBytes = bytes()
|
||||||
|
|
||||||
# Pack packet data
|
# Pack packet data
|
||||||
|
cdef list i
|
||||||
for i in __packetData:
|
for i in __packetData:
|
||||||
packetData += packData(i[0], i[1])
|
packetData += packData(i[0], i[1])
|
||||||
|
|
||||||
|
@ -168,42 +167,43 @@ def buildPacket(__packet, __packetData=None):
|
||||||
packetBytes += packetData # packet data
|
packetBytes += packetData # packet data
|
||||||
return packetBytes
|
return packetBytes
|
||||||
|
|
||||||
def readPacketID(stream):
|
cpdef int readPacketID(bytes stream):
|
||||||
"""
|
"""
|
||||||
Read packetID from stream (0-1 bytes)
|
Read packetID (first two bytes) from a packet
|
||||||
|
|
||||||
stream -- data stream
|
:param stream: packet bytes
|
||||||
return -- packet ID (int)
|
:return: packet ID
|
||||||
"""
|
"""
|
||||||
return unpackData(stream[0:2], dataTypes.UINT16)
|
return unpackData(stream[0:2], dataTypes.UINT16)
|
||||||
|
|
||||||
def readPacketLength(stream):
|
cpdef int readPacketLength(bytes stream):
|
||||||
"""
|
"""
|
||||||
Read packet length from stream (3-4-5-6 bytes)
|
Read packet data length (3:7 bytes) from a packet
|
||||||
|
|
||||||
stream -- data stream
|
:param stream: packet bytes
|
||||||
return -- packet length (int)
|
:return: packet data length
|
||||||
"""
|
"""
|
||||||
return unpackData(stream[3:7], dataTypes.UINT32)
|
return unpackData(stream[3:7], dataTypes.UINT32)
|
||||||
|
|
||||||
|
|
||||||
def readPacketData(stream, structure=None, hasFirstBytes = True):
|
cpdef readPacketData(bytes stream, list structure=None, bint hasFirstBytes = True):
|
||||||
"""
|
"""
|
||||||
Read packet data from stream according to structure
|
Read packet data from `stream` according to `structure`
|
||||||
|
:param stream: packet bytes
|
||||||
stream -- data stream
|
:param structure: packet structure: [[name, dataType], [name, dataType], ...]
|
||||||
structure -- [[name, dataType], [name, dataType], ...]
|
:param hasFirstBytes: if True, `stream` has packetID and length bytes.
|
||||||
hasFirstBytes -- if True, stream has packetID and length bytes.
|
if False, `stream` has only packet data. Default: True
|
||||||
if False, stream has only packetData.
|
:return: {name: unpackedValue, ...}
|
||||||
Optional. Default: True
|
|
||||||
return -- dictionary. key: name, value: read data
|
|
||||||
"""
|
"""
|
||||||
# Read packet ID (first 2 bytes)
|
# Default list argument
|
||||||
if structure is None:
|
if structure is None:
|
||||||
structure = []
|
structure = []
|
||||||
data = {}
|
|
||||||
|
# Read packet ID (first 2 bytes)
|
||||||
|
cdef dict data = {}
|
||||||
|
|
||||||
# Skip packet ID and packet length if needed
|
# Skip packet ID and packet length if needed
|
||||||
|
cdef start, end
|
||||||
if hasFirstBytes:
|
if hasFirstBytes:
|
||||||
end = 7
|
end = 7
|
||||||
start = 7
|
start = 7
|
||||||
|
@ -212,6 +212,8 @@ def readPacketData(stream, structure=None, hasFirstBytes = True):
|
||||||
start = 0
|
start = 0
|
||||||
|
|
||||||
# Read packet
|
# Read packet
|
||||||
|
cdef list i
|
||||||
|
cdef bint unpack
|
||||||
for i in structure:
|
for i in structure:
|
||||||
start = end
|
start = end
|
||||||
unpack = True
|
unpack = True
|
||||||
|
@ -246,7 +248,10 @@ def readPacketData(stream, structure=None, hasFirstBytes = True):
|
||||||
end = start+length[0]+length[1]+1
|
end = start+length[0]+length[1]+1
|
||||||
|
|
||||||
# Read bytes
|
# Read bytes
|
||||||
data[i[0]] = ''.join(chr(j) for j in stream[start+1+length[1]:end])
|
#data[i[0]] = ''.join(chr(j) for j in stream[start+1+length[1]:end])
|
||||||
|
data[i[0]] = ""
|
||||||
|
for j in stream[start+1+length[1]:end]:
|
||||||
|
data[i[0]] += chr(j)
|
||||||
elif i[1] == dataTypes.BYTE:
|
elif i[1] == dataTypes.BYTE:
|
||||||
end = start+1
|
end = start+1
|
||||||
elif i[1] == dataTypes.UINT16 or i[1] == dataTypes.SINT16:
|
elif i[1] == dataTypes.UINT16 or i[1] == dataTypes.SINT16:
|
|
@ -17,6 +17,7 @@ from objects import glob
|
||||||
def dispose():
|
def dispose():
|
||||||
"""
|
"""
|
||||||
Perform some clean up. Called on shutdown.
|
Perform some clean up. Called on shutdown.
|
||||||
|
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
print("> Disposing server... ")
|
print("> Disposing server... ")
|
||||||
|
@ -27,7 +28,7 @@ def runningUnderUnix():
|
||||||
"""
|
"""
|
||||||
Get if the server is running under UNIX or NT
|
Get if the server is running under UNIX or NT
|
||||||
|
|
||||||
return --- True if running under UNIX, otherwise False
|
:return: True if running under UNIX, otherwise False
|
||||||
"""
|
"""
|
||||||
return True if os.name == "posix" else False
|
return True if os.name == "posix" else False
|
||||||
|
|
||||||
|
@ -35,12 +36,14 @@ def scheduleShutdown(sendRestartTime, restart, message = "", delay=20):
|
||||||
"""
|
"""
|
||||||
Schedule a server shutdown/restart
|
Schedule a server shutdown/restart
|
||||||
|
|
||||||
sendRestartTime -- time (seconds) to wait before sending server restart packets to every client
|
:param sendRestartTime: time (seconds) to wait before sending server restart packets to every client
|
||||||
restart -- if True, server will restart. if False, server will shudown
|
:param restart: if True, server will restart. if False, server will shudown
|
||||||
message -- if set, send that message to every client to warn about the shutdown/restart
|
:param message: if set, send that message to every client to warn about the shutdown/restart
|
||||||
|
:param delay: additional restart delay in seconds. Default: 20
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Console output
|
# Console output
|
||||||
log.info("Pep.py will {} in {} seconds!".format("restart" if restart else "shutdown", sendRestartTime+delay))
|
log.info("Pep.py will {} in {} seconds!".format("restart" if restart else "shutdown", sendRestartTime+delay), "bunker")
|
||||||
log.info("Sending server restart packets in {} seconds...".format(sendRestartTime))
|
log.info("Sending server restart packets in {} seconds...".format(sendRestartTime))
|
||||||
|
|
||||||
# Send notification if set
|
# Send notification if set
|
||||||
|
@ -61,13 +64,21 @@ def scheduleShutdown(sendRestartTime, restart, message = "", delay=20):
|
||||||
threading.Timer(sendRestartTime+delay, action).start()
|
threading.Timer(sendRestartTime+delay, action).start()
|
||||||
|
|
||||||
def restartServer():
|
def restartServer():
|
||||||
"""Restart pep.py script"""
|
"""
|
||||||
|
Restart pep.py
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
log.info("Restarting pep.py...")
|
log.info("Restarting pep.py...")
|
||||||
dispose()
|
dispose()
|
||||||
os.execv(sys.executable, [sys.executable] + sys.argv)
|
os.execv(sys.executable, [sys.executable] + sys.argv)
|
||||||
|
|
||||||
def shutdownServer():
|
def shutdownServer():
|
||||||
"""Shutdown pep.py"""
|
"""
|
||||||
|
Shutdown pep.py
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
log.info("Shutting down pep.py...")
|
log.info("Shutting down pep.py...")
|
||||||
dispose()
|
dispose()
|
||||||
sig = signal.SIGKILL if runningUnderUnix() else signal.CTRL_C_EVENT
|
sig = signal.SIGKILL if runningUnderUnix() else signal.CTRL_C_EVENT
|
||||||
|
@ -77,7 +88,7 @@ def getSystemInfo():
|
||||||
"""
|
"""
|
||||||
Get a dictionary with some system/server info
|
Get a dictionary with some system/server info
|
||||||
|
|
||||||
return -- ["unix", "connectedUsers", "webServer", "cpuUsage", "totalMemory", "usedMemory", "loadAverage"]
|
:return: ["unix", "connectedUsers", "webServer", "cpuUsage", "totalMemory", "usedMemory", "loadAverage"]
|
||||||
"""
|
"""
|
||||||
data = {"unix": runningUnderUnix(), "connectedUsers": len(glob.tokens.tokens), "matches": len(glob.matches.matches)}
|
data = {"unix": runningUnderUnix(), "connectedUsers": len(glob.tokens.tokens), "matches": len(glob.matches.matches)}
|
||||||
|
|
||||||
|
|
178
irc/ircserver.py
178
irc/ircserver.py
|
@ -17,23 +17,21 @@ import traceback
|
||||||
import raven
|
import raven
|
||||||
|
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
|
from common.ripple import userUtils
|
||||||
from helpers import chatHelper as chat
|
from helpers import chatHelper as chat
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
|
|
||||||
class Client:
|
class Client:
|
||||||
"""
|
|
||||||
IRC Client object
|
|
||||||
"""
|
|
||||||
__linesep_regexp = re.compile(r"\r?\n")
|
__linesep_regexp = re.compile(r"\r?\n")
|
||||||
|
|
||||||
|
|
||||||
def __init__(self, server, sock):
|
def __init__(self, server, sock):
|
||||||
"""
|
"""
|
||||||
Initialize a Client object
|
Initialize a Client object
|
||||||
|
|
||||||
server -- server object
|
:param server: server object
|
||||||
sock -- socket connection object
|
:param sock: socket connection object
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
self.__timestamp = time.time()
|
self.__timestamp = time.time()
|
||||||
self.__readbuffer = ""
|
self.__readbuffer = ""
|
||||||
|
@ -47,6 +45,7 @@ class Client:
|
||||||
self.IRCUsername = ""
|
self.IRCUsername = ""
|
||||||
self.banchoUsername = ""
|
self.banchoUsername = ""
|
||||||
self.supposedUsername = ""
|
self.supposedUsername = ""
|
||||||
|
self.supposedUserID = 0
|
||||||
self.joinedChannels = []
|
self.joinedChannels = []
|
||||||
|
|
||||||
def messageChannel(self, channel, command, message, includeSelf=False):
|
def messageChannel(self, channel, command, message, includeSelf=False):
|
||||||
|
@ -60,7 +59,8 @@ class Client:
|
||||||
Add a message (basic string) to client buffer.
|
Add a message (basic string) to client buffer.
|
||||||
This is the lowest possible level.
|
This is the lowest possible level.
|
||||||
|
|
||||||
msg -- message to add
|
:param msg: message to add
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
self.__writebuffer += msg + "\r\n"
|
self.__writebuffer += msg + "\r\n"
|
||||||
|
|
||||||
|
@ -69,7 +69,7 @@ class Client:
|
||||||
"""
|
"""
|
||||||
Return this client's write buffer size
|
Return this client's write buffer size
|
||||||
|
|
||||||
return -- write buffer size
|
:return: write buffer size
|
||||||
"""
|
"""
|
||||||
return len(self.__writebuffer)
|
return len(self.__writebuffer)
|
||||||
|
|
||||||
|
@ -78,7 +78,8 @@ class Client:
|
||||||
"""
|
"""
|
||||||
Add an IRC-like message to client buffer.
|
Add an IRC-like message to client buffer.
|
||||||
|
|
||||||
msg -- message (without IRC stuff)
|
:param msg: message (without IRC stuff)
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
self.message(":{} {}".format(self.server.host, msg))
|
self.message(":{} {}".format(self.server.host, msg))
|
||||||
|
|
||||||
|
@ -87,10 +88,11 @@ class Client:
|
||||||
"""
|
"""
|
||||||
Add an IRC-like message to client buffer with code
|
Add an IRC-like message to client buffer with code
|
||||||
|
|
||||||
code -- response code
|
:param code: response code
|
||||||
message -- response message
|
:param message: response message
|
||||||
nickname -- receiver nickname
|
:param nickname: receiver nickname
|
||||||
channel -- optional
|
:param channel: optional
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
if nickname == "":
|
if nickname == "":
|
||||||
nickname = self.IRCUsername
|
nickname = self.IRCUsername
|
||||||
|
@ -103,7 +105,8 @@ class Client:
|
||||||
"""
|
"""
|
||||||
Add a 403 reply (no such channel) to client buffer.
|
Add a 403 reply (no such channel) to client buffer.
|
||||||
|
|
||||||
channel -- meh
|
:param channel:
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
self.replyCode(403, "{} :No such channel".format(channel))
|
self.replyCode(403, "{} :No such channel".format(channel))
|
||||||
|
|
||||||
|
@ -112,7 +115,8 @@ class Client:
|
||||||
"""
|
"""
|
||||||
Add a 461 reply (not enough parameters) to client buffer
|
Add a 461 reply (not enough parameters) to client buffer
|
||||||
|
|
||||||
command -- command that had not enough parameters
|
:param command: name of the command that had not enough parameters
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
self.replyCode(403, "{} :Not enough parameters".format(command))
|
self.replyCode(403, "{} :Not enough parameters".format(command))
|
||||||
|
|
||||||
|
@ -121,8 +125,9 @@ class Client:
|
||||||
"""
|
"""
|
||||||
Disconnects this client from the IRC server
|
Disconnects this client from the IRC server
|
||||||
|
|
||||||
quitmsg -- IRC quit message. Default: 'Client quit'
|
:param quitmsg: IRC quit message. Default: 'Client quit'
|
||||||
callLogout -- if True, call logoutEvent on bancho
|
:param callLogout: if True, call logoutEvent on bancho
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Send error to client and close socket
|
# Send error to client and close socket
|
||||||
self.message("ERROR :{}".format(quitmsg))
|
self.message("ERROR :{}".format(quitmsg))
|
||||||
|
@ -133,12 +138,16 @@ class Client:
|
||||||
self.server.removeClient(self, quitmsg)
|
self.server.removeClient(self, quitmsg)
|
||||||
|
|
||||||
# Bancho logout
|
# Bancho logout
|
||||||
if callLogout:
|
if callLogout and self.banchoUsername != "":
|
||||||
chat.IRCDisconnect(self.IRCUsername)
|
chat.IRCDisconnect(self.IRCUsername)
|
||||||
|
|
||||||
|
|
||||||
def readSocket(self):
|
def readSocket(self):
|
||||||
"""Read data coming from this client socket"""
|
"""
|
||||||
|
Read data coming from this client socket
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
try:
|
try:
|
||||||
# Try to read incoming data from socket
|
# Try to read incoming data from socket
|
||||||
data = self.socket.recv(2 ** 10)
|
data = self.socket.recv(2 ** 10)
|
||||||
|
@ -161,7 +170,11 @@ class Client:
|
||||||
|
|
||||||
|
|
||||||
def parseBuffer(self):
|
def parseBuffer(self):
|
||||||
"""Parse self.__readbuffer, get command, arguments and call its handler"""
|
"""
|
||||||
|
Parse self.__readbuffer, get command, arguments and call its handler
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
# Get lines from buffer
|
# Get lines from buffer
|
||||||
lines = self.__linesep_regexp.split(self.__readbuffer)
|
lines = self.__linesep_regexp.split(self.__readbuffer)
|
||||||
self.__readbuffer = lines[-1]
|
self.__readbuffer = lines[-1]
|
||||||
|
@ -198,7 +211,11 @@ class Client:
|
||||||
|
|
||||||
|
|
||||||
def writeSocket(self):
|
def writeSocket(self):
|
||||||
"""Write buffer to socket"""
|
"""
|
||||||
|
Write buffer to socket
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
try:
|
try:
|
||||||
sent = self.socket.send(self.__writebuffer.encode())
|
sent = self.socket.send(self.__writebuffer.encode())
|
||||||
log.debug("[IRC] [{}:{}] <- {}".format(self.ip, self.port, self.__writebuffer[:sent]))
|
log.debug("[IRC] [{}:{}] <- {}".format(self.ip, self.port, self.__writebuffer[:sent]))
|
||||||
|
@ -206,9 +223,13 @@ class Client:
|
||||||
except socket.error as x:
|
except socket.error as x:
|
||||||
self.disconnect(str(x))
|
self.disconnect(str(x))
|
||||||
|
|
||||||
|
|
||||||
def checkAlive(self):
|
def checkAlive(self):
|
||||||
"""Check if this client is still connected"""
|
"""
|
||||||
|
Check if this client is still connected.
|
||||||
|
If the client is dead, disconnect it.
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
now = time.time()
|
now = time.time()
|
||||||
if self.__timestamp + 180 < now:
|
if self.__timestamp + 180 < now:
|
||||||
self.disconnect("ping timeout")
|
self.disconnect("ping timeout")
|
||||||
|
@ -224,11 +245,19 @@ class Client:
|
||||||
|
|
||||||
|
|
||||||
def sendLusers(self):
|
def sendLusers(self):
|
||||||
"""Send lusers response to this client"""
|
"""
|
||||||
|
Send lusers response to this client
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
self.replyCode(251, "There are {} users and 0 services on 1 server".format(len(glob.tokens.tokens)))
|
self.replyCode(251, "There are {} users and 0 services on 1 server".format(len(glob.tokens.tokens)))
|
||||||
|
|
||||||
def sendMotd(self):
|
def sendMotd(self):
|
||||||
"""Send MOTD to this client"""
|
"""
|
||||||
|
Send MOTD to this client
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
self.replyCode(375, "- {} Message of the day - ".format(self.server.host))
|
self.replyCode(375, "- {} Message of the day - ".format(self.server.host))
|
||||||
if len(self.server.motd) == 0:
|
if len(self.server.motd) == 0:
|
||||||
self.replyCode(422, "MOTD File is missing")
|
self.replyCode(422, "MOTD File is missing")
|
||||||
|
@ -253,9 +282,10 @@ class Client:
|
||||||
m = hashlib.md5()
|
m = hashlib.md5()
|
||||||
m.update(arguments[0].encode("utf-8"))
|
m.update(arguments[0].encode("utf-8"))
|
||||||
tokenHash = m.hexdigest()
|
tokenHash = m.hexdigest()
|
||||||
supposedUsername = glob.db.fetch("SELECT users.username FROM users LEFT JOIN irc_tokens ON users.id = irc_tokens.userid WHERE irc_tokens.token = %s LIMIT 1", [tokenHash])
|
supposedUser = glob.db.fetch("SELECT users.username, users.id FROM users LEFT JOIN irc_tokens ON users.id = irc_tokens.userid WHERE irc_tokens.token = %s LIMIT 1", [tokenHash])
|
||||||
if supposedUsername:
|
if supposedUser:
|
||||||
self.supposedUsername = chat.fixUsernameForIRC(supposedUsername["username"])
|
self.supposedUsername = chat.fixUsernameForIRC(supposedUser["username"])
|
||||||
|
self.supposedUserID = supposedUser["id"]
|
||||||
self.__handleCommand = self.registerHandler
|
self.__handleCommand = self.registerHandler
|
||||||
else:
|
else:
|
||||||
# Wrong IRC Token
|
# Wrong IRC Token
|
||||||
|
@ -283,6 +313,11 @@ class Client:
|
||||||
self.reply("464 :Password incorrect")
|
self.reply("464 :Password incorrect")
|
||||||
return
|
return
|
||||||
|
|
||||||
|
# Make sure that the user is not banned/restricted:
|
||||||
|
if not userUtils.isAllowed(self.supposedUserID):
|
||||||
|
self.reply("465 :You're banned")
|
||||||
|
return
|
||||||
|
|
||||||
# Make sure we are not connected to Bancho
|
# Make sure we are not connected to Bancho
|
||||||
token = glob.tokens.getTokenFromUsername(chat.fixUsernameForBancho(nick), True)
|
token = glob.tokens.getTokenFromUsername(chat.fixUsernameForBancho(nick), True)
|
||||||
if token is not None:
|
if token is not None:
|
||||||
|
@ -296,7 +331,6 @@ class Client:
|
||||||
# Disconnect other IRC clients from the same user
|
# Disconnect other IRC clients from the same user
|
||||||
for _, value in self.server.clients.items():
|
for _, value in self.server.clients.items():
|
||||||
if value.IRCUsername.lower() == self.IRCUsername.lower() and value != self:
|
if value.IRCUsername.lower() == self.IRCUsername.lower() and value != self:
|
||||||
print("DISCONNECTERINOOOOOOOOOOOOOOOOOOOOO")
|
|
||||||
value.disconnect(quitmsg="Connected from another client")
|
value.disconnect(quitmsg="Connected from another client")
|
||||||
return
|
return
|
||||||
elif command == "USER":
|
elif command == "USER":
|
||||||
|
@ -324,11 +358,11 @@ class Client:
|
||||||
self.sendMotd()
|
self.sendMotd()
|
||||||
self.__handleCommand = self.mainHandler
|
self.__handleCommand = self.mainHandler
|
||||||
|
|
||||||
def quitHandler(self, command, arguments):
|
def quitHandler(self, _, arguments):
|
||||||
"""QUIT command handler"""
|
"""QUIT command handler"""
|
||||||
self.disconnect(self.IRCUsername if len(arguments) < 1 else arguments[0])
|
self.disconnect(self.IRCUsername if len(arguments) < 1 else arguments[0])
|
||||||
|
|
||||||
def joinHandler(self, command, arguments):
|
def joinHandler(self, _, arguments):
|
||||||
"""JOIN command handler"""
|
"""JOIN command handler"""
|
||||||
if len(arguments) < 1:
|
if len(arguments) < 1:
|
||||||
self.reply461("JOIN")
|
self.reply461("JOIN")
|
||||||
|
@ -341,13 +375,13 @@ class Client:
|
||||||
|
|
||||||
# TODO: Part all channels
|
# TODO: Part all channels
|
||||||
if arguments[0] == "0":
|
if arguments[0] == "0":
|
||||||
return
|
|
||||||
'''for (channelname, channel) in self.channels.items():
|
'''for (channelname, channel) in self.channels.items():
|
||||||
self.message_channel(channel, "PART", channelname, True)
|
self.message_channel(channel, "PART", channelname, True)
|
||||||
self.channel_log(channel, "left", meta=True)
|
self.channel_log(channel, "left", meta=True)
|
||||||
server.remove_member_from_channel(self, channelname)
|
server.remove_member_from_channel(self, channelname)
|
||||||
self.channels = {}
|
self.channels = {}
|
||||||
return'''
|
return'''
|
||||||
|
return
|
||||||
|
|
||||||
# Get channels to join list
|
# Get channels to join list
|
||||||
channels = arguments[0].split(",")
|
channels = arguments[0].split(",")
|
||||||
|
@ -376,13 +410,15 @@ class Client:
|
||||||
self.replyCode(332, description, channel=channel)
|
self.replyCode(332, description, channel=channel)
|
||||||
|
|
||||||
# Build connected users list
|
# Build connected users list
|
||||||
users = glob.channels.channels[channel].getConnectedUsers()[:]
|
if "chat/{}".format(channel) not in glob.streams.streams:
|
||||||
|
self.reply403(channel)
|
||||||
|
continue
|
||||||
|
users = glob.streams.streams["chat/{}".format(channel)].clients
|
||||||
usernames = []
|
usernames = []
|
||||||
for user in users:
|
for user in users:
|
||||||
token = glob.tokens.getTokenFromUserID(user)
|
if user not in glob.tokens.tokens:
|
||||||
if token is None:
|
|
||||||
continue
|
continue
|
||||||
usernames.append(chat.fixUsernameForIRC(token.username))
|
usernames.append(chat.fixUsernameForIRC(glob.tokens.tokens[user].username))
|
||||||
usernames = " ".join(usernames)
|
usernames = " ".join(usernames)
|
||||||
|
|
||||||
# Send IRC users list
|
# Send IRC users list
|
||||||
|
@ -393,7 +429,7 @@ class Client:
|
||||||
self.reply403(channel)
|
self.reply403(channel)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
def partHandler(self, command, arguments):
|
def partHandler(self, _, arguments):
|
||||||
"""PART command handler"""
|
"""PART command handler"""
|
||||||
if len(arguments) < 1:
|
if len(arguments) < 1:
|
||||||
self.reply461("PART")
|
self.reply461("PART")
|
||||||
|
@ -440,7 +476,6 @@ class Client:
|
||||||
|
|
||||||
# Send the message to bancho and reply
|
# Send the message to bancho and reply
|
||||||
if not recipientIRC.startswith("#"):
|
if not recipientIRC.startswith("#"):
|
||||||
print("PMPMPM!!!!!!!!!!")
|
|
||||||
recipientBancho = chat.fixUsernameForBancho(recipientIRC)
|
recipientBancho = chat.fixUsernameForBancho(recipientIRC)
|
||||||
else:
|
else:
|
||||||
recipientBancho = recipientIRC
|
recipientBancho = recipientIRC
|
||||||
|
@ -478,7 +513,7 @@ class Client:
|
||||||
"""LUSERS command handler"""
|
"""LUSERS command handler"""
|
||||||
self.sendLusers()
|
self.sendLusers()
|
||||||
|
|
||||||
def pingHandler(self, command, arguments):
|
def pingHandler(self, _, arguments):
|
||||||
"""PING command handler"""
|
"""PING command handler"""
|
||||||
if len(arguments) < 1:
|
if len(arguments) < 1:
|
||||||
self.replyCode(409, "No origin specified")
|
self.replyCode(409, "No origin specified")
|
||||||
|
@ -489,10 +524,17 @@ class Client:
|
||||||
"""(fake) PONG command handler"""
|
"""(fake) PONG command handler"""
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
def awayHandler(self, _, arguments):
|
||||||
|
"""AWAY command handler"""
|
||||||
|
response = chat.IRCAway(self.banchoUsername, " ".join(arguments))
|
||||||
|
self.replyCode(response, "You are no longer marked as being away" if response == 305 else "You have been marked as being away")
|
||||||
|
|
||||||
def mainHandler(self, command, arguments):
|
def mainHandler(self, command, arguments):
|
||||||
"""Handler for post-login commands"""
|
"""
|
||||||
|
Handler for post-login commands
|
||||||
|
"""
|
||||||
handlers = {
|
handlers = {
|
||||||
#"AWAY": away_handler,
|
"AWAY": self.awayHandler,
|
||||||
#"ISON": ison_handler,
|
#"ISON": ison_handler,
|
||||||
"JOIN": self.joinHandler,
|
"JOIN": self.joinHandler,
|
||||||
#"LIST": list_handler,
|
#"LIST": list_handler,
|
||||||
|
@ -518,14 +560,8 @@ class Client:
|
||||||
self.replyCode(421, "Unknown command ({})".format(command))
|
self.replyCode(421, "Unknown command ({})".format(command))
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class Server:
|
class Server:
|
||||||
def __init__(self, port):
|
def __init__(self, port):
|
||||||
#self.host = socket.getfqdn("127.0.0.1")[:63]
|
|
||||||
self.host = glob.conf.config["irc"]["hostname"]
|
self.host = glob.conf.config["irc"]["hostname"]
|
||||||
self.port = port
|
self.port = port
|
||||||
self.clients = {} # Socket - - > Client instance.
|
self.clients = {} # Socket - - > Client instance.
|
||||||
|
@ -535,7 +571,9 @@ class Server:
|
||||||
"""
|
"""
|
||||||
Disconnect someone from IRC if connected
|
Disconnect someone from IRC if connected
|
||||||
|
|
||||||
username -- victim
|
:param username: victim
|
||||||
|
:param isBanchoUsername: if True, username is a bancho username, else convert it to a bancho username
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
for _, value in self.clients.items():
|
for _, value in self.clients.items():
|
||||||
if (value.IRCUsername == username and not isBanchoUsername) or (value.banchoUsername == username and isBanchoUsername):
|
if (value.IRCUsername == username and not isBanchoUsername) or (value.banchoUsername == username and isBanchoUsername):
|
||||||
|
@ -546,8 +584,9 @@ class Server:
|
||||||
"""
|
"""
|
||||||
Let every IRC client connected to a specific client know that 'username' joined the channel from bancho
|
Let every IRC client connected to a specific client know that 'username' joined the channel from bancho
|
||||||
|
|
||||||
username -- username of bancho user
|
:param username: username of bancho user
|
||||||
channel -- joined channel name
|
:param channel: joined channel name
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
username = chat.fixUsernameForIRC(username)
|
username = chat.fixUsernameForIRC(username)
|
||||||
for _, value in self.clients.items():
|
for _, value in self.clients.items():
|
||||||
|
@ -558,8 +597,9 @@ class Server:
|
||||||
"""
|
"""
|
||||||
Let every IRC client connected to a specific client know that 'username' parted the channel from bancho
|
Let every IRC client connected to a specific client know that 'username' parted the channel from bancho
|
||||||
|
|
||||||
username -- username of bancho user
|
:param username: username of bancho user
|
||||||
channel -- joined channel name
|
:param channel: joined channel name
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
username = chat.fixUsernameForIRC(username)
|
username = chat.fixUsernameForIRC(username)
|
||||||
for _, value in self.clients.items():
|
for _, value in self.clients.items():
|
||||||
|
@ -570,9 +610,10 @@ class Server:
|
||||||
"""
|
"""
|
||||||
Send a message to IRC when someone sends it from bancho
|
Send a message to IRC when someone sends it from bancho
|
||||||
|
|
||||||
fro -- sender username
|
:param fro: sender username
|
||||||
to -- receiver username
|
:param to: receiver username
|
||||||
message -- text of the message
|
:param message: text of the message
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
fro = chat.fixUsernameForIRC(fro)
|
fro = chat.fixUsernameForIRC(fro)
|
||||||
to = chat.fixUsernameForIRC(to)
|
to = chat.fixUsernameForIRC(to)
|
||||||
|
@ -588,21 +629,26 @@ class Server:
|
||||||
value.message(":{} PRIVMSG {} :{}".format(fro, to, message))
|
value.message(":{} PRIVMSG {} :{}".format(fro, to, message))
|
||||||
|
|
||||||
|
|
||||||
def removeClient(self, client, quitmsg):
|
def removeClient(self, client, _):
|
||||||
"""
|
"""
|
||||||
Remove a client from connected clients
|
Remove a client from connected clients
|
||||||
|
|
||||||
client -- client object
|
:param client: client object
|
||||||
quitmsg -- QUIT argument, useless atm
|
:return:
|
||||||
"""
|
"""
|
||||||
if client.socket in self.clients:
|
if client.socket in self.clients:
|
||||||
del self.clients[client.socket]
|
del self.clients[client.socket]
|
||||||
|
|
||||||
def start(self):
|
def start(self):
|
||||||
"""Start IRC server main loop"""
|
"""
|
||||||
|
Start IRC server main loop
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
# Sentry
|
# Sentry
|
||||||
|
sentryClient = None
|
||||||
if glob.sentry:
|
if glob.sentry:
|
||||||
sentryClient = raven.Client(glob.conf.config["sentry"]["ircdns"])
|
sentryClient = raven.Client(glob.conf.config["sentry"]["ircdsn"])
|
||||||
|
|
||||||
serversocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
serversocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
||||||
serversocket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
|
serversocket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
|
||||||
|
@ -622,7 +668,7 @@ class Server:
|
||||||
[x.socket for x in self.clients.values()
|
[x.socket for x in self.clients.values()
|
||||||
if x.writeBufferSize() > 0],
|
if x.writeBufferSize() > 0],
|
||||||
[],
|
[],
|
||||||
2)
|
1)
|
||||||
|
|
||||||
# Handle incoming connections
|
# Handle incoming connections
|
||||||
for x in iwtd:
|
for x in iwtd:
|
||||||
|
@ -633,7 +679,7 @@ class Server:
|
||||||
try:
|
try:
|
||||||
self.clients[conn] = Client(self, conn)
|
self.clients[conn] = Client(self, conn)
|
||||||
log.info("[IRC] Accepted connection from {}:{}".format(addr[0], addr[1]))
|
log.info("[IRC] Accepted connection from {}:{}".format(addr[0], addr[1]))
|
||||||
except socket.error as e:
|
except socket.error:
|
||||||
try:
|
try:
|
||||||
conn.close()
|
conn.close()
|
||||||
except:
|
except:
|
||||||
|
@ -652,9 +698,15 @@ class Server:
|
||||||
lastAliveCheck = now
|
lastAliveCheck = now
|
||||||
except:
|
except:
|
||||||
log.error("[IRC] Unknown error!\n```\n{}\n{}```".format(sys.exc_info(), traceback.format_exc()))
|
log.error("[IRC] Unknown error!\n```\n{}\n{}```".format(sys.exc_info(), traceback.format_exc()))
|
||||||
if glob.sentry:
|
if glob.sentry and sentryClient is not None:
|
||||||
sentryClient.captureException()
|
sentryClient.captureException()
|
||||||
|
|
||||||
def main(port=6667):
|
def main(port=6667):
|
||||||
|
"""
|
||||||
|
Create and start an IRC server
|
||||||
|
|
||||||
|
:param port: IRC port. Default: 6667
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
glob.ircServer = Server(port)
|
glob.ircServer = Server(port)
|
||||||
glob.ircServer.start()
|
glob.ircServer.start()
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
# TODO: Rewrite this shit
|
# TODO: Rewrite this shit
|
||||||
from common import generalUtils
|
from common import generalUtils
|
||||||
|
from constants import serverPackets
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
from common.log import logUtils as log
|
||||||
|
|
||||||
|
|
||||||
class banchoConfig:
|
class banchoConfig:
|
||||||
|
@ -29,7 +31,12 @@ class banchoConfig:
|
||||||
"""
|
"""
|
||||||
self.config["banchoMaintenance"] = generalUtils.stringToBool(glob.db.fetch("SELECT value_int FROM bancho_settings WHERE name = 'bancho_maintenance'")["value_int"])
|
self.config["banchoMaintenance"] = generalUtils.stringToBool(glob.db.fetch("SELECT value_int FROM bancho_settings WHERE name = 'bancho_maintenance'")["value_int"])
|
||||||
self.config["freeDirect"] = generalUtils.stringToBool(glob.db.fetch("SELECT value_int FROM bancho_settings WHERE name = 'free_direct'")["value_int"])
|
self.config["freeDirect"] = generalUtils.stringToBool(glob.db.fetch("SELECT value_int FROM bancho_settings WHERE name = 'free_direct'")["value_int"])
|
||||||
self.config["menuIcon"] = glob.db.fetch("SELECT value_string FROM bancho_settings WHERE name = 'menu_icon'")["value_string"]
|
mainMenuIcon = glob.db.fetch("SELECT file_id, url FROM main_menu_icons WHERE is_current = 1 LIMIT 1")
|
||||||
|
if mainMenuIcon is None:
|
||||||
|
self.config["menuIcon"] = ""
|
||||||
|
else:
|
||||||
|
imageURL = "https://i.ppy.sh/{}.png".format(mainMenuIcon["file_id"])
|
||||||
|
self.config["menuIcon"] = "{}|{}".format(imageURL, mainMenuIcon["url"])
|
||||||
self.config["loginNotification"] = glob.db.fetch("SELECT value_string FROM bancho_settings WHERE name = 'login_notification'")["value_string"]
|
self.config["loginNotification"] = glob.db.fetch("SELECT value_string FROM bancho_settings WHERE name = 'login_notification'")["value_string"]
|
||||||
|
|
||||||
|
|
||||||
|
@ -41,3 +48,20 @@ class banchoConfig:
|
||||||
"""
|
"""
|
||||||
self.config["banchoMaintenance"] = maintenance
|
self.config["banchoMaintenance"] = maintenance
|
||||||
glob.db.execute("UPDATE bancho_settings SET value_int = %s WHERE name = 'bancho_maintenance'", [int(maintenance)])
|
glob.db.execute("UPDATE bancho_settings SET value_int = %s WHERE name = 'bancho_maintenance'", [int(maintenance)])
|
||||||
|
|
||||||
|
def reload(self):
|
||||||
|
# Reload settings from bancho_settings
|
||||||
|
glob.banchoConf.loadSettings()
|
||||||
|
|
||||||
|
# Reload channels too
|
||||||
|
glob.channels.loadChannels()
|
||||||
|
|
||||||
|
# And chat filters
|
||||||
|
glob.chatFilters.loadFilters()
|
||||||
|
|
||||||
|
# Send new channels and new bottom icon to everyone
|
||||||
|
glob.streams.broadcast("main", serverPackets.mainMenuIcon(glob.banchoConf.config["menuIcon"]))
|
||||||
|
glob.streams.broadcast("main", serverPackets.channelInfoEnd())
|
||||||
|
for key, value in glob.channels.channels.items():
|
||||||
|
if value.publicRead and not value.hidden:
|
||||||
|
glob.streams.broadcast("main", serverPackets.channelInfo(key))
|
|
@ -1,20 +1,19 @@
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from constants import exceptions
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
class channel:
|
class channel:
|
||||||
"""
|
|
||||||
A chat channel
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, name, description, publicRead, publicWrite, temp, hidden):
|
def __init__(self, name, description, publicRead, publicWrite, temp, hidden):
|
||||||
"""
|
"""
|
||||||
Create a new chat channel object
|
Create a new chat channel object
|
||||||
|
|
||||||
name -- channel name
|
:param name: channel name
|
||||||
description -- channel description
|
:param description: channel description
|
||||||
publicRead -- bool, if true channel can be read by everyone, if false it can be read only by mods/admins
|
:param publicRead: if True, this channel can be read by everyone. If False, it can be read only by mods/admins
|
||||||
publicWrite -- bool, same as public read but relative to write permissions
|
:param publicWrite: same as public read, but regards writing permissions
|
||||||
temp -- if True, channel will be deleted when there's no one in the channel
|
:param temp: if True, this channel will be deleted when there's no one in this channel
|
||||||
hidden -- if True, channel won't be shown in channels list
|
:param hidden: if True, thic channel won't be shown in channels list
|
||||||
"""
|
"""
|
||||||
self.name = name
|
self.name = name
|
||||||
self.description = description
|
self.description = description
|
||||||
|
@ -22,51 +21,24 @@ class channel:
|
||||||
self.publicWrite = publicWrite
|
self.publicWrite = publicWrite
|
||||||
self.moderated = False
|
self.moderated = False
|
||||||
self.temp = temp
|
self.temp = temp
|
||||||
self.connectedUsers = [999] # Fokabot is always connected to every channels (otherwise it doesn't show up in IRC users list)
|
|
||||||
self.hidden = hidden
|
self.hidden = hidden
|
||||||
|
|
||||||
# Client name (#spectator/#multiplayer)
|
# Make Foka join the channel
|
||||||
self.clientName = self.name
|
fokaToken = glob.tokens.getTokenFromUserID(999)
|
||||||
|
if fokaToken is not None:
|
||||||
|
try:
|
||||||
|
fokaToken.joinChannel(self)
|
||||||
|
except exceptions.userAlreadyInChannelException:
|
||||||
|
logging.warning("FokaBot has already joined channel {}".format(self.name))
|
||||||
|
|
||||||
|
@property
|
||||||
|
def isSpecial(self):
|
||||||
|
return any(self.name.startswith(x) for x in ("#spect_", "#multi_"))
|
||||||
|
|
||||||
|
@property
|
||||||
|
def clientName(self):
|
||||||
if self.name.startswith("#spect_"):
|
if self.name.startswith("#spect_"):
|
||||||
self.clientName = "#spectator"
|
return "#spectator"
|
||||||
elif self.name.startswith("#multi_"):
|
elif self.name.startswith("#multi_"):
|
||||||
self.clientName = "#multiplayer"
|
return "#multiplayer"
|
||||||
|
return self.name
|
||||||
def userJoin(self, userID):
|
|
||||||
"""
|
|
||||||
Add a user to connected users
|
|
||||||
|
|
||||||
userID -- user ID that joined the channel
|
|
||||||
"""
|
|
||||||
if userID not in self.connectedUsers:
|
|
||||||
self.connectedUsers.append(userID)
|
|
||||||
|
|
||||||
def userPart(self, userID):
|
|
||||||
"""
|
|
||||||
Remove a user from connected users
|
|
||||||
|
|
||||||
userID -- user ID that left the channel
|
|
||||||
"""
|
|
||||||
if userID in self.connectedUsers:
|
|
||||||
self.connectedUsers.remove(userID)
|
|
||||||
|
|
||||||
# Remove temp channels if empty or there's only fokabot connected
|
|
||||||
l = len(self.connectedUsers)
|
|
||||||
if self.temp == True and ((l == 0) or (l == 1 and 999 in self.connectedUsers)):
|
|
||||||
glob.channels.removeChannel(self.name)
|
|
||||||
|
|
||||||
def getConnectedUsers(self):
|
|
||||||
"""
|
|
||||||
Get connected user IDs list
|
|
||||||
|
|
||||||
return -- connectedUsers list
|
|
||||||
"""
|
|
||||||
return self.connectedUsers
|
|
||||||
|
|
||||||
def getConnectedUsersCount(self):
|
|
||||||
"""
|
|
||||||
Count connected users
|
|
||||||
|
|
||||||
return -- connected users number
|
|
||||||
"""
|
|
||||||
return len(self.connectedUsers)
|
|
|
@ -1,21 +1,18 @@
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from objects import channel
|
from objects import channel
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
from helpers import chatHelper as chat
|
||||||
|
|
||||||
|
|
||||||
class channelList:
|
class channelList:
|
||||||
"""
|
def __init__(self):
|
||||||
Channel list
|
self.channels = {}
|
||||||
|
|
||||||
channels -- dictionary. key: channel name, value: channel object
|
|
||||||
"""
|
|
||||||
channels = {}
|
|
||||||
|
|
||||||
def loadChannels(self):
|
def loadChannels(self):
|
||||||
"""
|
"""
|
||||||
Load chat channels from db and add them to channels dictionary
|
Load chat channels from db and add them to channels list
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Get channels from DB
|
# Get channels from DB
|
||||||
channels = glob.db.fetchAll("SELECT * FROM bancho_channels")
|
channels = glob.db.fetchAll("SELECT * FROM bancho_channels")
|
||||||
|
|
||||||
|
@ -28,15 +25,17 @@ class channelList:
|
||||||
|
|
||||||
def addChannel(self, name, description, publicRead, publicWrite, temp = False, hidden = False):
|
def addChannel(self, name, description, publicRead, publicWrite, temp = False, hidden = False):
|
||||||
"""
|
"""
|
||||||
Add a channel object to channels dictionary
|
Add a channel to channels list
|
||||||
|
|
||||||
name -- channel name
|
:param name: channel name
|
||||||
description -- channel description
|
:param description: channel description
|
||||||
publicRead -- bool, if true channel can be read by everyone, if false it can be read only by mods/admins
|
:param publicRead: if True, this channel can be read by everyone. If False, it can be read only by mods/admins
|
||||||
publicWrite -- bool, same as public read but relative to write permissions
|
:param publicWrite: same as public read, but regards writing permissions
|
||||||
temp -- if True, channel will be deleted when there's no one in the channel. Optional. Default = False.
|
:param temp: if True, this channel will be deleted when there's no one in this channel
|
||||||
hidden -- if True, channel will be hidden in channels list. Optional. Default = False.
|
:param hidden: if True, thic channel won't be shown in channels list
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
|
glob.streams.add("chat/{}".format(name))
|
||||||
self.channels[name] = channel.channel(name, description, publicRead, publicWrite, temp, hidden)
|
self.channels[name] = channel.channel(name, description, publicRead, publicWrite, temp, hidden)
|
||||||
log.info("Created channel {}".format(name))
|
log.info("Created channel {}".format(name))
|
||||||
|
|
||||||
|
@ -45,22 +44,46 @@ class channelList:
|
||||||
Add a temporary channel (like #spectator or #multiplayer), gets deleted when there's no one in the channel
|
Add a temporary channel (like #spectator or #multiplayer), gets deleted when there's no one in the channel
|
||||||
and it's hidden in channels list
|
and it's hidden in channels list
|
||||||
|
|
||||||
name -- channel name
|
:param name: channel name
|
||||||
return -- True if channel was created, False if failed
|
:return: True if the channel was created, otherwise False
|
||||||
"""
|
"""
|
||||||
if name in self.channels:
|
if name in self.channels:
|
||||||
return False
|
return False
|
||||||
|
glob.streams.add("chat/{}".format(name))
|
||||||
self.channels[name] = channel.channel(name, "Chat", True, True, True, True)
|
self.channels[name] = channel.channel(name, "Chat", True, True, True, True)
|
||||||
log.info("Created temp channel {}".format(name))
|
log.info("Created temp channel {}".format(name))
|
||||||
|
|
||||||
|
def addHiddenChannel(self, name):
|
||||||
|
"""
|
||||||
|
Add a hidden channel. It's like a normal channel and must be deleted manually,
|
||||||
|
but it's not shown in channels list.
|
||||||
|
|
||||||
|
:param name: channel name
|
||||||
|
:return: True if the channel was created, otherwise False
|
||||||
|
"""
|
||||||
|
if name in self.channels:
|
||||||
|
return False
|
||||||
|
glob.streams.add("chat/{}".format(name))
|
||||||
|
self.channels[name] = channel.channel(name, "Chat", True, True, False, True)
|
||||||
|
log.info("Created hidden channel {}".format(name))
|
||||||
|
|
||||||
def removeChannel(self, name):
|
def removeChannel(self, name):
|
||||||
"""
|
"""
|
||||||
Removes a channel from channels list
|
Removes a channel from channels list
|
||||||
|
|
||||||
name -- channel name
|
:param name: channel name
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
if name not in self.channels:
|
if name not in self.channels:
|
||||||
log.debug("{} is not in channels list".format(name))
|
log.debug("{} is not in channels list".format(name))
|
||||||
return
|
return
|
||||||
|
#glob.streams.broadcast("chat/{}".format(name), serverPackets.channelKicked(name))
|
||||||
|
stream = glob.streams.getStream("chat/{}".format(name))
|
||||||
|
if stream is not None:
|
||||||
|
for token in stream.clients:
|
||||||
|
if token in glob.tokens.tokens:
|
||||||
|
chat.partChannel(channel=name, token=glob.tokens.tokens[token], kick=True)
|
||||||
|
glob.streams.dispose("chat/{}".format(name))
|
||||||
|
glob.streams.remove("chat/{}".format(name))
|
||||||
self.channels.pop(name)
|
self.channels.pop(name)
|
||||||
log.info("Removed channel {}".format(name))
|
log.info("Removed channel {}".format(name))
|
||||||
|
|
|
@ -1,9 +1,20 @@
|
||||||
class chatFilters:
|
class chatFilters:
|
||||||
def __init__(self, fileName="filters.txt"):
|
def __init__(self, fileName="filters.txt"):
|
||||||
|
"""
|
||||||
|
Initialize chat filters
|
||||||
|
|
||||||
|
:param fileName: name of the file containing filters. Default: filters.txt
|
||||||
|
"""
|
||||||
self.filters = {}
|
self.filters = {}
|
||||||
self.loadFilters(fileName)
|
self.loadFilters(fileName)
|
||||||
|
|
||||||
def loadFilters(self, fileName="filters.txt"):
|
def loadFilters(self, fileName="filters.txt"):
|
||||||
|
"""
|
||||||
|
Load filters from a file
|
||||||
|
|
||||||
|
:param fileName: name of the file containing filters. Default: filters.txt
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
# Reset chat filters
|
# Reset chat filters
|
||||||
self.filters = {}
|
self.filters = {}
|
||||||
|
|
||||||
|
@ -19,6 +30,14 @@ class chatFilters:
|
||||||
self.filters[lineSplit[0].lower()] = lineSplit[1].replace("\n", "")
|
self.filters[lineSplit[0].lower()] = lineSplit[1].replace("\n", "")
|
||||||
|
|
||||||
def filterMessage(self, message):
|
def filterMessage(self, message):
|
||||||
|
"""
|
||||||
|
Replace forbidden words with filtered ones
|
||||||
|
|
||||||
|
:param message: normal message
|
||||||
|
:return: filtered message
|
||||||
|
"""
|
||||||
|
return message
|
||||||
|
"""
|
||||||
# Split words by spaces
|
# Split words by spaces
|
||||||
messageTemp = message.split(" ")
|
messageTemp = message.split(" ")
|
||||||
|
|
||||||
|
@ -32,3 +51,4 @@ class chatFilters:
|
||||||
|
|
||||||
# Return filtered message
|
# Return filtered message
|
||||||
return message
|
return message
|
||||||
|
"""
|
||||||
|
|
|
@ -12,30 +12,37 @@ from objects import glob
|
||||||
npRegex = re.compile("^https?:\\/\\/osu\\.ppy\\.sh\\/b\\/(\\d*)")
|
npRegex = re.compile("^https?:\\/\\/osu\\.ppy\\.sh\\/b\\/(\\d*)")
|
||||||
|
|
||||||
def connect():
|
def connect():
|
||||||
"""Add FokaBot to connected users and send userpanel/stats packet to everyone"""
|
"""
|
||||||
|
Connect FokaBot to Bancho
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
glob.BOT_NAME = userUtils.getUsername(999)
|
||||||
token = glob.tokens.addToken(999)
|
token = glob.tokens.addToken(999)
|
||||||
token.actionID = actions.IDLE
|
token.actionID = actions.IDLE
|
||||||
glob.streams.broadcast("main", serverPackets.userPanel(999))
|
glob.streams.broadcast("main", serverPackets.userPanel(999))
|
||||||
glob.streams.broadcast("main", serverPackets.userStats(999))
|
glob.streams.broadcast("main", serverPackets.userStats(999))
|
||||||
|
|
||||||
def disconnect():
|
def disconnect():
|
||||||
"""Remove FokaBot from connected users"""
|
"""
|
||||||
|
Disconnect FokaBot from Bancho
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
glob.tokens.deleteToken(glob.tokens.getTokenFromUserID(999))
|
glob.tokens.deleteToken(glob.tokens.getTokenFromUserID(999))
|
||||||
|
|
||||||
def fokabotResponse(fro, chan, message):
|
def fokabotResponse(fro, chan, message):
|
||||||
"""
|
"""
|
||||||
Check if a message has triggered fokabot (and return its response)
|
Check if a message has triggered FokaBot
|
||||||
|
|
||||||
fro -- sender username (for permissions stuff with admin commands)
|
:param fro: sender username
|
||||||
chan -- channel name
|
:param chan: channel name (or receiver username)
|
||||||
message -- message
|
:param message: chat mesage
|
||||||
|
:return: FokaBot's response or False if no response
|
||||||
return -- fokabot's response string or False
|
|
||||||
"""
|
"""
|
||||||
for i in fokabotCommands.commands:
|
for i in fokabotCommands.commands:
|
||||||
# Loop though all commands
|
# Loop though all commands
|
||||||
#if i["trigger"] in message:
|
if re.compile("^{}( (.+)?)?$".format(i["trigger"])).match(message.strip()):
|
||||||
if generalUtils.strContains(message, i["trigger"]):
|
|
||||||
# message has triggered a command
|
# message has triggered a command
|
||||||
|
|
||||||
# Make sure the user has right permissions
|
# Make sure the user has right permissions
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
"""Global objects and variables"""
|
"""Global objects and variables"""
|
||||||
|
|
||||||
import time
|
import time
|
||||||
|
|
||||||
from common.ddog import datadogClient
|
from common.ddog import datadogClient
|
||||||
from common.files import fileBuffer, fileLocks
|
from common.files import fileBuffer, fileLocks
|
||||||
from objects import channelList
|
from objects import channelList
|
||||||
|
@ -12,29 +11,28 @@ from common.web import schiavo
|
||||||
|
|
||||||
try:
|
try:
|
||||||
with open("version") as f:
|
with open("version") as f:
|
||||||
VERSION = f.read()
|
VERSION = f.read().strip()
|
||||||
if VERSION == "":
|
if VERSION == "":
|
||||||
raise
|
raise Exception
|
||||||
except:
|
except:
|
||||||
VERSION = "¯\_(xd)_/¯"
|
VERSION = "Unknown"
|
||||||
|
|
||||||
DATADOG_PREFIX = "peppy"
|
DATADOG_PREFIX = "peppy"
|
||||||
|
BOT_NAME = "FokaBot"
|
||||||
application = None
|
application = None
|
||||||
db = None
|
db = None
|
||||||
|
redis = None
|
||||||
conf = None
|
conf = None
|
||||||
banchoConf = None
|
banchoConf = None
|
||||||
tokens = tokenList.tokenList()
|
tokens = tokenList.tokenList()
|
||||||
channels = channelList.channelList()
|
channels = channelList.channelList()
|
||||||
matches = matchList.matchList()
|
matches = matchList.matchList()
|
||||||
restarting = False
|
|
||||||
fLocks = fileLocks.fileLocks()
|
fLocks = fileLocks.fileLocks()
|
||||||
fileBuffers = fileBuffer.buffersList()
|
fileBuffers = fileBuffer.buffersList()
|
||||||
schiavo = schiavo.schiavo()
|
schiavo = schiavo.schiavo()
|
||||||
dog = datadogClient.datadogClient()
|
dog = datadogClient.datadogClient()
|
||||||
verifiedCache = {}
|
verifiedCache = {}
|
||||||
cloudflare = False
|
|
||||||
chatFilters = None
|
chatFilters = None
|
||||||
userIDCache = {}
|
|
||||||
pool = None
|
pool = None
|
||||||
ircServer = None
|
ircServer = None
|
||||||
busyThreads = 0
|
busyThreads = 0
|
||||||
|
@ -46,8 +44,16 @@ gzip = False
|
||||||
localize = False
|
localize = False
|
||||||
sentry = False
|
sentry = False
|
||||||
irc = False
|
irc = False
|
||||||
|
restarting = False
|
||||||
|
|
||||||
startTime = int(time.time())
|
startTime = int(time.time())
|
||||||
|
|
||||||
|
|
||||||
streams = streamList.streamList()
|
streams = streamList.streamList()
|
||||||
|
|
||||||
|
# Additional modifications
|
||||||
|
COMMON_VERSION_REQ = "1.2.1"
|
||||||
|
try:
|
||||||
|
with open("common/version") as f:
|
||||||
|
COMMON_VERSION = f.read().strip()
|
||||||
|
except:
|
||||||
|
COMMON_VERSION = "Unknown"
|
492
objects/match.py
492
objects/match.py
|
@ -1,4 +1,9 @@
|
||||||
import copy
|
import copy
|
||||||
|
import json
|
||||||
|
import threading
|
||||||
|
|
||||||
|
import time
|
||||||
|
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from constants import dataTypes
|
from constants import dataTypes
|
||||||
from constants import matchModModes
|
from constants import matchModModes
|
||||||
|
@ -13,29 +18,31 @@ from objects import glob
|
||||||
|
|
||||||
class slot:
|
class slot:
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.status = slotStatuses.free
|
self.status = slotStatuses.FREE
|
||||||
self.team = 0
|
self.team = matchTeams.NO_TEAM
|
||||||
self.userID = -1
|
self.userID = -1
|
||||||
self.user = None
|
self.user = None
|
||||||
self.mods = 0
|
self.mods = 0
|
||||||
self.loaded = False
|
self.loaded = False
|
||||||
self.skip = False
|
self.skip = False
|
||||||
self.complete = False
|
self.complete = False
|
||||||
|
self.score = 0
|
||||||
|
self.failed = False
|
||||||
|
self.passed = True
|
||||||
|
|
||||||
class match:
|
class match:
|
||||||
"""Multiplayer match object"""
|
def __init__(self, matchID, matchName, matchPassword, beatmapID, beatmapName, beatmapMD5, gameMode, hostUserID, isTourney=False):
|
||||||
def __init__(self, matchID, matchName, matchPassword, beatmapID, beatmapName, beatmapMD5, gameMode, hostUserID):
|
|
||||||
"""
|
"""
|
||||||
Create a new match object
|
Create a new match object
|
||||||
|
|
||||||
matchID -- match progressive identifier
|
:param matchID: match progressive identifier
|
||||||
matchName -- match name, string
|
:param matchName: match name, string
|
||||||
matchPassword -- match md5 password. Leave empty for no password
|
:param matchPassword: match md5 password. Leave empty for no password
|
||||||
beatmapID -- beatmap ID
|
:param beatmapID: beatmap ID
|
||||||
beatmapName -- beatmap name, string
|
:param beatmapName: beatmap name, string
|
||||||
beatmapMD5 -- beatmap md5 hash, string
|
:param beatmapMD5: beatmap md5 hash, string
|
||||||
gameMode -- game mode ID. See gameModes.py
|
:param gameMode: game mode ID. See gameModes.py
|
||||||
hostUserID -- user id of the host
|
:param hostUserID: user id of the host
|
||||||
"""
|
"""
|
||||||
self.matchID = matchID
|
self.matchID = matchID
|
||||||
self.streamName = "multi/{}".format(self.matchID)
|
self.streamName = "multi/{}".format(self.matchID)
|
||||||
|
@ -49,11 +56,18 @@ class match:
|
||||||
self.beatmapMD5 = beatmapMD5
|
self.beatmapMD5 = beatmapMD5
|
||||||
self.hostUserID = hostUserID
|
self.hostUserID = hostUserID
|
||||||
self.gameMode = gameMode
|
self.gameMode = gameMode
|
||||||
self.matchScoringType = matchScoringTypes.score # default values
|
self.matchScoringType = matchScoringTypes.SCORE # default values
|
||||||
self.matchTeamType = matchTeamTypes.headToHead # default value
|
self.matchTeamType = matchTeamTypes.HEAD_TO_HEAD # default value
|
||||||
self.matchModMode = matchModModes.normal # default value
|
self.matchModMode = matchModModes.NORMAL # default value
|
||||||
self.seed = 0
|
self.seed = 0
|
||||||
self.matchDataCache = bytes()
|
self.matchDataCache = bytes()
|
||||||
|
self.isTourney = isTourney
|
||||||
|
self.isLocked = False # if True, users can't change slots/teams. Used in tourney matches
|
||||||
|
self.isStarting = False
|
||||||
|
self._lock = threading.Lock()
|
||||||
|
self.createTime = int(time.time())
|
||||||
|
self.vinseID = None
|
||||||
|
self.bloodcatAlert = False
|
||||||
|
|
||||||
# Create all slots and reset them
|
# Create all slots and reset them
|
||||||
self.slots = []
|
self.slots = []
|
||||||
|
@ -65,57 +79,67 @@ class match:
|
||||||
glob.streams.add(self.playingStreamName)
|
glob.streams.add(self.playingStreamName)
|
||||||
|
|
||||||
# Create #multiplayer channel
|
# Create #multiplayer channel
|
||||||
glob.channels.addTempChannel("#multi_{}".format(self.matchID))
|
glob.channels.addHiddenChannel("#multi_{}".format(self.matchID))
|
||||||
|
log.info("MPROOM{}: {} match created!".format(self.matchID, "Tourney" if self.isTourney else "Normal"))
|
||||||
|
|
||||||
def getMatchData(self):
|
def getMatchData(self, censored = False):
|
||||||
"""
|
"""
|
||||||
Return binary match data structure for packetHelper
|
Return binary match data structure for packetHelper
|
||||||
|
Return binary match data structure for packetHelper
|
||||||
|
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# General match info
|
# General match info
|
||||||
# TODO: Test without safe copy, the error might have been caused by outdated python bytecode cache
|
# TODO: Test without safe copy, the error might have been caused by outdated python bytecode cache
|
||||||
safeMatch = copy.deepcopy(self)
|
# safeMatch = copy.deepcopy(self)
|
||||||
struct = [
|
struct = [
|
||||||
[safeMatch.matchID, dataTypes.UINT16],
|
[self.matchID, dataTypes.UINT16],
|
||||||
[int(safeMatch.inProgress), dataTypes.BYTE],
|
[int(self.inProgress), dataTypes.BYTE],
|
||||||
[0, dataTypes.BYTE],
|
[0, dataTypes.BYTE],
|
||||||
[safeMatch.mods, dataTypes.UINT32],
|
[self.mods, dataTypes.UINT32],
|
||||||
[safeMatch.matchName, dataTypes.STRING],
|
[self.matchName, dataTypes.STRING]
|
||||||
[safeMatch.matchPassword, dataTypes.STRING],
|
|
||||||
[safeMatch.beatmapName, dataTypes.STRING],
|
|
||||||
[safeMatch.beatmapID, dataTypes.UINT32],
|
|
||||||
[safeMatch.beatmapMD5, dataTypes.STRING],
|
|
||||||
]
|
]
|
||||||
|
if censored and self.matchPassword:
|
||||||
|
struct.append(["redacted", dataTypes.STRING])
|
||||||
|
else:
|
||||||
|
struct.append([self.matchPassword, dataTypes.STRING])
|
||||||
|
|
||||||
|
struct.extend([
|
||||||
|
[self.beatmapName, dataTypes.STRING],
|
||||||
|
[self.beatmapID, dataTypes.UINT32],
|
||||||
|
[self.beatmapMD5, dataTypes.STRING]
|
||||||
|
])
|
||||||
|
|
||||||
# Slots status IDs, always 16 elements
|
# Slots status IDs, always 16 elements
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
struct.append([safeMatch.slots[i].status, dataTypes.BYTE])
|
struct.append([self.slots[i].status, dataTypes.BYTE])
|
||||||
|
|
||||||
# Slot teams, always 16 elements
|
# Slot teams, always 16 elements
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
struct.append([safeMatch.slots[i].team, dataTypes.BYTE])
|
struct.append([self.slots[i].team, dataTypes.BYTE])
|
||||||
|
|
||||||
# Slot user ID. Write only if slot is occupied
|
# Slot user ID. Write only if slot is occupied
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
if safeMatch.slots[i].user is not None and safeMatch.slots[i].user in glob.tokens.tokens:
|
if self.slots[i].user is not None and self.slots[i].user in glob.tokens.tokens:
|
||||||
struct.append([glob.tokens.tokens[safeMatch.slots[i].user].userID, dataTypes.UINT32])
|
struct.append([glob.tokens.tokens[self.slots[i].user].userID, dataTypes.UINT32])
|
||||||
|
|
||||||
# Other match data
|
# Other match data
|
||||||
struct.extend([
|
struct.extend([
|
||||||
[safeMatch.hostUserID, dataTypes.SINT32],
|
[self.hostUserID, dataTypes.SINT32],
|
||||||
[safeMatch.gameMode, dataTypes.BYTE],
|
[self.gameMode, dataTypes.BYTE],
|
||||||
[safeMatch.matchScoringType, dataTypes.BYTE],
|
[self.matchScoringType, dataTypes.BYTE],
|
||||||
[safeMatch.matchTeamType, dataTypes.BYTE],
|
[self.matchTeamType, dataTypes.BYTE],
|
||||||
[safeMatch.matchModMode, dataTypes.BYTE],
|
[self.matchModMode, dataTypes.BYTE],
|
||||||
])
|
])
|
||||||
|
|
||||||
# Slot mods if free mod is enabled
|
# Slot mods if free mod is enabled
|
||||||
if safeMatch.matchModMode == matchModModes.freeMod:
|
if self.matchModMode == matchModModes.FREE_MOD:
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
struct.append([safeMatch.slots[i].mods, dataTypes.UINT32])
|
struct.append([self.slots[i].mods, dataTypes.UINT32])
|
||||||
|
|
||||||
# Seed idk
|
# Seed idk
|
||||||
# TODO: Implement this, it should be used for mania "random" mod
|
# TODO: Implement this, it should be used for mania "random" mod
|
||||||
struct.append([safeMatch.seed, dataTypes.UINT32])
|
struct.append([self.seed, dataTypes.UINT32])
|
||||||
|
|
||||||
return struct
|
return struct
|
||||||
|
|
||||||
|
@ -123,19 +147,44 @@ class match:
|
||||||
"""
|
"""
|
||||||
Set room host to newHost and send him host packet
|
Set room host to newHost and send him host packet
|
||||||
|
|
||||||
newHost -- new host userID
|
:param newHost: new host userID
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
slotID = self.getUserSlotID(newHost)
|
slotID = self.getUserSlotID(newHost)
|
||||||
if slotID is None or self.slots[slotID].user not in glob.tokens.tokens:
|
if slotID is None or self.slots[slotID].user not in glob.tokens.tokens:
|
||||||
return
|
return False
|
||||||
token = glob.tokens.tokens[self.slots[slotID].user]
|
token = glob.tokens.tokens[self.slots[slotID].user]
|
||||||
self.hostUserID = newHost
|
self.hostUserID = newHost
|
||||||
token.enqueue(serverPackets.matchTransferHost())
|
token.enqueue(serverPackets.matchTransferHost())
|
||||||
self.sendUpdates()
|
self.sendUpdates()
|
||||||
log.info("MPROOM{}: {} is now the host".format(self.matchID, token.username))
|
log.info("MPROOM{}: {} is now the host".format(self.matchID, token.username))
|
||||||
|
return True
|
||||||
|
|
||||||
|
def removeHost(self):
|
||||||
|
"""
|
||||||
|
Removes the host (for tourney matches)
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
self.hostUserID = -1
|
||||||
|
self.sendUpdates()
|
||||||
|
log.info("MPROOM{}: Removed host".format(self.matchID))
|
||||||
|
|
||||||
def setSlot(self, slotID, status = None, team = None, user = "", mods = None, loaded = None, skip = None, complete = None):
|
def setSlot(self, slotID, status = None, team = None, user = "", mods = None, loaded = None, skip = None, complete = None):
|
||||||
#self.setSlot(i, slotStatuses.notReady, 0, user, 0)
|
"""
|
||||||
|
Set data for a specific slot.
|
||||||
|
All fields but slotID are optional.
|
||||||
|
Skipped fields won't be edited.
|
||||||
|
|
||||||
|
:param slotID: slot ID
|
||||||
|
:param status: new status
|
||||||
|
:param team: new team
|
||||||
|
:param user: new user id
|
||||||
|
:param mods: new mods
|
||||||
|
:param loaded: new loaded status
|
||||||
|
:param skip: new skip value
|
||||||
|
:param complete: new completed value
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
if status is not None:
|
if status is not None:
|
||||||
self.slots[slotID].status = status
|
self.slots[slotID].status = status
|
||||||
|
|
||||||
|
@ -161,8 +210,9 @@ class match:
|
||||||
"""
|
"""
|
||||||
Set slotID mods. Same as calling setSlot and then sendUpdate
|
Set slotID mods. Same as calling setSlot and then sendUpdate
|
||||||
|
|
||||||
slotID -- slot number
|
:param slotID: slot number
|
||||||
mods -- new mods
|
:param mods: new mods
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Set new slot data and send update
|
# Set new slot data and send update
|
||||||
self.setSlot(slotID, mods=mods)
|
self.setSlot(slotID, mods=mods)
|
||||||
|
@ -174,32 +224,36 @@ class match:
|
||||||
Switch slotID ready/not ready status
|
Switch slotID ready/not ready status
|
||||||
Same as calling setSlot and then sendUpdate
|
Same as calling setSlot and then sendUpdate
|
||||||
|
|
||||||
slotID -- slot number
|
:param slotID: slot number
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Update ready status and setnd update
|
# Update ready status and setnd update
|
||||||
|
if self.slots[slotID].user is None or self.isStarting:
|
||||||
|
return
|
||||||
oldStatus = self.slots[slotID].status
|
oldStatus = self.slots[slotID].status
|
||||||
if oldStatus == slotStatuses.ready:
|
if oldStatus == slotStatuses.READY:
|
||||||
newStatus = slotStatuses.notReady
|
newStatus = slotStatuses.NOT_READY
|
||||||
else:
|
else:
|
||||||
newStatus = slotStatuses.ready
|
newStatus = slotStatuses.READY
|
||||||
self.setSlot(slotID, newStatus)
|
self.setSlot(slotID, newStatus)
|
||||||
self.sendUpdates()
|
self.sendUpdates()
|
||||||
log.info("MPROOM{}: Slot{} changed ready status to {}".format(self.matchID, slotID, self.slots[slotID].status))
|
log.info("MPROOM{}: Slot{} changed ready status to {}".format(self.matchID, slotID, self.slots[slotID].status))
|
||||||
|
|
||||||
def toggleSlotLock(self, slotID):
|
def toggleSlotLocked(self, slotID):
|
||||||
"""
|
"""
|
||||||
Lock a slot
|
Lock a slot
|
||||||
Same as calling setSlot and then sendUpdate
|
Same as calling setSlot and then sendUpdate
|
||||||
|
|
||||||
slotID -- slot number
|
:param slotID: slot number
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Check if slot is already locked
|
# Check if slot is already locked
|
||||||
if self.slots[slotID].status == slotStatuses.locked:
|
if self.slots[slotID].status == slotStatuses.LOCKED:
|
||||||
newStatus = slotStatuses.free
|
newStatus = slotStatuses.FREE
|
||||||
else:
|
else:
|
||||||
newStatus = slotStatuses.locked
|
newStatus = slotStatuses.LOCKED
|
||||||
|
|
||||||
# Send updated settings to kicked user, so he returns to lobby
|
# Send updated settings to kicked user, so they will return to the lobby.
|
||||||
if self.slots[slotID].user is not None and self.slots[slotID].user in glob.tokens.tokens:
|
if self.slots[slotID].user is not None and self.slots[slotID].user in glob.tokens.tokens:
|
||||||
glob.tokens.tokens[self.slots[slotID].user].enqueue(serverPackets.updateMatch(self.matchID))
|
glob.tokens.tokens[self.slots[slotID].user].enqueue(serverPackets.updateMatch(self.matchID))
|
||||||
|
|
||||||
|
@ -208,13 +262,14 @@ class match:
|
||||||
|
|
||||||
# Send updates to everyone else
|
# Send updates to everyone else
|
||||||
self.sendUpdates()
|
self.sendUpdates()
|
||||||
log.info("MPROOM{}: Slot{} {}".format(self.matchID, slotID, "locked" if newStatus == slotStatuses.locked else "unlocked"))
|
log.info("MPROOM{}: Slot{} {}".format(self.matchID, slotID, "locked" if newStatus == slotStatuses.LOCKED else "unlocked"))
|
||||||
|
|
||||||
def playerLoaded(self, userID):
|
def playerLoaded(self, userID):
|
||||||
"""
|
"""
|
||||||
Set a player loaded status to True
|
Set a player loaded status to True
|
||||||
|
|
||||||
userID -- ID of user
|
:param userID: ID of user
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
slotID = self.getUserSlotID(userID)
|
slotID = self.getUserSlotID(userID)
|
||||||
if slotID is None:
|
if slotID is None:
|
||||||
|
@ -228,7 +283,7 @@ class match:
|
||||||
total = 0
|
total = 0
|
||||||
loaded = 0
|
loaded = 0
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
if self.slots[i].status == slotStatuses.playing:
|
if self.slots[i].status == slotStatuses.PLAYING:
|
||||||
total+=1
|
total+=1
|
||||||
if self.slots[i].loaded:
|
if self.slots[i].loaded:
|
||||||
loaded+=1
|
loaded+=1
|
||||||
|
@ -237,7 +292,11 @@ class match:
|
||||||
self.allPlayersLoaded()
|
self.allPlayersLoaded()
|
||||||
|
|
||||||
def allPlayersLoaded(self):
|
def allPlayersLoaded(self):
|
||||||
"""Send allPlayersLoaded packet to every playing usr in match"""
|
"""
|
||||||
|
Send allPlayersLoaded packet to every playing usr in match
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
glob.streams.broadcast(self.playingStreamName, serverPackets.allPlayersLoaded())
|
glob.streams.broadcast(self.playingStreamName, serverPackets.allPlayersLoaded())
|
||||||
log.info("MPROOM{}: All players loaded! Match starting...".format(self.matchID))
|
log.info("MPROOM{}: All players loaded! Match starting...".format(self.matchID))
|
||||||
|
|
||||||
|
@ -245,7 +304,8 @@ class match:
|
||||||
"""
|
"""
|
||||||
Set a player skip status to True
|
Set a player skip status to True
|
||||||
|
|
||||||
userID -- ID of user
|
:param userID: ID of user
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
slotID = self.getUserSlotID(userID)
|
slotID = self.getUserSlotID(userID)
|
||||||
if slotID is None:
|
if slotID is None:
|
||||||
|
@ -263,7 +323,7 @@ class match:
|
||||||
total = 0
|
total = 0
|
||||||
skipped = 0
|
skipped = 0
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
if self.slots[i].status == slotStatuses.playing:
|
if self.slots[i].status == slotStatuses.PLAYING:
|
||||||
total+=1
|
total+=1
|
||||||
if self.slots[i].skip:
|
if self.slots[i].skip:
|
||||||
skipped+=1
|
skipped+=1
|
||||||
|
@ -272,15 +332,39 @@ class match:
|
||||||
self.allPlayersSkipped()
|
self.allPlayersSkipped()
|
||||||
|
|
||||||
def allPlayersSkipped(self):
|
def allPlayersSkipped(self):
|
||||||
"""Send allPlayersSkipped packet to every playing usr in match"""
|
"""
|
||||||
|
Send allPlayersSkipped packet to every playing usr in match
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
glob.streams.broadcast(self.playingStreamName, serverPackets.allPlayersSkipped())
|
glob.streams.broadcast(self.playingStreamName, serverPackets.allPlayersSkipped())
|
||||||
log.info("MPROOM{}: All players have skipped!".format(self.matchID))
|
log.info("MPROOM{}: All players have skipped!".format(self.matchID))
|
||||||
|
|
||||||
|
def updateScore(self, slotID, score):
|
||||||
|
"""
|
||||||
|
Update score for a slot
|
||||||
|
|
||||||
|
:param slotID: the slot that the user that is updating their score is in
|
||||||
|
:param score: the new score to update
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
self.slots[slotID].score = score
|
||||||
|
|
||||||
|
def updateHP(self, slotID, hp):
|
||||||
|
"""
|
||||||
|
Update HP for a slot
|
||||||
|
|
||||||
|
:param slotID: the slot that the user that is updating their hp is in
|
||||||
|
:param hp: the new hp to update
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
self.slots[slotID].failed = True if hp == 254 else False
|
||||||
|
|
||||||
def playerCompleted(self, userID):
|
def playerCompleted(self, userID):
|
||||||
"""
|
"""
|
||||||
Set userID's slot completed to True
|
Set userID's slot completed to True
|
||||||
|
|
||||||
userID -- ID of user
|
:param userID: ID of user
|
||||||
"""
|
"""
|
||||||
slotID = self.getUserSlotID(userID)
|
slotID = self.getUserSlotID(userID)
|
||||||
if slotID is None:
|
if slotID is None:
|
||||||
|
@ -294,7 +378,7 @@ class match:
|
||||||
total = 0
|
total = 0
|
||||||
completed = 0
|
completed = 0
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
if self.slots[i].status == slotStatuses.playing:
|
if self.slots[i].status == slotStatuses.PLAYING:
|
||||||
total+=1
|
total+=1
|
||||||
if self.slots[i].complete:
|
if self.slots[i].complete:
|
||||||
completed+=1
|
completed+=1
|
||||||
|
@ -303,18 +387,40 @@ class match:
|
||||||
self.allPlayersCompleted()
|
self.allPlayersCompleted()
|
||||||
|
|
||||||
def allPlayersCompleted(self):
|
def allPlayersCompleted(self):
|
||||||
"""Cleanup match stuff and send match end packet to everyone"""
|
"""
|
||||||
|
Cleanup match stuff and send match end packet to everyone
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
# Collect some info about the match that just ended to send to the api
|
||||||
|
infoToSend = {
|
||||||
|
"id": self.matchID,
|
||||||
|
"name": self.matchName,
|
||||||
|
"beatmap_id": self.beatmapID,
|
||||||
|
"mods": self.mods,
|
||||||
|
"game_mode": self.gameMode,
|
||||||
|
"scores": {}
|
||||||
|
}
|
||||||
|
|
||||||
|
# Add score info for each player
|
||||||
|
for i in range(0,16):
|
||||||
|
if self.slots[i].user is not None and self.slots[i].status == slotStatuses.PLAYING:
|
||||||
|
infoToSend["scores"][glob.tokens.tokens[self.slots[i].user].userID] = {
|
||||||
|
"score": self.slots[i].score,
|
||||||
|
"mods": self.slots[i].mods,
|
||||||
|
"failed": self.slots[i].failed,
|
||||||
|
"pass": self.slots[i].passed,
|
||||||
|
"team": self.slots[i].team
|
||||||
|
}
|
||||||
|
|
||||||
|
# Send the info to the api
|
||||||
|
glob.redis.publish("api:mp_complete_match", json.dumps(infoToSend))
|
||||||
|
|
||||||
# Reset inProgress
|
# Reset inProgress
|
||||||
self.inProgress = False
|
self.inProgress = False
|
||||||
|
|
||||||
# Reset slots
|
# Reset slots
|
||||||
for i in range(0,16):
|
self.resetSlots()
|
||||||
if self.slots[i].user is not None and self.slots[i].status == slotStatuses.playing:
|
|
||||||
self.slots[i].status = slotStatuses.notReady
|
|
||||||
self.slots[i].loaded = False
|
|
||||||
self.slots[i].skip = False
|
|
||||||
self.slots[i].complete = False
|
|
||||||
|
|
||||||
# Send match update
|
# Send match update
|
||||||
self.sendUpdates()
|
self.sendUpdates()
|
||||||
|
@ -323,16 +429,51 @@ class match:
|
||||||
glob.streams.broadcast(self.streamName, serverPackets.matchComplete())
|
glob.streams.broadcast(self.streamName, serverPackets.matchComplete())
|
||||||
|
|
||||||
# Destroy playing stream
|
# Destroy playing stream
|
||||||
|
glob.streams.dispose(self.playingStreamName)
|
||||||
glob.streams.remove(self.playingStreamName)
|
glob.streams.remove(self.playingStreamName)
|
||||||
|
|
||||||
# Console output
|
# Console output
|
||||||
log.info("MPROOM{}: Match completed".format(self.matchID))
|
log.info("MPROOM{}: Match completed".format(self.matchID))
|
||||||
|
|
||||||
|
# Set vinse id if needed
|
||||||
|
chanName = "#multi_{}".format(self.matchID)
|
||||||
|
if self.vinseID is None:
|
||||||
|
self.vinseID = (int(time.time()) // (60 * 15)) << 32 | self.matchID
|
||||||
|
chat.sendMessage("FokaBot", chanName, "Match history available [{} here]".format(
|
||||||
|
"https://vinse.ripple.moe/match/{}".format(self.vinseID)
|
||||||
|
))
|
||||||
|
if not self.bloodcatAlert:
|
||||||
|
chat.sendMessage(
|
||||||
|
"FokaBot",
|
||||||
|
chanName,
|
||||||
|
"Oh by the way, in case you're playing unranked or broken maps "
|
||||||
|
"that are now available through ripple's osu!direct, you can "
|
||||||
|
"type '!bloodcat' in the chat to get a download link for the "
|
||||||
|
"currently selected map from Bloodcat!"
|
||||||
|
)
|
||||||
|
self.bloodcatAlert = True
|
||||||
|
|
||||||
|
# If this is a tournament match, then we send a notification in the chat
|
||||||
|
# saying that the match has completed.
|
||||||
|
if self.isTourney and (chanName in glob.channels.channels):
|
||||||
|
chat.sendMessage(glob.BOT_NAME, chanName, "Match has just finished.")
|
||||||
|
|
||||||
|
def resetSlots(self):
|
||||||
|
for i in range(0,16):
|
||||||
|
if self.slots[i].user is not None and self.slots[i].status == slotStatuses.PLAYING:
|
||||||
|
self.slots[i].status = slotStatuses.NOT_READY
|
||||||
|
self.slots[i].loaded = False
|
||||||
|
self.slots[i].skip = False
|
||||||
|
self.slots[i].complete = False
|
||||||
|
self.slots[i].score = 0
|
||||||
|
self.slots[i].failed = False
|
||||||
|
self.slots[i].passed = True
|
||||||
|
|
||||||
def getUserSlotID(self, userID):
|
def getUserSlotID(self, userID):
|
||||||
"""
|
"""
|
||||||
Get slot ID occupied by userID
|
Get slot ID occupied by userID
|
||||||
|
|
||||||
return -- slot id if found, None if user is not in room
|
:return: slot id if found, None if user is not in room
|
||||||
"""
|
"""
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
if self.slots[i].user is not None and self.slots[i].user in glob.tokens.tokens and glob.tokens.tokens[self.slots[i].user].userID == userID:
|
if self.slots[i].user is not None and self.slots[i].user in glob.tokens.tokens and glob.tokens.tokens[self.slots[i].user].userID == userID:
|
||||||
|
@ -343,21 +484,23 @@ class match:
|
||||||
"""
|
"""
|
||||||
Add someone to users in match
|
Add someone to users in match
|
||||||
|
|
||||||
userID -- user id of the user
|
:param user: user object of the user
|
||||||
return -- True if join success, False if fail (room is full)
|
:return: True if join success, False if fail (room is full)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Make sure we're not in this match
|
# Make sure we're not in this match
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
if self.slots[i].user == user.token:
|
if self.slots[i].user == user.token:
|
||||||
# Set bugged slot to free
|
# Set bugged slot to free
|
||||||
self.setSlot(i, slotStatuses.free, 0, None, 0)
|
self.setSlot(i, slotStatuses.FREE, 0, None, 0)
|
||||||
|
|
||||||
# Find first free slot
|
# Find first free slot
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
if self.slots[i].status == slotStatuses.free:
|
if self.slots[i].status == slotStatuses.FREE:
|
||||||
# Occupy slot
|
# Occupy slot
|
||||||
self.setSlot(i, slotStatuses.notReady, 0, user.token, 0)
|
team = matchTeams.NO_TEAM
|
||||||
|
if self.matchTeamType == matchTeamTypes.TEAM_VS or self.matchTeamType == matchTeamTypes.TAG_TEAM_VS:
|
||||||
|
team = matchTeams.RED if i % 2 == 0 else matchTeams.BLUE
|
||||||
|
self.setSlot(i, slotStatuses.NOT_READY, team, user.token, 0)
|
||||||
|
|
||||||
# Send updated match data
|
# Send updated match data
|
||||||
self.sendUpdates()
|
self.sendUpdates()
|
||||||
|
@ -368,11 +511,13 @@ class match:
|
||||||
|
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def userLeft(self, user):
|
def userLeft(self, user, disposeMatch=True):
|
||||||
"""
|
"""
|
||||||
Remove someone from users in match
|
Remove someone from users in match
|
||||||
|
|
||||||
userID -- user if of the user
|
:param user: user object of the user
|
||||||
|
:param disposeMatch: if `True`, will try to dispose match if there are no users in the room
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Make sure the user is in room
|
# Make sure the user is in room
|
||||||
slotID = self.getUserSlotID(user.userID)
|
slotID = self.getUserSlotID(user.userID)
|
||||||
|
@ -380,13 +525,13 @@ class match:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Set that slot to free
|
# Set that slot to free
|
||||||
self.setSlot(slotID, slotStatuses.free, 0, None, 0)
|
self.setSlot(slotID, slotStatuses.FREE, 0, None, 0)
|
||||||
|
|
||||||
# Check if everyone left
|
# Check if everyone left
|
||||||
if self.countUsers() == 0:
|
if self.countUsers() == 0 and disposeMatch and not self.isTourney:
|
||||||
# Dispose match
|
# Dispose match
|
||||||
glob.matches.disposeMatch(self.matchID)
|
glob.matches.disposeMatch(self.matchID)
|
||||||
log.info("MPROOM{}: Room disposed".format(self.matchID))
|
log.info("MPROOM{}: Room disposed because all users left".format(self.matchID))
|
||||||
return
|
return
|
||||||
|
|
||||||
# Check if host left
|
# Check if host left
|
||||||
|
@ -407,24 +552,29 @@ class match:
|
||||||
"""
|
"""
|
||||||
Change userID slot to newSlotID
|
Change userID slot to newSlotID
|
||||||
|
|
||||||
userID -- user that changed slot
|
:param userID: user that changed slot
|
||||||
newSlotID -- slot id of new slot
|
:param newSlotID: slot id of new slot
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
|
# Make sure the match is not locked
|
||||||
|
if self.isLocked or self.isStarting:
|
||||||
|
return False
|
||||||
|
|
||||||
# Make sure the user is in room
|
# Make sure the user is in room
|
||||||
oldSlotID = self.getUserSlotID(userID)
|
oldSlotID = self.getUserSlotID(userID)
|
||||||
if oldSlotID is None:
|
if oldSlotID is None:
|
||||||
return
|
return False
|
||||||
|
|
||||||
# Make sure there is no one inside new slot
|
# Make sure there is no one inside new slot
|
||||||
if self.slots[newSlotID].user is not None and self.slots[newSlotID].status != slotStatuses.free:
|
if self.slots[newSlotID].user is not None or self.slots[newSlotID].status != slotStatuses.FREE:
|
||||||
return
|
return False
|
||||||
|
|
||||||
# Get old slot data
|
# Get old slot data
|
||||||
#oldData = dill.copy(self.slots[oldSlotID])
|
#oldData = dill.copy(self.slots[oldSlotID])
|
||||||
oldData = copy.deepcopy(self.slots[oldSlotID])
|
oldData = copy.deepcopy(self.slots[oldSlotID])
|
||||||
|
|
||||||
# Free old slot
|
# Free old slot
|
||||||
self.setSlot(oldSlotID, slotStatuses.free, 0, None, 0, False, False, False)
|
self.setSlot(oldSlotID, slotStatuses.FREE, 0, None, 0, False, False, False)
|
||||||
|
|
||||||
# Occupy new slot
|
# Occupy new slot
|
||||||
self.setSlot(newSlotID, oldData.status, oldData.team, oldData.user, oldData.mods)
|
self.setSlot(newSlotID, oldData.status, oldData.team, oldData.user, oldData.mods)
|
||||||
|
@ -434,18 +584,16 @@ class match:
|
||||||
|
|
||||||
# Console output
|
# Console output
|
||||||
log.info("MPROOM{}: {} moved to slot {}".format(self.matchID, userID, newSlotID))
|
log.info("MPROOM{}: {} moved to slot {}".format(self.matchID, userID, newSlotID))
|
||||||
|
return True
|
||||||
|
|
||||||
def changePassword(self, newPassword):
|
def changePassword(self, newPassword):
|
||||||
"""
|
"""
|
||||||
Change match password to newPassword
|
Change match password to newPassword
|
||||||
|
|
||||||
newPassword -- new password string
|
:param newPassword: new password string
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
self.matchPassword = newPassword
|
self.matchPassword = newPassword
|
||||||
#if newPassword != "":
|
|
||||||
# self.matchPassword = generalUtils.stringMd5(newPassword)
|
|
||||||
#else:
|
|
||||||
# self.matchPassword = ""
|
|
||||||
|
|
||||||
# Send password change to every user in match
|
# Send password change to every user in match
|
||||||
glob.streams.broadcast(self.streamName, serverPackets.changeMatchPassword(self.matchPassword))
|
glob.streams.broadcast(self.streamName, serverPackets.changeMatchPassword(self.matchPassword))
|
||||||
|
@ -460,7 +608,8 @@ class match:
|
||||||
"""
|
"""
|
||||||
Set match global mods
|
Set match global mods
|
||||||
|
|
||||||
mods -- mods bitwise int thing
|
:param mods: mods bitwise int thing
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Set new mods and send update
|
# Set new mods and send update
|
||||||
self.mods = mods
|
self.mods = mods
|
||||||
|
@ -471,8 +620,9 @@ class match:
|
||||||
"""
|
"""
|
||||||
Set no beatmap status for userID
|
Set no beatmap status for userID
|
||||||
|
|
||||||
userID -- ID of user
|
:param userID: ID of user
|
||||||
has -- True if has beatmap, false if not
|
:param has: True if has beatmap, false if not
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Make sure the user is in room
|
# Make sure the user is in room
|
||||||
slotID = self.getUserSlotID(userID)
|
slotID = self.getUserSlotID(userID)
|
||||||
|
@ -480,7 +630,7 @@ class match:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Set slot
|
# Set slot
|
||||||
self.setSlot(slotID, slotStatuses.noMap if not has else slotStatuses.notReady)
|
self.setSlot(slotID, slotStatuses.NO_MAP if not has else slotStatuses.NOT_READY)
|
||||||
|
|
||||||
# Send updates
|
# Send updates
|
||||||
self.sendUpdates()
|
self.sendUpdates()
|
||||||
|
@ -489,7 +639,8 @@ class match:
|
||||||
"""
|
"""
|
||||||
Transfer host to slotID
|
Transfer host to slotID
|
||||||
|
|
||||||
slotID -- ID of slot
|
:param slotID: ID of slot
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Make sure there is someone in that slot
|
# Make sure there is someone in that slot
|
||||||
if self.slots[slotID].user is None or self.slots[slotID].user not in glob.tokens.tokens:
|
if self.slots[slotID].user is None or self.slots[slotID].user not in glob.tokens.tokens:
|
||||||
|
@ -499,19 +650,22 @@ class match:
|
||||||
self.setHost(glob.tokens.tokens[self.slots[slotID].user].userID)
|
self.setHost(glob.tokens.tokens[self.slots[slotID].user].userID)
|
||||||
|
|
||||||
# Send updates
|
# Send updates
|
||||||
self.sendUpdates()
|
# self.sendUpdates()
|
||||||
|
|
||||||
def playerFailed(self, userID):
|
def playerFailed(self, userID):
|
||||||
"""
|
"""
|
||||||
Send userID's failed packet to everyone in match
|
Send userID's failed packet to everyone in match
|
||||||
|
|
||||||
userID -- ID of user
|
:param userID: ID of user
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Make sure the user is in room
|
# Make sure the user is in room
|
||||||
slotID = self.getUserSlotID(userID)
|
slotID = self.getUserSlotID(userID)
|
||||||
if slotID is None:
|
if slotID is None:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
self.slots[slotID].passed = False
|
||||||
|
|
||||||
# Send packet to everyone
|
# Send packet to everyone
|
||||||
glob.streams.broadcast(self.playingStreamName, serverPackets.playerFailed(slotID))
|
glob.streams.broadcast(self.playingStreamName, serverPackets.playerFailed(slotID))
|
||||||
|
|
||||||
|
@ -522,10 +676,10 @@ class match:
|
||||||
"""
|
"""
|
||||||
Fro invites to in this match.
|
Fro invites to in this match.
|
||||||
|
|
||||||
fro -- sender userID
|
:param fro: sender userID
|
||||||
to -- receiver userID
|
:param to: receiver userID
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Get tokens
|
# Get tokens
|
||||||
froToken = glob.tokens.getTokenFromUserID(fro)
|
froToken = glob.tokens.getTokenFromUserID(fro)
|
||||||
toToken = glob.tokens.getTokenFromUserID(to)
|
toToken = glob.tokens.getTokenFromUserID(to)
|
||||||
|
@ -534,7 +688,7 @@ class match:
|
||||||
|
|
||||||
# FokaBot is too busy
|
# FokaBot is too busy
|
||||||
if to == 999:
|
if to == 999:
|
||||||
chat.sendMessage("FokaBot", froToken.username, "I would love to join your match, but I'm busy keeping ripple up and running. Sorry. Beep Boop.")
|
chat.sendMessage(glob.BOT_NAME, froToken.username, "I would love to join your match, but I'm busy keeping ripple up and running. Sorry. Beep Boop.")
|
||||||
|
|
||||||
# Send message
|
# Send message
|
||||||
message = "Come join my multiplayer match: \"[osump://{}/{} {}]\"".format(self.matchID, self.matchPassword.replace(" ", "_"), self.matchName)
|
message = "Come join my multiplayer match: \"[osump://{}/{} {}]\"".format(self.matchID, self.matchPassword.replace(" ", "_"), self.matchName)
|
||||||
|
@ -544,7 +698,7 @@ class match:
|
||||||
"""
|
"""
|
||||||
Return how many players are in that match
|
Return how many players are in that match
|
||||||
|
|
||||||
return -- number of users
|
:return: number of users
|
||||||
"""
|
"""
|
||||||
c = 0
|
c = 0
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
|
@ -552,27 +706,44 @@ class match:
|
||||||
c+=1
|
c+=1
|
||||||
return c
|
return c
|
||||||
|
|
||||||
def changeTeam(self, userID):
|
def changeTeam(self, userID, newTeam=None):
|
||||||
"""
|
"""
|
||||||
Change userID's team
|
Change userID's team
|
||||||
|
|
||||||
userID -- id of user
|
:param userID: id of user
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
|
# Make sure this match's mode has teams
|
||||||
|
if self.matchTeamType != matchTeamTypes.TEAM_VS and self.matchTeamType != matchTeamTypes.TAG_TEAM_VS:
|
||||||
|
return
|
||||||
|
|
||||||
|
# Make sure the match is not locked
|
||||||
|
if self.isLocked or self.isStarting:
|
||||||
|
return
|
||||||
|
|
||||||
# Make sure the user is in room
|
# Make sure the user is in room
|
||||||
slotID = self.getUserSlotID(userID)
|
slotID = self.getUserSlotID(userID)
|
||||||
if slotID is None:
|
if slotID is None:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Update slot and send update
|
# Update slot and send update
|
||||||
newTeam = matchTeams.blue if self.slots[slotID].team == matchTeams.red else matchTeams.red
|
if newTeam is None:
|
||||||
|
newTeam = matchTeams.BLUE if self.slots[slotID].team == matchTeams.RED else matchTeams.RED
|
||||||
self.setSlot(slotID, None, newTeam)
|
self.setSlot(slotID, None, newTeam)
|
||||||
self.sendUpdates()
|
self.sendUpdates()
|
||||||
|
|
||||||
def sendUpdates(self):
|
def sendUpdates(self):
|
||||||
|
"""
|
||||||
|
Send match updates packet to everyone in lobby and room streams
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
self.matchDataCache = serverPackets.updateMatch(self.matchID)
|
self.matchDataCache = serverPackets.updateMatch(self.matchID)
|
||||||
|
censoredDataCache = serverPackets.updateMatch(self.matchID, censored=True)
|
||||||
if self.matchDataCache is not None:
|
if self.matchDataCache is not None:
|
||||||
glob.streams.broadcast(self.streamName, self.matchDataCache)
|
glob.streams.broadcast(self.streamName, self.matchDataCache)
|
||||||
glob.streams.broadcast("lobby", self.matchDataCache)
|
if censoredDataCache is not None:
|
||||||
|
glob.streams.broadcast("lobby", censoredDataCache)
|
||||||
else:
|
else:
|
||||||
log.error("MPROOM{}: Can't send match update packet, match data is None!!!".format(self.matchID))
|
log.error("MPROOM{}: Can't send match update packet, match data is None!!!".format(self.matchID))
|
||||||
|
|
||||||
|
@ -580,16 +751,17 @@ class match:
|
||||||
"""
|
"""
|
||||||
Check if match teams are valid
|
Check if match teams are valid
|
||||||
|
|
||||||
return -- True if valid, False if invalid
|
:return: True if valid, False if invalid
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
if self.matchTeamType != matchTeamTypes.teamVs or self.matchTeamType != matchTeamTypes.tagTeamVs:
|
if self.matchTeamType != matchTeamTypes.TEAM_VS and self.matchTeamType != matchTeamTypes.TAG_TEAM_VS:
|
||||||
# Teams are always valid if we have no teams
|
# Teams are always valid if we have no teams
|
||||||
return True
|
return True
|
||||||
|
|
||||||
# We have teams, check if they are valid
|
# We have teams, check if they are valid
|
||||||
firstTeam = -1
|
firstTeam = -1
|
||||||
for i in range(0,16):
|
for i in range(0,16):
|
||||||
if self.slots[i].user is not None and (self.slots[i].status & slotStatuses.noMap) == 0:
|
if self.slots[i].user is not None and (self.slots[i].status & slotStatuses.NO_MAP) == 0:
|
||||||
if firstTeam == -1:
|
if firstTeam == -1:
|
||||||
firstTeam = self.slots[i].team
|
firstTeam = self.slots[i].team
|
||||||
elif firstTeam != self.slots[i].team:
|
elif firstTeam != self.slots[i].team:
|
||||||
|
@ -600,21 +772,29 @@ class match:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def start(self):
|
def start(self):
|
||||||
|
"""
|
||||||
|
Start the match
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
# Remove isStarting timer flag thingie
|
||||||
|
self.isStarting = False
|
||||||
|
|
||||||
# Make sure we have enough players
|
# Make sure we have enough players
|
||||||
if self.countUsers() < 2 or not self.checkTeams():
|
if self.countUsers() < 2 or not self.checkTeams():
|
||||||
return
|
return False
|
||||||
|
|
||||||
# Create playing channel
|
# Create playing channel
|
||||||
glob.streams.add(self.playingStreamName)
|
glob.streams.add(self.playingStreamName)
|
||||||
|
|
||||||
# Change inProgress value
|
# Change inProgress value
|
||||||
match.inProgress = True
|
self.inProgress = True
|
||||||
|
|
||||||
# Set playing to ready players and set load, skip and complete to False
|
# Set playing to ready players and set load, skip and complete to False
|
||||||
# Make clients join playing stream
|
# Make clients join playing stream
|
||||||
for i in range(0, 16):
|
for i in range(0, 16):
|
||||||
if (self.slots[i].status & slotStatuses.ready) > 0 and self.slots[i].user in glob.tokens.tokens:
|
if self.slots[i].user in glob.tokens.tokens:
|
||||||
self.slots[i].status = slotStatuses.playing
|
self.slots[i].status = slotStatuses.PLAYING
|
||||||
self.slots[i].loaded = False
|
self.slots[i].loaded = False
|
||||||
self.slots[i].skip = False
|
self.slots[i].skip = False
|
||||||
self.slots[i].complete = False
|
self.slots[i].complete = False
|
||||||
|
@ -625,3 +805,85 @@ class match:
|
||||||
|
|
||||||
# Send updates
|
# Send updates
|
||||||
self.sendUpdates()
|
self.sendUpdates()
|
||||||
|
return True
|
||||||
|
|
||||||
|
def forceSize(self, matchSize):
|
||||||
|
for i in range(0, matchSize):
|
||||||
|
if self.slots[i].status == slotStatuses.LOCKED:
|
||||||
|
self.toggleSlotLocked(i)
|
||||||
|
for i in range(matchSize, 16):
|
||||||
|
if self.slots[i].status != slotStatuses.LOCKED:
|
||||||
|
self.toggleSlotLocked(i)
|
||||||
|
|
||||||
|
def abort(self):
|
||||||
|
if not self.inProgress:
|
||||||
|
log.warning("MPROOM{}: Match is not in progress!".format(self.matchID))
|
||||||
|
return
|
||||||
|
self.inProgress = False
|
||||||
|
self.isStarting = False
|
||||||
|
self.resetSlots()
|
||||||
|
self.sendUpdates()
|
||||||
|
glob.streams.broadcast(self.playingStreamName, serverPackets.matchAbort())
|
||||||
|
glob.streams.dispose(self.playingStreamName)
|
||||||
|
glob.streams.remove(self.playingStreamName)
|
||||||
|
log.info("MPROOM{}: Match aborted".format(self.matchID))
|
||||||
|
|
||||||
|
def initializeTeams(self):
|
||||||
|
if self.matchTeamType == matchTeamTypes.TEAM_VS or self.matchTeamType == matchTeamTypes.TAG_TEAM_VS:
|
||||||
|
# Set teams
|
||||||
|
for i, _slot in enumerate(self.slots):
|
||||||
|
_slot.team = matchTeams.RED if i % 2 == 0 else matchTeams.BLUE
|
||||||
|
else:
|
||||||
|
# Reset teams
|
||||||
|
for _slot in self.slots:
|
||||||
|
_slot.team = matchTeams.NO_TEAM
|
||||||
|
|
||||||
|
def resetMods(self):
|
||||||
|
for _slot in self.slots:
|
||||||
|
_slot.mods = 0
|
||||||
|
|
||||||
|
def resetReady(self):
|
||||||
|
for _slot in self.slots:
|
||||||
|
if _slot.status == slotStatuses.READY:
|
||||||
|
_slot.status = slotStatuses.NOT_READY
|
||||||
|
|
||||||
|
def sendReadyStatus(self):
|
||||||
|
chanName = "#multi_{}".format(self.matchID)
|
||||||
|
|
||||||
|
# Make sure match exists before attempting to do anything else
|
||||||
|
if chanName not in glob.channels.channels:
|
||||||
|
return
|
||||||
|
|
||||||
|
totalUsers = 0
|
||||||
|
readyUsers = 0
|
||||||
|
|
||||||
|
for slot in self.slots:
|
||||||
|
# Make sure there is a user in this slot
|
||||||
|
if slot.user is None:
|
||||||
|
continue
|
||||||
|
|
||||||
|
# In this slot there is a user, so we increase the amount of total users
|
||||||
|
# in this multi room.
|
||||||
|
totalUsers += 1
|
||||||
|
|
||||||
|
if slot.status == slotStatuses.READY:
|
||||||
|
readyUsers += 1
|
||||||
|
|
||||||
|
message = "{} users ready out of {}.".format(readyUsers, totalUsers)
|
||||||
|
|
||||||
|
if totalUsers == readyUsers:
|
||||||
|
message += " All users ready!"
|
||||||
|
|
||||||
|
# Check whether there is anyone left in this match.
|
||||||
|
if totalUsers == 0:
|
||||||
|
message = "The match is now empty."
|
||||||
|
|
||||||
|
chat.sendMessage(glob.BOT_NAME, chanName, message)
|
||||||
|
|
||||||
|
def __enter__(self):
|
||||||
|
# 🌚🌚🌚🌚🌚
|
||||||
|
self._lock.acquire()
|
||||||
|
return self
|
||||||
|
|
||||||
|
def __exit__(self, exc_type, exc_val, exc_tb):
|
||||||
|
self._lock.release()
|
||||||
|
|
|
@ -1,3 +1,8 @@
|
||||||
|
import threading
|
||||||
|
import time
|
||||||
|
|
||||||
|
from common.sentry import sentry
|
||||||
|
from constants.exceptions import periodicLoopException
|
||||||
from objects import match
|
from objects import match
|
||||||
from objects import glob
|
from objects import glob
|
||||||
from constants import serverPackets
|
from constants import serverPackets
|
||||||
|
@ -9,39 +14,99 @@ class matchList:
|
||||||
self.matches = {}
|
self.matches = {}
|
||||||
self.lastID = 1
|
self.lastID = 1
|
||||||
|
|
||||||
def createMatch(self, matchName, matchPassword, beatmapID, beatmapName, beatmapMD5, gameMode, hostUserID):
|
def createMatch(self, matchName, matchPassword, beatmapID, beatmapName, beatmapMD5, gameMode, hostUserID, isTourney=False):
|
||||||
"""
|
"""
|
||||||
Add a new match to matches list
|
Add a new match to matches list
|
||||||
|
|
||||||
matchName -- match name, string
|
:param matchName: match name, string
|
||||||
matchPassword -- match md5 password. Leave empty for no password
|
:param matchPassword: match md5 password. Leave empty for no password
|
||||||
beatmapID -- beatmap ID
|
:param beatmapID: beatmap ID
|
||||||
beatmapName -- beatmap name, string
|
:param beatmapName: beatmap name, string
|
||||||
beatmapMD5 -- beatmap md5 hash, string
|
:param beatmapMD5: beatmap md5 hash, string
|
||||||
gameMode -- game mode ID. See gameModes.py
|
:param gameMode: game mode ID. See gameModes.py
|
||||||
hostUserID -- user id of who created the match
|
:param hostUserID: user id of who created the match
|
||||||
return -- match ID
|
:return: match ID
|
||||||
"""
|
"""
|
||||||
# Add a new match to matches list and create its stream
|
# Add a new match to matches list and create its stream
|
||||||
matchID = self.lastID
|
matchID = self.lastID
|
||||||
self.lastID+=1
|
self.lastID+=1
|
||||||
self.matches[matchID] = match.match(matchID, matchName, matchPassword, beatmapID, beatmapName, beatmapMD5, gameMode, hostUserID)
|
self.matches[matchID] = match.match(matchID, matchName, matchPassword, beatmapID, beatmapName, beatmapMD5, gameMode, hostUserID, isTourney)
|
||||||
return matchID
|
return matchID
|
||||||
|
|
||||||
def disposeMatch(self, matchID):
|
def disposeMatch(self, matchID):
|
||||||
"""
|
"""
|
||||||
Destroy match object with id = matchID
|
Destroy match object with id = matchID
|
||||||
|
|
||||||
matchID -- ID of match to dispose
|
:param matchID: ID of match to dispose
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Make sure the match exists
|
# Make sure the match exists
|
||||||
if matchID not in self.matches:
|
if matchID not in self.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Remove match object and stream
|
# Get match and disconnect all players
|
||||||
match = self.matches.pop(matchID)
|
_match = self.matches[matchID]
|
||||||
glob.streams.remove(match.streamName)
|
for slot in _match.slots:
|
||||||
glob.streams.remove(match.playingStreamName)
|
_token = glob.tokens.getTokenFromUserID(slot.userID, ignoreIRC=True)
|
||||||
|
if _token is None:
|
||||||
|
continue
|
||||||
|
_match.userLeft(_token, disposeMatch=False) # don't dispose the match twice when we remove all players
|
||||||
|
|
||||||
|
# Delete chat channel
|
||||||
|
glob.channels.removeChannel("#multi_{}".format(_match.matchID))
|
||||||
|
|
||||||
|
# Send matchDisposed packet before disposing streams
|
||||||
|
glob.streams.broadcast(_match.streamName, serverPackets.disposeMatch(_match.matchID))
|
||||||
|
|
||||||
|
# Dispose all streams
|
||||||
|
glob.streams.dispose(_match.streamName)
|
||||||
|
glob.streams.dispose(_match.playingStreamName)
|
||||||
|
glob.streams.remove(_match.streamName)
|
||||||
|
glob.streams.remove(_match.playingStreamName)
|
||||||
|
|
||||||
# Send match dispose packet to everyone in lobby
|
# Send match dispose packet to everyone in lobby
|
||||||
glob.streams.broadcast("lobby", serverPackets.disposeMatch(matchID))
|
glob.streams.broadcast("lobby", serverPackets.disposeMatch(matchID))
|
||||||
|
del self.matches[matchID]
|
||||||
|
log.info("MPROOM{}: Room disposed manually".format(_match.matchID))
|
||||||
|
|
||||||
|
@sentry.capture()
|
||||||
|
def cleanupLoop(self):
|
||||||
|
"""
|
||||||
|
Start match cleanup loop.
|
||||||
|
Empty matches that have been created more than 60 seconds ago will get deleted.
|
||||||
|
Useful when people create useless lobbies with `!mp make`.
|
||||||
|
The check is done every 30 seconds.
|
||||||
|
This method starts an infinite loop, call it only once!
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
log.debug("Checking empty matches")
|
||||||
|
t = int(time.time())
|
||||||
|
emptyMatches = []
|
||||||
|
exceptions = []
|
||||||
|
|
||||||
|
# Collect all empty matches
|
||||||
|
for key, m in self.matches.items():
|
||||||
|
if [x for x in m.slots if x.user is not None]:
|
||||||
|
continue
|
||||||
|
if t - m.createTime >= 120:
|
||||||
|
log.debug("Match #{} marked for cleanup".format(m.matchID))
|
||||||
|
emptyMatches.append(m.matchID)
|
||||||
|
|
||||||
|
# Dispose all empty matches
|
||||||
|
for matchID in emptyMatches:
|
||||||
|
try:
|
||||||
|
self.disposeMatch(matchID)
|
||||||
|
except Exception as e:
|
||||||
|
exceptions.append(e)
|
||||||
|
log.error(
|
||||||
|
"Something wrong happened while disposing a timed out match. Reporting to Sentry when "
|
||||||
|
"the loop ends."
|
||||||
|
)
|
||||||
|
|
||||||
|
# Re-raise exception if needed
|
||||||
|
if exceptions:
|
||||||
|
raise periodicLoopException(exceptions)
|
||||||
|
finally:
|
||||||
|
# Schedule a new check (endless loop)
|
||||||
|
threading.Timer(30, self.cleanupLoop).start()
|
||||||
|
|
|
@ -5,6 +5,7 @@ import uuid
|
||||||
from common.constants import gameModes, actions
|
from common.constants import gameModes, actions
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from common.ripple import userUtils
|
from common.ripple import userUtils
|
||||||
|
from constants import exceptions
|
||||||
from constants import serverPackets
|
from constants import serverPackets
|
||||||
from events import logoutEvent
|
from events import logoutEvent
|
||||||
from helpers import chatHelper as chat
|
from helpers import chatHelper as chat
|
||||||
|
@ -12,31 +13,35 @@ from objects import glob
|
||||||
|
|
||||||
|
|
||||||
class token:
|
class token:
|
||||||
|
|
||||||
def __init__(self, userID, token_ = None, ip ="", irc = False, timeOffset = 0, tournament = False):
|
def __init__(self, userID, token_ = None, ip ="", irc = False, timeOffset = 0, tournament = False):
|
||||||
"""
|
"""
|
||||||
Create a token object and set userID and token
|
Create a token object and set userID and token
|
||||||
|
|
||||||
userID -- user associated to this token
|
:param userID: user associated to this token
|
||||||
token -- if passed, set token to that value
|
:param token_: if passed, set token to that value
|
||||||
if not passed, token will be generated
|
if not passed, token will be generated
|
||||||
ip -- client ip. optional.
|
:param ip: client ip. optional.
|
||||||
irc -- if True, set this token as IRC client. optional.
|
:param irc: if True, set this token as IRC client. Default: False.
|
||||||
timeOffset -- the time offset from UTC for this user. optional.
|
:param timeOffset: the time offset from UTC for this user. Default: 0.
|
||||||
|
:param tournament: if True, flag this client as a tournement client. Default: True.
|
||||||
"""
|
"""
|
||||||
# Set stuff
|
# Set stuff
|
||||||
self.userID = userID
|
self.userID = userID
|
||||||
self.username = userUtils.getUsername(self.userID)
|
self.username = userUtils.getUsername(self.userID)
|
||||||
|
self.safeUsername = userUtils.getSafeUsername(self.userID)
|
||||||
self.privileges = userUtils.getPrivileges(self.userID)
|
self.privileges = userUtils.getPrivileges(self.userID)
|
||||||
self.admin = userUtils.isInPrivilegeGroup(self.userID, "developer") or userUtils.isInPrivilegeGroup(self.userID, "community manager")
|
self.admin = userUtils.isInPrivilegeGroup(self.userID, "developer")\
|
||||||
|
or userUtils.isInPrivilegeGroup(self.userID, "community manager")\
|
||||||
|
or userUtils.isInPrivilegeGroup(self.userID, "chat mod")
|
||||||
self.irc = irc
|
self.irc = irc
|
||||||
|
self.kicked = False
|
||||||
self.restricted = userUtils.isRestricted(self.userID)
|
self.restricted = userUtils.isRestricted(self.userID)
|
||||||
self.loginTime = int(time.time())
|
self.loginTime = int(time.time())
|
||||||
self.pingTime = self.loginTime
|
self.pingTime = self.loginTime
|
||||||
self.timeOffset = timeOffset
|
self.timeOffset = timeOffset
|
||||||
self.lock = threading.Lock() # Sync primitive
|
|
||||||
self.streams = []
|
self.streams = []
|
||||||
self.tournament = tournament
|
self.tournament = tournament
|
||||||
|
self.messagesBuffer = []
|
||||||
|
|
||||||
# Default variables
|
# Default variables
|
||||||
self.spectators = []
|
self.spectators = []
|
||||||
|
@ -51,6 +56,7 @@ class token:
|
||||||
self.country = 0
|
self.country = 0
|
||||||
self.location = [0,0]
|
self.location = [0,0]
|
||||||
self.awayMessage = ""
|
self.awayMessage = ""
|
||||||
|
self.sentAway = []
|
||||||
self.matchID = -1
|
self.matchID = -1
|
||||||
self.tillerino = [0,0,-1.0] # beatmap, mods, acc
|
self.tillerino = [0,0,-1.0] # beatmap, mods, acc
|
||||||
self.silenceEndTime = 0
|
self.silenceEndTime = 0
|
||||||
|
@ -79,6 +85,11 @@ class token:
|
||||||
else:
|
else:
|
||||||
self.token = str(uuid.uuid4())
|
self.token = str(uuid.uuid4())
|
||||||
|
|
||||||
|
# Locks
|
||||||
|
self.processingLock = threading.Lock() # Acquired while there's an incoming packet from this user
|
||||||
|
self._bufferLock = threading.Lock() # Acquired while writing to packets buffer
|
||||||
|
self._spectLock = threading.RLock()
|
||||||
|
|
||||||
# Set stats
|
# Set stats
|
||||||
self.updateCachedStats()
|
self.updateCachedStats()
|
||||||
|
|
||||||
|
@ -93,51 +104,72 @@ class token:
|
||||||
"""
|
"""
|
||||||
Add bytes (packets) to queue
|
Add bytes (packets) to queue
|
||||||
|
|
||||||
bytes -- (packet) bytes to enqueue
|
:param bytes_: (packet) bytes to enqueue
|
||||||
"""
|
"""
|
||||||
if not self.irc:
|
try:
|
||||||
|
# Acquire the buffer lock
|
||||||
|
self._bufferLock.acquire()
|
||||||
|
|
||||||
|
# Never enqueue for IRC clients or Foka
|
||||||
|
if self.irc or self.userID < 999:
|
||||||
|
return
|
||||||
|
|
||||||
|
# Avoid memory leaks
|
||||||
if len(bytes_) < 10 * 10 ** 6:
|
if len(bytes_) < 10 * 10 ** 6:
|
||||||
self.queue += bytes_
|
self.queue += bytes_
|
||||||
else:
|
else:
|
||||||
log.warning("{}'s packets buffer is above 10M!! Lost some data!".format(self.username))
|
log.warning("{}'s packets buffer is above 10M!! Lost some data!".format(self.username))
|
||||||
|
finally:
|
||||||
|
# Release the buffer lock
|
||||||
|
self._bufferLock.release()
|
||||||
|
|
||||||
def resetQueue(self):
|
def resetQueue(self):
|
||||||
"""Resets the queue. Call when enqueued packets have been sent"""
|
"""Resets the queue. Call when enqueued packets have been sent"""
|
||||||
|
try:
|
||||||
|
self._bufferLock.acquire()
|
||||||
self.queue = bytes()
|
self.queue = bytes()
|
||||||
|
finally:
|
||||||
|
self._bufferLock.release()
|
||||||
|
|
||||||
|
def joinChannel(self, channelObject):
|
||||||
def joinChannel(self, channel):
|
|
||||||
"""
|
"""
|
||||||
Add channel to joined channels list
|
Join a channel
|
||||||
|
|
||||||
channel -- channel name
|
:param channelObject: channel object
|
||||||
|
:raises: exceptions.userAlreadyInChannelException()
|
||||||
|
exceptions.channelNoPermissionsException()
|
||||||
"""
|
"""
|
||||||
if channel not in self.joinedChannels:
|
if channelObject.name in self.joinedChannels:
|
||||||
self.joinedChannels.append(channel)
|
raise exceptions.userAlreadyInChannelException()
|
||||||
|
if not channelObject.publicRead and not self.admin:
|
||||||
|
raise exceptions.channelNoPermissionsException()
|
||||||
|
self.joinedChannels.append(channelObject.name)
|
||||||
|
self.joinStream("chat/{}".format(channelObject.name))
|
||||||
|
self.enqueue(serverPackets.channelJoinSuccess(self.userID, channelObject.clientName))
|
||||||
|
|
||||||
def partChannel(self, channel):
|
def partChannel(self, channelObject):
|
||||||
"""
|
"""
|
||||||
Remove channel from joined channels list
|
Remove channel from joined channels list
|
||||||
|
|
||||||
channel -- channel name
|
:param channelObject: channel object
|
||||||
"""
|
"""
|
||||||
if channel in self.joinedChannels:
|
self.joinedChannels.remove(channelObject.name)
|
||||||
self.joinedChannels.remove(channel)
|
self.leaveStream("chat/{}".format(channelObject.name))
|
||||||
|
|
||||||
def setLocation(self, location):
|
def setLocation(self, latitude, longitude):
|
||||||
"""
|
"""
|
||||||
Set location (latitude and longitude)
|
Set client location
|
||||||
|
|
||||||
location -- [latitude, longitude]
|
:param latitude: latitude
|
||||||
|
:param longitude: longitude
|
||||||
"""
|
"""
|
||||||
self.location = location
|
self.location = (latitude, longitude)
|
||||||
|
|
||||||
def getLatitude(self):
|
def getLatitude(self):
|
||||||
"""
|
"""
|
||||||
Get latitude
|
Get latitude
|
||||||
|
|
||||||
return -- latitude
|
:return: latitude
|
||||||
"""
|
"""
|
||||||
return self.location[0]
|
return self.location[0]
|
||||||
|
|
||||||
|
@ -145,16 +177,20 @@ class token:
|
||||||
"""
|
"""
|
||||||
Get longitude
|
Get longitude
|
||||||
|
|
||||||
return -- longitude
|
:return: longitude
|
||||||
"""
|
"""
|
||||||
return self.location[1]
|
return self.location[1]
|
||||||
|
|
||||||
def startSpectating(self, host):
|
def startSpectating(self, host):
|
||||||
"""
|
"""
|
||||||
Set the spectating user to userID
|
Set the spectating user to userID, join spectator stream and chat channel
|
||||||
|
and send required packets to host
|
||||||
|
|
||||||
user -- user object
|
:param host: host osuToken object
|
||||||
"""
|
"""
|
||||||
|
try:
|
||||||
|
self._spectLock.acquire()
|
||||||
|
|
||||||
# Stop spectating old client
|
# Stop spectating old client
|
||||||
self.stopSpectating()
|
self.stopSpectating()
|
||||||
|
|
||||||
|
@ -176,10 +212,10 @@ class token:
|
||||||
|
|
||||||
# Create and join #spectator (#spect_userid) channel
|
# Create and join #spectator (#spect_userid) channel
|
||||||
glob.channels.addTempChannel("#spect_{}".format(host.userID))
|
glob.channels.addTempChannel("#spect_{}".format(host.userID))
|
||||||
chat.joinChannel(token=self, channel="#spect_{}".format(host.userID))
|
chat.joinChannel(token=self, channel="#spect_{}".format(host.userID), force=True)
|
||||||
if len(host.spectators) == 1:
|
if len(host.spectators) == 1:
|
||||||
# First spectator, send #spectator join to host too
|
# First spectator, send #spectator join to host too
|
||||||
chat.joinChannel(token=host, channel="#spect_{}".format(host.userID))
|
chat.joinChannel(token=host, channel="#spect_{}".format(host.userID), force=True)
|
||||||
|
|
||||||
# Send fellow spectator join to all clients
|
# Send fellow spectator join to all clients
|
||||||
glob.streams.broadcast(streamName, serverPackets.fellowSpectatorJoined(self.userID))
|
glob.streams.broadcast(streamName, serverPackets.fellowSpectatorJoined(self.userID))
|
||||||
|
@ -191,10 +227,21 @@ class token:
|
||||||
|
|
||||||
# Log
|
# Log
|
||||||
log.info("{} is spectating {}".format(self.username, host.username))
|
log.info("{} is spectating {}".format(self.username, host.username))
|
||||||
|
finally:
|
||||||
|
self._spectLock.release()
|
||||||
|
|
||||||
def stopSpectating(self):
|
def stopSpectating(self):
|
||||||
|
"""
|
||||||
|
Stop spectating, leave spectator stream and channel
|
||||||
|
and send required packets to host
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
self._spectLock.acquire()
|
||||||
|
|
||||||
# Remove our userID from host's spectators
|
# Remove our userID from host's spectators
|
||||||
if self.spectating is None:
|
if self.spectating is None or self.spectatingUserID <= 0:
|
||||||
return
|
return
|
||||||
if self.spectating in glob.tokens.tokens:
|
if self.spectating in glob.tokens.tokens:
|
||||||
hostToken = glob.tokens.tokens[self.spectating]
|
hostToken = glob.tokens.tokens[self.spectating]
|
||||||
|
@ -218,49 +265,35 @@ class token:
|
||||||
# If nobody is spectating the host anymore, close #spectator channel
|
# If nobody is spectating the host anymore, close #spectator channel
|
||||||
# and remove host from spect stream too
|
# and remove host from spect stream too
|
||||||
if len(hostToken.spectators) == 0:
|
if len(hostToken.spectators) == 0:
|
||||||
chat.partChannel(token=hostToken, channel="#spect_{}".format(hostToken.userID), kick=True)
|
chat.partChannel(token=hostToken, channel="#spect_{}".format(hostToken.userID), kick=True, force=True)
|
||||||
hostToken.leaveStream(streamName)
|
hostToken.leaveStream(streamName)
|
||||||
|
|
||||||
# Part #spectator channel
|
|
||||||
chat.partChannel(token=self, channel="#spect_{}".format(self.spectatingUserID), kick=True)
|
|
||||||
|
|
||||||
# Console output
|
# Console output
|
||||||
log.info("{} is no longer spectating {}".format(self.username, self.spectatingUserID))
|
log.info("{} is no longer spectating {}. Current spectators: {}".format(self.username, self.spectatingUserID, hostToken.spectators))
|
||||||
|
|
||||||
|
# Part #spectator channel
|
||||||
|
chat.partChannel(token=self, channel="#spect_{}".format(self.spectatingUserID), kick=True, force=True)
|
||||||
|
|
||||||
# Set our spectating user to 0
|
# Set our spectating user to 0
|
||||||
self.spectating = None
|
self.spectating = None
|
||||||
self.spectatingUserID = 0
|
self.spectatingUserID = 0
|
||||||
|
finally:
|
||||||
def setCountry(self, countryID):
|
self._spectLock.release()
|
||||||
"""
|
|
||||||
Set country to countryID
|
|
||||||
|
|
||||||
countryID -- numeric country ID. See countryHelper.py
|
|
||||||
"""
|
|
||||||
self.country = countryID
|
|
||||||
|
|
||||||
def getCountry(self):
|
|
||||||
"""
|
|
||||||
Get numeric country ID
|
|
||||||
|
|
||||||
return -- numeric country ID. See countryHelper.py
|
|
||||||
"""
|
|
||||||
return self.country
|
|
||||||
|
|
||||||
def updatePingTime(self):
|
def updatePingTime(self):
|
||||||
"""Update latest ping time"""
|
"""
|
||||||
|
Update latest ping time to current time
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
self.pingTime = int(time.time())
|
self.pingTime = int(time.time())
|
||||||
|
|
||||||
def setAwayMessage(self, __awayMessage):
|
|
||||||
"""Set a new away message"""
|
|
||||||
self.awayMessage = __awayMessage
|
|
||||||
|
|
||||||
|
|
||||||
def joinMatch(self, matchID):
|
def joinMatch(self, matchID):
|
||||||
"""
|
"""
|
||||||
Set match to matchID, join match stream and channel
|
Set match to matchID, join match stream and channel
|
||||||
|
|
||||||
matchID -- new match ID
|
:param matchID: new match ID
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Make sure the match exists
|
# Make sure the match exists
|
||||||
if matchID not in glob.matches.matches:
|
if matchID not in glob.matches.matches:
|
||||||
|
@ -285,9 +318,16 @@ class token:
|
||||||
# Set matchID, join stream, channel and send packet
|
# Set matchID, join stream, channel and send packet
|
||||||
self.matchID = matchID
|
self.matchID = matchID
|
||||||
self.joinStream(match.streamName)
|
self.joinStream(match.streamName)
|
||||||
chat.joinChannel(token=self, channel="#multi_{}".format(self.matchID))
|
chat.joinChannel(token=self, channel="#multi_{}".format(self.matchID), force=True)
|
||||||
self.enqueue(serverPackets.matchJoinSuccess(matchID))
|
self.enqueue(serverPackets.matchJoinSuccess(matchID))
|
||||||
|
|
||||||
|
if match.isTourney:
|
||||||
|
# Alert the user if we have just joined a tourney match
|
||||||
|
self.enqueue(serverPackets.notification("You are now in a tournament match."))
|
||||||
|
# If an user joins, then the ready status of the match changes and
|
||||||
|
# maybe not all users are ready.
|
||||||
|
match.sendReadyStatus()
|
||||||
|
|
||||||
def leaveMatch(self):
|
def leaveMatch(self):
|
||||||
"""
|
"""
|
||||||
Leave joined match, match stream and match channel
|
Leave joined match, match stream and match channel
|
||||||
|
@ -299,28 +339,37 @@ class token:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Part #multiplayer channel and streams (/ and /playing)
|
# Part #multiplayer channel and streams (/ and /playing)
|
||||||
chat.partChannel(token=self, channel="#multi_{}".format(self.matchID), kick=True)
|
chat.partChannel(token=self, channel="#multi_{}".format(self.matchID), kick=True, force=True)
|
||||||
self.leaveStream("multi/{}".format(self.matchID))
|
self.leaveStream("multi/{}".format(self.matchID))
|
||||||
self.leaveStream("multi/{}/playing".format(self.matchID)) # optional
|
self.leaveStream("multi/{}/playing".format(self.matchID)) # optional
|
||||||
|
|
||||||
|
# Set usertoken match to -1
|
||||||
|
leavingMatchID = self.matchID
|
||||||
|
self.matchID = -1
|
||||||
|
|
||||||
# Make sure the match exists
|
# Make sure the match exists
|
||||||
if self.matchID not in glob.matches.matches:
|
if leavingMatchID not in glob.matches.matches:
|
||||||
return
|
return
|
||||||
|
|
||||||
# The match exists, get object
|
# The match exists, get object
|
||||||
match = glob.matches.matches[self.matchID]
|
match = glob.matches.matches[leavingMatchID]
|
||||||
|
|
||||||
# Set slot to free
|
# Set slot to free
|
||||||
match.userLeft(self)
|
match.userLeft(self)
|
||||||
|
|
||||||
# Set usertoken match to -1
|
if match.isTourney:
|
||||||
self.matchID = -1
|
# If an user leaves, then the ready status of the match changes and
|
||||||
|
# maybe all users are ready. Or maybe nobody is in the match anymore
|
||||||
|
match.sendReadyStatus()
|
||||||
|
|
||||||
def kick(self, message="You have been kicked from the server. Please login again.", reason="kick"):
|
def kick(self, message="You have been kicked from the server. Please login again.", reason="kick"):
|
||||||
"""
|
"""
|
||||||
Kick this user from the server
|
Kick this user from the server
|
||||||
|
|
||||||
message -- Notification message to send to this user. Optional.
|
:param message: Notification message to send to this user.
|
||||||
|
Default: "You have been kicked from the server. Please login again."
|
||||||
|
:param reason: Kick reason, used in logs. Default: "kick"
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Send packet to target
|
# Send packet to target
|
||||||
log.info("{} has been disconnected. ({})".format(self.username, reason))
|
log.info("{} has been disconnected. ({})".format(self.username, reason))
|
||||||
|
@ -329,21 +378,28 @@ class token:
|
||||||
self.enqueue(serverPackets.loginFailed())
|
self.enqueue(serverPackets.loginFailed())
|
||||||
|
|
||||||
# Logout event
|
# Logout event
|
||||||
logoutEvent.handle(self, None)
|
logoutEvent.handle(self, deleteToken=self.irc)
|
||||||
|
|
||||||
def silence(self, seconds, reason, author = 999):
|
def silence(self, seconds = None, reason = "", author = 999):
|
||||||
"""
|
"""
|
||||||
Silences this user (db, packet and token)
|
Silences this user (db, packet and token)
|
||||||
|
|
||||||
seconds -- silence length in seconds
|
:param seconds: silence length in seconds. If None, get it from db. Default: None
|
||||||
reason -- silence reason
|
:param reason: silence reason. Default: empty string
|
||||||
author -- userID of who has silenced the target. Optional. Default: 999 (fokabot)
|
:param author: userID of who has silenced the user. Default: 999 (FokaBot)
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
|
if seconds is None:
|
||||||
|
# Get silence expire from db if needed
|
||||||
|
seconds = max(0, userUtils.getSilenceEnd(self.userID) - int(time.time()))
|
||||||
|
else:
|
||||||
# Silence in db and token
|
# Silence in db and token
|
||||||
self.silenceEndTime = int(time.time())+seconds
|
|
||||||
userUtils.silence(self.userID, seconds, reason, author)
|
userUtils.silence(self.userID, seconds, reason, author)
|
||||||
|
|
||||||
# Send silence packet to target
|
# Silence token
|
||||||
|
self.silenceEndTime = int(time.time()) + seconds
|
||||||
|
|
||||||
|
# Send silence packet to user
|
||||||
self.enqueue(serverPackets.silenceEndTime(seconds))
|
self.enqueue(serverPackets.silenceEndTime(seconds))
|
||||||
|
|
||||||
# Send silenced packet to everyone else
|
# Send silenced packet to everyone else
|
||||||
|
@ -353,7 +409,8 @@ class token:
|
||||||
"""
|
"""
|
||||||
Silences the user if is spamming.
|
Silences the user if is spamming.
|
||||||
|
|
||||||
increaseSpamRate -- pass True if the user has sent a new message. Optional. Default: True
|
:param increaseSpamRate: set to True if the user has sent a new message. Default: True
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
# Increase the spam rate if needed
|
# Increase the spam rate if needed
|
||||||
if increaseSpamRate:
|
if increaseSpamRate:
|
||||||
|
@ -367,7 +424,7 @@ class token:
|
||||||
"""
|
"""
|
||||||
Returns True if this user is silenced, otherwise False
|
Returns True if this user is silenced, otherwise False
|
||||||
|
|
||||||
return -- True/False
|
:return: True if this user is silenced, otherwise False
|
||||||
"""
|
"""
|
||||||
return self.silenceEndTime-int(time.time()) > 0
|
return self.silenceEndTime-int(time.time()) > 0
|
||||||
|
|
||||||
|
@ -376,12 +433,16 @@ class token:
|
||||||
Returns the seconds left for this user's silence
|
Returns the seconds left for this user's silence
|
||||||
(0 if user is not silenced)
|
(0 if user is not silenced)
|
||||||
|
|
||||||
return -- silence seconds left
|
:return: silence seconds left (or 0)
|
||||||
"""
|
"""
|
||||||
return max(0, self.silenceEndTime-int(time.time()))
|
return max(0, self.silenceEndTime-int(time.time()))
|
||||||
|
|
||||||
def updateCachedStats(self):
|
def updateCachedStats(self):
|
||||||
"""Update all cached stats for this token"""
|
"""
|
||||||
|
Update all cached stats for this token
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
stats = userUtils.getUserStats(self.userID, self.gameMode)
|
stats = userUtils.getUserStats(self.userID, self.gameMode)
|
||||||
log.debug(str(stats))
|
log.debug(str(stats))
|
||||||
if stats is None:
|
if stats is None:
|
||||||
|
@ -394,36 +455,110 @@ class token:
|
||||||
self.gameRank = stats["gameRank"]
|
self.gameRank = stats["gameRank"]
|
||||||
self.pp = stats["pp"]
|
self.pp = stats["pp"]
|
||||||
|
|
||||||
def checkRestricted(self, force=False):
|
def checkRestricted(self):
|
||||||
"""
|
"""
|
||||||
Check if this token is restricted. If so, send fokabot message
|
Check if this token is restricted. If so, send fokabot message
|
||||||
|
|
||||||
force -- If True, get restricted value from db.
|
:return:
|
||||||
If false, get the cached one. Optional. Default: False
|
|
||||||
"""
|
"""
|
||||||
if force:
|
oldRestricted = self.restricted
|
||||||
self.restricted = userUtils.isRestricted(self.userID)
|
self.restricted = userUtils.isRestricted(self.userID)
|
||||||
if self.restricted:
|
if self.restricted:
|
||||||
self.setRestricted()
|
self.setRestricted()
|
||||||
|
elif not self.restricted and oldRestricted != self.restricted:
|
||||||
|
self.resetRestricted()
|
||||||
|
|
||||||
|
def checkBanned(self):
|
||||||
|
"""
|
||||||
|
Check if this user is banned. If so, disconnect it.
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
if userUtils.isBanned(self.userID):
|
||||||
|
self.enqueue(serverPackets.loginBanned())
|
||||||
|
logoutEvent.handle(self, deleteToken=False)
|
||||||
|
|
||||||
|
|
||||||
def setRestricted(self):
|
def setRestricted(self):
|
||||||
"""
|
"""
|
||||||
Set this token as restricted, send FokaBot message to user
|
Set this token as restricted, send FokaBot message to user
|
||||||
and send offline packet to everyone
|
and send offline packet to everyone
|
||||||
|
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
self.restricted = True
|
self.restricted = True
|
||||||
chat.sendMessage("FokaBot",self.username, "Your account is currently in restricted mode. Please visit ripple's website for more information.")
|
chat.sendMessage(glob.BOT_NAME, self.username, "Your account is currently in restricted mode. Please visit ripple's website for more information.")
|
||||||
|
|
||||||
|
def resetRestricted(self):
|
||||||
|
"""
|
||||||
|
Send FokaBot message to alert the user that he has been unrestricted
|
||||||
|
and he has to log in again.
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
chat.sendMessage(glob.BOT_NAME, self.username, "Your account has been unrestricted! Please log in again.")
|
||||||
|
|
||||||
def joinStream(self, name):
|
def joinStream(self, name):
|
||||||
|
"""
|
||||||
|
Join a packet stream, or create it if the stream doesn't exist.
|
||||||
|
|
||||||
|
:param name: stream name
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
glob.streams.join(name, token=self.token)
|
glob.streams.join(name, token=self.token)
|
||||||
if name not in self.streams:
|
if name not in self.streams:
|
||||||
self.streams.append(name)
|
self.streams.append(name)
|
||||||
|
|
||||||
def leaveStream(self, name):
|
def leaveStream(self, name):
|
||||||
|
"""
|
||||||
|
Leave a packets stream
|
||||||
|
|
||||||
|
:param name: stream name
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
glob.streams.leave(name, token=self.token)
|
glob.streams.leave(name, token=self.token)
|
||||||
if name in self.streams:
|
if name in self.streams:
|
||||||
self.streams.remove(name)
|
self.streams.remove(name)
|
||||||
|
|
||||||
def leaveAllStreams(self):
|
def leaveAllStreams(self):
|
||||||
|
"""
|
||||||
|
Leave all joined packet streams
|
||||||
|
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
for i in self.streams:
|
for i in self.streams:
|
||||||
self.leaveStream(i)
|
self.leaveStream(i)
|
||||||
|
|
||||||
|
def awayCheck(self, userID):
|
||||||
|
"""
|
||||||
|
Returns True if userID doesn't know that we are away
|
||||||
|
Returns False if we are not away or if userID already knows we are away
|
||||||
|
|
||||||
|
:param userID: original sender userID
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
if self.awayMessage == "" or userID in self.sentAway:
|
||||||
|
return False
|
||||||
|
self.sentAway.append(userID)
|
||||||
|
return True
|
||||||
|
|
||||||
|
def addMessageInBuffer(self, chan, message):
|
||||||
|
"""
|
||||||
|
Add a message in messages buffer (10 messages, truncated at 50 chars).
|
||||||
|
Used as proof when the user gets reported.
|
||||||
|
|
||||||
|
:param chan: channel
|
||||||
|
:param message: message content
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
if len(self.messagesBuffer) > 9:
|
||||||
|
self.messagesBuffer = self.messagesBuffer[1:]
|
||||||
|
self.messagesBuffer.append("{time} - {user}@{channel}: {message}".format(time=time.strftime("%H:%M", time.localtime()), user=self.username, channel=chan, message=message[:50]))
|
||||||
|
|
||||||
|
def getMessagesBufferString(self):
|
||||||
|
"""
|
||||||
|
Get the content of the messages buffer as a string
|
||||||
|
|
||||||
|
:return: messages buffer content as a string
|
||||||
|
"""
|
||||||
|
return "\n".join(x for x in self.messagesBuffer)
|
|
@ -43,15 +43,29 @@ class stream:
|
||||||
log.info("{} has left stream {}".format(token, self.name))
|
log.info("{} has left stream {}".format(token, self.name))
|
||||||
self.clients.remove(token)
|
self.clients.remove(token)
|
||||||
|
|
||||||
def broadcast(self, data):
|
def broadcast(self, data, but=None):
|
||||||
"""
|
"""
|
||||||
Send some data to all clients connected to this stream
|
Send some data to all (or some) clients connected to this stream
|
||||||
|
|
||||||
:param data: data to send
|
:param data: data to send
|
||||||
|
:param but: array of tokens to ignore. Default: None (send to everyone)
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
if but is None:
|
||||||
|
but = []
|
||||||
|
for i in self.clients:
|
||||||
|
if i in glob.tokens.tokens:
|
||||||
|
if i not in but:
|
||||||
|
glob.tokens.tokens[i].enqueue(data)
|
||||||
|
else:
|
||||||
|
self.removeClient(token=i)
|
||||||
|
|
||||||
|
def dispose(self):
|
||||||
|
"""
|
||||||
|
Tell every client in this stream to leave the stream
|
||||||
|
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
for i in self.clients:
|
for i in self.clients:
|
||||||
if i in glob.tokens.tokens:
|
if i in glob.tokens.tokens:
|
||||||
glob.tokens.tokens[i].enqueue(data)
|
glob.tokens.tokens[i].leaveStream(self.name)
|
||||||
else:
|
|
||||||
self.removeClient(token=i)
|
|
|
@ -1,6 +1,7 @@
|
||||||
from objects import stream
|
from objects import stream
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
|
||||||
|
# TODO: use *args and **kwargs
|
||||||
class streamList:
|
class streamList:
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.streams = {}
|
self.streams = {}
|
||||||
|
@ -55,25 +56,39 @@ class streamList:
|
||||||
return
|
return
|
||||||
self.streams[streamName].removeClient(client=client, token=token)
|
self.streams[streamName].removeClient(client=client, token=token)
|
||||||
|
|
||||||
def broadcast(self, streamName, data):
|
def broadcast(self, streamName, data, but=None):
|
||||||
"""
|
"""
|
||||||
Send some data to all clients in a stream
|
Send some data to all clients in a stream
|
||||||
|
|
||||||
:param streamName: stream name
|
:param streamName: stream name
|
||||||
:param data: data to send
|
:param data: data to send
|
||||||
|
:param but: array of tokens to ignore. Default: None (send to everyone)
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
if streamName not in self.streams:
|
if streamName not in self.streams:
|
||||||
return
|
return
|
||||||
self.streams[streamName].broadcast(data)
|
self.streams[streamName].broadcast(data, but)
|
||||||
|
|
||||||
'''def getClients(self, streamName):
|
def dispose(self, streamName, *args, **kwargs):
|
||||||
"""
|
"""
|
||||||
Get all clients in a stream
|
Call `dispose` on `streamName`
|
||||||
|
|
||||||
:param streamName: name of the stream
|
:param streamName: name of the stream
|
||||||
|
:param args:
|
||||||
|
:param kwargs:
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
if streamName not in self.streams:
|
if streamName not in self.streams:
|
||||||
return
|
return
|
||||||
return self.streams[streamName].clients'''
|
self.streams[streamName].dispose(*args, **kwargs)
|
||||||
|
|
||||||
|
def getStream(self, streamName):
|
||||||
|
"""
|
||||||
|
Returns streamName's stream object or None if it doesn't exist
|
||||||
|
|
||||||
|
:param streamName:
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
if streamName in self.streams:
|
||||||
|
return self.streams[streamName]
|
||||||
|
return None
|
|
@ -1,59 +1,65 @@
|
||||||
import threading
|
import threading
|
||||||
import time
|
import time
|
||||||
|
|
||||||
|
import redis
|
||||||
|
|
||||||
from common.ripple import userUtils
|
from common.ripple import userUtils
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
|
from common.sentry import sentry
|
||||||
from constants import serverPackets
|
from constants import serverPackets
|
||||||
|
from constants.exceptions import periodicLoopException
|
||||||
from events import logoutEvent
|
from events import logoutEvent
|
||||||
from objects import glob
|
from objects import glob
|
||||||
from objects import osuToken
|
from objects import osuToken
|
||||||
|
|
||||||
|
|
||||||
class tokenList:
|
class tokenList:
|
||||||
"""
|
|
||||||
List of connected osu tokens
|
|
||||||
|
|
||||||
tokens -- dictionary. key: token string, value: token object
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
"""
|
|
||||||
Initialize a tokens list
|
|
||||||
"""
|
|
||||||
self.tokens = {}
|
self.tokens = {}
|
||||||
|
self._lock = threading.Lock()
|
||||||
|
|
||||||
|
def __enter__(self):
|
||||||
|
self._lock.acquire()
|
||||||
|
|
||||||
|
def __exit__(self, exc_type, exc_val, exc_tb):
|
||||||
|
self._lock.release()
|
||||||
|
|
||||||
def addToken(self, userID, ip = "", irc = False, timeOffset=0, tournament=False):
|
def addToken(self, userID, ip = "", irc = False, timeOffset=0, tournament=False):
|
||||||
"""
|
"""
|
||||||
Add a token object to tokens list
|
Add a token object to tokens list
|
||||||
|
|
||||||
userID -- user id associated to that token
|
:param userID: user id associated to that token
|
||||||
irc -- if True, set this token as IRC client
|
:param ip: ip address of the client
|
||||||
return -- token object
|
:param irc: if True, set this token as IRC client
|
||||||
|
:param timeOffset: the time offset from UTC for this user. Default: 0.
|
||||||
|
:param tournament: if True, flag this client as a tournement client. Default: True.
|
||||||
|
:return: token object
|
||||||
"""
|
"""
|
||||||
newToken = osuToken.token(userID, ip=ip, irc=irc, timeOffset=timeOffset, tournament=tournament)
|
newToken = osuToken.token(userID, ip=ip, irc=irc, timeOffset=timeOffset, tournament=tournament)
|
||||||
self.tokens[newToken.token] = newToken
|
self.tokens[newToken.token] = newToken
|
||||||
|
glob.redis.incr("ripple:online_users")
|
||||||
return newToken
|
return newToken
|
||||||
|
|
||||||
def deleteToken(self, token):
|
def deleteToken(self, token):
|
||||||
"""
|
"""
|
||||||
Delete a token from token list if it exists
|
Delete a token from token list if it exists
|
||||||
|
|
||||||
token -- token string
|
:param token: token string
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
if token in self.tokens:
|
if token in self.tokens:
|
||||||
# Delete session from DB
|
|
||||||
if self.tokens[token].ip != "":
|
if self.tokens[token].ip != "":
|
||||||
userUtils.deleteBanchoSessions(self.tokens[token].userID, self.tokens[token].ip)
|
userUtils.deleteBanchoSessions(self.tokens[token].userID, self.tokens[token].ip)
|
||||||
|
t = self.tokens.pop(token)
|
||||||
# Pop token from list
|
del t
|
||||||
self.tokens.pop(token)
|
glob.redis.decr("ripple:online_users")
|
||||||
|
|
||||||
def getUserIDFromToken(self, token):
|
def getUserIDFromToken(self, token):
|
||||||
"""
|
"""
|
||||||
Get user ID from a token
|
Get user ID from a token
|
||||||
|
|
||||||
token -- token to find
|
:param token: token to find
|
||||||
return -- false if not found, userID if found
|
:return: False if not found, userID if found
|
||||||
"""
|
"""
|
||||||
# Make sure the token exists
|
# Make sure the token exists
|
||||||
if token not in self.tokens:
|
if token not in self.tokens:
|
||||||
|
@ -62,63 +68,91 @@ class tokenList:
|
||||||
# Get userID associated to that token
|
# Get userID associated to that token
|
||||||
return self.tokens[token].userID
|
return self.tokens[token].userID
|
||||||
|
|
||||||
def getTokenFromUserID(self, userID, ignoreIRC=False):
|
def getTokenFromUserID(self, userID, ignoreIRC=False, _all=False):
|
||||||
"""
|
"""
|
||||||
Get token from a user ID
|
Get token from a user ID
|
||||||
|
|
||||||
userID -- user ID to find
|
:param userID: user ID to find
|
||||||
return -- False if not found, token object if found
|
:param ignoreIRC: if True, consider bancho clients only and skip IRC clients
|
||||||
|
:param _all: if True, return a list with all clients that match given username, otherwise return
|
||||||
|
only the first occurrence.
|
||||||
|
:return: False if not found, token object if found
|
||||||
"""
|
"""
|
||||||
# Make sure the token exists
|
# Make sure the token exists
|
||||||
|
ret = []
|
||||||
for _, value in self.tokens.items():
|
for _, value in self.tokens.items():
|
||||||
if value.userID == userID:
|
if value.userID == userID:
|
||||||
if ignoreIRC and value.irc:
|
if ignoreIRC and value.irc:
|
||||||
continue
|
continue
|
||||||
|
if _all:
|
||||||
|
ret.append(value)
|
||||||
|
else:
|
||||||
return value
|
return value
|
||||||
|
|
||||||
# Return none if not found
|
# Return full list or None if not found
|
||||||
|
if _all:
|
||||||
|
return ret
|
||||||
|
else:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def getTokenFromUsername(self, username, ignoreIRC=False):
|
def getTokenFromUsername(self, username, ignoreIRC=False, safe=False, _all=False):
|
||||||
"""
|
"""
|
||||||
Get token from a username
|
Get an osuToken object from an username
|
||||||
|
|
||||||
username -- username to find
|
:param username: normal username or safe username
|
||||||
return -- False if not found, token object if found
|
:param ignoreIRC: if True, consider bancho clients only and skip IRC clients
|
||||||
|
:param safe: if True, username is a safe username,
|
||||||
|
compare it with token's safe username rather than normal username
|
||||||
|
:param _all: if True, return a list with all clients that match given username, otherwise return
|
||||||
|
only the first occurrence.
|
||||||
|
:return: osuToken object or None
|
||||||
"""
|
"""
|
||||||
# lowercase
|
# lowercase
|
||||||
who = username.lower()
|
who = username.lower() if not safe else username
|
||||||
|
|
||||||
# Make sure the token exists
|
# Make sure the token exists
|
||||||
|
ret = []
|
||||||
for _, value in self.tokens.items():
|
for _, value in self.tokens.items():
|
||||||
if value.username.lower() == who:
|
if (not safe and value.username.lower() == who) or (safe and value.safeUsername == who):
|
||||||
if ignoreIRC and value.irc:
|
if ignoreIRC and value.irc:
|
||||||
continue
|
continue
|
||||||
|
if _all:
|
||||||
|
ret.append(value)
|
||||||
|
else:
|
||||||
return value
|
return value
|
||||||
|
|
||||||
# Return none if not found
|
# Return full list or None if not found
|
||||||
|
if _all:
|
||||||
|
return ret
|
||||||
|
else:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def deleteOldTokens(self, userID):
|
def deleteOldTokens(self, userID):
|
||||||
"""
|
"""
|
||||||
Delete old userID's tokens if found
|
Delete old userID's tokens if found
|
||||||
|
|
||||||
userID -- tokens associated to this user will be deleted
|
:param userID: tokens associated to this user will be deleted
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Delete older tokens
|
# Delete older tokens
|
||||||
|
delete = []
|
||||||
for key, value in list(self.tokens.items()):
|
for key, value in list(self.tokens.items()):
|
||||||
if value.userID == userID:
|
if value.userID == userID:
|
||||||
# Delete this token from the dictionary
|
# Delete this token from the dictionary
|
||||||
self.tokens[key].kick("You have logged in from somewhere else. You can't connect to Bancho/IRC from more than one device at the same time.", "kicked, multiple clients")
|
#self.tokens[key].kick("You have logged in from somewhere else. You can't connect to Bancho/IRC from more than one device at the same time.", "kicked, multiple clients")
|
||||||
|
delete.append(self.tokens[key])
|
||||||
|
|
||||||
|
for i in delete:
|
||||||
|
logoutEvent.handle(i)
|
||||||
|
|
||||||
def multipleEnqueue(self, packet, who, but = False):
|
def multipleEnqueue(self, packet, who, but = False):
|
||||||
"""
|
"""
|
||||||
Enqueue a packet to multiple users
|
Enqueue a packet to multiple users
|
||||||
|
|
||||||
packet -- packet bytes to enqueue
|
:param packet: packet bytes to enqueue
|
||||||
who -- userIDs array
|
:param who: userIDs array
|
||||||
but -- if True, enqueue to everyone but users in who array
|
:param but: if True, enqueue to everyone but users in `who` array
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
for _, value in self.tokens.items():
|
for _, value in self.tokens.items():
|
||||||
shouldEnqueue = False
|
shouldEnqueue = False
|
||||||
|
@ -134,26 +168,28 @@ class tokenList:
|
||||||
"""
|
"""
|
||||||
Enqueue packet(s) to every connected user
|
Enqueue packet(s) to every connected user
|
||||||
|
|
||||||
packet -- packet bytes to enqueue
|
:param packet: packet bytes to enqueue
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
for _, value in self.tokens.items():
|
for _, value in self.tokens.items():
|
||||||
value.enqueue(packet)
|
value.enqueue(packet)
|
||||||
|
|
||||||
def usersTimeoutCheckLoop(self, timeoutTime = 100, checkTime = 100):
|
@sentry.capture()
|
||||||
|
def usersTimeoutCheckLoop(self):
|
||||||
"""
|
"""
|
||||||
Deletes all timed out users.
|
Start timed out users disconnect loop.
|
||||||
If called once, will recall after checkTime seconds and so on, forever
|
This function will be called every `checkTime` seconds and so on, forever.
|
||||||
CALL THIS FUNCTION ONLY ONCE!
|
CALL THIS FUNCTION ONLY ONCE!
|
||||||
|
:return:
|
||||||
timeoutTime - seconds of inactivity required to disconnect someone (Default: 100)
|
|
||||||
checkTime - seconds between loops (Default: 100)
|
|
||||||
"""
|
"""
|
||||||
|
try:
|
||||||
log.debug("Checking timed out clients")
|
log.debug("Checking timed out clients")
|
||||||
|
exceptions = []
|
||||||
timedOutTokens = [] # timed out users
|
timedOutTokens = [] # timed out users
|
||||||
timeoutLimit = int(time.time())-timeoutTime
|
timeoutLimit = int(time.time()) - 100
|
||||||
for key, value in self.tokens.items():
|
for key, value in self.tokens.items():
|
||||||
# Check timeout (fokabot is ignored)
|
# Check timeout (fokabot is ignored)
|
||||||
if value.pingTime < timeoutLimit and value.userID != 999 and value.irc == False and value.tournament == False:
|
if value.pingTime < timeoutLimit and value.userID != 999 and not value.irc and not value.tournament:
|
||||||
# That user has timed out, add to disconnected tokens
|
# That user has timed out, add to disconnected tokens
|
||||||
# We can't delete it while iterating or items() throws an error
|
# We can't delete it while iterating or items() throws an error
|
||||||
timedOutTokens.append(key)
|
timedOutTokens.append(key)
|
||||||
|
@ -163,39 +199,62 @@ class tokenList:
|
||||||
for i in timedOutTokens:
|
for i in timedOutTokens:
|
||||||
log.debug("{} timed out!!".format(self.tokens[i].username))
|
log.debug("{} timed out!!".format(self.tokens[i].username))
|
||||||
self.tokens[i].enqueue(serverPackets.notification("Your connection to the server timed out."))
|
self.tokens[i].enqueue(serverPackets.notification("Your connection to the server timed out."))
|
||||||
|
try:
|
||||||
logoutEvent.handle(self.tokens[i], None)
|
logoutEvent.handle(self.tokens[i], None)
|
||||||
|
except Exception as e:
|
||||||
|
exceptions.append(e)
|
||||||
|
log.error(
|
||||||
|
"Something wrong happened while disconnecting a timed out client. Reporting to Sentry "
|
||||||
|
"when the loop ends."
|
||||||
|
)
|
||||||
|
del timedOutTokens
|
||||||
|
|
||||||
|
# Re-raise exceptions if needed
|
||||||
|
if exceptions:
|
||||||
|
raise periodicLoopException(exceptions)
|
||||||
|
finally:
|
||||||
# Schedule a new check (endless loop)
|
# Schedule a new check (endless loop)
|
||||||
threading.Timer(checkTime, self.usersTimeoutCheckLoop, [timeoutTime, checkTime]).start()
|
threading.Timer(100, self.usersTimeoutCheckLoop).start()
|
||||||
|
|
||||||
|
@sentry.capture()
|
||||||
def spamProtectionResetLoop(self):
|
def spamProtectionResetLoop(self):
|
||||||
"""
|
"""
|
||||||
Reset spam rate every 10 seconds.
|
Start spam protection reset loop.
|
||||||
|
Called every 10 seconds.
|
||||||
CALL THIS FUNCTION ONLY ONCE!
|
CALL THIS FUNCTION ONLY ONCE!
|
||||||
|
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
|
try:
|
||||||
# Reset spamRate for every token
|
# Reset spamRate for every token
|
||||||
for _, value in self.tokens.items():
|
for _, value in self.tokens.items():
|
||||||
value.spamRate = 0
|
value.spamRate = 0
|
||||||
|
finally:
|
||||||
# Schedule a new check (endless loop)
|
# Schedule a new check (endless loop)
|
||||||
threading.Timer(10, self.spamProtectionResetLoop).start()
|
threading.Timer(10, self.spamProtectionResetLoop).start()
|
||||||
|
|
||||||
def deleteBanchoSessions(self):
|
def deleteBanchoSessions(self):
|
||||||
"""
|
"""
|
||||||
Truncate bancho_sessions table.
|
Remove all `peppy:sessions:*` redis keys.
|
||||||
Call at bancho startup to delete old cached sessions
|
Call at bancho startup to delete old cached sessions
|
||||||
|
|
||||||
|
:return:
|
||||||
"""
|
"""
|
||||||
glob.db.execute("TRUNCATE TABLE bancho_sessions")
|
try:
|
||||||
|
# TODO: Make function or some redis meme
|
||||||
|
glob.redis.eval("return redis.call('del', unpack(redis.call('keys', ARGV[1])))", 0, "peppy:sessions:*")
|
||||||
|
except redis.RedisError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
def tokenExists(self, username = "", userID = -1):
|
def tokenExists(self, username = "", userID = -1):
|
||||||
"""
|
"""
|
||||||
Check if a token exists (aka check if someone is connected)
|
Check if a token exists
|
||||||
|
|
||||||
username -- Optional.
|
|
||||||
userID -- Optional.
|
|
||||||
return -- True if it exists, otherwise False
|
|
||||||
|
|
||||||
Use username or userid, not both at the same time.
|
Use username or userid, not both at the same time.
|
||||||
|
|
||||||
|
:param username: Optional.
|
||||||
|
:param userID: Optional.
|
||||||
|
:return: True if it exists, otherwise False
|
||||||
"""
|
"""
|
||||||
if userID > -1:
|
if userID > -1:
|
||||||
return True if self.getTokenFromUserID(userID) is not None else False
|
return True if self.getTokenFromUserID(userID) is not None else False
|
||||||
|
|
136
pep.py
136
pep.py
|
@ -1,21 +1,24 @@
|
||||||
"""Hello, pep.py here, ex-owner of ripple and prime minister of Ripwot."""
|
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
import threading
|
import threading
|
||||||
from multiprocessing.pool import ThreadPool
|
from multiprocessing.pool import ThreadPool
|
||||||
|
|
||||||
import tornado.gen
|
import tornado.gen
|
||||||
import tornado.httpserver
|
import tornado.httpserver
|
||||||
import tornado.ioloop
|
import tornado.ioloop
|
||||||
import tornado.web
|
import tornado.web
|
||||||
from raven.contrib.tornado import AsyncSentryClient
|
from raven.contrib.tornado import AsyncSentryClient
|
||||||
|
import redis
|
||||||
|
|
||||||
from common import generalUtils
|
import json
|
||||||
|
import shutil
|
||||||
|
from distutils.version import LooseVersion
|
||||||
|
|
||||||
|
from common import generalUtils, agpl
|
||||||
from common.constants import bcolors
|
from common.constants import bcolors
|
||||||
from common.db import dbConnector
|
from common.db import dbConnector
|
||||||
from common.ddog import datadogClient
|
from common.ddog import datadogClient
|
||||||
from common.log import logUtils as log
|
from common.log import logUtils as log
|
||||||
from common.ripple import userUtils
|
from common.redis import pubSub
|
||||||
from common.web import schiavo
|
from common.web import schiavo
|
||||||
from handlers import apiFokabotMessageHandler
|
from handlers import apiFokabotMessageHandler
|
||||||
from handlers import apiIsOnlineHandler
|
from handlers import apiIsOnlineHandler
|
||||||
|
@ -33,6 +36,13 @@ from objects import banchoConfig
|
||||||
from objects import chatFilters
|
from objects import chatFilters
|
||||||
from objects import fokabot
|
from objects import fokabot
|
||||||
from objects import glob
|
from objects import glob
|
||||||
|
from pubSubHandlers import changeUsernameHandler
|
||||||
|
|
||||||
|
from pubSubHandlers import disconnectHandler
|
||||||
|
from pubSubHandlers import banHandler
|
||||||
|
from pubSubHandlers import notificationHandler
|
||||||
|
from pubSubHandlers import updateSilenceHandler
|
||||||
|
from pubSubHandlers import updateStatsHandler
|
||||||
|
|
||||||
|
|
||||||
def make_app():
|
def make_app():
|
||||||
|
@ -47,7 +57,15 @@ def make_app():
|
||||||
(r"/stress", heavyHandler.handler)
|
(r"/stress", heavyHandler.handler)
|
||||||
])
|
])
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
# AGPL license agreement
|
||||||
|
try:
|
||||||
|
agpl.check_license("ripple", "pep.py")
|
||||||
|
except agpl.LicenseError as e:
|
||||||
|
print(str(e))
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
# Server start
|
# Server start
|
||||||
consoleHelper.printServerStartHeader(True)
|
consoleHelper.printServerStartHeader(True)
|
||||||
|
@ -72,6 +90,34 @@ if __name__ == "__main__":
|
||||||
else:
|
else:
|
||||||
consoleHelper.printDone()
|
consoleHelper.printDone()
|
||||||
|
|
||||||
|
# Read additional config file
|
||||||
|
consoleHelper.printNoNl("> Loading additional config file... ")
|
||||||
|
try:
|
||||||
|
if not os.path.isfile(glob.conf.config["custom"]["config"]):
|
||||||
|
consoleHelper.printWarning()
|
||||||
|
consoleHelper.printColored("[!] Missing config file at {}; A default one has been generated at this location.".format(glob.conf.config["custom"]["config"]), bcolors.YELLOW)
|
||||||
|
shutil.copy("common/default_config.json", glob.conf.config["custom"]["config"])
|
||||||
|
|
||||||
|
with open(glob.conf.config["custom"]["config"], "r") as f:
|
||||||
|
glob.conf.extra = json.load(f)
|
||||||
|
|
||||||
|
consoleHelper.printDone()
|
||||||
|
except:
|
||||||
|
consoleHelper.printWarning()
|
||||||
|
consoleHelper.printColored("[!] Unable to load custom config at {}".format(glob.conf.config["custom"]["config"]), bcolors.RED)
|
||||||
|
consoleHelper.printColored("[!] Make sure you have the latest osufx common submodule!", bcolors.RED)
|
||||||
|
sys.exit()
|
||||||
|
|
||||||
|
# Check if running common module is usable
|
||||||
|
if glob.COMMON_VERSION == "Unknown":
|
||||||
|
consoleHelper.printWarning()
|
||||||
|
consoleHelper.printColored("[!] You do not seem to be using osufx's common submodule... nothing will work...", bcolors.RED)
|
||||||
|
consoleHelper.printColored("[!] You can download or fork the submodule from {}https://github.com/osufx/ripple-python-common".format(bcolors.UNDERLINE), bcolors.RED)
|
||||||
|
sys.exit()
|
||||||
|
elif LooseVersion(glob.COMMON_VERSION_REQ) > LooseVersion(glob.COMMON_VERSION):
|
||||||
|
consoleHelper.printColored("[!] Your common submodule version is below the required version number for this version of pep.py.", bcolors.RED)
|
||||||
|
consoleHelper.printColored("[!] You are highly adviced to update your common submodule as stability may vary with outdated modules.", bcolors.RED)
|
||||||
|
|
||||||
# Create data folder if needed
|
# Create data folder if needed
|
||||||
consoleHelper.printNoNl("> Checking folders... ")
|
consoleHelper.printNoNl("> Checking folders... ")
|
||||||
paths = [".data"]
|
paths = [".data"]
|
||||||
|
@ -92,6 +138,31 @@ if __name__ == "__main__":
|
||||||
consoleHelper.printColored("[!] Error while connection to database. Please check your config.ini and run the server again", bcolors.RED)
|
consoleHelper.printColored("[!] Error while connection to database. Please check your config.ini and run the server again", bcolors.RED)
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
# Connect to redis
|
||||||
|
try:
|
||||||
|
consoleHelper.printNoNl("> Connecting to redis... ")
|
||||||
|
glob.redis = redis.Redis(glob.conf.config["redis"]["host"], glob.conf.config["redis"]["port"], glob.conf.config["redis"]["database"], glob.conf.config["redis"]["password"])
|
||||||
|
glob.redis.ping()
|
||||||
|
consoleHelper.printNoNl(" ")
|
||||||
|
consoleHelper.printDone()
|
||||||
|
except:
|
||||||
|
# Exception while connecting to db
|
||||||
|
consoleHelper.printError()
|
||||||
|
consoleHelper.printColored("[!] Error while connection to redis. Please check your config.ini and run the server again", bcolors.RED)
|
||||||
|
raise
|
||||||
|
|
||||||
|
# Empty redis cache
|
||||||
|
try:
|
||||||
|
# TODO: Make function or some redis meme
|
||||||
|
glob.redis.set("ripple:online_users", 0)
|
||||||
|
glob.redis.eval("return redis.call('del', unpack(redis.call('keys', ARGV[1])))", 0, "peppy:*")
|
||||||
|
except redis.exceptions.ResponseError:
|
||||||
|
# Script returns error if there are no keys starting with peppy:*
|
||||||
|
pass
|
||||||
|
|
||||||
|
# Save peppy version in redis
|
||||||
|
glob.redis.set("peppy:version", glob.VERSION)
|
||||||
|
|
||||||
# Load bancho_settings
|
# Load bancho_settings
|
||||||
try:
|
try:
|
||||||
consoleHelper.printNoNl("> Loading bancho settings from DB... ")
|
consoleHelper.printNoNl("> Loading bancho settings from DB... ")
|
||||||
|
@ -112,7 +183,7 @@ if __name__ == "__main__":
|
||||||
consoleHelper.printNoNl("> Creating threads pool... ")
|
consoleHelper.printNoNl("> Creating threads pool... ")
|
||||||
glob.pool = ThreadPool(int(glob.conf.config["server"]["threads"]))
|
glob.pool = ThreadPool(int(glob.conf.config["server"]["threads"]))
|
||||||
consoleHelper.printDone()
|
consoleHelper.printDone()
|
||||||
except:
|
except ValueError:
|
||||||
consoleHelper.printError()
|
consoleHelper.printError()
|
||||||
consoleHelper.printColored("[!] Error while creating threads pool. Please check your config.ini and run the server again", bcolors.RED)
|
consoleHelper.printColored("[!] Error while creating threads pool. Please check your config.ini and run the server again", bcolors.RED)
|
||||||
|
|
||||||
|
@ -125,6 +196,11 @@ if __name__ == "__main__":
|
||||||
consoleHelper.printColored("[!] Error while loading chat filters. Make sure there is a filters.txt file present", bcolors.RED)
|
consoleHelper.printColored("[!] Error while loading chat filters. Make sure there is a filters.txt file present", bcolors.RED)
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
# Start fokabot
|
||||||
|
consoleHelper.printNoNl("> Connecting bot... ")
|
||||||
|
fokabot.connect()
|
||||||
|
consoleHelper.printDone()
|
||||||
|
|
||||||
# Initialize chat channels
|
# Initialize chat channels
|
||||||
print("> Initializing chat channels... ")
|
print("> Initializing chat channels... ")
|
||||||
glob.channels.loadChannels()
|
glob.channels.loadChannels()
|
||||||
|
@ -136,11 +212,6 @@ if __name__ == "__main__":
|
||||||
glob.streams.add("lobby")
|
glob.streams.add("lobby")
|
||||||
consoleHelper.printDone()
|
consoleHelper.printDone()
|
||||||
|
|
||||||
# Start fokabot
|
|
||||||
consoleHelper.printNoNl("> Connecting FokaBot... ")
|
|
||||||
fokabot.connect()
|
|
||||||
consoleHelper.printDone()
|
|
||||||
|
|
||||||
# Initialize user timeout check loop
|
# Initialize user timeout check loop
|
||||||
consoleHelper.printNoNl("> Initializing user timeout check loop... ")
|
consoleHelper.printNoNl("> Initializing user timeout check loop... ")
|
||||||
glob.tokens.usersTimeoutCheckLoop()
|
glob.tokens.usersTimeoutCheckLoop()
|
||||||
|
@ -151,9 +222,9 @@ if __name__ == "__main__":
|
||||||
glob.tokens.spamProtectionResetLoop()
|
glob.tokens.spamProtectionResetLoop()
|
||||||
consoleHelper.printDone()
|
consoleHelper.printDone()
|
||||||
|
|
||||||
# Cache user ids
|
# Initialize multiplayer cleanup loop
|
||||||
consoleHelper.printNoNl("> Caching user IDs... ")
|
consoleHelper.printNoNl("> Initializing multiplayer cleanup loop... ")
|
||||||
userUtils.cacheUserIDs()
|
glob.matches.cleanupLoop()
|
||||||
consoleHelper.printDone()
|
consoleHelper.printDone()
|
||||||
|
|
||||||
# Localize warning
|
# Localize warning
|
||||||
|
@ -163,7 +234,7 @@ if __name__ == "__main__":
|
||||||
|
|
||||||
# Discord
|
# Discord
|
||||||
if generalUtils.stringToBool(glob.conf.config["discord"]["enable"]):
|
if generalUtils.stringToBool(glob.conf.config["discord"]["enable"]):
|
||||||
glob.schiavo = schiavo.schiavo(glob.conf.config["discord"]["boturl"])
|
glob.schiavo = schiavo.schiavo(glob.conf.config["discord"]["boturl"], "**pep.py**")
|
||||||
else:
|
else:
|
||||||
consoleHelper.printColored("[!] Warning! Discord logging is disabled!", bcolors.YELLOW)
|
consoleHelper.printColored("[!] Warning! Discord logging is disabled!", bcolors.YELLOW)
|
||||||
|
|
||||||
|
@ -187,7 +258,7 @@ if __name__ == "__main__":
|
||||||
try:
|
try:
|
||||||
glob.sentry = generalUtils.stringToBool(glob.conf.config["sentry"]["enable"])
|
glob.sentry = generalUtils.stringToBool(glob.conf.config["sentry"]["enable"])
|
||||||
if glob.sentry:
|
if glob.sentry:
|
||||||
glob.application.sentry_client = AsyncSentryClient(glob.conf.config["sentry"]["banchodns"], release=glob.VERSION)
|
glob.application.sentry_client = AsyncSentryClient(glob.conf.config["sentry"]["banchodsn"], release=glob.VERSION)
|
||||||
else:
|
else:
|
||||||
consoleHelper.printColored("[!] Warning! Sentry logging is disabled!", bcolors.YELLOW)
|
consoleHelper.printColored("[!] Warning! Sentry logging is disabled!", bcolors.YELLOW)
|
||||||
except:
|
except:
|
||||||
|
@ -202,39 +273,60 @@ if __name__ == "__main__":
|
||||||
[
|
[
|
||||||
datadogClient.periodicCheck("online_users", lambda: len(glob.tokens.tokens)),
|
datadogClient.periodicCheck("online_users", lambda: len(glob.tokens.tokens)),
|
||||||
datadogClient.periodicCheck("multiplayer_matches", lambda: len(glob.matches.matches)),
|
datadogClient.periodicCheck("multiplayer_matches", lambda: len(glob.matches.matches)),
|
||||||
|
|
||||||
|
#datadogClient.periodicCheck("ram_clients", lambda: generalUtils.getTotalSize(glob.tokens)),
|
||||||
|
#datadogClient.periodicCheck("ram_matches", lambda: generalUtils.getTotalSize(glob.matches)),
|
||||||
|
#datadogClient.periodicCheck("ram_channels", lambda: generalUtils.getTotalSize(glob.channels)),
|
||||||
|
#datadogClient.periodicCheck("ram_file_buffers", lambda: generalUtils.getTotalSize(glob.fileBuffers)),
|
||||||
|
#datadogClient.periodicCheck("ram_file_locks", lambda: generalUtils.getTotalSize(glob.fLocks)),
|
||||||
|
#datadogClient.periodicCheck("ram_datadog", lambda: generalUtils.getTotalSize(glob.datadogClient)),
|
||||||
|
#datadogClient.periodicCheck("ram_verified_cache", lambda: generalUtils.getTotalSize(glob.verifiedCache)),
|
||||||
|
#datadogClient.periodicCheck("ram_irc", lambda: generalUtils.getTotalSize(glob.ircServer)),
|
||||||
|
#datadogClient.periodicCheck("ram_tornado", lambda: generalUtils.getTotalSize(glob.application)),
|
||||||
|
#datadogClient.periodicCheck("ram_db", lambda: generalUtils.getTotalSize(glob.db)),
|
||||||
])
|
])
|
||||||
else:
|
else:
|
||||||
consoleHelper.printColored("[!] Warning! Datadog stats tracking is disabled!", bcolors.YELLOW)
|
consoleHelper.printColored("[!] Warning! Datadog stats tracking is disabled!", bcolors.YELLOW)
|
||||||
except:
|
except:
|
||||||
consoleHelper.printColored("[!] Error while starting Datadog client! Please check your config.ini and run the server again", bcolors.RED)
|
consoleHelper.printColored("[!] Error while starting Datadog client! Please check your config.ini and run the server again", bcolors.RED)
|
||||||
|
|
||||||
# Cloudflare memes
|
|
||||||
glob.cloudflare = generalUtils.stringToBool(glob.conf.config["server"]["cloudflare"])
|
|
||||||
|
|
||||||
# IRC start message and console output
|
# IRC start message and console output
|
||||||
glob.irc = generalUtils.stringToBool(glob.conf.config["irc"]["enable"])
|
glob.irc = generalUtils.stringToBool(glob.conf.config["irc"]["enable"])
|
||||||
if glob.irc:
|
if glob.irc:
|
||||||
# IRC port
|
# IRC port
|
||||||
|
ircPort = 0
|
||||||
try:
|
try:
|
||||||
ircPort = int(glob.conf.config["irc"]["port"])
|
ircPort = int(glob.conf.config["irc"]["port"])
|
||||||
except:
|
except ValueError:
|
||||||
consoleHelper.printColored("[!] Invalid IRC port! Please check your config.ini and run the server again", bcolors.RED)
|
consoleHelper.printColored("[!] Invalid IRC port! Please check your config.ini and run the server again", bcolors.RED)
|
||||||
log.logMessage("**pep.py** IRC server started!", discord="bunker", of="info.txt", stdout=False)
|
log.logMessage("IRC server started!", discord="bunker", of="info.txt", stdout=False)
|
||||||
consoleHelper.printColored("> IRC server listening on 127.0.0.1:{}...".format(ircPort), bcolors.GREEN)
|
consoleHelper.printColored("> IRC server listening on 127.0.0.1:{}...".format(ircPort), bcolors.GREEN)
|
||||||
threading.Thread(target=lambda: ircserver.main(port=ircPort)).start()
|
threading.Thread(target=lambda: ircserver.main(port=ircPort)).start()
|
||||||
else:
|
else:
|
||||||
consoleHelper.printColored("[!] Warning! IRC server is disabled!", bcolors.YELLOW)
|
consoleHelper.printColored("[!] Warning! IRC server is disabled!", bcolors.YELLOW)
|
||||||
|
|
||||||
# Server port
|
# Server port
|
||||||
|
serverPort = 0
|
||||||
try:
|
try:
|
||||||
serverPort = int(glob.conf.config["server"]["port"])
|
serverPort = int(glob.conf.config["server"]["port"])
|
||||||
except:
|
except ValueError:
|
||||||
consoleHelper.printColored("[!] Invalid server port! Please check your config.ini and run the server again", bcolors.RED)
|
consoleHelper.printColored("[!] Invalid server port! Please check your config.ini and run the server again", bcolors.RED)
|
||||||
|
|
||||||
# Server start message and console output
|
# Server start message and console output
|
||||||
log.logMessage("**pep.py** Server started!", discord="bunker", of="info.txt", stdout=False)
|
log.logMessage("Server started!", discord="bunker", of="info.txt", stdout=False)
|
||||||
consoleHelper.printColored("> Tornado listening for HTTP(s) clients on 127.0.0.1:{}...".format(serverPort), bcolors.GREEN)
|
consoleHelper.printColored("> Tornado listening for HTTP(s) clients on 127.0.0.1:{}...".format(serverPort), bcolors.GREEN)
|
||||||
|
|
||||||
|
# Connect to pubsub channels
|
||||||
|
pubSub.listener(glob.redis, {
|
||||||
|
"peppy:disconnect": disconnectHandler.handler(),
|
||||||
|
"peppy:change_username": changeUsernameHandler.handler(),
|
||||||
|
"peppy:reload_settings": lambda x: x == b"reload" and glob.banchoConf.reload(),
|
||||||
|
"peppy:update_cached_stats": updateStatsHandler.handler(),
|
||||||
|
"peppy:silence": updateSilenceHandler.handler(),
|
||||||
|
"peppy:ban": banHandler.handler(),
|
||||||
|
"peppy:notification": notificationHandler.handler(),
|
||||||
|
}).start()
|
||||||
|
|
||||||
# Start tornado
|
# Start tornado
|
||||||
glob.application.listen(serverPort)
|
glob.application.listen(serverPort)
|
||||||
tornado.ioloop.IOLoop.instance().start()
|
tornado.ioloop.IOLoop.instance().start()
|
||||||
|
|
0
pubSubHandlers/__init__.py
Normal file
0
pubSubHandlers/__init__.py
Normal file
18
pubSubHandlers/banHandler.py
Normal file
18
pubSubHandlers/banHandler.py
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
from common.redis import generalPubSubHandler
|
||||||
|
from common.ripple import userUtils
|
||||||
|
from objects import glob
|
||||||
|
|
||||||
|
class handler(generalPubSubHandler.generalPubSubHandler):
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__()
|
||||||
|
self.type = "int"
|
||||||
|
|
||||||
|
def handle(self, userID):
|
||||||
|
userID = super().parseData(userID)
|
||||||
|
if userID is None:
|
||||||
|
return
|
||||||
|
targetToken = glob.tokens.getTokenFromUserID(userID)
|
||||||
|
if targetToken is not None:
|
||||||
|
targetToken.privileges = userUtils.getPrivileges(userID)
|
||||||
|
targetToken.checkBanned()
|
||||||
|
targetToken.checkRestricted()
|
50
pubSubHandlers/changeUsernameHandler.py
Normal file
50
pubSubHandlers/changeUsernameHandler.py
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
from common.redis import generalPubSubHandler
|
||||||
|
from common.ripple import userUtils
|
||||||
|
from common.log import logUtils as log
|
||||||
|
from common.constants import actions
|
||||||
|
from objects import glob
|
||||||
|
|
||||||
|
def handleUsernameChange(userID, newUsername, targetToken=None):
|
||||||
|
try:
|
||||||
|
userUtils.appendNotes(userID, "Username change: '{}' -> '{}'".format(userUtils.getUsername(userID), newUsername))
|
||||||
|
userUtils.changeUsername(userID, newUsername=newUsername)
|
||||||
|
if targetToken is not None:
|
||||||
|
targetToken.kick("Your username has been changed to {}. Please log in again.".format(newUsername), "username_change")
|
||||||
|
except userUtils.usernameAlreadyInUseError:
|
||||||
|
log.rap(999, "Username change: {} is already in use!", through="Bancho")
|
||||||
|
if targetToken is not None:
|
||||||
|
targetToken.kick("There was a critical error while trying to change your username. Please contact a developer.", "username_change_fail")
|
||||||
|
except userUtils.invalidUsernameError:
|
||||||
|
log.rap(999, "Username change: {} is not a valid username!", through="Bancho")
|
||||||
|
if targetToken is not None:
|
||||||
|
targetToken.kick("There was a critical error while trying to change your username. Please contact a developer.", "username_change_fail")
|
||||||
|
|
||||||
|
class handler(generalPubSubHandler.generalPubSubHandler):
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__()
|
||||||
|
self.structure = {
|
||||||
|
"userID": 0,
|
||||||
|
"newUsername": ""
|
||||||
|
}
|
||||||
|
|
||||||
|
def handle(self, data):
|
||||||
|
data = super().parseData(data)
|
||||||
|
if data is None:
|
||||||
|
return
|
||||||
|
# Get the user's token
|
||||||
|
targetToken = glob.tokens.getTokenFromUserID(data["userID"])
|
||||||
|
if targetToken is None:
|
||||||
|
# If the user is offline change username immediately
|
||||||
|
handleUsernameChange(data["userID"], data["newUsername"])
|
||||||
|
else:
|
||||||
|
if targetToken.irc or (targetToken.actionID != actions.PLAYING and targetToken.actionID != actions.MULTIPLAYING):
|
||||||
|
# If the user is online and he's connected through IRC or he's not playing,
|
||||||
|
# change username and kick the user immediately
|
||||||
|
handleUsernameChange(data["userID"], data["newUsername"], targetToken)
|
||||||
|
else:
|
||||||
|
# If the user is playing, delay the username change until he submits the score
|
||||||
|
# On submit modular, lets will send the username change request again
|
||||||
|
# through redis once the score has been submitted
|
||||||
|
# The check is performed on bancho logout too, so if the user disconnects
|
||||||
|
# without submitting a score, the username gets changed on bancho logout
|
||||||
|
glob.redis.set("ripple:change_username_pending:{}".format(data["userID"]), data["newUsername"])
|
18
pubSubHandlers/disconnectHandler.py
Normal file
18
pubSubHandlers/disconnectHandler.py
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
from common.redis import generalPubSubHandler
|
||||||
|
from objects import glob
|
||||||
|
|
||||||
|
class handler(generalPubSubHandler.generalPubSubHandler):
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__()
|
||||||
|
self.structure = {
|
||||||
|
"userID": 0,
|
||||||
|
"reason": ""
|
||||||
|
}
|
||||||
|
|
||||||
|
def handle(self, data):
|
||||||
|
data = super().parseData(data)
|
||||||
|
if data is None:
|
||||||
|
return
|
||||||
|
targetToken = glob.tokens.getTokenFromUserID(data["userID"])
|
||||||
|
if targetToken is not None:
|
||||||
|
targetToken.kick(data["reason"], "pubsub_kick")
|
19
pubSubHandlers/notificationHandler.py
Normal file
19
pubSubHandlers/notificationHandler.py
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
from common.redis import generalPubSubHandler
|
||||||
|
from objects import glob
|
||||||
|
from constants import serverPackets
|
||||||
|
|
||||||
|
class handler(generalPubSubHandler.generalPubSubHandler):
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__()
|
||||||
|
self.structure = {
|
||||||
|
"userID": 0,
|
||||||
|
"message": ""
|
||||||
|
}
|
||||||
|
|
||||||
|
def handle(self, data):
|
||||||
|
data = super().parseData(data)
|
||||||
|
if data is None:
|
||||||
|
return
|
||||||
|
targetToken = glob.tokens.getTokenFromUserID(data["userID"])
|
||||||
|
if targetToken is not None:
|
||||||
|
targetToken.enqueue(serverPackets.notification(data["message"]))
|
15
pubSubHandlers/updateSilenceHandler.py
Normal file
15
pubSubHandlers/updateSilenceHandler.py
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
from common.redis import generalPubSubHandler
|
||||||
|
from objects import glob
|
||||||
|
|
||||||
|
class handler(generalPubSubHandler.generalPubSubHandler):
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__()
|
||||||
|
self.type = "int"
|
||||||
|
|
||||||
|
def handle(self, userID):
|
||||||
|
userID = super().parseData(userID)
|
||||||
|
if userID is None:
|
||||||
|
return
|
||||||
|
targetToken = glob.tokens.getTokenFromUserID(userID)
|
||||||
|
if targetToken is not None:
|
||||||
|
targetToken.silence()
|
15
pubSubHandlers/updateStatsHandler.py
Normal file
15
pubSubHandlers/updateStatsHandler.py
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
from common.redis import generalPubSubHandler
|
||||||
|
from objects import glob
|
||||||
|
|
||||||
|
class handler(generalPubSubHandler.generalPubSubHandler):
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__()
|
||||||
|
self.type = "int"
|
||||||
|
|
||||||
|
def handle(self, userID):
|
||||||
|
userID = super().parseData(userID)
|
||||||
|
if userID is None:
|
||||||
|
return
|
||||||
|
targetToken = glob.tokens.getTokenFromUserID(userID)
|
||||||
|
if targetToken is not None:
|
||||||
|
targetToken.updateCachedStats()
|
|
@ -1,6 +1,10 @@
|
||||||
requests
|
requests==2.18.1
|
||||||
tornado
|
tornado==4.4.2
|
||||||
mysqlclient
|
mysqlclient==1.3.9
|
||||||
psutil
|
psutil==5.2.2
|
||||||
raven
|
raven==5.32.0
|
||||||
bcrypt
|
bcrypt==3.1.1
|
||||||
|
dill==0.2.7.1
|
||||||
|
redis==2.10.5
|
||||||
|
cython==0.27.3
|
||||||
|
datadog==0.14.0
|
17
setup.py
Normal file
17
setup.py
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
"""Cython build file"""
|
||||||
|
from distutils.core import setup
|
||||||
|
from distutils.extension import Extension
|
||||||
|
from Cython.Build import cythonize
|
||||||
|
import os
|
||||||
|
|
||||||
|
cythonExt = []
|
||||||
|
for root, dirs, files in os.walk(os.getcwd()):
|
||||||
|
for file in files:
|
||||||
|
if file.endswith(".pyx") and ".pyenv" not in root: # im sorry
|
||||||
|
filePath = os.path.relpath(os.path.join(root, file))
|
||||||
|
cythonExt.append(Extension(filePath.replace("/", ".")[:-4], [filePath]))
|
||||||
|
|
||||||
|
setup(
|
||||||
|
name = "pep.pyx modules",
|
||||||
|
ext_modules = cythonize(cythonExt, nthreads = 4),
|
||||||
|
)
|
Loading…
Reference in New Issue
Block a user