/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Copyright (C) 2019 Michael Zanetti * * * * This file is part of nymea. * * * * nymea 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 2 of the License. * * * * nymea 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 nymea. If not, see . * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ #ifndef JSONHANDLER_H #define JSONHANDLER_H #include #include #include #include #include #include #include "jsonreply.h" class JsonHandler : public QObject { Q_OBJECT public: enum BasicType { Uuid, String, StringList, Int, Uint, Double, Bool, Variant, Color, Time, Object }; Q_ENUM(BasicType) explicit JsonHandler(QObject *parent = nullptr); virtual ~JsonHandler() = default; virtual QString name() const = 0; QVariantMap jsonEnums() const; QVariantMap jsonFlags() const; QVariantMap jsonObjects() const; QVariantMap jsonMethods() const; QVariantMap jsonNotifications() const; template static QString enumRef(); template static QString objectRef(); static QString objectRef(const QString &objectName); template static QString enumValueName(T value); template static T enumNameToValue(const QString &name); static BasicType variantTypeToBasicType(QVariant::Type variantType); static QVariant::Type basicTypeToVariantType(BasicType basicType); template QVariant pack(const T &value) const; template T unpack(const QVariantMap &map) const; protected: template void registerEnum(); template void registerEnum(); template void registerObject(); template void registerObject(); void registerObject(const QString &name, const QVariantMap &object); void registerMethod(const QString &name, const QString &description, const QVariantMap ¶ms, const QVariantMap &returns, bool deprecated = false); void registerNotification(const QString &name, const QString &description, const QVariantMap ¶ms, bool deprecated = false); JsonReply *createReply(const QVariantMap &data) const; JsonReply *createAsyncReply(const QString &method) const; private: QVariant pack(const QMetaObject &metaObject, const void *gadget) const; private: QVariantMap m_enums; QHash m_metaEnums; QVariantMap m_flags; QHash m_metaFlags; QHash m_flagsEnums; QVariantMap m_objects; QHash m_metaObjects; QHash m_listMetaObjects; QHash m_listEntryTypes; QVariantMap m_methods; QVariantMap m_notifications; }; Q_DECLARE_METATYPE(QVariant::Type) template void JsonHandler::registerEnum() { QMetaEnum metaEnum = QMetaEnum::fromType(); QStringList values; for (int i = 0; i < metaEnum.keyCount(); i++) { values << metaEnum.key(i); } m_enums.insert(metaEnum.name(), values); m_metaEnums.insert(metaEnum.name(), metaEnum); } template void JsonHandler::registerEnum() { registerEnum(); QMetaEnum metaEnum = QMetaEnum::fromType(); QMetaEnum metaFlags = QMetaEnum::fromType(); m_metaFlags.insert(metaFlags.name(), metaFlags); m_flagsEnums.insert(metaFlags.name(), metaEnum.name()); m_flags.insert(metaFlags.name(), QVariantList() << QString("$ref:%1").arg(metaEnum.name())); } template void JsonHandler::registerObject() { qRegisterMetaType(); QMetaObject metaObject = ObjectType::staticMetaObject; QString className = QString(metaObject.className()).split("::").last(); QVariantMap description; for (int i = 0; i < metaObject.propertyCount(); i++) { QMetaProperty metaProperty = metaObject.property(i); QString name = metaProperty.name(); if (name == "objectName") { continue; // Skip QObject's objectName property } if (metaProperty.isUser()) { name.prepend("o:"); } QVariant typeName; // qWarning() << ".-.-.-.-.-" << metaProperty.name() << metaProperty.type() << metaProperty.typeName(); if (metaProperty.type() == QVariant::UserType) { if (metaProperty.typeName() == QStringLiteral("QVariant::Type")) { typeName = QString("$ref:BasicType"); } else if (QString(metaProperty.typeName()).startsWith("QList")) { QString elementType = QString(metaProperty.typeName()).remove("QList<").remove(">"); QVariant::Type variantType = QVariant::nameToType(elementType.toUtf8()); typeName = QVariantList() << enumValueName(variantTypeToBasicType(variantType)); } else { typeName = QString("$ref:%1").arg(QString(metaProperty.typeName()).split("::").last()); } } else if (metaProperty.isEnumType()) { typeName = QString("$ref:%1").arg(QString(metaProperty.typeName()).split("::").last()); } else if (metaProperty.isFlagType()) { typeName = QVariantList() << "$ref:" + m_flagsEnums.value(metaProperty.name()); } else if (metaProperty.type() == QVariant::List) { typeName = QVariantList() << enumValueName(Variant); } else { typeName = enumValueName(variantTypeToBasicType(metaProperty.type())); } description.insert(name, typeName); } m_objects.insert(className, description); m_metaObjects.insert(className, metaObject); } template void JsonHandler::registerObject() { registerObject(); QMetaObject metaObject = ObjectType::staticMetaObject; QMetaObject listMetaObject = ListType::staticMetaObject; QString listTypeName = QString(listMetaObject.className()).split("::").last(); QString objectTypeName = QString(metaObject.className()).split("::").last(); m_objects.insert(listTypeName, QVariantList() << QVariant(QString("$ref:%1").arg(objectTypeName))); m_metaObjects.insert(listTypeName, listMetaObject); m_listMetaObjects.insert(listTypeName, listMetaObject); m_listEntryTypes.insert(listTypeName, objectTypeName); Q_ASSERT_X(listMetaObject.indexOfProperty("count") >= 0, "JsonHandler", QString("List type %1 does not implement \"count\" property!").arg(listTypeName).toUtf8()); Q_ASSERT_X(listMetaObject.indexOfMethod("get(int)") >= 0, "JsonHandler", QString("List type %1 does not implement \"Q_INVOKABLE QVariant get(int index)\" method!").arg(listTypeName).toUtf8()); } template QString JsonHandler::enumRef() { QMetaEnum metaEnum = QMetaEnum::fromType(); return QString("$ref:%1").arg(metaEnum.name()); } template QString JsonHandler::objectRef() { QMetaObject metaObject = T::staticMetaObject; return QString("$ref:%1").arg(QString(metaObject.className()).split("::").last()); } template QString JsonHandler::enumValueName(T value) { QMetaEnum metaEnum = QMetaEnum::fromType(); return metaEnum.valueToKey(value); } template T JsonHandler::enumNameToValue(const QString &name) { QMetaEnum metaEnum = QMetaEnum::fromType(); return static_cast(metaEnum.keyToValue(name.toUtf8())); } template QVariant JsonHandler::pack(const T &value) const { QMetaObject metaObject = T::staticMetaObject; return pack(metaObject, static_cast(&value)); } template T JsonHandler::unpack(const QVariantMap &map) const { T ret; QMetaObject metaObject = T::staticMetaObject; for (int i = 0; i < metaObject.propertyCount(); i++) { QMetaProperty metaProperty = metaObject.property(i); if (metaProperty.name() == QStringLiteral("objectName")) { continue; } if (!metaProperty.isWritable()) { continue; } if (!metaProperty.isUser()) { Q_ASSERT_X(map.contains(metaProperty.name()), this->metaObject()->className(), QString("Missing property %1 in map.").arg(metaProperty.name()).toUtf8()); } if (map.contains(metaProperty.name())) { // Special treatment for QDateTime (convert from time_t) QVariant variant = map.value(metaProperty.name()); if (metaProperty.type() == QVariant::DateTime) { variant = QDateTime::fromTime_t(variant.toUInt()); } metaProperty.writeOnGadget(&ret, variant); } } return ret; } #endif // JSONHANDLER_H