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文件:

https://blog.csdn.net/Aquester/article/details/48261609

查看回调接口TProcessorEventHandler和TServerEventHandler可了解更多的使用。

C++ Thrift服务端记录调用者IP和被调接口方法的更多相关文章

  1. 用http请求thrift服务端出现了内存溢出的情况

    记一次内存溢出的分析经历 - Janti - 博客园 https://www.cnblogs.com/superfj/p/8474288.html 说在前面的话 朋友,你经历过部署好的服务突然内存溢出 ...

  2. python thrift 服务端与客户端使用

    一.简介 thrift是一个软件框架,用来进行可扩展且跨语言的服务的开发.它结合了功能强大的软件堆栈和代码生成引擎,以构建在 C++, Java, Python, PHP, Ruby, Erlang, ...

  3. 常量,字段,构造方法 调试 ms 源代码 一个C#二维码图片识别的Demo 近期ASP.NET问题汇总及对应的解决办法 c# chart控件柱状图,改变柱子宽度 使用C#创建Windows服务 C#服务端判断客户端socket是否已断开的方法 线程 线程池 Task .NET 单元测试的利剑——模拟框架Moq

    常量,字段,构造方法   常量 1.什么是常量 ​ 常量是值从不变化的符号,在编译之前值就必须确定.编译后,常量值会保存到程序集元数据中.所以,常量必须是编译器识别的基元类型的常量,如:Boolean ...

  4. thrift 服务端linux C ++ 与客户端 windows python 环境配置(thrift 自带tutorial为例)

    关于Thrift文档化的确是做的不好.摸索了很久才终于把跨linux与windows跨C++与python语言的配置成功完成.以下是步骤: 1)                 Linux下环境配置 ...

  5. RPC框架 - thrift 服务端

    -------服务端程序 ------ 下载    下载 thrift 源代码包    下载 thrift 的bin包 准备描述文件(使用源代码包的示例文件)    \thrift-0.10.0\tu ...

  6. 简析服务端通过GT导入SHP至PG的方法

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 项目中需要在浏览器端直接上传SHP后服务端进行数据的自动入PG ...

  7. 简析服务端通过geotools导入SHP至PG的方法

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 项目中需要在浏览器端直接上传SHP后服务端进行数据的自动入PG ...

  8. C#服务端判断客户端socket是否已断开的方法

    刚开始,用Socket类的Connected属性来实现,却发现行不通,connected只表示  是在上次 还是 操作时连接到远程主机.如果在这之后[连接的另一方]断开了,它还一直返回true, 除非 ...

  9. 服务端获得客户端ip

    /** * 获取客户端ip地址(可以穿透代理) * * @param request * @return */ public static String getRemoteAddr(HttpServl ...

随机推荐

  1. Mysql的隔离级别

    一.首先什么是事务? 事务是应用程序中一系列严密的操作,所有操作必须成功完成,否则在每个操作中所作的所有更改都会被撤消.也就是事务具有原子性,一个事务中的一系列的操作要么全部成功,要么一个都不做. 事 ...

  2. Python学习—数据库篇之SQL补充

    一.SQL注入问题 在使用pymysql进行信息查询时,推荐使用传参的方式,禁止使用字符串拼接方式,因为字符串拼接往往会带来sql注入的问题 # -*- coding:utf-8 -*- # auth ...

  3. Rust2018

    Rust已经有2本中文书了 <<深入浅出Rust>> <<Rust编程之道>> 带着aync的rust 2019会更精彩 看async , 在这里htt ...

  4. Java中代理

    普通代理(最简单的代理) 需要有两个实现同一个接口的类,一个是被代理的类,一个是代理类 被代理类中我们按照自己想实现的功能重写接口中的方法 代理类中因为需要代理被代理类,所以在代理类中需要有个被代理类 ...

  5. div轮流滚动显示

    window.onload = function(){ var _box1 = document.getElementById("box1"); var _box2 = docum ...

  6. 贪吃蛇(简易版)Leslie5205912著

    # include <stdio.h># include <string.h># include <windows.h># include <stdlib.h ...

  7. 使用vue-cli@3启动elementui脚手架

    [vue3.x] 准备看elementui的源码,早上拉elementui提供的脚手架代码,于是下载了vue3.x(之前一直用2.x) 1.先把vue2.x卸载了 npm uninstall -g v ...

  8. 思科模拟器GNS3-2.1.8安装笔记 (适用于版本2.0.3以上的GNS3)

    当前现阶段学习经常使用的路由交换设备主要来自于思科.华为和华三三家,这三家的设备操作配置大致类似,却又不尽相同.因为实体设备通常都非常昂贵,所以作为学习,我们通常会使用它们提供的模拟器.华为的模拟器是 ...

  9. VBA实现表单自动编号

    效果:每次提交或刷新标段,表单案指定格式实现自动编号.如当天日期加数字编号的格式 即 2019年2月3日产生的第一张表单产生的编号格式为20190203-001 以此类推第二张表单编号为2019020 ...

  10. 使用react-handsontable

    新建一个项目 create-react-app myProject cd myProject npm install handsontable 或者 npm install handsontable- ...