mirror of
https://github.com/ppy/osu.git
synced 2025-01-26 20:13:20 +08:00
Add checks for audio formats
This commit is contained in:
parent
c7b1524b9f
commit
9ef27104ce
96
osu.Game.Tests/Editing/Checks/CheckHitsoundsFormatTest.cs
Normal file
96
osu.Game.Tests/Editing/Checks/CheckHitsoundsFormatTest.cs
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
// 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.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using ManagedBass;
|
||||||
|
using Moq;
|
||||||
|
using NUnit.Framework;
|
||||||
|
using osu.Game.Beatmaps;
|
||||||
|
using osu.Game.Rulesets.Edit;
|
||||||
|
using osu.Game.Rulesets.Edit.Checks;
|
||||||
|
using osu.Game.Rulesets.Objects;
|
||||||
|
using osu.Game.Tests.Beatmaps;
|
||||||
|
using osu.Game.Tests.Resources;
|
||||||
|
using osuTK.Audio;
|
||||||
|
|
||||||
|
namespace osu.Game.Tests.Editing.Checks
|
||||||
|
{
|
||||||
|
[TestFixture]
|
||||||
|
public class CheckHitsoundsFormatTest
|
||||||
|
{
|
||||||
|
private CheckHitsoundsFormat check = null!;
|
||||||
|
|
||||||
|
private IBeatmap beatmap = null!;
|
||||||
|
|
||||||
|
[SetUp]
|
||||||
|
public void Setup()
|
||||||
|
{
|
||||||
|
check = new CheckHitsoundsFormat();
|
||||||
|
beatmap = new Beatmap<HitObject>
|
||||||
|
{
|
||||||
|
BeatmapInfo = new BeatmapInfo
|
||||||
|
{
|
||||||
|
BeatmapSet = new BeatmapSetInfo
|
||||||
|
{
|
||||||
|
Files = { CheckTestHelpers.CreateMockFile("wav") }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 0 = No output device. This still allows decoding.
|
||||||
|
if (!Bass.Init(0) && Bass.LastError != Errors.Already)
|
||||||
|
throw new AudioException("Could not initialize Bass.");
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestMP3Audio()
|
||||||
|
{
|
||||||
|
using (var resourceStream = TestResources.OpenResource("Samples/test-sample-cut.mp3"))
|
||||||
|
{
|
||||||
|
var issues = check.Run(getContext(resourceStream)).ToList();
|
||||||
|
Assert.That(issues, Has.Count.EqualTo(1));
|
||||||
|
Assert.That(issues.Single().Template is CheckHitsoundsFormat.IssueTemplateIncorrectFormat);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestOGGAudio()
|
||||||
|
{
|
||||||
|
using (var resourceStream = TestResources.OpenResource("Samples/test-sample.ogg"))
|
||||||
|
{
|
||||||
|
var issues = check.Run(getContext(resourceStream)).ToList();
|
||||||
|
Assert.That(issues, Has.Count.EqualTo(0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestWAVAudio()
|
||||||
|
{
|
||||||
|
using (var resourceStream = TestResources.OpenResource("Samples/hitsound-delay.wav"))
|
||||||
|
{
|
||||||
|
var issues = check.Run(getContext(resourceStream)).ToList();
|
||||||
|
Assert.That(issues, Has.Count.EqualTo(0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestWEBMAudio()
|
||||||
|
{
|
||||||
|
using (var resourceStream = TestResources.OpenResource("Samples/test-sample.webm"))
|
||||||
|
{
|
||||||
|
var issues = check.Run(getContext(resourceStream)).ToList();
|
||||||
|
Assert.That(issues, Has.Count.EqualTo(1));
|
||||||
|
Assert.That(issues.Single().Template is CheckHitsoundsFormat.IssueTemplateFormatUnsupported);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private BeatmapVerifierContext getContext(Stream? resourceStream)
|
||||||
|
{
|
||||||
|
var mockWorkingBeatmap = new Mock<TestWorkingBeatmap>(beatmap, null, null);
|
||||||
|
mockWorkingBeatmap.Setup(w => w.GetStream(It.IsAny<string>())).Returns(resourceStream);
|
||||||
|
|
||||||
|
return new BeatmapVerifierContext(beatmap, mockWorkingBeatmap.Object);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
100
osu.Game.Tests/Editing/Checks/CheckSongFormatTest.cs
Normal file
100
osu.Game.Tests/Editing/Checks/CheckSongFormatTest.cs
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
// 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.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using ManagedBass;
|
||||||
|
using Moq;
|
||||||
|
using NUnit.Framework;
|
||||||
|
using osu.Game.Beatmaps;
|
||||||
|
using osu.Game.Rulesets.Edit;
|
||||||
|
using osu.Game.Rulesets.Edit.Checks;
|
||||||
|
using osu.Game.Rulesets.Objects;
|
||||||
|
using osu.Game.Tests.Beatmaps;
|
||||||
|
using osu.Game.Tests.Resources;
|
||||||
|
using osuTK.Audio;
|
||||||
|
|
||||||
|
namespace osu.Game.Tests.Editing.Checks
|
||||||
|
{
|
||||||
|
[TestFixture]
|
||||||
|
public partial class CheckSongFormatTest
|
||||||
|
{
|
||||||
|
private CheckSongFormat check = null!;
|
||||||
|
|
||||||
|
private IBeatmap beatmap = null!;
|
||||||
|
|
||||||
|
[SetUp]
|
||||||
|
public void Setup()
|
||||||
|
{
|
||||||
|
check = new CheckSongFormat();
|
||||||
|
beatmap = new Beatmap<HitObject>
|
||||||
|
{
|
||||||
|
BeatmapInfo = new BeatmapInfo
|
||||||
|
{
|
||||||
|
BeatmapSet = new BeatmapSetInfo
|
||||||
|
{
|
||||||
|
Files = { CheckTestHelpers.CreateMockFile("mp3") }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 0 = No output device. This still allows decoding.
|
||||||
|
if (!Bass.Init(0) && Bass.LastError != Errors.Already)
|
||||||
|
throw new AudioException("Could not initialize Bass.");
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestMP3Audio()
|
||||||
|
{
|
||||||
|
using (var resourceStream = TestResources.OpenResource("Samples/test-sample-cut.mp3"))
|
||||||
|
{
|
||||||
|
beatmap.Metadata.AudioFile = "abc123.mp3";
|
||||||
|
var issues = check.Run(getContext(resourceStream)).ToList();
|
||||||
|
Assert.That(issues, Has.Count.EqualTo(0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestOGGAudio()
|
||||||
|
{
|
||||||
|
using (var resourceStream = TestResources.OpenResource("Samples/test-sample.ogg"))
|
||||||
|
{
|
||||||
|
beatmap.Metadata.AudioFile = "abc123.mp3";
|
||||||
|
var issues = check.Run(getContext(resourceStream)).ToList();
|
||||||
|
Assert.That(issues, Has.Count.EqualTo(0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestWAVAudio()
|
||||||
|
{
|
||||||
|
using (var resourceStream = TestResources.OpenResource("Samples/hitsound-delay.wav"))
|
||||||
|
{
|
||||||
|
beatmap.Metadata.AudioFile = "abc123.mp3";
|
||||||
|
var issues = check.Run(getContext(resourceStream)).ToList();
|
||||||
|
Assert.That(issues, Has.Count.EqualTo(1));
|
||||||
|
Assert.That(issues.Single().Template is CheckSongFormat.IssueTemplateIncorrectFormat);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestWEBMAudio()
|
||||||
|
{
|
||||||
|
using (var resourceStream = TestResources.OpenResource("Samples/test-sample.webm"))
|
||||||
|
{
|
||||||
|
beatmap.Metadata.AudioFile = "abc123.mp3";
|
||||||
|
var issues = check.Run(getContext(resourceStream)).ToList();
|
||||||
|
Assert.That(issues, Has.Count.EqualTo(1));
|
||||||
|
Assert.That(issues.Single().Template is CheckSongFormat.IssueTemplateFormatUnsupported);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private BeatmapVerifierContext getContext(Stream? resourceStream)
|
||||||
|
{
|
||||||
|
var mockWorkingBeatmap = new Mock<TestWorkingBeatmap>(beatmap, null, null);
|
||||||
|
mockWorkingBeatmap.Setup(w => w.GetStream(It.IsAny<string>())).Returns(resourceStream);
|
||||||
|
|
||||||
|
return new BeatmapVerifierContext(beatmap, mockWorkingBeatmap.Object);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -28,6 +28,8 @@ namespace osu.Game.Rulesets.Edit
|
|||||||
new CheckTooShortAudioFiles(),
|
new CheckTooShortAudioFiles(),
|
||||||
new CheckAudioInVideo(),
|
new CheckAudioInVideo(),
|
||||||
new CheckDelayedHitsounds(),
|
new CheckDelayedHitsounds(),
|
||||||
|
new CheckSongFormat(),
|
||||||
|
new CheckHitsoundsFormat(),
|
||||||
|
|
||||||
// Files
|
// Files
|
||||||
new CheckZeroByteFiles(),
|
new CheckZeroByteFiles(),
|
||||||
|
93
osu.Game/Rulesets/Edit/Checks/CheckHitsoundsFormat.cs
Normal file
93
osu.Game/Rulesets/Edit/Checks/CheckHitsoundsFormat.cs
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
// 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.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using ManagedBass;
|
||||||
|
using osu.Framework.Audio.Callbacks;
|
||||||
|
using osu.Game.Beatmaps;
|
||||||
|
using osu.Game.Extensions;
|
||||||
|
using osu.Game.Rulesets.Edit.Checks.Components;
|
||||||
|
|
||||||
|
namespace osu.Game.Rulesets.Edit.Checks
|
||||||
|
{
|
||||||
|
public class CheckHitsoundsFormat : ICheck
|
||||||
|
{
|
||||||
|
public CheckMetadata Metadata => new CheckMetadata(CheckCategory.Audio, "Checks for hitsound formats.");
|
||||||
|
|
||||||
|
public IEnumerable<IssueTemplate> PossibleTemplates => new IssueTemplate[]
|
||||||
|
{
|
||||||
|
new IssueTemplateFormatUnsupported(this),
|
||||||
|
new IssueTemplateIncorrectFormat(this),
|
||||||
|
};
|
||||||
|
|
||||||
|
private IEnumerable<ChannelType> allowedFormats => new ChannelType[]
|
||||||
|
{
|
||||||
|
ChannelType.WavePCM,
|
||||||
|
ChannelType.WaveFloat,
|
||||||
|
ChannelType.OGG,
|
||||||
|
ChannelType.Wave | ChannelType.OGG,
|
||||||
|
};
|
||||||
|
|
||||||
|
public IEnumerable<Issue> Run(BeatmapVerifierContext context)
|
||||||
|
{
|
||||||
|
var beatmapSet = context.Beatmap.BeatmapInfo.BeatmapSet;
|
||||||
|
var audioFile = beatmapSet?.GetFile(context.Beatmap.Metadata.AudioFile);
|
||||||
|
|
||||||
|
if (beatmapSet == null) yield break;
|
||||||
|
|
||||||
|
foreach (var file in beatmapSet.Files)
|
||||||
|
{
|
||||||
|
if (audioFile != null && file.File == audioFile.File) continue;
|
||||||
|
|
||||||
|
using (Stream data = context.WorkingBeatmap.GetStream(file.File.GetStoragePath()))
|
||||||
|
{
|
||||||
|
if (data == null)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
var fileCallbacks = new FileCallbacks(new DataStreamFileProcedures(data));
|
||||||
|
int decodeStream = Bass.CreateStream(StreamSystem.NoBuffer, BassFlags.Decode | BassFlags.Prescan, fileCallbacks.Callbacks, fileCallbacks.Handle);
|
||||||
|
|
||||||
|
// If the format is not supported by BASS
|
||||||
|
if (decodeStream == 0)
|
||||||
|
{
|
||||||
|
if (AudioCheckUtils.HasAudioExtension(file.Filename) && probablyHasAudioData(data))
|
||||||
|
yield return new IssueTemplateFormatUnsupported(this).Create(file.Filename);
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var audioInfo = Bass.ChannelGetInfo(decodeStream);
|
||||||
|
|
||||||
|
if (!allowedFormats.Contains(audioInfo.ChannelType))
|
||||||
|
{
|
||||||
|
yield return new IssueTemplateIncorrectFormat(this).Create(file.Filename, audioInfo.ChannelType.ToString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool probablyHasAudioData(Stream data) => data.Length > 100;
|
||||||
|
|
||||||
|
public class IssueTemplateFormatUnsupported : IssueTemplate
|
||||||
|
{
|
||||||
|
public IssueTemplateFormatUnsupported(ICheck check)
|
||||||
|
: base(check, IssueType.Problem, "\"{0}\" is using a unsupported format; Use wav or ogg for hitsounds.")
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public Issue Create(string file) => new Issue(this, file);
|
||||||
|
}
|
||||||
|
|
||||||
|
public class IssueTemplateIncorrectFormat : IssueTemplate
|
||||||
|
{
|
||||||
|
public IssueTemplateIncorrectFormat(ICheck check)
|
||||||
|
: base(check, IssueType.Problem, "\"{0}\" is using a incorrect format ({1}); Use wav or ogg for hitsounds.")
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public Issue Create(string file, string format) => new Issue(this, file, format);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
81
osu.Game/Rulesets/Edit/Checks/CheckSongFormat.cs
Normal file
81
osu.Game/Rulesets/Edit/Checks/CheckSongFormat.cs
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
// 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;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using ManagedBass;
|
||||||
|
using osu.Framework.Audio.Callbacks;
|
||||||
|
using osu.Game.Beatmaps;
|
||||||
|
using osu.Game.Extensions;
|
||||||
|
using osu.Game.Rulesets.Edit.Checks.Components;
|
||||||
|
|
||||||
|
namespace osu.Game.Rulesets.Edit.Checks
|
||||||
|
{
|
||||||
|
public class CheckSongFormat : ICheck
|
||||||
|
{
|
||||||
|
public CheckMetadata Metadata => new CheckMetadata(CheckCategory.Audio, "Checks for song formats.");
|
||||||
|
|
||||||
|
public IEnumerable<IssueTemplate> PossibleTemplates => new IssueTemplate[]
|
||||||
|
{
|
||||||
|
new IssueTemplateFormatUnsupported(this),
|
||||||
|
new IssueTemplateIncorrectFormat(this),
|
||||||
|
};
|
||||||
|
|
||||||
|
private IEnumerable<ChannelType> allowedFormats => new ChannelType[]
|
||||||
|
{
|
||||||
|
ChannelType.MP3,
|
||||||
|
ChannelType.OGG,
|
||||||
|
};
|
||||||
|
|
||||||
|
public IEnumerable<Issue> Run(BeatmapVerifierContext context)
|
||||||
|
{
|
||||||
|
var beatmapSet = context.Beatmap.BeatmapInfo.BeatmapSet;
|
||||||
|
var audioFile = beatmapSet?.GetFile(context.Beatmap.Metadata.AudioFile);
|
||||||
|
|
||||||
|
if (beatmapSet == null) yield break;
|
||||||
|
if (audioFile == null) yield break;
|
||||||
|
|
||||||
|
using (Stream data = context.WorkingBeatmap.GetStream(audioFile.File.GetStoragePath()))
|
||||||
|
{
|
||||||
|
if (data == null || data.Length <= 0) yield break;
|
||||||
|
|
||||||
|
var fileCallbacks = new FileCallbacks(new DataStreamFileProcedures(data));
|
||||||
|
int decodeStream = Bass.CreateStream(StreamSystem.NoBuffer, BassFlags.Decode | BassFlags.Prescan, fileCallbacks.Callbacks, fileCallbacks.Handle);
|
||||||
|
|
||||||
|
// If the format is not supported by BASS
|
||||||
|
if (decodeStream == 0)
|
||||||
|
{
|
||||||
|
yield return new IssueTemplateFormatUnsupported(this).Create(audioFile.Filename);
|
||||||
|
yield break;
|
||||||
|
}
|
||||||
|
|
||||||
|
var audioInfo = Bass.ChannelGetInfo(decodeStream);
|
||||||
|
|
||||||
|
if (!allowedFormats.Contains(audioInfo.ChannelType))
|
||||||
|
yield return new IssueTemplateIncorrectFormat(this).Create(audioFile.Filename, audioInfo.ChannelType.ToString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class IssueTemplateFormatUnsupported : IssueTemplate
|
||||||
|
{
|
||||||
|
public IssueTemplateFormatUnsupported(ICheck check)
|
||||||
|
: base(check, IssueType.Problem, "\"{0}\" is using a unsupported format; Use mp3 or ogg for the song's audio.")
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public Issue Create(string file) => new Issue(this, file);
|
||||||
|
}
|
||||||
|
|
||||||
|
public class IssueTemplateIncorrectFormat : IssueTemplate
|
||||||
|
{
|
||||||
|
public IssueTemplateIncorrectFormat(ICheck check)
|
||||||
|
: base(check, IssueType.Problem, "\"{0}\" is using a incorrect format ({1}); Use mp3 or ogg for the song's audio.")
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public Issue Create(string file, string format) => new Issue(this, file, format);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user