tests/auto/qtemporaryfile/tst_qtemporaryfile.cpp
changeset 0 1918ee327afb
child 4 3b1da2848fc7
--- /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"