diff --git a/tests/audio/CMakeLists.txt b/tests/audio/CMakeLists.txt index 239bc41fd0..41dec1dcd2 100644 --- a/tests/audio/CMakeLists.txt +++ b/tests/audio/CMakeLists.txt @@ -1,7 +1,7 @@ # Declare dependencies macro (SETUP_TESTCASE_DEPENDENCIES) # link in the shared libraries - link_hifi_libraries(shared audio networking) + link_hifi_libraries(shared audio audio-client plugins networking) package_libraries_for_deployment() endmacro () diff --git a/tests/audio/src/AudioTests.cpp b/tests/audio/src/AudioTests.cpp new file mode 100644 index 0000000000..1e097855e8 --- /dev/null +++ b/tests/audio/src/AudioTests.cpp @@ -0,0 +1,87 @@ +#include +#include + +#include "AudioTests.h" +#include "AudioClient.h" +#include "DependencyManager.h" +#include "NodeList.h" +#include "plugins/CodecPlugin.h" +#include "plugins/PluginManager.h" + +QTEST_MAIN(AudioTests) + + +Q_DECLARE_METATYPE(QList) + + + + +void AudioTests::initTestCase() { + // AudioClient starts networking, but for the purposes of the tests here we don't care, + // so just got to use some port. + int listenPort = 10000; + + DependencyManager::registerInheritance(); + DependencyManager::set(NodeType::Agent, listenPort); + DependencyManager::set(); + DependencyManager::set(); + QSharedPointer ac = DependencyManager::get(); + + qRegisterMetaType>(); + + ac->startThread(); +} + +void AudioTests::listAudioDevices() { + QSharedPointer ac = DependencyManager::get(); + QVERIFY(!ac.isNull()); + +/* + // AudioClient::devicesChanged is declared as: + // void devicesChanged(QAudio::Mode mode, const QList& devices); + // + // Unfortunately with QSignalSpy we have to use the old SIGNAL() syntax, so it was a bit tricky + // to figure out how to get the signal to connect. The snippet below lists signals in the format + // that Qt understands. Turns out we lose the 'const &'. + + const QMetaObject *mo = ac->metaObject(); + QList signalSignatures; + + // Start from MyClass members + for(int methodIdx = mo->methodOffset(); methodIdx < mo->methodCount(); ++methodIdx) { + QMetaMethod mmTest = mo->method(methodIdx); + switch((int)mmTest.methodType()) { + case QMetaMethod::Signal: + signalSignatures.append(QString(mmTest.methodSignature())); + qDebug() << "SIG: " << QString(mmTest.methodSignature()); + break; + } + } +*/ + + QSignalSpy spy(ac.get(), SIGNAL(devicesChanged(QAudio::Mode,QList))); + + QVERIFY(spy.isValid()); // This checks that the signal has connected + spy.wait(15000); + + // We always get two events here, one for audio input, and one for output, + // but signals keep coming and we could potentially get more repetitions. + QVERIFY(spy.count() > 0); + qDebug() << "Received" << spy.count() << "device events"; + + // QSignalSpy is a QList, which stores the received signals. We can then examine it to see + // what we got. + for(auto event : spy) { + QAudio::Mode mode = qvariant_cast(event.at(0)); + QList devs = qvariant_cast>(event.at(1)); + + QVERIFY(devs.count() > 0); + + qDebug() << "Mode:" << mode; + for(auto dev : devs) { + qDebug() << "\t" << dev.deviceName(); + } + } + + +} diff --git a/tests/audio/src/AudioTests.h b/tests/audio/src/AudioTests.h new file mode 100644 index 0000000000..3dde1f04f2 --- /dev/null +++ b/tests/audio/src/AudioTests.h @@ -0,0 +1,27 @@ +// +// AudioTests.h +// tests/audio/src +// +// Created by Dale Glass on 27/8/2022 +// Copyright 2022 Overte e.V. +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#ifndef hifi_AudioTests_h +#define hifi_AudioTests_h + +#include + + + +class AudioTests : public QObject { + Q_OBJECT +private slots: + void initTestCase(); + + void listAudioDevices(); +}; + +#endif // hifi_AudioTests_h diff --git a/tests/audio/src/CodecTests.cpp b/tests/audio/src/CodecTests.cpp new file mode 100644 index 0000000000..072ee7fb9f --- /dev/null +++ b/tests/audio/src/CodecTests.cpp @@ -0,0 +1,125 @@ +#include +#include +#include +#include + + +#include "CodecTests.h" +#include "AudioClient.h" +#include "DependencyManager.h" +#include "NodeList.h" +#include "plugins/CodecPlugin.h" +#include "plugins/PluginManager.h" + +QTEST_MAIN(CodecTests) + + + + +void CodecTests::initTestCase() { + DependencyManager::set(); + + QDir testPath (QCoreApplication::applicationDirPath()); + QDir interfacePluginPath = testPath; + + + + qDebug() << "Our directory is" << testPath; + + interfacePluginPath.cdUp(); + interfacePluginPath.cdUp(); + interfacePluginPath.cd("interface"); + interfacePluginPath.cd("plugins"); + interfacePluginPath.makeAbsolute(); + + QString ourPluginPath = testPath.filePath("plugins"); + + + qDebug() << "Interface plugins are at" << interfacePluginPath; + qDebug() << "Our plugins are at" << ourPluginPath; + + + QFile::link(interfacePluginPath.path(), ourPluginPath); + +} + +void CodecTests::loadCodecs() { + const auto& codecPlugins = PluginManager::getInstance()->getCodecPlugins(); + + QVERIFY(codecPlugins.size() > 0); + + + for (const auto& plugin : codecPlugins) { + auto codecName = plugin->getName(); + qDebug() << "Codec:" << codecName << ", supported=" << plugin->isSupported(); + } +} + + + +void CodecTests::testEncoders() { + const auto& codecPlugins = PluginManager::getInstance()->getCodecPlugins(); + + QVERIFY(codecPlugins.size() > 0); + + + for (const auto& plugin : codecPlugins) { + if (!plugin->isSupported()) { + qWarning() << "Skipping unsupported plugin" << plugin->getName(); + continue; + } + + Encoder* encoder = plugin->createEncoder(AudioConstants::SAMPLE_RATE, AudioConstants::STEREO); + QVERIFY(encoder != nullptr); + + QByteArray data(AudioConstants::NETWORK_FRAME_BYTES_STEREO, 0); + QByteArray encoded; + + encoder->encode(data, encoded); + + QVERIFY(encoded.size() > 0); + + qDebug() << "Codec" << plugin->getName() << "encoded empty buffer of" << data.size() << "bytes into" << encoded.size(); + } +} + +void CodecTests::testDecoders () { + const auto& codecPlugins = PluginManager::getInstance()->getCodecPlugins(); + + QVERIFY(codecPlugins.size() > 0); + + + for (const auto& plugin : codecPlugins) { + if (!plugin->isSupported()) { + qWarning() << "Skipping unsupported plugin" << plugin->getName(); + continue; + } + + Encoder* encoder = plugin->createEncoder(AudioConstants::SAMPLE_RATE, AudioConstants::STEREO); + Decoder* decoder = plugin->createDecoder(AudioConstants::SAMPLE_RATE, AudioConstants::STEREO); + QVERIFY(encoder != nullptr); + + QByteArray data(AudioConstants::NETWORK_FRAME_BYTES_STEREO, 0); + QByteArray encoded; + QByteArray decoded; + QByteArray lost(AudioConstants::NETWORK_FRAME_BYTES_STEREO, 0); + + + encoder->encode(data, encoded); + decoder->decode(encoded, decoded); + + + QVERIFY(encoded.size() > 0); + QVERIFY(decoded.size() > 0); + QVERIFY(decoded.size() == data.size()); + QVERIFY(lost.size() > 0); + + + qDebug() << "Codec" << plugin->getName() << "encoded empty buffer of" << data.size() << "bytes into" << encoded.size() << "and decoded back into" << decoded.size(); + + // This is here mostly for valgrind testing -- we can't really validate anything, but we can see if it crashes. + decoder->lostFrame(lost); + QVERIFY(lost.size() > 0); + qDebug() << "Codec" << plugin->getName() << "decoded a lost frame"; + } +} diff --git a/tests/audio/src/CodecTests.h b/tests/audio/src/CodecTests.h new file mode 100644 index 0000000000..05280ddf0f --- /dev/null +++ b/tests/audio/src/CodecTests.h @@ -0,0 +1,32 @@ +// +// AudioTests.h +// tests/audio/src +// +// Created by Dale Glass on 27/8/2022 +// Copyright 2022 Overte e.V. +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#ifndef hifi_AudioTests_h +#define hifi_AudioTests_h + +#include + + + +class CodecTests : public QObject { + Q_OBJECT +private slots: + void initTestCase(); + + void loadCodecs(); + + + void testEncoders(); + void testDecoders(); + +}; + +#endif // hifi_AudioTests_h