提交项目
我们欢迎您提交自己的项目,请加入TG群联系管理员了解相关事宜
diff --git a/404.html b/404.html deleted file mode 100644 index b7073067..00000000 --- a/404.html +++ /dev/null @@ -1,26 +0,0 @@ - - -
- - -贡献者:
项目地址:nezha_api_tgbot
机器人可以通过API向面板请求服务器状态信息,得到信息后发送给用户
你可以搭建此机器人来方便地查看指定服务器的当前状态且不需要打开面板
当前版本:V1.0
贡献者:
使用iPhone或iPad扫描以下二维码,获取快捷指令
MacOS用户请访问这里,获取快捷指令
面板URL
、API Token
、服务器ID
WARNING
每个快捷指令只能监控一台服务器,如需要监控多个服务器,请多次复制此快捷指令并分别配置,然后分别修改快捷指令名称,如;一号服务器状态、二号服务器状态等
贡献者:
项目地址:nezha_telegram_bot
开源Telegram机器人项目,可以基于API实时查询哪吒面板的服务器信息。
命令 | 功能 | 仅私聊 |
---|---|---|
start | 开始使用键盘主菜单 | ✔️ |
help | 帮助列表 | ❌ |
add | 添加面板链接和token | ✔️ |
url | 添加面板链接 | ✔️ |
token | 添加面板token | ✔️ |
info | 获取保存的面板链接和token | ✔️ |
delete | 删除保存的面板链接和token | ✔️ |
id | 命令后面添加整数id,来进行单个服务器信息查询(私聊带刷新按钮,群聊不带) | ❌ |
all | 查询所有服务器的统计信息 | ❌ |
search | 在服务器名字中搜索关键字(支持多个,用空格分开) | ❌ |
还在因为自己的小鸡配置太差被朋友嘲笑吗?
还在想参加以针会友活动却因为小鸡不够排面而觉得羞耻吗?
还在因为在哪吒TG群里机器人排名太低而觉得自己低人一等吗?
今天开始!找回自信!
来自南京的名老中医 dysf888 独家秘方!让你找回男人本色,激情畅享!
安装 Fake Agent,可随意修改服务器监测数据的倍数上传到 Dashboard,让你的小鸡 脱!胎!换!骨! 让你在 MJJ 面前硬起来!
哪吒监控创始人奶爸也在用:
男人用了都说好:
贡献者:
项目地址:Argo-Nezha-Service-Container
网络
选项将 gRPC
开关打开面板域名加上 https://
开头,回调地址再加上 /oauth2/callback
结尾
Argo 隧道认证方式有 json 和 token,使用两个方式其中之一。推荐前者,理由脚本会处理好所有的 Argo 隧道参数和路径,后者需要到 Cloudflare 官网手动设置,容易出错。
proto.NezhaService
镜像 fscarmen/argo-nezha:latest
, 支持 amd64 和 arm64 架构
用到的变量
变量名 | 是否必须 | 备注 |
---|---|---|
GH_USER | 是 | github 的用户名,用于面板管理授权 |
GH_CLIENTID | 是 | 在 github 上申请 |
GH_CLIENTSECRET | 是 | 在 github 上申请 |
GH_BACKUP_USER | 否 | 在 github 上备份哪吒服务端数据库的 github 用户名,不填则与面板管理授权的账户 GH_USER 一致 |
GH_REPO | 否 | 在 github 上备份哪吒服务端数据库文件的 github 库 |
GH_EMAIL | 否 | github 的邮箱,用于备份的 git 推送到远程库 |
GH_PAT | 否 | github 的 PAT |
ARGO_AUTH | 是 | Json: 从 https://fscarmen.cloudflare.now.cc 获取的 Argo Json Token: 从 Cloudflare 官网获取 |
ARGO_DOMAIN | 是 | Argo 域名 |
Koyeb
docker run -dit \\
- --name nezha_dashboard \\
- --pull always \\
- --restart always \\
- -e GH_USER=<填 github 用户名> \\
- -e GH_EMAIL=<填 github 邮箱> \\
- -e GH_PAT=<填获取的> \\
- -e GH_REPO=<填自定义的> \\
- -e GH_CLIENTID=<填获取的> \\
- -e GH_CLIENTSECRET=<填获取的> \\
- -e ARGO_AUTH='<填获取的 Argo json 或者 token>' \\
- -e ARGO_DOMAIN=<填自定义的> \\
- -e GH_BACKUP_USER=<如与 GH_USER 一致,可以不要该环境变量> \\
- fscarmen/argo-nezha
version: '3.8'
-services:
- argo-nezha:
- image: fscarmen/argo-nezha
- pull: always
- container_name: nezha_dashboard
- restart: always
- environment:
- - GH_USER=<填 github 用户名>
- - GH_EMAIL=<<填 github 邮箱>
- - GH_PAT=<填获取的>
- - GH_REPO=<填自定义的>
- - GH_CLIENTID=<填获取的>
- - GH_CLIENTSECRET=<填获取的>
- - ARGO_AUTH='<填获取的 Argo json 或者 token>'
- - ARGO_DOMAIN=<填自定义的>
- - GH_BACKUP_USER=<如与 GH_USER 一致,可以不要该环境变量>
bash <(wget -qO- https://raw.githubusercontent.com/fscarmen2/Argo-Nezha-Service-Container/main/dashboard.sh)
通过gRPC传输,无需额外配置。使用面板给到的安装方式,举例
curl -L https://raw.githubusercontent.com/nezhahq/scripts/main/install.sh -o nezha.sh && chmod +x nezha.sh && sudo ./nezha.sh install_agent nezha.seales.nom.za 443 eAxO9IF519fKFODlW0 --tls
<file path>/cloudflared access ssh --hostname nezha.seales.nom.za/<GH_CLIENTID>
方法一: 把 Github 备份库里的 README.md
文件内容改为 backup
方法二: ssh 进去后,容器版本运行 /dashboard/backup.sh
; 非容器版本运行 /opt/nezha/dashboard/backup.sh
README.md
,定时服务会每分钟检测更新,并把上次同步的文件名记录在本地 /dbfile
处以与在线的文件内容作比对下图为以还原文件名为 dashboard-2023-04-23-13:08:37.tar.gz
作示例
bash /dashboard/restore.sh <文件名>
/dashboard
文件夹,压缩备份为 dashboard.tar.gz
文件tar czvf dashboard.tar.gz /dashboard
dashboard.tar.gz
/dashboard/
-|-- app # 哪吒面板主程序
-|-- argo.json # Argo 隧道 json 文件,记录着使用隧道的信息
-|-- argo.yml # Argo 隧道 yml 文件,用于在一同隧道下,根据不同域名来分流 web, gRPC 和 ssh 协议的作用
-|-- backup.sh # 备份数据脚本
-|-- restore.sh # 还原备份脚本
-|-- dbfile # 记录最新的还原或备份文件名
-|-- resource # 面板主题、语言和旗帜等资料的文件夹
-|-- data
-| |-- config.yaml # 哪吒面板的配置,如 Github OAuth2 / gRPC 域名 / 端口 / 是否启用 TLS 等信息
-| \`-- sqlite.db # SQLite 数据库文件,记录着面板设置的所有 severs 和 cron 等信息
-|-- entrypoint.sh # 主脚本,容器运行后执行
-|-- nezha.csr # SSL/TLS 证书签名请求
-|-- nezha.key # SSL/TLS 证书的私钥信息
-|-- nezha.pem # SSL/TLS 证书文件
-|-- cloudflared # Cloudflare Argo 隧道主程序
-|-- grpcwebproxy # gRPC 反代主程序
-\`-- nezha-agent # 哪吒客户端,用于监控本地 localhost
贡献者:
INFO
我们欢迎对项目的任何贡献,并将积极合并你的修改至下一次的Release,但我们可能会对不符合App Review Guidelines的内容(或潜在内容)进行一些修改。
WARNING
App Store版本的Nezha Mobile由于ICP备案原因不在中国大陆提供服务。
server.hidandelion.com
,不要添加协议头和末尾斜杠。API Token可通过Dashboard管理后台获取。 在Springboard添加小组件,长按后点击编辑小组件。
在随后的页面中选择一台你需要监控的服务器。你可以为多个小组件配置不同的服务器。
贡献者:
Broker for Nezha 是一个 Nezha Agent 的修改版本,其通过分离 Agent 的数据收集和 gRPC 上报功能,可使小型设备接入 Nezha Dashboard。
项目地址:https://codeberg.org/uubulb/broker
Broker for Nezha 需要与特定的数据收集软件一同使用。目前提供两种数据获取方式:
如需获取现有的数据收集软件,可以参考项目 README。
因数据获取方式缺乏安全性,建议仅在信任的网络环境使用。
如需自行编写数据收集端,除实现基本 HTTP 服务器或 TCP 客户端外,还需要使用特定的数据格式才可被 Broker 正常接收。具体请参考:数据类型
-1||Y(E,"default"))&&l.push(h)}}}const u=[o,l];return Z(e)&&r.set(e,u),u}function ts(e){return e[0]!=="$"&&!yt(e)}function ns(e){return e===null?"null":typeof e=="function"?e.name||"":typeof e=="object"&&e.constructor&&e.constructor.name||""}function rs(e,t){return ns(e)===ns(t)}function ss(e,t){return B(t)?t.findIndex(n=>rs(n,e)):W(t)&&rs(t,e)?0:-1}const Si=e=>e[0]==="_"||e==="$stable",Lr=e=>B(e)?e.map(Ae):[Ae(e)],$l=(e,t,n)=>{if(t._n)return t;const r=fl((...s)=>Lr(t(...s)),n);return r._c=!1,r},xi=(e,t,n)=>{const r=e._ctx;for(const s in e){if(Si(s))continue;const i=e[s];if(W(i))t[s]=$l(s,i,r);else if(i!=null){const o=Lr(i);t[s]=()=>o}}},Ti=(e,t)=>{const n=Lr(t);e.slots.default=()=>n},Hl=(e,t)=>{const n=e.slots=bi();if(e.vnode.shapeFlag&32){const r=t._;r?(le(n,t),Vs(n,"_",r,!0)):xi(t,n)}else t&&Ti(e,t)},jl=(e,t,n)=>{const{vnode:r,slots:s}=e;let i=!0,o=te;if(r.shapeFlag&32){const l=t._;l?n&&l===1?i=!1:(le(s,t),!n&&l===1&&delete s._):(i=!t.$stable,xi(t,s)),o=t}else t&&(Ti(e,t),o={default:1});if(i)for(const l in s)!Si(l)&&o[l]==null&&delete s[l]};function gn(e,t,n,r,s=!1){if(B(e)){e.forEach((y,E)=>gn(y,t&&(B(t)?t[E]:t),n,r,s));return}if(vt(r)&&!s)return;const i=r.shapeFlag&4?Pr(r.component):r.el,o=s?null:i,{i:l,r:c}=e,u=t&&t.r,d=l.refs===te?l.refs={}:l.refs,h=l.setupState;if(u!=null&&u!==c&&(ie(u)?(d[u]=null,Y(h,u)&&(h[u]=null)):pe(u)&&(u.value=null)),W(c))Xe(c,l,12,[o,d]);else{const y=ie(c),E=pe(c);if(y||E){const A=()=>{if(e.f){const P=y?Y(h,c)?h[c]:d[c]:c.value;s?B(P)&&gr(P,i):B(P)?P.includes(i)||P.push(i):y?(d[c]=[i],Y(h,c)&&(h[c]=d[c])):(c.value=[i],e.k&&(d[e.k]=c.value))}else y?(d[c]=o,Y(h,c)&&(h[c]=o)):E&&(c.value=o,e.k&&(d[e.k]=o))};o?(A.id=-1,ye(A,n)):A()}}}let is=!1;const pt=()=>{is||(console.error("Hydration completed but contains mismatches."),is=!0)},Vl=e=>e.namespaceURI.includes("svg")&&e.tagName!=="foreignObject",Dl=e=>e.namespaceURI.includes("MathML"),Jt=e=>{if(Vl(e))return"svg";if(Dl(e))return"mathml"},Qt=e=>e.nodeType===8;function Ul(e){const{mt:t,p:n,o:{patchProp:r,createText:s,nextSibling:i,parentNode:o,remove:l,insert:c,createComment:u}}=e,d=(g,m)=>{if(!m.hasChildNodes()){n(null,g,m),dn(),m._vnode=g;return}h(m.firstChild,g,null,null,null),dn(),m._vnode=g},h=(g,m,M,O,K,U=!1)=>{U=U||!!m.dynamicChildren;const I=Qt(g)&&g.data==="[",w=()=>P(g,m,M,O,K,I),{type:N,ref:x,shapeFlag:G,patchFlag:ne}=m;let ce=g.nodeType;m.el=g,ne===-2&&(U=!1,m.dynamicChildren=null);let j=null;switch(N){case wt:ce!==3?m.children===""?(c(m.el=s(""),o(g),g),j=g):j=w():(g.data!==m.children&&(pt(),g.data=m.children),j=i(g));break;case me:D(g)?(j=i(g),q(m.el=g.content.firstChild,g,M)):ce!==8||I?j=w():j=i(g);break;case It:if(I&&(g=i(g),ce=g.nodeType),ce===1||ce===3){j=g;const X=!m.children.length;for(let H=0;H 哪吒监控的 Dashboard 已经添加本地化,支持多个语言,你可以在开发新功能时遵循以下步骤来支持本地化 哪吒面板提供了主题开发环境,你可以使用它来创建新的哪吒监控主题 WARNING 请注意: 此开发环境仅支持 Contributors: Project: nezha_api_tgbot (Chinese) The bot can request server status information from the Dashboard through the API, and then send the information to the user. Current Version:V1.0 (Chinese) Scan the following QR code with your iPhone or iPad to get the shortcut WARNING Each shortcut can only monitor one server, if you need to monitor more than one server, please copy this shortcut several times and configure them separately, then modify the shortcut name separately, such as; Server 1 Status, Server 2 Status, etc. Contributor: GitHub project: nezha_telegram_bot(English is already supported) Contributors: GitHub project: Argo-Nezha-Service-Container Add Argo tunnel authentication methods include json and token, use one of the two methods. The former is recommended because the script will handle all the Argo tunnel parameters and paths, while the latter needs to be set manually on the Cloudflare website and is prone to errors. Image Variables used Koyeb Transfer via gRPC, no additional configuration required. Use the installation method given in the panel, for example Method 1: Change the contents of the Method 2: After ssh, run The following is an example of restoring a file with the name ! image Contributor: INFO We appreciate any contribution to the project, and we will actively merge your commits into next Release. However, we may modify your work in order to comply with App Review Guidelines. WARNING Due to ICP Filing Policy, Nezha Mobile is not available in China mainland. Add a widget on your Springboard. Long-tap the widget and then tap Edit Widget. Choose a server you want to monitor on the next view. You can choose different servers for each widget. Contributor: Broker for Nezha is a modified version of the Nezha Agent that separates data collection from gRPC reporting, enabling small devices to connect to the Nezha Dashboard. Project link: https://codeberg.org/uubulb/broker Online Documentation: Broker for Nezha Broker for Nezha needs to be used with specific data collection software. Currently, it provides two data collection methods: For existing data collection software, refer to the project README. Due to security concerns, it's recommended to use this only in a trusted network. If you need to write your own data collector, you'll need to implement a basic HTTP server or TCP client and use a specific data format that Broker can recognize. For details, see: Data types Nezha Monitoring's Dashboard has added localization to support multiple languages, and you can follow these steps to support localization when developing new features Nezha Monitoring provides a theme development environment that you can use to create new Nezha Monitoring themes WARNING Please note: This development environment only supports This document will introduce how to install the Agent on the monitored server and connect it to the Dashboard. TIP The repository for Agent binaries can be found at: https://github.com/nezhahq/agent/releases Nezha Monitoring supports one-click installation of the Agent on both Windows and Linux. By following the steps in this document, you can easily deploy it on your server. You need to set up a communication domain in the admin panel in advance, and this domain should not be connected to a CDN. This document uses the example communication domain “data.example.com”. WARNING If you encounter errors when running the one-click installation command in PowerShell, try the Manual Installation of the Agent on Windows below. First, get a copy of Nezha Agent: https://github.com/nezhahq/agent/releases After extracting the archive, run the following command to install the service (may require root permission): You can also add other arguments except the server address and password. For more details, refer to the documentation: Customizing Agent Monitoring Items. Uninstall the service: Start the service: Stop the service: Restart the service: The built-in service command of Agent supports most init systems, including FreeBSD rc.d and openrc, but still missing some of them. Here we take Void Linux's runit as an example: You can add other arguments here as well. Use the How to view logs: Refer to community articles: Using Systemd for DSM7 only: ‼️ Modify the corresponding information before running the above commands with the This section is adapted from 🐿️松鼠收集🌰 with the author's permission WARNING Please be sure to add environment variables before installing nezha-agent through Homebrew! Homebrew creates the service-required plist file during software installation, and if you add the environment variables after installation, it will fail to start due to missing parameters. DANGER Note that this Homebrew repository is maintained by a third party and is unrelated to Nezha Monitoring. The Nezha project team does not endorse this repository's usability, security, etc. Please evaluate the risks yourself before using! Since it has not yet been submitted to the Homebrew Core official library, it is temporarily placed in the third-party Homebrew repository maintained by the author of the blog: How to solve installation difficulties and issues in one step? How to enable autostart on older OpenWRT/LEDE? How to enable autostart on newer OpenWRT? Contributor: @艾斯德斯 The Agent does not currently have a Docker image. Please refer to Dashboard Related - Why is the IP Displayed in the Admin Panel Different from the Actual Agent IP?. This will not be repeated here. This mostly occurs on servers in mainland China. Currently, the one-click script fetches the installation script directly from Github. You may try several times, or manually install the Agent. Additionally, you can find third-party Github acceleration services or mirrors and set them in the one-click installation script. Please manually install sudo first, for example, in Ubuntu: Nezha Monitoring supports querying the status information of Agents using the API interface Nezha Monitoring's API interface allows for Token authentication and Cookies authentication. To create a new Token, go to the admin panel, click on the avatar in the top right corner, select "API Token," and enter the Token management page. Click "Add Token", add a custom note, and click "Add". To delete a Token, select the corresponding Token and click the delete icon on the right. WARNING Tokens are the authentication credentials for the API interface. They are critical to the security of your Dashboard's information, so do not share your Token with others. Ensure the request header contains Token authentication method: WARNING Negative timestamps in the following examples represent TIP The request method is Request: Parameters: Example response: Request: Parameters: Example response: This API does not require authentication, except for servers with Request: Parameters: Example response: Note: Description: Script for automating node registration. Example code is provided below. Reference: https://github.com/naiba/nezha/pull/472 Parameters: Request Payload: The payload can be completely empty ( You can also customize your data: With the above example code, you can easily obtain and process server status information, enabling automated monitoring and management. To set up Nezha Monitoring, you need: TIP If you want to use a CDN, prepare two domains: one configured with CDN for public access (CDN must support WebSocket protocol), and another domain not using CDN for communication between the Agent and Dashboard. This document uses "dashboard.example.com" and "data.example.com" as example domains. This document uses the aaPanel for reverse proxying the Dashboard as an example. As future versions change, some features may change their entry points. This document is for reference only. WARNING This project does not depend on the aaPanel; you can choose any server panel you prefer or manually install Nginx or Caddy to configure SSL and reverse proxy. Nezha Monitoring uses Github, Gitlab, or Gitee as admin accounts. If you encounter issues using GitHub, GitLab, or Gitee for admin account logins, consider switching to Cloudflare Access as your OAuth2 provider for authentication. WARNING The following steps are for users who have already started using Zero Trust. If you have not previously used Cloudflare Zero Trust, we strongly recommend that you first read the Guide on Using Cloudflare Access as an OAuth2 Provider to understand the configuration examples and setup process. If using this method, after installing the Dashboard, modify the configuration file Nazha supports custom OIDC authentication login. For configuration details, please refer to the documentation: Enable OIDC authorization. Run the installation script on the dashboard server: After Docker installation completes, enter the following values: After inputting the values, wait for the image to be pulled. In the future, if you need to run the script again, run: to open the management script. Create a new site in the aaPanel, with the domain filled in as the public access domain, such as “http://dashboard.example.com”. Then click “Settings” to enter the site settings options, select “Reverse Proxy” - “New Reverse Proxy”. Customize a proxy name and fill in Open the “Configuration File” on the right side of the newly created reverse proxy and replace the configuration file with the following content: Click “Save”. CaddyServer v1 (v2 does not require special configuration): First, temporarily disable the reverse proxy. Run the script First, let's explain how the IP displayed in the Admin panel is obtained: The Agent periodically requests IP-API to get IP information and reports it to the Dashboard. The currently used IP-API can be viewed here: myip.go. TIP For example, a common scenario is that the service provider gives you a high-defense server. To meet both high defense and low network interruption rate goals, the IP provided may be a mapped high-defense IP rather than your server's real exit IP. You can also test the exit IP on the Agent server by running the following commands: Please view or edit the First, try restarting Docker before proceeding: After restarting, try reinstalling the Dashboard. If this issue occurs, you can modify the configuration via the installation script or directly edit the If the Dashboard page layout is incorrect, it's usually due to missing or inaccessible CSS files. Find the site configured during Dashboard installation in Select Save the configuration, clear the browser, Nginx, and CDN caches, and refresh the page to see if it returns to normal. The value entered for the DDNS provider is incorrect. Currently, only The entered DDNS The installed versions of the Dashboard and Agent are not compatible and have a This could be a specific issue in Dashboard v0.17.10 - v0.18.0. Updating to the latest version could solve this. Appears in Agent v0.16.9+. Mostly caused by a malfunctioning This error indicates that no TCP-Ping or ICMP-Ping type monitoring has been set in the services page or monitoring data has not yet been generated. This is often due to an incomplete certificate. Add the -d parameter to the agent run command. If the log contains Common in scenarios like batch adding Agents, you can directly modify the database. DANGER Again, do not modify the database casually! If you need to modify data in the database, stop the dashboard container first. Agents typically update automatically, but the Dashboard does not and requires manual updates. When the Agent is offline, or the connection between the Agent and the Dashboard is unstable, it may cause issues with connecting to the Web Terminal. Please check if the Agent is running normally and maintaining a stable connection with the Dashboard. The DDNS functionality is designed for servers with dynamic IP addresses. When the Agent server reports a new IP (every 10 minutes), Dashboard will automatically update the DNS record based on the configuration. You can directly add DDNS configurations in the Dashboard management page. Webhook requires manually constructing HTTP requests, suitable for using other provider services when operations are simple. Webhook option descriptions: Supported Webhook placeholders: In the Dashboard logs, you can see the relevant logs for the DDNS functionality. When configured correctly, there will be corresponding log entries when updating DNS records. These issues can manifest in various ways, but ultimately the browser cannot display correctly after login. TIP What is a protocol? In the browser, the string that ends your domain with Ensure that the protocol+domain+port displayed in the browser before login and after the callback are consistent. The container DNS resolution failed, usually due to modified iptables configurations. This issue appears only when using Gitee login, and the reason is unclear. Switching to GitHub is recommended. This could be caused by various factors, most likely a network issue. Check your network and try again. You logged in with the wrong account or configured the wrong username. Note that the username is not an email, and you can use a script to modify it. This is a network issue. Try restarting Docker first, Same as above. Nezha Server Monitoring includes built-in NAT traversal capabilities, allowing users to expose internal services running on an Agent (such as home OpenWrt routers or NAS devices) to the public internet. This functionality facilitates remote access and is currently limited to supporting plain HTTP traversal. Prior to configuring NAT traversal, please ensure the following preparatory steps are completed: WARNING If your Dashboard server uses tools like the aaPanel or similar management platforms, ensure that your domain (e.g., Nezha Monitoring supports monitoring server load, CPU, memory, disk, traffic, monthly traffic, process count, and connection count, and sending notifications when user-defined thresholds are reached. Refer to the following notification method examples, and you can also flexibly set the push method according to your needs. Name: Bark URL structure: /:key/:body or /:key/:title/:body or /:key/:category/:title/:body Request Method: GET Request Type: Default Body: Empty Name: Bark URL structure: /push Request Method: POST Request Type: form Body: Prepare your Slack Workspace in advance and create an App for this Workspace. If you have not created one, you can create an App at Slack API. After creating the App, you need to add an Incoming Webhook to the App. In the App's settings page, find Incoming Webhooks, enable Activate Incoming Webhooks, and at the bottom of the page, find and click Add New Webhook to Workspace, choose a Channel, and click Allow. After creating, you will get a Webhook URL, which you will use to replace the example URL below. Create a bot in Telegram and get the bot's token and your Telegram user ID. The token and user ID are alphanumeric strings. You can get your user ID by chatting with @userinfobot on Telegram. Create a bot by chatting with @BotFather, and you will get the bot's token. Replace botXXXXXX with your bot token and YYYYYY with your user ID in the URL below. Note that you need to chat with the bot first, otherwise the bot cannot send messages to you. Note: SendCloud has a daily free email sending limit. This is just an example. You can choose a paid service or other similar free services. The usage method is similar. This example uses SendCloud as the email service. You need to register an account on SendCloud, create a sender email, and then obtain the APIUSER and APIKEY here. Replace Create a bot in DingTalk in advance and get the bot's token. The bot URL is obtained after creating a bot in the DingTalk group - Manage Bot - Create Bot. Choose custom keywords for the security method, and the Body content value must contain these keywords. Supported placeholders list This document is NOT available in English. Group Bot Configuration Instructions - Document - WeChat Work Developer Center You can remove or add relevant information as needed. Once configured, you will receive formatted Nezha dashboard information in your Matrix room whenever a notification is triggered. Complete Examples: Add an offline notification: Add an notification for CPU usage exceeding 50% for 10 seconds and memory usage below 20% for 20 seconds: Send notifications for specific servers to specific notification groups: Scenario example: First, set up notification groups A and B, then add two notification rules: Rule 1: Rule 2: Flexibly using parameters can make your notification function fully utilized Can be used for monthly traffic monitoring Example: For servers with IDs 3 and 4 (defined in ignore), if the monthly outbound traffic exceeds 1TB starting from the 1st of each month, send an notification: TIP If you still have trouble editing rules, you can try the rule generators below to simplify the process. Note that Nezha Monitoring does not guarantee the functionality of these generated rules. If you need to execute a task while sending an notification message, you can set this item. Modify the contents of Here is a step-by-step guide to configuring the OIDC settings in your Dashboard v0.18.3+ uses an embedded GeoIP database to query IP locations. An IPinfo format mmdb database is required to be placed at An IPinfo format mmdb is constructed by those fields: Dashboard only requires the It is recommended to use If your goal is to correct geographic information, it is recommended to modify directly based on the original IPinfo database. For similar operations, you can refer to the sing-geoip project. Using the correction of IPinfo data as an example, you need to use the A After writing the database information to a file, you can use the Once the format and information are confirmed to be correct, the database can be used for compiling the Dashboard. Afterwards, you can test if the query API is working correctly using Using Nginx or Caddy to reverse proxy gRPC Dashboard Configuration Agent Configuration Enabling Cloudflare CDN (Optional) According to Cloudflare gRPC requirements: gRPC services must listen on port 443 and must support TLS and HTTP/2. So, to enable CDN, you must use port 443 when configuring Nginx or Caddy to reverse proxy gRPC and configure the certificate (Caddy will automatically apply and configure the certificate). After enabling The traffic statistics on the homepage reset every time the server restarts. To achieve a monthly reset of the traffic counter, you can do the following: TIP This method allows you to set any period, including but not limited to hourly, daily, weekly, monthly, or yearly traffic statistics reset, offering great flexibility! Execute the following command to select network interfaces and partitions, then restart the Agent for changes to take effect: To view the supported runtime parameters, please execute the following command: If you installed the Agent using a one-click script, you can add or modify parameters by editing the system service configuration. Edit the file If you encounter issues logging in as an administrator using Github, Gitlab, or Gitee, you may consider switching to Cloudflare Access as the OAuth2 provider. Navigate to the Zero Trust Dashboard: https://one.dash.cloudflare.com/. Choose or create a new account, then follow these steps: After setting up the Dashboard, you need to configure identity verification policies in the Zero Trust Dashboard. Navigate to: Using email OTP as the default verification method: GPU monitoring is a new feature implemented in Nezha Monitoring v0.17.x. Before using the feature, please ensure your Dashboard version is higher than v0.17.2 and Agent version is higher than v0.17.0. Append the Execute the following command to modify Agent configuration to enable GPU monitoring. In the interactive menu returned, choose to enable GPU monitoring. GPU model and GPU utilization are two different monitor items, which uses different approaches to obtain their value. Windows and macOS supports getting GPU utilization without extra dependencies, and support multiple graphics card brands. Linux distros support only NVIDIA and AMD cards and need to install extra dependencies. Below are the instructions on how to enable GPU utilization monitoring on Linux for NVIDIA / AMD graphics cards. NVIDIA cards need the If you use unofficial drivers like AMD cards need to install the open source Mainstream distros have already packaged If your distro doesn't have the package, then you will need to compile Required dependencies: First, clone the git repository of Then compile the libraries and install them on your system. The server section is responsible for managing Agents, forming the most fundamental part of Nezha monitoring and serving as the basis for other functionalities. The first step is to add a server, where you can customize the name, group, sorting, and notes. Please refer to the previous section on installing the Agent. Agent update-related parameters include This is WebShell; the feature will not work if Added in Dashboard v0.19.1 / Agent v0.19.0. A pseudo file manager embedded in WebShell, supports file download/upload, directory navigation and copying current path. Access it by clicking the blue button in the bottom-right corner of the WebShell. The Services section is where you set up monitoring for external websites or servers using Agents. To add a new monitor, go to the "Services" page in the Admin dashboard and click "Add Monitor." When adding a new service monitor, you need to set the following parameters: Name: Customize a name. Type: Select a monitor type. Nezha Monitoring currently supports three types: "HTTP-GET," "ICMP-Ping," and "TCP-Ping." Target: Depending on the type you choose, the method for setting the target differs: Request Interval: Set the interval in seconds at which the Agent requests the target. Coverage: Select a rule to determine which Agents will request the target. Specific Servers: Used in conjunction with the coverage scope, select Agents within the rule to exclude. Notification Group: Choose the notification methods you have already set up on the "Notification" page. Click here for more details. Enable Fault Notification: Choose whether to receive fault notifications for the target as needed. The default is unchecked. After setting it up, click "Add." Wait a moment and go to the "Services" page on the homepage to view the monitoring results. Nezha Monitoring can monitor and record the latency between the Agent and the target server, sending notifications when there are significant changes. This feature helps you monitor if the server's network route has changed. If you need to execute tasks when service monitoring Notification are triggered, you can check "Enable Trigger Tasks" and select the pre-configured trigger tasks in "Task on Notification" and "Task on Recovery." The TCP-Ping and ICMP-Ping monitoring types set in the Services page will automatically enable the monitoring chart feature. On the "Network" page of the homepage, you can view historical network latency charts. The data in the charts is based on the latency from the Agent to the target server. You can click the Agent's name to toggle the chart display. In the chart, you can uncheck the target server's name to hide or show the corresponding data. To manage existing service monitors, go to the "Services" page in the Admin Panel. Select a monitor configuration and click the icons on the right to edit or delete it. You can customize the site title in this section. Select the theme for the frontend and dashboard here. If a theme that already exists is not in the options, update the Dashboard. To use a custom frontend theme, you need to create a theme description file. For more information on custom themes, please refer to Nezha Theme Development Environment. Before enabling the dashboard Custom theme, ensure you have installed the custom theme; otherwise, the Dashboard will not display correctly after enabling this option. Nezha Monitoring currently supports the following languages: We welcome corrections to translations or contributions of more languages. You can modify the LOGO, adjust colors, add beautification code, and add statistical code here. WARNING Custom code only affects the visitor homepage, not the admin panel. Due to differences in code between themes, if you need to modify the content in the admin panel, please modify the theme files in Docker. If you don't want to display your Dashboard directly to visitors, you can set a view password here. After setting the password, visitors need to enter the password to access the homepage. This setting is a prerequisite for using the one-click script to install the Agent. For details, please see here. If you want to receive notifications when a server's IP changes, you can set it up here. Select a rule to determine which servers to monitor. Choose according to your needs. Set exclusions for the selected rule in conjunction with the coverage scope. Choose a notification method, which should be set up in advance on the "Notification" page. WARNING Notifications take effect after enabling this setting. IP change notifications hide the full IP by default. If you don't want to hide it, check "Show Full IP Address in Notification." By default, the Dashboard allows visitors to change the theme. This feature only affects individual visitors and does not affect the theme set by the administrator in the admin panel. If you don't want visitors to switch themes, check this option. In the Tasks section, you can set up scheduled tasks, trigger tasks, and batch execute tasks on multiple servers. Nezha Monitoring supports pushing commands to Agents for execution. This feature is highly flexible and can be used for regular backups using tools like restic or rclone, periodically restarting a service to reset network connections, or executing a task when an notification is triggered, such as running a script when CPU usage is high for an extended period. Go to the "Tasks" page in the admin panel and click "Add Scheduled Task." When adding a scheduled task, you need to fill in the following parameters: Name: Customize a task name. Task Type: Choose the type of task. Schedule: Set the schedule time (not enable when using trigger task type). The time format is Command: Set the command to execute, similar to writing Shell/Bat scripts, but it's recommended not to use new lines; connect multiple commands with Coverage and Specific Servers: Select rules to determine which Agents execute the scheduled task, similar to the settings on the "Services" page. When using the trigger task type, you can choose "executed by the triggered server." Notification Group: Choose the notification methods you have set up on the "Notifications" page. Click here for more details. Send Success Notification: Check this option to trigger a notification upon successful task execution. To manage existing scheduled tasks, go to the "Tasks" page in the admin panel. For each task configuration, the three icons on the right are: 哪吒监控的被控端服务被称为 Agent,本文档将介绍如何在被控端服务器上安装 Agent,并与 Dashboard 连接。 TIP Agent 二进制文件仓库地址为:https://github.com/nezhahq/agent/releases 哪吒监控支持在 Windows 和 Linux 上一键安装 Agent。遵循本文档的步骤,你可以很轻松地在服务器上部署它。 你需要提前在管理面板中设置好通信域名,此域名不可以接入 CDN。本文档以示例通信域名 “data.example.com” 为例。 WARNING 如果在 PowerShell 中运行一键安装命令时遇到错误,请尝试下方的在 Windows 中手动安装 Agent。 如果你的被控服务器位于中国大陆,可以使用镜像: 首先获取 Agent 的二进制文件:https://github.com/nezhahq/agent/releases 解压后输入以下命令安装服务(可能需要 root 权限): 除了服务器地址和密码,还可以添加其它支持的参数。具体参考自定义 Agent 监控项目 卸载服务: 启动服务: 停止服务: 重启服务: 目前 Agent 内置的服务功能已经支持了绝大部分 init 系统,包括 FreeBSD rc.d 和 openrc。尽管如此还是有一些漏网之鱼。 这里使用 Void Linux 的 runit 作为示例: 这里同样可以添加其它参数。 之后可以通过 如何查看日志: Systemd 实现 仅适用于 DSM7: ‼️修改对应信息后‼️ 本节内容改编自 🐿️松鼠收集🌰,改编已获得原作者授权 WARNING 请务必先添加环境变量,再通过 Homebrew 安装 nezha-agent! 因 Homebrew 在软件安装时创建服务所需 plist 文件,若先安装再添加环境变量,会因缺少参数而启动失败。 DANGER 请注意,此 Homebrew 仓库由第三方维护,与哪吒监控无关。 Nezha 项目组不对该仓库的可用性和安全性等方面作出背书。在使用前,请自行评估风险! 由于暂未提交到 Homebrew Core 官方库,暂时放在上述博客作者参与维护的 第三方 Homebrew 仓库 中: 如何一步到位,解决安装过程中的疑难杂症? 如何使旧版 OpenWRT/LEDE 自启动? 如何使新版 OpenWRT 自启动? 贡献者:@艾斯德斯 Agent 目前没有推出 Docker 镜像。 请查看 Dashboard 相关-为什么管理面板中显示的 IP 和 Agent 实际 IP 不一致?。在此不再赘述。 多出现在中国大陆的服务器中,目前一键脚本只从 Github 直接获取安装脚本,您可以多尝试几次,或者手动安装 Agent。另外,也可以自行寻找第三方 Github 加速服务或者镜像,并在一键安装脚本中设置。 请先手动安装 sudo,例如在 Ubuntu 中: 哪吒监控支持使用 API 接口查询面板中 Agent 的状态信息 哪吒面板的 API 接口允许使用 Token 认证与 Cookies 认证。要新建一个 Token,在进入管理面板后,点击右上角的头像,选择 “API Token”,进入 Token 管理页面。点击 “API Token”,自定义备注后,点击 “添加”。 如需删除一个 Token,请选择相应的 Token,点击右侧的删除图标。 WARNING Token 是 API 接口的鉴权凭据,它对你的面板的信息安全非常重要,请不要泄漏你的 Token 给他人。 确保在请求头中包含 Token 认证方式: WARNING 下面示例中的负数时间戳为(0000-00-00),目前表示 Dashboard 上线后该 Agent 从未汇报过,但不建议用正负性判断状态。 TIP 请求方式为 请求: 参数: 返回示例: 请求: 参数: 返回示例: 此 API 无需认证。(除限制游客访问的服务器) 请求: 参数: 返回示例: 注: 说明: 用于自动化注册节点的脚本,示例代码详见下文。 参考: https://github.com/naiba/nezha/pull/472 参数: 请求体 (Payload): 你可以设置为空内容( 你也可以设置自己的请求参数: 通过以上示例代码,可以轻松获取和处理服务器的状态信息,从而实现自动化监控和管理。 搭建一个哪吒监控,你需要: TIP 如果你想使用 CDN,请准备两个域名,一个配置好 CDN 用作公开访问,CDN 需要支持 WebSocket 协议;另一个域名不要使用 CDN,用作 Agent 端与 Dashboard 的通信。 本文档分别以 "dashboard.example.com" 和 "data.example.com" 两个域名来演示。 本文档将以宝塔面板反代 Dashboard 的过程作为示范,随着未来版本的变化,部分功能的入口可能会发生改变,本文档仅供参考。 WARNING 本项目并不依赖宝塔,你可以选择使用你喜欢的任何服务器面板,或手动安装 Nginx 或 Caddy 来配置 SSL 和反代。 哪吒监控接入 Github、Gitlab、Gitee 作为后台管理员账号。 对于位于中国大陆的用户,直接连接到 GitHub 可能会遇到困难。如果您在使用 GitHub、GitLab 或 Gitee 作为管理员账户登录时遇到问题,建议切换到使用 Cloudflare Access 作为 OAuth2 提供方 进行登录。 WARNING 以下步骤适用于已经开始使用 Zero Trust 的用户。如果您尚未使用过 Cloudflare Zero Trust,强烈建议您首先阅读 Cloudflare Access 作为 OAuth2 提供方的使用指南,以了解 Cloudflare Access 的配置示例和流程。 如使用此方式,安装 Dashboard 完成后,需要修改配置文件 哪吒支持自定义 OIDC 验证登录。有关配置详情,请参考文档:启用 OIDC 认证。 在面板服务器中,运行安装脚本: 如果你的面板服务器位于中国大陆,可以使用镜像: 等待 Docker 安装完毕后,分别输入以下值: 输入完成后,等待拉取镜像。 将来如果需要再次运行脚本,可以运行: 来打开管理脚本。 在宝塔面板中新建一个站点,域名填写公开访问域名,如 “http://dashboard.example.com“ ,然后点击“设置”进入站点设置选项,选择“反向代理” - “新建反向代理”。 自定义一个代理名称,在下方“目标 URL”中填入 打开刚刚新建的反向代理右边的“配置文件”,将配置文件替换为以下内容: 点击“保存”。 CaddyServer v1(v2 无需特别配置): 首先,先暂时关闭反向代理。 运行脚本 首先解释管理面板中显示的IP是怎么得到的:Agent 会每隔一段时间请求一遍 IP-API,获取到 IP 信息后上报到 Dashboard,目前使用的 IP-API 可在此查看:myip.go。 TIP 举个简单也十分常见的例子,服务商给您提供的是一台高防服务器,为了同时满足高防和低网络中断率的目标,提供给您的 IP 可能是经过映射后的高防 IP 而并非您服务器的真实出口 IP。 您也可以在 Agent 服务器中运行以下命令测试出口 IP: 请查看或编辑 首先尝试重启 Docker 再操作: 重启后尝试重新安装面板。 如出现此问题,可以通过安装脚本修改配置,或者直接修改 如果出现 Dashboard 页面布局错误,通常是 CSS 文件丢失或无法被加载。 在 选择 保存配置,并清空浏览器、Nginx、CDN 中的缓存,此时刷新页面应恢复正常。 填入的 DDNS provider 的值有误,目前仅支持 填入的 DDNS Dashboard 与 Agent 版本不兼容导致含有对端不支持的 Dashboard v0.17.10 - v0.18.0 也存在此问题,更新至最新版本可以解决。 此报错见于 Agent v0.16.9+。原因为系统的 出现此错误说明没有在服务页中设置 TCP-Ping 和 ICMP-Ping 类型的监控或者监控数据还未生成。 常常是由于证书不完整造成的,请在 agent 运行参数中添加 -d,若 log 中有 常见于批量新建 Agent 等需求中,可以直接修改数据库。 DANGER 再重复一遍,请勿随意修改数据库! 如需要在数据库中修改数据,请先停止面板容器再修改。 Agent 通常情况下会自动更新,但 Dashboard 并不会,需要手动更新。 当 Agent 离线,或与 Dashboard 之间的连接不稳定时,可能会导致在线终端无法正常连接。 请检查 Agent 是否正常运行,是否与 Dashboard 保持稳定的连接。 DDNS 功能适用于使用动态 IP 的服务器,当 Agent 上报了一个新的 IP(10分钟一次),Dashboard 会根据配置自动更新 DNS 记录。 可以直接在 Dashboard 管理页面中添加 DDNS 配置。 Webhook 需要自行构建 HTTP 请求,适用于需要使用其它供应商的服务且操作较为简单的情况。 Webhook 相关选项说明: Webhook 支持的占位符: 在 Dashboard 的日志中,可以看到 DDNS 功能的相关日志,配置正确时,更新 DNS 记录时会有相应的日志记录。 还有其他一些表现形式,总之登录后浏览器无法正常显示。 TIP 什么是协议? 在浏览器中,您的域名以 请确保登录前浏览器显示的协议+域名+端口和登录后跳转到的协议+域名+端口一致。 容器 DNS 解析失败,多数情况下为修改了 iptables 相关配置。 只出现在 Gitee 登录方式中,原因不明,建议更换到 GitHub。 可能由多种因素引起,最大可能性是网络问题,建议检查网络后重试。 您登陆错了账号或者配置错了用户名,注意用户名不是邮箱,可使用脚本修改。 网络问题,可先重启 Docker, 同上。 哪吒服务器监控提供内置的内网穿透功能,允许用户通过设定 Dashboard 作为服务端,将运行在 Agent 上的内网服务(如家庭 OpenWrt 路由器、NAS 设备等)映射到公网上,便于外部访问。目前仅支持明文 HTTP 的穿透。 在配置内网穿透前,请确保完成以下准备工作: WARNING 如果您的 Dashboard 服务器使用宝塔面板或其他管理工具,请确保所用域名,如 哪吒监控支持对服务器的负载、CPU、内存、硬盘、流量、月流量、进程数、连接数进行监控,并在达到用户设定的阈值时发送告警通知。 请参考以下的通知方式示例,也可以根据自己的需求灵活设置推送方式。 名称:Bark URL 组成: /:key/:body or /:key/:title/:body or /:key/:category/:title/:body 请求方式: GET 请求类型: 默认 Body: 空 名称:Bark URL 组成: /push 请求方式: POST 请求类型: form Body: 请提前准备好 Slack 的 Workspace 并为这个 Workspace 创建一个 App。如果你还没有创建,可以在 Slack API 创建一个 App。 创建完成 App 后,需要为这个 App 添加一个 Incoming Webhook。在 App 的设置页面中找到 Incoming Webhooks,将 Activate Incoming Webhooks 勾选为 ON,在页面下方找到并点击 Add New Webhook to Workspace,选择一个 Channel,然后点击允许。完成创建后,你会得到一个 Webhook URL,使用这个 URL 替换下方的示例 URL。 Server 酱进阶 名称:Server 酱 请求方式: POST 请求类型: FORM Body: 请提前在 Telegram 中创建一个机器人,获取到机器人的 token 和你的 Telegram 用户 ID。 机器人的 token 和用户 ID 都是数字和字母的组合,可以在 Telegram 中与 @userinfobot 对话获取自己的用户 ID。与 @BotFather 对话,输入命令 /newbot 创建一个机器人,创建完成后可以获得机器人的 token。 得到的 token 和用户 ID 都是字符串,可以直接拼接到 URL 中,如下所示,将 botXXXXXX 中的 XXXXXX 替换为你的机器人 token,要保留其中的 bot,将 YYYYYY 替换为你的用户 ID。注意,你需要先与机器人对话,否则机器人无法发送消息给你。#NEZHA# 是占位符,不要修改和删除 需要提前关注你的应用 注意:SendCloud 有每日免费发送邮件限额限制,这里仅作示例,你可以选择付费服务或其他类似的免费服务,使用方法类似。 该示例使用 SendCloud 作为发信服务,需提前在 SendCloud 注册账号,创建发件邮箱,然后在这里获取 APIUSER 和 APIKEY。 替换示例 URL 中的 请提前在钉钉中创建一个机器人,获取到机器人的 token。 机器人 URL 在钉钉群 - 管理机器人 - 创建机器人后获取,安全方式选择自定义关键词,Body 中 content 值内需包含该关键词。 支持的占位符一览 根据需求删减相关内容信息即可。 机器人 URL 通过飞书群 - 群机器人 - 添加机器人 - 自定义机器人(webhook)创建后获取。 这样设置好后,每次触发通知时,你的 Matrix 房间中就会收到格式化的哪吒面板信息。 完整示例: 添加一个离线告警: 添加一个监控 CPU 持续 10 秒超过 50% 且 内存持续 20 秒占用低于 20% 的告警: 将特定的服务器通知发送到特定的通知分组: 示例场景: 首先你需要先设置好 A、B 两个通知组,然后添加两条告警规则: 规则一: 规则二: 灵活使用参数可以让你的告警功能被充分使用 可以用作月流量监控 示例: ID 为 3 和 4 的服务器(ignore 里面定义),以每月 1 号为统计周期,周期内统计的出站月流量达到 1TB 时告警: TIP 如果对告警配置仍感到有困难,可以试试以下第三方配置生成器简化操作。哪吒监控不对生成配置的功能性作任何保证。 如果需要在发出告警消息的同时执行某项任务,可以设置此项目。 如果你的 Dashboard 服务器无法访问 Telegram Bot API,但你依然想使用 Telegram 来推送通知,你可以尝试使用反向代理的方式解决这个问题。 这里介绍使用你自己的服务器进行反代的方法。你也可以选择使用 Cloudflare 的 Workers 进行反代,但可能对于中国大陆的用户来说网络连通性依然不佳。 要搭建一个 Telegram Bot API 反代,你需要准备以下内容: 编辑 Nginx 配置文件,在 执行 配置防火墙以防止他人盗用你的反代服务: 通过以上配置,可以有效防止未经授权的访问。 修改 以下是在 Dashboard v0.18.3+ 采用离线数据库获取 GeoIP 信息,如需编译正确显示地理位置信息的 Dashboard,必须在 IPinfo 格式包含以下字段: 其中 Dashboard 只需要用到 目前推荐使用 MaxMind 官方使用的 如果你的主要目的是为了校正地理信息,建议直接在 IPinfo 原版数据库的基础上修改。类似的操作可以参考 sing-geoip 项目。 这里以校正 IPinfo 数据为例,你需要使用 其中 编辑完成并将数据库信息写入到文件后,可以使用 确定格式、信息无误后便可用于 Dashboard 编译。之后可通过 使用 Nginx 或者 Caddy 反向代理 gRPC Dashboard 面板端配置 Agent 端配置 开启 Cloudflare CDN(可选) 根据 Cloudflare gRPC 的要求:gRPC 服务必须侦听 443 端口 且必须支持 TLS 和 HTTP/2。 所以如果需要开启 CDN,必须在配置 Nginx 或者 Caddy 反向代理 gRPC 时使用 443 端口,并配置证书(Caddy 会自动申请并配置证书)。 开启 主页中的流量统计每次服务器重启时都会重置,如果要实现每月重置一次流量计数,可以这样实现: TIP 此方式可以设置任何周期,包括且不限于每小时/每天/每周/每月/每年重置流量统计,非常灵活! 执行以下命令来选择网卡和分区,然后重启 Agent 即可生效: 要查看支持的运行参数,请执行以下命令: 如果您通过一键脚本安装了 Agent,可以通过编辑系统服务配置来添加或修改参数。编辑文件 TIP 如果您需要批量添加参数,可以利用 Dashboard 的计划任务功能。以禁用 Agent 自动更新功能为例,您可以设置一个计划任务。添加以下命令以修改系统服务配置,并触发该任务执行: 相较于 Github,Cloudflare Access 对于中国大陆用户更加友好。如您当前使用 Github、Gitlab、Gitee 作为管理员账户登录时遇到问题,您可以考虑切换 Cloudflare Access 作为 OAuth2 提供方 前往 Zero Trust Dashboard:https://one.dash.cloudflare.com/,选择或新建一个账户(Account),然后按照以下步骤操作: 在完成 Dashboard 设置后,您还需要在 Zero Trust Dashboard 中配置身份验证策略,路径为: 默认使用邮件 OTP 验证方式: GPU 监控是哪吒监控 v0.17.x 引入的新功能,使用前请检查您的 Dashboard 版本是否为 v0.17.2+ / Agent 版本是否为 v0.17.0+。 在 Agent 运行参数后添加 执行以下命令修改 Agent 配置文件以启用 GPU 监控: 在返回的互动菜单中选择启用 GPU 功能即可。 GPU 型号与 GPU 使用率为两个不同的监控项目,使用了不同实现获取。 其中 Windows 和 macOS 支持无依赖获取 GPU 使用率,并支持多个品牌显卡; Linux 平台则仅支持 NVIDIA / AMD 显卡,且需要安装额外依赖。 以下将介绍如何在 Linux 上为 NVIDIA / AMD 显卡启用 GPU 使用率监控。 NVIDIA 显卡获取 GPU 使用率需要用到 如果您使用的是非官方驱动,例如 AMD 显卡获取 GPU 使用率需要安装 主流系统均已打包 如果您的系统并没有相应包,那么则需要手动编译安装 您的系统需要安装这些依赖: 首先 Clone 然后进行编译并安装即可。 服务器区域负责管理 Agent,是哪吒探针中最基础的区域,也是其他功能的基础。 第一步是新增服务器,可以自定义名称、分组、排序和备注。 请参考前文安装 Agent。 Agent 更新相关的参数是 自定义 Agent 监控项目 中的 即 WebShell, Dashboard v0.19.1 / Agent v0.19.0 加入的新功能,是嵌入 WebShell 的一个伪文件管理器,提供文件下载和上传功能,也可以进行目录跳转或者复制路径。点击 WebShell 右下角的蓝色按钮就可以打开。 配置位置:后台-> 服务器 -> 编辑服务器 -> 公开备注 完整配置,包含过期时间和价格展示 单独配置过期时间 配置详细说明: startDate 账单开始时间 endDate 账单到期时间 Date格式说明 autoRenewal 自动续期 amount 价格 cycle 付费周期 服务区域是设置 Agent 监控外部网站或服务器的功能设置区 如需新增一个监控,可以进入管理面板中的“服务”页,点击“添加监控”。 新增一个服务监控时,你需要设置以下参数: 名称:自定义一个名称。 类型:选择一个监控类型。目前哪吒监控支持三种监控类型,分别是“HTTP-GET”、“ICMP-Ping”和“TCP-Ping”。 目标:根据你选择的类型不同,目标的设置方法也不同: 请求间隔:设定 Agent 每次请求目标的时间间隔,以秒为单位。 覆盖范围:选择一条规则来确定要使用哪些 Agent 来请求目标。 特定服务器:配合覆盖范围使用,选择规则内需要排除的 Agent。 通知方式组:选择你已经在“告警”页设置好的通知方式,点击这里了解详情。 启用故障通知:根据需要选择是否接收目标故障通知,默认为不勾选。 设置完成后,点击“添加”即可。稍等片刻前往主页的“服务”页查看监控结果。 哪吒监控可以监测并统计 Agent 到目标服务器之间的延迟,在发生较大变化的情况下发送通知。利用此功能可以帮助你监控服务器的线路是否发生了变化。 如果你需要在服务监控告警触发时执行一些任务,可以勾选“启用触发任务”,并在“告警时触发任务”和“恢复时触发任务”中选择你已经提前设置好的触发任务。 在服务页中设置的 TCP-Ping 和 ICMP-Ping 类型的监控都会自动启用监控图表功能。在主页的“网络”页中,你可以查看历史网络延迟监控图表。图表中的数据是根据 Agent 到目标服务器的延迟统计而来,你可以点击 Agent 的名称来切换图表。在图表中,可以取消勾选目标服务器的名称来隐藏或显示对应的数据。 如需对已有的服务监控进行管理,可以前往管理面板中的“服务”页。选择一条监控配置,点击右侧的图标进行编辑或删除。 你可以在此项中自定义站点标题。 在这里选择主页和管理界面主题,如果选项中没有某个已存在的主题,请更新面板。 使用自定义主页主题需要创建主题描述文件,关于自定义主题更多信息请参考哪吒主题开发环境。 启用管理界面 Custom 主题前,你需要提前安装自定义主题,否则启用该项后,面板将无法正常显示。 哪吒监控目前支持以下语言: 我们欢迎您对翻译进行勘误或贡献更多语言。 可以在此修改 LOGO、修改色调、添加美化代码和统计代码等。 WARNING 自定义代码仅在游客首页生效,管理面板不生效。由于不同主题的代码不同,如确需修改管理面板中的内容,请修改 Docker 中的主题文件。 如果你不想向游客直接展示你的 Dashboard,可以在这里设置一个查看密码。设置密码后,需要输入密码才可以访问主页。 此项设置是使用一键脚本安装 Agent 的前提,详情请查看这里。 如果你希望当某个服务器的 IP 发生变更时收到通知,可以在这里进行设置。 在这里选择一条规则,来确定需要监控哪些服务器,可以根据自己的需求进行选择。 配合覆盖范围的设置,在这里设置选定规则的排除项。 选择通知方式,通知方式请提前在“告警”页内设置。 WARNING 设置完成后,勾选启用时,通知生效。 IP 变更通知默认隐藏完整 IP,如果你不希望隐藏,可以勾选“通知信息中显示完整 IP 地址”。 Dashboard 默认启用允许访客修改显示主题的功能,此功能只会影响单一访客,不会影响管理员在后台设置的启用主题。如果你不希望访客切换主题,可以勾选此项。 在任务区域中,可以设置计划任务,触发任务,多服务器批量执行任务 哪吒监控支持推送命令到 Agent 执行,此功能非常灵活。使用此功能可以定期结合 restic、rclone 给服务器备份,定期重启某项服务来重置网络连接。也可以配合告警通知,在触发告警时执行某项任务,例如在 CPU 长时间高占用的情况下运行某个脚本。 进入管理面板的“任务”页,点击“添加计划任务”。添加计划任务时,你需要填入以下参数: 名称:自定义一个任务名称。 任务类型:选择任务的类型。 计划:设置计划时间(使用触发任务类型时不生效),时间格式为: 命令:设置需要执行的命令,就像写 Shell/Bat 脚本一样,但不推荐换行,多个命令使用 覆盖范围和特定服务器:选择规则来确定哪些 Agent 执行计划任务,类似于“服务”页中的设置。使用触发任务类型时,可选择“由触发的服务器执行”。 通知方式组:选择你已经在“告警”页设置好的通知方式,点击这里了解详情。 推送成功的消息:勾选此项,任务执行成功后会触发消息通知。 如需对已有的计划任务进行管理,可以前往管理面板中的“任务”页。选择一条任务配置,右侧的三个图标分别是: Community Project Nezha Monitoring has benefited from various projects contributed by the community, which have provided it with additional extensions. Open-source, lightweight, and easy-to-use server monitoring and operation tooll10n 本地化开发指南
介绍
/resource/l10n/zh-CN.toml
中已有的文本配置来替换新功能中的文本zh-CN.toml
的配置文本,将新文本拉取到 zh-CN.toml
等其他语言的配置文件中,并添加翻译新本地化文本的添加
',6),n=[r];function d(c,i,s,_,h,p){return o(),a("div",null,n)}const f=e(t,[["render",d]]);export{u as __pageData,f as default};
diff --git a/assets/developer_l10n.md.XzEs6w0F.lean.js b/assets/developer_l10n.md.XzEs6w0F.lean.js
deleted file mode 100644
index 7cb7c9eb..00000000
--- a/assets/developer_l10n.md.XzEs6w0F.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as a,o,a4 as l}from"./chunks/framework.BmdFiWrL.js";const u=JSON.parse('{"title":"l10n 本地化开发指南","description":"","frontmatter":{},"headers":[],"relativePath":"developer/l10n.md","filePath":"developer/l10n.md","lastUpdated":1732521523000}'),t={name:"developer/l10n.md"},r=l("",6),n=[r];function d(c,i,s,_,h,p){return o(),a("div",null,n)}const f=e(t,[["render",d]]);export{u as __pageData,f as default};
diff --git a/assets/developer_theme.md.yi2XIKJt.js b/assets/developer_theme.md.yi2XIKJt.js
deleted file mode 100644
index 0c2c4fba..00000000
--- a/assets/developer_theme.md.yi2XIKJt.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as o,o as a,a4 as t}from"./chunks/framework.BmdFiWrL.js";const u=JSON.parse('{"title":"哪吒主题开发环境","description":"","frontmatter":{},"headers":[],"relativePath":"developer/theme.md","filePath":"developer/theme.md","lastUpdated":1732521523000}'),c={name:"developer/theme.md"},d=t('/resource/l10n/
中添加新的语言文本配置哪吒主题开发环境
dashboard v0.19.20
及更新版本。使用说明
data/config.yaml
中的 Oauth2 配置(回调连接可以填 http://localhost
)docker-compose up
theme-custom
(前台主题)、static-custom
(前台主题静态文件) 和 dashboard-custom
(后台主题) 放置到服务器上的 /opt/nezha/dashboard/
中(如使用 Docker 安装)FAQ
',7),l=[d];function r(s,i,h,n,m,_){return a(),o("div",null,l)}const f=e(c,[["render",r]]);export{u as __pageData,f as default};
diff --git a/assets/developer_theme.md.yi2XIKJt.lean.js b/assets/developer_theme.md.yi2XIKJt.lean.js
deleted file mode 100644
index 307da4ee..00000000
--- a/assets/developer_theme.md.yi2XIKJt.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as o,o as a,a4 as t}from"./chunks/framework.BmdFiWrL.js";const u=JSON.parse('{"title":"哪吒主题开发环境","description":"","frontmatter":{},"headers":[],"relativePath":"developer/theme.md","filePath":"developer/theme.md","lastUpdated":1732521523000}'),c={name:"developer/theme.md"},d=t("",7),l=[d];function r(s,i,h,n,m,_){return a(),o("div",null,l)}const f=e(c,[["render",r]]);export{u as __pageData,f as default};
diff --git a/assets/en_US_case_case1.md.Ba8gewZ6.js b/assets/en_US_case_case1.md.Ba8gewZ6.js
deleted file mode 100644
index c09a7a52..00000000
--- a/assets/en_US_case_case1.md.Ba8gewZ6.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o as r,a4 as o}from"./chunks/framework.BmdFiWrL.js";const m=JSON.parse('{"title":"Build your own Telegram bot to query server information","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/case/case1.md","filePath":"en_US/case/case1.md","lastUpdated":1732521523000}'),a={name:"en_US/case/case1.md"},n=o('80
端口,在 docker-compose.yaml
中修改配置。Build your own Telegram bot to query server information
You can build this bot to easily view the current status of a given server without opening the Dashboard.Use Siri to run shortcut to check server status in iOS/MacOS
Contributor:Get shortcut command
MacOS users please visit here to get the shortcutHow to use
Dashboard URL
, API Token
, Server ID
in the three text boxesBuild your own server status query Telegram bot
Features
Commands list
',8),i=[o];function s(n,l,u,h,c,m){return r(),e("div",null,i)}const b=t(d,[["render",s]]);export{_ as __pageData,b as default};
diff --git a/assets/en_US_case_case3.md.CAI94Mh5.lean.js b/assets/en_US_case_case3.md.CAI94Mh5.lean.js
deleted file mode 100644
index ef329393..00000000
--- a/assets/en_US_case_case3.md.CAI94Mh5.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as t,c as e,o as r,a4 as a}from"./chunks/framework.BmdFiWrL.js";const _=JSON.parse('{"title":"Build your own server status query Telegram bot","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/case/case3.md","filePath":"en_US/case/case3.md","lastUpdated":1732521523000}'),d={name:"en_US/case/case3.md"},o=a("",8),i=[o];function s(n,l,u,h,c,m){return r(),e("div",null,i)}const b=t(d,[["render",s]]);export{_ as __pageData,b as default};
diff --git a/assets/en_US_case_case4.md.DB-wfY8y.js b/assets/en_US_case_case4.md.DB-wfY8y.js
deleted file mode 100644
index 3e563364..00000000
--- a/assets/en_US_case_case4.md.DB-wfY8y.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as t,c as n,o,j as e,a}from"./chunks/framework.BmdFiWrL.js";const x=JSON.parse('{"title":"Fake-agent, monitoring data cheater","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/case/case4.md","filePath":"en_US/case/case4.md","lastUpdated":1732521523000}'),r={name:"en_US/case/case4.md"},s=e("h1",{id:"fake-agent-monitoring-data-cheater",tabindex:"-1"},[a("Fake-agent, monitoring data cheater "),e("a",{class:"header-anchor",href:"#fake-agent-monitoring-data-cheater","aria-label":'Permalink to "Fake-agent, monitoring data cheater"'},"")],-1),c=e("p",null,"Contributor:",-1),i=e("ul",null,[e("li",null,[e("a",{href:"https://github.com/dysf888",target:"_blank",rel:"noreferrer"},"dysf888")])],-1),d=e("p",null,[a("GitHub project: "),e("a",{href:"https://github.com/dysf888/fake-nezha-agent",target:"_blank",rel:"noreferrer"},"fake-nezha-agent"),a("(Chinese)")],-1),h=e("p",null,[a("You can modify the monitoring data uploaded to Dashboard by the Agent"),e("br"),a(" Use it for cheating 😈")],-1),l=[s,c,i,d,h];function _(f,g,m,p,u,k){return o(),n("div",null,l)}const y=t(r,[["render",_]]);export{x as __pageData,y as default};
diff --git a/assets/en_US_case_case4.md.DB-wfY8y.lean.js b/assets/en_US_case_case4.md.DB-wfY8y.lean.js
deleted file mode 100644
index 3e563364..00000000
--- a/assets/en_US_case_case4.md.DB-wfY8y.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as t,c as n,o,j as e,a}from"./chunks/framework.BmdFiWrL.js";const x=JSON.parse('{"title":"Fake-agent, monitoring data cheater","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/case/case4.md","filePath":"en_US/case/case4.md","lastUpdated":1732521523000}'),r={name:"en_US/case/case4.md"},s=e("h1",{id:"fake-agent-monitoring-data-cheater",tabindex:"-1"},[a("Fake-agent, monitoring data cheater "),e("a",{class:"header-anchor",href:"#fake-agent-monitoring-data-cheater","aria-label":'Permalink to "Fake-agent, monitoring data cheater"'},"")],-1),c=e("p",null,"Contributor:",-1),i=e("ul",null,[e("li",null,[e("a",{href:"https://github.com/dysf888",target:"_blank",rel:"noreferrer"},"dysf888")])],-1),d=e("p",null,[a("GitHub project: "),e("a",{href:"https://github.com/dysf888/fake-nezha-agent",target:"_blank",rel:"noreferrer"},"fake-nezha-agent"),a("(Chinese)")],-1),h=e("p",null,[a("You can modify the monitoring data uploaded to Dashboard by the Agent"),e("br"),a(" Use it for cheating 😈")],-1),l=[s,c,i,d,h];function _(f,g,m,p,u,k){return o(),n("div",null,l)}const y=t(r,[["render",_]]);export{x as __pageData,y as default};
diff --git a/assets/en_US_case_case5.md.CSkituIs.js b/assets/en_US_case_case5.md.CSkituIs.js
deleted file mode 100644
index a4f65921..00000000
--- a/assets/en_US_case_case5.md.CSkituIs.js
+++ /dev/null
@@ -1,47 +0,0 @@
-import{_ as e,c as a,o as t,a4 as n}from"./chunks/framework.BmdFiWrL.js";const g=JSON.parse('{"title":"Nezha server over Argo tunnel","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/case/case5.md","filePath":"en_US/case/case5.md","lastUpdated":1732521523000}'),s={name:"en_US/case/case5.md"},o=n(`Command Description Private chat only start Getting started with the keyboard main menu ✔️ help help message ❌ add Add Nezha monitoring url link and token ✔️ url Add Nezha monitoring url link ✔️ token Add Nezha monitoring token ✔️ info Get saved Nezha monitoring url link and token ✔️ delete Delete saved Nezha monitoring url link and token ✔️ id Add an integer id after the command to query the information of a single server (refresh button only available in private chat) ❌ all Query statistics for all servers ❌ search Search for keywords in server names (multiple keywords supported, split by spaces) ❌ Nezha server over Argo tunnel
Catalog
Project Features.
Prepare variables to be used
network
option to turn the gRPC
switch on.https://
to the beginning of the panel's domain name and /oauth2/callback
to the end of the callback address.How to get Argo authentication: json or token
(Methods 1 - Json):
Easily get Argo tunnel json information through Cloudflare Json Generation Network: https://fscarmen.cloudflare.now.cc
(Methods 2 - Token): Manually generate Argo tunnel token information via Cloudflare website.
Go to the cf website: https://dash.cloudflare.com/
PaaS Deployment Example
fscarmen/argo-nezha:latest
, supports amd64 and arm64 architectures.Variable Name Required Remarks GH_USER Yes github username for panel admin authorization GH_CLIENTID yes apply on github GH_CLIENTSECRET yes apply on github GH_BACKUP_USER No The github username for backing up Nezha's server-side database on github, if not filled in, it is the same as the account GH_USER for panel management authorization GH_REPO No The github repository for backing up Nezha's server-side database files on github GH_EMAIL No github's mailbox for git push backups to remote repositories GH_PAT No github's PAT ARGO_AUTH Yes Argo Json from https://fscarmen.cloudflare.now.cc
Argo token from Cloudflare official websiteARGO_DOMAIN Yes Argo domain VPS Deployment Method 1 --- docker
docker deployment
docker run -dit \\
- --name nezha_dashboard \\
- --pull always \\
- --restart always \\
- -e GH_USER=<fill in github username> \\
- -e GH_EMAIL=<fill in github email> \\
- -e GH_PAT=<fill in the obtained> \\
- -e GH_REPO=<fill in customized> \\
- -e GH_CLIENTID=<fill in acquired> \\
- -e GH_CLIENTSECRET=<fill in acquired> \\
- -e ARGO_AUTH='<Fill in the fetched Argo json or token>' \\
- -e ARGO_DOMAIN=<fill in customized> \\
- -e GH_BACKUP_USER=<If it is consistent with GH_USER, you can leave it blank> \\
- fscarmen/argo-nezha
docker-compose deployment
version: '3.8'
-services.
- argo-nezha.
- image: fscarmen/argo-nezha
- --pull always
- container_name: nezha_dashboard
- restart: always
- environment:
- - GH_USER=<fill in github username>
- - GH_EMAIL=<fill in your github email>
- - GH_PAT=<<fill in obtained>
- - GH_REPO=<fill in customized>
- - GH_CLIENTID=<fill in obtained>
- - GH_CLIENTSECRET=<fill in fetched>
- - ARGO_AUTH='<Fill in the fetched Argo json or token>'
- - ARGO_DOMAIN=<fill in customized>
- - GH_BACKUP_USER=<If it is consistent with GH_USER, you can leave it blank>
VPS Deployment Method 2 --- hosts
bash <(wget -qO- https://raw.githubusercontent.com/fscarmen2/Argo-Nezha-Service-Container/main/dashboard.sh)
Client Access
curl -L https://raw.githubusercontent.com/nezhahq/scripts/main/install_en.sh -o nezha.sh && chmod +x nezha.sh && sudo ./nezha.sh install_agent data.seales.nom.za 443 eAxO9IF519fKFODlW0 --tls
SSH access
<filepath>/cloudflared access ssh --hostname ssh.seals.nom.za/<GH_CLIENTID>
Manually backing up your data
README.md
file in the Github backup repository to backup
/dashboard/backup.sh
for container version; /opt/nezha/dashboard/backup.sh
for VPS host version.Automatically restore backups
README.md
in the github backup repository, the timer service will check for updates every minute and record the last synchronized filename in the local /dbfile
to compare with the online file content.dashboard-2023-04-23-13:08:37.tar.gz
.Manually restore the backup
bash /dashboard/restore.sh <filename>
Migrating data
/dashboard
folder of the original Nezha and zip it up to dashboard.tar.gz
file.tar czvf dashboard.tar.gz /dashboard
dashboard.tar.gz
.Main catalog files and descriptions
/dashboard/
-|-- app # Nezha panel main program
-|-- argo.json # Argo tunnel json file, which records information about using the tunnel.
-|-- argo.yml # Argo tunnel yml file, used for streaming web, gRPC and ssh protocols under a single tunnel with different domains.
-|-- backup.sh # Backup data scripts
-|-- restore.sh # Restore backup scripts
-|-- dbfile # Record the name of the latest restore or backup file
-|-- resource # Folders of information on panel themes, languages, flags, etc.
-|-- data
-| |-- config.yaml # Configuration for the Nezha panel, e.g. Github OAuth2 / gRPC domain / port / TLS enabled or not.
-| \`-- sqlite.db # SQLite database file that records all severs and cron settings for the panel.
-|-- entrypoint.sh # The main script, which is executed after the container is run.
-|-- nezha.csr # SSL/TLS certificate signing request
-|-- nezha.key # Private key information for SSL/TLS certificate.
-|-- nezha.pem # SSL/TLS certificate file.
-|-- cloudflared # Cloudflare Argo tunnel main program.
-|-- grpcwebproxy # gRPC reverse proxy main program.
-\`-- nezha-agent # Nezha client, used to monitor the localhost.
Acknowledgements for articles and projects by
Disclaimer
`,88),r=[o];function i(l,c,h,d,p,u){return t(),a("div",null,r)}const f=e(s,[["render",i]]);export{g as __pageData,f as default};
diff --git a/assets/en_US_case_case5.md.CSkituIs.lean.js b/assets/en_US_case_case5.md.CSkituIs.lean.js
deleted file mode 100644
index ba42f0d1..00000000
--- a/assets/en_US_case_case5.md.CSkituIs.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as a,o as t,a4 as n}from"./chunks/framework.BmdFiWrL.js";const g=JSON.parse('{"title":"Nezha server over Argo tunnel","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/case/case5.md","filePath":"en_US/case/case5.md","lastUpdated":1732521523000}'),s={name:"en_US/case/case5.md"},o=n("",88),r=[o];function i(l,c,h,d,p,u){return t(),a("div",null,r)}const f=e(s,[["render",i]]);export{g as __pageData,f as default};
diff --git a/assets/en_US_case_case6.md.De9bTpTA.js b/assets/en_US_case_case6.md.De9bTpTA.js
deleted file mode 100644
index 08bb8650..00000000
--- a/assets/en_US_case_case6.md.De9bTpTA.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o,a4 as a}from"./chunks/framework.BmdFiWrL.js";const r="/assets/Download_on_the_App_Store_Badge_US-UK_RGB_blk_092917.CVyK0T4N.svg",i="/assets/1_en_US.CE_aSW8N.png",n="/assets/2_en_US.Dm8rLo_N.png",s="/assets/3_en_US.SUOW4yEM.png",k=JSON.parse('{"title":"Nezha Mobile - An iOS Client For Nezha Dashboard","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/case/case6.md","filePath":"en_US/case/case6.md","lastUpdated":1732521523000}'),l={name:"en_US/case/case6.md"},h=a('Nezha Mobile - An iOS Client For Nezha Dashboard
Contribute to the project
Download on the App Store
TestFlight Testing Program
Instructions
server.hidandelion.com
and API token on your first use. Do not add a protocol prefix or a trailing slash to the link. API Token can be retrieved from Admin Center of Dashboard. Configure Widget(Optional)
',15),c=[h];function d(p,_,g,u,m,b){return o(),t("div",null,c)}const w=e(l,[["render",d]]);export{k as __pageData,w as default};
diff --git a/assets/en_US_case_case6.md.De9bTpTA.lean.js b/assets/en_US_case_case6.md.De9bTpTA.lean.js
deleted file mode 100644
index 4b91bace..00000000
--- a/assets/en_US_case_case6.md.De9bTpTA.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o,a4 as a}from"./chunks/framework.BmdFiWrL.js";const r="/assets/Download_on_the_App_Store_Badge_US-UK_RGB_blk_092917.CVyK0T4N.svg",i="/assets/1_en_US.CE_aSW8N.png",n="/assets/2_en_US.Dm8rLo_N.png",s="/assets/3_en_US.SUOW4yEM.png",k=JSON.parse('{"title":"Nezha Mobile - An iOS Client For Nezha Dashboard","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/case/case6.md","filePath":"en_US/case/case6.md","lastUpdated":1732521523000}'),l={name:"en_US/case/case6.md"},h=a("",15),c=[h];function d(p,_,g,u,m,b){return o(),t("div",null,c)}const w=e(l,[["render",d]]);export{k as __pageData,w as default};
diff --git a/assets/en_US_case_case7.md.Cop8wLoV.js b/assets/en_US_case_case7.md.Cop8wLoV.js
deleted file mode 100644
index b29f9d6d..00000000
--- a/assets/en_US_case_case7.md.Cop8wLoV.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,a as r}from"./chunks/webssh.Cvv4PsOJ.js";import{_ as a,c as t,o,a4 as n}from"./chunks/framework.BmdFiWrL.js";const _=JSON.parse('{"title":"Broker for Nezha - Advancing on more platforms","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/case/case7.md","filePath":"en_US/case/case7.md","lastUpdated":1732521523000}'),i={name:"en_US/case/case7.md"},l=n('Broker for Nezha - Advancing on more platforms
Difference from Nezha Agent
Writing a Data Collector
Examples
Localization
Introduction
/resource/l10n/en-US.toml
to replace the text in the new feature.en-US.toml
, pull the new text into the configuration files of other languages such as en-US.toml
, and add translations.Adding a new localized text file
',6),l=[i];function r(d,c,s,u,h,f){return a(),t("div",null,l)}const p=e(n,[["render",r]]);export{g as __pageData,p as default};
diff --git a/assets/en_US_developer_l10n.md.XBQ_3Abu.lean.js b/assets/en_US_developer_l10n.md.XBQ_3Abu.lean.js
deleted file mode 100644
index 7fe7adc5..00000000
--- a/assets/en_US_developer_l10n.md.XBQ_3Abu.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o as a,a4 as o}from"./chunks/framework.BmdFiWrL.js";const g=JSON.parse('{"title":"Localization","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/developer/l10n.md","filePath":"en_US/developer/l10n.md","lastUpdated":1732521523000}'),n={name:"en_US/developer/l10n.md"},i=o("",6),l=[i];function r(d,c,s,u,h,f){return a(),t("div",null,l)}const p=e(n,[["render",r]]);export{g as __pageData,p as default};
diff --git a/assets/en_US_developer_theme.md.Cm-AKaYr.js b/assets/en_US_developer_theme.md.Cm-AKaYr.js
deleted file mode 100644
index cd920fcb..00000000
--- a/assets/en_US_developer_theme.md.Cm-AKaYr.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as o,o as t,a4 as a}from"./chunks/framework.BmdFiWrL.js";const _=JSON.parse('{"title":"Nezha Theme Development Environment","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/developer/theme.md","filePath":"en_US/developer/theme.md","lastUpdated":1732521523000}'),n={name:"en_US/developer/theme.md"},c=a('/resource/l10n/
.Nezha Theme Development Environment
dashboard v0.19.20
and newer versions.How to use
data/config.yaml
(The callback connection can be filled with http://localhost
)docker-compose up
theme-custom
(frontend theme), static-custom
(frontend theme static files), and dashboard-custom
(backend theme) into the /opt/nezha/dashboard/
directory on the server (if installed using Docker).FAQ
',7),i=[c];function r(l,d,h,s,m,p){return t(),o("div",null,i)}const f=e(n,[["render",r]]);export{_ as __pageData,f as default};
diff --git a/assets/en_US_developer_theme.md.Cm-AKaYr.lean.js b/assets/en_US_developer_theme.md.Cm-AKaYr.lean.js
deleted file mode 100644
index dbf111cb..00000000
--- a/assets/en_US_developer_theme.md.Cm-AKaYr.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as o,o as t,a4 as a}from"./chunks/framework.BmdFiWrL.js";const _=JSON.parse('{"title":"Nezha Theme Development Environment","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/developer/theme.md","filePath":"en_US/developer/theme.md","lastUpdated":1732521523000}'),n={name:"en_US/developer/theme.md"},c=a("",7),i=[c];function r(l,d,h,s,m,p){return t(),o("div",null,i)}const f=e(n,[["render",r]]);export{_ as __pageData,f as default};
diff --git a/assets/en_US_guide_agent.md.BOJ8LATb.js b/assets/en_US_guide_agent.md.BOJ8LATb.js
deleted file mode 100644
index a53276f0..00000000
--- a/assets/en_US_guide_agent.md.BOJ8LATb.js
+++ /dev/null
@@ -1,66 +0,0 @@
-import{_ as s,c as i,o as a,a4 as e}from"./chunks/framework.BmdFiWrL.js";const F=JSON.parse('{"title":"Install Agent","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/agent.md","filePath":"en_US/guide/agent.md","lastUpdated":1732521523000}'),n={name:"en_US/guide/agent.md"},t=e(`80
, change the configuration in docker-compose.yaml
.Install Agent
One-Click Installation of the Agent
Preparation
Go to the settings page in the admin panel, fill in the communication domain in the “Non-CDN Dashboard Server Domain/IP” field, and click "Save".One-Click Installation on Linux
One-Click Installation on macOS
One-Click Installation on Windows
Other Ways to Install the Agent
Installing the Agent on Linux (Support most distros)
Click to expand/collapse
curl -L https://raw.githubusercontent.com/nezhahq/scripts/main/install_en.sh -o nezha.sh && chmod +x nezha.sh && sudo ./nezha.sh
Installing the Agent using the built-in service command (Support most systems)
Click to expand/collapse
./nezha-agent service install -s server_name:port -p password
./nezha-agent service uninstall
./nezha-agent service start
./nezha-agent service stop
./nezha-agent service restart
Installing the Agent with runit
Click to expand/collapse
/etc/sv/nezha-agent
:mkdir /etc/sv/nezha-agent
/etc/sv/nezha-agent/run
, with following content:#!/bin/sh
-exec 2>&1
-exec /opt/nezha/agent/nezha-agent -s server_name:port -p password 2>&1
/etc/sv/nezha-agent/log/run
:#!/bin/sh
-exec vlogger -t nezha-agent -p daemon
sudo ln -s /etc/sv/nezha-agent/ /var/service
sv
command to manage the service.socklog
and enable it:sudo xbps-install -S socklog-void
-sudo ln -s /etc/sv/socklog-unix /var/service
svlogtail
:sudo svlogtail | grep nezha-agent
Manual Installation of the Agent on Windows
Nezha - Windows Client Installation (Chinese)Installing the Agent on Synology DSM
Click to expand/collapse
Installing Nezha Monitoring Agent on Synology DSM 7.x (Chinese)
Nezha - Synology Client (Agent) Installation Tutorial (Chinese)# Agent path
-EXEC="/PATH/TO/nezha-agent"
-# Log path
-LOG="\${EXEC}.log"
-# Additional execution parameters, can be empty
-ARGS=""
-# Nezha server gRPC address
-SERVER="HOST_OR_IP:gRPC_PORT"
-# The secret key obtained in the previous step
-SECRET="APP_SECRET"
-# User running the service, *strongly recommended to use non-root user*
-RUN_USER="nezha"
-
-# Write to systemd service file
-cat << EOF > /usr/lib/systemd/system/nezha.service
-[Unit]
-Description=Nezha Agent Service
-After=network.target
-
-[Service]
-Type=simple
-ExecStart=/bin/nohup \${EXEC} \${ARGS} -s \${SERVER} -p \${SECRET} &>> \${LOG} &
-ExecStop=ps -fe |grep nezha-agent|awk '{print \\$2}'|xargs kill
-User=\${RUN_USER}
-
-Restart=on-abort
-
-[Install]
-WantedBy=multi-user.target
-EOF
-
-# Reload service
-systemctl daemon-reload
-# Start service
-systemctl start nezha
-# Enable service startup
-systemctl enable nezha
root
account to complete the installation.Installing the Agent on macOS Using Homebrew
Click to expand/collapse
echo 'export HOMEBREW_NEZHA_AGENT_PASSWORD="Communication key, obtained from the service page"' >> ~/.zshrc
-echo 'export HOMEBREW_NEZHA_AGENT_SERVER="Your server and port, format your.domain:5555 "' >> ~/.zshrc
-source ~/.zshrc
brew install brewforge/chinese/nezha-agent
brew services start nezha-agent
brew services info nezha-agent
brew services stop nezha-agent
brew rm nezha-agent
echo $HOMEBREW_NEZHA_AGENT_PASSWORD
-echo $HOMEBREW_NEZHA_AGENT_SERVER
brew services stop nezha-agent
-brew reinstall nezha-agent
-brew services start nezha-agent
Installing the Agent on OpenWRT
Click to expand/collapse
NZ-OpenWrt
Nezha Monitoring for OpenWRT/root
.chmod +x /root/nezha-agent
to grant execution permission, then create /etc/init.d/nezha-service
:#!/bin/sh /etc/rc.common
-
-START=99
-USE_PROCD=1
-
-start_service() {
- procd_open_instance
- procd_set_param command /root/nezha-agent -s Dashboard communication domain:port -p Key -d
- procd_set_param respawn
- procd_close_instance
-}
-
-stop_service() {
- killall nezha-agent
-}
-
-restart() {
- stop
- sleep 2
- start
-}
chmod +x /etc/init.d/nezha-service
to grant execution permission./etc/init.d/nezha-service enable && /etc/init.d/nezha-service start
Does the Agent Have a Docker Image?
The design philosophy of the Agent is opposite to that of the Dashboard. While the Dashboard should minimally impact the server, the Agent needs to execute monitoring services and run commands within the server.
Running the Agent inside a container can still perform monitoring tasks, but features like WebShell will not function properly, so no Docker image is provided.Frequently Asked Questions about the Agent
The IP Displayed in the Admin Panel is Different from the Actual Agent IP?
Errors During One-Click Script Installation
curl: Failed to connect to raw.githubusercontent.com......
sudo: command not found
apt install sudo
API Interface
Creating a Token
Authentication Method
Authorization: Token
for authentication.Request Headers:
-Authorization: Token
Usage Instructions
0000-00-00
. This currently indicates that the Agent has never reported since the Dashboard went online, but it is not recommended to use the positive or negative value to determine the status.GET
, and the response format is JSON
.Get Server List
GET /api/v1/server/list?tag=
tag
(optional): ServerTag is the server group. Provide this parameter to query only servers in that group.{
- "code": 0,
- "message": "success",
- "result": [
- {
- "id": 1,
- "name": "Server1",
- "tag": "Tag1",
- "last_active": 1653014667,
- "ipv4": "1.1.1.1",
- "ipv6": "",
- "valid_ip": "1.1.1.1"
- },
- {
- "id": 2,
- "name": "Server2",
- "tag": "Tag2",
- "last_active": -62135596800,
- "ipv4": "",
- "ipv6": "",
- "valid_ip": ""
- }
- ]
-}
Get Server Details
GET /api/v1/server/details?id=&tag=
id
(optional): ServerID, multiple IDs separated by commas. Provide this parameter to query the server corresponding to that ID and ignore the tag
parameter.tag
(optional): ServerTag, provide this parameter to query only servers in that group.{
- "code": 0,
- "message": "success",
- "result": [
- {
- "id": 1,
- "name": "Server1",
- "tag": "Tag1",
- "last_active": 1653015042,
- "ipv4": "1.1.1.1",
- "ipv6": "",
- "valid_ip": "1.1.1.1",
- "host": {
- "Platform": "darwin",
- "PlatformVersion": "12.3.1",
- "CPU": [
- "Apple M1 Pro 1 Physical Core"
- ],
- "MemTotal": 17179869184,
- "DiskTotal": 2473496842240,
- "SwapTotal": 0,
- "Arch": "arm64",
- "Virtualization": "",
- "BootTime": 1652683962,
- "CountryCode": "hk",
- "Version": ""
- },
- "status": {
- "CPU": 17.33,
- "MemUsed": 14013841408,
- "SwapUsed": 0,
- "DiskUsed": 2335048912896,
- "NetInTransfer": 2710273234,
- "NetOutTransfer": 695454765,
- "NetInSpeed": 10806,
- "NetOutSpeed": 5303,
- "Uptime": 331080,
- "Load1": 5.23,
- "Load5": 4.87,
- "Load15": 3.99,
- "TcpConnCount": 195,
- "UdpConnCount": 70,
- "ProcessCount": 437
- }
- },
- {
- "id": 2,
- "name": "Server2",
- "tag": "Tag2",
- "last_active": -62135596800,
- "ipv4": "",
- "ipv6": "",
- "valid_ip": "",
- "host": {
- "Platform": "",
- "PlatformVersion": "",
- "CPU": null,
- "MemTotal": 0,
- "DiskTotal": 0,
- "SwapTotal": 0,
- "Arch": "",
- "Virtualization": "",
- "BootTime": 0,
- "CountryCode": "",
- "Version": ""
- },
- "status": {
- "CPU": 0,
- "MemUsed": 0,
- "SwapUsed": 0,
- "DiskUsed": 0,
- "NetInTransfer": 0,
- "NetOutTransfer": 0,
- "NetInSpeed": 0,
- "NetOutSpeed": 0,
- "Uptime": 0,
- "Load1": 0,
- "Load5": 0,
- "Load15": 0,
- "TcpConnCount": 0,
- "UdpConnCount": 0,
- "ProcessCount": 0
- }
- }
- ]
-}
Get ICMP Ping / TCPing monitor value
HideForGuest
option enabled.GET /api/v1/monitor/{id}
id
(required): ServerID, must be an positive integer.{
- "code": 0,
- "message": "success",
- "result": [
- {
- "monitor_id": 1,
- "server_id": 1,
- "monitor_name": "Monitor1",
- "server_name": "Server1",
- "created_at": [
- 1722142860000,
- 1722142920000
- ],
- "avg_delay": [
- 68.2275,
- 70.1129
- ]
- },
- {
- "monitor_id": 2,
- "server_id": 1,
- "monitor_name": "Monitor2",
- "server_name": "Server1",
- "created_at": [
- 1722142860000,
- 1722142920000
- ],
- "avg_delay": [
- 66.656,
- 68.2153
- ]
- },
- {
- "monitor_id": 3,
- "server_id": 1,
- "monitor_name": "Monitor3",
- "server_name": "Server1",
- "created_at": [
- 1722142860000,
- 1722142920000
- ],
- "avg_delay": [
- 61.4525,
- 62.342
- ]
- }
- ]
-}
created_at
corresponds with avg_delay
.Automatic Node Registration
POST /api/v1/server/register?simple=1
simple
: (optional): Specifies the format of the response data. simple=1
or simple=true
, the response will only include a Token string (e.g., 8GYwaxYuLfU7zl7ndC
).{"code": 200, "message": "Server created successfully","secret": "8GYwaxYuLfU7zl7ndC"}
{}
), and the system will apply the following default values:Name
: Defaults to the node’s IP address ($IP).Tag
: Defaults to "AutoRegister".Note
: Defaults to an empty string ("").HideForGuest
: Defaults to "on". "Name": "abcd", // Node name
- "Tag": "", // Tag (optional)
- "Note": "", // Note or description (optional)
- "HideForGuest": "on" // Whether to hide from guests
Usage Examples
Get All Server Information
import requests
-
-url = "http://your-dashboard/api/v1/server/list"
-headers = {
- "Authorization": "your_token"
-}
-
-response = requests.get(url, headers=headers)
-data = response.json()
-
-for server in data['result']:
- print(f"Server Name: {server['name']}, Last Active: {server['last_active']}, IP: {server['valid_ip']}")
Get Specific Server Details
import requests
-
-server_id = 1 # Replace with your server ID
-url = f"http://your-dashboard/api/v1/server/details?id={server_id}"
-headers = {
- "Authorization": "your_token"
-}
-
-response = requests.get(url, headers=headers)
-data = response.json()
-
-server = data['result'][0]
-print(f"Server Name: {server['name']}")
-print(f"CPU Usage: {server['status']['CPU']}%")
-print(f"Memory Used: {server['status']['MemUsed']} bytes")
-print(f"Disk Used: {server['status']['DiskUsed']} bytes")
-print(f"Network In Speed: {server['status']['NetInSpeed']} bytes/s")
-print(f"Network Out Speed: {server['status']['NetOutSpeed']} bytes/s")
Automatic Node Registration
nezha_register.sh
, and copy the following content into it:#!/bin/bash
-
-# Exit if NEZHA_TOKEN is not set
-if [ -z "\${NEZHA_TOKEN}" ]; then
- echo "NEZHA_TOKEN is not set. Exiting."
- exit 0
-fi
-
-# Set default values if variables are not set
-NEZHA_PROBE_ADDRESS="\${NEZHA_PROBE_ADDRESS:-probe.example.com}"
-NEZHA_PROBE_PORT="\${NEZHA_PROBE_PORT:-5555}"
-NEZHA_DASHBOARD_URL="\${NEZHA_DASHBOARD_URL:-https://nezha.example.com}"
-
-NODE_NAME=\${NODE_NAME:-$(hostname)}
-
-# Send POST request and capture response and HTTP status code
-response=$(curl -s -o response_body.json -w "%{http_code}" -X POST "\${NEZHA_DASHBOARD_URL}/api/v1/server/register?simple=true" \\
- -H "Content-Type: application/json" \\
- -H "Authorization: \${NEZHA_TOKEN}" \\
- -d "{\\"Name\\": \\"\${NODE_NAME}\\"}")
-
-# Extract HTTP status code and Nezha secret
-HTTP_CODE="$response"
-NEZHA_SECRET=$(cat response_body.json)
-rm -f response_body.json
-
-if [ "$HTTP_CODE" != "200" ]; then
- echo "Failed to get Nezha Secret. HTTP status code: $HTTP_CODE"
- exit 1
-fi
-
-# Additional check for NEZHA_SECRET to ensure it's not JSON-formatted (indicating failure)
-if [[ "\${NEZHA_SECRET:0:1}" == "{" ]]; then
- echo "Failed to get Nezha Secret. Received response: \${NEZHA_SECRET}"
- exit 1
-fi
-
-# Download and execute the install script with cleanup
-curl -fsSL https://raw.githubusercontent.com/nezhahq/scripts/main/install_en.sh -o nezha.sh && \\
-chmod +x nezha.sh || { echo "Failed to download or make the script executable"; exit 1; }
-
-# Clean up nezha.sh on exit
-trap 'rm -f nezha.sh' EXIT
-
-# Run the Nezha agent installation script
-CMD="./nezha.sh install_agent "\${NEZHA_PROBE_ADDRESS}" "\${NEZHA_PROBE_PORT}" "\${NEZHA_SECRET}" --tls"
-
-echo "Run commnad : \${CMD}"
-
-eval $CMD
chmod +x nezha_register.sh
Token
from Nezha dashboard, e.g. POXbxorKJBM8wPMKX8r2PdMblyXvpggB
export NEZHA_TOKEN="POXbxorKJBM8wPMKX8r2PdMblyXvpggB" # Obtain this token from the dashboard
-export NEZHA_PROBE_ADDRESS="your_probe_address" # Set your probe address
-export NEZHA_DASHBOARD_URL="https://nezha.example.com" # Replace with your dashboard URL
-export NEZHA_PROBE_PORT="5555" # Modify this port if needed
./nezha_register.sh
Run command: ./nezha.sh install_agent probe.example.com 5555 YOUR_SECRET --tls
Install Dashboard
Preparation
If you do not need to use ports 80 and 443 to access the Dashboard, you can directly use the installation script to install and run Nezha Monitoring without installing Nginx.Obtaining Github Client ID and Secret
Application name
- Fill in as you like.Homepage URL
- Fill in with the domain for accessing the dashboard, such as "http://dashboard.example.com" (your domain).Authorization callback URL
- Fill in with the callback address, such as "http://dashboard.example.com/oauth2/callback" (don't forget /oauth2/callback
).Using Cloudflare Access as an OAuth2 Provider
Setting Up a New SaaS-OIDC Application
My Team
-> Users
-> <specific user>
-> Get User ID
and save it.Access
-> Application
-> Add an Application
.SaaS
, enter a custom application name in Application
(e.g., nezha), select OIDC
, and click Add application
.Scopes
: openid
, email
, profile
, groups
.Redirect URLs
, such as https://dashboard.example.com/oauth2/callback
.Client ID
, Client Secret
, and Issuer
address (protocol and domain part), e.g., https://xxxxx.cloudflareaccess.com
./opt/nezha/dashboard/data/config.yaml
, and change the Endpoint
configuration to the Issuer
address saved earlier, e.g., https://xxxxx.cloudflareaccess.com
, and restart the Dashboard.OIDC Authentication Configuration (Optional)
Installing the Dashboard on the Server
curl -L https://raw.githubusercontent.com/nezhahq/scripts/main/install_en.sh -o nezha.sh && chmod +x nezha.sh && sudo ./nezha.sh
OAuth provider
- choose one from github, cloudflare, gitlab, gitee.Client ID
- the previously saved Client ID.Client Secret
- the previously saved Client Secret.Username
- the username/User ID from the OAuth provider.Site title
- custom site title.Access port
- public access port, customizable, default is 8008.Agent communication port
- port for Agent and Dashboard communication, default is 5555.
When the installation completes, you can access the dashboard by visiting your domain and port number, such as “http://dashboard.example.com:8008”../nezha.sh
Configuring Reverse Proxy
http://127.0.0.1
in the "Target URL" below, then click “Save”.#PROXY-START/
-location / {
- proxy_pass http://127.0.0.1:8008;
- proxy_set_header Host $http_host;
- proxy_set_header Upgrade $http_upgrade;
-}
-location ~ ^/(ws|terminal/.+|file/.+)$ {
- proxy_pass http://127.0.0.1:8008;
- proxy_http_version 1.1;
- proxy_set_header Upgrade $http_upgrade;
- proxy_set_header Connection "Upgrade";
- proxy_set_header Host $http_host;
-}
-#PROXY-END/
Now you should be able to access the dashboard directly using the domain, such as “http://dashboard.example.com”.Additional Content:
proxy /ws http://ip:8008 {
- websocket
- header_upstream -Origin
-}
-proxy /terminal/* http://ip:8008 {
- websocket
- header_upstream -Origin
-}
-proxy /file/* http://ip:8008 {
- websocket
- header_upstream -Origin
-}
Configuring SSL in the aaPanel
Like configuring SSL certificates for other websites, enter the “SSL” in the site settings, and you can choose to automatically apply for a Let’s Encrypt certificate or manually configure an existing certificate.
After completing the SSL settings, go back to Github OAuth Apps and edit the previously created OAuth application. Change all the domain parts in "Homepage URL" and "Authorization callback URL" from http
to https
, such as "https://dashboard.example.com" and "https://dashboard.example.com/oauth2/callback". Failing to change this may result in being unable to log in to the admin panel.Updating the Dashboard
./nezha.sh
, and select to restart and update the dashboard.Frequently Asked Questions about the Dashboard
Why is the IP Displayed in the Admin Panel Different from the Actual Agent IP?
If you find that the IP displayed in the Admin panel is different from the IP provided by your service provider, it is most likely that the service provider has given you an entry IP, but the Agent is testing your exit IP. This issue can also occur with multi-line servers and IPLC private lines.curl https://ipapi.co/ip/
-curl ip.sb
-curl ip-api.com
Forgot or Deleted Viewing Password
/opt/nezha/dashboard/data/config.yaml
file.
The password is located under the site-viewpassword
item.Dashboard Installation/Restart/Update Failure: iptables ......
systemctl status docker
-systemctl restart docker
-systemctl status docker
If iptables errors persist, consider disabling or removing iptables.
This issue might also be related to the kernel, so switching to the official kernel can be another solution.Dashboard Restart Failure: Invalid hostPort: nz_site_port
/opt/nezha/dashboard/docker-compose.yaml
file.Dashboard Layout Error, CSS Resources Not Loading
To resolve this, try restarting and updating the Dashboard
first.
If the problem persists after updating, the issue may be due to inappropriate configurations in your vhost file. You can edit the Nginx vhost file or use the aaPanel to:Websites
and click Settings
on the right.Configuration File
and remove the following lines:location ~ .*\\.(js|css)?$
- {
- expires 12h;
- error_log /dev/null;
- access_log /dev/null;
- }
Dashboard Cannot Start: panic: Unable to find the configured DDNS provider...
webhook
, cloudflare
, tencentcloud
, and dummy
are supported.Dashboard DDNS Update Crash: panic: interface conversion: interface {} is nil, not []interface {}
AccessID
or AccessSecret
is incorrect.Dashboard warning: NEZHA>> 错误的服务监控上报...
TaskType
that is unsupported by its counterpart. Updating both to the latest version could solve this problem.Unable to start the Agent service: Unix syslog delivery error
/dev/log
socket (or it does not exist at all). You can refer to https://unix.stackexchange.com/questions/317064/how-do-i-restore-dev-log-in-systemdrsyslog-host to solve this problem. Try avoiding using init systems like systemd
in a Docker installation.Network Monitoring Page Shows: server monitor history not found
If it has been set up, wait for some time and then check again.What to do if /terminal or /ws can't connect properly after enabling HTTPS?
x509:certificate signed by unknown authority
, replacing with a complete certificate will solve the problem.What if I'm not satisfied with the data modification/addition functionality provided by the dashboard and want to modify/add data myself?
Note that not everything in the database can be modified; incorrect modifications can lead to data corruption and inability to start the Dashboard. Do not modify the database casually!
The database type is sqlite3, located at /opt/nezha/dashboard/data/sqlite.db
. Backup before modifying.Will the Dashboard automatically update?
Agent Command Issuance Failed
When Connecting to Web Terminal DDNS
Why should I use Nezha's DDNS functionality?
Configuration
webhook
provider, the corresponding options need to be filled out as required. For detailed instructions, see Webhook Configuration.Webhook Configuration
GET
, POST
, PATCH
, DELETE
, and PUT
.JSON
or Form
.GET
and DELETE
. If you need to use a nested format, you must choose JSON
as the request type.#ip#
: Host IP.#domain#
: DDNS domain. Each request is made separately for each domain, so this value will be a single domain string.#type#
: IP type, either "ipv4"
or "ipv6"
.#record#
: Record type, either "A"
or "AAAA"
.#access_id#
: DDNS Credential 1.#access_secret#
: DDNS Credential 2.Oray Webhook Example
Click to expand/collapse
http://ddns.oray.com/ph/update?hostname=#domain#&myip=#ip#
GET
{"Authorization": "Basic pass"}
, replace pass
with the Base64-encoded userpass (e.g., user:pass
becomes dXNlcjpwYXNzCg==
).Provider List
Provider Credential 1 (ID) Credential 2 (Secret) dummy
❌️ ❌️ webhook
Optional Optional cloudflare
❌️ ✅ tencentcloud
✅ ✅ Viewing Logs
dashboard_1 | 2024/03/16 23:16:25 NEZHA>> 正在尝试更新域名(ddns.example.com)DDNS(1/3)
-dashboard_1 | 2024/03/16 23:16:28 NEZHA>> 尝试更新域名(ddns.example.com)DDNS成功
FAQ about logging into the Dashboard
Stuck Page/Connection Refused/Long Response Time After Login Callback
://
is the protocol, usually http
or https
. Since there may be multiple protocol+domain+port combinations available for accessing the Dashboard in a normal deployment, make sure to choose the most appropriate one as the callback.How to Check if My Callback Address is Wrong?
Ensure that your path is /oauth2/callback
, all in lowercase.Errors After Logging into the Admin Panel
http: named cookie not present
lookup xxx
It is recommended to restart Docker first, sudo systemctl restart docker
, then restart the Dashboard using the script.
If the lookup error persists, check if there are other tools controlling iptables, such as firewall.
This issue might also be related to the kernel, so try switching to the official kernel.Invalid authorization method, or the login callback address is invalid, expired, or has been revoked
oauth2: server response missing access_token
If unresolved, switching to Github or another method is recommended.The user is not an admin of this site and cannot log in
For Cloudflare Access users, note that your username is not an email but a User ID.dial tcp xxx:443 i/o timeout
sudo systemctl restart docker
, then restart the Dashboard using the script.
If you are configuring Github login on a server in mainland China, switching to Cloudflare Access is recommended to avoid network interference.net/http: TLS handshake timeout
Unable to receive email verification codes using Cloudflare Access as an OAuth2 Provider
',24),n=[r];function s(l,c,h,d,u,g){return o(),a("div",null,n)}const b=e(i,[["render",s]]);export{p as __pageData,b as default};
diff --git a/assets/en_US_guide_loginq.md.DK7A_9g2.lean.js b/assets/en_US_guide_loginq.md.DK7A_9g2.lean.js
deleted file mode 100644
index 39fb3990..00000000
--- a/assets/en_US_guide_loginq.md.DK7A_9g2.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as a,o,a4 as t}from"./chunks/framework.BmdFiWrL.js";const p=JSON.parse('{"title":"FAQ about logging into the Dashboard","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/loginq.md","filePath":"en_US/guide/loginq.md","lastUpdated":1732521523000}'),i={name:"en_US/guide/loginq.md"},r=t("",24),n=[r];function s(l,c,h,d,u,g){return o(),a("div",null,n)}const b=e(i,[["render",s]]);export{p as __pageData,b as default};
diff --git a/assets/en_US_guide_nat.md.Bm1a6Dwz.js b/assets/en_US_guide_nat.md.Bm1a6Dwz.js
deleted file mode 100644
index a872df12..00000000
--- a/assets/en_US_guide_nat.md.Bm1a6Dwz.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o as a,a4 as o}from"./chunks/framework.BmdFiWrL.js";const g=JSON.parse('{"title":"NAT Traversal Configuration","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/nat.md","filePath":"en_US/guide/nat.md","lastUpdated":1732521523000}'),i={name:"en_US/guide/nat.md"},n=o('Policies
.NAT Traversal Configuration
Preparation
service-1.example.com
. Point this domain name to the public IP address of your Dashboard server.service-2.example.com
.service-1.example.com
) is correctly linked to the Dashboard site within these tools to avoid access issues.NAT Traversal Configuration Steps
OpenWrt Login Page
.IP:port
format, such as 127.0.0.1:80
.service-1.example.com
. If a non-standard port is used by Dashboard (e.g. default 8008
), be sure to include it as well.http://service-1.example.com
to ensure the internal service on the Agent is successfully accessible.Usage Notes
',10),r=[n];function s(l,c,d,u,h,p){return a(),t("div",null,r)}const f=e(i,[["render",s]]);export{g as __pageData,f as default};
diff --git a/assets/en_US_guide_nat.md.Bm1a6Dwz.lean.js b/assets/en_US_guide_nat.md.Bm1a6Dwz.lean.js
deleted file mode 100644
index 0ec1d552..00000000
--- a/assets/en_US_guide_nat.md.Bm1a6Dwz.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o as a,a4 as o}from"./chunks/framework.BmdFiWrL.js";const g=JSON.parse('{"title":"NAT Traversal Configuration","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/nat.md","filePath":"en_US/guide/nat.md","lastUpdated":1732521523000}'),i={name:"en_US/guide/nat.md"},n=o("",10),r=[n];function s(l,c,d,u,h,p){return a(),t("div",null,r)}const f=e(i,[["render",s]]);export{g as __pageData,f as default};
diff --git a/assets/en_US_guide_notifications.md.CazaFiKA.js b/assets/en_US_guide_notifications.md.CazaFiKA.js
deleted file mode 100644
index 1e720325..00000000
--- a/assets/en_US_guide_notifications.md.CazaFiKA.js
+++ /dev/null
@@ -1,47 +0,0 @@
-import{_ as i,c as s,o as t,a4 as a}from"./chunks/framework.BmdFiWrL.js";const E=JSON.parse('{"title":"Notification Configuration","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/notifications.md","filePath":"en_US/guide/notifications.md","lastUpdated":1732521523000}'),e={name:"en_US/guide/notifications.md"},n=a(`Notification Configuration
Flexible Notification Methods
#DATETIME#
represents the timestamp of the event. When the notification is triggered, the Dashboard automatically replaces #DATETIME#
with the actual event time.#NEZHA#
is a placeholder for Dashboard messages, and the Dashboard automatically replaces the placeholder with the actual message when the notification is triggered.JSON
format: When the request type is FORM, the value is in key:value
form, and placeholders can be placed inside value
. The placeholders will be automatically replaced during notification. When the request type is JSON, only simple string replacement is performed before being submitted to the URL
.Bark Example
Click to expand/collapse
{"title": "#SERVER.NAME#","device_key":"xxxxxxxxx","body":"#NEZHA#","icon":"https://xxxxxxxx/nz.png"}
Slack Example Contributor: @白歌
Click to expand/collapse
URL Parameter Acquisition Instructions
{"text":"#NEZHA#"}
Telegram Example Contributor: @白歌
Click to expand/collapse
URL Parameter Acquisition Instructions
Email Notification Example - Outlook Contributor: @白歌
Click to expand/collapse
URL Parameter Acquisition Instructions
<replaceAPIUSER>
and <replaceAPIKEY>
in the example URL below with your APIUSER and APIKEY, and replace <customSenderEmail>
and <customRecipientEmail>
with any sender and recipient email addresses.{"Content-type":"application/json", "Authorization":"Bearer {Token}"}
{
- "message": {
- "subject": "Server Status Notification",
- "body": {
- "contentType": "Text",
- "content": "#NEZHA#"
- },
- "toRecipients": [
- {
- "emailAddress": {
- "address": "ADDRESS FOR RECEVING EMAILS"
- }
- }
- ]
- }
-}
DingTalk Group Bot Configuration Example
Click to expand/collapse
URL Parameter Acquisition Instructions
{"Content-Type": "application/json"}
{"msgtype": "text","text": {"content":"Nezha Probe:\\n#NEZHA#"}}
WeChat Work Group Bot Example Contributor: @ChowRex
Click to expand/collapse
{
- "content": "#NEZHA#",
- "ServerName": "#SERVER.NAME#",
- "ServerIP": "#SERVER.IP#",
- "ServerIPV4": "#SERVER.IPV4#",
- "ServerIPV6": "#SERVER.IPV6#",
- "CPU": "#SERVER.CPU#",
- "MEM": "#SERVER.MEM#",
- "SWAP": "#SERVER.SWAP#",
- "DISK": "#SERVER.DISK#",
- "NetInSpeed": "#SERVER.NETINSPEED#",
- "NetOutSpeed": "#SERVER.NETOUTSPEED#",
- "TransferIn": "#SERVER.TRANSFERIN#",
- "TranferOut": "#SERVER.TRANSFEROUT#",
- "Load1": "#SERVER.LOAD1#",
- "Load5": "#SERVER.LOAD5#",
- "Load15": "#SERVER.LOAD15#",
- "TCP_CONN_COUNT": "#SERVER.TCPCONNCOUNT", # invalid
- "UDP_CONN_COUNT": "#SERVER.UDPCONNCOUNT", # invalid
-}
{
- "msgtype": "markdown",
- "markdown": {
- "content": "# Nezha Notification\\n\\n\\"#NEZHA#\\"\\n\\n> Name: \\"#SERVER.NAME#\\"\\n> IP: \\"#SERVER.IP#\\"\\n> IPv4: \\"#SERVER.IPV4#\\"\\nIPv6: \\"#SERVER.IPV6#\\"\\n> CPU: \\"#SERVER.CPU#\\"\\n> Memory: \\"#SERVER.MEM#\\"\\n> Swap: \\"#SERVER.SWAP#\\"\\n> Disk: \\"#SERVER.DISK#\\"\\n> Upload Speed: \\"#SERVER.NETINSPEED#\\"\\n> Download Speed: \\"#SERVER.NETOUTSPEED#\\"\\n> Total Upload: \\"#SERVER.TRANSFERIN#\\"\\n> Total Download: \\"#SERVER.TRANSFEROUT#\\"\\n> Load1: \\"#SERVER.LOAD1#\\"\\n> Load5: \\"#SERVER.LOAD5#\\"\\n> Load15: \\"#SERVER.LOAD15#\\"\\n> TCP Connection Count: \\"#SERVER.TCPCONNCOUNT\\"\\n> UDP Connection Count: \\"#SERVER.UDPCONNCOUNT\\"\\n\\n"
- }
-}
Matrix Notification Example
Click to expand/collapse
Parameter Explanation
$
need to be replaced with your actual values. YOUR_HOME_SERVER
: The address of your Matrix server.YOUR_NEZHA_URL
: The URL of your Nezha dashboard.YOUR_MATRIX_USERNAME
and YOUR_MATRIX_PASSWD
: Your Matrix username and password.YOUR_MATRIX_TOKEN
retrieval method:curl -XPOST -d '{"type": "m.login.password", "identifier": {"user": "$YOUR_MATRIX_USERNAME", "type": "m.id.user"}, "password": "$YOUR_MATRIX_PASSWD"}' "https://$YOUR_HOME_SERVER/_matrix/client/r0/login"
Request Configuration
https://$YOUR_HOME_SERVER/_matrix/client/r0/rooms/$ROOM_ID/send/m.room.message
POST
JSON
{
- "Authorization": "Bearer $YOUR_MATRIX_TOKEN"
-}
{
- "msgtype": "m.text",
- "format": "org.matrix.custom.html",
- "formatted_body": "<html><head><title>Nezha Dashboard</title></head><body><h1><a href=\\"$YOUR_NEZHA_URL\\" target=\\"_blank\\">Nezha Dashboard</a></h1><ul><li>datetime: #DATETIME#</li><li>Message: #NEZHA#</li></ul></body></html>",
- "body": "#NEZHA#"
-}
Usage Steps
$YOUR_HOME_SERVER
, $YOUR_NEZHA_URL
, $YOUR_MATRIX_USERNAME
, $YOUR_MATRIX_PASSWD
, and $YOUR_MATRIX_TOKEN
with your own values.curl
command provided above to get YOUR_MATRIX_TOKEN
, making sure to replace all relevant variables.Notification Rule Explanation
Basic Rules
type
: You can choose one or more types. If multiple types are selected in one rule, all selected types must be satisfied to trigger a notification (refer to the examples below) cpu
, gpu
, memory
, swap
, disk
net_in_speed
inbound network speed, net_out_speed
outbound network speed, net_all_speed
total network speed, transfer_in
inbound traffic, transfer_out
outbound traffic, transfer_all
total trafficoffline
offline monitoringload1
, load5
, load15
loadprocess_count
process count (currently resource-intensive due to thread count, not supported temporarily)tcp_conn_count
, udp_conn_count
connection counttemperature_max
maximum temperatureduration
: Duration in seconds. An notification is triggered if 30% or more of the samples exceed the threshold within this duration (to prevent data spikes).min
or max
: cover
: 0
monitors all servers, use ignore
to exclude specific servers1
ignores all servers, use ignore
to monitor specific servers
Example: [{"type":"offline","duration":10, "cover":0, "ignore":{"5": true}}]
ignore
: Select specific servers to exclude, used with cover
, content is server ID and boolean value, e.g., {"1": true, "2": false}
[{"Type":"offline","Duration":10}]
[{"Type":"cpu","Min":0,"Max":50,"Duration":10},{"Type":"memory","Min":20,"Max":0,"Duration":20}]
There are 4 servers (1, 2, 3, 4) and two notification groups (A, B).
Notify group A if servers 1 and 2 are offline for 10 minutes.
Notify group B if servers 3 and 4 are offline for 10 minutes.[{"type":"offline","duration":600,"cover":1,"ignore":{"1":true,"2":true}}]
[{"type":"offline","duration":600,"cover":1,"ignore":{"3":true,"4":true}}]
Special: Any Cycle Traffic notification
type
: transfer_in_cycle
inbound traffic during the cycletransfer_out_cycle
outbound traffic during the cycletransfer_all_cycle
total traffic during the cyclecycle_start
: The start date of the statistical cycle (can be the start date of your server billing cycle). The time format is RFC3339, e.g., Beijing time is 2022-01-11T08:00:00.00+08:00
cycle_interval
: The number of statistical cycle units (e.g., if the cycle unit is days, and this value is 7, it means statistics are collected every 7 days)cycle_unit
: Statistical cycle unit, default is hour
, optional (hour
, day
, week
, month
, year
)min/max
, cover
, ignore
refer to basic rule configuration[{"type":"transfer_out_cycle","max":1099511627776,"cycle_start":"2022-01-01T00:00:00+08:00","cycle_interval":1,"cycle_unit":"month","cover":1,"ignore":{"3":true,"4":true}}]
Notification Trigger Modes
Set Tasks to Execute on notifications
`,47),o=[n];function l(h,r,p,k,d,u){return t(),s("div",null,o)}const g=i(e,[["render",l]]);export{E as __pageData,g as default};
diff --git a/assets/en_US_guide_notifications.md.CazaFiKA.lean.js b/assets/en_US_guide_notifications.md.CazaFiKA.lean.js
deleted file mode 100644
index 1dd038af..00000000
--- a/assets/en_US_guide_notifications.md.CazaFiKA.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as i,c as s,o as t,a4 as a}from"./chunks/framework.BmdFiWrL.js";const E=JSON.parse('{"title":"Notification Configuration","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/notifications.md","filePath":"en_US/guide/notifications.md","lastUpdated":1732521523000}'),e={name:"en_US/guide/notifications.md"},n=a("",47),o=[n];function l(h,r,p,k,d,u){return t(),s("div",null,o)}const g=i(e,[["render",l]]);export{E as __pageData,g as default};
diff --git a/assets/en_US_guide_q10.md.osfoQDQi.js b/assets/en_US_guide_q10.md.osfoQDQi.js
deleted file mode 100644
index 49e52c79..00000000
--- a/assets/en_US_guide_q10.md.osfoQDQi.js
+++ /dev/null
@@ -1,15 +0,0 @@
-import{_ as i,c as s,o as a,a4 as e}from"./chunks/framework.BmdFiWrL.js";const u=JSON.parse('{"title":"Configuring OIDC Authentication","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/q10.md","filePath":"en_US/guide/q10.md","lastUpdated":1732521523000}'),n={name:"en_US/guide/q10.md"},t=e(`Configuring OIDC Authentication
config.yaml
to configure OIDC (OpenID Connect) authentication. OIDC is an authentication layer on top of the OAuth 2.0 protocol, which allows applications to verify the identity of users based on the authentication performed by an Authorization Server. This method is widely used to implement single sign-on for modern applications.Configuration Details
config.yaml
:oauth2:
- type: oidc # (Required) Specifies the authentication type as OIDC
- oidcDisplayName: OIDC # (Optional, default: OIDC) The name displayed on the login page
- admin: "" # (Fill at least one of admin or adminGroups; default: empty) Admin usernames, separated by commas. Users listed here are considered administrators
- adminGroups: "" # (Fill at least one of admin or adminGroups; default: empty) Admin groups, separated by commas. Users in these groups are considered administrators. Omit if group management is not used
- clientid: # (Required) OIDC client ID
- clientsecret: # (Required) OIDC client secret
- oidcIssuer: https://auth.example.com/realms/master # (Required) The issuer URL of the OIDC provider, obtainable from your OIDC provider
- # oidcLogoutUrl: https://auth.example.com/realms/master/protocol/openid-connect/logout # (Currently unusable due to a bug)
- # oidcRegisterUrl: # (Optional) Registration link provided by the OIDC provider
- oidcScopes: openid,profile,email # (Optional, default: openid,profile,email) OIDC scopes requested, separated by commas
- oidcLoginClaim: sub # (Optional, default: sub) The username field returned by OIDC, can be preferred_username, sub, or email
- oidcGroupsClaim: groups # (Required if using adminGroups, default: groups) The user group information field returned by OIDC, can be groups or roles
- oidcAutoCreate: false # (Optional, default: false) Whether to automatically create a user if they do not exist
- oidcAutoLogin: false # (Optional, default: false) Whether to automatically redirect to the OIDC login page when the path is /login
Customize GeoIP database
pkg/geoip/geoip.db
to compile the Dashboard with working GeoIP queries.continent
:Continent codecontinent_name
:Continent namecountry
:Country codecountry_name
:Country namecountry
field, and other fields can be left empty.Editing database
mmdbwriter
and maxminddb-golang
libraries for editing mmdb databases as of now.ReplaceWith
inserter (which is the default value) and write the mmdbtype.Map
information:subnet := &net.IPNet{
- IP: net.ParseIP("114.5.1.4")
- Mask: net.CIDRMask(32, 32)
-}
-countryMap := make(map[mmdbtype.String]mmdbtype.String)
-
-countryMap[mmdbtype.String("country")] = mmdbtype.String("JP")
-
-if err := writer.Insert(subnet, mmdbtype.Map(countryMap)); err != nil {
- return err
-}
writer
is an instance of *mmdbwriter.Tree
and needs to be created using the mmdbwriter.New
method. The specific operations and the process of inserting other fields are not mentioned here.mmdbinspect
CLI tool to test the database information. For example:$ mmdbinspect -db ./country.mmdb 114.5.1.4
-[
- {
- "Database": "./country.mmdb",
- "Records": [
- {
- "Network": "114.5.1.4/32",
- "Record": {
- "continent": "AS",
- "continent_name": "Asia",
- "country": "JP",
- "country_name": "Japan"
- }
- }
- ],
- "Lookup": "114.5.1.4"
- }
-]
grpcurl
or similar tools:grpcurl -proto ./nezha/proto/nezha.proto -plaintext -H 'client_secret: test' -d '{"ip": "114.5.1.4"}' 127.0.0.1:5555 proto.NezhaService/LookupGeoIP
Agent Startup/Online Troubleshooting Process
',2),s=[a];function i(c,l,d,h,p,u){return o(),t("div",null,s)}const f=e(r,[["render",i]]);export{g as __pageData,f as default};
diff --git a/assets/en_US_guide_q2.md.BdYRs_s9.lean.js b/assets/en_US_guide_q2.md.BdYRs_s9.lean.js
deleted file mode 100644
index 7bce3d61..00000000
--- a/assets/en_US_guide_q2.md.BdYRs_s9.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o,a4 as n}from"./chunks/framework.BmdFiWrL.js";const g=JSON.parse('{"title":"Agent Startup/Online Troubleshooting Process","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/guide/q2.md","filePath":"en_US/guide/q2.md","lastUpdated":1732521523000}'),r={name:"en_US/guide/q2.md"},a=n("",2),s=[a];function i(c,l,d,h,p,u){return o(),t("div",null,s)}const f=e(r,[["render",i]]);export{g as __pageData,f as default};
diff --git a/assets/en_US_guide_q3.md.CdXGDmJL.js b/assets/en_US_guide_q3.md.CdXGDmJL.js
deleted file mode 100644
index e2545047..00000000
--- a/assets/en_US_guide_q3.md.CdXGDmJL.js
+++ /dev/null
@@ -1,74 +0,0 @@
-import{_ as s,c as i,o as a,a4 as n}from"./chunks/framework.BmdFiWrL.js";const E=JSON.parse('{"title":"Reverse Proxy gRPC Port (Supports Cloudflare CDN)","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/guide/q3.md","filePath":"en_US/guide/q3.md","lastUpdated":1732521523000}'),t={name:"en_US/guide/q3.md"},l=n(`/opt/nezha/agent/nezha-agent -s DashboardIP or non-CDN domain:RPCPort -p AgentKey -d
to check if the logs indicate timeouts due to DNS or poor network conditions.nc -v Domain/IP RPCPort
or telnet Domain/IP RPCPort
to check for network issues, inspect the inbound and outbound firewalls of the local machine and the panel server. If you cannot determine the issue, you can use the port checking tool provided by https://port.ping.pe/.Reverse Proxy gRPC Port (Supports Cloudflare CDN)
server {
- listen 443 ssl http2;
- listen [::]:443 ssl http2;
- server_name data.example.com; # Your domain that the Agent uses to connect to the Dashboard
-
- ssl_certificate /data/letsencrypt/fullchain.pem; # Path to your domain certificate
- ssl_certificate_key /data/letsencrypt/key.pem; # Path to your domain private key
- ssl_stapling on;
- ssl_session_timeout 1d;
- ssl_session_cache shared:SSL:10m; # This might conflict with other configuration files; comment it out if there are conflicts
- ssl_protocols TLSv1.1 TLSv1.2 TLSv1.3;
-
- underscores_in_headers on;
-
- keepalive_time 24h;
- keepalive_requests 100000;
- keepalive_timeout 120s;
-
- location / {
- grpc_read_timeout 300s;
- grpc_send_timeout 300s;
- grpc_socket_keepalive on;
- grpc_pass grpc://grpcservers;
- }
-}
-
-upstream grpcservers {
- server localhost:5555;
- keepalive 512;
-}
data.example.com:443 { # Your domain that the Agent uses to connect to the Dashboard
- reverse_proxy {
- to localhost:5555
- transport http {
- versions h2c 2
- }
- }
-}
Non-CDN Dashboard server domain/IP
field, enter the domain configured in Nginx or Caddy in the previous step, for example, data.example.com
, and save it./opt/nezha/dashboard/data/config.yaml
file. Modify proxygrpcport
to the port that Nginx or Caddy is listening to, for example, 443
. Since we enabled SSL/TLS in Nginx or Caddy, set tls
to true
. After making these changes, restart the Dashboard.Network
tab and turn on the gRPC
switch. Then, go to the DNS
tab, find the DNS record for the domain configured in Nginx or Caddy to reverse proxy gRPC, and enable the CDN by clicking the orange cloud.gRPC
, it might not be available immediately, and you may need to wait for a while. You can use curl
and nezha-agent -d
to verify:localhost:~/agent# curl -H "content-type: application/grpc+proto" -H "authorization: Bearer test" https://xxx.xxx.ovh -v
-* processing: https://xxx.xxx.ovh
-* Trying [2606:4700:3035::ac43:8bed]:443...
-* Connected to xxx.xxx.ovh (2606:4700:3035::ac43:8bed) port 443
-# ... SSL info
-* using HTTP/2
-* h2 [:method: GET]
-* h2 [:scheme: https]
-* h2 [:authority: xxx.xxx.ovh]
-* h2 [:path: /]
-* h2 [user-agent: curl/8.2.1]
-* h2 [accept: */*]
-* Using Stream ID: 1
-> GET / HTTP/2
-> Host: xxx.xxx.ovh
-> User-Agent: curl/8.4.0
-> Accept: */*
-> content-type: application/grpc+proto
-> authorization: Bearer test
->
-< HTTP/2 405
-< date: Wed, 20 Dec 2023 08:56:27 GMT
-< content-type: application/grpc+proto
-< cf-ray: 8386ac12dabd5ddc-HKG
-< cf-cache-status: DYNAMIC
-< grpc-message: Received a HEADERS frame with :method "GET" which should be POST
-< grpc-status: 13
-< report-to: {"endpoints":[{"url":"https:\\/\\/a.nel.cloudflare.com\\/report\\/v3?s=%2BTjgJvXWyRF11nUOYx9Lq7UDC1xOYBLtjvWrdjVJQIqu9YqnFJeZFran2KRs6zabQc%2BLV8AubNqYRYDb7hQAZe6bglmVz0wQjrb0tNovYf%2B59SAp%2BQfZnH%2BAFDydNT95ZCmTPnKgWetcwQiUfXU%3D"}],"group":"cf-nel","max_age":604800}
-< nel: {"success_fraction":0,"report_to":"cf-nel","max_age":604800}
-< vary: Accept-Encoding
-< server: cloudflare
-< alt-svc: h3=":443"; ma=86400
-<
-* Connection #0 to host xxx.xxx.ovh left intact
-localhost:~/agent# /opt/nezha/agent/nezha-agent -s nezha.xxx.xxx:443 -p YOUR_KEY --tls -d
-NEZHA@2023-12-20 05:14:00>> 检查更新: 0.15.14
-NEZHA@2023-12-20 05:14:01>> 上报系统信息失败: rpc error: code = Unknown desc = EOF # You need to modify the GRPCHost and TLS options in the Dashboard /opt/nezha/dashboard/data/config.yaml
-NEZHA@2023-12-20 05:14:01>> Error to close connection ...
How to Perform Data Migration and Backup Recovery?
',2),c=[n];function i(d,s,_,l,h,p){return a(),o("div",null,c)}const f=e(r,[["render",i]]);export{u as __pageData,f as default};
diff --git a/assets/en_US_guide_q5.md.dmeXbXkT.lean.js b/assets/en_US_guide_q5.md.dmeXbXkT.lean.js
deleted file mode 100644
index 21c625dc..00000000
--- a/assets/en_US_guide_q5.md.dmeXbXkT.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as o,o as a,a4 as t}from"./chunks/framework.BmdFiWrL.js";const u=JSON.parse('{"title":"How to Perform Data Migration and Backup Recovery?","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/guide/q5.md","filePath":"en_US/guide/q5.md","lastUpdated":1732521523000}'),r={name:"en_US/guide/q5.md"},n=t("",2),c=[n];function i(d,s,_,l,h,p){return a(),o("div",null,c)}const f=e(r,[["render",i]]);export{u as __pageData,f as default};
diff --git a/assets/en_US_guide_q6.md.B38wvOHW.js b/assets/en_US_guide_q6.md.B38wvOHW.js
deleted file mode 100644
index 48d5183f..00000000
--- a/assets/en_US_guide_q6.md.B38wvOHW.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as t,c as e,o,a4 as i}from"./chunks/framework.BmdFiWrL.js";const p=JSON.parse('{"title":"How to Reset Traffic Statistics Monthly?","description":"","frontmatter":{},"headers":[],"relativePath":"en_US/guide/q6.md","filePath":"en_US/guide/q6.md","lastUpdated":1732521523000}'),a={name:"en_US/guide/q6.md"},s=i('Stop Dashboard
./opt/nezha
folder, copy it to the same location in the new environment, and extract it.Start Dashboard
.How to Reset Traffic Statistics Monthly?
Notification
page in the management panel.Services
page, where you can see the monthly traffic statistics. The statistics here will not reset when the server restarts.Customizing Agent Monitoring Items
Customize Monitoring for Network Interfaces and Disk Partitions
/opt/nezha/agent/nezha-agent edit
Additional Runtime Parameters
Viewing Supported Parameters
./nezha-agent --help
Configuring Parameters
/etc/systemd/system/nezha-agent.service
and append your desired parameters at the end of the ExecStart=
line:
',11),s=[n];function r(d,c,l,u,h,p){return i(),t("div",null,s)}const f=e(o,[["render",r]]);export{m as __pageData,f as default};
diff --git a/assets/en_US_guide_q7.md.BiLpPPiv.lean.js b/assets/en_US_guide_q7.md.BiLpPPiv.lean.js
deleted file mode 100644
index 7a2dc288..00000000
--- a/assets/en_US_guide_q7.md.BiLpPPiv.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o as i,a4 as a}from"./chunks/framework.BmdFiWrL.js";const m=JSON.parse('{"title":"Customizing Agent Monitoring Items","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/q7.md","filePath":"en_US/guide/q7.md","lastUpdated":1732521523000}'),o={name:"en_US/guide/q7.md"},n=a("",11),s=[n];function r(d,c,l,u,h,p){return i(),t("div",null,s)}const f=e(o,[["render",r]]);export{m as __pageData,f as default};
diff --git a/assets/en_US_guide_q8.md.BbSzgA8n.js b/assets/en_US_guide_q8.md.BbSzgA8n.js
deleted file mode 100644
index 1204d1fe..00000000
--- a/assets/en_US_guide_q8.md.BbSzgA8n.js
+++ /dev/null
@@ -1,6 +0,0 @@
-import{_ as e,c as i,o as t,a4 as o}from"./chunks/framework.BmdFiWrL.js";const f=JSON.parse('{"title":"Cloudflare Access OAuth2 Configuration","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/q8.md","filePath":"en_US/guide/q8.md","lastUpdated":1732521523000}'),a={name:"en_US/guide/q8.md"},c=o(`--report-delay
: Sets the interval for reporting system information. The default is 1 second. To reduce system resource usage, you can set this to 3 (valid range: 1-4 seconds).--skip-conn
: Disables monitoring of network connections. Recommended for servers with high numbers of connections or high CPU usage.--skip-procs
: Disables monitoring of process counts, helping to reduce resource usage of the Agent.--disable-auto-update
: Disables the automatic update feature of the Agent, enhancing security.--disable-force-update
: Disables the forced update feature of the Agent, enhancing security.--disable-command-execute
: Prevents execution of any scheduled tasks or use of the WebSSH on the Agent, enhancing security.--tls
: Enables SSL/TLS encryption. This should be enabled when you are using nginx to reverse proxy the Agent's gRPC connections and nginx is configured with SSL/TLS.--use-ipv6-countrycode
: Forces the use of IPv6 addresses to query country codes. By default, the Agent uses IPv4 addresses to query country codes. If server supports IPv6 and the country code of the IPv4 address is different, can use this parameter.--gpu
: Enables GPU monitoring. Note: Monitoring GPU usage may require installation of additional dependencies. For more details, refer to the documentation: Enable GPU Monitoring.--temperature
: Enables hardware temperature monitoring. Only effective on supported hardware, some VPS may not be able to retrieve temperature information.-d
--debug
: Enables debug mode.-u
--ip-report-period
: Set IP update interval. Is only effective when the set value is higher than --report-delay
. The default is 1800 seconds.-k
--insecure
: Disable TLS certificate integrity check. Useful in situations involving a self-signed certificate.Cloudflare Access OAuth2 Configuration
Example Configuration:
Oauth2:
- Admin: 701b9ea6-9f56-48cd-af3e-cbb4bfc1475c
- ClientID: 3516291f53eca9b4901a01337e41be7dc52f565c8657d08a3fddb2178d13c5bf
- ClientSecret: 0568b67c7b6d0ed51c663e2fe935683007c28f947a27b7bd47a5ad3d8b56fb67
- Endpoint: "https://xxxxx.cloudflareaccess.com"
- Type: cloudflare
Configuration Description:
Parameter Retrieval Method Admin My Team
-> Users
-> <specific user>
-> User ID
ClientID/ClientSecret Access
-> Application
-> Add an Application
-> SaaS
-> OIDC
Endpoint Access
-> Application
-> Application URL
-> Only keep the protocol and domain, no path
Setting Up a New SaaS-OIDC Application
My Team
-> Users
-> Click <specific user>
-> Obtain and save the User ID
. (If this is your first time using Zero Trust, the Users list will be empty, and you can skip this step; users will appear after completing a verification.)Access
-> Applications
-> Add an Application
.SaaS
. In the Application
field, enter a custom application name (e.g., nezha
), select OIDC
, and then click Add application
.Scopes
, select openid
, email
, profile
, groups
.Redirect URLs
, enter your Dashboard Callback URL, such as https://dashboard.example.com/oauth2/callback
.Client ID
, Client Secret
, and the protocol and domain part of the Issuer
address, for example, https://xxxxx.cloudflareaccess.com
./opt/nezha/dashboard/data/config.yaml
), adjust the OAuth2
settings according to the example configuration, and restart the Dashboard service.Identity Verification Strategy Configuration
Access
-> Applications
-> <application name>
-> Policies
. You can choose from various SSO authentication methods, including email OTP and hardware key verification. For detailed configurations, refer to the Cloudflare Zero Trust Documentation.Policy Configuration Example (One-time PIN)
Access
-> Applications
-> <application name>
-> Policies
-> Add a policy
.Policy Name
, for example, OTP
, and set Action
to Allow
.Configure rules
, add a new Include
rule. Select Emails
as the Selector
and enter your email address in the textbox.Save policy
to save the configuration.Testing the Policy
`,16),n=[c];function d(s,l,r,h,p,u){return t(),i("div",null,n)}const m=e(a,[["render",d]]);export{f as __pageData,m as default};
diff --git a/assets/en_US_guide_q8.md.BbSzgA8n.lean.js b/assets/en_US_guide_q8.md.BbSzgA8n.lean.js
deleted file mode 100644
index 2174e66e..00000000
--- a/assets/en_US_guide_q8.md.BbSzgA8n.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as i,o as t,a4 as o}from"./chunks/framework.BmdFiWrL.js";const f=JSON.parse('{"title":"Cloudflare Access OAuth2 Configuration","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/q8.md","filePath":"en_US/guide/q8.md","lastUpdated":1732521523000}'),a={name:"en_US/guide/q8.md"},c=o("",16),n=[c];function d(s,l,r,h,p,u){return t(),i("div",null,n)}const m=e(a,[["render",d]]);export{f as __pageData,m as default};
diff --git a/assets/en_US_guide_q9.md.nn_5dMdf.js b/assets/en_US_guide_q9.md.nn_5dMdf.js
deleted file mode 100644
index f4fc2805..00000000
--- a/assets/en_US_guide_q9.md.nn_5dMdf.js
+++ /dev/null
@@ -1,14 +0,0 @@
-import{_ as i,c as a,o as s,a4 as e}from"./chunks/framework.BmdFiWrL.js";const u=JSON.parse('{"title":"Enable GPU monitoring","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/q9.md","filePath":"en_US/guide/q9.md","lastUpdated":1732521523000}'),n={name:"en_US/guide/q9.md"},t=e(`Send me a code
, and then enter the code received to log in to the Dashboard.User ID
was not specified in Admin
during previous steps, an error message will be displayed after login: "This user is not an administrator of this site and cannot log in." At this point, you need to go to My Team
-> Users
, find the corresponding user, click on the username to get the User ID
, and enter it into the Admin
section of the Dashboard configuration file. After restarting the Dashboard service, try logging in again.Enable GPU monitoring
Enable
From Command-Line Flag
--gpu
flag to the Agent argument. For example:/opt/nezha/agent/nezha-agent -s example.com:5555 -p example --gpu
From configuration file
/opt/nezha/agent/nezha-agent edit
Enable GPU utilization monitoring
NVIDIA
nvidia-smi
utility to get GPU utilization. This utility is included in the official driver by default.nouveau
, then it's not possible to get GPU utilization.AMD
amdgpu
driver and the rocm-smi
utility.rocm-smi
, below are commands to install the utility on these distros:# Arch Linux
-pacman -Sy rocm-smi-lib
-
-# Debian / Ubuntu
-apt install rocm-smi
-
-# Fedora / RHEL 8+
-dnf install rocm-smi
rocm_smi_lib
manually.git
cmake
gcc
rocm_smi_lib
:git clone https://github.com/ROCm/rocm_smi_lib
cd rocm_smi_lib
-mkdir -p build
-cd build
-cmake ..
-make -j $(nproc)
-# Install library file and header; default location is /opt/rocm
-make install
Server Management
Introduction
Adding a Server
Servers in the same group will be displayed together in supported themes. Notes will only be visible in the Admin Panel, so there's no need to worry about leaking information.Installing the Agent
It is recommended to use the one-click installation. After configuring the parameters, click the corresponding system icon in the one-click installation column of the server to copy the installation command and execute it on the respective server.Forced Update
--disable-auto-update
and --disable-force-update
as described in Custom Agent Monitoring Projects.
By default, the Agent will update automatically without intervention. However, if the user disables automatic updates, you can select specific servers for a forced update.
The forced update will not work if disable-force-update
is enabled.Data Columns
WebSSH Terminal
disable-command-execute
is enabled.
It is available for both Linux and Windows and supports Ctrl+Shift+V for pasting.
If the connection fails, refer to Real-Time Channel Disconnection/Online Terminal Connection Failure.FM
Public Note Example
ServerStatus Theme Agent Billing Information Display
Click to expand/collapse
Service Monitoring
Configured service monitors can be viewed on the "Services" page of the homepage, displaying the availability monitoring results for the past 30 days.How to Use
HTTP-GET
: For this type, you should enter a URL as the target, including http://
or https://
. If your target URL is https://
, the SSL certificate of that URL will also be monitored. Notifications will be triggered when the SSL certificate expires or changes. Example: https://example.com.ICMP-Ping
: For this type, you should enter a domain name or IP without a port number. Example: 1.1.1.1 or example.com.TCP-Ping
: For this type, you should enter a domain name or IP with a port number. Example: 1.1.1.1:80 or example.com:22.Latency Change Notification
Maximum Latency
or less than the Minimum Latency
.Trigger Tasks on Notification
Network Latency Chart
Managing Monitors
Settings
Site Title
Administrator List
user1,user2
./opt/nezha/dashboard/data/config.yaml
and reset the new administrator Client ID
and Client Secret
.Theme
Language
Custom Code (style, script)
Example of Changing Progress Bar Color, Background Image, Navigation Bar, etc., in the Default Theme
Click to expand/collapse
<style>
-/* Screen adaptation */
-@media only screen and (min-width: 1200px) {
- .ui.container {
- width: 80% !important;
-}
-}
-
-@media only screen and (max-width: 767px) {
- .ui.card>.content>.header:not(.ui), .ui.cards>.card>.content>.header:not(.ui) {
- margin-top: 0.4em !important;
- }
-}
-
-/* Overall icons */
-i.icon {
- color: #000;
- width: 1.2em !important;
-}
-
-/* Background image */
-body {
- content: " " !important;
- background: fixed !important;
- z-index: -1 !important;
- top: 0 !important;
- right: 0 !important;
- bottom: 0 !important;
- left: 0 !important;
- background-position: top !important;
- background-repeat: no-repeat !important;
- background-size: cover !important;
- background-image: url(https://backgroud.img) !important;
- font-family: Arial,Helvetica,sans-serif !important;
-}
-
-/* Navigation bar */
-.ui.large.menu {
- border: 0 !important;
- border-radius: 0px !important;
- background-color: rgba(255, 255, 255, 55%) !important;
-}
-
-/* Homepage buttons */
-.ui.menu .active.item {
- background-color: transparent !important;
-}
-
-/* Navigation bar dropdown */
-.ui.dropdown .menu {
- border: 0 !important;
- border-radius: 0 !important;
- background-color: rgba(255, 255, 255, 80%) !important;
-}
-
-/* Login button */
-.nezha-primary-btn {
- background-color: transparent !important;
- color: #000 !important;
-}
-
-/* Large card */
-#app .ui.fluid.accordion {
- background-color: #fbfbfb26 !important;
- border-radius: 0.4rem !important;
-}
-
-/* Small card */
-.ui.four.cards>.card {
- border-radius: 0.6rem !important;
- background-color: #fafafaa3 !important;
-}
-
-.status.cards .wide.column {
- padding-top: 0 !important;
- padding-bottom: 0 !important;
- height: 3.3rem !important;
-}
-
-.status.cards .three.wide.column {
- padding-right: 0rem !important;
-}
-
-.status.cards .wide.column:nth-child(1) {
- margin-top: 2rem !important;
-}
-
-.status.cards .wide.column:nth-child(2) {
- margin-top: 2rem !important;
-}
-
-.status.cards .description {
- padding-bottom: 0 !important;
-}
-
-/* Server name */
-.status.cards .flag {
- margin-right: 0.5rem !important;
-}
-
-/* Popup card icon */
-.status.cards .header > .info.icon {
- margin-right: 0 !important;
-}
-
-.nezha-secondary-font {
- color: #2175ba !important;
-}
-
-/* Upload/download */
-.status.cards .outline.icon {
- margin-right: 1px !important;
-}
-
-i.arrow.alternate.circle.down.outline.icon {
- color: #2175ba !important;
-}
-
-i.arrow.alternate.circle.up.outline.icon {
- color: red !important;
-}
-
-/* Popup card small arrow */
-.ui.right.center.popup {
- margin: -3px 0 0 0.914286em !important;
- -webkit-transform-origin: left 50% !important;
- transform-origin: left 50% !important;
-}
-
-.ui.bottom.left.popup {
- margin-left: 1px !important;
- margin-top: 3px !important;
-}
-
-.ui.top.left.popup {
- margin-left: 0 !important;
- margin-bottom: 10px !important;
-}
-
-.ui.top.right.popup {
- margin-right: 0 !important;
- margin-bottom: 8px !important;
-}
-
-.ui.left.center.popup {
- margin: -3px .91428571em 0 0 !important;
- -webkit-transform-origin: right 50% !important;
- transform-origin: right 50% !important;
-}
-
-.ui.right.center.popup:before,
-.ui.left.center.popup:before {
- border: 0px solid #fafafaeb !important;
- background: #fafafaeb !important;
-}
-
-.ui.top.popup:before {
- border-color: #fafafaeb transparent transparent !important;
-}
-
-.ui.popup:before {
- border-color: #fafafaeb transparent transparent !important;
-}
-
-.ui.bottom.left.popup:before {
- border-radius: 0 !important;
- border: 1px solid transparent !important;
- border-color: #fafafaeb transparent transparent !important;
- background: #fafafaeb !important;
- -webkit-box-shadow: 0px 0px 0 0 #fafafaeb !important;
- box-shadow: 0px 0px 0 0 #fafafaeb !important;
- -webkit-tap-highlight-color: rgba(0,0,0,0) !important;
-}
-
-.ui.bottom.right.popup:before {
- border-radius: 0 !important;
- border: 1px solid transparent !important;
- border-color: #fafafaeb transparent transparent !重要;
- background: #fafafaeb !重要
- -webkit-box-shadow: 0px 0px 0 0 #fafafaeb !重要;
- box-shadow: 0px 0px 0 0 #fafafaeb !重要;
- -webkit-tap-highlight-color: rgba(0,0,0,0) !重要;
-}
-
-.ui.top.left.popup:before {
- border-radius: 0 !重要;
- border: 1px solid transparent !重要;
- border-color: #fafafaeb transparent transparent !重要;
- background: #fafafaeb !重要;
- -webkit-box-shadow: 0px 0px 0 0 #fafafaeb !重要;
- box-shadow: 0px 0px 0 0 #fafafaeb !重要;
- -webkit-tap-highlight-color: rgba(0,0,0,0) !重要;
-}
-
-.ui.top.right.popup:before {
- border-radius: 0 !重要;
- border: 1px solid transparent !重要;
- border-color: #fafafaeb transparent transparent !重要;
- background: #fafafaeb !重要;
- -webkit-box-shadow: 0px 0px 0 0 #fafafaeb !重要;
- box-shadow: 0px 0px 0 0 #fafafaeb !重要;
- -webkit-tap-highlight-color: rgba(0,0,0,0) !重要;
-}
-
-.ui.left.center.popup:before {
- border-radius: 0 !重要;
- border: 1px solid transparent !重要;
- border-color: #fafafaeb transparent transparent !重要;
- background: #fafafaeb !重要;
- -webkit-box-shadow: 0px 0px 0 0 #fafafaeb !重要;
- box-shadow: 0px 0px 0 0 #fafafaeb !重要;
- -webkit-tap-highlight-color: rgba(0,0,0,0) !重要;
-}
-
-/* Popup card */
-.status.cards .ui.content.popup {
- min-width: 20rem !重要;
- line-height: 2rem !重要;
- border-radius: 5px !重要;
- border: 1px solid transparent !重要;
- background-color: #fafafaeb !重要;
- font-family: Arial,Helvetica,sans-serif !重要;
-}
-
-.ui.content {
- margin: 0 !重要;
- padding: 1em !重要;
-}
-
-/* Service page */
-.ui
-
-.table {
- background: RGB(225,225,225,0.6) !重要;
-}
-
-.ui.table thead th {
- background: transparent !重要;
-}
-
-/* Service page progress bar */
-.service-status .good {
- background-color: #2175ba !重要;
-}
-
-.service-status .danger {
- background-color: red !重要;
-}
-
-.service-status .warning {
- background-color: orange !重要;
-}
-
-/* Copyright */
-.ui.inverted.segment, .ui.primary.inverted.segment {
- color: #000 !重要;
- font-weight: bold !重要;
- background-color: #fafafaa3 !重要;
-}
-</style>
-
-<!--Logo and Copyright-->
-<script>
-window.onload = function(){
-var avatar=document.querySelector(".item img")
-var footer=document.querySelector("div.is-size-7")
-footer.innerHTML="Copyright info"
-footer.style.visibility="visible"
-avatar.src="https:/img.src"
-avatar.style.visibility="visible"
-}
-</script>
Example of Changing Progress Bar Color and Modifying Footer in DayNight Theme (by @hyt-allen-xu and @tech-fever)
Click to expand/collapse
<style>
-.ui.fine.progress> .progress-bar {
- background-color: #00a7d0 !important;
-}
-</style>
-
-<script>
-window.onload = function(){
- var avatar = document.querySelector("img");
- var footer = document.querySelector("div.footer-container");
- footer.innerHTML = "©2021 Your Name & Powered by Your Name";
- footer.style.visibility = "visible";
- avatar.src = "Your Image URL";
- avatar.style.visibility = "visible";
-}
-</script>
Example of Changing LOGO and Footer in Default Theme (by @iLay1678)
Click to expand/collapse
<style>
-.right.menu>a {
- visibility: hidden;
-}
-.footer .is-size-7 {
- visibility: hidden;
-}
-.item img {
- visibility: hidden;
-}
-</style>
-<script>
-window.onload = function() {
- var avatar = document.querySelector(".item img");
- var footer = document.querySelector("div.is-size-7");
- footer.innerHTML = "Powered by Your Name";
- footer.style.visibility = "visible";
- avatar.src = "Your Square Logo URL";
- avatar.style.visibility = "visible";
-}
-</script>
Example of Changing Background Image in Hotaru Theme
Click to expand/collapse
<style>
-.hotaru-cover {
- background: url(https://s3.ax1x.com/2020/12/08/DzHv6A.jpg) center;
-}
-</style>
View Password
Non-CDN Dashboard Server Domain/IP
IP Change Notifications
Coverage
Specific Servers
Send Notifications to a Specific Notification Group
Show Full IP Address in Notifications
Disable Homepage Theme Switching
Task Management
How to Use
* * * * * *
, corresponding to second minute hour day month weekday
. For more details, see Cron Expression Format.
For example: 0 0 3 * * *
means "3 AM every day."&&
or &
.
For example, to schedule a reboot, you can enter reboot
here.Managing Tasks
Frequently Asked Questions
',11),i=[s];function r(c,l,d,u,h,g){return o(),t("div",null,i)}const k=e(n,[["render",r]]);export{p as __pageData,k as default};
diff --git a/assets/en_US_guide_tasks.md.BBiXJhsc.lean.js b/assets/en_US_guide_tasks.md.BBiXJhsc.lean.js
deleted file mode 100644
index d13517bb..00000000
--- a/assets/en_US_guide_tasks.md.BBiXJhsc.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o,a4 as a}from"./chunks/framework.BmdFiWrL.js";const p=JSON.parse('{"title":"Task Management","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"en_US/guide/tasks.md","filePath":"en_US/guide/tasks.md","lastUpdated":1732521523000}'),n={name:"en_US/guide/tasks.md"},s=a("",11),i=[s];function r(c,l,d,u,h,g){return o(),t("div",null,i)}const k=e(n,[["render",r]]);export{p as __pageData,k as default};
diff --git a/assets/en_US_index.md.D1U7Cexx.js b/assets/en_US_index.md.D1U7Cexx.js
deleted file mode 100644
index a25de73b..00000000
--- a/assets/en_US_index.md.D1U7Cexx.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as t,c as e,o as a}from"./chunks/framework.BmdFiWrL.js";const p=JSON.parse('{"title":"Nezha Monitoring - Server monitoring tool","titleTemplate":"User Manual","description":"","frontmatter":{"layout":"home","title":"Nezha Monitoring - Server monitoring tool","titleTemplate":"User Manual","hero":{"name":"Nezha Monitoring","text":"Open-source, lightweight, and easy-to-use server monitoring and operation tool","image":"https://raw.githubusercontent.com/naiba/nezha/master/resource/static/brand.svg","actions":[{"theme":"brand","text":"Preview","link":"https://ops.naibahq.com"},{"theme":"alt","text":"Get Started →","link":"en_US/guide/dashboard"}]},"features":[{"title":"One-Click Installation","details":"Supports one-click installation of the Dashboard and Agent services, making it easy to operate. Compatible with mainstream systems, including Linux, Windows, macOS, OpenWRT, and Synology."},{"title":"Real-Time Monitoring","details":"Supports monitoring the status of multiple servers simultaneously, providing historical network status and latency charts, monitoring webpage and port availability, and SSL certificate status. Supports alerts for status such as faults and traffic, with notifications via Telegram, email, Slack, and more."},{"title":"Easy Operations","details":"Provides an API to get server status, supports WebSSH, DDNS, and traffic monitoring. Allows setting up scheduled and triggered tasks and executing server tasks in batches."}]},"headers":[],"relativePath":"en_US/index.md","filePath":"en_US/index.md","lastUpdated":1732521523000}'),i={name:"en_US/index.md"};function n(s,o,r,l,d,c){return a(),e("div")}const u=t(i,[["render",n]]);export{p as __pageData,u as default};
diff --git a/assets/en_US_index.md.D1U7Cexx.lean.js b/assets/en_US_index.md.D1U7Cexx.lean.js
deleted file mode 100644
index a25de73b..00000000
--- a/assets/en_US_index.md.D1U7Cexx.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as t,c as e,o as a}from"./chunks/framework.BmdFiWrL.js";const p=JSON.parse('{"title":"Nezha Monitoring - Server monitoring tool","titleTemplate":"User Manual","description":"","frontmatter":{"layout":"home","title":"Nezha Monitoring - Server monitoring tool","titleTemplate":"User Manual","hero":{"name":"Nezha Monitoring","text":"Open-source, lightweight, and easy-to-use server monitoring and operation tool","image":"https://raw.githubusercontent.com/naiba/nezha/master/resource/static/brand.svg","actions":[{"theme":"brand","text":"Preview","link":"https://ops.naibahq.com"},{"theme":"alt","text":"Get Started →","link":"en_US/guide/dashboard"}]},"features":[{"title":"One-Click Installation","details":"Supports one-click installation of the Dashboard and Agent services, making it easy to operate. Compatible with mainstream systems, including Linux, Windows, macOS, OpenWRT, and Synology."},{"title":"Real-Time Monitoring","details":"Supports monitoring the status of multiple servers simultaneously, providing historical network status and latency charts, monitoring webpage and port availability, and SSL certificate status. Supports alerts for status such as faults and traffic, with notifications via Telegram, email, Slack, and more."},{"title":"Easy Operations","details":"Provides an API to get server status, supports WebSSH, DDNS, and traffic monitoring. Allows setting up scheduled and triggered tasks and executing server tasks in batches."}]},"headers":[],"relativePath":"en_US/index.md","filePath":"en_US/index.md","lastUpdated":1732521523000}'),i={name:"en_US/index.md"};function n(s,o,r,l,d,c){return a(),e("div")}const u=t(i,[["render",n]]);export{p as __pageData,u as default};
diff --git a/assets/guide_agent.md.D5fn7vFK.js b/assets/guide_agent.md.D5fn7vFK.js
deleted file mode 100644
index bc6714f0..00000000
--- a/assets/guide_agent.md.D5fn7vFK.js
+++ /dev/null
@@ -1,67 +0,0 @@
-import{_ as s,c as i,o as a,a4 as n}from"./chunks/framework.BmdFiWrL.js";const F=JSON.parse('{"title":"安装 Agent","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/agent.md","filePath":"guide/agent.md","lastUpdated":1732521523000}'),e={name:"guide/agent.md"},t=n(`
If a command fails to run with a "command not found" error, it may be a PATH environment variable issue. On Linux servers, you can add source ~/.bashrc
at the beginning of the command or use the absolute path to execute the command.安装 Agent
一键安装 Agent
准备工作
进入后台管理面板,转到“设置”页,在“未接入 CDN 的面板服务器域名/IP”项中填入通信域名,然后点击"保存"。在 Linux 中一键安装
在 macOS 中一键安装
在 Windows 中一键安装
其他方式安装 Agent
在 Linux 中安装 Agent(支持大部分发行版)
点击展开/收起
curl -L https://raw.githubusercontent.com/nezhahq/scripts/main/install.sh -o nezha.sh && chmod +x nezha.sh && sudo ./nezha.sh
curl -L https://gitee.com/naibahq/scripts/raw/main/install.sh -o nezha.sh && chmod +x nezha.sh && sudo CN=true ./nezha.sh
使用 Agent 内置服务命令安装(支持主流系统)
点击展开/收起
./nezha-agent service install -s server_name:port -p password
./nezha-agent service uninstall
./nezha-agent service start
./nezha-agent service stop
./nezha-agent service restart
runit 安装 Agent
点击展开/收起
/etc/sv/nezha-agent
目录:mkdir /etc/sv/nezha-agent
/etc/sv/nezha-agent/run
服务文件,写入以下内容:#!/bin/sh
-exec 2>&1
-exec /opt/nezha/agent/nezha-agent -s server_name:port -p password 2>&1
/etc/sv/nezha-agent/log/run
:#!/bin/sh
-exec vlogger -t nezha-agent -p daemon
sudo ln -s /etc/sv/nezha-agent/ /var/service
sv
命令来操作服务。socklog
,并启用:sudo xbps-install -S socklog-void
-sudo ln -s /etc/sv/socklog-unix /var/service
svlogtail
查看日志:sudo svlogtail | grep nezha-agent
在 Windows 中手动安装 Agent
哪吒探针 - Windows 客户端安装在群晖 DSM 中安装 Agent
点击展开/收起
# Agent 路径
-EXEC="/PATH/TO/nezha-agent"
-# 日志路径地址
-LOG="\${EXEC}.log"
-# 额外执行参数, 可留空
-ARGS=""
-# 哪吒服务端 gRPC 地址
-SERVER="HOST_OR_IP:gRPC_PORT"
-# 上一步获取的主机密钥
-SECRET="APP_SECRET"
-# 运行服务的用户名, *强烈建议使用非root用户执行*
-RUN_USER="nezha"
-
-# 写入到 systemd 服务文件
-cat << EOF > /usr/lib/systemd/system/nezha.service
-[Unit]
-Description=Nezha Agent Service
-After=network.target
-
-[Service]
-Type=simple
-ExecStart=/bin/nohup \${EXEC} \${ARGS} -s \${SERVER} -p \${SECRET} &>> \${LOG} &
-ExecStop=ps -fe |grep nezha-agent|awk '{print \\$2}'|xargs kill
-User=\${RUN_USER
-
-}
-Restart=on-abort
-
-[Install]
-WantedBy=multi-user.target
-EOF
-
-# 重载服务
-systemctl daemon-reload
-# 启动服务
-systemctl start nezha
-# 服务自启动
-systemctl enable nezha
使用 root
账号执行上述命令即可安装完成。在 macOS 中使用 Homebrew 安装 Agent
点击展开/收起
echo 'export HOMEBREW_NEZHA_AGENT_PASSWORD="通信密钥,在服务页面获取"' >> ~/.zshrc
-echo 'export HOMEBREW_NEZHA_AGENT_SERVER="你的服务器和端口,格式 your.domain:5555 "' >> ~/.zshrc
-source ~/.zshrc
brew install brewforge/chinese/nezha-agent
brew services start nezha-agent
brew services info nezha-agent
brew services stop nezha-agent
brew rm nezha-agent
echo $HOMEBREW_NEZHA_AGENT_PASSWORD
-echo $HOMEBREW_NEZHA_AGENT_SERVER
brew services stop nezha-agent
-brew reinstall nezha-agent
-brew services start nezha-agent
在 OpenWRT 中安装 Agent
点击展开/收起
NZ-OpenWrt
哪吒监控 For OpenWRT/root
。chmod +x /root/nezha-agent
赋予执行权限,然后创建 /etc/init.d/nezha-service
:#!/bin/sh /etc/rc.common
-
-START=99
-USE_PROCD=1
-
-start_service() {
- procd_open_instance
- procd_set_param command /root/nezha-agent -s 面板通信地址:端口 -p 秘钥 -d
- procd_set_param respawn
- procd_close_instance
-}
-
-stop_service() {
- killall nezha-agent
-}
-
-restart() {
- stop
- sleep 2
- start
-}
chmod +x /etc/init.d/nezha-service
赋予执行权限。/etc/init.d/nezha-service enable && /etc/init.d/nezha-service start
Agent 有 Docker 镜像吗?
Agent 的设计思路和 Dashboard 相反,Dashboard 要尽可能不影响宿主机工作,但 Agent 则需要在宿主机中执行监控服务和运行命令。
将 Agent 放入容器中确实可以继续执行监控任务,但 WebShell 等功能无法正常运行,因此不提供 Docker 镜像。Agent 常见问题
后台显示的 IP 和 Agent 实际 IP 不一致?
一键脚本安装时出错
curl: Failed to connect to raw.githubusercontent.com......
sudo: command not found
apt install sudo
API 接口
创建 Token
认证方式
Authorization: Token
进行身份认证。Request Headers:
-Authorization: Token
使用说明
GET
,返回格式为 JSON
获取服务器列表
GET /api/v1/server/list?tag=
tag
(可选):ServerTag 是服务器的分组,提供此参数则仅查询该分组中的服务器。{
- "code": 0,
- "message": "success",
- "result": [
- {
- "id": 1,
- "name": "Server1",
- "tag": "Tag1",
- "last_active": 1653014667,
- "ipv4": "1.1.1.1",
- "ipv6": "",
- "valid_ip": "1.1.1.1"
- },
- {
- "id": 2,
- "name": "Server2",
- "tag": "Tag2",
- "last_active": -62135596800,
- "ipv4": "",
- "ipv6": "",
- "valid_ip": ""
- }
- ]
-}
获取服务器详情
GET /api/v1/server/details?id=&tag=
id
(可选):ServerID,多个 ID 以逗号分隔,提供此参数则查询该 ID 对应的服务器,同时无视 tag
参数。tag
(可选):ServerTag,提供此参数则仅查询该分组下的服务器。{
- "code": 0,
- "message": "success",
- "result": [
- {
- "id": 1,
- "name": "Server1",
- "tag": "Tag1",
- "last_active": 1653015042,
- "ipv4": "1.1.1.1",
- "ipv6": "",
- "valid_ip": "1.1.1.1",
- "host": {
- "Platform": "darwin",
- "PlatformVersion": "12.3.1",
- "CPU": [
- "Apple M1 Pro 1 Physical Core"
- ],
- "MemTotal": 17179869184,
- "DiskTotal": 2473496842240,
- "SwapTotal": 0,
- "Arch": "arm64",
- "Virtualization": "",
- "BootTime": 1652683962,
- "CountryCode": "hk",
- "Version": ""
- },
- "status": {
- "CPU": 17.33,
- "MemUsed": 14013841408,
- "SwapUsed": 0,
- "DiskUsed": 2335048912896,
- "NetInTransfer": 2710273234,
- "NetOutTransfer": 695454765,
- "NetInSpeed": 10806,
- "NetOutSpeed": 5303,
- "Uptime": 331080,
- "Load1": 5.23,
- "Load5": 4.87,
- "Load15": 3.99,
- "TcpConnCount": 195,
- "UdpConnCount": 70,
- "ProcessCount": 437
- }
- },
- {
- "id": 2,
- "name": "Server2",
- "tag": "Tag2",
- "last_active": -62135596800,
- "ipv4": "",
- "ipv6": "",
- "valid_ip": "",
- "host": {
- "Platform": "",
- "PlatformVersion": "",
- "CPU": null,
- "MemTotal": 0,
- "DiskTotal": 0,
- "SwapTotal": 0,
- "Arch": "",
- "Virtualization": "",
- "BootTime": 0,
- "CountryCode": "",
- "Version": ""
- },
- "status": {
- "CPU": 0,
- "MemUsed": 0,
- "SwapUsed": 0,
- "DiskUsed": 0,
- "NetInTransfer": 0,
- "NetOutTransfer": 0,
- "NetInSpeed": 0,
- "NetOutSpeed": 0,
- "Uptime": 0,
- "Load1": 0,
- "Load5": 0,
- "Load15": 0,
- "TcpConnCount": 0,
- "UdpConnCount": 0,
- "ProcessCount": 0
- }
- }
- ]
-}
获取 ICMP Ping / TCPing 数据
GET /api/v1/monitor/{id}
id
(必填):ServerID,只能是一个正整数。{
- "code": 0,
- "message": "success",
- "result": [
- {
- "monitor_id": 1,
- "server_id": 1,
- "monitor_name": "Monitor1",
- "server_name": "Server1",
- "created_at": [
- 1722142860000,
- 1722142920000
- ],
- "avg_delay": [
- 68.2275,
- 70.1129
- ]
- },
- {
- "monitor_id": 2,
- "server_id": 1,
- "monitor_name": "Monitor2",
- "server_name": "Server1",
- "created_at": [
- 1722142860000,
- 1722142920000
- ],
- "avg_delay": [
- 66.656,
- 68.2153
- ]
- },
- {
- "monitor_id": 3,
- "server_id": 1,
- "monitor_name": "Monitor3",
- "server_name": "Server1",
- "created_at": [
- 1722142860000,
- 1722142920000
- ],
- "avg_delay": [
- 61.4525,
- 62.342
- ]
- }
- ]
-}
created_at
和 avg_delay
为对应关系。自动注册节点
POST /api/v1/server/register?simple=1
simple
:(可选) 指定返回数据的格式。 simple=1
或者simple=true
时,返回的数据仅包含 Token 字符串(如 8GYwaxYuLfU7zl7ndC
)。{"code": 200, "message": "Server created successfully","secret": "8GYwaxYuLfU7zl7ndC"}
{}
), 然后nezha面板会使用默认值进行填充Name
: 默认是节点的IPTag
: 默认是 "AutoRegister".Note
:默认是 "".HideForGuest
: 默认是 "on".{
- "Name": "abcd", // 节点名称
- "Tag": "", // 标签,可选
- "Note": "", // 备注信息,可选
- "HideForGuest": "on" // 是否对访客隐藏
-}
使用案例
获取所有服务器信息
import requests
-
-url = "http://your-dashboard/api/v1/server/list"
-headers = {
- "Authorization": "your_token"
-}
-
-response = requests.get(url, headers=headers)
-data = response.json()
-
-for server in data['result']:
- print(f"Server Name: {server['name']}, Last Active: {server['last_active']}, IP: {server['valid_ip']}")
获取特定服务器详情
import requests
-
-server_id = 1 # 替换为你的服务器ID
-url = f"http://your-dashboard/api/v1/server/details?id={server_id}"
-headers = {
- "Authorization": "your_token"
-}
-
-response = requests.get(url, headers=headers)
-data = response.json()
-
-server = data['result'][0]
-print(f"Server Name: {server['name']}")
-print(f"CPU Usage: {server['status']['CPU']}%")
-print(f"Memory Used: {server['status']['MemUsed']} bytes")
-print(f"Disk Used: {server['status']['DiskUsed']} bytes")
-print(f"Network In Speed: {server['status']['NetInSpeed']} bytes/s")
-print(f"Network Out Speed: {server['status']['NetOutSpeed']} bytes/s")
自动注册节点
nezha_register.sh
,将以下内容复制到文件中:#!/bin/bash
-
-# Exit if NEZHA_TOKEN is not set
-if [ -z "\${NEZHA_TOKEN}" ]; then
- echo "NEZHA_TOKEN is not set. Exiting."
- exit 0
-fi
-
-# Set default values if variables are not set
-NEZHA_PROBE_ADDRESS="\${NEZHA_PROBE_ADDRESS:-probe.example.com}"
-NEZHA_PROBE_PORT="\${NEZHA_PROBE_PORT:-5555}"
-NEZHA_DASHBOARD_URL="\${NEZHA_DASHBOARD_URL:-https://nezha.example.com}"
-
-NODE_NAME=\${NODE_NAME:-$(hostname)}
-
-# Send POST request and capture response and HTTP status code
-response=$(curl -s -o response_body.json -w "%{http_code}" -X POST "\${NEZHA_DASHBOARD_URL}/api/v1/server/register?simple=true" \\
- -H "Content-Type: application/json" \\
- -H "Authorization: \${NEZHA_TOKEN}" \\
- -d "{\\"Name\\": \\"\${NODE_NAME}\\"}")
-
-# Extract HTTP status code and Nezha secret
-HTTP_CODE="$response"
-NEZHA_SECRET=$(cat response_body.json)
-rm -f response_body.json
-
-if [ "$HTTP_CODE" != "200" ]; then
- echo "Failed to get Nezha Secret. HTTP status code: $HTTP_CODE"
- exit 1
-fi
-
-# Additional check for NEZHA_SECRET to ensure it's not JSON-formatted (indicating failure)
-if [[ "\${NEZHA_SECRET:0:1}" == "{" ]]; then
- echo "Failed to get Nezha Secret. Received response: \${NEZHA_SECRET}"
- exit 1
-fi
-
-# Download and execute the install script with cleanup
-curl -fsSL https://raw.githubusercontent.com/nezhahq/scripts/main/install.sh -o nezha.sh && \\
-chmod +x nezha.sh || { echo "Failed to download or make the script executable"; exit 1; }
-
-# Clean up nezha.sh on exit
-trap 'rm -f nezha.sh' EXIT
-
-# Run the Nezha agent installation script
-CMD="./nezha.sh install_agent "\${NEZHA_PROBE_ADDRESS}" "\${NEZHA_PROBE_PORT}" "\${NEZHA_SECRET}" --tls"
-
-echo "Run commnad : \${CMD}"
-
-eval $CMD
chmod +x nezha_register.sh
POXbxorKJBM8wPMKX8r2PdMblyXvpggB
export NEZHA_TOKEN="POXbxorKJBM8wPMKX8r2PdMblyXvpggB" # 从面板获取
-export NEZHA_PROBE_ADDRESS="your_probe_address" # 填写探针地址
-export NEZHA_DASHBOARD_URL="https://nezha.example.com" # 修改为你的面板地址
-export NEZHA_PROBE_PORT="5555" # 修改为你的探针端口(如有不同)
./nezha_register.sh
Run command: ./nezha.sh install_agent probe.example.com 5555 YOUR_SECRET --tls
安装 Dashboard
准备工作
如果你认为没有必要使用 80、443 端口来访问 Dashboard,你甚至不需要安装 Nginx 就可以直接使用安装脚本安装并运行哪吒监控。获取 Github 的 Client ID 和密钥
Application name
- 随意填写。Homepage URL
- 填写面板的访问域名,如:"http://dashboard.example.com"(你的域名)。Authorization callback URL
- 填写回调地址,如:"http://dashboard.example.com/oauth2/callback"(不要忘记/oauth2/callback
)。使用 Cloudflare Access 作为 OAuth2 提供方
新建 SaaS-OIDC 应用流程
My Team
-> Users
-> <具体用户>
-> 获取 User ID
并保存。Access
-> Application
-> Add an Application
。SaaS
,在 Application
中输入自定义的应用名称(例如 nezha),选择 OIDC
后点击 Add application
。Scopes
选择 openid
, email
, profile
, groups
。Redirect URLs
填写你的 callback 地址,例如 https://dashboard.example.com/oauth2/callback
。Client ID
、Client Secret
、Issuer
地址中协议与域名的部分,例如 https://xxxxx.cloudflareaccess.com
。/opt/nezha/dashboard/data/config.yaml
,将 Endpoint
配置修改为之前保存的 Issuer
地址,例如 https://xxxxx.cloudflareaccess.com
,保存后需重启 Dashboard。OIDC 验证配置(可选)
在服务器中安装 Dashboard
curl -L https://raw.githubusercontent.com/nezhahq/scripts/main/install.sh -o nezha.sh && chmod +x nezha.sh && sudo ./nezha.sh
curl -L https://gitee.com/naibahq/scripts/raw/main/install.sh -o nezha.sh && chmod +x nezha.sh && sudo CN=true ./nezha.sh
OAuth提供商
- github、cloudflare、gitlab、gitee 中选择一个。Client ID
- 之前保存的 Client ID。Client Secret
- 之前保存的 Client Secret。用户名
- OAuth 提供商中的用户名/User ID。站点标题
- 自定义站点标题。访问端口
- 公开访问端口,可自定义,默认 8008。Agent的通信端口
- Agent 与 Dashboard 的通信端口,默认 5555。
安装结束后,如果一切正常,此时你可以访问域名+端口号,如 “http://dashboard.example.com:8008” 来查看面板。./nezha.sh
配置反向代理
http://127.0.0.1
然后点击“保存”。#PROXY-START/
-location / {
- proxy_pass http://127.0.0.1:8008;
- proxy_set_header Host $http_host;
- proxy_set_header Upgrade $http_upgrade;
-}
-location ~ ^/(ws|terminal/.+|file/.+)$ {
- proxy_pass http://127.0.0.1:8008;
- proxy_http_version 1.1;
- proxy_set_header Upgrade $http_upgrade;
- proxy_set_header Connection "Upgrade";
- proxy_set_header Host $http_host;
-}
-#PROXY-END/
现在,你应该可以直接使用域名,如:“http://dashboard.example.com“ 来访问面板了。扩展内容:
proxy /ws http://ip:8008 {
- websocket
- header_upstream -Origin
-}
-proxy /terminal/* http://ip:8008 {
- websocket
- header_upstream -Origin
-}
-proxy /file/* http://ip:8008 {
- websocket
- header_upstream -Origin
-}
在宝塔面板中配置 SSL
正如在其他网站中配置 SSL 证书一样,进入站点设置中的 “SSL”,你可以选择自动申请 Let´s Encrypt 证书或手动配置已有的证书。
完成 SSL 的设置后,你需要回到 https://github.com/settings/developers ,编辑之前创建的验证应用程序,将之前我们填入的 "Homepage URL" 和 "Authorization callback URL" 中的域名全部从 http
改为 https
,如:"https://dashboard.example.com" 和 "https://dashboard.example.com/oauth2/callback" ,不更改此项可能会导致你无法登录面板后台。更新 Dashboard
./nezha.sh
,选择重启面板并更新。Dashboard 常见问题
为什么管理面板中显示的 IP 和 Agent 实际 IP 不一致?
如您发现管理面板中显示的 IP 和服务商提供给您的 IP 不一致,最大的可能是服务商给您的是入口 IP,但 Agent 测试的是您的出口 IP。这个问题也可能会出现在多线服务器和 IPLC 专线中。curl https://ipapi.co/ip/
-curl ip.sb
-curl ip-api.com
忘记查看密码\\删除查看密码
/opt/nezha/dashboard/data/config.yaml
文件。
密码位于 site-viewpassword
项中。面板安装/重启/更新失败: iptables ......
systemctl status docker
-systemctl restart docker
-systemctl status docker
若依然出现 iptables... 等错误,则考虑直接关闭 iptables 甚至移除 iptables。
这个问题也可能与内核有关,也可以尝试更换官方内核。面板重启失败:Invalid hostPort: nz_site_port 等
/opt/nezha/dashboard/docker-compose.yaml
文件。面板布局错误、CSS 资源无法被加载
出现此类错误,可以先尝试 重启并更新面板
。
如果更新面板后问题没有得到解决,那么可能是你的 vhost 配置文件内有不适用的配置,你可以编辑 Nginx 的 vhost 文件或在宝塔面板内:网站
中找到安装 Dashboard 时配置的站点,点击右侧 设置
。配置文件
,删除配置文件中的:location ~ .*\\.(js|css)?$
- {
- expires 12h;
- error_log /dev/null;
- access_log /dev/null;
- }
面板无法启动:panic:无法找到配置的 DDNS 提供者...
webhook
、cloudflare
、tencentcloud
和 dummy
。面板更新 DDNS 崩溃:panic:interface conversion: interface {} is nil, not []interface {}
AccessID
或 AccessSecret
有误。面板警告:NEZHA>> 错误的服务监控上报...
TaskType
导致,全部更新至最新版本即可解决。无法启动 Agent 服务:Unix syslog delivery error
/dev/log
套接字工作不正常或不存在。可参考 https://unix.stackexchange.com/questions/317064/how-do-i-restore-dev-log-in-systemdrsyslog-host 解决。如果使用的是 Docker,请避免使用 systemd
等 init 系统。打开网络监控页显示:server monitor history not found
如已经设置完毕,可以等待一段时间后再查看。启用 HTTPS 后 /terminal 或 /ws 不能正常连接
x509:certificate signed by unknown authority
,更换完整证书则可解决该问题。对面板提供的数据修改/增加功能不满意,想要修改/增加数据
请注意,数据库中并非什么都可以修改,错误的修改会导致数据混乱无法启动 Dashboard,请勿随意修改数据库!
数据库类型是 sqlite3,位于 /opt/nezha/dashboard/data/sqlite.db
,修改前请备份。Dashboard 会自动更新吗?
连接在线终端时提示:
Agent 信令下发失败
DDNS
为什么我要使用哪吒监控的 DDNS 功能?
配置说明
,
分隔。支持 IDN(国际化域名)。webhook
供应商,那么需要按需填写 Webhook 的对应选项。具体的说明请见 Webhook 配置说明。Webhook 配置说明
GET
、POST
、PATCH
、DELETE
以及 PUT
。JSON
或者 Form
。GET
和 DELETE
不会使用。如需使用嵌套格式,必须选择 JSON
作为请求类型。#ip#
:主机 IP。#domain#
:DDNS 域名。每个域名的请求是分别进行的,所以此项的值只会是单个域名字符串。#type#
:IP 类型,值为 "ipv4"
或 "ipv6"
。#record#
:记录类型,值为 "A"
或 "AAAA"
。#access_id#
:DDNS 凭据 1。#access_secret#
:DDNS 凭据 2。花生壳 Webhook 示例
点击展开/收起
http://ddns.oray.com/ph/update?hostname=#domain#&myip=#ip#
GET
{"Authorization": "Basic pass"}
,把 pass
替换成你的用户名:密码的 Base64 编码(例如,user:pass
为 dXNlcjpwYXNzCg==
)。供应商列表
供应商 凭据1(ID) 凭据2(Secret) dummy
❌️ ❌️ webhook
可选 可选 cloudflare
❌️ ✅ tencentcloud
✅ ✅ 查看日志
dashboard_1 | 2024/03/16 23:16:25 NEZHA>> 正在尝试更新域名(ddns.example.com)DDNS(1/3)
-dashboard_1 | 2024/03/16 23:16:28 NEZHA>> 尝试更新域名(ddns.example.com)DDNS成功
登录常见问题
登录回调后页面卡住\\拒绝连接\\响应时间过长
://
结尾的字符串即为协议,通常为 http
和 https
两种。由于正常部署情况下面板可能有多种协议+域名+端口组合均可访问,请务必选一个最合适的作为回调。如何检查我的回调地址是否错误?
请确保您的路径为 /oauth2/callback
,全部小写。登录后面板报错
http: named cookie not present
lookup xxx
建议先重启 docker,sudo systemctl restart docker
,再使用脚本重启面板。
仍然出现 lookup 错误建议查看是否有其他控制 iptables 的工具,如宝塔防火墙等。
这个问题也可能与内核有关系,请尝试更换官方内核。授权方式无效,或者登录回调地址无效、过期或已被撤销
oauth2: server response missing access_token
无法解决的话建议更换 Github 等。该用户不是本站点管理员,无法登录
Cloudflare Access 用户请注意,您的用户名不是邮箱,而是 User ID。dial tcp xxx:443 i/o timeout
sudo systemctl restart docker
,再使用脚本重启面板。
如为国内服务器配置 Github 登陆方式,则建议切换到 Cloudflare Access 以避免网络干扰。net/http: TLS handshake timeout
使用 Cloudflare Access 作为 OAuth2 提供方时无法收到邮件验证码
',24),s=[r];function l(c,n,h,d,u,p){return t(),a("div",null,s)}const k=e(i,[["render",l]]);export{m as __pageData,k as default};
diff --git a/assets/guide_loginq.md.CnJ6gnLD.lean.js b/assets/guide_loginq.md.CnJ6gnLD.lean.js
deleted file mode 100644
index 3375a51f..00000000
--- a/assets/guide_loginq.md.CnJ6gnLD.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as a,o as t,a4 as o}from"./chunks/framework.BmdFiWrL.js";const m=JSON.parse('{"title":"登录常见问题","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/loginq.md","filePath":"guide/loginq.md","lastUpdated":1732521523000}'),i={name:"guide/loginq.md"},r=o("",24),s=[r];function l(c,n,h,d,u,p){return t(),a("div",null,s)}const k=e(i,[["render",l]]);export{m as __pageData,k as default};
diff --git a/assets/guide_nat.md.DpUGasQ9.js b/assets/guide_nat.md.DpUGasQ9.js
deleted file mode 100644
index 05fc95a9..00000000
--- a/assets/guide_nat.md.DpUGasQ9.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as a,o,a4 as t}from"./chunks/framework.BmdFiWrL.js";const u=JSON.parse('{"title":"内网穿透功能配置","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/nat.md","filePath":"guide/nat.md","lastUpdated":1732521523000}'),l={name:"guide/nat.md"},i=t('Policies
中已正确配置了邮件验证策略。内网穿透功能配置
准备工作
service-1.example.com
。将此域名解析指向 Dashboard 服务器的公网 IP 地址。service-2.example.com
。service-1.example.com
已在该面板中正确绑定至 Dashboard 站点,否则可能导致无法访问。内网穿透配置步骤
OpenWrt 登录页
。IP:端口
,例如 127.0.0.1:80
。service-1.example.com
。如果面板使用了非标准端口访问(比如默认的 8008
),需要手动添加才可以生效。http://service-1.example.com
,以验证是否成功映射至 Agent 上的内网服务。使用须知
',10),r=[i];function c(d,s,n,h,p,_){return o(),a("div",null,r)}const g=e(l,[["render",c]]);export{u as __pageData,g as default};
diff --git a/assets/guide_nat.md.DpUGasQ9.lean.js b/assets/guide_nat.md.DpUGasQ9.lean.js
deleted file mode 100644
index a150ee25..00000000
--- a/assets/guide_nat.md.DpUGasQ9.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as a,o,a4 as t}from"./chunks/framework.BmdFiWrL.js";const u=JSON.parse('{"title":"内网穿透功能配置","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/nat.md","filePath":"guide/nat.md","lastUpdated":1732521523000}'),l={name:"guide/nat.md"},i=t("",10),r=[i];function c(d,s,n,h,p,_){return o(),a("div",null,r)}const g=e(l,[["render",c]]);export{u as __pageData,g as default};
diff --git a/assets/guide_notifications.md.Z1WGTu70.js b/assets/guide_notifications.md.Z1WGTu70.js
deleted file mode 100644
index 5bfe739d..00000000
--- a/assets/guide_notifications.md.Z1WGTu70.js
+++ /dev/null
@@ -1,35 +0,0 @@
-import{_ as i,c as s,o as a,a4 as t}from"./chunks/framework.BmdFiWrL.js";const c=JSON.parse('{"title":"通知设置","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/notifications.md","filePath":"guide/notifications.md","lastUpdated":1732521523000}'),l={name:"guide/notifications.md"},e=t(`通知设置
灵活的通知方式
#DATETIME#
代表事件发生的时间戳。当通知被触发时,面板会自动将 #DATETIME#
替换为事件的实际时间。#NEZHA#
是面板消息占位符,面板触发通知时会自动用实际消息替换占位符。JSON
格式的:当请求类型为 FORM 时,值为 key:value
的形式,value
里面可放置占位符,通知时会自动替换。当请求类型为 JSON 时 只会简单进行字符串替换后直接提交到 URL
。Bark 示例
点击展开/收起
{"title": "#SERVER.NAME#","device_key":"xxxxxxxxx","body":"#NEZHA#","icon":"https://xxxxxxxx/nz.png"}
Slack 示例 贡献者:@白歌
点击展开/收起
URL 参数获取说明
{"text":"#NEZHA#"}
Server 酱示例
点击展开/收起
{
- "title": "#SERVER.NAME#",
- "desp": "**#NEZHA#\\n\\n平均负载: \\"#SERVER.LOAD1#\\",\\"#SERVER.LOAD5#\\",\\"#SERVER.LOAD15#\\"\\n\\n## [点击访问面板](https://你的面板域名)\\n\\n"
-}
Telegram 示例 贡献者:@白歌
点击展开/收起
URL 参数获取说明
wxpusher 示例
点击展开/收起
{"appToken":"你的appToken","topicIds":[],"content":"#NEZHA#","contentType":"1","uids":["你的uid"]}
邮件通知示例 - SendCloud 贡献者:@白歌
点击展开/收起
URL 参数获取说明
<替换APIUSER>
和 <替换APIKEY>
为自己的 APIUSER 和 APIKEY,替换 URL 中的 <自定义发件邮箱>
和 <自定义收件邮箱>
为任意的的发件邮箱和收件邮箱。钉钉群机器人配置 示例
点击展开/收起
URL 参数获取说明
{"Content-Type": "application/json"}
{"msgtype": "text","text": {"content":"哪吒探针:\\n#NEZHA#"}}
企业微信群机器人 示例 贡献者:@ChowRex
点击展开/收起
{
- "content": "#NEZHA#",
- "ServerName": "#SERVER.NAME#",
- "ServerIP": "#SERVER.IP#",
- "ServerIPV4": "#SERVER.IPV4#",
- "ServerIPV6": "#SERVER.IPV6#",
- "CPU": "#SERVER.CPU#",
- "MEM": "#SERVER.MEM#",
- "SWAP": "#SERVER.SWAP#",
- "DISK": "#SERVER.DISK#",
- "NetInSpeed": "#SERVER.NETINSPEED#",
- "NetOutSpeed": "#SERVER.NETOUTSPEED#",
- "TransferIn": "#SERVER.TRANSFERIN#",
- "TranferOut": "#SERVER.TRANSFEROUT#",
- "Load1": "#SERVER.LOAD1#",
- "Load5": "#SERVER.LOAD5#",
- "Load15": "#SERVER.LOAD15#",
- "TCP_CONN_COUNT": "#SERVER.TCPCONNCOUNT", # 无效
- "UDP_CONN_COUNT": "#SERVER.UDPCONNCOUNT", # 无效
-}
{
- "msgtype": "markdown",
- "markdown": {
- "content": "# 哪吒通知消息\\n\\n\\"#NEZHA#\\"\\n\\n> 名称: \\"#SERVER.NAME#\\"\\n> IP: \\"#SERVER.IP#\\"\\n> IPv4: \\"#SERVER.IPV4#\\"\\n> IPv6: \\"#SERVER.IPV6#\\"\\n> CPU: \\"#SERVER.CPU#\\"\\n> 内存: \\"#SERVER.MEM#\\"\\n> 交换分区: \\"#SERVER.SWAP#\\"\\n> 存储: \\"#SERVER.DISK#\\"\\n> 实时上传速度: \\"#SERVER.NETINSPEED#\\"\\n> 实时下载速度: \\"#SERVER.NETOUTSPEED#\\"\\n> 总上传: \\"#SERVER.TRANSFERIN#\\"\\n> 总下载: \\"#SERVER.TRANSFEROUT#\\"\\n> 1分钟内负载: \\"#SERVER.LOAD1#\\"\\n> 5分钟内负载: \\"#SERVER.LOAD5#\\"\\n> 15分钟内负载: \\"#SERVER.LOAD15#\\"\\n> TCP连接数: \\"#SERVER.TCPCONNCOUNT\\"\\n> UDP连接数: \\"#SERVER.UDPCONNCOUNT\\"\\n\\n"
- }
-}
飞书群机器人配置 示例 贡献者:@eya46
点击展开/收起
URL 参数获取说明
{"content":{"text":"#NEZHA#\\n#DATETIME#"},"msg_type":"text"}
Matrix 通知示例
点击展开/收起
参数说明
$
开头的大写变量都需要替换为你的实际值。 YOUR_HOME_SERVER
: Matrix 服务器的地址。YOUR_NEZHA_URL
: 你的哪吒面板的 URL。YOUR_MATRIX_USERNAME
和 YOUR_MATRIX_PASSWD
: Matrix 用户名和密码。YOUR_MATRIX_TOKEN
获取方式:curl -XPOST -d '{"type": "m.login.password", "identifier": {"user": "$YOUR_MATRIX_USERNAME", "type": "m.id.user"}, "password": "$YOUR_MATRIX_PASSWD"}' "https://$YOUR_HOME_SERVER/_matrix/client/r0/login"
请求配置
https://$YOUR_HOME_SERVER/_matrix/client/r0/rooms/$ROOM_ID/send/m.room.message
POST
JSON
{
- "Authorization": "Bearer $YOUR_MATRIX_TOKEN"
-}
{
- "msgtype": "m.text",
- "format": "org.matrix.custom.html",
- "formatted_body": "<html><head><title>Nezha Dashboard</title></head><body><h1><a href=\\"$YOUR_NEZHA_URL\\" target=\\"_blank\\">Nezha Dashboard</a></h1><ul><li>datetime: #DATETIME#</li><li>Message: #NEZHA#</li></ul></body></html>",
- "body": "#NEZHA#"
-}
使用步骤
$YOUR_HOME_SERVER
, $YOUR_NEZHA_URL
, $YOUR_MATRIX_USERNAME
, $YOUR_MATRIX_PASSWD
, 以及 $YOUR_MATRIX_TOKEN
替换为你自己的值。curl
命令来获取 YOUR_MATRIX_TOKEN
,确保替换所有相关的变量。告警规则说明
基本规则
type
:可选取一个或多个类型,如在一个规则中选择了多个类型,需要同时满足所有选择的类型才会触发通知(可参考后面的示例) cpu
、gpu
、memory
、swap
、disk
net_in_speed
入站网速、net_out_speed
出站网速、net_all_speed
双向网速、transfer_in
入站流量、transfer_out
出站流量、transfer_all
双向流量offline
离线监控load1
、load5
、load15
负载process_count
进程数(目前取线程数占用资源太多,暂时不支持)tcp_conn_count
、udp_conn_count
连接数temperature_max
最高温度值duration
:持续数秒,数秒内采样记录 30% 以上触发阈值才会告警(防数据插针)min
或 max
: cover
: 0
监控所有服务器,通过 ignore
忽略特定服务器1
忽略所有服务器,通过 ignore
监控特定服务器
例如:[{"type":"offline","duration":10, "cover":0, "ignore":{"5": true}}]
ignore
:选择忽略特定服务器,搭配 cover
使用,内容为服务器 ID 和布尔值,例如:{"1": true, "2": false}
[{"Type":"offline","Duration":10}]
[{"Type":"cpu","Min":0,"Max":50,"Duration":10},{"Type":"memory","Min":20,"Max":0,"Duration":20}]
有 1、2、3、4 四台服务器和 A、B 两个不同的通知组。
1、2 这两台服务器掉线十分钟后给通知组 A 发送通知。
3、4 这两台服务器掉线十分钟后给通知组 B 发送通知。[{"type":"offline","duration":600,"cover":1,"ignore":{"1":true,"2":true}}]
[{"type":"offline","duration":600,"cover":1,"ignore":{"3":true,"4":true}}]
特殊:任意周期流量告警
type
: transfer_in_cycle
周期内的入站流量transfer_out_cycle
周期内的出站流量transfer_all_cycle
周期内双向流量的和cycle_start
:统计周期开始日期(可以是你机器计费周期的开始日期),时间格式为 RFC3339,例如北京时间为 2022-01-11T08:00:00.00+08:00
cycle_interval
:统计周期单位的数量(例如,周期单位为天,该值为 7,则代表每隔 7 天统计一次)cycle_unit
:统计周期单位,默认 hour
,可选(hour
, day
, week
, month
, year
)min/max
、cover
、ignore
参考基本规则配置[{"type":"transfer_out_cycle","max":1099511627776,"cycle_start":"2022-01-01T00:00:00+08:00","cycle_interval":1,"cycle_unit":"month","cover":1,"ignore":{"3":true,"4":true}}]
通知触发模式
设置告警时执行任务
`,54),n=[e];function h(o,k,p,r,d,u){return a(),s("div",null,n)}const g=i(l,[["render",h]]);export{c as __pageData,g as default};
diff --git a/assets/guide_notifications.md.Z1WGTu70.lean.js b/assets/guide_notifications.md.Z1WGTu70.lean.js
deleted file mode 100644
index 185d0630..00000000
--- a/assets/guide_notifications.md.Z1WGTu70.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as i,c as s,o as a,a4 as t}from"./chunks/framework.BmdFiWrL.js";const c=JSON.parse('{"title":"通知设置","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/notifications.md","filePath":"guide/notifications.md","lastUpdated":1732521523000}'),l={name:"guide/notifications.md"},e=t("",54),n=[e];function h(o,k,p,r,d,u){return a(),s("div",null,n)}const g=i(l,[["render",h]]);export{c as __pageData,g as default};
diff --git a/assets/guide_q1.md.Cwu7c2wW.js b/assets/guide_q1.md.Cwu7c2wW.js
deleted file mode 100644
index 49bae31f..00000000
--- a/assets/guide_q1.md.Cwu7c2wW.js
+++ /dev/null
@@ -1,46 +0,0 @@
-import{_ as s,c as i,o as a,a4 as n}from"./chunks/framework.BmdFiWrL.js";const c=JSON.parse('{"title":"反向代理 Telegram Bot API","description":"","frontmatter":{},"headers":[],"relativePath":"guide/q1.md","filePath":"guide/q1.md","lastUpdated":1732521523000}'),l={name:"guide/q1.md"},p=n(`反向代理 Telegram Bot API
准备工作
NGINX 配置
http{}
中添加如下配置:# HTTP 强制跳转到 HTTPS
-server {
- listen 80;
- listen [::]:80;
- server_name <yourDomainName>;
-
- # 强制 HTTPS
- return 301 https://$server_name$request_uri;
-}
-
-# HTTPS 配置
-server {
- listen 443 ssl;
- listen [::]:443 ssl;
- server_name <yourDomainName>;
-
- # SSL 证书路径
- ssl_certificate </path/to/your/server.pem>;
- ssl_certificate_key </path/to/your/server.key>;
-
- # Root 非必要
- root /var/www/tgbot/;
-
- # 必须配置 DNS,否则会报 502 错误
- resolver 8.8.8.8;
-
- # 以 /bot 开头的请求会被正则匹配
- location ~* ^/bot {
- proxy_buffering off;
- proxy_pass https://api.telegram.org$request_uri;
- proxy_http_version 1.1;
- }
-
- # Root 非必要,主要用于确认服务器状态。也可以改为 return 403
- location / {
- try_files $uri $uri /index.html;
- }
-
- # 错误日志
- error_log /var/log/tg.log error;
-}
yourDomainName
:你准备的域名ssl_certificate
:SSL 证书路径ssl_certificate_key
:SSL 证书路径使用方式
systemctl restart nginx
重启 Nginx。然后在 Nezha 中将原来的 https://api.telegram.org/
替换为 https://<yourDomainName>/
,即可正常推送消息。防止盗用
serverIp
:Agent 的 IP 地址。根据你的系统选择适用的命令,ufw
或 iptables
均可。# Ubuntu
-ufw allow proto tcp from <serverIp> to any port 443
-
-# CentOS
-iptables -I INPUT -p tcp --dport 443 -j DROP
-iptables -I INPUT -s <serverIp> -p tcp --dport 443 -j ACCEPT
配置OIDC认证
config.yaml
文件的内容来配置 OIDC(OpenID Connect)认证。OIDC 是建立在 OAuth 2.0 协议之上的认证层,允许应用程序通过授权服务器执行的认证来验证用户的身份。这种方法广泛用于实现现代应用程序的单点登录。配置详细信息
config.yaml
中配置 OIDC 设置的逐步指南:oauth2:
- type: oidc # (必填)指定认证类型为 OIDC
- oidcDisplayName: OIDC # (可选,默认值为 OIDC)登录页面显示的名称
- admin: "" # (至少填写 admin 或 adminGroups 中的一个;默认为空)管理员用户名,用逗号分隔。列在此处的用户被视为管理员
- adminGroups: "" # (至少填写 admin 或 adminGroups 中的一个;默认为空)管理员组,用逗号分隔。属于这些组的用户被视为管理员。如果不使用组管理,可以省略此项
- clientid: # (必填)OIDC 客户端 ID
- clientsecret: # (必填)OIDC 客户端密钥
- oidcIssuer: https://auth.example.com/realms/master # (必填)OIDC 提供商的 issuer URL,可以从您的OIDC 提供商处获取
- # oidcLogoutUrl: https://auth.example.com/realms/master/protocol/openid-connect/logout # (因 bug 目前不能使用)
- # oidcRegisterUrl: # (可选)OIDC 提供商的注册链接
- oidcScopes: openid,profile,email # (可选,默认值为 openid,profile,email )OIDC 请求的 scope,用逗号分隔
- oidcLoginClaim: sub # (可选,默认值为 sub)OIDC 返回的用户名字段,可以是 preferred_username、sub 或email
- oidcGroupsClaim: groups # (如果使用 adminGroups,则必填,默认值为 groups )OIDC 返回的用户组信息字段,可以是 groups 或 roles
- oidcAutoCreate: false # (可选,默认值为 false)如果用户不存在,是否自动创建用户
- oidcAutoLogin: false # (可选,默认值为 false)当地址是 /login 时,是否自动跳转到OIDC登录页面
自定义 IP 地理位置数据库
pkg/geoip/geoip.db
路径下放置符合 IPinfo 格式的 mmdb 数据库。continent
:大洲代码continent_name
:大洲名称country
:国家/地区代码country_name
:国家/地区名称country
信息,其它字段可留空。编辑数据库
mmdbwriter
和 maxminddb-golang
这两个库进行 mmdb 的编辑操作。ReplaceWith
inserter(此为默认值)并写入 mmdbtype.Map
信息:subnet := &net.IPNet{
- IP: net.ParseIP("114.5.1.4")
- Mask: net.CIDRMask(32, 32)
-}
-countryMap := make(map[mmdbtype.String]mmdbtype.String)
-
-countryMap[mmdbtype.String("country")] = mmdbtype.String("JP")
-
-if err := writer.Insert(subnet, mmdbtype.Map(countryMap)); err != nil {
- return err
-}
writer
为一个 *mmdbwriter.Tree
的实例,需要通过 mmdbwriter.New
方法创建。具体操作和插入其它字段的过程不再赘述。mmdbinspect
命令行工具测试数据库信息。例如:$ mmdbinspect -db ./country.mmdb 114.5.1.4
-[
- {
- "Database": "./country.mmdb",
- "Records": [
- {
- "Network": "114.5.1.4/32",
- "Record": {
- "continent": "AS",
- "continent_name": "Asia",
- "country": "JP",
- "country_name": "Japan"
- }
- }
- ],
- "Lookup": "114.5.1.4"
- }
-]
grpcurl
或类似工具测试查询 API 是否工作正常:grpcurl -proto ./nezha/proto/nezha.proto -plaintext -H 'client_secret: test' -d '{"ip": "114.5.1.4"}' 127.0.0.1:5555 proto.NezhaService/LookupGeoIP
Agent 启动/上线 问题自检流程
',2),i=[r];function c(s,_,d,l,p,g){return a(),t("div",null,i)}const u=e(n,[["render",c]]);export{f as __pageData,u as default};
diff --git a/assets/guide_q2.md.DtMO0dtm.lean.js b/assets/guide_q2.md.DtMO0dtm.lean.js
deleted file mode 100644
index 73e1aff3..00000000
--- a/assets/guide_q2.md.DtMO0dtm.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o as a,a4 as o}from"./chunks/framework.BmdFiWrL.js";const f=JSON.parse('{"title":"Agent 启动/上线 问题自检流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/q2.md","filePath":"guide/q2.md","lastUpdated":1732521523000}'),n={name:"guide/q2.md"},r=o("",2),i=[r];function c(s,_,d,l,p,g){return a(),t("div",null,i)}const u=e(n,[["render",c]]);export{f as __pageData,u as default};
diff --git a/assets/guide_q3.md.0iIre3Mh.js b/assets/guide_q3.md.0iIre3Mh.js
deleted file mode 100644
index 530a0908..00000000
--- a/assets/guide_q3.md.0iIre3Mh.js
+++ /dev/null
@@ -1,74 +0,0 @@
-import{_ as s,c as i,o as a,a4 as n}from"./chunks/framework.BmdFiWrL.js";const o=JSON.parse('{"title":"反向代理 gRPC 端口(支持 Cloudflare CDN)","description":"","frontmatter":{},"headers":[],"relativePath":"guide/q3.md","filePath":"guide/q3.md","lastUpdated":1732521523000}'),l={name:"guide/q3.md"},p=n(`/opt/nezha/agent/nezha-agent -s 面板IP或非CDN域名:面板RPC端口 -p Agent密钥 -d
查看日志是否是因为 DNS、网络不佳导致超时(timeout)nc -v 域名/IP 面板RPC端口
或者 telnet 域名/IP 面板RPC端口
来检验是否是网络问题,检查本机与面板服务器的出入站防火墙,如果无法判断问题可借助 https://port.ping.pe/ 提供的端口检查工具进行检测。反向代理 gRPC 端口(支持 Cloudflare CDN)
server {
- listen 443 ssl http2;
- listen [::]:443 ssl http2;
- server_name data.example.com; # 你的 Agent 连接 Dashboard 的域名
-
- ssl_certificate /data/letsencrypt/fullchain.pem; # 你的域名证书路径
- ssl_certificate_key /data/letsencrypt/key.pem; # 你的域名私钥路径
- ssl_stapling on;
- ssl_session_timeout 1d;
- ssl_session_cache shared:SSL:10m; # 此项可能会和其他配置文件冲突,如冲突请注释此项
- ssl_protocols TLSv1.1 TLSv1.2 TLSv1.3;
-
- underscores_in_headers on;
-
- keepalive_time 24h;
- keepalive_requests 100000;
- keepalive_timeout 120s;
-
- location / {
- grpc_read_timeout 300s;
- grpc_send_timeout 300s;
- grpc_socket_keepalive on;
- grpc_pass grpc://grpcservers;
- }
-}
-
-upstream grpcservers {
- server localhost:5555;
- keepalive 512;
-}
data.example.com:443 { # 你的 Agent 连接 Dashboard 的域名
- reverse_proxy {
- to localhost:5555
- transport http {
- versions h2c 2
- }
- }
-}
未接入CDN的面板服务器域名/IP
中填入上一步在 Nginx 或 Caddy 中配置的域名 比如 data.example.com
,并保存。proxygrpcport
修改为 Nginx 或 Caddy 监听的端口,比如上一步设置的 443
;因为我们在 Nginx 或 Caddy 中开启了 SSL/TLS,所以需要将 tls
设置为 true
;修改完成后重启面板。网络
选项将 gRPC
开关打开,打开 DNS
选项,找到 Nginx 或 Caddy 反代 gRPC 配置的域名的解析记录,打开橙色云启用 CDN。gRPC
后,可能不能立即可用,需要等待一段时间。具体可通过curl
和使用nezha-agent -d
来验证:localhost:~/agent# curl -H "content-type: application/grpc+proto" -H "authorization: Bearer test" https://xxx.xxx.ovh -v
-* processing: https://xxx.xxx.ovh
-* Trying [2606:4700:3035::ac43:8bed]:443...
-* Connected to xxx.xxx.ovh (2606:4700:3035::ac43:8bed) port 443
-# ... SSL info
-* using HTTP/2
-* h2 [:method: GET]
-* h2 [:scheme: https]
-* h2 [:authority: xxx.xxx.ovh]
-* h2 [:path: /]
-* h2 [user-agent: curl/8.2.1]
-* h2 [accept: */*]
-* Using Stream ID: 1
-> GET / HTTP/2
-> Host: xxx.xxx.ovh
-> User-Agent: curl/8.4.0
-> Accept: */*
-> content-type: application/grpc+proto
-> authorization: Bearer test
->
-< HTTP/2 405
-< date: Wed, 20 Dec 2023 08:56:27 GMT
-< content-type: application/grpc+proto
-< cf-ray: 8386ac12dabd5ddc-HKG
-< cf-cache-status: DYNAMIC
-< grpc-message: Received a HEADERS frame with :method "GET" which should be POST
-< grpc-status: 13
-< report-to: {"endpoints":[{"url":"https:\\/\\/a.nel.cloudflare.com\\/report\\/v3?s=%2BTjgJvXWyRF11nUOYx9Lq7UDC1xOYBLtjvWrdjVJQIqu9YqnFJeZFran2KRs6zabQc%2BLV8AubNqYRYDb7hQAZe6bglmVz0wQjrb0tNovYf%2B59SAp%2BQfZnH%2BAFDydNT95ZCmTPnKgWetcwQiUfXU%3D"}],"group":"cf-nel","max_age":604800}
-< nel: {"success_fraction":0,"report_to":"cf-nel","max_age":604800}
-< vary: Accept-Encoding
-< server: cloudflare
-< alt-svc: h3=":443"; ma=86400
-<
-* Connection #0 to host xxx.xxx.ovh left intact
-localhost:~/agent# /opt/nezha/agent/nezha-agent -s nezha.xxx.xxx:443 -p YOUR_KEY --tls -d
-NEZHA@2023-12-20 05:14:00>> 检查更新: 0.15.14
-NEZHA@2023-12-20 05:14:01>> 上报系统信息失败: rpc error: code = Unknown desc = EOF # 需要修改主控端 /opt/nezha/dashboard/data/config.yaml的 GRPCHost 和 TLS 选项
-NEZHA@2023-12-20 05:14:01>> Error to close connection ...
如何每月重置流量统计?
告警
页服务
页中就可以看到月流量统计了,这里的统计数据不会因为服务器重启而重置自定义 Agent 监控项目
自定义监控的网卡和硬盘分区
/opt/nezha/agent/nezha-agent edit
其他运行参数
查看支持的参数
./nezha-agent --help
配置参数
/etc/systemd/system/nezha-agent.service
,并在 ExecStart=
行的末尾添加您需要的参数:sed -i '/^ExecStart=/ s/$/ --disable-auto-update/' /etc/systemd/system/nezha-agent.service && systemctl daemon-reload
',12),o=[d];function l(n,c,h,p,r,g){return s(),a("div",null,o)}const b=e(t,[["render",l]]);export{u as __pageData,b as default};
diff --git a/assets/guide_q7.md.BrDNRaw0.lean.js b/assets/guide_q7.md.BrDNRaw0.lean.js
deleted file mode 100644
index 5ea93f39..00000000
--- a/assets/guide_q7.md.BrDNRaw0.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as a,o as s,a4 as i}from"./chunks/framework.BmdFiWrL.js";const u=JSON.parse('{"title":"自定义 Agent 监控项目","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/q7.md","filePath":"guide/q7.md","lastUpdated":1732521523000}'),t={name:"guide/q7.md"},d=i("",12),o=[d];function l(n,c,h,p,r,g){return s(),a("div",null,o)}const b=e(t,[["render",l]]);export{u as __pageData,b as default};
diff --git a/assets/guide_q8.md.BsCGKX4S.js b/assets/guide_q8.md.BsCGKX4S.js
deleted file mode 100644
index 1a129491..00000000
--- a/assets/guide_q8.md.BsCGKX4S.js
+++ /dev/null
@@ -1,6 +0,0 @@
-import{_ as e,c as a,o as d,a4 as o}from"./chunks/framework.BmdFiWrL.js";const g=JSON.parse('{"title":"使用 Cloudflare Access 作为 OAuth2 提供方","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/q8.md","filePath":"guide/q8.md","lastUpdated":1732521523000}'),c={name:"guide/q8.md"},s=o(`--report-delay
:设置系统信息上报的间隔时间。默认为 1 秒。为了降低系统资源占用,可以设置为 3(有效范围:1-4 秒)。--skip-conn
:不监控网络连接数。建议在连接数较多或 CPU 资源占用较高的服务器上使用此参数。--skip-procs
:不监控进程数,有助于降低 Agent 的资源占用。--disable-auto-update
:禁用 Agent 的自动更新功能,增强安全性。--disable-force-update
:禁用 Agent 的强制更新功能,增强安全性。--disable-command-execute
:禁止在 Agent 上执行任何定时任务或使用在线终端,增强安全性。--tls
:启用 SSL/TLS 加密。当您使用 nginx 反向代理 Agent 的 gRPC 连接且 nginx 配置了 SSL/TLS 时,应启用此配置。--use-ipv6-countrycode
:强制使用 IPv6 地址查询国家代码。默认情况下,Agent 使用 IPv4 地址查询国家代码,如果服务器支持 IPv6 且与 IPv4 地址的国家代码不同,可以使用此参数。--gpu
:启用 GPU 监控。注意:GPU 使用率监控可能需要安装额外的依赖包,详细信息可以参考文档:启用 GPU 监控。--temperature
:启用硬件温度监控。仅支持的硬件有效,部分 VPS 可能无法获取温度信息。-d
--debug
:启用调试模式。-u
--ip-report-period
:本地IP更新间隔, 如果这个值小于 --report-delay
设置的值,那么以 --report-delay
的值为准。默认为1800秒(30分钟)。-k
--insecure
:禁用证书检查,适用于使用自签证书的场景。使用 Cloudflare Access 作为 OAuth2 提供方
示例配置:
Oauth2:
- Admin: 701b9ea6-9f56-48cd-af3e-cbb4bfc1475c
- ClientID: 3516291f53eca9b4901a01337e41be7dc52f565c8657d08a3fddb2178d13c5bf
- ClientSecret: 0568b67c7b6d0ed51c663e2fe935683007c28f947a27b7bd47a5ad3d8b56fb67
- Endpoint: "https://xxxxx.cloudflareaccess.com"
- Type: cloudflare
配置说明:
参数 获取方式 Admin My Team
-> Users
-> <具体用户>
-> User ID
ClientID/ClientSecret Access
-> Application
-> Add an Application
-> SaaS
-> OIDC
Endpoint Access
-> Application
-> Application URL
-> 只保留协议+域名的部分,路径不需要
新建 SaaS-OIDC 应用流程
My Team
-> Users
-> 点击<具体用户>
-> 获取 User ID
并保存 (如果是第一次使用 Zero Trust,Users 列表会为空,可暂时跳过这一步;你需要完成一次验证后,用户才会出现在 Users 列表中);Access
-> Applications
-> Add an Application
;SaaS
,在 Application
字段中输入自定义的应用名称(例如 nezha
),选择 OIDC
后点击 Add application
;Scopes
选择 openid
, email
, profile
, groups
;Redirect URLs
中填写你的 Dashboard Callback 地址,例如 https://dashboard.example.com/oauth2/callback
;Client ID
、Client Secret
和 Issuer
地址中的协议与域名部分,例如 https://xxxxx.cloudflareaccess.com
;/opt/nezha/dashboard/data/config.yaml
),参照示例配置修改 OAuth2
设置,并重启 Dashboard 服务。身份验证策略配置
Access
-> Applications
-> <应用名>
-> Policies
。您可以选择多种 SSO 验证方式,包括邮件 OTP 验证、硬件密钥验证等,详细配置请参考 Cloudflare Zero Trust 文档。策略配置示例(One-time PIN)
Access
-> Applications
-> <应用名>
-> Policies
-> Add a policy
;Policy Name
,例如 OTP
,Action
设置为 Allow
;Configure rules
下新增一条 Include
规则,Selector
选择 Emails
,在文本框中输入你的邮箱地址;Save policy
保存策略。测试策略
`,16),t=[s];function i(l,r,n,h,p,u){return d(),a("div",null,t)}const b=e(c,[["render",i]]);export{g as __pageData,b as default};
diff --git a/assets/guide_q8.md.BsCGKX4S.lean.js b/assets/guide_q8.md.BsCGKX4S.lean.js
deleted file mode 100644
index a0bfbdde..00000000
--- a/assets/guide_q8.md.BsCGKX4S.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as a,o as d,a4 as o}from"./chunks/framework.BmdFiWrL.js";const g=JSON.parse('{"title":"使用 Cloudflare Access 作为 OAuth2 提供方","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/q8.md","filePath":"guide/q8.md","lastUpdated":1732521523000}'),c={name:"guide/q8.md"},s=o("",16),t=[s];function i(l,r,n,h,p,u){return d(),a("div",null,t)}const b=e(c,[["render",i]]);export{g as __pageData,b as default};
diff --git a/assets/guide_q9.md.EqT3EdyS.js b/assets/guide_q9.md.EqT3EdyS.js
deleted file mode 100644
index a780e245..00000000
--- a/assets/guide_q9.md.EqT3EdyS.js
+++ /dev/null
@@ -1,14 +0,0 @@
-import{_ as s,c as a,o as i,a4 as e}from"./chunks/framework.BmdFiWrL.js";const F=JSON.parse('{"title":"启用 GPU 监控","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/q9.md","filePath":"guide/q9.md","lastUpdated":1732521523000}'),n={name:"guide/q9.md"},t=e(`使用 Cloudflare 账号登录
,点击登录会跳转到 Cloudflare Access 登录页面;Send me a code
,输入收到的验证码,即可登录 Dashboard;Admin
中未填写 User ID
,登录后会提示错误信息:“该用户不是本站点管理员,无法登录”。此时需要在 My Team
-> Users
中找到对应的用户,点击用户名获取 User ID
并填写到 Dashboard 配置文件里的 Admin
部分,重启 Dashboard 服务后再次尝试登录。启用 GPU 监控
启用
通过启动参数
--gpu
即可。例如:/opt/nezha/agent/nezha-agent -s example.com:5555 -p example --gpu
通过配置文件
/opt/nezha/agent/nezha-agent edit
打开 GPU 占用率监控支持
NVIDIA
nvidia-smi
工具,一般为官方驱动自带。nouveau
,那么将无法获取 GPU 使用率。AMD
amdgpu
开源驱动和 rocm-smi
工具。rocm-smi
,以下是部分系统的安装命令:# Arch Linux
-pacman -Sy rocm-smi-lib
-
-# Debian / Ubuntu
-apt install rocm-smi
-
-# Fedora / RHEL 8+
-dnf install rocm-smi
rocm_smi_lib
。git
cmake
gcc
rocm_smi_lib
的 git 仓库:git clone https://github.com/ROCm/rocm_smi_lib
cd rocm_smi_lib
-mkdir -p build
-cd build
-cmake ..
-make -j $(nproc)
-# Install library file and header; default location is /opt/rocm
-make install
服务器
介绍
新增服务器
拥有相同分组的服务器会在受支持的主题中划分到一起进行显示,备注仅会在后台显示,无需担心泄露信息。安装 Agent
推荐使用一键安装,即配置好参数后,点击服务器一键安装列上的对应的系统图标即可复制安装命令,在相应服务器上进行安装。强制更新
--disable-auto-update
和 --disable-force-update
。
默认情况下,Agent 会自动更新,无需干预。但当用户关闭自动更新后,也可以选中指定服务器进行强制更新。disable-force-update
开启时此功能不生效。数据列
在线终端
disable-command-execute
开启时此功能不生效。
Linux 和 Windows 均可用,可使用 Ctrl+Shift+V 粘贴。
连接失败请参考实时通道断开/在线终端连接失败。FM
公开备注示例
ServerStatus主题agent账单信息展示
点击展开/收起
{
- "billingDataMod": {
- "startDate": "2024-10-01T00:00:00+08:00",
- "endDate": "2024-11-01T00:00:00+08:00",
- "autoRenewal": "1",
- "cycle": "月",
- "amount": "$3.99"
- }
-}
{
- "billingDataMod": {
- "startDate": "2024-10-01T00:00:00+08:00",
- "endDate": "2024-11-01T00:00:00+08:00",
- "autoRenewal": "1",
- "cycle": "月"
- }
-}
格式如 2022-04-01T23:59:59+08:00
格式如 2022-05-01T23:59:59+08:00
-当以0000-00-00开头时,表示账单无期限,适用于永久免费的vps,如0000-00-00T23:59:59+08:00
分3部分
-1. 日期:2022-04-01
-2. 时间:T23:59:59
-3. 时区:+08:00
-例如:
-2024-10-01T23:59:59+05:00
-2024-09-21T12:00:00-05:00
-2024-03-15T23:59:59+00:00
支持值
-1:自动续期
-0:不自动续期
-当设置 "autoRenewal": 1 时,程序会根据当前时间自动判断,vps账单到期后无需手动更改startDate和endDate
格式如 $9.99 、€19.92 、¥199、19.99HKD、9.99USD、49.99CNY
-如果vps是免费的,请设置为"0" , 如 "amount": "0", 前台展示为 FREE
-如果vps是按量收费,请设置为"-1" , 如 "amount": "-1", 前台展示为PAYG
支持格式(英文大小写字母都支持,大小写字母混用也支持)
-1. 月、month、monthly、m、mo
-2. 季、quarterly、q
-3. 半年、半、half、semi-annually、h
-4. 年、year、annually、y、yr
服务监控
设置好的服务监控可以在主页中的“服务”页查看过去 30 天的可用性监控结果使用方法
HTTP-GET
:选择此类型时,你应输入一个 URL 作为目标,URL 需添加 http://
或 https://
。如果你的目标 URL 是 https://
,将会同时监控该 URL 的 SSL 证书,当 SSL 证书到期或发生变更,会触发提醒。例如:https://example.com。ICMP-Ping
:选择此类型时,你应输入一个域名或 IP,不含端口号。例如:1.1.1.1 或 example.com。TCP-Ping
:选择此类型时,你应输入一个域名或 IP 并包含端口号。例如:1.1.1.1:80 或 example.com:22。延迟变化告警
最高延迟
或小于最低延迟
时,将会发送告警通知。告警时触发任务
网络延迟图表
管理监控
设置
站点标题
管理员列表
user1,user2
。/opt/nezha/dashboard/data/config.yaml
,重新设置新的管理员 Client ID
和 Client Secret
。界面主题
Language
自定义代码 (style, script)
默认主题更改进度条颜色、背景图片、导航栏等效果示例
点击展开/收起
<style>
-/* 屏幕适配 */
-@media only screen and (min-width: 1200px) {
- .ui.container {
- width: 80% !important;
-}
-}
-
-@media only screen and (max-width: 767px) {
- .ui.card>.content>.header:not(.ui), .ui.cards>.card>.content>.header:not(.ui) {
- margin-top: 0.4em !important;
- }
-}
-
-/* 整体图标 */
-i.icon {
- color: #000;
- width: 1.2em !important;
-}
-
-/* 背景图片 */
-body {
- content: " " !important;
- background: fixed !important;
- z-index: -1 !important;
- top: 0 !important;
- right: 0 !important;
- bottom: 0 !important;
- left: 0 !important;
- background-position: top !important;
- background-repeat: no-repeat !important;
- background-size: cover !important;
- background-image: url(https://backgroud.img) !important;
- font-family: Arial,Helvetica,sans-serif !important;
-}
-
-/* 导航栏 */
-.ui.large.menu {
- border: 0 !important;
- border-radius: 0px !important;
- background-color: rgba(255, 255, 255, 55%) !important;
-}
-
-/* 首页按钮 */
-.ui.menu .active.item {
- background-color: transparent !important;
-}
-
-/* 导航栏下拉框 */
-.ui.dropdown .menu {
- border: 0 !important;
- border-radius: 0 !important;
- background-color: rgba(255, 255, 255, 80%) !important;
-}
-
-/* 登录按钮 */
-.nezha-primary-btn {
- background-color: transparent !important;
- color: #000 !important;
-}
-
-/* 大卡片 */
-#app .ui.fluid.accordion {
- background-color: #fbfbfb26 !important;
- border-radius: 0.4rem !important;
-}
-
-/* 小卡片 */
-.ui.four.cards>.card {
- border-radius: 0.6rem !important;
- background-color: #fafafaa3 !important;
-}
-
-.status.cards .wide.column {
- padding-top: 0 !important;
- padding-bottom: 0 !important;
- height: 3.3rem !important;
-}
-
-.status.cards .three.wide.column {
- padding-right: 0rem !important;
-}
-
-.status.cards .wide.column:nth-child(1) {
- margin-top: 2rem !important;
-}
-
-.status.cards .wide.column:nth-child(2) {
- margin-top: 2rem !important;
-}
-
-.status.cards .description {
- padding-bottom: 0 !important;
-}
-
-/* 服务器名 */
-.status.cards .flag {
- margin-right: 0.5rem !important;
-}
-
-/* 弹出卡片图标 */
-.status.cards .header > .info.icon {
- margin-right: 0 !important;
-}
-
-.nezha-secondary-font {
- color: #2175ba !important;
-}
-
-/* 上传下载 */
-.status.cards .outline.icon {
- margin-right: 1px !important;
-}
-
-i.arrow.alternate.circle.down.outline.icon {
- color: #2175ba !important;
-}
-
-i.arrow.alternate.circle.up.outline.icon {
- color: red !important;
-}
-
-/* 弹出卡片小箭头 */
-.ui.right.center.popup {
- margin: -3px 0 0 0.914286em !important;
- -webkit-transform-origin: left 50% !important;
- transform-origin: left 50% !important;
-}
-
-.ui.bottom.left.popup {
- margin-left: 1px !important;
- margin-top: 3px !important;
-}
-
-.ui.top.left.popup {
- margin-left: 0 !important;
- margin-bottom: 10px !important;
-}
-
-.ui.top.right.popup {
- margin-right: 0 !important;
- margin-bottom: 8px !important;
-}
-
-.ui.left.center.popup {
- margin: -3px .91428571em 0 0 !important;
- -webkit-transform-origin: right 50% !important;
- transform-origin: right 50% !important;
-}
-
-.ui.right.center.popup:before,
-.ui.left.center.popup:before {
- border: 0px solid #fafafaeb !important;
- background: #fafafaeb !important;
-}
-
-.ui.top.popup:before {
- border-color: #fafafaeb transparent transparent !important;
-}
-
-.ui.popup:before {
- border-color: #fafafaeb transparent transparent !important;
-}
-
-.ui.bottom.left.popup:before {
- border-radius: 0 !important;
- border: 1px solid transparent !important;
- border-color: #fafafaeb transparent transparent !important;
- background: #fafafaeb !important;
- -webkit-box-shadow: 0px 0px 0 0 #fafafaeb !important;
- box-shadow: 0px 0px 0 0 #fafafaeb !important;
- -webkit-tap-highlight-color: rgba(0,0,0,0) !important;
-}
-
-.ui.bottom.right.popup:before {
- border-radius: 0 !important;
- border: 1px solid transparent !important;
- border-color: #fafafaeb transparent transparent !important;
- background: #fafafaeb !important
- -webkit-box-shadow: 0px 0px 0 0 #fafafaeb !important;
- box-shadow: 0px 0px 0 0 #fafafaeb !important;
- -webkit-tap-highlight-color: rgba(0,0,0,0) !important;
-}
-
-.ui.top.left.popup:before {
- border-radius: 0 !important;
- border: 1px solid transparent !important;
- border-color: #fafafaeb transparent transparent !important;
- background: #fafafaeb !important;
- -webkit-box-shadow: 0px 0px 0 0 #fafafaeb !important;
- box-shadow: 0px 0px 0 0 #fafafaeb !important;
- -webkit-tap-highlight-color: rgba(0,0,0,0) !important;
-}
-
-.ui.top.right.popup:before {
- border-radius: 0 !important;
- border: 1px solid transparent !important;
- border-color: #fafafaeb transparent transparent !important;
- background: #fafafaeb !important;
- -webkit-box-shadow: 0px 0px 0 0 #fafafaeb !important;
- box-shadow: 0px 0px 0 0 #fafafaeb !important;
- -webkit-tap-highlight-color: rgba(0,0,0,0) !important;
-}
-
-.ui.left.center.popup:before {
- border-radius: 0 !important;
- border: 1px solid transparent !important;
- border-color: #fafafaeb transparent transparent !important;
- background: #fafafaeb !important;
- -webkit-box-shadow: 0px 0px 0 0 #fafafaeb !important;
- box-shadow: 0px 0px 0 0 #fafafaeb !important;
- -webkit-tap-highlight-color: rgba(0,0,0,0) !important;
-}
-
-/* 弹出卡片 */
-.status.cards .ui.content.popup {
- min-width: 20rem !important;
- line-height: 2rem !important;
- border-radius: 5px !important;
- border: 1px solid transparent !important;
- background-color: #fafafaeb !important;
- font-family: Arial,Helvetica,sans-serif !important;
-}
-
-.ui.content {
- margin: 0 !important;
- padding: 1em !important;
-}
-
-/* 服务页 */
-.ui.table {
- background: RGB(225,225,225,0.6) !important;
-}
-
-.ui.table thead th {
- background: transparent !important;
-}
-
-/* 服务页进度条 */
-.service-status .good {
- background-color: #2175ba !important;
-}
-
-.service-status .danger {
- background-color: red !important;
-}
-
-.service-status .warning {
- background-color: orange !important;
-}
-
-/* 版权 */
-.ui.inverted.segment, .ui.primary.inverted.segment {
- color: #000 !important;
- font-weight: bold !important;
- background-color: #fafafaa3 !important;
-}
-</style>
-
-<!--Logo和版权-->
-<script>
-window.onload = function(){
-var avatar=document.querySelector(".item img")
-var footer=document.querySelector("div.is-size-7")
-footer.innerHTML="Copyright info"
-footer.style.visibility="visible"
-avatar.src="https:/img.src"
-avatar.style.visibility="visible"
-}
-</script>
DayNight 主题更改进度条颜色、修改页脚示例(来自 @hyt-allen-xu @tech-fever)
点击展开/收起
<style>
-.ui.fine.progress> .progress-bar {
- background-color: #00a7d0 !important;
-}
-</style>
-
-<script>
-window.onload = function(){
- var avatar = document.querySelector("img");
- var footer = document.querySelector("div.footer-container");
- footer.innerHTML = "©2021 你的名字 & Powered by 你的名字";
- footer.style.visibility = "visible";
- avatar.src = "你的图片连接";
- avatar.style.visibility = "visible";
-}
-</script>
默认主题修改 LOGO、修改页脚示例(来自 @iLay1678)
点击展开/收起
<style>
-.right.menu>a {
- visibility: hidden;
-}
-.footer .is-size-7 {
- visibility: hidden;
-}
-.item img {
- visibility: hidden;
-}
-</style>
-<script>
-window.onload = function() {
- var avatar = document.querySelector(".item img");
- var footer = document.querySelector("div.is-size-7");
- footer.innerHTML = "Powered by 你的名字";
- footer.style.visibility = "visible";
- avatar.src = "你的方形 logo 地址";
- avatar.style.visibility = "visible";
-}
-</script>
hotaru 主题更改背景图片示例
点击展开/收起
<style>
-.hotaru-cover {
- background: url(https://s3.ax1x.com/2020/12/08/DzHv6A.jpg) center;
-}
-</style>
前台查看密码
未接入 CDN 的面板服务器域名/IP
IP 变更提醒
覆盖范围
特定服务器
提醒发送至指定的通知分组
通知中显示完整 IP 地址
禁止主页切换主题
任务管理
使用方法
* * * * * *
,分别对应秒 分 时 天 月 星期
,详情见计划表达式格式。
例如:0 0 3 * * *
对应为“每天 3 点”。&&/&
连接。
例如执行定期重启命令,可以在这里输入 reboot
。管理任务
常见问题
',11),n=[s];function i(l,c,d,g,p,h){return e(),o("div",null,n)}const m=t(a,[["render",i]]);export{u as __pageData,m as default};
diff --git a/assets/guide_tasks.md.BzPghW25.lean.js b/assets/guide_tasks.md.BzPghW25.lean.js
deleted file mode 100644
index dca8a5ed..00000000
--- a/assets/guide_tasks.md.BzPghW25.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as t,c as o,o as e,a4 as r}from"./chunks/framework.BmdFiWrL.js";const u=JSON.parse('{"title":"任务管理","description":"","frontmatter":{"outline":"deep"},"headers":[],"relativePath":"guide/tasks.md","filePath":"guide/tasks.md","lastUpdated":1732521523000}'),a={name:"guide/tasks.md"},s=r("",11),n=[s];function i(l,c,d,g,p,h){return e(),o("div",null,n)}const m=t(a,[["render",i]]);export{u as __pageData,m as default};
diff --git a/assets/index.md.YzXpf9jM.js b/assets/index.md.YzXpf9jM.js
deleted file mode 100644
index 03daab15..00000000
--- a/assets/index.md.YzXpf9jM.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o as a}from"./chunks/framework.BmdFiWrL.js";const p=JSON.parse('{"title":"哪吒监控 - 服务器监控与运维工具","titleTemplate":"使用文档","description":"","frontmatter":{"layout":"home","title":"哪吒监控 - 服务器监控与运维工具","titleTemplate":"使用文档","hero":{"name":"哪吒监控","text":"开源、轻量、易用的服务器监控与运维工具","image":"https://raw.githubusercontent.com/naiba/nezha/master/resource/static/brand.svg","actions":[{"theme":"brand","text":"预览","link":"https://ops.naibahq.com"},{"theme":"alt","text":"开始使用 →","link":"/guide/dashboard"}]},"features":[{"title":"一键安装","details":"支持一键安装面板和监控服务,操作便捷。兼容主流系统,包括 Linux、Windows、macOS、OpenWRT 以及群晖。"},{"title":"实时监控","details":"支持同时监控多个服务器的状态,提供历史网络状态和延迟图表,监控网页、端口可用性和 SSL 证书状态。支持故障和流量等状态告警,可通过 Telegram、邮件、微信等多种方式提醒。"},{"title":"轻松运维","details":"提供 API 获取服务器状态,支持WebSSH、DDNS 和流量监控。可设置定时和触发任务,并批量执行服务器任务。"}]},"headers":[],"relativePath":"index.md","filePath":"index.md","lastUpdated":1732521523000}'),i={name:"index.md"};function n(s,r,o,d,l,c){return a(),t("div")}const h=e(i,[["render",n]]);export{p as __pageData,h as default};
diff --git a/assets/index.md.YzXpf9jM.lean.js b/assets/index.md.YzXpf9jM.lean.js
deleted file mode 100644
index 03daab15..00000000
--- a/assets/index.md.YzXpf9jM.lean.js
+++ /dev/null
@@ -1 +0,0 @@
-import{_ as e,c as t,o as a}from"./chunks/framework.BmdFiWrL.js";const p=JSON.parse('{"title":"哪吒监控 - 服务器监控与运维工具","titleTemplate":"使用文档","description":"","frontmatter":{"layout":"home","title":"哪吒监控 - 服务器监控与运维工具","titleTemplate":"使用文档","hero":{"name":"哪吒监控","text":"开源、轻量、易用的服务器监控与运维工具","image":"https://raw.githubusercontent.com/naiba/nezha/master/resource/static/brand.svg","actions":[{"theme":"brand","text":"预览","link":"https://ops.naibahq.com"},{"theme":"alt","text":"开始使用 →","link":"/guide/dashboard"}]},"features":[{"title":"一键安装","details":"支持一键安装面板和监控服务,操作便捷。兼容主流系统,包括 Linux、Windows、macOS、OpenWRT 以及群晖。"},{"title":"实时监控","details":"支持同时监控多个服务器的状态,提供历史网络状态和延迟图表,监控网页、端口可用性和 SSL 证书状态。支持故障和流量等状态告警,可通过 Telegram、邮件、微信等多种方式提醒。"},{"title":"轻松运维","details":"提供 API 获取服务器状态,支持WebSSH、DDNS 和流量监控。可设置定时和触发任务,并批量执行服务器任务。"}]},"headers":[],"relativePath":"index.md","filePath":"index.md","lastUpdated":1732521523000}'),i={name:"index.md"};function n(s,r,o,d,l,c){return a(),t("div")}const h=e(i,[["render",n]]);export{p as __pageData,h as default};
diff --git a/assets/inter-italic-cyrillic-ext.r48I6akx.woff2 b/assets/inter-italic-cyrillic-ext.r48I6akx.woff2
deleted file mode 100644
index b6b603d5..00000000
Binary files a/assets/inter-italic-cyrillic-ext.r48I6akx.woff2 and /dev/null differ
diff --git a/assets/inter-italic-cyrillic.By2_1cv3.woff2 b/assets/inter-italic-cyrillic.By2_1cv3.woff2
deleted file mode 100644
index def40a4f..00000000
Binary files a/assets/inter-italic-cyrillic.By2_1cv3.woff2 and /dev/null differ
diff --git a/assets/inter-italic-greek-ext.1u6EdAuj.woff2 b/assets/inter-italic-greek-ext.1u6EdAuj.woff2
deleted file mode 100644
index e070c3d3..00000000
Binary files a/assets/inter-italic-greek-ext.1u6EdAuj.woff2 and /dev/null differ
diff --git a/assets/inter-italic-greek.DJ8dCoTZ.woff2 b/assets/inter-italic-greek.DJ8dCoTZ.woff2
deleted file mode 100644
index a3c16ca4..00000000
Binary files a/assets/inter-italic-greek.DJ8dCoTZ.woff2 and /dev/null differ
diff --git a/assets/inter-italic-latin-ext.CN1xVJS-.woff2 b/assets/inter-italic-latin-ext.CN1xVJS-.woff2
deleted file mode 100644
index 2210a899..00000000
Binary files a/assets/inter-italic-latin-ext.CN1xVJS-.woff2 and /dev/null differ
diff --git a/assets/inter-italic-latin.C2AdPX0b.woff2 b/assets/inter-italic-latin.C2AdPX0b.woff2
deleted file mode 100644
index 790d62dc..00000000
Binary files a/assets/inter-italic-latin.C2AdPX0b.woff2 and /dev/null differ
diff --git a/assets/inter-italic-vietnamese.BSbpV94h.woff2 b/assets/inter-italic-vietnamese.BSbpV94h.woff2
deleted file mode 100644
index 1eec0775..00000000
Binary files a/assets/inter-italic-vietnamese.BSbpV94h.woff2 and /dev/null differ
diff --git a/assets/inter-roman-cyrillic-ext.BBPuwvHQ.woff2 b/assets/inter-roman-cyrillic-ext.BBPuwvHQ.woff2
deleted file mode 100644
index 2cfe6153..00000000
Binary files a/assets/inter-roman-cyrillic-ext.BBPuwvHQ.woff2 and /dev/null differ
diff --git a/assets/inter-roman-cyrillic.C5lxZ8CY.woff2 b/assets/inter-roman-cyrillic.C5lxZ8CY.woff2
deleted file mode 100644
index e3886dd1..00000000
Binary files a/assets/inter-roman-cyrillic.C5lxZ8CY.woff2 and /dev/null differ
diff --git a/assets/inter-roman-greek-ext.CqjqNYQ-.woff2 b/assets/inter-roman-greek-ext.CqjqNYQ-.woff2
deleted file mode 100644
index 36d67487..00000000
Binary files a/assets/inter-roman-greek-ext.CqjqNYQ-.woff2 and /dev/null differ
diff --git a/assets/inter-roman-greek.BBVDIX6e.woff2 b/assets/inter-roman-greek.BBVDIX6e.woff2
deleted file mode 100644
index 2bed1e85..00000000
Binary files a/assets/inter-roman-greek.BBVDIX6e.woff2 and /dev/null differ
diff --git a/assets/inter-roman-latin-ext.4ZJIpNVo.woff2 b/assets/inter-roman-latin-ext.4ZJIpNVo.woff2
deleted file mode 100644
index 9a8d1e2b..00000000
Binary files a/assets/inter-roman-latin-ext.4ZJIpNVo.woff2 and /dev/null differ
diff --git a/assets/inter-roman-latin.Di8DUHzh.woff2 b/assets/inter-roman-latin.Di8DUHzh.woff2
deleted file mode 100644
index 07d3c53a..00000000
Binary files a/assets/inter-roman-latin.Di8DUHzh.woff2 and /dev/null differ
diff --git a/assets/inter-roman-vietnamese.BjW4sHH5.woff2 b/assets/inter-roman-vietnamese.BjW4sHH5.woff2
deleted file mode 100644
index 57bdc22a..00000000
Binary files a/assets/inter-roman-vietnamese.BjW4sHH5.woff2 and /dev/null differ
diff --git a/assets/qrcode.b_bFVwtg.png b/assets/qrcode.b_bFVwtg.png
deleted file mode 100644
index bf852b9c..00000000
Binary files a/assets/qrcode.b_bFVwtg.png and /dev/null differ
diff --git a/assets/style.CuK3IKp4.css b/assets/style.CuK3IKp4.css
deleted file mode 100644
index 266509b2..00000000
--- a/assets/style.CuK3IKp4.css
+++ /dev/null
@@ -1 +0,0 @@
-@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(/assets/inter-roman-cyrillic-ext.BBPuwvHQ.woff2) format("woff2");unicode-range:U+0460-052F,U+1C80-1C88,U+20B4,U+2DE0-2DFF,U+A640-A69F,U+FE2E-FE2F}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(/assets/inter-roman-cyrillic.C5lxZ8CY.woff2) format("woff2");unicode-range:U+0301,U+0400-045F,U+0490-0491,U+04B0-04B1,U+2116}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(/assets/inter-roman-greek-ext.CqjqNYQ-.woff2) format("woff2");unicode-range:U+1F00-1FFF}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(/assets/inter-roman-greek.BBVDIX6e.woff2) format("woff2");unicode-range:U+0370-0377,U+037A-037F,U+0384-038A,U+038C,U+038E-03A1,U+03A3-03FF}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(/assets/inter-roman-vietnamese.BjW4sHH5.woff2) format("woff2");unicode-range:U+0102-0103,U+0110-0111,U+0128-0129,U+0168-0169,U+01A0-01A1,U+01AF-01B0,U+0300-0301,U+0303-0304,U+0308-0309,U+0323,U+0329,U+1EA0-1EF9,U+20AB}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(/assets/inter-roman-latin-ext.4ZJIpNVo.woff2) format("woff2");unicode-range:U+0100-02AF,U+0304,U+0308,U+0329,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(/assets/inter-roman-latin.Di8DUHzh.woff2) format("woff2");unicode-range:U+0000-00FF,U+0131,U+0152-0153,U+02BB-02BC,U+02C6,U+02DA,U+02DC,U+0304,U+0308,U+0329,U+2000-206F,U+2074,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Inter;font-style:italic;font-weight:100 900;font-display:swap;src:url(/assets/inter-italic-cyrillic-ext.r48I6akx.woff2) format("woff2");unicode-range:U+0460-052F,U+1C80-1C88,U+20B4,U+2DE0-2DFF,U+A640-A69F,U+FE2E-FE2F}@font-face{font-family:Inter;font-style:italic;font-weight:100 900;font-display:swap;src:url(/assets/inter-italic-cyrillic.By2_1cv3.woff2) format("woff2");unicode-range:U+0301,U+0400-045F,U+0490-0491,U+04B0-04B1,U+2116}@font-face{font-family:Inter;font-style:italic;font-weight:100 900;font-display:swap;src:url(/assets/inter-italic-greek-ext.1u6EdAuj.woff2) format("woff2");unicode-range:U+1F00-1FFF}@font-face{font-family:Inter;font-style:italic;font-weight:100 900;font-display:swap;src:url(/assets/inter-italic-greek.DJ8dCoTZ.woff2) format("woff2");unicode-range:U+0370-0377,U+037A-037F,U+0384-038A,U+038C,U+038E-03A1,U+03A3-03FF}@font-face{font-family:Inter;font-style:italic;font-weight:100 900;font-display:swap;src:url(/assets/inter-italic-vietnamese.BSbpV94h.woff2) format("woff2");unicode-range:U+0102-0103,U+0110-0111,U+0128-0129,U+0168-0169,U+01A0-01A1,U+01AF-01B0,U+0300-0301,U+0303-0304,U+0308-0309,U+0323,U+0329,U+1EA0-1EF9,U+20AB}@font-face{font-family:Inter;font-style:italic;font-weight:100 900;font-display:swap;src:url(/assets/inter-italic-latin-ext.CN1xVJS-.woff2) format("woff2");unicode-range:U+0100-02AF,U+0304,U+0308,U+0329,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Inter;font-style:italic;font-weight:100 900;font-display:swap;src:url(/assets/inter-italic-latin.C2AdPX0b.woff2) format("woff2");unicode-range:U+0000-00FF,U+0131,U+0152-0153,U+02BB-02BC,U+02C6,U+02DA,U+02DC,U+0304,U+0308,U+0329,U+2000-206F,U+2074,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Punctuation SC;font-weight:400;src:local("PingFang SC Regular"),local("Noto Sans CJK SC"),local("Microsoft YaHei");unicode-range:U+201C,U+201D,U+2018,U+2019,U+2E3A,U+2014,U+2013,U+2026,U+00B7,U+007E,U+002F}@font-face{font-family:Punctuation SC;font-weight:500;src:local("PingFang SC Medium"),local("Noto Sans CJK SC"),local("Microsoft YaHei");unicode-range:U+201C,U+201D,U+2018,U+2019,U+2E3A,U+2014,U+2013,U+2026,U+00B7,U+007E,U+002F}@font-face{font-family:Punctuation SC;font-weight:600;src:local("PingFang SC Semibold"),local("Noto Sans CJK SC Bold"),local("Microsoft YaHei Bold");unicode-range:U+201C,U+201D,U+2018,U+2019,U+2E3A,U+2014,U+2013,U+2026,U+00B7,U+007E,U+002F}@font-face{font-family:Punctuation SC;font-weight:700;src:local("PingFang SC Semibold"),local("Noto Sans CJK SC Bold"),local("Microsoft YaHei Bold");unicode-range:U+201C,U+201D,U+2018,U+2019,U+2E3A,U+2014,U+2013,U+2026,U+00B7,U+007E,U+002F}:root{--vp-c-white: #ffffff;--vp-c-black: #000000;--vp-c-neutral: var(--vp-c-black);--vp-c-neutral-inverse: var(--vp-c-white)}.dark{--vp-c-neutral: var(--vp-c-white);--vp-c-neutral-inverse: var(--vp-c-black)}:root{--vp-c-gray-1: #dddde3;--vp-c-gray-2: #e4e4e9;--vp-c-gray-3: #ebebef;--vp-c-gray-soft: rgba(142, 150, 170, .14);--vp-c-indigo-1: #3451b2;--vp-c-indigo-2: #3a5ccc;--vp-c-indigo-3: #5672cd;--vp-c-indigo-soft: rgba(100, 108, 255, .14);--vp-c-purple-1: #6f42c1;--vp-c-purple-2: #7e4cc9;--vp-c-purple-3: #8e5cd9;--vp-c-purple-soft: rgba(159, 122, 234, .14);--vp-c-green-1: #18794e;--vp-c-green-2: #299764;--vp-c-green-3: #30a46c;--vp-c-green-soft: rgba(16, 185, 129, .14);--vp-c-yellow-1: #915930;--vp-c-yellow-2: #946300;--vp-c-yellow-3: #9f6a00;--vp-c-yellow-soft: rgba(234, 179, 8, .14);--vp-c-red-1: #b8272c;--vp-c-red-2: #d5393e;--vp-c-red-3: #e0575b;--vp-c-red-soft: rgba(244, 63, 94, .14);--vp-c-sponsor: #db2777}.dark{--vp-c-gray-1: #515c67;--vp-c-gray-2: #414853;--vp-c-gray-3: #32363f;--vp-c-gray-soft: rgba(101, 117, 133, .16);--vp-c-indigo-1: #a8b1ff;--vp-c-indigo-2: #5c73e7;--vp-c-indigo-3: #3e63dd;--vp-c-indigo-soft: rgba(100, 108, 255, .16);--vp-c-purple-1: #c8abfa;--vp-c-purple-2: #a879e6;--vp-c-purple-3: #8e5cd9;--vp-c-purple-soft: rgba(159, 122, 234, .16);--vp-c-green-1: #3dd68c;--vp-c-green-2: #30a46c;--vp-c-green-3: #298459;--vp-c-green-soft: rgba(16, 185, 129, .16);--vp-c-yellow-1: #f9b44e;--vp-c-yellow-2: #da8b17;--vp-c-yellow-3: #a46a0a;--vp-c-yellow-soft: rgba(234, 179, 8, .16);--vp-c-red-1: #f66f81;--vp-c-red-2: #f14158;--vp-c-red-3: #b62a3c;--vp-c-red-soft: rgba(244, 63, 94, .16)}:root{--vp-c-bg: #ffffff;--vp-c-bg-alt: #f6f6f7;--vp-c-bg-elv: #ffffff;--vp-c-bg-soft: #f6f6f7}.dark{--vp-c-bg: #1b1b1f;--vp-c-bg-alt: #161618;--vp-c-bg-elv: #202127;--vp-c-bg-soft: #202127}:root{--vp-c-border: #c2c2c4;--vp-c-divider: #e2e2e3;--vp-c-gutter: #e2e2e3}.dark{--vp-c-border: #3c3f44;--vp-c-divider: #2e2e32;--vp-c-gutter: #000000}:root{--vp-c-text-1: rgba(60, 60, 67);--vp-c-text-2: rgba(60, 60, 67, .78);--vp-c-text-3: rgba(60, 60, 67, .56)}.dark{--vp-c-text-1: rgba(255, 255, 245, .86);--vp-c-text-2: rgba(235, 235, 245, .6);--vp-c-text-3: rgba(235, 235, 245, .38)}:root{--vp-c-default-1: var(--vp-c-gray-1);--vp-c-default-2: var(--vp-c-gray-2);--vp-c-default-3: var(--vp-c-gray-3);--vp-c-default-soft: var(--vp-c-gray-soft);--vp-c-brand-1: var(--vp-c-indigo-1);--vp-c-brand-2: var(--vp-c-indigo-2);--vp-c-brand-3: var(--vp-c-indigo-3);--vp-c-brand-soft: var(--vp-c-indigo-soft);--vp-c-brand: var(--vp-c-brand-1);--vp-c-tip-1: var(--vp-c-brand-1);--vp-c-tip-2: var(--vp-c-brand-2);--vp-c-tip-3: var(--vp-c-brand-3);--vp-c-tip-soft: var(--vp-c-brand-soft);--vp-c-note-1: var(--vp-c-brand-1);--vp-c-note-2: var(--vp-c-brand-2);--vp-c-note-3: var(--vp-c-brand-3);--vp-c-note-soft: var(--vp-c-brand-soft);--vp-c-success-1: var(--vp-c-green-1);--vp-c-success-2: var(--vp-c-green-2);--vp-c-success-3: var(--vp-c-green-3);--vp-c-success-soft: var(--vp-c-green-soft);--vp-c-important-1: var(--vp-c-purple-1);--vp-c-important-2: var(--vp-c-purple-2);--vp-c-important-3: var(--vp-c-purple-3);--vp-c-important-soft: var(--vp-c-purple-soft);--vp-c-warning-1: var(--vp-c-yellow-1);--vp-c-warning-2: var(--vp-c-yellow-2);--vp-c-warning-3: var(--vp-c-yellow-3);--vp-c-warning-soft: var(--vp-c-yellow-soft);--vp-c-danger-1: var(--vp-c-red-1);--vp-c-danger-2: var(--vp-c-red-2);--vp-c-danger-3: var(--vp-c-red-3);--vp-c-danger-soft: var(--vp-c-red-soft);--vp-c-caution-1: var(--vp-c-red-1);--vp-c-caution-2: var(--vp-c-red-2);--vp-c-caution-3: var(--vp-c-red-3);--vp-c-caution-soft: var(--vp-c-red-soft)}:root{--vp-font-family-base: "Inter", ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";--vp-font-family-mono: ui-monospace, "Menlo", "Monaco", "Consolas", "Liberation Mono", "Courier New", monospace;font-optical-sizing:auto}:root:where(:lang(zh)){--vp-font-family-base: "Punctuation SC", "Inter", ui-sans-serif, system-ui, "PingFang SC", "Noto Sans CJK SC", "Noto Sans SC", "Heiti SC", "Microsoft YaHei", "DengXian", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji"}:root{--vp-shadow-1: 0 1px 2px rgba(0, 0, 0, .04), 0 1px 2px rgba(0, 0, 0, .06);--vp-shadow-2: 0 3px 12px rgba(0, 0, 0, .07), 0 1px 4px rgba(0, 0, 0, .07);--vp-shadow-3: 0 12px 32px rgba(0, 0, 0, .1), 0 2px 6px rgba(0, 0, 0, .08);--vp-shadow-4: 0 14px 44px rgba(0, 0, 0, .12), 0 3px 9px rgba(0, 0, 0, .12);--vp-shadow-5: 0 18px 56px rgba(0, 0, 0, .16), 0 4px 12px rgba(0, 0, 0, .16)}:root{--vp-z-index-footer: 10;--vp-z-index-local-nav: 20;--vp-z-index-nav: 30;--vp-z-index-layout-top: 40;--vp-z-index-backdrop: 50;--vp-z-index-sidebar: 60}@media (min-width: 960px){:root{--vp-z-index-sidebar: 25}}:root{--vp-layout-max-width: 1440px}:root{--vp-header-anchor-symbol: "#"}:root{--vp-code-line-height: 1.7;--vp-code-font-size: .875em;--vp-code-color: var(--vp-c-brand-1);--vp-code-link-color: var(--vp-c-brand-1);--vp-code-link-hover-color: var(--vp-c-brand-2);--vp-code-bg: var(--vp-c-default-soft);--vp-code-block-color: var(--vp-c-text-2);--vp-code-block-bg: var(--vp-c-bg-alt);--vp-code-block-divider-color: var(--vp-c-gutter);--vp-code-lang-color: var(--vp-c-text-3);--vp-code-line-highlight-color: var(--vp-c-default-soft);--vp-code-line-number-color: var(--vp-c-text-3);--vp-code-line-diff-add-color: var(--vp-c-success-soft);--vp-code-line-diff-add-symbol-color: var(--vp-c-success-1);--vp-code-line-diff-remove-color: var(--vp-c-danger-soft);--vp-code-line-diff-remove-symbol-color: var(--vp-c-danger-1);--vp-code-line-warning-color: var(--vp-c-warning-soft);--vp-code-line-error-color: var(--vp-c-danger-soft);--vp-code-copy-code-border-color: var(--vp-c-divider);--vp-code-copy-code-bg: var(--vp-c-bg-soft);--vp-code-copy-code-hover-border-color: var(--vp-c-divider);--vp-code-copy-code-hover-bg: var(--vp-c-bg);--vp-code-copy-code-active-text: var(--vp-c-text-2);--vp-code-copy-copied-text-content: "Copied";--vp-code-tab-divider: var(--vp-code-block-divider-color);--vp-code-tab-text-color: var(--vp-c-text-2);--vp-code-tab-bg: var(--vp-code-block-bg);--vp-code-tab-hover-text-color: var(--vp-c-text-1);--vp-code-tab-active-text-color: var(--vp-c-text-1);--vp-code-tab-active-bar-color: var(--vp-c-brand-1)}:root{--vp-button-brand-border: transparent;--vp-button-brand-text: var(--vp-c-white);--vp-button-brand-bg: var(--vp-c-brand-3);--vp-button-brand-hover-border: transparent;--vp-button-brand-hover-text: var(--vp-c-white);--vp-button-brand-hover-bg: var(--vp-c-brand-2);--vp-button-brand-active-border: transparent;--vp-button-brand-active-text: var(--vp-c-white);--vp-button-brand-active-bg: var(--vp-c-brand-1);--vp-button-alt-border: transparent;--vp-button-alt-text: var(--vp-c-text-1);--vp-button-alt-bg: var(--vp-c-default-3);--vp-button-alt-hover-border: transparent;--vp-button-alt-hover-text: var(--vp-c-text-1);--vp-button-alt-hover-bg: var(--vp-c-default-2);--vp-button-alt-active-border: transparent;--vp-button-alt-active-text: var(--vp-c-text-1);--vp-button-alt-active-bg: var(--vp-c-default-1);--vp-button-sponsor-border: var(--vp-c-text-2);--vp-button-sponsor-text: var(--vp-c-text-2);--vp-button-sponsor-bg: transparent;--vp-button-sponsor-hover-border: var(--vp-c-sponsor);--vp-button-sponsor-hover-text: var(--vp-c-sponsor);--vp-button-sponsor-hover-bg: transparent;--vp-button-sponsor-active-border: var(--vp-c-sponsor);--vp-button-sponsor-active-text: var(--vp-c-sponsor);--vp-button-sponsor-active-bg: transparent}:root{--vp-custom-block-font-size: 14px;--vp-custom-block-code-font-size: 13px;--vp-custom-block-info-border: transparent;--vp-custom-block-info-text: var(--vp-c-text-1);--vp-custom-block-info-bg: var(--vp-c-default-soft);--vp-custom-block-info-code-bg: var(--vp-c-default-soft);--vp-custom-block-note-border: transparent;--vp-custom-block-note-text: var(--vp-c-text-1);--vp-custom-block-note-bg: var(--vp-c-default-soft);--vp-custom-block-note-code-bg: var(--vp-c-default-soft);--vp-custom-block-tip-border: transparent;--vp-custom-block-tip-text: var(--vp-c-text-1);--vp-custom-block-tip-bg: var(--vp-c-tip-soft);--vp-custom-block-tip-code-bg: var(--vp-c-tip-soft);--vp-custom-block-important-border: transparent;--vp-custom-block-important-text: var(--vp-c-text-1);--vp-custom-block-important-bg: var(--vp-c-important-soft);--vp-custom-block-important-code-bg: var(--vp-c-important-soft);--vp-custom-block-warning-border: transparent;--vp-custom-block-warning-text: var(--vp-c-text-1);--vp-custom-block-warning-bg: var(--vp-c-warning-soft);--vp-custom-block-warning-code-bg: var(--vp-c-warning-soft);--vp-custom-block-danger-border: transparent;--vp-custom-block-danger-text: var(--vp-c-text-1);--vp-custom-block-danger-bg: var(--vp-c-danger-soft);--vp-custom-block-danger-code-bg: var(--vp-c-danger-soft);--vp-custom-block-caution-border: transparent;--vp-custom-block-caution-text: var(--vp-c-text-1);--vp-custom-block-caution-bg: var(--vp-c-caution-soft);--vp-custom-block-caution-code-bg: var(--vp-c-caution-soft);--vp-custom-block-details-border: var(--vp-custom-block-info-border);--vp-custom-block-details-text: var(--vp-custom-block-info-text);--vp-custom-block-details-bg: var(--vp-custom-block-info-bg);--vp-custom-block-details-code-bg: var(--vp-custom-block-info-code-bg)}:root{--vp-input-border-color: var(--vp-c-border);--vp-input-bg-color: var(--vp-c-bg-alt);--vp-input-switch-bg-color: var(--vp-c-default-soft)}:root{--vp-nav-height: 64px;--vp-nav-bg-color: var(--vp-c-bg);--vp-nav-screen-bg-color: var(--vp-c-bg);--vp-nav-logo-height: 24px}.hide-nav{--vp-nav-height: 0px}.hide-nav .VPSidebar{--vp-nav-height: 22px}:root{--vp-local-nav-bg-color: var(--vp-c-bg)}:root{--vp-sidebar-width: 272px;--vp-sidebar-bg-color: var(--vp-c-bg-alt)}:root{--vp-backdrop-bg-color: rgba(0, 0, 0, .6)}:root{--vp-home-hero-name-color: var(--vp-c-brand-1);--vp-home-hero-name-background: transparent;--vp-home-hero-image-background-image: none;--vp-home-hero-image-filter: none}:root{--vp-badge-info-border: transparent;--vp-badge-info-text: var(--vp-c-text-2);--vp-badge-info-bg: var(--vp-c-default-soft);--vp-badge-tip-border: transparent;--vp-badge-tip-text: var(--vp-c-tip-1);--vp-badge-tip-bg: var(--vp-c-tip-soft);--vp-badge-warning-border: transparent;--vp-badge-warning-text: var(--vp-c-warning-1);--vp-badge-warning-bg: var(--vp-c-warning-soft);--vp-badge-danger-border: transparent;--vp-badge-danger-text: var(--vp-c-danger-1);--vp-badge-danger-bg: var(--vp-c-danger-soft)}:root{--vp-carbon-ads-text-color: var(--vp-c-text-1);--vp-carbon-ads-poweredby-color: var(--vp-c-text-2);--vp-carbon-ads-bg-color: var(--vp-c-bg-soft);--vp-carbon-ads-hover-text-color: var(--vp-c-brand-1);--vp-carbon-ads-hover-poweredby-color: var(--vp-c-text-1)}:root{--vp-local-search-bg: var(--vp-c-bg);--vp-local-search-result-bg: var(--vp-c-bg);--vp-local-search-result-border: var(--vp-c-divider);--vp-local-search-result-selected-bg: var(--vp-c-bg);--vp-local-search-result-selected-border: var(--vp-c-brand-1);--vp-local-search-highlight-bg: var(--vp-c-brand-1);--vp-local-search-highlight-text: var(--vp-c-neutral-inverse)}@media (prefers-reduced-motion: reduce){*,:before,:after{animation-delay:-1ms!important;animation-duration:1ms!important;animation-iteration-count:1!important;background-attachment:initial!important;scroll-behavior:auto!important;transition-duration:0s!important;transition-delay:0s!important}}*,:before,:after{box-sizing:border-box}html{line-height:1.4;font-size:16px;-webkit-text-size-adjust:100%}html.dark{color-scheme:dark}body{margin:0;width:100%;min-width:320px;min-height:100vh;line-height:24px;font-family:var(--vp-font-family-base);font-size:16px;font-weight:400;color:var(--vp-c-text-1);background-color:var(--vp-c-bg);font-synthesis:style;text-rendering:optimizeLegibility;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}main{display:block}h1,h2,h3,h4,h5,h6{margin:0;line-height:24px;font-size:16px;font-weight:400}p{margin:0}strong,b{font-weight:600}a,area,button,[role=button],input,label,select,summary,textarea{touch-action:manipulation}a{color:inherit;text-decoration:inherit}ol,ul{list-style:none;margin:0;padding:0}blockquote{margin:0}pre,code,kbd,samp{font-family:var(--vp-font-family-mono)}img,svg,video,canvas,audio,iframe,embed,object{display:block}figure{margin:0}img,video{max-width:100%;height:auto}button,input,optgroup,select,textarea{border:0;padding:0;line-height:inherit;color:inherit}button{padding:0;font-family:inherit;background-color:transparent;background-image:none}button:enabled,[role=button]:enabled{cursor:pointer}button:focus,button:focus-visible{outline:1px dotted;outline:4px auto -webkit-focus-ring-color}button:focus:not(:focus-visible){outline:none!important}input:focus,textarea:focus,select:focus{outline:none}table{border-collapse:collapse}input{background-color:transparent}input:-ms-input-placeholder,textarea:-ms-input-placeholder{color:var(--vp-c-text-3)}input::-ms-input-placeholder,textarea::-ms-input-placeholder{color:var(--vp-c-text-3)}input::placeholder,textarea::placeholder{color:var(--vp-c-text-3)}input::-webkit-outer-spin-button,input::-webkit-inner-spin-button{-webkit-appearance:none;margin:0}input[type=number]{-moz-appearance:textfield}textarea{resize:vertical}select{-webkit-appearance:none}fieldset{margin:0;padding:0}h1,h2,h3,h4,h5,h6,li,p{overflow-wrap:break-word}vite-error-overlay{z-index:9999}mjx-container{overflow-x:auto}mjx-container>svg{display:inline-block;margin:auto}[class^=vpi-],[class*=" vpi-"],.vp-icon{width:1em;height:1em}[class^=vpi-].bg,[class*=" vpi-"].bg,.vp-icon.bg{background-size:100% 100%;background-color:transparent}[class^=vpi-]:not(.bg),[class*=" vpi-"]:not(.bg),.vp-icon:not(.bg){-webkit-mask:var(--icon) no-repeat;mask:var(--icon) no-repeat;-webkit-mask-size:100% 100%;mask-size:100% 100%;background-color:currentColor;color:inherit}.vpi-align-left{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='M21 6H3M15 12H3M17 18H3'/%3E%3C/svg%3E")}.vpi-arrow-right,.vpi-arrow-down,.vpi-arrow-left,.vpi-arrow-up{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='M5 12h14M12 5l7 7-7 7'/%3E%3C/svg%3E")}.vpi-chevron-right,.vpi-chevron-down,.vpi-chevron-left,.vpi-chevron-up{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='m9 18 6-6-6-6'/%3E%3C/svg%3E")}.vpi-chevron-down,.vpi-arrow-down{transform:rotate(90deg)}.vpi-chevron-left,.vpi-arrow-left{transform:rotate(180deg)}.vpi-chevron-up,.vpi-arrow-up{transform:rotate(-90deg)}.vpi-square-pen{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='M12 3H5a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h14a2 2 0 0 0 2-2v-7'/%3E%3Cpath d='M18.375 2.625a2.121 2.121 0 1 1 3 3L12 15l-4 1 1-4Z'/%3E%3C/svg%3E")}.vpi-plus{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='M5 12h14M12 5v14'/%3E%3C/svg%3E")}.vpi-sun{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Ccircle cx='12' cy='12' r='4'/%3E%3Cpath d='M12 2v2M12 20v2M4.93 4.93l1.41 1.41M17.66 17.66l1.41 1.41M2 12h2M20 12h2M6.34 17.66l-1.41 1.41M19.07 4.93l-1.41 1.41'/%3E%3C/svg%3E")}.vpi-moon{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='M12 3a6 6 0 0 0 9 9 9 9 0 1 1-9-9Z'/%3E%3C/svg%3E")}.vpi-more-horizontal{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Ccircle cx='12' cy='12' r='1'/%3E%3Ccircle cx='19' cy='12' r='1'/%3E%3Ccircle cx='5' cy='12' r='1'/%3E%3C/svg%3E")}.vpi-languages{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='m5 8 6 6M4 14l6-6 2-3M2 5h12M7 2h1M22 22l-5-10-5 10M14 18h6'/%3E%3C/svg%3E")}.vpi-heart{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='M19 14c1.49-1.46 3-3.21 3-5.5A5.5 5.5 0 0 0 16.5 3c-1.76 0-3 .5-4.5 2-1.5-1.5-2.74-2-4.5-2A5.5 5.5 0 0 0 2 8.5c0 2.3 1.5 4.05 3 5.5l7 7Z'/%3E%3C/svg%3E")}.vpi-search{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Ccircle cx='11' cy='11' r='8'/%3E%3Cpath d='m21 21-4.3-4.3'/%3E%3C/svg%3E")}.vpi-layout-list{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Crect width='7' height='7' x='3' y='3' rx='1'/%3E%3Crect width='7' height='7' x='3' y='14' rx='1'/%3E%3Cpath d='M14 4h7M14 9h7M14 15h7M14 20h7'/%3E%3C/svg%3E")}.vpi-delete{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='M20 5H9l-7 7 7 7h11a2 2 0 0 0 2-2V7a2 2 0 0 0-2-2ZM18 9l-6 6M12 9l6 6'/%3E%3C/svg%3E")}.vpi-corner-down-left{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='m9 10-5 5 5 5'/%3E%3Cpath d='M20 4v7a4 4 0 0 1-4 4H4'/%3E%3C/svg%3E")}:root{--vp-icon-copy: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='rgba(128,128,128,1)' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Crect width='8' height='4' x='8' y='2' rx='1' ry='1'/%3E%3Cpath d='M16 4h2a2 2 0 0 1 2 2v14a2 2 0 0 1-2 2H6a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h2'/%3E%3C/svg%3E");--vp-icon-copied: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='rgba(128,128,128,1)' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Crect width='8' height='4' x='8' y='2' rx='1' ry='1'/%3E%3Cpath d='M16 4h2a2 2 0 0 1 2 2v14a2 2 0 0 1-2 2H6a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h2'/%3E%3Cpath d='m9 14 2 2 4-4'/%3E%3C/svg%3E")}.vpi-social-discord{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M20.317 4.37a19.791 19.791 0 0 0-4.885-1.515.074.074 0 0 0-.079.037c-.21.375-.444.864-.608 1.25a18.27 18.27 0 0 0-5.487 0 12.64 12.64 0 0 0-.617-1.25.077.077 0 0 0-.079-.037A19.736 19.736 0 0 0 3.677 4.37a.07.07 0 0 0-.032.027C.533 9.046-.32 13.58.099 18.057a.082.082 0 0 0 .031.057 19.9 19.9 0 0 0 5.993 3.03.078.078 0 0 0 .084-.028c.462-.63.874-1.295 1.226-1.994a.076.076 0 0 0-.041-.106 13.107 13.107 0 0 1-1.872-.892.077.077 0 0 1-.008-.128 10.2 10.2 0 0 0 .372-.292.074.074 0 0 1 .077-.01c3.928 1.793 8.18 1.793 12.062 0a.074.074 0 0 1 .078.01c.12.098.246.198.373.292a.077.077 0 0 1-.006.127 12.299 12.299 0 0 1-1.873.892.077.077 0 0 0-.041.107c.36.698.772 1.362 1.225 1.993a.076.076 0 0 0 .084.028 19.839 19.839 0 0 0 6.002-3.03.077.077 0 0 0 .032-.054c.5-5.177-.838-9.674-3.549-13.66a.061.061 0 0 0-.031-.03zM8.02 15.33c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.956-2.419 2.157-2.419 1.21 0 2.176 1.096 2.157 2.42 0 1.333-.956 2.418-2.157 2.418zm7.975 0c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.955-2.419 2.157-2.419 1.21 0 2.176 1.096 2.157 2.42 0 1.333-.946 2.418-2.157 2.418Z'/%3E%3C/svg%3E")}.vpi-social-facebook{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M9.101 23.691v-7.98H6.627v-3.667h2.474v-1.58c0-4.085 1.848-5.978 5.858-5.978.401 0 .955.042 1.468.103a8.68 8.68 0 0 1 1.141.195v3.325a8.623 8.623 0 0 0-.653-.036 26.805 26.805 0 0 0-.733-.009c-.707 0-1.259.096-1.675.309a1.686 1.686 0 0 0-.679.622c-.258.42-.374.995-.374 1.752v1.297h3.919l-.386 2.103-.287 1.564h-3.246v8.245C19.396 23.238 24 18.179 24 12.044c0-6.627-5.373-12-12-12s-12 5.373-12 12c0 5.628 3.874 10.35 9.101 11.647Z'/%3E%3C/svg%3E")}.vpi-social-github{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M12 .297c-6.63 0-12 5.373-12 12 0 5.303 3.438 9.8 8.205 11.385.6.113.82-.258.82-.577 0-.285-.01-1.04-.015-2.04-3.338.724-4.042-1.61-4.042-1.61C4.422 18.07 3.633 17.7 3.633 17.7c-1.087-.744.084-.729.084-.729 1.205.084 1.838 1.236 1.838 1.236 1.07 1.835 2.809 1.305 3.495.998.108-.776.417-1.305.76-1.605-2.665-.3-5.466-1.332-5.466-5.93 0-1.31.465-2.38 1.235-3.22-.135-.303-.54-1.523.105-3.176 0 0 1.005-.322 3.3 1.23.96-.267 1.98-.399 3-.405 1.02.006 2.04.138 3 .405 2.28-1.552 3.285-1.23 3.285-1.23.645 1.653.24 2.873.12 3.176.765.84 1.23 1.91 1.23 3.22 0 4.61-2.805 5.625-5.475 5.92.42.36.81 1.096.81 2.22 0 1.606-.015 2.896-.015 3.286 0 .315.21.69.825.57C20.565 22.092 24 17.592 24 12.297c0-6.627-5.373-12-12-12'/%3E%3C/svg%3E")}.vpi-social-instagram{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M7.03.084c-1.277.06-2.149.264-2.91.563a5.874 5.874 0 0 0-2.124 1.388 5.878 5.878 0 0 0-1.38 2.127C.321 4.926.12 5.8.064 7.076.008 8.354-.005 8.764.001 12.023c.007 3.259.021 3.667.083 4.947.061 1.277.264 2.149.563 2.911.308.789.72 1.457 1.388 2.123a5.872 5.872 0 0 0 2.129 1.38c.763.295 1.636.496 2.913.552 1.278.056 1.689.069 4.947.063 3.257-.007 3.668-.021 4.947-.082 1.28-.06 2.147-.265 2.91-.563a5.881 5.881 0 0 0 2.123-1.388 5.881 5.881 0 0 0 1.38-2.129c.295-.763.496-1.636.551-2.912.056-1.28.07-1.69.063-4.948-.006-3.258-.02-3.667-.081-4.947-.06-1.28-.264-2.148-.564-2.911a5.892 5.892 0 0 0-1.387-2.123 5.857 5.857 0 0 0-2.128-1.38C19.074.322 18.202.12 16.924.066 15.647.009 15.236-.006 11.977 0 8.718.008 8.31.021 7.03.084m.14 21.693c-1.17-.05-1.805-.245-2.228-.408a3.736 3.736 0 0 1-1.382-.895 3.695 3.695 0 0 1-.9-1.378c-.165-.423-.363-1.058-.417-2.228-.06-1.264-.072-1.644-.08-4.848-.006-3.204.006-3.583.061-4.848.05-1.169.246-1.805.408-2.228.216-.561.477-.96.895-1.382a3.705 3.705 0 0 1 1.379-.9c.423-.165 1.057-.361 2.227-.417 1.265-.06 1.644-.072 4.848-.08 3.203-.006 3.583.006 4.85.062 1.168.05 1.804.244 2.227.408.56.216.96.475 1.382.895.421.42.681.817.9 1.378.165.422.362 1.056.417 2.227.06 1.265.074 1.645.08 4.848.005 3.203-.006 3.583-.061 4.848-.051 1.17-.245 1.805-.408 2.23-.216.56-.477.96-.896 1.38a3.705 3.705 0 0 1-1.378.9c-.422.165-1.058.362-2.226.418-1.266.06-1.645.072-4.85.079-3.204.007-3.582-.006-4.848-.06m9.783-16.192a1.44 1.44 0 1 0 1.437-1.442 1.44 1.44 0 0 0-1.437 1.442M5.839 12.012a6.161 6.161 0 1 0 12.323-.024 6.162 6.162 0 0 0-12.323.024M8 12.008A4 4 0 1 1 12.008 16 4 4 0 0 1 8 12.008'/%3E%3C/svg%3E")}.vpi-social-linkedin{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M20.447 20.452h-3.554v-5.569c0-1.328-.027-3.037-1.852-3.037-1.853 0-2.136 1.445-2.136 2.939v5.667H9.351V9h3.414v1.561h.046c.477-.9 1.637-1.85 3.37-1.85 3.601 0 4.267 2.37 4.267 5.455v6.286zM5.337 7.433a2.062 2.062 0 0 1-2.063-2.065 2.064 2.064 0 1 1 2.063 2.065zm1.782 13.019H3.555V9h3.564v11.452zM22.225 0H1.771C.792 0 0 .774 0 1.729v20.542C0 23.227.792 24 1.771 24h20.451C23.2 24 24 23.227 24 22.271V1.729C24 .774 23.2 0 22.222 0h.003z'/%3E%3C/svg%3E")}.vpi-social-mastodon{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M23.268 5.313c-.35-2.578-2.617-4.61-5.304-5.004C17.51.242 15.792 0 11.813 0h-.03c-3.98 0-4.835.242-5.288.309C3.882.692 1.496 2.518.917 5.127.64 6.412.61 7.837.661 9.143c.074 1.874.088 3.745.26 5.611.118 1.24.325 2.47.62 3.68.55 2.237 2.777 4.098 4.96 4.857 2.336.792 4.849.923 7.256.38.265-.061.527-.132.786-.213.585-.184 1.27-.39 1.774-.753a.057.057 0 0 0 .023-.043v-1.809a.052.052 0 0 0-.02-.041.053.053 0 0 0-.046-.01 20.282 20.282 0 0 1-4.709.545c-2.73 0-3.463-1.284-3.674-1.818a5.593 5.593 0 0 1-.319-1.433.053.053 0 0 1 .066-.054c1.517.363 3.072.546 4.632.546.376 0 .75 0 1.125-.01 1.57-.044 3.224-.124 4.768-.422.038-.008.077-.015.11-.024 2.435-.464 4.753-1.92 4.989-5.604.008-.145.03-1.52.03-1.67.002-.512.167-3.63-.024-5.545zm-3.748 9.195h-2.561V8.29c0-1.309-.55-1.976-1.67-1.976-1.23 0-1.846.79-1.846 2.35v3.403h-2.546V8.663c0-1.56-.617-2.35-1.848-2.35-1.112 0-1.668.668-1.67 1.977v6.218H4.822V8.102c0-1.31.337-2.35 1.011-3.12.696-.77 1.608-1.164 2.74-1.164 1.311 0 2.302.5 2.962 1.498l.638 1.06.638-1.06c.66-.999 1.65-1.498 2.96-1.498 1.13 0 2.043.395 2.74 1.164.675.77 1.012 1.81 1.012 3.12z'/%3E%3C/svg%3E")}.vpi-social-npm{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M1.763 0C.786 0 0 .786 0 1.763v20.474C0 23.214.786 24 1.763 24h20.474c.977 0 1.763-.786 1.763-1.763V1.763C24 .786 23.214 0 22.237 0zM5.13 5.323l13.837.019-.009 13.836h-3.464l.01-10.382h-3.456L12.04 19.17H5.113z'/%3E%3C/svg%3E")}.vpi-social-slack{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M5.042 15.165a2.528 2.528 0 0 1-2.52 2.523A2.528 2.528 0 0 1 0 15.165a2.527 2.527 0 0 1 2.522-2.52h2.52v2.52zm1.271 0a2.527 2.527 0 0 1 2.521-2.52 2.527 2.527 0 0 1 2.521 2.52v6.313A2.528 2.528 0 0 1 8.834 24a2.528 2.528 0 0 1-2.521-2.522v-6.313zM8.834 5.042a2.528 2.528 0 0 1-2.521-2.52A2.528 2.528 0 0 1 8.834 0a2.528 2.528 0 0 1 2.521 2.522v2.52H8.834zm0 1.271a2.528 2.528 0 0 1 2.521 2.521 2.528 2.528 0 0 1-2.521 2.521H2.522A2.528 2.528 0 0 1 0 8.834a2.528 2.528 0 0 1 2.522-2.521h6.312zm10.122 2.521a2.528 2.528 0 0 1 2.522-2.521A2.528 2.528 0 0 1 24 8.834a2.528 2.528 0 0 1-2.522 2.521h-2.522V8.834zm-1.268 0a2.528 2.528 0 0 1-2.523 2.521 2.527 2.527 0 0 1-2.52-2.521V2.522A2.527 2.527 0 0 1 15.165 0a2.528 2.528 0 0 1 2.523 2.522v6.312zm-2.523 10.122a2.528 2.528 0 0 1 2.523 2.522A2.528 2.528 0 0 1 15.165 24a2.527 2.527 0 0 1-2.52-2.522v-2.522h2.52zm0-1.268a2.527 2.527 0 0 1-2.52-2.523 2.526 2.526 0 0 1 2.52-2.52h6.313A2.527 2.527 0 0 1 24 15.165a2.528 2.528 0 0 1-2.522 2.523h-6.313z'/%3E%3C/svg%3E")}.vpi-social-twitter,.vpi-social-x{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M18.901 1.153h3.68l-8.04 9.19L24 22.846h-7.406l-5.8-7.584-6.638 7.584H.474l8.6-9.83L0 1.154h7.594l5.243 6.932ZM17.61 20.644h2.039L6.486 3.24H4.298Z'/%3E%3C/svg%3E")}.vpi-social-youtube{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M23.498 6.186a3.016 3.016 0 0 0-2.122-2.136C19.505 3.545 12 3.545 12 3.545s-7.505 0-9.377.505A3.017 3.017 0 0 0 .502 6.186C0 8.07 0 12 0 12s0 3.93.502 5.814a3.016 3.016 0 0 0 2.122 2.136c1.871.505 9.376.505 9.376.505s7.505 0 9.377-.505a3.015 3.015 0 0 0 2.122-2.136C24 15.93 24 12 24 12s0-3.93-.502-5.814zM9.545 15.568V8.432L15.818 12l-6.273 3.568z'/%3E%3C/svg%3E")}.visually-hidden{position:absolute;width:1px;height:1px;white-space:nowrap;clip:rect(0 0 0 0);clip-path:inset(50%);overflow:hidden}.custom-block{border:1px solid transparent;border-radius:8px;padding:16px 16px 8px;line-height:24px;font-size:var(--vp-custom-block-font-size);color:var(--vp-c-text-2)}.custom-block.info{border-color:var(--vp-custom-block-info-border);color:var(--vp-custom-block-info-text);background-color:var(--vp-custom-block-info-bg)}.custom-block.info a,.custom-block.info code{color:var(--vp-c-brand-1)}.custom-block.info a:hover,.custom-block.info a:hover>code{color:var(--vp-c-brand-2)}.custom-block.info code{background-color:var(--vp-custom-block-info-code-bg)}.custom-block.note{border-color:var(--vp-custom-block-note-border);color:var(--vp-custom-block-note-text);background-color:var(--vp-custom-block-note-bg)}.custom-block.note a,.custom-block.note code{color:var(--vp-c-brand-1)}.custom-block.note a:hover,.custom-block.note a:hover>code{color:var(--vp-c-brand-2)}.custom-block.note code{background-color:var(--vp-custom-block-note-code-bg)}.custom-block.tip{border-color:var(--vp-custom-block-tip-border);color:var(--vp-custom-block-tip-text);background-color:var(--vp-custom-block-tip-bg)}.custom-block.tip a,.custom-block.tip code{color:var(--vp-c-tip-1)}.custom-block.tip a:hover,.custom-block.tip a:hover>code{color:var(--vp-c-tip-2)}.custom-block.tip code{background-color:var(--vp-custom-block-tip-code-bg)}.custom-block.important{border-color:var(--vp-custom-block-important-border);color:var(--vp-custom-block-important-text);background-color:var(--vp-custom-block-important-bg)}.custom-block.important a,.custom-block.important code{color:var(--vp-c-important-1)}.custom-block.important a:hover,.custom-block.important a:hover>code{color:var(--vp-c-important-2)}.custom-block.important code{background-color:var(--vp-custom-block-important-code-bg)}.custom-block.warning{border-color:var(--vp-custom-block-warning-border);color:var(--vp-custom-block-warning-text);background-color:var(--vp-custom-block-warning-bg)}.custom-block.warning a,.custom-block.warning code{color:var(--vp-c-warning-1)}.custom-block.warning a:hover,.custom-block.warning a:hover>code{color:var(--vp-c-warning-2)}.custom-block.warning code{background-color:var(--vp-custom-block-warning-code-bg)}.custom-block.danger{border-color:var(--vp-custom-block-danger-border);color:var(--vp-custom-block-danger-text);background-color:var(--vp-custom-block-danger-bg)}.custom-block.danger a,.custom-block.danger code{color:var(--vp-c-danger-1)}.custom-block.danger a:hover,.custom-block.danger a:hover>code{color:var(--vp-c-danger-2)}.custom-block.danger code{background-color:var(--vp-custom-block-danger-code-bg)}.custom-block.caution{border-color:var(--vp-custom-block-caution-border);color:var(--vp-custom-block-caution-text);background-color:var(--vp-custom-block-caution-bg)}.custom-block.caution a,.custom-block.caution code{color:var(--vp-c-caution-1)}.custom-block.caution a:hover,.custom-block.caution a:hover>code{color:var(--vp-c-caution-2)}.custom-block.caution code{background-color:var(--vp-custom-block-caution-code-bg)}.custom-block.details{border-color:var(--vp-custom-block-details-border);color:var(--vp-custom-block-details-text);background-color:var(--vp-custom-block-details-bg)}.custom-block.details a{color:var(--vp-c-brand-1)}.custom-block.details a:hover,.custom-block.details a:hover>code{color:var(--vp-c-brand-2)}.custom-block.details code{background-color:var(--vp-custom-block-details-code-bg)}.custom-block-title{font-weight:600}.custom-block p+p{margin:8px 0}.custom-block.details summary{margin:0 0 8px;font-weight:700;cursor:pointer;-webkit-user-select:none;user-select:none}.custom-block.details summary+p{margin:8px 0}.custom-block a{color:inherit;font-weight:600;text-decoration:underline;text-underline-offset:2px;transition:opacity .25s}.custom-block a:hover{opacity:.75}.custom-block code{font-size:var(--vp-custom-block-code-font-size)}.custom-block.custom-block th,.custom-block.custom-block blockquote>p{font-size:var(--vp-custom-block-font-size);color:inherit}.dark .vp-code span{color:var(--shiki-dark, inherit)}html:not(.dark) .vp-code span{color:var(--shiki-light, inherit)}.vp-code-group{margin-top:16px}.vp-code-group .tabs{position:relative;display:flex;margin-right:-24px;margin-left:-24px;padding:0 12px;background-color:var(--vp-code-tab-bg);overflow-x:auto;overflow-y:hidden;box-shadow:inset 0 -1px var(--vp-code-tab-divider)}@media (min-width: 640px){.vp-code-group .tabs{margin-right:0;margin-left:0;border-radius:8px 8px 0 0}}.vp-code-group .tabs input{position:fixed;opacity:0;pointer-events:none}.vp-code-group .tabs label{position:relative;display:inline-block;border-bottom:1px solid transparent;padding:0 12px;line-height:48px;font-size:14px;font-weight:500;color:var(--vp-code-tab-text-color);white-space:nowrap;cursor:pointer;transition:color .25s}.vp-code-group .tabs label:after{position:absolute;right:8px;bottom:-1px;left:8px;z-index:1;height:2px;border-radius:2px;content:"";background-color:transparent;transition:background-color .25s}.vp-code-group label:hover{color:var(--vp-code-tab-hover-text-color)}.vp-code-group input:checked+label{color:var(--vp-code-tab-active-text-color)}.vp-code-group input:checked+label:after{background-color:var(--vp-code-tab-active-bar-color)}.vp-code-group div[class*=language-],.vp-block{display:none;margin-top:0!important;border-top-left-radius:0!important;border-top-right-radius:0!important}.vp-code-group div[class*=language-].active,.vp-block.active{display:block}.vp-block{padding:20px 24px}.vp-doc h1,.vp-doc h2,.vp-doc h3,.vp-doc h4,.vp-doc h5,.vp-doc h6{position:relative;font-weight:600;outline:none}.vp-doc h1{letter-spacing:-.02em;line-height:40px;font-size:28px}.vp-doc h2{margin:48px 0 16px;border-top:1px solid var(--vp-c-divider);padding-top:24px;letter-spacing:-.02em;line-height:32px;font-size:24px}.vp-doc h3{margin:32px 0 0;letter-spacing:-.01em;line-height:28px;font-size:20px}.vp-doc .header-anchor{position:absolute;top:0;left:0;margin-left:-.87em;font-weight:500;-webkit-user-select:none;user-select:none;opacity:0;text-decoration:none;transition:color .25s,opacity .25s}.vp-doc .header-anchor:before{content:var(--vp-header-anchor-symbol)}.vp-doc h1:hover .header-anchor,.vp-doc h1 .header-anchor:focus,.vp-doc h2:hover .header-anchor,.vp-doc h2 .header-anchor:focus,.vp-doc h3:hover .header-anchor,.vp-doc h3 .header-anchor:focus,.vp-doc h4:hover .header-anchor,.vp-doc h4 .header-anchor:focus,.vp-doc h5:hover .header-anchor,.vp-doc h5 .header-anchor:focus,.vp-doc h6:hover .header-anchor,.vp-doc h6 .header-anchor:focus{opacity:1}@media (min-width: 768px){.vp-doc h1{letter-spacing:-.02em;line-height:40px;font-size:32px}}.vp-doc h2 .header-anchor{top:24px}.vp-doc p,.vp-doc summary{margin:16px 0}.vp-doc p{line-height:28px}.vp-doc blockquote{margin:16px 0;border-left:2px solid var(--vp-c-divider);padding-left:16px;transition:border-color .5s}.vp-doc blockquote>p{margin:0;font-size:16px;color:var(--vp-c-text-2);transition:color .5s}.vp-doc a{font-weight:500;color:var(--vp-c-brand-1);text-decoration:underline;text-underline-offset:2px;transition:color .25s,opacity .25s}.vp-doc a:hover{color:var(--vp-c-brand-2)}.vp-doc strong{font-weight:600}.vp-doc ul,.vp-doc ol{padding-left:1.25rem;margin:16px 0}.vp-doc ul{list-style:disc}.vp-doc ol{list-style:decimal}.vp-doc li+li{margin-top:8px}.vp-doc li>ol,.vp-doc li>ul{margin:8px 0 0}.vp-doc table{display:block;border-collapse:collapse;margin:20px 0;overflow-x:auto}.vp-doc tr{background-color:var(--vp-c-bg);border-top:1px solid var(--vp-c-divider);transition:background-color .5s}.vp-doc tr:nth-child(2n){background-color:var(--vp-c-bg-soft)}.vp-doc th,.vp-doc td{border:1px solid var(--vp-c-divider);padding:8px 16px}.vp-doc th{text-align:left;font-size:14px;font-weight:600;color:var(--vp-c-text-2);background-color:var(--vp-c-bg-soft)}.vp-doc td{font-size:14px}.vp-doc hr{margin:16px 0;border:none;border-top:1px solid var(--vp-c-divider)}.vp-doc .custom-block{margin:16px 0}.vp-doc .custom-block p{margin:8px 0;line-height:24px}.vp-doc .custom-block p:first-child{margin:0}.vp-doc .custom-block div[class*=language-]{margin:8px 0;border-radius:8px}.vp-doc .custom-block div[class*=language-] code{font-weight:400;background-color:transparent}.vp-doc .custom-block .vp-code-group .tabs{margin:0;border-radius:8px 8px 0 0}.vp-doc :not(pre,h1,h2,h3,h4,h5,h6)>code{font-size:var(--vp-code-font-size);color:var(--vp-code-color)}.vp-doc :not(pre)>code{border-radius:4px;padding:3px 6px;background-color:var(--vp-code-bg);transition:color .25s,background-color .5s}.vp-doc a>code{color:var(--vp-code-link-color)}.vp-doc a:hover>code{color:var(--vp-code-link-hover-color)}.vp-doc h1>code,.vp-doc h2>code,.vp-doc h3>code{font-size:.9em}.vp-doc div[class*=language-],.vp-block{position:relative;margin:16px -24px;background-color:var(--vp-code-block-bg);overflow-x:auto;transition:background-color .5s}@media (min-width: 640px){.vp-doc div[class*=language-],.vp-block{border-radius:8px;margin:16px 0}}@media (max-width: 639px){.vp-doc li div[class*=language-]{border-radius:8px 0 0 8px}}.vp-doc div[class*=language-]+div[class*=language-],.vp-doc div[class$=-api]+div[class*=language-],.vp-doc div[class*=language-]+div[class$=-api]>div[class*=language-]{margin-top:-8px}.vp-doc [class*=language-] pre,.vp-doc [class*=language-] code{direction:ltr;text-align:left;white-space:pre;word-spacing:normal;word-break:normal;word-wrap:normal;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-hyphens:none;-moz-hyphens:none;-ms-hyphens:none;hyphens:none}.vp-doc [class*=language-] pre{position:relative;z-index:1;margin:0;padding:20px 0;background:transparent;overflow-x:auto}.vp-doc [class*=language-] code{display:block;padding:0 24px;width:fit-content;min-width:100%;line-height:var(--vp-code-line-height);font-size:var(--vp-code-font-size);color:var(--vp-code-block-color);transition:color .5s}.vp-doc [class*=language-] code .highlighted{background-color:var(--vp-code-line-highlight-color);transition:background-color .5s;margin:0 -24px;padding:0 24px;width:calc(100% + 48px);display:inline-block}.vp-doc [class*=language-] code .highlighted.error{background-color:var(--vp-code-line-error-color)}.vp-doc [class*=language-] code .highlighted.warning{background-color:var(--vp-code-line-warning-color)}.vp-doc [class*=language-] code .diff{transition:background-color .5s;margin:0 -24px;padding:0 24px;width:calc(100% + 48px);display:inline-block}.vp-doc [class*=language-] code .diff:before{position:absolute;left:10px}.vp-doc [class*=language-] .has-focused-lines .line:not(.has-focus){filter:blur(.095rem);opacity:.4;transition:filter .35s,opacity .35s}.vp-doc [class*=language-] .has-focused-lines .line:not(.has-focus){opacity:.7;transition:filter .35s,opacity .35s}.vp-doc [class*=language-]:hover .has-focused-lines .line:not(.has-focus){filter:blur(0);opacity:1}.vp-doc [class*=language-] code .diff.remove{background-color:var(--vp-code-line-diff-remove-color);opacity:.7}.vp-doc [class*=language-] code .diff.remove:before{content:"-";color:var(--vp-code-line-diff-remove-symbol-color)}.vp-doc [class*=language-] code .diff.add{background-color:var(--vp-code-line-diff-add-color)}.vp-doc [class*=language-] code .diff.add:before{content:"+";color:var(--vp-code-line-diff-add-symbol-color)}.vp-doc div[class*=language-].line-numbers-mode{padding-left:32px}.vp-doc .line-numbers-wrapper{position:absolute;top:0;bottom:0;left:0;z-index:3;border-right:1px solid var(--vp-code-block-divider-color);padding-top:20px;width:32px;text-align:center;font-family:var(--vp-font-family-mono);line-height:var(--vp-code-line-height);font-size:var(--vp-code-font-size);color:var(--vp-code-line-number-color);transition:border-color .5s,color .5s}.vp-doc [class*=language-]>button.copy{direction:ltr;position:absolute;top:12px;right:12px;z-index:3;border:1px solid var(--vp-code-copy-code-border-color);border-radius:4px;width:40px;height:40px;background-color:var(--vp-code-copy-code-bg);opacity:0;cursor:pointer;background-image:var(--vp-icon-copy);background-position:50%;background-size:20px;background-repeat:no-repeat;transition:border-color .25s,background-color .25s,opacity .25s}.vp-doc [class*=language-]:hover>button.copy,.vp-doc [class*=language-]>button.copy:focus{opacity:1}.vp-doc [class*=language-]>button.copy:hover,.vp-doc [class*=language-]>button.copy.copied{border-color:var(--vp-code-copy-code-hover-border-color);background-color:var(--vp-code-copy-code-hover-bg)}.vp-doc [class*=language-]>button.copy.copied,.vp-doc [class*=language-]>button.copy:hover.copied{border-radius:0 4px 4px 0;background-color:var(--vp-code-copy-code-hover-bg);background-image:var(--vp-icon-copied)}.vp-doc [class*=language-]>button.copy.copied:before,.vp-doc [class*=language-]>button.copy:hover.copied:before{position:relative;top:-1px;transform:translate(calc(-100% - 1px));display:flex;justify-content:center;align-items:center;border:1px solid var(--vp-code-copy-code-hover-border-color);border-right:0;border-radius:4px 0 0 4px;padding:0 10px;width:fit-content;height:40px;text-align:center;font-size:12px;font-weight:500;color:var(--vp-code-copy-code-active-text);background-color:var(--vp-code-copy-code-hover-bg);white-space:nowrap;content:var(--vp-code-copy-copied-text-content)}.vp-doc [class*=language-]>span.lang{position:absolute;top:2px;right:8px;z-index:2;font-size:12px;font-weight:500;color:var(--vp-code-lang-color);transition:color .4s,opacity .4s}.vp-doc [class*=language-]:hover>button.copy+span.lang,.vp-doc [class*=language-]>button.copy:focus+span.lang{opacity:0}.vp-doc .VPTeamMembers{margin-top:24px}.vp-doc .VPTeamMembers.small.count-1 .container{margin:0!important;max-width:calc((100% - 24px)/2)!important}.vp-doc .VPTeamMembers.small.count-2 .container,.vp-doc .VPTeamMembers.small.count-3 .container{max-width:100%!important}.vp-doc .VPTeamMembers.medium.count-1 .container{margin:0!important;max-width:calc((100% - 24px)/2)!important}:is(.vp-external-link-icon,.vp-doc a[href*="://"],.vp-doc a[target=_blank]):not(.no-icon):after{display:inline-block;margin-top:-1px;margin-left:4px;width:11px;height:11px;background:currentColor;color:var(--vp-c-text-3);flex-shrink:0;--icon: url("data:image/svg+xml, %3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' %3E%3Cpath d='M0 0h24v24H0V0z' fill='none' /%3E%3Cpath d='M9 5v2h6.59L4 18.59 5.41 20 17 8.41V15h2V5H9z' /%3E%3C/svg%3E");-webkit-mask-image:var(--icon);mask-image:var(--icon)}.vp-external-link-icon:after{content:""}.external-link-icon-enabled :is(.vp-doc a[href*="://"],.vp-doc a[target=_blank]):after{content:"";color:currentColor}.vp-sponsor{border-radius:16px;overflow:hidden}.vp-sponsor.aside{border-radius:12px}.vp-sponsor-section+.vp-sponsor-section{margin-top:4px}.vp-sponsor-tier{margin:0 0 4px!important;text-align:center;letter-spacing:1px!important;line-height:24px;width:100%;font-weight:600;color:var(--vp-c-text-2);background-color:var(--vp-c-bg-soft)}.vp-sponsor.normal .vp-sponsor-tier{padding:13px 0 11px;font-size:14px}.vp-sponsor.aside .vp-sponsor-tier{padding:9px 0 7px;font-size:12px}.vp-sponsor-grid+.vp-sponsor-tier{margin-top:4px}.vp-sponsor-grid{display:flex;flex-wrap:wrap;gap:4px}.vp-sponsor-grid.xmini .vp-sponsor-grid-link{height:64px}.vp-sponsor-grid.xmini .vp-sponsor-grid-image{max-width:64px;max-height:22px}.vp-sponsor-grid.mini .vp-sponsor-grid-link{height:72px}.vp-sponsor-grid.mini .vp-sponsor-grid-image{max-width:96px;max-height:24px}.vp-sponsor-grid.small .vp-sponsor-grid-link{height:96px}.vp-sponsor-grid.small .vp-sponsor-grid-image{max-width:96px;max-height:24px}.vp-sponsor-grid.medium .vp-sponsor-grid-link{height:112px}.vp-sponsor-grid.medium .vp-sponsor-grid-image{max-width:120px;max-height:36px}.vp-sponsor-grid.big .vp-sponsor-grid-link{height:184px}.vp-sponsor-grid.big .vp-sponsor-grid-image{max-width:192px;max-height:56px}.vp-sponsor-grid[data-vp-grid="2"] .vp-sponsor-grid-item{width:calc((100% - 4px)/2)}.vp-sponsor-grid[data-vp-grid="3"] .vp-sponsor-grid-item{width:calc((100% - 4px * 2) / 3)}.vp-sponsor-grid[data-vp-grid="4"] .vp-sponsor-grid-item{width:calc((100% - 12px)/4)}.vp-sponsor-grid[data-vp-grid="5"] .vp-sponsor-grid-item{width:calc((100% - 16px)/5)}.vp-sponsor-grid[data-vp-grid="6"] .vp-sponsor-grid-item{width:calc((100% - 4px * 5) / 6)}.vp-sponsor-grid-item{flex-shrink:0;width:100%;background-color:var(--vp-c-bg-soft);transition:background-color .25s}.vp-sponsor-grid-item:hover{background-color:var(--vp-c-default-soft)}.vp-sponsor-grid-item:hover .vp-sponsor-grid-image{filter:grayscale(0) invert(0)}.vp-sponsor-grid-item.empty:hover{background-color:var(--vp-c-bg-soft)}.dark .vp-sponsor-grid-item:hover{background-color:var(--vp-c-white)}.dark .vp-sponsor-grid-item.empty:hover{background-color:var(--vp-c-bg-soft)}.vp-sponsor-grid-link{display:flex}.vp-sponsor-grid-box{display:flex;justify-content:center;align-items:center;width:100%}.vp-sponsor-grid-image{max-width:100%;filter:grayscale(1);transition:filter .25s}.dark .vp-sponsor-grid-image{filter:grayscale(1) invert(1)}.VPBadge{display:inline-block;margin-left:2px;border:1px solid transparent;border-radius:12px;padding:0 10px;line-height:22px;font-size:12px;font-weight:500;transform:translateY(-2px)}.VPBadge.small{padding:0 6px;line-height:18px;font-size:10px;transform:translateY(-8px)}.VPDocFooter .VPBadge{display:none}.vp-doc h1>.VPBadge{margin-top:4px;vertical-align:top}.vp-doc h2>.VPBadge{margin-top:3px;padding:0 8px;vertical-align:top}.vp-doc h3>.VPBadge{vertical-align:middle}.vp-doc h4>.VPBadge,.vp-doc h5>.VPBadge,.vp-doc h6>.VPBadge{vertical-align:middle;line-height:18px}.VPBadge.info{border-color:var(--vp-badge-info-border);color:var(--vp-badge-info-text);background-color:var(--vp-badge-info-bg)}.VPBadge.tip{border-color:var(--vp-badge-tip-border);color:var(--vp-badge-tip-text);background-color:var(--vp-badge-tip-bg)}.VPBadge.warning{border-color:var(--vp-badge-warning-border);color:var(--vp-badge-warning-text);background-color:var(--vp-badge-warning-bg)}.VPBadge.danger{border-color:var(--vp-badge-danger-border);color:var(--vp-badge-danger-text);background-color:var(--vp-badge-danger-bg)}.VPBackdrop[data-v-c79a1216]{position:fixed;top:0;right:0;bottom:0;left:0;z-index:var(--vp-z-index-backdrop);background:var(--vp-backdrop-bg-color);transition:opacity .5s}.VPBackdrop.fade-enter-from[data-v-c79a1216],.VPBackdrop.fade-leave-to[data-v-c79a1216]{opacity:0}.VPBackdrop.fade-leave-active[data-v-c79a1216]{transition-duration:.25s}@media (min-width: 1280px){.VPBackdrop[data-v-c79a1216]{display:none}}.NotFound[data-v-d6be1790]{padding:64px 24px 96px;text-align:center}@media (min-width: 768px){.NotFound[data-v-d6be1790]{padding:96px 32px 168px}}.code[data-v-d6be1790]{line-height:64px;font-size:64px;font-weight:600}.title[data-v-d6be1790]{padding-top:12px;letter-spacing:2px;line-height:20px;font-size:20px;font-weight:700}.divider[data-v-d6be1790]{margin:24px auto 18px;width:64px;height:1px;background-color:var(--vp-c-divider)}.quote[data-v-d6be1790]{margin:0 auto;max-width:256px;font-size:14px;font-weight:500;color:var(--vp-c-text-2)}.action[data-v-d6be1790]{padding-top:20px}.link[data-v-d6be1790]{display:inline-block;border:1px solid var(--vp-c-brand-1);border-radius:16px;padding:3px 16px;font-size:14px;font-weight:500;color:var(--vp-c-brand-1);transition:border-color .25s,color .25s}.link[data-v-d6be1790]:hover{border-color:var(--vp-c-brand-2);color:var(--vp-c-brand-2)}.root[data-v-b933a997]{position:relative;z-index:1}.nested[data-v-b933a997]{padding-right:16px;padding-left:16px}.outline-link[data-v-b933a997]{display:block;line-height:32px;font-size:14px;font-weight:400;color:var(--vp-c-text-2);white-space:nowrap;overflow:hidden;text-overflow:ellipsis;transition:color .5s}.outline-link[data-v-b933a997]:hover,.outline-link.active[data-v-b933a997]{color:var(--vp-c-text-1);transition:color .25s}.outline-link.nested[data-v-b933a997]{padding-left:13px}.VPDocAsideOutline[data-v-a5bbad30]{display:none}.VPDocAsideOutline.has-outline[data-v-a5bbad30]{display:block}.content[data-v-a5bbad30]{position:relative;border-left:1px solid var(--vp-c-divider);padding-left:16px;font-size:13px;font-weight:500}.outline-marker[data-v-a5bbad30]{position:absolute;top:32px;left:-1px;z-index:0;opacity:0;width:2px;border-radius:2px;height:18px;background-color:var(--vp-c-brand-1);transition:top .25s cubic-bezier(0,1,.5,1),background-color .5s,opacity .25s}.outline-title[data-v-a5bbad30]{line-height:32px;font-size:14px;font-weight:600}.VPDocAside[data-v-3f215769]{display:flex;flex-direction:column;flex-grow:1}.spacer[data-v-3f215769]{flex-grow:1}.VPDocAside[data-v-3f215769] .spacer+.VPDocAsideSponsors,.VPDocAside[data-v-3f215769] .spacer+.VPDocAsideCarbonAds{margin-top:24px}.VPDocAside[data-v-3f215769] .VPDocAsideSponsors+.VPDocAsideCarbonAds{margin-top:16px}.VPLastUpdated[data-v-7e05ebdb]{line-height:24px;font-size:14px;font-weight:500;color:var(--vp-c-text-2)}@media (min-width: 640px){.VPLastUpdated[data-v-7e05ebdb]{line-height:32px;font-size:14px;font-weight:500}}.VPDocFooter[data-v-d4a0bba5]{margin-top:64px}.edit-info[data-v-d4a0bba5]{padding-bottom:18px}@media (min-width: 640px){.edit-info[data-v-d4a0bba5]{display:flex;justify-content:space-between;align-items:center;padding-bottom:14px}}.edit-link-button[data-v-d4a0bba5]{display:flex;align-items:center;border:0;line-height:32px;font-size:14px;font-weight:500;color:var(--vp-c-brand-1);transition:color .25s}.edit-link-button[data-v-d4a0bba5]:hover{color:var(--vp-c-brand-2)}.edit-link-icon[data-v-d4a0bba5]{margin-right:8px}.prev-next[data-v-d4a0bba5]{border-top:1px solid var(--vp-c-divider);padding-top:24px;display:grid;grid-row-gap:8px}@media (min-width: 640px){.prev-next[data-v-d4a0bba5]{grid-template-columns:repeat(2,1fr);grid-column-gap:16px}}.pager-link[data-v-d4a0bba5]{display:block;border:1px solid var(--vp-c-divider);border-radius:8px;padding:11px 16px 13px;width:100%;height:100%;transition:border-color .25s}.pager-link[data-v-d4a0bba5]:hover{border-color:var(--vp-c-brand-1)}.pager-link.next[data-v-d4a0bba5]{margin-left:auto;text-align:right}.desc[data-v-d4a0bba5]{display:block;line-height:20px;font-size:12px;font-weight:500;color:var(--vp-c-text-2)}.title[data-v-d4a0bba5]{display:block;line-height:20px;font-size:14px;font-weight:500;color:var(--vp-c-brand-1);transition:color .25s}.VPDoc[data-v-39a288b8]{padding:32px 24px 96px;width:100%}@media (min-width: 768px){.VPDoc[data-v-39a288b8]{padding:48px 32px 128px}}@media (min-width: 960px){.VPDoc[data-v-39a288b8]{padding:48px 32px 0}.VPDoc:not(.has-sidebar) .container[data-v-39a288b8]{display:flex;justify-content:center;max-width:992px}.VPDoc:not(.has-sidebar) .content[data-v-39a288b8]{max-width:752px}}@media (min-width: 1280px){.VPDoc .container[data-v-39a288b8]{display:flex;justify-content:center}.VPDoc .aside[data-v-39a288b8]{display:block}}@media (min-width: 1440px){.VPDoc:not(.has-sidebar) .content[data-v-39a288b8]{max-width:784px}.VPDoc:not(.has-sidebar) .container[data-v-39a288b8]{max-width:1104px}}.container[data-v-39a288b8]{margin:0 auto;width:100%}.aside[data-v-39a288b8]{position:relative;display:none;order:2;flex-grow:1;padding-left:32px;width:100%;max-width:256px}.left-aside[data-v-39a288b8]{order:1;padding-left:unset;padding-right:32px}.aside-container[data-v-39a288b8]{position:fixed;top:0;padding-top:calc(var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + var(--vp-doc-top-height, 0px) + 48px);width:224px;height:100vh;overflow-x:hidden;overflow-y:auto;scrollbar-width:none}.aside-container[data-v-39a288b8]::-webkit-scrollbar{display:none}.aside-curtain[data-v-39a288b8]{position:fixed;bottom:0;z-index:10;width:224px;height:32px;background:linear-gradient(transparent,var(--vp-c-bg) 70%)}.aside-content[data-v-39a288b8]{display:flex;flex-direction:column;min-height:calc(100vh - (var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + 48px));padding-bottom:32px}.content[data-v-39a288b8]{position:relative;margin:0 auto;width:100%}@media (min-width: 960px){.content[data-v-39a288b8]{padding:0 32px 128px}}@media (min-width: 1280px){.content[data-v-39a288b8]{order:1;margin:0;min-width:640px}}.content-container[data-v-39a288b8]{margin:0 auto}.VPDoc.has-aside .content-container[data-v-39a288b8]{max-width:688px}.VPButton[data-v-cad61b99]{display:inline-block;border:1px solid transparent;text-align:center;font-weight:600;white-space:nowrap;transition:color .25s,border-color .25s,background-color .25s}.VPButton[data-v-cad61b99]:active{transition:color .1s,border-color .1s,background-color .1s}.VPButton.medium[data-v-cad61b99]{border-radius:20px;padding:0 20px;line-height:38px;font-size:14px}.VPButton.big[data-v-cad61b99]{border-radius:24px;padding:0 24px;line-height:46px;font-size:16px}.VPButton.brand[data-v-cad61b99]{border-color:var(--vp-button-brand-border);color:var(--vp-button-brand-text);background-color:var(--vp-button-brand-bg)}.VPButton.brand[data-v-cad61b99]:hover{border-color:var(--vp-button-brand-hover-border);color:var(--vp-button-brand-hover-text);background-color:var(--vp-button-brand-hover-bg)}.VPButton.brand[data-v-cad61b99]:active{border-color:var(--vp-button-brand-active-border);color:var(--vp-button-brand-active-text);background-color:var(--vp-button-brand-active-bg)}.VPButton.alt[data-v-cad61b99]{border-color:var(--vp-button-alt-border);color:var(--vp-button-alt-text);background-color:var(--vp-button-alt-bg)}.VPButton.alt[data-v-cad61b99]:hover{border-color:var(--vp-button-alt-hover-border);color:var(--vp-button-alt-hover-text);background-color:var(--vp-button-alt-hover-bg)}.VPButton.alt[data-v-cad61b99]:active{border-color:var(--vp-button-alt-active-border);color:var(--vp-button-alt-active-text);background-color:var(--vp-button-alt-active-bg)}.VPButton.sponsor[data-v-cad61b99]{border-color:var(--vp-button-sponsor-border);color:var(--vp-button-sponsor-text);background-color:var(--vp-button-sponsor-bg)}.VPButton.sponsor[data-v-cad61b99]:hover{border-color:var(--vp-button-sponsor-hover-border);color:var(--vp-button-sponsor-hover-text);background-color:var(--vp-button-sponsor-hover-bg)}.VPButton.sponsor[data-v-cad61b99]:active{border-color:var(--vp-button-sponsor-active-border);color:var(--vp-button-sponsor-active-text);background-color:var(--vp-button-sponsor-active-bg)}html:not(.dark) .VPImage.dark[data-v-8426fc1a]{display:none}.dark .VPImage.light[data-v-8426fc1a]{display:none}.VPHero[data-v-303bb580]{margin-top:calc((var(--vp-nav-height) + var(--vp-layout-top-height, 0px)) * -1);padding:calc(var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + 48px) 24px 48px}@media (min-width: 640px){.VPHero[data-v-303bb580]{padding:calc(var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + 80px) 48px 64px}}@media (min-width: 960px){.VPHero[data-v-303bb580]{padding:calc(var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + 80px) 64px 64px}}.container[data-v-303bb580]{display:flex;flex-direction:column;margin:0 auto;max-width:1152px}@media (min-width: 960px){.container[data-v-303bb580]{flex-direction:row}}.main[data-v-303bb580]{position:relative;z-index:10;order:2;flex-grow:1;flex-shrink:0}.VPHero.has-image .container[data-v-303bb580]{text-align:center}@media (min-width: 960px){.VPHero.has-image .container[data-v-303bb580]{text-align:left}}@media (min-width: 960px){.main[data-v-303bb580]{order:1;width:calc((100% / 3) * 2)}.VPHero.has-image .main[data-v-303bb580]{max-width:592px}}.name[data-v-303bb580],.text[data-v-303bb580]{max-width:392px;letter-spacing:-.4px;line-height:40px;font-size:32px;font-weight:700;white-space:pre-wrap}.VPHero.has-image .name[data-v-303bb580],.VPHero.has-image .text[data-v-303bb580]{margin:0 auto}.name[data-v-303bb580]{color:var(--vp-home-hero-name-color)}.clip[data-v-303bb580]{background:var(--vp-home-hero-name-background);-webkit-background-clip:text;background-clip:text;-webkit-text-fill-color:var(--vp-home-hero-name-color)}@media (min-width: 640px){.name[data-v-303bb580],.text[data-v-303bb580]{max-width:576px;line-height:56px;font-size:48px}}@media (min-width: 960px){.name[data-v-303bb580],.text[data-v-303bb580]{line-height:64px;font-size:56px}.VPHero.has-image .name[data-v-303bb580],.VPHero.has-image .text[data-v-303bb580]{margin:0}}.tagline[data-v-303bb580]{padding-top:8px;max-width:392px;line-height:28px;font-size:18px;font-weight:500;white-space:pre-wrap;color:var(--vp-c-text-2)}.VPHero.has-image .tagline[data-v-303bb580]{margin:0 auto}@media (min-width: 640px){.tagline[data-v-303bb580]{padding-top:12px;max-width:576px;line-height:32px;font-size:20px}}@media (min-width: 960px){.tagline[data-v-303bb580]{line-height:36px;font-size:24px}.VPHero.has-image .tagline[data-v-303bb580]{margin:0}}.actions[data-v-303bb580]{display:flex;flex-wrap:wrap;margin:-6px;padding-top:24px}.VPHero.has-image .actions[data-v-303bb580]{justify-content:center}@media (min-width: 640px){.actions[data-v-303bb580]{padding-top:32px}}@media (min-width: 960px){.VPHero.has-image .actions[data-v-303bb580]{justify-content:flex-start}}.action[data-v-303bb580]{flex-shrink:0;padding:6px}.image[data-v-303bb580]{order:1;margin:-76px -24px -48px}@media (min-width: 640px){.image[data-v-303bb580]{margin:-108px -24px -48px}}@media (min-width: 960px){.image[data-v-303bb580]{flex-grow:1;order:2;margin:0;min-height:100%}}.image-container[data-v-303bb580]{position:relative;margin:0 auto;width:320px;height:320px}@media (min-width: 640px){.image-container[data-v-303bb580]{width:392px;height:392px}}@media (min-width: 960px){.image-container[data-v-303bb580]{display:flex;justify-content:center;align-items:center;width:100%;height:100%;transform:translate(-32px,-32px)}}.image-bg[data-v-303bb580]{position:absolute;top:50%;left:50%;border-radius:50%;width:192px;height:192px;background-image:var(--vp-home-hero-image-background-image);filter:var(--vp-home-hero-image-filter);transform:translate(-50%,-50%)}@media (min-width: 640px){.image-bg[data-v-303bb580]{width:256px;height:256px}}@media (min-width: 960px){.image-bg[data-v-303bb580]{width:320px;height:320px}}[data-v-303bb580] .image-src{position:absolute;top:50%;left:50%;max-width:192px;max-height:192px;transform:translate(-50%,-50%)}@media (min-width: 640px){[data-v-303bb580] .image-src{max-width:256px;max-height:256px}}@media (min-width: 960px){[data-v-303bb580] .image-src{max-width:320px;max-height:320px}}.VPFeature[data-v-a3976bdc]{display:block;border:1px solid var(--vp-c-bg-soft);border-radius:12px;height:100%;background-color:var(--vp-c-bg-soft);transition:border-color .25s,background-color .25s}.VPFeature.link[data-v-a3976bdc]:hover{border-color:var(--vp-c-brand-1)}.box[data-v-a3976bdc]{display:flex;flex-direction:column;padding:24px;height:100%}.box[data-v-a3976bdc]>.VPImage{margin-bottom:20px}.icon[data-v-a3976bdc]{display:flex;justify-content:center;align-items:center;margin-bottom:20px;border-radius:6px;background-color:var(--vp-c-default-soft);width:48px;height:48px;font-size:24px;transition:background-color .25s}.title[data-v-a3976bdc]{line-height:24px;font-size:16px;font-weight:600}.details[data-v-a3976bdc]{flex-grow:1;padding-top:8px;line-height:24px;font-size:14px;font-weight:500;color:var(--vp-c-text-2)}.link-text[data-v-a3976bdc]{padding-top:8px}.link-text-value[data-v-a3976bdc]{display:flex;align-items:center;font-size:14px;font-weight:500;color:var(--vp-c-brand-1)}.link-text-icon[data-v-a3976bdc]{margin-left:6px}.VPFeatures[data-v-a6181336]{position:relative;padding:0 24px}@media (min-width: 640px){.VPFeatures[data-v-a6181336]{padding:0 48px}}@media (min-width: 960px){.VPFeatures[data-v-a6181336]{padding:0 64px}}.container[data-v-a6181336]{margin:0 auto;max-width:1152px}.items[data-v-a6181336]{display:flex;flex-wrap:wrap;margin:-8px}.item[data-v-a6181336]{padding:8px;width:100%}@media (min-width: 640px){.item.grid-2[data-v-a6181336],.item.grid-4[data-v-a6181336],.item.grid-6[data-v-a6181336]{width:50%}}@media (min-width: 768px){.item.grid-2[data-v-a6181336],.item.grid-4[data-v-a6181336]{width:50%}.item.grid-3[data-v-a6181336],.item.grid-6[data-v-a6181336]{width:calc(100% / 3)}}@media (min-width: 960px){.item.grid-4[data-v-a6181336]{width:25%}}.container[data-v-8e2d4988]{margin:auto;width:100%;max-width:1280px;padding:0 24px}@media (min-width: 640px){.container[data-v-8e2d4988]{padding:0 48px}}@media (min-width: 960px){.container[data-v-8e2d4988]{width:100%;padding:0 64px}}.vp-doc[data-v-8e2d4988] .VPHomeSponsors,.vp-doc[data-v-8e2d4988] .VPTeamPage{margin-left:var(--vp-offset, calc(50% - 50vw) );margin-right:var(--vp-offset, calc(50% - 50vw) )}.vp-doc[data-v-8e2d4988] .VPHomeSponsors h2{border-top:none;letter-spacing:normal}.vp-doc[data-v-8e2d4988] .VPHomeSponsors a,.vp-doc[data-v-8e2d4988] .VPTeamPage a{text-decoration:none}.VPHome[data-v-686f80a6]{margin-bottom:96px}@media (min-width: 768px){.VPHome[data-v-686f80a6]{margin-bottom:128px}}.VPContent[data-v-1428d186]{flex-grow:1;flex-shrink:0;margin:var(--vp-layout-top-height, 0px) auto 0;width:100%}.VPContent.is-home[data-v-1428d186]{width:100%;max-width:100%}.VPContent.has-sidebar[data-v-1428d186]{margin:0}@media (min-width: 960px){.VPContent[data-v-1428d186]{padding-top:var(--vp-nav-height)}.VPContent.has-sidebar[data-v-1428d186]{margin:var(--vp-layout-top-height, 0px) 0 0;padding-left:var(--vp-sidebar-width)}}@media (min-width: 1440px){.VPContent.has-sidebar[data-v-1428d186]{padding-right:calc((100vw - var(--vp-layout-max-width)) / 2);padding-left:calc((100vw - var(--vp-layout-max-width)) / 2 + var(--vp-sidebar-width))}}.VPFooter[data-v-e315a0ad]{position:relative;z-index:var(--vp-z-index-footer);border-top:1px solid var(--vp-c-gutter);padding:32px 24px;background-color:var(--vp-c-bg)}.VPFooter.has-sidebar[data-v-e315a0ad]{display:none}.VPFooter[data-v-e315a0ad] a{text-decoration-line:underline;text-underline-offset:2px;transition:color .25s}.VPFooter[data-v-e315a0ad] a:hover{color:var(--vp-c-text-1)}@media (min-width: 768px){.VPFooter[data-v-e315a0ad]{padding:32px}}.container[data-v-e315a0ad]{margin:0 auto;max-width:var(--vp-layout-max-width);text-align:center}.message[data-v-e315a0ad],.copyright[data-v-e315a0ad]{line-height:24px;font-size:14px;font-weight:500;color:var(--vp-c-text-2)}.VPLocalNavOutlineDropdown[data-v-17a5e62e]{padding:12px 20px 11px}@media (min-width: 960px){.VPLocalNavOutlineDropdown[data-v-17a5e62e]{padding:12px 36px 11px}}.VPLocalNavOutlineDropdown button[data-v-17a5e62e]{display:block;font-size:12px;font-weight:500;line-height:24px;color:var(--vp-c-text-2);transition:color .5s;position:relative}.VPLocalNavOutlineDropdown button[data-v-17a5e62e]:hover{color:var(--vp-c-text-1);transition:color .25s}.VPLocalNavOutlineDropdown button.open[data-v-17a5e62e]{color:var(--vp-c-text-1)}.icon[data-v-17a5e62e]{display:inline-block;vertical-align:middle;margin-left:2px;font-size:14px;transform:rotate(0);transition:transform .25s}@media (min-width: 960px){.VPLocalNavOutlineDropdown button[data-v-17a5e62e]{font-size:14px}.icon[data-v-17a5e62e]{font-size:16px}}.open>.icon[data-v-17a5e62e]{transform:rotate(90deg)}.items[data-v-17a5e62e]{position:absolute;top:40px;right:16px;left:16px;display:grid;gap:1px;border:1px solid var(--vp-c-border);border-radius:8px;background-color:var(--vp-c-gutter);max-height:calc(var(--vp-vh, 100vh) - 86px);overflow:hidden auto;box-shadow:var(--vp-shadow-3)}@media (min-width: 960px){.items[data-v-17a5e62e]{right:auto;left:calc(var(--vp-sidebar-width) + 32px);width:320px}}.header[data-v-17a5e62e]{background-color:var(--vp-c-bg-soft)}.top-link[data-v-17a5e62e]{display:block;padding:0 16px;line-height:48px;font-size:14px;font-weight:500;color:var(--vp-c-brand-1)}.outline[data-v-17a5e62e]{padding:8px 0;background-color:var(--vp-c-bg-soft)}.flyout-enter-active[data-v-17a5e62e]{transition:all .2s ease-out}.flyout-leave-active[data-v-17a5e62e]{transition:all .15s ease-in}.flyout-enter-from[data-v-17a5e62e],.flyout-leave-to[data-v-17a5e62e]{opacity:0;transform:translateY(-16px)}.VPLocalNav[data-v-a6f0e41e]{position:sticky;top:0;left:0;z-index:var(--vp-z-index-local-nav);border-bottom:1px solid var(--vp-c-gutter);padding-top:var(--vp-layout-top-height, 0px);width:100%;background-color:var(--vp-local-nav-bg-color)}.VPLocalNav.fixed[data-v-a6f0e41e]{position:fixed}@media (min-width: 960px){.VPLocalNav[data-v-a6f0e41e]{top:var(--vp-nav-height)}.VPLocalNav.has-sidebar[data-v-a6f0e41e]{padding-left:var(--vp-sidebar-width)}.VPLocalNav.empty[data-v-a6f0e41e]{display:none}}@media (min-width: 1280px){.VPLocalNav[data-v-a6f0e41e]{display:none}}@media (min-width: 1440px){.VPLocalNav.has-sidebar[data-v-a6f0e41e]{padding-left:calc((100vw - var(--vp-layout-max-width)) / 2 + var(--vp-sidebar-width))}}.container[data-v-a6f0e41e]{display:flex;justify-content:space-between;align-items:center}.menu[data-v-a6f0e41e]{display:flex;align-items:center;padding:12px 24px 11px;line-height:24px;font-size:12px;font-weight:500;color:var(--vp-c-text-2);transition:color .5s}.menu[data-v-a6f0e41e]:hover{color:var(--vp-c-text-1);transition:color .25s}@media (min-width: 768px){.menu[data-v-a6f0e41e]{padding:0 32px}}@media (min-width: 960px){.menu[data-v-a6f0e41e]{display:none}}.menu-icon[data-v-a6f0e41e]{margin-right:8px;font-size:14px}.VPOutlineDropdown[data-v-a6f0e41e]{padding:12px 24px 11px}@media (min-width: 768px){.VPOutlineDropdown[data-v-a6f0e41e]{padding:12px 32px 11px}}.VPSwitch[data-v-1d5665e3]{position:relative;border-radius:11px;display:block;width:40px;height:22px;flex-shrink:0;border:1px solid var(--vp-input-border-color);background-color:var(--vp-input-switch-bg-color);transition:border-color .25s!important}.VPSwitch[data-v-1d5665e3]:hover{border-color:var(--vp-c-brand-1)}.check[data-v-1d5665e3]{position:absolute;top:1px;left:1px;width:18px;height:18px;border-radius:50%;background-color:var(--vp-c-neutral-inverse);box-shadow:var(--vp-shadow-1);transition:transform .25s!important}.icon[data-v-1d5665e3]{position:relative;display:block;width:18px;height:18px;border-radius:50%;overflow:hidden}.icon[data-v-1d5665e3] [class^=vpi-]{position:absolute;top:3px;left:3px;width:12px;height:12px;color:var(--vp-c-text-2)}.dark .icon[data-v-1d5665e3] [class^=vpi-]{color:var(--vp-c-text-1);transition:opacity .25s!important}.sun[data-v-d1f28634]{opacity:1}.moon[data-v-d1f28634],.dark .sun[data-v-d1f28634]{opacity:0}.dark .moon[data-v-d1f28634]{opacity:1}.dark .VPSwitchAppearance[data-v-d1f28634] .check{transform:translate(18px)}.VPNavBarAppearance[data-v-e6aabb21]{display:none}@media (min-width: 1280px){.VPNavBarAppearance[data-v-e6aabb21]{display:flex;align-items:center}}.VPMenuGroup+.VPMenuLink[data-v-43f1e123]{margin:12px -12px 0;border-top:1px solid var(--vp-c-divider);padding:12px 12px 0}.link[data-v-43f1e123]{display:block;border-radius:6px;padding:0 12px;line-height:32px;font-size:14px;font-weight:500;color:var(--vp-c-text-1);white-space:nowrap;transition:background-color .25s,color .25s}.link[data-v-43f1e123]:hover{color:var(--vp-c-brand-1);background-color:var(--vp-c-default-soft)}.link.active[data-v-43f1e123]{color:var(--vp-c-brand-1)}.VPMenuGroup[data-v-69e747b5]{margin:12px -12px 0;border-top:1px solid var(--vp-c-divider);padding:12px 12px 0}.VPMenuGroup[data-v-69e747b5]:first-child{margin-top:0;border-top:0;padding-top:0}.VPMenuGroup+.VPMenuGroup[data-v-69e747b5]{margin-top:12px;border-top:1px solid var(--vp-c-divider)}.title[data-v-69e747b5]{padding:0 12px;line-height:32px;font-size:14px;font-weight:600;color:var(--vp-c-text-2);white-space:nowrap;transition:color .25s}.VPMenu[data-v-e7ea1737]{border-radius:12px;padding:12px;min-width:128px;border:1px solid var(--vp-c-divider);background-color:var(--vp-c-bg-elv);box-shadow:var(--vp-shadow-3);transition:background-color .5s;max-height:calc(100vh - var(--vp-nav-height));overflow-y:auto}.VPMenu[data-v-e7ea1737] .group{margin:0 -12px;padding:0 12px 12px}.VPMenu[data-v-e7ea1737] .group+.group{border-top:1px solid var(--vp-c-divider);padding:11px 12px 12px}.VPMenu[data-v-e7ea1737] .group:last-child{padding-bottom:0}.VPMenu[data-v-e7ea1737] .group+.item{border-top:1px solid var(--vp-c-divider);padding:11px 16px 0}.VPMenu[data-v-e7ea1737] .item{padding:0 16px;white-space:nowrap}.VPMenu[data-v-e7ea1737] .label{flex-grow:1;line-height:28px;font-size:12px;font-weight:500;color:var(--vp-c-text-2);transition:color .5s}.VPMenu[data-v-e7ea1737] .action{padding-left:24px}.VPFlyout[data-v-b6c34ac9]{position:relative}.VPFlyout[data-v-b6c34ac9]:hover{color:var(--vp-c-brand-1);transition:color .25s}.VPFlyout:hover .text[data-v-b6c34ac9]{color:var(--vp-c-text-2)}.VPFlyout:hover .icon[data-v-b6c34ac9]{fill:var(--vp-c-text-2)}.VPFlyout.active .text[data-v-b6c34ac9]{color:var(--vp-c-brand-1)}.VPFlyout.active:hover .text[data-v-b6c34ac9]{color:var(--vp-c-brand-2)}.VPFlyout:hover .menu[data-v-b6c34ac9],.button[aria-expanded=true]+.menu[data-v-b6c34ac9]{opacity:1;visibility:visible;transform:translateY(0)}.button[aria-expanded=false]+.menu[data-v-b6c34ac9]{opacity:0;visibility:hidden;transform:translateY(0)}.button[data-v-b6c34ac9]{display:flex;align-items:center;padding:0 12px;height:var(--vp-nav-height);color:var(--vp-c-text-1);transition:color .5s}.text[data-v-b6c34ac9]{display:flex;align-items:center;line-height:var(--vp-nav-height);font-size:14px;font-weight:500;color:var(--vp-c-text-1);transition:color .25s}.option-icon[data-v-b6c34ac9]{margin-right:0;font-size:16px}.text-icon[data-v-b6c34ac9]{margin-left:4px;font-size:14px}.icon[data-v-b6c34ac9]{font-size:20px;transition:fill .25s}.menu[data-v-b6c34ac9]{position:absolute;top:calc(var(--vp-nav-height) / 2 + 20px);right:0;opacity:0;visibility:hidden;transition:opacity .25s,visibility .25s,transform .25s}.VPSocialLink[data-v-eee4e7cb]{display:flex;justify-content:center;align-items:center;width:36px;height:36px;color:var(--vp-c-text-2);transition:color .5s}.VPSocialLink[data-v-eee4e7cb]:hover{color:var(--vp-c-text-1);transition:color .25s}.VPSocialLink[data-v-eee4e7cb]>svg,.VPSocialLink[data-v-eee4e7cb]>[class^=vpi-social-]{width:20px;height:20px;fill:currentColor}.VPSocialLinks[data-v-7bc22406]{display:flex;justify-content:center}.VPNavBarExtra[data-v-d0bd9dde]{display:none;margin-right:-12px}@media (min-width: 768px){.VPNavBarExtra[data-v-d0bd9dde]{display:block}}@media (min-width: 1280px){.VPNavBarExtra[data-v-d0bd9dde]{display:none}}.trans-title[data-v-d0bd9dde]{padding:0 24px 0 12px;line-height:32px;font-size:14px;font-weight:700;color:var(--vp-c-text-1)}.item.appearance[data-v-d0bd9dde],.item.social-links[data-v-d0bd9dde]{display:flex;align-items:center;padding:0 12px}.item.appearance[data-v-d0bd9dde]{min-width:176px}.appearance-action[data-v-d0bd9dde]{margin-right:-2px}.social-links-list[data-v-d0bd9dde]{margin:-4px -8px}.VPNavBarHamburger[data-v-e5dd9c1c]{display:flex;justify-content:center;align-items:center;width:48px;height:var(--vp-nav-height)}@media (min-width: 768px){.VPNavBarHamburger[data-v-e5dd9c1c]{display:none}}.container[data-v-e5dd9c1c]{position:relative;width:16px;height:14px;overflow:hidden}.VPNavBarHamburger:hover .top[data-v-e5dd9c1c]{top:0;left:0;transform:translate(4px)}.VPNavBarHamburger:hover .middle[data-v-e5dd9c1c]{top:6px;left:0;transform:translate(0)}.VPNavBarHamburger:hover .bottom[data-v-e5dd9c1c]{top:12px;left:0;transform:translate(8px)}.VPNavBarHamburger.active .top[data-v-e5dd9c1c]{top:6px;transform:translate(0) rotate(225deg)}.VPNavBarHamburger.active .middle[data-v-e5dd9c1c]{top:6px;transform:translate(16px)}.VPNavBarHamburger.active .bottom[data-v-e5dd9c1c]{top:6px;transform:translate(0) rotate(135deg)}.VPNavBarHamburger.active:hover .top[data-v-e5dd9c1c],.VPNavBarHamburger.active:hover .middle[data-v-e5dd9c1c],.VPNavBarHamburger.active:hover .bottom[data-v-e5dd9c1c]{background-color:var(--vp-c-text-2);transition:top .25s,background-color .25s,transform .25s}.top[data-v-e5dd9c1c],.middle[data-v-e5dd9c1c],.bottom[data-v-e5dd9c1c]{position:absolute;width:16px;height:2px;background-color:var(--vp-c-text-1);transition:top .25s,background-color .5s,transform .25s}.top[data-v-e5dd9c1c]{top:0;left:0;transform:translate(0)}.middle[data-v-e5dd9c1c]{top:6px;left:0;transform:translate(8px)}.bottom[data-v-e5dd9c1c]{top:12px;left:0;transform:translate(4px)}.VPNavBarMenuLink[data-v-9c663999]{display:flex;align-items:center;padding:0 12px;line-height:var(--vp-nav-height);font-size:14px;font-weight:500;color:var(--vp-c-text-1);transition:color .25s}.VPNavBarMenuLink.active[data-v-9c663999],.VPNavBarMenuLink[data-v-9c663999]:hover{color:var(--vp-c-brand-1)}.VPNavBarMenu[data-v-7f418b0f]{display:none}@media (min-width: 768px){.VPNavBarMenu[data-v-7f418b0f]{display:flex}}/*! @docsearch/css 3.6.0 | MIT License | © Algolia, Inc. and contributors | https://docsearch.algolia.com */:root{--docsearch-primary-color:#5468ff;--docsearch-text-color:#1c1e21;--docsearch-spacing:12px;--docsearch-icon-stroke-width:1.4;--docsearch-highlight-color:var(--docsearch-primary-color);--docsearch-muted-color:#969faf;--docsearch-container-background:rgba(101,108,133,.8);--docsearch-logo-color:#5468ff;--docsearch-modal-width:560px;--docsearch-modal-height:600px;--docsearch-modal-background:#f5f6f7;--docsearch-modal-shadow:inset 1px 1px 0 0 hsla(0,0%,100%,.5),0 3px 8px 0 #555a64;--docsearch-searchbox-height:56px;--docsearch-searchbox-background:#ebedf0;--docsearch-searchbox-focus-background:#fff;--docsearch-searchbox-shadow:inset 0 0 0 2px var(--docsearch-primary-color);--docsearch-hit-height:56px;--docsearch-hit-color:#444950;--docsearch-hit-active-color:#fff;--docsearch-hit-background:#fff;--docsearch-hit-shadow:0 1px 3px 0 #d4d9e1;--docsearch-key-gradient:linear-gradient(-225deg,#d5dbe4,#f8f8f8);--docsearch-key-shadow:inset 0 -2px 0 0 #cdcde6,inset 0 0 1px 1px #fff,0 1px 2px 1px rgba(30,35,90,.4);--docsearch-key-pressed-shadow:inset 0 -2px 0 0 #cdcde6,inset 0 0 1px 1px #fff,0 1px 1px 0 rgba(30,35,90,.4);--docsearch-footer-height:44px;--docsearch-footer-background:#fff;--docsearch-footer-shadow:0 -1px 0 0 #e0e3e8,0 -3px 6px 0 rgba(69,98,155,.12)}html[data-theme=dark]{--docsearch-text-color:#f5f6f7;--docsearch-container-background:rgba(9,10,17,.8);--docsearch-modal-background:#15172a;--docsearch-modal-shadow:inset 1px 1px 0 0 #2c2e40,0 3px 8px 0 #000309;--docsearch-searchbox-background:#090a11;--docsearch-searchbox-focus-background:#000;--docsearch-hit-color:#bec3c9;--docsearch-hit-shadow:none;--docsearch-hit-background:#090a11;--docsearch-key-gradient:linear-gradient(-26.5deg,#565872,#31355b);--docsearch-key-shadow:inset 0 -2px 0 0 #282d55,inset 0 0 1px 1px #51577d,0 2px 2px 0 rgba(3,4,9,.3);--docsearch-key-pressed-shadow:inset 0 -2px 0 0 #282d55,inset 0 0 1px 1px #51577d,0 1px 1px 0 rgba(3,4,9,.30196078431372547);--docsearch-footer-background:#1e2136;--docsearch-footer-shadow:inset 0 1px 0 0 rgba(73,76,106,.5),0 -4px 8px 0 rgba(0,0,0,.2);--docsearch-logo-color:#fff;--docsearch-muted-color:#7f8497}.DocSearch-Button{align-items:center;background:var(--docsearch-searchbox-background);border:0;border-radius:40px;color:var(--docsearch-muted-color);cursor:pointer;display:flex;font-weight:500;height:36px;justify-content:space-between;margin:0 0 0 16px;padding:0 8px;-webkit-user-select:none;user-select:none}.DocSearch-Button:active,.DocSearch-Button:focus,.DocSearch-Button:hover{background:var(--docsearch-searchbox-focus-background);box-shadow:var(--docsearch-searchbox-shadow);color:var(--docsearch-text-color);outline:none}.DocSearch-Button-Container{align-items:center;display:flex}.DocSearch-Search-Icon{stroke-width:1.6}.DocSearch-Button .DocSearch-Search-Icon{color:var(--docsearch-text-color)}.DocSearch-Button-Placeholder{font-size:1rem;padding:0 12px 0 6px}.DocSearch-Button-Keys{display:flex;min-width:calc(40px + .8em)}.DocSearch-Button-Key{align-items:center;background:var(--docsearch-key-gradient);border-radius:3px;box-shadow:var(--docsearch-key-shadow);color:var(--docsearch-muted-color);display:flex;height:18px;justify-content:center;margin-right:.4em;position:relative;padding:0 0 2px;border:0;top:-1px;width:20px}.DocSearch-Button-Key--pressed{transform:translate3d(0,1px,0);box-shadow:var(--docsearch-key-pressed-shadow)}@media (max-width:768px){.DocSearch-Button-Keys,.DocSearch-Button-Placeholder{display:none}}.DocSearch--active{overflow:hidden!important}.DocSearch-Container,.DocSearch-Container *{box-sizing:border-box}.DocSearch-Container{background-color:var(--docsearch-container-background);height:100vh;left:0;position:fixed;top:0;width:100vw;z-index:200}.DocSearch-Container a{text-decoration:none}.DocSearch-Link{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;color:var(--docsearch-highlight-color);cursor:pointer;font:inherit;margin:0;padding:0}.DocSearch-Modal{background:var(--docsearch-modal-background);border-radius:6px;box-shadow:var(--docsearch-modal-shadow);flex-direction:column;margin:60px auto auto;max-width:var(--docsearch-modal-width);position:relative}.DocSearch-SearchBar{display:flex;padding:var(--docsearch-spacing) var(--docsearch-spacing) 0}.DocSearch-Form{align-items:center;background:var(--docsearch-searchbox-focus-background);border-radius:4px;box-shadow:var(--docsearch-searchbox-shadow);display:flex;height:var(--docsearch-searchbox-height);margin:0;padding:0 var(--docsearch-spacing);position:relative;width:100%}.DocSearch-Input{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:transparent;border:0;color:var(--docsearch-text-color);flex:1;font:inherit;font-size:1.2em;height:100%;outline:none;padding:0 0 0 8px;width:80%}.DocSearch-Input::placeholder{color:var(--docsearch-muted-color);opacity:1}.DocSearch-Input::-webkit-search-cancel-button,.DocSearch-Input::-webkit-search-decoration,.DocSearch-Input::-webkit-search-results-button,.DocSearch-Input::-webkit-search-results-decoration{display:none}.DocSearch-LoadingIndicator,.DocSearch-MagnifierLabel,.DocSearch-Reset{margin:0;padding:0}.DocSearch-MagnifierLabel,.DocSearch-Reset{align-items:center;color:var(--docsearch-highlight-color);display:flex;justify-content:center}.DocSearch-Container--Stalled .DocSearch-MagnifierLabel,.DocSearch-LoadingIndicator{display:none}.DocSearch-Container--Stalled .DocSearch-LoadingIndicator{align-items:center;color:var(--docsearch-highlight-color);display:flex;justify-content:center}@media screen and (prefers-reduced-motion:reduce){.DocSearch-Reset{animation:none;-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;border-radius:50%;color:var(--docsearch-icon-color);cursor:pointer;right:0;stroke-width:var(--docsearch-icon-stroke-width)}}.DocSearch-Reset{animation:fade-in .1s ease-in forwards;-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;border-radius:50%;color:var(--docsearch-icon-color);cursor:pointer;padding:2px;right:0;stroke-width:var(--docsearch-icon-stroke-width)}.DocSearch-Reset[hidden]{display:none}.DocSearch-Reset:hover{color:var(--docsearch-highlight-color)}.DocSearch-LoadingIndicator svg,.DocSearch-MagnifierLabel svg{height:24px;width:24px}.DocSearch-Cancel{display:none}.DocSearch-Dropdown{max-height:calc(var(--docsearch-modal-height) - var(--docsearch-searchbox-height) - var(--docsearch-spacing) - var(--docsearch-footer-height));min-height:var(--docsearch-spacing);overflow-y:auto;overflow-y:overlay;padding:0 var(--docsearch-spacing);scrollbar-color:var(--docsearch-muted-color) var(--docsearch-modal-background);scrollbar-width:thin}.DocSearch-Dropdown::-webkit-scrollbar{width:12px}.DocSearch-Dropdown::-webkit-scrollbar-track{background:transparent}.DocSearch-Dropdown::-webkit-scrollbar-thumb{background-color:var(--docsearch-muted-color);border:3px solid var(--docsearch-modal-background);border-radius:20px}.DocSearch-Dropdown ul{list-style:none;margin:0;padding:0}.DocSearch-Label{font-size:.75em;line-height:1.6em}.DocSearch-Help,.DocSearch-Label{color:var(--docsearch-muted-color)}.DocSearch-Help{font-size:.9em;margin:0;-webkit-user-select:none;user-select:none}.DocSearch-Title{font-size:1.2em}.DocSearch-Logo a{display:flex}.DocSearch-Logo svg{color:var(--docsearch-logo-color);margin-left:8px}.DocSearch-Hits:last-of-type{margin-bottom:24px}.DocSearch-Hits mark{background:none;color:var(--docsearch-highlight-color)}.DocSearch-HitsFooter{color:var(--docsearch-muted-color);display:flex;font-size:.85em;justify-content:center;margin-bottom:var(--docsearch-spacing);padding:var(--docsearch-spacing)}.DocSearch-HitsFooter a{border-bottom:1px solid;color:inherit}.DocSearch-Hit{border-radius:4px;display:flex;padding-bottom:4px;position:relative}@media screen and (prefers-reduced-motion:reduce){.DocSearch-Hit--deleting{transition:none}}.DocSearch-Hit--deleting{opacity:0;transition:all .25s linear}@media screen and (prefers-reduced-motion:reduce){.DocSearch-Hit--favoriting{transition:none}}.DocSearch-Hit--favoriting{transform:scale(0);transform-origin:top center;transition:all .25s linear;transition-delay:.25s}.DocSearch-Hit a{background:var(--docsearch-hit-background);border-radius:4px;box-shadow:var(--docsearch-hit-shadow);display:block;padding-left:var(--docsearch-spacing);width:100%}.DocSearch-Hit-source{background:var(--docsearch-modal-background);color:var(--docsearch-highlight-color);font-size:.85em;font-weight:600;line-height:32px;margin:0 -4px;padding:8px 4px 0;position:sticky;top:0;z-index:10}.DocSearch-Hit-Tree{color:var(--docsearch-muted-color);height:var(--docsearch-hit-height);opacity:.5;stroke-width:var(--docsearch-icon-stroke-width);width:24px}.DocSearch-Hit[aria-selected=true] a{background-color:var(--docsearch-highlight-color)}.DocSearch-Hit[aria-selected=true] mark{text-decoration:underline}.DocSearch-Hit-Container{align-items:center;color:var(--docsearch-hit-color);display:flex;flex-direction:row;height:var(--docsearch-hit-height);padding:0 var(--docsearch-spacing) 0 0}.DocSearch-Hit-icon{height:20px;width:20px}.DocSearch-Hit-action,.DocSearch-Hit-icon{color:var(--docsearch-muted-color);stroke-width:var(--docsearch-icon-stroke-width)}.DocSearch-Hit-action{align-items:center;display:flex;height:22px;width:22px}.DocSearch-Hit-action svg{display:block;height:18px;width:18px}.DocSearch-Hit-action+.DocSearch-Hit-action{margin-left:6px}.DocSearch-Hit-action-button{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;border-radius:50%;color:inherit;cursor:pointer;padding:2px}svg.DocSearch-Hit-Select-Icon{display:none}.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-Select-Icon{display:block}.DocSearch-Hit-action-button:focus,.DocSearch-Hit-action-button:hover{background:#0003;transition:background-color .1s ease-in}@media screen and (prefers-reduced-motion:reduce){.DocSearch-Hit-action-button:focus,.DocSearch-Hit-action-button:hover{transition:none}}.DocSearch-Hit-action-button:focus path,.DocSearch-Hit-action-button:hover path{fill:#fff}.DocSearch-Hit-content-wrapper{display:flex;flex:1 1 auto;flex-direction:column;font-weight:500;justify-content:center;line-height:1.2em;margin:0 8px;overflow-x:hidden;position:relative;text-overflow:ellipsis;white-space:nowrap;width:80%}.DocSearch-Hit-title{font-size:.9em}.DocSearch-Hit-path{color:var(--docsearch-muted-color);font-size:.75em}.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-action,.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-icon,.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-path,.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-text,.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-title,.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-Tree,.DocSearch-Hit[aria-selected=true] mark{color:var(--docsearch-hit-active-color)!important}@media screen and (prefers-reduced-motion:reduce){.DocSearch-Hit-action-button:focus,.DocSearch-Hit-action-button:hover{background:#0003;transition:none}}.DocSearch-ErrorScreen,.DocSearch-NoResults,.DocSearch-StartScreen{font-size:.9em;margin:0 auto;padding:36px 0;text-align:center;width:80%}.DocSearch-Screen-Icon{color:var(--docsearch-muted-color);padding-bottom:12px}.DocSearch-NoResults-Prefill-List{display:inline-block;padding-bottom:24px;text-align:left}.DocSearch-NoResults-Prefill-List ul{display:inline-block;padding:8px 0 0}.DocSearch-NoResults-Prefill-List li{list-style-position:inside;list-style-type:"» "}.DocSearch-Prefill{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;border-radius:1em;color:var(--docsearch-highlight-color);cursor:pointer;display:inline-block;font-size:1em;font-weight:700;padding:0}.DocSearch-Prefill:focus,.DocSearch-Prefill:hover{outline:none;text-decoration:underline}.DocSearch-Footer{align-items:center;background:var(--docsearch-footer-background);border-radius:0 0 8px 8px;box-shadow:var(--docsearch-footer-shadow);display:flex;flex-direction:row-reverse;flex-shrink:0;height:var(--docsearch-footer-height);justify-content:space-between;padding:0 var(--docsearch-spacing);position:relative;-webkit-user-select:none;user-select:none;width:100%;z-index:300}.DocSearch-Commands{color:var(--docsearch-muted-color);display:flex;list-style:none;margin:0;padding:0}.DocSearch-Commands li{align-items:center;display:flex}.DocSearch-Commands li:not(:last-of-type){margin-right:.8em}.DocSearch-Commands-Key{align-items:center;background:var(--docsearch-key-gradient);border-radius:2px;box-shadow:var(--docsearch-key-shadow);display:flex;height:18px;justify-content:center;margin-right:.4em;padding:0 0 1px;color:var(--docsearch-muted-color);border:0;width:20px}.DocSearch-VisuallyHiddenForAccessibility{clip:rect(0 0 0 0);clip-path:inset(50%);height:1px;overflow:hidden;position:absolute;white-space:nowrap;width:1px}@media (max-width:768px){:root{--docsearch-spacing:10px;--docsearch-footer-height:40px}.DocSearch-Dropdown{height:100%}.DocSearch-Container{height:100vh;height:-webkit-fill-available;height:calc(var(--docsearch-vh, 1vh)*100);position:absolute}.DocSearch-Footer{border-radius:0;bottom:0;position:absolute}.DocSearch-Hit-content-wrapper{display:flex;position:relative;width:80%}.DocSearch-Modal{border-radius:0;box-shadow:none;height:100vh;height:-webkit-fill-available;height:calc(var(--docsearch-vh, 1vh)*100);margin:0;max-width:100%;width:100%}.DocSearch-Dropdown{max-height:calc(var(--docsearch-vh, 1vh)*100 - var(--docsearch-searchbox-height) - var(--docsearch-spacing) - var(--docsearch-footer-height))}.DocSearch-Cancel{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;color:var(--docsearch-highlight-color);cursor:pointer;display:inline-block;flex:none;font:inherit;font-size:1em;font-weight:500;margin-left:var(--docsearch-spacing);outline:none;overflow:hidden;padding:0;-webkit-user-select:none;user-select:none;white-space:nowrap}.DocSearch-Commands,.DocSearch-Hit-Tree{display:none}}@keyframes fade-in{0%{opacity:0}to{opacity:1}}[class*=DocSearch]{--docsearch-primary-color: var(--vp-c-brand-1);--docsearch-highlight-color: var(--docsearch-primary-color);--docsearch-text-color: var(--vp-c-text-1);--docsearch-muted-color: var(--vp-c-text-2);--docsearch-searchbox-shadow: none;--docsearch-searchbox-background: transparent;--docsearch-searchbox-focus-background: transparent;--docsearch-key-gradient: transparent;--docsearch-key-shadow: none;--docsearch-modal-background: var(--vp-c-bg-soft);--docsearch-footer-background: var(--vp-c-bg)}.dark [class*=DocSearch]{--docsearch-modal-shadow: none;--docsearch-footer-shadow: none;--docsearch-logo-color: var(--vp-c-text-2);--docsearch-hit-background: var(--vp-c-default-soft);--docsearch-hit-color: var(--vp-c-text-2);--docsearch-hit-shadow: none}.DocSearch-Button{display:flex;justify-content:center;align-items:center;margin:0;padding:0;width:48px;height:55px;background:transparent;transition:border-color .25s}.DocSearch-Button:hover{background:transparent}.DocSearch-Button:focus{outline:1px dotted;outline:5px auto -webkit-focus-ring-color}.DocSearch-Button-Key--pressed{transform:none;box-shadow:none}.DocSearch-Button:focus:not(:focus-visible){outline:none!important}@media (min-width: 768px){.DocSearch-Button{justify-content:flex-start;border:1px solid transparent;border-radius:8px;padding:0 10px 0 12px;width:100%;height:40px;background-color:var(--vp-c-bg-alt)}.DocSearch-Button:hover{border-color:var(--vp-c-brand-1);background:var(--vp-c-bg-alt)}}.DocSearch-Button .DocSearch-Button-Container{display:flex;align-items:center}.DocSearch-Button .DocSearch-Search-Icon{position:relative;width:16px;height:16px;color:var(--vp-c-text-1);fill:currentColor;transition:color .5s}.DocSearch-Button:hover .DocSearch-Search-Icon{color:var(--vp-c-text-1)}@media (min-width: 768px){.DocSearch-Button .DocSearch-Search-Icon{top:1px;margin-right:8px;width:14px;height:14px;color:var(--vp-c-text-2)}}.DocSearch-Button .DocSearch-Button-Placeholder{display:none;margin-top:2px;padding:0 16px 0 0;font-size:13px;font-weight:500;color:var(--vp-c-text-2);transition:color .5s}.DocSearch-Button:hover .DocSearch-Button-Placeholder{color:var(--vp-c-text-1)}@media (min-width: 768px){.DocSearch-Button .DocSearch-Button-Placeholder{display:inline-block}}.DocSearch-Button .DocSearch-Button-Keys{direction:ltr;display:none;min-width:auto}@media (min-width: 768px){.DocSearch-Button .DocSearch-Button-Keys{display:flex;align-items:center}}.DocSearch-Button .DocSearch-Button-Key{display:block;margin:2px 0 0;border:1px solid var(--vp-c-divider);border-right:none;border-radius:4px 0 0 4px;padding-left:6px;min-width:0;width:auto;height:22px;line-height:22px;font-family:var(--vp-font-family-base);font-size:12px;font-weight:500;transition:color .5s,border-color .5s}.DocSearch-Button .DocSearch-Button-Key+.DocSearch-Button-Key{border-right:1px solid var(--vp-c-divider);border-left:none;border-radius:0 4px 4px 0;padding-left:2px;padding-right:6px}.DocSearch-Button .DocSearch-Button-Key:first-child{font-size:0!important}.DocSearch-Button .DocSearch-Button-Key:first-child:after{content:"Ctrl";font-size:12px;letter-spacing:normal;color:var(--docsearch-muted-color)}.mac .DocSearch-Button .DocSearch-Button-Key:first-child:after{content:"⌘"}.DocSearch-Button .DocSearch-Button-Key:first-child>*{display:none}.DocSearch-Search-Icon{--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' stroke-width='1.6' viewBox='0 0 20 20'%3E%3Cpath fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' d='m14.386 14.386 4.088 4.088-4.088-4.088A7.533 7.533 0 1 1 3.733 3.733a7.533 7.533 0 0 1 10.653 10.653z'/%3E%3C/svg%3E")}.VPNavBarSearch{display:flex;align-items:center}@media (min-width: 768px){.VPNavBarSearch{flex-grow:1;padding-left:24px}}@media (min-width: 960px){.VPNavBarSearch{padding-left:32px}}.dark .DocSearch-Footer{border-top:1px solid var(--vp-c-divider)}.DocSearch-Form{border:1px solid var(--vp-c-brand-1);background-color:var(--vp-c-white)}.dark .DocSearch-Form{background-color:var(--vp-c-default-soft)}.DocSearch-Screen-Icon>svg{margin:auto}.VPNavBarSocialLinks[data-v-0394ad82]{display:none}@media (min-width: 1280px){.VPNavBarSocialLinks[data-v-0394ad82]{display:flex;align-items:center}}.title[data-v-ab179fa1]{display:flex;align-items:center;border-bottom:1px solid transparent;width:100%;height:var(--vp-nav-height);font-size:16px;font-weight:600;color:var(--vp-c-text-1);transition:opacity .25s}@media (min-width: 960px){.title[data-v-ab179fa1]{flex-shrink:0}.VPNavBarTitle.has-sidebar .title[data-v-ab179fa1]{border-bottom-color:var(--vp-c-divider)}}[data-v-ab179fa1] .logo{margin-right:8px;height:var(--vp-nav-logo-height)}.VPNavBarTranslations[data-v-88af2de4]{display:none}@media (min-width: 1280px){.VPNavBarTranslations[data-v-88af2de4]{display:flex;align-items:center}}.title[data-v-88af2de4]{padding:0 24px 0 12px;line-height:32px;font-size:14px;font-weight:700;color:var(--vp-c-text-1)}.VPNavBar[data-v-ccf7ddec]{position:relative;height:var(--vp-nav-height);pointer-events:none;white-space:nowrap;transition:background-color .5s}.VPNavBar[data-v-ccf7ddec]:not(.home){background-color:var(--vp-nav-bg-color)}@media (min-width: 960px){.VPNavBar[data-v-ccf7ddec]:not(.home){background-color:transparent}.VPNavBar[data-v-ccf7ddec]:not(.has-sidebar):not(.home.top){background-color:var(--vp-nav-bg-color)}}.wrapper[data-v-ccf7ddec]{padding:0 8px 0 24px}@media (min-width: 768px){.wrapper[data-v-ccf7ddec]{padding:0 32px}}@media (min-width: 960px){.VPNavBar.has-sidebar .wrapper[data-v-ccf7ddec]{padding:0}}.container[data-v-ccf7ddec]{display:flex;justify-content:space-between;margin:0 auto;max-width:calc(var(--vp-layout-max-width) - 64px);height:var(--vp-nav-height);pointer-events:none}.container>.title[data-v-ccf7ddec],.container>.content[data-v-ccf7ddec]{pointer-events:none}.container[data-v-ccf7ddec] *{pointer-events:auto}@media (min-width: 960px){.VPNavBar.has-sidebar .container[data-v-ccf7ddec]{max-width:100%}}.title[data-v-ccf7ddec]{flex-shrink:0;height:calc(var(--vp-nav-height) - 1px);transition:background-color .5s}@media (min-width: 960px){.VPNavBar.has-sidebar .title[data-v-ccf7ddec]{position:absolute;top:0;left:0;z-index:2;padding:0 32px;width:var(--vp-sidebar-width);height:var(--vp-nav-height);background-color:transparent}}@media (min-width: 1440px){.VPNavBar.has-sidebar .title[data-v-ccf7ddec]{padding-left:max(32px,calc((100% - (var(--vp-layout-max-width) - 64px)) / 2));width:calc((100% - (var(--vp-layout-max-width) - 64px)) / 2 + var(--vp-sidebar-width) - 32px)}}.content[data-v-ccf7ddec]{flex-grow:1}@media (min-width: 960px){.VPNavBar.has-sidebar .content[data-v-ccf7ddec]{position:relative;z-index:1;padding-right:32px;padding-left:var(--vp-sidebar-width)}}@media (min-width: 1440px){.VPNavBar.has-sidebar .content[data-v-ccf7ddec]{padding-right:calc((100vw - var(--vp-layout-max-width)) / 2 + 32px);padding-left:calc((100vw - var(--vp-layout-max-width)) / 2 + var(--vp-sidebar-width))}}.content-body[data-v-ccf7ddec]{display:flex;justify-content:flex-end;align-items:center;height:var(--vp-nav-height);transition:background-color .5s}@media (min-width: 960px){.VPNavBar:not(.home.top) .content-body[data-v-ccf7ddec]{position:relative;background-color:var(--vp-nav-bg-color)}.VPNavBar:not(.has-sidebar):not(.home.top) .content-body[data-v-ccf7ddec]{background-color:transparent}}@media (max-width: 767px){.content-body[data-v-ccf7ddec]{column-gap:.5rem}}.menu+.translations[data-v-ccf7ddec]:before,.menu+.appearance[data-v-ccf7ddec]:before,.menu+.social-links[data-v-ccf7ddec]:before,.translations+.appearance[data-v-ccf7ddec]:before,.appearance+.social-links[data-v-ccf7ddec]:before{margin-right:8px;margin-left:8px;width:1px;height:24px;background-color:var(--vp-c-divider);content:""}.menu+.appearance[data-v-ccf7ddec]:before,.translations+.appearance[data-v-ccf7ddec]:before{margin-right:16px}.appearance+.social-links[data-v-ccf7ddec]:before{margin-left:16px}.social-links[data-v-ccf7ddec]{margin-right:-8px}.divider[data-v-ccf7ddec]{width:100%;height:1px}@media (min-width: 960px){.VPNavBar.has-sidebar .divider[data-v-ccf7ddec]{padding-left:var(--vp-sidebar-width)}}@media (min-width: 1440px){.VPNavBar.has-sidebar .divider[data-v-ccf7ddec]{padding-left:calc((100vw - var(--vp-layout-max-width)) / 2 + var(--vp-sidebar-width))}}.divider-line[data-v-ccf7ddec]{width:100%;height:1px;transition:background-color .5s}.VPNavBar:not(.home) .divider-line[data-v-ccf7ddec]{background-color:var(--vp-c-gutter)}@media (min-width: 960px){.VPNavBar:not(.home.top) .divider-line[data-v-ccf7ddec]{background-color:var(--vp-c-gutter)}.VPNavBar:not(.has-sidebar):not(.home.top) .divider[data-v-ccf7ddec]{background-color:var(--vp-c-gutter)}}.VPNavScreenAppearance[data-v-2d7af913]{display:flex;justify-content:space-between;align-items:center;border-radius:8px;padding:12px 14px 12px 16px;background-color:var(--vp-c-bg-soft)}.text[data-v-2d7af913]{line-height:24px;font-size:12px;font-weight:500;color:var(--vp-c-text-2)}.VPNavScreenMenuLink[data-v-7f31e1f6]{display:block;border-bottom:1px solid var(--vp-c-divider);padding:12px 0 11px;line-height:24px;font-size:14px;font-weight:500;color:var(--vp-c-text-1);transition:border-color .25s,color .25s}.VPNavScreenMenuLink[data-v-7f31e1f6]:hover{color:var(--vp-c-brand-1)}.VPNavScreenMenuGroupLink[data-v-19976ae1]{display:block;margin-left:12px;line-height:32px;font-size:14px;font-weight:400;color:var(--vp-c-text-1);transition:color .25s}.VPNavScreenMenuGroupLink[data-v-19976ae1]:hover{color:var(--vp-c-brand-1)}.VPNavScreenMenuGroupSection[data-v-8133b170]{display:block}.title[data-v-8133b170]{line-height:32px;font-size:13px;font-weight:700;color:var(--vp-c-text-2);transition:color .25s}.VPNavScreenMenuGroup[data-v-ff6087d4]{border-bottom:1px solid var(--vp-c-divider);height:48px;overflow:hidden;transition:border-color .5s}.VPNavScreenMenuGroup .items[data-v-ff6087d4]{visibility:hidden}.VPNavScreenMenuGroup.open .items[data-v-ff6087d4]{visibility:visible}.VPNavScreenMenuGroup.open[data-v-ff6087d4]{padding-bottom:10px;height:auto}.VPNavScreenMenuGroup.open .button[data-v-ff6087d4]{padding-bottom:6px;color:var(--vp-c-brand-1)}.VPNavScreenMenuGroup.open .button-icon[data-v-ff6087d4]{transform:rotate(45deg)}.button[data-v-ff6087d4]{display:flex;justify-content:space-between;align-items:center;padding:12px 4px 11px 0;width:100%;line-height:24px;font-size:14px;font-weight:500;color:var(--vp-c-text-1);transition:color .25s}.button[data-v-ff6087d4]:hover{color:var(--vp-c-brand-1)}.button-icon[data-v-ff6087d4]{transition:transform .25s}.group[data-v-ff6087d4]:first-child{padding-top:0}.group+.group[data-v-ff6087d4],.group+.item[data-v-ff6087d4]{padding-top:4px}.VPNavScreenTranslations[data-v-858fe1a4]{height:24px;overflow:hidden}.VPNavScreenTranslations.open[data-v-858fe1a4]{height:auto}.title[data-v-858fe1a4]{display:flex;align-items:center;font-size:14px;font-weight:500;color:var(--vp-c-text-1)}.icon[data-v-858fe1a4]{font-size:16px}.icon.lang[data-v-858fe1a4]{margin-right:8px}.icon.chevron[data-v-858fe1a4]{margin-left:4px}.list[data-v-858fe1a4]{padding:4px 0 0 24px}.link[data-v-858fe1a4]{line-height:32px;font-size:13px;color:var(--vp-c-text-1)}.VPNavScreen[data-v-cc5739dd]{position:fixed;top:calc(var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + 1px);right:0;bottom:0;left:0;padding:0 32px;width:100%;background-color:var(--vp-nav-screen-bg-color);overflow-y:auto;transition:background-color .5s;pointer-events:auto}.VPNavScreen.fade-enter-active[data-v-cc5739dd],.VPNavScreen.fade-leave-active[data-v-cc5739dd]{transition:opacity .25s}.VPNavScreen.fade-enter-active .container[data-v-cc5739dd],.VPNavScreen.fade-leave-active .container[data-v-cc5739dd]{transition:transform .25s ease}.VPNavScreen.fade-enter-from[data-v-cc5739dd],.VPNavScreen.fade-leave-to[data-v-cc5739dd]{opacity:0}.VPNavScreen.fade-enter-from .container[data-v-cc5739dd],.VPNavScreen.fade-leave-to .container[data-v-cc5739dd]{transform:translateY(-8px)}@media (min-width: 768px){.VPNavScreen[data-v-cc5739dd]{display:none}}.container[data-v-cc5739dd]{margin:0 auto;padding:24px 0 96px;max-width:288px}.menu+.translations[data-v-cc5739dd],.menu+.appearance[data-v-cc5739dd],.translations+.appearance[data-v-cc5739dd]{margin-top:24px}.menu+.social-links[data-v-cc5739dd]{margin-top:16px}.appearance+.social-links[data-v-cc5739dd]{margin-top:16px}.VPNav[data-v-ae24b3ad]{position:relative;top:var(--vp-layout-top-height, 0px);left:0;z-index:var(--vp-z-index-nav);width:100%;pointer-events:none;transition:background-color .5s}@media (min-width: 960px){.VPNav[data-v-ae24b3ad]{position:fixed}}.VPSidebarItem.level-0[data-v-b8d55f3b]{padding-bottom:24px}.VPSidebarItem.collapsed.level-0[data-v-b8d55f3b]{padding-bottom:10px}.item[data-v-b8d55f3b]{position:relative;display:flex;width:100%}.VPSidebarItem.collapsible>.item[data-v-b8d55f3b]{cursor:pointer}.indicator[data-v-b8d55f3b]{position:absolute;top:6px;bottom:6px;left:-17px;width:2px;border-radius:2px;transition:background-color .25s}.VPSidebarItem.level-2.is-active>.item>.indicator[data-v-b8d55f3b],.VPSidebarItem.level-3.is-active>.item>.indicator[data-v-b8d55f3b],.VPSidebarItem.level-4.is-active>.item>.indicator[data-v-b8d55f3b],.VPSidebarItem.level-5.is-active>.item>.indicator[data-v-b8d55f3b]{background-color:var(--vp-c-brand-1)}.link[data-v-b8d55f3b]{display:flex;align-items:center;flex-grow:1}.text[data-v-b8d55f3b]{flex-grow:1;padding:4px 0;line-height:24px;font-size:14px;transition:color .25s}.VPSidebarItem.level-0 .text[data-v-b8d55f3b]{font-weight:700;color:var(--vp-c-text-1)}.VPSidebarItem.level-1 .text[data-v-b8d55f3b],.VPSidebarItem.level-2 .text[data-v-b8d55f3b],.VPSidebarItem.level-3 .text[data-v-b8d55f3b],.VPSidebarItem.level-4 .text[data-v-b8d55f3b],.VPSidebarItem.level-5 .text[data-v-b8d55f3b]{font-weight:500;color:var(--vp-c-text-2)}.VPSidebarItem.level-0.is-link>.item>.link:hover .text[data-v-b8d55f3b],.VPSidebarItem.level-1.is-link>.item>.link:hover .text[data-v-b8d55f3b],.VPSidebarItem.level-2.is-link>.item>.link:hover .text[data-v-b8d55f3b],.VPSidebarItem.level-3.is-link>.item>.link:hover .text[data-v-b8d55f3b],.VPSidebarItem.level-4.is-link>.item>.link:hover .text[data-v-b8d55f3b],.VPSidebarItem.level-5.is-link>.item>.link:hover .text[data-v-b8d55f3b]{color:var(--vp-c-brand-1)}.VPSidebarItem.level-0.has-active>.item>.text[data-v-b8d55f3b],.VPSidebarItem.level-1.has-active>.item>.text[data-v-b8d55f3b],.VPSidebarItem.level-2.has-active>.item>.text[data-v-b8d55f3b],.VPSidebarItem.level-3.has-active>.item>.text[data-v-b8d55f3b],.VPSidebarItem.level-4.has-active>.item>.text[data-v-b8d55f3b],.VPSidebarItem.level-5.has-active>.item>.text[data-v-b8d55f3b],.VPSidebarItem.level-0.has-active>.item>.link>.text[data-v-b8d55f3b],.VPSidebarItem.level-1.has-active>.item>.link>.text[data-v-b8d55f3b],.VPSidebarItem.level-2.has-active>.item>.link>.text[data-v-b8d55f3b],.VPSidebarItem.level-3.has-active>.item>.link>.text[data-v-b8d55f3b],.VPSidebarItem.level-4.has-active>.item>.link>.text[data-v-b8d55f3b],.VPSidebarItem.level-5.has-active>.item>.link>.text[data-v-b8d55f3b]{color:var(--vp-c-text-1)}.VPSidebarItem.level-0.is-active>.item .link>.text[data-v-b8d55f3b],.VPSidebarItem.level-1.is-active>.item .link>.text[data-v-b8d55f3b],.VPSidebarItem.level-2.is-active>.item .link>.text[data-v-b8d55f3b],.VPSidebarItem.level-3.is-active>.item .link>.text[data-v-b8d55f3b],.VPSidebarItem.level-4.is-active>.item .link>.text[data-v-b8d55f3b],.VPSidebarItem.level-5.is-active>.item .link>.text[data-v-b8d55f3b]{color:var(--vp-c-brand-1)}.caret[data-v-b8d55f3b]{display:flex;justify-content:center;align-items:center;margin-right:-7px;width:32px;height:32px;color:var(--vp-c-text-3);cursor:pointer;transition:color .25s;flex-shrink:0}.item:hover .caret[data-v-b8d55f3b]{color:var(--vp-c-text-2)}.item:hover .caret[data-v-b8d55f3b]:hover{color:var(--vp-c-text-1)}.caret-icon[data-v-b8d55f3b]{font-size:18px;transform:rotate(90deg);transition:transform .25s}.VPSidebarItem.collapsed .caret-icon[data-v-b8d55f3b]{transform:rotate(0)}.VPSidebarItem.level-1 .items[data-v-b8d55f3b],.VPSidebarItem.level-2 .items[data-v-b8d55f3b],.VPSidebarItem.level-3 .items[data-v-b8d55f3b],.VPSidebarItem.level-4 .items[data-v-b8d55f3b],.VPSidebarItem.level-5 .items[data-v-b8d55f3b]{border-left:1px solid var(--vp-c-divider);padding-left:16px}.VPSidebarItem.collapsed .items[data-v-b8d55f3b]{display:none}.VPSidebar[data-v-575e6a36]{position:fixed;top:var(--vp-layout-top-height, 0px);bottom:0;left:0;z-index:var(--vp-z-index-sidebar);padding:32px 32px 96px;width:calc(100vw - 64px);max-width:320px;background-color:var(--vp-sidebar-bg-color);opacity:0;box-shadow:var(--vp-c-shadow-3);overflow-x:hidden;overflow-y:auto;transform:translate(-100%);transition:opacity .5s,transform .25s ease;overscroll-behavior:contain}.VPSidebar.open[data-v-575e6a36]{opacity:1;visibility:visible;transform:translate(0);transition:opacity .25s,transform .5s cubic-bezier(.19,1,.22,1)}.dark .VPSidebar[data-v-575e6a36]{box-shadow:var(--vp-shadow-1)}@media (min-width: 960px){.VPSidebar[data-v-575e6a36]{padding-top:var(--vp-nav-height);width:var(--vp-sidebar-width);max-width:100%;background-color:var(--vp-sidebar-bg-color);opacity:1;visibility:visible;box-shadow:none;transform:translate(0)}}@media (min-width: 1440px){.VPSidebar[data-v-575e6a36]{padding-left:max(32px,calc((100% - (var(--vp-layout-max-width) - 64px)) / 2));width:calc((100% - (var(--vp-layout-max-width) - 64px)) / 2 + var(--vp-sidebar-width) - 32px)}}@media (min-width: 960px){.curtain[data-v-575e6a36]{position:sticky;top:-64px;left:0;z-index:1;margin-top:calc(var(--vp-nav-height) * -1);margin-right:-32px;margin-left:-32px;height:var(--vp-nav-height);background-color:var(--vp-sidebar-bg-color)}}.nav[data-v-575e6a36]{outline:0}.group+.group[data-v-575e6a36]{border-top:1px solid var(--vp-c-divider);padding-top:10px}@media (min-width: 960px){.group[data-v-575e6a36]{padding-top:10px;width:calc(var(--vp-sidebar-width) - 64px)}}.VPSkipLink[data-v-0f60ec36]{top:8px;left:8px;padding:8px 16px;z-index:999;border-radius:8px;font-size:12px;font-weight:700;text-decoration:none;color:var(--vp-c-brand-1);box-shadow:var(--vp-shadow-3);background-color:var(--vp-c-bg)}.VPSkipLink[data-v-0f60ec36]:focus{height:auto;width:auto;clip:auto;clip-path:none}@media (min-width: 1280px){.VPSkipLink[data-v-0f60ec36]{top:14px;left:16px}}.Layout[data-v-5d98c3a5]{display:flex;flex-direction:column;min-height:100vh}.VPHomeSponsors[data-v-3d121b4a]{border-top:1px solid var(--vp-c-gutter);padding-top:88px!important}.VPHomeSponsors[data-v-3d121b4a]{margin:96px 0}@media (min-width: 768px){.VPHomeSponsors[data-v-3d121b4a]{margin:128px 0}}.VPHomeSponsors[data-v-3d121b4a]{padding:0 24px}@media (min-width: 768px){.VPHomeSponsors[data-v-3d121b4a]{padding:0 48px}}@media (min-width: 960px){.VPHomeSponsors[data-v-3d121b4a]{padding:0 64px}}.container[data-v-3d121b4a]{margin:0 auto;max-width:1152px}.love[data-v-3d121b4a]{margin:0 auto;width:fit-content;font-size:28px;color:var(--vp-c-text-3)}.icon[data-v-3d121b4a]{display:inline-block}.message[data-v-3d121b4a]{margin:0 auto;padding-top:10px;max-width:320px;text-align:center;line-height:24px;font-size:16px;font-weight:500;color:var(--vp-c-text-2)}.sponsors[data-v-3d121b4a]{padding-top:32px}.action[data-v-3d121b4a]{padding-top:40px;text-align:center}.VPTeamPage[data-v-7c57f839]{margin:96px 0}@media (min-width: 768px){.VPTeamPage[data-v-7c57f839]{margin:128px 0}}.VPHome .VPTeamPageTitle[data-v-7c57f839-s]{border-top:1px solid var(--vp-c-gutter);padding-top:88px!important}.VPTeamPageSection+.VPTeamPageSection[data-v-7c57f839-s],.VPTeamMembers+.VPTeamPageSection[data-v-7c57f839-s]{margin-top:64px}.VPTeamMembers+.VPTeamMembers[data-v-7c57f839-s]{margin-top:24px}@media (min-width: 768px){.VPTeamPageTitle+.VPTeamPageSection[data-v-7c57f839-s]{margin-top:16px}.VPTeamPageSection+.VPTeamPageSection[data-v-7c57f839-s],.VPTeamMembers+.VPTeamPageSection[data-v-7c57f839-s]{margin-top:96px}}.VPTeamMembers[data-v-7c57f839-s]{padding:0 24px}@media (min-width: 768px){.VPTeamMembers[data-v-7c57f839-s]{padding:0 48px}}@media (min-width: 960px){.VPTeamMembers[data-v-7c57f839-s]{padding:0 64px}}.VPTeamPageTitle[data-v-bf2cbdac]{padding:48px 32px;text-align:center}@media (min-width: 768px){.VPTeamPageTitle[data-v-bf2cbdac]{padding:64px 48px 48px}}@media (min-width: 960px){.VPTeamPageTitle[data-v-bf2cbdac]{padding:80px 64px 48px}}.title[data-v-bf2cbdac]{letter-spacing:0;line-height:44px;font-size:36px;font-weight:500}@media (min-width: 768px){.title[data-v-bf2cbdac]{letter-spacing:-.5px;line-height:56px;font-size:48px}}.lead[data-v-bf2cbdac]{margin:0 auto;max-width:512px;padding-top:12px;line-height:24px;font-size:16px;font-weight:500;color:var(--vp-c-text-2)}@media (min-width: 768px){.lead[data-v-bf2cbdac]{max-width:592px;letter-spacing:.15px;line-height:28px;font-size:20px}}.VPTeamPageSection[data-v-b1a88750]{padding:0 32px}@media (min-width: 768px){.VPTeamPageSection[data-v-b1a88750]{padding:0 48px}}@media (min-width: 960px){.VPTeamPageSection[data-v-b1a88750]{padding:0 64px}}.title[data-v-b1a88750]{position:relative;margin:0 auto;max-width:1152px;text-align:center;color:var(--vp-c-text-2)}.title-line[data-v-b1a88750]{position:absolute;top:16px;left:0;width:100%;height:1px;background-color:var(--vp-c-divider)}.title-text[data-v-b1a88750]{position:relative;display:inline-block;padding:0 24px;letter-spacing:0;line-height:32px;font-size:20px;font-weight:500;background-color:var(--vp-c-bg)}.lead[data-v-b1a88750]{margin:0 auto;max-width:480px;padding-top:12px;text-align:center;line-height:24px;font-size:16px;font-weight:500;color:var(--vp-c-text-2)}.members[data-v-b1a88750]{padding-top:40px}.VPTeamMembersItem[data-v-f3fa364a]{display:flex;flex-direction:column;gap:2px;border-radius:12px;width:100%;height:100%;overflow:hidden}.VPTeamMembersItem.small .profile[data-v-f3fa364a]{padding:32px}.VPTeamMembersItem.small .data[data-v-f3fa364a]{padding-top:20px}.VPTeamMembersItem.small .avatar[data-v-f3fa364a]{width:64px;height:64px}.VPTeamMembersItem.small .name[data-v-f3fa364a]{line-height:24px;font-size:16px}.VPTeamMembersItem.small .affiliation[data-v-f3fa364a]{padding-top:4px;line-height:20px;font-size:14px}.VPTeamMembersItem.small .desc[data-v-f3fa364a]{padding-top:12px;line-height:20px;font-size:14px}.VPTeamMembersItem.small .links[data-v-f3fa364a]{margin:0 -16px -20px;padding:10px 0 0}.VPTeamMembersItem.medium .profile[data-v-f3fa364a]{padding:48px 32px}.VPTeamMembersItem.medium .data[data-v-f3fa364a]{padding-top:24px;text-align:center}.VPTeamMembersItem.medium .avatar[data-v-f3fa364a]{width:96px;height:96px}.VPTeamMembersItem.medium .name[data-v-f3fa364a]{letter-spacing:.15px;line-height:28px;font-size:20px}.VPTeamMembersItem.medium .affiliation[data-v-f3fa364a]{padding-top:4px;font-size:16px}.VPTeamMembersItem.medium .desc[data-v-f3fa364a]{padding-top:16px;max-width:288px;font-size:16px}.VPTeamMembersItem.medium .links[data-v-f3fa364a]{margin:0 -16px -12px;padding:16px 12px 0}.profile[data-v-f3fa364a]{flex-grow:1;background-color:var(--vp-c-bg-soft)}.data[data-v-f3fa364a]{text-align:center}.avatar[data-v-f3fa364a]{position:relative;flex-shrink:0;margin:0 auto;border-radius:50%;box-shadow:var(--vp-shadow-3)}.avatar-img[data-v-f3fa364a]{position:absolute;top:0;right:0;bottom:0;left:0;border-radius:50%;object-fit:cover}.name[data-v-f3fa364a]{margin:0;font-weight:600}.affiliation[data-v-f3fa364a]{margin:0;font-weight:500;color:var(--vp-c-text-2)}.org.link[data-v-f3fa364a]{color:var(--vp-c-text-2);transition:color .25s}.org.link[data-v-f3fa364a]:hover{color:var(--vp-c-brand-1)}.desc[data-v-f3fa364a]{margin:0 auto}.desc[data-v-f3fa364a] a{font-weight:500;color:var(--vp-c-brand-1);text-decoration-style:dotted;transition:color .25s}.links[data-v-f3fa364a]{display:flex;justify-content:center;height:56px}.sp-link[data-v-f3fa364a]{display:flex;justify-content:center;align-items:center;text-align:center;padding:16px;font-size:14px;font-weight:500;color:var(--vp-c-sponsor);background-color:var(--vp-c-bg-soft);transition:color .25s,background-color .25s}.sp .sp-link.link[data-v-f3fa364a]:hover,.sp .sp-link.link[data-v-f3fa364a]:focus{outline:none;color:var(--vp-c-white);background-color:var(--vp-c-sponsor)}.sp-icon[data-v-f3fa364a]{margin-right:8px;font-size:16px}.VPTeamMembers.small .container[data-v-6cb0dbc4]{grid-template-columns:repeat(auto-fit,minmax(224px,1fr))}.VPTeamMembers.small.count-1 .container[data-v-6cb0dbc4]{max-width:276px}.VPTeamMembers.small.count-2 .container[data-v-6cb0dbc4]{max-width:576px}.VPTeamMembers.small.count-3 .container[data-v-6cb0dbc4]{max-width:876px}.VPTeamMembers.medium .container[data-v-6cb0dbc4]{grid-template-columns:repeat(auto-fit,minmax(256px,1fr))}@media (min-width: 375px){.VPTeamMembers.medium .container[data-v-6cb0dbc4]{grid-template-columns:repeat(auto-fit,minmax(288px,1fr))}}.VPTeamMembers.medium.count-1 .container[data-v-6cb0dbc4]{max-width:368px}.VPTeamMembers.medium.count-2 .container[data-v-6cb0dbc4]{max-width:760px}.container[data-v-6cb0dbc4]{display:grid;gap:24px;margin:0 auto;max-width:1152px}.VPDocAsideSponsors[data-v-9a4b6acc]{margin-top:8px!important}.banner[data-v-9a4b6acc]{margin:.25rem 0;padding:.4rem 0;border-radius:.8rem;position:relative;display:flex;align-items:center;justify-content:center;width:100%;gap:1rem;background-color:var(--vp-c-bg-soft);border:2px solid var(--vp-c-bg-soft);transition:border-color .5s}.banner[data-v-9a4b6acc]:last-of-type{margin-bottom:1rem}.banner[data-v-9a4b6acc]:hover{border:2px solid var(--vp-c-brand-1)}.banner img[data-v-9a4b6acc]{transition:transform .5s;transform:scale(1.25)}.banner:hover img[data-v-9a4b6acc]{transform:scale(1.75)}.banner .name[data-v-9a4b6acc]{background-image:linear-gradient(120deg,var(--vp-c-brand-3) 32%,var(--vp-c-brand-2),var(--vp-c-brand-1));background-clip:text;-webkit-background-clip:text;-webkit-text-fill-color:transparent;font-size:.9rem;font-weight:600;margin-left:1rem;writing-mode:vertical-rl;text-orientation:upright}.banner .info[data-v-9a4b6acc]{display:flex;flex-direction:column;align-items:center;justify-content:center;flex-grow:.5;height:100%;position:relative}.banner .version[data-v-9a4b6acc]{color:var(--vp-c-text-2);font-size:.8rem;font-weight:700;white-space:nowrap;margin-top:auto;margin-bottom:auto}.banner .change[data-v-9a4b6acc]{color:var(--vp-c-text-1);font-size:.8rem;font-weight:700;margin-top:auto;text-align:center}
diff --git a/assets/webssh.DHR7F3te.jpg b/assets/webssh.DHR7F3te.jpg
deleted file mode 100644
index 4acdd67f..00000000
Binary files a/assets/webssh.DHR7F3te.jpg and /dev/null differ
diff --git a/case/case1.html b/case/case1.html
deleted file mode 100644
index 66068c83..00000000
--- a/case/case1.html
+++ /dev/null
@@ -1,29 +0,0 @@
-
-
-
-
-
-
如果运行失败,提示 xxx 命令找不到,可能是 PATH 环境变量的问题。在 Linux 服务器上,可以在命令开头加入 source ~/.bashrc
,或者使用绝对路径执行命令。Nezha Monitoring
Nezha Monitoring