/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** 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 General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** 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-3.0.html. ** ****************************************************************************/ #include "pastebindotcomprotocol.h" #include #include #include #include #include #include #include enum { debug = 0 }; static const char PASTEBIN_BASE[]="https://pastebin.com/"; static const char PASTEBIN_API[]="api/api_post.php"; static const char PASTEBIN_RAW[]="raw/"; static const char PASTEBIN_ARCHIVE[]="archive"; static const char API_KEY[]="api_dev_key=516686fc461fb7f9341fd7cf2af6f829&"; // user: qtcreator_apikey static const char PROTOCOL_NAME[] = "Pastebin.Com"; namespace CodePaster { QString PasteBinDotComProtocol::protocolName() { return QLatin1String(PROTOCOL_NAME); } unsigned PasteBinDotComProtocol::capabilities() const { return ListCapability | PostDescriptionCapability; } static inline QByteArray format(Protocol::ContentType ct) { QByteArray format = "api_paste_format="; switch (ct) { case Protocol::C: format += 'c'; break; case Protocol::Cpp: format += "cpp-qt"; break; case Protocol::JavaScript: format += "javascript"; break; case Protocol::Diff: format += "diff"; break; case Protocol::Xml: format += "xml"; break; case Protocol::Text: // fallthrough! default: format += "text"; } format += '&'; return format; } // According to documentation, Pastebin.com accepts only fixed expiry specifications: // N = Never, 10M = 10 Minutes, 1H = 1 Hour, 1D = 1 Day, 1W = 1 Week, 2W = 2 Weeks, 1M = 1 Month, // however, 1W and 2W do not work. static inline QByteArray expirySpecification(int expiryDays) { if (expiryDays <= 1) return QByteArray("1D"); if (expiryDays <= 31) return QByteArray("1M"); return QByteArray("N"); } void PasteBinDotComProtocol::paste(const QString &text, ContentType ct, int expiryDays, const QString & /* username */, // Not used unless registered user const QString &comment, const QString &description) { Q_UNUSED(comment) Q_UNUSED(description) QTC_ASSERT(!m_pasteReply, return); // Format body QByteArray pasteData = API_KEY; pasteData += "api_option=paste&"; pasteData += "api_paste_expire_date="; pasteData += expirySpecification(expiryDays); pasteData += '&'; pasteData += format(ct); pasteData += "api_paste_name="; // Title or name. pasteData += QUrl::toPercentEncoding(description); pasteData += "&api_paste_code="; pasteData += QUrl::toPercentEncoding(fixNewLines(text)); // fire request m_pasteReply = httpPost(QLatin1String(PASTEBIN_BASE) + QLatin1String(PASTEBIN_API), pasteData); connect(m_pasteReply, &QNetworkReply::finished, this, &PasteBinDotComProtocol::pasteFinished); if (debug) qDebug() << "paste: sending " << m_pasteReply << pasteData; } void PasteBinDotComProtocol::pasteFinished() { if (m_pasteReply->error()) qWarning("%s protocol error: %s", PROTOCOL_NAME, qPrintable(m_pasteReply->errorString())); else emit pasteDone(QString::fromLatin1(m_pasteReply->readAll())); m_pasteReply->deleteLater(); m_pasteReply = nullptr; } void PasteBinDotComProtocol::fetch(const QString &id) { // Did we get a complete URL or just an id. Insert a call to the php-script QString link = QLatin1String(PASTEBIN_BASE) + QLatin1String(PASTEBIN_RAW); if (id.startsWith(QLatin1String("http://"))) link.append(id.midRef(id.lastIndexOf(QLatin1Char('/')) + 1)); else link.append(id); if (debug) qDebug() << "fetch: sending " << link; m_fetchReply = httpGet(link); connect(m_fetchReply, &QNetworkReply::finished, this, &PasteBinDotComProtocol::fetchFinished); m_fetchId = id; } void PasteBinDotComProtocol::fetchFinished() { QString title; QString content; const bool error = m_fetchReply->error(); if (error) { content = m_fetchReply->errorString(); if (debug) qDebug() << "fetchFinished: error" << m_fetchId << content; } else { title = QLatin1String(PROTOCOL_NAME) + QLatin1String(": ") + m_fetchId; content = QString::fromUtf8(m_fetchReply->readAll()); if (debug) { QDebug nsp = qDebug().nospace(); nsp << "fetchFinished: " << content.size() << " Bytes"; if (debug > 1) nsp << content; } } m_fetchReply->deleteLater(); m_fetchReply = nullptr; emit fetchDone(title, content, error); } void PasteBinDotComProtocol::list() { QTC_ASSERT(!m_listReply, return); const QString url = QLatin1String(PASTEBIN_BASE) + QLatin1String(PASTEBIN_ARCHIVE); m_listReply = httpGet(url); connect(m_listReply, &QNetworkReply::finished, this, &PasteBinDotComProtocol::listFinished); if (debug) qDebug() << "list: sending " << url << m_listReply; } /* Quick & dirty: Parse out the 'archive' table as of 16.3.2016: \code ... -\endcode */ enum ParseState { OutSideTable, WithinTable, WithinTableRow, WithinTableHeaderElement, WithinTableElement, WithinTableElementAnchor, ParseError }; QDebug operator<<(QDebug d, const QXmlStreamAttributes &al) { QDebug nospace = d.nospace(); foreach (const QXmlStreamAttribute &a, al) nospace << a.name().toString() << '=' << a.value().toString() << ' '; return d; } static inline ParseState nextOpeningState(ParseState current, const QXmlStreamReader &reader) { const QStringRef &element = reader.name(); switch (current) { case OutSideTable: // Trigger on main table only. if (element == QLatin1String("table") && reader.attributes().value(QLatin1String("class")) == QLatin1String("maintable")) return WithinTable; return OutSideTable; case WithinTable: if (element == QLatin1String("tr")) return WithinTableRow; break; case WithinTableRow: if (element == QLatin1String("td")) return WithinTableElement; if (element == QLatin1String("th")) return WithinTableHeaderElement; break; case WithinTableElement: if (element == QLatin1String("img")) return WithinTableElement; if (element == QLatin1String("a")) return WithinTableElementAnchor; break; case WithinTableHeaderElement: case WithinTableElementAnchor: case ParseError: break; } return ParseError; } static inline ParseState nextClosingState(ParseState current, const QStringRef &element) { switch (current) { case OutSideTable: return OutSideTable; case WithinTable: if (element == QLatin1String("table")) return OutSideTable; break; case WithinTableRow: if (element == QLatin1String("tr")) return WithinTable; break; case WithinTableElement: if (element == QLatin1String("td")) return WithinTableRow; if (element == QLatin1String("img")) return WithinTableElement; break; case WithinTableHeaderElement: if (element == QLatin1String("th")) return WithinTableRow; break; case WithinTableElementAnchor: if (element == QLatin1String("a")) return WithinTableElement; break; case ParseError: break; } return ParseError; } static inline QStringList parseLists(QIODevice *io, QString *errorMessage) { enum { maxEntries = 200 }; // Limit the archive, which can grow quite large. QStringList rc; // Read answer and delete part up to the main table since the input form has // parts that can no longer be parsed using XML parsers () QByteArray data = io->readAll(); const int tablePos = data.indexOf("
Name / Title Posted Syntax
Vector 1 2 sec ago C++
"); if (tablePos < 0) { *errorMessage = QLatin1String("Could not locate beginning of table."); return rc; } data.remove(0, tablePos); QXmlStreamReader reader(data); ParseState state = OutSideTable; int tableRow = 0; int tableColumn = 0; const QString hrefAttribute = QLatin1String("href"); QString link; QString title; QString age; while (!reader.atEnd()) { switch (reader.readNext()) { case QXmlStreamReader::StartElement: state = nextOpeningState(state, reader); switch (state) { case WithinTableRow: tableColumn = 0; break; case OutSideTable: case WithinTable: case WithinTableHeaderElement: case WithinTableElement: break; case WithinTableElementAnchor: // 'href="/svb5K8wS"' if (tableColumn == 0) { link = reader.attributes().value(hrefAttribute).toString(); if (link.startsWith(QLatin1Char('/'))) link.remove(0, 1); } break; case ParseError: return rc; } // switch startelement state break; case QXmlStreamReader::EndElement: state = nextClosingState(state, reader.name()); switch (state) { case OutSideTable: if (tableRow) // Seen the table, bye. return rc; break; case WithinTable: // User can occasionally be empty. if (tableRow && !link.isEmpty() && !title.isEmpty() && !age.isEmpty()) { QString entry = link; entry += QLatin1Char(' '); entry += title; entry += QLatin1String(" ("); entry += age; entry += QLatin1Char(')'); rc.push_back(entry); if (rc.size() >= maxEntries) return rc; } tableRow++; age.clear(); link.clear(); title.clear(); break; case WithinTableRow: tableColumn++; break; case WithinTableHeaderElement: case WithinTableElement: case WithinTableElementAnchor: break; case ParseError: return rc; } // switch endelement state break; case QXmlStreamReader::Characters: switch (state) { case WithinTableElement: if (tableColumn == 1) age = reader.text().toString(); break; case WithinTableElementAnchor: if (tableColumn == 0) title = reader.text().toString(); break; default: break; } // switch characters read state break; default: break; } // switch reader state } if (reader.hasError()) *errorMessage = QString::fromLatin1("Error at line %1:%2").arg(reader.lineNumber()).arg(reader.errorString()); return rc; } void PasteBinDotComProtocol::listFinished() { const bool error = m_listReply->error(); if (error) { if (debug) qDebug() << "listFinished: error" << m_listReply->errorString(); } else { QString errorMessage; const QStringList list = parseLists(m_listReply, &errorMessage); if (list.isEmpty()) qWarning().nospace() << "Failed to read list from " << PASTEBIN_BASE << ':' << errorMessage; emit listDone(name(), list); if (debug) qDebug() << list; } m_listReply->deleteLater(); m_listReply = nullptr; } } // namespace CodePaster