MythTV master
mythhttpmetamethod.cpp
Go to the documentation of this file.
1// Qt
2#include <QDateTime>
3#include <QFileInfo>
4#include <QTimeZone>
5#include <algorithm>
6
7// MythTV
8#include "mythlogging.h"
10
11#define LOC QString("MetaMethod: ")
12
20MythHTTPMetaMethod::MythHTTPMetaMethod(int Index, QMetaMethod& Method, int RequestTypes,
21 const QString& ReturnName, bool Slot)
22 : m_index(Index),
23 m_requestTypes(RequestTypes),
24 m_method(Method)
25{
26 // Static list of unsupported return types (cannot be serialised (QHash) or of no use (pointers))
27 static const std::vector<int> s_invalidTypes =
28 {
29 QMetaType::UnknownType, QMetaType::VoidStar, QMetaType::QObjectStar,
30 QMetaType::QVariantHash, QMetaType::QRect, QMetaType::QRectF,
31 QMetaType::QSize, QMetaType::QSizeF, QMetaType::QLine,
32 QMetaType::QLineF, QMetaType::QPoint, QMetaType::QPointF
33 };
34
35 // Static list of unsupported parameters (all invalid types plus extras)
36 static const std::vector<int> s_invalidParams =
37 {
38 QMetaType::UnknownType, QMetaType::VoidStar, QMetaType::QObjectStar,
39 QMetaType::QVariantHash, QMetaType::QRect, QMetaType::QRectF,
40 QMetaType::QSize, QMetaType::QSizeF, QMetaType::QLine,
41 QMetaType::QLineF, QMetaType::QPoint, QMetaType::QPointF,
42 QMetaType::QVariantMap, QMetaType::QStringList, QMetaType::QVariantList
43 };
44
45 int returntype = Method.returnType();
46
47 // Discard methods with an unsupported return type
48#ifdef __cpp_lib_ranges_contains
49 if (std::ranges::contains(s_invalidTypes, returntype))
50#else
51 if (std::ranges::any_of(s_invalidTypes,
52 [&returntype](int Type) { return Type == returntype; }))
53#endif
54 {
55 LOG(VB_HTTP, LOG_ERR, LOC + QString("Method '%1' has unsupported return type '%2'").arg(Method.name().constData(), Method.typeName()));
56 return;
57 }
58
59 // Warn about complicated methods not supported by QMetaMethod - these will
60 // fail if all arguments are required and used
61 if (Method.parameterCount() > (Q_METAMETHOD_INVOKE_MAX_ARGS - 1))
62 {
63 LOG(VB_HTTP, LOG_WARNING, LOC + QString("Method '%1' takes more than %2 parameters; will probably fail")
64 .arg(Method.name().constData()).arg(Q_METAMETHOD_INVOKE_MAX_ARGS - 1));
65 }
66
67 // Decide on the name of the returned type
68 if (Slot && ValidReturnType(returntype))
69 {
70 // Explicitly set via Q_CLASSINFO (name=XXX)
71 m_returnTypeName = ReturnName;
72 if (m_returnTypeName.isEmpty())
73 {
74 // If this is a user type, we assume its name should be used, otherwise
75 // prefer deduction from methodname
76 if (returntype <= QMetaType::User)
77 if (QString(Method.name()).startsWith(QStringLiteral("Get"), Qt::CaseInsensitive))
78 m_returnTypeName = Method.name().mid(3);
79 // Default to the type name
80 if (m_returnTypeName.isEmpty())
81 m_returnTypeName = Method.typeName();
82 // Duplicated from MythXMLSerialiser
83 if (m_returnTypeName.startsWith("Q"))
85 m_returnTypeName.remove("DTC::");
86 m_returnTypeName.remove(QChar('*'));
87 }
88 }
89
90 // Add method name and return type
91 m_types.emplace_back(returntype >= 0 ? returntype : 0);
92 m_names.emplace_back(Method.name());
93
94 // Add type/value for each method parameter
95 auto names = Method.parameterNames();
96 auto types = Method.parameterTypes();
97
98 // Add type/value for each method parameter
99 for (int i = 0; i < names.size(); ++i)
100 {
101#if QT_VERSION < QT_VERSION_CHECK(6,0,0)
102 int type = QMetaType::type(types[i]);
103#else
104 int type = QMetaType::fromName(types[i]).id();
105#endif
106
107 // Discard methods that use unsupported parameter types.
108 // Note: slots only - these are supportable for signals
109#ifdef __cpp_lib_ranges_contains
110 if (Slot && std::ranges::contains(s_invalidParams, type))
111#else
112 if (Slot && std::ranges::any_of(s_invalidParams,
113 [&type](int Type) { return type == Type; }))
114#endif
115 {
116 LOG(VB_GENERAL, LOG_ERR, LOC + QString("Method '%1' has unsupported parameter type '%2' (%3)")
117 .arg(Method.name().constData(), types[i].constData()).arg(type));
118 return;
119 }
120
121 m_names.emplace_back(names[i]);
122 m_types.emplace_back(type);
123 }
124
125 m_valid = true;
126}
127
128HTTPMethodPtr MythHTTPMetaMethod::Create(int Index, QMetaMethod &Method, int RequestTypes,
129 const QString &ReturnName, bool Slot)
130{
131 HTTPMethodPtr result =
132 std::shared_ptr<MythHTTPMetaMethod>(new MythHTTPMetaMethod(Index, Method, RequestTypes, ReturnName, Slot));
133 if (result->m_valid)
134 return result;
135 return nullptr;
136}
137
143void* MythHTTPMetaMethod::CreateParameter(void* Parameter, int Type, const QString& Value)
144{
145#if QT_VERSION < QT_VERSION_CHECK(6,0,0)
146 QByteArray typeName = QMetaType::typeName(Type);
147#else
148 QByteArray typeName = QMetaType(Type).name();
149#endif
150
151 // Enum types
152#if QT_VERSION < QT_VERSION_CHECK(6,0,0)
153 auto typeflags = QMetaType::typeFlags(Type);
154#else
155 auto typeflags = QMetaType(Type).flags();
156#endif
157 if ((typeflags & QMetaType::IsEnumeration) == QMetaType::IsEnumeration)
158 {
159 // QMetaEnum::keyToValue will return -1 for an unrecognised enumerant, so
160 // default to -1 for all error cases
161 int value = -1;
162 if (int index = typeName.lastIndexOf("::" ); index > -1)
163 {
164 QString enumname = typeName.mid(index + 2);
165#if QT_VERSION < QT_VERSION_CHECK(6,0,0)
166 const auto * metaobject = QMetaType::metaObjectForType(Type);
167#else
168 const auto * metaobject = QMetaType(Type).metaObject();
169#endif
170 if (metaobject)
171 {
172 int enumindex = metaobject->indexOfEnumerator(enumname.toUtf8().constData());
173 if (enumindex >= 0)
174 {
175 QMetaEnum metaEnum = metaobject->enumerator(enumindex);
176 value = metaEnum.keyToValue(Value.toUtf8().constData());
177 }
178 }
179 }
180 *(static_cast<int*>(Parameter)) = value;
181 return Parameter;
182 }
183
184 // Handle parameters of type std::chrono::seconds
185 if (typeName == "std::chrono::seconds")
186 {
187 *(static_cast<std::chrono::seconds*>(Parameter)) = std::chrono::seconds(Value.toInt());
188 return Parameter;
189 }
190
191 switch (Type)
192 {
193 case QMetaType::QVariant : *(static_cast<QVariant *>(Parameter)) = QVariant(Value); break;
194 case QMetaType::Bool : *(static_cast<bool *>(Parameter)) = ToBool(Value ); break;
195 case QMetaType::Char : *(static_cast<char *>(Parameter)) = (Value.length() > 0) ? Value.at(0).toLatin1() : 0; break;
196 case QMetaType::UChar : *(static_cast<unsigned char*>(Parameter)) = (Value.length() > 0) ? static_cast<unsigned char>(Value.at(0).toLatin1()) : 0; break;
197 case QMetaType::QChar : *(static_cast<QChar *>(Parameter)) = (Value.length() > 0) ? Value.at(0) : QChar(0); break;
198 case QMetaType::Short : *(static_cast<short *>(Parameter)) = Value.toShort(); break;
199 case QMetaType::UShort : *(static_cast<ushort *>(Parameter)) = Value.toUShort(); break;
200 case QMetaType::Int : *(static_cast<int *>(Parameter)) = Value.toInt(); break;
201 case QMetaType::UInt : *(static_cast<uint *>(Parameter)) = Value.toUInt(); break;
202 case QMetaType::Long : *(static_cast<long *>(Parameter)) = Value.toLong(); break;
203 case QMetaType::ULong : *(static_cast<ulong *>(Parameter)) = Value.toULong(); break;
204 case QMetaType::LongLong : *(static_cast<qlonglong *>(Parameter)) = Value.toLongLong(); break;
205 case QMetaType::ULongLong : *(static_cast<qulonglong *>(Parameter)) = Value.toULongLong(); break;
206 case QMetaType::Double : *(static_cast<double *>(Parameter)) = Value.toDouble(); break;
207 case QMetaType::Float : *(static_cast<float *>(Parameter)) = Value.toFloat(); break;
208 case QMetaType::QString : *(static_cast<QString *>(Parameter)) = Value; break;
209 case QMetaType::QByteArray: *(static_cast<QByteArray *>(Parameter)) = Value.toUtf8(); break;
210 case QMetaType::QTime : *(static_cast<QTime *>(Parameter)) = QTime::fromString(Value, Qt::ISODate ); break;
211 case QMetaType::QDate : *(static_cast<QDate *>(Parameter)) = QDate::fromString(Value, Qt::ISODate ); break;
212 case QMetaType::QDateTime :
213 {
214 QDateTime dt = QDateTime::fromString(Value, Qt::ISODate);
215#if QT_VERSION < QT_VERSION_CHECK(6,5,0)
216 dt.setTimeSpec(Qt::UTC);
217#else
218 dt.setTimeZone(QTimeZone(QTimeZone::UTC));
219#endif
220 *(static_cast<QDateTime*>(Parameter)) = dt;
221 break;
222 }
223 default:
224 LOG(VB_GENERAL, LOG_WARNING, LOC + QString("Unknown QMetaType:%1 %2").arg(Type).arg(QString(typeName)));
225 break;
226 }
227 return Parameter;
228}
229
230QVariant MythHTTPMetaMethod::CreateReturnValue(int Type, void* Value)
231{
232 if (!(ValidReturnType(Type)))
233 return {};
234
235 // This assumes any user type will be derived from QObject...
236 // (Exception for QFileInfo)
237 if (
238#if QT_VERSION < QT_VERSION_CHECK(6,0,0)
239 Type == QMetaType::type("QFileInfo")
240#else
241 Type == QMetaType::fromName("QFileInfo").id()
242#endif
243 )
244 return QVariant::fromValue<QFileInfo>(*(static_cast<QFileInfo*>(Value)));
245
246 if (Type > QMetaType::User)
247 {
248 QObject* object = *(static_cast<QObject**>(Value));
249 return QVariant::fromValue<QObject*>(object);
250 }
251
252#if QT_VERSION < QT_VERSION_CHECK(6,0,0)
253 return {Type, Value};
254#else
255 return QVariant(QMetaType(Type), Value);
256#endif
257}
std::vector< int > m_types
static void * CreateParameter(void *Parameter, int Type, const QString &Value)
Populate the QMetaType object referenced by Parameter with Value.
static QVariant CreateReturnValue(int Type, void *Value)
static bool ToBool(const QString &Value)
static HTTPMethodPtr Create(int Index, QMetaMethod &Method, int RequestTypes, const QString &ReturnName={}, bool Slot=true)
std::vector< QString > m_names
MythHTTPMetaMethod(int Index, QMetaMethod &Method, int RequestTypes, const QString &ReturnName, bool Slot)
static bool ValidReturnType(int Type)
unsigned int uint
Definition: compat.h:60
static const struct wl_interface * types[]
#define LOC
std::shared_ptr< MythHTTPMetaMethod > HTTPMethodPtr
#define LOG(_MASK_, _LEVEL_, _QSTRING_)
Definition: mythlogging.h:39
@ ISODate
Default UTC.
Definition: mythdate.h:17
QDateTime fromString(const QString &dtstr)
Converts kFilename && kISODate formats to QDateTime.
Definition: mythdate.cpp:39