Files
2nd/10_Wiki/Topics/Coding/DB_DuckDB_Embedded.md
T
2026-05-09 21:08:02 +09:00

6.7 KiB

id, title, category, status, source_trust_level, verification_status, created_at, updated_at, tags, tech_stack, applied_in, aliases
id title category status source_trust_level verification_status created_at updated_at tags tech_stack applied_in aliases
db-duckdb-embedded DuckDB — Embedded OLAP / Local Analytics Coding draft B conceptual 2026-05-09 2026-05-09
database
duckdb
olap
embedded
vibe-coding
language applicable_to
TS / Python / SQL
Backend
Frontend
DuckDB
MotherDuck
embedded analytics
columnar SQLite
Parquet query

DuckDB

SQLite 의 OLAP 버전. Embedded columnar DB — 단일 파일, in-process, 매우 빠른 analytic query. Parquet / CSV 직접 query. ClickHouse / BigQuery 의 단일 노드 alternative.

📖 핵심 개념

  • Embedded: process 안 (no server).
  • Columnar: analytic 빠름.
  • File: .duckdb 단일 파일 또는 in-memory.
  • Federation: Parquet / CSV / S3 직접 query.

💻 코드 패턴

Node 사용

yarn add @duckdb/node-api
import { DuckDBInstance } from '@duckdb/node-api';

const db = await DuckDBInstance.create('app.duckdb');
const conn = await db.connect();

await conn.run(`
  CREATE TABLE orders (
    id UUID,
    user_id UUID,
    amount DECIMAL(10, 2),
    created_at TIMESTAMP
  )
`);

await conn.run(`INSERT INTO orders VALUES (?, ?, ?, ?)`, [id, userId, 99.50, new Date()]);

const result = await conn.run(`SELECT user_id, SUM(amount) FROM orders GROUP BY user_id`);
const rows = result.getRows();

Python

import duckdb

con = duckdb.connect('app.duckdb')
con.execute('CREATE TABLE orders (...)')
con.execute('INSERT INTO orders VALUES (...)', [...])

df = con.execute('SELECT * FROM orders').df()  # pandas

Parquet 직접 query

-- 파일 직접 (no import)
SELECT * FROM 'data.parquet';

-- 여러 파일
SELECT * FROM 'data/*.parquet';

-- Hive-partitioned
SELECT * FROM 'data/year=*/month=*/data.parquet';

-- Aggregate
SELECT date, count(*) FROM 'events_*.parquet' GROUP BY date;

S3 / HTTP

INSTALL httpfs; LOAD httpfs;

SELECT * FROM 's3://bucket/data.parquet';
SELECT * FROM 'https://example.com/data.csv';

-- Credentials
SET s3_region = 'us-east-1';
SET s3_access_key_id = '...';
SET s3_secret_access_key = '...';

Iceberg / Delta

INSTALL iceberg; LOAD iceberg;
SELECT * FROM iceberg_scan('s3://bucket/orders');

INSTALL delta; LOAD delta;
SELECT * FROM delta_scan('s3://bucket/orders');

→ Lakehouse 직접 query. 작은 cluster 또는 dev.

Postgres 직접 (federate)

INSTALL postgres; LOAD postgres;
ATTACH 'postgresql://user:pw@host/db' AS pg;

SELECT * FROM pg.public.users WHERE created_at > '2026-01-01';

-- DuckDB 가 push down 가능한 filter 그렇게.

→ Postgres 안 데이터 + DuckDB 의 analytic 함께.

CSV / JSON

SELECT * FROM read_csv('data.csv', header=true);
SELECT * FROM read_csv_auto('data.csv');  -- 자동 schema

SELECT * FROM read_json('data.json');
SELECT * FROM read_json_auto('data.ndjson');

Window / analytic

SELECT
    user_id,
    amount,
    SUM(amount) OVER (PARTITION BY user_id ORDER BY created_at) AS running_total,
    LAG(amount) OVER (PARTITION BY user_id ORDER BY created_at) AS prev_amount,
    RANK() OVER (ORDER BY amount DESC) AS rank
FROM orders;

→ Window function full 지원.

MotherDuck (managed)

ATTACH 'md:my_database' AS cloud;

SELECT * FROM cloud.orders WHERE date > '2026-05-01';

-- Local + cloud 혼합
SELECT * FROM local_orders UNION ALL SELECT * FROM cloud.orders;

→ DuckDB 클라우드 — local query + cloud sync.

Use case

1. ETL / data prep:
   - Parquet 변환
   - Aggregate 계산
   - dbt 와 통합

2. Local analytics:
   - 큰 CSV 분석
   - Notebook 안

3. Embedded analytics in app:
   - Small / medium dataset (~100GB)
   - 빠른 query (BigQuery 같지만 local)

4. Test fixture:
   - dbt local dev
   - Production analytic 모델 검증

5. Edge analytics:
   - Cloudflare D1 alternative (analytic)

Use case 안 적합

- OLTP (transaction, write-heavy concurrent)
- 매우 큰 (TB+) — Snowflake / BigQuery
- 분산 cluster 필요

Performance

1B rows aggregate: ~10s on laptop.
10M rows complex query: ms.

vs Postgres: 5-50x analytic.
vs Pandas: 메모리 효율, parallel.

vs SQLite

SQLite:  OLTP, row-oriented.
DuckDB:  OLAP, columnar.

DuckDB 가 SQLite read.
INSTALL sqlite; LOAD sqlite;
ATTACH 'app.sqlite' AS s (TYPE SQLITE);
SELECT * FROM s.users;

Concurrent access

DuckDB 는 single-writer (concurrent read OK).
Concurrent write = lock.

→ Process 1개 또는 외부 sync.

React (browser)

import * as duckdb from '@duckdb/duckdb-wasm';

const bundles = duckdb.getJsDelivrBundles();
const bundle = await duckdb.selectBundle(bundles);
const worker = new Worker(bundle.mainWorker!);
const logger = new duckdb.ConsoleLogger();
const db = new duckdb.AsyncDuckDB(logger, worker);
await db.instantiate(bundle.mainModule);

const conn = await db.connect();
await conn.query(`SELECT * FROM 'https://example.com/data.parquet'`);

→ Browser 안 SQL 분석. WASM 빌드.

Dataframe-like API

con.sql('SELECT * FROM orders').df()      # pandas
con.sql('SELECT * FROM orders').arrow()   # PyArrow
con.sql('SELECT * FROM orders').pl()       # polars

CLI

duckdb my.duckdb
> SELECT * FROM 'data.parquet';
> .mode json
> SELECT * FROM users LIMIT 5;
> .schema users

Persistent vs in-memory

const db = await DuckDBInstance.create();          // in-memory
const db = await DuckDBInstance.create('app.db');  // file

Migration / schema

-- DuckDB 도 일반 DDL
ALTER TABLE orders ADD COLUMN status VARCHAR;
CREATE INDEX orders_user ON orders(user_id);

-- Constraint
CREATE TABLE users (
    id UUID PRIMARY KEY,
    email VARCHAR UNIQUE NOT NULL CHECK (email LIKE '%@%')
);

Backup

# 단순 — file copy
cp app.duckdb app.duckdb.bak

# 또는 export
EXPORT DATABASE 'export_dir';
IMPORT DATABASE 'export_dir';

🤔 의사결정 기준

상황 추천
Embedded analytics DuckDB
큰 CSV / Parquet 분석 DuckDB
ETL / dbt local DuckDB
OLTP Postgres / SQLite
분산 / TB+ Snowflake / BigQuery / ClickHouse
Browser analytics DuckDB-wasm
Edge Cloudflare D1 (SQLite)

안티패턴

  • OLTP write 많음: SQLite 가 낫다.
  • Concurrent writers: lock contention.
  • 모든 데이터 in-memory + 큰 dataset: OOM. 파일.
  • Schema drift (read auto-detect 매번): 고정 schema.
  • Index 없는 큰 join: composite index.
  • No backup: file 손실 = 영원.

🤖 LLM 활용 힌트

  • ETL / 분석 / dbt local = DuckDB.
  • Parquet / S3 / Iceberg 직접 query.
  • Postgres + DuckDB federation.
  • WASM = browser 안 SQL.

🔗 관련 문서