osu/osu.Game/Database/UserLookupCache.cs

87 lines
3.0 KiB
C#
Raw Normal View History

2020-11-06 07:38:57 +00:00
// 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.
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using osu.Framework.Allocation;
using osu.Game.Online.API;
using osu.Game.Online.API.Requests;
using osu.Game.Users;
namespace osu.Game.Database
{
public class UserLookupCache : MemoryCachingComponent<int, User>
{
[Resolved]
private IAPIProvider api { get; set; }
2020-11-09 01:40:16 +00:00
public Task<User> GetUserAsync(int userId, CancellationToken token = default) => GetAsync(userId, token);
2020-11-06 07:38:57 +00:00
protected override async Task<User> ComputeValueAsync(int lookup, CancellationToken token = default)
=> await queryUser(lookup);
2020-11-16 11:52:51 +00:00
private readonly Queue<(int id, TaskCompletionSource<User>)> pendingUserTasks = new Queue<(int, TaskCompletionSource<User>)>();
private Task pendingRequestTask;
private readonly object taskAssignmentLock = new object();
2020-11-06 07:38:57 +00:00
private Task<User> queryUser(int userId)
2020-11-06 07:38:57 +00:00
{
lock (taskAssignmentLock)
{
var tcs = new TaskCompletionSource<User>();
2020-11-06 07:38:57 +00:00
// Add to the queue.
2020-11-16 11:52:51 +00:00
pendingUserTasks.Enqueue((userId, tcs));
2020-11-06 07:38:57 +00:00
// Create a request task if there's not already one.
if (pendingRequestTask == null)
createNewTask();
2020-11-06 07:38:57 +00:00
return tcs.Task;
2020-11-06 07:38:57 +00:00
}
}
private void performLookup()
2020-11-06 07:38:57 +00:00
{
var userTasks = new List<(int id, TaskCompletionSource<User> task)>();
// Grab at most 50 users from the queue.
2020-11-06 07:38:57 +00:00
lock (taskAssignmentLock)
{
while (pendingUserTasks.Count > 0 && userTasks.Count < 50)
2020-11-06 07:38:57 +00:00
{
2020-11-16 11:52:51 +00:00
(int id, TaskCompletionSource<User> task) next = pendingUserTasks.Dequeue();
// Perform a secondary check for existence, in case the user was queried in a previous batch.
if (CheckExists(next.id, out var existing))
next.task.SetResult(existing);
else
userTasks.Add(next);
2020-11-06 07:38:57 +00:00
}
}
// Query the users.
var request = new GetUsersRequest(userTasks.Select(t => t.id).ToArray());
2020-11-06 07:38:57 +00:00
// rather than queueing, we maintain our own single-threaded request stream.
api.Perform(request);
// Create a new request task if there's still more users to query.
lock (taskAssignmentLock)
{
pendingRequestTask = null;
if (pendingUserTasks.Count > 0)
createNewTask();
2020-11-06 07:38:57 +00:00
}
// Notify of completion.
foreach (var (id, task) in userTasks)
task.SetResult(request.Result?.Users?.FirstOrDefault(u => u.Id == id));
2020-11-06 07:38:57 +00:00
}
private void createNewTask() => pendingRequestTask = Task.Run(performLookup);
2020-11-06 07:38:57 +00:00
}
}