Skip to content
Snippets Groups Projects
xdgurl.cpp 9.65 KiB
Newer Older
  • Learn to ignore specific revisions
  • akiraohgaki's avatar
    akiraohgaki committed
    #include <QDebug>
    
    akiraohgaki's avatar
    akiraohgaki committed
    #include <QUrl>
    #include <QUrlQuery>
    
    akiraohgaki's avatar
    akiraohgaki committed
    #include <QTemporaryFile>
    
    #include <QMimeDatabase>
    
    akiraohgaki's avatar
    akiraohgaki committed
    #include <QNetworkReply>
    
    akiraohgaki's avatar
    akiraohgaki committed
    
    #include "../core/config.h"
    #include "../core/network.h"
    #include "../utility/file.h"
    #include "../utility/json.h"
    
    akiraohgaki's avatar
    akiraohgaki committed
    #include "../utility/package.h"
    
    akiraohgaki's avatar
    akiraohgaki committed
    
    #include "xdgurl.h"
    
    namespace Handlers {
    
    
    akiraohgaki's avatar
    akiraohgaki committed
    XdgUrl::XdgUrl(const QString &xdgUrl, Core::Config *appConfig, Core::Config *userConfig, Core::Network *asyncNetwork, QObject *parent) :
        QObject(parent), _xdgUrl(xdgUrl), _appConfig(appConfig), _userConfig(userConfig), _asyncNetwork(asyncNetwork)
    
    akiraohgaki's avatar
    akiraohgaki committed
    {
    
    akiraohgaki's avatar
    akiraohgaki committed
        _metadata = _parse();
    
    akiraohgaki's avatar
    akiraohgaki committed
        _destinations = _loadDestinations();
    
    akiraohgaki's avatar
    akiraohgaki committed
    }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
    QJsonObject XdgUrl::_parse()
    {
    
    akiraohgaki's avatar
    akiraohgaki committed
        QUrl url(_xdgUrl);
        QUrlQuery query(url);
    
    akiraohgaki's avatar
    akiraohgaki committed
        QJsonObject metadata;
    
    akiraohgaki's avatar
    akiraohgaki committed
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        metadata["scheme"] = QString("xdg");
        metadata["command"] = QString("download");
        metadata["url"] = QString("");
        metadata["type"] = QString("downloads");
        metadata["filename"] = QString("");
    
    akiraohgaki's avatar
    akiraohgaki committed
    
        if (!url.scheme().isEmpty()) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            metadata["scheme"] = url.scheme();
    
    akiraohgaki's avatar
    akiraohgaki committed
        }
    
        if (!url.host().isEmpty()) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            metadata["command"] = url.host();
    
    akiraohgaki's avatar
    akiraohgaki committed
        }
    
        if (query.hasQueryItem("url") && !query.queryItemValue("url").isEmpty()) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            metadata["url"] = query.queryItemValue("url", QUrl::FullyDecoded);
    
    akiraohgaki's avatar
    akiraohgaki committed
        }
    
        if (query.hasQueryItem("type") && !query.queryItemValue("type").isEmpty()) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            metadata["type"] = query.queryItemValue("type", QUrl::FullyDecoded);
    
    akiraohgaki's avatar
    akiraohgaki committed
        }
    
        if (query.hasQueryItem("filename") && !query.queryItemValue("filename").isEmpty()) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            metadata["filename"] = QUrl(query.queryItemValue("filename", QUrl::FullyDecoded)).fileName();
    
    akiraohgaki's avatar
    akiraohgaki committed
        }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        if (!metadata["url"].toString().isEmpty() && metadata["filename"].toString().isEmpty()) {
            metadata["filename"] = QUrl(metadata["url"].toString()).fileName();
    
    akiraohgaki's avatar
    akiraohgaki committed
        }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        return metadata;
    
    akiraohgaki's avatar
    akiraohgaki committed
    QString XdgUrl::_convertPathString(const QString &path)
    {
        QString newPath = path;
    
    akiraohgaki's avatar
    akiraohgaki committed
        if (newPath.contains("$HOME")) {
            newPath.replace("$HOME", Utility::File::homePath());
        }
        else if (newPath.contains("$XDG_DATA")) {
            newPath.replace("$XDG_DATA", Utility::File::xdgDataHomePath());
        }
        else if (newPath.contains("$KDE_DATA")) {
            newPath.replace("$KDE_DATA", Utility::File::kdeDataHomePath());
        }
    
    akiraohgaki's avatar
    akiraohgaki committed
        return newPath;
    }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
    QJsonObject XdgUrl::_loadDestinations()
    
    {
        QJsonObject destinations;
        QJsonObject appConfigDestinations = _appConfig->get("destinations");
        QJsonObject appConfigDestinationsAlias = _appConfig->get("destinations_alias");
    
    akiraohgaki's avatar
    akiraohgaki committed
        QJsonObject userConfigDestinations = _userConfig->get("destinations");
        QJsonObject userConfigDestinationsAlias = _userConfig->get("destinations_alias");
    
    
        foreach (const QString key, appConfigDestinations.keys()) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            destinations[key] = _convertPathString(appConfigDestinations[key].toString());
    
        }
    
        foreach (const QString key, appConfigDestinationsAlias.keys()) {
            QString value = appConfigDestinationsAlias[key].toString();
            if (destinations.contains(value)) {
    
    akiraohgaki's avatar
    akiraohgaki committed
                destinations[key] = destinations.value(value);
            }
        }
    
        if (!userConfigDestinations.isEmpty()) {
            foreach (const QString key, userConfigDestinations.keys()) {
                destinations[key] = _convertPathString(userConfigDestinations[key].toString());
            }
        }
    
        if (!userConfigDestinationsAlias.isEmpty()) {
            foreach (const QString key, userConfigDestinationsAlias.keys()) {
                QString value = userConfigDestinationsAlias[key].toString();
                if (destinations.contains(value)) {
                    destinations[key] = destinations.value(value);
                }
    
    akiraohgaki's avatar
    akiraohgaki committed
    void XdgUrl::_saveDownloadedFile(QNetworkReply *reply)
    
    akiraohgaki's avatar
    akiraohgaki committed
    {
    
    akiraohgaki's avatar
    akiraohgaki committed
        QJsonObject result;
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        QTemporaryFile temporaryFile;
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        if (!temporaryFile.open() || temporaryFile.write(reply->readAll()) == -1) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            result["error"] = QString("save_error");
            emit finished(Utility::Json::convertObjToStr(result));
            return;
        }
    
        QMimeDatabase mimeDb;
        QString mimeType = mimeDb.mimeTypeForFile(temporaryFile.fileName()).name();
    
        if (mimeType == "text/html" || mimeType == "application/xhtml+xml") {
            result["error"] = QString("filetype_error");
            emit finished(Utility::Json::convertObjToStr(result));
            return;
        }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        QString type = _metadata["type"].toString();
        QString destination = _destinations[type].toString();
    
    akiraohgaki's avatar
    akiraohgaki committed
        QString path = destination + "/" + _metadata["filename"].toString();
    
        Utility::File::makeDir(destination);
        Utility::File::remove(path); // Remove previous downloaded file
    
        if (!temporaryFile.copy(path)) {
            result["error"] = QString("save_error");
            emit finished(Utility::Json::convertObjToStr(result));
            return;
        }
    
        result["success"] = QString("download_success");
        result["destination"] = destination;
    
    akiraohgaki's avatar
    akiraohgaki committed
        result["path"] = path;
    
    akiraohgaki's avatar
    akiraohgaki committed
        emit finished(Utility::Json::convertObjToStr(result));
    
    akiraohgaki's avatar
    akiraohgaki committed
    void XdgUrl::_installDownloadedFile(QNetworkReply *reply)
    
    akiraohgaki's avatar
    akiraohgaki committed
    {
    
    akiraohgaki's avatar
    akiraohgaki committed
        QJsonObject result;
    
        QTemporaryFile temporaryFile;
    
        if (!temporaryFile.open() || temporaryFile.write(reply->readAll()) == -1) {
            result["error"] = QString("save_error");
            emit finished(Utility::Json::convertObjToStr(result));
            return;
        }
    
        QMimeDatabase mimeDb;
        QString mimeType = mimeDb.mimeTypeForFile(temporaryFile.fileName()).name();
    
        if (mimeType == "text/html" || mimeType == "application/xhtml+xml") {
            result["error"] = QString("filetype_error");
            emit finished(Utility::Json::convertObjToStr(result));
            return;
        }
    
        QString type = _metadata["type"].toString();
        QString destination = _destinations[type].toString();
        QString path = destination + "/" + _metadata["filename"].toString();
    
        Utility::File::makeDir(destination);
        Utility::File::remove(path); // Remove previous downloaded file
    
        if ((type == "plasma_plasmoids" || type == "plasma4_plasmoids" || type == "plasma5_plasmoids")
                && Utility::Package::installPlasmapkg(temporaryFile.fileName(), "plasmoid")) {
            qInfo("The plasmoid has been installed");
        }
        else if ((type == "plasma_look_and_feel" || type == "plasma5_look_and_feel")
                 && Utility::Package::installPlasmapkg(temporaryFile.fileName(), "lookandfeel")) {
            qInfo("The plasma look and feel has been installed");
        }
        else if ((type == "plasma_desktopthemes" || type == "plasma5_desktopthemes")
                 && Utility::Package::installPlasmapkg(temporaryFile.fileName(), "theme")) {
            qInfo("The plasma desktop theme has been installed");
        }
        else if (type == "kwin_effects"
                 && Utility::Package::installPlasmapkg(temporaryFile.fileName(), "kwineffect")) {
            qInfo("The KWin effect has been installed");
        }
        else if (type == "kwin_scripts"
                 && Utility::Package::installPlasmapkg(temporaryFile.fileName(), "kwinscript")) {
            qInfo("The KWin script has been installed");
        }
        else if (type == "kwin_tabbox"
                 && Utility::Package::installPlasmapkg(temporaryFile.fileName(), "windowswitcher")) {
            qInfo("The KWin window switcher has been installed");
        }
        else if (Utility::Package::uncompressArchive(temporaryFile.fileName(), destination)) {
            qInfo("The archive file has been uncompressed into " + destination);
        }
        else if (temporaryFile.copy(path)) {
            qInfo("Saved the file as " + path);
        }
        else {
            result["error"] = QString("install_error");
            emit finished(Utility::Json::convertObjToStr(result));
            return;
        }
    
        result["success"] = QString("install_success");
        result["destination"] = destination;
        result["path"] = path;
        emit finished(Utility::Json::convertObjToStr(result));
    
    akiraohgaki's avatar
    akiraohgaki committed
    /**
     * Private slots
     */
    
    
    akiraohgaki's avatar
    akiraohgaki committed
    void XdgUrl::_downloaded(QNetworkReply *reply)
    
    akiraohgaki's avatar
    akiraohgaki committed
    {
    
    akiraohgaki's avatar
    akiraohgaki committed
        QJsonObject result;
    
    akiraohgaki's avatar
    akiraohgaki committed
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        if (reply->error() != QNetworkReply::NoError) {
            result["error"] = QString("network_error");
            emit finished(Utility::Json::convertObjToStr(result));
            return;
        }
    
        // If the network reply has a refresh header, retry download
        if (reply->hasRawHeader("Refresh")) {
            QString refreshUrl = QString(reply->rawHeader("Refresh")).split("url=").last();
            if (refreshUrl.startsWith("/")) {
    
    akiraohgaki's avatar
    akiraohgaki committed
                refreshUrl = reply->url().authority() + refreshUrl;
    
    akiraohgaki's avatar
    akiraohgaki committed
            }
            _asyncNetwork->get(QUrl(refreshUrl));
            return;
        }
    
        if (_metadata["command"].toString() == "download") {
    
    akiraohgaki's avatar
    akiraohgaki committed
            _saveDownloadedFile(reply);
    
    akiraohgaki's avatar
    akiraohgaki committed
        }
        else if (_metadata["command"].toString() == "install") {
    
    akiraohgaki's avatar
    akiraohgaki committed
            _installDownloadedFile(reply);
    
    akiraohgaki's avatar
    akiraohgaki committed
        }
    
    akiraohgaki's avatar
    akiraohgaki committed
    }
    
    /**
     * Public slots
     */
    
    
    akiraohgaki's avatar
    akiraohgaki committed
    QString XdgUrl::getXdgUrl()
    {
        return _xdgUrl;
    }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
    QString XdgUrl::getMetadata()
    {
        return Utility::Json::convertObjToStr(_metadata);
    }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
    bool XdgUrl::isValid()
    {
        bool isValid = true;
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        if (_metadata["scheme"].toString() != "xdg" && _metadata["scheme"].toString() != "xdgs") {
    
    akiraohgaki's avatar
    akiraohgaki committed
            isValid = false;
        }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        if (_metadata["command"].toString() != "download" && _metadata["command"].toString() != "install") {
    
    akiraohgaki's avatar
    akiraohgaki committed
            isValid = false;
        }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        if (!QUrl(_metadata["url"].toString()).isValid()) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            isValid = false;
        }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        if (!_destinations.contains(_metadata["type"].toString())) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            isValid = false;
        }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
        if (_metadata["filename"].toString().isEmpty()) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            isValid = false;
        }
    
        return isValid;
    }
    
    
    akiraohgaki's avatar
    akiraohgaki committed
    void XdgUrl::process()
    
    akiraohgaki's avatar
    akiraohgaki committed
    {
    
    akiraohgaki's avatar
    akiraohgaki committed
        /**
         * xdgs scheme is a reserved name, so the process of xdgs
         * is the same process of the xdg scheme currently.
         */
    
        if (isValid()) {
    
    akiraohgaki's avatar
    akiraohgaki committed
            connect(_asyncNetwork, &Core::Network::finished, this, &XdgUrl::_downloaded);
    
    akiraohgaki's avatar
    akiraohgaki committed
            _asyncNetwork->get(QUrl(_metadata["url"].toString()));
    
    akiraohgaki's avatar
    akiraohgaki committed
        }
    
    akiraohgaki's avatar
    akiraohgaki committed
    } // namespace Handlers