summaryrefslogtreecommitdiffstats
path: root/tests/auto/integration/shared/mediafileselector.h
diff options
context:
space:
mode:
Diffstat (limited to 'tests/auto/integration/shared/mediafileselector.h')
-rw-r--r--tests/auto/integration/shared/mediafileselector.h191
1 files changed, 143 insertions, 48 deletions
diff --git a/tests/auto/integration/shared/mediafileselector.h b/tests/auto/integration/shared/mediafileselector.h
index 8a9c3e86a..e36677f34 100644
--- a/tests/auto/integration/shared/mediafileselector.h
+++ b/tests/auto/integration/shared/mediafileselector.h
@@ -1,75 +1,170 @@
-/****************************************************************************
-**
-** Copyright (C) 2017 The Qt Company Ltd.
-** Contact: https://www.qt.io/licensing/
-**
-** This file is part of the test suite of the Qt Toolkit.
-**
-** $QT_BEGIN_LICENSE:GPL-EXCEPT$
-** Commercial License Usage
-** Licensees holding valid commercial Qt licenses may use this file in
-** accordance with the commercial license agreement provided with the
-** Software or, alternatively, in accordance with the terms contained in
-** a written agreement between you and The Qt Company. For licensing terms
-** and conditions see https://www.qt.io/terms-conditions. For further
-** information use the contact form at https://www.qt.io/contact-us.
-**
-** GNU General Public License Usage
-** Alternatively, this file may be used under the terms of the GNU
-** General Public License version 3 as published by the Free Software
-** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
-** included in the packaging of this file. Please review the following
-** information to ensure the GNU General Public License requirements will
-** be met: https://www.gnu.org/licenses/gpl-3.0.html.
-**
-** $QT_END_LICENSE$
-**
-****************************************************************************/
+// Copyright (C) 2017 The Qt Company Ltd.
+// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only
#ifndef MEDIAFILESELECTOR_H
#define MEDIAFILESELECTOR_H
#include <QUrl>
-#include <QMediaPlayer>
-#include <QAudioOutput>
+#include <qmediaplayer.h>
+#include <qaudiooutput.h>
+#include <qvideosink.h>
#include <qsignalspy.h>
#include <qfileinfo.h>
#include <qtest.h>
+#include <private/qmultimediautils_p.h>
+
+#include <unordered_map>
QT_BEGIN_NAMESPACE
-namespace MediaFileSelector {
+using MaybeUrl = QMaybe<QUrl, QString>;
+
+#define CHECK_SELECTED_URL(maybeUrl) \
+ if (!maybeUrl) \
+ QSKIP((QLatin1String("\nUnable to select none of the media candidates:\n") + maybeUrl.error()) \
+ .toLocal8Bit() \
+ .data())
-static QUrl selectMediaFile(const QStringList& mediaCandidates)
+class MediaFileSelector
{
- QMediaPlayer player;
- QAudioOutput audioOutput;
- QVideoSink videoOutput;
- player.setAudioOutput(&audioOutput);
- player.setVideoOutput(&videoOutput);
+public:
+ int failedSelectionsCount() const { return m_failedSelectionsCount; }
- QSignalSpy errorSpy(&player, SIGNAL(errorOccurred(QMediaPlayer::Error, const QString&)));
+ QString dumpErrors() const
+ {
+ QStringList failedMedias;
+ for (const auto &mediaToError : m_mediaToErrors)
+ if (!mediaToError.second.isEmpty())
+ failedMedias.emplace_back(mediaToError.first);
- for (const QString &media : mediaCandidates) {
- player.setSource(media);
- player.play();
+ failedMedias.sort();
+ return dumpErrors(failedMedias);
+ }
+
+ template <typename... Media>
+ MaybeUrl select(Media... media)
+ {
+ return select({ std::move(nativeFileName(media))... });
+ }
+
+ MaybeUrl select(const QStringList &candidates)
+ {
+ QUrl foundUrl;
+ for (const auto &media : candidates) {
+ auto emplaceRes = m_mediaToErrors.try_emplace(media, QString());
+ if (emplaceRes.second) {
+ auto maybeUrl = selectMediaFile(media);
+ if (!maybeUrl) {
+ Q_ASSERT(!maybeUrl.error().isEmpty());
+ emplaceRes.first->second = maybeUrl.error();
+ }
+ }
- for (int i = 0; i < 2000 && player.mediaStatus() != QMediaPlayer::BufferedMedia && errorSpy.isEmpty(); i+=50) {
- QTest::qWait(50);
+ if (foundUrl.isEmpty() && emplaceRes.first->second.isEmpty())
+ foundUrl = media;
}
- if (player.mediaStatus() == QMediaPlayer::BufferedMedia && errorSpy.isEmpty()) {
- return media;
+ if (!foundUrl.isEmpty())
+ return foundUrl;
+
+ ++m_failedSelectionsCount;
+ return { QUnexpect{}, dumpErrors(candidates) };
+ }
+
+private:
+ QString dumpErrors(const QStringList &medias) const
+ {
+ using namespace Qt::StringLiterals;
+ QString result;
+
+ for (const auto &media : medias) {
+ auto it = m_mediaToErrors.find(media);
+ if (it != m_mediaToErrors.end() && !it->second.isEmpty())
+ result.append("\t"_L1)
+ .append(it->first)
+ .append(": "_L1)
+ .append(it->second)
+ .append("\n"_L1);
}
- errorSpy.clear();
+
+ return result;
+ }
+
+ static MaybeUrl selectMediaFile(QString media)
+ {
+ if (qEnvironmentVariableIsSet("QTEST_SKIP_MEDIA_VALIDATION"))
+ return QUrl(media);
+
+ using namespace Qt::StringLiterals;
+
+ QAudioOutput audioOutput;
+ QVideoSink videoOutput;
+ QMediaPlayer player;
+ player.setAudioOutput(&audioOutput);
+ player.setVideoOutput(&videoOutput);
+
+ player.setSource(media);
+ player.play();
+
+ const auto waitingFinished = QTest::qWaitFor([&]() {
+ const auto status = player.mediaStatus();
+ return status == QMediaPlayer::BufferedMedia || status == QMediaPlayer::EndOfMedia
+ || status == QMediaPlayer::InvalidMedia
+ || player.error() != QMediaPlayer::NoError;
+ });
+
+ auto enumValueToString = [](auto enumValue) {
+ return QString(QMetaEnum::fromType<decltype(enumValue)>().valueToKey(enumValue));
+ };
+
+ if (!waitingFinished)
+ return { QUnexpect{},
+ "The media got stuck in the status "_L1
+ + enumValueToString(player.mediaStatus()) };
+
+ if (player.mediaStatus() == QMediaPlayer::InvalidMedia)
+ return { QUnexpect{},
+ "Unable to load the media. Error ["_L1 + enumValueToString(player.error())
+ + " "_L1 + player.errorString() + "]"_L1 };
+
+ if (player.error() != QMediaPlayer::NoError)
+ return { QUnexpect{},
+ "Unable to start playing the media, codecs issues. Error ["_L1
+ + enumValueToString(player.error()) + " "_L1 + player.errorString()
+ + "]"_L1 };
+
+ return QUrl(media);
}
- return QUrl();
-}
+ QString nativeFileName(const QString &media)
+ {
+#ifdef Q_OS_ANDROID
+ auto it = m_nativeFiles.find(media);
+ if (it != m_nativeFiles.end())
+ return it->second->fileName();
+
+ QFile file(media);
+ if (file.open(QIODevice::ReadOnly)) {
+ m_nativeFiles.insert({ media, std::unique_ptr<QTemporaryFile>(QTemporaryFile::createNativeFile(file))});
+ return m_nativeFiles[media]->fileName();
+ }
+ qWarning() << "Failed to create temporary file";
+#endif // Q_OS_ANDROID
+
+ return media;
+ }
-} // MediaFileSelector namespace
+private:
+#ifdef Q_OS_ANDROID
+ std::unordered_map<QString, std::unique_ptr<QTemporaryFile>> m_nativeFiles;
+#endif
+ std::unordered_map<QString, QString> m_mediaToErrors;
+ int m_failedSelectionsCount = 0;
+};
QT_END_NAMESPACE
+Q_DECLARE_METATYPE(MaybeUrl)
+
#endif