Seed 文件格式

本页内容

Seed 文件是初始化 EmDash 站点的 JSON 文档。它们定义集合、字段、分类法、菜单、重定向、小部件区域、站点设置和可选的示例内容。

根结构

Seed 文件具有以下顶级结构:

{
	"$schema": "https://emdashcms.com/seed.schema.json",
	"version": "1",
	"meta": {},
	"settings": {},
	"collections": [],
	"taxonomies": [],
	"bylines": [],
	"menus": [],
	"redirects": [],
	"widgetAreas": [],
	"sections": [],
	"content": {}
}
字段类型必需描述
$schemastring用于编辑器验证的 JSON 架构 URL
version"1"Seed 格式版本
metaobject关于 seed 的元数据
settingsobject站点设置
collectionsarray集合定义
taxonomiesarray分类法定义
bylinesarray署名档案定义
menusarray导航菜单
redirectsarray重定向规则
widgetAreasarray小部件区域定义
sectionsarray可重用的内容块
contentobject示例内容条目

Meta

meta 对象包含有关 seed 的可选描述性元数据:

{
	"meta": {
		"name": "Blog Starter",
		"description": "A simple blog with posts, pages, and categories",
		"author": "EmDash"
	}
}

Settings

settings 对象包含站点范围的配置值:

{
	"settings": {
		"title": "My Site",
		"tagline": "A modern CMS",
		"postsPerPage": 10,
		"dateFormat": "MMMM d, yyyy"
	}
}

设置应用于带有 site: 前缀的 options 表。设置向导将从 seed 文件(如果提供)预填充 titletagline,允许用户在初始设置期间覆盖它们。

Collections

每个集合定义在数据库中创建一个内容类型:

{
	"collections": [
		{
			"slug": "posts",
			"label": "Posts",
			"labelSingular": "Post",
			"description": "Blog posts",
			"icon": "file-text",
			"supports": ["drafts", "revisions"],
			"fields": [
				{
					"slug": "title",
					"label": "Title",
					"type": "string",
					"required": true
				},
				{
					"slug": "content",
					"label": "Content",
					"type": "portableText"
				},
				{
					"slug": "featured_image",
					"label": "Featured Image",
					"type": "image"
				}
			]
		}
	]
}

集合属性

属性类型必需描述
slugstringURL 安全标识符(小写、下划线)
labelstring复数显示名称
labelSingularstring单数显示名称
descriptionstring管理界面描述
iconstringLucide 图标名称
supportsarray功能:"drafts""revisions"
fieldsarray字段定义

字段属性

属性类型必需描述
slugstring列名(小写、下划线)
labelstring显示名称
typestring字段类型
requiredboolean验证:字段必须有值
uniqueboolean验证:值必须唯一
defaultValueany新条目的默认值
validationobject其他验证规则
widgetstring管理界面小部件覆盖
optionsobject小部件特定配置

字段类型

类型描述存储为
string短文本TEXT
text长文本(文本区域)TEXT
number数值REAL
integer整数INTEGER
boolean真/假INTEGER
date日期值TEXT (ISO 8601)
datetime日期和时间TEXT (ISO 8601)
email电子邮件地址TEXT
urlURLTEXT
slugURL 安全字符串TEXT
portableText富文本内容JSON
image图像引用JSON
file文件引用JSON
json任意 JSONJSON
reference对另一个条目的引用TEXT

Taxonomies

分类法是内容的分类系统:

{
	"taxonomies": [
		{
			"name": "category",
			"label": "Categories",
			"labelSingular": "Category",
			"hierarchical": true,
			"collections": ["posts"],
			"terms": [
				{ "slug": "news", "label": "News" },
				{ "slug": "tutorials", "label": "Tutorials" },
				{
					"slug": "advanced",
					"label": "Advanced Tutorials",
					"parent": "tutorials"
				}
			]
		},
		{
			"name": "tag",
			"label": "Tags",
			"labelSingular": "Tag",
			"hierarchical": false,
			"collections": ["posts"]
		}
	]
}

分类法属性

属性类型必需描述
namestring唯一标识符
labelstring复数显示名称
labelSingularstring单数显示名称
hierarchicalboolean允许嵌套术语(类别)或扁平(标签)
collectionsarray此分类法适用的集合
termsarray预定义术语

术语属性

属性类型必需描述
slugstringURL 安全标识符
labelstring显示名称
descriptionstring术语描述
parentstring父术语 slug(仅分层)

menus 数组定义可从管理员编辑的导航菜单:

{
	"menus": [
		{
			"name": "primary",
			"label": "Primary Navigation",
			"items": [
				{ "type": "custom", "label": "Home", "url": "/" },
				{ "type": "page", "ref": "about" },
				{ "type": "custom", "label": "Blog", "url": "/posts" },
				{
					"type": "custom",
					"label": "External",
					"url": "https://example.com",
					"target": "_blank"
				}
			]
		}
	]
}

菜单项类型

类型描述必需字段
custom自定义 URLurl
page链接到页面条目ref
post链接到文章条目ref
taxonomy链接到分类法存档refcollection
collection链接到集合存档collection

菜单项属性

属性类型描述
typestring项目类型(见上文)
labelstring显示文本(页面/文章引用自动生成)
urlstring自定义 URL(用于 custom 类型)
refstringseed 中的内容 ID(用于 page/post 类型)
collectionstring集合 slug
targetstring新窗口使用 "_blank"
titleAttrstringHTML title 属性
cssClassesstring自定义 CSS 类
childrenarray嵌套菜单项

Bylines

署名档案与所有权(author_id)分开。定义一次可重用的署名身份,然后从内容条目引用它们。

{
	"bylines": [
		{
			"id": "editorial",
			"slug": "emdash-editorial",
			"displayName": "EmDash Editorial"
		},
		{
			"id": "guest",
			"slug": "guest-contributor",
			"displayName": "Guest Contributor",
			"isGuest": true
		}
	]
}
属性类型必需描述
idstringcontent[].bylines 使用的 seed 本地 ID
slugstringURL 安全的署名 slug
displayNamestring模板和 API 中显示的名称
biostring可选的档案简介
websiteUrlstring可选的网站 URL
isGuestboolean将署名标记为访客档案

Redirects

redirects 数组定义在迁移后保留旧版 URL 的重定向规则:

{
	"redirects": [
		{ "source": "/old-about", "destination": "/about" },
		{ "source": "/legacy-feed", "destination": "/rss.xml", "type": 308 },
		{
			"source": "/category/news",
			"destination": "/categories/news",
			"groupName": "migration"
		}
	]
}

重定向属性

属性类型必需描述
sourcestring源路径(必须以 / 开头)
destinationstring目标路径(必须以 / 开头)
typenumberHTTP 状态:301302307308
enabledboolean重定向是否处于活动状态(默认:true
groupNamestring用于管理过滤/搜索的可选分组标签

Widget Areas

widgetAreas 数组定义可配置的内容区域:

{
	"widgetAreas": [
		{
			"name": "sidebar",
			"label": "Main Sidebar",
			"description": "Appears on blog posts and pages",
			"widgets": [
				{
					"type": "component",
					"title": "Recent Posts",
					"componentId": "core:recent-posts",
					"props": { "count": 5 }
				},
				{
					"type": "menu",
					"title": "Quick Links",
					"menuName": "footer"
				},
				{
					"type": "content",
					"title": "About",
					"content": [
						{
							"_type": "block",
							"style": "normal",
							"children": [{ "_type": "span", "text": "Welcome to our site!" }]
						}
					]
				}
			]
		}
	]
}

小部件类型

类型描述必需字段
content富文本内容content(Portable Text)
menu渲染菜单menuName
component注册的组件componentId

内置组件

组件 ID描述
core:recent-posts最近文章列表
core:categories类别列表
core:tags标签云
core:search搜索表单
core:archives月度存档

Sections

Sections 是编辑者通过 /section 斜杠命令插入到 Portable Text 字段中的可重用内容块:

{
	"sections": [
		{
			"slug": "hero-centered",
			"title": "Centered Hero",
			"description": "Full-width hero with centered heading and CTA button",
			"keywords": ["hero", "banner", "header", "landing"],
			"content": [
				{
					"_type": "block",
					"style": "h1",
					"children": [{ "_type": "span", "text": "Welcome to Our Site" }]
				},
				{
					"_type": "block",
					"children": [
						{ "_type": "span", "text": "Your compelling tagline goes here." }
					]
				}
			]
		}
	]
}

Section 属性

属性类型必需描述
slugstringURL 安全标识符
titlestring在 section 选择器中显示的显示名称
descriptionstring解释何时使用此 section
keywordsarray用于查找 section 的搜索词
contentarrayPortable Text 块
sourcestring"theme"(seed 的默认值)或 "import"

来自 seed 文件的 sections 标记为 source: "theme",不能从管理界面删除。编辑者可以创建自己的 sections(source: "user"),并在编辑内容时插入任何 section 类型。

Content

content 对象包含按集合组织的示例内容条目:

{
	"content": {
		"posts": [
			{
				"id": "hello-world",
				"slug": "hello-world",
				"status": "published",
				"bylines": [
					{ "byline": "editorial" },
					{ "byline": "guest", "roleLabel": "Guest essay" }
				],
				"data": {
					"title": "Hello World",
					"content": [
						{
							"_type": "block",
							"style": "normal",
							"children": [{ "_type": "span", "text": "Welcome!" }]
						}
					],
					"excerpt": "Your first post."
				},
				"taxonomies": {
					"category": ["news"],
					"tag": ["welcome", "first-post"]
				}
			}
		],
		"pages": [
			{
				"id": "about",
				"slug": "about",
				"status": "published",
				"data": {
					"title": "About Us",
					"content": [
						{
							"_type": "block",
							"style": "normal",
							"children": [{ "_type": "span", "text": "About page content." }]
						}
					]
				}
			}
		]
	}
}

内容条目属性

属性类型必需描述
idstring用于引用的 seed 本地 ID
slugstringURL slug
statusstring"published""draft"(默认:"published"
dataobject字段值
bylinesarray有序的署名积分(byline、可选 roleLabel
taxonomiesobject按分类法名称的术语分配

Content References

使用 $ref: 前缀引用其他内容条目:

{
	"data": {
		"related_posts": ["$ref:another-post", "$ref:third-post"]
	}
}

$ref: 前缀在种子期间将 seed ID 解析为数据库 ID。

Media References

从 URL 包含图像:

{
	"data": {
		"featured_image": {
			"$media": {
				"url": "https://images.unsplash.com/photo-xxx",
				"alt": "Description of the image",
				"filename": "hero.jpg",
				"caption": "Photo by Someone"
			}
		}
	}
}

.emdash/media/ 包含本地图像:

{
	"data": {
		"featured_image": {
			"$media": {
				"file": "hero.jpg",
				"alt": "Description of the image"
			}
		}
	}
}

媒体属性

属性类型必需描述
urlstring是*要下载的远程 URL
filestring是*.emdash/media/ 中的本地文件名
altstring用于可访问性的替代文本
filenamestring覆盖文件名
captionstring媒体标题

*需要 urlfile 之一,但不能同时需要两者。

Applying Seeds Programmatically

对 CLI 工具或脚本使用 seed API:

import { applySeed, validateSeed } from "emdash/seed";
import seedData from "./.emdash/seed.json";

// 首先验证
const validation = validateSeed(seedData);
if (!validation.valid) {
	console.error(validation.errors);
	process.exit(1);
}

// 应用 seed
const result = await applySeed(db, seedData, {
	includeContent: true,
	onConflict: "skip",
	storage: myStorage,
	baseUrl: "http://localhost:4321",
});

console.log(result);
// {
//   collections: { created: 2, skipped: 0 },
//   fields: { created: 8, skipped: 0 },
//   taxonomies: { created: 2, terms: 5 },
//   bylines: { created: 2, skipped: 0 },
//   menus: { created: 1, items: 4 },
//   redirects: { created: 3, skipped: 0 },
//   widgetAreas: { created: 1, widgets: 3 },
//   settings: { applied: 3 },
//   content: { created: 3, skipped: 0 },
//   media: { created: 2, skipped: 0 }
// }

Apply 选项

选项类型默认值描述
includeContentbooleanfalse创建示例内容条目
onConflictstring"skip""skip""update""error"
mediaBasePathstring本地媒体文件的基本路径
storageStorage用于媒体上传的存储适配器
baseUrlstring媒体 URL 的基本 URL

Idempotency

种子可以安全地运行多次。按实体类型的冲突行为:

实体行为
集合如果 slug 存在则跳过
字段如果 collection + slug 存在则跳过
分类法定义如果名称存在则跳过
分类法术语如果名称 + slug 存在则跳过
署名档案如果 slug 存在则跳过
菜单如果名称存在则跳过
菜单项全部替换(菜单重新创建)
重定向如果源存在则跳过
小部件区域如果名称存在则跳过
小部件全部替换(区域重新创建)
Section如果 slug 存在则跳过
设置更新(设置旨在更改)
内容如果 slug 在集合中存在则跳过

Validation

Seed 文件在应用之前进行验证:

import { validateSeed } from "emdash/seed";

const { valid, errors, warnings } = validateSeed(seedData);

if (!valid) {
	errors.forEach((e) => console.error(e));
}

warnings.forEach((w) => console.warn(w));

验证检查:

  • 必需字段存在
  • Slug 对其类型有效(集合和字段 slug 允许小写字母、数字和下划线;其他 slug 还允许连字符)
  • 字段类型有效
  • 引用指向现有内容
  • 分层术语父级存在
  • 重定向路径是安全的本地 URL
  • 重定向源是唯一的
  • 集合中没有重复的 slug

CLI Commands

.emdash/seed.jsonpackage.json#emdash.seedseed/seed.json 中的 seed 文件被内联到构建中,并在数据库为空时的第一个请求时应用。要将现有站点的架构(以及可选的其内容)导出为 seed 文件:

# `mkdir -p` 因为在新项目上 .emdash/ 可能还不存在
mkdir -p .emdash

# 将当前架构导出为 seed 文件
npx emdash export-seed > .emdash/seed.json

# 导出包含内容
npx emdash export-seed --with-content > .emdash/seed.json

Next Steps