Apache Pulsar 连接器
Pulsar Source 当前支持 Pulsar 2.8.1 之后的版本,但是 Pulsar Source 使用到了 Pulsar 的事务机制,建议在 Pulsar 2.9.2 及其之后的版本上使用 Pulsar Source 进行数据读取。
如果想要了解更多关于 Pulsar API 兼容性设计,可以阅读文档 。
Copied to clipboard!
Flink 的流连接器并不会放到发行文件里面一同发布,阅读此文档,了解如何将连接器添加到集群实例内。
Pulsar Source 提供了 builder 类来构造 实例。下面的代码实例使用 builder 类创建的实例会从 “persistent://public/default/my-topic” 的数据开始端进行消费。对应的 Pulsar Source 使用了 Exclusive(独占)的订阅方式消费消息,订阅名称为 my-subscription
,并把消息体的二进制字节流以 UTF-8 的方式编码为字符串。
PulsarSource<String> pulsarSource = PulsarSource.builder()
.setServiceUrl(serviceUrl)
.setAdminUrl(adminUrl)
.setStartCursor(StartCursor.earliest())
.setTopics("my-topic")
.setDeserializationSchema(PulsarDeserializationSchema.flinkSchema(new SimpleStringSchema()))
.setSubscriptionName("my-subscription")
.setSubscriptionType(SubscriptionType.Exclusive)
.build();
env.fromSource(source, WatermarkStrategy.noWatermarks(), "Pulsar Source");
如果使用构造类构造 PulsarSource
,一定要提供下面几个属性:
- Pulsar 数据消费的地址,使用
setServiceUrl(String)
方法提供。 - Pulsar HTTP 管理地址,使用
setAdminUrl(String)
方法提供。 - Pulsar 订阅名称,使用
setSubscriptionName(String)
方法提供。 - 需要消费的 Topic 或者是 Topic 下面的分区,详见。
- 解码 Pulsar 消息的反序列化器,详见反序列化器。
指定消费的 Topic 或者 Topic 分区
Pulsar Source 提供了两种订阅 Topic 或 Topic 分区的方式。
Topic 列表,从这个 Topic 的所有分区上消费消息,例如:
PulsarSource.builder().setTopics("some-topic1", "some-topic2");
// 从 topic "topic-a" 的 0 和 1 分区上消费
PulsarSource.builder().setTopics("topic-a-partition-0", "topic-a-partition-2");
Topic 正则,Pulsar Source 使用给定的正则表达式匹配出所有合规的 Topic,例如:
PulsarSource.builder().setTopicPattern("topic-*");
Topic 名称简写
从 Pulsar 2.0 之后,完整的 Topic 名称格式为 {persistent|non-persistent}://租户/命名空间/topic
。但是 Pulsar Source 不需要提供 Topic 名称的完整定义,因为 Topic 类型、租户、命名空间都设置了默认值。
下面的表格提供了当前 Pulsar Topic 支持的简写方式:
Topic 名称简写 | 翻译后的 Topic 名称 |
---|---|
my-topic | persistent://public/default/my-topic |
my-tenant/my-namespace/my-topic | persistent://my-tenant/my-namespace/my-topic |
对于 Non-persistent(非持久化)Topic,Pulsar Source 不支持简写名称。所以无法将
non-persistent://public/default/my-topic
简写成non-persistent://my-topic
。
Pulsar Topic 层次结构
对于 Pulsar 而言,Topic 分区也是一种 Topic。Pulsar 会将一个有分区的 Topic 在内部按照分区的大小拆分成等量的无分区 Topic。
由于 Pulsar 内部的分区实际实现为一个 Topic,我们将用“分区”来指代“仅有一个分区的 Topic(Non-partitioned Topic)”和“具有多个分区的 Topic 下属的分区”。
例如,在 Pulsar 的 sample
租户下面的 flink
命名空间里面创建了一个有 3 个分区的 Topic,给它起名为 simple-string
。可以在 Pulsar 上看到如下的 Topic 列表:
Topic 名称 | 是否分区 |
---|---|
persistent://sample/flink/simple-string | 是 |
persistent://sample/flink/simple-string-partition-0 | 否 |
persistent://sample/flink/simple-string-partition-1 | 否 |
persistent://sample/flink/simple-string-partition-2 | 否 |
这意味着,用户可以用上面的子 Topic 去直接消费分区里面的数据,不需要再去基于上层的父 Topic 去消费全部分区的数据。例如:使用 PulsarSource.builder().setTopics("sample/flink/simple-string-partition-1", "sample/flink/simple-string-partition-2")
将会只消费 Topic sample/flink/simple-string
分区 1 和 2 里面的消息。
配置 Topic 正则表达式
前面提到了 Pulsar Topic 有 persistent
、non-persistent
两种类型,使用正则表达式消费数据的时候,Pulsar Source 会尝试从正则表达式里面解析出消息的类型。例如:PulsarSource.builder().setTopicPattern("non-persistent://my-topic*")
会解析出 non-persistent
这个 Topic 类型。如果用户使用 Topic 名称简写的方式,Pulsar Source 会使用默认的消息类型 persistent
。
如果想用正则去消费 persistent
和 non-persistent
类型的 Topic,需要使用 RegexSubscriptionMode
定义 Topic 类型,例如:setTopicPattern("topic-*", RegexSubscriptionMode.AllTopics)
。
反序列化器
反序列化器用于解析 Pulsar 消息,Pulsar Source 使用 PulsarDeserializationSchema
来定义反序列化器。用户可以在 builder 类中使用 setDeserializationSchema(PulsarDeserializationSchema)
方法配置反序列化器。
如果用户只关心消息体的二进制字节流,并不需要其他属性来解析数据。可以直接使用预定义的 PulsarDeserializationSchema
。Pulsar Source里面提供了 3 种预定义的反序列化器。
使用 Pulsar 的 Schema 解析消息。
// 基础数据类型
PulsarDeserializationSchema.pulsarSchema(Schema);
// 结构类型 (JSON, Protobuf, Avro, etc.)
PulsarDeserializationSchema.pulsarSchema(Schema, Class);
PulsarDeserializationSchema.pulsarSchema(Schema, Class, Class);
使用 Flink 的
DeserializationSchema
解析消息。PulsarDeserializationSchema.flinkSchema(DeserializationSchema);
使用 Flink 的
TypeInformation
解析消息。PulsarDeserializationSchema.flinkTypeInfo(TypeInformation, ExecutionConfig);
Pulsar 的 Message<byte[]>
包含了很多 。例如,消息的 key、消息发送时间、消息生产时间、用户在消息上自定义的键值对属性等。可以使用 Message<byte[]>
接口来获取这些属性。
如果用户需要基于这些额外的属性来解析一条消息,可以实现 PulsarDeserializationSchema
接口。并一定要确保 PulsarDeserializationSchema.getProducedType()
方法返回的 TypeInformation
是正确的结果。Flink 使用 TypeInformation
将解析出来的结果序列化传递到下游算子。
Pulsar 订阅
订阅是命名好的配置规则,指导消息如何投递给消费者。Pulsar Source 需要提供一个独立的订阅名称,支持 Pulsar 的四种订阅模式:
当前 Pulsar Source 里,独占
和 灾备
的实现没有区别,如果 Flink 的一个 reader 挂了,Pulsar Source 会把所有未消费的数据交给其他的 reader 来消费数据。
默认情况下,如果没有指定订阅类型,Pulsar Source 使用共享订阅类型(SubscriptionType.Shared
)。
// 名为 "my-shared" 的共享订阅
PulsarSource.builder().setSubscriptionName("my-shared");
// 名为 "my-exclusive" 的独占订阅
PulsarSource.builder().setSubscriptionName("my-exclusive").setSubscriptionType(SubscriptionType.Exclusive);
如果想在 Pulsar Source 里面使用 key 共享
订阅,需要提供 RangeGenerator
实例。RangeGenerator
会生成一组消息 key 的 hash 范围,Pulsar Source 会基于给定的范围来消费数据。
Pulsar Source 也提供了一个名为 UniformRangeGenerator
的默认实现,它会基于 flink 数据源的并行度将 hash 范围均分。
起始消费位置
Pulsar Source 使用 setStartCursor(StartCursor)
方法给定开始消费的位置。内置的开始消费位置有:
从 Topic 里面最早的一条消息开始消费。
StartCursor.earliest();
从 Topic 里面最新的一条消息开始消费。
从给定的消息开始消费。
StartCursor.fromMessageId(MessageId);
与前者不同的是,给定的消息可以跳过,再进行消费。
StartCursor.fromMessageId(MessageId, boolean);
从给定的消息时间开始消费。
StartCursor.fromMessageTime(long);
每条消息都有一个固定的序列号,这个序列号在 Pulsar 上有序排列,其包含了 ledger、entry、partition 等原始信息,用于在 Pulsar 底层存储上查找到具体的消息。
Pulsar 称这个序列号为
MessageId
,用户可以使用DefaultImplementation.newMessageId(long ledgerId, long entryId, int partitionIndex)
创建它。
边界
Pulsar Source 默认情况下使用流的方式消费数据。除非任务失败或者被取消,否则将持续消费数据。用户可以使用 setBoundedStopCursor(StopCursor)
给定停止消费的位置,这种情况下会使用批的方式进行消费。使用流的方式一样可以给定停止位置,使用 setUnboundedStopCursor(StopCursor)
方法即可。
在批模式下,使用 setBoundedStopCursor(StopCursor)
来指定一个消费停止位置。
内置的停止消费位置如下:
-
StopCursor.never();
停止于 Pulsar 启动时 Topic 里面最新的那条数据。
StopCursor.latest();
停止于某条消息,结果里不包含此消息。
StopCursor.atMessageId(MessageId);
停止于某条消息之后,结果里包含此消息。
StopCursor.afterMessageId(MessageId);
停止于某个给定的消息发布时间戳,比如
Message<byte[]>.getPublishTime()
。StopCursor.atPublishTime(long);
除了前面提到的配置选项,Pulsar Source 还提供了丰富的选项供 Pulsar 专家使用,在 builder 类里通过 setConfig(ConfigOption<T>, T)
和 setConfig(Configuration)
方法给定下述的全部配置。
Pulsar Java 客户端配置项
Pulsar Source 使用 Java 客户端来创建消费实例,相关的配置定义于 Pulsar 的 内。在 PulsarOptions
选项中,定义大部分的可供用户定义的配置。
Pulsar 管理 API 配置项
管理 API 用于查询 Topic 的元数据和用正则订阅的时候的 Topic 查找,它与 Java 客户端共享大部分配置。下面列举的配置只供管理 API 使用,PulsarOptions
包含了这些配置 。
Key | Default | Type | Description |
---|---|---|---|
pulsar.admin.adminUrl | (none) | String | The Pulsar service HTTP URL for the admin endpoint. For example, , or https://my-broker.example.com:8443 for TLS. |
pulsar.admin.autoCertRefreshTime | 300000 | Integer | The auto cert refresh time (in ms) if Pulsar admin supports TLS authentication. |
pulsar.admin.connectTimeout | 60000 | Integer | The connection time out (in ms) for the PulsarAdmin client. |
pulsar.admin.readTimeout | 60000 | Integer | The server response read timeout (in ms) for the PulsarAdmin client for any request. |
pulsar.admin.requestTimeout | 300000 | Integer | The server request timeout (in ms) for the PulsarAdmin client for any request. |
Pulsar 消费者 API 配置项
Pulsar 提供了消费者 API 和读者 API 两套 API 来进行数据消费,它们可用于不同的业务场景。Flink 上的 Pulsar Source 使用消费者 API 进行消费,它的配置定义于 Pulsar 的 ConsumerConfigurationData
内。Pulsar Source 将其中大部分的可供用户定义的配置定义于 PulsarSourceOptions
内。
Key | Default | Type | Description |
---|---|---|---|
pulsar.consumer.ackReceiptEnabled | false | Boolean | Acknowledgement will return a receipt but this does not mean that the message will not be resent after getting the receipt. |
pulsar.consumer.ackTimeoutMillis | 0 | Long | The timeout (in ms) for unacknowledged messages, truncated to the nearest millisecond. The timeout needs to be greater than 1 second. By default, the acknowledge timeout is disabled and that means that messages delivered to a consumer will not be re-delivered unless the consumer crashes. When acknowledgement timeout being enabled, if a message is not acknowledged within the specified timeout it will be re-delivered to the consumer (possibly to a different consumer in case of a shared subscription). |
pulsar.consumer.acknowledgementsGroupTimeMicros | 100000 | Long | Group a consumer acknowledgment for a specified time (in μs). By default, a consumer uses 100μs grouping time to send out acknowledgments to a broker. If the group time is set to 0 , acknowledgments are sent out immediately. A longer ack group time is more efficient at the expense of a slight increase in message re-deliveries after a failure. |
pulsar.consumer.autoAckOldestChunkedMessageOnQueueFull | false | Boolean | Buffering a large number of outstanding uncompleted chunked messages can bring memory pressure and it can be guarded by providing this pulsar.consumer.maxPendingChunkedMessage threshold. Once a consumer reaches this threshold, it drops the outstanding unchunked-messages by silently acknowledging if pulsar.consumer.autoAckOldestChunkedMessageOnQueueFull is true. Otherwise, it marks them for redelivery. |
pulsar.consumer.autoUpdatePartitionsIntervalSeconds | 60 | Integer | The interval (in seconds) of updating partitions. This only works if autoUpdatePartitions is enabled. |
pulsar.consumer.consumerName | (none) | String | The consumer name is informative and it can be used to identify a particular consumer instance from the topic stats. |
pulsar.consumer.cryptoFailureAction | FAIL | Enum | The consumer should take action when it receives a message that can not be decrypted.
Fail to decompress the messages. If messages contain batch messages, a client is not be able to retrieve individual messages in batch. The delivered encrypted message contains EncryptionContext which contains encryption and compression information in. You can use an application to decrypt the consumed message payload.Possible values:
|
pulsar.consumer.deadLetterPolicy.deadLetterTopic | (none) | String | Name of the dead topic where the failed messages are sent. |
pulsar.consumer.deadLetterPolicy.maxRedeliverCount | 0 | Integer | The maximum number of times that a message are redelivered before being sent to the dead letter queue. |
pulsar.consumer.deadLetterPolicy.retryLetterTopic | (none) | String | Name of the retry topic where the failed messages are sent. |
pulsar.consumer.expireTimeOfIncompleteChunkedMessageMillis | 60000 | Long | If a producer fails to publish all the chunks of a message, the consumer can expire incomplete chunks if the consumer cannot receive all chunks in expire times (default 1 hour, in ms). |
pulsar.consumer.maxPendingChunkedMessage | 10 | Integer | The consumer buffers chunk messages into memory until it receives all the chunks of the original message. While consuming chunk-messages, chunks from the same message might not be contiguous in the stream and they might be mixed with other messages’ chunks. So, consumer has to maintain multiple buffers to manage chunks coming from different messages. This mainly happens when multiple publishers are publishing messages on the topic concurrently or publishers failed to publish all chunks of the messages. For example, there are M1-C1, M2-C1, M1-C2, M2-C2 messages.Messages M1-C1 and M1-C2 belong to the M1 original message while M2-C1 and M2-C2 belong to the M2 message. Buffering a large number of outstanding uncompleted chunked messages can bring memory pressure and it can be guarded by providing this pulsar.consumer.maxPendingChunkedMessage threshold. Once, a consumer reaches this threshold, it drops the outstanding unchunked messages by silently acknowledging or asking the broker to redeliver messages later by marking it unacknowledged. This behavior can be controlled by the pulsar.consumer.autoAckOldestChunkedMessageOnQueueFull option. |
pulsar.consumer.maxTotalReceiverQueueSizeAcrossPartitions | 50000 | Integer | The maximum total receiver queue size across partitions. This setting reduces the receiver queue size for individual partitions if the total receiver queue size exceeds this value. |
pulsar.consumer.negativeAckRedeliveryDelayMicros | 60000000 | Long | Delay (in μs) to wait before redelivering messages that failed to be processed. When an application uses Consumer.negativeAcknowledge(Message) , failed messages are redelivered after a fixed timeout. |
pulsar.consumer.poolMessages | false | Boolean | Enable pooling of messages and the underlying data buffers. |
pulsar.consumer.priorityLevel | 0 | Integer | Priority level for a consumer to which a broker gives more priorities while dispatching messages in the shared subscription type. The broker follows descending priorities. For example, 0=max-priority, 1, 2,… In shared subscription mode, the broker first dispatches messages to the consumers on the highest priority level if they have permits. Otherwise, the broker considers consumers on the next priority level. Example 1 If a subscription has consumer A with priorityLevel 0 and consumer B with priorityLevel 1, then the broker only dispatches messages to consumer A until it runs out permits and then starts dispatching messages to consumer B.Example 2 Consumer Priority, Level, Permits C1, 0, 2 C2, 0, 1 C3, 0, 1 C4, 1, 2 C5, 1, 1 The order in which a broker dispatches messages to consumers is: C1, C2, C3, C1, C4, C5, C4. |
pulsar.consumer.properties | Map | A name or value property of this consumer. properties is application defined metadata attached to a consumer. When getting a topic stats, associate this metadata with the consumer stats for easier identification. | |
pulsar.consumer.readCompacted | false | Boolean | If enabling readCompacted , a consumer reads messages from a compacted topic rather than reading a full message backlog of a topic.A consumer only sees the latest value for each key in the compacted topic, up until reaching the point in the topic message when compacting backlog. Beyond that point, send messages as normal. Only enabling readCompacted on subscriptions to persistent topics, which have a single active consumer (like failure or exclusive subscriptions).Attempting to enable it on subscriptions to non-persistent topics or on shared subscriptions leads to a subscription call throwing a PulsarClientException . |
pulsar.consumer.receiverQueueSize | 1000 | Integer | Size of a consumer’s receiver queue. For example, the number of messages accumulated by a consumer before an application calls Receive .A value higher than the default value increases consumer throughput, though at the expense of more memory utilization. |
pulsar.consumer.replicateSubscriptionState | false | Boolean | If replicateSubscriptionState is enabled, a subscription state is replicated to geo-replicated clusters. |
pulsar.consumer.retryEnable | false | Boolean | If enabled, the consumer will automatically retry messages. |
pulsar.consumer.subscriptionInitialPosition | Latest | Enum | Initial position at which to set cursor when subscribing to a topic at first time. Possible values:
|
pulsar.consumer.subscriptionMode | Durable | Enum | Select the subscription mode to be used when subscribing to the topic.
Possible values:
|
pulsar.consumer.subscriptionName | (none) | String | Specify the subscription name for this consumer. This argument is required when constructing the consumer. |
pulsar.consumer.subscriptionType | Shared | Enum | Subscription type. Four subscription types are available:
Possible values:
|
pulsar.consumer.tickDurationMillis | 1000 | Long | Granularity (in ms) of the ack-timeout redelivery. A greater (for example, 1 hour) tickDurationMillis reduces the memory overhead to track messages. |
Pulsar Source配置项
下述配置主要用于性能调优或者是控制消息确认的行为。如非必要,可以不用强制配置。
动态分区发现
为了能在启动 Flink 任务之后还能发现在 Pulsar 上扩容的分区或者是新创建的 Topic,Pulsar Source 提供了动态分区发现机制。该机制不需要重启 Flink 任务。对选项 PulsarSourceOptions.PULSAR_PARTITION_DISCOVERY_INTERVAL_MS
设置一个正整数即可启用。
默认情况下,Pulsar 启用动态分区发现,查询间隔为 30 秒。用户可以给定一个负数,将该功能禁用。如果使用批的方式消费数据,将无法启用该功能。
事件时间和水位线
默认情况下,Pulsar Source 使用 Pulsar 的 Message<byte[]>
里面的时间作为解析结果的时间戳。用户可以使用 WatermarkStrategy
来自行解析出想要的消息时间,并向下游传递对应的水位线。
env.fromSource(pulsarSource, new CustomWatermarkStrategy(), "Pulsar Source With Custom Watermark Strategy");
详细讲解了如何定义 WatermarkStrategy
。
消息确认
一旦在 Topic 上创建了订阅,消息便会在 Pulsar 里。即使没有消费者,消息也不会被丢弃。只有当 Pulsar Source 同 Pulsar 确认此条消息已经被消费,该消息才以某种机制会被移除。Pulsar Source 支持四种订阅方式,它们的消息确认方式也大不相同。
独占和灾备订阅下的消息确认
独占
和 灾备
订阅下,Pulsar Source 使用累进式确认方式。确认某条消息已经被处理时,其前面消息会自动被置为已读。Pulsar Source 会在 Flink 完成检查点时将对应时刻消费的消息置为已读,以此来保证 Pulsar 状态与 Flink 状态一致。
如果用户没有在 Flink 上启用检查点,Pulsar Source 可以使用周期性提交来将消费状态提交给 Pulsar,使用配置 PulsarSourceOptions.PULSAR_AUTO_COMMIT_CURSOR_INTERVAL
来进行定义。
需要注意的是,此种场景下,Pulsar Source 并不依赖于提交到 Pulsar 的状态来做容错。消息确认只是为了能在 Pulsar 端看到对应的消费处理情况。
共享和 key 共享订阅下的消息确认
共享
和 key 共享
需要依次确认每一条消息,所以 Pulsar Source 在 Pulsar 事务里面进行消息确认,然后将事务提交到 Pulsar。
首先需要在 Pulsar 的 borker.conf
文件里面启用事务:
transactionCoordinatorEnabled=true
Pulsar Source 创建的事务的默认超时时间为 3 小时,请确保这个时间大于 Flink 检查点的间隔。用户可以使用 PulsarSourceOptions.PULSAR_TRANSACTION_TIMEOUT_MILLIS
来设置事务的超时时间。
如果用户无法启用 Pulsar 的事务,或者是因为项目禁用了检查点,需要将 PulsarSourceOptions.PULSAR_ENABLE_AUTO_ACKNOWLEDGE_MESSAGE
选项设置为 true
,消息从 Pulsar 消费后会被立刻置为已读。Pulsar Source 无法保证此种场景下的消息一致性。
Pulsar Source 在 Pulsar 上使用日志的形式记录某个事务下的消息确认,为了更好的性能,请缩短 Flink 做检查点的间隔。
Pulsar Sink 连接器可以将经过 Flink 处理后的数据写入一个或多个 Pulsar Topic 或者 Topic 下的某些分区。
Pulsar Sink 基于 Flink 最新的 Sink API 实现。
如果想要使用旧版的使用
SinkFuntion
接口实现的 Sink 连接器,可以使用 StreamNative 维护的 。
使用示例
Pulsar Sink 使用 builder 类来创建 PulsarSink
实例。
下面示例展示了如何通过 Pulsar Sink 以“至少一次”的语义将字符串类型的数据发送给 topic1。
DataStream<String> stream = ...
PulsarSink<String> sink = PulsarSink.builder()
.setAdminUrl(adminUrl)
.setTopics("topic1")
.setSerializationSchema(PulsarSerializationSchema.flinkSchema(new SimpleStringSchema()))
.setDeliverGuarantee(DeliveryGuarantee.AT_LEAST_ONCE)
.build();
stream.sinkTo(sink);
下列为创建一个 PulsarSink
实例必需的属性:
- Pulsar 数据消费的地址,使用
setServiceUrl(String)
方法提供。 - Pulsar HTTP 管理地址,使用
setAdminUrl(String)
方法提供。 - 需要发送到的 Topic 或者是 Topic 下面的分区,详见。
- 编码 Pulsar 消息的序列化器,详见序列化器。
在创建 PulsarSink
时,建议使用 setProducerName(String)
来指定 PulsarSink
内部使用的 Pulsar 生产者名称。这样方便在数据监控页面找到对应的生产者监控指标。
指定写入的 Topic 或者 Topic 分区
PulsarSink
指定写入 Topic 的方式和 Pulsar Source 指定消费的 Topic 或者 Topic 分区的方式类似。PulsarSink
支持以 mixin 风格指定写入的 Topic 或分区。因此,可以指定一组 Topic 或者分区或者是两者都有。
// Topic "some-topic1" 和 "some-topic2"
PulsarSink.builder().setTopics("some-topic1", "some-topic2")
PulsarSink.builder().setTopics("topic-a-partition-0", "topic-a-partition-2")
// Topic "topic-a" 以及 Topic "some-topic2" 分区 0 和 2
PulsarSink.builder().setTopics("topic-a-partition-0", "topic-a-partition-2", "some-topic2")
动态分区发现默认处于开启状态,这意味着 PulsarSink
将会周期性地从 Pulsar 集群中查询 Topic 的元数据来获取可能有的分区数量变更信息。使用 PulsarSinkOptions.PULSAR_TOPIC_METADATA_REFRESH_INTERVAL
配置项来指定查询的间隔时间。
可以选择实现 TopicRouter
接口来自定义。此外,阅读 Topic 名称简写将有助于理解 Pulsar 的分区在 Pulsar 连接器中的配置方式。
序列化器(PulsarSerializationSchema
)负责将 Flink 中的每条记录序列化成 byte 数组,并通过网络发送至指定的写入 Topic。和 Pulsar Source 类似的是,序列化器同时支持使用基于 Flink 的 SerializationSchema
接口实现序列化器和使用 Pulsar 原生的 Schema
类型实现的序列化器。不过序列化器并不支持 Pulsar 的 Schema.AUTO_PRODUCE_BYTES()
。
使用 Pulsar 的 来序列化 Flink 中的数据。
// 原始数据类型
PulsarSerializationSchema.pulsarSchema(Schema)
// 有结构数据类型(JSON、Protobuf、Avro 等)
PulsarSerializationSchema.pulsarSchema(Schema, Class)
// 键值对类型
PulsarSerializationSchema.pulsarSchema(Schema, Class, Class)
使用 Flink 的
SerializationSchema
来序列化数据。PulsarSerializationSchema.flinkSchema(SerializationSchema)
同时使用 PulsarSerializationSchema.pulsarSchema()
以及在 builder 中指定 PulsarSinkBuilder.enableSchemaEvolution()
可以启用 Schema evolution 特性。该特性会使用 Pulsar Broker 端提供的 Schema 版本兼容性检测以及 Schema 版本演进。下列示例展示了如何启用 Schema Evolution。
Schema<SomePojo> schema = Schema.AVRO(SomePojo.class);
PulsarSerializationSchema<SomePojo> pulsarSchema = PulsarSerializationSchema.pulsarSchema(schema, SomePojo.class);
PulsarSink<String> sink = PulsarSink.builder()
...
.setSerializationSchema(pulsarSchema)
.enableSchemaEvolution()
.build();
如果想要使用 Pulsar 原生的 Schema 序列化消息而不需要 Schema Evolution 特性,那么写入的 Topic 会使用
Schema.BYTES
作为消息的 Schema,对应 Topic 的消费者需要自己负责反序列化的工作。例如,如果使用
PulsarSerializationSchema.pulsarSchema(Schema.STRING)
而不使用PulsarSinkBuilder.enableSchemaEvolution()
。那么在写入 Topic 中所记录的消息 Schema 将会是Schema.BYTES
。
消息路由策略
在 Pulsar Sink 中,消息路由发生在于分区之间,而非上层 Topic。对于给定 Topic 的情况,路由算法会首先会查询出 Topic 之上所有的分区信息,并在这些分区上实现消息的路由。Pulsar Sink 默认提供 2 种路由策略的实现。
KeyHashTopicRouter
:使用消息的 key 对应的哈希值来取模计算出消息对应的 Topic 分区。使用此路由可以将具有相同 key 的消息发送至同一个 Topic 分区。消息的 key 可以在自定义
PulsarSerializationSchema
时,在serialize()
方法内使用PulsarMessageBuilder.key(String key)
来予以指定。如果消息没有包含 key,此路由策略将从 Topic 分区中随机选择一个发送。
可以使用
MessageKeyHash.JAVA_HASH
或者MessageKeyHash.MURMUR3_32_HASH
两种不同的哈希算法来计算消息 key 的哈希值。使用PulsarSinkOptions.PULSAR_MESSAGE_KEY_HASH
配置项来指定想要的哈希算法。RoundRobinRouter
:轮换使用用户给定的 Topic 分区。消息将会轮替地选取 Topic 分区,当往某个 Topic 分区里写入指定数量的消息后,将会轮换至下一个 Topic 分区。使用
PulsarSinkOptions.PULSAR_BATCHING_MAX_MESSAGES
指定向一个 Topic 分区中写入的消息数量。
还可以通过实现 TopicRouter
接口来自定义消息路由策略,请注意 TopicRouter 的实现需要能被序列化。
在 TopicRouter
内可以指定任意的 Topic 分区(即使这个 Topic 分区不在 setTopics()
指定的列表中)。因此,当使用自定义的 TopicRouter
时,PulsarSinkBuilder.setTopics
选项是可选的。
@PublicEvolving
public interface TopicRouter<IN> extends Serializable {
String route(IN in, List<String> partitions, PulsarSinkContext context);
default void open(SinkConfiguration sinkConfiguration) {
// 默认无操作
}
}
如前文所述,Pulsar 分区的内部被实现为一个无分区的 Topic,一般情况下 Pulsar 客户端会隐藏这个实现,并且提供内置的消息路由策略。Pulsar Sink 并没有使用 Pulsar 客户端提供的路由策略和封装,而是使用了 Pulsar 客户端更底层的 API 自行实现了消息路由逻辑。这样做的主要目的是能够在属于不同 Topic 的分区之间定义更灵活的消息路由策略。
详情请参考 Pulsar 的 partitioned topics。
发送一致性
PulsarSink
支持三种发送一致性。
NONE
:Flink 应用运行时可能出现数据丢失的情况。在这种模式下,Pulsar Sink 发送消息后并不会检查消息是否发送成功。此模式具有最高的吞吐量,可用于一致性没有要求的场景。AT_LEAST_ONCE
:每条消息至少有一条对应消息发送至 Pulsar,发送至 Pulsar 的消息可能会因为 Flink 应用重启而出现重复。EXACTLY_ONCE
:每条消息有且仅有一条对应消息发送至 Pulsar。发送至 Pulsar 的消息不会有重复也不会丢失。Pulsar Sink 内部依赖 Pulsar 事务和两阶段提交协议来保证每条记录都能正确发往 Pulsar。
消息延时发送
消息延时发送特性可以让指定发送的每一条消息需要延时一段时间后才能被下游的消费者所消费。当延时消息发送特性启用时,Pulsar Sink 会立刻将消息发送至 Pulsar Broker。但该消息在指定的延迟时间到达前将会保持对下游消费者不可见。
消息延时发送仅在 Shared
订阅模式下有效,在 Exclusive
和 Failover
模式下该特性无效。
可以使用 MessageDelayer.fixed(Duration)
创建一个 MessageDelayer
来为所有消息指定恒定的接收时延,或者实现 MessageDelayer
接口来为不同的消息指定不同的接收时延。
Sink 配置项
可以在 builder 类里通过 setConfig(ConfigOption<T>, T)
和 setConfig(Configuration)
方法给定下述的全部配置。
PulsarClient 和 PulsarAdmin 配置项
Pulsar Sink 和 Pulsar Source 公用的配置选项可参考
Pulsar 生产者 API 配置项
Pulsar Sink 使用生产者 API 来发送消息。Pulsar 的 ProducerConfigurationData
中大部分的配置项被映射为 PulsarSinkOptions
里的选项。
Key | Default | Type | Description |
---|---|---|---|
pulsar.producer.batchingEnabled | true | Boolean | Enable batch send ability, it was enabled by default. |
pulsar.producer.batchingMaxBytes | 131072 | Integer | The maximum size of messages permitted in a batch. Keep the maximum consistent as previous versions. |
pulsar.producer.batchingMaxMessages | 1000 | Integer | The maximum number of messages permitted in a batch. |
pulsar.producer.batchingMaxPublishDelayMicros | 1000 | Long | Batching time period of sending messages. |
pulsar.producer.batchingPartitionSwitchFrequencyByPublishDelay | 10 | Integer | The maximum wait time for switching topic partitions. |
pulsar.producer.chunkingEnabled | false | Boolean | |
pulsar.producer.compressionType | NONE | Enum | Message data compression type used by a producer.Available options: Possible values:
|
pulsar.producer.initialSequenceId | (none) | Long | The sequence id for avoiding the duplication, it’s used when Pulsar doesn’t have transaction. |
pulsar.producer.maxPendingMessages | 1000 | Integer | The maximum size of a queue holding pending messages. For example, a message waiting to receive an acknowledgment from a https://pulsar.apache.org/docs/en/reference-terminology#broker. By default, when the queue is full, all calls to the Send and SendAsync methods fail unless you set BlockIfQueueFull to true. |
pulsar.producer.maxPendingMessagesAcrossPartitions | 50000 | Integer | The maximum number of pending messages across partitions. Use the setting to lower the max pending messages for each partition ( setMaxPendingMessages ) if the total number exceeds the configured value. |
pulsar.producer.producerName | (none) | String | A producer name which would be displayed in the Pulsar’s dashboard. If no producer name was provided, we would use a Pulsar generated name instead. |
pulsar.producer.properties | Map | A name or value property of this consumer. properties is application defined metadata attached to a consumer. When getting a topic stats, associate this metadata with the consumer stats for easier identification. | |
pulsar.producer.sendTimeoutMs | 30000 | Long | Message send timeout in ms.If a message is not acknowledged by a server before the sendTimeout expires, an error occurs. |
Pulsar Sink 配置项
下述配置主要用于性能调优或者是控制消息确认的行为。如非必要,可以不用考虑配置。
Key | Default | Type | Description |
---|---|---|---|
pulsar.sink.deliveryGuarantee | none | Enum | Optional delivery guarantee when committing. Possible values:
|
pulsar.sink.enableSchemaEvolution | false | Boolean | If you enable this option and use PulsarSerializationSchema.pulsarSchema(), we would consume and deserialize the message by using Pulsar’s Schema . |
pulsar.sink.maxRecommitTimes | 5 | Integer | The allowed transaction recommit times if we meet some retryable exception. This is used in Pulsar Transaction. |
pulsar.sink.messageKeyHash | murmur-3-32-hash | Enum | The hash policy for routing message by calculating the hash code of message key. Possible values:
|
pulsar.sink.topicMetadataRefreshInterval | 1800000 | Long | Auto update the topic metadata in a fixed interval (in ms). The default value is 30 minutes. |
pulsar.sink.transactionTimeoutMillis | 10800000 | Long | This option is used when the user require the DeliveryGuarantee.EXACTLY_ONCE semantic.We would use transaction for making sure the message could be write only once. |
Sink 监控指标
下列表格列出了当前 Sink 支持的监控指标,前 6 个指标是 FLIP-33: Standardize Connector Metrics 中规定的 Sink 连接器应当支持的标准指标。
指标
numBytesOut
、numRecordsOut
和numRecordsOutErrors
从 Pulsar Producer 实例的监控指标中获得。
currentSendTime
记录了最近一条消息从放入生产者的缓冲队列到消息被消费确认所耗费的时间。这项指标在NONE
发送一致性下不可用。
默认情况下,Pulsar 生产者每隔 60 秒才会刷新一次监控数据,然而 Pulsar Sink 每 500 毫秒就会从 Pulsar 生产者中获得最新的监控数据。因此 numRecordsOut
、numBytesOut
、numAcksReceived
以及 numRecordsOutErrors
4 个指标实际上每 60 秒才会刷新一次。
如果想要更高地刷新评率,可以通过 builder.setConfig(PulsarOptions.PULSAR_STATS_INTERVAL_SECONDS. 1L)
来将 Pulsar 生产者的监控数据刷新频率调整至相应值(最低为1s)。
numBytesOutRate
和 numRecordsOutRate
指标是 Flink 内部通过 numBytesOut
和 计数器,在一个 60 秒的窗口内计算得到的。
Pulsar Sink 遵循 中定义的 Sink API 设计。
无状态的 SinkWriter
在 EXACTLY_ONCE
一致性下,Pulsar Sink 不会将事务相关的信息存放于检查点快照中。这意味着当 Flink 应用重启时,Pulsar Sink 会创建新的事务实例。上一次运行过程中任何未提交事务中的消息会因为超时中止而无法被下游的消费者所消费。这样的设计保证了 SinkWriter 是无状态的。
Pulsar Schema Evolution
Pulsar Schema Evolution 允许用户在一个 Flink 应用程序中使用的数据模型发生特定改变后(比如向基于 ARVO 的 POJO 类中增加或删除一个字段),仍能使用同一个 Flink 应用程序的代码。
可以在 Pulsar 集群内指定哪些类型的数据模型的改变是被允许的,详情请参阅 。
常见的升级步骤,请参阅升级应用程序和 Flink 版本。Pulsar 连接器没有在 Flink 端存储消费的状态,所有的消费信息都推送到了 Pulsar。所以需要注意下面的事项:
- 不要同时升级 Pulsar 连接器和 Pulsar 服务端的版本。
使用 Flink 和 Pulsar 交互时如果遇到问题,由于 Flink 内部实现只是基于 Pulsar 的 和管理 API 而开发的。
用户遇到的问题可能与 Flink 无关,请先升级 Pulsar 的版本、Pulsar 客户端的版本,或者修改 Pulsar 的配置、Pulsar 连接器的配置来尝试解决问题。
在 Java 11 上使用不稳定
Pulsar connector 在 Java 11 中有一些尚未修复的问题。我们当前推荐在 Java 8 环境中运行Pulsar connector.