summaryrefslogtreecommitdiffstats
path: root/src/protobuf/qprotobufserializerbase.cpp
blob: b929287e3beeecc77d4419aa4d0c5459833c6414 (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
// Copyright (C) 2024 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only

#include <QtProtobuf/private/qprotobufserializerbase_p.h>

#include <QtProtobuf/private/qprotobufregistration_p.h>
#include <QtProtobuf/private/qtprotobufdefs_p.h>
#include <QtProtobuf/private/qtprotobuflogging_p.h>
#include <QtProtobuf/private/qtprotobufserializerhelpers_p.h>

#include <QtProtobuf/qprotobufmessage.h>
#include <QtProtobuf/qprotobufpropertyordering.h>
#include <QtProtobuf/qprotobufrepeatediterator.h>
#include <QtProtobuf/qtprotobuftypes.h>

QT_BEGIN_NAMESPACE

QProtobufSerializerBase::QProtobufSerializerBase() = default;
QProtobufSerializerBase::~QProtobufSerializerBase() = default;

void QProtobufSerializerBase::serializeMessage(const QProtobufMessage *message)
{
    Q_ASSERT(message != nullptr);

    const auto *ordering = message->propertyOrdering();
    Q_ASSERT(ordering != nullptr);

    for (int index = 0; index < ordering->fieldCount(); ++index) {
        int fieldNumber = ordering->fieldNumber(index);
        Q_ASSERT_X(fieldNumber <= ProtobufFieldNumMax && fieldNumber >= ProtobufFieldNumMin, "",
                   "Protobuf field number is out of range");
        QtProtobufPrivate::QProtobufFieldInfo fieldInfo(*ordering, index);
        QVariant value = QtProtobufSerializerHelpers::messageProperty(message, fieldInfo);
        QMetaType metaType = value.metaType();

        // Empty value
        if (metaType.id() == QMetaType::UnknownType || value.isNull())
            continue;

        if (metaType.flags().testFlag(QMetaType::IsPointer)) {
            serializeMessageField(value.value<QProtobufMessage *>(), fieldInfo);
            continue;
        }

        if (value.canView(QMetaType::fromType<QProtobufRepeatedIterator>())) {
            QProtobufRepeatedIterator propertyIt = value.view<QProtobufRepeatedIterator>();
            while (propertyIt.hasNext())
                serializeMessageField(propertyIt.next(), fieldInfo);
            continue;
        }

        QtProtobufPrivate::FieldFlags flags = fieldInfo.fieldFlags();
        // Convert enums to something known by scalar type serializers
        if (flags.testFlag(QtProtobufPrivate::FieldFlag::Enum)) {
            if (!serializeEnum(value, fieldInfo))
                qProtoWarning() << metaType << " is not registered as protobuf enum";
            continue;
        }

        if (serializeScalarField(value, fieldInfo))
            continue;

        auto handler = QtProtobufPrivate::findHandler(metaType);
        if (!handler.serializer) {
            qProtoWarning("No serializer found for type %s", value.typeName());
            continue;
        }

        handler.serializer([this](const QProtobufMessage *message,
                                  const QtProtobufPrivate::QProtobufFieldInfo
                                      &fieldInfo) { serializeMessageField(message, fieldInfo); },
                           value.constData(), fieldInfo);
    }
}

void QProtobufSerializerBase::serializeMessageField(const QProtobufMessage *message,
                                                    const QtProtobufPrivate::QProtobufFieldInfo
                                                        &fieldInfo)
{
    serializeMessageFieldBegin();
    serializeMessage(message);
    serializeMessageFieldEnd(message, fieldInfo);
}

QT_END_NAMESPACE