mirror of
https://github.com/drmext/MonkeyBusiness.git
synced 2026-03-21 09:54:41 -05:00
229 lines
5.9 KiB
Python
229 lines
5.9 KiB
Python
import config
|
|
|
|
import time
|
|
|
|
from lxml.builder import ElementMaker
|
|
|
|
from kbinxml import KBinXML
|
|
|
|
from utils.arc4 import EamuseARC4
|
|
from utils.lz77 import lz77_decode, lz77_encode
|
|
|
|
|
|
def _add_val_as_str(elm, val):
|
|
new_val = str(val)
|
|
|
|
if elm is not None:
|
|
elm.text = new_val
|
|
|
|
else:
|
|
return new_val
|
|
|
|
|
|
def _add_bool_as_str(elm, val):
|
|
return _add_val_as_str(elm, 1 if val else 0)
|
|
|
|
|
|
def _add_list_as_str(elm, vals):
|
|
new_val = " ".join([str(val) for val in vals])
|
|
|
|
if elm is not None:
|
|
elm.text = new_val
|
|
elm.attrib["__count"] = str(len(vals))
|
|
|
|
else:
|
|
return new_val
|
|
|
|
|
|
def _prng():
|
|
state = 0x41C64E6D
|
|
while True:
|
|
x = (state * 0x838C9CDA) + 0x6072
|
|
# state = (state * 0x41C64E6D + 0x3039)
|
|
# state = (state * 0x41C64E6D + 0x3039)
|
|
state = (state * 0xC2A29A69 + 0xD3DC167E) & 0xFFFFFFFF
|
|
yield (x & 0x7FFF0000) | state >> 0xF & 0xFFFF
|
|
prng_init = _prng()
|
|
|
|
|
|
E = ElementMaker(
|
|
typemap={
|
|
int: _add_val_as_str,
|
|
bool: _add_bool_as_str,
|
|
list: _add_list_as_str,
|
|
float: _add_val_as_str,
|
|
}
|
|
)
|
|
|
|
|
|
async def core_get_game_version_from_software_version(software_version):
|
|
_, model, dest, spec, rev, ext = software_version
|
|
ext = int(ext)
|
|
|
|
if model == "LDJ":
|
|
if ext >= 2025091700:
|
|
return 33
|
|
elif ext >= 2024100900:
|
|
return 32
|
|
elif ext >= 2023101800:
|
|
return 31
|
|
elif ext >= 2022101700:
|
|
return 30
|
|
elif ext >= 2021101300:
|
|
return 29
|
|
# TODO: Consolidate IIDX modules to easily support versions 21-28 (probably never)
|
|
elif ext >= 2020102800:
|
|
return 28
|
|
elif ext >= 2019101600:
|
|
return 27
|
|
elif ext >= 2018110700:
|
|
return 26
|
|
elif ext >= 2017122100:
|
|
return 25
|
|
elif ext >= 2016102400:
|
|
return 24
|
|
elif ext >= 2015111100:
|
|
return 23
|
|
elif ext >= 2014091700:
|
|
return 22
|
|
elif ext >= 2013100200:
|
|
return 21
|
|
elif ext >= 2012010100:
|
|
return 20
|
|
elif model == "KDZ":
|
|
return 19
|
|
elif model == "JDZ":
|
|
return 18
|
|
|
|
elif model == "M32":
|
|
if ext >= 2024031300:
|
|
return 10
|
|
elif ext >= 2022121400:
|
|
return 9
|
|
elif ext >= 2021042100:
|
|
return 8
|
|
elif ext >= 2019100200:
|
|
return 7
|
|
elif ext >= 2018072700:
|
|
return 6
|
|
# TODO: Support versions 1-5 (never)
|
|
elif ext >= 2017090600:
|
|
return 5
|
|
elif ext >= 2017011800:
|
|
return 4
|
|
elif ext >= 2015042100:
|
|
return 3
|
|
elif ext >= 2014021400:
|
|
return 2
|
|
elif ext >= 2013012400:
|
|
return 1
|
|
|
|
elif model == "MDX":
|
|
if ext >= 2024061200 and ext not in (2024042069, 2025042069): # GF
|
|
return 20
|
|
if ext >= 2019022600: # ???
|
|
return 19
|
|
|
|
elif model == "KFC":
|
|
if ext >= 2020090402: # ???
|
|
return 6
|
|
|
|
elif model == "REC":
|
|
return 1
|
|
|
|
# TODO: ???
|
|
# elif model == "PAN":
|
|
# return 0
|
|
|
|
else:
|
|
return 0
|
|
|
|
|
|
async def core_process_request(request):
|
|
cl = request.headers.get("Content-Length")
|
|
data = await request.body()
|
|
|
|
if not cl or not data:
|
|
return {}
|
|
|
|
request.compress = request.headers.get("X-Compress", "none") # intentionally lowercase 'none' (NOT None)
|
|
|
|
if "X-Eamuse-Info" in request.headers:
|
|
xeamuseinfo = request.headers.get("X-Eamuse-Info")
|
|
version, unix_time, prng = xeamuseinfo.split("-")
|
|
xml_dec = EamuseARC4(bytes.fromhex(unix_time), bytes.fromhex(prng)).decrypt(data[: int(cl)])
|
|
request.is_encrypted = True
|
|
else:
|
|
xml_dec = data[: int(cl)]
|
|
request.is_encrypted = False
|
|
|
|
if request.compress == "lz77":
|
|
xml_dec = lz77_decode(xml_dec)
|
|
|
|
xml = KBinXML(xml_dec, convert_illegal_things=True)
|
|
root = xml.xml_doc
|
|
xml_text = xml.to_text()
|
|
request.is_binxml = KBinXML.is_binary_xml(xml_dec)
|
|
|
|
if config.verbose_log:
|
|
print()
|
|
print("\033[94mREQUEST\033[0m:")
|
|
print(xml_text)
|
|
|
|
model_parts = (root.attrib["model"], *root.attrib["model"].split(":"))
|
|
module = root[0].tag
|
|
method = root[0].attrib["method"] if "method" in root[0].attrib else None
|
|
command = root[0].attrib["command"] if "command" in root[0].attrib else None
|
|
game_version = await core_get_game_version_from_software_version(model_parts)
|
|
|
|
return {
|
|
"root": root,
|
|
"text": xml_text,
|
|
"module": module,
|
|
"method": method,
|
|
"command": command,
|
|
"model": model_parts[1],
|
|
"dest": model_parts[2],
|
|
"spec": model_parts[3],
|
|
"rev": model_parts[4],
|
|
"ext": model_parts[5],
|
|
"game_version": game_version,
|
|
}
|
|
|
|
|
|
async def core_prepare_response(request, xml):
|
|
binxml = KBinXML(xml)
|
|
|
|
if request.is_binxml:
|
|
xml_binary = binxml.to_binary()
|
|
else:
|
|
xml_binary = binxml.to_text().encode("utf-8") # TODO: Proper encoding
|
|
|
|
if config.verbose_log:
|
|
print("\033[91mRESPONSE\033[0m:")
|
|
print(binxml.to_text())
|
|
|
|
response_headers = {"User-Agent": "EAMUSE.Httpac/1.0"}
|
|
|
|
if config.response_compression:
|
|
response_headers["X-Compress"] = request.compress
|
|
if request.compress == "lz77":
|
|
response = lz77_encode(xml_binary) # very slow
|
|
else:
|
|
response = xml_binary
|
|
else:
|
|
response_headers["X-Compress"] = "none" # intentionally lowercase 'none' (NOT None)
|
|
response = xml_binary
|
|
|
|
|
|
if request.is_encrypted:
|
|
version = 1
|
|
unix_time = int(time.time())
|
|
prng = next(prng_init) & 0xFFFF
|
|
response_headers["X-Eamuse-Info"] = f"{version}-{unix_time:04x}-{prng:02x}"
|
|
response = EamuseARC4(unix_time.to_bytes(4), prng.to_bytes(2)).encrypt(response)
|
|
else:
|
|
response = bytes(response)
|
|
|
|
return response, response_headers
|