mirror of
https://github.com/RetroShare/RetroShare.git
synced 2025-01-16 09:57:19 -05:00
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:
parent
ed3fa92096
commit
31b950a8fe
@ -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;
|
||||||
|
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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 ;
|
||||||
|
Loading…
Reference in New Issue
Block a user