mirror of
https://github.com/jxxghp/MoviePilot.git
synced 2026-06-15 04:32:09 +08:00
docs: update moviepilot plugin creation skill
This commit is contained in:
@@ -1,14 +1,16 @@
|
||||
---
|
||||
name: create-moviepilot-plugin
|
||||
version: 1
|
||||
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, forms, pages, dashboards, commands, services, workflow
|
||||
actions, agent tools, and local install/reload flows. Also use for Chinese
|
||||
requests mentioning 编写插件、本地插件源、插件开发、V2插件、插件市场、本地安装插件、插件热加载.
|
||||
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
|
||||
---
|
||||
|
||||
@@ -22,21 +24,45 @@ a local plugin source and installed into the running MoviePilot instance.
|
||||
- Host plugin contract: `app/plugins/__init__.py`, especially `_PluginBase`.
|
||||
- Host plugin discovery, local source sync, install, reload: `app/core/plugin.py`
|
||||
and `app/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-Plugins` uses `plugins.v2/` with
|
||||
`package.v2.json` for V2 plugins; legacy or cross-generation entries may use
|
||||
`plugins/` with `package.json`.
|
||||
- When working in or from `MoviePilot-Plugins`, read its `README.md`,
|
||||
`docs/Repository_Guide.md`, and `docs/V2_Plugin_Development.md`. For
|
||||
scenario-specific extensions, read the matching `docs/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`, and
|
||||
`MoviePilot-Frontend/src/pages/plugin-app.vue`.
|
||||
- Repository boundaries: `MoviePilot` owns runtime loading, API registration,
|
||||
events, services, data, and permissions; `MoviePilot-Frontend` owns plugin UI
|
||||
rendering, federation loading, and sidebar pages; `MoviePilot-Plugins` owns
|
||||
plugin source, icons, package indexes, and release metadata.
|
||||
|
||||
## Pre-Flight
|
||||
|
||||
1. Understand the user request: plugin purpose, trigger mode, configuration,
|
||||
output UI, whether it needs a scheduler, API, command, workflow action, or
|
||||
agent tool.
|
||||
2. Inspect existing plugins before creating a new one:
|
||||
2. 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.
|
||||
3. Inspect existing plugins before creating a new one:
|
||||
- Local runtime examples: `app/plugins/<plugin>/__init__.py`
|
||||
- Market/local source candidates: use `query_market_plugins` when the
|
||||
running instance is available.
|
||||
3. Determine the target source path:
|
||||
- For Vue federation examples, prefer current compliant plugins such as
|
||||
`MoviePilot-Plugins/plugins.v2/agenttokens/` and the frontend example
|
||||
`MoviePilot-Frontend/examples/plugin-component/`.
|
||||
4. Determine the target source path:
|
||||
- Query `PLUGIN_LOCAL_REPO_PATHS` with `query_system_settings` when possible.
|
||||
- If exactly one local plugin repository is configured, prefer that path.
|
||||
- If several are configured, choose the one the user named; otherwise ask
|
||||
@@ -47,11 +73,41 @@ a local plugin source and installed into the running MoviePilot instance.
|
||||
plugin source loader. Create that source directory and write the plugin
|
||||
under it; do not write new plugin source directly into `app/plugins/`
|
||||
unless the user explicitly asks for a runtime-only experiment.
|
||||
4. Choose the plugin ID:
|
||||
5. 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__`.
|
||||
|
||||
## 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:
|
||||
|
||||
```text
|
||||
这个插件 UI 用哪种方式实现?
|
||||
1. JSON 配置:后端返回 Vuetify JSON,适合普通配置表单、简单详情页和轻量仪表板。
|
||||
2. 联邦 UI:独立 Vue 远程组件,适合复杂交互、自定义布局、侧栏全页或多页面。
|
||||
```
|
||||
|
||||
Selection rules:
|
||||
|
||||
- **JSON config / Vuetify JSON**: implement `get_form()`, `get_page()`, and
|
||||
`get_dashboard()` with JSON component schemas. No frontend build or
|
||||
`dist/assets/remoteEntry.js` is needed.
|
||||
- **Federation UI / Vue remote component**: implement `get_render_mode()`,
|
||||
expose Vue components through Vite federation, build frontend assets into the
|
||||
plugin directory, and use `get_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
|
||||
|
||||
@@ -67,6 +123,25 @@ Default to V2 layout for new local plugins:
|
||||
└── ... # helper modules, schemas, static assets
|
||||
```
|
||||
|
||||
For a Vue federation plugin, the runtime requirement is the built remote assets
|
||||
under the plugin directory:
|
||||
|
||||
```text
|
||||
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:
|
||||
|
||||
```text
|
||||
@@ -107,21 +182,29 @@ Rules:
|
||||
|
||||
- The package object key must match the plugin class name.
|
||||
- `version` must match `plugin_version`.
|
||||
- `name`, `description`, `icon`, `author`, and `level` should match the plugin
|
||||
class attributes when those attributes exist.
|
||||
- `name`, `description`, `icon`, `author`, `labels`, and `level` should match
|
||||
the plugin class attributes when those attributes exist (`plugin_name`,
|
||||
`plugin_desc`, `plugin_icon`, `plugin_author`, `plugin_label`, `auth_level`).
|
||||
- `history` should record user-readable changes for each published version.
|
||||
- Use `system_version` when the plugin depends on a host capability introduced
|
||||
in a specific MoviePilot version.
|
||||
in a specific MoviePilot version, including new backend APIs, helpers, events,
|
||||
Vue federation behavior, sidebar nav, dashboard behavior, or agent tools.
|
||||
- Use `"release": true` only 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.txt` changes, 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 public classes,
|
||||
public methods, and public functions need Chinese docstrings.
|
||||
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.
|
||||
|
||||
```python
|
||||
from typing import Any, Dict, List, Optional, Tuple
|
||||
@@ -224,7 +307,8 @@ Use only the extension points the requested plugin actually needs:
|
||||
- Notification: use `post_message()` instead of directly calling message
|
||||
modules.
|
||||
- APIs: return route definitions from `get_api()`; default auth is `apikey`
|
||||
when `auth` is omitted.
|
||||
when `auth` is omitted. Vue component APIs should normally use
|
||||
`auth: "bear"` and be called through the `api` prop 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
|
||||
@@ -239,6 +323,118 @@ Use only the extension points the requested plugin actually needs:
|
||||
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:
|
||||
|
||||
```python
|
||||
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:
|
||||
|
||||
```python
|
||||
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"`.
|
||||
- `section` must be one of `start`, `discovery`, `subscribe`, `organize`,
|
||||
`system`; invalid values fall back to `system`.
|
||||
- `permission` may be `subscribe`, `discovery`, `search`, `manage`, or `admin`;
|
||||
invalid values are ignored.
|
||||
- `nav_key` defaults to `main` and must not contain `/`, `?`, `#`, or spaces.
|
||||
- Multiple sidebar entries are allowed; each entry needs a stable `nav_key`.
|
||||
|
||||
Frontend federation requirements:
|
||||
|
||||
```js
|
||||
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.target` to `esnext` because federation uses top-level await.
|
||||
- Use `cssCodeSplit: true` and scoped/component-local styles where possible.
|
||||
- Build with the frontend project's documented command, then keep `remoteEntry.js`
|
||||
and every JS/CSS/asset file it references under `dist/assets`.
|
||||
- Do not add frontend runtime dependencies to the plugin Python
|
||||
`requirements.txt`; keep frontend dependencies in the frontend build project.
|
||||
|
||||
Component contracts:
|
||||
|
||||
- `Page` renders the plugin detail dialog and may emit `action`, `switch`, and
|
||||
`close`.
|
||||
- `Config` renders plugin settings, receives `initialConfig` and `api`, and
|
||||
emits `save`, `close`, and `switch`.
|
||||
- `Dashboard` receives `config` and `allowRefresh`.
|
||||
- `AppPage` renders the main-layout sidebar page and receives `api`, `pluginId`,
|
||||
and `navKey`.
|
||||
- For sidebar `nav_key=main`, the frontend loads `./AppPage` then `./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 `AppPage` may branch on `navKey`, or separate
|
||||
`AppPage{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_TOKEN` into Vue components for browser-side calls.
|
||||
|
||||
## Local Install And Reload
|
||||
|
||||
1. After writing files in a configured local plugin repository, call
|
||||
@@ -258,6 +454,17 @@ Use only the extension points the requested plugin actually needs:
|
||||
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 that
|
||||
`dist/assets/remoteEntry.js` exists.
|
||||
- For sidebar plugins, confirm the plugin is enabled, `get_state()` returns
|
||||
`True`, `get_sidebar_nav()` returns valid items, and matching `AppPage`
|
||||
exposes exist for all non-main `nav_key` values or a generic `AppPage` handles
|
||||
them.
|
||||
- Confirm frontend-facing API routes use `auth: "bear"` and browser code calls
|
||||
them through the provided `api` prop.
|
||||
- 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
|
||||
@@ -266,6 +473,27 @@ Use only the extension points the requested plugin actually needs:
|
||||
- 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=moviepilot` should include the plugin with a
|
||||
URL ending in `/plugin/file/<plugin_id_lower>/<dist_path>/remoteEntry.js`.
|
||||
- `GET /api/v1/plugin/sidebar_nav` should include sidebar entries for enabled
|
||||
Vue plugins with valid `nav_key`, `section`, and `permission`.
|
||||
- If the console says `Module name 'vue' does not resolve to a valid URL`, check
|
||||
the federation `shared` config and use `requiredVersion: false`.
|
||||
- If the console says top-level await is unavailable, set `build.target` to
|
||||
`esnext`.
|
||||
- If dynamic import fails, check the remote file request status, the computed
|
||||
`remoteEntry.js` path, 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_key` and fallbacks (`AppPage{PascalCase}` -> `AppPage` -> `Page`).
|
||||
|
||||
## Final Report
|
||||
|
||||
@@ -273,5 +501,7 @@ Report:
|
||||
|
||||
- Plugin ID, source path, and runtime path if installed.
|
||||
- Package file changed (`package.v2.json` or `package.json`).
|
||||
- UI mode used (`vuetify` JSON or `vue` federation), 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.
|
||||
|
||||
Reference in New Issue
Block a user