Corrected bugs in chat:

- sending large messages now works, thanks to RsChatMsgItem splitting (not 100% backward compatible, but avoids crashing)
	- removed crash due to dynamic_cast onto a deleted pointer.



git-svn-id: http://svn.code.sf.net/p/retroshare/code/trunk@4187 b45a01b8-16f6-495d-af2f-9b41ad6348cc
This commit is contained in:
csoler 2011-05-04 20:52:45 +00:00
parent ed3fa92096
commit 31b950a8fe
3 changed files with 89 additions and 9 deletions

View File

@ -43,6 +43,7 @@ const uint32_t RS_CHAT_FLAG_CUSTOM_STATE = 0x0010; // used for transm
const uint32_t RS_CHAT_FLAG_PUBLIC = 0x0020; const uint32_t RS_CHAT_FLAG_PUBLIC = 0x0020;
const uint32_t RS_CHAT_FLAG_REQUEST_CUSTOM_STATE = 0x0040; const uint32_t RS_CHAT_FLAG_REQUEST_CUSTOM_STATE = 0x0040;
const uint32_t RS_CHAT_FLAG_CUSTOM_STATE_AVAILABLE = 0x0080; const uint32_t RS_CHAT_FLAG_CUSTOM_STATE_AVAILABLE = 0x0080;
const uint32_t RS_CHAT_FLAG_PARTIAL_MESSAGE = 0x0100;
const uint32_t RS_CHATMSG_CONFIGFLAG_INCOMING = 0x0001; const uint32_t RS_CHATMSG_CONFIGFLAG_INCOMING = 0x0001;

View File

@ -204,6 +204,34 @@ void p3ChatService::sendStatusString( const std::string& id , const std::string&
sendItem(cs); sendItem(cs);
} }
void p3ChatService::checkSizeAndSendMessage(RsChatMsgItem *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 ;
while(msg->message.size() > MAX_STRING_SIZE)
{
// chop off the first 15000 wchars
RsChatMsgItem *item = new RsChatMsgItem(*msg) ;
item->message = item->message.substr(0,MAX_STRING_SIZE) ;
msg->message = msg->message.substr(MAX_STRING_SIZE,msg->message.size()-MAX_STRING_SIZE) ;
// Clear out any one time flags that should not be copied into multiple objects. This is
// a precaution, in case the receivign peer does not yet handle split messages transparently.
//
item->chatFlags &= (RS_CHAT_FLAG_PRIVATE | RS_CHAT_FLAG_PUBLIC) ;
// Indicate that the message is to be continued.
//
item->chatFlags |= RS_CHAT_FLAG_PARTIAL_MESSAGE ;
sendItem(item) ;
}
sendItem(msg) ;
}
bool p3ChatService::sendPrivateChat(std::string &id, std::wstring &msg) bool p3ChatService::sendPrivateChat(std::string &id, std::wstring &msg)
{ {
// make chat item.... // make chat item....
@ -262,7 +290,7 @@ bool p3ChatService::sendPrivateChat(std::string &id, std::wstring &msg)
std::cerr << std::endl; std::cerr << std::endl;
#endif #endif
sendItem(ci); checkSizeAndSendMessage(ci);
// Check if custom state string has changed, in which case it should be sent to the peer. // Check if custom state string has changed, in which case it should be sent to the peer.
bool should_send_state_string = false ; bool should_send_state_string = false ;
@ -297,6 +325,50 @@ bool p3ChatService::sendPrivateChat(std::string &id, std::wstring &msg)
return true; return true;
} }
bool p3ChatService::checkAndRebuildPartialMessage(RsChatMsgItem *ci)
{
// Check is the item is ending an incomplete item.
//
std::map<std::string,RsChatMsgItem*>::iterator it = _pendingPartialMessages.find(ci->PeerId()) ;
bool ci_is_partial = ci->chatFlags & RS_CHAT_FLAG_PARTIAL_MESSAGE ;
if(it != _pendingPartialMessages.end())
{
#ifdef CHAT_DEBUG
std::cerr << "Pending messahe found. Happending it." << std::endl;
#endif
// Yes, there is. Append the item to ci.
ci->message = it->second->message + ci->message ;
ci->chatFlags |= it->second->chatFlags ;
delete it->second ;
if(!ci_is_partial)
_pendingPartialMessages.erase(it) ;
}
if(ci_is_partial)
{
#ifdef CHAT_DEBUG
std::cerr << "Message is partial, storing for later." << std::endl;
#endif
// The item is a partial message. Push it, and wait for the rest.
//
_pendingPartialMessages[ci->PeerId()] = ci ;
return false ;
}
else
{
#ifdef CHAT_DEBUG
std::cerr << "Message is complete, using it now." << std::endl;
#endif
return true ;
}
}
void p3ChatService::receiveChatQueue() void p3ChatService::receiveChatQueue()
{ {
bool publicChanged = false; bool publicChanged = false;
@ -321,11 +393,14 @@ void p3ChatService::receiveChatQueue()
std::cerr << std::endl; std::cerr << std::endl;
std::cerr << "Got msg. Flags = " << ci->chatFlags << std::endl ; std::cerr << "Got msg. Flags = " << ci->chatFlags << std::endl ;
#endif #endif
if(!checkAndRebuildPartialMessage(ci))
continue ;
if(ci->chatFlags & RS_CHAT_FLAG_REQUESTS_AVATAR) // no msg here. Just an avatar request. if(ci->chatFlags & RS_CHAT_FLAG_REQUESTS_AVATAR) // no msg here. Just an avatar request.
{ {
sendAvatarJpegData(ci->PeerId()) ; sendAvatarJpegData(ci->PeerId()) ;
delete item ; delete item ;
continue ;
} }
else // normal msg. Return it normally. else // normal msg. Return it normally.
{ {
@ -371,14 +446,12 @@ void p3ChatService::receiveChatQueue()
} }
} /* UNLOCK */ } /* UNLOCK */
} }
continue ;
} }
RsChatStatusItem *cs = dynamic_cast<RsChatStatusItem*>(item) ; RsChatStatusItem *cs = dynamic_cast<RsChatStatusItem*>(item) ;
if(cs != NULL){ if(cs != NULL)
{
#ifdef CHAT_DEBUG #ifdef CHAT_DEBUG
std::cerr << "Received status string \"" << cs->status_string << "\"" << std::endl ; std::cerr << "Received status string \"" << cs->status_string << "\"" << std::endl ;
#endif #endif
@ -388,8 +461,8 @@ void p3ChatService::receiveChatQueue()
} }
else // Check if new custom string is available at peer's. If so, send a request to get the custom string. else // Check if new custom string is available at peer's. If so, send a request to get the custom string.
if(cs->flags & RS_CHAT_FLAG_CUSTOM_STATE){ if(cs->flags & RS_CHAT_FLAG_CUSTOM_STATE)
{
receiveStateString(cs->PeerId(),cs->status_string) ; // store it receiveStateString(cs->PeerId(),cs->status_string) ; // store it
rsicontrol->getNotify().notifyCustomState(cs->PeerId(), cs->status_string) ; rsicontrol->getNotify().notifyCustomState(cs->PeerId(), cs->status_string) ;
}else }else
@ -406,7 +479,6 @@ void p3ChatService::receiveChatQueue()
delete item; delete item;
continue ; continue ;
} }
RsChatAvatarItem *ca = dynamic_cast<RsChatAvatarItem*>(item) ; RsChatAvatarItem *ca = dynamic_cast<RsChatAvatarItem*>(item) ;
@ -876,7 +948,7 @@ void p3ChatService::sendAvatarJpegData(const std::string& peer_id)
} }
else { else {
#ifdef CHAT_DEBUG #ifdef CHAT_DEBUG
std::cerr << "Doing nothing" << std::endl ; std::cerr << "We have no avatar yet: Doing nothing" << std::endl ;
#endif #endif
} }
} }

View File

@ -189,6 +189,12 @@ class p3ChatService: public p3Service, public p3Config, public pqiMonitor
/// Send a request for custom status string /// Send a request for custom status string
void sendCustomStateRequest(const std::string& peer_id); void sendCustomStateRequest(const std::string& peer_id);
/// called as a proxy to sendItem(). Possibly splits item into multiple items of size lower than the maximum item size.
void checkSizeAndSendMessage(RsChatMsgItem *item) ;
/// Called when a RsChatMsgItem is received. The item may be collapsed with any waiting partial chat item from the same peer.
bool checkAndRebuildPartialMessage(RsChatMsgItem*) ;
RsChatAvatarItem *makeOwnAvatarItem() ; RsChatAvatarItem *makeOwnAvatarItem() ;
RsChatStatusItem *makeOwnCustomStateStringItem() ; RsChatStatusItem *makeOwnCustomStateStringItem() ;
@ -200,6 +206,7 @@ class p3ChatService: public p3Service, public p3Config, public pqiMonitor
AvatarInfo *_own_avatar ; AvatarInfo *_own_avatar ;
std::map<std::string,AvatarInfo *> _avatars ; std::map<std::string,AvatarInfo *> _avatars ;
std::map<std::string,RsChatMsgItem *> _pendingPartialMessages ;
std::string _custom_status_string ; std::string _custom_status_string ;
std::map<std::string,StateStringInfo> _state_strings ; std::map<std::string,StateStringInfo> _state_strings ;