This module contains:
- Catch2 matchers that operate on
juce::dsp::AudioBlock
and rely on... - ...a collection of C++ free functions that take an
dsp::dsp::AudioBlock
.
My projects rely on this module, but I wanted to publish it to share how I'm using Catch2 and JUCE.
The Catch2 matchers gives a solid amount of detail on AudioBlocks when a test fails. This includes displaying summary stats and sparkline for any AudioBlock:
REQUIRE_THAT(myAudioBlock, isEqualTo (someOtherBlock))
with expansion:
Block is 1 channel, 480 samples, min -0.766242, max 0.289615, 100% filled
[0—⎻—x—⎼⎽_⎽⎼—]
is equal to
Block is 1 channel, 480 samples, min -1, max 1, 100% filled
[0—⎻⎺‾⎺⎻—x—⎼⎽_⎽⎼—]
REQUIRE_THAT (myAudioBlock, isEqualTo (someOtherBlock));
This compares each channel of the two blocks and confirms each sample is within a tolerance of std::numeric_limits<float>::epsilon() * 100
. Don't @ me, but assuming you are doing things like multiplying a number of floats together to arrive at the end result, you probably don't want this tolerance any tighter.
However, you can loosen or tighten to your desire by passing an absolute tolerance:
REQUIRE_THAT (myAudioBlock, isEqualTo (someOtherBlock), 0.0005f));
REQUIRE_THAT (myAudioBlock, isFilled);
Passes when the block is completely filled. No more than 1 consecutive zero is allowed.
REQUIRE_THAT (myAudioBlock, isFilledUntil(256));
Passes when the block is filled (doesn't contain consecutive zeros) up to to and including sample 256.
REQUIRE_THAT (myAudioBlock, isFilledAfter(256));
Passes if, starting with this sample, the block is filled and doesn't contain consecutive zeros.
If the sample specified is zero (eg, start of a sine wave) but there are no consecutive zeros, it'll pass.
Fails if the block ends at the sample specified.
REQUIRE_THAT (myAudioBlock, isFilledBetween(64, 128));
Passes when sample values 64 and 128 have a non zero value for all channels.
REQUIRE_THAT (myAudioBlock, isEmpty);
The block only has 0s.
REQUIRE_THAT (myAudioBlock, isEmptyUntil(256));
Passes when the block has only zeros up to and including this sample number.
REQUIRE_THAT (myAudioBlock, isEmptyAfter(256));
Passes when the block only contains zeros after this sample number, or when the block ends at this point.
The matchers above call out to free functions test helpers (prepended with block
) which can be used seperately.
So for example you can do use the Catch2 matcher style:
REQUIRE_THAT (myAudioBlock, isFilledUntil(256));
Or the free function style:
REQUIRE (blockIsFilledUntil(myAudioBlock, 256);
Additionally, there are some other helpers:
Returns the peak absolute value (magnitude) from any channel in the block.
REQUIRE (maxMagnitude (myAudioBlock) <= Catch::Approx (1.0f));
REQUIRE (magnitudeOfFrequency (myAudioBlock, 440.f, sampleRate) < 0.005);
This one is my favorite.
Use this if you know the frequencies you expect to encounter in the AudioBlock. You'll need to pass it the sample rate you are using.
This uses frequency correlation and compares your AudioBlock against sine and cosine probes. It's essentially the same thing as what goes on under the hood with DFT, but for exactly the frequency you specify.
This is a much more accurate way to confirm the presence of a known frequency than FFT.
There are various FFT related functions available which rely on creating an instance of the FFT class.
Tip: Prefer the magnitudeOfFrequency
helper if you know the frequency you are expecting and wish to somewhat accurately confirm magnitude. FFT is inherently messy. You'll get better results when your expected frequencies are in the middle of FFT bins.
However, you can still sorta sloppily check for the strongest frequency:
REQUIRE (FFT (myAudioBlock, 44100.0f).strongestFrequencyIs (200.f));
auto fft = FFT (myAudioBlock, 44100.f);
fft.printDebug(); // prints the bins
REQUIRE (fft.strongestFrequencyBin() == fft.frequencyBinFor (220.f));
auto fft = FFT (myAudioBlock, 44100.f);
REQUIRE_THAT (fft.strongFrequencyBins(), Catch::Matchers::Contains (fft.frequencyBinFor (220.f)));
Prerequisites:
- Catch2
- The melatonin_audio_sparklines JUCE module. It provides Catch2 with fancy descriptions of AudioBlocks when tests fail.
Set up with a git submodule tracking the main
branch:
git submodule add -b main https://github.com/sudara/melatonin_audio_block_test_helpers modules/melatonin_audio_block_test_helpers
git commit -m "Added melatonin_audio_block_test_helpers submodule."
To update these test helpers, you can:
git submodule update --remote --merge modules/melatonin_audio_block_test_helpers
If you use CMake, you can inform JUCE about the module in your CMakeLists.txt
:
juce_add_module("modules/melatonin_audio_block_test_helpers")
Add the following to any .cpp you'd like to use the helpers:
#include "melatonin_audio_block_test_helpers/melatonin_audio_block_test_helpers.h"
// make your life easier while you are at it...
using namespace melatonin;
- The matchers currently require Catch2 v3, which is currently the devel branch in Catch2 (and has been for a year).
- This is just a random assortment of things I actually use for my tests.
AudioBlock
only (I don't useAudioBuffer
much, but would be willing to look at a PR!).