Skip to content
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

Merged
merged 16 commits into from
Feb 28, 2025

Conversation

smoogipoo
Copy link
Contributor

@smoogipoo smoogipoo commented Feb 26, 2025

Fixes #31951
Fixes #16480

Three important changes are made here...

Refactoring results screens to use Tasks

On master, results screens are very closely tied to their API requests. Implementations are required to return an APIRequest 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:

var req = FetchScores(fetchScoresCallback);
if (req != null)
api.Queue(req);

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:

Schedule(() =>
{
PerformSuccessCallback(scoresCallback, allScores);
hideLoadingSpinners();
});

This abstraction turned out to be quite a mess that refactoring it to use Tasks seemed appropriate. Now, the FetchScores() methods both return a Task<ScoreInfo[]> which also indicates the completion, allowing removal of the extra lastFetchCompleted 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 beatmaps

Aside from querying scores appropriately, this screen also handles transforming from MultiplayerScore models to ScoreInfo. The former model contains a BeatmapId parameter that was previously unused.

Now, the screen will perform a second request to fetch corresponding beatmaps before performing the transformation. Of particular importance:

  • Local beatmaps are preferred if they exist.
  • Online beatmaps are allowed to return 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 through BeatmapInfo so that it can be attached to a ScoreInfo. BeatmapDifficultyCache does not understand this scenario and will not return the expected fallback value, as it expects BeatmapInfo 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:

  • The ongoing FA playlist to ensure scrolling/basic lookups are still working as expected.
  • This freestyle room to check that multiple beatmaps are displayed correctly on the screen.

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.
@bdach bdach self-requested a review February 28, 2025 08:23
Copy link
Collaborator

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

Comment on lines +73 to +76
// 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;
Copy link
Collaborator

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...

Comment on lines 223 to 236
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
};
Copy link
Collaborator

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?

Copy link
Contributor Author

@smoogipoo smoogipoo Feb 28, 2025

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.

Copy link
Collaborator

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

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.

@bdach
Copy link
Collaborator

bdach commented Feb 28, 2025

Operationally speaking, it should be fine to merge this immediately without looking at the other multiplayer PRs, right?

@smoogipoo
Copy link
Contributor Author

it should be fine to merge this immediately without looking at the other multiplayer PRs

Yep, this should be independent of all the others.

@bdach bdach merged commit c4e37a1 into ppy:master Feb 28, 2025
8 of 10 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
2 participants