summaryrefslogtreecommitdiffstats
path: root/common/resource.cpp
blob: 533a1329b4791ca98dbf33f5db64700f64db2491 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
/*
 * Copyright (C) 2014 Aaron Seigo <aseigo@kde.org>
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) version 3, or any
 * later version accepted by the membership of KDE e.V. (or its
 * successor approved by the membership of KDE e.V.), which shall
 * act as a proxy defined in Section 6 of version 3 of the license.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library.  If not, see <http://www.gnu.org/licenses/>.
 */

#include "resource.h"

#include <QCoreApplication>
#include <QDir>
#include <QPluginLoader>
#include <QPointer>

#include "facadefactory.h"
#include "adaptorfactoryregistry.h"

namespace Sink {

Resource::Resource() : QObject(), d(0)
{
    Q_UNUSED(d);
}

Resource::~Resource()
{
    // delete d;
}

void Resource::processCommand(int commandId, const QByteArray &data)
{
    Q_UNUSED(commandId)
    Q_UNUSED(data)
}

void Resource::setLowerBoundRevision(qint64 revision)
{
    Q_UNUSED(revision)
}


class ResourceFactory::Private
{
public:
    static QHash<QString, QPointer<ResourceFactory>> s_loadedFactories;
};

QHash<QString, QPointer<ResourceFactory>> ResourceFactory::Private::s_loadedFactories;

ResourceFactory::ResourceFactory(QObject *parent) : QObject(parent), d(0)
{
    Q_UNUSED(d);
}

ResourceFactory::~ResourceFactory()
{
    // delete d;
}

ResourceFactory *ResourceFactory::load(const QByteArray &resourceName)
{
    ResourceFactory *factory = Private::s_loadedFactories.value(resourceName);
    if (factory) {
        return factory;
    }

    for (auto const &path : QCoreApplication::instance()->libraryPaths()) {
        if (path.endsWith(QLatin1String("plugins"))) {
            QDir pluginDir(path);
            // TODO: centralize this so that it is easy to change centrally
            //      also ref'd in cmake as ${SINK_RESOURCE_PLUGINS_PATH}
            pluginDir.cd(QStringLiteral("sink"));
            pluginDir.cd(QStringLiteral("resources"));

            for (const QString &fileName : pluginDir.entryList(QDir::Files)) {
                const QString path = pluginDir.absoluteFilePath(fileName);
                QPluginLoader loader(path);

                const QString id = loader.metaData()[QStringLiteral("IID")].toString();
                if (id == resourceName) {
                    QObject *object = loader.instance();
                    if (object) {
                        factory = qobject_cast<ResourceFactory *>(object);
                        if (factory) {
                            Private::s_loadedFactories.insert(resourceName, factory);
                            //TODO: Instead of always loading both facades and adaptorfactories into the respective singletons, we could also leave this up to the caller. (ResourceFactory::loadFacades(...))
                            factory->registerFacades(resourceName, FacadeFactory::instance());
                            factory->registerAdaptorFactories(resourceName, AdaptorFactoryRegistry::instance());
                            // TODO: if we need more data on it const QJsonObject json = loader.metaData()[QStringLiteral("MetaData")].toObject();
                            return factory;
                        } else {
                            qWarning() << "Plugin for" << resourceName << "from plugin" << loader.fileName() << "produced the wrong object type:" << object;
                            delete object;
                        }
                    } else {
                        qWarning() << "Could not load factory for" << resourceName << "from plugin" << loader.fileName() << "due to the following error:" << loader.errorString();
                    }
                }
            }
        }
    }

    qWarning() << "Failed to find factory for resource:" << resourceName;
    return nullptr;
}

} // namespace Sink

// Ignore warning I don't know how to fix in a moc file
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wundefined-reinterpret-cast"
#include "moc_resource.cpp"
#pragma clang diagnostic pop