/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** 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 Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/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 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #include "pastebindotcaprotocol.h" #include #include #include #include #include static const char urlC[] = "http://pastebin.ca/"; namespace CodePaster { PasteBinDotCaProtocol::PasteBinDotCaProtocol(const NetworkAccessManagerProxyPtr &nw) : NetworkProtocol(nw), m_fetchReply(0), m_listReply(0), m_pasteReply(0), m_hostChecked(false) { } unsigned PasteBinDotCaProtocol::capabilities() const { return ListCapability | PostDescriptionCapability; } void PasteBinDotCaProtocol::fetch(const QString &id) { QTC_ASSERT(!m_fetchReply, return); const QString url = QLatin1String(urlC); const QString rawPostFix = QLatin1String("raw/"); // Create link as ""http://pastebin.ca/raw/[id]" // If we get a complete URL, just insert 'raw', else build URL. QString link = id; if (link.startsWith(url)) { const int lastSlashPos = link.lastIndexOf(QLatin1Char('/')); if (lastSlashPos != -1) link.insert(lastSlashPos + 1, rawPostFix); } else { link.insert(0, rawPostFix); link.insert(0, url); } m_fetchReply = httpGet(link); connect(m_fetchReply, SIGNAL(finished()), this, SLOT(fetchFinished())); m_fetchId = id; } static QByteArray toTypeId(Protocol::ContentType ct) { if (ct == Protocol::C) return QByteArray(1, '3'); if (ct == Protocol::Cpp) return QByteArray(1, '4'); if (ct == Protocol::JavaScript) return QByteArray("27"); if (ct == Protocol::Diff) return QByteArray( "34"); if (ct == Protocol::Xml) return QByteArray("15"); return QByteArray(1, '1'); } void PasteBinDotCaProtocol::paste(const QString &text, ContentType ct, const QString &username, const QString & /* comment */, const QString &description) { QTC_ASSERT(!m_pasteReply, return); QByteArray data = "api=+xKvWG+1UFXkr2Kn3Td4AnpYtCIjA4qt&"; data += "content="; data += QUrl::toPercentEncoding(fixNewLines(text)); data += "&type="; data += toTypeId(ct); data += "&description="; data += QUrl::toPercentEncoding(description); data += "expiry=1%20day&name="; data += QUrl::toPercentEncoding(username); // fire request const QString link = QLatin1String(urlC) + QLatin1String("quiet-paste.php"); m_pasteReply = httpPost(link, data); connect(m_pasteReply, SIGNAL(finished()), this, SLOT(pasteFinished())); } void PasteBinDotCaProtocol::pasteFinished() { if (m_pasteReply->error()) { qWarning("Pastebin.ca protocol error: %s", qPrintable(m_pasteReply->errorString())); } else { /// returns ""SUCCESS:[id]"" const QByteArray data = m_pasteReply->readAll(); const QString link = QString::fromLatin1(urlC) + QString::fromAscii(data).remove(QLatin1String("SUCCESS:")); emit pasteDone(link); } m_pasteReply->deleteLater(); m_pasteReply = 0; } void PasteBinDotCaProtocol::fetchFinished() { QString title; QString content; bool error = m_fetchReply->error(); if (error) { content = m_fetchReply->errorString(); } else { title = QString::fromLatin1("Pastebin.ca: %1").arg(m_fetchId); const QByteArray data = m_fetchReply->readAll(); content = QString::fromUtf8(data); content.remove(QLatin1Char('\r')); } m_fetchReply->deleteLater(); m_fetchReply = 0; emit fetchDone(title, content, error); } void PasteBinDotCaProtocol::list() { QTC_ASSERT(!m_listReply, return); m_listReply = httpGet(QLatin1String(urlC)); connect(m_listReply, SIGNAL(finished()), this, SLOT(listFinished())); } bool PasteBinDotCaProtocol::checkConfiguration(QString *errorMessage) { if (m_hostChecked) // Check the host once. return true; const bool ok = httpStatus(QLatin1String(urlC), errorMessage); if (ok) m_hostChecked = true; return ok; } /* Quick & dirty: Parse the
-elements with the "Recent Posts" listing * out of the page. \code
...

Create a New Pastebin Post

\endcode */ static inline QStringList parseLists(QIODevice *io) { enum State { OutsideRecentLink, InsideRecentLink }; QStringList rc; const QString classAttribute = QLatin1String("class"); const QString divElement = QLatin1String("div"); const QString anchorElement = QLatin1String("a"); // Start parsing at the 'recent posts' entry as the HTML above is not well-formed // as of 8.4.2010. This will then terminate with an error. QByteArray data = io->readAll(); const QByteArray recentPosts("

Recent Posts

"); const int recentPostsPos = data.indexOf(recentPosts); if (recentPostsPos == -1) return rc; data.remove(0, recentPostsPos + recentPosts.size()); QXmlStreamReader reader(data); State state = OutsideRecentLink; while (!reader.atEnd()) { switch(reader.readNext()) { case QXmlStreamReader::StartElement: // Inside a
of an entry: Anchor or description if (state == InsideRecentLink && reader.name() == anchorElement) { // Anchor // Strip host from link QString link = reader.attributes().value(QLatin1String("href")).toString(); if (link.startsWith(QLatin1Char('/'))) link.remove(0, 1); const QString nameTitle = reader.readElementText(); rc.push_back(link + QLatin1Char(' ') + nameTitle); } else if (state == OutsideRecentLink && reader.name() == divElement) { // "
" state switching if (reader.attributes().value(classAttribute) == QLatin1String("recentlink")) state = InsideRecentLink; } // divElement break; default: break; } // switch reader } // while reader.atEnd() return rc; } void PasteBinDotCaProtocol::listFinished() { const bool error = m_listReply->error(); if (error) { qWarning("pastebin.ca list failed: %s", qPrintable(m_listReply->errorString())); } else { emit listDone(name(), parseLists(m_listReply)); } m_listReply->deleteLater(); m_listReply = 0; } } // namespace CodePaster