Skip to content

Commit

Permalink
Merge branch 'master' into beta
Browse files Browse the repository at this point in the history
  • Loading branch information
Jezza34000 authored Jun 20, 2024
2 parents 1a1b558 + d7b97f7 commit 359c894
Show file tree
Hide file tree
Showing 7 changed files with 231 additions and 41 deletions.
7 changes: 3 additions & 4 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -43,10 +43,9 @@ Vous pouvez tester si les paramètres sont correctes et si le plugin arrive à a
* Générez les commandes de la caméra : bouton ![image](https://user-images.githubusercontent.com/54839700/174434177-5f433dc2-91b5-4cdb-9bcc-8a42d6a48f0d.png)


**!! NB TRES IMPORTANT !! :**
<BR>
Lors de mise(s) à jour du plugin Reolink, de nouvelles commandes peuvent être ajouter (cf. Changelog du plugin) pour faciliter une meilleure gestion de la caméra.
**Pour bénéficier des nouvelles commandes, relancer le processus de création de commandes pour chacun de la (des) caméra(s)**
> IMPORTANT : Lors de mise(s) à jour du plugin Reolink, de nouvelles commandes peuvent être ajouter (cf. Changelog du plugin) pour faciliter une meilleure gestion de la caméra. **Pour bénéficier des nouvelles commandes, relancer le processus de création de commandes pour chacun de la (des) caméra(s)**
> ATTENTION : La remontée de détection de mouvement depuis la caméra vers le plugin n'est pas fiable, et ne dois aucunement être utilisé en tant que système de sécurité.
<BR><P>

Expand Down
14 changes: 14 additions & 0 deletions changelog.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,19 @@
# Change log

10/03/2021
- Ajout de nouvelles détection depuis la souscrption ONVIF
- Amélioration communication entre daemon et plugin
Merci @t0urista

21/12/2023
- Update vulnerable package in requirements.txt

28/10/2022
- Bug fix RefreshInfo

18/10/2022
- Bug fix API (is_loggedin)

27/07/2022
- Bug fix for Atlas Jeedom (py wheel)

Expand Down
96 changes: 96 additions & 0 deletions core/config/reolinkapicmd.json
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,102 @@
"isVisible": 1,
"logicalId": "MdState"
},


{
"abilityneed":"alarmMd",
"name": "event Motion (etat)",
"type": "info",
"subtype": "binary",
"IsHistorized": 0,
"isVisible": 1,
"logicalId": "EvMotion"
},
{
"abilityneed":"alarmMd",
"name": "event FaceDetect (etat)",
"type": "info",
"subtype": "binary",
"IsHistorized": 0,
"isVisible": 1,
"logicalId": "EvFaceDetect"
},
{
"abilityneed":"alarmMd",
"name": "event PeopleDetect (etat)",
"type": "info",
"subtype": "binary",
"IsHistorized": 0,
"isVisible": 1,
"logicalId": "EvPeopleDetect"
},
{
"abilityneed":"alarmMd",
"name": "event VehicleDetect (etat)",
"type": "info",
"subtype": "binary",
"IsHistorized": 0,
"isVisible": 1,
"logicalId": "EvVehicleDetect"
},
{
"abilityneed":"alarmMd",
"name": "event DogCatDetect (etat)",
"type": "info",
"subtype": "binary",
"IsHistorized": 0,
"isVisible": 1,
"logicalId": "EvDogCatDetect"
},
{
"abilityneed":"alarmMd",
"name": "event MotionAlarm (etat)",
"type": "info",
"subtype": "binary",
"IsHistorized": 0,
"isVisible": 1,
"logicalId": "EvMotionAlarm"
},
{
"abilityneed":"alarmMd",
"name": "event Visitor (etat)",
"type": "info",
"subtype": "binary",
"IsHistorized": 0,
"isVisible": 1,
"logicalId": "EvVisitor"
},

{
"abilityneed":"alarmMd",
"name": "last Onvif Event (full) ",
"type": "info",
"subtype": "string",
"IsHistorized": 0,
"isVisible": 1,
"logicalId": "EvLastOnvifFull"
},

{
"abilityneed":"alarmMd",
"name": "last Onvif Event (name) ",
"type": "info",
"subtype": "string",
"IsHistorized": 0,
"isVisible": 1,
"logicalId": "EvLastOnvifName"
},

{
"abilityneed":"alarmMd",
"name": "last Onvif Event (etat)",
"type": "info",
"subtype": "binary",
"IsHistorized": 0,
"isVisible": 1,
"logicalId": "EvLastOnvifState"
},

{
"abilityneed":"performance",
"name": "Débit codec (Kbps)",
Expand Down
31 changes: 26 additions & 5 deletions core/php/jeeReolink.php
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,10 @@
die();
}

if (isset($result['message']) && $result['message'] == "motion") {
# BEGIN modified by t0urista to handle ONVIF events
if (isset($result['message']) && (($result['message']=="motion") || (strpos($result['message'], 'Ev') !== false) )) {
# END modified by t0urista to handle ONVIF events

$plugin = plugin::byId('reolink');
$eqLogics = eqLogic::byType($plugin->getId());

Expand All @@ -31,16 +34,34 @@
}

if ($camera_ip == $result['ip']) {
log::add('reolink', 'debug', 'Evènement MotionState reçu depuis le daemon. Cam IP='.$result['ip'].' état='.$result['motionstate']);
$eqLogic->checkAndUpdateCmd('MdState', $result['motionstate']);
if ($result['message'] == "motion") {
log::add('reolink', 'debug', 'Cam IP='.$result['ip']. ' Onvif event reçu depuis le daemon. name= MDstate, état='.$result['motionstate']);
$eqLogic->checkAndUpdateCmd('MdState', $result['motionstate']);
}

# BEGIN added by t0urista to handle ONVIF events

# catch any ONVIF events in the 3 genreic ONVIF commands, can cover unknown ONVIF events
$eqLogic->checkAndUpdateCmd('EvLastOnvifName', $result['message']);
$eqLogic->checkAndUpdateCmd('EvLastOnvifState', $result['motionstate']);
$eqLogic->checkAndUpdateCmd('EvLastOnvifFull', $result['message'] . '-' . $result['motionstate']);

# catch all pre-defined ONVIF events with their dedicated commands, does only cover knwon ONVIF events

if (strpos($result['message'], 'Ev') !== false) {
log::add('reolink', 'debug', 'Cam IP='.$result['ip']. ' Onvif event reçu depuis le daemon. name= ' . $result['message'] . ', etat='.$result['motionstate']);
$eqLogic->checkAndUpdateCmd($result['message'], $result['motionstate']);
}
# END added by t0urista to handle ONVIF events

#log::add('reolink', 'debug', 'IP : ' . $camera_contact_point . ' / IsCamAI : ' . $camera_AI . ' / EqId : ' . $EqId . ' / Channel : ' . $channel);
if ($camera_AI == "Oui") {
$camcnx = reolink::getReolinkConnection($eqLogic->getId());
$channel = $eqLogic->getConfiguration('channelNum') - 1;
$res = $camcnx->SendCMD('[{"cmd":"GetAiState","action":0,"param":{"channel":'.$channel.'}}]');
if (isset($res[0]['value'])) {
$eqLogic->checkAndUpdateCmd('MdPersonState', $res[0]['value']['people']['alarm_state']);
$eqLogic->checkAndUpdateCmd('MdVehicleState', $res[0]['value']['vehicle']['alarm_state']);
$eqLogic->checkAndUpdateCmd('EvPeopleDetect', $res[0]['value']['people']['alarm_state']);
$eqLogic->checkAndUpdateCmd('EvVehicleDetect', $res[0]['value']['vehicle']['alarm_state']);
}
log::add('reolink', 'debug', 'Cam AI : Evènements Motion | Personne : ' . $res[0]['value']['people']['alarm_state'] . ' / Vehicule : ' . $res[0]['value']['vehicle']['alarm_state']);
}
Expand Down
6 changes: 6 additions & 0 deletions renovate.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
{
"$schema": "https://docs.renovatebot.com/renovate-schema.json",
"extends": [
"config:recommended"
]
}
116 changes: 85 additions & 31 deletions resources/demond/camhook.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,49 +4,103 @@
import re

try:
from jeedom.jeedom import *
from jeedom.jeedom import *
except ImportError:
print("Error: importing module jeedom.jeedom")
sys.exit(1)
print("Error: importing module jeedom.jeedom")
sys.exit(1)

try:
f = open('jeedomcreds', 'r')
_callback = f.readline().rstrip("\n")
_apikey = f.readline().rstrip("\n")
f.close()
f = open('jeedomcreds', 'r')
_callback = f.readline().rstrip("\n")
_apikey = f.readline().rstrip("\n")
f.close()
except:
logging.error(f"Unable to read credentials jeedom file, retry...")
sys.exit(1)
logging.error(f"Unable to read credentials jeedom file, retry...")
sys.exit(1)

try:
os.remove("jeedomcreds")
os.remove("jeedomcreds")
except:
pass
pass

jeedom_cnx = jeedom_com(_apikey, _callback)
detect_state = 0
# BEGIN added by t0urista to handle ONVIF events
eventTable = {}
# END added by t0urista to handle ONVIF events
app = FastAPI()


@app.post("/inbound_events", status_code=200)
async def get_body(request: Request):
global detect_state
ip = request.client.host
logging.debug(f"Incoming XML camera event on webhook from IP={ip}")
xml_answer = await request.body()

new_detect_state = 0
if re.search('IsMotion" Value="true"', xml_answer.decode('utf-8')):
new_detect_state = 1

if detect_state != new_detect_state:
detect_state = new_detect_state
send_frame = {
"message": "motion",
"ip": ip,
"motionstate": detect_state
}
# convert into JSON:
message = json.dumps(send_frame)
jeedom_cnx.send_change_immediate(json.loads(message))
return

global detect_state

# BEGIN added by t0urista to handle ONVIF events
global eventTable
# END added by t0urista to handle ONVIF events


ip = request.client.host
logging.debug(f"Incoming XML camera event on webhook from IP={ip}")
xml_answer = await request.body()

new_detect_state = 0
if re.search('IsMotion" Value="true"', xml_answer.decode('utf-8')):
new_detect_state = 1
if detect_state != new_detect_state:
detect_state = new_detect_state
send_frame = {
"message": "motion",
"ip": ip,
"motionstate": detect_state
}
# convert into JSON:
message = json.dumps(send_frame)
jeedom_cnx.send_change_immediate(json.loads(message))


# BEGIN added by t0urista to handle ONVIF events
pattern=r'<wsnt:.*?ConcreteSet">(.*?)</wsnt:Topic>.*?<tt:Data>.*?Value="(.*?)" /></tt:Data>'
event_status_all=re.findall(pattern,xml_answer.decode('utf-8'))

# logging.debug(event_status_all)
for event_status in event_status_all:
# logging.debug(event_status)
event=re.findall(r'tns1:.*/(.*)', event_status[0])
event="Ev" + event[0]
# logging.debug(event[0])
status=0
if(event_status[1]!="false"):
status=1
logging.debug(f"received event {event} {status}")
eventFound=0
for eventName in eventTable:
# print(eventName)
if (eventName==event):
eventFound=1
if (eventTable[eventName]!=status):
logging.debug(f"{event} --> {status}")
eventTable[eventName]=status
send_frame = {
"message": event,
"ip": ip,
"motionstate": status
}
# convert into JSON:
message = json.dumps(send_frame)
jeedom_cnx.send_change_immediate(json.loads(message))
if (eventFound==0):
eventTable[event] = status
logging.debug(f"{event} --> {status}")
send_frame = {
"message": event,
"ip": ip,
"motionstate": status
}
# convert into JSON:
message = json.dumps(send_frame)
jeedom_cnx.send_change_immediate(json.loads(message))
# END added by t0urista to handle ONVIF events

return
2 changes: 1 addition & 1 deletion resources/requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -5,4 +5,4 @@ aiohttp>=3.9.1
aiosignal~=1.3.1
fastapi~=0.111.0
uvicorn~=0.30.1
urllib3~=2.2.2
urllib3~=2.2.2

0 comments on commit 359c894

Please sign in to comment.