MESReport / utils /utils.py
ChenyuRabbitLove's picture
feat: add completion reward
115ff47
raw
history blame
8.72 kB
import os
import json
import logging
from datetime import datetime, timedelta, date
from typing import List
from dataclasses import asdict
import gradio as gr
from google.oauth2.service_account import Credentials
from google.cloud import bigquery
from utils.mes_player_model import Player
AVATAR_PATH = "avatar/"
AVATAR_FILE_TYPE = ".png"
MEDIA_PATH = "medias/"
MEDIA_FILE_TYPE = ".png"
SCOPES = ["https://www.googleapis.com/auth/bigquery"]
SERVICE_ACCOUNT_INFO = os.getenv("GBQ_TOKEN")
service_account_info_dict = json.loads(SERVICE_ACCOUNT_INFO)
creds = Credentials.from_service_account_info(service_account_info_dict, scopes=SCOPES)
client = bigquery.Client(
credentials=creds, project=service_account_info_dict["project_id"]
)
def get_content(file_name: str) -> str:
with open(file_name, "r", encoding="utf-8") as file:
content = file.read()
return content
def get_player_partners(player_info: gr.State) -> List[str]:
return [
f"{MEDIA_PATH}{partner}{MEDIA_FILE_TYPE}" for partner in player_info["partners"]
]
def get_player_badges(player_info: gr.State) -> List[str]:
return [f"{MEDIA_PATH}{badge}{MEDIA_FILE_TYPE}" for badge in player_info["badges"]]
def get_player_avatar(player_info: gr.State) -> str:
return f"{AVATAR_PATH}avatar_{player_info['player_group'] + 1}{AVATAR_FILE_TYPE}"
def get_player_adventure_logs(player_info: gr.State) -> List[str]:
log_template = """<div class="adventure"><p>{player_log}</p></div>"""
return [
log_template.format(player_log=player_log)
for player_log in player_info["adventure_logs"]
]
def get_player_adventure_logs_html(player_info: gr.State) -> str:
adventure_logs = "".join(get_player_adventure_logs(player_info))
template_content = get_content("htmls/adventure_template.html")
return template_content.replace("{logs}", adventure_logs)
def get_player_achievements(player_info: gr.State) -> List[str]:
achivement_name_map = {
"participation_star": "參賽之星",
"star_score_settler": "星際積分領航者",
"interstellar_traveler_I": "星際旅行者 I",
"interstellar_traveler_II": "星際旅行者 II",
"interstellar_traveler_III": "星際旅行者 III",
"interstellar_traveler_IV": "星際旅行者 IV",
"climbers_club_I": "爬升俱樂部 I",
"climbers_club_II": "爬升俱樂部 II",
"climbers_club_III": "爬升俱樂部 III",
"star_cluster_detector": "星團探測官",
"starry_vigilante": "群星瞭望者",
"planetary_decoder": "行星解碼",
"galactic_librarian": "星系圖書館員",
"energy_enthusiast_I": "能量狂熱者 I",
"energy_enthusiast_II": "能量狂熱者 II",
"energy_enthusiast_III": "能量狂熱者 III",
"energy_enthusiast_IV": "能量狂熱者 IV",
"knowledge_planet_explorer_I": "知識星球探險家 I",
"knowledge_planet_explorer_II": "知識星球探險家 II",
"scientific_expedition_explorer_I": "科學探險探險家 I",
"scientific_expedition_explorer_II": "科學探險探險家 II",
"cultural_celebration_explorer_I": "文化慶典探險家 I",
"cultural_celebration_explorer_II": "文化慶典探險家 II",
"youth_literature_explorer_I": "青春文學探險家 I",
"youth_literature_explorer_II": "青春文學探險家 II",
"path_to_wealth_explorer_I": "財富之路探險家 I",
"path_to_wealth_explorer_II": "財富之路探險家 II",
"cultivation_universe_explorer_I": "素養宇宙探險家 I",
"cultivation_universe_explorer_II": "素養宇宙探險家 II",
"electronic_and_information_college_explorer_I": "電資學院探險家 I",
"electronic_and_information_college_explorer_II": "電資學院探險家 II",
"star_warrior": "星空艦長",
}
if not isinstance(player_info["rewards_status"], dict):
rewards_status = json.loads(player_info["rewards_status"])
else:
rewards_status = player_info["rewards_status"]
if "routine_checker" in rewards_status:
del rewards_status["routine_checker"]
return [
(
achivement_name_map[achievement_key],
"完成" if achievement_value["is_completed"] else "未完成",
)
for achievement_key, achievement_value in rewards_status.items()
]
def get_current_story():
with open("story.json", "r", encoding="utf-8") as file:
story = json.load(file)
storyline_date = {
(datetime(2023, 12, 4).date(), datetime(2023, 12, 5).date()): 1,
(datetime(2023, 12, 6).date(), datetime(2023, 12, 7).date()): 2,
(datetime(2023, 12, 8).date(), datetime(2023, 12, 9).date()): 3,
(datetime(2023, 12, 10).date(), datetime(2023, 12, 11).date()): 4,
(datetime(2023, 12, 12).date(), datetime(2023, 12, 13).date()): 5,
(datetime(2023, 12, 14).date(), datetime(2023, 12, 15).date()): 6,
(datetime(2023, 12, 16).date(), datetime(2023, 12, 17).date()): 7,
(datetime(2023, 12, 18).date(), datetime(2023, 12, 19).date()): 8,
(datetime(2023, 12, 20).date(), datetime(2023, 12, 22).date()): 9,
(datetime(2023, 12, 23).date(), datetime(2023, 12, 25).date()): 10,
(datetime(2023, 12, 26).date(), datetime(2023, 12, 27).date()): 11,
(datetime(2023, 12, 28).date(), datetime(2023, 12, 29).date()): 12,
}
def get_stage(storyline_date):
current_date = datetime.now().date()
for (start_date, end_date), stage in storyline_date.items():
if start_date <= current_date <= end_date:
return stage
return None
stage = get_stage(storyline_date)
if stage:
return gr.Slider(
value=stage / 12 * 100,
show_label=False,
interactive=False,
info=story[str(stage)],
)
else:
return gr.Slider(
value=0,
show_label=False,
interactive=False,
info="狐貍貓與光束守護者的旅程將於 2023/12/04 開始!敬請期待!",
)
def query_bq_table(client, sql):
try:
query_job = client.query(sql)
query_job.result()
return query_job.to_dataframe()
except Exception as e:
logging.error(f"Query Failed: {e}")
raise
def load_player_statuses(client):
table_name = f"mes_report_20231229"
sql = f"SELECT * FROM `data_mart.{table_name}`"
return {
row["player_backend_user_id"]: Player.from_dict(row)
for _, row in query_bq_table(client, sql).iterrows()
}
def save_latest_player_data():
latest_player_data = load_player_statuses(client)
latest_player_data_as_dict = {
key: asdict(value) for key, value in latest_player_data.items()
}
def date_serializer(obj):
if isinstance(obj, date):
return obj.isoformat()
raise TypeError("Type not serializable")
with open("latest_player_data.json", "w") as fp:
print("Saving latest player data...")
json.dump(latest_player_data_as_dict, fp, default=date_serializer)
return "finished"
def render_player_data(player_info: gr.State):
player_avatar = get_player_avatar(player_info)
player_partners = get_player_partners(player_info)
player_badges = get_player_badges(player_info)
player_adventure_logs = get_player_adventure_logs_html(player_info)
player_achievements = get_player_achievements(player_info)
current_story = get_current_story()
return (
player_avatar,
player_partners,
player_badges,
player_adventure_logs,
player_achievements,
current_story,
)
def insert_data_into_bigquery(client, dataset_id, table_id, rows_to_insert):
# Specify the destination table
table_ref = client.dataset(dataset_id).table(table_id)
table = client.get_table(table_ref)
# Insert data into the table
errors = client.insert_rows(table, rows_to_insert)
# Check if any errors occurred during insertion
if errors:
logging.info("Errors occurred while inserting rows:")
for error in errors:
print(error)
else:
logging.info(f"Inserted {len(rows_to_insert)} rows successfully.")
def render_finished(player_activity, *args):
player_activity.render_finished(*args)
insert_row = player_activity.to_dict()
insert_data_into_bigquery(
client, "streaming_log", "log_mes_player_login_activity", [insert_row]
)
logging.info(
f"Player {insert_row['player_backend_user_id']} rendered successfully."
)