-
-
Notifications
You must be signed in to change notification settings - Fork 2.4k
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
Fix multiplayer results screen displaying same beatmap for all users #32109
Conversation
This is a very special case where online beatmap/ruleset models are being ferried via `ScoreInfo` in what appear to `BeatmapDifficultyCache` as local `BeatmapInfo`/`RulesetInfo` models. Here, BDC will incorrectly attempt to proceed with calculating true difficulty where it cannot, and return 0. This is fixed locally because `ScoreInfo` is a very weird model, and I'm not sure whether BDC should contain logic to work around this.
- Handling all errors matches master a little bit better. Logging exceptions in any case. - Not throwing when beatmaps are missing simplifies tests.
Somewhat informal because it isn't super easy to handle.
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.
Remark, no action required: I can't understate how much I stand behind these changes, the coupling of the results screen to API requests is so terrible
// In some cases, the beatmap ferried through ScoreInfo actually represents an online beatmap. | ||
// If it isn't, we may be able to compute a more accurate difficulty from the ruleset and mods. | ||
if (realmAccess.Run(r => r.Find<BeatmapInfo>(score.BeatmapInfo!.ID)) != null) | ||
starDifficulty = beatmapDifficultyCache.GetDifficultyAsync(score.BeatmapInfo!, score.Ruleset, score.Mods).GetResultSafely() ?? starDifficulty; |
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.
Remark, no action required: I am wincing at this a lot because as far as I can tell this working predicates on GUID conflicts never happening out of thin air. But also I know how tightly everything in the results screen hierarchy is coupled to ScoreInfo
and have tried and failed to refactor everything so that it's not so I'll probably just bite the bullet and ignore this...
Difficulty = new BeatmapDifficulty(beatmap.Difficulty), | ||
Metadata = | ||
{ | ||
Author = new RealmUser | ||
{ | ||
Username = beatmap.Metadata.Author.Username, | ||
OnlineID = beatmap.Metadata.Author.OnlineID, | ||
} | ||
}, | ||
DifficultyName = beatmap.DifficultyName, | ||
StarRating = beatmap.StarRating, | ||
Length = beatmap.Length, | ||
BPM = beatmap.BPM | ||
}; |
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.
Suggestion / moderate severity concern: Think this may be missing a few? At least
diff --git a/osu.Game/Screens/OnlinePlay/Playlists/PlaylistItemResultsScreen.cs b/osu.Game/Screens/OnlinePlay/Playlists/PlaylistItemResultsScreen.cs
index 572bf535f7..184de2f50c 100644
--- a/osu.Game/Screens/OnlinePlay/Playlists/PlaylistItemResultsScreen.cs
+++ b/osu.Game/Screens/OnlinePlay/Playlists/PlaylistItemResultsScreen.cs
@@ -223,6 +223,8 @@ private async Task<ScoreInfo[]> transformScores(List<MultiplayerScore> scores)
Difficulty = new BeatmapDifficulty(beatmap.Difficulty),
Metadata =
{
+ Artist = beatmap.Metadata.Artist,
+ Title = beatmap.Metadata.Title,
Author = new RealmUser
{
Username = beatmap.Metadata.Author.Username,
The aforementioned patch would "fix" a particular white whale of mine in #16480. It's a much more direct fix than what I was going for (which was weaning off the entirety of the results monstrosity off of ScoreInfo
and then giving up seeing the scope of work required, at which point I decided to just belay that until the "inevitable" redesign of resutls), but it appears to work well enough?
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.
Applied in 993473c.
One thing I forgot to mention in the OP, is I originally started going down the path of adding copy constructors to all these models, similar to BeatmapDifficulty
except inlined instead. In theory it would catch this right out of the gate.
I didn't know how to feel about that from the outset/thought it would be heavily contested in review, so I didn't end up doing it and went with this local solution instead.
Curious on what your take would be.
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.
going down the path of adding copy constructors to all these models
You mean a copy constructor to BeatmapInfo
I guess? I dunno where I'd be with that.
We already have
osu/osu.Game/Beatmaps/BeatmapInfo.cs
Line 236 in 89b6d7c
public BeatmapInfo Clone() => (BeatmapInfo)this.Detach().MemberwiseClone(); |
which is half-baked; we already have #25467 which was pronounced too ugly to live; and in general I'm skeptical as to having cloning be a solution to any of this. The ideal way out for me would be to redo everything so that we can start using IScoreInfo
on these screens and skip all the weird games with model instances and collating data and so on, but as I said like twice here, I tried that and it was painful.
I dunno. I'd just get this in as is rather than go down that conversation, so I tentatively agree with the decision of keeping that out of this pull.
Operationally speaking, it should be fine to merge this immediately without looking at the other multiplayer PRs, right? |
Yep, this should be independent of all the others. |
Fixes #31951
Fixes #16480
Three important changes are made here...
Refactoring results screens to use
Task
sOn
master
, results screens are very closely tied to their API requests. Implementations are required to return anAPIRequest
to be handled by the base class, which doesn't work well when the playlists implementation needs to do a second lookup for the beatmaps:osu/osu.Game/Screens/Ranking/ResultsScreen.cs
Lines 240 to 243 in 38d807e
Furthermore, callbacks are passed around that are expected to be called in just the right way by implementations to actually add new scores to the list:
osu/osu.Game/Screens/OnlinePlay/Playlists/PlaylistItemResultsScreen.cs
Lines 116 to 120 in 38d807e
This abstraction turned out to be quite a mess that refactoring it to use
Task
s seemed appropriate. Now, theFetchScores()
methods both return aTask<ScoreInfo[]>
which also indicates the completion, allowing removal of the extralastFetchCompleted
state.Important things to look out for: The code for querying when scrolled to the left/right of the list has changed significantly. There should be no functional changes.
Making
PlaylistItemResultsScreen
query beatmapsAside from querying scores appropriately, this screen also handles transforming from
MultiplayerScore
models toScoreInfo
. The former model contains aBeatmapId
parameter that was previously unused.Now, the screen will perform a second request to fetch corresponding beatmaps before performing the transformation. Of particular importance:
null
, in which case the score will default to displaying the global beatmap while logging an error. This is done primarily to simplify testing flows, and I don't expect this to be hit in practice.Making score panel difficulty rating display not depend on local beatmap availability
Hopefully the explanation in 9029099 is sufficient, but the gist is that getting this to work means essentially ferrying
APIBeatmap
throughBeatmapInfo
so that it can be attached to aScoreInfo
.BeatmapDifficultyCache
does not understand this scenario and will not return the expected fallback value, as it expectsBeatmapInfo
to always represent a database-local model.I've experimented with making it understand this scenario in https://github.com/smoogipoo/osu/tree/fix-bdc-fallback, but I'm unsure of that path.
Testing
I've updated
TestScenePlaylistsResultsScreen
to show multiple beatmaps, but most of my testing has been on the live servers. In particular: