kde-playground/kdepimlibs/kblog/gdata.cpp
2015-04-14 21:49:29 +00:00

1005 lines
35 KiB
C++

/*
This file is part of the kblog library.
Copyright (c) 2007 Christian Weilbach <christian_weilbach@web.de>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include "gdata.h"
#include "gdata_p.h"
#include "blogpost.h"
#include "blogcomment.h"
#include <syndication/loader.h>
#include <syndication/item.h>
#include <syndication/category.h>
#include <kio/netaccess.h>
#include <kio/job.h>
#include <KDebug>
#include <KLocalizedString>
#include <KDateTime>
#include <QByteArray>
#include <QRegExp>
#include <QDomDocument>
#define TIMEOUT 600
using namespace KBlog;
GData::GData( const KUrl &server, QObject *parent )
: Blog( server, *new GDataPrivate, parent )
{
kDebug();
setUrl( server );
}
GData::~GData()
{
kDebug();
}
QString GData::interfaceName() const
{
kDebug();
return QLatin1String( "Google Blogger Data" );
}
QString GData::fullName() const
{
kDebug();
return d_func()->mFullName;
}
void GData::setFullName( const QString &fullName )
{
kDebug();
Q_D( GData );
d->mFullName = fullName;
}
QString GData::profileId() const
{
kDebug();
return d_func()->mProfileId;
}
void GData::setProfileId( const QString &pid )
{
kDebug();
Q_D( GData );
d->mProfileId = pid;
}
void GData::fetchProfileId()
{
kDebug();
QByteArray data;
KIO::StoredTransferJob *job = KIO::storedGet( url(), KIO::NoReload, KIO::HideProgressInfo );
KUrl blogUrl = url();
connect( job, SIGNAL(result(KJob*)),
this, SLOT(slotFetchProfileId(KJob*)) );
}
void GData::listBlogs()
{
kDebug();
Syndication::Loader *loader = Syndication::Loader::create();
connect( loader,
SIGNAL(loadingComplete(Syndication::Loader*,Syndication::FeedPtr,Syndication::ErrorCode)),
this,
SLOT(slotListBlogs(Syndication::Loader*,Syndication::FeedPtr,Syndication::ErrorCode)) );
loader->loadFrom( QString::fromLatin1("http://www.blogger.com/feeds/%1/blogs").arg(profileId()) );
}
void GData::listRecentPosts( const QStringList &labels, int number,
const KDateTime &upMinTime, const KDateTime &upMaxTime,
const KDateTime &pubMinTime, const KDateTime &pubMaxTime )
{
kDebug();
Q_D( GData );
QString urlString( QLatin1String("http://www.blogger.com/feeds/") + blogId() + QLatin1String("/posts/default") );
if ( ! labels.empty() ) {
urlString += QLatin1String("/-/") + labels.join( QLatin1String("/") );
}
kDebug() << "listRecentPosts()";
KUrl url( urlString );
if ( !upMinTime.isNull() ) {
url.addQueryItem( QLatin1String("updated-min"), upMinTime.toString() );
}
if ( !upMaxTime.isNull() ) {
url.addQueryItem( QLatin1String("updated-max"), upMaxTime.toString() );
}
if ( !pubMinTime.isNull() ) {
url.addQueryItem( QLatin1String("published-min"), pubMinTime.toString() );
}
if ( !pubMaxTime.isNull() ) {
url.addQueryItem( QLatin1String("published-max"), pubMaxTime.toString() );
}
Syndication::Loader *loader = Syndication::Loader::create();
if ( number > 0 ) {
d->mListRecentPostsMap[ loader ] = number;
}
connect( loader,
SIGNAL(loadingComplete(Syndication::Loader*,Syndication::FeedPtr,Syndication::ErrorCode)),
this,
SLOT(slotListRecentPosts(Syndication::Loader*,Syndication::FeedPtr,Syndication::ErrorCode)) );
loader->loadFrom( url.url() );
}
void GData::listRecentPosts( int number )
{
kDebug();
listRecentPosts( QStringList(), number );
}
void GData::listComments( KBlog::BlogPost *post )
{
kDebug();
Q_D( GData );
Syndication::Loader *loader = Syndication::Loader::create();
d->mListCommentsMap[ loader ] = post;
connect( loader,
SIGNAL(loadingComplete(Syndication::Loader*,Syndication::FeedPtr,Syndication::ErrorCode)),
this,
SLOT(slotListComments(Syndication::Loader*,Syndication::FeedPtr,Syndication::ErrorCode)) );
loader->loadFrom( QString(QLatin1String("http://www.blogger.com/feeds/") + blogId() + QLatin1Char('/') +
post->postId() + QLatin1String("/comments/default")) );
}
void GData::listAllComments()
{
kDebug();
Syndication::Loader *loader = Syndication::Loader::create();
connect( loader,
SIGNAL(loadingComplete(Syndication::Loader*,Syndication::FeedPtr,Syndication::ErrorCode)),
this,
SLOT(slotListAllComments(Syndication::Loader*,Syndication::FeedPtr,Syndication::ErrorCode)) );
loader->loadFrom( QString::fromLatin1("http://www.blogger.com/feeds/%1/comments/default").arg(blogId()) );
}
void GData::fetchPost( KBlog::BlogPost *post )
{
kDebug();
Q_D( GData );
if ( !post ) {
kError() << "post is null pointer";
return;
}
kDebug();
Syndication::Loader *loader = Syndication::Loader::create();
d->mFetchPostMap[ loader ] = post;
connect( loader,
SIGNAL(loadingComplete(Syndication::Loader*,Syndication::FeedPtr,Syndication::ErrorCode)),
this,
SLOT(slotFetchPost(Syndication::Loader*,Syndication::FeedPtr,Syndication::ErrorCode)) );
loader->loadFrom( QString::fromLatin1("http://www.blogger.com/feeds/%1/posts/default").arg(blogId()));
}
void GData::modifyPost( KBlog::BlogPost *post )
{
kDebug();
Q_D( GData );
if ( !post ) {
kError() << "post is null pointer";
return;
}
if ( !d->authenticate() ) {
kError() << "Authentication failed.";
emit errorPost( Atom, i18n( "Authentication failed." ), post );
return;
}
QString atomMarkup = QLatin1String("<entry xmlns='http://www.w3.org/2005/Atom'>");
atomMarkup += QLatin1String("<id>tag:blogger.com,1999:blog-") + blogId();
atomMarkup += QLatin1String(".post-") + post->postId() + QLatin1String("</id>");
atomMarkup += QLatin1String("<published>") + post->creationDateTime().toString() + QLatin1String("</published>");
atomMarkup += QLatin1String("<updated>") + post->modificationDateTime().toString() + QLatin1String("</updated>");
atomMarkup += QLatin1String("<title type='text'>") + post->title() + QLatin1String("</title>");
if ( post->isPrivate() ) {
atomMarkup += QLatin1String("<app:control xmlns:app='http://purl.org/atom/app#'>");
atomMarkup += QLatin1String("<app:draft>yes</app:draft></app:control>");
}
atomMarkup += QLatin1String("<content type='xhtml'>");
atomMarkup += QLatin1String("<div xmlns='http://www.w3.org/1999/xhtml'>");
atomMarkup += post->content();
atomMarkup += QLatin1String("</div></content>");
QList<QString>::ConstIterator it = post->tags().constBegin();
QList<QString>::ConstIterator end = post->tags().constEnd();
for ( ; it != end; ++it ) {
atomMarkup += QLatin1String("<category scheme='http://www.blogger.com/atom/ns#' term='") + ( *it ) + QLatin1String("' />");
}
atomMarkup += QLatin1String("<author>");
if ( !fullName().isEmpty() ) {
atomMarkup += QLatin1String("<name>") + fullName() + QLatin1String("</name>");
}
atomMarkup += QLatin1String("<email>") + username() + QLatin1String("</email>");
atomMarkup += QLatin1String("</author>");
atomMarkup += QLatin1String("</entry>");
QByteArray postData;
QDataStream stream( &postData, QIODevice::WriteOnly );
stream.writeRawData( atomMarkup.toUtf8(), atomMarkup.toUtf8().length() );
KIO::StoredTransferJob *job = KIO::storedHttpPost( postData,
KUrl( QLatin1String("http://www.blogger.com/feeds/") + blogId() + QLatin1String("/posts/default/") + post->postId() ),
KIO::HideProgressInfo );
Q_ASSERT( job );
d->mModifyPostMap[ job ] = post;
job->addMetaData( QLatin1String("content-type"), QLatin1String("Content-Type: application/atom+xml; charset=utf-8") );
job->addMetaData( QLatin1String("ConnectTimeout"), QLatin1String("50") );
job->addMetaData( QLatin1String("UserAgent"), userAgent() );
job->addMetaData( QLatin1String("customHTTPHeader"),
QLatin1String("Authorization: GoogleLogin auth=") + d->mAuthenticationString +
QLatin1String("\r\nX-HTTP-Method-Override: PUT") );
connect( job, SIGNAL(result(KJob*)),
this, SLOT(slotModifyPost(KJob*)) );
}
void GData::createPost( KBlog::BlogPost *post )
{
kDebug();
Q_D( GData );
if ( !post ) {
kError() << "post is null pointer";
return;
}
if ( !d->authenticate() ) {
kError() << "Authentication failed.";
emit errorPost( Atom, i18n( "Authentication failed." ), post );
return;
}
QString atomMarkup = QLatin1String("<entry xmlns='http://www.w3.org/2005/Atom'>");
atomMarkup += QLatin1String("<title type='text'>") + post->title() + QLatin1String("</title>");
if ( post->isPrivate() ) {
atomMarkup += QLatin1String("<app:control xmlns:app='http://purl.org/atom/app#'>");
atomMarkup += QLatin1String("<app:draft>yes</app:draft></app:control>");
}
atomMarkup += QLatin1String("<content type='xhtml'>");
atomMarkup += QLatin1String("<div xmlns='http://www.w3.org/1999/xhtml'>");
atomMarkup += post->content(); // FIXME check for Utf
atomMarkup += QLatin1String("</div></content>");
QList<QString>::ConstIterator it = post->tags().constBegin();
QList<QString>::ConstIterator end = post->tags().constEnd();
for ( ; it != end; ++it ) {
atomMarkup += QLatin1String("<category scheme='http://www.blogger.com/atom/ns#' term='") + ( *it ) + QLatin1String("' />");
}
atomMarkup += QLatin1String("<author>");
if ( !fullName().isEmpty() ) {
atomMarkup += QLatin1String("<name>") + fullName() + QLatin1String("</name>");
}
atomMarkup += QLatin1String("<email>") + username() + QLatin1String("</email>");
atomMarkup += QLatin1String("</author>");
atomMarkup += QLatin1String("</entry>");
QByteArray postData;
QDataStream stream( &postData, QIODevice::WriteOnly );
stream.writeRawData( atomMarkup.toUtf8(), atomMarkup.toUtf8().length() );
KIO::StoredTransferJob *job = KIO::storedHttpPost( postData,
KUrl( QLatin1String("http://www.blogger.com/feeds/") + blogId() + QLatin1String("/posts/default") ),
KIO::HideProgressInfo );
Q_ASSERT ( job );
d->mCreatePostMap[ job ] = post;
job->addMetaData( QLatin1String("content-type"), QLatin1String("Content-Type: application/atom+xml; charset=utf-8") );
job->addMetaData( QLatin1String("ConnectTimeout"), QLatin1String("50") );
job->addMetaData( QLatin1String("UserAgent"), userAgent() );
job->addMetaData( QLatin1String("customHTTPHeader"),
QLatin1String("Authorization: GoogleLogin auth=") + d->mAuthenticationString );
connect( job, SIGNAL(result(KJob*)),
this, SLOT(slotCreatePost(KJob*)) );
}
void GData::removePost( KBlog::BlogPost *post )
{
kDebug();
Q_D( GData );
if ( !post ) {
kError() << "post is null pointer";
return;
}
if ( !d->authenticate() ) {
kError() << "Authentication failed.";
emit errorPost( Atom, i18n( "Authentication failed." ), post );
return;
}
QByteArray postData;
KIO::StoredTransferJob *job = KIO::storedHttpPost( postData,
KUrl( QLatin1String("http://www.blogger.com/feeds/") + blogId() + QLatin1String("/posts/default/") + post->postId() ),
KIO::HideProgressInfo );
d->mRemovePostMap[ job ] = post;
if ( !job ) {
kWarning() << "Unable to create KIO job for http://www.blogger.com/feeds/"
<< blogId() << QLatin1String("/posts/default/") + post->postId();
}
job->addMetaData( QLatin1String("ConnectTimeout"), QLatin1String("50") );
job->addMetaData( QLatin1String("UserAgent"), userAgent() );
job->addMetaData( QLatin1String("customHTTPHeader"),
QLatin1String("Authorization: GoogleLogin auth=") + d->mAuthenticationString +
QLatin1String("\r\nX-HTTP-Method-Override: DELETE") );
connect( job, SIGNAL(result(KJob*)),
this, SLOT(slotRemovePost(KJob*)) );
}
void GData::createComment( KBlog::BlogPost *post, KBlog::BlogComment *comment )
{
kDebug();
if ( !comment ) {
kError() << "comment is null pointer";
return;
}
if ( !post ) {
kError() << "post is null pointer";
return;
}
Q_D( GData );
if ( !d->authenticate() ) {
kError() << "Authentication failed.";
emit errorComment( Atom, i18n( "Authentication failed." ), post, comment );
return;
}
QString atomMarkup = QLatin1String("<entry xmlns='http://www.w3.org/2005/Atom'>");
atomMarkup += QLatin1String("<title type=\"text\">") + comment->title() + QLatin1String("</title>");
atomMarkup += QLatin1String("<content type=\"html\">") + comment->content() + QLatin1String("</content>");
atomMarkup += QLatin1String("<author>");
atomMarkup += QLatin1String("<name>") + comment->name() + QLatin1String("</name>");
atomMarkup += QLatin1String("<email>") + comment->email() + QLatin1String("</email>");
atomMarkup += QLatin1String("</author></entry>");
QByteArray postData;
kDebug() << postData;
QDataStream stream( &postData, QIODevice::WriteOnly );
stream.writeRawData( atomMarkup.toUtf8(), atomMarkup.toUtf8().length() );
KIO::StoredTransferJob *job = KIO::storedHttpPost( postData,
KUrl( QLatin1String("http://www.blogger.com/feeds/") + blogId() + QLatin1String("/") + post->postId() + QLatin1String("/comments/default") ),
KIO::HideProgressInfo );
d->mCreateCommentMap[ job ][post] = comment;
if ( !job ) {
kWarning() << "Unable to create KIO job for http://www.blogger.com/feeds/"
<< blogId() << "/" << post->postId() << "/comments/default";
}
job->addMetaData( QLatin1String("content-type"), QLatin1String("Content-Type: application/atom+xml; charset=utf-8") );
job->addMetaData( QLatin1String("ConnectTimeout"), QLatin1String("50") );
job->addMetaData( QLatin1String("customHTTPHeader"),
QLatin1String("Authorization: GoogleLogin auth=") + d->mAuthenticationString );
job->addMetaData( QLatin1String("UserAgent"), userAgent() );
connect( job, SIGNAL(result(KJob*)),
this, SLOT(slotCreateComment(KJob*)) );
}
void GData::removeComment( KBlog::BlogPost *post, KBlog::BlogComment *comment )
{
kDebug();
Q_D( GData );
kDebug();
if ( !comment ) {
kError() << "comment is null pointer";
return;
}
if ( !post ) {
kError() << "post is null pointer";
return;
}
if ( !d->authenticate() ) {
kError() << "Authentication failed.";
emit errorComment( Atom, i18n( "Authentication failed." ), post, comment );
return;
}
QByteArray postData;
KIO::StoredTransferJob *job = KIO::storedHttpPost(postData,
KUrl( QLatin1String("http://www.blogger.com/feeds/") + blogId() + QLatin1String("/") + post->postId() +
QLatin1String("/comments/default/") + comment->commentId() ), KIO::HideProgressInfo );
d->mRemoveCommentMap[ job ][ post ] = comment;
if ( !job ) {
kWarning() << "Unable to create KIO job for http://www.blogger.com/feeds/"
<< blogId() << post->postId()
<< "/comments/default/" << comment->commentId();
}
job->addMetaData( QLatin1String("ConnectTimeout"), QLatin1String("50") );
job->addMetaData( QLatin1String("UserAgent"), userAgent() );
job->addMetaData( QLatin1String("customHTTPHeader"),
QLatin1String("Authorization: GoogleLogin auth=") +
d->mAuthenticationString + QLatin1String("\r\nX-HTTP-Method-Override: DELETE") );
connect( job, SIGNAL(result(KJob*)),
this, SLOT(slotRemoveComment(KJob*)) );
}
GDataPrivate::GDataPrivate():mAuthenticationString(), mAuthenticationTime()
{
kDebug();
}
GDataPrivate::~GDataPrivate()
{
kDebug();
}
bool GDataPrivate::authenticate()
{
kDebug();
Q_Q( GData );
QByteArray data;
KUrl authGateway( QLatin1String("https://www.google.com/accounts/ClientLogin") );
authGateway.addQueryItem( QLatin1String("Email"), q->username() );
authGateway.addQueryItem( QLatin1String("Passwd"), q->password() );
authGateway.addQueryItem( QLatin1String("source"), q->userAgent() );
authGateway.addQueryItem( QLatin1String("service"), QLatin1String("blogger") );
if ( !mAuthenticationTime.isValid() ||
QDateTime::currentDateTime().toTime_t() - mAuthenticationTime.toTime_t() > TIMEOUT ||
mAuthenticationString.isEmpty() ) {
KIO::Job *job = KIO::http_post( authGateway, QByteArray(), KIO::HideProgressInfo );
if ( KIO::NetAccess::synchronousRun( job, (QWidget*)0, &data, &authGateway ) ) {
QRegExp rx( QLatin1String("Auth=(.+)") );
if ( rx.indexIn( QLatin1String(data) ) != -1 ) {
kDebug() << "RegExp got authentication string:" << rx.cap( 1 );
mAuthenticationString = rx.cap( 1 );
mAuthenticationTime = QDateTime::currentDateTime();
return true;
}
}
return false;
}
return true;
}
void GDataPrivate::slotFetchProfileId( KJob *job )
{
kDebug();
if ( !job ) {
kError() << "job is a null pointer.";
return;
}
Q_Q( GData );
KIO::StoredTransferJob *stj = qobject_cast<KIO::StoredTransferJob*>( job );
const QString data = QString::fromUtf8( stj->data(), stj->data().size() );
if ( !job->error() ) {
QRegExp pid( QLatin1String("http://www.blogger.com/profile/(\\d+)") );
if ( pid.indexIn( data ) != -1 ) {
q->setProfileId( pid.cap( 1 ) );
kDebug() << "QRegExp bid( 'http://www.blogger.com/profile/(\\d+)' matches" << pid.cap( 1 );
emit q->fetchedProfileId( pid.cap( 1 ) );
} else {
kError() << "QRegExp bid( 'http://www.blogger.com/profile/(\\d+)' "
<< " could not regexp the Profile ID";
emit q->error( GData::Other, i18n( "Could not regexp the Profile ID." ) );
emit q->fetchedProfileId( QString() );
}
} else {
kError() << "Job Error: " << job->errorString();
emit q->error( GData::Other, job->errorString() );
emit q->fetchedProfileId( QString() );
}
}
void GDataPrivate::slotListBlogs( Syndication::Loader *loader,
Syndication::FeedPtr feed,
Syndication::ErrorCode status ) {
kDebug();
Q_Q( GData );
if ( !loader ) {
kError() << "loader is a null pointer.";
return;
}
if ( status != Syndication::Success ) {
emit q->error( GData::Atom, i18n( "Could not get blogs." ) );
return;
}
QList<QMap<QString,QString> > blogsList;
QList<Syndication::ItemPtr> items = feed->items();
QList<Syndication::ItemPtr>::ConstIterator it = items.constBegin();
QList<Syndication::ItemPtr>::ConstIterator end = items.constEnd();
for ( ; it != end; ++it ) {
QRegExp rx( QLatin1String("blog-(\\d+)") );
QMap<QString,QString> blogInfo;
if ( rx.indexIn( ( *it )->id() ) != -1 ) {
kDebug() << "QRegExp rx( 'blog-(\\d+)' matches" << rx.cap( 1 );
blogInfo[QLatin1String("id")] = rx.cap( 1 );
blogInfo[QLatin1String("title")] = ( *it )->title();
blogInfo[QLatin1String("url")] = ( *it )->link();
blogInfo[QLatin1String("summary")] = ( *it )->description(); //TODO fix/add more
blogsList << blogInfo;
} else {
kError() << "QRegExp rx( 'blog-(\\d+)' does not match anything in:"
<< ( *it )->id();
emit q->error( GData::Other, i18n( "Could not regexp the blog id path." ) );
}
}
kDebug() << "Emitting listedBlogs(); ";
emit q->listedBlogs( blogsList );
}
void GDataPrivate::slotListComments( Syndication::Loader *loader,
Syndication::FeedPtr feed,
Syndication::ErrorCode status )
{
kDebug();
Q_Q( GData );
if ( !loader ) {
kError() << "loader is a null pointer.";
return;
}
BlogPost *post = mListCommentsMap[ loader ];
mListCommentsMap.remove( loader );
if ( status != Syndication::Success ) {
emit q->errorPost( GData::Atom, i18n( "Could not get comments." ), post );
return;
}
QList<KBlog::BlogComment> commentList;
QList<Syndication::ItemPtr> items = feed->items();
QList<Syndication::ItemPtr>::ConstIterator it = items.constBegin();
QList<Syndication::ItemPtr>::ConstIterator end = items.constEnd();
for ( ; it != end; ++it ) {
BlogComment comment;
QRegExp rx( QLatin1String("post-(\\d+)") );
if ( rx.indexIn( ( *it )->id() ) == -1 ) {
kError() << "QRegExp rx( 'post-(\\d+)' does not match" << rx.cap( 1 );
emit q->error( GData::Other, i18n( "Could not regexp the comment id path." ) );
} else {
comment.setCommentId( rx.cap( 1 ) );
}
kDebug() << "QRegExp rx( 'post-(\\d+)' matches" << rx.cap( 1 );
comment.setTitle( ( *it )->title() );
comment.setContent( ( *it )->content() );
// FIXME: assuming UTC for now
comment.setCreationDateTime(
KDateTime( QDateTime::fromTime_t( ( *it )->datePublished() ),
KDateTime::Spec::UTC() ) );
comment.setModificationDateTime(
KDateTime( QDateTime::fromTime_t( ( *it )->dateUpdated() ),
KDateTime::Spec::UTC() ) );
commentList.append( comment );
}
kDebug() << "Emitting listedComments()";
emit q->listedComments( post, commentList );
}
void GDataPrivate::slotListAllComments( Syndication::Loader *loader,
Syndication::FeedPtr feed,
Syndication::ErrorCode status )
{
kDebug();
Q_Q( GData );
if ( !loader ) {
kError() << "loader is a null pointer.";
return;
}
if ( status != Syndication::Success ) {
emit q->error( GData::Atom, i18n( "Could not get comments." ) );
return;
}
QList<KBlog::BlogComment> commentList;
QList<Syndication::ItemPtr> items = feed->items();
QList<Syndication::ItemPtr>::ConstIterator it = items.constBegin();
QList<Syndication::ItemPtr>::ConstIterator end = items.constEnd();
for ( ; it != end; ++it ) {
BlogComment comment;
QRegExp rx( QLatin1String("post-(\\d+)") );
if ( rx.indexIn( ( *it )->id() ) == -1 ) {
kError() << "QRegExp rx( 'post-(\\d+)' does not match" << rx.cap( 1 );
emit q->error( GData::Other, i18n( "Could not regexp the comment id path." ) );
} else {
comment.setCommentId( rx.cap( 1 ) );
}
kDebug() << "QRegExp rx( 'post-(\\d+)' matches" << rx.cap( 1 );
comment.setTitle( ( *it )->title() );
comment.setContent( ( *it )->content() );
// FIXME: assuming UTC for now
comment.setCreationDateTime(
KDateTime( QDateTime::fromTime_t( ( *it )->datePublished() ),
KDateTime::Spec::UTC() ) );
comment.setModificationDateTime(
KDateTime( QDateTime::fromTime_t( ( *it )->dateUpdated() ),
KDateTime::Spec::UTC() ) );
commentList.append( comment );
}
kDebug() << "Emitting listedAllComments()";
emit q->listedAllComments( commentList );
}
void GDataPrivate::slotListRecentPosts( Syndication::Loader *loader,
Syndication::FeedPtr feed,
Syndication::ErrorCode status ) {
kDebug();
Q_Q( GData );
if ( !loader ) {
kError() << "loader is a null pointer.";
return;
}
if ( status != Syndication::Success ) {
emit q->error( GData::Atom, i18n( "Could not get posts." ) );
return;
}
int number = 0;
if ( mListRecentPostsMap.contains( loader ) ) {
number = mListRecentPostsMap[ loader ];
}
mListRecentPostsMap.remove( loader );
QList<KBlog::BlogPost> postList;
QList<Syndication::ItemPtr> items = feed->items();
QList<Syndication::ItemPtr>::ConstIterator it = items.constBegin();
QList<Syndication::ItemPtr>::ConstIterator end = items.constEnd();
for ( ; it != end; ++it ) {
BlogPost post;
QRegExp rx( QLatin1String("post-(\\d+)") );
if ( rx.indexIn( ( *it )->id() ) == -1 ) {
kError() << "QRegExp rx( 'post-(\\d+)' does not match" << rx.cap( 1 );
emit q->error( GData::Other, i18n( "Could not regexp the post id path." ) );
} else {
post.setPostId( rx.cap( 1 ) );
}
kDebug() << "QRegExp rx( 'post-(\\d+)' matches" << rx.cap( 1 );
post.setTitle( ( *it )->title() );
post.setContent( ( *it )->content() );
post.setLink( ( *it )->link() );
QStringList labels;
int catCount = ( *it )->categories().count();
QList< Syndication::CategoryPtr > cats = ( *it )->categories();
for ( int i=0; i < catCount; ++i ) {
if ( cats[i].get()->label().isEmpty() ) {
labels.append( cats[i].get()->term() );
} else {
labels.append( cats[i].get()->label() );
}
}
post.setTags( labels );
// FIXME: assuming UTC for now
post.setCreationDateTime(
KDateTime( QDateTime::fromTime_t( ( *it )->datePublished() ),
KDateTime::Spec::UTC() ).toLocalZone() );
post.setModificationDateTime(
KDateTime( QDateTime::fromTime_t( ( *it )->dateUpdated() ),
KDateTime::Spec::UTC() ).toLocalZone() );
post.setStatus( BlogPost::Fetched );
postList.append( post );
if ( number-- == 0 ) {
break;
}
}
kDebug() << "Emitting listedRecentPosts()";
emit q->listedRecentPosts( postList );
}
void GDataPrivate::slotFetchPost( Syndication::Loader *loader,
Syndication::FeedPtr feed,
Syndication::ErrorCode status )
{
kDebug();
Q_Q( GData );
if ( !loader ) {
kError() << "loader is a null pointer.";
return;
}
bool success = false;
BlogPost *post = mFetchPostMap.take( loader );
kError() << "Post" << post;
post->postId();
if ( status != Syndication::Success ) {
emit q->errorPost( GData::Atom, i18n( "Could not get posts." ), post );
return;
}
QString postId = post->postId();
QList<Syndication::ItemPtr> items = feed->items();
QList<Syndication::ItemPtr>::ConstIterator it = items.constBegin();
QList<Syndication::ItemPtr>::ConstIterator end = items.constEnd();
for ( ; it != end; ++it ) {
QRegExp rx( QLatin1String("post-(\\d+)") );
if ( rx.indexIn( ( *it )->id() ) != -1 &&
rx.cap( 1 ) == postId ) {
kDebug() << "QRegExp rx( 'post-(\\d+)' matches" << rx.cap( 1 );
post->setPostId( rx.cap( 1 ) );
post->setTitle( ( *it )->title() );
post->setContent( ( *it )->content() );
post->setStatus( BlogPost::Fetched );
post->setLink( ( *it )->link() );
post->setCreationDateTime(
KDateTime( QDateTime::fromTime_t( ( *it )->datePublished() ),
KDateTime::Spec::UTC() ).toLocalZone() );
post->setModificationDateTime(
KDateTime( QDateTime::fromTime_t( ( *it )->dateUpdated() ),
KDateTime::Spec::UTC() ).toLocalZone() );
kDebug() << "Emitting fetchedPost( postId=" << postId << ");";
success = true;
emit q->fetchedPost( post );
break;
}
}
if ( !success ) {
kError() << "QRegExp rx( 'post-(\\d+)' does not match"
<< mFetchPostMap[ loader ]->postId() << ".";
emit q->errorPost( GData::Other, i18n( "Could not regexp the blog id path." ), post );
}
}
void GDataPrivate::slotCreatePost( KJob *job )
{
kDebug();
if ( !job ) {
kError() << "job is a null pointer.";
return;
}
KIO::StoredTransferJob *stj = qobject_cast<KIO::StoredTransferJob*>( job );
const QString data = QString::fromUtf8( stj->data(), stj->data().size() );
Q_Q( GData );
KBlog::BlogPost *post = mCreatePostMap[ job ];
mCreatePostMap.remove( job );
if ( job->error() != 0 ) {
kError() << "slotCreatePost error:" << job->errorString();
emit q->errorPost( GData::Atom, job->errorString(), post );
return;
}
QRegExp rxId( QLatin1String("post-(\\d+)") ); //FIXME check and do better handling, esp the creation date time
if ( rxId.indexIn( data ) == -1 ) {
kError() << "Could not regexp the id out of the result:" << data;
emit q->errorPost( GData::Atom,
i18n( "Could not regexp the id out of the result." ), post );
return;
}
kDebug() << "QRegExp rx( 'post-(\\d+)' ) matches" << rxId.cap( 1 );
QRegExp rxPub( QLatin1String("<published>(.+)</published>") );
if ( rxPub.indexIn( data ) == -1 ) {
kError() << "Could not regexp the published time out of the result:" << data;
emit q->errorPost( GData::Atom,
i18n( "Could not regexp the published time out of the result." ), post );
return;
}
kDebug() << "QRegExp rx( '<published>(.+)</published>' ) matches" << rxPub.cap( 1 );
QRegExp rxUp( QLatin1String("<updated>(.+)</updated>") );
if ( rxUp.indexIn( data ) == -1 ) {
kError() << "Could not regexp the update time out of the result:" << data;
emit q->errorPost( GData::Atom,
i18n( "Could not regexp the update time out of the result." ), post );
return;
}
kDebug() << "QRegExp rx( '<updated>(.+)</updated>' ) matches" << rxUp.cap( 1 );
post->setPostId( rxId.cap( 1 ) );
post->setCreationDateTime( KDateTime().fromString( rxPub.cap( 1 ) ).toLocalZone() );
post->setModificationDateTime( KDateTime().fromString( rxUp.cap( 1 ) ) );
post->setStatus( BlogPost::Created );
kDebug() << "Emitting createdPost()";
emit q->createdPost( post );
}
void GDataPrivate::slotModifyPost( KJob *job )
{
kDebug();
if ( !job ) {
kError() << "job is a null pointer.";
return;
}
KIO::StoredTransferJob *stj = qobject_cast<KIO::StoredTransferJob*>( job );
const QString data = QString::fromUtf8( stj->data(), stj->data().size() );
KBlog::BlogPost *post = mModifyPostMap[ job ];
mModifyPostMap.remove( job );
Q_Q( GData );
if ( job->error() != 0 ) {
kError() << "slotModifyPost error:" << job->errorString();
emit q->errorPost( GData::Atom, job->errorString(), post );
return;
}
QRegExp rxId( QLatin1String("post-(\\d+)") ); //FIXME check and do better handling, esp creation date time
if ( rxId.indexIn( data ) == -1 ) {
kError() << "Could not regexp the id out of the result:" << data;
emit q->errorPost( GData::Atom,
i18n( "Could not regexp the id out of the result." ), post );
return;
}
kDebug() << "QRegExp rx( 'post-(\\d+)' ) matches" << rxId.cap( 1 );
QRegExp rxPub( QLatin1String("<published>(.+)</published>") );
if ( rxPub.indexIn( data ) == -1 ) {
kError() << "Could not regexp the published time out of the result:" << data;
emit q->errorPost( GData::Atom,
i18n( "Could not regexp the published time out of the result." ), post );
return;
}
kDebug() << "QRegExp rx( '<published>(.+)</published>' ) matches" << rxPub.cap( 1 );
QRegExp rxUp( QLatin1String("<updated>(.+)</updated>") );
if ( rxUp.indexIn( data ) == -1 ) {
kError() << "Could not regexp the update time out of the result:" << data;
emit q->errorPost( GData::Atom,
i18n( "Could not regexp the update time out of the result." ), post );
return;
}
kDebug() << "QRegExp rx( '<updated>(.+)</updated>' ) matches" << rxUp.cap( 1 );
post->setPostId( rxId.cap( 1 ) );
post->setCreationDateTime( KDateTime().fromString( rxPub.cap( 1 ) ) );
post->setModificationDateTime( KDateTime().fromString( rxUp.cap( 1 ) ) );
post->setStatus( BlogPost::Modified );
emit q->modifiedPost( post );
}
void GDataPrivate::slotRemovePost( KJob *job )
{
kDebug();
if ( !job ) {
kError() << "job is a null pointer.";
return;
}
KIO::StoredTransferJob *stj = qobject_cast<KIO::StoredTransferJob*>( job );
const QString data = QString::fromUtf8( stj->data(), stj->data().size() );
KBlog::BlogPost *post = mRemovePostMap[ job ];
mRemovePostMap.remove( job );
Q_Q( GData );
if ( job->error() != 0 ) {
kError() << "slotRemovePost error:" << job->errorString();
emit q->errorPost( GData::Atom, job->errorString(), post );
return;
}
post->setStatus( BlogPost::Removed );
kDebug() << "Emitting removedPost()";
emit q->removedPost( post );
}
void GDataPrivate::slotCreateComment( KJob *job )
{
kDebug();
if ( !job ) {
kError() << "job is a null pointer.";
return;
}
KIO::StoredTransferJob *stj = qobject_cast<KIO::StoredTransferJob*>( job );
const QString data = QString::fromUtf8( stj->data(), stj->data().size() );
kDebug() << "Dump data: " << data;
Q_Q( GData );
KBlog::BlogComment *comment = mCreateCommentMap[ job ].values().first();
KBlog::BlogPost *post = mCreateCommentMap[ job ].keys().first();
mCreateCommentMap.remove( job );
if ( job->error() != 0 ) {
kError() << "slotCreateComment error:" << job->errorString();
emit q->errorComment( GData::Atom, job->errorString(), post, comment );
return;
}
// TODO check for result and fit appropriately
QRegExp rxId( QLatin1String("post-(\\d+)") );
if ( rxId.indexIn( data ) == -1 ) {
kError() << "Could not regexp the id out of the result:" << data;
emit q->errorPost( GData::Atom,
i18n( "Could not regexp the id out of the result." ), post );
return;
}
kDebug() << "QRegExp rx( 'post-(\\d+)' ) matches" << rxId.cap( 1 );
QRegExp rxPub( QLatin1String("<published>(.+)</published>") );
if ( rxPub.indexIn( data ) == -1 ) {
kError() << "Could not regexp the published time out of the result:" << data;
emit q->errorPost( GData::Atom,
i18n( "Could not regexp the published time out of the result." ), post );
return;
}
kDebug() << "QRegExp rx( '<published>(.+)</published>' ) matches" << rxPub.cap( 1 );
QRegExp rxUp( QLatin1String("<updated>(.+)</updated>") );
if ( rxUp.indexIn( data ) == -1 ) {
kError() << "Could not regexp the update time out of the result:" << data;
emit q->errorPost( GData::Atom,
i18n( "Could not regexp the update time out of the result." ), post );
return;
}
kDebug() << "QRegExp rx( '<updated>(.+)</updated>' ) matches" << rxUp.cap( 1 );
comment->setCommentId( rxId.cap( 1 ) );
comment->setCreationDateTime( KDateTime().fromString( rxPub.cap( 1 ) ) );
comment->setModificationDateTime( KDateTime().fromString( rxUp.cap( 1 ) ) );
comment->setStatus( BlogComment::Created );
kDebug() << "Emitting createdComment()";
emit q->createdComment( post, comment );
}
void GDataPrivate::slotRemoveComment( KJob *job )
{
kDebug();
if ( !job ) {
kError() << "job is a null pointer.";
return;
}
KIO::StoredTransferJob *stj = qobject_cast<KIO::StoredTransferJob*>( job );
const QString data = QString::fromUtf8( stj->data(), stj->data().size() );
Q_Q( GData );
KBlog::BlogComment *comment = mRemoveCommentMap[ job ].values().first();
KBlog::BlogPost *post = mRemoveCommentMap[ job ].keys().first();
mRemoveCommentMap.remove( job );
if ( job->error() != 0 ) {
kError() << "slotRemoveComment error:" << job->errorString();
emit q->errorComment( GData::Atom, job->errorString(), post, comment );
return;
}
comment->setStatus( BlogComment::Created );
kDebug() << "Emitting removedComment()";
emit q->removedComment( post, comment );
}
#include "moc_gdata.cpp"