178 lines
4.5 KiB
Markdown
178 lines
4.5 KiB
Markdown
---
|
|
id: node-streams-patterns
|
|
title: Node Streams — Readable / Writable / Transform / Web Streams
|
|
category: Coding
|
|
status: draft
|
|
source_trust_level: B
|
|
verification_status: conceptual
|
|
created_at: 2026-05-09
|
|
updated_at: 2026-05-09
|
|
tags: [node, streams, async-iterator, vibe-coding]
|
|
tech_stack: { language: "TS / Node", applicable_to: ["Backend"] }
|
|
applied_in: []
|
|
aliases: [Readable, Writable, Transform, pipeline, async iterator, Web Streams, ReadableStream]
|
|
---
|
|
|
|
# Node Streams
|
|
|
|
> 큰 파일 / 무한 데이터 = stream. **`pipeline()` + async iterator + Web Streams** 가 modern. Backpressure 자동 처리. 옛 .pipe() / event listener 보다 안전.
|
|
|
|
## 📖 핵심 개념
|
|
- Readable: 읽기 (file, network response).
|
|
- Writable: 쓰기.
|
|
- Transform: 변환 (gzip, parser).
|
|
- Async iterator: `for await (const chunk of stream)`.
|
|
- Backpressure: writable 가 느리면 readable 도 멈춤.
|
|
|
|
## 💻 코드 패턴
|
|
|
|
### pipeline (가장 안전)
|
|
```ts
|
|
import { pipeline } from 'node:stream/promises';
|
|
import { createReadStream, createWriteStream } from 'node:fs';
|
|
import { createGzip } from 'node:zlib';
|
|
|
|
await pipeline(
|
|
createReadStream('input.txt'),
|
|
createGzip(),
|
|
createWriteStream('output.txt.gz'),
|
|
);
|
|
// 한쪽 에러 → 모든 stream cleanup
|
|
```
|
|
|
|
### Async iterator (read)
|
|
```ts
|
|
import { createReadStream } from 'node:fs';
|
|
import { createInterface } from 'node:readline';
|
|
|
|
const rl = createInterface({ input: createReadStream('big.txt'), crlfDelay: Infinity });
|
|
for await (const line of rl) {
|
|
if (line.includes('error')) console.log(line);
|
|
}
|
|
```
|
|
|
|
### Transform (변환)
|
|
```ts
|
|
import { Transform } from 'node:stream';
|
|
|
|
const upper = new Transform({
|
|
transform(chunk, _enc, cb) {
|
|
cb(null, chunk.toString().toUpperCase());
|
|
},
|
|
});
|
|
|
|
await pipeline(stdin, upper, stdout);
|
|
```
|
|
|
|
### 직접 Readable 만들기
|
|
```ts
|
|
import { Readable } from 'node:stream';
|
|
|
|
async function* generate() {
|
|
for (let i = 0; i < 1_000_000; i++) yield `${i}\n`;
|
|
}
|
|
|
|
const stream = Readable.from(generate());
|
|
await pipeline(stream, createWriteStream('out.txt'));
|
|
```
|
|
|
|
### Web Streams (modern, fetch / 브라우저 호환)
|
|
```ts
|
|
const r = await fetch('https://example.com/big.json');
|
|
const reader = r.body!.getReader();
|
|
const decoder = new TextDecoder();
|
|
|
|
while (true) {
|
|
const { value, done } = await reader.read();
|
|
if (done) break;
|
|
process(decoder.decode(value, { stream: true }));
|
|
}
|
|
|
|
// 또는 async iterator (Node 18+)
|
|
for await (const chunk of r.body!) { ... }
|
|
```
|
|
|
|
### Web ↔ Node 변환
|
|
```ts
|
|
import { Readable, Writable } from 'node:stream';
|
|
|
|
const nodeReadable = Readable.fromWeb(webReadable);
|
|
const webReadable = Readable.toWeb(nodeReadable);
|
|
```
|
|
|
|
### Backpressure 직접 제어
|
|
```ts
|
|
async function writeLines(writable: Writable, lines: AsyncIterable<string>) {
|
|
for await (const line of lines) {
|
|
if (!writable.write(line)) {
|
|
await new Promise(r => writable.once('drain', r));
|
|
}
|
|
}
|
|
writable.end();
|
|
}
|
|
```
|
|
|
|
### CSV streaming parse
|
|
```ts
|
|
import { parse } from 'csv-parse';
|
|
|
|
await pipeline(
|
|
createReadStream('big.csv'),
|
|
parse({ columns: true }),
|
|
async function* (rows) {
|
|
for await (const row of rows) {
|
|
yield JSON.stringify(row) + '\n';
|
|
}
|
|
},
|
|
createWriteStream('out.ndjson'),
|
|
);
|
|
```
|
|
|
|
### HTTP response streaming
|
|
```ts
|
|
import { Readable } from 'node:stream';
|
|
|
|
app.get('/big', async (req, res) => {
|
|
res.setHeader('content-type', 'application/x-ndjson');
|
|
const data = streamFromDB();
|
|
await pipeline(data, res); // 자동 backpressure
|
|
});
|
|
```
|
|
|
|
### Error 처리
|
|
```ts
|
|
try {
|
|
await pipeline(...);
|
|
} catch (e) {
|
|
// 모든 stream 에서 발생한 에러 캡처
|
|
}
|
|
```
|
|
|
|
## 🤔 의사결정 기준
|
|
| 상황 | 추천 |
|
|
|---|---|
|
|
| 큰 파일 read/write | createReadStream + pipeline |
|
|
| HTTP body | for await (chunk of req) |
|
|
| 변환 | Transform 또는 async generator |
|
|
| Browser 호환 | Web Streams |
|
|
| 메모리 한계 | stream 필수 |
|
|
| 작은 데이터 | string / Buffer 직접 |
|
|
|
|
## ❌ 안티패턴
|
|
- **`.pipe(...)` 직접 + error handler 누락**: dangling stream.
|
|
- **String concat 으로 GB 파일 read**: OOM.
|
|
- **on('data') 만 + flow mode**: backpressure 무시. async iterator.
|
|
- **Transform `_flush` 누락 + 마지막 chunk**: 데이터 일부 잃음.
|
|
- **paused readable 그대로 await**: hang.
|
|
- **String chunk 가정**: encoding 불일치 가능. setEncoding('utf8').
|
|
|
|
## 🤖 LLM 활용 힌트
|
|
- pipeline + async iterator + Web Streams 3종.
|
|
- 큰 데이터는 항상 stream.
|
|
- Error = pipeline catch.
|
|
|
|
## 🔗 관련 문서
|
|
- [[Backend_WebSocket_Scaling]]
|
|
- [[AI_Streaming_LLM_Response]]
|
|
- [[Backpressure_Patterns]]
|