f8b21af4be
10_Wiki/Topics 대규모 정리: - 오류 캡처/미완성 stub 문서 227개 제거 - 교차폴더 중복 43클러스터 병합 (63파일 → redirect) - 링크명 정규화: 깨진 링크 수정·redirect 직결·개념 매핑 ~2,400건 - 카테고리 MOC 6개 신규 생성 - Graph 섹션 미해결 related-keyword 링크 10,058건 제거 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
203 lines
5.8 KiB
Markdown
203 lines
5.8 KiB
Markdown
---
|
|
id: wiki-2026-0508-broker-topology
|
|
title: Broker Topology
|
|
category: 10_Wiki/Topics
|
|
status: verified
|
|
canonical_id: self
|
|
aliases: [Event Broker Topology, Pub-Sub Topology]
|
|
duplicate_of: none
|
|
source_trust_level: A
|
|
confidence_score: 0.9
|
|
verification_status: applied
|
|
tags: [event-driven, architecture, kafka, messaging]
|
|
raw_sources: []
|
|
last_reinforced: 2026-05-10
|
|
github_commit: pending
|
|
tech_stack:
|
|
language: typescript
|
|
framework: kafka-nats-rabbitmq
|
|
---
|
|
|
|
# Broker Topology
|
|
|
|
## 매 한 줄
|
|
> **"매 distributed pub-sub mesh — 매 central mediator 없이, broker (queue/topic) 가 routing fabric."**. Broker topology는 Event-Driven Architecture 의 두 종류 중 하나 (vs Mediator topology). 매 single-purpose events 의 fan-out 에 최적. 2026년 Kafka, NATS JetStream, AWS EventBridge, Redpanda, Pulsar 가 dominant brokers.
|
|
|
|
## 매 핵심
|
|
|
|
### 매 Broker vs Mediator
|
|
| 측면 | Broker | Mediator |
|
|
|---|---|---|
|
|
| Coordination | None — events fan out | Central mediator orchestrates |
|
|
| Coupling | Loosest | Some (mediator knows steps) |
|
|
| Use case | Simple fan-out, async notify | Complex multi-step workflow |
|
|
| Failure recovery | Each consumer handles | Mediator retries/compensates |
|
|
| Examples | Kafka topics, NATS subjects | Apache Camel, Step Functions |
|
|
|
|
### 매 컴포넌트
|
|
- **Producer**: events 의 publish.
|
|
- **Broker**: topic/subject/queue store + routing.
|
|
- **Consumer**: subscribe + react.
|
|
- **Schema Registry** (Confluent, Apicurio): event contract 의 versioning.
|
|
- **Dead Letter Queue (DLQ)**: failed messages.
|
|
|
|
### 매 broker 종류
|
|
- **Log-based** (Kafka, Redpanda, Pulsar): replay 가능, partitioned, ordered per partition.
|
|
- **Queue-based** (RabbitMQ, SQS): consumed once, no replay.
|
|
- **Subject-based** (NATS): lightweight, JetStream 으로 persistence.
|
|
- **Cloud-native** (EventBridge, Pub/Sub, EventHubs).
|
|
|
|
### 매 응용
|
|
1. Order events (e-commerce fan-out).
|
|
2. CDC (Debezium → Kafka → multiple sinks).
|
|
3. IoT telemetry ingestion.
|
|
4. Analytics event pipeline.
|
|
|
|
## 💻 패턴
|
|
|
|
### Kafka producer (TypeScript / KafkaJS)
|
|
```typescript
|
|
import { Kafka } from 'kafkajs';
|
|
|
|
const kafka = new Kafka({
|
|
clientId: 'orders',
|
|
brokers: ['kafka-1:9092', 'kafka-2:9092'],
|
|
});
|
|
const producer = kafka.producer();
|
|
|
|
await producer.connect();
|
|
await producer.send({
|
|
topic: 'order.created',
|
|
messages: [{
|
|
key: order.id,
|
|
value: JSON.stringify(order),
|
|
headers: { 'content-type': 'application/json' },
|
|
}],
|
|
});
|
|
```
|
|
|
|
### Kafka consumer (consumer group fan-out)
|
|
```typescript
|
|
const consumer = kafka.consumer({ groupId: 'shipping-svc' });
|
|
await consumer.connect();
|
|
await consumer.subscribe({ topic: 'order.created' });
|
|
|
|
await consumer.run({
|
|
eachMessage: async ({ message }) => {
|
|
const order = JSON.parse(message.value!.toString());
|
|
await createShipment(order);
|
|
},
|
|
});
|
|
```
|
|
|
|
### NATS JetStream
|
|
```typescript
|
|
import { connect, JSONCodec } from 'nats';
|
|
|
|
const nc = await connect({ servers: 'nats://localhost:4222' });
|
|
const js = nc.jetstream();
|
|
const jc = JSONCodec();
|
|
|
|
await js.publish('orders.created', jc.encode(order));
|
|
|
|
const sub = await js.subscribe('orders.>', {
|
|
config: { durable_name: 'shipping' },
|
|
});
|
|
for await (const m of sub) {
|
|
const order = jc.decode(m.data);
|
|
await createShipment(order);
|
|
m.ack();
|
|
}
|
|
```
|
|
|
|
### Schema Registry (Avro + Confluent)
|
|
```typescript
|
|
import { SchemaRegistry } from '@kafkajs/confluent-schema-registry';
|
|
|
|
const registry = new SchemaRegistry({ host: 'http://schema-registry:8081' });
|
|
const schema = `{
|
|
"type": "record",
|
|
"name": "OrderCreated",
|
|
"fields": [
|
|
{ "name": "id", "type": "string" },
|
|
{ "name": "amount", "type": "double" }
|
|
]
|
|
}`;
|
|
|
|
const { id } = await registry.register({ type: 'AVRO', schema });
|
|
const encoded = await registry.encode(id, order);
|
|
await producer.send({ topic: 'order.created', messages: [{ value: encoded }] });
|
|
```
|
|
|
|
### DLQ pattern
|
|
```typescript
|
|
await consumer.run({
|
|
eachMessage: async ({ topic, message }) => {
|
|
try {
|
|
await handleOrder(JSON.parse(message.value!.toString()));
|
|
} catch (err) {
|
|
await producer.send({
|
|
topic: `${topic}.dlq`,
|
|
messages: [{
|
|
value: message.value,
|
|
headers: {
|
|
...message.headers,
|
|
error: String(err),
|
|
retryCount: '0',
|
|
},
|
|
}],
|
|
});
|
|
}
|
|
},
|
|
});
|
|
```
|
|
|
|
### Partitioning for ordering
|
|
```typescript
|
|
// Same orderId always goes to same partition → ordered per order
|
|
await producer.send({
|
|
topic: 'order.events',
|
|
messages: [{
|
|
key: order.id, // partitioner uses key hash
|
|
value: JSON.stringify(event),
|
|
}],
|
|
});
|
|
```
|
|
|
|
## 매 결정 기준
|
|
| 상황 | Approach |
|
|
|---|---|
|
|
| High throughput, replay 필요 | Kafka / Redpanda |
|
|
| Lightweight, low-latency | NATS JetStream |
|
|
| Cloud serverless | EventBridge, Pub/Sub |
|
|
| Strong work-queue + ack | RabbitMQ, SQS |
|
|
| Multi-tenant, geo | Pulsar |
|
|
|
|
**기본값**: Kafka (or Redpanda — Kafka API compat) + Schema Registry + Avro/Protobuf for events.
|
|
|
|
## 🔗 Graph
|
|
- 부모: [[Event-Driven Architecture]]
|
|
- 변형: [[Mediator Topology]] · [[Choreography]]
|
|
- 응용: [[Kafka]] · [[NATS]]
|
|
- Adjacent: [[CDC]]
|
|
|
|
## 🤖 LLM 활용
|
|
**언제**: fan-out events, decoupled microservices, CDC pipelines, telemetry ingestion.
|
|
**언제 X**: synchronous request-reply (use gRPC/HTTP), small monolith (events 가 overhead).
|
|
|
|
## ❌ 안티패턴
|
|
- **No schema registry**: 모든 producer/consumer 가 schema drift — production 깨짐.
|
|
- **Topic per service**: scalability impossible — topic per event type.
|
|
- **No DLQ**: poison messages 가 consumer halt — DLQ + alerting.
|
|
- **Sync over async**: HTTP request-reply 를 broker 위에 — synchronous 면 broker 의 X.
|
|
|
|
## 🧪 검증 / 중복
|
|
- Verified (Richards "Software Architecture Patterns" / Confluent docs / NATS docs 2026).
|
|
- 신뢰도 A.
|
|
|
|
## 🕓 Changelog
|
|
| 날짜 | 변경 |
|
|
|---|---|
|
|
| 2026-05-08 | Phase 1 |
|
|
| 2026-05-10 | Manual cleanup — Broker vs Mediator + Kafka/NATS/DLQ 패턴 |
|