Files correlati : cg0.exe cg0700a.msk cg0700b.msk cg3.exe cg4.exe Bug : Commento: Merge 1.0 libraries
423 lines
14 KiB
C++
423 lines
14 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2015 The Qt Company Ltd.
|
|
** Contact: http://www.qt.io/licensing/
|
|
**
|
|
** This file is part of the test suite of the Qt Toolkit.
|
|
**
|
|
** $QT_BEGIN_LICENSE:LGPL$
|
|
** 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 http://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at http://www.qt.io/contact-us.
|
|
**
|
|
** GNU Lesser General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
** following information to ensure the GNU Lesser General Public License
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
**
|
|
** As a special exception, The Qt Company gives you certain additional
|
|
** rights. These rights are described in The Qt Company LGPL Exception
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3.0 as published by the Free Software
|
|
** Foundation and appearing in the file LICENSE.GPL included in the
|
|
** packaging of this file. Please review the following information to
|
|
** ensure the GNU General Public License version 3.0 requirements will be
|
|
** met: http://www.gnu.org/copyleft/gpl.html.
|
|
**
|
|
** $QT_END_LICENSE$
|
|
**
|
|
****************************************************************************/
|
|
#include <qtest.h>
|
|
#include <QDeclarativeView>
|
|
#include <QApplication>
|
|
#include <QLibraryInfo>
|
|
#include <QDir>
|
|
#include <QDebug>
|
|
#include <QProcess>
|
|
#include <QFile>
|
|
|
|
#ifdef Q_OS_SYMBIAN
|
|
// In Symbian OS test data is located in applications private dir
|
|
#define QT_TEST_SOURCE_DIR "."
|
|
#endif
|
|
|
|
enum Mode { Record, RecordNoVisuals, RecordSnapshot, Play, TestVisuals, RemoveVisuals, UpdateVisuals, UpdatePlatformVisuals, Test };
|
|
|
|
static QString testdir;
|
|
class tst_qmlvisual : public QObject
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
tst_qmlvisual();
|
|
|
|
static QString toTestScript(const QString &, Mode=Test);
|
|
static QString viewer();
|
|
|
|
static QStringList findQmlFiles(const QDir &d);
|
|
private slots:
|
|
void visual_data();
|
|
void visual();
|
|
|
|
private:
|
|
QString qmlruntime;
|
|
};
|
|
|
|
|
|
tst_qmlvisual::tst_qmlvisual()
|
|
{
|
|
qmlruntime = viewer();
|
|
}
|
|
|
|
QString tst_qmlvisual::viewer()
|
|
{
|
|
QString binaries = QLibraryInfo::location(QLibraryInfo::BinariesPath);
|
|
|
|
QString qmlruntime;
|
|
|
|
#if defined(Q_WS_MAC)
|
|
qmlruntime = QDir(binaries).absoluteFilePath("QMLViewer.app/Contents/MacOS/QMLViewer");
|
|
#elif defined(Q_WS_WIN) || defined(Q_WS_S60)
|
|
qmlruntime = QDir(binaries).absoluteFilePath("qmlviewer.exe");
|
|
#else
|
|
qmlruntime = QDir(binaries).absoluteFilePath("qmlviewer");
|
|
#endif
|
|
|
|
return qmlruntime;
|
|
}
|
|
|
|
void tst_qmlvisual::visual_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
QTest::addColumn<QString>("testdata");
|
|
|
|
QStringList files;
|
|
files << findQmlFiles(QDir(QT_TEST_SOURCE_DIR));
|
|
if (qgetenv("QMLVISUAL_ALL") != "1") {
|
|
#if defined(Q_WS_MAC)
|
|
//Text on Mac varies per version. Only check the text on 10.6
|
|
if(QSysInfo::MacintoshVersion != QSysInfo::MV_10_6)
|
|
foreach(const QString &str, files.filter(QRegExp(".*text.*")))
|
|
files.removeAll(str);
|
|
#endif
|
|
#if defined(Q_WS_QWS)
|
|
//We don't want QWS test results to mire down the CI system
|
|
files.clear();
|
|
//Needs at least one test data or it fails anyways
|
|
files << QT_TEST_SOURCE_DIR "/selftest_noimages/selftest_noimages.qml";
|
|
#endif
|
|
}
|
|
|
|
foreach (const QString &file, files) {
|
|
QString testdata = toTestScript(file);
|
|
if (testdata.isEmpty())
|
|
continue;
|
|
|
|
QTest::newRow(file.toLatin1().constData()) << file << testdata;
|
|
}
|
|
}
|
|
|
|
void tst_qmlvisual::visual()
|
|
{
|
|
QFETCH(QString, file);
|
|
QFETCH(QString, testdata);
|
|
|
|
QStringList arguments;
|
|
#ifdef Q_WS_MAC
|
|
arguments << "-no-opengl";
|
|
#endif
|
|
arguments << "-script" << testdata
|
|
<< "-scriptopts" << "play,testimages,testerror,testskip,exitoncomplete,exitonfailure"
|
|
<< file;
|
|
#ifdef Q_WS_QWS
|
|
arguments << "-qws";
|
|
#endif
|
|
|
|
QProcess p;
|
|
p.start(qmlruntime, arguments);
|
|
bool finished = p.waitForFinished();
|
|
QByteArray output = p.readAllStandardOutput() + p.readAllStandardError();
|
|
QVERIFY2(finished, output.data());
|
|
if (p.exitCode() != 0)
|
|
qDebug() << output;
|
|
QCOMPARE(p.exitStatus(), QProcess::NormalExit);
|
|
QCOMPARE(p.exitCode(), 0);
|
|
}
|
|
|
|
QString tst_qmlvisual::toTestScript(const QString &file, Mode mode)
|
|
{
|
|
if (!file.endsWith(".qml"))
|
|
return QString();
|
|
|
|
int index = file.lastIndexOf(QDir::separator());
|
|
if (index == -1)
|
|
index = file.lastIndexOf('/');
|
|
if (index == -1)
|
|
return QString();
|
|
|
|
const char* platformsuffix=0; // platforms with different fonts
|
|
#if defined(Q_WS_MACX)
|
|
platformsuffix = "-MAC";
|
|
#elif defined(Q_WS_X11)
|
|
platformsuffix = "-X11";
|
|
#elif defined(Q_WS_WIN32)
|
|
platformsuffix = "-WIN";
|
|
#elif defined(Q_WS_QWS)
|
|
platformsuffix = "-QWS";
|
|
#elif defined(Q_WS_S60)
|
|
platformsuffix = "-S60";
|
|
#endif
|
|
|
|
QString testdata = file.left(index + 1) +
|
|
QString("data");
|
|
QString testname = file.mid(index + 1, file.length() - index - 5);
|
|
|
|
if (platformsuffix && (mode == UpdatePlatformVisuals || QFile::exists(testdata+QLatin1String(platformsuffix)+QDir::separator()+testname+".qml"))) {
|
|
QString platformdir = testdata + QLatin1String(platformsuffix);
|
|
if (mode == UpdatePlatformVisuals) {
|
|
if (!QDir().mkpath(platformdir)) {
|
|
qFatal("Cannot make path %s", qPrintable(platformdir));
|
|
}
|
|
// Copy from base
|
|
QDir dir(testdata,testname+".*");
|
|
dir.setFilter(QDir::Files);
|
|
QFileInfoList list = dir.entryInfoList();
|
|
for (int i = 0; i < list.size(); ++i) {
|
|
QFile in(list.at(i).filePath());
|
|
if (!in.open(QIODevice::ReadOnly)) {
|
|
qFatal("Cannot open file %s: %s", qPrintable(in.fileName()), qPrintable(in.errorString()));
|
|
}
|
|
QFile out(platformdir + QDir::separator() + list.at(i).fileName());
|
|
if (!out.open(QIODevice::WriteOnly)) {
|
|
qFatal("Cannot open file %s: %s", qPrintable(out.fileName()), qPrintable(out.errorString()));
|
|
}
|
|
out.write(in.readAll());
|
|
}
|
|
}
|
|
testdata = platformdir;
|
|
}
|
|
|
|
testdata += QDir::separator() + testname;
|
|
|
|
return testdata;
|
|
}
|
|
|
|
QStringList tst_qmlvisual::findQmlFiles(const QDir &d)
|
|
{
|
|
QStringList rv;
|
|
|
|
QStringList files = d.entryList(QStringList() << QLatin1String("*.qml"),
|
|
QDir::Files);
|
|
foreach (const QString &file, files) {
|
|
if (file.at(0).isLower()) {
|
|
rv << d.absoluteFilePath(file);
|
|
}
|
|
}
|
|
|
|
QStringList dirs = d.entryList(QDir::Dirs | QDir::NoDotAndDotDot |
|
|
QDir::NoSymLinks);
|
|
foreach (const QString &dir, dirs) {
|
|
if (dir.left(4) == "data")
|
|
continue;
|
|
|
|
QDir sub = d;
|
|
sub.cd(dir);
|
|
rv << findQmlFiles(sub);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
void action(Mode mode, const QString &file)
|
|
{
|
|
QString testdata = tst_qmlvisual::toTestScript(file,mode);
|
|
|
|
QStringList arguments;
|
|
#ifdef Q_WS_MAC
|
|
arguments << "-no-opengl";
|
|
#endif
|
|
switch (mode) {
|
|
case Test:
|
|
// Don't run qml
|
|
break;
|
|
case Record:
|
|
arguments << "-script" << testdata
|
|
<< "-scriptopts" << "record,testimages,saveonexit"
|
|
<< file;
|
|
break;
|
|
case RecordNoVisuals:
|
|
arguments << "-script" << testdata
|
|
<< "-scriptopts" << "record,saveonexit"
|
|
<< file;
|
|
break;
|
|
case RecordSnapshot:
|
|
arguments << "-script" << testdata
|
|
<< "-scriptopts" << "record,testimages,snapshot,saveonexit"
|
|
<< file;
|
|
break;
|
|
case Play:
|
|
arguments << "-script" << testdata
|
|
<< "-scriptopts" << "play,testimages,testerror,testskip,exitoncomplete"
|
|
<< file;
|
|
break;
|
|
case TestVisuals:
|
|
arguments << "-script" << testdata
|
|
<< "-scriptopts" << "play"
|
|
<< file;
|
|
break;
|
|
case UpdateVisuals:
|
|
case UpdatePlatformVisuals:
|
|
arguments << "-script" << testdata
|
|
<< "-scriptopts" << "play,record,testimages,exitoncomplete,saveonexit"
|
|
<< file;
|
|
break;
|
|
case RemoveVisuals:
|
|
arguments << "-script" << testdata
|
|
<< "-scriptopts" << "play,record,exitoncomplete,saveonexit"
|
|
<< file;
|
|
break;
|
|
}
|
|
if (!arguments.isEmpty()) {
|
|
QProcess p;
|
|
p.setProcessChannelMode(QProcess::ForwardedChannels);
|
|
p.start(tst_qmlvisual::viewer(), arguments);
|
|
p.waitForFinished();
|
|
}
|
|
}
|
|
|
|
void usage()
|
|
{
|
|
fprintf(stderr, "\n");
|
|
fprintf(stderr, "QML related options\n");
|
|
fprintf(stderr, " -listtests : list all the tests seen by tst_qmlvisual, and then exit immediately\n");
|
|
fprintf(stderr, " -record file : record new test data for file\n");
|
|
fprintf(stderr, " -recordnovisuals file : record new test data for file, but ignore visuals\n");
|
|
fprintf(stderr, " -recordsnapshot file : record new snapshot for file (like record but only records a single frame and then exits)\n");
|
|
fprintf(stderr, " -play file : playback test data for file, printing errors\n");
|
|
fprintf(stderr, " -testvisuals file : playback test data for file, without errors\n");
|
|
fprintf(stderr, " -updatevisuals file : playback test data for file, accept new visuals for file\n");
|
|
fprintf(stderr, " -updateplatformvisuals file : playback test data for file, accept new visuals for file only on current platform (MacOSX/Win32/X11/QWS/S60)\n");
|
|
fprintf(stderr, "\n"
|
|
"Visual tests are recordings of manual interactions with a QML test,\n"
|
|
"that can then be run automatically. To record a new test, run:\n"
|
|
"\n"
|
|
" tst_qmlvisual -record yourtestdir/yourtest.qml\n"
|
|
"\n"
|
|
"This records everything you do (try to keep it short).\n"
|
|
"To play back a test, run:\n"
|
|
"\n"
|
|
" tst_qmlvisual -play yourtestdir/yourtest.qml\n"
|
|
"\n"
|
|
"Your test may include QML code to test itself, reporting any error to an\n"
|
|
"'error' property on the root object - the test will fail if this property\n"
|
|
"gets set to anything non-empty.\n"
|
|
"\n"
|
|
"If your test changes slightly but is still correct (check with -play), you\n"
|
|
"can update the visuals by running:\n"
|
|
"\n"
|
|
" tst_qmlvisual -updatevisuals yourtestdir/yourtest.qml\n"
|
|
"\n"
|
|
"If your test includes platform-sensitive visuals (eg. text in system fonts),\n"
|
|
"you should create platform-specific visuals, using -updateplatformvisuals\n"
|
|
"instead.\n"
|
|
"\n"
|
|
"If you ONLY wish to use the 'error' property, you can record your test with\n"
|
|
"-recordnovisuals, or discard existing visuals with -removevisuals; the test\n"
|
|
"will then only fail on a syntax error, crash, or non-empty 'error' property.\n"
|
|
"\n"
|
|
"If your test has anything set to the 'skip' property on the root object then\n"
|
|
"test failures will be ignored. This allows for an opt-out of automated\n"
|
|
"aggregation of test results. The value of the 'skip' property (usually a\n"
|
|
"string) will then be printed to stdout when the test is run as part of the\n"
|
|
"message saying the test has been skipped.\n"
|
|
);
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
QApplication app(argc, argv);
|
|
|
|
Mode mode = Test;
|
|
QString file;
|
|
bool showHelp = false;
|
|
|
|
int newArgc = 1;
|
|
char **newArgv = new char*[argc];
|
|
newArgv[0] = argv[0];
|
|
|
|
for (int ii = 1; ii < argc; ++ii) {
|
|
QString arg(argv[ii]);
|
|
if (arg == "-play" && (ii + 1) < argc) {
|
|
mode = Play;
|
|
file = argv[++ii];
|
|
} else if (arg == "-record" && (ii + 1) < argc) {
|
|
mode = Record;
|
|
file = argv[++ii];
|
|
} else if (arg == "-recordnovisuals" && (ii + 1) < argc) {
|
|
mode = RecordNoVisuals;
|
|
file = argv[++ii];
|
|
} else if (arg == "-recordsnapshot" && (ii + 1) < argc) {
|
|
mode = RecordSnapshot;
|
|
file = argv[++ii];
|
|
} else if (arg == "-testvisuals" && (ii + 1) < argc) {
|
|
mode = TestVisuals;
|
|
file = argv[++ii];
|
|
} else if (arg == "-removevisuals" && (ii + 1) < argc) {
|
|
mode = RemoveVisuals;
|
|
file = argv[++ii];
|
|
} else if (arg == "-updatevisuals" && (ii + 1) < argc) {
|
|
mode = UpdateVisuals;
|
|
file = argv[++ii];
|
|
} else if (arg == "-updateplatformvisuals" && (ii + 1) < argc) {
|
|
mode = UpdatePlatformVisuals;
|
|
file = argv[++ii];
|
|
} else {
|
|
newArgv[newArgc++] = argv[ii];
|
|
}
|
|
|
|
if (arg == "-help" || arg == "-?" || arg == "--help") {
|
|
atexit(usage);
|
|
showHelp = true;
|
|
}
|
|
|
|
if (arg == "-listtests") {
|
|
QStringList list = tst_qmlvisual::findQmlFiles(QDir(QT_TEST_SOURCE_DIR));
|
|
foreach (QString test, list) {
|
|
qWarning() << qPrintable(test);
|
|
}
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
if (mode == Test || showHelp) {
|
|
tst_qmlvisual tc;
|
|
return QTest::qExec(&tc, newArgc, newArgv);
|
|
} else {
|
|
if (!file.endsWith(QLatin1String(".qml"))) {
|
|
qWarning() << "Error: Invalid file name (must end in .qml)";
|
|
return -1;
|
|
}
|
|
QDir d = QDir::current();
|
|
QString absFile = d.absoluteFilePath(file);
|
|
if (!QFile::exists(absFile)) {
|
|
qWarning() << "Error: File does not exist";
|
|
return -1;
|
|
}
|
|
|
|
action(mode, absFile);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
#include "tst_qmlvisual.moc"
|