3b4539de79
Signed-off-by: Sefa Eyeoglu <contact@scrumplex.net>
534 lines
14 KiB
C++
534 lines
14 KiB
C++
// SPDX-License-Identifier: GPL-3.0-only
|
|
/*
|
|
* PolyMC - Minecraft Launcher
|
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, version 3.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*
|
|
* This file incorporates work covered by the following copyright and
|
|
* permission notice:
|
|
*
|
|
* Copyright 2013-2021 MultiMC Contributors
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "FileSystem.h"
|
|
|
|
#include <QDir>
|
|
#include <QFile>
|
|
#include <QSaveFile>
|
|
#include <QFileInfo>
|
|
#include <QDebug>
|
|
#include <QUrl>
|
|
#include <QStandardPaths>
|
|
#include <QTextStream>
|
|
|
|
#if defined Q_OS_WIN32
|
|
#include <windows.h>
|
|
#include <string>
|
|
#include <sys/utime.h>
|
|
#include <winnls.h>
|
|
#include <shobjidl.h>
|
|
#include <objbase.h>
|
|
#include <objidl.h>
|
|
#include <shlguid.h>
|
|
#include <shlobj.h>
|
|
#else
|
|
#include <utime.h>
|
|
#endif
|
|
|
|
namespace FS {
|
|
|
|
void ensureExists(const QDir &dir)
|
|
{
|
|
if (!QDir().mkpath(dir.absolutePath()))
|
|
{
|
|
throw FileSystemException("Unable to create folder " + dir.dirName() + " (" +
|
|
dir.absolutePath() + ")");
|
|
}
|
|
}
|
|
|
|
void write(const QString &filename, const QByteArray &data)
|
|
{
|
|
ensureExists(QFileInfo(filename).dir());
|
|
QSaveFile file(filename);
|
|
if (!file.open(QSaveFile::WriteOnly))
|
|
{
|
|
throw FileSystemException("Couldn't open " + filename + " for writing: " +
|
|
file.errorString());
|
|
}
|
|
if (data.size() != file.write(data))
|
|
{
|
|
throw FileSystemException("Error writing data to " + filename + ": " +
|
|
file.errorString());
|
|
}
|
|
if (!file.commit())
|
|
{
|
|
throw FileSystemException("Error while committing data to " + filename + ": " +
|
|
file.errorString());
|
|
}
|
|
}
|
|
|
|
QByteArray read(const QString &filename)
|
|
{
|
|
QFile file(filename);
|
|
if (!file.open(QFile::ReadOnly))
|
|
{
|
|
throw FileSystemException("Unable to open " + filename + " for reading: " +
|
|
file.errorString());
|
|
}
|
|
const qint64 size = file.size();
|
|
QByteArray data(int(size), 0);
|
|
const qint64 ret = file.read(data.data(), size);
|
|
if (ret == -1 || ret != size)
|
|
{
|
|
throw FileSystemException("Error reading data from " + filename + ": " +
|
|
file.errorString());
|
|
}
|
|
return data;
|
|
}
|
|
|
|
bool updateTimestamp(const QString& filename)
|
|
{
|
|
#ifdef Q_OS_WIN32
|
|
std::wstring filename_utf_16 = filename.toStdWString();
|
|
return (_wutime64(filename_utf_16.c_str(), nullptr) == 0);
|
|
#else
|
|
QByteArray filenameBA = QFile::encodeName(filename);
|
|
return (utime(filenameBA.data(), nullptr) == 0);
|
|
#endif
|
|
}
|
|
|
|
bool ensureFilePathExists(QString filenamepath)
|
|
{
|
|
QFileInfo a(filenamepath);
|
|
QDir dir;
|
|
QString ensuredPath = a.path();
|
|
bool success = dir.mkpath(ensuredPath);
|
|
return success;
|
|
}
|
|
|
|
bool ensureFolderPathExists(QString foldernamepath)
|
|
{
|
|
QFileInfo a(foldernamepath);
|
|
QDir dir;
|
|
QString ensuredPath = a.filePath();
|
|
bool success = dir.mkpath(ensuredPath);
|
|
return success;
|
|
}
|
|
|
|
bool copy::operator()(const QString &offset)
|
|
{
|
|
//NOTE always deep copy on windows. the alternatives are too messy.
|
|
#if defined Q_OS_WIN32
|
|
m_followSymlinks = true;
|
|
#endif
|
|
|
|
auto src = PathCombine(m_src.absolutePath(), offset);
|
|
auto dst = PathCombine(m_dst.absolutePath(), offset);
|
|
|
|
QFileInfo currentSrc(src);
|
|
if (!currentSrc.exists())
|
|
return false;
|
|
|
|
if(!m_followSymlinks && currentSrc.isSymLink())
|
|
{
|
|
qDebug() << "creating symlink" << src << " - " << dst;
|
|
if (!ensureFilePathExists(dst))
|
|
{
|
|
qWarning() << "Cannot create path!";
|
|
return false;
|
|
}
|
|
return QFile::link(currentSrc.symLinkTarget(), dst);
|
|
}
|
|
else if(currentSrc.isFile())
|
|
{
|
|
qDebug() << "copying file" << src << " - " << dst;
|
|
if (!ensureFilePathExists(dst))
|
|
{
|
|
qWarning() << "Cannot create path!";
|
|
return false;
|
|
}
|
|
return QFile::copy(src, dst);
|
|
}
|
|
else if(currentSrc.isDir())
|
|
{
|
|
qDebug() << "recursing" << offset;
|
|
if (!ensureFolderPathExists(dst))
|
|
{
|
|
qWarning() << "Cannot create 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;
|
|
}
|
|
if(!operator()(inner_offset))
|
|
{
|
|
qWarning() << "Failed to copy" << inner_offset;
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
qCritical() << "Copy ERROR: Unknown filesystem object:" << src;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool deletePath(QString path)
|
|
{
|
|
bool OK = true;
|
|
QFileInfo finfo(path);
|
|
if(finfo.isFile()) {
|
|
return QFile::remove(path);
|
|
}
|
|
|
|
QDir dir(path);
|
|
|
|
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;
|
|
}
|
|
|
|
|
|
QString PathCombine(const QString & path1, const QString & path2)
|
|
{
|
|
if(!path1.size())
|
|
return path2;
|
|
if(!path2.size())
|
|
return path1;
|
|
return QDir::cleanPath(path1 + QDir::separator() + path2);
|
|
}
|
|
|
|
QString PathCombine(const QString & path1, const QString & path2, const QString & path3)
|
|
{
|
|
return PathCombine(PathCombine(path1, path2), path3);
|
|
}
|
|
|
|
QString PathCombine(const QString & path1, const QString & path2, const QString & path3, const QString & path4)
|
|
{
|
|
return PathCombine(PathCombine(path1, path2, path3), path4);
|
|
}
|
|
|
|
QString AbsolutePath(QString path)
|
|
{
|
|
return QFileInfo(path).absolutePath();
|
|
}
|
|
|
|
QString ResolveExecutable(QString path)
|
|
{
|
|
if (path.isEmpty())
|
|
{
|
|
return QString();
|
|
}
|
|
if(!path.contains('/'))
|
|
{
|
|
path = QStandardPaths::findExecutable(path);
|
|
}
|
|
QFileInfo pathInfo(path);
|
|
if(!pathInfo.exists() || !pathInfo.isExecutable())
|
|
{
|
|
return QString();
|
|
}
|
|
return pathInfo.absoluteFilePath();
|
|
}
|
|
|
|
/**
|
|
* Normalize path
|
|
*
|
|
* Any paths inside the current folder will be normalized to relative paths (to current)
|
|
* Other paths will be made absolute
|
|
*/
|
|
QString NormalizePath(QString path)
|
|
{
|
|
QDir a = QDir::currentPath();
|
|
QString currentAbsolute = a.absolutePath();
|
|
|
|
QDir b(path);
|
|
QString newAbsolute = b.absolutePath();
|
|
|
|
if (newAbsolute.startsWith(currentAbsolute))
|
|
{
|
|
return a.relativeFilePath(newAbsolute);
|
|
}
|
|
else
|
|
{
|
|
return newAbsolute;
|
|
}
|
|
}
|
|
|
|
QString badFilenameChars = "\"\\/?<>:;*|!+\r\n";
|
|
|
|
QString RemoveInvalidFilenameChars(QString string, QChar replaceWith)
|
|
{
|
|
for (int i = 0; i < string.length(); i++)
|
|
{
|
|
if (badFilenameChars.contains(string[i]))
|
|
{
|
|
string[i] = replaceWith;
|
|
}
|
|
}
|
|
return string;
|
|
}
|
|
|
|
QString DirNameFromString(QString string, QString inDir)
|
|
{
|
|
int num = 0;
|
|
QString baseName = RemoveInvalidFilenameChars(string, '-');
|
|
QString dirName;
|
|
do
|
|
{
|
|
if(num == 0)
|
|
{
|
|
dirName = baseName;
|
|
}
|
|
else
|
|
{
|
|
dirName = baseName + QString::number(num);;
|
|
}
|
|
|
|
// If it's over 9000
|
|
if (num > 9000)
|
|
return "";
|
|
num++;
|
|
} while (QFileInfo(PathCombine(inDir, dirName)).exists());
|
|
return dirName;
|
|
}
|
|
|
|
// Does the folder path contain any '!'? If yes, return true, otherwise false.
|
|
// (This is a problem for Java)
|
|
bool checkProblemticPathJava(QDir folder)
|
|
{
|
|
QString pathfoldername = folder.absolutePath();
|
|
return pathfoldername.contains("!", Qt::CaseInsensitive);
|
|
}
|
|
|
|
// Win32 crap
|
|
#ifdef Q_OS_WIN
|
|
|
|
bool called_coinit = false;
|
|
|
|
HRESULT CreateLink(LPCSTR linkPath, LPCSTR targetPath, LPCSTR 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;
|
|
}
|
|
}
|
|
|
|
IShellLinkA *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()
|
|
{
|
|
return QStandardPaths::writableLocation(QStandardPaths::DesktopLocation);
|
|
}
|
|
|
|
// Cross-platform Shortcut creation
|
|
bool createShortCut(QString location, QString dest, QStringList args, QString name,
|
|
QString icon)
|
|
{
|
|
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
|
location = PathCombine(location, name + ".desktop");
|
|
|
|
QFile f(location);
|
|
f.open(QIODevice::WriteOnly | QIODevice::Text);
|
|
QTextStream stream(&f);
|
|
|
|
QString argstring;
|
|
if (!args.empty())
|
|
argstring = " '" + args.join("' '") + "'";
|
|
|
|
stream << "[Desktop Entry]"
|
|
<< "\n";
|
|
stream << "Type=Application"
|
|
<< "\n";
|
|
stream << "TryExec=" << dest.toLocal8Bit() << "\n";
|
|
stream << "Exec=" << dest.toLocal8Bit() << argstring.toLocal8Bit() << "\n";
|
|
stream << "Name=" << name.toLocal8Bit() << "\n";
|
|
stream << "Icon=" << icon.toLocal8Bit() << "\n";
|
|
|
|
stream.flush();
|
|
f.close();
|
|
|
|
f.setPermissions(f.permissions() | QFileDevice::ExeOwner | QFileDevice::ExeGroup |
|
|
QFileDevice::ExeOther);
|
|
|
|
return true;
|
|
#elif defined Q_OS_WIN
|
|
// TODO: Fix
|
|
// QFile file(PathCombine(location, name + ".lnk"));
|
|
// WCHAR *file_w;
|
|
// WCHAR *dest_w;
|
|
// WCHAR *args_w;
|
|
// file.fileName().toWCharArray(file_w);
|
|
// dest.toWCharArray(dest_w);
|
|
|
|
// QString argStr;
|
|
// for (int i = 0; i < args.count(); i++)
|
|
// {
|
|
// argStr.append(args[i]);
|
|
// argStr.append(" ");
|
|
// }
|
|
// argStr.toWCharArray(args_w);
|
|
|
|
// return SUCCEEDED(CreateLink(file_w, dest_w, args_w));
|
|
return false;
|
|
#else
|
|
qWarning("Desktop Shortcuts not supported on your platform!");
|
|
return false;
|
|
#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)
|
|
{
|
|
if (!FS::ensureFolderPathExists(overwritten_path))
|
|
return false;
|
|
|
|
QStringList paths_to_override;
|
|
QDir root_override (override_path);
|
|
for (auto file : listFolderPaths(root_override)) {
|
|
QString destination = file;
|
|
destination.replace(override_path, overwritten_path);
|
|
|
|
qDebug() << QString("Applying override %1 in %2").arg(file, destination);
|
|
|
|
if (QFile::exists(destination))
|
|
QFile::remove(destination);
|
|
if (!QFile::rename(file, destination)) {
|
|
qCritical() << QString("Failed to apply override from %1 to %2").arg(file, destination);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
}
|