qt5base-lts/tests/manual/filetest/main.cpp
Shawn Rutledge 092ef06e00 Fix directory listing in manual filetest; add recursive find
- QFileInfo::fileName() is not the name of a directory, so we need
  absoluteFilePath() to construct a QDir
- QDir::entryList() returns only the name suffix, not the whole path
- stop at any arbitrary depth, and add a find command which does full
  recursion

Amends 04a5a74685

Change-Id: I9870db092125a797e8b654e98954ac611dde1ab2
Reviewed-by: Volker Hilsheimer <volker.hilsheimer@qt.io>
2022-07-11 14:42:27 +02:00

212 lines
7.0 KiB
C++

// Copyright (C) 2021 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
#include <QDateTime>
#include <QDebug>
#include <QCoreApplication>
#include <QFileInfo>
#include <QFile>
#include <QDir>
#include <iostream>
#include <string>
static const char usage1[] =
"\nTests various file functionality in Qt\n\n"
"Usage: ";
static const char usage2[] =" [KEYWORD] [ARGUMENTS]\n\n"
"Keywords: ls FILES list file information\n"
" find FILES list file information recursively\n"
" stat FILES print detailed file information\n"
" mv SOURCE TARGET rename files using QFile::rename\n"
" cp SOURCE TARGET copy files using QFile::copy\n"
" rm FILE remove file using QFile::remove\n"
" rmr DIR remove directory recursively\n"
" using QDir::removeRecursively\n";
std::ostream &operator<<(std::ostream &o, const QString &str)
{
return o << qPrintable(str);
}
std::ostream &operator<<(std::ostream &o, const QDateTime &dt)
{
if (dt.isValid())
return o << dt.toString(Qt::ISODateWithMs);
return o << '-';
}
static inline std::string permissions(QFile::Permissions permissions)
{
std::string result(10, '-');
if (permissions & QFile::ReadOwner)
result[1] = 'r';
if (permissions & QFile::WriteOwner)
result[2] = 'w';
if (permissions & QFile::ExeOwner)
result[3] = 'x';
if (permissions & QFile::ReadGroup)
result[4] = 'r';
if (permissions & QFile::WriteGroup)
result[5] = 'w';
if (permissions & QFile::ExeGroup)
result[6] = 'x';
if (permissions & QFile::ReadOther)
result[7] = 'r';
if (permissions & QFile::WriteOther)
result[8] = 'w';
if (permissions & QFile::ExeOther)
result[9] = 'x';
return result;
}
static inline std::string permissions(const QFileInfo &fi)
{
std::string result = permissions(fi.permissions());
if (fi.isSymLink())
result[0] = 'l';
else if (fi.isDir())
result[0] = 'd';
return result;
}
static int ls(int argCount, const char **args, int depth = 0, int maxDepth = 1)
{
for (int i = 0 ; i < argCount; ++i) {
const QFileInfo fi(QString::fromLocal8Bit(args[i]));
std::cout << QDir::toNativeSeparators(fi.absoluteFilePath()).toStdString() << ' ' << fi.size()
<< ' ' << permissions(fi);
if (fi.exists())
std::cout << " [exists]";
if (fi.isFile())
std::cout << " [file]";
if (fi.isSymLink()) {
std::cout << " [symlink to "
<< QDir::toNativeSeparators(fi.symLinkTarget()).toStdString() << ']';
}
if (fi.isDir())
std::cout << " [dir]";
std::cout << std::endl;
if (depth < maxDepth && fi.isDir()) {
QDir dir(fi.absoluteFilePath());
const QStringList entries = dir.entryList(QDir::AllEntries | QDir::NoDotAndDotDot);
for (const QString &s : entries) {
QByteArray encoded = QFile::encodeName(dir.filePath(s));
const char *ptr = encoded.constData();
ls(1, &ptr, depth + 1, maxDepth);
}
}
}
return 0;
}
static int stat(int argCount, char **args)
{
for (int i = 0 ; i < argCount; ++i) {
const QFileInfo fi(QFile::decodeName(args[i]));
std::cout << "Name:\t" << fi.fileName() << std::endl;
std::cout << "Path:\t" << QDir::toNativeSeparators(fi.path())
<< " (" << QDir::toNativeSeparators(fi.absolutePath()) << ')' << std::endl;
std::cout << "Size:\t" << fi.size()
<< "\tType: "
<< (fi.isSymLink() && !fi.exists() ? "Broken symlink" :
!fi.exists() ? "Non-existent" :
fi.isSymLink() ? "Symlink to " : "")
<< (!fi.exists() ? "" :
fi.isFile() ? "Regular file" :
fi.isDir() ? "Directory" : "Special node")
<< std::endl;
if (fi.isSymLink())
std::cout << "Target:\t" << fi.symLinkTarget() << std::endl;
std::cout << "Attrs: "
<< (fi.isReadable() ? " readable" : "")
<< (fi.isWritable() ? " writable" : "")
<< (fi.isExecutable() ? " executable" : "")
<< (fi.isHidden() ? " hidden" : "")
<< (fi.isNativePath() ? " nativepath" : "")
<< (fi.isRoot() ? " root" : "")
<< (fi.isBundle() ? " bundle" : "")
<< std::endl;
std::cout << "Mode:\t" << permissions(fi) << std::endl;
std::cout << "Owner:\t" << fi.owner() << " (" << fi.ownerId()
<< ")\tGroup:\t" << fi.group() << " (" << fi.groupId() << ')' << std::endl;
std::cout << "Access:\t" << fi.lastRead() << std::endl;
std::cout << "Birth:\t" << fi.birthTime() << std::endl;
std::cout << "Change:\t" << fi.metadataChangeTime() << std::endl;
std::cout << "Modified: " << fi.lastModified() << std::endl;
}
return 0;
}
static int mv(const char *sourceFileName, const char *targetFileName)
{
QFile sourceFile(QString::fromLocal8Bit(sourceFileName));
if (!sourceFile.rename(QString::fromLocal8Bit(targetFileName))) {
qWarning().nospace() << sourceFile.errorString();
return -1;
}
return 0;
}
static int cp(const char *sourceFileName, const char *targetFileName)
{
QFile sourceFile(QString::fromLocal8Bit(sourceFileName));
if (!sourceFile.copy(QString::fromLocal8Bit(targetFileName))) {
qWarning().nospace() << sourceFile.errorString();
return -1;
}
return 0;
}
static int rm(const char *fileName)
{
QFile file(QString::fromLocal8Bit(fileName));
if (!file.remove()) {
qWarning().nospace() << file.errorString();
return -1;
}
return 0;
}
static int rmr(const char *dirName)
{
QDir dir(QString::fromLocal8Bit(dirName));
if (!dir.removeRecursively()) {
qWarning().nospace() << "Failed to remove " << dir.absolutePath();
return -1;
}
return 0;
}
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
Q_UNUSED(a);
if (argc >= 3 && !qstrcmp(argv[1], "ls"))
return ls(argc -2, const_cast<const char **>(argv + 2));
if (argc >= 3 && !qstrcmp(argv[1], "find"))
return ls(argc -2, const_cast<const char **>(argv + 2), 0, std::numeric_limits<int>::max());
if (argc >= 3 && !qstrcmp(argv[1], "stat"))
return stat(argc -2, argv + 2);
if (argc == 4 && !qstrcmp(argv[1], "mv"))
return mv(argv[2], argv[3]);
if (argc == 4 && !qstrcmp(argv[1], "cp"))
return cp(argv[2], argv[3]);
if (argc == 3 && !qstrcmp(argv[1], "rm"))
return rm(argv[2]);
if (argc == 3 && !qstrcmp(argv[1], "rmr"))
return rmr(argv[2]);
std::cerr << usage1 << argv[0] << usage2;
return 0;
}