/* This file is part of the kblog library. Copyright (c) 2007 Christian Weilbach 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 #include #include #include #include #include #include #include #include #include #include #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(""); atomMarkup += QLatin1String("tag:blogger.com,1999:blog-") + blogId(); atomMarkup += QLatin1String(".post-") + post->postId() + QLatin1String(""); atomMarkup += QLatin1String("") + post->creationDateTime().toString() + QLatin1String(""); atomMarkup += QLatin1String("") + post->modificationDateTime().toString() + QLatin1String(""); atomMarkup += QLatin1String("") + post->title() + QLatin1String(""); if ( post->isPrivate() ) { atomMarkup += QLatin1String(""); atomMarkup += QLatin1String("yes"); } atomMarkup += QLatin1String(""); atomMarkup += QLatin1String("
"); atomMarkup += post->content(); atomMarkup += QLatin1String("
"); QList::ConstIterator it = post->tags().constBegin(); QList::ConstIterator end = post->tags().constEnd(); for ( ; it != end; ++it ) { atomMarkup += QLatin1String(""); } atomMarkup += QLatin1String(""); if ( !fullName().isEmpty() ) { atomMarkup += QLatin1String("") + fullName() + QLatin1String(""); } atomMarkup += QLatin1String("") + username() + QLatin1String(""); atomMarkup += QLatin1String(""); atomMarkup += QLatin1String("
"); 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(""); atomMarkup += QLatin1String("") + post->title() + QLatin1String(""); if ( post->isPrivate() ) { atomMarkup += QLatin1String(""); atomMarkup += QLatin1String("yes"); } atomMarkup += QLatin1String(""); atomMarkup += QLatin1String("
"); atomMarkup += post->content(); // FIXME check for Utf atomMarkup += QLatin1String("
"); QList::ConstIterator it = post->tags().constBegin(); QList::ConstIterator end = post->tags().constEnd(); for ( ; it != end; ++it ) { atomMarkup += QLatin1String(""); } atomMarkup += QLatin1String(""); if ( !fullName().isEmpty() ) { atomMarkup += QLatin1String("") + fullName() + QLatin1String(""); } atomMarkup += QLatin1String("") + username() + QLatin1String(""); atomMarkup += QLatin1String(""); atomMarkup += QLatin1String("
"); 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(""); atomMarkup += QLatin1String("") + comment->title() + QLatin1String(""); atomMarkup += QLatin1String("") + comment->content() + QLatin1String(""); atomMarkup += QLatin1String(""); atomMarkup += QLatin1String("") + comment->name() + QLatin1String(""); atomMarkup += QLatin1String("") + comment->email() + QLatin1String(""); atomMarkup += QLatin1String(""); 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( 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 > blogsList; QList items = feed->items(); QList::ConstIterator it = items.constBegin(); QList::ConstIterator end = items.constEnd(); for ( ; it != end; ++it ) { QRegExp rx( QLatin1String("blog-(\\d+)") ); QMap 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 commentList; QList items = feed->items(); QList::ConstIterator it = items.constBegin(); QList::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 commentList; QList items = feed->items(); QList::ConstIterator it = items.constBegin(); QList::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 postList; QList items = feed->items(); QList::ConstIterator it = items.constBegin(); QList::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 items = feed->items(); QList::ConstIterator it = items.constBegin(); QList::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( 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("(.+)") ); 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( '(.+)' ) matches" << rxPub.cap( 1 ); QRegExp rxUp( QLatin1String("(.+)") ); 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( '(.+)' ) 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( 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("(.+)") ); 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( '(.+)' ) matches" << rxPub.cap( 1 ); QRegExp rxUp( QLatin1String("(.+)") ); 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( '(.+)' ) 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( 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( 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("(.+)") ); 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( '(.+)' ) matches" << rxPub.cap( 1 ); QRegExp rxUp( QLatin1String("(.+)") ); 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( '(.+)' ) 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( 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"