osu/osu.Game.Tournament/IPC/FileBasedIPC.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

268 lines
9.7 KiB
C#
Raw Normal View History

2019-03-04 04:24:19 +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.
2018-11-07 16:23:00 +00:00
using System;
using System.IO;
using System.Linq;
using Microsoft.Win32;
2018-11-07 16:23:00 +00:00
using osu.Framework.Allocation;
using osu.Framework.Extensions.ObjectExtensions;
2018-11-10 15:45:48 +00:00
using osu.Framework.Logging;
using osu.Framework.Platform;
using osu.Framework.Threading;
2018-11-07 16:23:00 +00:00
using osu.Game.Beatmaps.Legacy;
using osu.Game.Online.API;
using osu.Game.Online.API.Requests;
using osu.Game.Online.API.Requests.Responses;
2018-11-07 16:23:00 +00:00
using osu.Game.Rulesets;
2019-06-18 05:51:48 +00:00
using osu.Game.Tournament.Models;
2018-11-07 16:23:00 +00:00
namespace osu.Game.Tournament.IPC
{
2018-11-15 12:28:42 +00:00
public partial class FileBasedIPC : MatchIPCInfo
2018-11-07 16:23:00 +00:00
{
2023-07-25 11:50:55 +00:00
public Storage? IPCStorage { get; private set; }
2018-11-07 16:23:00 +00:00
[Resolved]
2023-07-25 11:50:55 +00:00
protected IAPIProvider API { get; private set; } = null!;
2018-11-07 16:23:00 +00:00
[Resolved]
2023-07-25 11:50:55 +00:00
protected IRulesetStore Rulesets { get; private set; } = null!;
2018-11-07 16:23:00 +00:00
[Resolved]
2023-07-25 11:50:55 +00:00
private GameHost host { get; set; } = null!;
[Resolved]
2023-07-25 11:50:55 +00:00
private LadderInfo ladder { get; set; } = null!;
[Resolved]
2023-07-25 11:50:55 +00:00
private StableInfo stableInfo { get; set; } = null!;
private int lastBeatmapId;
2023-07-25 11:50:55 +00:00
private ScheduledDelegate? scheduled;
private GetBeatmapRequest? beatmapLookupRequest;
2018-11-07 16:23:00 +00:00
[BackgroundDependencyLoader]
private void load()
{
2023-07-25 11:50:55 +00:00
string? stablePath = stableInfo.StablePath ?? findStablePath();
initialiseIPCStorage(stablePath);
}
2023-07-25 11:50:55 +00:00
private Storage? initialiseIPCStorage(string? path)
2018-11-07 16:23:00 +00:00
{
scheduled?.Cancel();
IPCStorage = null;
try
{
if (string.IsNullOrEmpty(path))
return null;
IPCStorage = new DesktopStorage(path, host as DesktopGameHost);
2018-11-07 16:23:00 +00:00
const string file_ipc_filename = "ipc.txt";
const string file_ipc_state_filename = "ipc-state.txt";
const string file_ipc_scores_filename = "ipc-scores.txt";
const string file_ipc_channel_filename = "ipc-channel.txt";
2018-11-07 16:23:00 +00:00
if (IPCStorage.Exists(file_ipc_filename))
2019-11-11 11:53:22 +00:00
{
scheduled = Scheduler.AddDelayed(delegate
2018-11-07 16:23:00 +00:00
{
try
2018-11-07 16:23:00 +00:00
{
using (var stream = IPCStorage.GetStream(file_ipc_filename))
using (var sr = new StreamReader(stream))
2018-11-07 16:23:00 +00:00
{
int beatmapId = int.Parse(sr.ReadLine().AsNonNull());
int mods = int.Parse(sr.ReadLine().AsNonNull());
if (lastBeatmapId != beatmapId)
{
beatmapLookupRequest?.Cancel();
lastBeatmapId = beatmapId;
2023-07-25 11:50:55 +00:00
var existing = ladder.CurrentMatch.Value?.Round.Value?.Beatmaps.FirstOrDefault(b => b.ID == beatmapId);
2023-08-09 10:11:51 +00:00
if (existing != null)
Beatmap.Value = existing.Beatmap;
else
{
beatmapLookupRequest = new GetBeatmapRequest(new APIBeatmap { OnlineID = beatmapId });
beatmapLookupRequest.Success += b =>
{
if (lastBeatmapId == beatmapId)
Beatmap.Value = new TournamentBeatmap(b);
};
beatmapLookupRequest.Failure += _ =>
{
if (lastBeatmapId == beatmapId)
Beatmap.Value = null;
};
API.Queue(beatmapLookupRequest);
}
}
2018-11-07 16:23:00 +00:00
Mods.Value = (LegacyMods)mods;
}
}
catch
{
// file might be in use.
2018-11-07 16:23:00 +00:00
}
2018-11-10 15:45:48 +00:00
try
2018-11-17 12:26:56 +00:00
{
using (var stream = IPCStorage.GetStream(file_ipc_channel_filename))
using (var sr = new StreamReader(stream))
{
2023-07-25 11:50:55 +00:00
ChatChannel.Value = sr.ReadLine().AsNonNull();
}
}
catch (Exception)
{
// file might be in use.
2018-11-17 12:26:56 +00:00
}
try
2018-11-10 15:45:48 +00:00
{
using (var stream = IPCStorage.GetStream(file_ipc_state_filename))
using (var sr = new StreamReader(stream))
{
2022-12-26 19:36:39 +00:00
State.Value = Enum.Parse<TourneyState>(sr.ReadLine().AsNonNull());
}
}
catch (Exception)
{
// file might be in use.
2018-11-10 15:45:48 +00:00
}
try
2018-11-10 15:45:48 +00:00
{
using (var stream = IPCStorage.GetStream(file_ipc_scores_filename))
using (var sr = new StreamReader(stream))
{
2023-07-25 11:50:55 +00:00
Score1.Value = int.Parse(sr.ReadLine().AsNonNull());
Score2.Value = int.Parse(sr.ReadLine().AsNonNull());
}
2018-11-10 15:45:48 +00:00
}
catch (Exception)
{
// file might be in use.
}
}, 250, true);
2019-11-11 11:53:22 +00:00
}
}
catch (Exception e)
{
Logger.Error(e, "Stable installation could not be found; disabling file based IPC");
}
return IPCStorage;
2018-11-07 16:23:00 +00:00
}
2020-06-13 14:20:59 +00:00
/// <summary>
/// Manually sets the path to the directory used for inter-process communication with a cutting-edge install.
/// </summary>
/// <param name="path">Path to the IPC directory</param>
/// <returns>Whether the supplied path was a valid IPC directory.</returns>
2023-07-25 11:50:55 +00:00
public bool SetIPCLocation(string? path)
{
if (path == null || !ipcFileExistsInDirectory(path))
return false;
var newStorage = initialiseIPCStorage(stableInfo.StablePath = path);
if (newStorage == null)
return false;
stableInfo.SaveChanges();
return true;
}
2020-06-13 14:20:59 +00:00
/// <summary>
/// Tries to automatically detect the path to the directory used for inter-process communication
/// with a cutting-edge install.
/// </summary>
/// <returns>Whether an IPC directory was successfully auto-detected.</returns>
public bool AutoDetectIPCLocation() => SetIPCLocation(findStablePath());
private static bool ipcFileExistsInDirectory(string? p) => p != null && File.Exists(Path.Combine(p, "ipc.txt"));
2023-07-25 11:50:55 +00:00
private string? findStablePath()
2018-11-07 16:23:00 +00:00
{
2023-07-25 11:50:55 +00:00
string? stableInstallPath = findFromEnvVar() ??
findFromRegistry() ??
findFromLocalAppData() ??
findFromDotFolder();
2020-06-22 07:32:27 +00:00
Logger.Log($"Stable path for tourney usage: {stableInstallPath}");
return stableInstallPath;
}
2018-11-07 16:23:00 +00:00
2023-07-25 11:50:55 +00:00
private string? findFromEnvVar()
{
try
{
Logger.Log("Trying to find stable with environment variables");
2023-07-25 11:50:55 +00:00
string? stableInstallPath = Environment.GetEnvironmentVariable("OSU_STABLE_PATH");
2020-06-03 21:49:06 +00:00
if (ipcFileExistsInDirectory(stableInstallPath))
return stableInstallPath!;
}
catch
{
}
return null;
}
2023-07-25 11:50:55 +00:00
private string? findFromLocalAppData()
{
Logger.Log("Trying to find stable in %LOCALAPPDATA%");
string stableInstallPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData), @"osu!");
2020-06-03 21:49:06 +00:00
if (ipcFileExistsInDirectory(stableInstallPath))
return stableInstallPath;
return null;
}
2023-07-25 11:50:55 +00:00
private string? findFromDotFolder()
{
Logger.Log("Trying to find stable in dotfolders");
string stableInstallPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), ".osu");
2020-06-03 21:49:06 +00:00
if (ipcFileExistsInDirectory(stableInstallPath))
return stableInstallPath;
return null;
}
2023-07-25 11:50:55 +00:00
private string? findFromRegistry()
{
Logger.Log("Trying to find stable in registry");
try
{
2023-07-25 11:50:55 +00:00
string? stableInstallPath;
#pragma warning disable CA1416
2023-07-25 11:50:55 +00:00
using (RegistryKey? key = Registry.ClassesRoot.OpenSubKey("osu"))
2020-12-01 21:39:04 +00:00
stableInstallPath = key?.OpenSubKey(@"shell\open\command")?.GetValue(string.Empty)?.ToString()?.Split('"')[1].Replace("osu!.exe", "");
#pragma warning restore CA1416
if (ipcFileExistsInDirectory(stableInstallPath))
return stableInstallPath;
}
catch
{
}
return null;
2018-11-07 16:23:00 +00:00
}
}
}