1
0
mirror of synced 2024-12-11 05:55:58 +01:00
bemaniutils/bemani/frontend/ddr/endpoints.py

650 lines
20 KiB
Python

# vim: set fileencoding=utf-8
import re
from typing import Any, Dict, List, Optional
from flask import Blueprint, request, Response, url_for, abort
from bemani.common import ID, GameConstants
from bemani.data import Link, UserID
from bemani.frontend.app import loginrequired, jsonify, render_react
from bemani.frontend.ddr.ddr import DDRFrontend
from bemani.frontend.templates import templates_location
from bemani.frontend.static import static_location
from bemani.frontend.types import g
ddr_pages = Blueprint(
'ddr_pages',
__name__,
url_prefix='/ddr',
template_folder=templates_location,
static_folder=static_location,
)
@ddr_pages.route('/scores')
@loginrequired
def viewnetworkscores() -> Response:
# Only load the last 100 results for the initial fetch, so we can render faster
frontend = DDRFrontend(g.data, g.config, g.cache)
network_scores = frontend.get_network_scores(limit=100)
if len(network_scores['attempts']) > 10:
network_scores['attempts'] = frontend.round_to_ten(network_scores['attempts'])
return render_react(
'Global DDR Scores',
'ddr/scores.react.js',
{
'attempts': network_scores['attempts'],
'songs': frontend.get_all_songs(),
'players': network_scores['players'],
'versions': {version: name for (game, version, name) in frontend.all_games()},
'shownames': True,
'shownewrecords': False,
},
{
'refresh': url_for('ddr_pages.listnetworkscores'),
'player': url_for('ddr_pages.viewplayer', userid=-1),
'individual_score': url_for('ddr_pages.viewtopscores', musicid=-1),
},
)
@ddr_pages.route('/scores/list')
@jsonify
@loginrequired
def listnetworkscores() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
return frontend.get_network_scores()
@ddr_pages.route('/scores/<int:userid>')
@loginrequired
def viewscores(userid: UserID) -> Response:
frontend = DDRFrontend(g.data, g.config, g.cache)
info = frontend.get_latest_player_info([userid]).get(userid)
if info is None:
abort(404)
scores = frontend.get_scores(userid, limit=100)
if len(scores) > 10:
scores = frontend.round_to_ten(scores)
return render_react(
f'{info["name"]}\'s DDR Scores',
'ddr/scores.react.js',
{
'attempts': scores,
'songs': frontend.get_all_songs(),
'players': {},
'versions': {version: name for (game, version, name) in frontend.all_games()},
'shownames': False,
'shownewrecords': True,
},
{
'refresh': url_for('ddr_pages.listscores', userid=userid),
'player': url_for('ddr_pages.viewplayer', userid=-1),
'individual_score': url_for('ddr_pages.viewtopscores', musicid=-1),
},
)
@ddr_pages.route('/scores/<int:userid>/list')
@jsonify
@loginrequired
def listscores(userid: UserID) -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
return {
'attempts': frontend.get_scores(userid),
'players': {},
}
@ddr_pages.route('/records')
@loginrequired
def viewnetworkrecords() -> Response:
frontend = DDRFrontend(g.data, g.config, g.cache)
network_records = frontend.get_network_records()
return render_react(
'Global DDR Records',
'ddr/records.react.js',
{
'records': network_records['records'],
'songs': frontend.get_all_songs(),
'players': network_records['players'],
'versions': {version: name for (game, version, name) in frontend.all_games()},
'shownames': True,
'showpersonalsort': False,
'filterempty': False,
},
{
'refresh': url_for('ddr_pages.listnetworkrecords'),
'player': url_for('ddr_pages.viewplayer', userid=-1),
'individual_score': url_for('ddr_pages.viewtopscores', musicid=-1),
},
)
@ddr_pages.route('/records/list')
@jsonify
@loginrequired
def listnetworkrecords() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
return frontend.get_network_records()
@ddr_pages.route('/records/<int:userid>')
@loginrequired
def viewrecords(userid: UserID) -> Response:
frontend = DDRFrontend(g.data, g.config, g.cache)
info = frontend.get_latest_player_info([userid]).get(userid)
if info is None:
abort(404)
return render_react(
f'{info["name"]}\'s DDR Records',
'ddr/records.react.js',
{
'records': frontend.get_records(userid),
'songs': frontend.get_all_songs(),
'players': {},
'versions': {version: name for (game, version, name) in frontend.all_games()},
'shownames': False,
'showpersonalsort': True,
'filterempty': True,
},
{
'refresh': url_for('ddr_pages.listrecords', userid=userid),
'player': url_for('ddr_pages.viewplayer', userid=-1),
'individual_score': url_for('ddr_pages.viewtopscores', musicid=-1),
},
)
@ddr_pages.route('/records/<int:userid>/list')
@jsonify
@loginrequired
def listrecords(userid: UserID) -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
return {
'records': frontend.get_records(userid),
'players': {},
}
@ddr_pages.route('/topscores/<int:musicid>')
@loginrequired
def viewtopscores(musicid: int) -> Response:
# We just want to find the latest mix that this song exists in
frontend = DDRFrontend(g.data, g.config, g.cache)
name = None
artist = None
genre = None
difficulties: List[int] = [0] * 10
groove: List[Dict[str, int]] = [{}] * 10
for chart in frontend.valid_charts:
details = g.data.local.music.get_song(GameConstants.DDR, 0, musicid, chart)
if details is not None:
name = details.name
artist = details.artist
genre = details.genre
difficulties[chart] = details.data.get_int('difficulty', 13)
groove[chart] = details.data.get_dict('groove')
if name is None:
# Not a real song!
abort(404)
top_scores = frontend.get_top_scores(musicid)
return render_react(
f'Top DDR Scores for {artist} - {name}',
'ddr/topscores.react.js',
{
'name': name,
'artist': artist,
'genre': genre,
'difficulties': difficulties,
'groove': groove,
'players': top_scores['players'],
'topscores': top_scores['topscores'],
},
{
'refresh': url_for('ddr_pages.listtopscores', musicid=musicid),
'player': url_for('ddr_pages.viewplayer', userid=-1),
},
)
@ddr_pages.route('/topscores/<int:musicid>/list')
@jsonify
@loginrequired
def listtopscores(musicid: int) -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
return frontend.get_top_scores(musicid)
@ddr_pages.route('/players')
@loginrequired
def viewplayers() -> Response:
frontend = DDRFrontend(g.data, g.config, g.cache)
return render_react(
'All DDR Players',
'ddr/allplayers.react.js',
{
'players': frontend.get_all_players()
},
{
'refresh': url_for('ddr_pages.listplayers'),
'player': url_for('ddr_pages.viewplayer', userid=-1),
},
)
@ddr_pages.route('/players/list')
@jsonify
@loginrequired
def listplayers() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
return {
'players': frontend.get_all_players(),
}
@ddr_pages.route('/players/<int:userid>')
@loginrequired
def viewplayer(userid: UserID) -> Response:
frontend = DDRFrontend(g.data, g.config, g.cache)
info = frontend.get_all_player_info([userid])[userid]
if not info:
abort(404)
latest_version = sorted(info.keys(), reverse=True)[0]
return render_react(
f'{info[latest_version]["name"]}\'s DDR Profile',
'ddr/player.react.js',
{
'playerid': userid,
'own_profile': userid == g.userID,
'player': info,
'versions': {version: name for (game, version, name) in frontend.all_games()},
},
{
'refresh': url_for('ddr_pages.listplayer', userid=userid),
'records': url_for('ddr_pages.viewrecords', userid=userid),
'scores': url_for('ddr_pages.viewscores', userid=userid),
},
)
@ddr_pages.route('/players/<int:userid>/list')
@jsonify
@loginrequired
def listplayer(userid: UserID) -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
info = frontend.get_all_player_info([userid])[userid]
return {
'player': info,
}
@ddr_pages.route('/options')
@loginrequired
def viewsettings() -> Response:
frontend = DDRFrontend(g.data, g.config, g.cache)
userid = g.userID
info = frontend.get_all_player_info([userid])[userid]
if not info:
abort(404)
return render_react(
'DDR Game Settings',
'ddr/settings.react.js',
{
'player': info,
'versions': {version: name for (game, version, name) in frontend.all_games()},
},
{
'updatename': url_for('ddr_pages.updatename'),
'updateweight': url_for('ddr_pages.updateweight'),
'updateearlylate': url_for('ddr_pages.updateearlylate'),
'updatebackgroundcombo': url_for('ddr_pages.updatebackgroundcombo'),
'updatesettings': url_for('ddr_pages.updatesettings'),
},
)
@ddr_pages.route('/options/name/update', methods=['POST'])
@jsonify
@loginrequired
def updatename() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
version = int(request.get_json()['version'])
name = request.get_json()['name']
user = g.data.local.user.get_user(g.userID)
if user is None:
raise Exception('Unable to find user to update!')
# Grab profile and update name
profile = g.data.local.user.get_profile(GameConstants.DDR, version, user.id)
if profile is None:
raise Exception('Unable to find profile to update!')
if len(name) == 0 or len(name) > 8:
raise Exception('Invalid profile name!')
if re.match(r'^[-&$\\.\\?!A-Z0-9 ]*$', name) is None:
raise Exception('Invalid profile name!')
profile = frontend.update_name(profile, name)
g.data.local.user.put_profile(GameConstants.DDR, version, user.id, profile)
# Return that we updated
return {
'version': version,
'name': name,
}
@ddr_pages.route('/options/weight/update', methods=['POST'])
@jsonify
@loginrequired
def updateweight() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
version = int(request.get_json()['version'])
weight = int(float(request.get_json()['weight']) * 10)
enabled = request.get_json()['enabled']
user = g.data.local.user.get_user(g.userID)
if user is None:
raise Exception('Unable to find user to update!')
# Grab profile and update weight
profile = g.data.local.user.get_profile(GameConstants.DDR, version, user.id)
if profile is None:
raise Exception('Unable to find profile to update!')
if enabled:
if weight <= 0 or weight > 9999:
raise Exception('Invalid weight!')
profile = frontend.update_weight(profile, weight, enabled)
g.data.local.user.put_profile(GameConstants.DDR, version, user.id, profile)
# Return that we updated
return {
'version': version,
'weight': weight,
'enabled': enabled,
}
@ddr_pages.route('/options/earlylate/update', methods=['POST'])
@jsonify
@loginrequired
def updateearlylate() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
version = int(request.get_json()['version'])
value = request.get_json()['value']
user = g.data.local.user.get_user(g.userID)
if user is None:
raise Exception('Unable to find user to update!')
# Grab profile and update early/late indicator
profile = g.data.local.user.get_profile(GameConstants.DDR, version, user.id)
if profile is None:
raise Exception('Unable to find profile to update!')
profile = frontend.update_early_late(profile, value)
g.data.local.user.put_profile(GameConstants.DDR, version, user.id, profile)
# Return that we updated
return {
'version': version,
'value': value != 0,
}
@ddr_pages.route('/options/backgroundcombo/update', methods=['POST'])
@jsonify
@loginrequired
def updatebackgroundcombo() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
version = int(request.get_json()['version'])
value = request.get_json()['value']
user = g.data.local.user.get_user(g.userID)
if user is None:
raise Exception('Unable to find user to update!')
# Grab profile and update combo position
profile = g.data.local.user.get_profile(GameConstants.DDR, version, user.id)
if profile is None:
raise Exception('Unable to find profile to update!')
profile = frontend.update_background_combo(profile, value)
g.data.local.user.put_profile(GameConstants.DDR, version, user.id, profile)
# Return that we updated
return {
'version': version,
'value': value != 0,
}
@ddr_pages.route('/options/settings/update', methods=['POST'])
@jsonify
@loginrequired
def updatesettings() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
settings = request.get_json()['settings']
version = int(request.get_json()['version'])
user = g.data.local.user.get_user(g.userID)
if user is None:
raise Exception('Unable to find user to update!')
# Grab profile and settings dict that needs updating
profile = g.data.local.user.get_profile(GameConstants.DDR, version, user.id)
if profile is None:
raise Exception('Unable to find profile to update!')
profile = frontend.update_settings(profile, settings)
g.data.local.user.put_profile(GameConstants.DDR, version, user.id, profile)
# Return updated settings
info = frontend.get_all_player_info([user.id])[user.id]
return {
'player': info,
'version': version,
}
@ddr_pages.route('/rivals')
@loginrequired
def viewrivals() -> Response:
frontend = DDRFrontend(g.data, g.config, g.cache)
rivals, info = frontend.get_rivals(g.userID)
return render_react(
'DDR Rivals',
'ddr/rivals.react.js',
{
'userid': str(g.userID),
'rivals': rivals,
'max_active_rivals': frontend.max_active_rivals,
'players': info,
'versions': {version: name for (game, version, name) in frontend.all_games()},
},
{
'refresh': url_for('ddr_pages.listrivals'),
'search': url_for('ddr_pages.searchrivals'),
'player': url_for('ddr_pages.viewplayer', userid=-1),
'addrival': url_for('ddr_pages.addrival'),
'removerival': url_for('ddr_pages.removerival'),
'setactiverival': url_for('ddr_pages.setactiverival'),
'setinactiverival': url_for('ddr_pages.setinactiverival'),
},
)
@ddr_pages.route('/rivals/list')
@jsonify
@loginrequired
def listrivals() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
rivals, info = frontend.get_rivals(g.userID)
return {
'rivals': rivals,
'players': info,
}
@ddr_pages.route('/rivals/search', methods=['POST'])
@jsonify
@loginrequired
def searchrivals() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
version = int(request.get_json()['version'])
name = request.get_json()['term']
# Try to treat the term as an extid
extid = ID.parse_extid(name)
matches = set()
profiles = g.data.remote.user.get_all_profiles(GameConstants.DDR, version)
for (userid, profile) in profiles:
if profile.get_int('extid') == extid or profile.get_str('name').lower() == name.lower():
matches.add(userid)
info = frontend.get_all_player_info(list(matches), allow_remote=True)
return {
'results': info,
}
@ddr_pages.route('/rivals/add', methods=['POST'])
@jsonify
@loginrequired
def addrival() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
version = int(request.get_json()['version'])
other_userid = UserID(int(request.get_json()['userid']))
userid = g.userID
# Find a slot to put the rival into
occupied: List[Optional[Link]] = [None] * 10
for link in g.data.local.user.get_links(GameConstants.DDR, version, userid):
if link.type[:7] != 'friend_':
continue
pos = int(link.type[7:])
if pos >= 0 and pos < 10:
occupied[pos] = link
# Put rival in the first slot
newrivalpos = -1
for i in range(len(occupied)):
if occupied[i] is None:
newrivalpos = i
break
if newrivalpos == -1:
raise Exception('No room for another rival!')
# Add this rival link
profile = g.data.remote.user.get_profile(GameConstants.DDR, version, other_userid)
if profile is None:
raise Exception('Unable to find profile for rival!')
g.data.local.user.put_link(
GameConstants.DDR,
version,
userid,
f'friend_{newrivalpos}',
other_userid,
{},
)
# Now return updated rival info
rivals, info = frontend.get_rivals(userid)
return {
'rivals': rivals,
'players': info,
}
@ddr_pages.route('/rivals/remove', methods=['POST'])
@jsonify
@loginrequired
def removerival() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
version = int(request.get_json()['version'])
other_userid = UserID(int(request.get_json()['userid']))
position = int(request.get_json()['position'])
userid = g.userID
# Remove this rival link
g.data.local.user.destroy_link(
GameConstants.DDR,
version,
userid,
f'friend_{position}',
other_userid,
)
profile = g.data.local.user.get_profile(GameConstants.DDR, version, userid)
if profile is None:
raise Exception('Unable to find profile to update!')
profile = frontend.deactivate_rival(profile, position)
g.data.local.user.put_profile(GameConstants.DDR, version, userid, profile)
# Now return updated rival info
rivals, info = frontend.get_rivals(userid)
return {
'rivals': rivals,
'players': info,
}
@ddr_pages.route('/rivals/activate', methods=['POST'])
@jsonify
@loginrequired
def setactiverival() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
version = int(request.get_json()['version'])
position = int(request.get_json()['position'])
userid = g.userID
profile = g.data.local.user.get_profile(GameConstants.DDR, version, userid)
if profile is None:
raise Exception('Unable to find profile to update!')
profile = frontend.activate_rival(profile, position)
g.data.local.user.put_profile(GameConstants.DDR, version, userid, profile)
# Now return updated rival info
rivals, info = frontend.get_rivals(userid)
return {
'rivals': rivals,
'players': info,
}
@ddr_pages.route('/rivals/inactivate', methods=['POST'])
@jsonify
@loginrequired
def setinactiverival() -> Dict[str, Any]:
frontend = DDRFrontend(g.data, g.config, g.cache)
version = int(request.get_json()['version'])
position = int(request.get_json()['position'])
userid = g.userID
profile = g.data.local.user.get_profile(GameConstants.DDR, version, userid)
if profile is None:
raise Exception('Unable to find profile to update!')
profile = frontend.deactivate_rival(profile, position)
g.data.local.user.put_profile(GameConstants.DDR, version, userid, profile)
# Now return updated rival info
rivals, info = frontend.get_rivals(userid)
return {
'rivals': rivals,
'players': info,
}