You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
503 lines
19 KiB
503 lines
19 KiB
/* |
|
* This file is free software: you can redistribute it and/or modify it |
|
* under the terms of the GNU General Public License as published by the |
|
* Free Software Foundation, either version 3 of the License, or |
|
* (at your option) any later version. |
|
* |
|
* This file is distributed in the hope that it will be useful, but |
|
* WITHOUT ANY WARRANTY; without even the implied warranty of |
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. |
|
* See the GNU General Public License for more details. |
|
* |
|
* You should have received a copy of the GNU General Public License along |
|
* with this program. If not, see <http://www.gnu.org/licenses/>. |
|
* |
|
* Code by Michael Oborne and Siddharth Bharat Purohit, Cubepilot Pty. |
|
*/ |
|
|
|
#include "AP_ONVIF.h" |
|
#if ENABLE_ONVIF |
|
#include <AP_ONVIF/MediaBinding.nsmap> |
|
|
|
#include "onvifhelpers.h" |
|
// For ChibiOS we will use HW RND # generator |
|
#include <GCS_MAVLink/GCS.h> |
|
|
|
extern const AP_HAL::HAL &hal; |
|
|
|
#if 0 |
|
#define DEBUG_PRINT(fmt,args...) do {GCS_SEND_TEXT(MAV_SEVERITY_INFO ,"AP_ONVIF:" fmt "\n", ## args); } while(0) |
|
#else |
|
#define DEBUG_PRINT(fmt,args...) |
|
#endif |
|
|
|
#define ERROR_PRINT(fmt,args...) do {GCS_SEND_TEXT(MAV_SEVERITY_ERROR , "AP_ONVIF:" fmt "\n", ## args); } while(0) |
|
|
|
const char *wsse_PasswordDigestURI = "http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-username-token-profile-1.0#PasswordDigest"; |
|
const char *wsse_Base64BinaryURI = "http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-soap-message-security-1.0#Base64Binary"; |
|
|
|
AP_ONVIF *AP_ONVIF::_singleton; |
|
static AP_ONVIF onvif; |
|
|
|
#define DEVICE_ENDPOINT_LOC "/onvif/device_service" |
|
#define MEDIA_ENDPOINT_LOC "/onvif/media_service" |
|
#define PTZ_ENDPOINT_LOC "/onvif/ptz_service" |
|
// Default constructor |
|
AP_ONVIF::AP_ONVIF() |
|
{ |
|
if (_singleton != nullptr) { |
|
AP_HAL::panic("AP_ONVIF must be singleton"); |
|
} |
|
_singleton = this; |
|
} |
|
|
|
// Start ONVIF client with username, password and service host url |
|
bool AP_ONVIF::start(const char *user, const char *pass, const char *hostname) |
|
{ |
|
if (!initialised) { |
|
soap = soap_new1(SOAP_XML_CANONICAL | SOAP_C_UTFSTRING); |
|
if (soap == nullptr) { |
|
ERROR_PRINT("AP_ONVIF: Failed to allocate soap"); |
|
return false; |
|
} |
|
soap->connect_timeout = soap->recv_timeout = soap->send_timeout = 30; // 30 sec |
|
|
|
if (proxy_device == nullptr) { |
|
proxy_device = new DeviceBindingProxy(soap); |
|
} |
|
if (proxy_media == nullptr) { |
|
proxy_media = new MediaBindingProxy(soap); |
|
} |
|
if (proxy_ptz == nullptr) { |
|
proxy_ptz = new PTZBindingProxy(soap); |
|
} |
|
|
|
if (proxy_device == nullptr || |
|
proxy_media == nullptr || |
|
proxy_ptz == nullptr) { |
|
ERROR_PRINT("AP_ONVIF: Failed to allocate gSOAP Proxy objects."); |
|
return false; |
|
} |
|
initialised = true; |
|
} |
|
|
|
if (strlen(user) > username_len) { |
|
if (username != nullptr) { |
|
free(username); |
|
username = nullptr; |
|
} |
|
} |
|
username_len = strlen(user); |
|
if (username == nullptr) { |
|
username = (char*)malloc(username_len + 1); |
|
} |
|
|
|
if (strlen(pass) > password_len) { |
|
if (password != nullptr) { |
|
free(password); |
|
password = nullptr; |
|
} |
|
} |
|
password_len = strlen(pass); |
|
if (password == nullptr) { |
|
password = (char*)malloc(password_len+1); |
|
} |
|
|
|
if (strlen(hostname) > hostname_len) { |
|
// free if not nullptr |
|
if (device_endpoint != nullptr) { |
|
free(device_endpoint); |
|
device_endpoint = nullptr; |
|
} |
|
if (media_endpoint != nullptr) { |
|
free(media_endpoint); |
|
media_endpoint = nullptr; |
|
} |
|
if (ptz_endpoint != nullptr) { |
|
free(ptz_endpoint); |
|
ptz_endpoint = nullptr; |
|
} |
|
} |
|
|
|
hostname_len = strlen(hostname); |
|
|
|
if (device_endpoint == nullptr) { |
|
device_endpoint = (char*)malloc(hostname_len + strlen(DEVICE_ENDPOINT_LOC) + 1); |
|
} |
|
if (media_endpoint == nullptr) { |
|
media_endpoint = (char*)malloc(hostname_len + strlen(MEDIA_ENDPOINT_LOC) + 1); |
|
} |
|
if (ptz_endpoint == nullptr) { |
|
ptz_endpoint = (char*)malloc(hostname_len + strlen(PTZ_ENDPOINT_LOC) + 1); |
|
} |
|
|
|
if (device_endpoint == nullptr || |
|
media_endpoint == nullptr || |
|
ptz_endpoint == nullptr || |
|
username == nullptr || |
|
password == nullptr) { |
|
ERROR_PRINT("Failed to Allocate strings"); |
|
return false; |
|
} |
|
|
|
strcpy(username, user); |
|
strcpy(password, pass); |
|
snprintf(device_endpoint, hostname_len + strlen(DEVICE_ENDPOINT_LOC) + 1, "%s" DEVICE_ENDPOINT_LOC, hostname); |
|
snprintf(media_endpoint, hostname_len + strlen(MEDIA_ENDPOINT_LOC) + 1, "%s" MEDIA_ENDPOINT_LOC, hostname); |
|
snprintf(ptz_endpoint, hostname_len + strlen(PTZ_ENDPOINT_LOC) + 1, "%s" PTZ_ENDPOINT_LOC, hostname); |
|
|
|
/// TODO: Need to find a way to store this in parameter system |
|
// or it could be just storage, we will see |
|
proxy_device->soap_endpoint = device_endpoint; |
|
if (!probe_onvif_server()) { |
|
ERROR_PRINT("Failed to probe onvif server."); |
|
return false; |
|
} |
|
|
|
return true; |
|
} |
|
|
|
void AP_ONVIF::report_error() |
|
{ |
|
ERROR_PRINT("ONVIF ERROR:"); |
|
if (soap_check_state(soap)) { |
|
ERROR_PRINT("Error: soap struct state not initialized"); |
|
} else if (soap->error) { |
|
const char **c, *v = NULL, *s, *d; |
|
c = soap_faultcode(soap); |
|
if (!*c) { |
|
soap_set_fault(soap); |
|
c = soap_faultcode(soap); |
|
} |
|
if (soap->version == 2) { |
|
v = soap_fault_subcode(soap); |
|
} |
|
s = soap_fault_string(soap); |
|
d = soap_fault_detail(soap); |
|
ERROR_PRINT("%s%d fault %s [%s]\n%s\nDetail: %s",(soap->version ? "SOAP 1." : "Error "), |
|
(soap->version ? (int)soap->version : soap->error), |
|
*c, (v ? v : "no subcode"), (s ? s : "[no reason]"), |
|
(d ? d : "[no detail]")); |
|
} |
|
} |
|
|
|
// detect onvif server present on the network |
|
bool AP_ONVIF::probe_onvif_server() |
|
{ |
|
{ |
|
_tds__GetDeviceInformation GetDeviceInformation; |
|
_tds__GetDeviceInformationResponse GetDeviceInformationResponse; |
|
if (!set_credentials()) { |
|
ERROR_PRINT("Failed to setup credentials"); |
|
goto err; |
|
} |
|
if (proxy_device->GetDeviceInformation(&GetDeviceInformation, GetDeviceInformationResponse)) { |
|
ERROR_PRINT("Failed to fetch Device Information"); |
|
report_error(); |
|
goto err; |
|
} |
|
|
|
DEBUG_PRINT("Manufacturer: %s",GetDeviceInformationResponse.Manufacturer); |
|
DEBUG_PRINT("Model: %s",GetDeviceInformationResponse.Model); |
|
DEBUG_PRINT("FirmwareVersion: %s",GetDeviceInformationResponse.FirmwareVersion); |
|
DEBUG_PRINT("SerialNumber: %s",GetDeviceInformationResponse.SerialNumber); |
|
DEBUG_PRINT("HardwareId: %s",GetDeviceInformationResponse.HardwareId); |
|
} |
|
|
|
// get device capabilities and print media |
|
{ |
|
_tds__GetCapabilities GetCapabilities; |
|
_tds__GetCapabilitiesResponse GetCapabilitiesResponse; |
|
if (!set_credentials()) { |
|
ERROR_PRINT("Failed to setup credentials"); |
|
goto err; |
|
} |
|
if (proxy_device->GetCapabilities(&GetCapabilities, GetCapabilitiesResponse)) { |
|
ERROR_PRINT("Failed to fetch Device Capabilities"); |
|
report_error(); |
|
goto err; |
|
} |
|
|
|
if (!GetCapabilitiesResponse.Capabilities || !GetCapabilitiesResponse.Capabilities->Media) { |
|
ERROR_PRINT("Missing device capabilities info"); |
|
goto err; |
|
} else { |
|
DEBUG_PRINT("XAddr: %s", GetCapabilitiesResponse.Capabilities->Media->XAddr); |
|
if (GetCapabilitiesResponse.Capabilities->Media->StreamingCapabilities) { |
|
if (GetCapabilitiesResponse.Capabilities->Media->StreamingCapabilities->RTPMulticast) { |
|
DEBUG_PRINT("RTPMulticast: %s",(*GetCapabilitiesResponse.Capabilities->Media->StreamingCapabilities->RTPMulticast ? "yes" : "no")); |
|
} |
|
if (GetCapabilitiesResponse.Capabilities->Media->StreamingCapabilities->RTP_USCORETCP) { |
|
DEBUG_PRINT("RTP_TCP: %s", (*GetCapabilitiesResponse.Capabilities->Media->StreamingCapabilities->RTP_USCORETCP ? "yes" : "no")); |
|
} |
|
if (GetCapabilitiesResponse.Capabilities->Media->StreamingCapabilities->RTP_USCORERTSP_USCORETCP) { |
|
DEBUG_PRINT("RTP_RTSP_TCP: %s", (*GetCapabilitiesResponse.Capabilities->Media->StreamingCapabilities->RTP_USCORERTSP_USCORETCP ? "yes" : "no")); |
|
} |
|
} |
|
} |
|
|
|
// set the Media proxy endpoint to XAddr |
|
proxy_media->soap_endpoint = media_endpoint; |
|
} |
|
|
|
// get device profiles |
|
{ |
|
_trt__GetProfiles GetProfiles; |
|
_trt__GetProfilesResponse GetProfilesResponse; |
|
if (!set_credentials()) { |
|
ERROR_PRINT("Failed to setup credentials"); |
|
goto err; |
|
} |
|
if (proxy_media->GetProfiles(&GetProfiles, GetProfilesResponse)){ |
|
ERROR_PRINT("Failed to fetch profiles"); |
|
report_error(); |
|
goto err; |
|
} |
|
|
|
if (GetProfilesResponse.__sizeProfiles > 0) { |
|
DEBUG_PRINT("Profiles Received %lu", (unsigned long)GetProfilesResponse.__sizeProfiles); |
|
} else { |
|
ERROR_PRINT("Error: No Profiles Received"); |
|
goto err; |
|
} |
|
|
|
// for each profile get snapshot |
|
for (uint32_t i = 0; i < (uint32_t)GetProfilesResponse.__sizeProfiles; i++) { |
|
DEBUG_PRINT("Profile name: %s", GetProfilesResponse.Profiles[i]->Name); |
|
} |
|
|
|
// Just use first one for now |
|
if (profile_token_size < (strlen(GetProfilesResponse.Profiles[0]->token) + 1)) { |
|
if (profile_token != nullptr) { |
|
free(profile_token); |
|
} |
|
profile_token = (char*)malloc(strlen(GetProfilesResponse.Profiles[0]->token) + 1); |
|
profile_token_size = strlen(GetProfilesResponse.Profiles[0]->token) + 1; |
|
if (profile_token == nullptr) { |
|
goto err; |
|
} |
|
} |
|
|
|
strcpy(profile_token, GetProfilesResponse.Profiles[0]->token); |
|
|
|
proxy_ptz->soap_endpoint = ptz_endpoint; |
|
} |
|
|
|
// get PTZ Token |
|
{ |
|
_tptz__GetConfigurations GetConfigurations; |
|
_tptz__GetConfigurationsResponse GetConfigurationsResponse; |
|
if (!set_credentials()) { |
|
ERROR_PRINT("Failed to setup credentials"); |
|
goto err; |
|
} |
|
if (proxy_ptz->GetConfigurations(&GetConfigurations, GetConfigurationsResponse)) { |
|
ERROR_PRINT("Failed to fetch Configurations"); |
|
report_error(); |
|
goto err; |
|
} |
|
|
|
if (GetConfigurationsResponse.__sizePTZConfiguration > 0) { |
|
DEBUG_PRINT("PTZ Tokens Received"); |
|
} else { |
|
ERROR_PRINT("Error: No Profiles Received"); |
|
goto err; |
|
} |
|
|
|
for (uint32_t i = 0; i < (uint32_t)GetConfigurationsResponse.__sizePTZConfiguration; i++) { |
|
DEBUG_PRINT("PTZ: %s", GetConfigurationsResponse.PTZConfiguration[i]->Name); |
|
} |
|
//GetConfigurationsResponse.PTZConfiguration[0]->token |
|
pan_tilt_limit_max = Vector2f(GetConfigurationsResponse.PTZConfiguration[0]->PanTiltLimits->Range->XRange->Max, |
|
GetConfigurationsResponse.PTZConfiguration[0]->PanTiltLimits->Range->YRange->Max); |
|
pan_tilt_limit_min = Vector2f(GetConfigurationsResponse.PTZConfiguration[0]->PanTiltLimits->Range->XRange->Min, |
|
GetConfigurationsResponse.PTZConfiguration[0]->PanTiltLimits->Range->YRange->Min); |
|
zoom_min = GetConfigurationsResponse.PTZConfiguration[0]->ZoomLimits->Range->XRange->Min; |
|
zoom_max = GetConfigurationsResponse.PTZConfiguration[0]->ZoomLimits->Range->XRange->Max; |
|
|
|
DEBUG_PRINT("Pan: %f %f Tilt: %f %f", pan_tilt_limit_min.x, pan_tilt_limit_max.x, |
|
pan_tilt_limit_min.y, pan_tilt_limit_max.y); |
|
} |
|
|
|
// Get PTZ status |
|
{ |
|
_tptz__GetStatus GetStatus; |
|
_tptz__GetStatusResponse GetStatusResponse; |
|
|
|
GetStatus.ProfileToken = profile_token; |
|
if (!set_credentials()) { |
|
ERROR_PRINT("Failed to setup credentials"); |
|
goto err; |
|
} |
|
if (proxy_ptz->GetStatus(&GetStatus, GetStatusResponse)) { |
|
DEBUG_PRINT("Failed to recieve PTZ status"); |
|
goto err; |
|
} |
|
|
|
if (GetStatusResponse.PTZStatus->Error) { |
|
DEBUG_PRINT("ErrorStatus: %s", GetStatusResponse.PTZStatus->Error); |
|
} |
|
if (GetStatusResponse.PTZStatus->MoveStatus->PanTilt) { |
|
DEBUG_PRINT("PTStatus: %d", *GetStatusResponse.PTZStatus->MoveStatus->PanTilt); |
|
} |
|
if (GetStatusResponse.PTZStatus->MoveStatus->Zoom) { |
|
DEBUG_PRINT("ZoomStatus: %d", *GetStatusResponse.PTZStatus->MoveStatus->Zoom); |
|
} |
|
DEBUG_PRINT("Pan: %f Tilt: %f", GetStatusResponse.PTZStatus->Position->PanTilt->x, |
|
GetStatusResponse.PTZStatus->Position->PanTilt->y); |
|
DEBUG_PRINT("Zoom: %f", GetStatusResponse.PTZStatus->Position->Zoom->x); |
|
} |
|
|
|
soap_destroy(soap); |
|
soap_end(soap); |
|
return true; |
|
err: |
|
soap_destroy(soap); |
|
soap_end(soap); |
|
return false; |
|
} |
|
|
|
// Generate Random Nonce value |
|
bool AP_ONVIF::rand_nonce(char *nonce, size_t noncelen) |
|
{ |
|
if (noncelen <= 4) { |
|
// invalid size |
|
return false; |
|
} |
|
uint32_t r = (uint32_t)(hal.util->get_hw_rtc()/1000000ULL); |
|
(void)memcpy((void *)nonce, (const void *)&r, 4); |
|
return hal.util->get_random_vals((uint8_t*)&nonce[4], noncelen - 4); |
|
} |
|
|
|
#define TEST_NONCE "LKqI6G/AikKCQrN0zqZFlg==" |
|
#define TEST_TIME "2010-09-16T07:50:45Z" |
|
#define TEST_PASS "userpassword" |
|
#define TEST_RESULT "tuOSpGlFlIXsozq4HFNeeGeFLEI=" |
|
#define TEST 0 |
|
bool AP_ONVIF::set_credentials() |
|
{ |
|
soap_wsse_delete_Security(soap); |
|
soap_wsse_add_Timestamp(soap, "Time", 60); |
|
|
|
_wsse__Security *security = soap_wsse_add_Security(soap); |
|
const char *created = soap_dateTime2s(soap, (time_t)(hal.util->get_hw_rtc()/1000000ULL)); |
|
char HA[SHA1_DIGEST_SIZE] {}; |
|
char HABase64fin[29] {}; |
|
char nonce[16]; |
|
char *nonceBase64enc = nullptr; |
|
char *nonceBase64fin; |
|
sha1_ctx ctx; |
|
uint16_t HABase64len; |
|
char *HABase64enc = nullptr; |
|
uint16_t noncelen; |
|
|
|
/* generate a nonce */ |
|
if (!rand_nonce(nonce, 16)) { |
|
return false; |
|
} |
|
|
|
sha1_begin(&ctx); |
|
#if TEST |
|
char* test_nonce = (char*)base64_decode((const unsigned char*)TEST_NONCE, strlen(TEST_NONCE), &noncelen); |
|
sha1_hash((const unsigned char*)test_nonce, noncelen, &ctx); |
|
sha1_hash((const unsigned char*)TEST_TIME, strlen(TEST_TIME), &ctx); |
|
sha1_hash((const unsigned char*)TEST_PASS, strlen(TEST_PASS), &ctx); |
|
nonceBase64enc = (char*)base64_encode((unsigned char*)test_nonce, 16, &noncelen); // this call also mallocs |
|
DEBUG_PRINT("Created:%s Hash64:%s", TEST_TIME, HABase64fin); |
|
#else |
|
sha1_hash((const unsigned char*)nonce, 16, &ctx); |
|
sha1_hash((const unsigned char*)created, strlen(created), &ctx); |
|
sha1_hash((const unsigned char*)password, strlen(password), &ctx); |
|
nonceBase64enc = (char*)base64_encode((unsigned char*)nonce, 16, &noncelen); // this call also mallocs |
|
#endif |
|
if (nonceBase64enc == nullptr) { |
|
return false; |
|
} |
|
// move to something we can track |
|
nonceBase64fin = (char*)soap_malloc(soap, noncelen); |
|
memcpy(nonceBase64fin, nonceBase64enc, noncelen); |
|
free(nonceBase64enc); |
|
|
|
sha1_end((unsigned char*)HA, &ctx); |
|
HABase64enc = (char*)base64_encode((unsigned char*)HA, SHA1_DIGEST_SIZE, &HABase64len); |
|
if (HABase64enc == nullptr) { |
|
return false; |
|
} |
|
if (HABase64len > 29) { |
|
//things have gone truly bad time to panic |
|
ERROR_PRINT("Error: Invalid Base64 Encode!"); |
|
free(HABase64enc); |
|
return false; |
|
} |
|
|
|
memcpy(HABase64fin, HABase64enc, HABase64len); |
|
free(HABase64enc); |
|
|
|
if (soap_wsse_add_UsernameTokenText(soap, "Auth", username, HABase64fin)) { |
|
report_error(); |
|
return false; |
|
} |
|
/* populate the remainder of the password, nonce, and created */ |
|
security->UsernameToken->Password->Type = (char*)wsse_PasswordDigestURI; |
|
security->UsernameToken->Nonce = (struct wsse__EncodedString*)soap_malloc(soap, sizeof(struct wsse__EncodedString)); |
|
security->UsernameToken->Salt = NULL; |
|
security->UsernameToken->Iteration = NULL; |
|
if (!security->UsernameToken->Nonce) { |
|
ERROR_PRINT("Failed to allocate NONCE"); |
|
return false; |
|
} |
|
soap_default_wsse__EncodedString(soap, security->UsernameToken->Nonce); |
|
security->UsernameToken->Nonce->__item = nonceBase64fin; |
|
security->UsernameToken->Nonce->EncodingType = (char*)wsse_Base64BinaryURI; |
|
security->UsernameToken->wsu__Created = soap_strdup(soap, created); |
|
return true; |
|
} |
|
|
|
// Turn ONVIF camera to mentioned pan, tilt and zoom, normalised |
|
// between limits |
|
bool AP_ONVIF::set_absolutemove(float x, float y, float z) |
|
{ |
|
_tptz__AbsoluteMove AbsoluteMove; |
|
_tptz__AbsoluteMoveResponse AbsoluteMoveResponse; |
|
AbsoluteMove.Position = soap_new_tt__PTZVector(soap); |
|
if (AbsoluteMove.Position == nullptr) { |
|
ERROR_PRINT("Failed to allocate AbsoluteMove.Position"); |
|
goto err; |
|
} |
|
AbsoluteMove.Position->PanTilt = soap_new_tt__Vector2D(soap); |
|
if (AbsoluteMove.Position->PanTilt == nullptr) { |
|
ERROR_PRINT("Failed to allocate AbsoluteMove.Position->PanTilt"); |
|
goto err; |
|
} |
|
AbsoluteMove.Position->Zoom = soap_new_tt__Vector1D(soap); |
|
if (AbsoluteMove.Position->Zoom == nullptr) { |
|
ERROR_PRINT("Failed to allocate AbsoluteMove.Position->Zoom"); |
|
goto err; |
|
} |
|
|
|
AbsoluteMove.Position->PanTilt->x = constrain_float(x, pan_tilt_limit_min.x, pan_tilt_limit_max.x); |
|
AbsoluteMove.Position->PanTilt->y = constrain_float(y, pan_tilt_limit_min.y, pan_tilt_limit_max.y); |
|
AbsoluteMove.Position->Zoom->x = constrain_float(z, zoom_min, zoom_max); |
|
AbsoluteMove.Speed = NULL; |
|
AbsoluteMove.ProfileToken = profile_token; |
|
// DEBUG_PRINT("Setting AbsoluteMove: %f %f %f", AbsoluteMove.Position->PanTilt->x, |
|
// AbsoluteMove.Position->PanTilt->y, |
|
// AbsoluteMove.Position->Zoom->x); |
|
if (!set_credentials()) { |
|
ERROR_PRINT("Failed to setup credentials"); |
|
goto err; |
|
} |
|
if (proxy_ptz->AbsoluteMove(&AbsoluteMove, AbsoluteMoveResponse)) { |
|
ERROR_PRINT("Failed to sent AbsoluteMove cmd"); |
|
report_error(); |
|
goto err; |
|
} |
|
soap_destroy(soap); |
|
soap_end(soap); |
|
return true; |
|
err: |
|
soap_destroy(soap); |
|
soap_end(soap); |
|
return false; |
|
} |
|
#endif //#if ENABLE_ONVIF
|
|
|