Files
archived-MoviePilot-Plugins/docs/Repository_Guide.md
2026-04-20 21:32:50 +08:00

9.2 KiB
Raw Blame History

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. 目录结构

本仓库当前采用如下结构:

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需要在条目上声明

{
  "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:用户可见级别
  • history:更新日志
  • release:是否使用 GitHub Release 压缩包发布
  • v2:默认索引中的插件是否兼容 V2

这些字段是“插件市场展示元数据”,而不是运行时唯一真相。真正加载后的插件类仍然需要在代码里声明自己的 plugin_nameplugin_descplugin_version 等属性。两者必须同步。

4. 版本选择与加载规则

MoviePilot 当前的插件版本选择逻辑可以概括为:

  1. 先确定当前宿主版本标识,例如 v2
  2. 优先检查 package.v2.json 中是否存在该插件
  3. 若不存在,再检查 package.json
  4. 只有当 package.json 中对应条目显式声明 "v2": true 时,才会作为 V2 兼容插件继续使用

这意味着:

  • 同一个插件若在 package.v2.json 中已有专用实现,就不要再依赖 package.json 中的兼容声明做“隐式覆盖”。
  • 新写的 V2 专用插件,优先放 plugins.v2/,并把元数据写入 package.v2.json
  • 真正跨版本共用一套实现时,再使用 package.json + "v2": true 的方式。

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.jsonpackage.v2.json 中补齐元数据
  5. 如有插件文档,在插件目录补充 README.md
  6. 如有 Vue UI构建后把产物放进 dist/assets/

6.3 维护版本一致性

发布前至少核对以下三处是否一致:

  • 索引里的 version
  • 插件类里的 plugin_version
  • history 中最新一条变更说明

7. 校验建议

这个仓库没有独立的完整测试宿主,因此校验应该尽量贴近真实运行层。

7.1 Python 插件代码

建议在宿主环境里做最小校验:

# 对修改过的插件文件做语法检查
python3 -m py_compile plugins.v2/myplugin/__init__.py

# 或者对整个插件目录做批量编译检查
python3 -m compileall plugins.v2/myplugin

# 顺手检查 diff 中是否有空白符问题
git diff --check

7.2 Vue 远程组件

如果插件使用独立的前端工程,建议至少执行:

# 类型检查
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.jsonpackage.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:总览与 FAQ
  • 本仓库 docs/Repository_Guide.md:仓库维护与发布规则
  • 本仓库 docs/V2_Plugin_Development.mdV2 插件开发主文档
  • 前端仓库 docs/module-federation-guide.mdVue 联邦远程组件开发规范

10. 开始之前先读哪一份

  • 想知道“这个仓库该怎么维护、改哪个文件、怎么发布”:看本文档
  • 想直接开发一个 V2 插件:看 docs/V2_Plugin_Development.md
  • 想做 Vue 远程组件或侧栏全页:看前端仓库模块联邦文档
  • 想按功能场景抄现成模式:回到根目录 README.md 的 FAQ