name: create-moviepilot-plugin version: 2 description: >- Use this skill when the user asks to create, modify, debug, validate, or scaffold a MoviePilot local plugin. Covers MoviePilot V2 plugin development, _PluginBase implementations, package.v2.json/package.json market metadata, plugins.v2/plugins source layout, PLUGIN_LOCAL_REPO_PATHS local plugin sources, plugin APIs, Vuetify JSON forms/pages/dashboards, Vue module federation remote components, get_render_mode, get_sidebar_nav, plugin sidebar pages, commands, services, workflow actions, agent tools, and local install/reload flows. Also use for Chinese requests mentioning 编写插件、本地插件源, 插件开发, V2插件, 插件市场, 本地安装插件, 插件热加载, 前端联邦, 侧栏入口, Vue插件页面. allowed-tools: list_directory read_file write_file edit_file execute_command query_system_settings update_system_settings query_market_plugins install_plugin reload_plugin query_installed_plugins
Create MoviePilot Plugin
Use this skill to build or revise MoviePilot plugins that can be developed from a local plugin source and installed into the running MoviePilot instance.
Ground Truth
- Host plugin contract:
app/plugins/__init__.py, especially_PluginBase. - Host plugin discovery, local source sync, install, reload:
app/core/plugin.pyandapp/helper/plugin.py. - Host plugin endpoints, API auth, static files, remotes, and sidebar nav:
app/api/endpoints/plugin.py. - Local development note:
docs/development-setup.md. - Plugin repository conventions:
MoviePilot-Pluginsusesplugins.v2/withpackage.v2.jsonfor V2 plugins; legacy or cross-generation entries may useplugins/withpackage.json. - When working in or from
MoviePilot-Plugins, read itsREADME.md,docs/Repository_Guide.md, anddocs/V2_Plugin_Development.md. For scenario-specific extensions, read the matchingdocs/faq/*.md. - When the plugin uses Vue federation, also read
MoviePilot-Frontend/docs/module-federation-guide.md,MoviePilot-Frontend/docs/federation-troubleshooting.md,MoviePilot-Frontend/src/utils/federationLoader.ts, andMoviePilot-Frontend/src/pages/plugin-app.vue. - Repository boundaries:
MoviePilotowns runtime loading, API registration, events, services, data, and permissions;MoviePilot-Frontendowns plugin UI rendering, federation loading, and sidebar pages;MoviePilot-Pluginsowns plugin source, icons, package indexes, and release metadata.
Pre-Flight
- Understand the user request: plugin purpose, trigger mode, configuration, output UI, whether it needs a scheduler, API, command, workflow action, or agent tool.
- Run the UI Mode Selection Gate before writing any UI code.
- If the user already explicitly chose JSON config/Vuetify JSON or Vue federation, follow that choice.
- If the plugin has any UI surface and the user has not chosen a mode, ask them to choose between the two modes below and wait for the answer before implementing UI files or schemas.
- Do not silently default to either mode just because one seems easier.
- Inspect existing plugins before creating a new one:
- Local runtime examples:
app/plugins/<plugin>/__init__.py - Market/local source candidates: use
query_market_pluginswhen the running instance is available. - For Vue federation examples, prefer current compliant plugins such as
MoviePilot-Plugins/plugins.v2/agenttokens/and the frontend exampleMoviePilot-Frontend/examples/plugin-component/.
- Local runtime examples:
- Determine the target source path:
- Query
PLUGIN_LOCAL_REPO_PATHSwithquery_system_settingswhen possible. - If exactly one local plugin repository is configured, prefer that path.
- If several are configured, choose the one the user named; otherwise ask which repository to use.
- If none is configured, set it before writing plugin code:
update_system_settings(setting_key="PLUGIN_LOCAL_REPO_PATHS", value="local-plugins", operation="replace").local-pluginsis resolved relative to the MoviePilot root by the local plugin source loader. Create that source directory and write the plugin under it; do not write new plugin source directly intoapp/plugins/unless the user explicitly asks for a runtime-only experiment.
- Query
- Choose the plugin ID:
- Class name is the plugin ID, for example
MyNotifier. - Directory name is the class name lowercased, for example
mynotifier. - Avoid collisions with installed or market plugins unless the user is explicitly modifying that plugin.
- Do not hardcode the original plugin ID for data/config namespaces when the
plugin may support clones; use
self.__class__.__name__.
- Class name is the plugin ID, for example
UI Mode Selection Gate
MoviePilot plugin UI has exactly two implementation modes. Make the user choose one whenever the request includes configuration, detail pages, dashboards, sidebar pages, or any other plugin UI and the mode is not already explicit.
Ask a concise question like:
这个插件 UI 用哪种方式实现?
1. JSON 配置:后端返回 Vuetify JSON,适合普通配置表单、简单详情页和轻量仪表板。
2. 联邦 UI:独立 Vue 远程组件,适合复杂交互、自定义布局、侧栏全页或多页面。
Selection rules:
- JSON config / Vuetify JSON: implement
get_form(),get_page(), andget_dashboard()with JSON component schemas. No frontend build ordist/assets/remoteEntry.jsis needed. - Federation UI / Vue remote component: implement
get_render_mode(), expose Vue components through Vite federation, build frontend assets into the plugin directory, and useget_sidebar_nav()only when a sidebar page is requested. - If the plugin truly has no user-facing UI, state that no UI mode is needed and implement only the backend extension points the request requires.
- Backend-only work may proceed while waiting only if it cannot constrain or preclude either UI mode.
Local Source Layout
Default to V2 layout for new local plugins:
<local-plugin-repo>/
├── package.v2.json
└── plugins.v2/
└── <plugin_id_lower>/
├── __init__.py
├── requirements.txt # only when extra runtime dependencies are necessary
└── ... # helper modules, schemas, static assets
For a Vue federation plugin, the runtime requirement is the built remote assets under the plugin directory:
plugins.v2/<plugin_id_lower>/
├── __init__.py
├── dist/
│ └── assets/
│ ├── remoteEntry.js
│ └── ... # JS/CSS/assets referenced by remoteEntry
├── package.json # optional frontend build project metadata
├── vite.config.js # optional frontend build config
└── src/ # optional source, not required at runtime
Do not rely on frontend source files at runtime. If the source is kept in the
plugin repository for maintainability, still build and ship the dist/assets
files required by remoteEntry.js.
Only use the legacy layout when the user explicitly needs it:
<local-plugin-repo>/
├── package.json
└── plugins/
└── <plugin_id_lower>/
└── __init__.py
For legacy package.json entries that should work on V2, include "v2": true.
For V2-first work, prefer package.v2.json and plugins.v2/.
Package Metadata
Add or update the package entry for the plugin ID. Keep the package version and
the class plugin_version synchronized.
{
"MyNotifier": {
"name": "通知示例",
"description": "根据用户配置发送示例通知。",
"labels": "消息通知",
"version": "1.0.0",
"icon": "mynotifier.png",
"author": "local",
"level": 1,
"system_version": ">=2.12.0",
"history": {
"v1.0.0": "初始版本"
}
}
}
Rules:
- The package object key must match the plugin class name.
versionmust matchplugin_version.name,description,icon,author,labels, andlevelshould match the plugin class attributes when those attributes exist (plugin_name,plugin_desc,plugin_icon,plugin_author,plugin_label,auth_level).historyshould record user-readable changes for each published version.- Use
system_versionwhen the plugin depends on a host capability introduced in a specific MoviePilot version, including new backend APIs, helpers, events, Vue federation behavior, sidebar nav, dashboard behavior, or agent tools. - Use
"release": trueonly when the plugin is intentionally distributed by a GitHub Release archive. - New plugin entries should usually be appended to the package index so they appear as newer marketplace items.
- Do not add dependencies unless they are actually required. If
requirements.txtchanges, the user must reinstall the plugin; hot reload is not enough to install dependencies. - Plugin dependencies are installed into the shared MoviePilot Python environment. Do not pin or downgrade packages already provided by MoviePilot unless the user has explicitly accepted the compatibility risk.
Implementation Skeleton
Implement all abstract methods from _PluginBase. All new functions and
methods need Chinese docstrings; public classes, public methods, and public
functions are a hard review gate.
from typing import Any, Dict, List, Optional, Tuple
from app.plugins import _PluginBase
class MyNotifier(_PluginBase):
"""通知示例插件。"""
plugin_name = "通知示例"
plugin_desc = "根据用户配置发送示例通知。"
plugin_icon = "mynotifier.png"
plugin_version = "1.0.0"
plugin_label = "消息通知"
plugin_author = "local"
plugin_config_prefix = "mynotifier_"
plugin_order = 100
auth_level = 1
_enabled = False
_message = ""
def init_plugin(self, config: dict = None) -> None:
"""根据插件配置初始化运行状态。"""
self.stop_service()
self._enabled = False
self._message = ""
if not config:
return
self._enabled = bool(config.get("enabled"))
self._message = str(config.get("message") or "")
def get_state(self) -> bool:
"""获取插件启用状态。"""
return self._enabled
@staticmethod
def get_command() -> List[Dict[str, Any]]:
"""返回插件远程命令列表。"""
return []
def get_api(self) -> List[Dict[str, Any]]:
"""返回插件 API 列表。"""
return []
def get_form(self) -> Tuple[Optional[List[dict]], Dict[str, Any]]:
"""返回插件配置表单与默认配置。"""
return [
{
"component": "VForm",
"content": [
{
"component": "VSwitch",
"props": {
"model": "enabled",
"label": "启用插件"
}
},
{
"component": "VTextField",
"props": {
"model": "message",
"label": "通知内容"
}
}
]
}
], {
"enabled": False,
"message": ""
}
def get_page(self) -> Optional[List[dict]]:
"""返回插件详情页面。"""
if not self._enabled:
return None
return [
{
"component": "VAlert",
"props": {
"type": "info",
"text": self._message or "插件已启用"
}
}
]
def stop_service(self) -> None:
"""停止插件后台服务并释放资源。"""
return None
Extension Points
Use only the extension points the requested plugin actually needs:
- Configuration:
get_form()returns Vuetify form schema and default data;init_plugin()reads config;update_config()persists internal changes. - Data: use
save_data(),get_data(),del_data(), andget_data_path(). - Notification: use
post_message()instead of directly calling message modules. - APIs: return route definitions from
get_api(); default auth isapikeywhenauthis omitted. Vue component APIs should normally useauth: "bear"and be called through theapiprop passed by the frontend. - Commands: return slash-command definitions from
get_command()and dispatch through MoviePilot events. - Services: return scheduler services from
get_service()and always clean them up instop_service(). - Dashboards: use
get_dashboard_meta()andget_dashboard()for homepage widgets. - Workflow actions: use
get_actions(); action functions receiveActionContentfirst and return(success, action_content). - Agent tools: use
get_agent_tools(); each tool class must inheritapp.agent.tools.base.MoviePilotTool. - Custom Vue UI: implement
get_render_mode()only when Vuetify schema cannot satisfy the request. Return("vue", "<compiled-assets-path>")and include built frontend assets in the plugin directory.
Vue Federation UI
Use Vue federation only after the Pre-Flight UI decision says JSON schema is not enough. A Vue plugin must align backend methods, built files, and federation exposes.
Backend requirements:
from typing import Any, Dict, List, Tuple
@staticmethod
def get_render_mode() -> Tuple[str, str]:
"""声明插件使用 Vue 联邦组件渲染。"""
return "vue", "dist/assets"
def get_form(self) -> Tuple[List[dict], Dict[str, Any]]:
"""Vue 模式下返回默认配置模型。"""
return [], self._current_config()
def get_page(self) -> List[dict]:
"""Vue 模式下详情页由远程 Page 组件渲染。"""
return []
When the plugin needs a main-layout sidebar page, also implement:
def get_sidebar_nav(self) -> List[Dict[str, Any]]:
"""声明插件在主界面左侧导航栏中的全页入口。"""
if not self.get_state():
return []
return [
{
"nav_key": "main",
"title": "我的插件",
"icon": "mdi-puzzle",
"section": "system",
"permission": "manage",
"order": 10,
}
]
Sidebar rules:
- Sidebar entries are only aggregated for enabled plugins whose
get_render_mode()returns"vue". sectionmust be one ofstart,discovery,subscribe,organize,system; invalid values fall back tosystem.permissionmay besubscribe,discovery,search,manage, oradmin; invalid values are ignored.nav_keydefaults tomainand must not contain/,?,#, or spaces.- Multiple sidebar entries are allowed; each entry needs a stable
nav_key.
Frontend federation requirements:
federation({
name: 'MyPlugin',
filename: 'remoteEntry.js',
exposes: {
'./Page': './src/components/Page.vue',
'./Config': './src/components/Config.vue',
'./Dashboard': './src/components/Dashboard.vue',
'./AppPage': './src/components/AppPage.vue',
'./AppPageSettings': './src/components/AppPageSettings.vue',
},
shared: {
vue: { requiredVersion: false, generate: false },
vuetify: { requiredVersion: false, generate: false, singleton: true },
'vuetify/styles': { requiredVersion: false, generate: false, singleton: true },
},
format: 'esm',
})
Build requirements:
- Set Vite
build.targettoesnextbecause federation uses top-level await. - Use
cssCodeSplit: trueand scoped/component-local styles where possible. - Build with the frontend project's documented command, then keep
remoteEntry.jsand every JS/CSS/asset file it references underdist/assets. - Do not add frontend runtime dependencies to the plugin Python
requirements.txt; keep frontend dependencies in the frontend build project.
Component contracts:
Pagerenders the plugin detail dialog and may emitaction,switch, andclose.Configrenders plugin settings, receivesinitialConfigandapi, and emitssave,close, andswitch.DashboardreceivesconfigandallowRefresh.AppPagerenders the main-layout sidebar page and receivesapi,pluginId, andnavKey.- For sidebar
nav_key=main, the frontend loads./AppPagethen./Page. - For any other
nav_key, the frontend loads./AppPage{PascalCase(nav_key)}, then./AppPage, then./Page. Examples:settings -> AppPageSettings,my_tool -> AppPageMyTool. - A single
AppPagemay branch onnavKey, or separateAppPage{PascalCase}files may be exposed for specific entries.
Vue API calls:
- Define frontend-facing plugin APIs with
auth: "bear". - Call them with the injected API object, for example
props.api.get(\plugin/${props.pluginId}/history`)`. - Do not pass
settings.API_TOKENinto Vue components for browser-side calls.
Local Install And Reload
- After writing files in a configured local plugin repository, call
query_market_plugins(query="<PluginID>", force_refresh=True)to confirm the local source is visible. - Install or reinstall with
install_plugin(plugin_id="<PluginID>", force=True). The install flow copies the source intoapp/plugins/<plugin_id_lower>/. - If
PLUGIN_AUTO_RELOADor development mode is enabled, Python source changes in an installed local plugin can auto-sync and reload. If it is not enabled, callreload_plugin(plugin_id="<PluginID>")after editing runtime files. - When
requirements.txtchanges, reinstall withforce=True; reloading alone does not install new dependencies.
Validation
- Re-read the changed files and confirm class name, directory name, package ID, and package version are consistent.
- Confirm every public class, public method, and public function has a Chinese docstring.
- Confirm every newly written function or method has a Chinese docstring, even when it is private helper code.
- For Vue federation plugins, confirm
get_render_mode()returns("vue", "dist/assets")or the actual built asset path, and thatdist/assets/remoteEntry.jsexists. - For sidebar plugins, confirm the plugin is enabled,
get_state()returnsTrue,get_sidebar_nav()returns valid items, and matchingAppPageexposes exist for all non-mainnav_keyvalues or a genericAppPagehandles them. - Confirm frontend-facing API routes use
auth: "bear"and browser code calls them through the providedapiprop. - Keep external HTTP calls behind MoviePilot utilities and avoid real network calls in tests.
- If the plugin has non-trivial logic, add or update pytest-native tests. Plugin
repositories can use
app.testing.bootstrap.prepare_v2_backend()to prepare a temporary MoviePilot backend and inject<repo>/plugins.v2intosys.path. - Run the narrowest allowed validation for the touched area. In this repository,
follow
docs/rules/03-commands.md; for plugin-only repositories, follow their own documented validation commands. - For plugin repository Python changes, use the host Python environment when possible and run at least syntax compilation for touched plugin files.
- For Vue federation changes, run the frontend project's documented typecheck and build commands when available, then verify the built assets were copied to the plugin directory.
Vue Federation Troubleshooting
GET /api/v1/plugin/remotes?token=moviepilotshould include the plugin with a URL ending in/plugin/file/<plugin_id_lower>/<dist_path>/remoteEntry.js.GET /api/v1/plugin/sidebar_navshould include sidebar entries for enabled Vue plugins with validnav_key,section, andpermission.- If the console says
Module name 'vue' does not resolve to a valid URL, check the federationsharedconfig and userequiredVersion: false. - If the console says top-level await is unavailable, set
build.targettoesnext. - If dynamic import fails, check the remote file request status, the computed
remoteEntry.jspath, and whether the installed runtime plugin directory actually contains the built assets. - If a sidebar page is blank, check the expose name resolution for the current
nav_keyand fallbacks (AppPage{PascalCase}->AppPage->Page).
Final Report
Report:
- Plugin ID, source path, and runtime path if installed.
- Package file changed (
package.v2.jsonorpackage.json). - UI mode used (
vuetifyJSON orvuefederation), and for Vue plugins the exposed components and built asset path. - Whether the plugin was installed or reloaded.
- Validation commands run, or why validation was not run.