Intruder
Intruder14 листопада 2019 р. 11:43

Как в Qt провести процедуру валидации с помощью XSD схемы XML файла

Всем доброго времени суток.

Как провести валидацию (offline) файла XML? С чего начать, может быть есть какие-нибудь примеры?
Заранее благодарен за помощь.

Рекомендуємо хостинг TIMEWEB
Рекомендуємо хостинг TIMEWEB
Стабільний хостинг, на якому розміщується соціальна мережа EVILEG. Для проектів на Django радимо VDS хостинг.

Вам це подобається? Поділіться в соціальних мережах!

12
Evgenii Legotckoi
  • 15 листопада 2019 р. 03:16

Добрый день.

А что в вашем понимании означает валидацию?

Вы можете считать файл и использовать QDomDocument для считывания структуры XML. Если структура разбита, то метод setContent может помочь выйвить ошибку в определённой строке.

Вот сигнатура метода

bool QDomDocument::setContent(const QString &text, bool namespaceProcessing, QString *errorMsg = nullptr, int *errorLine = nullptr, int *errorColumn = nullptr)

Передаёте указатели на строку для сообщения об ошибки, на int переменную для номера строки и на int переменную для номера колонки. Если метод возвращает false, то можете уже смотреть в те переменные, там будет ошибка, её место и описание.

    Intruder
    • 15 листопада 2019 р. 03:43

    Евгений, добрый день.

    Думаю, что Ваш вопрос правильный, я не совсем корректно сформулировал свой вопрос. Суть заключается в том, что я хочу проверить правильный ли файл мне подсовывается с точки зрения его схемы XSD. Дело в том, что я могу на вход подать любой файл формата XML. И если входной файл не соответсвует заданной схеме, то завершить дальнейшую его обработку.
    Я уже написал небольшую програмку, которая этим занимается, но столкнулся в другой проблемой.
    Для начала я решил проверить свою задачу вот так:

    #include <QCoreApplication>
    
    ....
    {
    ....
    QFile xsdfile("D:\\XML_Schema_package\\Schemas\\wrngdata.xsd");
        xsdfile.open(QIODevice::ReadOnly);
        QXmlSchema schema;
        if(schema.load(&xsdfile, QUrl::fromLocalFile(xsdfile.fileName()))){
            if(schema.isValid()){
                qDebug() << "schema is valid";
                QFile xmlfile("D:\\S1000D_4-1_Bike-Samples\\DMC-S1000DLIGHTING-AAA-D00-00-00-00AA-057A-A_008-00_EN-US.XML");
                xmlfile.open(QIODevice::ReadOnly);
                QXmlSchemaValidator xmlvalidator(schema);
                if(xmlvalidator.validate(&xmlfile, QUrl::fromLocalFile(xmlfile.fileName()))){
                    qDebug() << "Is valid";
                }
            }else {
                qDebug() << "schema is not valid";
            }
        }
    
        return a.exec();
        }
    

    Получил утвердительный результат.
    В моем проекте есть динамическая библиотека и я решил перенести эту часть кода в библиотеку. Вот ее объявление и описание:

    static bool validate(const QString &xsdschema, const QString &xmlschema);
    
    bool Wrngdatalib::validate(const QString &xsdschema, const QString &xmlschema)
    {
        bool flag = false;
        QFile xsdfile(xsdschema);
        if(!xsdfile.open(QIODevice::ReadOnly)){
            flag = false;
        }
        QXmlSchema schema;
        if(schema.load(&xsdfile, QUrl::fromLocalFile(xsdfile.fileName()))){
            if(schema.isValid()){
                QFile xmlfile(xmlschema);
                if(!xmlfile.open(QIODevice::ReadOnly)){
                    flag = false;
                }
                QXmlSchemaValidator xmlvalidator(schema);
                if(xmlvalidator.validate(&xmlfile, QUrl::fromLocalFile(xmlfile.fileName()))){
                    flag = true;
                }
            }
        }
        return flag;
    }
    

    В теле программы я поправил код вот таким образом:

    if(Wrngdatalib::validate("D:\\XML_Schema_package\\Schemas\\wrngdata.xsd", "D:\\S1000D_4-1_Bike-Samples\\DMC-S1000DLIGHTING-AAA-D00-00-00-00AA-057A-A_008-00_EN-US.XML")){
        qDebug() << "Is valid";
    }
    

    Но при попытке выполнить программу получаю вот такие ошибки:

    Object is exist
    QEventLoop: Cannot be used without QApplication
    QEventLoop: Cannot be used without QApplication
    QEventLoop: Cannot be used without QApplication
    QEventLoop: Cannot be used without QApplication
    QEventLoop: Cannot be used without QApplication
    Error XSDError in http://www.s1000d.org/S1000D_4-1/xml_schema_flat/wrngdata.xsd, at line 1, column 0: Premature end of document.
    Error XSDError in file:///D:/S1000D_4-1_Bike-Samples/DMC-S1000DLIGHTING-AAA-D00-00-00-00AA-057A-A_008-00_EN-US.XML, at line 8, column 304: Loaded schema file is invalid.
    

    Почитав в Интернете нашел описание того, что моя функция может выполнится только в окне приложения, потому что создает еще потоко сообщений или что-то в этом роде. И меня возник вопрос, можно ли это исправить или часть кода, который я перенес в библиотеку придется возвращать в приложение?

    Заранее благодарен за ответ и помощь.

      Evgenii Legotckoi
      • 15 листопада 2019 р. 04:06
      • (відредаговано)

      Что это такое Wrngdatalib ? Это namespace ?

      Скорее всего проблема в том, что те объекты тех классов, которые там присутствуют для обработки xml наследованы от QObject, но в том методе внутри библиотеки они не находят возможности использования QEventLoop. Попробуйте в библиотеке написать класс для валидации XML, который будет наследован от QObject. А в коде основной программы создавайте инстанс этого класса и только потом вызывайте метод validate -

        Intruder
        • 15 листопада 2019 р. 05:55

        Что это такое Wrngdatalib ? Это namespace ?
        Wrngdatalib - Это class.
        Что-то не получилось. Может я снова делаю что-то не то?
        Создал класс:

        #ifndef XMLFUNCTION_H
        #define XMLFUNCTION_H
        
        #include <QObject>
        #include "wrngdatalib_global.h"
        
        class WRNGDATALIBSHARED_EXPORT xmlfunction : public QObject
        {
            Q_OBJECT
        public:
            explicit xmlfunction(QObject *parent = nullptr);
        
            bool validate(const QString &xsdschema, const QString &xmlschema);
        
        signals:
        
        public slots:
        
        };
        
        #endif // XMLFUNCTION_H
        

        Описание функции:

        #include "xmlfunction.h"
        #include <QtXmlPatterns/QXmlSchema>
        #include <QtXmlPatterns/QXmlSchemaValidator>
        #include <QFile>
        xmlfunction::xmlfunction(QObject *parent) : QObject(parent)
        {
        
        }
        
        bool xmlfunction::validate(const QString &xsdschema, const QString &xmlschema)
        {
            bool flag = false;
            QFile xsdfile(xsdschema);
            if(!xsdfile.open(QIODevice::ReadOnly)){
                flag = false;
            }
            QXmlSchema schema;
            if(schema.load(&xsdfile, QUrl::fromLocalFile(xsdfile.fileName()))){
                if(schema.isValid()){
                    QFile xmlfile(xmlschema);
                    if(!xmlfile.open(QIODevice::ReadOnly)){
                        flag = false;
                    }
                    QXmlSchemaValidator xmlvalidator(schema);
                    if(xmlvalidator.validate(&xmlfile, QUrl::fromLocalFile(xmlfile.fileName()))){
                       flag = true;
                    }
                }
            }
            return flag;
        }
        

        И сама программа:

        #include <QCoreApplication>
        #include <QVector>
        #include <QDebug>
        #include <QtXmlPatterns/QXmlSchema>
        #include <QtXmlPatterns/QXmlSchemaValidator>
        #include "wrngdatalib.h"
        #include "xmlfunction.h"
        
        int main(int argc, char *argv[])
        {
            QCoreApplication a(argc, argv);
        
            xmlfunction xmlObj;
            bool aaa = xmlObj.validate("D:\\XML_Schema_package\\Schemas\\wrngdata.xsd", "D:\\S1000D_4-1_Bike-Samples\\DMC-S1000DLIGHTING-AAA-D00-00-00-00AA-057A-A_008-00_EN-US.XML");
            if(aaa){
                qDebug() << "Is valid";
            }
            return a.exec();
        }
        

        Получаю туже самую ошибку.

          Evgenii Legotckoi
          • 15 листопада 2019 р. 06:02

          Ага... слушайте. Дело-то походу в том, что вы используете QCoreApplication, поскольку у вас консольная программа.

          Попробуйте так

          int main(int argc, char *argv[])
          {
              QCoreApplication a(argc, argv);
          
              xmlfunction xmlObj;
              bool aaa = xmlObj.validate("D:\\XML_Schema_package\\Schemas\\wrngdata.xsd", "D:\\S1000D_4-1_Bike-Samples\\DMC-S1000DLIGHTING-AAA-D00-00-00-00AA-057A-A_008-00_EN-US.XML");
              if(aaa){
                  qDebug() << "Is valid";
              }
              return 0;
          }
          
            Intruder
            • 15 листопада 2019 р. 06:18

            Нет, так тоже не работает. Ведь когда у меня эта часть кода по валидации была в теле программы, то все работало. Может это как-то связано с сигналами и слотами? Я просто в этом пока ну очень сильно плаваю.

              Evgenii Legotckoi
              • 15 листопада 2019 р. 06:33

              Это не с сигналами и слотами связано, а с системой обработки событий в Qt.
              Если использовать QApplication вместо QCoreApplication, то работает?

                Intruder
                • 15 листопада 2019 р. 07:37

                Евгений, почитав про эту проблему пришел к выводу, что либо нужно говорить очередь, либо все вернуть из библиотеки (dll в моем случае) в приложение, потому что в приложении все работает просто замечательно. И еще как вариант курить QEventLoop.

                  Evgenii Legotckoi
                  • 18 листопада 2019 р. 06:04

                  Я бы вернул в приложение.
                  Ещё задумался бы, а нужна ли действительно эта динамическая библиотека вам. Возможно, что вам достаточно будет сделать статическую библиотеку.

                    Intruder
                    • 18 листопада 2019 р. 13:27

                    Евгений, доброго дня.
                    В динамической библиотеке помимо этого много чего еще есть. Поэтому просто хотел добавить функцию валидации перед парсингом. Функцию оставлю в приложении - это не критично. Красиво чтоли хотелось, не знаю.
                    Спасибо за помощь. А в чем разница между статической и динамичческой библиотекой?

                      Evgenii Legotckoi
                      • 18 листопада 2019 р. 13:34

                      Добрый день

                      • динамическая - это в случае Windows dll, он лежит рядом с испоолняемым файлом.
                      • статическая библиотека компилируется как самостоятельная библиотека в проекте, и будучи предкомпилированной может подключаться в другие проекты без наличия исходников, достаточно будет только заголовочных файлов. При этом статическая библиотека включается в исполняемый файл.
                        Intruder
                        • 18 листопада 2019 р. 13:36

                        Понятно. Я кстати так сейчас и сделал, добавил проект в проект и т.д.
                        Спасибо!

                          Коментарі

                          Only authorized users can post comments.
                          Please, Log in or Sign up
                          sf

                          C++ - Тест 004. Указатели, Массивы и Циклы

                          • Результат:90бали,
                          • Рейтинг балів8
                          МВ

                          Qt - Тест 001. Сигналы и слоты

                          • Результат:68бали,
                          • Рейтинг балів-1
                          ЛС

                          C++ - Тест 001. Первая программа и типы данных

                          • Результат:53бали,
                          • Рейтинг балів-4
                          Останні коментарі
                          A
                          ALO1ZE19 жовтня 2024 р. 15:19
                          Читалка файлів fb3 на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                          ИМ
                          Игорь Максимов05 жовтня 2024 р. 14:51
                          Django - Урок 064. Як написати розширення для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                          d
                          dblas505 липня 2024 р. 18:02
                          QML - Урок 016. База даних SQLite та робота з нею в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                          k
                          kmssr09 лютого 2024 р. 02:43
                          Qt Linux - Урок 001. Автозапуск програми Qt під Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                          АК
                          Анатолий Кононенко05 лютого 2024 р. 09:50
                          Qt WinAPI - Урок 007. Робота з ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                          Тепер обговоріть на форумі
                          J
                          JacobFib17 жовтня 2024 р. 10:27
                          добавить qlineseries в функции Пользователь может получить любые разъяснения по интересующим вопросам, касающимся обработки его персональных данных, обратившись к Оператору с помощью электронной почты https://topdecorpro.ru…
                          ИМ
                          Игорь Максимов03 жовтня 2024 р. 11:05
                          Реализация навигации по разделам Спасибо Евгений!
                          JW
                          Jhon Wick01 жовтня 2024 р. 22:52
                          Indian Food Restaurant In Columbus OH| Layla’s Kitchen Indian Restaurant If you're looking for a truly authentic https://www.laylaskitchenrestaurantohio.com/ , Layla’s Kitchen Indian Restaurant is your go-to destination. Located at 6152 Cleveland Ave, Colu…
                          КГ
                          Кирилл Гусарев27 вересня 2024 р. 16:09
                          Не запускается программа на Qt: точка входа в процедуру не найдена в библиотеке DLL Написал программу на C++ Qt в Qt Creator, сбилдил Release с помощью MinGW 64-bit, бинарнику напихал dll-ки с помощью windeployqt.exe. При попытке запуска моей сбилженной программы выдаёт три оши…
                          F
                          Fynjy22 липня 2024 р. 11:15
                          при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …

                          Слідкуйте за нами в соціальних мережах