C++ Thrift服务端记录调用者IP和被调接口方法
Apache开源的Thrift(http://thrift.apache.org)有着广泛的使用,有时候需要知道谁调用了指定的函数,比如在下线一起老的接口之前,需要确保对这些老接口的访问已全部迁移到新口。Thrift提供了支持,在《Thrift结构分析及增加取客户端IP功能实现》一文中已做过介绍,但不够具体。
本文对这个做一个详细的介绍,过程中使用到了开源的C++ Thrift服务端的辅助类CThriftServerHelper(对应的客户端辅助类为CThriftClientHelper),源代码网址为:
|
https://github.com/eyjian/libmooon/blob/master/include/mooon/net/thrift_helper.h |
为达到目标,需要提供一个Context结构体和两个回调接口实现类。
1) Contex结构体ThriftServerContext
结构体的内容完成自定义,这里定义一个peer成员用来保存客户端的IP和端口号,根据实际需要也可分成两个字段。
|
struct ThriftServerContext { std::string peer; // 客户端的IP和端口号,格式为标准的“IP:PORT” }; |
2) ServerEvent回调接口实现类MyServerEventHandler
|
class MyServerEventHandler: public apache::thrift::server::TServerEventHandler { private: virtual void* createContext( boost::shared_ptr<apache::thrift::server::TProtocol> input, boost::shared_ptr<apache::thrift::server::TProtocol> output); virtual void deleteContext( void* serverContext, boost::shared_ptr<apache::thrift::server::TProtocol>input, boost::shared_ptr<apache::thrift::server::TProtocol>output); virtual void processContext(void* serverContext, boost::shared_ptr<apache::thrift::server::TTransport> transport); }; |
3) ProcessorEvent回调接口实现类MyProcessorEventHandler
|
class MyProcessorEventHandler: public apache::thrift::TProcessorEventHandler { private: virtual void* getContext(const char* fn_name, void* serverContext); }; |
4) 相关的实现
|
void* MyServerEventHandler::createContext( boost::shared_ptr<apache::thrift::server::TProtocol> input, boost::shared_ptr<apache::thrift::server::TProtocol> output) { // 以下针对TNonblockingServer // in_transport和out_transport实际为apache::thrift::server::TMemoryBuffer //boost::shared_ptr<apache::thrift::server::TTransport> in_transport = input->getTransport(); //boost::shared_ptr<apache::thrift::server::TTransport> output_transport = output->getTransport(); return new ThriftServerContext; // 创建Context,Context每一个客户端连接是一对一的关系 } void MyServerEventHandler::deleteContext( void* serverContext, boost::shared_ptr<apache::thrift::server::TProtocol>input, boost::shared_ptr<apache::thrift::server::TProtocol>output) { delete (ThriftServerContext*)serverContext; // 释放Context,否则内存泄漏,连接被关闭时调用 } void MyServerEventHandler::processContext( void* serverContext, boost::shared_ptr<apache::thrift::server::TTransport> transport) { #if 1 // 如果是TNonblockingServer,则TTransport::getOrigin返回的是IP地址 //MYLOG_DEBUG("Called from %s\n", transport->getOrigin().c_str()); ThriftServerContext* ctx = (ThriftServerContext*)serverContext; // 保存客户端的IP和端口号,以便MyProcessorEventHandler::getContext中可用 ctx->peer = transport->getOrigin(); #else // 以下针对TNonblockingServer有效 apache::thrift::server::TSocket* socket = dynamic_cast<apache::thrift::server::TSocket*>(transport.get()); if (socket != NULL) { // TSocket::getPeerAddress返回的是IP地址, // 如果调用TSocket::getPeerHost(),则返回的可能是IP对应的hostname MYLOG_DEBUG("Called from %s:%d\n", socket->getPeerAddress().c_str(), socket->getPeerPort()); } #endif } // 参数fn_name为被调用接口名 // serverContext承载了客户端的IP和端口号数据 // // 在getContext中,还可为每个调用创建自己的Context,但注意区别Server的Context void* MyProcessorEventHandler::getContext(const char* fn_name, void* serverContext) { ThriftServerContext* ctx = (ThriftServerContext*)serverContext; MYLOG_INFO("%s called by %s\n", fn_name, ctx->peer.c_str()); return NULL; // 如果为本次调用创建Context,则需要实现freeContext以释放Context } |
5) 应用示例
|
class CMyServer { public: CMyServer(); void start(); private: boost::shared_ptr<MyServerEventHandler> _server_event_handler; boost::shared_ptr<MyProcessorEventHandler> _processor_event_handler; mooon::net::CThriftServerHelper<CMyHandler, MyServiceProcessor> _thrift_server; }; CMyServer::CMyServer() : _server_event_handler(new MyServerEventHandler), _processor_event_handler(new MyProcessorEventHandler), _thrift_server(_server_event_handler, _processor_event_handler) { } void CMyServer::start() { // 启动Thrift服务, // 注意调用线程在这里会阻塞, // 直到调用_thrift_server.stop()停止Thrift服务。 _thrift_server.serve( mooon::argument::port->value(), // 监听端口号 mooon::argument::wkthreads->value(), // 工作线程数 mooon::argument::iothreads->value()); // IO线程数 } |
有关细节请参见《Thrift结构分析及增加取客户端IP功能实现》,以及编译thrift文件后生成的Service.cpp文件:
查看回调接口TProcessorEventHandler和TServerEventHandler可了解更多的使用。
C++ Thrift服务端记录调用者IP和被调接口方法的更多相关文章
- 用http请求thrift服务端出现了内存溢出的情况
记一次内存溢出的分析经历 - Janti - 博客园 https://www.cnblogs.com/superfj/p/8474288.html 说在前面的话 朋友,你经历过部署好的服务突然内存溢出 ...
- python thrift 服务端与客户端使用
一.简介 thrift是一个软件框架,用来进行可扩展且跨语言的服务的开发.它结合了功能强大的软件堆栈和代码生成引擎,以构建在 C++, Java, Python, PHP, Ruby, Erlang, ...
- 常量,字段,构造方法 调试 ms 源代码 一个C#二维码图片识别的Demo 近期ASP.NET问题汇总及对应的解决办法 c# chart控件柱状图,改变柱子宽度 使用C#创建Windows服务 C#服务端判断客户端socket是否已断开的方法 线程 线程池 Task .NET 单元测试的利剑——模拟框架Moq
常量,字段,构造方法 常量 1.什么是常量 常量是值从不变化的符号,在编译之前值就必须确定.编译后,常量值会保存到程序集元数据中.所以,常量必须是编译器识别的基元类型的常量,如:Boolean ...
- thrift 服务端linux C ++ 与客户端 windows python 环境配置(thrift 自带tutorial为例)
关于Thrift文档化的确是做的不好.摸索了很久才终于把跨linux与windows跨C++与python语言的配置成功完成.以下是步骤: 1) Linux下环境配置 ...
- RPC框架 - thrift 服务端
-------服务端程序 ------ 下载 下载 thrift 源代码包 下载 thrift 的bin包 准备描述文件(使用源代码包的示例文件) \thrift-0.10.0\tu ...
- 简析服务端通过GT导入SHP至PG的方法
文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 项目中需要在浏览器端直接上传SHP后服务端进行数据的自动入PG ...
- 简析服务端通过geotools导入SHP至PG的方法
文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 项目中需要在浏览器端直接上传SHP后服务端进行数据的自动入PG ...
- C#服务端判断客户端socket是否已断开的方法
刚开始,用Socket类的Connected属性来实现,却发现行不通,connected只表示 是在上次 还是 操作时连接到远程主机.如果在这之后[连接的另一方]断开了,它还一直返回true, 除非 ...
- 服务端获得客户端ip
/** * 获取客户端ip地址(可以穿透代理) * * @param request * @return */ public static String getRemoteAddr(HttpServl ...
随机推荐
- 【SVN】关于提交代码时的问题
将项目上传到SVN弹出框提示One added/edited TODO item was found. Would you like to review it? 然后百度说这样解决: 造成这个原因是因 ...
- MySQL如何解决1209错误
1209 - The MySQL server is running with the--read-only option so it cannot execute this sta ...
- vue踩坑(二):跨域以及携带cookie
最近后台需求要在请求的时候传cooki给后台,正常情况下拿到cookie后存在cookie里,同域名下是会自己带到请求头里的,但是因为要在本地调试,那么问题就来了,localhost:8080下面的c ...
- Huawei BGP和OSPF双边界重分布(一)
网络拓扑: PS:本例使用明细前缀列表双边界引入 S5700-LSW1 ================================================================ ...
- Educational Codeforces Round 35 (Rated for Div. 2)
Educational Codeforces Round 35 (Rated for Div. 2) https://codeforces.com/contest/911 A 模拟 #include& ...
- iOS多图上传
iOS多图上传涉及到多线程问题,个人比较喜欢使用GCD操作,下边是最近写的一个多图上传代码,附带相关注释 __block BOOL allSucc = YES; __block int m = 0; ...
- Eclipse oxygen 版本汉化教程
Eclipse oxygen 版本汉化步骤如下: 第一步:打开Eclipse 第二步:浏览器打开网址 http://www.eclipse.org/babel/downloads.php 1.复制对应 ...
- MVC开发T4代码生成之一----文本模板基础
T4文本模板 T4全写为Text Template Transformation Toolkit,是一种编程辅助工具,用来使程序代码自(懒)动(猿)生(福)成(利)的工具.MVC开发中大量使用了T4模 ...
- VSCode的使用
前后端分离的,先打开vs,打开你的项目,在项目根目录中找到.vs文件加,找到.vs\config\applicationhost.config,然后打开找到你项目的IIS Express配置,例如:& ...
- LCMapString/LCMapStringEx实现简体字、繁体字的转换。
c#环境下想要最小程度不使用第三方库.程序性能,于是选择了这个Windows API. 转载自https://coolong124220.nidbox.com/diary/read/8045380 对 ...