插件清单

本页内容

每个沙盒插件在其 package.json 旁边都有一个 emdash-plugin.jsonc。它是手动编辑的,包含插件的身份标识、信任契约(capabilities、hosts、storage)以及注册表显示的配置字段。emdash-plugin init 会创建一个;CLI 会自动读取 ./emdash-plugin.jsonc 用于 builddevvalidatebundlepublish

该文件是 JSONC:允许注释和尾随逗号。

以下示例显示了一个图片画廊插件的完整清单:

{
	"$schema": "./node_modules/@emdash-cms/plugin-cli/schemas/emdash-plugin.schema.json",

	"slug": "gallery",
	"publisher": "did:plc:abc123def456",

	"license": "MIT",
	"author": { "name": "Jane Doe", "url": "https://example.com" },
	"security": { "email": "[email protected]" },

	// Optional profile
	"name": "Gallery",
	"description": "Image gallery block for EmDash.",
	"keywords": ["gallery", "images"],
	"repo": "https://github.com/example/plugin-gallery",

	// Trust contract
	"capabilities": ["content:read"],
	"allowedHosts": [],
	"storage": {}
}

身份标识

字段必需说明
slug发布者命名空间内的 URL 安全 ID。/^[a-z][a-z0-9_-]*$/,最多 64 个字符。
publisher您的 Atmosphere 账户的 DID 或句柄。参见发布者固定
version不含构建元数据的 Semver 2.0。通常省略 — 见下文。

slugpublisher 共同构成包的身份。EmDash 会自动从中派生包的完整标识符。

version 位于 package.json

构建会将清单的 versionpackage.json#version 进行协调:

  • 两者都设置且相等 → 正常。
  • 两者都设置但不同 → 硬错误。
  • 只设置一个 → 该值生效。
  • 两者都未设置 → 硬错误。

npm 分发插件的推荐模式是从清单中省略 version,让 package.json 成为唯一的真实来源(你的发布工具已经在那里提升了版本)。没有 package.json 的纯注册表插件必须在清单中设置 version — 没有其他地方可以放置它。

配置文件

这些为注册表列表提供信息。license、作者(authorauthors)和安全联系人(securitysecurityContacts)是必需的;其余是可选的。

字段必需说明
licenseSPDX 表达式("MIT""Apache-2.0""MIT OR Apache-2.0")。在首次发布时使用;在后续发布中现有配置文件优先。
author / authors二选一。单个作者使用 author: { name, url?, email? };多个作者使用 authors: [...](≤ 32)。同时设置两者会出错。
security / securityContacts二选一。每个联系人至少需要 emailurl 之一。多个联系人使用 securityContacts: [...](≤ 8)。同时设置两者会出错。
name显示名称。默认为 slug。
description保持简短(约 140 个字符)。长值可能在列表中被截断。
keywords≤ 5 个条目。
repo源代码仓库的 https:// URL。

除非你真的有多个作者,否则使用单数形式 author / security — 这是常见情况,脚手架会生成它。

信任契约

信任契约包括 capabilitiesallowedHostsstorage。这三者默认都为空,因此不需要额外权限的插件可以完全省略它们。

{
	"capabilities": ["network:request", "content:read"],
	"allowedHosts": ["api.example.com", "*.cdn.example.com"],
	"storage": {
		"events": { "indexes": ["timestamp"] },
		"submissions": { "indexes": ["email"], "uniqueIndexes": ["token"] }
	}
}

Capabilities

已识别的名称:

Capability授予
content:read / content:write通过 ctx 读取/修改站点内容。
media:read / media:write读取/写入媒体。
users:read读取用户记录。
email:send通过 ctx 发送电子邮件。
network:request通过 ctx.http 进行出站 HTTP,限制为 allowedHosts
network:request:unrestricted到任何主机的出站 HTTP。用于替代 network:request
hooks.email-transport:register注册电子邮件传输钩子。
hooks.email-events:register注册电子邮件生命周期钩子。
hooks.page-fragments:register注册 page:fragments 钩子(仅原生)。

CLI 强制执行的两条跨字段规则(编辑器的 JSON-Schema 检查不执行 — 运行 emdash-plugin validate):

  • network:request 需要非空的 allowedHosts。如果插件确实必须访问任何主机,请改用 network:request:unrestricted
  • network:request:unrestricted 需要 allowedHosts 为空 — 不受限制的 capability 已经授予所有主机,因此列表会与之矛盾。

主机模式是纯主机名(无方案、路径或空格)。前导 *. 允许子域:*.cdn.example.com

Storage

集合名称 → 索引配置的映射。集合名称遵循相同的 /^[a-z][a-z0-9_]*$/ 规则(运行时使用名称作为 SQL 表后缀)。索引是字段名称或复合数组;uniqueIndexes 也可查询 — 不要在 indexes 中也列出它们。

"storage": {
	"events": { "indexes": ["timestamp", ["collection", "timestamp"]] }
}

管理界面

可选。沙盒插件通过 Block Kit 渲染管理页面和仪表板小部件;清单仅声明它们出现的位置。如果插件没有管理 UI,请完全省略 admin 键。

"admin": {
	"pages": [{ "path": "/gallery", "label": "Gallery", "icon": "image" }],
	"widgets": [{ "id": "recent-uploads", "title": "Recent uploads", "size": "half" }]
}

声明 admin.pagesadmin.widgets 的插件还必须在 src/plugin.ts 中提供渲染 Block Kit 内容的 admin 路由 — schema 无法强制执行这一点(路由名称是从源代码而非清单中探测的),但运行时会检查它。

发布者固定

publisher 固定发布身份,这样你就不会意外地在错误的账户下发布插件。

在你的首次成功发布时,如果清单的 publisher 与活动会话匹配,它会保持原样。如果你使用 emdash-plugin init 创建脚手架并将其留空,CLI 会将活动会话的 DID 写回清单。

以下示例显示了 CLI 写入的行,为了可读性添加了已解析的句柄作为注释:

"publisher": "did:plc:abc123def456", // jane.example.com

每次后续发布时,CLI 会将活动会话和固定的 publisher 解析为 DID 并进行比较。不匹配会立即以 MANIFEST_PUBLISHER_MISMATCH 失败 — 没有覆盖标志。有意解决它:

  • 会话错误:emdash-plugin switch <did>,然后再次发布。
  • 真正将插件转移到新发布者:编辑清单中的 publisher

不发布的情况下验证

emdash-plugin validate          # ./emdash-plugin.jsonc
emdash-plugin validate path/    # 特定目录

使用 tsc 风格的 file:line:column 诊断进行离线 schema 检查,包括跨字段规则。适用于预提交钩子或 CI 步骤。重复键和未知键是错误(严格模式捕获 "licens" 拼写错误)。

CLI 标志仍然优先

显式标志(--license--author-name、…)在两者都设置时会覆盖清单值 — 对 CI 覆盖有用。--no-manifest 完全跳过清单(如果在默认路径存在清单则发出警告,这样发布者固定安全故事保持可见)。

下一步