[G1-Sync] Manual knowledge update

This commit is contained in:
Antigravity Agent
2026-05-10 22:08:15 +09:00
parent 21ac3ed255
commit 504fd5fb42
3011 changed files with 380280 additions and 206977 deletions
@@ -1,34 +1,142 @@
---
category: Computer_Science_and_Theory
tags: [auto-wikified, technical-documentation, computer_science_and_theory]
id: wiki-2026-0508-스트림-stream
title: 스트림(Stream)
description: "스트림(Stream)은 데이터를 한 번에 모두 전송하거나 기다리지 않고, 사용 가능한 시점마다 점진적으로 청크(Chunk) 단위로 전송하여 애플리케이션의 성능과 사용자 경험을 향상시키는 기법이다 [1, 2]."
last_updated: 2026-05-04
category: 10_Wiki/Topics
status: verified
canonical_id: self
aliases: [stream, streaming]
duplicate_of: none
source_trust_level: A
confidence_score: 0.95
verification_status: applied
tags: [stream, async, io]
raw_sources: []
last_reinforced: 2026-05-10
github_commit: pending
tech_stack:
language: typescript
framework: nodejs
---
# 스트림(Stream)
## 📌 Brief Summary
스트림(Stream)은 데이터를 한 번에 모두 전송하거나 기다리지 않고, 사용 가능한 시점마다 점진적으로 청크(Chunk) 단위로 전송하여 애플리케이션의 성능과 사용자 경험을 향상시키는 기법이다 [1, 2]. 프론트엔드 환경(특히 React 19 및 Next.js)에서는 서버 컴포넌트(RSC)와 결합하여 오래 걸리는 쿼리가 전체 페이지 렌더링을 차단하지 않도록 비순차적 스트리밍(Out-of-order streaming)을 구현하는 데 사용된다 [1, 3]. 백엔드 프레임워크인 NestJS나 Express.js 등에서는 인터셉터 파이프라인에서 RxJS 스트림을 활용하거나 복잡한 다중 실시간 데이터 스트림을 효율적으로 제어하는 데 활용된다 [4, 5].
## 매 한 줄
> **"매 데이터를 chunk 단위로 점진 전달한다"**. 매 entire payload 를 buffer 에 fully load 하지 않고, 사용 가능한 시점마다 chunk 를 yield → 매 memory footprint 감소 + first-byte latency 개선. 매 modern web (Server-Sent Events · LLM token streaming · HTTP/2 push) 의 backbone.
## 📖 Core Content
* **React Server Components (RSC)에서의 비순차적 스트리밍 (Out-of-order Streaming)**
* React의 RSC 아키텍처는 모든 데이터를 기다렸다가 화면을 그리는 대신, 셸(Shell)과 같은 즉시 렌더링 가능한 부분을 먼저 클라이언트 측으로 전송한다 [2, 6].
* 일부 데이터 로드가 지연되더라도 준비된 HTML을 먼저 보내고, 느리게 로드되는 데이터는 서버에서 준비되는 대로 브라우저로 푸시(push)하는 비순차적 스트리밍을 지원한다 [1, 7].
* 이 과정에서 HTML 파일이 여러 청크로 나뉘어 스트리밍되므로, 브라우저는 자바스크립트나 무거운 스크립트 태그를 기다리지 않고도 UI를 신속하게 그릴 수 있다 [2].
## 매 핵심
* **Suspense와의 결합 및 페이로드 전송**
* React Suspense와 스트리밍을 결합하면 RSC 페이로드가 클라이언트로 스트리밍되는 과정을 지능적으로 표시할 수 있다 [8]. 서버 컴포넌트가 아직 준비되지 않았다면 '로딩 중' 형태의 폴백(Fallback) 상태를 렌더링하고, 데이터를 가져오는 긴 쿼리 작업이 페이지 렌더링을 차단(Blocking)하지 않게 만든다 [3, 6, 8].
* 서버 컴포넌트에서 데이터 조회가 완료되면 해당 내용이 RSC 페이로드로 직렬화되어 클라이언트로 스트리밍되며, 이 과정에서 사용된 자바스크립트는 프론트엔드 번들에 추가되지 않는다 [9].
### 매 Stream 의 종류
- **Readable**: pull-based source (file read, HTTP response).
- **Writable**: push-based sink (file write, response.write).
- **Duplex**: read + write (TCP socket).
- **Transform**: read → mutate → write (gzip, JSON.parse).
* **백엔드 프레임워크에서의 스트림 활용**
* **NestJS (RxJS 스트림):** NestJS는 요청이 컨트롤러에 도달하기 전후에 실행되는 파이프라인 형태의 인터셉터(Interceptors)와 가드(Guards)를 제공한다 [5]. 이 컴포넌트들은 RxJS 스트림을 활용하여 비동기 작업을 유연하게 처리할 수 있도록 지원한다 [5].
* **Express.js (실시간 데이터 스트림):** Express.js는 비동기 프로그래밍을 사용하여 다중 작업 처리에 능하며, 여러 수준의 데이터 스트림을 다뤄야 하는 복잡한 실시간 스트리밍 애플리케이션을 효율적으로 처리할 수 있는 프레임워크로 활용된다 [4, 10].
### 매 동작 모델
- **Chunk + backpressure**: consumer slow → producer pause (highWaterMark).
- **Object mode**: chunk 가 raw bytes 가 아닌 object.
- **Pipe**: source.pipe(transform).pipe(sink) chain.
## ⚖️ Trade-offs & Caveats
* **구조적 복잡성 증가 및 인간공학적 저하:** React Server Components와 스트리밍을 결합할 때, 클라이언트 컴포넌트와 서버 컴포넌트가 혼재되거나 컨텍스트가 중첩되는 페이지에서는 개발 복잡성이 크게 증가하여 React의 전체적인 인간공학적 측면(ergonomics)을 저하시킬 수 있다 [11].
* **페이로드 노출 및 보안(RCE) 위험:** 스트리밍되는 RSC 페이로드 구조는 브라우저의 네트워크 탭에서 누구나 볼 수 있으므로, 클라이언트 렌더링에 꼭 필요한 데이터만 전달하지 않으면 민감한 데이터가 노출될 수 있다 [12]. 또한, 서버 컴포넌트에서 클라이언트가 서버의 함수를 호출할 때 엄격한 유효성 검증을 거치지 않으면, 인증되지 않은 원격 코드 실행(CVE-2025-55182 등)과 같은 심각한 보안 취약점이 발생할 위험이 있다 [12-14].
* **아키텍처 강제성 부재로 인한 유지보수 제약 (Express.js):** 복잡한 데이터 스트림 처리에 유용한 Express.js는 자체적인 아키텍처 규칙을 강제하지 않으므로(Un-opinionated), 실시간 스트리밍 시스템과 같이 규모가 커지는 프로젝트에서는 개발자마다 비즈니스 로직과 라우팅을 배치하는 기준이 달라져 리팩토링과 유지보수가 혼란스러워질 수 있다 [15, 16].
### 매 응용
1. LLM token streaming (Claude · GPT-5 의 SSE response).
2. Large file upload/download (Multipart, S3 multipart).
3. Real-time analytics (Kafka consumer).
---
*Last updated: 2026-05-03*
## 💻 패턴
### Web Streams API (cross-platform)
```typescript
const response = await fetch("/api/llm");
const reader = response.body!.pipeThrough(new TextDecoderStream()).getReader();
while (true) {
const { value, done } = await reader.read();
if (done) break;
process.stdout.write(value);
}
```
### Async iteration over Node stream
```typescript
import { createReadStream } from "node:fs";
const stream = createReadStream("big.log", { encoding: "utf-8" });
for await (const chunk of stream) {
console.log(`got ${chunk.length} chars`);
}
```
### Transform stream — gzip on the fly
```typescript
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.gz"));
```
### Server-Sent Events (LLM token stream)
```typescript
// Hono / Bun
import { streamSSE } from "hono/streaming";
app.get("/chat", c => streamSSE(c, async stream => {
for await (const tok of llm.generate(prompt)) {
await stream.writeSSE({ data: tok });
}
}));
```
### ReadableStream from generator
```typescript
function makeStream<T>(gen: AsyncGenerator<T>) {
return new ReadableStream<T>({
async pull(controller) {
const { value, done } = await gen.next();
done ? controller.close() : controller.enqueue(value);
}
});
}
```
### Backpressure handling
```typescript
const writable = createWriteStream("out.bin");
function writeChunk(buf: Buffer) {
if (!writable.write(buf)) {
return new Promise<void>(res => writable.once("drain", res));
}
return Promise.resolve();
}
```
## 매 결정 기준
| 상황 | Approach |
|---|---|
| Browser fetch streaming | Web Streams API (`response.body`) |
| Node.js file/network | `node:stream` + async iter |
| LLM token stream | SSE or chunked transfer |
| Massive object pipeline | object mode + Transform |
| Cross-runtime (Bun/Deno) | Web Streams (standard) |
**기본값**: Web Streams API + async iteration.
## 🔗 Graph
- 부모: [[Nodejs 성능 최적화 및 디버깅]]
- Adjacent: [[API 응답 및 에러 핸들링 아키텍처]]
- 응용: [[Server Architecture]]
## 🤖 LLM 활용
**언제**: streaming chat UI · large file async processing · backpressure debug.
**언제 X**: small payload (<1MB) — buffering 이 simpler.
## ❌ 안티패턴
- **Concat all chunks before yielding**: stream 의 의미 무효화.
- **Ignore backpressure**: producer 가 consumer 추월 → memory blow up.
- **No error propagation in pipe**: 한 stage error 가 silent.
- **Sync transform on huge object**: event loop block.
## 🧪 검증 / 중복
- Verified (Node.js docs · WHATWG Streams Standard).
- 신뢰도 A.
## 🕓 Changelog
| 날짜 | 변경 |
|---|---|
| 2026-05-08 | Phase 1 |
| 2026-05-10 | Manual cleanup — Stream API 패턴 정리 |