update
This commit is contained in:
@@ -0,0 +1,9 @@
|
|||||||
|
|
||||||
|
```sh
|
||||||
|
# 安装 python 环境 ,安装项目
|
||||||
|
uv sync
|
||||||
|
|
||||||
|
uv run python main.py
|
||||||
|
|
||||||
|
pm2 start "uv run app.py" --name "dnake-socket-app"
|
||||||
|
```
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@kevisual/me-dnake-docs",
|
"name": "@kevisual/me-dnake-docs",
|
||||||
"version": "0.0.1",
|
"version": "0.0.2",
|
||||||
"description": "",
|
"description": "",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"basename": "/root/me-dnake-docs",
|
"basename": "/root/me-dnake-docs",
|
||||||
@@ -8,7 +8,7 @@
|
|||||||
"dev": "astro dev",
|
"dev": "astro dev",
|
||||||
"build": "astro build",
|
"build": "astro build",
|
||||||
"preview": "astro preview",
|
"preview": "astro preview",
|
||||||
"pub": "envision deploy ./dist -k me-dnake-docs -v 0.0.1 -u",
|
"pub": "envision deploy ./dist -k me-dnake-docs -v 0.0.2 -u",
|
||||||
"sn": "pnpm dlx shadcn@latest add "
|
"sn": "pnpm dlx shadcn@latest add "
|
||||||
},
|
},
|
||||||
"keywords": [],
|
"keywords": [],
|
||||||
|
|||||||
48
docs/src/pages/home.mdx
Normal file
48
docs/src/pages/home.mdx
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
# home配置文档
|
||||||
|
|
||||||
|
## configuration.yaml
|
||||||
|
|
||||||
|
配置调用rest_command服务,文件地址在 docker 内部根目录下,`config/configuration.yaml` 位置
|
||||||
|
|
||||||
|
添加以下内容
|
||||||
|
```yaml
|
||||||
|
rest_command:
|
||||||
|
set_floor_notify:
|
||||||
|
url: http://host.docker.internal:3001/api/router?path=floor
|
||||||
|
method: POST
|
||||||
|
content_type: application/json
|
||||||
|
```
|
||||||
|
然后配置脚本位置 `config/scripts.yaml` 文件,添加以下内容
|
||||||
|
(scripts 被 configuration.yaml 引用)
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
notify_floor:
|
||||||
|
alias: "发送楼层消息"
|
||||||
|
description: "通过本地服务发送楼层通知"
|
||||||
|
sequence:
|
||||||
|
- service: rest_command.set_floor_notify
|
||||||
|
```
|
||||||
|
## 触发使用
|
||||||
|
home 可视化页面可以编辑自动化,编辑后台也可以,在 automations.yaml 文件中添加以下类似内容
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
- id: '1760892271622'
|
||||||
|
alias: 打开电梯
|
||||||
|
description: 打开电梯,上13楼来
|
||||||
|
triggers:
|
||||||
|
- trigger: state
|
||||||
|
entity_id:
|
||||||
|
- event.xiaomi_cn_1076466578_hub1_virtual_event_e_4_1
|
||||||
|
from:
|
||||||
|
to:
|
||||||
|
conditions:
|
||||||
|
- condition: state
|
||||||
|
entity_id: event.xiaomi_cn_1076466578_hub1_virtual_event_e_4_1
|
||||||
|
attribute: 事件名称
|
||||||
|
state: 打开电梯
|
||||||
|
actions:
|
||||||
|
- action: script.notify_floor
|
||||||
|
metadata: {}
|
||||||
|
data: {}
|
||||||
|
mode: single
|
||||||
|
```
|
||||||
@@ -2,6 +2,7 @@
|
|||||||
import '../styles/global.css';
|
import '../styles/global.css';
|
||||||
|
|
||||||
const links = [
|
const links = [
|
||||||
|
|
||||||
{
|
{
|
||||||
link: './adb/',
|
link: './adb/',
|
||||||
title: 'ADB 文档',
|
title: 'ADB 文档',
|
||||||
@@ -13,41 +14,52 @@ const links = [
|
|||||||
{
|
{
|
||||||
link: './ip/',
|
link: './ip/',
|
||||||
title: 'IP 相关文档, IPTABLES 配置,如何从A 局域网 ping 同B 局域网',
|
title: 'IP 相关文档, IPTABLES 配置,如何从A 局域网 ping 同B 局域网',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
link: './python/',
|
||||||
|
title: 'Python 脚本运行文档',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
link: './home/',
|
||||||
|
title: 'homeassistant 触发脚本文档方法',
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
---
|
---
|
||||||
|
|
||||||
<html lang='zh-CN'>
|
<html lang='zh-CN'>
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8" />
|
<meta charset='UTF-8' />
|
||||||
<title>My Homepage</title>
|
<title>My Homepage</title>
|
||||||
<script src="https://cdn.tailwindcss.com"></script>
|
<script src='https://cdn.tailwindcss.com'></script>
|
||||||
</head>
|
</head>
|
||||||
<body class="bg-gray-50 min-h-screen">
|
<body class='bg-gray-50 min-h-screen'>
|
||||||
<div class="container mx-auto px-4 py-8">
|
<div class='container mx-auto px-4 py-8'>
|
||||||
<h1 class="text-3xl font-bold text-gray-800 mb-8 text-center">文档导航</h1>
|
<h1 class='text-3xl font-bold text-gray-800 mb-8 text-center'>文档导航</h1>
|
||||||
|
|
||||||
<div class="max-w-2xl mx-auto">
|
<div class='max-w-2xl mx-auto'>
|
||||||
<ul class="space-y-4">
|
<ul class='space-y-4'>
|
||||||
{links.map((item) => (
|
{
|
||||||
<li>
|
links.map((item) => (
|
||||||
<a
|
<li>
|
||||||
href={item.link}
|
<a
|
||||||
class="block p-6 bg-white rounded-lg shadow-md hover:shadow-lg transition-shadow duration-300 border border-gray-200 hover:border-blue-300 group"
|
href={item.link}
|
||||||
>
|
class='block p-6 bg-white rounded-lg shadow-md hover:shadow-lg transition-shadow duration-300 border border-gray-200 hover:border-blue-300 group'>
|
||||||
<h2 class="text-xl font-semibold text-gray-800 group-hover:text-blue-600 transition-colors duration-200">
|
<h2 class='text-xl font-semibold text-gray-800 group-hover:text-blue-600 transition-colors duration-200'>{item.title}</h2>
|
||||||
{item.title}
|
<p class='text-gray-600 mt-2'>点击查看详细文档</p>
|
||||||
</h2>
|
<div class='flex items-center mt-3 text-blue-500 group-hover:text-blue-700'>
|
||||||
<p class="text-gray-600 mt-2">点击查看详细文档</p>
|
<span class='text-sm'>查看文档</span>
|
||||||
<div class="flex items-center mt-3 text-blue-500 group-hover:text-blue-700">
|
<svg
|
||||||
<span class="text-sm">查看文档</span>
|
class='w-4 h-4 ml-2 transform group-hover:translate-x-1 transition-transform duration-200'
|
||||||
<svg class="w-4 h-4 ml-2 transform group-hover:translate-x-1 transition-transform duration-200" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
fill='none'
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"></path>
|
stroke='currentColor'
|
||||||
</svg>
|
viewBox='0 0 24 24'>
|
||||||
</div>
|
<path stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M9 5l7 7-7 7' />
|
||||||
</a>
|
</svg>
|
||||||
</li>
|
</div>
|
||||||
))}
|
</a>
|
||||||
|
</li>
|
||||||
|
))
|
||||||
|
}
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
144
docs/src/pages/python.mdx
Normal file
144
docs/src/pages/python.mdx
Normal file
@@ -0,0 +1,144 @@
|
|||||||
|
# python 代码示例
|
||||||
|
|
||||||
|
```sh
|
||||||
|
# 安装 python 环境 ,安装项目
|
||||||
|
uv sync
|
||||||
|
```
|
||||||
|
|
||||||
|
## 运行 python 脚本
|
||||||
|
|
||||||
|
main.py 主模块
|
||||||
|
|
||||||
|
```python
|
||||||
|
import socket
|
||||||
|
import time
|
||||||
|
|
||||||
|
def send_sip_message():
|
||||||
|
# 配置参数
|
||||||
|
target_ip = "192.168.3.3" # 设备eth1的IP(转发入口)
|
||||||
|
target_port = 5060 # 转发端口
|
||||||
|
mac_local_ip = "192.168.3.33"# Mac的本地IP(与Via头部一致)
|
||||||
|
sender_sip = "3011302" # 发送方SIP用户
|
||||||
|
sender_ip = "192.168.9.57" # 设备eth0的IP(转发后源IP)
|
||||||
|
receiver_sip = "3019901" # 接收方SIP用户
|
||||||
|
receiver_ip = "192.168.9.4" # 目标IP
|
||||||
|
|
||||||
|
# 生成唯一标识
|
||||||
|
call_id = f"{int(time.time())}" # 基于时间戳的Call-ID
|
||||||
|
tag = f"{int(time.time() % 1000000)}" # 随机tag
|
||||||
|
branch = f"z9hG4bK{int(time.time())}" # Via分支标识
|
||||||
|
|
||||||
|
# XML消息体(业务数据)
|
||||||
|
xml_body = '''<?xml version="1.0" encoding="UTF-8" ?>
|
||||||
|
<params>
|
||||||
|
<to>sip:{receiver_sip}@{receiver_ip}:5060</to>
|
||||||
|
<elev>1</elev>
|
||||||
|
<direct>2</direct>
|
||||||
|
<floor>13</floor>
|
||||||
|
<family>2</family>
|
||||||
|
<app>elev</app>
|
||||||
|
<event>appoint</event>
|
||||||
|
<event_url>/elev/appoint</event_url>
|
||||||
|
</params>'''.format(receiver_sip=receiver_sip, receiver_ip=receiver_ip)
|
||||||
|
|
||||||
|
# 计算消息体长度(字节数)
|
||||||
|
content_length = len(xml_body.encode('utf-8'))
|
||||||
|
|
||||||
|
# 构造SIP MESSAGE请求(严格使用\r\n换行)
|
||||||
|
sip_message = f'''MESSAGE sip:{receiver_sip}@{receiver_ip}:5060 SIP/2.0
|
||||||
|
Via: SIP/2.0/UDP {mac_local_ip}:5060;rport;branch={branch}
|
||||||
|
From: <sip:{sender_sip}@{sender_ip}:5060>;tag={tag}
|
||||||
|
To: <sip:{receiver_sip}@{receiver_ip}:5060>
|
||||||
|
Call-ID: {call_id}@{mac_local_ip}
|
||||||
|
CSeq: 20 MESSAGE
|
||||||
|
Content-Type: text/plain
|
||||||
|
Max-Forwards: 70
|
||||||
|
User-Agent: DnakeVoip v1.0
|
||||||
|
Content-Length: {content_length}
|
||||||
|
|
||||||
|
{xml_body}'''
|
||||||
|
|
||||||
|
# 替换换行符为\r\n(确保符合SIP协议)
|
||||||
|
sip_message = sip_message.replace('\n', '\r\n')
|
||||||
|
|
||||||
|
try:
|
||||||
|
# 创建UDP socket并发送消息
|
||||||
|
with socket.socket(socket.AF_INET, socket.SOCK_DGRAM) as sock:
|
||||||
|
# 设置超时时间(5秒)
|
||||||
|
sock.settimeout(5)
|
||||||
|
# 发送消息
|
||||||
|
sock.sendto(sip_message.encode('utf-8'), (target_ip, target_port))
|
||||||
|
print(f"已发送SIP消息到 {target_ip}:{target_port}")
|
||||||
|
print("\n发送的消息内容:")
|
||||||
|
print(sip_message)
|
||||||
|
|
||||||
|
# 尝试接收响应(可选)
|
||||||
|
try:
|
||||||
|
response, addr = sock.recvfrom(4096)
|
||||||
|
print(f"\n收到来自 {addr} 的响应:")
|
||||||
|
print(response.decode('utf-8', errors='ignore'))
|
||||||
|
except socket.timeout:
|
||||||
|
print("\n未收到响应(可能对方未回复或网络延迟)")
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
print(f"发送失败:{str(e)}")
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
send_sip_message()
|
||||||
|
```
|
||||||
|
|
||||||
|
app.py
|
||||||
|
|
||||||
|
```python
|
||||||
|
from fastapi import FastAPI, Query
|
||||||
|
import httpx
|
||||||
|
import asyncio
|
||||||
|
import uvicorn
|
||||||
|
from main import send_sip_message
|
||||||
|
|
||||||
|
app = FastAPI()
|
||||||
|
|
||||||
|
async def main():
|
||||||
|
"""主函数内容"""
|
||||||
|
print("执行main函数...")
|
||||||
|
# 执行发送SIP消息
|
||||||
|
send_sip_message()
|
||||||
|
return {"message": "main function executed - SIP message sent"}
|
||||||
|
|
||||||
|
async def test():
|
||||||
|
"""测试函数内容"""
|
||||||
|
print("执行test函数...")
|
||||||
|
# 在这里添加你的test函数逻辑
|
||||||
|
return {"message": "test function executed"}
|
||||||
|
|
||||||
|
@app.api_route("/api/router", methods=["GET", "POST", "PUT", "DELETE", "PATCH", "HEAD", "OPTIONS"])
|
||||||
|
async def router(path: str = Query(...)):
|
||||||
|
"""路由处理器"""
|
||||||
|
try:
|
||||||
|
# 根据path参数执行不同的函数
|
||||||
|
if path == "floor":
|
||||||
|
result = await main()
|
||||||
|
elif path == "test":
|
||||||
|
result = await test()
|
||||||
|
else:
|
||||||
|
return {"error": f"Unknown path: {path}"}
|
||||||
|
|
||||||
|
return {"success": True, "data": result}
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
return {"success": False, "error": str(e)}
|
||||||
|
|
||||||
|
@app.api_route("/", methods=["GET", "POST", "PUT", "DELETE", "PATCH", "HEAD", "OPTIONS"])
|
||||||
|
async def root():
|
||||||
|
"""健康检查端点"""
|
||||||
|
return {"message": "FastAPI server is running"}
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
# 监听所有IPv6地址的3001端口
|
||||||
|
uvicorn.run(
|
||||||
|
"app:app",
|
||||||
|
host="::", # 监听所有IPv6地址
|
||||||
|
port=3001,
|
||||||
|
reload=True
|
||||||
|
)
|
||||||
|
```
|
||||||
Reference in New Issue
Block a user