osu/osu.Game/Rulesets/RulesetStore.cs

182 lines
6.5 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.IO;
using System.Linq;
using System.Reflection;
using osu.Framework.Logging;
2018-04-13 09:19:50 +00:00
using osu.Game.Database;
namespace osu.Game.Rulesets
{
public class RulesetStore : DatabaseBackedStore, IDisposable
2018-04-13 09:19:50 +00:00
{
private const string ruleset_library_prefix = "osu.Game.Rulesets";
2018-04-13 09:19:50 +00:00
private readonly Dictionary<Assembly, Type> loadedAssemblies = new Dictionary<Assembly, Type>();
2018-04-13 09:19:50 +00:00
public RulesetStore(IDatabaseContextFactory factory)
: base(factory)
{
2019-07-03 09:36:04 +00:00
// On android in release configuration assemblies are loaded from the apk directly into memory.
2019-07-03 09:41:01 +00:00
// We cannot read assemblies from cwd, so should check loaded assemblies instead.
2019-07-03 09:42:10 +00:00
loadFromAppDomain();
loadFromDisk();
addMissingRulesets();
AppDomain.CurrentDomain.AssemblyResolve += resolveRulesetAssembly;
2018-04-13 09:19:50 +00:00
}
/// <summary>
/// Retrieve a ruleset using a known ID.
/// </summary>
/// <param name="id">The ruleset's internal ID.</param>
/// <returns>A ruleset, if available, else null.</returns>
public RulesetInfo GetRuleset(int id) => AvailableRulesets.FirstOrDefault(r => r.ID == id);
/// <summary>
/// Retrieve a ruleset using a known short name.
/// </summary>
/// <param name="shortName">The ruleset's short name.</param>
/// <returns>A ruleset, if available, else null.</returns>
public RulesetInfo GetRuleset(string shortName) => AvailableRulesets.FirstOrDefault(r => r.ShortName == shortName);
/// <summary>
/// All available rulesets.
/// </summary>
public IEnumerable<RulesetInfo> AvailableRulesets { get; private set; }
2018-04-13 09:19:50 +00:00
private Assembly resolveRulesetAssembly(object sender, ResolveEventArgs args) => loadedAssemblies.Keys.FirstOrDefault(a => a.FullName == args.Name);
2018-04-13 09:19:50 +00:00
private void addMissingRulesets()
2018-04-13 09:19:50 +00:00
{
using (var usage = ContextFactory.GetForWrite())
{
var context = usage.Context;
var instances = loadedAssemblies.Values.Select(r => (Ruleset)Activator.CreateInstance(r)).ToList();
2018-04-13 09:19:50 +00:00
//add all legacy rulesets first to ensure they have exclusive choice of primary key.
2019-12-24 04:48:27 +00:00
foreach (var r in instances.Where(r => r is ILegacyRuleset))
2018-04-13 09:19:50 +00:00
{
if (context.RulesetInfo.SingleOrDefault(dbRuleset => dbRuleset.ID == r.RulesetInfo.ID) == null)
2018-04-13 09:19:50 +00:00
context.RulesetInfo.Add(r.RulesetInfo);
}
context.SaveChanges();
//add any other modes
2019-12-24 04:48:27 +00:00
foreach (var r in instances.Where(r => !(r is ILegacyRuleset)))
2019-11-11 11:53:22 +00:00
{
2018-04-13 09:19:50 +00:00
if (context.RulesetInfo.FirstOrDefault(ri => ri.InstantiationInfo == r.RulesetInfo.InstantiationInfo) == null)
context.RulesetInfo.Add(r.RulesetInfo);
2019-11-11 11:53:22 +00:00
}
2018-04-13 09:19:50 +00:00
context.SaveChanges();
//perform a consistency check
foreach (var r in context.RulesetInfo)
{
try
{
var instanceInfo = ((Ruleset)Activator.CreateInstance(Type.GetType(r.InstantiationInfo, asm =>
{
// for the time being, let's ignore the version being loaded.
// this allows for debug builds to successfully load rulesets (even though debug rulesets have a 0.0.0 version).
asm.Version = null;
return Assembly.Load(asm);
}, null))).RulesetInfo;
2018-04-13 09:19:50 +00:00
r.Name = instanceInfo.Name;
r.ShortName = instanceInfo.ShortName;
r.InstantiationInfo = instanceInfo.InstantiationInfo;
2018-04-13 09:19:50 +00:00
r.Available = true;
}
catch
{
r.Available = false;
}
}
context.SaveChanges();
AvailableRulesets = context.RulesetInfo.Where(r => r.Available).ToList();
}
}
private void loadFromAppDomain()
{
2019-07-02 15:25:12 +00:00
foreach (var ruleset in AppDomain.CurrentDomain.GetAssemblies())
{
string rulesetName = ruleset.GetName().Name;
if (!rulesetName.StartsWith(ruleset_library_prefix, StringComparison.InvariantCultureIgnoreCase) || ruleset.GetName().Name.Contains("Tests"))
continue;
addRuleset(ruleset);
2019-07-02 15:25:12 +00:00
}
}
private void loadFromDisk()
2019-07-03 09:42:10 +00:00
{
try
{
string[] files = Directory.GetFiles(Environment.CurrentDirectory, $"{ruleset_library_prefix}.*.dll");
foreach (string file in files.Where(f => !Path.GetFileName(f).Contains("Tests")))
loadRulesetFromFile(file);
}
2019-10-01 06:41:01 +00:00
catch (Exception e)
2019-07-03 09:42:10 +00:00
{
Logger.Error(e, $"Could not load rulesets from directory {Environment.CurrentDirectory}");
2019-07-03 09:42:10 +00:00
}
}
private void loadRulesetFromFile(string file)
2018-04-13 09:19:50 +00:00
{
var filename = Path.GetFileNameWithoutExtension(file);
if (loadedAssemblies.Values.Any(t => t.Namespace == filename))
2018-04-13 09:19:50 +00:00
return;
try
{
addRuleset(Assembly.LoadFrom(file));
2018-04-13 09:19:50 +00:00
}
catch (Exception e)
2018-04-13 09:19:50 +00:00
{
Logger.Error(e, $"Failed to load ruleset {filename}");
2018-04-13 09:19:50 +00:00
}
}
private void addRuleset(Assembly assembly)
{
if (loadedAssemblies.ContainsKey(assembly))
return;
try
{
loadedAssemblies[assembly] = assembly.GetTypes().First(t => t.IsPublic && t.IsSubclassOf(typeof(Ruleset)));
}
catch (Exception e)
{
Logger.Error(e, $"Failed to add ruleset {assembly}");
}
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
protected virtual void Dispose(bool disposing)
{
AppDomain.CurrentDomain.AssemblyResolve -= resolveRulesetAssembly;
}
2018-04-13 09:19:50 +00:00
}
}