TarsCloud_TarsCpp/servant/libservant/Transceiver.cpp

669 lines
17 KiB
C++
Raw Normal View History

2018-09-05 11:26:21 +08:00
/**
* Tencent is pleased to support the open source community by making Tars available.
*
* Copyright (C) 2016THL A29 Limited, a Tencent company. All rights reserved.
*
* Licensed under the BSD 3-Clause License (the "License"); you may not use this file except
* in compliance with the License. You may obtain a copy of the License at
*
* https://opensource.org/licenses/BSD-3-Clause
*
* Unless required by applicable law or agreed to in writing, software distributed
* under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
* CONDITIONS OF ANY KIND, either express or implied. See the License for the
* specific language governing permissions and limitations under the License.
*/
#include "servant/Transceiver.h"
#include "servant/AdapterProxy.h"
#include "servant/Application.h"
#include "servant/TarsLogger.h"
//#include "servant/AuthLogic.h"
#include "servant/Auth.h"
#if TARS_SSL
#include "util/tc_openssl.h"
#endif
#if TARS_HTTP2
#include "util/tc_nghttp2.h"
#include "util/tc_http2clientmgr.h"
#endif
namespace tars
{
///////////////////////////////////////////////////////////////////////
Transceiver::Transceiver(AdapterProxy * pAdapterProxy,const EndpointInfo &ep)
: _adapterProxy(pAdapterProxy)
, _ep(ep)
, _fd(-1)
, _connStatus(eUnconnected)
, _conTimeoutTime(0)
, _authState(AUTH_INIT)
{
_fdInfo.iType = FDInfo::ET_C_NET;
_fdInfo.p = (void *)this;
_fdInfo.fd = -1;
}
Transceiver::~Transceiver()
{
close();
}
void Transceiver::checkTimeout()
{
if(eConnecting == _connStatus && TNOWMS > _conTimeoutTime)
{
//链接超时
TLOGERROR("[TARS][Transceiver::checkTimeout ep:"<<_adapterProxy->endpoint().desc()<<" , connect timeout]"<<endl);
_adapterProxy->setConTimeout(true);
close();
}
}
bool Transceiver::isSSL() const
{
return _adapterProxy->endpoint().type() == EndpointInfo::SSL;
}
void Transceiver::reconnect()
{
close();
connect();
}
void Transceiver::connect()
{
if(isValid())
{
return;
}
if(_connStatus == eConnecting || _connStatus == eConnected)
{
return;
}
int fd = -1;
if (_ep.type() == EndpointInfo::UDP)
{
fd = NetworkUtil::createSocket(true);
NetworkUtil::setBlock(fd, false);
_connStatus = eConnected;
}
else
{
fd = NetworkUtil::createSocket(false);
NetworkUtil::setBlock(fd, false);
bool bConnected = NetworkUtil::doConnect(fd, _ep.addr());
if(bConnected)
{
setConnected();
}
else
{
_connStatus = Transceiver::eConnecting;
_conTimeoutTime = TNOWMS + _adapterProxy->getConTimeout();
}
}
_fd = fd;
TLOGINFO("[TARS][Transceiver::connect objname:" << _adapterProxy->getObjProxy()->name()
<< ",connect:" << _ep.desc() << ",fd:" << _fd << "]" << endl);
//设置网络qos的dscp标志
if(0 != _ep.qos())
{
int iQos=_ep.qos();
::setsockopt(fd,SOL_IP,IP_TOS,&iQos,sizeof(iQos));
}
//设置套接口选项
vector<SocketOpt> &socketOpts = _adapterProxy->getObjProxy()->getSocketOpt();
for(size_t i=0; i<socketOpts.size(); ++i)
{
if(setsockopt(_fd,socketOpts[i].level,socketOpts[i].optname,socketOpts[i].optval,socketOpts[i].optlen) == -1)
{
TLOGERROR("[TARS][setsockopt error:" << NetworkUtil::errorToString(errno)
<< ",objname:" << _adapterProxy->getObjProxy()->name()
<< ",desc:" << _ep.desc()
<< ",fd:" << _fd
<< ",level:" << socketOpts[i].level
<< ",optname:" << socketOpts[i].optname
<< ",optval:" << socketOpts[i].optval
<<" ]"<< endl);
}
}
_adapterProxy->getObjProxy()->getCommunicatorEpoll()->addFd(fd, &_fdInfo, EPOLLIN|EPOLLOUT);
}
void Transceiver::setConnected()
{
_connStatus = eConnected;
_adapterProxy->setConTimeout(false);
_adapterProxy->addConnExc(false);
_onConnect();
}
void Transceiver::_onConnect()
{
#if TARS_SSL
if (isSSL())
{
// 分配ssl对象
SSL* ssl = NewSSL("client");
if (!ssl)
{
ObjectProxy* obj = _adapterProxy->getObjProxy();
TLOGERROR("[TARS][_onConnect:" << obj->name() << " can't find client SSL_CTX " << endl);
this->close();
return;
}
_openssl.reset(new TC_OpenSSL());
_openssl->Init(ssl, false);
std::string out = _openssl->DoHandshake();
if (_openssl->HasError())
{
TLOGERROR("[TARS] SSL_connect failed " << endl);
this->close();
return;
}
// send the encrypt data from write buffer
if (!out.empty())
{
this->sendRequest(out.data(), out.size(), true);
}
return;
}
#endif
_doAuthReq();
}
void Transceiver::_doAuthReq()
{
ObjectProxy* obj = _adapterProxy->getObjProxy();
TLOGINFO("[TARS][_onConnect:" << obj->name() << " auth Type is " << _adapterProxy->endpoint().authType() << endl);
_adapterProxy->doInvoke();
}
void Transceiver::close()
{
if(!isValid()) return;
#if TARS_SSL
if (_openssl)
{
_openssl->Release();
_openssl.reset();
}
#endif
#if TARS_HTTP2
Http2ClientSessionManager::getInstance()->delSession(_adapterProxy->getId());
#endif
_adapterProxy->getObjProxy()->getCommunicatorEpoll()->delFd(_fd,&_fdInfo,EPOLLIN|EPOLLOUT);
NetworkUtil::closeSocketNoThrow(_fd);
_connStatus = eUnconnected;
_fd = -1;
_sendBuffer.Clear();
_recvBuffer.Clear();
_authState = AUTH_INIT;
TLOGINFO("[TARS][trans close:"<< _adapterProxy->getObjProxy()->name()<< "," << _ep.desc() << "]" << endl);
}
int Transceiver::doRequest()
{
if(!isValid())
{
return -1;
}
int iRet = 0;
//buf不为空,先发生buffer的内容
if(!_sendBuffer.IsEmpty())
{
size_t length = 0;
void* data = NULL;
_sendBuffer.PeekData(data, length);
iRet = this->send(data, length, 0);
//失败,直接返回
if(iRet < 0)
{
return iRet;
}
if(iRet > 0)
{
_sendBuffer.Consume(iRet);
if (_sendBuffer.IsEmpty())
_sendBuffer.Shrink();
else
return 0;
}
}
//取adapter里面积攒的数据
_adapterProxy->doInvoke();
//object里面应该是空的
assert(_adapterProxy->getObjProxy()->timeoutQSize() == 0);
return 0;
}
int Transceiver::sendRequest(const char * pData, size_t iSize, bool forceSend)
{
//空数据 直接返回成功
if(iSize == 0)
{
return eRetOk;
}
if(_connStatus != eConnected)
{
return eRetError;
}
//buf不为空,直接返回失败
//等buffer可写了,epoll会通知写时间
if(!_sendBuffer.IsEmpty())
{
return eRetError;
}
#if TARS_SSL
// 握手数据已加密,直接发送,会话数据需加密
std::string out;
if (isSSL() && _openssl->IsHandshaked())
{
out = _openssl->Write(pData, iSize);
pData = out.data();
iSize = out.size();
}
#endif
int iRet = this->send(pData,iSize,0);
//失败,直接返回
if(iRet < 0)
{
return eRetError;
}
//没有全部发送完,写buffer 返回成功
if(iRet < (int)iSize)
{
_sendBuffer.PushData(pData+iRet,iSize-iRet);
return eRetFull;
}
return eRetOk;
}
//////////////////////////////////////////////////////////
TcpTransceiver::TcpTransceiver(AdapterProxy * pAdapterProxy, const EndpointInfo &ep)
: Transceiver(pAdapterProxy, ep)
{
}
int TcpTransceiver::doResponse(list<ResponsePacket>& done)
{
if(!isValid())
{
return -1;
}
int iRet = 0;
done.clear();
do
{
_recvBuffer.AssureSpace(8 * 1024);
char stackBuffer[64 * 1024];
struct iovec vecs[2];
vecs[0].iov_base = _recvBuffer.WriteAddr();
vecs[0].iov_len = _recvBuffer.WritableSize();
vecs[1].iov_base = stackBuffer;
vecs[1].iov_len = sizeof stackBuffer;
if ((iRet = this->readv(vecs, 2)) > 0)
{
if (static_cast<size_t>(iRet) <= vecs[0].iov_len)
{
_recvBuffer.Produce(iRet);
}
else
{
_recvBuffer.Produce(vecs[0].iov_len);
size_t stackBytes = static_cast<size_t>(iRet) - vecs[0].iov_len;
_recvBuffer.PushData(stackBuffer, stackBytes);
}
}
}
while (iRet>0);
TLOGINFO("[TARS][tcp doResponse objname:" << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ",recvbuf:" << _recvBuffer.ReadableSize() << "]" << endl);
if(!_recvBuffer.IsEmpty())
{
try
{
const char* data = _recvBuffer.ReadAddr();
size_t len = _recvBuffer.ReadableSize();
#if TARS_SSL
if (isSSL())
{
const bool preNotHandshake = !_openssl->IsHandshaked();
std::string out;
if (!_openssl->Read(_recvBuffer.ReadAddr(), _recvBuffer.ReadableSize(), out))
{
TLOGERROR("[TARS][SSL_connect Failed: " << _adapterProxy->getObjProxy()->name() << endl);
this->close();
return -1;
}
else
{
sendRequest(out.data(), out.size(), true);
}
_recvBuffer.Clear();
if (!_openssl->IsHandshaked())
return 0;
if (preNotHandshake)
_doAuthReq();
std::string* plainBuf = _openssl->RecvBuffer();
data = plainBuf->data();
len = plainBuf->size();
}
#endif
size_t pos = 0;
ProxyProtocol& proto = _adapterProxy->getObjProxy()->getProxyProtocol();
if (proto.responseExFunc)
{
long id = _adapterProxy->getId();
pos = proto.responseExFunc(data, len, done, (void*)id);
}
else
{
pos = proto.responseFunc(data, len, done);
}
if(pos > 0)
{
#if TARS_SSL
if (isSSL())
{
std::string* plainBuf = _openssl->RecvBuffer();
plainBuf->erase(0, pos);
}
else
#endif
{
_recvBuffer.Consume(pos);
if (_recvBuffer.Capacity() > 8 * 1024 * 1024)
_recvBuffer.Shrink();
}
}
}
catch (exception &ex)
{
TLOGERROR("[TARS][tcp doResponse objname:" << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ",desc:" << _ep.desc() << ",tcp recv decode error:" << ex.what() << endl);
close();
}
catch (...)
{
TLOGERROR("[TARS][tcp doResponse objname:" << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ",desc" << _ep.desc() << ",tcp recv decode error." << endl);
close();
}
}
return done.empty()?0:1;
}
int TcpTransceiver::send(const void* buf, uint32_t len, uint32_t flag)
{
//只有是连接状态才能收发数据
if(_connStatus != eConnected)
{
return -1;
}
int iRet = ::send(_fd, buf, len, flag);
if (iRet < 0 && errno != EAGAIN)
{
TLOGINFO("[TARS][tcp send objname:" << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ",desc:" << _ep.desc()
<< ",fail! errno:" << errno << "," << strerror(errno) << ",close]" << endl);
close();
return iRet;
}
// EAGAIN不能认为是错误
if (iRet < 0 && errno == EAGAIN)
iRet = 0;
TLOGINFO("[TARS][tcp send," << _adapterProxy->getObjProxy()->name() << ",fd:" << _fd
<< ",desc:" << _ep.desc() << ",len:" << iRet << "]" << endl);
return iRet;
}
int TcpTransceiver::readv(const struct iovec* vecs, int32_t vcnt)
{
//只有是连接状态才能收发数据
if(eConnected != _connStatus)
return -1;
int iRet = ::readv(_fd, vecs, vcnt);
if (iRet == 0 || (iRet < 0 && errno != EAGAIN))
{
TLOGINFO("[TARS][tcp readv, " << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ", " << _ep.desc() <<",ret " << iRet
<< ", fail! errno:" << errno << "," << strerror(errno) << ",close]" << endl);
close();
return 0;
}
TLOGINFO("[TARS][tcp readv," << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << "," << _ep.desc() << ",ret:" << iRet << "]" << endl);
return iRet;
}
int TcpTransceiver::recv(void* buf, uint32_t len, uint32_t flag)
{
//只有是连接状态才能收发数据
if(_connStatus != eConnected)
{
return -1;
}
int iRet = ::recv(_fd, buf, len, flag);
if (iRet == 0 || (iRet < 0 && errno != EAGAIN))
{
TLOGINFO("[TARS][tcp recv objname:" << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ", " << _ep.desc() <<",ret " << iRet
<< ", fail! errno:" << errno << "," << strerror(errno) << ",close]" << endl);
close();
return 0;
}
TLOGINFO("[TARS][tcp recv objname:" << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ",desc:" << _ep.desc() << ",ret:" << iRet << "]" << endl);
return iRet;
}
/////////////////////////////////////////////////////////////////
UdpTransceiver::UdpTransceiver(AdapterProxy * pAdapterProxy, const EndpointInfo &ep)
: Transceiver(pAdapterProxy, ep)
, _recvBuffer(NULL)
{
// UDP不支持鉴权
_authState = AUTH_SUCC;
if(!_recvBuffer)
{
_recvBuffer = new char[DEFAULT_RECV_BUFFERSIZE];
if(!_recvBuffer)
{
throw TC_Exception("objproxy '" + _adapterProxy->getObjProxy()->name() + "' malloc udp receive buffer fail");
}
}
}
UdpTransceiver::~UdpTransceiver()
{
if(_recvBuffer)
{
delete _recvBuffer;
_recvBuffer = NULL;
}
}
int UdpTransceiver::doResponse(list<ResponsePacket>& done)
{
if(!isValid())
{
return -1;
}
int recv = 0;
done.clear();
do
{
if ((recv = this->recv(_recvBuffer, DEFAULT_RECV_BUFFERSIZE, 0)) > 0)
{
TLOGINFO("[TARS][udp doResponse objname:" << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ",recvbuf:" << recv << "]" << endl);
try
{
_adapterProxy->getObjProxy()->getProxyProtocol().responseFunc(_recvBuffer, recv, done);
}
catch (exception &ex)
{
TLOGERROR("[TARS][udp doResponse, " << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ",desc:" << _ep.desc()
<< ", udp recv decode error:" << ex.what() << endl);
}
catch (...)
{
TLOGERROR("[TARS][udp doResponse, " << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ",desc:" << _ep.desc()
<< ", udp recv decode error." << endl);
}
}
}
while (recv > 0);
return done.empty()?0:1;
}
int UdpTransceiver::send(const void* buf, uint32_t len, uint32_t flag)
{
if(!isValid())
{
return -1;
}
int iRet = ::sendto(_fd, buf, len, flag, (struct sockaddr*) &(_ep.addr()), sizeof(sockaddr));
if (iRet<0)
{
if(errno != EAGAIN)
{
TLOGERROR("[TARS][udp send objname:" << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ",desc:" << _ep.desc()
<< ", fail! errno:" << errno << "," << strerror(errno) << ",close]" << endl);
close();
return iRet;
}
iRet = 0;
}
else if(iRet > 0 && iRet != (int)len)
{
TLOGERROR("[TARS][udp send objname:" << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << "," << _ep.desc() << ", send error."
<< ", len:" << len << ", sendLen:" << iRet << endl);
//udp只发一次 发送一半也算全部发送成功
iRet = len;
}
return iRet;
}
int UdpTransceiver::recv(void* buf, uint32_t len, uint32_t flag)
{
if(!isValid())
{
return -1;
}
int iRet = ::recvfrom(_fd, buf, len, flag, NULL, NULL); //need check from_ip & port
if (iRet < 0 && errno != EAGAIN)
{
TLOGERROR("[TARS][udp recv objname:" << _adapterProxy->getObjProxy()->name()
<< ",fd:" << _fd << ",desc:" << _ep.desc()
<< ", fail! errno:" << errno << "," << strerror(errno) << ",close]" << endl);
close();
return 0;
}
return iRet;
}
/////////////////////////////////////////////////////////////////
}