osu/osu.Game/Online/API/APIAccess.cs

441 lines
14 KiB
C#
Raw Normal View History

// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.
2018-04-13 09:19:50 +00:00
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Net;
using System.Net.Http;
2018-04-13 09:19:50 +00:00
using System.Threading;
using System.Threading.Tasks;
using Newtonsoft.Json.Linq;
2019-02-21 10:04:31 +00:00
using osu.Framework.Bindables;
using osu.Framework.Extensions.ExceptionExtensions;
using osu.Framework.Extensions.ObjectExtensions;
2018-04-13 09:19:50 +00:00
using osu.Framework.Graphics;
using osu.Framework.Logging;
using osu.Game.Configuration;
using osu.Game.Online.API.Requests;
using osu.Game.Users;
namespace osu.Game.Online.API
{
public class APIAccess : Component, IAPIProvider
{
private readonly OsuConfigManager config;
2020-02-14 13:27:21 +00:00
2018-04-13 09:19:50 +00:00
private readonly OAuth authentication;
public string Endpoint => @"https://osu.ppy.sh";
2018-04-13 09:19:50 +00:00
private const string client_id = @"5";
private const string client_secret = @"FGc9GAtyHzeQDshWP5Ah7dega8hJACAJpQtw6OXk";
private readonly Queue<APIRequest> queue = new Queue<APIRequest>();
2018-04-13 09:19:50 +00:00
/// <summary>
/// The username/email provided by the user when initiating a login.
/// </summary>
public string ProvidedUsername { get; private set; }
private string password;
public Bindable<User> LocalUser { get; } = new Bindable<User>(createGuestUser());
public Bindable<UserActivity> Activity { get; } = new Bindable<UserActivity>();
protected bool HasLogin => authentication.Token.Value != null || (!string.IsNullOrEmpty(ProvidedUsername) && !string.IsNullOrEmpty(password));
2018-04-13 09:19:50 +00:00
private readonly CancellationTokenSource cancellationToken = new CancellationTokenSource();
private readonly Logger log;
public APIAccess(OsuConfigManager config)
{
this.config = config;
authentication = new OAuth(client_id, client_secret, Endpoint);
log = Logger.GetLogger(LoggingTarget.Network);
ProvidedUsername = config.Get<string>(OsuSetting.Username);
authentication.TokenString = config.Get<string>(OsuSetting.Token);
authentication.Token.ValueChanged += onTokenChanged;
LocalUser.BindValueChanged(u =>
{
u.OldValue?.Activity.UnbindFrom(Activity);
u.NewValue.Activity.BindTo(Activity);
}, true);
var thread = new Thread(run)
{
Name = "APIAccess",
IsBackground = true
};
thread.Start();
2018-04-13 09:19:50 +00:00
}
2019-02-21 09:56:34 +00:00
private void onTokenChanged(ValueChangedEvent<OAuthToken> e) => config.Set(OsuSetting.Token, config.Get<bool>(OsuSetting.SavePassword) ? authentication.TokenString : string.Empty);
2018-04-13 09:19:50 +00:00
private readonly List<IOnlineComponent> components = new List<IOnlineComponent>();
internal new void Schedule(Action action) => base.Schedule(action);
/// <summary>
/// Register a component to receive API events.
2019-03-08 03:44:33 +00:00
/// Fires <see cref="IOnlineComponent.APIStateChanged"/> once immediately to ensure a correct state.
/// </summary>
/// <param name="component"></param>
2018-04-13 09:19:50 +00:00
public void Register(IOnlineComponent component)
{
2019-05-09 04:33:18 +00:00
Schedule(() => components.Add(component));
component.APIStateChanged(this, state);
2018-04-13 09:19:50 +00:00
}
public void Unregister(IOnlineComponent component)
{
2019-05-09 04:33:18 +00:00
Schedule(() => components.Remove(component));
2018-04-13 09:19:50 +00:00
}
public string AccessToken => authentication.RequestAccessToken();
/// <summary>
/// Number of consecutive requests which failed due to network issues.
/// </summary>
private int failureCount;
private void run()
{
while (!cancellationToken.IsCancellationRequested)
{
switch (State)
{
case APIState.Failing:
//todo: replace this with a ping request.
log.Add(@"In a failing state, waiting a bit before we try again...");
Thread.Sleep(5000);
if (!IsLoggedIn) goto case APIState.Connecting;
2018-04-13 09:19:50 +00:00
if (queue.Count == 0)
{
log.Add(@"Queueing a ping request");
Queue(new GetUserRequest());
2018-04-13 09:19:50 +00:00
}
2018-04-13 09:19:50 +00:00
break;
2019-04-01 03:16:05 +00:00
2018-04-13 09:19:50 +00:00
case APIState.Offline:
case APIState.Connecting:
2020-05-05 01:31:11 +00:00
// work to restore a connection...
2018-04-13 09:19:50 +00:00
if (!HasLogin)
{
State = APIState.Offline;
Thread.Sleep(50);
continue;
}
State = APIState.Connecting;
// save the username at this point, if the user requested for it to be.
config.Set(OsuSetting.Username, config.Get<bool>(OsuSetting.SaveUsername) ? ProvidedUsername : string.Empty);
if (!authentication.HasValidAccessToken && !authentication.AuthenticateWithLogin(ProvidedUsername, password))
{
//todo: this fails even on network-related issues. we should probably handle those differently.
//NotificationOverlay.ShowMessage("Login failed!");
log.Add(@"Login failed!");
password = null;
authentication.Clear();
continue;
}
var userReq = new GetUserRequest();
userReq.Success += u =>
{
LocalUser.Value = u;
2019-12-18 05:07:03 +00:00
// todo: save/pull from settings
LocalUser.Value.Status.Value = new UserStatusOnline();
2018-04-13 09:19:50 +00:00
failureCount = 0;
//we're connected!
State = APIState.Online;
};
if (!handleRequest(userReq))
{
2018-12-26 07:06:39 +00:00
if (State == APIState.Connecting)
State = APIState.Failing;
2018-04-13 09:19:50 +00:00
continue;
}
// 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
// before actually going online.
2018-06-26 11:13:44 +00:00
while (State > APIState.Offline && State < APIState.Online)
2018-04-13 09:19:50 +00:00
Thread.Sleep(500);
break;
}
2020-05-05 01:31:11 +00:00
// hard bail if we can't get a valid access token.
2018-04-13 09:19:50 +00:00
if (authentication.RequestAccessToken() == null)
{
2018-12-22 08:54:19 +00:00
Logout();
2018-04-13 09:19:50 +00:00
continue;
}
while (true)
{
APIRequest req;
lock (queue)
{
if (queue.Count == 0) break;
2019-02-28 04:31:40 +00:00
req = queue.Dequeue();
}
handleRequest(req);
2018-04-13 09:19:50 +00:00
}
Thread.Sleep(50);
2018-04-13 09:19:50 +00:00
}
}
public void Perform(APIRequest request)
{
try
{
request.Perform(this);
}
catch (Exception e)
{
// todo: fix exception handling
request.Fail(e);
}
}
public Task PerformAsync(APIRequest request) =>
Task.Factory.StartNew(() => Perform(request), TaskCreationOptions.LongRunning);
2018-04-13 09:19:50 +00:00
public void Login(string username, string password)
{
Debug.Assert(State == APIState.Offline);
ProvidedUsername = username;
this.password = password;
}
public RegistrationRequest.RegistrationRequestErrors CreateAccount(string email, string username, string password)
{
Debug.Assert(State == APIState.Offline);
2018-12-05 04:08:35 +00:00
var req = new RegistrationRequest
{
Url = $@"{Endpoint}/users",
Method = HttpMethod.Post,
Username = username,
Email = email,
Password = password
};
try
{
req.Perform();
}
catch (Exception e)
{
try
{
return JObject.Parse(req.GetResponseString()).SelectToken("form_error", true).AsNonNull().ToObject<RegistrationRequest.RegistrationRequestErrors>();
}
catch
{
// if we couldn't deserialize the error message let's throw the original exception outwards.
e.Rethrow();
}
}
2018-12-05 04:08:35 +00:00
return null;
}
2018-04-13 09:19:50 +00:00
/// <summary>
/// Handle a single API request.
/// Ensures all exceptions are caught and dealt with correctly.
2018-04-13 09:19:50 +00:00
/// </summary>
/// <param name="req">The request.</param>
/// <returns>true if the request succeeded.</returns>
2018-04-13 09:19:50 +00:00
private bool handleRequest(APIRequest req)
{
try
{
req.Perform(this);
2020-05-05 01:31:11 +00:00
// we could still be in initialisation, at which point we don't want to say we're Online yet.
if (IsLoggedIn) State = APIState.Online;
2018-04-13 09:19:50 +00:00
failureCount = 0;
return true;
}
catch (WebException we)
{
handleWebException(we);
return false;
2018-04-13 09:19:50 +00:00
}
catch (Exception ex)
2018-04-13 09:19:50 +00:00
{
Logger.Error(ex, "Error occurred while handling an API request.");
return false;
2018-04-13 09:19:50 +00:00
}
}
private APIState state;
2018-04-13 09:19:50 +00:00
public APIState State
{
get => state;
2018-04-13 09:19:50 +00:00
private set
{
if (state == value)
return;
2018-04-13 09:19:50 +00:00
APIState oldState = state;
2018-04-13 09:19:50 +00:00
state = value;
log.Add($@"We just went {state}!");
2019-05-09 04:33:18 +00:00
Schedule(() =>
2018-04-13 09:19:50 +00:00
{
components.ForEach(c => c.APIStateChanged(this, state));
OnStateChange?.Invoke(oldState, state);
});
2018-04-13 09:19:50 +00:00
}
}
private bool handleWebException(WebException we)
{
HttpStatusCode statusCode = (we.Response as HttpWebResponse)?.StatusCode
?? (we.Status == WebExceptionStatus.UnknownError ? HttpStatusCode.NotAcceptable : HttpStatusCode.RequestTimeout);
// special cases for un-typed but useful message responses.
switch (we.Message)
{
case "Unauthorized":
case "Forbidden":
statusCode = HttpStatusCode.Unauthorized;
break;
}
switch (statusCode)
{
case HttpStatusCode.Unauthorized:
2018-12-22 08:54:19 +00:00
Logout();
return true;
2019-04-01 03:16:05 +00:00
case HttpStatusCode.RequestTimeout:
failureCount++;
log.Add($@"API failure count is now {failureCount}");
if (failureCount < 3)
2020-05-05 01:31:11 +00:00
// we might try again at an api level.
return false;
if (State == APIState.Online)
{
State = APIState.Failing;
flushQueue();
}
return true;
}
return true;
}
2018-04-13 09:19:50 +00:00
public bool IsLoggedIn => LocalUser.Value.Id > 1;
public void Queue(APIRequest request)
{
lock (queue) queue.Enqueue(request);
}
2018-04-13 09:19:50 +00:00
public event StateChangeDelegate OnStateChange;
public delegate void StateChangeDelegate(APIState oldState, APIState newState);
private void flushQueue(bool failOldRequests = true)
{
lock (queue)
{
var oldQueueRequests = queue.ToArray();
2018-04-13 09:19:50 +00:00
queue.Clear();
2018-04-13 09:19:50 +00:00
if (failOldRequests)
{
foreach (var req in oldQueueRequests)
req.Fail(new WebException(@"Disconnected from server"));
}
2018-04-13 09:19:50 +00:00
}
}
2018-12-22 08:54:19 +00:00
public void Logout()
2018-04-13 09:19:50 +00:00
{
flushQueue();
2018-04-13 09:19:50 +00:00
password = null;
authentication.Clear();
2019-05-09 04:42:04 +00:00
// Scheduled prior to state change such that the state changed event is invoked with the correct user present
Schedule(() => LocalUser.Value = createGuestUser());
2019-05-09 04:42:04 +00:00
State = APIState.Offline;
2018-04-13 09:19:50 +00:00
}
private static User createGuestUser() => new GuestUser();
2018-04-13 09:19:50 +00:00
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
flushQueue();
cancellationToken.Cancel();
}
}
internal class GuestUser : User
{
public GuestUser()
{
Username = @"Guest";
Id = 1;
}
}
2018-04-13 09:19:50 +00:00
public enum APIState
{
/// <summary>
/// We cannot login (not enough credentials).
/// </summary>
Offline,
/// <summary>
/// We are having connectivity issues.
/// </summary>
Failing,
/// <summary>
/// We are in the process of (re-)connecting.
/// </summary>
Connecting,
/// <summary>
/// We are online.
/// </summary>
Online
}
}