PHP怎么用海康威视AI视频分析_推送流地址获行为检测结果【方案】

海康威视AI分析结果推送流是基于HTTP SSE的JSON文本事件流,非RTSP视频流;PHP需用cURL长连接配合超时禁用、自定义WRITEFUNCTION回调逐行解析,并做字段容错与进程守护。

海康威视AI分析结果推送流是HTTP长连接,不是RTSP

很多人一上来就用 ffplayffmpeg -i rtsp://... 去接“推送流地址”,发现根本连不上——因为海康的AI行为检测结果推送走的是 HTTP SSE(Server-Sent Events) 协议,不是视频流。你拿到的那个“推送流地址”形如 http://192.168.1.64/ISAPI/AI/behaviorAnalysis/pushStream?channel=1&streamType=0,它返回的是持续不断的 JSON 文本事件流,不是二进制视频帧。

所以第一步必须明确:这不是用来播画面的,而是用来收结构化数据的。

PHP用cURL保持长连接读取SSE流要禁用超时和缓冲

PHP默认的 curl_exec() 会等完整响应结束才返回,但SSE是永不断开的流式响应。你需要手动配置cURL为“不等待结束、边收边处理”:

  • CURLOPT_TIMEOUT 必须设为 0(永不超时)或足够大(如 300
  • CURLOPT_CONNECTTIMEOUT 可设为 10,只控制建连阶段
  • CURLOPT_RETURNTRANSFER 设为 false,让数据直接输出到回调函数
  • CURLOPT_WRITEFUNCTION 指定自定义回调,逐行解析 data: {...}
  • ob_end_flush()flush() 在CLI模式下通常无效,别白费劲
function handleSseLine($ch, $data) {
    if (strpos($data, 'data: ') === 0) {
        $json = trim(substr($data, 6));
        if (!empty($json) && $j

son !== '}') { // 海康偶尔发单个 } $event = json_decode($json, true); if (json_last_error() === JSON_ERROR_NONE && !empty($event['channelID'])) { echo "[DETECT] " . $event['behaviorType'] . " at " . date('Y-m-d H:i:s') . "\n"; // 这里写入数据库、触发告警、调用Webhook等 } } } return strlen($data); } $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, 'http://192.168.1.64/ISAPI/AI/behaviorAnalysis/pushStream?channel=1&streamType=0'); curl_setopt($ch, CURLOPT_USERPWD, 'admin:12345'); // 注意:海康默认需要基础认证 curl_setopt($ch, CURLOPT_HTTPAUTH, CURLAUTH_BASIC); curl_setopt($ch, CURLOPT_TIMEOUT, 0); curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10); curl_setopt($ch, CURLOPT_RETURNTRANSFER, false); curl_setopt($ch, CURLOPT_WRITEFUNCTION, 'handleSseLine'); curl_setopt($ch, CURLOPT_HEADER, false); curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); curl_exec($ch); curl_close($ch);

海康推送JSON字段不稳定,必须做容错校验

实测中,同一台设备在不同固件版本下,behaviorType 可能是 "peopleClimb""climb" 或空字符串;timeStamp 字段可能不存在,或格式为毫秒时间戳(13位整数)或ISO字符串;部分事件甚至不带 objects 数组。硬解析会崩。

关键防护点:

  • 每次 json_decode 后必须检查 json_last_error()
  • 访问字段前一律用 isset()array_key_exists(),别直接 $event['behaviorType']
  • 时间字段优先 fallback 到 date('U')(当前秒级时间),避免因字段缺失导致逻辑中断
  • 把原始 $json 字符串存一份到日志文件,方便后期比对设备行为

PHP常驻进程需配合systemd或supervisord管理,不能裸跑

这个cURL长连接一旦断开(网络抖动、设备重启、防火墙中断),PHP脚本就退出了,没人自动拉起。裸跑 php detector.php 在终端关闭后也会终止。

生产环境必须用进程守护:

  • Linux下推荐用 systemd 写 service 文件,设置 Restart=alwaysRestartSec=5
  • 若用 supervisord,注意 autorestart=true + startsecs=10(给SSE连接留够建连时间)
  • 脚本开头加 declare(ticks=1); pcntl_signal(SIGTERM, function(){ exit(0); });,支持优雅退出
  • 避免用 nohup php detector.php & —— 它无法自动恢复崩溃,且stdout/stderr易丢失

海康的推送流没有心跳保活机制,实际运行中每小时左右大概率断连一次,靠守护进程自动重连是最省心的做法。