Pulsar-Function

参考链接

笔记

  • 概述
    轻量级的计算方法,依赖于Pulsar消息系统,处理逻辑如下:
1. 消费一个或多个Topic
2. 每个消息都执行用户提供的处理逻辑
3. 发布计算后的结果到另一个Topic

目标:

1. 开发人员开发效率
2. 排查故障方便
3. 操作简单(无需外部系统)

灵感:

1. 流式处理引擎(Apache Storm,Apache Heron,Apache Flink)
2. "Serverless", "Function as a Service"(Amazon Web Services Lambda,Google Cloud Functions,Azure Cloud Functions)
Function可如下描述:
a. Lambda样式的函数
b. 使用Pulsar为消息总线而设计

程序设计模型:

提供了广泛的功能,核心编程模型简单.
Functions可以从一个或多个输入Topic中获取数据,并且对于每个收到的消息可以执行如下任务:
1. 对输入消息做逻辑处理,并输出数据至指定的Topic或Bookkeeper
2. 往日志Topic写入处理日志(用于调试)
3. 递增计数器
可以将多个Function组合起来,实现一个消息处理链,如:
输入Message -> 关键字过滤白名单Function->白名单结果Topic->消息统计Function->统计结果Topic->结果入库Function

Functions处理消息类型:

Pulsar Functions默认是以字节数组的形式来接收消息及发送消息;也可以绑定消息的类型(有两种方式)
1. Schema Registry
  pulsar内置的schema注册中心
2. SerDe
自定义Serialization,Deserialization函数(必须跟function打在同一个包里),并在发布Function时指定处理类,使用参数--output-serde-classname,如: --output-serde-classname com.example.serde.TweetSerde 

FQFN(完全限定函数名)

包含三元素:function tenant, namespace, function name
故相同的函数名可以部署在不同的命名空间下。

支持的开发语言

Java
Python
Go

消息处理三种模式保证

At-most-once delivery: 最多一次投递,发送到函数的每个消息都可能被处理,或者不被处理;处理过程失败后将没有重试机制;
At-least-once delivery:最少一次投递,发送到函数的每个消息都可以被多次处理,Function默认的模式;处理过程失败后消息将会被重新投递处理,故存在消息被多次投递、处理的情况;
Effectively-once delivery:只有一次有效投递,发送到函数的每条消息都将有一个与之关联的输出;

Exactly-once delivery:只投递一次
基本实现的方式有两种:

  1. 分布式快照\状态检查点 (Chandy-Lamport分布式快照算法)
    定期检查,出现失败情况,回滚至最近的一次全局一致性快照/检查点,重新消费处理对应offset的消息
    优点: 较少的性能和资源开销
    缺点:故障中恢复时对性能的影响更大;计算网络拓扑越大,对性能的影响越大

  2. 基于“最少投递一次”增加消息去重插件
    消息处理前,先进行去重处理;对于失败的消息直接重新投递,若有计算节点已经处理过,则不处理该消息;需要更多的资源,尤其是存储。
    优点:故障对性能的影响是局部的;计算网络拓扑增大,并不会提高故障对性能的影响;
    缺点:需要大量存储和基础架构来支持;每个事件/消息都需要额外的性能开销;

Pulsar Broker消息去重:

Broker的重复数据消除逻辑基于 record-keeping系统。每个Broker都会跟踪每个消息生产者最后一个发布“successfully”的消息ID。这些信息存储于内存(以producer -> last sequence ID形式),Pulsar会定期存储快照,并持久化同时复制多份。快照与快照消息日志相结合,可确保Broker崩溃后正确状态的重建。
同时要实现Exactly-once delivery,应用也需要做相应支持,简单样例:
Producer producer = client.createProducer(TOPIC_NAME, conf);
此方式创建的producer,系统将为其分配一个唯一的名称,并且发布消息的序列ID是从0开始;;若应用重启则名称将会变化,故需要做下改进:
设置producer名称,并从最后的序列ID开始发送消息;
并且应用系统需要将发送的消息和序列ID关联起来;
消费端需要将处理结果同序列ID关联起来(如写入DB),故障恢复时可以通过Reader API读取指定ID的消息;
  • Functions工作原理
    1.) 运行模式
    【functions】
    线程:function以线程方式运行于Functions worker,使用同一个JVM;只支持java function。
    进程:由Functions worker fork出新进程运行function,在相同的宿主机上;支持Java, Python, and Go functions。
    Kubernetes:以docker容器化方式运行functions。
    【Functions worker】
    模式一: 运行于Broker中,可以理解为Broker的一部分;简单方便;
    模式二:独立的Functions worker集群模式运行;更好的资源隔离;
    2.) functions开发
    提供接口类型:
    Language-native interface:无需pulsar指定的库,故也没有对应的context;
import java.util.function.Function;
public class JavaNativeExclamationFunction implements Function<String, String> {
   @Override
   public String apply(String input) {
       return String.format("%s!", input);
   }
}

Pulsar Function SDK:引入pulsar指定的库,实现的函数带有对应的context;

import org.apache.pulsar.functions.api.Context;
import org.apache.pulsar.functions.api.Function;

public class ExclamationFunction implements Function<String, String> {
   @Override
   public String process(String input, Context context) {
       return String.format("%s!", input);
   }
}

消息序列化:
a.) Pulsar内置的schema注册中心
b.) 自定义的序列化、反序列化函数(SerDe)
注意:需要将SerDe打包到function的包中
3.) 调试
a.) unit Test

@Test
public void testJavaNativeExclamationFunction() {
   JavaNativeExclamationFunction exclamation = new JavaNativeExclamationFunction();
   String output = exclamation.apply("foo");
   Assert.assertEquals(output, "foo!");
}
@Test
public void testExclamationFunction() {
   ExclamationFunction exclamation = new ExclamationFunction();
   String output = exclamation.process("foo", mock(Context.class));
   Assert.assertEquals(output, "foo!");
}

b.) localrun mode

public class ExclamationFunction implements Function<String, String> {

   @Override
   public String process(String s, Context context) throws Exception {
       return s + "!";
   }

public static void main(String[] args) throws Exception {
    FunctionConfig functionConfig = new FunctionConfig();
    functionConfig.setName("exclamation");
    functionConfig.setInputs(Collections.singleton("input"));   functionConfig.setClassName(ExclamationFunction.class.getName());
    functionConfig.setRuntime(FunctionConfig.Runtime.JAVA);
    functionConfig.setOutput("output");
    LocalRunner localRunner = LocalRunner.builder().functionConfig(functionConfig).build();
    localRunner.start(false);
}

4.) 部署
使用pulsar-admin functions命令,主要参数如下:

Parameter Default 备注
classname The function's class name
jar Path to the jar file for the function
name The function's name
tenant The function's tenant
namespace The function's namespace
output The function's output topic
inputs The function's input topic or topics
subs-name Pulsar source subscription name
processing-guarantees The processing guarantees (aka delivery semantics) applied to the function [ATLEAST_ONCE, ATMOST_ONCE, EFFECTIVELY_ONCE
parallelism The function's parallelism factor
cpu The cpu in cores that need to be allocated per function instance
log-topic The topic to which the function's logs are produced
schema-type The builtin schema type or custom schema class name to be used for messages output by the function

特别说明
【Subscription type】
For at-least-once and at-most-once [processing guarantees], the [SHARED]mode is applied by default; for effectively-once guarantees, the [FAILOVER] mode is applied.

支持本地运行(localrun,运行于连接的broker)和集群运行(create,伴随broker集群运行)
并行度设置(Parallelism):
localrun,多次调用命令,命令亦提供参数项parallelism
create,设置参数项parallelism

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容