EmDash CLI는 EmDash CMS 인스턴스 관리를 위한 명령을 제공합니다 — 데이터베이스 설정, 타입 생성, 콘텐츠 CRUD, 스키마 관리, 미디어 등.
설치
CLI는 emdash 패키지에 포함되어 있습니다. 다음 명령으로 설치하세요:
npm install emdash
npx emdash로 명령을 실행하거나 package.json에 스크립트를 추가하세요. 바이너리는 간결함을 위해 em으로도 사용 가능합니다.
인증
실행 중인 EmDash 인스턴스와 통신하는 명령은 다음 순서로 인증을 해결합니다:
--token플래그 — 커맨드라인의 명시적 토큰EMDASH_TOKEN환경 변수- 저장된 자격 증명
~/.config/emdash/auth.json에서 (emdash login으로 저장됨) - 개발 바이패스 — URL이 localhost이고 토큰이 없는 경우 개발 바이패스 엔드포인트를 통해 자동 인증
대부분의 명령은 --url (기본값 http://localhost:4321)과 --token 플래그를 허용합니다. 로컬 개발 서버를 대상으로 할 때는 토큰이 필요하지 않습니다.
공통 플래그
이 플래그들은 모든 원격 명령에서 사용 가능합니다:
| 플래그 | 별칭 | 설명 | 기본값 |
|---|---|---|---|
--url | -u | EmDash 인스턴스 URL | http://localhost:4321 |
--token | -t | 인증 토큰 | env/저장된 자격 증명에서 |
--json | JSON으로 출력 (파이핑용) | TTY에서 자동 감지 |
출력
stdout이 TTY인 경우 CLI는 consola로 결과를 예쁘게 출력합니다. 파이프되거나 --json이 설정된 경우 stdout으로 원시 JSON을 출력합니다 — jq나 다른 도구에 적합합니다.
명령
emdash dev
자동 데이터베이스 설정으로 개발 서버를 시작합니다.
npx emdash dev [options]
옵션
| 옵션 | 별칭 | 설명 | 기본값 |
|---|---|---|---|
--database | -d | 데이터베이스 파일 경로 | ./data.db |
--types | -t | 시작 전 원격에서 타입 생성 | false |
--port | -p | 개발 서버 포트 | 4321 |
--cwd | 작업 디렉토리 | 현재 디렉토리 |
예시
# 개발 서버 시작
npx emdash dev
# 커스텀 포트
npx emdash dev --port 3000
# 시작 전 원격에서 타입 생성
npx emdash dev --types
동작
- 대기 중인 데이터베이스 마이그레이션을 확인하고 실행
--types가 설정된 경우 원격 인스턴스에서 TypeScript 타입 생성 (EMDASH_URLenv 또는package.json의emdash.url에서 URL)EMDASH_DATABASE_URL설정으로 Astro 개발 서버 시작
emdash types
실행 중인 EmDash 인스턴스의 스키마에서 TypeScript 타입을 생성합니다.
npx emdash types [options]
옵션
| 옵션 | 별칭 | 설명 | 기본값 |
|---|---|---|---|
--url | -u | EmDash 인스턴스 URL | http://localhost:4321 |
--token | -t | 인증 토큰 | env/저장된 자격 증명에서 |
--output | -o | 타입 출력 경로 | .emdash/types.ts |
--cwd | 작업 디렉토리 | 현재 디렉토리 |
예시
# 로컬 개발 서버에서 타입 생성
npx emdash types
# 원격 인스턴스에서 생성
npx emdash types --url https://my-site.pages.dev
# 커스텀 출력 경로
npx emdash types --output src/types/emdash.ts
동작
- 인스턴스에서 스키마 가져오기
- TypeScript 타입 정의 생성
- 출력 파일에 타입 작성
- 참조용으로 옆에
schema.json작성
emdash login
OAuth Device Flow를 사용하여 EmDash 인스턴스에 로그인합니다.
npx emdash login [options]
옵션
| 옵션 | 별칭 | 설명 | 기본값 |
|---|---|---|---|
--url | -u | EmDash 인스턴스 URL | http://localhost:4321 |
동작
- 인스턴스에서 인증 엔드포인트 검색
- localhost이고 인증이 구성되지 않은 경우 자동으로 개발 바이패스 사용
- 그렇지 않으면 OAuth Device Flow 시작 — 코드를 표시하고 브라우저 열기
- 인증을 폴링한 후
~/.config/emdash/auth.json에 자격 증명 저장
저장된 자격 증명은 같은 인스턴스를 대상으로 하는 모든 후속 명령에서 자동으로 사용됩니다.
emdash logout
로그아웃하고 저장된 자격 증명을 제거합니다.
npx emdash logout [options]
옵션
| 옵션 | 별칭 | 설명 | 기본값 |
|---|---|---|---|
--url | -u | EmDash 인스턴스 URL | http://localhost:4321 |
emdash whoami
현재 인증된 사용자를 표시합니다.
npx emdash whoami [options]
옵션
| 옵션 | 별칭 | 설명 | 기본값 |
|---|---|---|---|
--url | -u | EmDash 인스턴스 URL | http://localhost:4321 |
--token | -t | 인증 토큰 | env/저장된 자격 증명에서 |
--json | JSON으로 출력 |
이메일, 이름, 역할, 인증 방법, 인스턴스 URL을 표시합니다.
emdash content
콘텐츠 항목을 관리합니다. 모든 하위 명령은 EmDashClient를 통해 원격 API를 사용합니다.
content list <collection>
npx emdash content list posts
npx emdash content list posts --status published --limit 10
| 옵션 | 설명 |
|---|---|
--status | 상태별 필터 |
--limit | 최대 항목 |
--cursor | 페이지네이션 커서 |
content get <collection> <id>
npx emdash content get posts 01ABC123
npx emdash content get posts 01ABC123 --raw
| 옵션 | 설명 |
|---|---|
--raw | 원시 Portable Text 반환 (마크다운 변환 건너뜀) |
응답에는 _rev 토큰이 포함됩니다. 덮어쓰기 전에 현재 상태를 확인했음을 증명하기 위해 content update에 전달하세요.
content create <collection>
npx emdash content create posts --data '{"title": "Hello"}'
npx emdash content create posts --file post.json --slug hello-world
cat post.json | npx emdash content create posts --stdin
| 옵션 | 설명 |
|---|---|
--data | 콘텐츠 데이터가 포함된 JSON 문자열 |
--file | JSON 파일에서 데이터 읽기 |
--stdin | stdin에서 데이터 읽기 |
--slug | 콘텐츠 슬러그 |
--locale | 콘텐츠 로케일 |
--translation-of | 이것을 번역으로 연결할 콘텐츠 항목의 ID |
--draft | 자동 게시 대신 초안으로 유지 |
--data, --file, --stdin 중 정확히 하나를 통해 데이터를 제공하세요. --draft가 설정되지 않은 한 새 항목은 자동으로 게시됩니다.
content update <collection> <id>
현재 상태를 확인했음을 증명하기 위해 이전 get의 _rev 토큰을 제공해야 합니다. 이는 보지 못한 변경 사항을 덮어쓰는 것을 방지합니다. 다음 단계는 항목을 읽은 다음 해당 토큰으로 업데이트합니다:
# 1. 항목 읽기, _rev 기록
npx emdash content get posts 01ABC123
# 2. 1단계의 _rev로 업데이트
npx emdash content update posts 01ABC123 \
--rev MToyMDI2LTAyLTE0... \
--data '{"title": "Updated"}'
| 옵션 | 설명 |
|---|---|
--rev | get의 리비전 토큰 (필수) |
--data | 콘텐츠 데이터가 포함된 JSON 문자열 |
--file | JSON 파일에서 데이터 읽기 |
get 이후 항목이 변경된 경우 서버는 409 Conflict를 반환합니다 — 다시 읽고 재시도하세요.
content delete <collection> <id>
npx emdash content delete posts 01ABC123
콘텐츠 항목을 소프트 삭제합니다 (휴지통으로 이동).
content publish <collection> <id>
npx emdash content publish posts 01ABC123
content unpublish <collection> <id>
npx emdash content unpublish posts 01ABC123
content schedule <collection> <id>
npx emdash content schedule posts 01ABC123 --at 2026-03-01T09:00:00Z
| 옵션 | 설명 |
|---|---|
--at | ISO 8601 날짜/시간 (필수) |
content restore <collection> <id>
npx emdash content restore posts 01ABC123
휴지통에 있는 콘텐츠 항목을 복원합니다.
emdash schema
컬렉션과 필드를 관리합니다.
schema list
npx emdash schema list
모든 컬렉션을 나열합니다.
schema get <collection>
npx emdash schema get posts
모든 필드가 포함된 컬렉션을 표시합니다.
schema create <collection>
npx emdash schema create articles --label Articles
npx emdash schema create articles --label Articles --label-singular Article --description "Blog articles"
| 옵션 | 설명 |
|---|---|
--label | 컬렉션 레이블 (필수) |
--label-singular | 단수 레이블 |
--description | 컬렉션 설명 |
schema delete <collection>
npx emdash schema delete articles
npx emdash schema delete articles --force
| 옵션 | 설명 |
|---|---|
--force | 확인 건너뛰기 |
--force가 설정되지 않은 한 확인을 요청합니다.
schema add-field <collection> <field>
npx emdash schema add-field posts body --type portableText --label "Body Content"
npx emdash schema add-field posts featured --type boolean --required
| 옵션 | 설명 |
|---|---|
--type | 필드 타입: string, text, number, integer, boolean, datetime, image, reference, portableText, json (필수) |
--label | 필드 레이블 (기본값은 필드 슬러그) |
--required | 필드가 필수인지 여부 |
schema remove-field <collection> <field>
npx emdash schema remove-field posts featured
emdash media
미디어 항목을 관리합니다.
media list
npx emdash media list
npx emdash media list --mime image/png --limit 20
| 옵션 | 설명 |
|---|---|
--mime | MIME 타입별 필터 |
--limit | 항목 수 |
--cursor | 페이지네이션 커서 |
media upload <file>
npx emdash media upload ./photo.jpg
npx emdash media upload ./photo.jpg --alt "A sunset" --caption "Taken in Bristol"
| 옵션 | 설명 |
|---|---|
--alt | 대체 텍스트 |
--caption | 캡션 텍스트 |
media get <id>
npx emdash media get 01MEDIA123
media delete <id>
npx emdash media delete 01MEDIA123
emdash search
콘텐츠 전체의 전체 텍스트 검색.
npx emdash search "hello world"
npx emdash search "hello" --collection posts --limit 5
| 옵션 | 별칭 | 설명 |
|---|---|---|
--collection | -c | 컬렉션별 필터 |
--limit | -l | 최대 결과 |
emdash taxonomy
분류체계와 용어를 관리합니다.
taxonomy list
npx emdash taxonomy list
taxonomy terms <name>
npx emdash taxonomy terms categories
npx emdash taxonomy terms tags --limit 50
| 옵션 | 별칭 | 설명 |
|---|---|---|
--limit | -l | 최대 용어 |
--cursor | 페이지네이션 커서 |
taxonomy add-term <taxonomy>
npx emdash taxonomy add-term categories --name "Tech" --slug tech
npx emdash taxonomy add-term categories --name "Frontend" --parent 01PARENT123
| 옵션 | 설명 |
|---|---|
--name | 용어 레이블 (필수) |
--slug | 용어 슬러그 (기본값은 슬러그화된 이름) |
--parent | 상위 용어 ID (계층적 분류체계용) |
emdash menu
내비게이션 메뉴를 관리합니다.
menu list
npx emdash menu list
menu get <name>
npx emdash menu get primary
모든 항목이 포함된 메뉴를 반환합니다.
emdash export-seed
데이터베이스 스키마와 콘텐츠를 시드 파일로 내보냅니다. 로컬 SQLite 파일에서 직접 작동합니다.
npx emdash export-seed [options] > seed.json
옵션
| 옵션 | 별칭 | 설명 | 기본값 |
|---|---|---|---|
--database | -d | 데이터베이스 파일 경로 | ./data.db |
--cwd | 작업 디렉토리 | 현재 디렉토리 | |
--with-content | 콘텐츠 포함 (모두 또는 쉼표로 구분된 컬렉션) | ||
--no-pretty | JSON 형식화 비활성화 | false |
출력 형식
내보낸 시드 파일에는 다음이 포함됩니다:
- 설정: 사이트 제목, 태그라인, 소셜 링크
- 컬렉션: 필드가 포함된 모든 컬렉션 정의
- 분류체계: 분류체계 정의 및 용어
- 메뉴: 항목이 포함된 내비게이션 메뉴
- 위젯 영역: 위젯 영역 및 위젯
- 콘텐츠 (요청된 경우): 이식성을 위한
$media참조 및$ref:구문이 포함된 항목
emdash secrets generate
배포를 위한 EMDASH_ENCRYPTION_KEY를 생성합니다. 이 키는 플러그인 시크릿을 저장 시 암호화하는 데 사용됩니다.
npx emdash secrets generate
새 키를 stdout으로 출력합니다. 시크릿 저장소로 파이프하거나 --write로 개발 파일에 직접 작성하세요. 다음 명령은 키를 .dev.vars 또는 .env에 작성합니다:
npx emdash secrets generate --write .dev.vars
npx emdash secrets generate --write .env
--write는 --force 없이 기존 항목을 덮어쓰는 것을 거부합니다. 기존 암호화된 데이터가 있는 배포에서 키를 교체하면 해당 시크릿을 읽을 수 없게 되므로 이 보호는 의도적입니다.
emdash secrets fingerprint <key>
값을 노출하지 않고 키의 8자 지문(kid)을 출력합니다. 이는 CI에서 올바른 키가 배포되었는지 확인하는 데 유용합니다. 다음 명령은 키의 지문을 출력합니다:
npx emdash secrets fingerprint emdash_enc_v1_...
생성된 파일
.emdash/types.ts
emdash types 명령은 각 컬렉션에 대한 TypeScript 인터페이스를 생성합니다:
// Generated by EmDash CLI
// Do not edit manually - run `emdash types` to regenerate
import type { PortableTextBlock } from "emdash";
export interface Post {
id: string;
title: string;
content: PortableTextBlock[];
publishedAt: Date | null;
}
.emdash/schema.json
명령은 도구용으로 원시 스키마 내보내기도 작성합니다:
{
"version": "a1b2c3d4",
"collections": [
{
"slug": "posts",
"label": "Posts",
"fields": [...]
}
]
}
환경 변수
| 변수 | 설명 |
|---|---|
EMDASH_DATABASE_URL | 데이터베이스 URL (dev에 의해 자동 설정) |
EMDASH_TOKEN | 원격 작업용 인증 토큰 |
EMDASH_URL | types 및 dev --types의 기본 원격 URL |
EMDASH_ENCRYPTION_KEY | 플러그인 시크릿을 저장 시 암호화하기 위한 키. 운영자가 제공 — 데이터베이스에 절대 저장되지 않음. emdash secrets generate로 생성. |
EMDASH_PREVIEW_SECRET | 미리보기 HMAC 시크릿의 선택적 재정의. 설정되지 않은 경우 EmDash가 옵션 테이블에 생성하고 저장. |
EMDASH_IP_SALT | 댓글 작성자 IP 해시 솔트의 선택적 재정의. 설정되지 않은 경우 EmDash가 옵션 테이블에 생성하고 저장. |
EMDASH_AUTH_SECRET | 레거시. 설정된 경우 IP 솔트 소스로 사용되어 기존 설치가 업그레이드 후에도 안정적인 댓글 작성자 IP 해시를 유지. 새 설치는 이를 설정하지 않아야 합니다. |
패키지 스크립트
편의를 위해 package.json 스크립트로 CLI 명령을 추가하세요:
{
"scripts": {
"dev": "emdash dev",
"types": "emdash types",
"export-seed": "emdash export-seed",
"db:reset": "rm -f data.db"
}
}
종료 코드
| 코드 | 설명 |
|---|---|
0 | 성공 |
1 | 오류 (구성, 네트워크, 데이터베이스) |