1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-06 07:02:54 +08:00

Refactor APIAccess main loop to read better

This commit is contained in:
Dean Herbert 2022-08-11 15:43:39 +09:00
parent 47196b19a5
commit 865d63f768

View File

@ -104,30 +104,23 @@ namespace osu.Game.Online.API
/// </summary> /// </summary>
private int failureCount; private int failureCount;
/// <summary>
/// The main API thread loop, which will continue to run until the game is shut down.
/// </summary>
private void run() private void run()
{ {
while (!cancellationToken.IsCancellationRequested) while (!cancellationToken.IsCancellationRequested)
{ {
switch (State.Value) if (state.Value == APIState.Failing)
{ {
case APIState.Failing: // To recover from a failing state, falling through and running the full reconnection process seems safest for now.
//todo: replace this with a ping request. // This could probably be replaced with a ping-style request if we want to avoid the reconnection overheads.
log.Add(@"In a failing state, waiting a bit before we try again..."); log.Add($@"{nameof(APIAccess)} is in a failing state, waiting a bit before we try again...");
Thread.Sleep(5000); Thread.Sleep(5000);
if (!IsLoggedIn) goto case APIState.Connecting;
if (queue.Count == 0)
{
log.Add(@"Queueing a ping request");
Queue(new GetUserRequest());
} }
break; // Ensure that we have valid credentials.
// If not, setting the offline state will allow the game to prompt the user to provide new credentials.
case APIState.Offline:
case APIState.Connecting:
// work to restore a connection...
if (!HasLogin) if (!HasLogin)
{ {
state.Value = APIState.Offline; state.Value = APIState.Offline;
@ -135,6 +128,61 @@ namespace osu.Game.Online.API
continue; continue;
} }
Debug.Assert(HasLogin);
// Ensure that we are in an online state. If not, attempt a connect.
if (state.Value != APIState.Online)
{
attemptConnect();
if (state.Value != APIState.Online)
continue;
}
// hard bail if we can't get a valid access token.
if (authentication.RequestAccessToken() == null)
{
Logout();
continue;
}
processQueuedRequests();
Thread.Sleep(50);
}
}
/// <summary>
/// Dequeue from the queue and run each request synchronously until the queue is empty.
/// </summary>
private void processQueuedRequests()
{
while (true)
{
APIRequest req;
lock (queue)
{
if (queue.Count == 0) return;
req = queue.Dequeue();
}
handleRequest(req);
}
}
/// <summary>
/// From a non-connected state, perform a full connection flow, obtaining OAuth tokens and populating the local user and friends.
/// </summary>
/// <remarks>
/// This method takes control of <see cref="state"/> and transitions from <see cref="APIState.Connecting"/> to either
/// - <see cref="APIState.Online"/> (successful connection)
/// - <see cref="APIState.Failing"/> (failed connection but retrying)
/// - <see cref="APIState.Offline"/> (failed and can't retry, clear credentials and require user interaction)
/// </remarks>
/// <returns>Whether the connection attempt was successful.</returns>
private void attemptConnect()
{
state.Value = APIState.Connecting; state.Value = APIState.Connecting;
if (localUser.IsDefault) if (localUser.IsDefault)
@ -163,21 +211,20 @@ namespace osu.Game.Online.API
{ {
//todo: this fails even on network-related issues. we should probably handle those differently. //todo: this fails even on network-related issues. we should probably handle those differently.
LastLoginError = e; LastLoginError = e;
log.Add(@"Login failed!"); log.Add($@"Login failed for username {ProvidedUsername} ({LastLoginError.Message})!");
password = null;
authentication.Clear(); Logout();
continue; return;
} }
} }
var userReq = new GetUserRequest(); var userReq = new GetUserRequest();
userReq.Failure += ex => userReq.Failure += ex =>
{ {
if (ex is APIException) if (ex is APIException)
{ {
LastLoginError = ex; LastLoginError = ex;
log.Add("Login failed on local user retrieval!"); log.Add($@"Login failed for username {ProvidedUsername} on user retrieval ({LastLoginError.Message})!");
Logout(); Logout();
} }
else if (ex is WebException webException && webException.Message == @"Unauthorized") else if (ex is WebException webException && webException.Message == @"Unauthorized")
@ -186,7 +233,9 @@ namespace osu.Game.Online.API
Logout(); Logout();
} }
else else
failConnectionProcess(); {
state.Value = APIState.Failing;
}
}; };
userReq.Success += user => userReq.Success += user =>
{ {
@ -202,61 +251,25 @@ namespace osu.Game.Online.API
if (!handleRequest(userReq)) if (!handleRequest(userReq))
{ {
failConnectionProcess(); state.Value = APIState.Failing;
continue; return;
} }
// getting user's friends is considered part of the connection process.
var friendsReq = new GetFriendsRequest(); var friendsReq = new GetFriendsRequest();
friendsReq.Failure += _ => state.Value = APIState.Failing;
friendsReq.Failure += _ => failConnectionProcess();
friendsReq.Success += res => friends.AddRange(res); friendsReq.Success += res => friends.AddRange(res);
if (!handleRequest(friendsReq)) if (!handleRequest(friendsReq))
{ {
failConnectionProcess(); state.Value = APIState.Failing;
continue; return;
} }
// The Success callback event is fired on the main thread, so we should wait for that to run before proceeding. // The Success callback event is fired on the main thread, so we should wait for that to run before proceeding.
// Without this, we will end up circulating this Connecting loop multiple times and queueing up many web requests // Without this, we will end up circulating this Connecting loop multiple times and queueing up many web requests
// before actually going online. // before actually going online.
while (State.Value > APIState.Offline && State.Value < APIState.Online) while (State.Value == APIState.Connecting && !cancellationToken.IsCancellationRequested)
Thread.Sleep(500); Thread.Sleep(500);
break;
}
// hard bail if we can't get a valid access token.
if (authentication.RequestAccessToken() == null)
{
Logout();
continue;
}
while (true)
{
APIRequest req;
lock (queue)
{
if (queue.Count == 0) break;
req = queue.Dequeue();
}
handleRequest(req);
}
Thread.Sleep(50);
}
void failConnectionProcess()
{
// if something went wrong during the connection process, we want to reset the state (but only if still connecting).
if (State.Value == APIState.Connecting)
state.Value = APIState.Failing;
}
} }
public void Perform(APIRequest request) public void Perform(APIRequest request)