--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/auto/qtemporaryfile/tst_qtemporaryfile.cpp Mon Jan 11 14:00:40 2010 +0000
@@ -0,0 +1,562 @@
+/****************************************************************************
+**
+** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
+** All rights reserved.
+** Contact: Nokia Corporation (qt-info@nokia.com)
+**
+** This file is part of the test suite of the Qt Toolkit.
+**
+** $QT_BEGIN_LICENSE:LGPL$
+** No Commercial Usage
+** This file contains pre-release code and may not be distributed.
+** You may use this file in accordance with the terms and conditions
+** contained in the Technology Preview License Agreement accompanying
+** this package.
+**
+** 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 as published by the Free Software
+** Foundation and appearing in the file LICENSE.LGPL included in the
+** packaging of this file. Please review the following information to
+** ensure the GNU Lesser General Public License version 2.1 requirements
+** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
+**
+** In addition, as a special exception, Nokia gives you certain additional
+** rights. These rights are described in the Nokia Qt LGPL Exception
+** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
+**
+** If you have questions regarding the use of this file, please contact
+** Nokia at qt-info@nokia.com.
+**
+**
+**
+**
+**
+**
+**
+**
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+
+
+#include <QtTest/QtTest>
+#include <qcoreapplication.h>
+#include <qstring.h>
+#include <qtemporaryfile.h>
+#include <qfile.h>
+#include <qdir.h>
+#include <qset.h>
+
+#if defined(Q_OS_WIN)
+# include <windows.h>
+#endif
+#if defined(Q_OS_UNIX)
+# include <sys/types.h>
+# include <sys/stat.h>
+# include <errno.h>
+# include <fcntl.h> // open(2)
+# include <unistd.h> // close(2)
+#endif
+
+#ifdef Q_OS_SYMBIAN
+#define SRCDIR ""
+#endif
+
+//TESTED_CLASS=
+//TESTED_FILES=
+
+class tst_QTemporaryFile : public QObject
+{
+ Q_OBJECT
+public:
+ tst_QTemporaryFile();
+ virtual ~tst_QTemporaryFile();
+public slots:
+ void init();
+ void cleanup();
+private slots:
+ void construction();
+ void fileTemplate();
+ void fileTemplate_data();
+ void getSetCheck();
+ void fileName();
+ void fileNameIsEmpty();
+ void autoRemove();
+ void write();
+ void openCloseOpenClose();
+ void size();
+ void resize();
+ void openOnRootDrives();
+ void stressTest();
+ void rename();
+ void renameFdLeak();
+ void reOpenThroughQFile();
+ void keepOpenMode();
+ void resetTemplateAfterError();
+ void setTemplateAfterOpen();
+
+public:
+};
+
+void tst_QTemporaryFile::construction()
+{
+ QTemporaryFile file(0);
+ QString tmp = QDir::tempPath();
+ QCOMPARE(file.fileTemplate().left(tmp.size()), tmp);
+ QCOMPARE(file.fileTemplate().at(tmp.size()), QChar('/'));
+}
+
+// Testing get/set functions
+void tst_QTemporaryFile::getSetCheck()
+{
+ QTemporaryFile obj1;
+ // bool QTemporaryFile::autoRemove()
+ // void QTemporaryFile::setAutoRemove(bool)
+ obj1.setAutoRemove(false);
+ QCOMPARE(false, obj1.autoRemove());
+ obj1.setAutoRemove(true);
+ QCOMPARE(true, obj1.autoRemove());
+}
+
+tst_QTemporaryFile::tst_QTemporaryFile()
+{
+}
+
+tst_QTemporaryFile::~tst_QTemporaryFile()
+{
+
+}
+
+void tst_QTemporaryFile::init()
+{
+// TODO: Add initialization code here.
+// This will be executed immediately before each test is run.
+}
+
+void tst_QTemporaryFile::cleanup()
+{
+// TODO: Add cleanup code here.
+// This will be executed immediately after each test is run.
+}
+
+void tst_QTemporaryFile::fileTemplate_data()
+{
+ QTest::addColumn<QString>("constructorTemplate");
+ QTest::addColumn<QString>("suffix");
+ QTest::addColumn<QString>("fileTemplate");
+
+ QTest::newRow("constructor default") << "" << "" << "";
+ QTest::newRow("constructor with xxx sufix") << "qt_XXXXXXxxx" << "xxx" << "";
+ QTest::newRow("constructor with xXx sufix") << "qt_XXXXXXxXx" << "xXx" << "";
+ QTest::newRow("constructor with no sufix") << "qt_XXXXXX" << "" << "";
+ QTest::newRow("constructor with >6 X's and xxx suffix") << "qt_XXXXXXXXXXxxx" << "xxx" << "";
+ QTest::newRow("constructor with >6 X's, no suffix") << "qt_XXXXXXXXXX" << "" << "";
+
+ QTest::newRow("set template, no suffix") << "" << "" << "foo";
+ QTest::newRow("set template, with lowercase XXXXXX") << "" << "xxxxxx" << "qt_XXXXXXxxxxxx";
+ QTest::newRow("set template, with xxx") << "" << ".xxx" << "qt_XXXXXX.xxx";
+ QTest::newRow("set template, with >6 X's") << "" << ".xxx" << "qt_XXXXXXXXXXXXXX.xxx";
+ QTest::newRow("set template, with >6 X's, no suffix") << "" << "" << "qt_XXXXXXXXXXXXXX";
+}
+
+void tst_QTemporaryFile::fileTemplate()
+{
+ QFETCH(QString, constructorTemplate);
+ QFETCH(QString, suffix);
+ QFETCH(QString, fileTemplate);
+
+ QTemporaryFile file(constructorTemplate);
+ if (!fileTemplate.isEmpty())
+ file.setFileTemplate(fileTemplate);
+
+ QCOMPARE(file.open(), true);
+
+ QCOMPARE(file.fileName().right(suffix.length()), suffix);
+ file.close();
+}
+
+
+/*
+ This tests whether the temporary file really gets placed in QDir::tempPath
+*/
+void tst_QTemporaryFile::fileName()
+{
+ // Get QDir::tempPath and make an absolute path.
+ QString tempPath = QDir::tempPath();
+ QString absoluteTempPath = QDir(tempPath).absolutePath();
+ QTemporaryFile file;
+ file.setAutoRemove(true);
+ file.open();
+ QString fileName = file.fileName();
+ QVERIFY(QFile::exists(fileName));
+ // Get path to the temp file, whithout the file name.
+ QString absoluteFilePath = QFileInfo(fileName).absolutePath();
+#if defined(Q_OS_WIN) || defined(Q_OS_SYMBIAN)
+ absoluteFilePath = absoluteFilePath.toLower();
+ absoluteTempPath = absoluteTempPath.toLower();
+#endif
+ QCOMPARE(absoluteFilePath, absoluteTempPath);
+}
+
+void tst_QTemporaryFile::fileNameIsEmpty()
+{
+ QString filename;
+ {
+ QTemporaryFile file;
+ QVERIFY(file.fileName().isEmpty());
+
+ QVERIFY(file.open());
+ QVERIFY(!file.fileName().isEmpty());
+
+ filename = file.fileName();
+ QVERIFY(QFile::exists(filename));
+
+ file.close();
+ QVERIFY(!file.isOpen());
+ QVERIFY(QFile::exists(filename));
+ QVERIFY(!file.fileName().isEmpty());
+ }
+ QVERIFY(!QFile::exists(filename));
+}
+
+void tst_QTemporaryFile::autoRemove()
+{
+ // Test auto remove
+ QString fileName;
+ {
+ QTemporaryFile file("tempXXXXXX");
+ file.setAutoRemove(true);
+ QVERIFY(file.open());
+ fileName = file.fileName();
+ file.close();
+ }
+ QVERIFY(!QFile::exists(fileName));
+
+ // Test if disabling auto remove works.
+ {
+ QTemporaryFile file("tempXXXXXX");
+ file.setAutoRemove(false);
+ QVERIFY(file.open());
+ fileName = file.fileName();
+ file.close();
+ }
+ QVERIFY(QFile::exists(fileName));
+ QVERIFY(QFile::remove(fileName));
+
+
+ // Do not explicitly call setAutoRemove (tests if it really is the default as documented)
+ {
+ QTemporaryFile file("tempXXXXXX");
+ QVERIFY(file.open());
+ fileName = file.fileName();
+ file.close();
+ }
+ QVERIFY(!QFile::exists(fileName));
+
+}
+
+void tst_QTemporaryFile::write()
+{
+ QByteArray data("OLE\nOLE\nOLE");
+ QTemporaryFile file;
+ QVERIFY(file.open());
+ QCOMPARE((int)file.write(data), data.size());
+ file.reset();
+ QFile compare(file.fileName());
+ compare.open(QIODevice::ReadOnly);
+ QCOMPARE(compare.readAll() , data);
+ file.close();
+}
+
+void tst_QTemporaryFile::openCloseOpenClose()
+{
+ QString fileName;
+ {
+ // Create a temp file
+ QTemporaryFile file("tempXXXXXX");
+ file.setAutoRemove(true);
+ QVERIFY(file.open());
+ file.write("OLE");
+ fileName = file.fileName();
+ QVERIFY(QFile::exists(fileName));
+ file.close();
+
+ // Check that it still exists after being closed
+ QVERIFY(QFile::exists(fileName));
+ QVERIFY(!file.isOpen());
+ QVERIFY(file.open());
+ QCOMPARE(file.readAll(), QByteArray("OLE"));
+ // Check that it's still the same file after being opened again.
+ QCOMPARE(file.fileName(), fileName);
+ }
+ QVERIFY(!QFile::exists(fileName));
+}
+
+void tst_QTemporaryFile::size()
+{
+ QTemporaryFile file;
+ QVERIFY(file.open());
+ QVERIFY(file.exists());
+ QVERIFY(!file.isSequential());
+ QByteArray str("foobar");
+ file.write(str);
+ QVERIFY(QFile::exists(file.fileName()));
+ // On CE it takes more time for the filesystem to update
+ // the information. Usually you have to close it or seek
+ // to get latest information. flush() does not help either.
+#if !defined(Q_OS_WINCE)
+ QCOMPARE(file.size(), qint64(6));
+#endif
+ file.seek(0);
+ QCOMPARE(file.size(), qint64(6));
+}
+
+void tst_QTemporaryFile::resize()
+{
+ QTemporaryFile file;
+ file.setAutoRemove(true);
+ QVERIFY(file.open());
+ QVERIFY(file.resize(100));
+
+ QCOMPARE(QFileInfo(file.fileName()).size(), qint64(100));
+
+ file.close();
+}
+
+void tst_QTemporaryFile::openOnRootDrives()
+{
+#if defined(Q_OS_WIN) && !defined(Q_OS_WINCE)
+ unsigned int lastErrorMode = SetErrorMode(SEM_FAILCRITICALERRORS);
+#endif
+ // If it's possible to create a file in the root directory, it
+ // must be possible to create a temp file there too.
+ foreach (QFileInfo driveInfo, QDir::drives()) {
+ QFile testFile(driveInfo.filePath() + "XXXXXX.txt");
+ if (testFile.open(QIODevice::ReadWrite)) {
+ testFile.remove();
+ QTemporaryFile file(driveInfo.filePath() + "XXXXXX.txt");
+ file.setAutoRemove(true);
+ QVERIFY(file.open());
+ }
+ }
+#if defined(Q_OS_WIN) && !defined(Q_OS_WINCE)
+ SetErrorMode(lastErrorMode);
+#endif
+}
+
+void tst_QTemporaryFile::stressTest()
+{
+#if defined(Q_OS_WINCE) || defined(Q_OS_SYMBIAN)
+ // 200 is still ok, first colision happens after ~30
+ const int iterations = 200;
+#else
+ const int iterations = 1000;
+#endif
+
+ QSet<QString> names;
+ for (int i = 0; i < iterations; ++i) {
+ QTemporaryFile file;
+ file.setAutoRemove(false);
+ if (!file.open()) {
+ qDebug() << "Could not open File:" << file.fileName();
+ continue;
+ }
+ QVERIFY(!names.contains(file.fileName()));
+ names.insert(file.fileName());
+ }
+ for (QSet<QString>::const_iterator it = names.constBegin(); it != names.constEnd(); ++it) {
+ QFile::remove(*it);
+ }
+}
+
+void tst_QTemporaryFile::rename()
+{
+ // This test checks that the temporary file is deleted, even after a
+ // rename.
+
+ QDir dir;
+ QVERIFY(!dir.exists("temporary-file.txt"));
+
+ QString tempname;
+ {
+ QTemporaryFile file(dir.filePath("temporary-file.XXXXXX"));
+
+ QVERIFY(file.open());
+ tempname = file.fileName();
+ QVERIFY(dir.exists(tempname));
+
+ QVERIFY(file.rename("temporary-file.txt"));
+ QVERIFY(!dir.exists(tempname));
+ QVERIFY(dir.exists("temporary-file.txt"));
+ QCOMPARE(file.fileName(), QString("temporary-file.txt"));
+ }
+
+ QVERIFY(!dir.exists(tempname));
+ QVERIFY(!dir.exists("temporary-file.txt"));
+}
+
+void tst_QTemporaryFile::renameFdLeak()
+{
+#ifdef Q_OS_UNIX
+ // Test this on Unix only
+
+ // Open a bunch of files to force the fd count to go up
+ static const int count = 10;
+ int bunch_of_files[count];
+ for (int i = 0; i < count; ++i) {
+ bunch_of_files[i] = ::open(SRCDIR "tst_qtemporaryfile.cpp", O_RDONLY);
+ QVERIFY(bunch_of_files[i] != -1);
+ }
+
+ int fd;
+ {
+ QTemporaryFile file;
+ file.setAutoRemove(false);
+ QVERIFY(file.open());
+
+ // close the bunch of files
+ for (int i = 0; i < count; ++i)
+ ::close(bunch_of_files[i]);
+
+ // save the file descriptor for later
+ fd = file.handle();
+
+ // rename the file to something
+ QString newPath = QDir::tempPath() + "/tst_qtemporaryfile-renameFdLeak-" + QString::number(getpid());
+ file.rename(newPath);
+ QFile::remove(newPath);
+ }
+
+ // check if QTemporaryFile closed the file
+ QVERIFY(::close(fd) == -1 && errno == EBADF);
+#endif
+}
+
+void tst_QTemporaryFile::reOpenThroughQFile()
+{
+ QByteArray data("abcdefghij");
+
+ QTemporaryFile file;
+ QVERIFY(((QFile &)file).open(QIODevice::WriteOnly));
+ QCOMPARE(file.write(data), (qint64)data.size());
+
+ file.close();
+ QVERIFY(file.open());
+ QCOMPARE(file.readAll(), data);
+}
+
+void tst_QTemporaryFile::keepOpenMode()
+{
+ QByteArray data("abcdefghij");
+
+ {
+ QTemporaryFile file;
+ QVERIFY(((QFile &)file).open(QIODevice::WriteOnly));
+ QVERIFY(QIODevice::WriteOnly == file.openMode());
+
+ QCOMPARE(file.write(data), (qint64)data.size());
+ file.close();
+
+ QVERIFY(((QFile &)file).open(QIODevice::ReadOnly));
+ QVERIFY(QIODevice::ReadOnly == file.openMode());
+ QCOMPARE(file.readAll(), data);
+ }
+
+ {
+ QTemporaryFile file;
+ QVERIFY(file.open());
+ QCOMPARE(file.write(data), (qint64)data.size());
+ QVERIFY(file.rename("temporary-file.txt"));
+
+ QVERIFY(((QFile &)file).open(QIODevice::ReadOnly));
+ QVERIFY(QIODevice::ReadOnly == file.openMode());
+ QCOMPARE(file.readAll(), data);
+
+ QVERIFY(((QFile &)file).open(QIODevice::WriteOnly));
+ QVERIFY(QIODevice::WriteOnly == file.openMode());
+ }
+}
+
+void tst_QTemporaryFile::resetTemplateAfterError()
+{
+ // calling setFileTemplate on a failed open
+
+ QString tempPath = QDir::tempPath();
+
+ QString const fileTemplate("destination/qt_temp_file_test.XXXXXX");
+ QString const fileTemplate2(tempPath + "/qt_temp_file_test.XXXXXX");
+
+ QVERIFY2( QDir(tempPath).exists() || QDir().mkpath(tempPath), "Test precondition" );
+ QVERIFY2( !QFile::exists("destination"), "Test precondition" );
+ QVERIFY2( !QFile::exists(fileTemplate2) || QFile::remove(fileTemplate2), "Test precondition" );
+
+ QFile file(fileTemplate2);
+ QByteArray fileContent("This file is intentionally NOT left empty.");
+
+ QVERIFY( file.open(QIODevice::ReadWrite | QIODevice::Truncate) );
+ QCOMPARE( file.write(fileContent), (qint64)fileContent.size() );
+ QVERIFY( file.flush() );
+
+ QString fileName;
+ {
+ QTemporaryFile temp;
+
+ QVERIFY( temp.fileName().isEmpty() );
+ QVERIFY( !temp.fileTemplate().isEmpty() );
+
+ temp.setFileTemplate( fileTemplate );
+
+ QVERIFY( temp.fileName().isEmpty() );
+ QCOMPARE( temp.fileTemplate(), fileTemplate );
+
+ QVERIFY( !temp.open() );
+
+ QVERIFY( temp.fileName().isEmpty() );
+ QCOMPARE( temp.fileTemplate(), fileTemplate );
+
+ temp.setFileTemplate( fileTemplate2 );
+ QVERIFY( temp.open() );
+
+ fileName = temp.fileName();
+ QVERIFY( QFile::exists(fileName) );
+ QVERIFY( !fileName.isEmpty() );
+ QVERIFY2( fileName != fileTemplate2,
+ ("Generated name shouldn't be same as template: " + fileTemplate2).toLocal8Bit().constData() );
+ }
+
+ QVERIFY( !QFile::exists(fileName) );
+
+ file.seek(0);
+ QCOMPARE( QString(file.readAll()), QString(fileContent) );
+ QVERIFY( file.remove() );
+}
+
+void tst_QTemporaryFile::setTemplateAfterOpen()
+{
+ QTemporaryFile temp;
+
+ QVERIFY( temp.fileName().isEmpty() );
+ QVERIFY( !temp.fileTemplate().isEmpty() );
+
+ QVERIFY( temp.open() );
+
+ QString const fileName = temp.fileName();
+ QString const newTemplate("funny-path/funny-name-XXXXXX.tmp");
+
+ QVERIFY( !fileName.isEmpty() );
+ QVERIFY( QFile::exists(fileName) );
+ QVERIFY( !temp.fileTemplate().isEmpty() );
+ QVERIFY( temp.fileTemplate() != newTemplate );
+
+ temp.close(); // QTemporaryFile::setFileTemplate will assert on isOpen() up to 4.5.2
+ temp.setFileTemplate(newTemplate);
+ QCOMPARE( temp.fileTemplate(), newTemplate );
+
+ QVERIFY( temp.open() );
+ QCOMPARE( temp.fileName(), fileName );
+ QCOMPARE( temp.fileTemplate(), newTemplate );
+}
+
+QTEST_MAIN(tst_QTemporaryFile)
+#include "tst_qtemporaryfile.moc"