teamtalk文件服务器通信机制

1.消息服务器与文件服务器的连接
  消息服务器中的文件服务器连接会主动获取文件服务器的地址参数,保存在GetFileServerIPList
 
  
                                         客户端
 
  
                                               CID_FILE_REQUEST
 
  
                                                             msg_server
              file_server
 
  
 
  
 
  
1.client向msg_server发送case CID_FILE_REQUEST
2.msg_server收到CID_FILE_REQUEST进行处理,获取文件服务器的连接,然后转发CID_OTHER_FILE_TRANSFER_REQ
    HandleClientFileRequest(this, pPdu);
 
  
void CFileHandler::HandleClientFileRequest(CMsgConn* pMsgConn, CImPdu* pPdu)
{
    IM::File::IMFileReq msg;
    CHECK_PB_PARSE_MSG(msg.ParseFromArray(pPdu->GetBodyData(), pPdu->GetBodyLength()));
 
  
    uint32_t from_id = pMsgConn->GetUserId();
    uint32_t to_id = msg.to_user_id();
    string file_name = msg.file_name();
    uint32_t file_size = msg.file_size();
    uint32_t trans_mode = msg.trans_mode();
    log("HandleClientFileRequest, %u->%u, fileName: %s, trans_mode: %u.", from_id, to_id, file_name.c_str(), trans_mode);
 
  
    CDbAttachData attach(ATTACH_TYPE_HANDLE, pMsgConn->GetHandle());
    CFileServConn* pFileConn = get_random_file_serv_conn();
    if (pFileConn)
    {
        IM::Server::IMFileTransferReq msg2;
        msg2.set_from_user_id(from_id);
        msg2.set_to_user_id(to_id);
        msg2.set_file_name(file_name);
        msg2.set_file_size(file_size);
        msg2.set_trans_mode((IM::BaseDefine::TransferFileType)trans_mode);
        msg2.set_attach_data(attach.GetBuffer(), attach.GetLength());
        CImPdu pdu;
        pdu.SetPBMsg(&msg2);
        pdu.SetServiceId(SID_OTHER);
        pdu.SetCommandId(CID_OTHER_FILE_TRANSFER_REQ);
        pdu.SetSeqNum(pPdu->GetSeqNum());
 
  
        if (IM::BaseDefine::FILE_TYPE_OFFLINE == trans_mode)
        {
            pFileConn->SendPdu(&pdu);
        }
    }
}
3.文件服务器收到CID_OTHER_FILE_TRANSFER_REQ,新建一条任务,然后发回给msg_server
void FileMsgServerConn::_HandleMsgFileTransferReq(CImPdu* pdu) {
    IM::Server::IMFileTransferReq transfer_req;
    CHECK_PB_PARSE_MSG(transfer_req.ParseFromArray(pdu->GetBodyData(), pdu->GetBodyLength()));
 
  
 
  
    uint32_t from_id = transfer_req.from_user_id();
    uint32_t to_id = transfer_req.to_user_id();
 
  
    IM::Server::IMFileTransferRsp transfer_rsp;
    transfer_rsp.set_result_code(1);
    transfer_rsp.set_from_user_id(from_id);
    transfer_rsp.set_to_user_id(to_id);
    transfer_rsp.set_file_name(transfer_req.file_name());
    transfer_rsp.set_file_size(transfer_req.file_size());
    transfer_rsp.set_task_id("");
    transfer_rsp.set_trans_mode(transfer_req.trans_mode());
    transfer_rsp.set_attach_data(transfer_req.attach_data());
 
  
 
  
    bool rv = false;
    do {
        std::string task_id = GenerateUUID();
        if (task_id.empty()) {
            log("Create task id failed");
            break;
        }
        log("trams_mode=%d, task_id=%s, from_id=%d, to_id=%d, file_name=%s, file_size=%d", transfer_req.trans_mode(), task_id.c_str(), from_id, to_id, transfer_req.file_name().c_str(), transfer_req.file_size());
 
  
        BaseTransferTask* transfer_task = TransferTaskManager::GetInstance()->NewTransferTask(
                                                                                              transfer_req.trans_mode(),
                                                                                              task_id,
                                                                                              from_id,
                                                                                              to_id,
                                                                                              transfer_req.file_name(),
                                                                                              transfer_req.file_size());
 
  
        if (transfer_task == NULL) {
            // 创建未成功
            // close connection with msg svr
            // need_close = true;
            log("Create task failed");
            break;
        }
 
  
        transfer_rsp.set_result_code(0);
        transfer_rsp.set_task_id(task_id);
        rv = true;
        // need_seq_no = false;
 
  
        log("Create task succeed, task id %s, task type %d, from user %d, to user %d", task_id.c_str(), transfer_req.trans_mode(), from_id, to_id);
    } while (0);
 
  
    ::SendMessageLite(this, SID_OTHER, CID_OTHER_FILE_TRANSFER_RSP, pdu->GetSeqNum(), &transfer_rsp);
 
  
    if (!rv) {
        // 未创建成功,关闭连接
        Close();
    }
}
4.msg_server收到CID_OTHER_FILE_TRANSFER_RSP,给发送方返回CID_FILE_RESPONSE,给接收方返回CID_FILE_NOTIFY
void CFileServConn::_HandleFileMsgTransRsp(CImPdu* pPdu)
{
    IM::Server::IMFileTransferRsp msg;
    CHECK_PB_PARSE_MSG(msg.ParseFromArray(pPdu->GetBodyData(), pPdu->GetBodyLength()));
 
  
    uint32_t result = msg.result_code();
    uint32_t from_id = msg.from_user_id();
    uint32_t to_id = msg.to_user_id();
    string file_name = msg.file_name();
    uint32_t file_size = msg.file_size();
    string task_id = msg.task_id();
    uint32_t trans_mode = msg.trans_mode();
    CDbAttachData attach((uchar_t*)msg.attach_data().c_str(), msg.attach_data().length());
    log("HandleFileMsgTransRsp, result: %u, from_user_id: %u, to_user_id: %u, file_name: %s, \
        task_id: %s, trans_mode: %u. ", result, from_id, to_id,
        file_name.c_str(), task_id.c_str(), trans_mode);
 
  
    const list* ip_addr_list = GetFileServerIPList();
 
  
    IM::File::IMFileRsp msg2;
    msg2.set_result_code(result);
    msg2.set_from_user_id(from_id);
    msg2.set_to_user_id(to_id);
    msg2.set_file_name(file_name);
    msg2.set_task_id(task_id);
    msg2.set_trans_mode((IM::BaseDefine::TransferFileType)trans_mode);
    for (list::const_iterator it = ip_addr_list->begin(); it != ip_addr_list->end(); it++)
    {
        IM::BaseDefine::IpAddr ip_addr_tmp = *it;
        IM::BaseDefine::IpAddr* ip_addr = msg2.add_ip_addr_list();
        ip_addr->set_ip(ip_addr_tmp.ip());
        ip_addr->set_port(ip_addr_tmp.port());
    }
    CImPdu pdu;
    pdu.SetPBMsg(&msg2);
    pdu.SetServiceId(SID_FILE);
    pdu.SetCommandId(CID_FILE_RESPONSE);
    pdu.SetSeqNum(pPdu->GetSeqNum());
    uint32_t handle = attach.GetHandle();
 
  
    CMsgConn* pFromConn = CImUserManager::GetInstance()->GetMsgConnByHandle(from_id, handle);
    if (pFromConn)
    {
        pFromConn->SendPdu(&pdu);
    }
 
  
    if (result == 0)
    {
        IM::File::IMFileNotify msg3;
        msg3.set_from_user_id(from_id);
        msg3.set_to_user_id(to_id);
        msg3.set_file_name(file_name);
        msg3.set_file_size(file_size);
        msg3.set_task_id(task_id);
        msg3.set_trans_mode((IM::BaseDefine::TransferFileType)trans_mode);
        msg3.set_offline_ready(0);
        for (list::const_iterator it = ip_addr_list->begin(); it != ip_addr_list->end(); it++)
        {
            IM::BaseDefine::IpAddr ip_addr_tmp = *it;
            IM::BaseDefine::IpAddr* ip_addr = msg3.add_ip_addr_list();
            ip_addr->set_ip(ip_addr_tmp.ip());
            ip_addr->set_port(ip_addr_tmp.port());
        }
        CImPdu pdu2;
        pdu2.SetPBMsg(&msg3);
        pdu2.SetServiceId(SID_FILE);
        pdu2.SetCommandId(CID_FILE_NOTIFY);
 
  
        //send notify to target user
        CImUser* pToUser = CImUserManager::GetInstance()->GetImUserById(to_id);
        if (pToUser)
        {
            pToUser->BroadcastPduWithOutMobile(&pdu2);
        }
 
  
        //send to route server
        CRouteServConn* pRouteConn = get_route_serv_conn();
        if (pRouteConn) {
            pRouteConn->SendPdu(&pdu2);
        }
    }
}
5.客户端收到CID_FILE_RESPONSE,
发送方开始连接文件服务器
void FileTransferModule_Impl::_sendfileResponse(IN std::string& body)
{
        IM::File::IMFileRsp imFileRsp;
        if (!imFileRsp.ParseFromString(body))
        {
                LOG__(ERR, _T("parse failed,body:%s"), util::stringToCString(body));
                return;
        }
 
  
        UInt32 nResult = imFileRsp.result_code();
        if (nResult != 0)
        {
                LOG__(ERR, _T("_sendfileResponse result != 0"));
                module::getFileTransferModule()->asynNotifyObserver(module::KEY_FILESEVER_UPLOAD_FAILED);
        }
 
  
        TransferFileEntity fileEntity;
        fileEntity.sTaskID = imFileRsp.task_id();
    assert(!fileEntity.sTaskID.empty());
        fileEntity.sFromID = util::uint32ToString(imFileRsp.from_user_id());
        fileEntity.sToID = util::uint32ToString(imFileRsp.to_user_id());
        fileEntity.sFileName = imFileRsp.file_name();
        fileEntity.setSaveFilePath(util::stringToCString(fileEntity.sFileName));//发送方文件地址,就是保存地址
        fileEntity.time = static_cast(time(0));
        uint32_t transMode = imFileRsp.trans_mode();
        if (IM::BaseDefine::TransferFileType::FILE_TYPE_ONLINE == transMode)
        {
                fileEntity.nClientMode = IM::BaseDefine::ClientFileRole::CLIENT_REALTIME_SENDER;
        }
        else if (IM::BaseDefine::TransferFileType::FILE_TYPE_OFFLINE == transMode)
        {
                fileEntity.nClientMode = IM::BaseDefine::ClientFileRole::CLIENT_OFFLINE_UPLOAD;
        }
        fileEntity.pFileObject = new TransferFile(util::stringToCString(fileEntity.sFileName),FALSE);
        if (fileEntity.pFileObject)
        {
                fileEntity.nFileSize = fileEntity.pFileObject->length();
        }
 
  
        UINT32 nIPCount = imFileRsp.ip_addr_list_size();
        if (nIPCount <= 0)
        {
                return;
        }
        const IM::BaseDefine::IpAddr& ipAdd = imFileRsp.ip_addr_list(0);
        fileEntity.sIP = ipAdd.ip();
        fileEntity.nPort = ipAdd.port();
 
  
        if (!TransferFileEntityManager::getInstance()->pushTransferFileEntity(fileEntity))
                TransferFileEntityManager::getInstance()->updateFileInfoBysTaskID(fileEntity);
 
  
        LOG__(DEBG, _T("FileTransferSevice_Impl::准备连接文件服务器 sTaskId = %s"), util::stringToCString(fileEntity.sTaskID));
        TransferFileEntityManager::getInstance()->openFileSocketByTaskId(fileEntity.sTaskID);
}
接收方收到通知,开始连接文件服务器
void FileTransferModule_Impl::_fileNotify(IN std::string& body)
{
        IM::File::IMFileNotify imFileNotify;
        if (!imFileNotify.ParseFromString(body))
        {
                LOG__(ERR, _T("parse failed,body:%s"), util::stringToCString(body));
                return;
        }
        TransferFileEntity file;
        file.sFileName = imFileNotify.file_name();
        file.sFromID = util::uint32ToString(imFileNotify.from_user_id());
        file.sToID = util::uint32ToString(imFileNotify.to_user_id());
        file.sTaskID = imFileNotify.task_id();
        file.nFileSize = imFileNotify.file_size();
 
  
        UINT32 nIPCount = imFileNotify.ip_addr_list_size();
        if (nIPCount <= 0)
        {
                return;
        }
        const IM::BaseDefine::IpAddr& ipAdd = imFileNotify.ip_addr_list(0);
        file.sIP = ipAdd.ip();
        file.nPort = ipAdd.port();
 
  
        uint32_t transMode = imFileNotify.trans_mode();
        if (IM::BaseDefine::TransferFileType::FILE_TYPE_ONLINE == transMode)
        {
                file.nClientMode = IM::BaseDefine::ClientFileRole::CLIENT_REALTIME_RECVER;
        }
        else if (IM::BaseDefine::TransferFileType::FILE_TYPE_OFFLINE == transMode)
        {
                file.nClientMode = IM::BaseDefine::ClientFileRole::CLIENT_OFFLINE_DOWNLOAD;
        }
        file.time = static_cast(time(0));
        TransferFileEntityManager::getInstance()->pushTransferFileEntity(file);
        LOG__(DEBG, _T("FileTransferSevice_Impl::给你发文件 sFileID = %s"), util::stringToCString(file.sTaskID));
 
  
        if (1 == imFileNotify.offline_ready())
        {
                //TODO离线文件传输结束
        }
 
  
        //连接服务器
        TransferFileEntityManager::getInstance()->openFileSocketByTaskId(file.sTaskID);
}
6.接收方开始接受文件,向文件服务器发文件拉取请求 CID_FILE_PULL_DATA_REQ
BOOL FileTransferUIThread::acceptFileTransfer(const std::string& taskId)
{
        FileTransferSocket* pFileSocket = _findFileSocketByTaskId(taskId);
        if (pFileSocket)
        {
                imcore::IMLibCoreStartOperationWithLambda(
                        [=]()
                {
                        TransferFileEntity fileEntity;
                        if (TransferFileEntityManager::getInstance()->getFileInfoByTaskId(taskId, fileEntity))
                        {
                                int mode = fileEntity.nClientMode == IM::BaseDefine::ClientFileRole::CLIENT_OFFLINE_DOWNLOAD ? IM::BaseDefine::TransferFileType::FILE_TYPE_OFFLINE : IM::BaseDefine::TransferFileType::FILE_TYPE_ONLINE;
                LOG__(APP, _T("IMFilePullDataReq,taskId:%s"),util::stringToCString(taskId));
                IM::File::IMFilePullDataReq imFilePullDataReq;
                                imFilePullDataReq.set_task_id(taskId);
                                imFilePullDataReq.set_user_id(util::stringToInt32(fileEntity.sToID));
                                imFilePullDataReq.set_trans_mode(static_cast(mode));
                                imFilePullDataReq.set_offset(0);
                fileEntity.nFileSize > FILE_TRANSFER_BLOCK_SIZE ? imFilePullDataReq.set_data_size(FILE_TRANSFER_BLOCK_SIZE) : imFilePullDataReq.set_data_size(fileEntity.nFileSize);
                                //发包
                pFileSocket->sendPacket(IM::BaseDefine::ServiceID::SID_FILE
                    , IM::BaseDefine::FileCmdID::CID_FILE_PULL_DATA_REQ, &imFilePullDataReq);
                                //CImPduClientFilePullDataReq pduPullDataReq(taskId.c_str(), fileEntity.sToID.c_str()
                                //	, mode, 0, FILE_TRANSFER_BLOCK_SIZE);
                                //pFileSocket->sendPacket(&pduPullDataReq);
                        }
                });
        }
 
  
        return FALSE;
}
7.服务端收到请其开始处理,是在线请求的话直接把请求发个发送方
void FileClientConn::_HandleClientFilePullFileReq(CImPdu *pdu) {
        if (transfer_task_->GetTransMode() == FILE_TYPE_ONLINE) {
            OnlineTransferTask* online = reinterpret_cast(transfer_task_);
            online->SetSeqNum(pdu->GetSeqNum());
            CImConn* conn = transfer_task_->GetOpponentConn(user_id);
            if (conn) {
                conn->SendPdu(pdu);
                // SendMessageLite(conn, SID_FILE, CID_FILE_PULL_DATA_RSP, pdu->GetSeqNum(), &pull_data_rsp);
            }
            // SendPdu(&pdu);
        } else {
            SendMessageLite(this, SID_FILE, CID_FILE_PULL_DATA_RSP, pdu->GetSeqNum(), &pull_data_rsp);
            if (rv == 1) {
                _StatesNotify(CLIENT_FILE_DONE, task_id, transfer_task_->from_user_id(), this);
            }
        }
 }
 8.发送方收到服务器转发的CID_FILE_PULL_DATA_REQ,读取指定大小文件进行发送
 
  
fileEntity.pFileObject->readBlock(fileOffset, fileSize, buff);//读取本地文件的数据块
 
  
 void FileTransferSocket::_filePullDataReqResponse(IN std::string& body)//发
 {
         IM::File::IMFilePullDataReq imFilePullDataReq;
     if (!imFilePullDataReq.ParseFromString(body))
     {
         LOG__(ERR, _T("parse failed,body:%s"), util::stringToCString(body));
         return;
     }
         UInt32 fileSize = imFilePullDataReq.data_size();
         UInt32 fileOffset = imFilePullDataReq.offset();
         std::string taskId = imFilePullDataReq.task_id();
 
  
         TransferFileEntity fileEntity;
         if (!TransferFileEntityManager::getInstance()->getFileInfoByTaskId(taskId, fileEntity))
         {
                 LOG__(ERR, _T("PullDataReqResponse: can't find the fileInfo"));
                 return;
         }
         LOG__(DEBG, _T("send:taskId=%s,filesize=%d,name=%s,BolckSize=%d")
                 ,util::stringToCString(fileEntity.sTaskID)
                 ,fileEntity.nFileSize
                 ,fileEntity.getRealFileName()
         ,fileSize);
         std::string buff;
         if (nullptr == fileEntity.pFileObject)
         {
                 LOG__(ERR, _T("PullDataReqResponse: file boject Destoryed!"));
                 return;
         }
         fileEntity.pFileObject->readBlock(fileOffset, fileSize, buff);//读取本地文件的数据块
         IM::File::IMFilePullDataRsp imFilePullDataRsp;//todo check
     imFilePullDataRsp.set_result_code(0);
         imFilePullDataRsp.set_task_id(taskId);
         imFilePullDataRsp.set_user_id(util::stringToInt32(fileEntity.sFromID));
         imFilePullDataRsp.set_offset(fileOffset);
     imFilePullDataRsp.set_file_data((void*)buff.data(), fileSize);
 
  
     //send packet
     sendPacket(IM::BaseDefine::ServiceID::SID_FILE, IM::BaseDefine::FileCmdID::CID_FILE_PULL_DATA_RSP
         , &imFilePullDataRsp);
 
  
         fileEntity.nProgress = fileOffset + fileSize;
         if (fileEntity.nProgress < fileEntity.nFileSize)
         {
                 //更新进度条
                 TransferFileEntityManager::getInstance()->updateFileInfoBysTaskID(fileEntity);//保存当前进度
                 module::getFileTransferModule()->asynNotifyObserver(module::KEY_FILESEVER_UPDATA_PROGRESSBAR
             , fileEntity.sTaskID);
         }
         else//传输完成
         {
                 if (fileEntity.pFileObject)
                 {
                         delete fileEntity.pFileObject;
                         fileEntity.pFileObject = nullptr;
                 }
                 module::getFileTransferModule()->asynNotifyObserver(module::KEY_FILESEVER_PROGRESSBAR_FINISHED
             , fileEntity.sTaskID);
         }
         TransferFileEntityManager::getInstance()->updateFileInfoBysTaskID(fileEntity);
 }
 9.接收方收到数据CID_FILE_PULL_DATA_RSP,如果没有接收完继续发CID_FILE_PULL_DATA_REQ
 void FileTransferSocket::_filePullDataRspResponse(IN std::string& body)//收
 {
         IM::File::IMFilePullDataRsp imFilePullDataRsp;
     if (!imFilePullDataRsp.ParseFromString(body))
     {
         LOG__(ERR, _T("parse failed,body:%s"), util::stringToCString(body));
         return;
     }
         UInt32 nRes = imFilePullDataRsp.result_code();
         if (0 != nRes)
         {
                 LOG__(ERR, _T("PullDataRspResponse: error result:%d"),nRes);
                 return;
         }
         std::string taskId = imFilePullDataRsp.task_id();
         const std::string& strData = imFilePullDataRsp.file_data();//todo ?????要长度
         void* pData = (void*)(strData.data());
         UInt32 nBlockSize = strData.size();
         UInt32 fileOffset = imFilePullDataRsp.offset();
         TransferFileEntity fileEntity;
         if (!TransferFileEntityManager::getInstance()->getFileInfoByTaskId(taskId, fileEntity))
         {
                 LOG__(ERR, _T("can't find the fileInfo"));
                 return;
         }
         LOG__(DEBG, _T("receive:taskId=%s,filesize=%d,name=%s,BolckSize=%d")
                 , util::stringToCString(fileEntity.sTaskID)
                 , fileEntity.nFileSize
                 , fileEntity.getRealFileName()
         , nBlockSize);
 
  
         //存文件...
         if (!fileEntity.pFileObject->writeBlock(fileOffset, nBlockSize, pData))
         {
                 LOG__(DEBG, _T("writeBlock failed "));
                 return;
         }
 
  
         fileEntity.nProgress = fileOffset + nBlockSize;
         if (fileEntity.nProgress < fileEntity.nFileSize)
         {
                 //更新进度条
                 TransferFileEntityManager::getInstance()->updateFileInfoBysTaskID(fileEntity);//保存当前进度
                 module::getFileTransferModule()->asynNotifyObserver(module::KEY_FILESEVER_UPDATA_PROGRESSBAR
             , fileEntity.sTaskID);
 
  
                 //继续发file block req...
                 int mode = fileEntity.nClientMode == IM::BaseDefine::ClientFileRole::CLIENT_OFFLINE_DOWNLOAD ? IM::BaseDefine::TransferFileType::FILE_TYPE_OFFLINE : IM::BaseDefine::TransferFileType::FILE_TYPE_ONLINE;
                 IM::File::IMFilePullDataReq imFilePullDataReq;
                 imFilePullDataReq.set_task_id(taskId);
                 imFilePullDataReq.set_user_id(util::stringToInt32(fileEntity.sToID));
                 imFilePullDataReq.set_trans_mode(static_cast(mode));
                 imFilePullDataReq.set_offset(fileEntity.nProgress);
 
  
         UInt32 pullSize = fileEntity.nFileSize - fileEntity.nProgress;
         pullSize > nBlockSize ? imFilePullDataReq.set_data_size(nBlockSize) : imFilePullDataReq.set_data_size(pullSize);
 
  
                 // 发包
         sendPacket(IM::BaseDefine::ServiceID::SID_FILE, IM::BaseDefine::FileCmdID::CID_FILE_PULL_DATA_REQ, &imFilePullDataReq);
         }
         else//传输完成
         {
                 if (fileEntity.pFileObject)
                 {
                         delete fileEntity.pFileObject;
                         fileEntity.pFileObject = nullptr;
                 }
 
  
         //告知对方文件传输完成了。
         IM::File::IMFileState imFileState;
         imFileState.set_state(IM::BaseDefine::ClientFileState::CLIENT_FILE_DONE);
         imFileState.set_task_id(taskId);
         imFileState.set_user_id(util::stringToInt32(fileEntity.sToID));
         sendPacket(IM::BaseDefine::ServiceID::SID_FILE, IM::BaseDefine::FileCmdID::CID_FILE_STATE, &imFileState);
 
  
                 TransferFileEntityManager::getInstance()->updateFileInfoBysTaskID(fileEntity);
                 module::getFileTransferModule()->asynNotifyObserver(module::KEY_FILESEVER_PROGRESSBAR_FINISHED, fileEntity.sTaskID);
         }
 }
 
  
 
 

你可能感兴趣的:(c++)