为什么现在不工作了?
广告侦听器(如KAFKA_ADVERTISED_LISTENERS 中定义)是代理在其初始连接中返回给客户端以供其在后续连接中使用的主机和端口。
如果您希望外部客户端使用 9093,那么 KAFKA_ADVERTISED_LISTENERS=…PLAINTEXT_HOST://localhost:9093 是正确的。 然而,你还没有配置你的KAFKA_LISTENERS,如果你在启动时检查代理日志,它将默认为KAFKA_ADVERTISED_LISTENERS设置的值:
kafka-0 | listeners = PLAINTEXT://0.0.0.0:29094,PLAINTEXT_HOST://0.0.0.0:9093
因此,在这种状态下,代理正在侦听端口 9093,但使用此 Docker Compose 指令,您已将外部连接重定向到容器内 9093 到 9092 上的容器:
ports:
- "9093:9092"
➜ docker ps
CONTAINER ID IMAGE … PORTS NAMES
8b934ef4145c confluentinc/cp-kafka:5.4.1 … 0.0.0.0:9093->9092/tcp kafka-0
因此,您的外部连接将转到容器中的 9092 端口,并且代理未在此端口上侦听。您可以通过nc 验证这一点:
-- Port 9093 is open on the host machine
➜ nc -vz localhost 9093
Connection to localhost port 9093 [tcp/*] succeeded!
-- Port 9092 is _not_ open on the Kafka container
➜ docker exec -it kafka-0 nc -vz localhost 9092
localhost [127.0.0.1] 9092 (?) : Connection refused
❌您会看到客户端连接失败
➜ kafkacat -b localhost:9093 -L
% ERROR: Failed to acquire metadata: Local: Broker transport failure
如何解决?
您可以:
- 将侦听器更改为使用 Docker 端口重定向定位的端口。这会起作用,但我个人认为更令人困惑。
- 更改 Docker 端口重定向以定位侦听器所在的端口。这是我会使用的选项,因为它更清晰(例如,端口
9093 始终使用,而不是将9092 和9093 混合在一起)
选项 1:将侦听器更改为使用 Docker 端口重定向的目标端口
version: '3'
services:
kafka-0:
image: confluentinc/cp-kafka:5.4.1
container_name: kafka-0
ports:
- "9093:9092"
environment:
- KAFKA_BROKER_ID=1
- KAFKA_ZOOKEEPER_CONNECT=zookeeper:2181
- KAFKA_LISTENER_SECURITY_PROTOCOL_MAP=PLAINTEXT:PLAINTEXT,PLAINTEXT_HOST:PLAINTEXT
- KAFKA_ADVERTISED_LISTENERS=PLAINTEXT://kafka-0:29094,PLAINTEXT_HOST://localhost:9093
- KAFKA_LISTENERS=PLAINTEXT://0.0.0.0:29094,PLAINTEXT_HOST://0.0.0.0:9092
- KAFKA_OFFSETS_TOPIC_REPLICATION_FACTOR=1
depends_on:
- zookeeper
zookeeper:
image: confluentinc/cp-zookeeper:5.4.1
container_name: zookeeper
ports:
- "2182:2181"
environment:
- ZOOKEEPER_CLIENT_PORT=2181
✅测试:
➜ kafkacat -b localhost:9093 -L
Metadata for all topics (from broker 1: localhost:9093/1):
1 brokers:
broker 1 at localhost:9093 (controller)
选项 2:更改 Docker 端口重定向以针对侦听器所在的端口
version: '3'
services:
kafka-0:
image: confluentinc/cp-kafka:5.4.1
container_name: kafka-0
ports:
- "9093:9093"
environment:
- KAFKA_BROKER_ID=1
- KAFKA_ZOOKEEPER_CONNECT=zookeeper:2181
- KAFKA_LISTENER_SECURITY_PROTOCOL_MAP=PLAINTEXT:PLAINTEXT,PLAINTEXT_HOST:PLAINTEXT
- KAFKA_ADVERTISED_LISTENERS=PLAINTEXT://kafka-0:29094,PLAINTEXT_HOST://localhost:9093
# If you don't specify KAFKA_LISTENERS it will default to the ports used in
# KAFKA_ADVERTISED_LISTENERS, but IMO it's better to be explicit about these settings
- KAFKA_LISTENERS=PLAINTEXT://0.0.0.0:29094,PLAINTEXT_HOST://0.0.0.0:9093
- KAFKA_OFFSETS_TOPIC_REPLICATION_FACTOR=1
depends_on:
- zookeeper
zookeeper:
image: confluentinc/cp-zookeeper:5.4.1
container_name: zookeeper
ports:
- "2182:2181"
environment:
- ZOOKEEPER_CLIENT_PORT=2181
✅测试
➜ kafkacat -b localhost:9093 -L
Metadata for all topics (from broker 1: localhost:9093/1):
1 brokers:
broker 1 at localhost:9093 (controller)
从在 Docker 网络中连接到 Kafka
以上示例是关于从 Docker 主机 连接到 Kafka。如果您想从 Docker 网络中(例如另一个容器)连接到它,您需要使用 kafka-0:29094 作为代理主机和 IP。如果您尝试使用localhost:9093,那么客户端容器会将localhost 解析为它自己的容器,因此会失败。
多个经纪人
See here 示例 Docker Compose 与多个 Kafka 代理。
参考文献
https://rmoff.net/2018/08/02/kafka-listeners-explained/