2018-05-29 15:54:27 -04:00
/*******************************************************************************
* libretroshare / src / services : p3msgservice . cc *
* *
* libretroshare : retroshare core library *
* *
* Copyright 2004 - 2008 Robert Fernie < retroshare @ lunamutt . com > *
* *
* This program is free software : you can redistribute it and / or modify *
* it under the terms of the GNU Lesser General Public License as *
* published by the Free Software Foundation , either version 3 of the *
* License , or ( at your option ) any later version . *
* *
* This program 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 Lesser General Public License for more details . *
* *
* You should have received a copy of the GNU Lesser General Public License *
* along with this program . If not , see < https : //www.gnu.org/licenses/>. *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-08-06 05:40:23 -04:00
# include "retroshare/rsiface.h"
2013-05-01 17:13:40 -04:00
# include "retroshare/rspeers.h"
2014-03-29 10:18:05 -04:00
# include "retroshare/rsidentity.h"
2009-02-22 12:36:39 -05:00
2007-12-11 20:43:17 -05:00
# include "pqi/pqibin.h"
2011-07-09 14:39:34 -04:00
# include "pqi/p3linkmgr.h"
2013-04-29 16:44:48 -04:00
# include "pqi/authgpg.h"
2014-03-29 10:18:05 -04:00
# include "pqi/p3cfgmgr.h"
# include "gxs/gxssecurity.h"
2007-12-11 20:43:17 -05:00
2014-03-29 10:18:05 -04:00
# include "services/p3idservice.h"
2007-12-11 20:43:17 -05:00
# include "services/p3msgservice.h"
2014-03-29 10:18:05 -04:00
2013-05-03 17:02:44 -04:00
# include "pgp/pgpkeyutil.h"
2014-01-07 17:51:22 -05:00
# include "rsserver/p3face.h"
2017-04-24 16:47:08 -04:00
# include "rsitems/rsconfigitems.h"
2014-03-29 10:18:05 -04:00
2014-01-30 17:14:37 -05:00
# include "grouter/p3grouter.h"
# include "grouter/groutertypes.h"
2007-12-11 20:43:17 -05:00
2008-07-10 12:29:18 -04:00
# include "util/rsdebug.h"
2008-02-04 12:55:13 -05:00
# include "util/rsdir.h"
2012-03-30 19:02:52 -04:00
# include "util/rsstring.h"
2013-05-04 18:03:48 -04:00
# include "util/radix64.h"
2013-04-29 16:44:48 -04:00
# include "util/rsrandom.h"
2016-01-09 10:58:49 -05:00
# include "util/rsmemory.h"
2014-03-29 10:18:05 -04:00
# include "util/rsprint.h"
2016-01-09 10:58:49 -05:00
# include "util/rsthreads.h"
2008-02-04 12:55:13 -05:00
2014-05-19 18:16:04 -04:00
# include <unistd.h>
2007-12-11 20:43:17 -05:00
# include <iomanip>
2010-09-25 08:54:34 -04:00
# include <map>
2014-03-29 10:18:05 -04:00
# include <sstream>
2007-12-11 20:43:17 -05:00
2013-11-29 17:09:13 -05:00
//#define MSG_DEBUG 1
2013-08-09 13:02:29 -04:00
//#define DEBUG_DISTANT_MSG
2013-08-04 10:11:59 -04:00
//#define DISABLE_DISTANT_MESSAGES
2014-05-17 17:36:07 -04:00
//#define DEBUG_DISTANT_MSG
2013-05-08 12:58:22 -04:00
2017-02-27 03:06:35 -05:00
typedef unsigned int uint ;
2015-03-22 00:45:01 -04:00
using namespace Rs : : Msgs ;
2016-06-18 12:04:18 -04:00
static struct RsLog : : logInfo msgservicezoneInfo = { RsLog : : Default , " msgservice " } ;
2016-06-18 07:00:15 -04:00
# define msgservicezone &msgservicezoneInfo
2007-12-11 20:43:17 -05:00
2016-01-09 10:58:49 -05:00
static const uint32_t RS_MSG_DISTANT_MESSAGE_HASH_KEEP_TIME = 2 * 30 * 86400 ; // keep msg hashes for 2 months to avoid re-sent msgs
2007-12-11 20:43:17 -05:00
/* Another little hack ..... unique message Ids
* will be handled in this class . . . . .
* These are unique within this run of the server ,
* and are not stored long term . . . .
*
* Only 3 entry points :
* ( 1 ) from network . . . .
* ( 2 ) from local send
* ( 3 ) from storage . . .
*/
2017-02-21 17:08:02 -05:00
p3MsgService : : p3MsgService ( p3ServiceControl * sc , p3IdService * id_serv ,
2017-03-01 20:37:53 -05:00
p3GxsTrans & gxsMS )
2017-02-26 07:06:38 -05:00
: p3Service ( ) , p3Config ( ) ,
gxsOngoingMutex ( " p3MsgService Gxs Outgoing Mutex " ) , mIdService ( id_serv ) ,
mServiceCtrl ( sc ) , mMsgMtx ( " p3MsgService " ) , mMsgUniqueId ( 0 ) ,
recentlyReceivedMutex ( " p3MsgService recently received hash mutex " ) ,
2017-03-01 20:37:53 -05:00
mGxsTransServ ( gxsMS )
2007-12-11 20:43:17 -05:00
{
2017-04-24 08:14:34 -04:00
_serialiser = new RsMsgSerialiser ( RsServiceSerializer : : SERIALIZATION_FLAG_NONE ) ; // this serialiser is used for services. It's not the same than the one returned by setupSerialiser(). We need both!!
2013-05-01 17:13:40 -04:00
addSerialType ( _serialiser ) ;
2010-09-23 17:27:34 -04:00
2017-02-21 17:08:02 -05:00
/* MsgIds are not transmitted, but only used locally as a storage index.
* As such , thay do not need to be different at friends nodes . */
mMsgUniqueId = 1 ;
mShouldEnableDistantMessaging = true ;
mDistantMessagingEnabled = false ;
mDistantMessagePermissions = RS_DISTANT_MESSAGING_CONTACT_PERMISSION_FLAG_FILTER_NONE ;
2013-09-05 14:45:40 -04:00
2017-02-21 17:08:02 -05:00
if ( sc ) initStandardTagTypes ( ) ; // Initialize standard tag types
2014-01-30 17:14:37 -05:00
2017-03-01 20:37:53 -05:00
mGxsTransServ . registerGxsTransClient ( GxsTransSubServices : : P3_MSG_SERVICE ,
this ) ;
2007-12-11 20:43:17 -05:00
}
2014-03-21 23:53:44 -04:00
const std : : string MSG_APP_NAME = " msg " ;
const uint16_t MSG_APP_MAJOR_VERSION = 1 ;
const uint16_t MSG_APP_MINOR_VERSION = 0 ;
const uint16_t MSG_MIN_MAJOR_VERSION = 1 ;
const uint16_t MSG_MIN_MINOR_VERSION = 0 ;
RsServiceInfo p3MsgService : : getServiceInfo ( )
{
return RsServiceInfo ( RS_SERVICE_TYPE_MSG ,
MSG_APP_NAME ,
MSG_APP_MAJOR_VERSION ,
MSG_APP_MINOR_VERSION ,
MSG_MIN_MAJOR_VERSION ,
MSG_MIN_MINOR_VERSION ) ;
}
2008-02-04 16:40:34 -05:00
uint32_t p3MsgService : : getNewUniqueMsgId ( )
2008-02-04 12:55:13 -05:00
{
2016-01-09 10:58:49 -05:00
RS_STACK_MUTEX ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2008-02-04 16:40:34 -05:00
return mMsgUniqueId + + ;
}
2007-12-11 20:43:17 -05:00
int p3MsgService : : tick ( )
{
pqioutput ( PQL_DEBUG_BASIC , msgservicezone ,
" p3MsgService::tick() " ) ;
/* don't worry about increasing tick rate!
* ( handled by p3service )
*/
incomingMsgs ( ) ;
2013-04-29 16:44:48 -04:00
2018-10-06 19:34:05 -04:00
static rstime_t last_management_time = 0 ;
rstime_t now = time ( NULL ) ;
2013-04-29 16:44:48 -04:00
if ( now > last_management_time + 5 )
{
2017-02-21 17:08:02 -05:00
manageDistantPeers ( ) ;
checkOutgoingMessages ( ) ;
cleanListOfReceivedMessageHashes ( ) ;
2013-08-31 17:10:04 -04:00
2017-02-21 17:08:02 -05:00
last_management_time = now ;
2013-04-29 16:44:48 -04:00
}
2007-12-11 20:43:17 -05:00
return 0 ;
}
2016-01-09 10:58:49 -05:00
void p3MsgService : : cleanListOfReceivedMessageHashes ( )
{
2017-02-21 17:08:02 -05:00
RS_STACK_MUTEX ( recentlyReceivedMutex ) ;
2016-01-09 10:58:49 -05:00
2018-10-06 19:34:05 -04:00
rstime_t now = time ( NULL ) ;
2017-02-21 17:08:02 -05:00
for ( auto it = mRecentlyReceivedMessageHashes . begin ( ) ;
it ! = mRecentlyReceivedMessageHashes . end ( ) ; )
if ( now > RS_MSG_DISTANT_MESSAGE_HASH_KEEP_TIME + it - > second )
{
std : : cerr < < " p3MsgService(): cleanListOfReceivedMessageHashes(). "
< < " Removing old hash " < < it - > first < < " , aged "
< < now - it - > second < < " secs ago " < < std : : endl ;
it = mRecentlyReceivedMessageHashes . erase ( it ) ;
}
else + + it ;
2016-01-09 10:58:49 -05:00
}
2007-12-11 20:43:17 -05:00
int p3MsgService : : status ( )
{
pqioutput ( PQL_DEBUG_BASIC , msgservicezone ,
" p3MsgService::status() " ) ;
return 1 ;
}
2016-01-07 00:04:52 -05:00
void p3MsgService : : processIncomingMsg ( RsMsgItem * mi )
2011-05-21 12:26:00 -04:00
{
mi - > recvTime = time ( NULL ) ;
mi - > msgId = getNewUniqueMsgId ( ) ;
{
2012-04-26 19:41:14 -04:00
RsStackMutex stack ( mMsgMtx ) ; /*** STACK LOCKED MTX ***/
2011-05-21 12:26:00 -04:00
2016-01-07 00:04:52 -05:00
/* from a peer */
2012-05-01 05:18:55 -04:00
2016-01-07 00:04:52 -05:00
mi - > msgFlags & = ( RS_MSG_FLAGS_DISTANT | RS_MSG_FLAGS_SYSTEM ) ; // remove flags except those
mi - > msgFlags | = RS_MSG_FLAGS_NEW ;
2012-04-26 19:41:14 -04:00
2016-01-07 00:04:52 -05:00
p3Notify * notify = RsServer : : notify ( ) ;
if ( notify )
2012-05-01 05:18:55 -04:00
{
2016-01-07 00:04:52 -05:00
notify - > AddPopupMessage ( RS_POPUP_MSG , mi - > PeerId ( ) . toStdString ( ) , mi - > subject , mi - > message ) ;
std : : string out ;
rs_sprintf ( out , " %lu " , mi - > msgId ) ;
notify - > AddFeedItem ( RS_FEED_ITEM_MESSAGE , out , " " , " " ) ;
2012-05-01 05:18:55 -04:00
}
2012-04-26 19:41:14 -04:00
imsg [ mi - > msgId ] = mi ;
RsMsgSrcId * msi = new RsMsgSrcId ( ) ;
msi - > msgId = mi - > msgId ;
msi - > srcId = mi - > PeerId ( ) ;
mSrcIds . insert ( std : : pair < uint32_t , RsMsgSrcId * > ( msi - > msgId , msi ) ) ;
2016-01-07 00:04:52 -05:00
2012-04-26 19:41:14 -04:00
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
2015-02-18 14:07:58 -05:00
2015-09-06 15:28:34 -04:00
/**** STACK UNLOCKED ***/
}
2015-02-18 14:07:58 -05:00
2015-09-06 15:28:34 -04:00
// If the peer is allowed to push files, then auto-download the recommended files.
if ( rsPeers - > servicePermissionFlags ( mi - > PeerId ( ) ) & RS_NODE_PERM_ALLOW_PUSH )
{
std : : list < RsPeerId > srcIds ;
srcIds . push_back ( mi - > PeerId ( ) ) ;
2015-02-18 14:07:58 -05:00
2015-09-06 15:28:34 -04:00
for ( std : : list < RsTlvFileItem > : : const_iterator it ( mi - > attachment . items . begin ( ) ) ; it ! = mi - > attachment . items . end ( ) ; + + it )
rsFiles - > FileRequest ( ( * it ) . name , ( * it ) . hash , ( * it ) . filesize , std : : string ( ) , RS_FILE_REQ_ANONYMOUS_ROUTING , srcIds ) ;
}
2011-05-21 12:26:00 -04:00
2014-01-07 17:51:22 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST , NOTIFY_TYPE_ADD ) ;
2011-05-21 12:26:00 -04:00
}
2015-09-06 15:28:34 -04:00
2011-09-09 15:37:38 -04:00
bool p3MsgService : : checkAndRebuildPartialMessage ( RsMsgItem * ci )
{
// Check is the item is ending an incomplete item.
//
2014-03-17 16:56:06 -04:00
std : : map < RsPeerId , RsMsgItem * > : : iterator it = _pendingPartialMessages . find ( ci - > PeerId ( ) ) ;
2011-09-09 15:37:38 -04:00
bool ci_is_partial = ci - > msgFlags & RS_MSG_FLAGS_PARTIAL ;
if ( it ! = _pendingPartialMessages . end ( ) )
{
2013-11-29 17:09:13 -05:00
# ifdef MSG_DEBUG
std : : cerr < < " Pending message found. Appending it. " < < std : : endl ;
# endif
2011-09-09 15:37:38 -04:00
// Yes, there is. Append the item to ci.
ci - > message = it - > second - > message + ci - > message ;
ci - > msgFlags | = it - > second - > msgFlags ;
delete it - > second ;
if ( ! ci_is_partial )
_pendingPartialMessages . erase ( it ) ;
}
if ( ci_is_partial )
{
2013-11-29 17:09:13 -05:00
# ifdef MSG_DEBUG
2011-09-09 15:37:38 -04:00
std : : cerr < < " Message is partial, storing for later. " < < std : : endl ;
2013-11-29 17:09:13 -05:00
# endif
2011-09-09 15:37:38 -04:00
// The item is a partial message. Push it, and wait for the rest.
//
_pendingPartialMessages [ ci - > PeerId ( ) ] = ci ;
return false ;
}
else
{
2013-11-29 17:09:13 -05:00
# ifdef MSG_DEBUG
2011-09-09 15:37:38 -04:00
std : : cerr < < " Message is complete, using it now. " < < std : : endl ;
2013-11-29 17:09:13 -05:00
# endif
2011-09-09 15:37:38 -04:00
return true ;
}
}
2011-05-21 12:26:00 -04:00
int p3MsgService : : incomingMsgs ( )
2007-12-11 20:43:17 -05:00
{
RsMsgItem * mi ;
int i = 0 ;
2009-02-22 12:36:39 -05:00
2007-12-11 20:43:17 -05:00
while ( ( mi = ( RsMsgItem * ) recvItem ( ) ) ! = NULL )
{
2013-04-29 16:44:48 -04:00
handleIncomingItem ( mi ) ;
+ + i ;
}
2008-02-04 16:40:34 -05:00
2013-04-29 16:44:48 -04:00
return i ;
}
void p3MsgService : : handleIncomingItem ( RsMsgItem * mi )
{
bool changed = false ;
if ( checkAndRebuildPartialMessage ( mi ) ) // only returns true when a msg is complete.
{
2016-01-07 00:04:52 -05:00
processIncomingMsg ( mi ) ;
2013-04-29 16:44:48 -04:00
changed = true ;
2007-12-11 20:43:17 -05:00
}
2009-02-22 12:36:39 -05:00
if ( changed )
2014-01-07 17:51:22 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST , NOTIFY_TYPE_MOD ) ;
2007-12-11 20:43:17 -05:00
}
2014-03-28 23:57:44 -04:00
void p3MsgService : : statusChange ( const std : : list < pqiServicePeer > & plist )
2008-02-07 11:18:34 -05:00
{
/* should do it properly! */
2014-03-28 23:57:44 -04:00
/* only do this when a new peer is connected */
bool newPeers = false ;
std : : list < pqiServicePeer > : : const_iterator it ;
2014-10-24 18:07:26 -04:00
for ( it = plist . begin ( ) ; it ! = plist . end ( ) ; + + it )
2014-03-28 23:57:44 -04:00
{
if ( it - > actions & RS_SERVICE_PEER_CONNECTED )
{
newPeers = true ;
}
}
if ( newPeers )
{
checkOutgoingMessages ( ) ;
}
2008-02-07 11:18:34 -05:00
}
2011-09-09 15:37:38 -04:00
void p3MsgService : : checkSizeAndSendMessage ( RsMsgItem * msg )
{
// We check the message item, and possibly split it into multiple messages, if the message is too big.
static const uint32_t MAX_STRING_SIZE = 15000 ;
2015-01-11 17:18:28 -05:00
std : : cerr < < " Msg is size " < < msg - > message . size ( ) < < std : : endl ;
2014-03-17 16:56:06 -04:00
2011-09-09 15:37:38 -04:00
while ( msg - > message . size ( ) > MAX_STRING_SIZE )
{
// chop off the first 15000 wchars
RsMsgItem * item = new RsMsgItem ( * msg ) ;
item - > message = item - > message . substr ( 0 , MAX_STRING_SIZE ) ;
msg - > message = msg - > message . substr ( MAX_STRING_SIZE , msg - > message . size ( ) - MAX_STRING_SIZE ) ;
2013-09-08 07:42:13 -04:00
# ifdef DEBUG_DISTANT_MSG
2011-09-09 15:37:38 -04:00
std : : cerr < < " Chopped off msg of size " < < item - > message . size ( ) < < std : : endl ;
2013-09-08 07:42:13 -04:00
# endif
2011-09-09 15:37:38 -04:00
// Indicate that the message is to be continued.
//
item - > msgFlags | = RS_MSG_FLAGS_PARTIAL ;
2015-01-11 17:18:28 -05:00
sendItem ( item ) ;
2011-09-09 15:37:38 -04:00
}
2013-09-08 07:42:13 -04:00
# ifdef DEBUG_DISTANT_MSG
2011-09-09 15:37:38 -04:00
std : : cerr < < " Chopped off msg of size " < < msg - > message . size ( ) < < std : : endl ;
2013-09-08 07:42:13 -04:00
# endif
2013-04-29 18:32:58 -04:00
2015-01-11 17:18:28 -05:00
sendItem ( msg ) ;
2011-09-09 15:37:38 -04:00
}
2017-02-21 17:08:02 -05:00
int p3MsgService : : checkOutgoingMessages ( )
2007-12-11 20:43:17 -05:00
{
2017-02-21 17:08:02 -05:00
bool changed = false ;
std : : list < RsMsgItem * > output_queue ;
2007-12-11 20:43:17 -05:00
2017-02-21 17:08:02 -05:00
{
RS_STACK_MUTEX ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2010-05-07 18:23:38 -04:00
2017-02-21 17:08:02 -05:00
const RsPeerId & ownId = mServiceCtrl - > getOwnId ( ) ;
2016-01-08 20:48:47 -05:00
2017-02-21 17:08:02 -05:00
std : : list < uint32_t > : : iterator it ;
std : : list < uint32_t > toErase ;
2016-01-08 20:48:47 -05:00
2017-02-21 17:08:02 -05:00
std : : map < uint32_t , RsMsgItem * > : : iterator mit ;
for ( mit = msgOutgoing . begin ( ) ; mit ! = msgOutgoing . end ( ) ; + + mit )
{
if ( mit - > second - > msgFlags & RS_MSG_FLAGS_TRASH ) continue ;
2016-01-08 20:48:47 -05:00
2017-02-21 17:08:02 -05:00
/* find the certificate */
RsPeerId pid = mit - > second - > PeerId ( ) ;
bool should_send = false ;
2016-01-08 20:48:47 -05:00
2017-02-21 17:08:02 -05:00
if ( pid = = ownId ) should_send = true ;
2016-01-08 20:48:47 -05:00
2017-02-21 17:08:02 -05:00
// FEEDBACK Msg to Ourselves
if ( mServiceCtrl - > isPeerConnected ( getServiceInfo ( ) . mServiceType ,
pid ) )
should_send = true ;
2016-01-08 20:48:47 -05:00
2017-02-21 17:08:02 -05:00
if ( ( mit - > second - > msgFlags & RS_MSG_FLAGS_DISTANT ) & &
! ( mit - > second - > msgFlags & RS_MSG_FLAGS_ROUTED ) )
should_send = true ;
2016-01-08 20:48:47 -05:00
2017-02-21 17:08:02 -05:00
if ( should_send )
{
/* send msg */
pqioutput ( PQL_DEBUG_BASIC , msgservicezone ,
" p3MsgService::checkOutGoingMessages() Sending out message " ) ;
/* remove the pending flag */
output_queue . push_back ( mit - > second ) ;
/* When the message is a distant msg, dont remove it yet from
* the list . Only mark it as being sent , so that we don ' t send
* it again . */
if ( ! ( mit - > second - > msgFlags & RS_MSG_FLAGS_DISTANT ) )
{
( mit - > second ) - > msgFlags & = ~ RS_MSG_FLAGS_PENDING ;
toErase . push_back ( mit - > first ) ;
changed = true ;
}
else
{
2014-04-21 09:08:42 -04:00
# ifdef DEBUG_DISTANT_MSG
2017-02-21 17:08:02 -05:00
std : : cerr < < " Message id " < < mit - > first < < " is distant: "
< < " kept in outgoing, and marked as ROUTED "
< < std : : endl ;
2014-04-21 09:08:42 -04:00
# endif
2017-02-21 17:08:02 -05:00
mit - > second - > msgFlags | = RS_MSG_FLAGS_ROUTED ;
}
}
else
{
pqioutput ( PQL_DEBUG_BASIC , msgservicezone ,
" p3MsgService::checkOutGoingMessages() Delaying until available... " ) ;
}
}
2016-01-08 20:48:47 -05:00
2017-02-21 17:08:02 -05:00
/* clean up */
for ( it = toErase . begin ( ) ; it ! = toErase . end ( ) ; + + it )
{
mit = msgOutgoing . find ( * it ) ;
if ( mit ! = msgOutgoing . end ( ) ) msgOutgoing . erase ( mit ) ;
2016-01-08 20:48:47 -05:00
2017-02-21 17:08:02 -05:00
std : : map < uint32_t , RsMsgSrcId * > : : iterator srcIt = mSrcIds . find ( * it ) ;
if ( srcIt ! = mSrcIds . end ( ) )
{
delete ( srcIt - > second ) ;
mSrcIds . erase ( srcIt ) ;
}
}
2016-01-08 20:48:47 -05:00
2017-02-21 17:08:02 -05:00
if ( toErase . size ( ) > 0 ) IndicateConfigChanged ( ) ;
}
2008-02-04 12:55:13 -05:00
2017-02-21 17:08:02 -05:00
for ( std : : list < RsMsgItem * > : : const_iterator it ( output_queue . begin ( ) ) ;
it ! = output_queue . end ( ) ; + + it )
if ( ( * it ) - > msgFlags & RS_MSG_FLAGS_DISTANT ) // don't split distant messages. The global router takes care of it.
sendDistantMsgItem ( * it ) ;
else
checkSizeAndSendMessage ( * it ) ;
2013-05-01 17:13:40 -04:00
2017-02-21 17:08:02 -05:00
if ( changed )
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST , NOTIFY_TYPE_MOD ) ;
2008-02-07 11:18:34 -05:00
2017-02-21 17:08:02 -05:00
return 0 ;
2007-12-11 20:43:17 -05:00
}
2017-02-25 18:46:02 -05:00
bool p3MsgService : : saveList ( bool & cleanup , std : : list < RsItem * > & itemList )
2007-12-11 20:43:17 -05:00
{
2017-02-25 18:46:02 -05:00
RsMsgGRouterMap * gxsmailmap = new RsMsgGRouterMap ;
{
RS_STACK_MUTEX ( gxsOngoingMutex ) ;
gxsmailmap - > ongoing_msgs = gxsOngoingMessages ;
}
itemList . push_front ( gxsmailmap ) ;
2007-12-11 20:43:17 -05:00
2010-09-20 18:11:09 -04:00
std : : map < uint32_t , RsMsgItem * > : : iterator mit ;
std : : map < uint32_t , RsMsgTagType * > : : iterator mit2 ;
std : : map < uint32_t , RsMsgTags * > : : iterator mit3 ;
2010-11-10 15:37:02 -05:00
std : : map < uint32_t , RsMsgSrcId * > : : iterator lit ;
2010-11-02 17:11:11 -04:00
std : : map < uint32_t , RsMsgParentId * > : : iterator mit4 ;
2008-02-04 16:40:34 -05:00
2017-02-25 18:46:02 -05:00
cleanup = true ;
2007-12-11 20:43:17 -05:00
2010-09-20 18:11:09 -04:00
mMsgMtx . lock ( ) ;
2007-12-11 20:43:17 -05:00
2014-10-24 18:07:26 -04:00
for ( mit = imsg . begin ( ) ; mit ! = imsg . end ( ) ; + + mit )
2015-04-28 17:46:10 -04:00
itemList . push_back ( new RsMsgItem ( * mit - > second ) ) ;
2007-12-11 20:43:17 -05:00
2014-10-24 18:07:26 -04:00
for ( lit = mSrcIds . begin ( ) ; lit ! = mSrcIds . end ( ) ; + + lit )
2015-04-28 17:46:10 -04:00
itemList . push_back ( new RsMsgSrcId ( * lit - > second ) ) ;
2010-10-31 15:29:26 -04:00
2014-10-24 18:07:26 -04:00
for ( mit = msgOutgoing . begin ( ) ; mit ! = msgOutgoing . end ( ) ; + + mit )
2015-04-28 17:46:10 -04:00
itemList . push_back ( new RsMsgItem ( * mit - > second ) ) ;
2010-08-19 17:47:26 -04:00
2014-10-24 18:07:26 -04:00
for ( mit2 = mTags . begin ( ) ; mit2 ! = mTags . end ( ) ; + + mit2 )
2015-04-28 17:46:10 -04:00
itemList . push_back ( new RsMsgTagType ( * mit2 - > second ) ) ;
2010-08-19 17:47:26 -04:00
2014-10-24 18:07:26 -04:00
for ( mit3 = mMsgTags . begin ( ) ; mit3 ! = mMsgTags . end ( ) ; + + mit3 )
2015-04-28 17:46:10 -04:00
itemList . push_back ( new RsMsgTags ( * mit3 - > second ) ) ;
2010-08-19 17:47:26 -04:00
2014-10-24 18:07:26 -04:00
for ( mit4 = mParentId . begin ( ) ; mit4 ! = mParentId . end ( ) ; + + mit4 )
2015-04-28 17:46:10 -04:00
itemList . push_back ( new RsMsgParentId ( * mit4 - > second ) ) ;
RsMsgGRouterMap * grmap = new RsMsgGRouterMap ;
grmap - > ongoing_msgs = _ongoing_messages ;
itemList . push_back ( grmap ) ;
2017-02-21 17:08:02 -05:00
RsMsgDistantMessagesHashMap * ghm = new RsMsgDistantMessagesHashMap ;
{
RS_STACK_MUTEX ( recentlyReceivedMutex ) ;
ghm - > hash_map = mRecentlyReceivedMessageHashes ;
}
itemList . push_back ( ghm ) ;
2010-11-02 17:11:11 -04:00
2013-08-09 13:02:29 -04:00
RsConfigKeyValueSet * vitem = new RsConfigKeyValueSet ;
RsTlvKeyValue kv ;
kv . key = " DISTANT_MESSAGES_ENABLED " ;
2015-04-18 15:59:27 -04:00
kv . value = mShouldEnableDistantMessaging ? " YES " : " NO " ;
2013-08-09 13:02:29 -04:00
vitem - > tlvkvs . pairs . push_back ( kv ) ;
2015-12-25 22:37:06 -05:00
kv . key = " DISTANT_MESSAGE_PERMISSION_FLAGS " ;
kv . value = RsUtil : : NumberToString ( mDistantMessagePermissions ) ;
vitem - > tlvkvs . pairs . push_back ( kv ) ;
2013-08-09 13:02:29 -04:00
2017-02-25 18:46:02 -05:00
itemList . push_back ( vitem ) ;
2013-08-09 13:02:29 -04:00
2010-12-18 14:35:07 -05:00
return true ;
2010-09-20 18:11:09 -04:00
}
2009-03-20 19:37:06 -04:00
2010-09-20 18:11:09 -04:00
void p3MsgService : : saveDone ( )
{
// unlocks mutex which has been locked by savelist
mMsgMtx . unlock ( ) ;
}
2009-03-20 19:37:06 -04:00
2016-01-07 22:22:05 -05:00
RsSerialiser * p3MsgService : : setupSerialiser ( ) // this serialiser is used for config. So it adds somemore info in the serialised items
2010-09-20 18:11:09 -04:00
{
2013-08-09 13:02:29 -04:00
RsSerialiser * rss = new RsSerialiser ;
2017-04-24 08:14:34 -04:00
rss - > addSerialType ( new RsMsgSerialiser ( RsServiceSerializer : : SERIALIZATION_FLAG_CONFIG ) ) ;
2013-08-09 13:02:29 -04:00
rss - > addSerialType ( new RsGeneralConfigSerialiser ( ) ) ;
2010-09-20 18:11:09 -04:00
return rss ;
2007-12-11 20:43:17 -05:00
}
2010-08-22 18:12:26 -04:00
// build list of standard tag types
static void getStandardTagTypes ( MsgTagType & tags )
{
/* create standard tag types, the text must be translated in the GUI */
tags . types [ RS_MSGTAGTYPE_IMPORTANT ] = std : : pair < std : : string , uint32_t > ( " Important " , 0xFF0000 ) ;
tags . types [ RS_MSGTAGTYPE_WORK ] = std : : pair < std : : string , uint32_t > ( " Work " , 0xFF9900 ) ;
tags . types [ RS_MSGTAGTYPE_PERSONAL ] = std : : pair < std : : string , uint32_t > ( " Personal " , 0x009900 ) ;
tags . types [ RS_MSGTAGTYPE_TODO ] = std : : pair < std : : string , uint32_t > ( " Todo " , 0x3333FF ) ;
tags . types [ RS_MSGTAGTYPE_LATER ] = std : : pair < std : : string , uint32_t > ( " Later " , 0x993399 ) ;
}
// Initialize the standard tag types after load
void p3MsgService : : initStandardTagTypes ( )
{
bool bChanged = false ;
2014-03-28 23:57:44 -04:00
const RsPeerId & ownId = mServiceCtrl - > getOwnId ( ) ;
2010-08-22 18:12:26 -04:00
MsgTagType tags ;
getStandardTagTypes ( tags ) ;
std : : map < uint32_t , std : : pair < std : : string , uint32_t > > : : iterator tit ;
2014-10-24 18:07:26 -04:00
for ( tit = tags . types . begin ( ) ; tit ! = tags . types . end ( ) ; + + tit ) {
2010-08-22 18:12:26 -04:00
std : : map < uint32_t , RsMsgTagType * > : : iterator mit = mTags . find ( tit - > first ) ;
if ( mit = = mTags . end ( ) ) {
RsMsgTagType * tagType = new RsMsgTagType ( ) ;
2014-03-17 16:56:06 -04:00
tagType - > PeerId ( ownId ) ;
2010-08-22 18:12:26 -04:00
tagType - > tagId = tit - > first ;
tagType - > text = tit - > second . first ;
tagType - > rgb_color = tit - > second . second ;
mTags . insert ( std : : pair < uint32_t , RsMsgTagType * > ( tit - > first , tagType ) ) ;
bChanged = true ;
}
}
if ( bChanged ) {
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
}
}
2017-02-25 18:46:02 -05:00
bool p3MsgService : : loadList ( std : : list < RsItem * > & load )
2007-12-11 20:43:17 -05:00
{
2017-02-25 18:46:02 -05:00
auto gxsmIt = load . begin ( ) ;
RsMsgGRouterMap * gxsmailmap = dynamic_cast < RsMsgGRouterMap * > ( * gxsmIt ) ;
if ( gxsmailmap )
{
{
RS_STACK_MUTEX ( gxsOngoingMutex ) ;
gxsOngoingMessages = gxsmailmap - > ongoing_msgs ;
}
delete * gxsmIt ; load . erase ( gxsmIt ) ;
}
2010-08-19 17:47:26 -04:00
RsMsgItem * mitem ;
RsMsgTagType * mtt ;
RsMsgTags * mti ;
2010-10-31 15:29:26 -04:00
RsMsgSrcId * msi ;
2010-11-02 17:11:11 -04:00
RsMsgParentId * msp ;
2015-04-28 17:46:10 -04:00
RsMsgGRouterMap * grm ;
2016-01-07 22:22:05 -05:00
RsMsgDistantMessagesHashMap * ghm ;
2007-12-11 20:43:17 -05:00
2010-09-20 18:11:09 -04:00
std : : list < RsMsgItem * > items ;
2017-02-25 18:46:02 -05:00
std : : list < RsItem * > : : iterator it ;
2010-09-25 08:54:34 -04:00
std : : map < uint32_t , RsMsgTagType * > : : iterator tagIt ;
2014-03-17 16:56:06 -04:00
std : : map < uint32_t , RsPeerId > srcIdMsgMap ;
std : : map < uint32_t , RsPeerId > : : iterator srcIt ;
2010-10-31 15:29:26 -04:00
2016-01-07 22:22:05 -05:00
uint32_t max_msg_id = 0 ;
2015-12-30 18:20:09 -05:00
// load items and calculate next unique msgId
2017-02-25 18:46:02 -05:00
for ( it = load . begin ( ) ; it ! = load . end ( ) ; + + it )
2015-12-30 18:20:09 -05:00
{
2017-02-25 18:46:02 -05:00
if ( NULL ! = ( mitem = dynamic_cast < RsMsgItem * > ( * it ) ) )
2015-12-30 18:20:09 -05:00
{
/* STORE MsgID */
2016-01-07 22:22:05 -05:00
if ( mitem - > msgId > max_msg_id )
max_msg_id = mitem - > msgId ;
2015-12-30 18:20:09 -05:00
items . push_back ( mitem ) ;
}
2017-02-25 18:46:02 -05:00
else if ( NULL ! = ( grm = dynamic_cast < RsMsgGRouterMap * > ( * it ) ) )
2015-12-30 18:20:09 -05:00
{
2017-03-13 17:57:33 -04:00
typedef std : : map < GRouterMsgPropagationId , uint32_t > tT ;
for ( tT : : const_iterator bit = grm - > ongoing_msgs . begin ( ) ;
bit ! = grm - > ongoing_msgs . end ( ) ; + + bit )
_ongoing_messages . insert ( * bit ) ;
delete * it ;
continue ;
2017-02-21 17:08:02 -05:00
}
else if ( NULL ! = ( ghm = dynamic_cast < RsMsgDistantMessagesHashMap * > ( * it ) ) )
{
{
RS_STACK_MUTEX ( recentlyReceivedMutex ) ;
mRecentlyReceivedMessageHashes = ghm - > hash_map ;
}
2016-01-09 10:58:49 -05:00
# ifdef DEBUG_DISTANT_MSG
std : : cerr < < " loaded recently received message map: " < < std : : endl ;
for ( std : : map < Sha1CheckSum , uint32_t > : : const_iterator it ( mRecentlyReceivedDistantMessageHashes . begin ( ) ) ; it ! = mRecentlyReceivedDistantMessageHashes . end ( ) ; + + it )
std : : cerr < < " " < < it - > first < < " received " < < time ( NULL ) - it - > second < < " secs ago. " < < std : : endl ;
# endif
2017-03-04 14:52:06 -05:00
delete * it ;
continue ;
2016-01-07 22:22:05 -05:00
}
2015-12-30 18:20:09 -05:00
else if ( NULL ! = ( mtt = dynamic_cast < RsMsgTagType * > ( * it ) ) )
{
// delete standard tags as they are now save in config
if ( mTags . end ( ) = = ( tagIt = mTags . find ( mtt - > tagId ) ) )
{
mTags . insert ( std : : pair < uint32_t , RsMsgTagType * > ( mtt - > tagId , mtt ) ) ;
}
else
{
delete mTags [ mtt - > tagId ] ;
mTags . erase ( tagIt ) ;
mTags . insert ( std : : pair < uint32_t , RsMsgTagType * > ( mtt - > tagId , mtt ) ) ;
}
}
2017-02-25 18:46:02 -05:00
else if ( NULL ! = ( mti = dynamic_cast < RsMsgTags * > ( * it ) ) )
2015-12-30 18:20:09 -05:00
{
mMsgTags . insert ( std : : pair < uint32_t , RsMsgTags * > ( mti - > msgId , mti ) ) ;
}
2017-02-25 18:46:02 -05:00
else if ( NULL ! = ( msi = dynamic_cast < RsMsgSrcId * > ( * it ) ) )
2015-12-30 18:20:09 -05:00
{
srcIdMsgMap . insert ( std : : pair < uint32_t , RsPeerId > ( msi - > msgId , msi - > srcId ) ) ;
mSrcIds . insert ( std : : pair < uint32_t , RsMsgSrcId * > ( msi - > msgId , msi ) ) ; // does not need to be kept
}
2017-02-25 18:46:02 -05:00
else if ( NULL ! = ( msp = dynamic_cast < RsMsgParentId * > ( * it ) ) )
2015-12-30 18:20:09 -05:00
{
mParentId . insert ( std : : pair < uint32_t , RsMsgParentId * > ( msp - > msgId , msp ) ) ;
}
RsConfigKeyValueSet * vitem = NULL ;
2017-02-25 18:46:02 -05:00
if ( NULL ! = ( vitem = dynamic_cast < RsConfigKeyValueSet * > ( * it ) ) )
2015-12-30 18:20:09 -05:00
{
for ( std : : list < RsTlvKeyValue > : : const_iterator kit = vitem - > tlvkvs . pairs . begin ( ) ; kit ! = vitem - > tlvkvs . pairs . end ( ) ; + + kit )
{
if ( kit - > key = = " DISTANT_MESSAGES_ENABLED " )
{
2013-11-29 17:09:13 -05:00
# ifdef MSG_DEBUG
2015-12-30 18:20:09 -05:00
std : : cerr < < " Loaded config default nick name for distant chat: " < < kit - > value < < std : : endl ;
2013-08-09 13:02:29 -04:00
# endif
2015-12-30 18:20:09 -05:00
mShouldEnableDistantMessaging = ( kit - > value = = " YES " ) ;
}
if ( kit - > key = = " DISTANT_MESSAGE_PERMISSION_FLAGS " )
{
2015-12-25 22:37:06 -05:00
# ifdef MSG_DEBUG
2015-12-30 18:20:09 -05:00
std : : cerr < < " Loaded distant message permission flags: " < < kit - > value < < std : : endl ;
2015-12-25 22:37:06 -05:00
# endif
2015-12-30 18:20:09 -05:00
if ( ! kit - > value . empty ( ) )
{
std : : istringstream is ( kit - > value ) ;
uint32_t tmp ;
is > > tmp ;
if ( tmp < 3 )
mDistantMessagePermissions = tmp ;
else
std : : cerr < < " (EE) Invalid value read for DistantMessagePermission flags in config: " < < tmp < < std : : endl ;
}
}
}
2017-02-25 18:46:02 -05:00
delete * it ;
2015-12-30 18:20:09 -05:00
continue ;
}
}
2016-01-07 22:22:05 -05:00
mMsgUniqueId = max_msg_id + 1 ; // make it unique with respect to what was loaded. Not totally safe, but works 99.9999% of the cases.
2015-12-30 18:20:09 -05:00
load . clear ( ) ;
// sort items into lists
std : : list < RsMsgItem * > : : iterator msgIt ;
for ( msgIt = items . begin ( ) ; msgIt ! = items . end ( ) ; + + msgIt )
{
mitem = * msgIt ;
/* STORE MsgID */
if ( mitem - > msgId = = 0 ) {
2010-06-04 19:39:33 -04:00
mitem - > msgId = getNewUniqueMsgId ( ) ;
2015-12-30 18:20:09 -05:00
}
2008-02-07 11:18:34 -05:00
2017-02-25 18:46:02 -05:00
RS_STACK_MUTEX ( mMsgMtx ) ;
2010-11-10 15:37:02 -05:00
2015-12-30 18:20:09 -05:00
srcIt = srcIdMsgMap . find ( mitem - > msgId ) ;
if ( srcIt ! = srcIdMsgMap . end ( ) ) {
mitem - > PeerId ( srcIt - > second ) ;
srcIdMsgMap . erase ( srcIt ) ;
}
2010-11-10 15:37:02 -05:00
2015-12-30 18:20:09 -05:00
/* switch depending on the PENDING
2010-06-04 19:39:33 -04:00
* flags
*/
2015-12-30 18:20:09 -05:00
if ( mitem - > msgFlags & RS_MSG_FLAGS_PENDING )
{
//std::cerr << "MSG_PENDING";
//std::cerr << std::endl;
//mitem->print(std::cerr);
msgOutgoing [ mitem - > msgId ] = mitem ;
}
else
{
imsg [ mitem - > msgId ] = mitem ;
}
}
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
/* remove missing msgId in mSrcIds */
for ( srcIt = srcIdMsgMap . begin ( ) ; srcIt ! = srcIdMsgMap . end ( ) ; + + srcIt ) {
std : : map < uint32_t , RsMsgSrcId * > : : iterator it = mSrcIds . find ( srcIt - > first ) ;
if ( it ! = mSrcIds . end ( ) ) {
delete ( it - > second ) ;
mSrcIds . erase ( it ) ;
}
}
/* remove missing msgId in mParentId */
std : : map < uint32_t , RsMsgParentId * > : : iterator mit = mParentId . begin ( ) ;
while ( mit ! = mParentId . end ( ) ) {
if ( imsg . find ( mit - > first ) = = imsg . end ( ) ) {
if ( msgOutgoing . find ( mit - > first ) = = msgOutgoing . end ( ) ) {
/* not found */
mParentId . erase ( mit + + ) ;
continue ;
}
}
+ + mit ;
}
return true ;
2007-12-11 20:43:17 -05:00
}
void p3MsgService : : loadWelcomeMsg ( )
{
/* Load Welcome Message */
RsMsgItem * msg = new RsMsgItem ( ) ;
2014-03-28 23:57:44 -04:00
//msg -> PeerId(mServiceCtrl->getOwnId());
2007-12-11 20:43:17 -05:00
2011-03-06 07:58:18 -05:00
msg - > sendTime = time ( NULL ) ;
msg - > recvTime = time ( NULL ) ;
msg - > msgFlags = RS_MSG_FLAGS_NEW ;
2007-12-11 20:43:17 -05:00
2013-10-03 17:41:34 -04:00
msg - > subject = " Welcome to Retroshare " ;
2007-12-11 20:43:17 -05:00
2013-10-03 17:41:34 -04:00
msg - > message = " Send and receive messages with your friends... \n " ;
msg - > message + = " These can hold recommendations from your local shared files. \n \n " ;
msg - > message + = " Add recommendations through the Local Files Dialog. \n \n " ;
msg - > message + = " Enjoy. " ;
2007-12-11 20:43:17 -05:00
2008-02-04 12:55:13 -05:00
msg - > msgId = getNewUniqueMsgId ( ) ;
2008-02-04 16:40:34 -05:00
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2010-11-04 14:03:54 -04:00
imsg [ msg - > msgId ] = msg ;
IndicateConfigChanged ( ) ;
2008-02-04 12:55:13 -05:00
}
/***********************************************************************/
/***********************************************************************/
/***********************************************************************/
/***********************************************************************/
/****************************************/
/****************************************/
bool p3MsgService : : getMessageSummaries ( std : : list < MsgInfoSummary > & msgList )
{
/* do stuff */
msgList . clear ( ) ;
2008-02-04 16:40:34 -05:00
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2008-02-04 12:55:13 -05:00
std : : map < uint32_t , RsMsgItem * > : : iterator mit ;
2014-10-24 18:07:26 -04:00
for ( mit = imsg . begin ( ) ; mit ! = imsg . end ( ) ; + + mit )
2008-02-04 12:55:13 -05:00
{
MsgInfoSummary mis ;
initRsMIS ( mit - > second , mis ) ;
msgList . push_back ( mis ) ;
}
2014-10-24 18:07:26 -04:00
for ( mit = msgOutgoing . begin ( ) ; mit ! = msgOutgoing . end ( ) ; + + mit )
2008-02-04 12:55:13 -05:00
{
MsgInfoSummary mis ;
initRsMIS ( mit - > second , mis ) ;
msgList . push_back ( mis ) ;
}
2010-05-13 15:20:40 -04:00
return true ;
2007-12-11 20:43:17 -05:00
}
2010-11-02 17:11:11 -04:00
bool p3MsgService : : getMessage ( const std : : string & mId , MessageInfo & msg )
2008-02-04 12:55:13 -05:00
{
std : : map < uint32_t , RsMsgItem * > : : iterator mit ;
uint32_t msgId = atoi ( mId . c_str ( ) ) ;
2008-02-04 16:40:34 -05:00
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2008-02-04 12:55:13 -05:00
mit = imsg . find ( msgId ) ;
if ( mit = = imsg . end ( ) )
{
mit = msgOutgoing . find ( msgId ) ;
if ( mit = = msgOutgoing . end ( ) )
{
return false ;
}
}
/* mit valid */
initRsMI ( mit - > second , msg ) ;
2014-05-07 16:53:16 -04:00
std : : map < uint32_t , RsMsgSrcId * > : : const_iterator it = mSrcIds . find ( msgId ) ;
if ( it ! = mSrcIds . end ( ) )
2019-01-03 16:20:00 -05:00
msg . rsgxsid_srcId = RsGxsId ( it - > second - > srcId ) ; // (cyril) this is a hack. Not good. I'm not removing it because it may have consequences, but I dont like this.
2014-05-07 16:53:16 -04:00
2008-02-04 12:55:13 -05:00
return true ;
}
2018-10-03 15:52:54 -04:00
void p3MsgService : : getMessageCount ( uint32_t & nInbox , uint32_t & nInboxNew , uint32_t & nOutbox , uint32_t & nDraftbox , uint32_t & nSentbox , uint32_t & nTrashbox )
2010-05-13 15:20:40 -04:00
{
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2018-10-03 15:52:54 -04:00
nInbox = 0 ;
nInboxNew = 0 ;
nOutbox = 0 ;
nDraftbox = 0 ;
nSentbox = 0 ;
nTrashbox = 0 ;
2010-05-13 15:20:40 -04:00
std : : map < uint32_t , RsMsgItem * > : : iterator mit ;
std : : map < uint32_t , RsMsgItem * > * apMsg [ 2 ] = { & imsg , & msgOutgoing } ;
for ( int i = 0 ; i < 2 ; i + + ) {
2014-10-24 18:07:26 -04:00
for ( mit = apMsg [ i ] - > begin ( ) ; mit ! = apMsg [ i ] - > end ( ) ; + + mit ) {
2010-05-13 15:20:40 -04:00
MsgInfoSummary mis ;
initRsMIS ( mit - > second , mis ) ;
2010-05-28 10:42:54 -04:00
if ( mis . msgflags & RS_MSG_TRASH ) {
2018-10-03 15:52:54 -04:00
+ + nTrashbox ;
2010-05-28 10:42:54 -04:00
continue ;
}
2010-05-13 15:20:40 -04:00
switch ( mis . msgflags & RS_MSG_BOXMASK ) {
case RS_MSG_INBOX :
2018-10-03 15:52:54 -04:00
+ + nInbox ;
if ( ( mis . msgflags & RS_MSG_NEW ) = = RS_MSG_NEW )
+ + nInboxNew ;
2010-05-13 15:20:40 -04:00
break ;
case RS_MSG_OUTBOX :
2018-10-03 15:52:54 -04:00
+ + nOutbox ;
2010-05-13 15:20:40 -04:00
break ;
case RS_MSG_DRAFTBOX :
2018-10-03 15:52:54 -04:00
+ + nDraftbox ;
2010-05-13 15:20:40 -04:00
break ;
case RS_MSG_SENTBOX :
2018-10-03 15:52:54 -04:00
+ + nSentbox ;
2010-05-13 15:20:40 -04:00
break ;
}
}
}
}
2008-02-04 12:55:13 -05:00
/* remove based on the unique mid (stored in sid) */
2010-11-02 17:11:11 -04:00
bool p3MsgService : : removeMsgId ( const std : : string & mid )
2008-02-04 12:55:13 -05:00
{
std : : map < uint32_t , RsMsgItem * > : : iterator mit ;
uint32_t msgId = atoi ( mid . c_str ( ) ) ;
2010-08-22 18:12:26 -04:00
if ( msgId = = 0 ) {
std : : cerr < < " p3MsgService::removeMsgId: Unknown msgId " < < msgId < < std : : endl ;
return false ;
}
bool changed = false ;
2008-02-04 12:55:13 -05:00
{
2009-02-22 12:36:39 -05:00
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2008-02-04 12:55:13 -05:00
2009-02-22 12:36:39 -05:00
mit = imsg . find ( msgId ) ;
if ( mit ! = imsg . end ( ) )
{
2010-08-22 18:12:26 -04:00
changed = true ;
2009-02-22 12:36:39 -05:00
RsMsgItem * mi = mit - > second ;
imsg . erase ( mit ) ;
delete mi ;
}
2008-02-04 12:55:13 -05:00
2009-02-22 12:36:39 -05:00
mit = msgOutgoing . find ( msgId ) ;
if ( mit ! = msgOutgoing . end ( ) )
{
changed = true ;
RsMsgItem * mi = mit - > second ;
msgOutgoing . erase ( mit ) ;
delete mi ;
}
2010-11-10 15:37:02 -05:00
std : : map < uint32_t , RsMsgSrcId * > : : iterator srcIt = mSrcIds . find ( msgId ) ;
if ( srcIt ! = mSrcIds . end ( ) ) {
changed = true ;
delete ( srcIt - > second ) ;
mSrcIds . erase ( srcIt ) ;
}
2008-02-04 12:55:13 -05:00
}
2010-08-22 18:12:26 -04:00
if ( changed ) {
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
setMessageTag ( mid , 0 , false ) ;
2010-11-02 17:11:11 -04:00
setMsgParentId ( msgId , 0 ) ;
2010-08-22 18:12:26 -04:00
2014-01-07 17:51:22 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST , NOTIFY_TYPE_MOD ) ;
2010-08-22 18:12:26 -04:00
}
2009-02-22 12:36:39 -05:00
2010-08-22 18:12:26 -04:00
return changed ;
2008-02-04 12:55:13 -05:00
}
2011-05-23 19:45:31 -04:00
bool p3MsgService : : markMsgIdRead ( const std : : string & mid , bool unreadByUser )
2008-02-04 12:55:13 -05:00
{
2009-02-22 12:36:39 -05:00
std : : map < uint32_t , RsMsgItem * > : : iterator mit ;
2008-02-04 12:55:13 -05:00
uint32_t msgId = atoi ( mid . c_str ( ) ) ;
2010-08-22 18:12:26 -04:00
bool changed = false ;
2008-02-04 12:55:13 -05:00
{
2009-02-22 12:36:39 -05:00
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
mit = imsg . find ( msgId ) ;
if ( mit ! = imsg . end ( ) )
{
RsMsgItem * mi = mit - > second ;
2010-08-22 18:12:26 -04:00
uint32_t msgFlags = mi - > msgFlags ;
/* remove new state */
mi - > msgFlags & = ~ ( RS_MSG_FLAGS_NEW ) ;
/* set state from user */
2011-05-23 19:45:31 -04:00
if ( unreadByUser ) {
2010-08-22 18:12:26 -04:00
mi - > msgFlags | = RS_MSG_FLAGS_UNREAD_BY_USER ;
} else {
mi - > msgFlags & = ~ RS_MSG_FLAGS_UNREAD_BY_USER ;
}
if ( mi - > msgFlags ! = msgFlags )
{
changed = true ;
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
}
} else {
return false ;
2009-02-22 12:36:39 -05:00
}
2010-08-22 18:12:26 -04:00
} /* UNLOCKED */
2009-02-22 12:36:39 -05:00
2010-08-22 18:12:26 -04:00
if ( changed ) {
2014-01-07 17:51:22 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST , NOTIFY_TYPE_MOD ) ;
2010-08-09 08:16:21 -04:00
}
2010-08-22 18:12:26 -04:00
return true ;
2008-02-04 12:55:13 -05:00
}
2010-11-02 17:11:11 -04:00
bool p3MsgService : : setMsgFlag ( const std : : string & mid , uint32_t flag , uint32_t mask )
{
std : : map < uint32_t , RsMsgItem * > : : iterator mit ;
uint32_t msgId = atoi ( mid . c_str ( ) ) ;
bool changed = false ;
{
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
mit = imsg . find ( msgId ) ;
if ( mit = = imsg . end ( ) )
{
mit = msgOutgoing . find ( msgId ) ;
if ( mit = = msgOutgoing . end ( ) )
{
return false ;
}
}
uint32_t oldFlag = mit - > second - > msgFlags ;
mit - > second - > msgFlags & = ~ mask ;
mit - > second - > msgFlags | = flag ;
if ( mit - > second - > msgFlags ! = oldFlag ) {
changed = true ;
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
}
} /* UNLOCKED */
if ( changed ) {
2014-01-07 17:51:22 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST , NOTIFY_TYPE_MOD ) ;
2010-11-02 17:11:11 -04:00
}
return true ;
}
bool p3MsgService : : getMsgParentId ( const std : : string & msgId , std : : string & msgParentId )
{
msgParentId . clear ( ) ;
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
std : : map < uint32_t , RsMsgParentId * > : : iterator mit = mParentId . find ( atoi ( msgId . c_str ( ) ) ) ;
if ( mit = = mParentId . end ( ) ) {
return false ;
}
2012-04-17 17:00:54 -04:00
rs_sprintf ( msgParentId , " %lu " , mit - > second - > msgParentId ) ;
2010-11-02 17:11:11 -04:00
return true ;
}
bool p3MsgService : : setMsgParentId ( uint32_t msgId , uint32_t msgParentId )
{
std : : map < uint32_t , RsMsgParentId * > : : iterator mit ;
bool changed = false ;
{
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
mit = mParentId . find ( msgId ) ;
if ( mit = = mParentId . end ( ) )
{
if ( msgParentId ) {
RsMsgParentId * msp = new RsMsgParentId ( ) ;
2014-03-28 23:57:44 -04:00
msp - > PeerId ( mServiceCtrl - > getOwnId ( ) ) ;
2010-11-02 17:11:11 -04:00
msp - > msgId = msgId ;
msp - > msgParentId = msgParentId ;
mParentId . insert ( std : : pair < uint32_t , RsMsgParentId * > ( msgId , msp ) ) ;
changed = true ;
}
} else {
if ( msgParentId ) {
if ( mit - > second - > msgParentId ! = msgParentId ) {
mit - > second - > msgParentId = msgParentId ;
changed = true ;
}
} else {
delete mit - > second ;
mParentId . erase ( mit ) ;
changed = true ;
}
}
} /* UNLOCKED */
if ( changed ) {
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
}
return true ;
}
2008-02-04 12:55:13 -05:00
/****************************************/
/****************************************/
/* Message Items */
2016-01-07 00:04:52 -05:00
uint32_t p3MsgService : : sendMessage ( RsMsgItem * item ) // no from field because it's implicitly our own PeerId
2008-02-04 12:55:13 -05:00
{
2014-03-29 10:18:05 -04:00
if ( ! item )
2016-01-07 00:04:52 -05:00
return 0 ;
2014-03-29 10:18:05 -04:00
2016-01-07 00:04:52 -05:00
pqioutput ( PQL_DEBUG_BASIC , msgservicezone , " p3MsgService::sendMessage() " ) ;
item - > msgId = getNewUniqueMsgId ( ) ; /* grabs Mtx as well */
item - > msgFlags | = ( RS_MSG_FLAGS_OUTGOING | RS_MSG_FLAGS_PENDING ) ; /* add pending flag */
{
RS_STACK_MUTEX ( mMsgMtx ) ;
/* STORE MsgID */
msgOutgoing [ item - > msgId ] = item ;
if ( item - > PeerId ( ) ! = mServiceCtrl - > getOwnId ( ) )
{
/* not to the loopback device */
RsMsgSrcId * msi = new RsMsgSrcId ( ) ;
msi - > msgId = item - > msgId ;
msi - > srcId = mServiceCtrl - > getOwnId ( ) ;
mSrcIds . insert ( std : : pair < uint32_t , RsMsgSrcId * > ( msi - > msgId , msi ) ) ;
}
}
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST , NOTIFY_TYPE_ADD ) ;
return item - > msgId ;
}
2017-02-21 17:08:02 -05:00
uint32_t p3MsgService : : sendDistantMessage ( RsMsgItem * item , const RsGxsId & from )
2016-01-07 00:04:52 -05:00
{
2017-02-21 17:08:02 -05:00
if ( ! item ) return 0 ;
2008-02-04 12:55:13 -05:00
2017-02-21 17:08:02 -05:00
item - > msgId = getNewUniqueMsgId ( ) ; /* grabs Mtx as well */
item - > msgFlags | = ( RS_MSG_FLAGS_DISTANT | RS_MSG_FLAGS_OUTGOING |
RS_MSG_FLAGS_PENDING ) ; /* add pending flag */
2008-02-07 11:18:34 -05:00
2010-11-10 15:37:02 -05:00
{
2017-02-21 17:08:02 -05:00
RS_STACK_MUTEX ( mMsgMtx ) ;
2008-02-04 16:40:34 -05:00
2010-11-10 15:37:02 -05:00
/* STORE MsgID */
msgOutgoing [ item - > msgId ] = item ;
2016-01-07 00:04:52 -05:00
mDistantOutgoingMsgSigners [ item - > msgId ] = from ;
2010-11-10 15:37:02 -05:00
2016-01-07 00:04:52 -05:00
if ( item - > PeerId ( ) ! = mServiceCtrl - > getOwnId ( ) )
{
2010-11-10 15:37:02 -05:00
/* not to the loopback device */
2016-01-07 00:04:52 -05:00
2010-11-10 15:37:02 -05:00
RsMsgSrcId * msi = new RsMsgSrcId ( ) ;
msi - > msgId = item - > msgId ;
2017-02-21 17:08:02 -05:00
msi - > srcId = RsPeerId ( from ) ;
2010-11-10 15:37:02 -05:00
mSrcIds . insert ( std : : pair < uint32_t , RsMsgSrcId * > ( msi - > msgId , msi ) ) ;
2016-01-07 00:04:52 -05:00
}
2010-11-10 15:37:02 -05:00
}
2008-02-07 11:18:34 -05:00
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
2017-02-21 17:08:02 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST ,
NOTIFY_TYPE_ADD ) ;
2016-01-07 00:04:52 -05:00
return item - > msgId ;
2008-02-04 12:55:13 -05:00
}
bool p3MsgService : : MessageSend ( MessageInfo & info )
{
2016-01-08 22:50:03 -05:00
for ( std : : set < RsPeerId > : : const_iterator pit = info . rspeerid_msgto . begin ( ) ; pit ! = info . rspeerid_msgto . end ( ) ; + + pit ) sendMessage ( initMIRsMsg ( info , * pit ) ) ;
for ( std : : set < RsPeerId > : : const_iterator pit = info . rspeerid_msgcc . begin ( ) ; pit ! = info . rspeerid_msgcc . end ( ) ; + + pit ) sendMessage ( initMIRsMsg ( info , * pit ) ) ;
for ( std : : set < RsPeerId > : : const_iterator pit = info . rspeerid_msgbcc . begin ( ) ; pit ! = info . rspeerid_msgbcc . end ( ) ; + + pit ) sendMessage ( initMIRsMsg ( info , * pit ) ) ;
2013-08-31 08:49:04 -04:00
2016-01-08 22:50:03 -05:00
for ( std : : set < RsGxsId > : : const_iterator pit = info . rsgxsid_msgto . begin ( ) ; pit ! = info . rsgxsid_msgto . end ( ) ; + + pit ) sendDistantMessage ( initMIRsMsg ( info , * pit ) , info . rsgxsid_srcId ) ;
for ( std : : set < RsGxsId > : : const_iterator pit = info . rsgxsid_msgcc . begin ( ) ; pit ! = info . rsgxsid_msgcc . end ( ) ; + + pit ) sendDistantMessage ( initMIRsMsg ( info , * pit ) , info . rsgxsid_srcId ) ;
for ( std : : set < RsGxsId > : : const_iterator pit = info . rsgxsid_msgbcc . begin ( ) ; pit ! = info . rsgxsid_msgbcc . end ( ) ; + + pit ) sendDistantMessage ( initMIRsMsg ( info , * pit ) , info . rsgxsid_srcId ) ;
2008-02-04 12:55:13 -05:00
2016-01-08 22:50:03 -05:00
// store message in outgoing list. In order to appear as sent the message needs to have the OUTGOING flg, but no pending flag on.
RsMsgItem * msg = initMIRsMsg ( info , mServiceCtrl - > getOwnId ( ) ) ;
if ( msg )
{
std : : list < RsPgpId > : : iterator it ;
if ( msg - > msgFlags & RS_MSG_FLAGS_SIGNED )
msg - > msgFlags | = RS_MSG_FLAGS_SIGNATURE_CHECKS ; // this is always true, since we are sending the message
/* use processMsg to get the new msgId */
msg - > recvTime = time ( NULL ) ;
msg - > msgId = getNewUniqueMsgId ( ) ;
msg - > msgFlags | = RS_MSG_OUTGOING ;
imsg [ msg - > msgId ] = msg ;
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST , NOTIFY_TYPE_ADD ) ;
//
// // return new message id
// rs_sprintf(info.msgId, "%lu", msg->msgId);
}
2008-02-04 12:55:13 -05:00
return true ;
}
2013-10-03 17:41:34 -04:00
bool p3MsgService : : SystemMessage ( const std : : string & title , const std : : string & message , uint32_t systemFlag )
2012-05-01 05:18:55 -04:00
{
if ( ( systemFlag & RS_MSG_SYSTEM ) = = 0 ) {
/* no flag specified */
return false ;
}
2014-03-28 23:57:44 -04:00
const RsPeerId & ownId = mServiceCtrl - > getOwnId ( ) ;
2012-05-01 05:18:55 -04:00
RsMsgItem * msg = new RsMsgItem ( ) ;
msg - > PeerId ( ownId ) ;
msg - > msgFlags = 0 ;
if ( systemFlag & RS_MSG_USER_REQUEST ) {
msg - > msgFlags | = RS_MSG_FLAGS_USER_REQUEST ;
}
2012-05-05 18:20:05 -04:00
if ( systemFlag & RS_MSG_FRIEND_RECOMMENDATION ) {
msg - > msgFlags | = RS_MSG_FLAGS_FRIEND_RECOMMENDATION ;
}
2015-02-07 20:01:48 -05:00
if ( systemFlag & RS_MSG_PUBLISH_KEY ) {
msg - > msgFlags | = RS_MSG_FLAGS_PUBLISH_KEY ;
}
2012-05-01 05:18:55 -04:00
msg - > msgId = 0 ;
msg - > sendTime = time ( NULL ) ;
msg - > recvTime = 0 ;
msg - > subject = title ;
msg - > message = message ;
2015-04-17 17:36:22 -04:00
msg - > rspeerid_msgto . ids . insert ( ownId ) ;
2012-05-01 05:18:55 -04:00
2016-01-07 00:04:52 -05:00
processIncomingMsg ( msg ) ;
2012-05-01 05:18:55 -04:00
return true ;
}
2010-11-02 17:11:11 -04:00
bool p3MsgService : : MessageToDraft ( MessageInfo & info , const std : : string & msgParentId )
2010-05-25 05:32:14 -04:00
{
2014-03-28 23:57:44 -04:00
RsMsgItem * msg = initMIRsMsg ( info , mServiceCtrl - > getOwnId ( ) ) ;
2010-05-25 05:32:14 -04:00
if ( msg )
{
uint32_t msgId = 0 ;
if ( info . msgId . empty ( ) = = false ) {
2010-05-28 10:42:54 -04:00
msgId = atoi ( info . msgId . c_str ( ) ) ;
2010-05-25 05:32:14 -04:00
}
if ( msgId ) {
msg - > msgId = msgId ;
} else {
msg - > msgId = getNewUniqueMsgId ( ) ; /* grabs Mtx as well */
}
{
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
/* add pending flag */
msg - > msgFlags | = ( RS_MSG_OUTGOING | RS_MSG_FLAGS_DRAFT ) ;
if ( msgId ) {
// remove existing message
2010-05-28 10:42:54 -04:00
std : : map < uint32_t , RsMsgItem * > : : iterator mit ;
mit = imsg . find ( msgId ) ;
if ( mit ! = imsg . end ( ) ) {
RsMsgItem * mi = mit - > second ;
imsg . erase ( mit ) ;
delete mi ;
2010-05-25 05:32:14 -04:00
}
}
/* STORE MsgID */
imsg [ msg - > msgId ] = msg ;
2010-09-11 06:39:40 -04:00
// return new message id
2012-04-17 17:00:54 -04:00
rs_sprintf ( info . msgId , " %lu " , msg - > msgId ) ;
2010-05-25 05:32:14 -04:00
}
2010-11-02 17:11:11 -04:00
setMsgParentId ( msg - > msgId , atoi ( msgParentId . c_str ( ) ) ) ;
2010-05-25 05:32:14 -04:00
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
2014-01-07 17:51:22 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST , NOTIFY_TYPE_MOD ) ;
2010-05-25 05:32:14 -04:00
return true ;
}
return false ;
}
2010-08-22 18:12:26 -04:00
bool p3MsgService : : getMessageTagTypes ( MsgTagType & tags )
2010-08-19 17:47:26 -04:00
{
2010-08-22 18:12:26 -04:00
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2010-08-19 17:47:26 -04:00
std : : map < uint32_t , RsMsgTagType * > : : iterator mit ;
2014-10-24 18:07:26 -04:00
for ( mit = mTags . begin ( ) ; mit ! = mTags . end ( ) ; + + mit ) {
2010-08-19 17:47:26 -04:00
std : : pair < std : : string , uint32_t > p ( mit - > second - > text , mit - > second - > rgb_color ) ;
tags . types . insert ( std : : pair < uint32_t , std : : pair < std : : string , uint32_t > > ( mit - > first , p ) ) ;
}
return true ;
}
2010-08-22 18:12:26 -04:00
bool p3MsgService : : setMessageTagType ( uint32_t tagId , std : : string & text , uint32_t rgb_color )
2010-08-19 17:47:26 -04:00
{
2010-08-22 18:12:26 -04:00
int nNotifyType = 0 ;
{
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2010-08-19 17:47:26 -04:00
2010-08-22 18:12:26 -04:00
std : : map < uint32_t , RsMsgTagType * > : : iterator mit ;
mit = mTags . find ( tagId ) ;
2010-08-19 17:47:26 -04:00
2010-08-22 18:12:26 -04:00
if ( mit = = mTags . end ( ) ) {
if ( tagId < RS_MSGTAGTYPE_USER ) {
std : : cerr < < " p3MsgService::MessageSetTagType: Standard tag type " < < tagId < < " cannot be inserted " < < std : : endl ;
return false ;
}
2010-08-19 17:47:26 -04:00
2010-08-22 18:12:26 -04:00
/* new tag */
RsMsgTagType * tagType = new RsMsgTagType ( ) ;
2014-03-28 23:57:44 -04:00
tagType - > PeerId ( mServiceCtrl - > getOwnId ( ) ) ;
2010-08-22 18:12:26 -04:00
tagType - > rgb_color = rgb_color ;
tagType - > tagId = tagId ;
tagType - > text = text ;
mTags . insert ( std : : pair < uint32_t , RsMsgTagType * > ( tagId , tagType ) ) ;
nNotifyType = NOTIFY_TYPE_ADD ;
} else {
if ( mit - > second - > text ! = text | | mit - > second - > rgb_color ! = rgb_color ) {
/* modify existing tag */
if ( tagId > = RS_MSGTAGTYPE_USER ) {
mit - > second - > text = text ;
} else {
/* don't change text for standard tag types */
if ( mit - > second - > text ! = text ) {
std : : cerr < < " p3MsgService::MessageSetTagType: Text " < < text < < " for standard tag type " < < tagId < < " cannot be changed " < < std : : endl ;
}
}
mit - > second - > rgb_color = rgb_color ;
nNotifyType = NOTIFY_TYPE_MOD ;
}
}
} /* UNLOCKED */
if ( nNotifyType ) {
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
2010-08-19 17:47:26 -04:00
2014-01-07 17:51:22 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGE_TAGS , nNotifyType ) ;
2010-08-22 18:12:26 -04:00
2010-08-19 17:47:26 -04:00
return true ;
}
return false ;
}
2010-08-22 18:12:26 -04:00
bool p3MsgService : : removeMessageTagType ( uint32_t tagId )
2010-08-19 17:47:26 -04:00
{
2010-08-22 18:12:26 -04:00
if ( tagId < RS_MSGTAGTYPE_USER ) {
std : : cerr < < " p3MsgService::MessageRemoveTagType: Can't delete standard tag type " < < tagId < < std : : endl ;
return false ;
}
2010-08-19 17:47:26 -04:00
2010-08-22 18:12:26 -04:00
{
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2010-08-19 17:47:26 -04:00
2010-08-22 18:12:26 -04:00
std : : map < uint32_t , RsMsgTagType * > : : iterator mit ;
mit = mTags . find ( tagId ) ;
2010-08-19 17:47:26 -04:00
2010-08-22 18:12:26 -04:00
if ( mit = = mTags . end ( ) ) {
/* tag id not found */
std : : cerr < < " p3MsgService::MessageRemoveTagType: Tag Id not found " < < tagId < < std : : endl ;
return false ;
}
/* search for messages with this tag type */
std : : map < uint32_t , RsMsgTags * > : : iterator mit1 ;
for ( mit1 = mMsgTags . begin ( ) ; mit1 ! = mMsgTags . end ( ) ; ) {
RsMsgTags * tag = mit1 - > second ;
std : : list < uint32_t > : : iterator lit ;
lit = std : : find ( tag - > tagIds . begin ( ) , tag - > tagIds . end ( ) , tagId ) ;
if ( lit ! = tag - > tagIds . end ( ) ) {
tag - > tagIds . erase ( lit ) ;
2014-10-21 18:33:02 -04:00
if ( tag - > tagIds . empty ( ) ) {
2010-08-22 18:12:26 -04:00
/* remove empty tag */
delete ( tag ) ;
2010-10-11 11:52:12 -04:00
mMsgTags . erase ( mit1 + + ) ;
2010-08-22 18:12:26 -04:00
continue ;
}
}
2014-10-24 18:07:26 -04:00
+ + mit1 ;
2010-08-22 18:12:26 -04:00
}
/* remove tag type */
delete ( mit - > second ) ;
mTags . erase ( mit ) ;
} /* UNLOCKED */
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
2014-01-07 17:51:22 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGE_TAGS , NOTIFY_TYPE_DEL ) ;
2010-08-22 18:12:26 -04:00
return true ;
}
2010-11-02 17:11:11 -04:00
bool p3MsgService : : getMessageTag ( const std : : string & msgId , MsgTagInfo & info )
2010-08-22 18:12:26 -04:00
{
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
uint32_t mid = atoi ( msgId . c_str ( ) ) ;
if ( mid = = 0 ) {
std : : cerr < < " p3MsgService::MessageGetMsgTag: Unknown msgId " < < msgId < < std : : endl ;
return false ;
}
std : : map < uint32_t , RsMsgTags * > : : iterator mit ;
if ( mMsgTags . end ( ) ! = ( mit = mMsgTags . find ( mid ) ) ) {
2012-04-17 17:00:54 -04:00
rs_sprintf ( info . msgId , " %lu " , mit - > second - > msgId ) ;
2010-08-22 18:12:26 -04:00
info . tagIds = mit - > second - > tagIds ;
return true ;
}
return false ;
2010-08-19 17:47:26 -04:00
}
2010-08-22 18:12:26 -04:00
/* set == false && tagId == 0 --> remove all */
2010-11-02 17:11:11 -04:00
bool p3MsgService : : setMessageTag ( const std : : string & msgId , uint32_t tagId , bool set )
2010-08-19 17:47:26 -04:00
{
2010-08-22 18:12:26 -04:00
uint32_t mid = atoi ( msgId . c_str ( ) ) ;
if ( mid = = 0 ) {
std : : cerr < < " p3MsgService::MessageSetMsgTag: Unknown msgId " < < msgId < < std : : endl ;
return false ;
}
if ( tagId = = 0 ) {
if ( set = = true ) {
std : : cerr < < " p3MsgService::MessageSetMsgTag: No valid tagId given " < < tagId < < std : : endl ;
return false ;
}
}
int nNotifyType = 0 ;
{
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
std : : map < uint32_t , RsMsgTags * > : : iterator mit ;
mit = mMsgTags . find ( mid ) ;
if ( mit = = mMsgTags . end ( ) ) {
if ( set ) {
/* new msg */
RsMsgTags * tag = new RsMsgTags ( ) ;
2014-03-28 23:57:44 -04:00
tag - > PeerId ( mServiceCtrl - > getOwnId ( ) ) ;
2010-08-22 18:12:26 -04:00
tag - > msgId = mid ;
tag - > tagIds . push_back ( tagId ) ;
mMsgTags . insert ( std : : pair < uint32_t , RsMsgTags * > ( tag - > msgId , tag ) ) ;
nNotifyType = NOTIFY_TYPE_ADD ;
}
} else {
RsMsgTags * tag = mit - > second ;
/* search existing tagId */
std : : list < uint32_t > : : iterator lit ;
if ( tagId ) {
lit = std : : find ( tag - > tagIds . begin ( ) , tag - > tagIds . end ( ) , tagId ) ;
} else {
lit = tag - > tagIds . end ( ) ;
}
if ( set ) {
if ( lit = = tag - > tagIds . end ( ) ) {
tag - > tagIds . push_back ( tagId ) ;
/* keep the list sorted */
tag - > tagIds . sort ( ) ;
nNotifyType = NOTIFY_TYPE_ADD ;
}
} else {
if ( tagId = = 0 ) {
/* remove all */
delete ( tag ) ;
mMsgTags . erase ( mit ) ;
nNotifyType = NOTIFY_TYPE_DEL ;
} else {
if ( lit ! = tag - > tagIds . end ( ) ) {
tag - > tagIds . erase ( lit ) ;
nNotifyType = NOTIFY_TYPE_DEL ;
2014-10-21 18:33:02 -04:00
if ( tag - > tagIds . empty ( ) ) {
2010-08-22 18:12:26 -04:00
/* remove empty tag */
delete ( tag ) ;
mMsgTags . erase ( mit ) ;
}
}
}
}
}
} /* UNLOCKED */
if ( nNotifyType ) {
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
2010-08-19 17:47:26 -04:00
2014-01-07 17:51:22 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGE_TAGS , nNotifyType ) ;
2010-08-19 17:47:26 -04:00
2010-08-22 18:12:26 -04:00
return true ;
}
return false ;
2010-08-19 17:47:26 -04:00
}
2010-08-22 18:12:26 -04:00
bool p3MsgService : : resetMessageStandardTagTypes ( MsgTagType & tags )
{
MsgTagType standardTags ;
2010-09-25 08:54:34 -04:00
getStandardTagTypes ( standardTags ) ;
2010-08-22 18:12:26 -04:00
std : : map < uint32_t , std : : pair < std : : string , uint32_t > > : : iterator mit ;
2014-10-24 18:07:26 -04:00
for ( mit = standardTags . types . begin ( ) ; mit ! = standardTags . types . end ( ) ; + + mit ) {
2010-08-22 18:12:26 -04:00
tags . types [ mit - > first ] = mit - > second ;
}
return true ;
}
2010-08-19 17:47:26 -04:00
2010-05-28 10:42:54 -04:00
/* move message to trash based on the unique mid */
2010-11-02 17:11:11 -04:00
bool p3MsgService : : MessageToTrash ( const std : : string & mid , bool bTrash )
2010-05-28 10:42:54 -04:00
{
std : : map < uint32_t , RsMsgItem * > : : iterator mit ;
uint32_t msgId = atoi ( mid . c_str ( ) ) ;
bool bChanged = false ;
bool bFound = false ;
{
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
RsMsgItem * mi = NULL ;
mit = imsg . find ( msgId ) ;
if ( mit ! = imsg . end ( ) ) {
mi = mit - > second ;
} else {
mit = msgOutgoing . find ( msgId ) ;
if ( mit ! = msgOutgoing . end ( ) ) {
mi = mit - > second ;
}
}
if ( mi ) {
bFound = true ;
if ( bTrash ) {
if ( ( mi - > msgFlags & RS_MSG_FLAGS_TRASH ) = = 0 ) {
mi - > msgFlags | = RS_MSG_FLAGS_TRASH ;
bChanged = true ;
}
} else {
if ( mi - > msgFlags & RS_MSG_FLAGS_TRASH ) {
mi - > msgFlags & = ~ RS_MSG_FLAGS_TRASH ;
bChanged = true ;
}
}
}
}
if ( bChanged ) {
IndicateConfigChanged ( ) ; /**** INDICATE MSG CONFIG CHANGED! *****/
checkOutgoingMessages ( ) ;
2014-01-07 17:51:22 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST , NOTIFY_TYPE_MOD ) ;
2010-05-28 10:42:54 -04:00
}
return bFound ;
}
2008-02-04 12:55:13 -05:00
/****************************************/
/****************************************/
/****************************************/
/**** HELPER FNS For Chat/Msg/Channel Lists ************
* These aren ' t required to be locked , unless
* the data used is from internal stores - > then they should be .
*/
void p3MsgService : : initRsMI ( RsMsgItem * msg , MessageInfo & mi )
{
mi . msgflags = 0 ;
/* translate flags, if we sent it... outgoing */
2013-08-02 19:00:36 -04:00
2014-03-29 10:18:05 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_OUTGOING ) mi . msgflags | = RS_MSG_OUTGOING ;
if ( msg - > msgFlags & RS_MSG_FLAGS_PENDING ) mi . msgflags | = RS_MSG_PENDING ; /* if it has a pending flag, then its in the outbox */
if ( msg - > msgFlags & RS_MSG_FLAGS_DRAFT ) mi . msgflags | = RS_MSG_DRAFT ;
if ( msg - > msgFlags & RS_MSG_FLAGS_NEW ) mi . msgflags | = RS_MSG_NEW ;
if ( msg - > msgFlags & RS_MSG_FLAGS_SIGNED ) mi . msgflags | = RS_MSG_SIGNED ;
if ( msg - > msgFlags & RS_MSG_FLAGS_SIGNATURE_CHECKS ) mi . msgflags | = RS_MSG_SIGNATURE_CHECKS ;
2015-02-20 05:34:11 -05:00
if ( msg - > msgFlags & RS_MSG_FLAGS_DISTANT ) mi . msgflags | = RS_MSG_DISTANT ;
2014-03-29 10:18:05 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_TRASH ) mi . msgflags | = RS_MSG_TRASH ;
if ( msg - > msgFlags & RS_MSG_FLAGS_UNREAD_BY_USER ) mi . msgflags | = RS_MSG_UNREAD_BY_USER ;
if ( msg - > msgFlags & RS_MSG_FLAGS_REPLIED ) mi . msgflags | = RS_MSG_REPLIED ;
if ( msg - > msgFlags & RS_MSG_FLAGS_FORWARDED ) mi . msgflags | = RS_MSG_FORWARDED ;
if ( msg - > msgFlags & RS_MSG_FLAGS_STAR ) mi . msgflags | = RS_MSG_STAR ;
if ( msg - > msgFlags & RS_MSG_FLAGS_USER_REQUEST ) mi . msgflags | = RS_MSG_USER_REQUEST ;
if ( msg - > msgFlags & RS_MSG_FLAGS_FRIEND_RECOMMENDATION ) mi . msgflags | = RS_MSG_FRIEND_RECOMMENDATION ;
2015-02-07 20:01:48 -05:00
if ( msg - > msgFlags & RS_MSG_FLAGS_PUBLISH_KEY ) mi . msgflags | = RS_MSG_PUBLISH_KEY ;
2014-03-29 10:18:05 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_LOAD_EMBEDDED_IMAGES ) mi . msgflags | = RS_MSG_LOAD_EMBEDDED_IMAGES ;
2008-02-04 12:55:13 -05:00
mi . ts = msg - > sendTime ;
2014-03-29 10:18:05 -04:00
mi . rspeerid_srcId = msg - > PeerId ( ) ;
2008-02-04 12:55:13 -05:00
2014-03-29 10:18:05 -04:00
mi . rspeerid_msgto = msg - > rspeerid_msgto . ids ;
mi . rspeerid_msgcc = msg - > rspeerid_msgcc . ids ;
mi . rspeerid_msgbcc = msg - > rspeerid_msgbcc . ids ;
2008-02-04 12:55:13 -05:00
2014-03-29 10:18:05 -04:00
mi . rsgxsid_msgto = msg - > rsgxsid_msgto . ids ;
mi . rsgxsid_msgcc = msg - > rsgxsid_msgcc . ids ;
mi . rsgxsid_msgbcc = msg - > rsgxsid_msgbcc . ids ;
2008-02-04 12:55:13 -05:00
mi . title = msg - > subject ;
mi . msg = msg - > message ;
2015-03-22 00:45:01 -04:00
{
//msg->msgId;
rs_sprintf ( mi . msgId , " %lu " , msg - > msgId ) ;
}
2008-02-04 12:55:13 -05:00
mi . attach_title = msg - > attachment . title ;
mi . attach_comment = msg - > attachment . comment ;
mi . count = 0 ;
mi . size = 0 ;
2014-10-24 18:07:26 -04:00
for ( std : : list < RsTlvFileItem > : : iterator it = msg - > attachment . items . begin ( ) ; it ! = msg - > attachment . items . end ( ) ; + + it )
2008-02-04 12:55:13 -05:00
{
FileInfo fi ;
2011-03-06 07:58:18 -05:00
fi . fname = RsDirUtil : : getTopDir ( it - > name ) ;
2008-02-04 12:55:13 -05:00
fi . size = it - > filesize ;
fi . hash = it - > hash ;
fi . path = it - > path ;
mi . files . push_back ( fi ) ;
mi . count + + ;
mi . size + = fi . size ;
}
}
void p3MsgService : : initRsMIS ( RsMsgItem * msg , MsgInfoSummary & mis )
{
mis . msgflags = 0 ;
2016-10-22 17:48:19 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_DISTANT )
mis . msgflags | = RS_MSG_DISTANT ;
2013-10-16 16:02:28 -04:00
2013-08-02 19:00:36 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_SIGNED )
mis . msgflags | = RS_MSG_SIGNED ;
if ( msg - > msgFlags & RS_MSG_FLAGS_SIGNATURE_CHECKS )
mis . msgflags | = RS_MSG_SIGNATURE_CHECKS ;
2008-02-04 12:55:13 -05:00
/* translate flags, if we sent it... outgoing */
if ( ( msg - > msgFlags & RS_MSG_FLAGS_OUTGOING )
2014-03-28 23:57:44 -04:00
/*|| (msg->PeerId() == mServiceCtrl->getOwnId())*/ )
2008-02-04 12:55:13 -05:00
{
mis . msgflags | = RS_MSG_OUTGOING ;
}
/* if it has a pending flag, then its in the outbox */
if ( msg - > msgFlags & RS_MSG_FLAGS_PENDING )
{
mis . msgflags | = RS_MSG_PENDING ;
}
2011-03-06 07:58:18 -05:00
if ( msg - > msgFlags & RS_MSG_FLAGS_DRAFT )
{
mis . msgflags | = RS_MSG_DRAFT ;
}
if ( msg - > msgFlags & RS_MSG_FLAGS_NEW )
2008-02-04 12:55:13 -05:00
{
mis . msgflags | = RS_MSG_NEW ;
}
2010-05-28 10:42:54 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_TRASH )
{
mis . msgflags | = RS_MSG_TRASH ;
}
2012-04-26 19:41:14 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_UNREAD_BY_USER )
2010-08-22 18:12:26 -04:00
{
mis . msgflags | = RS_MSG_UNREAD_BY_USER ;
}
2010-11-02 17:11:11 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_REPLIED )
{
mis . msgflags | = RS_MSG_REPLIED ;
}
if ( msg - > msgFlags & RS_MSG_FLAGS_FORWARDED )
{
mis . msgflags | = RS_MSG_FORWARDED ;
}
2011-05-23 19:45:31 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_STAR )
{
mis . msgflags | = RS_MSG_STAR ;
}
2012-05-01 05:18:55 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_USER_REQUEST )
{
mis . msgflags | = RS_MSG_USER_REQUEST ;
}
2012-05-05 18:20:05 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_FRIEND_RECOMMENDATION )
{
mis . msgflags | = RS_MSG_FRIEND_RECOMMENDATION ;
}
2015-02-07 20:01:48 -05:00
if ( msg - > msgFlags & RS_MSG_FLAGS_PUBLISH_KEY )
{
mis . msgflags | = RS_MSG_PUBLISH_KEY ;
}
2013-09-24 16:19:21 -04:00
if ( msg - > msgFlags & RS_MSG_FLAGS_LOAD_EMBEDDED_IMAGES )
{
mis . msgflags | = RS_MSG_LOAD_EMBEDDED_IMAGES ;
}
2008-02-04 12:55:13 -05:00
mis . srcId = msg - > PeerId ( ) ;
{
//msg->msgId;
2012-04-17 17:00:54 -04:00
rs_sprintf ( mis . msgId , " %lu " , msg - > msgId ) ;
2008-02-04 12:55:13 -05:00
}
mis . title = msg - > subject ;
mis . count = msg - > attachment . items . size ( ) ;
mis . ts = msg - > sendTime ;
}
2014-03-29 10:18:05 -04:00
void p3MsgService : : initMIRsMsg ( RsMsgItem * msg , const MessageInfo & info )
2008-02-04 12:55:13 -05:00
{
msg - > msgFlags = 0 ;
msg - > msgId = 0 ;
msg - > sendTime = time ( NULL ) ;
msg - > recvTime = 0 ;
msg - > subject = info . title ;
2014-03-29 10:18:05 -04:00
msg - > message = info . msg ;
2013-05-03 17:02:44 -04:00
2014-03-29 10:18:05 -04:00
msg - > rspeerid_msgto . ids = info . rspeerid_msgto ;
msg - > rspeerid_msgcc . ids = info . rspeerid_msgcc ;
2008-02-04 12:55:13 -05:00
2014-03-29 10:18:05 -04:00
msg - > rsgxsid_msgto . ids = info . rsgxsid_msgto ;
msg - > rsgxsid_msgcc . ids = info . rsgxsid_msgcc ;
2008-02-04 12:55:13 -05:00
/* We don't fill in bcc (unless to ourselves) */
2014-03-29 10:18:05 -04:00
if ( msg - > PeerId ( ) = = mServiceCtrl - > getOwnId ( ) )
2008-02-04 12:55:13 -05:00
{
2014-03-29 10:18:05 -04:00
msg - > rsgxsid_msgbcc . ids = info . rsgxsid_msgbcc ;
msg - > rspeerid_msgbcc . ids = info . rspeerid_msgbcc ;
2008-02-04 12:55:13 -05:00
}
msg - > attachment . title = info . attach_title ;
msg - > attachment . comment = info . attach_comment ;
2014-10-24 18:07:26 -04:00
for ( std : : list < FileInfo > : : const_iterator it = info . files . begin ( ) ; it ! = info . files . end ( ) ; + + it )
2008-02-04 12:55:13 -05:00
{
RsTlvFileItem mfi ;
mfi . hash = it - > hash ;
mfi . name = it - > fname ;
mfi . filesize = it - > size ;
msg - > attachment . items . push_back ( mfi ) ;
}
2012-05-01 05:18:55 -04:00
/* translate flags from outside */
if ( info . msgflags & RS_MSG_USER_REQUEST )
2015-01-11 17:18:28 -05:00
msg - > msgFlags | = RS_MSG_FLAGS_USER_REQUEST ;
2013-04-29 18:32:58 -04:00
2012-05-05 18:20:05 -04:00
if ( info . msgflags & RS_MSG_FRIEND_RECOMMENDATION )
msg - > msgFlags | = RS_MSG_FLAGS_FRIEND_RECOMMENDATION ;
2014-03-29 10:18:05 -04:00
}
RsMsgItem * p3MsgService : : initMIRsMsg ( const MessageInfo & info , const RsGxsId & to )
{
2015-01-11 17:18:28 -05:00
RsMsgItem * msg = new RsMsgItem ( ) ;
2014-03-29 10:18:05 -04:00
2015-01-11 17:18:28 -05:00
initMIRsMsg ( msg , info ) ;
2014-03-29 10:18:05 -04:00
2015-01-11 17:18:28 -05:00
msg - > PeerId ( RsPeerId ( to ) ) ;
msg - > msgFlags | = RS_MSG_FLAGS_DISTANT ;
2012-05-01 05:18:55 -04:00
2015-01-11 17:18:28 -05:00
if ( info . msgflags & RS_MSG_SIGNED )
msg - > msgFlags | = RS_MSG_FLAGS_SIGNED ;
2013-08-02 19:00:36 -04:00
2015-01-11 17:18:28 -05:00
// // We replace the msg text by the whole message serialized possibly signed,
// // and binary encrypted, so as to obfuscate all its content.
// //
// if(!createDistantMessage(to,info.rsgxsid_srcId,msg))
// {
// std::cerr << "Cannot encrypt distant message. Something went wrong." << std::endl;
// delete msg ;
// return NULL ;
// }
2014-03-29 10:18:05 -04:00
return msg ;
}
RsMsgItem * p3MsgService : : initMIRsMsg ( const MessageInfo & info , const RsPeerId & to )
{
2015-01-11 17:18:28 -05:00
RsMsgItem * msg = new RsMsgItem ( ) ;
2014-03-29 10:18:05 -04:00
initMIRsMsg ( msg , info ) ;
2013-05-03 17:02:44 -04:00
2014-03-29 10:18:05 -04:00
msg - > PeerId ( to ) ;
/* load embedded images from own messages */
msg - > msgFlags | = RS_MSG_FLAGS_LOAD_EMBEDDED_IMAGES ;
2013-05-03 17:02:44 -04:00
2008-02-04 12:55:13 -05:00
return msg ;
}
2013-04-29 16:44:48 -04:00
2014-03-29 10:18:05 -04:00
void p3MsgService : : connectToGlobalRouter ( p3GRouter * gr )
2013-04-29 16:44:48 -04:00
{
2014-03-29 10:18:05 -04:00
mGRouter = gr ;
gr - > registerClientService ( GROUTER_CLIENT_ID_MESSAGES , this ) ;
}
2013-04-29 16:44:48 -04:00
2014-03-29 10:18:05 -04:00
void p3MsgService : : enableDistantMessaging ( bool b )
{
2015-04-18 15:59:27 -04:00
// We use a temporary variable because the call to OwnIds() might fail.
2013-04-29 16:44:48 -04:00
2015-04-18 15:59:27 -04:00
mShouldEnableDistantMessaging = b ;
IndicateConfigChanged ( ) ;
2014-03-29 10:18:05 -04:00
}
2015-04-18 15:59:27 -04:00
2014-03-29 10:18:05 -04:00
bool p3MsgService : : distantMessagingEnabled ( )
{
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
2015-04-18 15:59:27 -04:00
return mShouldEnableDistantMessaging ;
2013-04-29 16:44:48 -04:00
}
void p3MsgService : : manageDistantPeers ( )
{
// now possibly flush pending messages
2015-04-18 15:59:27 -04:00
if ( mShouldEnableDistantMessaging = = mDistantMessagingEnabled )
return ;
2013-05-08 12:58:22 -04:00
# ifdef DEBUG_DISTANT_MSG
2013-04-29 16:44:48 -04:00
std : : cerr < < " p3MsgService::manageDistantPeers() " < < std : : endl ;
2013-05-08 12:58:22 -04:00
# endif
2015-04-18 15:59:27 -04:00
std : : list < RsGxsId > own_id_list ;
if ( mIdService - > getOwnIds ( own_id_list ) )
{
# ifdef DEBUG_DISTANT_MSG
for ( std : : list < RsGxsId > : : const_iterator it ( own_id_list . begin ( ) ) ; it ! = own_id_list . end ( ) ; + + it )
2016-01-09 10:58:49 -05:00
std : : cerr < < ( mShouldEnableDistantMessaging ? " Enabling " : " Disabling " ) < < " distant messaging, with peer id = " < < * it < < std : : endl ;
2015-04-18 15:59:27 -04:00
# endif
for ( std : : list < RsGxsId > : : const_iterator it ( own_id_list . begin ( ) ) ; it ! = own_id_list . end ( ) ; + + it )
{
if ( mShouldEnableDistantMessaging )
mGRouter - > registerKey ( * it , GROUTER_CLIENT_ID_MESSAGES , " Messaging contact " ) ;
else
mGRouter - > unregisterKey ( * it , GROUTER_CLIENT_ID_MESSAGES ) ;
}
RsStackMutex stack ( mMsgMtx ) ; /********** STACK LOCKED MTX ******/
mDistantMessagingEnabled = mShouldEnableDistantMessaging ;
}
2013-04-29 16:44:48 -04:00
}
2017-02-21 17:08:02 -05:00
void p3MsgService : : notifyDataStatus ( const GRouterMsgPropagationId & id ,
const RsGxsId & signer_id ,
uint32_t data_status )
2014-04-19 12:02:11 -04:00
{
2017-02-21 17:08:02 -05:00
if ( data_status = = GROUTER_CLIENT_SERVICE_DATA_STATUS_FAILED )
{
RS_STACK_MUTEX ( mMsgMtx ) ;
2016-01-07 22:22:05 -05:00
2017-02-21 17:08:02 -05:00
std : : cerr < < " (WW) p3MsgService::notifyDataStatus: Global router tells "
< < " us that item ID " < < id
< < " could not be delivered on time. " ;
2016-01-07 22:22:05 -05:00
2017-02-21 17:08:02 -05:00
auto it = _ongoing_messages . find ( id ) ;
if ( it = = _ongoing_messages . end ( ) )
{
std : : cerr < < " (EE) cannot find pending message to acknowledge. "
< < " Weird. grouter id = " < < id < < std : : endl ;
return ;
}
2016-01-07 22:22:05 -05:00
2017-02-21 17:08:02 -05:00
uint32_t msg_id = it - > second ;
std : : cerr < < " message id = " < < msg_id < < std : : endl ;
2016-01-07 22:22:05 -05:00
2017-02-21 17:08:02 -05:00
/* this is needed because it's not saved in config, but we should
* probably include it in _ongoing_messages */
mDistantOutgoingMsgSigners [ msg_id ] = signer_id ;
std : : map < uint32_t , RsMsgItem * > : : iterator mit = msgOutgoing . find ( msg_id ) ;
if ( mit = = msgOutgoing . end ( ) )
{
2017-02-25 18:46:02 -05:00
std : : cerr < < " (II) message has been notified as not delivered, "
< < " but it's not in outgoing list. Probably it has been "
< < " delivered successfully by other means. "
2017-02-21 17:08:02 -05:00
< < std : : endl ;
}
2017-02-25 18:46:02 -05:00
else
{
std : : cerr < < " reseting the ROUTED flag so that the message is "
< < " requested again " < < std : : endl ;
2017-02-21 17:08:02 -05:00
2017-02-25 18:46:02 -05:00
// clear the routed flag so that the message is requested again
mit - > second - > msgFlags & = ~ RS_MSG_FLAGS_ROUTED ;
}
2017-02-21 17:08:02 -05:00
return ;
}
if ( data_status = = GROUTER_CLIENT_SERVICE_DATA_STATUS_RECEIVED )
{
RS_STACK_MUTEX ( mMsgMtx ) ;
2014-04-19 12:02:11 -04:00
# ifdef DEBUG_DISTANT_MSG
2017-02-21 17:08:02 -05:00
std : : cerr < < " p3MsgService::acknowledgeDataReceived(): acknowledging data received for msg propagation id " < < id < < std : : endl ;
2014-04-19 12:02:11 -04:00
# endif
2017-02-21 17:08:02 -05:00
auto it = _ongoing_messages . find ( id ) ;
if ( it = = _ongoing_messages . end ( ) )
{
std : : cerr < < " (EE) cannot find pending message to acknowledge. "
< < " Weird. grouter id = " < < id < < std : : endl ;
return ;
}
2014-04-19 12:02:11 -04:00
2017-02-25 18:46:02 -05:00
uint32_t msg_id = it - > second ;
2014-04-19 12:02:11 -04:00
2017-02-21 17:08:02 -05:00
// we should now remove the item from the msgOutgoing list.
std : : map < uint32_t , RsMsgItem * > : : iterator it2 = msgOutgoing . find ( msg_id ) ;
if ( it2 = = msgOutgoing . end ( ) )
{
2017-02-25 18:46:02 -05:00
std : : cerr < < " (II) message has been notified as delivered, but it's "
< < " not in outgoing list. Probably it has been delivered "
< < " successfully by other means. " < < std : : endl ;
2017-02-21 17:08:02 -05:00
return ;
}
2014-04-19 12:02:11 -04:00
2017-02-21 17:08:02 -05:00
delete it2 - > second ;
msgOutgoing . erase ( it2 ) ;
2014-04-19 12:02:11 -04:00
2017-02-21 17:08:02 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST ,
NOTIFY_TYPE_ADD ) ;
IndicateConfigChanged ( ) ;
2016-01-07 22:22:05 -05:00
2017-02-21 17:08:02 -05:00
return ;
}
std : : cerr < < " p3MsgService: unhandled data status info from global router "
< < " for msg ID " < < id < < " : this is a bug. " < < std : : endl ;
2013-04-29 16:44:48 -04:00
}
2017-02-21 17:08:02 -05:00
2015-12-25 22:37:06 -05:00
bool p3MsgService : : acceptDataFromPeer ( const RsGxsId & to_gxs_id )
{
if ( mDistantMessagePermissions & RS_DISTANT_MESSAGING_CONTACT_PERMISSION_FLAG_FILTER_NON_CONTACTS )
return ( rsIdentity ! = NULL ) & & rsIdentity - > isARegularContact ( to_gxs_id ) ;
if ( mDistantMessagePermissions & RS_DISTANT_MESSAGING_CONTACT_PERMISSION_FLAG_FILTER_EVERYBODY )
return false ;
return true ;
}
void p3MsgService : : setDistantMessagingPermissionFlags ( uint32_t flags )
{
if ( flags ! = mDistantMessagePermissions )
{
mDistantMessagePermissions = flags ;
IndicateConfigChanged ( ) ;
}
}
uint32_t p3MsgService : : getDistantMessagingPermissionFlags ( )
{
return mDistantMessagePermissions ;
}
2017-02-21 17:08:02 -05:00
2017-03-01 20:37:53 -05:00
bool p3MsgService : : receiveGxsTransMail ( const RsGxsId & authorId ,
const RsGxsId & recipientId ,
const uint8_t * data , uint32_t dataSize )
2013-12-27 15:06:47 -05:00
{
2017-03-01 20:37:53 -05:00
std : : cout < < __PRETTY_FUNCTION__ < < " " < < authorId < < " , " < < recipientId
< < " ,, " < < dataSize < < std : : endl ;
2017-02-25 18:46:02 -05:00
2017-02-21 17:08:02 -05:00
Sha1CheckSum hash = RsDirUtil : : sha1sum ( data , dataSize ) ;
2015-01-11 17:18:28 -05:00
2017-02-21 17:08:02 -05:00
{
RS_STACK_MUTEX ( recentlyReceivedMutex ) ;
if ( mRecentlyReceivedMessageHashes . find ( hash ) ! =
mRecentlyReceivedMessageHashes . end ( ) )
{
2017-03-01 20:37:53 -05:00
std : : cerr < < __PRETTY_FUNCTION__ < < " (II) receiving "
2017-02-25 18:46:02 -05:00
< < " message of hash " < < hash < < " more than once. "
< < " Probably it has arrived before by other means. "
2017-02-21 17:08:02 -05:00
< < std : : endl ;
return true ;
}
mRecentlyReceivedMessageHashes [ hash ] = time ( NULL ) ;
}
IndicateConfigChanged ( ) ;
RsItem * item = _serialiser - > deserialise ( const_cast < uint8_t * > ( data ) , & dataSize ) ;
RsMsgItem * msg_item = dynamic_cast < RsMsgItem * > ( item ) ;
if ( msg_item )
{
2017-03-01 20:37:53 -05:00
std : : cerr < < __PRETTY_FUNCTION__ < < " Encrypted item correctly "
< < " deserialised. Passing on to incoming list. "
2017-02-21 17:08:02 -05:00
< < std : : endl ;
msg_item - > msgFlags | = RS_MSG_FLAGS_DISTANT ;
/* we expect complete msgs - remove partial flag just in case
* someone has funny ideas */
msg_item - > msgFlags & = ~ RS_MSG_FLAGS_PARTIAL ;
// hack to pass on GXS id.
2017-03-01 17:07:53 -05:00
msg_item - > PeerId ( RsPeerId ( authorId ) ) ;
2017-02-21 17:08:02 -05:00
handleIncomingItem ( msg_item ) ;
}
else
2017-02-26 07:06:38 -05:00
{
2017-03-01 20:37:53 -05:00
std : : cerr < < __PRETTY_FUNCTION__ < < " Item could not be "
2017-02-21 17:08:02 -05:00
< < " deserialised. Format error?? " < < std : : endl ;
2017-02-26 07:06:38 -05:00
return false ;
}
return true ;
2017-02-21 17:08:02 -05:00
}
2017-03-01 20:37:53 -05:00
bool p3MsgService : : notifyGxsTransSendStatus ( RsGxsTransId mailId ,
GxsTransSendStatus status )
2017-02-21 17:08:02 -05:00
{
2017-03-01 20:37:53 -05:00
std : : cout < < __PRETTY_FUNCTION__ < < " " < < mailId < < " , "
< < static_cast < uint > ( status ) < < std : : endl ;
2017-02-25 18:46:02 -05:00
2017-03-01 20:37:53 -05:00
if ( status = = GxsTransSendStatus : : RECEIPT_RECEIVED )
2017-02-21 17:08:02 -05:00
{
uint32_t msg_id ;
{
RS_STACK_MUTEX ( gxsOngoingMutex ) ;
2017-03-01 20:37:53 -05:00
auto it = gxsOngoingMessages . find ( mailId ) ;
2017-02-21 17:08:02 -05:00
if ( it = = gxsOngoingMessages . end ( ) )
{
2017-03-01 20:37:53 -05:00
std : : cerr < < __PRETTY_FUNCTION__ < < " "
< < mailId
< < " , " < < static_cast < uint > ( status )
< < " (EE) cannot find pending message to acknowledge! "
2017-02-21 17:08:02 -05:00
< < std : : endl ;
return false ;
}
msg_id = it - > second ;
}
2017-02-25 18:46:02 -05:00
// we should now remove the item from the msgOutgoing list.
2017-02-21 17:08:02 -05:00
{
RS_STACK_MUTEX ( mMsgMtx ) ;
2017-02-25 18:46:02 -05:00
auto it2 = msgOutgoing . find ( msg_id ) ;
if ( it2 = = msgOutgoing . end ( ) )
2017-02-21 17:08:02 -05:00
{
2017-03-01 20:37:53 -05:00
std : : cerr < < __PRETTY_FUNCTION__ < < " " < < mailId
< < " , " < < static_cast < uint > ( status ) < < " (II) "
2017-02-25 18:46:02 -05:00
< < " received receipt for message that is not in "
< < " outgoing list, probably it has been acknoweldged "
< < " before by other means. " < < std : : endl ;
2017-02-21 17:08:02 -05:00
return true ;
}
2017-02-25 18:46:02 -05:00
delete it2 - > second ;
msgOutgoing . erase ( it2 ) ;
2017-02-21 17:08:02 -05:00
}
2017-02-25 18:46:02 -05:00
RsServer : : notify ( ) - > notifyListChange ( NOTIFY_LIST_MESSAGELIST ,
NOTIFY_TYPE_ADD ) ;
IndicateConfigChanged ( ) ;
return true ;
2017-02-21 17:08:02 -05:00
}
2017-03-01 20:37:53 -05:00
if ( status > = GxsTransSendStatus : : FAILED_RECEIPT_SIGNATURE )
2017-02-21 17:08:02 -05:00
{
uint32_t msg_id ;
{
RS_STACK_MUTEX ( gxsOngoingMutex ) ;
2017-03-01 20:37:53 -05:00
std : : cerr < < __PRETTY_FUNCTION__ < < " mail delivery "
< < " mailId: " < < mailId
2017-02-25 18:46:02 -05:00
< < " failed with " < < static_cast < uint32_t > ( status ) ;
2017-03-01 20:37:53 -05:00
auto it = gxsOngoingMessages . find ( mailId ) ;
2017-02-21 17:08:02 -05:00
if ( it = = gxsOngoingMessages . end ( ) )
{
2017-02-25 18:46:02 -05:00
std : : cerr < < " cannot find pending message to notify "
2017-02-21 17:08:02 -05:00
< < std : : endl ;
return false ;
}
msg_id = it - > second ;
}
2017-02-25 18:46:02 -05:00
std : : cerr < < " message id = " < < msg_id < < std : : endl ;
2017-02-21 17:08:02 -05:00
{
RS_STACK_MUTEX ( mMsgMtx ) ;
2017-02-25 18:46:02 -05:00
auto mit = msgOutgoing . find ( msg_id ) ;
if ( mit = = msgOutgoing . end ( ) )
2017-02-21 17:08:02 -05:00
{
2017-02-25 18:46:02 -05:00
std : : cerr < < " message has been notified as not delivered, "
< < " but it not on outgoing list. "
< < std : : endl ;
2017-02-21 17:08:02 -05:00
return true ;
}
2017-02-25 18:46:02 -05:00
std : : cerr < < " reseting the ROUTED flag so that the message is "
< < " requested again " < < std : : endl ;
2017-02-21 17:08:02 -05:00
2017-02-25 18:46:02 -05:00
// clear the routed flag so that the message is requested again
mit - > second - > msgFlags & = ~ RS_MSG_FLAGS_ROUTED ;
return true ;
2017-02-21 17:08:02 -05:00
}
}
2017-02-26 07:06:38 -05:00
return true ;
2017-02-21 17:08:02 -05:00
}
void p3MsgService : : receiveGRouterData ( const RsGxsId & destination_key ,
const RsGxsId & signing_key ,
GRouterServiceId & /*client_id*/ ,
uint8_t * data , uint32_t data_size )
{
std : : cerr < < " p3MsgService::receiveGRouterData(): received message item of "
< < " size " < < data_size < < " , for key " < < destination_key
< < std : : endl ;
/* first make sure that we havn't already received the data. Since we allow
* to re - send messages , it ' s necessary to check . */
Sha1CheckSum hash = RsDirUtil : : sha1sum ( data , data_size ) ;
{
RS_STACK_MUTEX ( recentlyReceivedMutex ) ;
if ( mRecentlyReceivedMessageHashes . find ( hash ) ! =
mRecentlyReceivedMessageHashes . end ( ) )
{
2017-02-25 18:46:02 -05:00
std : : cerr < < " p3MsgService::receiveGRouterData(...) (II) receiving "
2017-02-21 17:08:02 -05:00
< < " distant message of hash " < < hash < < " more than once "
2017-02-25 18:46:02 -05:00
< < " . Probably it has arrived before by other means. "
2017-02-21 17:08:02 -05:00
< < std : : endl ;
free ( data ) ;
return ;
}
mRecentlyReceivedMessageHashes [ hash ] = time ( NULL ) ;
}
IndicateConfigChanged ( ) ;
RsItem * item = _serialiser - > deserialise ( data , & data_size ) ;
2016-01-07 22:22:05 -05:00
free ( data ) ;
2013-12-27 15:06:47 -05:00
2017-02-21 17:08:02 -05:00
RsMsgItem * msg_item = dynamic_cast < RsMsgItem * > ( item ) ;
2013-12-27 15:06:47 -05:00
2017-02-21 17:08:02 -05:00
if ( msg_item ! = NULL )
{
std : : cerr < < " Encrypted item correctly deserialised. Passing on to incoming list. " < < std : : endl ;
2013-04-29 16:44:48 -04:00
2017-02-21 17:08:02 -05:00
msg_item - > msgFlags | = RS_MSG_FLAGS_DISTANT ;
/* we expect complete msgs - remove partial flag just in case someone has funny ideas */
msg_item - > msgFlags & = ~ RS_MSG_FLAGS_PARTIAL ;
2015-02-20 05:34:11 -05:00
2017-02-21 17:08:02 -05:00
msg_item - > PeerId ( RsPeerId ( signing_key ) ) ; // hack to pass on GXS id.
handleIncomingItem ( msg_item ) ;
}
else
std : : cerr < < " Item could not be deserialised. Format error?? " < < std : : endl ;
2014-03-17 16:56:06 -04:00
}
2015-01-11 17:18:28 -05:00
void p3MsgService : : sendDistantMsgItem ( RsMsgItem * msgitem )
2013-04-29 16:44:48 -04:00
{
2017-02-21 17:08:02 -05:00
RsGxsId destination_key_id ( msgitem - > PeerId ( ) ) ;
RsGxsId signing_key_id ;
2015-01-11 17:18:28 -05:00
2017-02-21 17:08:02 -05:00
/* just in case, but normally we should always have this flag set, when
* ending up here . */
msgitem - > msgFlags | = RS_MSG_FLAGS_DISTANT ;
2015-01-11 17:18:28 -05:00
2017-02-21 17:08:02 -05:00
{
RS_STACK_MUTEX ( mMsgMtx ) ;
2015-01-11 17:18:28 -05:00
2017-02-21 17:08:02 -05:00
std : : map < uint32_t , RsGxsId > : : const_iterator it =
mDistantOutgoingMsgSigners . find ( msgitem - > msgId ) ;
if ( it = = mDistantOutgoingMsgSigners . end ( ) )
{
std : : cerr < < " (EE) no signer registered for distant message "
< < msgitem - > msgId < < " . Cannot send! " < < std : : endl ;
return ;
}
signing_key_id = it - > second ;
if ( signing_key_id . isNull ( ) )
{
std : : cerr < < " ERROR: cannot find signing key id for msg id "
< < msgitem - > msgId < < " available keys are: " < < std : : endl ;
typedef std : : map < uint32_t , RsGxsId > : : const_iterator itT ;
for ( itT it = mDistantOutgoingMsgSigners . begin ( ) ;
it ! = mDistantOutgoingMsgSigners . end ( ) ; + + it )
std : : cerr < < " \t " < < it - > first < < " " < < it - > second
< < std : : endl ;
return ;
}
}
2013-05-08 12:58:22 -04:00
# ifdef DEBUG_DISTANT_MSG
2017-02-21 17:08:02 -05:00
std : : cerr < < " p3MsgService::sendDistanteMsgItem(): sending distant msg item "
< < " msg ID: " < < msgitem - > msgId < < " to peer: "
< < destination_key_id < < " signing: " < < signing_key_id
< < std : : endl ;
2013-05-08 12:58:22 -04:00
# endif
2013-08-31 17:10:04 -04:00
2017-02-21 17:08:02 -05:00
/* The item is serialized and turned into a generic turtle item. Use use the
* explicit serialiser to make sure that the msgId is not included */
2015-01-11 17:18:28 -05:00
2017-04-24 08:14:34 -04:00
uint32_t msg_serialized_rssize = RsMsgSerialiser ( RsServiceSerializer : : SERIALIZATION_FLAG_NONE ) . size ( msgitem ) ;
2016-01-09 10:58:49 -05:00
RsTemporaryMemory msg_serialized_data ( msg_serialized_rssize ) ;
2015-01-11 17:18:28 -05:00
2017-04-24 08:14:34 -04:00
if ( ! RsMsgSerialiser ( RsServiceSerializer : : SERIALIZATION_FLAG_NONE ) . serialise ( msgitem , msg_serialized_data , & msg_serialized_rssize ) )
2015-01-11 17:18:28 -05:00
{
std : : cerr < < " (EE) p3MsgService::sendTurtleData(): Serialization error. " < < std : : endl ;
return ;
}
2013-08-31 17:10:04 -04:00
# ifdef DEBUG_DISTANT_MSG
2017-02-21 17:08:02 -05:00
std : : cerr < < " serialised size : " < < msg_serialized_rssize < < std : : endl ;
2013-08-31 17:10:04 -04:00
# endif
2014-03-29 10:18:05 -04:00
2017-02-21 17:08:02 -05:00
GRouterMsgPropagationId grouter_message_id ;
mGRouter - > sendData ( destination_key_id , GROUTER_CLIENT_ID_MESSAGES ,
msg_serialized_data , msg_serialized_rssize ,
signing_key_id , grouter_message_id ) ;
2017-03-01 20:37:53 -05:00
RsGxsTransId gxsMailId ;
2017-05-27 16:19:52 -04:00
mGxsTransServ . sendData ( gxsMailId , GxsTransSubServices : : P3_MSG_SERVICE ,
2017-02-25 18:46:02 -05:00
signing_key_id , destination_key_id ,
msg_serialized_data , msg_serialized_rssize ) ;
2015-01-11 17:18:28 -05:00
2017-02-21 17:08:02 -05:00
/* now store the grouter id along with the message id, so that we can keep
* track of received messages */
2015-01-11 17:18:28 -05:00
2017-02-21 17:08:02 -05:00
{
RS_STACK_MUTEX ( mMsgMtx ) ;
_ongoing_messages [ grouter_message_id ] = msgitem - > msgId ;
}
{
RS_STACK_MUTEX ( gxsOngoingMutex ) ;
gxsOngoingMessages [ gxsMailId ] = msgitem - > msgId ;
}
IndicateConfigChanged ( ) ; // save _ongoing_messages
2013-04-29 16:44:48 -04:00
}