2021-09-30 06:40:41 +00:00
|
|
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
2019-01-24 08:43:03 +00:00
|
|
|
// See the LICENCE file in the repository root for full licence text.
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
using System;
|
2021-04-17 15:47:13 +00:00
|
|
|
using System.IO;
|
2021-10-04 08:00:22 +00:00
|
|
|
using System.Linq;
|
2021-09-30 06:40:41 +00:00
|
|
|
using JetBrains.Annotations;
|
|
|
|
using osu.Framework.Audio;
|
2018-04-13 09:19:50 +00:00
|
|
|
using osu.Framework.Audio.Track;
|
|
|
|
using osu.Framework.Graphics.Textures;
|
2021-09-30 06:40:41 +00:00
|
|
|
using osu.Framework.IO.Stores;
|
|
|
|
using osu.Framework.Lists;
|
2018-04-13 09:19:50 +00:00
|
|
|
using osu.Framework.Logging;
|
2021-09-30 06:40:41 +00:00
|
|
|
using osu.Framework.Platform;
|
|
|
|
using osu.Framework.Statistics;
|
2020-10-16 05:39:02 +00:00
|
|
|
using osu.Framework.Testing;
|
2018-04-13 09:19:50 +00:00
|
|
|
using osu.Game.Beatmaps.Formats;
|
2021-12-14 05:19:43 +00:00
|
|
|
using osu.Game.Database;
|
2019-09-09 22:43:30 +00:00
|
|
|
using osu.Game.IO;
|
2018-04-13 09:19:50 +00:00
|
|
|
using osu.Game.Skinning;
|
|
|
|
using osu.Game.Storyboards;
|
|
|
|
|
|
|
|
namespace osu.Game.Beatmaps
|
|
|
|
{
|
2021-09-30 07:45:32 +00:00
|
|
|
public class WorkingBeatmapCache : IBeatmapResourceProvider, IWorkingBeatmapCache
|
2018-04-13 09:19:50 +00:00
|
|
|
{
|
2021-09-30 06:40:41 +00:00
|
|
|
private readonly WeakList<BeatmapManagerWorkingBeatmap> workingCache = new WeakList<BeatmapManagerWorkingBeatmap>();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// A default representation of a WorkingBeatmap to use when no beatmap is available.
|
|
|
|
/// </summary>
|
|
|
|
public readonly WorkingBeatmap DefaultBeatmap;
|
|
|
|
|
2021-09-30 07:45:32 +00:00
|
|
|
public BeatmapModelManager BeatmapManager { private get; set; }
|
2021-09-30 06:40:41 +00:00
|
|
|
|
|
|
|
private readonly AudioManager audioManager;
|
|
|
|
private readonly IResourceStore<byte[]> resources;
|
|
|
|
private readonly LargeTextureStore largeTextureStore;
|
|
|
|
private readonly ITrackStore trackStore;
|
|
|
|
private readonly IResourceStore<byte[]> files;
|
|
|
|
|
|
|
|
[CanBeNull]
|
|
|
|
private readonly GameHost host;
|
|
|
|
|
2021-11-09 08:27:07 +00:00
|
|
|
public WorkingBeatmapCache(ITrackStore trackStore, AudioManager audioManager, IResourceStore<byte[]> resources, IResourceStore<byte[]> files, WorkingBeatmap defaultBeatmap = null, GameHost host = null)
|
2021-09-30 06:40:41 +00:00
|
|
|
{
|
|
|
|
DefaultBeatmap = defaultBeatmap;
|
|
|
|
|
|
|
|
this.audioManager = audioManager;
|
|
|
|
this.resources = resources;
|
|
|
|
this.host = host;
|
|
|
|
this.files = files;
|
|
|
|
largeTextureStore = new LargeTextureStore(host?.CreateTextureLoaderStore(files));
|
2021-11-09 08:27:07 +00:00
|
|
|
this.trackStore = trackStore;
|
2021-09-30 06:40:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void Invalidate(BeatmapSetInfo info)
|
|
|
|
{
|
|
|
|
foreach (var b in info.Beatmaps)
|
|
|
|
Invalidate(b);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Invalidate(BeatmapInfo info)
|
|
|
|
{
|
|
|
|
lock (workingCache)
|
|
|
|
{
|
2021-11-24 03:49:57 +00:00
|
|
|
var working = workingCache.FirstOrDefault(w => info.Equals(w.BeatmapInfo));
|
2021-10-14 04:58:36 +00:00
|
|
|
|
2021-09-30 06:40:41 +00:00
|
|
|
if (working != null)
|
2021-10-14 04:58:36 +00:00
|
|
|
{
|
|
|
|
Logger.Log($"Invalidating working beatmap cache for {info}");
|
2021-09-30 06:40:41 +00:00
|
|
|
workingCache.Remove(working);
|
2021-10-14 04:58:36 +00:00
|
|
|
}
|
2021-09-30 06:40:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public virtual WorkingBeatmap GetWorkingBeatmap(BeatmapInfo beatmapInfo)
|
|
|
|
{
|
|
|
|
// if there are no files, presume the full beatmap info has not yet been fetched from the database.
|
|
|
|
if (beatmapInfo?.BeatmapSet?.Files.Count == 0)
|
|
|
|
{
|
|
|
|
int lookupId = beatmapInfo.ID;
|
|
|
|
beatmapInfo = BeatmapManager.QueryBeatmap(b => b.ID == lookupId);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (beatmapInfo?.BeatmapSet == null)
|
|
|
|
return DefaultBeatmap;
|
|
|
|
|
|
|
|
lock (workingCache)
|
|
|
|
{
|
2021-11-24 03:49:57 +00:00
|
|
|
var working = workingCache.FirstOrDefault(w => beatmapInfo.Equals(w.BeatmapInfo));
|
2021-10-14 04:58:36 +00:00
|
|
|
|
2021-09-30 06:40:41 +00:00
|
|
|
if (working != null)
|
|
|
|
return working;
|
|
|
|
|
|
|
|
beatmapInfo.Metadata ??= beatmapInfo.BeatmapSet.Metadata;
|
|
|
|
|
|
|
|
workingCache.Add(working = new BeatmapManagerWorkingBeatmap(beatmapInfo, this));
|
|
|
|
|
|
|
|
// best effort; may be higher than expected.
|
|
|
|
GlobalStatistics.Get<int>(nameof(Beatmaps), $"Cached {nameof(WorkingBeatmap)}s").Value = workingCache.Count();
|
|
|
|
|
|
|
|
return working;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#region IResourceStorageProvider
|
|
|
|
|
|
|
|
TextureStore IBeatmapResourceProvider.LargeTextureStore => largeTextureStore;
|
|
|
|
ITrackStore IBeatmapResourceProvider.Tracks => trackStore;
|
|
|
|
AudioManager IStorageResourceProvider.AudioManager => audioManager;
|
2021-12-14 05:19:43 +00:00
|
|
|
RealmContextFactory IStorageResourceProvider.RealmContextFactory => null;
|
2021-09-30 06:40:41 +00:00
|
|
|
IResourceStore<byte[]> IStorageResourceProvider.Files => files;
|
|
|
|
IResourceStore<byte[]> IStorageResourceProvider.Resources => resources;
|
|
|
|
IResourceStore<TextureUpload> IStorageResourceProvider.CreateTextureLoaderStore(IResourceStore<byte[]> underlyingStore) => host?.CreateTextureLoaderStore(underlyingStore);
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
2020-10-16 05:39:02 +00:00
|
|
|
[ExcludeFromDynamicCompile]
|
2020-08-11 04:48:57 +00:00
|
|
|
private class BeatmapManagerWorkingBeatmap : WorkingBeatmap
|
2018-04-13 09:19:50 +00:00
|
|
|
{
|
2020-12-22 03:06:10 +00:00
|
|
|
[NotNull]
|
2020-12-21 05:06:50 +00:00
|
|
|
private readonly IBeatmapResourceProvider resources;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
2020-12-22 03:06:10 +00:00
|
|
|
public BeatmapManagerWorkingBeatmap(BeatmapInfo beatmapInfo, [NotNull] IBeatmapResourceProvider resources)
|
|
|
|
: base(beatmapInfo, resources.AudioManager)
|
2018-04-13 09:19:50 +00:00
|
|
|
{
|
2020-12-21 05:06:50 +00:00
|
|
|
this.resources = resources;
|
2018-04-13 09:19:50 +00:00
|
|
|
}
|
|
|
|
|
2018-04-19 11:44:38 +00:00
|
|
|
protected override IBeatmap GetBeatmap()
|
2018-04-13 09:19:50 +00:00
|
|
|
{
|
2020-08-24 10:38:05 +00:00
|
|
|
if (BeatmapInfo.Path == null)
|
2020-09-04 04:13:53 +00:00
|
|
|
return new Beatmap { BeatmapInfo = BeatmapInfo };
|
2020-08-24 10:38:05 +00:00
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
try
|
|
|
|
{
|
2021-04-17 15:49:10 +00:00
|
|
|
using (var stream = new LineBufferedReader(GetStream(BeatmapSetInfo.GetPathForFile(BeatmapInfo.Path))))
|
2018-04-13 09:19:50 +00:00
|
|
|
return Decoder.GetDecoder<Beatmap>(stream).Decode(stream);
|
|
|
|
}
|
2020-02-10 08:25:11 +00:00
|
|
|
catch (Exception e)
|
2018-04-13 09:19:50 +00:00
|
|
|
{
|
2020-02-10 08:25:11 +00:00
|
|
|
Logger.Error(e, "Beatmap failed to load");
|
2018-04-13 09:19:50 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override Texture GetBackground()
|
|
|
|
{
|
2021-11-04 05:11:21 +00:00
|
|
|
if (string.IsNullOrEmpty(Metadata?.BackgroundFile))
|
2018-04-13 09:19:50 +00:00
|
|
|
return null;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2021-04-17 15:49:10 +00:00
|
|
|
return resources.LargeTextureStore.Get(BeatmapSetInfo.GetPathForFile(Metadata.BackgroundFile));
|
2018-04-13 09:19:50 +00:00
|
|
|
}
|
2020-02-10 08:25:11 +00:00
|
|
|
catch (Exception e)
|
2018-04-13 09:19:50 +00:00
|
|
|
{
|
2020-02-10 08:25:11 +00:00
|
|
|
Logger.Error(e, "Background failed to load");
|
2018-04-13 09:19:50 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
2019-08-30 20:19:34 +00:00
|
|
|
|
2020-08-07 13:31:41 +00:00
|
|
|
protected override Track GetBeatmapTrack()
|
2018-04-13 09:19:50 +00:00
|
|
|
{
|
2021-11-04 05:01:01 +00:00
|
|
|
if (string.IsNullOrEmpty(Metadata?.AudioFile))
|
2020-09-01 06:48:13 +00:00
|
|
|
return null;
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
try
|
|
|
|
{
|
2021-04-17 15:49:10 +00:00
|
|
|
return resources.Tracks.Get(BeatmapSetInfo.GetPathForFile(Metadata.AudioFile));
|
2018-04-13 09:19:50 +00:00
|
|
|
}
|
2020-02-10 08:25:11 +00:00
|
|
|
catch (Exception e)
|
2018-04-13 09:19:50 +00:00
|
|
|
{
|
2020-02-10 08:25:11 +00:00
|
|
|
Logger.Error(e, "Track failed to load");
|
2018-06-27 07:02:49 +00:00
|
|
|
return null;
|
2018-04-13 09:19:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-27 07:07:18 +00:00
|
|
|
protected override Waveform GetWaveform()
|
|
|
|
{
|
2021-11-04 05:01:01 +00:00
|
|
|
if (string.IsNullOrEmpty(Metadata?.AudioFile))
|
2021-12-22 10:14:18 +00:00
|
|
|
return null;
|
2020-09-01 06:48:13 +00:00
|
|
|
|
2018-06-27 07:07:18 +00:00
|
|
|
try
|
|
|
|
{
|
2021-04-17 15:49:10 +00:00
|
|
|
var trackData = GetStream(BeatmapSetInfo.GetPathForFile(Metadata.AudioFile));
|
2021-12-22 10:14:18 +00:00
|
|
|
return trackData == null ? null : new Waveform(trackData);
|
2018-06-27 07:07:18 +00:00
|
|
|
}
|
2020-02-10 08:25:11 +00:00
|
|
|
catch (Exception e)
|
2018-06-27 07:07:18 +00:00
|
|
|
{
|
2020-02-10 08:25:11 +00:00
|
|
|
Logger.Error(e, "Waveform failed to load");
|
2021-12-22 10:14:18 +00:00
|
|
|
return null;
|
2018-06-27 07:07:18 +00:00
|
|
|
}
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
protected override Storyboard GetStoryboard()
|
|
|
|
{
|
|
|
|
Storyboard storyboard;
|
2019-04-01 03:16:05 +00:00
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
try
|
|
|
|
{
|
2021-04-17 15:49:10 +00:00
|
|
|
using (var stream = new LineBufferedReader(GetStream(BeatmapSetInfo.GetPathForFile(BeatmapInfo.Path))))
|
2018-04-13 09:19:50 +00:00
|
|
|
{
|
|
|
|
var decoder = Decoder.GetDecoder<Storyboard>(stream);
|
|
|
|
|
2021-10-27 04:04:41 +00:00
|
|
|
string storyboardFilename = BeatmapSetInfo?.Files.FirstOrDefault(f => f.Filename.EndsWith(".osb", StringComparison.OrdinalIgnoreCase))?.Filename;
|
2021-10-04 07:50:29 +00:00
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
// todo: support loading from both set-wide storyboard *and* beatmap specific.
|
2021-10-04 07:50:29 +00:00
|
|
|
if (string.IsNullOrEmpty(storyboardFilename))
|
2018-04-13 09:19:50 +00:00
|
|
|
storyboard = decoder.Decode(stream);
|
|
|
|
else
|
|
|
|
{
|
2021-10-04 07:50:29 +00:00
|
|
|
using (var secondaryStream = new LineBufferedReader(GetStream(BeatmapSetInfo.GetPathForFile(storyboardFilename))))
|
2018-04-13 09:19:50 +00:00
|
|
|
storyboard = decoder.Decode(stream, secondaryStream);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (Exception e)
|
|
|
|
{
|
|
|
|
Logger.Error(e, "Storyboard failed to load");
|
|
|
|
storyboard = new Storyboard();
|
|
|
|
}
|
|
|
|
|
|
|
|
storyboard.BeatmapInfo = BeatmapInfo;
|
|
|
|
|
|
|
|
return storyboard;
|
|
|
|
}
|
|
|
|
|
2021-08-15 16:38:01 +00:00
|
|
|
protected internal override ISkin GetSkin()
|
2018-04-13 09:19:50 +00:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2020-12-21 06:14:32 +00:00
|
|
|
return new LegacyBeatmapSkin(BeatmapInfo, resources.Files, resources);
|
2018-04-13 09:19:50 +00:00
|
|
|
}
|
|
|
|
catch (Exception e)
|
|
|
|
{
|
|
|
|
Logger.Error(e, "Skin failed to load");
|
2019-08-26 05:25:35 +00:00
|
|
|
return null;
|
2018-04-13 09:19:50 +00:00
|
|
|
}
|
|
|
|
}
|
2021-04-17 15:47:13 +00:00
|
|
|
|
|
|
|
public override Stream GetStream(string storagePath) => resources.Files.GetStream(storagePath);
|
2018-04-13 09:19:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|