Files
archived-MoviePilot-Plugins/docs/Repository_Guide.md
2026-05-20 19:56:21 +08:00

268 lines
9.7 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
# MoviePilot-Plugins 仓库指南
本文档面向维护者和插件开发者,说明 `MoviePilot-Plugins` 在整个 MoviePilot 体系中的职责、目录约定、元数据规则、发布流程,以及与 `MoviePilot` / `MoviePilot-Frontend` 两个主仓库的边界。
## 1. 仓库职责
`MoviePilot-Plugins` 不是独立运行时,而是插件市场和插件源码仓库。
- `MoviePilot` 后端仓库负责:
- 插件类加载与生命周期管理
- 事件与链式扩展
- 插件 API / 服务 / 仪表板注册
- 配置、插件数据、权限控制
- 插件安装、升级、分身、远程组件静态资源服务
- `MoviePilot-Frontend` 前端仓库负责:
- 插件市场与插件卡片展示
- 插件配置页、详情页、仪表板渲染
- Vue 联邦远程组件加载
- 插件侧栏全页入口
- `MoviePilot-Plugins` 负责:
- 插件源码目录
- 插件市场索引文件
- 插件图标资源
- 插件开发与维护文档
因此,开发插件时要避免把“宿主逻辑”误写进本仓库文档。例如:
- 某个 `get_api()` 为什么没有被挂载,应该先看 `MoviePilot/app/api/endpoints/plugin.py`
- 某个 Vue 远程页面为什么没有出现在侧栏,应该先看 `MoviePilot-Frontend` 的联邦加载与菜单逻辑
- 某个插件为什么在插件市场里没显示,才应该先看本仓库的 `package.json` / `package.v2.json`
## 2. 目录结构
本仓库当前采用如下结构:
```text
MoviePilot-Plugins/
├── plugins/ # 默认插件目录
├── plugins.v2/ # V2 专用插件目录
├── icons/ # 插件图标
├── docs/ # 文档
├── package.json # 默认插件索引
├── package.v2.json # V2 优先插件索引
└── .github/workflows/ # 自动发布工作流
```
关键约定:
- 一个插件一个目录。
- 目录名必须是插件类名的小写,例如 `class AutoSignIn` 对应目录 `autosignin/`
- 插件主类必须定义在该目录的 `__init__.py` 中。
- 插件目录内可附带:
- `requirements.txt`:额外 Python 依赖
- `README.md`:插件专属使用说明
- `dist/assets/`Vue 联邦构建产物
- 其他运行时所需静态文件
## 3. 元数据文件说明
### 3.1 `package.json`
默认插件索引文件,用于:
- 旧版兼容或默认版本插件
- 对 V2 兼容但不需要单独维护代码目录的插件
如果某个默认插件也能用于 V2需要在条目上声明
```json
{
"MyPlugin": {
"version": "1.2.3",
"v2": true
}
}
```
### 3.2 `package.v2.json`
V2 优先插件索引文件。MoviePilot 在 V2 环境下会优先读取这里的条目;找不到时,才会回退到 `package.json` 中声明了 `"v2": true` 的兼容插件。
### 3.3 常用字段
每个索引条目通常包含:
- `name`:插件展示名
- `description`:插件简介
- `labels`:标签,多个标签使用英文逗号分隔
- `version`:插件版本
- `icon`:图标文件名或完整 HTTP URL
- `author`:作者
- `level`:用户可见级别
- `system_version`:可安装的 MoviePilot 主系统版本范围,格式参考 pip 依赖版本约束,例如 `">=2.12.0,<3"`
- `history`:更新日志
- `release`:是否使用 GitHub Release 压缩包发布
- `v2`:默认索引中的插件是否兼容 V2
这些字段是“插件市场展示元数据”,而不是运行时唯一真相。真正加载后的插件类仍然需要在代码里声明自己的 `plugin_name``plugin_desc``plugin_version` 等属性。两者必须同步。
## 4. 版本选择与加载规则
MoviePilot 当前的插件版本选择逻辑可以概括为:
1. 先确定当前宿主版本标识,例如 `v2`
2. 优先检查 `package.v2.json` 中是否存在该插件
3. 若不存在,再检查 `package.json`
4. 只有当 `package.json` 中对应条目显式声明 `"v2": true` 时,才会作为 V2 兼容插件继续使用
5. 如果条目声明了 `system_version`,安装、更新检测和本地插件同步会继续检查当前 MoviePilot 主程序版本是否落在该范围内;未声明则不检查
这意味着:
- 同一个插件若在 `package.v2.json` 中已有专用实现,就不要再依赖 `package.json` 中的兼容声明做“隐式覆盖”。
- 新写的 V2 专用插件,优先放 `plugins.v2/`,并把元数据写入 `package.v2.json`
- 真正跨版本共用一套实现时,再使用 `package.json + "v2": true` 的方式。
- 依赖宿主新增能力的插件需要同步声明 `system_version`,否则旧版 MoviePilot 仍可能看到更新入口但安装后无法加载。
## 5. 与宿主仓库的协作边界
### 5.1 与 `MoviePilot` 后端的边界
本仓库只保存插件实现,不应复制宿主的公共能力。插件应优先复用后端仓库已经提供的抽象,例如:
- `_PluginBase`
- `eventmanager`
- `DownloaderHelper` / `MediaServerHelper` / `NotificationHelper`
- `save_data()` / `get_data()` / `get_data_path()`
- 插件 API 动态注册
- 插件仪表板、服务、工作流动作、智能体工具扩展点
如果插件需要新增宿主接口,例如:
- 新的链式事件
- 新的插件 API 渲染能力
- 新的工作流动作契约
- 新的智能体工具注入点
应先在 `MoviePilot` 中补齐宿主能力,再回到本仓库落插件实现。
### 5.2 与 `MoviePilot-Frontend` 的边界
插件有两种主要 UI 方式:
- Vuetify JSON 配置
- Vue 联邦远程组件
前者的宿主渲染在 `MoviePilot-Frontend` 已经实现,插件只需要返回 JSON 结构;后者需要遵守前端仓库的联邦组件暴露规范、共享依赖规范和侧栏入口规范。
如果你在本仓库写了 Vue 模式插件,需要同时关注:
- `MoviePilot-Frontend/docs/module-federation-guide.md`
- `MoviePilot-Frontend/src/utils/federationLoader.ts`
- `MoviePilot-Frontend` 中与插件页面、侧栏导航、仪表板相关的组件
## 6. 开发一个插件时的推荐流程
### 6.1 先判断插件形态
- 只是扩展后端能力、配置项简单:优先写 Vuetify JSON 模式插件
- 需要复杂交互或完整页面:使用 Vue 联邦模式
- 只是给现有插件补 V2 兼容:优先评估能否复用 `package.json + "v2": true`
- 已经与 V1 / 默认版本差异很大:直接转为 `plugins.v2/ + package.v2.json`
### 6.2 再落目录与元数据
最小步骤通常是:
1.`plugins/``plugins.v2/` 下新建目录
2.`__init__.py` 中实现插件类
3. 如有依赖,增加 `requirements.txt`
4.`package.json``package.v2.json` 中补齐元数据
5. 如有插件文档,在插件目录补充 `README.md`
6. 如有 Vue UI构建后把产物放进 `dist/assets/`
### 6.3 维护版本一致性
发布前至少核对以下三处是否一致:
- 索引里的 `version`
- 插件类里的 `plugin_version`
- `history` 中最新一条变更说明
## 7. 校验建议
这个仓库没有独立的完整测试宿主,因此校验应该尽量贴近真实运行层。
### 7.1 Python 插件代码
建议在宿主环境里做最小校验:
```bash
# 对修改过的插件文件做语法检查
python3 -m py_compile plugins.v2/myplugin/__init__.py
# 或者对整个插件目录做批量编译检查
python3 -m compileall plugins.v2/myplugin
# 顺手检查 diff 中是否有空白符问题
git diff --check
```
### 7.2 Vue 远程组件
如果插件使用独立的前端工程,建议至少执行:
```bash
# 类型检查
yarn typecheck
# 构建联邦产物
yarn build
```
然后再把构建产物拷贝到插件目录中的 `dist/assets/`
### 7.3 宿主联调
以下场景必须回到宿主仓库验证:
- `get_api()` 是否真正注册成功
- `get_service()` 是否出现在服务列表
- `get_dashboard()` / `get_dashboard_meta()` 是否正常显示
- `get_render_mode() == "vue"` 的远程组件是否能成功加载
- `get_sidebar_nav()` 是否正确出现在前端侧栏
## 8. 发布流程
本仓库的自动发布逻辑位于 `.github/workflows/release.yml`,当前规则如下:
- 只有当 `package.json``package.v2.json` 发生变更时,工作流才会触发
- 只有索引条目中声明了 `"release": true` 的插件会参与自动打包
- 工作流会尝试在 `plugins/<plugin_id_lower>``plugins.v2/<plugin_id_lower>` 中寻找插件目录
- Release Tag 格式为 `插件ID_v插件版本号`
- 压缩包文件名格式为 `插件目录小写_v插件版本号.zip`
- 若插件目录自上一个 Tag 以来没有变化,则会跳过打包
- 若同名 Release / Tag 已存在,工作流会先删除旧对象再重新创建
这意味着发布一个可下载压缩包的插件时,最少要确认:
1. 插件目录存在且名称正确
2. 索引条目中已声明 `"release": true`
3. 索引版本号与代码版本号一致
4. 目标目录自上一个同插件 Tag 以来确实有代码变化
## 9. 文档维护建议
如果一次改动同时涉及:
- 插件能力扩展点变更
- 宿主后端新增接口或新契约
- 前端新增加载规则或侧栏行为
应同步更新对应仓库文档,不要只改本仓库 README。
推荐文档分工:
- 本仓库 `README.md`:总览与主入口
- 本仓库 `docs/FAQ.md`FAQ 索引与场景入口
- 本仓库 `docs/Repository_Guide.md`:仓库维护与发布规则
- 本仓库 `docs/V2_Plugin_Development.md`V2 插件开发主文档
- 前端仓库 `docs/module-federation-guide.md`Vue 联邦远程组件开发规范
## 10. 开始之前先读哪一份
- 想知道“这个仓库该怎么维护、改哪个文件、怎么发布”:看本文档
- 想直接开发一个 V2 插件:看 `docs/V2_Plugin_Development.md`
- 想做 Vue 远程组件或侧栏全页:看前端仓库模块联邦文档
- 想按功能场景抄现成模式:看 `docs/FAQ.md``docs/faq/` 下的独立 FAQ 文档