-
Notifications
You must be signed in to change notification settings - Fork 390
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add user season score calculation workflow #11768
base: master
Are you sure you want to change the base?
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
mostly commenting on the db structure, I didn't review all the code too closely
app/Models/SeasonScoreFactor.php
Outdated
|
||
/** | ||
* @property int $id | ||
* @property float $factor |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
also want to question whether the concept of "factors" here is important to the score calculation because it seems a little complicated to me (as a player) and I can't imagine what would cause you to tune the factors individually. would something more straightforward like how total pp is weighted not work?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
factors should really be consistent across the seasons these days, though used to be changed in the past (e.g. i believe 4-playlist format we had before used a different set than what we use now), so this is mostly for backwards compatibility i think
app/Models/UserSeasonScore.php
Outdated
$factors = $this->season->scoreFactorsOrderedForCalculation(); | ||
$parentRooms = $this->season->rooms->where('parent_id', null); | ||
|
||
if ($parentRooms->count() > count($factors)) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
seems like these have to be equal?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
having too many factors shouldn't be an issue when they are ordered in descending order for calculation, it is handy to just set them all per-season without having to update with every new playlist launch 🤔
database/migrations/2024_10_22_190453_create_user_season_score_aggregates_table.php
Outdated
Show resolved
Hide resolved
'room_id' => $room, | ||
]); | ||
|
||
$this->roomAddPlay($this->user, $playlistItem, [ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
more like static::roomAddPlay
but either works I guess 🤷
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
UserScoreAggregateTest used $this->roomAddPlay()
so just mimicked it here, but static probably makes more sense indeed 🤔
Schema::create('user_season_score_aggregates', function (Blueprint $table) { | ||
$table->bigInteger('user_id')->unsigned(); | ||
$table->integer('season_id')->unsigned(); | ||
$table->float('total_score'); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this explodes on values as small as 7M
SQLSTATE[22003]: Numeric value out of range: 1264 Out of range value for column 'total_score' at row 1 (Connection: mysql, SQL: insert into `user_season_score_aggregates` (`user_id`, `season_id`, `total_score`) values (189617, 40, 7643867))
It needs to be either actual float or double (float
here is double(8,2)
). Or maybe rounded to int instead. Maybe @peppy can weigh in.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is that a laravel thing which maps float
to double(8,2)
?
i'm fine with any suitable data type. i don't think this table should get big enough for the choice to matter in terms of infrastructure concerns.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yeah afaict up to laravel 10 it's not possible to use this syntax to create actual float column
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
went ahead and just changed it to regular double
Part of #8736.
Serving as a base for implementation of seasonal rankings in nearing feature.
This mimics the Beatmap Spotlights seasonal rankings calculation process that used to be done via an external script.
Also adds an artisan command for score recalculating.
Tried to optimize the calculation process as much as possible, any further potential improvements on that matter would be vastly appreciated. 🤔
How does this work?
Rooms
The seasons are designed as a series of few (currently: three) playlists that are usually labeled by consecutive letters of alphabet (A, B, C).
Each of the playlists is ran twice over a course of the season. For the sake of this implementation, the rooms of the first run of the playlists are referred to as "parent rooms", while the rooms of the second run are called "children rooms".
This PR adds
group_indicator
to theseason_rooms
table, to group parent and children rooms together.Score calculation
Total (user) season score is sum of best playlist total scores achieved throughout the season, with added factor-based weighting.
Total scores of a each playlist are compared between the parent and child rooms, with higher being taken for total season score calculation.
Said scores are sorted from highest to lowest, and season-specified factors are being applied in descending order to each of them (e.g. 1.0, 0.75, 0.5). This results in final total season score.
This PR adds the following:
user_season_score_aggregates
table,total_score
of which is calculated using related modelcalculate()
function, that is called whenever user places a score on related season's room.score_factors
field onseasons
table that stores factors used for total season score calculation.Example user total season score calculation
Let's say a user achieves following scores during the six playlists of the season:
The season's score factors are set to following: 1, 0.75, 0.5.
Playlist final scores are being sorted in descending order, resulting in following values: 40, 20, 12. The final season score calculation goes as below:
The total season score is 61. That value would be used for sake of placing user in seasonal leaderboard.