自定义Flume Sink:ElasticSearch Sink
Flume Sink的目的是从Flume Channel中获取数据然后输出到存储或者其他Flume Source中。Flume Agent启动的时候,它会为每一个Sink都启动一个SinkRunner的对象,SinkRunner.start()方法会启动一个新的线程去管理每一个Sink的生命周期。每一个Sink需要实现start()、Stop()和process()方法。你可以在start方法中去初始化Sink的参数和状态,在stop方法中清理Sink的资源。最关键的是process方法,它将处理从Channel中拿出来的数据。另外如果Sink有一些配置则需要实现Configurable接口。
由于Flume官方提供的Sink往往不能满足要求,所以我们自定义Sink来实现定制化的需求,这里以ElasticSearch为例。在Sink中实现所以文档的简单的Insert功能。例子使用Flume 1.7。
1. 编写代码
首先新建类ElasticSearchSink类继承AbstractSink类,由于还希望有自定义的Sink的配置,所以实现Configurable接口。
public class ElasticSearchSink extends AbstractSink implements Configurable
ElasticSearch的IP以及索引的名称可以配置在配置文件里面,配置文件就是使用flume的conf文件。你可以重写Configurable的configure的方法去获取配置,代码如下:
@Override
public void configure(Context context)
{
esHost = context.getString("es_host");
esIndex = context.getString("es_index");
}
注意里面的配置项“es_host”和“es_index”在conf配置文件中的语法:
agent.sinks = sink1
agent.sinks.sink1.type = nick.test.flume.ElasticSearchSink
agent.sinks.sink1.es_host = 192.168.50.213
agent.sinks.sink1.es_index = vehicle_event_test
接下来就是实现process方法,在这个方法中需要获取channel,因为数据都是从channel中获得的。获取消息之前,需要先获取一个Channel是事务,处理完成之后需要commit和关闭这个事务。这样才能让channel知道这个消息已经消费完成,它可以从它的内部队列中删除这个消息。如果消费失败,需要重新消费的话,可以rollback这个事务。事务的引入是flume对消息可靠性保证的关键。
process方法需要返回一个Status类型的枚举,Ready和BackOff。如果你到了一个消息,并正常处理了,需要使用Ready。如果拿到的消息是null,则可以返回BackOff。所谓BackOff(失效补偿)就是当sink获取不到 消息的时候, Sink的PollingRunner 线程需要等待一段backoff时间,等channel中的数据得到了补偿再来进行pollling 操作。
完整的代码如下:
public class ElasticSearchSink extends AbstractSink implements Configurable
{
private String esHost;
private String esIndex; private TransportClient client; @Override
public Status process() throws EventDeliveryException
{ Status status = null;
// Start transaction
Channel ch = getChannel();
Transaction txn = ch.getTransaction();
txn.begin();
try
{
Event event = ch.take(); if (event != null)
{
String body = new String(event.getBody(), "UTF-8"); BulkRequestBuilder bulkRequest = client.prepareBulk();
List<JSONObject> jsons = new ArrayList<JSONObject>(); JSONObject obj = JSONObject.parseObject(body); String vehicleId = obj.getString("vehicle_id");
String eventBeginCode = obj.getString("event_begin_code");
String eventBeginTime = obj.getString("event_begin_time"); //doc id in index
String id = (vehicleId + "_" + eventBeginTime + "_" + eventBeginCode).trim(); JSONObject json = new JSONObject();
json.put("vehicle_id", vehicleId); bulkRequest.add(client.prepareIndex(esIndex, esIndex).setSource(json)); BulkResponse bulkResponse = bulkRequest.get(); status = Status.READY;
}
else
{
status = Status.BACKOFF;
} txn.commit();
}
catch (Throwable t)
{
txn.rollback();
t.getCause().printStackTrace(); status = Status.BACKOFF;
}
finally
{
txn.close();
} return status; } @Override
public void configure(Context context)
{
esHost = context.getString("es_host");
esIndex = context.getString("es_index");
} @Override
public synchronized void stop()
{
super.stop();
} @Override
public synchronized void start()
{
try
{
Settings settings = Settings.builder().put("cluster.name", "elasticsearch").build();
client = new PreBuiltTransportClient(settings).addTransportAddress(new InetSocketTransportAddress(InetAddress.getByName(esHost), 9300));
super.start(); System.out.println("finish start");
}
catch (Exception ex)
{
ex.printStackTrace();
}
}
}
2. 打包、配置和运行
由于是自定义的Sink,所以需要打成jar包,然后copy到flume的lib文件夹下。然后配置agent的配置文件,最后启动flume就可以了。本例中,我使用了kafkasource、memorychannel和自定义的sink,完整的配置文件如下:
agent.sources = source1
agent.channels = channel1
agent.sinks = sink1 agent.sources.source1.type = org.apache.flume.source.kafka.KafkaSource
agent.sources.source1.channels = channel1
agent.sources.source1.batchSize = 1
agent.sources.source1.batchDurationMillis = 2000
agent.sources.source1.kafka.bootstrap.servers = 192.168.50.116:9092,192.168.50.117:9092,192.168.50.118:9092,192.168.50.226:9092
agent.sources.source1.kafka.topics = iov-vehicle-event
agent.sources.source1.kafka.consumer.group.id = flume-vehicle-event-nick agent.sinks.sink1.type = nick.test.flume.ElasticSearchSink
agent.sinks.sink1.es_host = 192.168.50.213
agent.sinks.sink1.es_index = vehicle_event_test agent.sinks.sink1.channel = channel1 agent.channels.channel1.type = memory
agent.channels.channel1.capacity = 1000

自定义Flume Sink:ElasticSearch Sink的更多相关文章
- 自定义flume的hbase sink 的序列化程序
package com.hello.hbase; import java.nio.charset.Charset; import java.text.SimpleDateFormat; import ...
- Flume NG中的ElasticSearch Sink
ElasticSearch是一个基于Lucene的搜索服务器.它提供了一个分布式多用户能力的全文搜索引擎,基于RESTful web接口.Elasticsearch是用Java开发的,并作为Apach ...
- flink-----实时项目---day07-----1.Flink的checkpoint原理分析 2. 自定义两阶段提交sink(MySQL) 3 将数据写入Hbase(使用幂等性结合at least Once实现精确一次性语义) 4 ProtoBuf
1.Flink中exactly once实现原理分析 生产者从kafka拉取数据以及消费者往kafka写数据都需要保证exactly once.目前flink中支持exactly once的sourc ...
- Flume的Avro Sink和Avro Source研究之二 : Avro Sink
啊,AvroSink要复杂好多:< 好吧,先确定主要问题: AvroSink为啥这么多代码?有必要吗?它都有哪些逻辑需要实现? 你看,avro-rpc-quickstart里是这么建client ...
- Hadoop实战-Flume之Hdfs Sink(十)
a1.sources = r1 a1.sinks = k1 a1.channels = c1 # Describe/configure the source a1.sources.r1.type = ...
- flume 测试 hive sink
测试flume,将数据送到hive表中,首先建表. create table order_flume( order_id string, user_id string, eval_set string ...
- Flume配置Failover Sink Processor
1 官网内容 2 看一张图一目了然 3 详细配置 source配置文件 #配置文件: a1.sources= r1 a1.sinks= k1 k2 a1.channels= c1 #负载平衡 a1.s ...
- Flume的Avro Sink和Avro Source研究之一: Avro Source
问题 : Avro Source提供了怎么样RPC服务,是怎么提供的? 问题 1.1 Flume Source是如何启动一个Netty Server来提供RPC服务. 由GitHub上avro-rpc ...
- 基于Flume+Kafka+ Elasticsearch+Storm的海量日志实时分析平台(转)
0背景介绍 随着机器个数的增加.各种服务.各种组件的扩容.开发人员的递增,日志的运维问题是日渐尖锐.通常,日志都是存储在服务运行的本地机器上,使用脚本来管理,一般非压缩日志保留最近三天,压缩保留最近1 ...
随机推荐
- C# WebApi+Task+WebSocket实战项目演练(四)
一.课程介绍 本次分享课程属于<C#高级编程实战技能开发宝典课程系列>中的第四部分,阿笨后续会计划将实际项目中的一些比较实用的关于C#高级编程的技巧分享出来给大家进行学习,不断的收集.整理 ...
- [Asp.net core]使用ssh命令发布asp.net core项目
命令 # 移除之前发布的包 rm -rf ./.Publish rm -rf ./Wolfy.Blog.tar.gz # 编译并发布 将发布包打包在.Publish目录下 -o "../.P ...
- hadoop ha 读取 activce状态的活动节点
方式一 package com.xxx.hadoop; import com.sun.demo.jvmti.hprof.Tracker; import com.sun.xml.bind.util.Wh ...
- 【Jenkins】新版本的特性:自定义流水线
#!/usr/bin/env groovy pipeline { agent none stages { stage('stage-01') { agent { label 'master' } st ...
- 《闲聊 ASP.NET Core》系列直播清单
[闲聊 ASP.NET Core]第一期:项目与应用结构 [闲聊 ASP.NET Core]第二期:Web Host 初始化与生命周期事件 [闲聊ASP.NET Core]第三期:应用程序配置 [闲聊 ...
- docker 端口映射 及外部无法访问问题
docker容器内提供服务并监听8888端口,要使外部能够访问,需要做端口映射. docker run -it --rm -p : server:v1 此时出现问题,在虚机A上部署后,在A内能够访问8 ...
- Visual Studio 2015 update 3各版本下载地址
微软在06月27日发布了Visual Studio 2015 Update 3 .在MSDN中微软也提供下载,而且MSDN的Visual Studio 2015 Update 3与官方免费下载的文件是 ...
- Android App 安全的HTTPS 通信
漏洞描述 对于数字证书相关概念.Android 里 https 通信代码就不再复述了,直接讲问题.缺少相应的安全校验很容易导致中间人攻击,而漏洞的形式主要有以下3种: 自定义X509TrustMana ...
- 自建证书配置HTTPS服务器
1.写这篇博客的初衷是因为最近iOS9出来了,苹果官方默认要求使用HTTPS,所以自己想整一个HTTPS服务器,也想好好了解一下HTTPS通信,也知道了HTTPS其实就是在HTTP的基础上加上了SSL ...
- linux下 彻底修改python的包/模块导入路径
python模式下,有时候需要导入 import某些模块或者包.明明这个模块/包是存在的,却提示导入错误,比如,“ImportError: No module named lxml”. 但是当你在命 ...