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

海康威视AI分析结果推送流是HTTP长连接,不是RTSP
很多人一上来就用 ffplay 或 ffmpeg -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) && $json !== '}') { // 海康偶尔发单个 }
$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 数组。硬解析会崩。
立即学习“PHP免费学习笔记(深入)”;
关键防护点:
- 每次
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=always和RestartSec=5 - 若用
supervisord,注意autorestart=true+startsecs=10(给SSE连接留够建连时间) - 脚本开头加
declare(ticks=1); pcntl_signal(SIGTERM, function(){ exit(0); });,支持优雅退出 - 避免用
nohup php detector.php &—— 它无法自动恢复崩溃,且stdout/stderr易丢失
海康的推送流没有心跳保活机制,实际运行中每小时左右大概率断连一次,靠守护进程自动重连是最省心的做法。











