QDBus: Add diagnostics reporting interface

Add an abstract class QDBusIntrospection::DiagnosticsReporter
for reporting errors and warnings. Extend QDBusXmlParser
to accept optional diagnostics reporter and use it when
available.

Report unexpected elements as warnings and the rest of
problems as errors.

This will allow tools like qdbusxml2cpp to show parsing
diagnostics using a custom format.

While at it, s/D-BUS/D-Bus/a

Task-number: QTBUG-2597
Change-Id: Ibec3f5f282cf717d1127eb64c82b4fc695f14a74
Reviewed-by: Thiago Macieira <thiago.macieira@intel.com>
This commit is contained in:
Ievgenii Meshcheriakov 2023-06-27 16:17:59 +02:00
parent 1572c420f3
commit 248d2103b5
3 changed files with 76 additions and 37 deletions

View File

@ -58,6 +58,15 @@ public:
qint64 columnNumber = 0; qint64 columnNumber = 0;
}; };
class DiagnosticsReporter
{
public:
virtual void warning(const SourceLocation &location, const char *msg, ...)
Q_ATTRIBUTE_FORMAT_PRINTF(3, 4) = 0;
virtual void error(const SourceLocation &location, const char *msg, ...)
Q_ATTRIBUTE_FORMAT_PRINTF(3, 4) = 0;
};
struct Argument struct Argument
{ {
SourceLocation location; SourceLocation location;

View File

@ -17,7 +17,21 @@ using namespace Qt::StringLiterals;
Q_LOGGING_CATEGORY(dbusParser, "dbus.parser", QtWarningMsg) Q_LOGGING_CATEGORY(dbusParser, "dbus.parser", QtWarningMsg)
#define qDBusParserError(...) qCDebug(dbusParser, ##__VA_ARGS__) #define qDBusParserWarning(format, ...) \
do { \
if (m_reporter) \
m_reporter->warning(m_currentLocation, format "\n", ##__VA_ARGS__); \
else \
qCDebug(dbusParser, "Warning: " format, ##__VA_ARGS__); \
} while (0)
#define qDBusParserError(format, ...) \
do { \
if (m_reporter) \
m_reporter->error(m_currentLocation, format "\n", ##__VA_ARGS__); \
else \
qCDebug(dbusParser, "Error: " format, ##__VA_ARGS__); \
} while (0)
bool QDBusXmlParser::parseArg(const QXmlStreamAttributes &attributes, bool QDBusXmlParser::parseArg(const QXmlStreamAttributes &attributes,
QDBusIntrospection::Argument &argData) QDBusIntrospection::Argument &argData)
@ -28,8 +42,8 @@ bool QDBusXmlParser::parseArg(const QXmlStreamAttributes &attributes,
bool ok = QDBusUtil::isValidSingleSignature(argType); bool ok = QDBusUtil::isValidSingleSignature(argType);
if (!ok) { if (!ok) {
qDBusParserError("Invalid D-BUS type signature '%s' found while parsing introspection", qDBusParserError("Invalid D-Bus type signature '%s' found while parsing introspection",
qPrintable(argType)); qPrintable(argType));
} }
argData.name = attributes.value("name"_L1).toString(); argData.name = attributes.value("name"_L1).toString();
@ -58,8 +72,8 @@ bool QDBusXmlParser::parseAnnotation(QDBusIntrospection::Annotations &annotation
const QString name = attributes.value("name"_L1).toString(); const QString name = attributes.value("name"_L1).toString();
if (!QDBusUtil::isValidInterfaceName(name)) { if (!QDBusUtil::isValidInterfaceName(name)) {
qDBusParserError("Invalid D-BUS annotation '%s' found while parsing introspection", qDBusParserError("Invalid D-Bus annotation '%s' found while parsing introspection",
qPrintable(name)); qPrintable(name));
return false; return false;
} }
const QString value = attributes.value("value"_L1).toString(); const QString value = attributes.value("value"_L1).toString();
@ -78,8 +92,9 @@ bool QDBusXmlParser::parseProperty(QDBusIntrospection::Property &propertyData)
QXmlStreamAttributes attributes = m_xml.attributes(); QXmlStreamAttributes attributes = m_xml.attributes();
const QString propertyName = attributes.value("name"_L1).toString(); const QString propertyName = attributes.value("name"_L1).toString();
if (!QDBusUtil::isValidMemberName(propertyName)) { if (!QDBusUtil::isValidMemberName(propertyName)) {
qDBusParserError("Invalid D-BUS member name '%s' found in interface '%s' while parsing introspection", qDBusParserError("Invalid D-Bus member name '%s' found in interface '%s' while parsing "
qPrintable(propertyName), qPrintable(m_currentInterface->name)); "introspection",
qPrintable(propertyName), qPrintable(m_currentInterface->name));
m_xml.skipCurrentElement(); m_xml.skipCurrentElement();
return false; return false;
} }
@ -90,9 +105,10 @@ bool QDBusXmlParser::parseProperty(QDBusIntrospection::Property &propertyData)
if (!QDBusUtil::isValidSingleSignature(propertyData.type)) { if (!QDBusUtil::isValidSingleSignature(propertyData.type)) {
// cannot be! // cannot be!
qDBusParserError("Invalid D-BUS type signature '%s' found in property '%s.%s' while parsing introspection", qDBusParserError("Invalid D-Bus type signature '%s' found in property '%s.%s' while "
qPrintable(propertyData.type), qPrintable(m_currentInterface->name), "parsing introspection",
qPrintable(propertyName)); qPrintable(propertyData.type), qPrintable(m_currentInterface->name),
qPrintable(propertyName));
} }
const QString access = attributes.value("access"_L1).toString(); const QString access = attributes.value("access"_L1).toString();
@ -103,9 +119,10 @@ bool QDBusXmlParser::parseProperty(QDBusIntrospection::Property &propertyData)
else if (access == "readwrite"_L1) else if (access == "readwrite"_L1)
propertyData.access = QDBusIntrospection::Property::ReadWrite; propertyData.access = QDBusIntrospection::Property::ReadWrite;
else { else {
qDBusParserError("Invalid D-BUS property access '%s' found in property '%s.%s' while parsing introspection", qDBusParserError("Invalid D-Bus property access '%s' found in property '%s.%s' while "
qPrintable(access), qPrintable(m_currentInterface->name), "parsing introspection",
qPrintable(propertyName)); qPrintable(access), qPrintable(m_currentInterface->name),
qPrintable(propertyName));
return false; // invalid one! return false; // invalid one!
} }
@ -120,7 +137,8 @@ bool QDBusXmlParser::parseProperty(QDBusIntrospection::Property &propertyData)
if (m_xml.name() == "annotation"_L1) { if (m_xml.name() == "annotation"_L1) {
parseAnnotation(propertyData.annotations); parseAnnotation(propertyData.annotations);
} else if (m_xml.prefix().isEmpty()) { } else if (m_xml.prefix().isEmpty()) {
qDBusParserError() << "Unknown element" << m_xml.name() << "while checking for annotations"; qDBusParserWarning("Unknown element '%s' while checking for annotations",
qPrintable(m_xml.name().toString()));
} }
m_xml.skipCurrentElement(); m_xml.skipCurrentElement();
} while (readNextStartElement()); } while (readNextStartElement());
@ -129,7 +147,7 @@ bool QDBusXmlParser::parseProperty(QDBusIntrospection::Property &propertyData)
} }
if (!m_xml.isEndElement() || m_xml.name() != "property"_L1) { if (!m_xml.isEndElement() || m_xml.name() != "property"_L1) {
qDBusParserError() << "Invalid property specification" << m_xml.tokenString() << m_xml.name(); qDBusParserError("Invalid property specification: '%s'", qPrintable(m_xml.tokenString()));
return false; return false;
} }
@ -144,8 +162,9 @@ bool QDBusXmlParser::parseMethod(QDBusIntrospection::Method &methodData)
const QXmlStreamAttributes attributes = m_xml.attributes(); const QXmlStreamAttributes attributes = m_xml.attributes();
const QString methodName = attributes.value("name"_L1).toString(); const QString methodName = attributes.value("name"_L1).toString();
if (!QDBusUtil::isValidMemberName(methodName)) { if (!QDBusUtil::isValidMemberName(methodName)) {
qDBusParserError("Invalid D-BUS member name '%s' found in interface '%s' while parsing introspection", qDBusParserError("Invalid D-Bus member name '%s' found in interface '%s' while parsing "
qPrintable(methodName), qPrintable(m_currentInterface->name)); "introspection",
qPrintable(methodName), qPrintable(m_currentInterface->name));
return false; return false;
} }
@ -177,7 +196,8 @@ bool QDBusXmlParser::parseMethod(QDBusIntrospection::Method &methodData)
outArguments << argument; outArguments << argument;
} }
} else if (m_xml.prefix().isEmpty()) { } else if (m_xml.prefix().isEmpty()) {
qDBusParserError() << "Unknown element" << m_xml.name() << "while checking for method arguments"; qDBusParserWarning("Unknown element '%s' while checking for method arguments",
qPrintable(m_xml.name().toString()));
} }
m_xml.skipCurrentElement(); m_xml.skipCurrentElement();
} while (readNextStartElement()); } while (readNextStartElement());
@ -201,8 +221,9 @@ bool QDBusXmlParser::parseSignal(QDBusIntrospection::Signal &signalData)
const QString signalName = attributes.value("name"_L1).toString(); const QString signalName = attributes.value("name"_L1).toString();
if (!QDBusUtil::isValidMemberName(signalName)) { if (!QDBusUtil::isValidMemberName(signalName)) {
qDBusParserError("Invalid D-BUS member name '%s' found in interface '%s' while parsing introspection", qDBusParserError("Invalid D-Bus member name '%s' found in interface '%s' while parsing "
qPrintable(signalName), qPrintable(m_currentInterface->name)); "introspection",
qPrintable(signalName), qPrintable(m_currentInterface->name));
return false; return false;
} }
@ -230,7 +251,8 @@ bool QDBusXmlParser::parseSignal(QDBusIntrospection::Signal &signalData)
arguments << argument; arguments << argument;
} }
} else { } else {
qDBusParserError() << "Unknown element" << m_xml.name() << "while checking for signal arguments"; qDBusParserWarning("Unknown element '%s' while checking for signal arguments",
qPrintable(m_xml.name().toString()));
} }
m_xml.skipCurrentElement(); m_xml.skipCurrentElement();
} while (readNextStartElement()); } while (readNextStartElement());
@ -250,8 +272,8 @@ void QDBusXmlParser::readInterface()
const QString ifaceName = m_xml.attributes().value("name"_L1).toString(); const QString ifaceName = m_xml.attributes().value("name"_L1).toString();
if (!QDBusUtil::isValidInterfaceName(ifaceName)) { if (!QDBusUtil::isValidInterfaceName(ifaceName)) {
qDBusParserError("Invalid D-BUS interface name '%s' found while parsing introspection", qDBusParserError("Invalid D-Bus interface name '%s' found while parsing introspection",
qPrintable(ifaceName)); qPrintable(ifaceName));
return; return;
} }
@ -283,7 +305,8 @@ void QDBusXmlParser::readInterface()
m_xml.skipCurrentElement(); // skip over annotation object m_xml.skipCurrentElement(); // skip over annotation object
} else { } else {
if (m_xml.prefix().isEmpty()) { if (m_xml.prefix().isEmpty()) {
qDBusParserError() << "Unknown element while parsing interface" << m_xml.name(); qDBusParserWarning("Unknown element '%s' while parsing interface",
qPrintable(m_xml.name().toString()));
} }
m_xml.skipCurrentElement(); m_xml.skipCurrentElement();
} }
@ -296,7 +319,7 @@ void QDBusXmlParser::readInterface()
QSharedDataPointer<QDBusIntrospection::Interface>(m_currentInterface.release())); QSharedDataPointer<QDBusIntrospection::Interface>(m_currentInterface.release()));
if (!m_xml.isEndElement() || m_xml.name() != "interface"_L1) { if (!m_xml.isEndElement() || m_xml.name() != "interface"_L1) {
qDBusParserError() << "Invalid Interface specification"; qDBusParserError("Invalid Interface specification");
} }
} }
@ -307,8 +330,8 @@ void QDBusXmlParser::readNode(int nodeLevel)
? (m_object->path + objName) ? (m_object->path + objName)
: QString(m_object->path + u'/' + objName); : QString(m_object->path + u'/' + objName);
if (!QDBusUtil::isValidObjectPath(fullName)) { if (!QDBusUtil::isValidObjectPath(fullName)) {
qDBusParserError("Invalid D-BUS object path '%s' found while parsing introspection", qDBusParserError("Invalid D-Bus object path '%s' found while parsing introspection",
qPrintable(fullName)); qPrintable(fullName));
return; return;
} }
@ -340,8 +363,13 @@ bool QDBusXmlParser::readNextStartElement()
return false; return false;
} }
QDBusXmlParser::QDBusXmlParser(const QString &service, const QString &path, const QString &xmlData) QDBusXmlParser::QDBusXmlParser(const QString &service, const QString &path, const QString &xmlData,
: m_service(service), m_path(path), m_object(new QDBusIntrospection::Object), m_xml(xmlData) QDBusIntrospection::DiagnosticsReporter *reporter)
: m_service(service),
m_path(path),
m_object(new QDBusIntrospection::Object),
m_xml(xmlData),
m_reporter(reporter)
{ {
m_object->service = m_service; m_object->service = m_service;
m_object->path = m_path; m_object->path = m_path;
@ -360,7 +388,8 @@ QDBusXmlParser::QDBusXmlParser(const QString &service, const QString &path, cons
readInterface(); readInterface();
} else { } else {
if (m_xml.prefix().isEmpty()) { if (m_xml.prefix().isEmpty()) {
qDBusParserError() << "skipping unknown element" << m_xml.name(); qDBusParserWarning("Skipping unknown element '%s'",
qPrintable(m_xml.name().toString()));
} }
m_xml.skipCurrentElement(); m_xml.skipCurrentElement();
} }
@ -369,7 +398,8 @@ QDBusXmlParser::QDBusXmlParser(const QString &service, const QString &path, cons
if (m_xml.name() == "node"_L1) { if (m_xml.name() == "node"_L1) {
--nodeLevel; --nodeLevel;
} else { } else {
qDBusParserError() << "Invalid Node declaration" << m_xml.name(); qDBusParserError("Invalid node declaration '%s'",
qPrintable(m_xml.name().toString()));
} }
break; break;
case QXmlStreamReader::StartDocument: case QXmlStreamReader::StartDocument:
@ -386,14 +416,13 @@ QDBusXmlParser::QDBusXmlParser(const QString &service, const QString &path, cons
break; break;
Q_FALLTHROUGH(); Q_FALLTHROUGH();
default: default:
qDBusParserError() << "unknown token" << m_xml.name() << m_xml.tokenString(); qDBusParserError("Unknown token: '%s'", qPrintable(m_xml.tokenString()));
break; break;
} }
} }
if (m_xml.hasError()) { if (m_xml.hasError())
qDBusParserError() << "xml error" << m_xml.errorString() << "doc" << xmlData; qDBusParserError("XML error: %s", qPrintable(m_xml.errorString()));
}
} }
QT_END_NAMESPACE QT_END_NAMESPACE

View File

@ -39,10 +39,11 @@ class QDBusXmlParser
QDBusIntrospection::Interfaces m_interfaces; QDBusIntrospection::Interfaces m_interfaces;
QXmlStreamReader m_xml; QXmlStreamReader m_xml;
QDBusIntrospection::SourceLocation m_currentLocation; QDBusIntrospection::SourceLocation m_currentLocation;
QDBusIntrospection::DiagnosticsReporter *m_reporter;
public: public:
QDBusXmlParser(const QString& service, const QString& path, QDBusXmlParser(const QString &service, const QString &path, const QString &xmlData,
const QString& xmlData); QDBusIntrospection::DiagnosticsReporter *reporter = nullptr);
inline QDBusIntrospection::Interfaces interfaces() const { return m_interfaces; } inline QDBusIntrospection::Interfaces interfaces() const { return m_interfaces; }
inline QSharedDataPointer<QDBusIntrospection::Object> object() const { return m_object; } inline QSharedDataPointer<QDBusIntrospection::Object> object() const { return m_object; }