/* * libretroshare/src/services: p3Blogs.cc * * RetroShare C++ Interface. * * Copyright 2008 by Robert Fernie. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License Version 2 as published by the Free Software Foundation. * * 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; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 * USA. * * Please report all bugs and problems to "retroshare@lunamutt.com". * */ #include "services/p3blogs.h" #include "util/rsdir.h" std::ostream &operator<<(std::ostream &out, const BlogInfo &info) { std::string name(info.blogName.begin(), info.blogName.end()); std::string desc(info.blogDesc.begin(), info.blogDesc.end()); out << "BlogInfo:"; out << std::endl; out << "BlogId: " << info.blogId << std::endl; out << "BlogName: " << name << std::endl; out << "BlogDesc: " << desc << std::endl; out << "BlogFlags: " << info.blogFlags << std::endl; out << "Pop: " << info.pop << std::endl; out << "LastPost: " << info.lastPost << std::endl; return out; } std::ostream &operator<<(std::ostream &out, const BlogMsgSummary &info) { out << "BlogMsgSummary:"; out << std::endl; out << "BlogId: " << info.blogId << std::endl; return out; } std::ostream &operator<<(std::ostream &out, const BlogMsgInfo &info) { out << "BlogMsgInfo:"; out << std::endl; out << "BlogId: " << info.blogId << std::endl; return out; } RsBlogs *rsBlogs = NULL; /* Blogs will be initially stored for 1 year * remember 2^16 = 64K max units in store period. * PUBPERIOD * 2^16 = max STORE PERIOD */ #define BLOG_STOREPERIOD (90*24*3600) /* 30 * 24 * 3600 - secs in a year */ #define BLOG_PUBPERIOD 600 /* 10 minutes ... (max = 455 days) */ p3Blogs::p3Blogs(uint16_t type, CacheStrapper *cs, CacheTransfer *cft, std::string srcdir, std::string storedir) :p3GroupDistrib(type, cs, cft, srcdir, storedir, "", CONFIG_TYPE_QBLOG, BLOG_STOREPERIOD, BLOG_PUBPERIOD) { return; } p3Blogs::~p3Blogs() { return; } /****************************************/ bool p3Blogs::blogsChanged(std::list &blogIds) { return groupsChanged(blogIds); } bool p3Blogs::getBlogInfo(std::string cId, BlogInfo &ci) { RsStackMutex stack(distribMtx); /***** STACK LOCKED MUTEX *****/ /* extract details */ GroupInfo *gi = locked_getGroupInfo(cId); if (!gi) return false; ci.blogId = gi->grpId; ci.blogName = gi->grpName; ci.blogDesc = gi->grpDesc; ci.blogFlags = gi->flags; ci.pop = gi->sources.size(); ci.lastPost = gi->lastPost; ci.pngChanImage = gi->grpIcon.pngImageData; if(ci.pngChanImage != NULL) ci.pngImageLen = gi->grpIcon.imageSize; else ci.pngImageLen = 0; return true; } bool p3Blogs::getBlogList(std::list &blogList) { std::list grpIds; std::list::iterator it; getAllGroupList(grpIds); for(it = grpIds.begin(); it != grpIds.end(); it++) { BlogInfo ci; if (getBlogInfo(*it, ci)) { blogList.push_back(ci); } } return true; } bool p3Blogs::getBlogMsgList(std::string cId, std::list &msgs) { std::list msgIds; std::list::iterator it; getAllMsgList(cId, msgIds); RsStackMutex stack(distribMtx); /***** STACK LOCKED MUTEX *****/ for(it = msgIds.begin(); it != msgIds.end(); it++) { /* get details */ RsDistribMsg *msg = locked_getGroupMsg(cId, *it); RsBlogMsg *cmsg = dynamic_cast(msg); if (!cmsg) continue; BlogMsgSummary tis; tis.blogId = msg->grpId; tis.msgId = msg->msgId; tis.ts = msg->timestamp; /* the rest must be gotten from the derived Msg */ tis.subject = cmsg->subject; tis.msg = cmsg->message; msgs.push_back(tis); } return true; } bool p3Blogs::getBlogMessage(std::string fId, std::string mId, BlogMsgInfo &info) { std::list msgIds; std::list::iterator it; RsStackMutex stack(distribMtx); /***** STACK LOCKED MUTEX *****/ RsDistribMsg *msg = locked_getGroupMsg(fId, mId); RsBlogMsg *cmsg = dynamic_cast(msg); if (!cmsg) return false; info.blogId = msg->grpId; info.msgId = msg->msgId; info.ts = msg->timestamp; /* the rest must be gotten from the derived Msg */ info.subject = cmsg->subject; info.msg = cmsg->message; return true; } bool p3Blogs::BlogMessageSend(BlogMsgInfo &info) { RsBlogMsg *cmsg = new RsBlogMsg(); cmsg->grpId = info.blogId; cmsg->subject = info.subject; cmsg->message = info.msg; cmsg->timestamp = time(NULL); std::string msgId = publishMsg(cmsg, true); return true; } bool p3Blogs::BlogMessageReply(BlogMsgInfo& reply){ // ensure it has a value if(isReply(reply)){ std::cerr << "p3Blogs::BlogMessageReply()" << " This is not a reply " << std::endl; return false; } // also check that msgId exists for group return BlogMessageSend(reply); } bool p3Blogs::isReply(BlogMsgInfo& info){ // if replies list is empty then this is not a reply to a blog return !info.msgIdReply.empty(); } std::string p3Blogs::createBlog(std::wstring blogName, std::wstring blogDesc, uint32_t blogFlags, unsigned char* pngImageData, uint32_t imageSize) { std::string id = createGroup(blogName, blogDesc, blogFlags, pngImageData, imageSize); return id; } RsSerialType *p3Blogs::createSerialiser() { return new RsBlogSerialiser(); } bool p3Blogs::locked_checkDistribMsg(RsDistribMsg *msg) { return true; } RsDistribGrp *p3Blogs::locked_createPublicDistribGrp(GroupInfo &info) { RsDistribGrp *grp = NULL; //new RsChannelGrp(); return grp; } RsDistribGrp *p3Blogs::locked_createPrivateDistribGrp(GroupInfo &info) { RsDistribGrp *grp = NULL; //new RsChannelGrp(); return grp; } bool p3Blogs::blogSubscribe(std::string cId, bool subscribe) { std::cerr << "p3Blogs::channelSubscribe() "; std::cerr << cId; std::cerr << std::endl; return subscribeToGroup(cId, subscribe); } /***************************************************************************************/ /****************** Event Feedback (Overloaded form p3distrib) *************************/ /***************************************************************************************/ /* only download in the first week of channel * older stuff can be manually downloaded. */ const uint32_t DOWNLOAD_PERIOD = 7 * 24 * 3600; /* This is called when we receive a msg, and also recalled * */ bool p3Blogs::locked_eventDuplicateMsg(GroupInfo *grp, RsDistribMsg *msg, const std::string& id, bool historical) { return true; } #include "pqi/pqinotify.h" bool p3Blogs::locked_eventNewMsg(GroupInfo *grp, RsDistribMsg *msg, const std::string& id, bool historical) { std::string grpId = msg->grpId; std::string msgId = msg->msgId; std::string nullId; std::cerr << "p3Blogs::locked_eventNewMsg() "; std::cerr << " grpId: " << grpId; std::cerr << " msgId: " << msgId; std::cerr << " peerId: " << id; std::cerr << std::endl; if (!historical) { getPqiNotify()->AddFeedItem(RS_FEED_ITEM_BLOG_MSG, grpId, msgId, nullId); } /* request the files * NB: This could result in duplicates. * which must be handled by ft side. * * this is exactly what DuplicateMsg does. * */ return locked_eventDuplicateMsg(grp, msg, id, historical); } void p3Blogs::locked_notifyGroupChanged(GroupInfo &grp, uint32_t flags, bool historical) { std::string grpId = grp.grpId; std::string msgId; std::string nullId; std::cerr << "p3Blogs::locked_notifyGroupChanged() "; std::cerr << grpId; std::cerr << " flags:" << flags; std::cerr << std::endl; switch(flags) { case GRP_NEW_UPDATE: std::cerr << "p3Blogs::locked_notifyGroupChanged() NEW UPDATE"; std::cerr << std::endl; if (!historical) { getPqiNotify()->AddFeedItem(RS_FEED_ITEM_BLOG_NEW, grpId, msgId, nullId); } break; case GRP_UPDATE: std::cerr << "p3Blogs::locked_notifyGroupChanged() UPDATE"; std::cerr << std::endl; if (!historical) { getPqiNotify()->AddFeedItem(RS_FEED_ITEM_BLOG_UPDATE, grpId, msgId, nullId); } break; case GRP_LOAD_KEY: std::cerr << "p3Blogs::locked_notifyGroupChanged() LOAD_KEY"; std::cerr << std::endl; break; case GRP_NEW_MSG: std::cerr << "p3Blogs::locked_notifyGroupChanged() NEW MSG"; std::cerr << std::endl; break; case GRP_SUBSCRIBED: std::cerr << "p3Blogs::locked_notifyGroupChanged() SUBSCRIBED"; std::cerr << std::endl; break; case GRP_UNSUBSCRIBED: std::cerr << "p3Blogs::locked_notifyGroupChanged() UNSUBSCRIBED"; std::cerr << std::endl; /* won't stop downloads... */ break; default: std::cerr << "p3Blogs::locked_notifyGroupChanged() Unknown DEFAULT"; std::cerr << std::endl; break; } return p3GroupDistrib::locked_notifyGroupChanged(grp, flags); } //TODO: if you want to enable config saving and loading implement this bool p3Blogs::childLoadList(std::list& configSaves) { return true; } //TODO: std::list p3Blogs::childSaveList() { std::list saveL; return saveL; } /****************************************/