Merge pull request #1142 from flowln/better_fs
This commit is contained in:
commit
fe9a4fece4
2
.github/workflows/build.yml
vendored
2
.github/workflows/build.yml
vendored
@ -39,7 +39,7 @@ jobs:
|
|||||||
qt_ver: 6
|
qt_ver: 6
|
||||||
|
|
||||||
- os: macos-12
|
- os: macos-12
|
||||||
macosx_deployment_target: 10.14
|
macosx_deployment_target: 10.15
|
||||||
qt_ver: 6
|
qt_ver: 6
|
||||||
qt_host: mac
|
qt_host: mac
|
||||||
qt_version: '6.3.0'
|
qt_version: '6.3.0'
|
||||||
|
@ -37,6 +37,7 @@
|
|||||||
|
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
|
#include <QDirIterator>
|
||||||
#include <QFile>
|
#include <QFile>
|
||||||
#include <QFileInfo>
|
#include <QFileInfo>
|
||||||
#include <QSaveFile>
|
#include <QSaveFile>
|
||||||
@ -58,6 +59,24 @@
|
|||||||
#include <utime.h>
|
#include <utime.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <filesystem>
|
||||||
|
|
||||||
|
#if defined Q_OS_WIN32
|
||||||
|
|
||||||
|
std::wstring toStdString(QString s)
|
||||||
|
{
|
||||||
|
return s.toStdWString();
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
std::string toStdString(QString s)
|
||||||
|
{
|
||||||
|
return s.toStdString();
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
namespace FS {
|
namespace FS {
|
||||||
|
|
||||||
void ensureExists(const QDir& dir)
|
void ensureExists(const QDir& dir)
|
||||||
@ -128,6 +147,8 @@ bool ensureFolderPathExists(QString foldernamepath)
|
|||||||
|
|
||||||
bool copy::operator()(const QString& offset)
|
bool copy::operator()(const QString& offset)
|
||||||
{
|
{
|
||||||
|
using copy_opts = std::filesystem::copy_options;
|
||||||
|
|
||||||
// NOTE always deep copy on windows. the alternatives are too messy.
|
// NOTE always deep copy on windows. the alternatives are too messy.
|
||||||
#if defined Q_OS_WIN32
|
#if defined Q_OS_WIN32
|
||||||
m_followSymlinks = true;
|
m_followSymlinks = true;
|
||||||
@ -136,94 +157,53 @@ bool copy::operator()(const QString& offset)
|
|||||||
auto src = PathCombine(m_src.absolutePath(), offset);
|
auto src = PathCombine(m_src.absolutePath(), offset);
|
||||||
auto dst = PathCombine(m_dst.absolutePath(), offset);
|
auto dst = PathCombine(m_dst.absolutePath(), offset);
|
||||||
|
|
||||||
QFileInfo currentSrc(src);
|
std::error_code err;
|
||||||
if (!currentSrc.exists())
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (!m_followSymlinks && currentSrc.isSymLink()) {
|
std::filesystem::copy_options opt = copy_opts::none;
|
||||||
qDebug() << "creating symlink" << src << " - " << dst;
|
|
||||||
if (!ensureFilePathExists(dst)) {
|
// The default behavior is to follow symlinks
|
||||||
qWarning() << "Cannot create path!";
|
if (!m_followSymlinks)
|
||||||
return false;
|
opt |= copy_opts::copy_symlinks;
|
||||||
}
|
|
||||||
return QFile::link(currentSrc.symLinkTarget(), dst);
|
|
||||||
} else if (currentSrc.isFile()) {
|
// We can't use copy_opts::recursive because we need to take into account the
|
||||||
qDebug() << "copying file" << src << " - " << dst;
|
// blacklisted paths, so we iterate over the source directory, and if there's no blacklist
|
||||||
if (!ensureFilePathExists(dst)) {
|
// match, we copy the file.
|
||||||
qWarning() << "Cannot create path!";
|
QDir src_dir(src);
|
||||||
return false;
|
QDirIterator source_it(src, QDir::Filter::Files, QDirIterator::Subdirectories);
|
||||||
}
|
|
||||||
return QFile::copy(src, dst);
|
while (source_it.hasNext()) {
|
||||||
} else if (currentSrc.isDir()) {
|
auto src_path = source_it.next();
|
||||||
qDebug() << "recursing" << offset;
|
auto relative_path = src_dir.relativeFilePath(src_path);
|
||||||
if (!ensureFolderPathExists(dst)) {
|
|
||||||
qWarning() << "Cannot create path!";
|
if (m_blacklist && m_blacklist->matches(relative_path))
|
||||||
return false;
|
|
||||||
}
|
|
||||||
QDir currentDir(src);
|
|
||||||
for (auto& f : currentDir.entryList(QDir::Files | QDir::Dirs | QDir::NoDotAndDotDot | QDir::Hidden | QDir::System)) {
|
|
||||||
auto inner_offset = PathCombine(offset, f);
|
|
||||||
// ignore and skip stuff that matches the blacklist.
|
|
||||||
if (m_blacklist && m_blacklist->matches(inner_offset)) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
|
||||||
if (!operator()(inner_offset)) {
|
auto dst_path = PathCombine(dst, relative_path);
|
||||||
qWarning() << "Failed to copy" << inner_offset;
|
ensureFilePathExists(dst_path);
|
||||||
return false;
|
|
||||||
|
std::filesystem::copy(toStdString(src_path), toStdString(dst_path), opt, err);
|
||||||
|
if (err) {
|
||||||
|
qWarning() << "Failed to copy files:" << QString::fromStdString(err.message());
|
||||||
|
qDebug() << "Source file:" << src_path;
|
||||||
|
qDebug() << "Destination file:" << dst_path;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
qCritical() << "Copy ERROR: Unknown filesystem object:" << src;
|
return err.value() == 0;
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool deletePath(QString path)
|
bool deletePath(QString path)
|
||||||
{
|
{
|
||||||
bool OK = true;
|
std::error_code err;
|
||||||
QFileInfo finfo(path);
|
|
||||||
if (finfo.isFile()) {
|
std::filesystem::remove_all(toStdString(path), err);
|
||||||
return QFile::remove(path);
|
|
||||||
|
if (err) {
|
||||||
|
qWarning() << "Failed to remove files:" << QString::fromStdString(err.message());
|
||||||
}
|
}
|
||||||
|
|
||||||
QDir dir(path);
|
return err.value() == 0;
|
||||||
|
|
||||||
if (!dir.exists()) {
|
|
||||||
return OK;
|
|
||||||
}
|
|
||||||
auto allEntries = dir.entryInfoList(QDir::NoDotAndDotDot | QDir::System | QDir::Hidden | QDir::AllDirs | QDir::Files, QDir::DirsFirst);
|
|
||||||
|
|
||||||
for (auto& info : allEntries) {
|
|
||||||
#if defined Q_OS_WIN32
|
|
||||||
QString nativePath = QDir::toNativeSeparators(info.absoluteFilePath());
|
|
||||||
auto wString = nativePath.toStdWString();
|
|
||||||
DWORD dwAttrs = GetFileAttributesW(wString.c_str());
|
|
||||||
// Windows: check for junctions, reparse points and other nasty things of that sort
|
|
||||||
if (dwAttrs & FILE_ATTRIBUTE_REPARSE_POINT) {
|
|
||||||
if (info.isFile()) {
|
|
||||||
OK &= QFile::remove(info.absoluteFilePath());
|
|
||||||
} else if (info.isDir()) {
|
|
||||||
OK &= dir.rmdir(info.absoluteFilePath());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
// We do not trust Qt with reparse points, but do trust it with unix symlinks.
|
|
||||||
if (info.isSymLink()) {
|
|
||||||
OK &= QFile::remove(info.absoluteFilePath());
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
else if (info.isDir()) {
|
|
||||||
OK &= deletePath(info.absoluteFilePath());
|
|
||||||
} else if (info.isFile()) {
|
|
||||||
OK &= QFile::remove(info.absoluteFilePath());
|
|
||||||
} else {
|
|
||||||
OK = false;
|
|
||||||
qCritical() << "Delete ERROR: Unknown filesystem object:" << info.absoluteFilePath();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
OK &= dir.rmdir(dir.absolutePath());
|
|
||||||
return OK;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool trash(QString path, QString *pathInTrash = nullptr)
|
bool trash(QString path, QString *pathInTrash = nullptr)
|
||||||
@ -316,8 +296,7 @@ QString DirNameFromString(QString string, QString inDir)
|
|||||||
if (num == 0) {
|
if (num == 0) {
|
||||||
dirName = baseName;
|
dirName = baseName;
|
||||||
} else {
|
} else {
|
||||||
dirName = baseName + QString::number(num);
|
dirName = baseName + "(" + QString::number(num) + ")";
|
||||||
;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// If it's over 9000
|
// If it's over 9000
|
||||||
@ -336,50 +315,6 @@ bool checkProblemticPathJava(QDir folder)
|
|||||||
return pathfoldername.contains("!", Qt::CaseInsensitive);
|
return pathfoldername.contains("!", Qt::CaseInsensitive);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Win32 crap
|
|
||||||
#ifdef Q_OS_WIN
|
|
||||||
|
|
||||||
bool called_coinit = false;
|
|
||||||
|
|
||||||
HRESULT CreateLink(LPCCH linkPath, LPCWSTR targetPath, LPCWSTR args)
|
|
||||||
{
|
|
||||||
HRESULT hres;
|
|
||||||
|
|
||||||
if (!called_coinit) {
|
|
||||||
hres = CoInitialize(NULL);
|
|
||||||
called_coinit = true;
|
|
||||||
|
|
||||||
if (!SUCCEEDED(hres)) {
|
|
||||||
qWarning("Failed to initialize COM. Error 0x%08lX", hres);
|
|
||||||
return hres;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
IShellLink* link;
|
|
||||||
hres = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_IShellLink, (LPVOID*)&link);
|
|
||||||
|
|
||||||
if (SUCCEEDED(hres)) {
|
|
||||||
IPersistFile* persistFile;
|
|
||||||
|
|
||||||
link->SetPath(targetPath);
|
|
||||||
link->SetArguments(args);
|
|
||||||
|
|
||||||
hres = link->QueryInterface(IID_IPersistFile, (LPVOID*)&persistFile);
|
|
||||||
if (SUCCEEDED(hres)) {
|
|
||||||
WCHAR wstr[MAX_PATH];
|
|
||||||
|
|
||||||
MultiByteToWideChar(CP_ACP, 0, linkPath, -1, wstr, MAX_PATH);
|
|
||||||
|
|
||||||
hres = persistFile->Save(wstr, TRUE);
|
|
||||||
persistFile->Release();
|
|
||||||
}
|
|
||||||
link->Release();
|
|
||||||
}
|
|
||||||
return hres;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
QString getDesktopDir()
|
QString getDesktopDir()
|
||||||
{
|
{
|
||||||
return QStandardPaths::writableLocation(QStandardPaths::DesktopLocation);
|
return QStandardPaths::writableLocation(QStandardPaths::DesktopLocation);
|
||||||
@ -439,47 +374,24 @@ bool createShortCut(QString location, QString dest, QStringList args, QString na
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
QStringList listFolderPaths(QDir root)
|
|
||||||
{
|
|
||||||
auto createAbsPath = [](QFileInfo const& entry) { return FS::PathCombine(entry.path(), entry.fileName()); };
|
|
||||||
|
|
||||||
QStringList entries;
|
|
||||||
|
|
||||||
root.refresh();
|
|
||||||
for (auto entry : root.entryInfoList(QDir::Filter::Files)) {
|
|
||||||
entries.append(createAbsPath(entry));
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto entry : root.entryInfoList(QDir::Filter::AllDirs | QDir::Filter::NoDotAndDotDot)) {
|
|
||||||
entries.append(listFolderPaths(createAbsPath(entry)));
|
|
||||||
}
|
|
||||||
|
|
||||||
return entries;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool overrideFolder(QString overwritten_path, QString override_path)
|
bool overrideFolder(QString overwritten_path, QString override_path)
|
||||||
{
|
{
|
||||||
|
using copy_opts = std::filesystem::copy_options;
|
||||||
|
|
||||||
if (!FS::ensureFolderPathExists(overwritten_path))
|
if (!FS::ensureFolderPathExists(overwritten_path))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
QStringList paths_to_override;
|
std::error_code err;
|
||||||
QDir root_override (override_path);
|
std::filesystem::copy_options opt = copy_opts::recursive | copy_opts::overwrite_existing;
|
||||||
for (auto file : listFolderPaths(root_override)) {
|
|
||||||
QString destination = file;
|
|
||||||
destination.replace(override_path, overwritten_path);
|
|
||||||
ensureFilePathExists(destination);
|
|
||||||
|
|
||||||
qDebug() << QString("Applying override %1 in %2").arg(file, destination);
|
std::filesystem::copy(toStdString(override_path), toStdString(overwritten_path), opt, err);
|
||||||
|
|
||||||
if (QFile::exists(destination))
|
if (err) {
|
||||||
QFile::remove(destination);
|
qCritical() << QString("Failed to apply override from %1 to %2").arg(override_path, overwritten_path);
|
||||||
if (!QFile::rename(file, destination)) {
|
qCritical() << "Reason:" << QString::fromStdString(err.message());
|
||||||
qCritical() << QString("Failed to apply override from %1 to %2").arg(file, destination);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return err.value() == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -4,6 +4,8 @@
|
|||||||
|
|
||||||
#include <FileSystem.h>
|
#include <FileSystem.h>
|
||||||
|
|
||||||
|
#include <pathmatcher/RegexpMatcher.h>
|
||||||
|
|
||||||
class FileSystemTest : public QObject
|
class FileSystemTest : public QObject
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
@ -111,6 +113,40 @@ slots:
|
|||||||
f();
|
f();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void test_copy_with_blacklist()
|
||||||
|
{
|
||||||
|
QString folder = QFINDTESTDATA("testdata/FileSystem/test_folder");
|
||||||
|
auto f = [&folder]()
|
||||||
|
{
|
||||||
|
QTemporaryDir tempDir;
|
||||||
|
tempDir.setAutoRemove(true);
|
||||||
|
qDebug() << "From:" << folder << "To:" << tempDir.path();
|
||||||
|
|
||||||
|
QDir target_dir(FS::PathCombine(tempDir.path(), "test_folder"));
|
||||||
|
qDebug() << tempDir.path();
|
||||||
|
qDebug() << target_dir.path();
|
||||||
|
FS::copy c(folder, target_dir.path());
|
||||||
|
c.blacklist(new RegexpMatcher("[.]?mcmeta"));
|
||||||
|
c();
|
||||||
|
|
||||||
|
for(auto entry: target_dir.entryList())
|
||||||
|
{
|
||||||
|
qDebug() << entry;
|
||||||
|
}
|
||||||
|
QVERIFY(!target_dir.entryList().contains("pack.mcmeta"));
|
||||||
|
QVERIFY(target_dir.entryList().contains("assets"));
|
||||||
|
};
|
||||||
|
|
||||||
|
// first try variant without trailing /
|
||||||
|
QVERIFY(!folder.endsWith('/'));
|
||||||
|
f();
|
||||||
|
|
||||||
|
// then variant with trailing /
|
||||||
|
folder.append('/');
|
||||||
|
QVERIFY(folder.endsWith('/'));
|
||||||
|
f();
|
||||||
|
}
|
||||||
|
|
||||||
void test_getDesktop()
|
void test_getDesktop()
|
||||||
{
|
{
|
||||||
QCOMPARE(FS::getDesktopDir(), QStandardPaths::writableLocation(QStandardPaths::DesktopLocation));
|
QCOMPARE(FS::getDesktopDir(), QStandardPaths::writableLocation(QStandardPaths::DesktopLocation));
|
||||||
|
Loading…
x
Reference in New Issue
Block a user