File indexing completed on 2025-10-26 03:48:03

0001 /*
0002     This file is part of the syndication library
0003     SPDX-FileCopyrightText: 2006 Frank Osterfeld <osterfeld@kde.org>
0004 
0005     SPDX-License-Identifier: LGPL-2.0-or-later
0006 */
0007 
0008 #include "parser.h"
0009 #include "document.h"
0010 #include "model.h"
0011 #include "modelmaker.h"
0012 #include "property.h"
0013 #include "rdfvocab.h"
0014 #include "resource.h"
0015 #include "rssvocab.h"
0016 #include "statement.h"
0017 
0018 #include <documentsource.h>
0019 
0020 #include <QDomDocument>
0021 #include <QDomNodeList>
0022 #include <QHash>
0023 #include <QList>
0024 #include <QMap>
0025 #include <QString>
0026 #include <QStringList>
0027 
0028 namespace Syndication
0029 {
0030 namespace RDF
0031 {
0032 class SYNDICATION_NO_EXPORT Parser::ParserPrivate
0033 {
0034 public:
0035     QDomDocument addEnumeration(const QDomDocument &doc);
0036     void map09to10(Model model);
0037     void addSequenceFor09(Model model);
0038 
0039     QString strInternalNs;
0040     QString strItemIndex;
0041 };
0042 
0043 bool Parser::accept(const DocumentSource &source) const
0044 {
0045     QDomDocument doc = source.asDomDocument();
0046 
0047     if (doc.isNull()) {
0048         return false;
0049     }
0050     QDomElement root = doc.documentElement();
0051 
0052     if (!root.isElement()) {
0053         return false;
0054     }
0055 
0056     return root.namespaceURI() == RDFVocab::self()->namespaceURI();
0057 }
0058 
0059 SpecificDocumentPtr Parser::parse(const DocumentSource &source) const
0060 {
0061     QDomDocument doc = source.asDomDocument();
0062 
0063     if (doc.isNull()) {
0064         return Syndication::SpecificDocumentPtr(new Document());
0065     }
0066 
0067     doc = d->addEnumeration(doc);
0068 
0069     ModelMaker maker;
0070     Model model = maker.createFromXML(doc);
0071 
0072     bool is09 = !model.resourcesWithType(RSS09Vocab::self()->channel()).isEmpty();
0073 
0074     if (is09) {
0075         d->map09to10(model);
0076         d->addSequenceFor09(model);
0077     }
0078 
0079     QList<ResourcePtr> channels = model.resourcesWithType(RSSVocab::self()->channel());
0080 
0081     if (channels.isEmpty()) {
0082         return Syndication::SpecificDocumentPtr(new Document());
0083     }
0084 
0085     return DocumentPtr(new Document(*(channels.begin())));
0086 }
0087 
0088 QDomDocument Parser::ParserPrivate::addEnumeration(const QDomDocument &docp)
0089 {
0090     QDomDocument doc(docp);
0091 
0092     const QDomNodeList list = doc.elementsByTagNameNS(RSS09Vocab::self()->namespaceURI(), QStringLiteral("item"));
0093 
0094     for (int i = 0; i < list.size(); ++i) {
0095         QDomElement item = list.item(i).toElement();
0096         if (!item.isNull()) {
0097             QDomElement ie = doc.createElementNS(strInternalNs, strItemIndex);
0098             item.appendChild(ie);
0099             ie.appendChild(doc.createTextNode(QString::number(i)));
0100         }
0101     }
0102 
0103     return doc;
0104 }
0105 
0106 void Parser::ParserPrivate::map09to10(Model model)
0107 {
0108     QHash<QString, PropertyPtr> hash;
0109 
0110     hash.insert(RSS09Vocab::self()->title()->uri(), RSSVocab::self()->title());
0111     hash.insert(RSS09Vocab::self()->description()->uri(), RSSVocab::self()->description());
0112     hash.insert(RSS09Vocab::self()->link()->uri(), RSSVocab::self()->link());
0113     hash.insert(RSS09Vocab::self()->name()->uri(), RSSVocab::self()->name());
0114     hash.insert(RSS09Vocab::self()->url()->uri(), RSSVocab::self()->url());
0115     hash.insert(RSS09Vocab::self()->image()->uri(), RSSVocab::self()->image());
0116     hash.insert(RSS09Vocab::self()->textinput()->uri(), RSSVocab::self()->textinput());
0117 
0118     QStringList uris09 = RSS09Vocab::self()->properties();
0119 
0120     // map statement predicates to RSS 1.0
0121 
0122     const QList<StatementPtr> &statements = model.statements();
0123 
0124     for (const auto &stmt : statements) {
0125         const QString predUri = stmt->predicate()->uri();
0126         if (uris09.contains(predUri)) {
0127             model.addStatement(stmt->subject(), hash[predUri], stmt->object());
0128         }
0129     }
0130     // map channel type
0131     QList<ResourcePtr> channels = model.resourcesWithType(RSS09Vocab::self()->channel());
0132 
0133     ResourcePtr channel;
0134 
0135     if (!channels.isEmpty()) {
0136         channel = *(channels.begin());
0137 
0138         model.removeStatement(channel, RDFVocab::self()->type(), RSS09Vocab::self()->channel());
0139         model.addStatement(channel, RDFVocab::self()->type(), RSSVocab::self()->channel());
0140     }
0141 }
0142 
0143 void Parser::ParserPrivate::addSequenceFor09(Model model)
0144 {
0145     // RDF 0.9 doesn't contain an item sequence, and the items don't have rdf:about, so add both
0146 
0147     const QList<ResourcePtr> items = model.resourcesWithType(RSS09Vocab::self()->item());
0148 
0149     if (items.isEmpty()) {
0150         return;
0151     }
0152 
0153     const QList<ResourcePtr> channels = model.resourcesWithType(RSSVocab::self()->channel());
0154 
0155     if (channels.isEmpty()) {
0156         return;
0157     }
0158 
0159     PropertyPtr itemIndex = model.createProperty(strInternalNs + strItemIndex);
0160 
0161     // use QMap here, not QHash. as we need the sorting functionality
0162     QMap<uint, ResourcePtr> sorted;
0163 
0164     for (const ResourcePtr &i : items) {
0165         QString numstr = i->property(itemIndex)->asString();
0166         bool ok = false;
0167         uint num = numstr.toUInt(&ok);
0168         if (ok) {
0169             sorted[num] = i;
0170         }
0171     }
0172 
0173     SequencePtr seq = model.createSequence();
0174     model.addStatement(channels.first(), RSSVocab::self()->items(), seq);
0175 
0176     for (const ResourcePtr &i : std::as_const(sorted)) {
0177         seq->append(i);
0178         // add rdf:about (type)
0179         model.addStatement(i, RDFVocab::self()->type(), RSSVocab::self()->item());
0180 
0181         // add to items sequence
0182         model.addStatement(seq, RDFVocab::self()->li(), i);
0183     }
0184 }
0185 
0186 Parser::Parser()
0187     : d(new ParserPrivate)
0188 {
0189     d->strInternalNs = QStringLiteral("http://akregator.sf.net/libsyndication/internal#");
0190     d->strItemIndex = QStringLiteral("itemIndex");
0191 }
0192 
0193 Parser::~Parser() = default;
0194 
0195 Parser::Parser(const Parser &other)
0196     : AbstractParser(other)
0197     , d(nullptr)
0198 {
0199 }
0200 Parser &Parser::operator=(const Parser & /*other*/)
0201 {
0202     return *this;
0203 }
0204 
0205 QString Parser::format() const
0206 {
0207     return QStringLiteral("rdf");
0208 }
0209 
0210 } // namespace RDF
0211 } // namespace Syndication