How to: Horizontally scale subscribers with StatefulSets

学习如何使用StatefulSet进行订阅,并使用一致的消费者ID进行水平扩展

与 Deployment 不同,Pod在Deployments中是短暂的,StatefulSets 允许在Kubernetes上部署有状态的应用程序,为每个Pod保持一个固定的标识。

以下是一个带有 Dapr 的 StatefulSet 示例:

  1. apiVersion: apps/v1
  2. kind: StatefulSet
  3. metadata:
  4. name: python-subscriber
  5. spec:
  6. selector:
  7. matchLabels:
  8. app: python-subscriber # has to match .spec.template.metadata.labels
  9. serviceName: "python-subscriber"
  10. replicas: 3
  11. template:
  12. metadata:
  13. labels:
  14. app: python-subscriber # has to match .spec.selector.matchLabels
  15. annotations:
  16. dapr.io/enabled: "true"
  17. dapr.io/app-id: "python-subscriber"
  18. dapr.io/app-port: "5001"
  19. spec:
  20. containers:
  21. - name: python-subscriber
  22. image: ghcr.io/dapr/samples/pubsub-python-subscriber:latest
  23. ports:
  24. - containerPort: 5001
  25. imagePullPolicy: Always

当通过 Dapr 订阅 pub/sub 主题时,应用程序可以定义 consumerID,该 consumerID 决定了订阅者在队列或主题中的位置。 使用 Pod 的 StatefulSets 粘性标识,你可以拥有一个唯一的 consumerID 每个 Pod,允许订阅者应用程序的每个水平缩放。 Dapr会跟踪每个Pod的名称,在使用{podName}标记时可以用来声明组件。

在扩展给定主题的订阅者数量时,每个 Dapr 组件都有确定其行为的唯一设置。 通常,对于多个消费者,有两个选项:

  • 广播:发布到主题的每条消息都将被所有订阅者消费。
  • 共享:消息被任何订阅者消费(但不是全部)。

Kafka通过consumerID将每个订阅者与主题中的自己位置隔离。 当实例重新启动时,它将重用相同的consumerID,并从其上次已知的位置继续,而不跳过消息。 下面的组件演示了如何让多个 Pod 使用 Kafka 组件:

  1. apiVersion: dapr.io/v1alpha1
  2. kind: Component
  3. metadata:
  4. name: pubsub
  5. spec:
  6. type: pubsub.kafka
  7. version: v1
  8. metadata:
  9. - name: brokers
  10. value: my-cluster-kafka-bootstrap.kafka.svc.cluster.local:9092
  11. - name: consumerID
  12. value: "{podName}"
  13. - name: authRequired
  14. value: "false"

MQTT3协议具有共享主题,允许多个订阅者对主题的消息进行”竞争”,这意味着消息只会被其中一个订阅者处理。 例如:

  1. apiVersion: dapr.io/v1alpha1
  2. kind: Component
  3. metadata:
  4. name: mqtt-pubsub
  5. spec:
  6. type: pubsub.mqtt3
  7. version: v1
  8. metadata:
  9. - name: consumerID
  10. value: "{podName}"
  11. - name: cleanSession
  12. value: "true"
  13. - name: url
  14. value: "tcp://admin:public@localhost:1883"
  15. - name: qos
  16. value: 1
  17. - name: retain
  18. value: "false"

下一步