/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtXmlPatterns module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include #include #include #include "qatomicstring_p.h" #include "qautoptr_p.h" #include "qcommonsequencetypes_p.h" #include "qacceltreeresourceloader_p.h" QT_BEGIN_NAMESPACE using namespace QPatternist; AccelTreeResourceLoader::AccelTreeResourceLoader(const NamePool::Ptr &np, const NetworkAccessDelegator::Ptr &manager, AccelTreeBuilder::Features features) : m_namePool(np) , m_networkAccessDelegator(manager) , m_features(features) { Q_ASSERT(m_namePool); Q_ASSERT(m_networkAccessDelegator); } bool AccelTreeResourceLoader::retrieveDocument(const QUrl &uri, const ReportContext::Ptr &context) { Q_ASSERT(uri.isValid()); AccelTreeBuilder builder(uri, uri, m_namePool, context.data(), m_features); const AutoPtr reply(load(uri, m_networkAccessDelegator, context)); if(!reply) return false; bool success = false; success = streamToReceiver(reply.data(), &builder, m_namePool, context, uri); m_loadedDocuments.insert(uri, builder.builtDocument()); return success; } bool AccelTreeResourceLoader::retrieveDocument(QIODevice *source, const QUrl &documentUri, const ReportContext::Ptr &context) { Q_ASSERT(source); Q_ASSERT(source->isReadable()); Q_ASSERT(documentUri.isValid()); AccelTreeBuilder builder(documentUri, documentUri, m_namePool, context.data(), m_features); bool success = false; success = streamToReceiver(source, &builder, m_namePool, context, documentUri); m_loadedDocuments.insert(documentUri, builder.builtDocument()); return success; } QNetworkReply *AccelTreeResourceLoader::load(const QUrl &uri, const NetworkAccessDelegator::Ptr &networkDelegator, const ReportContext::Ptr &context, ErrorHandling errorHandling) { return load(uri, networkDelegator->managerFor(uri), context, errorHandling); } QNetworkReply *AccelTreeResourceLoader::load(const QUrl &uri, QNetworkAccessManager *const networkManager, const ReportContext::Ptr &context, ErrorHandling errorHandling) { Q_ASSERT(networkManager); Q_ASSERT(uri.isValid()); const bool ftpSchemeUsed = (uri.scheme() == QStringLiteral("ftp")); // QNAM doesn't have support for SynchronousRequestAttribute in its ftp backend. QEventLoop ftpNetworkLoop; QNetworkRequest request(uri); if (!ftpSchemeUsed) request.setAttribute(QNetworkRequest::SynchronousRequestAttribute, true); QNetworkReply *const reply = networkManager->get(request); if (ftpSchemeUsed) { ftpNetworkLoop.connect(reply, SIGNAL(error(QNetworkReply::NetworkError)), SLOT(quit())); ftpNetworkLoop.connect(reply, SIGNAL(finished()), SLOT(quit())); ftpNetworkLoop.exec(QEventLoop::ExcludeUserInputEvents); } if (reply->error() != QNetworkReply::NoError) { const QString errorMessage(escape(reply->errorString())); /* Note, we delete reply before we exit this function with error(). */ delete reply; const QSourceLocation location(uri); if(context && (errorHandling == FailOnError)) context->error(errorMessage, ReportContext::FODC0002, location); return 0; } else return reply; } bool AccelTreeResourceLoader::streamToReceiver(QIODevice *const dev, AccelTreeBuilder *const receiver, const NamePool::Ptr &np, const ReportContext::Ptr &context, const QUrl &uri) { Q_ASSERT(dev); Q_ASSERT(receiver); Q_ASSERT(np); QXmlStreamReader reader(dev); /* Optimize: change NamePool to take QStringRef such that we don't have to call toString() below. That * will save us a gazillion of temporary QStrings. */ while(!reader.atEnd()) { reader.readNext(); switch(reader.tokenType()) { case QXmlStreamReader::StartElement: { /* Send the name. */ receiver->startElement(np->allocateQName(reader.namespaceUri().toString(), reader.name().toString(), reader.prefix().toString()), reader.lineNumber(), reader.columnNumber()); /* Send namespace declarations. */ const QXmlStreamNamespaceDeclarations &nss = reader.namespaceDeclarations(); /* The far most common case, is for it to be empty. */ if(!nss.isEmpty()) { const int len = nss.size(); for(int i = 0; i < len; ++i) { const QXmlStreamNamespaceDeclaration &ns = nss.at(i); receiver->namespaceBinding(np->allocateBinding(ns.prefix().toString(), ns.namespaceUri().toString())); } } /* Send attributes. */ const QXmlStreamAttributes &attrs = reader.attributes(); const int len = attrs.size(); for(int i = 0; i < len; ++i) { const QXmlStreamAttribute &attr = attrs.at(i); receiver->attribute(np->allocateQName(attr.namespaceUri().toString(), attr.name().toString(), attr.prefix().toString()), attr.value()); } continue; } case QXmlStreamReader::EndElement: { receiver->endElement(); continue; } case QXmlStreamReader::Characters: { if(reader.isWhitespace()) receiver->whitespaceOnly(reader.text()); else receiver->characters(reader.text()); continue; } case QXmlStreamReader::Comment: { receiver->comment(reader.text().toString()); continue; } case QXmlStreamReader::ProcessingInstruction: { receiver->processingInstruction(np->allocateQName(QString(), reader.processingInstructionTarget().toString()), reader.processingInstructionData().toString()); continue; } case QXmlStreamReader::StartDocument: { receiver->startDocument(); continue; } case QXmlStreamReader::EndDocument: { receiver->endDocument(); continue; } case QXmlStreamReader::EntityReference: case QXmlStreamReader::DTD: { /* We just ignore any DTD and entity references. */ continue; } case QXmlStreamReader::Invalid: { if(context) context->error(escape(reader.errorString()), ReportContext::FODC0002, QSourceLocation(uri, reader.lineNumber(), reader.columnNumber())); return false; } case QXmlStreamReader::NoToken: { Q_ASSERT_X(false, Q_FUNC_INFO, "This token is never expected to be received."); return false; } } } return true; } Item AccelTreeResourceLoader::openDocument(const QUrl &uri, const ReportContext::Ptr &context) { const AccelTree::Ptr doc(m_loadedDocuments.value(uri)); if(doc) return doc->root(QXmlNodeModelIndex()); /* Pass in dummy object. We know AccelTree doesn't use it. */ else { if(retrieveDocument(uri, context)) return m_loadedDocuments.value(uri)->root(QXmlNodeModelIndex()); /* Pass in dummy object. We know AccelTree doesn't use it. */ else return Item(); } } Item AccelTreeResourceLoader::openDocument(QIODevice *source, const QUrl &documentUri, const ReportContext::Ptr &context) { const AccelTree::Ptr doc(m_loadedDocuments.value(documentUri)); if(doc) return doc->root(QXmlNodeModelIndex()); /* Pass in dummy object. We know AccelTree doesn't use it. */ else { if(retrieveDocument(source, documentUri, context)) return m_loadedDocuments.value(documentUri)->root(QXmlNodeModelIndex()); /* Pass in dummy object. We know AccelTree doesn't use it. */ else return Item(); } } SequenceType::Ptr AccelTreeResourceLoader::announceDocument(const QUrl &uri, const Usage) { // TODO deal with the usage thingy Q_ASSERT(uri.isValid()); Q_ASSERT(!uri.isRelative()); Q_UNUSED(uri); /* Needed when compiling in release mode. */ return CommonSequenceTypes::ZeroOrOneDocumentNode; } bool AccelTreeResourceLoader::isDocumentAvailable(const QUrl &uri) { return retrieveDocument(uri, ReportContext::Ptr()); } bool AccelTreeResourceLoader::retrieveUnparsedText(const QUrl &uri, const QString &encoding, const ReportContext::Ptr &context, const SourceLocationReflection *const where) { const AutoPtr reply(load(uri, m_networkAccessDelegator, context)); if(!reply) return false; const QTextCodec * codec; if(encoding.isEmpty()) { /* XSL Transformations (XSLT) Version 2.0 16.2 Reading Text Files: * * "if the media type of the resource is text/xml or application/xml * (see [RFC2376]), or if it matches the conventions text/\*+xml or * application/\*+xml (see [RFC3023] and/or its successors), then the * encoding is recognized as specified in [XML 1.0]" */ codec = QTextCodec::codecForMib(106); } else { codec = QTextCodec::codecForName(encoding.toLatin1()); if(codec && context) { context->error(QtXmlPatterns::tr("%1 is an unsupported encoding.").arg(formatURI(encoding)), ReportContext::XTDE1190, where); } else return false; } QTextCodec::ConverterState converterState; const QByteArray inData(reply->readAll()); const QString result(codec->toUnicode(inData.constData(), inData.length(), &converterState)); if(converterState.invalidChars) { if(context) { context->error(QtXmlPatterns::tr("%1 contains octets which are disallowed in " "the requested encoding %2.").arg(formatURI(uri), formatURI(encoding)), ReportContext::XTDE1190, where); } else return false; } const int len = result.length(); /* This code is a candidate for threading. Divide and conqueror. */ for(int i = 0; i < len; ++i) { if(!QXmlUtils::isChar(result.at(i))) { if(context) { context->error(QtXmlPatterns::tr("The codepoint %1, occurring in %2 using encoding %3, " "is an invalid XML character.").arg(formatData(result.at(i)), formatURI(uri), formatURI(encoding)), ReportContext::XTDE1190, where); } else return false; } } m_unparsedTexts.insert(qMakePair(uri, encoding), result); return true; } bool AccelTreeResourceLoader::isUnparsedTextAvailable(const QUrl &uri, const QString &encoding) { return retrieveUnparsedText(uri, encoding, ReportContext::Ptr(), 0); } Item AccelTreeResourceLoader::openUnparsedText(const QUrl &uri, const QString &encoding, const ReportContext::Ptr &context, const SourceLocationReflection *const where) { const QString &text = m_unparsedTexts.value(qMakePair(uri, encoding)); if(text.isNull()) { if(retrieveUnparsedText(uri, encoding, context, where)) return openUnparsedText(uri, encoding, context, where); else return Item(); } else return AtomicString::fromValue(text); } QSet AccelTreeResourceLoader::deviceURIs() const { QHash::const_iterator it(m_loadedDocuments.constBegin()); const QHash::const_iterator end(m_loadedDocuments.constEnd()); QSet retval; while (it != end) { if(it.key().toString().startsWith(QLatin1String("tag:trolltech.com,2007:QtXmlPatterns:QIODeviceVariable:"))) retval.insert(it.key()); ++it; } return retval; } void AccelTreeResourceLoader::clear(const QUrl &uri) { m_loadedDocuments.remove(uri); } QT_END_NAMESPACE