|
www.lissyara.su
—> статьи
—> FreeBSD
—> Мелочи
—> Видеонаблюдение
Бюджетный вариант видеонаблюдения на удаленном объекте
Автор: BAV_Lug.
У компании, где я сейчас работаю, несколько объектов по стране. Естественно не хватает контроля со стороны руководства за персоналом этих объектов. Было принято решение о установке видеонаблюдения, с несколькими условиями:
- бюджет минимальный
- возможность удаленного просмотра в режиме on-line
- возможность просмотра архива, за определенный период
Обычные системы видеонаблюдения по выше перечисленным условиям мне не подошли (в основном из-за первого пункта). Выбор пал на WEB камеры Dlink DCS-900. Это камеры имеющие Ethernet-интерфейс. В целом камеры неплохие, за свои деньги, но у них есть большой недостаток - отсутствие вменяемого ПО. Одну такую камеру я взял на тесты почти год назад, но столкнувшись, с отсутствием нормального ПО, отложил ее до лучших времен.
И вот это время настало - руководство сказало: "Срочно ставь!"
Поехал поставил камеры и начал думать, что же дальше с ними делать.
1. Просмотр on-line
Из-за того, что на объектах ширина канала обычно не более 512k, то on-line видео явно не посмотришь. Решил сделать WEB-страничку на которой просто менялась бы картинка с камеры (например 1 раз 5 секунд), в принципе для того, чтобы понять что происходит на объекте этого достаточно. Камеры умеют отдавать картинку по http и ftp (могут скидывать картинку через определенный интервал на ftp сервер). Выбор пал на http с авторизацией по логину/паролю, благо камера это позволяет.
Сразу оговорюсь, что все приведенные ниже листинги, я уверен, можно было написать намного красивее. Но я не программист, а всего лишь админ (к тому же c JS я столкнулся впервые). А потому прошу сильно не пинать.
На сервере объекта был развернут apache+php. И написан следующий код (все было написано под две камеры, но легко расширяется на большее количество)
cam1.php cam2.php
<?php
$login = 'xxxxx';
$passw = 'xxxxx';
$authBasic = 'Authorization: Basic '. base64_encode($login .':'. $passw);
$opts = array
(
'http' => array
(
'method' => 'GET',
'header' => $authBasic ."\r\n"
)
);
$context = stream_context_create($opts);
$file = file_get_contents('http://адрес камеры/image.jpg', false, $context);
header('Content-type: image/jpeg', true);
header('Expires: Mon, 26 Jul 1997 05:00:00 GMT');
header('Last-Modified: ' . gmdate("D, d M Y H:i:s") . " GMT");
header('Cache-Control: no-cache, must-revalidate');
header('Pragma: no-cache');
echo $file;
?>
| index.html
<html>
<head>
<script type="text/javascript">
var timei = 30000;
var timeint;
var camname = "cam1.php?";
var count = 1;
function checktime(timeimg)
{
if (timeimg==0)
{
clearInterval(timeint);
}
else
{
timei = timeimg;
clearInterval(timeint);
init();
}
}
function checkcam(camera)
{
camname = camera;
reloadImage();
}
function reloadImage()
{
img5 = document.getElementById("webimg");
img5.src = camname + count;
count = count + 1;
}
function init()
{
timeint = setInterval("reloadImage()",timei);
}
</script>
</head>
<body>
<table>
<th COLSPAN=2>
</th>
<tr>
<td>
Камера <br />
<form>
<input type="radio" name="camera" onclick="checkcam(this.value)"
value="cam1.php?" CHECKED>1<br />
<input type="radio" name="camera" onclick="checkcam(this.value)"
value="cam2.php?">2
</form>
</td>
<td ROWSPAN=3>
<img src="cam1.php?0" id="webimg" />
</td>
</tr>
<tr>
<td>
Интервал обновления
<form>
<input type="radio" name="timeimg" onclick="checktime(this.value)"
value="0" CHECKED> Без обновления<br />
<input type="radio" name="timeimg" onclick="checktime(this.value)"
value="3000">3 сек<br />
<input type="radio" name="timeimg" onclick="checktime(this.value)"
value="5000">5 сек<br />
<input type="radio" name="timeimg" onclick="checktime(this.value)"
value="10000">10 сек<br />
<input type="radio" name="timeimg" onclick="checktime(this.value)"
value="20000">20 сек<br />
<input type="radio" name="timeimg" onclick="checktime(this.value)"
value="30000">30 сек<br />
<input type="radio" name="timeimg" onclick="checktime(this.value)"
value="60000">1 мин<br />
</form>
</td>
</tr>
<tr>
<td>
</td>
</tr>
</table>
</body>
</html>
| Вот так это выглядит
2. Ведение архива
Для создания архива я воспользовался возможностью камер передавать снимки по ftp. Настроил их так, чтобы они каждые 15 секунд сбрасывали снимок на локальный ftp сервер объекта. Затем ночью (во время минимальной загрузки канала эти снимки забираются на сервер головного офиса).
Снимки передаются с камер с названием, например 2009041510131001.jpg, где первые 4 символа - год, 2 символа - месяц, 2 символа - день, 2 символа - час, 2 символа - минуты, 2 символа - секунды и еще 2 - это номер кадра в текущую секунду.
Так как кадров за день получается очень много, то для ускорения последующей работы, используется следующий скрипт для раскладывания их по папкам - дням (спасибо zg за подсказку как более правильно написать этот скрипт).
#!/bin/sh
SDIR="/home/web/usr/home/arhivwebcam/cam"
DDIR="/home/web/usr/home/arhivwebcam/sort/cam"
for i in 1 2
do
ls -1 "$SDIR$i" | awk '{print substr($1,1,4), substr($1,5,2), substr($1,7,2), $1}' \
| while read YEAR MONTH DAY FILE
do
DIR=$DDIR$i/$YEAR/$MONTH/$DAY
[ -d "$DIR" ] || mkdir -p "$DIR"
mv $SDIR$i/$FILE $DIR/$FILE
done
done
| Где SDIR это папки в которые изначально складываются снимки полученные с фтп сервера объекта.
Ну и написан следующий код для просмотра этого архива.
<?php
$date_t = getdate();
$month_name = array(1 => "Январь", "Февраль", "Март",
"Апрель", "Май", "Июнь", "Июль",
"Август", "Сентябрь", "Октябрь",
"Ноябрь", "Декабрь");
$time_name = array(7 => "07:00-08:00", "08:00-09:00", "09:00-10:00",
"10:00-11:00", "11:00-12:00", "12:00-13:00",
"13:00-14:00", "14:00-15:00", "15:00-16:00",
"16:00-17:00", "17:00-18:00", "18:00-19:00",
"19:00-20:00", "20:00-21:00", "21:00-22:00");
$cam_dir = array(1 => "sort/cam1", "sort/cam2");
$year = $_GET[year];
if ($year != "")
{
$month = sprintf("%02d", $_GET[month]);
$day = sprintf("%02d", $_GET[day]);
$time = sprintf("%02d", $_GET[time]);
$cam = $_GET[cam];
$dirt = $cam_dir[$cam]."/".$year."/".$month."/".$day."/";
$printerror = "";
if (! @chdir($dirt))
{
$printerror = " Not archive for this time";
};
$listjpg = glob($year.$month.$day.$time."*.jpg");
}
?>
<html>
<head>
<script type="text/javascript">
imgslide=new Array();
var count = -1;
var timeint;
<?php
$count_img = count($listjpg);
echo "var img_max = $count_img;";
if ($count_img > 0)
{
for ($i = 0; $i < $count_img; $i++)
{
echo "imgslide[$i]=new Image()"."\n";
echo "imgslide[$i].src=\"$dirt$listjpg[$i]\""."\n";
}
}
?>
function viewImage()
{
img5 = document.getElementById("webimg");
img5.src = imgslide[count].src;
time_o = document.getElementById("time_cadr");
time_out = imgslide[count].src;
l = time_out.length;
h = time_out.substring((l-12),(l-10));
m = time_out.substring((l-10),(l-8));
s = time_out.substring((l-8),(l-6));
time_o.innerHTML = h + ":" + m + ":" + s;
}
function reloadImage()
{
if (count != img_max)
{
count = count + 1;
viewImage()
if (count == (img_max-1))
{
clearInterval(timeint);
}
}
}
function init()
{
if (img_max > 0)
{
timeint = setInterval("reloadImage()",750);
}
}
function pause()
{
clearInterval(timeint);
}
function play()
{
if (img_max > 0)
{
if (count >= (img_max-1))
{
count = 0;
}
pause();
init();
}
}
function play_max()
{
if (img_max > 0)
{
if (count >= (img_max-1))
{
count = 0;
}
pause();
timeint = setInterval("reloadImage()",400);
}
}
function cadr_down()
{
if (img_max > 0)
{
pause();
if (count == 0)
{
count = img_max;
}
count = count - 1;
viewImage();
}
}
function cadr_up()
{
if (img_max > 0)
{
pause();
if (count >= (img_max-1))
{
count = 0;
}
count = count + 1;
viewImage();
}
}
</script>
</head>
<body onload="init()">
<table>
<tr>
<td>
<form>
<select name="day">
<?php
for ($i = 1; $i <= 31; $i++)
{
echo "<option value=\"$i\" ";
if (($i == $day) || (($day == "") && ($i == $date_t[mday])))
{
echo "selected";
}
echo ">$i</option>"."\n";
}
?>
</select>
<select name="month">
<?php
for ($i = 1; $i <= 12; $i++)
{
echo "<option value=\"$i\" ";
if (($i == $month) || (($month == "") && ($i == $date_t[mon])))
{
echo "selected";
}
echo ">$month_name[$i]</option>"."\n";
}
?>
</select>
<select name="year">
<?php
for ($i = 2009; $i <= 2015; $i++)
{
echo "<option value=\"$i\" ";
if (($i == $year) || (($month == "") && ($i == $date_t[year])))
{
echo "selected";
}
echo ">$i</option>"."\n";
}
?>
</select>
Время
<select name="time">
<?php
for ($i = 7; $i <= 21; $i++)
{
echo "<option value=\"$i\" ";
if (($i == $time) || (($time == "") && ($i == $date_t[hours])))
{
echo "selected";
}
echo ">$time_name[$i]</option>"."\n";
}
?>
</select>
Камера
<select name="cam">
<?php
for ($i = 1; $i <= 2; $i++)
{
echo "<option value=\"$i\" ";
if ($i == $cam)
{
echo "selected";
}
echo ">$i</option>"."\n";
}
?>
</select>
<input type="submit" value="Просмотр">
<?php
echo $printerror;
?>
</form>
</td>
</tr>
<tr>
<td>
<img src="black.jpg" id="webimg" />
</td>
</tr>
<tr>
<td>
<button onclick="play()"> > </button>
<button onclick="play_max()">->></button>
<button onclick="pause()"> || </button>
<button onclick="cadr_down()"><-</button>
<button onclick="cadr_up()">-></button>
<span id="time_cadr"></span>
</td>
</tr>
</table>
</body>
</html>
| Вот, что получилось
размещено: 2009-07-29,
последнее обновление: 2009-07-29,
автор: BAV_Lug
|
|
Комментарии пользователей [17 шт.]