artemis/titles/sao/base.py

2626 lines
133 KiB
Python

import logging
from csv import *
from random import choice, randint
from typing import Dict, List
from os import path
import re
from sqlalchemy.engine import Row
from core import CoreConfig
from .config import SaoConfig
from .database import SaoData
from .const import GameconnectCmd, RewardType, ExBonusCondition, QuestType
from titles.sao.handlers.base import *
import csv
class SaoBase:
DATA_LIST = {}
def __init__(self, core_cfg: CoreConfig, game_cfg: SaoConfig) -> None:
self.core_cfg = core_cfg
self.game_cfg = game_cfg
self.data = SaoData(core_cfg)
self.logger = logging.getLogger("sao")
def load_data_csv(self, file: str, version: int = 1, base_ver: int = 1) -> List[Dict]:
if base_ver > version:
self.logger.warning(f"load_data_csv: Cannot use base version higher then requested version ({base_ver} > {version})")
return []
for x in range(version, base_ver - 1, -1):
ret = self.DATA_LIST.get(x, {}).get(file, [])
if ret:
break
if not ret and base_ver != 1:
ret = self.DATA_LIST.get(1, {}).get(file, [])
if ret:
return ret
found = False
for x in range(version, base_ver - 1, -1):
fname = f"./titles/sao/data/{x}/{file}.csv"
if path.exists(fname):
found_ver = x
found = True
break
if not found and base_ver != 1: # v1 will always be fallback if it isn't already
fname = f"./titles/sao/data/1/{file}.csv"
if path.exists(fname):
found_ver = 1
found = True
if not found:
self.logger.warning(f"load_data_csv: Failed to find v{version} csv file {fname}")
return []
ret = []
with open(fname, "r", encoding="utf8") as f:
data = csv.DictReader(f, delimiter=',')
for x in data:
newdict = {}
for k, v in x.items():
newkey = k
if k.startswith("// "):
newkey = k.replace("// ", "")
if v.isdigit():
newdict[newkey] = int(v)
elif v.lower() == "true":
newdict[newkey] = True
elif v.lower() == "false":
newdict[newkey] = False
elif re.match(r"^\d\d\d\d\/\d\d\/\d\d \d{1,2}:\d\d:\d\d$", v):
newdict[newkey] = datetime.strptime(v, "%Y/%m/%d %H:%M:%S")
elif re.match(r"^\d\d\d\d\/\d\d\/\d\d$", v):
newdict[newkey] = datetime.strptime(v, "%Y/%m/%d")
else:
newdict[newkey] = v
ret.append(newdict)
# Cache the CSV data in memory
if found_ver not in self.DATA_LIST:
self.DATA_LIST[found_ver] = {}
self.DATA_LIST[found_ver][file] = ret
return ret
async def add_reward(self, reward: Dict, user_id: int):
reward_type = int(reward.get("CommonRewardType", "0"))
if reward_type == RewardType.HeroLog:
reward_hero_data = await self.data.static.get_hero_by_id(reward['CommonRewardId'])
now_have_skills = await self.hero_default_skills(reward_hero_data['SkillTableSubId'])
new_hero_id = await self.data.item.put_hero_log(
user_id,
reward['CommonRewardId'],
1,
0,
None,
None,
now_have_skills[0],
now_have_skills[1],
now_have_skills[2],
now_have_skills[3],
now_have_skills[4],
)
self.logger.info(f"Rewarded user {user_id} with hero {reward['CommonRewardId']} (ID {new_hero_id})")
# TODO: add properties
elif reward_type == RewardType.Equipment:
new_equip_id = await self.data.item.put_equipment(user_id, reward['CommonRewardId'])
self.logger.info(f"Rewarded user {user_id} with equipment {reward['CommonRewardId']} (ID {new_equip_id})")
elif reward_type == RewardType.Item:
new_item_id = await self.data.item.put_item(user_id, reward['CommonRewardId'])
self.logger.info(f"Rewarded user {user_id} with item {reward['CommonRewardId']} (ID {new_item_id})")
elif reward_type == RewardType.Col:
col_num = int(reward['CommonRewardNum'])
self.logger.info(f"Rewarded user {user_id} with {col_num} Col")
await self.data.profile.add_col(user_id, col_num)
elif reward_type == RewardType.VP:
vp_num = int(reward['CommonRewardNum'])
self.logger.info(f"Rewarded user {user_id} with {vp_num} VP")
await self.data.profile.add_vp(user_id, vp_num)
elif reward_type == RewardType.YuiMadal:
medal_num = int(reward['CommonRewardNum'])
self.logger.info(f"Rewarded user {user_id} with {medal_num} Yui Medals")
await self.data.profile.add_yui_medals(user_id, medal_num)
else:
self.logger.warn(f"User {user_id} Unhandled reward type {reward_type} -> {reward}")
async def hero_default_skills(self, skill_table_id: int) -> List[int]:
skills = await self.data.static.get_skill_table_by_subid(skill_table_id)
if not skills:
self.logger.error(f"Failed to find skill table {skill_table_id}! Please run the reader")
return [None, None, None, None, None]
default_skills = []
now_have_skills = [None, None, None, None, None]
for skill in skills:
if skill['LevelObtained'] == 1 and skill['AwakeningId'] == 0:
default_skills.append(skill['SkillId'])
for skill in default_skills:
skill_info = await self.data.static.get_skill_by_id(skill)
skill_slot = skill_info['Level'] - 1
if now_have_skills[skill_slot] is not None:
now_have_skills[skill]
return now_have_skills
async def handle_noop(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
return SaoNoopResponse(header.cmd + 1).make()
async def handle_c000(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#ticket/ticket
req = SaoTicketRequest(header, request)
return SaoTicketResponse().make()
async def handle_c100(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#common/get_app_versions
resp = SaoGetAppVersionsResponse()
resp.data_list.append(AppVersionData.from_args(self.game_cfg.server.game_version, datetime.fromtimestamp(0)))
return resp.make()
async def handle_c102(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#common/master_data_version_check
req = SaoMasterDataVersionCheckRequest(header, request)
self.logger.info(f"Cab at {src_ip} checked in with master data v{req.current_data_version}")
return SaoMasterDataVersionCheckResponse(self.game_cfg.server.data_version, req.current_data_version).make()
async def handle_c104(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#common/login
req = SaoLoginRequest(header, request)
user_id = await self.data.card.get_user_id_from_card( req.access_code )
if not user_id:
card = await self.data.card.get_card_by_idm(req.chip_id[:16])
if card:
user_id = card['user']
card_id = card['id']
await self.data.card.set_access_code_by_access_code(card['access_code'], req.access_code)
else:
user_id = await self.data.user.create_user() #works
card_id = await self.data.card.create_card(user_id, req.access_code)
if card_id is None:
user_id = -1
self.logger.error("Failed to register card!")
self.logger.info(f"Registered card {req.access_code} to user {user_id} from {req.serial_no}")
if req.access_code.startswith("5"):
await self.data.card.set_idm_by_access_code(req.access_code, req.chip_id[:16])
elif (req.access_code.startswith("010") or req.access_code.startswith("3")) and int(req.chip_id[:8], 16) != 0x04030201:
await self.data.card.set_chip_id_by_access_code(req.access_code, int(req.chip_id[:8], 16))
profile_data = await self.data.profile.get_profile(user_id)
if not profile_data:
profile_id = await self.data.profile.create_profile(user_id)
if profile_id:
equip1 = await self.data.item.put_equipment(user_id, 101000000)
equip2 = await self.data.item.put_equipment(user_id, 102000000)
equip3 = await self.data.item.put_equipment(user_id, 109000000)
if not equip1 or not equip2 or not equip3:
self.logger.error(f"Failed to create profile for user {user_id} from {req.serial_no} (could not add equipment)")
return SaoNoopResponse(GameconnectCmd.LOGIN_RESPONSE).make()
hero1 = await self.data.item.put_hero_log(user_id, 101000010, 1, 0, equip1, None, 1002, 1003, 1014, None, None)
hero2 = await self.data.item.put_hero_log(user_id, 102000010, 1, 0, equip2, None, 3001, 3002, 3004, None, None)
hero3 = await self.data.item.put_hero_log(user_id, 105000010, 1, 0, equip3, None, 10005, 10002, 10004, None, None)
if not hero1 or not hero2 or not hero3:
self.logger.error(f"Failed to create profile for user {user_id} from {req.serial_no} (could not add heros)")
return SaoNoopResponse(GameconnectCmd.LOGIN_RESPONSE).make()
await self.data.item.put_hero_party(user_id, 0, hero1, hero2, hero3)
self.logger.info(f"Create profile {profile_id} for user {user_id} from {req.serial_no}")
else:
self.logger.error(f"Failed to create profile for user {user_id} from {req.serial_no}")
return SaoNoopResponse(GameconnectCmd.LOGIN_RESPONSE).make()
resp = SaoLoginResponse(user_id, True, False)
else:
is_login_today = False
if profile_data['last_login_date']:
last_login_time = int(profile_data["last_login_date"].timestamp())
midnight_today_ts = int(
datetime.now()
.replace(hour=0, minute=0, second=0, microsecond=0)
.timestamp()
)
if last_login_time > midnight_today_ts:
is_login_today = True
if not is_login_today:
await self.data.profile.add_vp(user_id, 100)
resp = SaoLoginResponse(user_id, profile_data['login_ct'] < 1, is_login_today)
await self.data.profile.user_login(user_id)
return resp.make()
async def handle_c106(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/logout
req = SaoLogoutRequest(header, request)
self.logger.info(f"User {req.user_id} Logout from {'game' if req.cabinet_type == 0 else 'terminal'} @ {src_ip} with {req.remaining_ticket_num} tickets remaining")
return SaoNoopResponse(GameconnectCmd.LOGOUT_RESPONSE).make()
async def handle_c108(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/logout_ticket_unpurchased
req = SaoLogoutTicketUnpurchasedRequest(header, request)
self.logger.info(f"User {req.user_id} Logout from {'game' if req.cabinet_type == 0 else 'terminal'} @ {src_ip} without buying a ticket")
return SaoNoopResponse(GameconnectCmd.LOGOUT_TICKET_UNPURCHASED_RESPONSE).make()
async def handle_c10a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/paying_play_start
req = SaoPayingPlayStartRequest(header, request)
self.logger.info(f"User {req.paying_user_id} started paying session @ {req.store_name} ({src_ip}) on cab {req.serial_no}")
resp = SaoPayingPlayStartResponse()
# TODO: session management
return resp.make()
async def handle_c10c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/paying_play_end
req = SaoPayingPlayEndRequest(header, request)
self.logger.info(f"User {req.paying_user_id} ended paying session {req.paying_session_id} @ {req.store_name} ({src_ip}) on cab {req.serial_no} after {req.played_amount} {req.played_type} type games")
return SaoNoopResponse(GameconnectCmd.PAYING_PLAY_END_RESPONSE).make()
async def handle_c10e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/purchase_ticket
req = SaoPurchaseTicketRequest(header, request)
self.logger.info(f"User {req.user_id} pruchased {req.purchase_num} tickets (ID {req.ticket_id}) @ {src_ip} with discout type {req.discount_type}")
return SaoNoopResponse(GameconnectCmd.PURCHASE_TICKET_RESPONSE).make()
async def handle_c110(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/consume_ticket
req = SaoConsumeTicketRequest(header, request)
self.logger.info(f"User {req.user_id} consumed {req.consume_num} tickets (ID {req.ticket_id}) @ {src_ip} with discout type {req.discount_type} on {req.act_type}")
return SaoNoopResponse(GameconnectCmd.CONSUME_TICKET_RESPONSE).make()
async def handle_c112(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/add_credit
req = SaoAddCreditRequest(header, request)
self.logger.info(f"User {req.user_id} added {req.add_num} credits to a {'game' if req.cabinet_type == 0 else 'terminal'} @ {src_ip}")
return SaoNoopResponse(GameconnectCmd.ADD_CREDIT_RESPONSE).make()
async def handle_c114(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/consume_credit
req = SaoConsumeCreditRequest(header, request)
self.logger.info(f"User {req.user_id} consumed {req.consume_num} credits on a {'game' if req.cabinet_type == 0 else 'terminal'} @ {req.store_id} ({src_ip}) on {req.act_type}")
return SaoNoopResponse(GameconnectCmd.CONSUME_CREDIT_RESPONSE).make()
async def handle_c116(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/purchase_ticket_guest
req = SaoPurchaseTicketGuestRequest(header, request)
self.logger.info(f"Guest purchased {req.purchase_num} tickets on a {'game' if req.cabinet_type == 0 else 'terminal'} @ {req.store_id} ({src_ip} | SN {req.serial_no})")
return SaoNoopResponse(GameconnectCmd.PURCHASE_TICKET_GUEST_RESPONSE).make()
async def handle_c118(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/consume_ticket_guest
req = SaoConsumeTicketGuestRequest(header, request)
self.logger.info(f"Guest consumed {req.consume_num} tickets @ {req.store_id} ({src_ip} | SN {req.serial_no}) with discout type {req.discount_type} on {req.act_type}")
return SaoNoopResponse(GameconnectCmd.CONSUME_TICKET_GUEST_RESPONSE).make()
async def handle_c11a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/add_credit_guest
req = SaoAddCreditGuestRequest(header, request)
self.logger.info(f"Guest added {req.add_num} credits to a {'game' if req.cabinet_type == 0 else 'terminal'} @ {req.store_id} ({src_ip} | SN {req.serial_no})")
return SaoNoopResponse(GameconnectCmd.ADD_CREDIT_GUEST_RESPONSE).make()
async def handle_c11c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/consume_credit_guest
req = SaoConsumeCreditGuestRequest(header, request)
self.logger.info(f"Guest consumed {req.consume_num} credits on a {'game' if req.cab_type == 0 else 'terminal'} @ {req.shop_id} ({src_ip} | SN {req.serial_num}) on {req.act_type}")
return SaoNoopResponse(GameconnectCmd.CONSUME_CREDIT_GUEST_RESPONSE).make()
async def handle_c11e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#common/get_auth_card_data
req = SaoGetAuthCardDataRequest(header, request)
#Check authentication
card = await self.data.card.get_card_by_access_code( req.access_code )
if not card:
card = await self.data.card.get_card_by_idm(req.chip_id[:16])
if not card:
self.logger.info(f"Unregistered card {req.access_code} authenticated from {req.serial_no}")
return SaoGetAuthCardDataResponse("NEW PLAYER", 0).make()
await self.data.card.set_access_code_by_access_code(card['access_code'], req.access_code)
else:
user_id = card['user']
card_id = card['id']
if req.access_code.startswith("5") and not card['idm']:
await self.data.card.set_idm_by_access_code(card_id, req.chip_id[:16])
elif (req.access_code.startswith("010") or req.access_code.startswith("3")) and not card['chip_id'] and int(req.chip_id[:8], 16) != 0x04030201:
await self.data.card.set_chip_id_by_access_code(card_id, int(req.chip_id[:8], 16))
self.logger.info(f"User Authenticated from {req.serial_no}: { req.access_code } | { user_id }")
#Grab values from profile
profile_data = await self.data.profile.get_profile(user_id)
if not profile_data:
self.logger.info(f"Unregistered user {user_id} with card {req.access_code} authenticated from {req.serial_no}")
return SaoGetAuthCardDataResponse("NEW PLAYER", user_id).make()
return SaoGetAuthCardDataResponse(profile_data['nick_name'], user_id).make()
async def handle_c120(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/get_access_code_by_keitai
req = SaoGetAccessCodeByKeitaiRequest(header, request)
cid = req.chip_id
idm = cid[:16]
pmm = cid[16:]
card = await self.data.card.get_card_by_idm(idm)
# If we don't have that card saved locally, check aimedb
if not card:
# Validate that we're talking to a phone
if not int(pmm[2:4], 16) in self.data.card.moble_os_codes:
self.logger.warn(f"{req.serial_no} looked up non-moble chip ID {cid}!")
return SaoGetAccessCodeByKeitaiResponse("").make()
# TODO: Actual felica moble registration
return SaoGetAccessCodeByKeitaiResponse("").make()
#ac = await self.data.card.register_felica_moble_ac(idm, pmm)
# if we didn't get an access code, fail hard
if not ac:
self.logger.warn(f"Failed to register access code for chip ID {cid} requested by {req.serial_no}")
return SaoGetAccessCodeByKeitaiResponse("").make()
self.logger.info(f"Successfully registered moble felica access code {ac} for chip ID {cid} requested by {req.serial_no}")
uid = await self.data.user.create_user()
if not uid:
self.logger.error(f"Failed to create user for chip ID {cid} (access code {ac}) @ LoadAccessCode request from {req.serial_no}")
return SaoGetAccessCodeByKeitaiResponse("").make()
cardid = await self.data.card.create_card(uid, ac)
if not cardid:
self.logger.error(f"Failed to create card for user {uid} with chip ID {cid} (access code {ac}) @ LoadAccessCode request from {req.serial_no}")
await self.data.user.delete_user(uid)
return SaoGetAccessCodeByKeitaiResponse("").make()
self.logger.info(f"Moble Felica access code lookup for {cid} -> {ac} (user {uid}) requested by {req.serial_no}")
else:
ac = card['access_code']
uid = card['user']
self.logger.info(f"Moble Felica access code for {cid} -> {ac} (user {uid}) requested by {req.serial_no}")
return SaoGetAccessCodeByKeitaiResponse(ac).make()
async def handle_c122(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#common/get_maintenance_info
resp = SaoGetMaintenanceInfoResponse()
return resp.make()
async def handle_c124(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#common/get_resource_path_info
resp = SaoGetResourcePathInfoResponse(f"https://{self.core_cfg.server.hostname}/saoresource/")
return resp.make()
async def handle_c126(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/validation_error_notification
req = SaoValidationErrorNotificationRequest(header, request)
self.logger.warn(f"User {req.user_id} on {'game' if req.cabinet_type == 0 else 'terminal'} {req.serial_no} @ {req.store_name} ({src_ip} | Place ID {req.place_id}) " \
+ f"Validation error: {req.send_protocol_name} || {req.send_data_to_fraud_value} || {req.send_data_to_modification_value}")
return SaoNoopResponse(GameconnectCmd.VALIDATION_ERROR_NOTIFICATION_RESPONSE).make()
async def handle_c128(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/power_cutting_return_notification
req = SaoPowerCuttingReturnNotification(header, request)
self.logger.warn(f"User {req.user_id} on {'game' if req.cabinet_type == 0 else 'terminal'} {req.serial_no} @ {req.store_name} ({src_ip} | Place ID {req.place_id}) " \
+ f"Power outage return: Act Type {req.last_act_type} || {req.remaining_ticket_num} Remaining Tickets || {req.remaining_credit_num} Remaining Credits")
return SaoNoopResponse(GameconnectCmd.POWER_CUTTING_RETURN_NOTIFICATION_RESPONSE).make()
async def handle_c12a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#common/give_free_ticket
req = SaoGiveFreeTicketRequest(header, request)
self.logger.info(f"Give {req.give_num} free tickets (id {req.ticket_id}) to user {req.user_id}")
resp = SaoNoopResponse(GameconnectCmd.GIVE_FREE_TICKET_RESPONSE)
return resp.make()
async def handle_c12c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# common/matching_error_notification
req = SaoMatchingErrorNotificationRequest(header, request)
self.logger.warn(f"{'game' if req.cabinet_type == 0 else 'terminal'} {req.serial_no} @ {req.store_name} ({src_ip} | Place ID {req.place_id}) " \
+ f"Matching error: {req.matching_error_data_list[0]}")
return SaoNoopResponse(GameconnectCmd.MATCHING_ERROR_NOTIFICATION_RESPONSE).make()
async def handle_c12e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#common/ac_cabinet_boot_notification
req = SaoCommonAcCabinetBootNotificationRequest(header, request)
if req.current_version_app_id < self.game_cfg.server.game_version:
self.logger.info(f"!!OUTDATED!! {'Game' if req.cabinet_type == 0 else 'Terminal'} {req.serial_no} Booted v{req.current_version_app_id} (Master data v{req.current_master_data_version}): {req.store_name} ({src_ip} | Place/Shop ID {req.place_id}/{req.store_id})")
if req.current_version_app_id > self.game_cfg.server.game_version:
self.logger.info(f"!!TOO NEW!! {'Game' if req.cabinet_type == 0 else 'Terminal'} {req.serial_no} Booted v{req.current_version_app_id} (Master data v{req.current_master_data_version}): {req.store_name} ({src_ip} | Place/Shop ID {req.place_id}/{req.store_id})")
self.logger.info(f"{'Game' if req.cabinet_type == 0 else 'Terminal'} {req.serial_no} Booted v{req.current_version_app_id} (Master data v{req.current_master_data_version}): {req.store_name} ({src_ip} | Place/Shop ID {req.place_id}/{req.store_id})")
resp = SaoNoopResponse(GameconnectCmd.AC_CABINET_BOOT_NOTIFICATION_RESPONSE)
return resp.make()
async def handle_c200(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# tutorial/first_tutorial_end
req = SaoGenericUserTicketRequest(header, request)
self.logger.info(f"User {req.user_id} (ticket {req.ticket_id}) finished first tutorial")
return SaoNoopResponse(GameconnectCmd.FIRST_TUTORIAL_END_RESPONSE).make()
async def handle_c202(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# tutorial/various_tutorial_end
req = SaoVariousTutorialEndRequest(header, request)
await self.data.profile.add_tutorial_byte(int(req.user_id), req.tutorial_type)
return SaoNoopResponse(GameconnectCmd.VARIOUS_TUTORIAL_END_RESPONSE).make()
async def handle_c204(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# tutorial/get_various_tutorial_data_list
req = SaoGenericUserRequest(header, request)
tuts = await self.data.profile.get_tutorial_bytes(int(req.user_id))
resp = SaoGetVariousTutorialDataListResponse()
if tuts:
for t in tuts:
resp.end_tutorial_type_list.append(t['tutorial_byte'])
return resp.make()
async def handle_c300(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# card/discharge_profile_card
req = SaoDischargeProfileCardRequest(header, request)
# Real cards seem to start with 10-17 as the first 2 digits, so we'll anchor ours with 2 to ensure no overlap
sn = f"2{str(randint(1, 999999999999999999)).zfill(18)}"
while await self.data.profile.get_hero_card(sn):
sn = f"2{str(randint(1, 999999999999999999)).zfill(18)}"
resp = SaoDischargeProfileCardResponse(sn)
db_hero = await self.data.item.get_hero_log(req.user_id, req.hero_log_user_hero_log_id)
if not db_hero:
hero_statc = await self.data.static.get_hero_by_id(db_hero['hero_log_id'])
if not hero_statc:
self.logger.error(f"Failed to find hero log {db_hero['hero_log_id']}! Please run the reader")
resp.header.error_type = ProtocolErrorNum.RESOURCE_CARD_ERR1
return resp.make()
now_have_skills = await self.hero_default_skills(hero_statc['SkillTableSubId'])
db_hero_id = await self.data.item.put_hero_log(
req.user_id,
db_hero['hero_log_id'],
1,
0,
None,
None,
now_have_skills[0],
now_have_skills[1],
now_have_skills[2],
now_have_skills[3],
now_have_skills[4]
)
if not db_hero_id:
self.logger.error(f"Failed to give user {req.user_id} hero {db_hero['hero_log_id']}!")
resp.header.error_type = ProtocolErrorNum.RESOURCE_CARD_ERR6
return resp.make()
else:
db_hero_id = db_hero['id']
await self.data.profile.put_hero_card(req.user_id, sn, db_hero_id, req.holographic_flag)
self.logger.info(f"User {req.user_id} printed {'holo ' if req.holographic_flag == 1 else ''}profile card {req.hero_log_user_hero_log_id} {req.execute_print_type}, code {sn}")
return resp.make()
async def handle_c302(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# card/discharge_resource_card
req = SaoDischargeResourceCardRequest(header, request)
for x in req.common_reward_user_data:
sn = f"2{str(randint(1, 999999999999999999)).zfill(18)}"
self.logger.info(f"User {req.user_id} printed {'holo ' if req.holographic_flag == 1 else ''}resource card {x.user_common_reward_id} {req.execute_print_type}, code {sn}")
await self.data.profile.put_resource_card(req.user_id, sn, x.common_reward_type, x.user_common_reward_id, req.holographic_flag)
return SaoDischargeProfileCardResponse(sn).make()
async def handle_c304(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# card/discharge_resource_card_complete
req = SaoDischargeResourceCardCompleteRequest(header, request)
self.logger.info(f"User {req.user_id} finished printing resource card {req.resource_card_code}")
return SaoNoopResponse(GameconnectCmd.DISCHARGE_RESOURCE_CARD_COMPLETE_RESPONSE).make()
async def handle_c306(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#card/scan_qr_quest_profile_card
req = SaoScanQrQuestProfileCardRequest(header, request)
resp = SaoScanQrQuestProfileCardResponse()
card = await self.data.profile.get_hero_card(req.profile_card_code)
if not card:
self.logger.warn(f"User {req.user_id} scanned unregistered QR code {req.profile_card_code}")
return resp.make()
hero = await self.data.item.get_hero_log_by_id(card['user_hero_id'])
if not hero: # Shouldn't happen
self.logger.warn(f"User {req.user_id} scanned QR code {req.profile_card_code} but does not have hero entry {card['user_hero_id']}")
return resp.make()
hero_static_data = await self.data.static.get_hero_by_id(hero['hero_log_id'])
if not hero_static_data: # Shouldn't happen
self.logger.warn(f"No entry for hero {hero['hero_log_id']}, please run read.py")
return resp.make()
profile = await self.data.profile.get_profile(card['user'])
if not profile: # Shouldn't happen
self.logger.warn(f"No profile for user {card['user']}, something broke")
return resp.make()
self.logger.info(f"User {req.user_id} scanned QR code {req.profile_card_code}")
card_resp = ReadProfileCard.from_args(req.profile_card_code, profile['nick_name'])
card_resp.rank_num = profile['rank_num']
card_resp.setting_title_id = profile['setting_title_id']
card_resp.skill_id = hero['skill_slot1_skill_id']
card_resp.hero_log_hero_log_id = hero['hero_log_id']
card_resp.hero_log_log_level = hero['log_level']
#TODO: Awakening
#card_resp.hero_log_awakening_stage = hero['log_level']
resp.profile_card_data.append(card_resp)
return resp.make()
async def handle_c308(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# card/scan_qr_shop_resource_card
req = SaoScanQrShopResourceCardRequest(header, request)
self.logger.info(f"User {req.user_id} scanned shop resource card {req.resource_card_code}")
resp = SaoNoopResponse(GameconnectCmd.SCAN_QR_SHOP_RESOURCE_CARD_RESPONSE)
# On official, resource cards have limited uses, but we don't track that currently (tho we should)
card = await self.data.profile.get_resource_card(req.resource_card_code) # TODO: use count
if not card:
self.logger.warn(f"No resource card with serial {req.resource_card_code} exists!")
resp.header.err_status = 4832 # Theres a few error codes but none seem to do anything?
# Also not sure if it should be this or result
return resp.make()
async def handle_c30a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# card/scan_qr_quest_resource_card
req = SaoScanQrQuestResourceCardRequest(header, request)
self.logger.info(f"User {req.user_id} scanned quest resource card {req.resource_card_code}")
card = await self.data.profile.get_resource_card(req.resource_card_code)
if not card:
resp = SaoScanQrQuestResourceCardResponse(card['common_reward_type'], card['common_reward_id'], card['holographic_flag'])
else:
self.logger.warn(f"No resource card with serial {req.resource_card_code} exists!")
resp = SaoScanQrQuestResourceCardResponse()
resp.header.err_status = 4832 # Theres a few error codes but none seem to do anything?
# Also not sure if it should be this or result
return resp.make()
async def handle_c400(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#home/check_yui_medal_get_condition
req = SaoCheckYuiMedalGetConditionRequest(header, request)
profile = await self.data.profile.get_profile(req.user_id)
if profile['last_yui_medal_date']:
last_check_ts = int(profile['last_yui_medal_date'].timestamp())
day_ts = int(datetime.now().replace(hour=0, minute=0, second=0, microsecond=0).timestamp())
diff_ts = day_ts - last_check_ts
if diff_ts > 0:
num_days = diff_ts / 86400
else:
num_days = 0
else:
num_days = 1
if num_days > 1:
await self.data.profile.add_yui_medals(req.user_id)
await self.data.profile.update_yui_medal_date(req.user_id)
return SaoCheckYuiMedalGetConditionResponse(num_days, 1 if num_days > 1 else 0).make()
async def handle_c402(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#home/get_yui_medal_bonus_user_data
req = SaoGetYuiMedalBonusUserDataRequest(header, request)
resp = SaoGetYuiMedalBonusUserDataResponse() # TODO: Track yui login bonus
return resp.make()
async def handle_c404(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# home/check_comeback_event
req = SaoGenericUserTicketRequest(header, request)
resp = SaoCheckComebackEventResponse()
#resp.get_comeback_event_id_list += [1,2,3,4] # TODO: track comeback date
return resp.make()
async def handle_c406(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# home/change_my_store
req = SaoChangeMyStoreRequest(header, request)
self.logger.info(f"User {req.user_id} changed My Store to {req.store_id}")
shop_id = int(req.store_id[3:], 16)
await self.data.profile.set_my_shop(req.user_id, shop_id)
return SaoNoopResponse(GameconnectCmd.CHANGE_MY_STORE_RESPONSE).make()
async def handle_c408(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# home/check_title_get_decision
req = SaoCheckTitleGetDecisionRequest(header, request)
resp = SaoCheckTitleGetDecisionResponse() # TODO: titles
return resp.make()
async def handle_c40a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# home/check_profile_card_used_reward
req = SaoCheckProfileCardUsedRewardRequest(header, request)
resp = SaoCheckProfileCardUsedRewardResponse() # TODO: check_profile_card_used_reward
return resp.make()
async def handle_c40c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# home/check_ac_login_bonus
req = SaoGenericUserTicketRequest(header, request)
resp = SaoCheckAcLoginBonusResponse()
#resp.get_ac_login_bonus_id_list.append(1) # TODO: track login bonus date
#resp.get_ac_login_bonus_id_list.append(2)
return resp.make()
async def handle_c500(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# user_info/get_user_basic_data
req = SaoGetUserBasicDataRequest(header, request)
profile_data = await self.data.profile.get_profile(req.user_id)
player_rank_data = self.load_data_csv("PlayerRank")
resp = SaoGetUserBasicDataResponse(profile_data)
for e in player_rank_data:
if resp.user_basic_data[0].rank_num == int(e['PlayerRankId']):
resp.user_basic_data[0].rank_exp = resp.user_basic_data[0].rank_exp - int(e["TotalExp"])
break
if profile_data['my_shop']:
ac = await self.data.arcade.get_arcade(profile_data['my_shop'])
if ac:
resp.user_basic_data[0].my_store_name = ac['name']
return resp.make()
async def handle_c502(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# user_info/get_vp_gasha_ticket_data_list
req = SaoGetVpGashaTicketDataListRequest(header, request)
# TODO: gasha tickets
resp = SaoGetVpGashaTicketDataListResponse()
return resp.make()
async def handle_c504(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# user_info/get_present_box_num
req = SaoGenericUserRequest(header, request)
# TODO: presents
return SaoGetPresentBoxNumResponse().make()
async def handle_c600(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# have_object/get_hero_log_user_data_list
req = SaoGenericUserRequest(header, request)
hero_level_data = self.load_data_csv("HeroLogLevel")
hero_data = await self.data.item.get_hero_logs(req.user_id)
resp = SaoGetHeroLogUserDataListResponse()
for hero in hero_data:
append = HeroLogUserData.from_args(hero)
hero_static = await self.data.static.get_hero_by_id(hero['hero_log_id'])
if not hero_static:
self.logger.warn(f"No hero for id {hero['hero_log_id']}, please run reader")
resp.hero_log_user_data_list.append(append)
continue
append.property1_property_id = hero_static['Property1PropertyId'] if hero_static['Property1PropertyId'] else 0
append.property1_value1 = hero_static['Property1Value1'] if hero_static['Property1Value1'] else 0
append.property1_value2 = hero_static['Property1Value2'] if hero_static['Property1Value2'] else 0
append.property2_property_id = hero_static['Property2PropertyId'] if hero_static['Property2PropertyId'] else 0
append.property2_value1 = hero_static['Property2Value1'] if hero_static['Property2Value1'] else 0
append.property2_value2 = hero_static['Property2Value2'] if hero_static['Property2Value2'] else 0
append.property3_property_id = hero_static['Property3PropertyId'] if hero_static['Property3PropertyId'] else 0
append.property3_value1 = hero_static['Property3Value1'] if hero_static['Property3Value1'] else 0
append.property3_value2 = hero_static['Property3Value2'] if hero_static['Property3Value2'] else 0
append.property4_property_id = hero_static['Property4PropertyId'] if hero_static['Property4PropertyId'] else 0
append.property4_value1 = hero_static['Property4Value1'] if hero_static['Property4Value1'] else 0
append.property4_value2 = hero_static['Property4Value2'] if hero_static['Property4Value2'] else 0
for e in hero_level_data:
if hero['log_level'] == int(e['HeroLogLevelId']):
append.log_exp = hero['log_exp'] - int(e["TotalExp"])
break
resp.hero_log_user_data_list.append(append)
return resp.make()
async def handle_c602(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# have_object/get_equipment_user_data_list
req = SaoGenericUserRequest(header, request)
resp = SaoGetEquipmentUserDataListResponse()
equip_level_data = self.load_data_csv("EquipmentLevel")
equipment_data = await self.data.item.get_user_equipments(req.user_id)
if equipment_data:
for equipment in equipment_data:
e = EquipmentUserData.from_args(equipment)
weapon_static = await self.data.static.get_equipment_by_id(equipment['equipment_id'])
if not weapon_static:
self.logger.warn(f"No equipment for id {equipment['equipment_id']}, please run reader")
resp.equipment_user_data_list.append(e)
continue
if not e.property1_property_id:
e.property1_property_id = weapon_static['Property1PropertyId'] if weapon_static['Property1PropertyId'] else 0
e.property1_value1 = weapon_static['Property1Value1'] if weapon_static['Property1Value1'] else 0
e.property1_value2 = weapon_static['Property1Value2'] if weapon_static['Property1Value2'] else 0
if not e.property2_property_id:
e.property2_property_id = weapon_static['Property2PropertyId'] if weapon_static['Property2PropertyId'] else 0
e.property2_value1 = weapon_static['Property2Value1'] if weapon_static['Property2Value1'] else 0
e.property2_value2 = weapon_static['Property2Value2'] if weapon_static['Property2Value2'] else 0
if e.property3_property_id:
e.property3_property_id = weapon_static['Property3PropertyId'] if weapon_static['Property3PropertyId'] else 0
e.property3_value1 = weapon_static['Property3Value1'] if weapon_static['Property3Value1'] else 0
e.property3_value2 = weapon_static['Property3Value2'] if weapon_static['Property3Value2'] else 0
if e.property4_property_id:
e.property4_property_id = weapon_static['Property4PropertyId'] if weapon_static['Property4PropertyId'] else 0
e.property4_value1 = weapon_static['Property4Value1'] if weapon_static['Property4Value1'] else 0
e.property4_value2 = weapon_static['Property4Value2'] if weapon_static['Property4Value2'] else 0
for f in equip_level_data:
if equipment['enhancement_value'] == int(f['EquipmentLevelId']):
e.enhancement_exp = equipment['enhancement_exp'] - int(f["TotalExp"])
break
resp.equipment_user_data_list.append(e)
return resp.make()
async def handle_c604(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# have_object/get_item_user_data_list
req = SaoGenericUserRequest(header, request)
item_data = await self.data.item.get_user_items(req.user_id)
resp = SaoGetItemUserDataListResponse(item_data)
return resp.make()
async def handle_c606(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# have_object/get_support_log_user_data_list
req = SaoGenericUserRequest(header, request)
resp = SaoGetSupportLogUserDataListResponse()
supports = self.load_data_csv("SupportLog")
# TODO: Save supports
for x in range(len(supports)):
tmp = SupportLogUserData.from_args(f"{req.user_id}{x}", supports[x]['SupportLogId'])
resp.support_log_user_data_list.append(tmp)
return resp.make()
async def handle_c608(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#have_object/get_episode_append_data_list
req = SaoGenericUserRequest(header, request)
# TODO: Appends
resp = SaoGetEpisodeAppendDataListResponse()
return resp.make()
async def handle_c60a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# have_object/get_event_item_data_list
req = SaoGenericUserRequest(header, request)
res = SaoGetEventItemDataListResponse()
# TODO: Event items maybe
return res.make()
async def handle_c60c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# have_object/get_gasha_medal_user_data_list
req = SaoGenericUserRequest(header, request)
res = SaoGetGashaMedalUserDataListResponse()
# TODO: Gasha Medal data
return res.make()
async def handle_c700(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# shop/get_shop_resource_sales_data_list
# TODO: Get user shop data
req = SaoGenericUserRequest(header, request)
resp = SaoGetShopResourceSalesDataListResponse()
return resp.make()
async def handle_c702(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# shop/purchase_shop_resource
req = SaoPurchaseShopResourceRequest(header, request)
self.logger.infof(f"User {req.user_id} (ticket {req.ticket_id}) purchased shop resourse {req.user_shop_resource_id}")
# TODO: Shop purchases
return SaoNoopResponse(GameconnectCmd.PURCHASE_SHOP_RESOURCE_RESPONSE).make()
async def handle_c704(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# shop/discard_shop_resource
req = SaoPurchaseShopResourceRequest(header, request)
self.logger.infof(f"User {req.user_id} (ticket {req.ticket_id}) discarded shop resourse {req.user_shop_resource_id}")
return SaoNoopResponse(GameconnectCmd.DISCARD_SHOP_RESOURCE_RESPONSE).make()
async def handle_c800(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# custom/get_title_user_data_list
req = SaoGenericUserRequest(header, request)
titleIdsData = await self.data.static.get_title_ids(0, True)
# TODO: Save titles
resp = SaoGetTitleUserDataListResponse(req.user_id, titleIdsData)
return resp.make()
async def handle_c802(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# custom/change_title
req = SaoChangeTitleRequest(header, request)
self.logger.info(f"User {req.user_id} (ticket {req.ticket_id}) changed their title to {req.user_title_id}")
await self.data.profile.set_title(req.user_id, req.user_title_id)
return SaoNoopResponse(GameconnectCmd.CHANGE_TITLE_RESPONSE).make()
async def handle_c804(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# custom/get_party_data_list
req = SaoGenericUserRequest(header, request)
resp = SaoGetPartyDataListResponse()
hero_parties = await self.data.item.get_hero_party(req.user_id)
for party in hero_parties:
hero1_data = await self.data.item.get_user_hero_by_id(party['user_hero_log_id_1'])
hero2_data = await self.data.item.get_user_hero_by_id(party['user_hero_log_id_2'])
hero3_data = await self.data.item.get_user_hero_by_id(party['user_hero_log_id_3'])
resp.party_data_list.append(PartyData.from_args(party['id'], party['user_party_team_id'], hero1_data._asdict(), hero2_data._asdict(), hero3_data._asdict()))
return resp.make()
async def handle_c808(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# custom/get_support_log_party_data_list
req = SaoGenericUserRequest(header, request)
resp = SaoGetSupportLogPartyDataListResponse()
# TODO: Support logs
return resp.make()
async def handle_c812(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# custom/disposal_resource
req = SaoDisposalResourceRequest(header, request)
get_col = 0
for disposal in req.disposal_common_reward_user_data_list:
if disposal.common_reward_type == RewardType.HeroLog:
removed_hero = await self.data.item.remove_hero_log(disposal.user_common_reward_id)
if removed_hero:
static_hero = await self.data.static.get_hero_by_id(removed_hero)
get_col += static_hero['SalePrice']
elif disposal.common_reward_type == RewardType.Equipment:
removed_equip = await self.data.item.remove_equipment(disposal.user_common_reward_id)
if removed_equip:
static_equip = await self.data.static.get_equipment_by_id(removed_equip)
get_col += static_equip['SalePrice']
elif disposal.common_reward_type == RewardType.Item:
removed_equip = await self.data.item.remove_item(disposal.user_common_reward_id)
if removed_equip:
static_equip = await self.data.static.get_equipment_by_id(removed_equip)
get_col += static_equip['SalePrice']
elif disposal.common_reward_type == RewardType.SupportLog:
continue
else:
self.logger.warn(f"Unhandled disposal type {disposal.common_reward_type}")
await self.data.profile.add_col(req.user_id, get_col)
return SaoDisposalResourceResponse(get_col).make()
async def handle_c814(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#custom/synthesize_enhancement_hero_log
req = SaoSynthesizeEnhancementHeroLogRequest(header, request)
resp = SaoSynthesizeEnhancementHeroLogResponse()
hero_level_data = self.load_data_csv("HeroLogLevel")
equip_level_data = self.load_data_csv("EquipmentLevel")
hero_exp = 0
col_cost = 0
for x in req.material_common_reward_user_data_list:
if x.common_reward_type == RewardType.Item:
user_item = await self.data.item.get_user_item_by_id(x.user_common_reward_id)
static_item = await self.data.static.get_item_id(user_item['item_id'])
if int(static_item['ItemTypeId']) == 7:
hero_exp += int(static_item['Value'])
self.logger.info(f"Remove item {x.user_common_reward_id} and add {int(static_item['Value'])} XP (running {hero_exp})")
await self.data.item.remove_item(x.user_common_reward_id)
elif x.common_reward_type == RewardType.Equipment:
equipment_data = await self.data.item.get_user_equipment_by_id(x.user_common_reward_id)
if equipment_data is None:
self.logger.error(f"Failed to find equipment {x.user_common_reward_id} for user {req.user_id}!")
continue
req_exp = 0
for e in range(len(equip_level_data)):
if equipment_data['enhancement_value'] == int(equip_level_data[e]['EquipmentLevelId']):
req_exp = equip_level_data[e + 1]['RequireExp']
break
static_equip_data = await self.data.static.get_equipment_by_id(equipment_data['equipment_id'])
hero_exp += int(static_equip_data['CompositionExp']) + req_exp
self.logger.info(f"Remove equipment {x.user_common_reward_id} and add {int(static_equip_data['CompositionExp']) + req_exp} XP (running {hero_exp})")
await self.data.item.remove_equipment(x.user_common_reward_id)
elif x.common_reward_type == RewardType.HeroLog:
hero_data = await self.data.item.get_hero_log_by_id(x.user_common_reward_id)
if hero_data is None:
self.logger.error(f"Failed to find hero {x.user_common_reward_id} for user {req.user_id}!")
continue
req_exp = 0
for e in range(len(hero_level_data)):
if hero_data['log_level'] == int(hero_level_data[e]['HeroLogLevelId']):
req_exp = hero_level_data[e + 1]['RequireExp']
break
static_hero_data = await self.data.static.get_hero_by_id(hero_data['hero_log_id'])
hero_exp += int(static_hero_data['CompositionExp']) + req_exp
self.logger.info(f"Remove hero {x.user_common_reward_id} and add {int(static_hero_data['CompositionExp']) + req_exp} XP (running {hero_exp})")
await self.data.item.remove_hero_log(x.user_common_reward_id)
else:
self.logger.warn(f"Unhandled ype {x.common_reward_type}! (running {hero_exp})")
hero_exp = int(hero_exp * 1.5)
await self.data.item.add_hero_xp(req.origin_user_hero_log_id, hero_exp)
log_exp = await self.data.item.get_hero_xp(req.origin_user_hero_log_id)
pre_synth_xp = log_exp - hero_exp
pre_synth_level = 1
for e in range(len(hero_level_data)):
if log_exp>=int(hero_level_data[e]["TotalExp"]) and log_exp<int(hero_level_data[e+1]["TotalExp"]):
self.logger.info(f"Set hero {req.origin_user_hero_log_id} level {hero_level_data[e]['HeroLogLevelId']}")
remain_exp = log_exp - int(hero_level_data[e]["TotalExp"])
await self.data.item.set_hero_level(req.origin_user_hero_log_id, hero_level_data[e]["HeroLogLevelId"])
break
for e in range(len(hero_level_data)):
if pre_synth_xp>=int(hero_level_data[e]["TotalExp"]) and pre_synth_xp<int(hero_level_data[e+1]["TotalExp"]):
pre_synth_level = int(hero_level_data[e]["HeroLogLevelId"])
break
# Load the item again to push to the response handler
synthesize_hero_log_data = await self.data.item.get_hero_log_by_id(req.origin_user_hero_log_id)
col_cost = pre_synth_level * (100 * len(req.material_common_reward_user_data_list)) * (1 - synthesize_hero_log_data['awakening_stage'] * 0.1)
if pre_synth_level >= 100:
col_cost = col_cost * 10
col_cost = col_cost * 1000
col_cost = max(100, int(col_cost / 1000))
self.logger.info(f"Synthesize {hero_exp} exp for hero {req.origin_user_hero_log_id}, costing {col_cost} col")
await self.data.profile.add_col(req.user_id, -col_cost)
if synthesize_hero_log_data is not None:
resp.after_hero_log_user_data.append(HeroLogUserData.from_args(synthesize_hero_log_data))
resp.after_hero_log_user_data[0].log_exp = remain_exp
return resp.make()
async def handle_c816(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#custom/synthesize_enhancement_equipment
req_data = SaoSynthesizeEnhancementEquipmentRequest(header, request)
resp = SaoSynthesizeEnhancementEquipmentResponse()
hero_level_data = self.load_data_csv("HeroLogLevel")
equip_level_data = self.load_data_csv("EquipmentLevel")
equipment_exp = 0
col_cost = 0
for x in req_data.material_common_reward_user_data_list:
if x.common_reward_type == RewardType.Item:
user_item = await self.data.item.get_user_item_by_id(x.user_common_reward_id)
static_item = await self.data.static.get_item_id(user_item['item_id'])
if int(static_item['ItemTypeId']) == 7:
equipment_exp += int(static_item['Value'])
self.logger.info(f"Remove item {x.user_common_reward_id} and add {int(static_item['Value'])} XP (running {equipment_exp})")
await self.data.item.remove_item(x.user_common_reward_id)
elif x.common_reward_type == RewardType.Equipment:
equipment_data = await self.data.item.get_user_equipment_by_id(x.user_common_reward_id)
if equipment_data is None:
self.logger.error(f"Failed to find equipment {x.user_common_reward_id} for user {req_data.user_id}!")
continue
req_exp = 0
for e in range(len(equip_level_data)):
if equipment_data['enhancement_value'] == int(equip_level_data[e]['EquipmentLevelId']):
req_exp = equip_level_data[e + 1]['RequireExp']
break
static_equip_data = await self.data.static.get_equipment_by_id(equipment_data['equipment_id'])
equipment_exp += int(static_equip_data['CompositionExp']) + req_exp
self.logger.info(f"Remove equipment {x.user_common_reward_id} and add {int(static_equip_data['CompositionExp']) + req_exp} XP (running {equipment_exp})")
await self.data.item.remove_equipment(x.user_common_reward_id)
elif x.common_reward_type == RewardType.HeroLog:
hero_data = await self.data.item.get_hero_log_by_id(x.user_common_reward_id)
if hero_data is None:
self.logger.error(f"Failed to find hero {x.user_common_reward_id} for user {req_data.user_id}!")
continue
req_exp = 0
for e in range(len(hero_level_data)):
if hero_data['log_level'] == int(hero_level_data[e]['HeroLogLevelId']):
req_exp = hero_level_data[e + 1]['RequireExp']
break
static_hero_data = await self.data.static.get_hero_by_id(hero_data['hero_log_id'])
equipment_exp += int(static_hero_data['CompositionExp']) + req_exp
self.logger.info(f"Remove hero {x.user_common_reward_id} and add {int(static_hero_data['CompositionExp']) + req_exp} XP (running {equipment_exp})")
await self.data.item.remove_hero_log(x.user_common_reward_id)
else:
self.logger.warn(f"Unhandled ype {x.common_reward_type}! (running {equipment_exp})")
equipment_exp = int(equipment_exp * 1.5)
await self.data.item.add_equipment_enhancement_exp(req_data.origin_user_equipment_id, equipment_exp)
synthesize_equipment_data = await self.data.item.get_user_equipment(req_data.user_id, req_data.origin_user_equipment_id)
equip_exp_new = synthesize_equipment_data['enhancement_exp']
pre_synth_level = synthesize_equipment_data['enhancement_value']
new_synth_level = 1
for e in range(len(equip_level_data)):
if equip_exp_new>=int(equip_level_data[e]["TotalExp"]) and equip_exp_new<int(equip_level_data[e+1]["TotalExp"]):
self.logger.info(f"Set equipment {req_data.origin_user_equipment_id} level {equip_level_data[e]['EquipmentLevelId']}")
new_synth_level = equip_level_data[e]['EquipmentLevelId']
remain_exp = equip_exp_new - int(equip_level_data[e]["TotalExp"])
await self.data.item.set_equipment_enhancement_value(req_data.origin_user_equipment_id, equip_level_data[e]["EquipmentLevelId"])
break
# Load the item again to push to the response handler
col_cost = pre_synth_level * (100 * len(req_data.material_common_reward_user_data_list)) * (1 - synthesize_equipment_data['awakening_stage'] * 0.1)
if pre_synth_level >= 100:
col_cost = col_cost * 10
col_cost = col_cost * 1000
col_cost = max(100, int(col_cost / 1000))
self.logger.info(f"Synthesize {equipment_exp} exp for equipment {req_data.origin_user_equipment_id}, costing {col_cost} col")
await self.data.profile.add_col(req_data.user_id, -col_cost)
if synthesize_equipment_data is not None:
resp.after_equipment_user_data.append(EquipmentUserData.from_args(synthesize_equipment_data))
resp.after_equipment_user_data[0].enhancement_exp = remain_exp
resp.after_equipment_user_data[0].enhancement_value = new_synth_level
return resp.make()
async def handle_c806(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#custom/change_party
req_data = SaoChangePartyRequest(header, request)
for party_team in req_data.party_data_list:
for hero in party_team.party_team_data_list:
await self.data.item.set_user_hero_weapons(
int(hero.user_hero_log_id),
int(hero.main_weapon_user_equipment_id) if hero.main_weapon_user_equipment_id and int(hero.main_weapon_user_equipment_id) > 0 else None,
int(hero.sub_equipment_user_equipment_id) if hero.sub_equipment_user_equipment_id and int(hero.sub_equipment_user_equipment_id) > 0 else None
)
await self.data.item.set_user_hero_skills(
int(hero.user_hero_log_id),
hero.skill_slot1_skill_id if hero.skill_slot1_skill_id > 0 else None,
hero.skill_slot2_skill_id if hero.skill_slot2_skill_id > 0 else None,
hero.skill_slot3_skill_id if hero.skill_slot3_skill_id > 0 else None,
hero.skill_slot4_skill_id if hero.skill_slot4_skill_id > 0 else None,
hero.skill_slot5_skill_id if hero.skill_slot5_skill_id > 0 else None
)
await self.data.item.put_hero_party(
req_data.user_id,
party_team.team_no,
party_team.party_team_data_list[0].user_hero_log_id,
party_team.party_team_data_list[1].user_hero_log_id,
party_team.party_team_data_list[2].user_hero_log_id,
)
resp = SaoNoopResponse(GameconnectCmd.CHANGE_PARTY_RESPONSE)
return resp.make()
async def handle_c900(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#quest/get_quest_scene_user_data_list
req = SaoGenericUserRequest(header, request)
quest_data = await self.data.item.get_quest_logs(req.user_id)
resp = SaoGetQuestSceneUserDataListResponse()
for quest in quest_data:
ex_bonus_data = await self.data.item.get_player_ex_bonus_by_quest(req.user_id, quest["quest_scene_id"])
tmp = QuestSceneUserData.from_args(quest)
for ex_bonus in ex_bonus_data:
tmp.quest_scene_ex_bonus_user_data_list.append(QuestSceneExBonusUserData.from_args(ex_bonus['ex_bonus_table_id'], ex_bonus['quest_clear_flag']))
resp.quest_scene_user_data_list.append(tmp)
return resp.make()
async def handle_c902(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes: # for whatever reason, having all entries empty or filled changes nothing
#quest/get_quest_scene_prev_scan_profile_card
resp = SaoGetQuestScenePrevScanProfileCardResponse()
resp.profile_card_data.append(ReadProfileCardData.from_args({}, {}))
return resp.make()
async def handle_c904(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#quest/episode_play_start
req_data = SaoEpisodePlayStartRequest(header, request)
user_id = req_data.user_id
profile_data = await self.data.profile.get_profile(user_id)
sesh_id = await self.data.item.create_session(
user_id,
int(req_data.play_start_request_data[0].user_party_id),
req_data.episode_id,
req_data.play_mode,
req_data.play_start_request_data[0].quest_drop_boost_apply_flag
)
resp = SaoEpisodePlayStartResponse()
resp.play_start_response_data.append(QuestScenePlayStartResponseData.from_args(sesh_id, profile_data['nick_name']))
resp.multi_play_start_response_data.append(QuestSceneMultiPlayStartResponseData.from_args())
return resp.make()
async def handle_c908(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes: # Level calculation missing for the profile and heroes
#quest/episode_play_end
req_data = SaoEpisodePlayEndRequest(header, request)
resp = SaoEpisodePlayEndResponse()
user_id = req_data.user_id
episode_id = req_data.episode_id
play_end = req_data.play_end_request_data[0]
base_get_data = play_end.base_get_data[0]
score_data = play_end.score_data[0]
exp = 0
cleared_mission_ct = 0
highest_mission_diff_cleared = 0
num_monsters_defeated = 0
monsters_defeated_data = {}
json_data = {"data": []}
ep_data = await self.data.static.get_episode_by_id(req_data.episode_id)
quest_scene = await self.data.static.get_quest_by_id(int(ep_data['QuestSceneId']))
reward_table = await self.data.static.get_rewards_by_subtable(int(quest_scene['RewardTableSubId']))
ex_bonus_table = await self.data.static.get_ex_bonuses_by_subtable(int(ep_data['ExBonusTableSubId']))
await self.data.profile.add_col(user_id, base_get_data.get_col)
quest_clear_flag = score_data.clear_time > 0
clear_time = score_data.clear_time
combo_num = score_data.combo_num
total_damage = score_data.total_damage
concurrent_destroying_num = score_data.concurrent_destroying_num
if quest_clear_flag is True:
await self.data.profile.add_vp(user_id, quest_scene['SingleRewardVp'])
await self.data.profile.add_exp(user_id, quest_scene['SuccessPlayerExp'])
exp = await self.data.profile.get_exp(user_id)
else:
await self.data.profile.add_exp(user_id, quest_scene['FailedPlayerExp'])
exp = await self.data.profile.get_exp(user_id)
# Calculate level based off experience and the CSV list
player_level_data = self.load_data_csv("PlayerRank")
for i in range(0,len(player_level_data)):
if exp>=int(player_level_data[i]["TotalExp"]) and exp<int(player_level_data[i+1]["TotalExp"]):
await self.data.profile.set_level(user_id, int(player_level_data[i]["PlayerRankId"]))
break
current_data = await self.data.item.get_quest_log(user_id, ep_data['QuestSceneId'])
if current_data:
await self.data.item.put_player_quest(
user_id,
int(QuestType.EPISODE),
ep_data['QuestSceneId'],
False if not quest_clear_flag and not current_data['quest_clear_flag'] else True,
min(clear_time, current_data['clear_time']) if quest_clear_flag else current_data['clear_time'],
max(combo_num, current_data['combo_num']),
max(int(total_damage), current_data['total_damage']),
max(concurrent_destroying_num, current_data['concurrent_destroying_num'])
)
else:
await self.data.item.put_player_quest(
user_id,
int(QuestType.EPISODE),
ep_data['QuestSceneId'],
quest_clear_flag,
clear_time,
combo_num,
total_damage,
concurrent_destroying_num
)
for mission in play_end.mission_data_list:
if mission.clear_flag == 1:
cleared_mission_ct += 1
if mission.mission_difficulty_id > highest_mission_diff_cleared:
highest_mission_diff_cleared = mission.mission_difficulty_id
for monster_data in play_end.discovery_enemy_data_list:
num_monsters_defeated += monster_data.destroy_num
monsters_defeated_data[monster_data.enemy_kind_id] = monster_data.destroy_num
for bonus in ex_bonus_table:
table_id = int(bonus['ExBonusTableId'])
ach_thing = 0
condition = int(bonus['ExBonusConditionId'])
val1 = int(bonus['ConditionValue1'])
val2 = int(bonus['ConditionValue2'])
if condition == ExBonusCondition.CLEAR_UNDER_X_SECS:
if quest_clear_flag and int(score_data.clear_time / 1000) < val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.DEFEAT_X_MONSTER_Y_TIMES:
if monsters_defeated_data.get(val1, 0) >= val2:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.DEFEAT_X_MONSTERS:
if num_monsters_defeated >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.CLEAR_X_MISSIONS:
if cleared_mission_ct >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.CLEAR_MISSION_DIFFICULTY_X:
if highest_mission_diff_cleared >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.COLLECT_X_LOGS:
if len(play_end.get_unanalyzed_log_tmp_reward_data_list) >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.CLEAR_SKILL_LEVEL_X:
if score_data.reaching_skill_level >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.NO_LOSSES:
if score_data.total_loss_num == 0:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.ACCEL_X_TIMES:
if score_data.acceleration_invocation_num >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.MAX_COMBO_X:
if score_data.combo_num >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.MULTIPLAYER_CLEAR_X:
# TODO
pass
else:
self.logger.warn(f"Unhandled EX Bonus condition {condition}")
resp.play_end_response_data[0].ex_bonus_data_list.append(QuestScenePlayEndExBonusData.from_args(table_id, ach_thing))
self.logger.info(f"User {user_id} {'cleared' if quest_clear_flag else 'ended'} episode {episode_id}")
for rare_drop in play_end.get_rare_drop_data_list:
rewardList = self.load_data_csv("QuestRareDrop")
for drop in rewardList:
if int(drop['QuestRareDropId']) == rare_drop.quest_rare_drop_id:
await self.add_reward(drop, user_id)
break
for unanalyzed_log in play_end.get_unanalyzed_log_tmp_reward_data_list:
able_rewards: List[Row] = []
for reward in reward_table:
if int(reward['UnanalyzedLogGradeId']) == unanalyzed_log.unanalyzed_log_grade_id:
able_rewards.append(reward)
randomized_unanalyzed_id = choice(able_rewards)
await self.add_reward(randomized_unanalyzed_id._asdict(), user_id)
json_data["data"].append(randomized_unanalyzed_id._asdict())
trace_table = await self.data.static.get_player_trace_by_subid(quest_scene['PlayerTraceTableSubId'])
for trace in play_end.get_player_trace_data_list:
self.logger.info(f"User {user_id} obtained trace {trace.user_quest_scene_player_trace_id}")
resp.play_end_response_data[0].play_end_player_trace_reward_data_list.append(QuestScenePlayEndPlayerTraceRewardData.from_args(choice(trace_table)._asdict()))
await self.data.item.create_end_session(user_id, ep_data['QuestSceneId'], play_end.play_result_flag, json_data["data"])
# Update heroes from the used party
play_session = await self.data.item.get_session(user_id)
session_party = await self.data.item.get_hero_party_by_id(play_session["user_party_team_id"])
if session_party:
hero_level_data = self.load_data_csv("HeroLogLevel")
hero_list = []
hero_list.append(session_party["user_hero_log_id_1"])
hero_list.append(session_party["user_hero_log_id_2"])
hero_list.append(session_party["user_hero_log_id_3"])
for i in range(0,3):
await self.data.item.add_hero_xp(hero_list[i], base_get_data.get_hero_log_exp)
log_exp = await self.data.item.get_hero_xp(hero_list[i])
# Calculate hero level based off experience and the CSV list
if log_exp:
for e in range(0,len(hero_level_data)):
if log_exp>=int(hero_level_data[e]["TotalExp"]) and log_exp<int(hero_level_data[e+1]["TotalExp"]):
self.logger.info(f"Set hero {hero_list[i]} level {hero_level_data[e]['HeroLogLevelId']} ({log_exp} total XP)")
await self.data.item.set_hero_level(hero_list[i], hero_level_data[e]['HeroLogLevelId'])
break
else:
self.logger.error(f"Failed to get session party {play_session['user_party_team_id']} data for user {user_id}!")
return resp.make()
async def handle_c90a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#quest/episode_play_end_unanalyzed_log_fixed
req = SaoEpisodePlayEndUnanalyzedLogFixedRequest(header, request)
resp = SaoEpisodePlayEndUnanalyzedLogFixedResponse()
end_session_data = await self.data.item.get_end_session(req.user_id)
for data in end_session_data['reward_data']:
resp.play_end_unanalyzed_log_reward_data_list.append(QuestScenePlayEndUnanalyzedLogRewardData.from_args(data['UnanalyzedLogGradeId'], data))
return resp.make()
async def handle_c914(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#quest/trial_tower_play_start
req_data = SaoTrialTowerPlayStartRequest(header, request)
user_id = req_data.user_id
profile_data = await self.data.profile.get_profile(user_id)
sesh_id = await self.data.item.create_session(
user_id,
int(req_data.play_start_request_data[0].user_party_id),
req_data.trial_tower_id,
req_data.play_mode,
req_data.play_start_request_data[0].quest_drop_boost_apply_flag
)
self.logger.info(f"User {req_data.user_id} started trial tower on floor {req_data.trial_tower_id}")
resp = SaoTrialTowerPlayStartResponse(sesh_id, profile_data['nick_name'])
return resp.make()
async def handle_c918(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#quest/trial_tower_play_end
req_data = SaoTrialTowerPlayEndRequest(header, request)
resp = SaoTrialTowerPlayEndResponse()
user_id = req_data.user_id
trial_tower_id = req_data.trial_tower_id
play_end = req_data.play_end_request_data[0]
base_get_data = play_end.base_get_data[0]
score_data = play_end.score_data[0]
exp = 0
cleared_mission_ct = 0
highest_mission_diff_cleared = 0
num_monsters_defeated = 0
monsters_defeated_data = {}
json_data = {"data": []}
ep_data = await self.data.static.get_tower_by_id(trial_tower_id)
quest_scene = await self.data.static.get_quest_by_id(int(ep_data['QuestSceneId']))
reward_table = await self.data.static.get_rewards_by_subtable(int(quest_scene['RewardTableSubId']))
ex_bonus_table = await self.data.static.get_ex_bonuses_by_subtable(int(ep_data['ExBonusTableSubId']))
await self.data.profile.add_col(user_id, base_get_data.get_col)
quest_clear_flag = score_data.clear_time > 0
clear_time = score_data.clear_time
combo_num = score_data.combo_num
total_damage = score_data.total_damage
concurrent_destroying_num = score_data.concurrent_destroying_num
if quest_clear_flag is True:
await self.data.profile.add_vp(user_id, quest_scene['SingleRewardVp'])
await self.data.profile.add_exp(user_id, quest_scene['SuccessPlayerExp'])
exp = await self.data.profile.get_exp(user_id)
else:
await self.data.profile.add_exp(user_id, quest_scene['FailedPlayerExp'])
exp = await self.data.profile.get_exp(user_id)
# Calculate level based off experience and the CSV list
player_level_data = self.load_data_csv("PlayerRank")
for i in range(0,len(player_level_data)):
if exp>=int(player_level_data[i]["TotalExp"]) and exp<int(player_level_data[i+1]["TotalExp"]):
await self.data.profile.set_level(user_id, int(player_level_data[i]["PlayerRankId"]))
break
current_data = await self.data.item.get_quest_log(user_id, ep_data['QuestSceneId'])
if current_data:
await self.data.item.put_player_quest(
user_id,
int(QuestType.TRIAL_TOWER),
ep_data['QuestSceneId'],
False if not quest_clear_flag and not current_data['quest_clear_flag'] else True,
min(clear_time, current_data['clear_time']) if quest_clear_flag else current_data['clear_time'],
max(combo_num, current_data['combo_num']),
max(int(total_damage), current_data['total_damage']),
max(concurrent_destroying_num, current_data['concurrent_destroying_num'])
)
else:
await self.data.item.put_player_quest(
user_id,
int(QuestType.TRIAL_TOWER),
ep_data['QuestSceneId'],
quest_clear_flag,
clear_time,
combo_num,
total_damage,
concurrent_destroying_num
)
for mission in play_end.mission_data_list:
if mission.clear_flag == 1:
cleared_mission_ct += 1
if mission.mission_difficulty_id > highest_mission_diff_cleared:
highest_mission_diff_cleared = mission.mission_difficulty_id
for monster_data in play_end.discovery_enemy_data_list:
num_monsters_defeated += monster_data.destroy_num
monsters_defeated_data[monster_data.enemy_kind_id] = monster_data.destroy_num
for bonus in ex_bonus_table:
table_id = int(bonus['ExBonusTableId'])
ach_thing = 0
condition = int(bonus['ExBonusConditionId'])
val1 = int(bonus['ConditionValue1'])
val2 = int(bonus['ConditionValue2'])
if condition == ExBonusCondition.CLEAR_UNDER_X_SECS:
if quest_clear_flag and int(score_data.clear_time / 1000) < val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.DEFEAT_X_MONSTER_Y_TIMES:
if monsters_defeated_data.get(val1, 0) >= val2:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.DEFEAT_X_MONSTERS:
if num_monsters_defeated >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.CLEAR_X_MISSIONS:
if cleared_mission_ct >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.CLEAR_MISSION_DIFFICULTY_X:
if highest_mission_diff_cleared >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.COLLECT_X_LOGS:
if len(play_end.get_unanalyzed_log_tmp_reward_data_list) >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.CLEAR_SKILL_LEVEL_X:
if score_data.reaching_skill_level >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.NO_LOSSES:
if score_data.total_loss_num == 0:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.ACCEL_X_TIMES:
if score_data.acceleration_invocation_num >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.MAX_COMBO_X:
if score_data.combo_num >= val1:
await self.add_reward(bonus._asdict(), user_id)
await self.data.item.put_ex_bonus(user_id, int(ep_data['QuestSceneId']), table_id, True)
ach_thing = 2
elif condition == ExBonusCondition.MULTIPLAYER_CLEAR_X:
# TODO
pass
else:
self.logger.warn(f"Unhandled EX Bonus condition {condition}")
resp.play_end_response_data[0].ex_bonus_data_list.append(QuestScenePlayEndExBonusData.from_args(table_id, ach_thing))
self.logger.info(f"User {user_id} {'cleared' if quest_clear_flag else 'ended'} trial tower {trial_tower_id}")
for rare_drop in play_end.get_rare_drop_data_list:
rewardList = self.load_data_csv("QuestRareDrop")
for drop in rewardList:
if int(drop['QuestRareDropId']) == rare_drop.quest_rare_drop_id:
await self.add_reward(drop, user_id)
break
for unanalyzed_log in play_end.get_unanalyzed_log_tmp_reward_data_list:
able_rewards: List[Row] = []
for reward in reward_table:
if int(reward['UnanalyzedLogGradeId']) == unanalyzed_log.unanalyzed_log_grade_id:
able_rewards.append(reward)
randomized_unanalyzed_id = choice(able_rewards)
await self.add_reward(randomized_unanalyzed_id._asdict(), user_id)
json_data["data"].append(randomized_unanalyzed_id._asdict())
trace_table = await self.data.static.get_player_trace_by_subid(quest_scene['PlayerTraceTableSubId'])
for trace in play_end.get_player_trace_data_list:
self.logger.info(f"User {user_id} obtained trace {trace.user_quest_scene_player_trace_id}")
resp.play_end_response_data[0].play_end_player_trace_reward_data_list.append(QuestScenePlayEndPlayerTraceRewardData.from_args(choice(trace_table)._asdict()))
await self.data.item.create_end_session(user_id, ep_data['QuestSceneId'], play_end.play_result_flag, json_data["data"])
# Update heroes from the used party
play_session = await self.data.item.get_session(user_id)
session_party = await self.data.item.get_hero_party_by_id(play_session["user_party_team_id"])
if session_party:
hero_level_data = self.load_data_csv("HeroLogLevel")
hero_list = []
hero_list.append(session_party["user_hero_log_id_1"])
hero_list.append(session_party["user_hero_log_id_2"])
hero_list.append(session_party["user_hero_log_id_3"])
for i in range(0,3):
self.logger.info(f"Give hero {hero_list[i]} {base_get_data.get_hero_log_exp}")
await self.data.item.add_hero_xp(hero_list[i], base_get_data.get_hero_log_exp)
log_exp = await self.data.item.get_hero_xp(hero_list[i])
# Calculate hero level based off experience and the CSV list
if log_exp:
for e in range(0,len(hero_level_data)):
if log_exp>=int(hero_level_data[e]["TotalExp"]) and log_exp<int(hero_level_data[e+1]["TotalExp"]):
self.logger.info(f"Set hero {hero_list[i]} level {hero_level_data[e]['HeroLogLevelId']} ({log_exp} total XP)")
await self.data.item.set_hero_level(hero_list[i], hero_level_data[e]['HeroLogLevelId'])
break
else:
self.logger.error(f"Failed to get session party {play_session['user_party_team_id']} data for user {user_id}!")
return resp.make()
async def handle_c91a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes: # handler is identical to the episode
#quest/trial_tower_play_end_unanalyzed_log_fixed
req = SaoTrialTowerPlayEndUnanalyzedLogFixedRequest(header, request)
resp = SaoTrialTowerPlayEndUnanalyzedLogFixedResponse()
end_session_data = await self.data.item.get_end_session(req.user_id)
for data in end_session_data['reward_data']:
resp.play_end_unanalyzed_log_reward_data_list.append(QuestScenePlayEndUnanalyzedLogRewardData.from_args(data['UnanalyzedLogGradeId'], data))
return resp.make()
async def handle_c930(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# quest/get_chat_side_story_user_data_list
req = SaoGenericUserRequest(header, request)
resp = SaoGetChatSideStoryUserDataListResponse()
return resp.make()
async def handle_ca02(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#quest_multi_play_room/get_quest_scene_multi_play_photon_server
resp = SaoGetQuestSceneMultiPlayPhotonServerResponse(self.game_cfg.server.photon_app_id)
return resp.make()
async def handle_cb02(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# quest_ranking/get_quest_hierarchy_progress_degrees_ranking_list
req = SaoGetQuestHierarchyProgressDegreesRankingListRequest(header, request)
return SaoGetQuestHierarchyProgressDegreesRankingListResponse(GameconnectCmd.GET_QUEST_HIERARCHY_PROGRESS_DEGREES_RANKING_LIST_RESPONSE).make()
async def handle_cb04(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# quest_ranking/get_quest_popular_hero_log_ranking_list
req = SaoGetQuestPopularHeroLogRankingListRequest(header, request)
return SaoGetQuestPopularHeroLogRankingListResponse(GameconnectCmd.GET_QUEST_POPULAR_HERO_LOG_RANKING_LIST_RESPONSE).make()
async def handle_cd00(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#defrag_match/get_defrag_match_basic_data
resp = SaoGetDefragMatchBasicDataResponse()
data = DefragMatchBasicUserData.from_args()
return resp.make()
async def handle_cd02(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#defrag_match/get_defrag_match_ranking_user_data
# TODO: League points
req = SaoGetDefragMatchRankingUserDataRequest(header, request)
profile = await self.data.profile.get_profile(req.user_id)
resp = SaoGetDefragMatchRankingUserDataResponse(profile._asdict())
return resp.make()
async def handle_cd04(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#defrag_match/get_defrag_match_league_point_ranking_list
resp = SaoGetDefragMatchLeaguePointRankingListResponse()
return resp.make()
async def handle_cd06(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
#defrag_match/get_defrag_match_league_score_ranking_list
resp = SaoGetDefragMatchLeagueScoreRankingListResponse()
return resp.make()
async def handle_cf0e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# gasha/get_gasha_medal_shop_user_data_list
# TODO: Get user shop data
req = GetGashaMedalShopUserDataListRequest(header, request)
resp = GetGashaMedalShopUserDataListResponse(GameconnectCmd.GET_GASHA_MEDAL_SHOP_USER_DATA_LIST_RESPONSE)
return resp.make()
async def handle_d000(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
req = SaoGenericUserRequest(header, request)
resp = SaoGetAdventureExecUserDataResponse()
return resp.make()
async def handle_d100(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# shop/get_yui_medal_shop_user_data_list
# TODO: Get user shop data
req = GetYuiMedalShopUserDataListRequest(header, request)
resp = GetYuiMedalShopUserDataListResponse(GameconnectCmd.GET_YUI_MEDAL_SHOP_USER_DATA_LIST_RESPONSE)
return resp.make()
async def handle_d200(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# mission/get_beginner_mission_user_data
req = SaoGetBeginnerMissionUserDataRequest(header, request)
resp = SaoGetBeginnerMissionUserDataResponse()
profile = await self.data.profile.get_profile(req.user_id)
if profile:
if profile['ad_confirm_date']:
resp.data[0].ad_confirm_date = profile['ad_confirm_date']
resp.data[0].ad_confirm_flag = 1
return resp.make()
async def handle_d202(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# mission/get_beginner_mission_progresses_user_data_list
req = SaoGetBeginnerMissionProgressesUserDataListRequest(header, request)
resp = SaoGetBeginnerMissionProgressesUserDataListResponse()
return resp.make()
async def handle_d204(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# mission/get_beginner_mission_seat_progresses_user_data_list
req = SaoGetBeginnerMissionSeatProgressesUserDataListRequest(header, request)
resp = SaoGetBeginnerMissionSeatProgressesUserDataListResponse()
return resp.make()
async def handle_d206(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# mission/beginner_mission_ad_confirm_notification
req = SaoBeginnerMissionAdConfirmNotificationRequest(header, request)
self.logger.info(f"User {req.user_id} confirmed ad for beginner mission {req.beginner_mission_id}")
await self.data.profile.update_beginner_mission_date(req.user_id)
return SaoNoopResponse(GameconnectCmd.BEGINNER_MISSION_AD_CONFIRM_NOTIFICATION_RESPONSE).make()
async def handle_d312(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# campaign/get_linked_site_reg_campaign_user_data
req = SaoGetLinkedSiteRegCampaignUserDataRequest(header, request)
resp = SaoGetLinkedSiteRegCampaignUserDataResponse()
return resp.make()
async def handle_d400(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# other/get_hero_log_unit_user_data_list
req = SaoGenericUserRequest(header, request)
resp = SaoGetHeroLogUnitUserDataListResponse()
return resp.make()
async def handle_d402(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# other/get_chara_unit_user_data_list
req = SaoGenericUserRequest(header, request)
resp = SaoGetCharaUnitUserDataListResponse()
return resp.make()
async def handle_d404(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# other/bnid_serial_code_check
req = SaoBnidSerialCodeCheckRequest()
resp = SaoBnidSerialCodeCheckResponse()
return resp.make()
async def handle_d404(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# other/bnid_serial_code_entry_by_appendix_card
req = SaoBnidSerialCodeEntryByAppendixCardRequest()
resp = SaoBnidSerialCodeEntryByAppendixCardResponse()
return resp.make()
async def handle_d500(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_player_ranks
tbl = self.load_data_csv('PlayerRank')
resp = SaoGetMPlayerRanksResponse(tbl)
return resp.make()
async def handle_d502(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_titles
tbl = self.load_data_csv('Title')
resp = SaoGetMTitlesResponse(tbl)
return resp.make()
async def handle_d504(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_fragments
tbl = self.load_data_csv('Fragment')
resp = SaoGetMFragmentsResponse(tbl)
return resp.make()
async def handle_d506(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_reward_tables
tbl = self.load_data_csv('RewardTable')
resp = SaoGetMRewardTablesResponse(tbl)
return resp.make()
async def handle_d508(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_reward_sets
tbl = self.load_data_csv('RewardSet')
resp = SaoGetMRewardSetsResponse(tbl)
return resp.make()
async def handle_d50a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_unanalyzed_log_grades
tbl = self.load_data_csv('UnanalyzedLogGrade')
resp = SaoGetMUnanalyzedLogGradesResponse(tbl)
return resp.make()
async def handle_d50c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_appoint_leader_params
tbl = self.load_data_csv('AppointLeaderParam')
resp = SaoGetMAppointLeaderParamsResponse(tbl)
return resp.make()
async def handle_d50e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_appoint_leader_effects
tbl = self.load_data_csv('AppointLeaderEffect')
resp = SaoGetMAppointLeaderEffectsResponse(tbl)
return resp.make()
async def handle_d510(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_appoint_leader_effect_types
tbl = self.load_data_csv('AppointLeaderEffectType')
resp = SaoGetMAppointLeaderEffectTypesResponse(tbl)
return resp.make()
async def handle_d512(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_rarities
tbl = self.load_data_csv('Rarity')
resp = SaoGetMRaritiesResponse(tbl)
return resp.make()
async def handle_d514(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_composition_events
tbl = self.load_data_csv('CompositionEvent')
resp = SaoGetMCompositionEventsResponse(tbl)
return resp.make()
async def handle_d516(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_composition_params
tbl = self.load_data_csv('CompositionParam')
resp = SaoGetMCompositionParamsResponse(tbl)
return resp.make()
async def handle_d518(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_game_play_prices
tbl = self.load_data_csv('GamePlayPrice')
resp = SaoGetMGamePlayPricesResponse(tbl)
return resp.make()
async def handle_d51a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_buy_tickets
tbl = self.load_data_csv('BuyTicket')
resp = SaoGetMBuyTicketsResponse(tbl)
return resp.make()
async def handle_d51c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_tips
tbl = self.load_data_csv('Tips')
resp = SaoGetMTipsResponse(tbl)
return resp.make()
async def handle_d51e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_caps
tbl = self.load_data_csv('Cap')
resp = SaoGetMCapsResponse(tbl)
return resp.make()
async def handle_d520(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_hero_log
tbl = self.load_data_csv('HeroLog')
resp = SaoGetMHeroLogResponse(tbl)
return resp.make()
async def handle_d522(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_hero_log_levels
tbl = self.load_data_csv('HeroLogLevel')
resp = SaoGetMHeroLogLevelsResponse(tbl)
return resp.make()
async def handle_d524(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_hero_log_roles
tbl = self.load_data_csv('HeroLogRole')
resp = SaoGetMHeroLogRolesResponse(tbl)
return resp.make()
async def handle_d526(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_hero_log_trust_ranks
tbl = self.load_data_csv('HeroLogTrustRank')
resp = SaoGetMHeroLogTrustRanksResponse(tbl)
return resp.make()
async def handle_d528(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_charas
tbl = self.load_data_csv('Chara')
resp = SaoGetMCharasResponse(tbl)
return resp.make()
async def handle_d52a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_chara_friendly_ranks
tbl = self.load_data_csv('CharaFriendlyRank')
resp = SaoGetMCharaFriendlyRanksResponse(tbl)
return resp.make()
async def handle_d52c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_equipments
tbl = self.load_data_csv('Equipment')
resp = SaoGetMEquipmentsResponse(tbl)
return resp.make()
async def handle_d52e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_equipment_levels
tbl = self.load_data_csv('EquipmentLevel')
resp = SaoGetMEquipmentLevelsResponse(tbl)
return resp.make()
async def handle_d530(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_weapon_types
tbl = self.load_data_csv('WeaponType')
resp = SaoGetMWeaponTypesResponse(tbl)
return resp.make()
async def handle_d532(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_items
tbl = self.load_data_csv('Item')
resp = SaoGetMItemsResponse(tbl)
return resp.make()
async def handle_d534(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_item_types
tbl = self.load_data_csv('ItemType')
resp = SaoGetMItemTypesResponse(tbl)
return resp.make()
async def handle_d536(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_buff_items
tbl = self.load_data_csv('BuffItem')
resp = SaoGetMBuffItemsResponse(tbl)
return resp.make()
async def handle_d538(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_enemies
tbl = self.load_data_csv('Enemy')
resp = SaoGetMEnemiesResponse(tbl)
return resp.make()
async def handle_d53a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_enemy_sets
tbl = self.load_data_csv('EnemySet')
resp = SaoGetMEnemySetsResponse(tbl)
return resp.make()
async def handle_d53c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_enemy_kinds
tbl = self.load_data_csv('EnemyKind')
resp = SaoGetMEnemyKindsResponse(tbl)
return resp.make()
async def handle_d53e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_enemy_categories
tbl = self.load_data_csv('EnemyCategory')
resp = SaoGetMEnemyCategoriesResponse(tbl)
return resp.make()
async def handle_d540(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_units
tbl = self.load_data_csv('Unit')
resp = SaoGetMUnitsResponse(tbl)
return resp.make()
async def handle_d542(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_unit_gimmicks
tbl = self.load_data_csv('UnitGimmick')
resp = SaoGetMUnitGimmicksResponse(tbl)
return resp.make()
async def handle_d544(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_unit_collisions
tbl = self.load_data_csv('UnitCollision')
resp = SaoGetMUnitCollisionsResponse(tbl)
return resp.make()
async def handle_d546(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_unit_powers
tbl = self.load_data_csv('UnitPower')
resp = SaoGetMUnitPowersResponse(tbl)
return resp.make()
async def handle_d548(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gimmick_attacks
tbl = self.load_data_csv('GimmickAttack')
resp = SaoGetMGimmickAttacksResponse(tbl)
return resp.make()
async def handle_d54a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_chara_attacks
tbl = self.load_data_csv('CharaAttack')
resp = SaoGetMCharaAttacksResponse(tbl)
return resp.make()
async def handle_d54c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_boss_attacks
tbl = self.load_data_csv('BossAttack')
resp = SaoGetMBossAttacksResponse(tbl)
return resp.make()
async def handle_d54e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_monster_attacks
tbl = self.load_data_csv('MonsterAttack')
resp = SaoGetMMonsterAttacksResponse(tbl)
return resp.make()
async def handle_d550(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_monster_actions
tbl = self.load_data_csv('MonsterAction')
resp = SaoGetMMonsterActionsResponse(tbl)
return resp.make()
async def handle_d552(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_properties
tbl = self.load_data_csv('Property')
resp = SaoGetMPropertiesResponse(tbl)
return resp.make()
async def handle_d554(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_property_tables
tbl = self.load_data_csv('PropertyTable')
resp = SaoGetMPropertyTablesResponse(tbl)
return resp.make()
async def handle_d556(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_property_types
tbl = self.load_data_csv('PropertyType')
resp = SaoGetMPropertyTypesResponse(tbl)
return resp.make()
async def handle_d558(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_skills
tbl = self.load_data_csv('Skill')
resp = SaoGetMSkillsResponse(tbl)
return resp.make()
async def handle_d55a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_skill_tables
tbl = self.load_data_csv('SkillTable')
resp = SaoGetMSkillTablesResponse(tbl)
return resp.make()
async def handle_d55c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_skill_levels
tbl = self.load_data_csv('SkillLevel')
resp = SaoGetMSkillLevelsResponse(tbl)
return resp.make()
async def handle_d55e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_awakenings
tbl = self.load_data_csv('Awakening')
resp = SaoGetMAwakeningsResponse(tbl)
return resp.make()
async def handle_d560(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_synchro_skills
tbl = self.load_data_csv('SynchroSkill')
resp = SaoGetMSynchroSkillsResponse(tbl)
return resp.make()
async def handle_d562(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_sound_skill_cut_in_voices
tbl = self.load_data_csv('Sound_SkillCutInVoice')
resp = SaoGetMSoundSkillCutInVoicesResponse(tbl)
return resp.make()
async def handle_d564(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_scenes
tbl = self.load_data_csv('QuestScene')
resp = SaoGetMQuestScenesResponse(tbl)
return resp.make()
async def handle_d566(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_exist_units
tbl = self.load_data_csv('QuestExistUnit')
resp = SaoGetMQuestExistUnitsResponse(tbl)
return resp.make()
async def handle_d568(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_episode_append_rewards
tbl = self.load_data_csv('QuestEpisodeAppendRewards')
resp = SaoGetMQuestEpisodeAppendRewardsResponse(tbl)
return resp.make()
async def handle_d56a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_side_quests
tbl = self.load_data_csv('SideQuest')
resp = SaoGetMSideQuestsResponse(tbl)
return resp.make()
async def handle_d56c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_episodes
tbl = self.load_data_csv('Episode')
resp = SaoGetMEpisodesResponse(tbl)
return resp.make()
async def handle_d56e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_episode_chapters
tbl = self.load_data_csv('EpisodeChapter')
resp = SaoGetMEpisodeChaptersResponse(tbl)
return resp.make()
async def handle_d570(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_episode_parts
tbl = self.load_data_csv('EpisodePart')
resp = SaoGetMEpisodePartsResponse(tbl)
return resp.make()
async def handle_d572(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_trial_towers
tbl = self.load_data_csv('TrialTower')
resp = SaoGetMTrialTowersResponse(tbl)
return resp.make()
async def handle_d574(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_ex_towers
req = SaoGetMExTowersRequest(header, request)
tbl = self.load_data_csv('ExTowers')
resp = SaoGetMExTowersResponse(tbl)
return resp.make()
async def handle_d576(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_ex_tower_quests
req = SaoGetMExTowerQuestsRequest(header, request)
tbl = self.load_data_csv('ExTowerQuests')
resp = SaoGetMExTowerQuestsResponse(tbl)
return resp.make()
async def handle_d578(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_menu_display_enemies
tbl = self.load_data_csv('MenuDisplayEnemy')
resp = SaoGetMMenuDisplayEnemiesResponse(tbl)
return resp.make()
async def handle_d57a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_missions
tbl = self.load_data_csv('Mission')
resp = SaoGetMMissionsResponse(tbl)
return resp.make()
async def handle_d57c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_mission_tables
tbl = self.load_data_csv('MissionTable')
resp = SaoGetMMissionTablesResponse(tbl)
return resp.make()
async def handle_d57e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_mission_difficulties
tbl = self.load_data_csv('MissionDifficulty')
resp = SaoGetMMissionDifficultiesResponse(tbl)
return resp.make()
async def handle_d580(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_battle_cameras
tbl = self.load_data_csv('BattleCamera')
resp = SaoGetMBattleCamerasResponse(tbl)
return resp.make()
async def handle_d582(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_chat_main_stories
tbl = self.load_data_csv('ChatMainStory')
resp = SaoGetMChatMainStoriesResponse(tbl)
return resp.make()
async def handle_d584(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_chat_side_stories
tbl = self.load_data_csv('ChatSideStory')
resp = SaoGetMChatSideStoriesResponse(tbl)
return resp.make()
async def handle_d586(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_chat_event_stories
req = SaoGetMChatEventStoriesRequest(header, request)
tbl = self.load_data_csv('ChatEventStory')
resp = SaoGetMChatEventStoriesResponse(tbl)
return resp.make()
async def handle_d588(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_navigator_charas
tbl = self.load_data_csv('NavigatorChara')
resp = SaoGetMNavigatorCharasResponse(tbl)
return resp.make()
async def handle_d58a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_navigator_comments
tbl = self.load_data_csv('NavigatorComment')
resp = SaoGetMNavigatorCommentsResponse(tbl)
return resp.make()
async def handle_d58c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_ex_bonus_tables
tbl = self.load_data_csv('ExBonusTable')
resp = SaoGetMExBonusTablesResponse(tbl)
return resp.make()
async def handle_d58e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_ex_bonus_conditions
tbl = self.load_data_csv('ExBonusCondition')
resp = SaoGetMExBonusConditionsResponse(tbl)
return resp.make()
async def handle_d590(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_rare_drops
tbl = self.load_data_csv('QuestRareDrop')
resp = SaoGetMQuestRareDropsResponse(tbl)
return resp.make()
async def handle_d592(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_special_rare_drop_settings
tbl = self.load_data_csv('QuestSpecialRareDropSettings')
resp = SaoGetMQuestSpecialRareDropSettingsResponse(tbl)
return resp.make()
async def handle_d594(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_special_rare_drops
tbl = self.load_data_csv('QuestSpecialRareDrops')
resp = SaoGetMQuestSpecialRareDropsResponse(tbl)
return resp.make()
async def handle_d596(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_tutorials
tbl = self.load_data_csv('QuestTutorial')
resp = SaoGetMQuestTutorialsResponse(tbl)
return resp.make()
async def handle_d598(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_player_trace_tables
tbl = self.load_data_csv('PlayerTraceTable')
resp = SaoGetMQuestPlayerTraceTablesResponse(tbl)
return resp.make()
async def handle_d59a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_stills
tbl = self.load_data_csv('QuestStill')
resp = SaoGetMQuestStillsResponse(tbl)
return resp.make()
async def handle_d59c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gashas
tbl = self.load_data_csv('Gasha')
resp = SaoGetMGashasResponse(tbl)
return resp.make()
async def handle_d59e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gasha_headers
tbl = self.load_data_csv('GashaHeader')
resp = SaoGetMGashaHeadersResponse(tbl)
return resp.make()
async def handle_d5a0(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gasha_lottery_rarities
tbl = self.load_data_csv('GashaLotteryRarity')
resp = SaoGetMGashaLotteryRaritiesResponse(tbl)
return resp.make()
async def handle_d5a2(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gasha_prizes
tbl = self.load_data_csv('GashaPrize')
resp = SaoGetMGashaPrizesResponse(tbl)
return resp.make()
async def handle_d5a4(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_comeback_events
tbl = self.load_data_csv('ComebackEvent')
resp = SaoGetMComebackEventsResponse(tbl)
return resp.make()
async def handle_d5a6(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_ad_banners
tbl = self.load_data_csv('AdBanners')
resp = SaoGetMAdBannersResponse(tbl)
return resp.make()
async def handle_d5a8(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_events
tbl = self.load_data_csv('Event')
resp = SaoGetMEventsResponse(tbl)
return resp.make()
async def handle_d5aa(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_treasure_hunts
req = SaoGetMTreasureHuntsRequest(header, request)
tbl = self.load_data_csv('TreasureHunt')
resp = SaoGetMTreasureHuntsResponse(tbl)
return resp.make()
async def handle_d5ac(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_treasure_hunt_whole_tasks
req = SaoGetMTreasureHuntWholeTasksRequest(header, request)
tbl = self.load_data_csv('TreasureHuntWholeTask')
resp = SaoGetMTreasureHuntWholeTasksResponse(tbl)
return resp.make()
async def handle_d5ae(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_treasure_hunt_individual_tasks
req = SaoGetMTreasureHuntIndividualTasksRequest(header, request)
tbl = self.load_data_csv('TreasureHuntIndividualTask')
resp = SaoGetMTreasureHuntIndividualTasksResponse(tbl)
return resp.make()
async def handle_d5b0(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_treasure_hunt_special_effects
req = SaoGetMTreasureHuntSpecialEffectsRequest(header, request)
tbl = self.load_data_csv('TreasureHuntSpecialEffect')
resp = SaoGetMTreasureHuntSpecialEffectsResponse(tbl)
return resp.make()
async def handle_d5b2(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_treasure_hunt_event_point_reward_common_rewards
req = SaoGetMTreasureHuntEventPointRewardCommonRewardsRequest(header, request)
tbl = self.load_data_csv('TreasureHuntEventPointRewardCommonReward')
resp = SaoGetMTreasureHuntEventPointRewardCommonRewardsResponse(tbl)
return resp.make()
async def handle_d5b4(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_treasure_hunt_event_point_reward_titles
req = SaoGetMTreasureHuntEventPointRewardTitlesRequest(header, request)
tbl = self.load_data_csv('TreasureHuntEventPointRewardTitle')
resp = SaoGetMTreasureHuntEventPointRewardTitlesResponse(tbl)
return resp.make()
async def handle_d5b6(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_treasure_hunt_task_texts
tbl = self.load_data_csv('TreasureHuntTaskText')
resp = SaoGetMTreasureHuntTaskTextsResponse(tbl)
return resp.make()
async def handle_d5b8(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_bnid_serial_codes
tbl = self.load_data_csv('BnidSerialCodes')
resp = SaoGetMBnidSerialCodesResponse(tbl)
return resp.make()
async def handle_d5ba(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_bnid_serial_code_rewards
tbl = self.load_data_csv('BnidSerialCodeRewards')
resp = SaoGetMBnidSerialCodeRewardsResponse(tbl)
return resp.make()
async def handle_d5bc(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_support_log
tbl = self.load_data_csv('SupportLog')
resp = SaoGetMSupportLogResponse(tbl)
return resp.make()
async def handle_d5be(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_support_log_types
tbl = self.load_data_csv('SupportLogType')
resp = SaoGetMSupportLogTypesResponse(tbl)
return resp.make()
async def handle_d5c0(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_episode_appends
tbl = self.load_data_csv('EpisodeAppends')
resp = SaoGetMEpisodeAppendsResponse(tbl)
return resp.make()
async def handle_d5c2(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_defrag_match_quests
tbl = self.load_data_csv('DefragMatchQuest')
resp = SaoGetMQuestDefragMatchQuestsResponse(tbl)
return resp.make()
async def handle_d5c4(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_quest_defrag_match_quest_boss_tables
tbl = self.load_data_csv('DefragMatchBossTable')
resp = SaoGetMQuestDefragMatchQuestBossTablesResponse(tbl)
return resp.make()
async def handle_d5c6(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_defrag_matches
req = SaoGetMDefragMatchesRequest(header, request)
tbl = self.load_data_csv('DefragMatchs')
resp = SaoGetMDefragMatchesResponse(tbl)
return resp.make()
async def handle_d5c8(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_defrag_match_seed
req = SaoGetMDefragMatchSeedRequest(header, request)
tbl = self.load_data_csv('DefragMatchSeed')
resp = SaoGetMDefragMatchSeedResponse(tbl)
return resp.make()
async def handle_d5ca(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_defrag_match_special_effects
req = SaoGetMDefragMatchSpecialEffectsRequest(header, request)
tbl = self.load_data_csv('DefragMatchSpecialEffects')
resp = SaoGetMDefragMatchSpecialEffectsResponse(tbl)
return resp.make()
async def handle_d5cc(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_defrag_match_grades
req = SaoGetMDefragMatchGradesRequest(header, request)
tbl = self.load_data_csv('DefragMatchGrade')
resp = SaoGetMDefragMatchGradesResponse(tbl)
return resp.make()
async def handle_d5ce(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_defrag_match_cpu_units
tbl = self.load_data_csv('DefragMatchCpuUnits')
resp = SaoGetMDefragMatchCpuUnitsResponse(tbl)
return resp.make()
async def handle_d5d0(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_defrag_match_cpu_support_logs
tbl = self.load_data_csv('DefragMatchCpuSupportLogs')
resp = SaoGetMDefragMatchCpuSupportLogsResponse(tbl)
return resp.make()
async def handle_d5d2(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_defrag_match_period_bonuses
req = SaoGetMDefragMatchPeriodBonusesRequest(header, request)
tbl = self.load_data_csv('DefragMatchPeriodBonuses')
resp = SaoGetMDefragMatchPeriodBonusesResponse(tbl)
return resp.make()
async def handle_d5d4(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_defrag_match_random_bonus_tables
req = SaoGetMDefragMatchRandomBonusTablesRequest(header, request)
tbl = self.load_data_csv('DefragMatchRandomBonusTables')
resp = SaoGetMDefragMatchRandomBonusTablesResponse(tbl)
return resp.make()
async def handle_d5d6(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_defrag_match_random_bonus_conditions
tbl = self.load_data_csv('DefragMatchRandomBonusConditions')
resp = SaoGetMDefragMatchRandomBonusConditionsResponse(tbl)
return resp.make()
async def handle_d5d8(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_defrag_match_rare_drops
req = SaoGetMDefragMatchRareDropsRequest(header, request)
tbl = self.load_data_csv('DefragMatchRareDrops')
resp = SaoGetMDefragMatchRareDropsResponse(tbl)
return resp.make()
async def handle_d5da(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_yui_medal_shops
tbl = self.load_data_csv('YuiMedalShops')
resp = SaoGetMYuiMedalShopsResponse(tbl)
return resp.make()
async def handle_d5dc(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_yui_medal_shop_items
tbl = self.load_data_csv('YuiMedalShopItems')
resp = SaoGetMYuiMedalShopItemsResponse(tbl)
return resp.make()
async def handle_d5de(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_event_scenes
req = SaoGetMEventScenesRequest(header, request)
tbl = self.load_data_csv('EventScenes')
resp = SaoGetMEventScenesResponse(tbl)
return resp.make()
async def handle_d5e0(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_generic_campaign_periods
tbl = self.load_data_csv('GenericCampaignPeriods')
resp = SaoGetMGenericCampaignPeriodsResponse(tbl)
return resp.make()
async def handle_d5e2(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_beginner_missions
tbl = self.load_data_csv('BeginnerMissions')
resp = SaoGetMBeginnerMissionsResponse(tbl)
return resp.make()
async def handle_d5e4(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_beginner_mission_conditions
req = SaoGetMBeginnerMissionConditionsRequest(header, request)
tbl = self.load_data_csv('BeginnerMissionConditions')
resp = SaoGetMBeginnerMissionConditionsResponse(tbl)
return resp.make()
async def handle_d5e6(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_beginner_mission_rewards
req = SaoGetMBeginnerMissionRewardsRequest(header, request)
tbl = self.load_data_csv('BeginnerMissionRewards')
resp = SaoGetMBeginnerMissionRewardsResponse(tbl)
return resp.make()
async def handle_d5e8(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_beginner_mission_seat_conditions
req = SaoGetMBeginnerMissionSeatConditionsRequest(header, request)
tbl = self.load_data_csv('BeginnerMissionSeatConditions')
resp = SaoGetMBeginnerMissionSeatConditionsResponse(tbl)
return resp.make()
async def handle_d5ea(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_beginner_mission_seat_rewards
req = SaoGetMBeginnerMissionSeatRewardsRequest(header, request)
tbl = self.load_data_csv('BeginnerMissionSeatRewards')
resp = SaoGetMBeginnerMissionSeatRewardsResponse(tbl)
return resp.make()
async def handle_d5ec(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_event_items
tbl = self.load_data_csv('EventItems')
resp = SaoGetMEventItemsResponse(tbl)
return resp.make()
async def handle_d5ee(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_event_monsters
req = SaoGetMEventMonstersRequest(header, request)
tbl = self.load_data_csv('EventMonsters')
resp = SaoGetMEventMonstersResponse(tbl)
return resp.make()
async def handle_d5f0(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_yui_medal_bonuses
tbl = self.load_data_csv('YuiMedalBonus')
resp = SaoGetMYuiMedalBonusesResponse(tbl)
return resp.make()
async def handle_d5f2(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_yui_medal_bonus_conditions
tbl = self.load_data_csv('YuiMedalBonusCondition')
resp = SaoGetMYuiMedalBonusConditionsResponse(tbl)
return resp.make()
async def handle_d5f4(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gasha_medals
tbl = self.load_data_csv('GashaMedals')
resp = SaoGetMGashaMedalsResponse(tbl)
return resp.make()
async def handle_d5f6(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gasha_medal_types
tbl = self.load_data_csv('GashaMedalTypes')
resp = SaoGetMGashaMedalTypesResponse(tbl)
return resp.make()
async def handle_d5f8(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gasha_medal_settings
req = SaoGetMGashaMedalSettingsRequest(header, request)
tbl = self.load_data_csv('GashaMedalSettings')
resp = SaoGetMGashaMedalSettingsResponse(tbl)
return resp.make()
async def handle_d5fa(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gasha_medal_bonuses
tbl = self.load_data_csv('GashaMedalBonuses')
resp = SaoGetMGashaMedalBonusesResponse(tbl)
return resp.make()
async def handle_d5fc(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gasha_medal_shops
tbl = self.load_data_csv('GashaMedalShops')
resp = SaoGetMGashaMedalShopsResponse(tbl)
return resp.make()
async def handle_d5fe(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data/get_m_gasha_medal_shop_items
req = SaoGetMGashaMedalShopItemsRequest(header, request)
tbl = self.load_data_csv('GashaMedalShopItems')
resp = SaoGetMGashaMedalShopItemsResponse(tbl)
return resp.make()
async def handle_d600(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mres_earn_campaign_applications
tbl = self.load_data_csv('ResEarnCampaignApplications')
resp = SaoGetMResEarnCampaignApplicationsResponse(tbl)
return resp.make()
async def handle_d602(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mres_earn_campaign_application_products
req = SaoGetMResEarnCampaignApplicationProductsRequest(header, request)
tbl = self.load_data_csv('ResEarnCampaignApplicationProducts')
resp = SaoGetMResEarnCampaignApplicationProductsResponse(tbl)
return resp.make()
async def handle_d604(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mres_earn_campaign_shops
tbl = self.load_data_csv('ResEarnCampaignShops')
resp = SaoGetMResEarnCampaignShopsResponse(tbl)
return resp.make()
async def handle_d606(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mres_earn_campaign_shop_items
req = SaoGetMResEarnCampaignShopItemsRequest(header, request)
tbl = self.load_data_csv('ResEarnCampaignShopItems')
resp = SaoGetMResEarnCampaignShopItemsResponse(tbl)
return resp.make()
async def handle_d608(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mpaying_yui_medal_bonuses
tbl = self.load_data_csv('PayingYuiMedalBonuses')
resp = SaoGetMPayingYuiMedalBonusesResponse(tbl)
return resp.make()
async def handle_d60a(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mac_login_bonuses
tbl = self.load_data_csv('AcLoginBonuses')
resp = SaoGetMAcLoginBonusesResponse(tbl)
return resp.make()
async def handle_d60c(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mplay_campaigns
tbl = self.load_data_csv('PlayCampaigns')
resp = SaoGetMPlayCampaignsResponse(tbl)
return resp.make()
async def handle_d60e(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mplay_campaign_rewards
req = SaoGetMPlayCampaignRewardsRequest(header, request)
tbl = self.load_data_csv('PlayCampaignRewards')
resp = SaoGetMPlayCampaignRewardsResponse(tbl)
return resp.make()
async def handle_d610(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mgasha_free_campaigns
tbl = self.load_data_csv('GashaFreeCampaigns')
resp = SaoGetMGashaFreeCampaignsResponse(tbl)
return resp.make()
async def handle_d612(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mquest_drop_boost_campaigns
tbl = self.load_data_csv('QuestDropBoostCampaigns')
resp = SaoGetMQuestDropBoostCampaignsResponse(tbl)
return resp.make()
async def handle_d614(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mfirst_ticket_purchase_campaigns
tbl = self.load_data_csv('FirstTicketPurchaseCampaigns')
resp = SaoGetMFirstTicketPurchaseCampaignsResponse(tbl)
return resp.make()
async def handle_d616(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mlinked_site_reg_campaigns
tbl = self.load_data_csv('LinkedSiteRegCampaigns')
resp = SaoGetMLinkedSiteRegCampaignsResponse(tbl)
return resp.make()
async def handle_d618(self, header: SaoRequestHeader, request: bytes, src_ip: str) -> bytes:
# master_data2/get_mlinked_site_reg_campaign_rewards
req = SaoGetMLinkedSiteRegCampaignRewardsRequest(header, request)
tbl = self.load_data_csv('LinkedSiteRegCampaignRewards')
resp = SaoGetMLinkedSiteRegCampaignRewardsResponse(tbl)
return resp.make()