-
-
Notifications
You must be signed in to change notification settings - Fork 16
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Question about media platform. #9
Comments
@xannor thank you very much for reaching out and your contributions to the projects on which this library is based on! I was not planning on adding the media platform on the short term, but I would be more than happy to review and accept PRs (both in this library and in HomeAssistant) if you are willing to work on it. Your idea sounds good and was indeed already pretty much incoporated in the fwesterberg custom integration. |
I agree, I did not like it either, but "live" thumbs on a raspberry at the time were much worse, as FFMPEG was incredibly slow. But good to know, I'll first tool around with an "extension" integration to start with, and see if I can piggyback off the the official. Then when things are stable we can see about PR's to push parts over. |
I started working on this and I have a couple of questions.
|
@xannor good to hear you have started to work on it!
|
As for the DUO, it reports channels like an NVR, if you need and debug dumps from it I can provide them, and possibly answer some questions about it as that is what I primarily tested and developed against. |
@xannor did you setup the DUO 1st gen in the new build in HomeAssistant integration? |
@xannor about the time, if you now the time_diffrence in seconds between the utc.now() on HomeAssistant and the camera internal time you can always calculate the time you need. Just take the datetime from the recording and add/substract the time diffrence in seconds, to get UTC on the HomeAssistant device. Normally this time diffrence will only be a second or so, but if the camera timezone is set incorrectly lets say 2 hours in front the timediffrence will become 7200 seconds (two hours). But I think the conversion/correction to UTC timezone schould happen inside the reolink-aio library in the I have not yet looked at the |
Also cleaned up get_vod_source method a bit in this commit: 80affcd |
Time: DUO: --Edit: Also, this is a minor thing, but it surprised me. I often reset the DUO, so I have not "initialized it" and your integration requires a password, so the default admin/no password is impossible to use. Not a big issue, but it is something (you could even use that as an initialization approach and require a password to be setup.) Also, I have not tested this, but there was an issue reported in my CI repo about some special characters not working, possibly due to the default url encoding. I dont know if this is something mentioned or tested in your integration, or just something I did wrong in mine. |
I have found something to add though, for channel capabilities, recDownload and recReplay. These should show if a channel is allowed to use the Download action or if it needs to replay via url (which I think may be less optimal in most cases.) I see you have code to get the vod_source, but not to download the file direct (3.6.6 in the api doc) I know the browser interface does the download call, and I think we can "stream" the download. This may work around the rtmp HD issue as well. |
I started going down the path of using a Calendar as the "entity" managing the search (it mostly fist concept wise.) and in the process I re-build my tz code from my other integration. If you are curios about and/or want to critique it: DateTime utils Seems to work so far but it is still very much experimental. |
If we use the download option, where does the file get downloaded to? |
From the API Docks the download action just provides the MP4 as a data stream over http as apposed to pushing through one of the rtmp/rtsp servers. Since the the files are complete I figure this would be better on the camera. |
Have you tried this with HomeAssistant? I am starting to think it would be wise to have a configuration option to specify the |
I have not tried the download command from the API yet, but I have used the replay through RTMP, and I am hoping the download method will solve some of the issues I found.
I dont plan on actually downloading full files to the HA install, just streaming the download through the camera stream service. I am hoping this would both load faster and provide the ability to seek the playback, in addition to providing the full quality, and possibly the ability of the user to save the playback somewhere if they choose. |
OffTopic> On the topic front, I am learning to "Decode" the vod filenames, I believe they have the motion sensor data encoded in them, and I can use that in the search to "client side" filter and display the triggers for the file. Unfortunatly I can only get filenames for pet/person/motion/and timer, I dont have a camera in a location that can do vehicle detection, and I dont have anything that supports the other types of detection in the API (face, animal, etc). I think the format has space for those types but I cannot verify. |
ran into a snag with your library. You code assumes that if a VOD search returns no files but files were requested, it is an error. I think the cameras (at least the duo I am testing with) will also return no files and just a status if there are too many results. While technically correct, it also eats the Status entry so I cannot trap and work around the error (i.e. know which days of the month do contain files and split my request up). If you could either not throw when no file results and pass an empty list, or throw a custom error with the status attached that I can trap, that would help "Work around" this issue. Update: |
@xannor thanks for figuring out this bug, my NVR always returns a (very long) list of recordings, but if I chose a time-intervall where there are no recordings I would indeed get the error as you posted. I have fixed by simply return a empty list. Besides I already did some of the parsing in this library using the new VOD_file class. Please have a look at it, and add properties if you need more in a PR. My simple test code:
If your DUO cam does not return any results if there schould be results because there are too many, please make some additonal code that re-requests the vod_files in chunks and stitches them together afterwords. |
Please add this decoding to the new VOD_file class such that we have a simple property (preferably a ENUM) that indicates the type of the recording, we can default to unkown if not available. |
I looked over your code, and even though you mean well, your simplified UTC code has issues. Mostly, it does not handle daylight savings time, so a camera in a DST zone would be an hour off half the year if the VODs looked up were recorded in before/after a time change. This is because the camera always stores and reports in local time. This is why I went through the trouble of parsing the DST field so I can use the actual TZ info of the camera along with the date/time of the file to determine the actual local time of the file. For the "flags" because a video can be flagged as multiple "triggers" I am looking into if I should use a intflag or an array of enums, a VOD could have multiple, even potentially none. defaulting to an empty file list is fine, though I think there is a meaning difference in the API between no files and a missing files. I have noticed that when my search range exceeds the camera history, it would just report the statuses (i.e. as a hint of how far back you can search) even when the range would include files. |
Some feedback on your VOD_file class, What is playback time? I have not seen that field on my camera, I assume it is an NVR thing (I dont have one of those) Pulling that field, when it is not present would error. Also, why are you taking a dict[str,any] instead of using your SearchFile typeddict? |
I also did some basic work to continue with the files improvements. Its not ready to submit as I am still working on the filename decoding (and I had to do some other cleanup to handle your recent changes.) hopefully I can have something to push this weekend. I did get the calendar integration basically working with the changes in this fork |
I did check for daylight savings time, and it apear to work, the
Ah right, did not consider that.
I never got back a empty file list from my NVR, did you? |
Actually I am not entirely sure yet, it is another timestamp, I think a motion recording (short recording 1 min or so) will have the playback time refering to the start of the bigger 1 hour recording file of which it is part, or something simular, but have not figured out what it actually means. It was just in the return data from the NVR.
We schould default it to None when it is not present, can be done in a followup PR.
Yea that schould probably indeed be the SearchFile typing, but I just wanted to provide some basics to you of what I had in mind. I did not check if the SearchFile typing is consistant with the NVR response. |
Be carefull relying on "it works on the NVR" it is better hardware than the cameras (and tends to have more bells an whistles) so it may do things the cameras cant. I.E. store in UTC and adjust to local (I dont own an NVR so I dont know what it is fully capable of, I can only guess/assume). The cameras I have tested with only seem to store in local time so changing the time or DST enabled does not seem to affect existing recordings, only new ones. The code I provided just implements their DST rules as a python timezone, with that you can convert the camera localtime to any time.
I would prefer to at least use enums in a list as I dont have enough sources to test and determine what all the flags values could be, and there could be more in the future. So far I can verify Timer Motion Pet and Person. I have a couple of cameras that support vehicle, but are not in a location that has any to detect. I know in the API there is also animal (maybe the Trax does this) and face. The only issue with intflags is, depending in the python version. 0 could mean none or all, and I think it may be possible to get a recording with no triggers, not sure how but I have gotten ones with pet and person but no motion.
I have seen it in the past but that may have been older firmware. I will have to test and see what I get for a response in a range with no recordings, but within the status window. Not that it matters, as long as I get the status I can see if there would have been matches. |
On a side note, what is the target of this lib ,10 or 11? if 11 I would like to use StrEnums where possible instead of literal strings, if 10 is a possible target, we can either go with Enums and "fake" an StrEnum or use a backport lib. I kinda wish the one in HA was a separate lib instead of module code, then we could use that because it would already be present. I wasnt sure what your thoughts are on this so I haven used any yet. |
Currently this lib supports python 3.9, 3.10 and 3.11. I think having the last 3 python versions supported is a good thing, once 3.12 is out, I will probably drop 3.9. There are also pylint, flake8 and mypy tests beeing run for all 3 python versions. |
It would also be good to keep track of which direction we came from, so we only have to query in one direction on the camera, which is why I had the question about session/temp (in discord), otherwise could encode it in the identifier, but wanted to avoid polluting it if possible. |
Got the basics for DL support here https://github.com/xannor/ha-core/tree/reolink_vod_dl however, having an issue where the download is aborted about 30 seconds in, dont know if it is because of aio timeouts or something in the api side. Just get a connection closed notice. Give it a test and see if it causes any problems with NVRs though. |
@xannor sorry for taking so long, I just tested your reolink_vod_dl branch with my NVR but it does not work. So we will need to check using |
Thats what I was not sure of, and why I needed you to test. does the NVR fail with a command error or does the D/L start but fail after some point? The API's should match the "abilities" and so I want to make sure we go down the right path and not just make assumptions. |
This commit implemented download support for the NVR: 27e5124 However I would not recommend it for an NVR since it takes a really long time to start a playback (NvrDownload takes something like 20 seconds before completing). I will make a PR for your branch to suggest the nessesary changes. Maybe we can disable download for NVRs at first and in a seperate PR once this is merged make a new config flow option to let the user chose between streaming/downloading for playback (the NVR will have streaming as default and other cams downloading as default). |
See my PR to your branch here: xannor/ha-core#1 |
I think for the time being, just not doing the download on the NVR is the better approach, as there is little advantage in that instance. I still feel that an NVR is fundementally the same approach as here and it would be better to manage/use that interface than try to use the limited API and have HA do it. I mostly want this for those who do not have/want an NVR and want to have some of the functionality in concept. Also, I am not comfortable doing changes that affect/use the NVR as I do not have one and could not test any issues related to it. |
I looked at your PR. the live ability should be 1 if main/sub/ext is supported and 2 if only main/sub according to the API docs. you said the NVR does not support ext, does it report a 1 for that value, because I though I was explicitly checking for 1 in hasExt. |
Yes the NVR was reporting 1 for the "live" ability, but it only has main/sub playback, not ext. |
its nice to see the NVR does not follow the documented API correctly.. :) I'll add the NVR check to that as well. Ahh, I see your edit, so it does support ext for live but not playback. Given my tests where ext and sub seem the same on HD and 4MP cameras, maybe I should just skip ext for recordings? They do seems smaller so I dont know if it is a client side resize that makes them appear the same to me. |
I did some small edits to disable NVR DL and fully disabled ext playback, for now. I still need to investigate the DL issues I have seen, but could you check that the NVR is working as it should now? |
Ok, interesting results. I changed the player to pass the url to the camera though (since the command can be called via get like caps can.) for the non-hd it played right away and could play videos longer than 30sec, works wonderfully. For HD because the h265 codec, firefox refused to play, but could download if I open link in new tab, and MS edge would only play audio. So it looks like the HD cams will be an issue with playback. The downloaded file plays fine though. I also could not open the file via VLC (with the link) but that might be because the cams only allow 1 request to download they will deny and throw fits to multiple requests, which VLC might be doing. I did determine the issues I was having in passing the stream through HA are due to the connection timeouts, but I am not sure I can control them entirely. The alternative is to pass the url through (with token) but that both potentially exposes security info, and would not work with clients that do not have direct access to the camera themselves. Also if the camera is setup for the default https cert, the browser will reject the connection anyway. I am not sure how to be proceed from this point, |
@xannor I left some comments here: xannor/ha-core#1 Yes h265 encoding can be a problem to play in some browsers, that is also why the RTSP stream from Reolink does not yet support h265 encoding I think. I would just do only sub stream for cameras that have h265 encoding, just like what is done for the NVR.
I don't quite understand this...
I also don't understand this, could you give an example of the URL you are talking about? (you can replace token/password with xxxx) |
I beleive the RTSP does do h265, as that is the only way to pull the full stream, RTMP does not support h265.
Yeah, might have to but I dont like it. Would be nice if HA had a download component, for downloading from the HA, and not just downloading to it.
Just like with the screen capture API command, the download command can be passed to the camera as a GET style url request. I manually built that in the code to test, and it does work, except for the afformentioned h265 problem.
When passing the download command as a url, you have to supply the token parameter as well. If we send this url to the client, it means internal security information (normally the token is not exposed anywhere) would be sent over the web socket. Even though this should be safe, I dont like the thought of doing that, if it can be avoided. The reason I tried this approach is because all view requests in HA, by default use the default timeout of 30 seconds for the total life of the session connection. Since proxing the file through HA means two connections, one on the camera side the integration can control, and one on the HA side that core controlls, if the file takes longer than 30 seconds to transfer, it gets killed by the timeout. Most players will only buffer about 10-15 seconds and pause thedownload, so almost all transfer of files larger than a 15 second video will fail. Passing the url through works around this because now HA is not involved with the transfer, however, this will not work in situations where the client must access content though HA (such as remote connections) as the url would not be properly resolvable to the client. In addition if the connection is going over SSL, and a self-signed cert is used. then the player will not work anyways because the browser would reject the cert, proxying through HA also worked around this issue. Also, passing the url like that to potentially a remote client means that url would now be re-transmitted over the internet when the client attempts to connect. Both of these situations make me not want to try to hand the url off to the client as it would lead to a bad experience that a regular user may not realize is not an error in the integration. Thus needless bug tickets. |
From the absence in the write-up, I'm guessing the media support did not make the cut for this release? Updating right now so I will know for sure in a few minutes. |
@xannor I just updated and it is included, it is just not mentioned in the release notes. |
Yep, i see it, surprised there was no mention in the write-up. Hah, the snowflake camera work with it, but boy is it ugly, just a wall of icons and times. Not the integrations fault, the camera only reports motion events and it sure detects a lot of leaves moving during the day. I wish there was a way to localize the info better, for instance, show times in 12hr instead of 24hr as that is more "natural" for over here. What you can and cannot localize has been somewhat of a mystery to me. |
We could localize if you want. |
is there a way to see the user preferences? Going that route I would prefer to use user settings instead of system, if possible. I have tried to dig through other integrations to see how they handle that sort of thing, but mostly when localization has been involved, they only talk about states and other things done through the lang files, not regular output. Also, is there a way to use the lang data as well, would be good to use localized text for the AI states from the lang for the user than forcing english, for the VOD info. |
I was re-reading the API and I see a "new" command called playback that I didnt see before. It is structured exactly like download but does not list the same limitations. I am going to test with it and expand the vod_source function to handle generate one of the three possible urls. Also in vod source I see you check for rtmp and enable it, this should be the responsibility of the integration to en/force not as a side-effect of the library (generally side-effects are a bad idea). I am going to remove this from my changes. Also, also, I am going to remove the download support from the get_chunk as I think using vod_source and letting the integration deal with the url is better than passing back an abusable stream (solves some of the hacks to get the stream out of get_chunk properly.) |
Ok got a little farther, and feel a little better about it. I have a new PR to this lib for the changes I need (and I updated my branch of core). If you could test what I have with your NVR to make sure I didnt break anything, this would be a good start for the second round of media support. |
I will try looking at it soon, am a bit swamped with work at the moment tho... Seems like a nice way to go forward. |
@xannor I just merged the PR (in a seperate comment), I will release a new reolink_aio version shortly. |
Ahh, I see where you thought things didnt work. I was not enforcing the RTMP/DL limit in API but in the component, I figure the API should be closer to "dumb" and the calling code should enforce any device specific rules, except where the API docs state such a rule, or a work around is necessary. |
I like to put as much logic in the library and make the HomeAssistant code as "dumb" as possible. |
Yeah, I see where your coming from now. Its just an old habit of mine, to write as close to the API and UI as possible and bridge the difference. |
@xannor version bump PR in HA is here: home-assistant/core#106747 |
@xannor could you check this issue: home-assistant/core#110939 |
With this latest PR, a Proxy is added such that playback inside HomeAssistant is now basically always possible: home-assistant/core#133916 |
Do you have any plans to build out a media platform using the search command? If you dont, or dont plan on doing it for a while, I might create a custom integration that wraps around the "built in" reolink and this api to implement one. I had built the original one in the fwesterberg custom integration, and has plans to build one in my own CI until your built in one surpassed it faster than I could work on it.
My idea is the media platform providing date sorted access to recordings and a custom sensor that provides info on the last recording for a channel. The only real drawback is that the rest api does not provide any trigger information other than a date/time, (but the client one does) and no potential thumbnails.
any thoughts?
The text was updated successfully, but these errors were encountered: