Merge branch 'onvif-wsse-created-race'

This commit is contained in:
Isaac Connor
2026-06-14 09:15:01 -05:00
3 changed files with 122 additions and 2 deletions

View File

@@ -1241,11 +1241,34 @@ void ONVIF::set_credentials(struct soap *soap) {
return;
}
soap_wsse_delete_Security(soap);
// Pin a single timestamp for the whole security header. gSOAP's
// soap_wsse_add_Timestamp() and soap_wsse_add_UsernameTokenDigest() each call
// time(NULL) independently, so when the two calls straddle a one-second
// boundary the wsu:Timestamp Created and the UsernameToken Created differ by a
// second. Hikvision (and some other cameras) reject that mismatch as
// NotAuthorized, which surfaced as intermittent PullMessages auth failures
// every few thousand requests despite correct credentials and synced clocks.
// Capturing time(NULL) once and forcing both Created values to it removes the
// race.
time_t wsse_now = time(nullptr);
// Use configurable timestamp validity (default 60 seconds) to handle clock drift
// between ZoneMinder and the camera. The old value of 10 seconds was too short
// and caused "not authorized" errors when clocks were slightly out of sync.
soap_wsse_add_Timestamp(soap, "Time", timestamp_validity_seconds);
// soap_wsse_add_Timestamp() stamped Created/Expires from its own time(NULL).
// Re-stamp them from wsse_now so they match the UsernameToken Created below.
_wsse__Security *security = soap_wsse_add_Security(soap);
if (security && security->wsu__Timestamp) {
security->wsu__Timestamp->Created = soap_strdup(soap, soap_dateTime2s(soap, wsse_now));
if (security->wsu__Timestamp->Expires) {
security->wsu__Timestamp->Expires =
soap_strdup(soap, soap_dateTime2s(soap, wsse_now + timestamp_validity_seconds));
}
}
const char *username = parent->onvif_username.empty() ? parent->user.c_str() : parent->onvif_username.c_str();
const char *password = parent->onvif_username.empty() ? parent->pass.c_str() : parent->onvif_password.c_str();
@@ -1254,9 +1277,11 @@ void ONVIF::set_credentials(struct soap *soap) {
Debug(2, "ONVIF: Using UsernameToken (plain) authentication");
soap_wsse_add_UsernameTokenText(soap, "Auth", username, password);
} else {
// Try UsernameTokenDigest authentication (default)
// Try UsernameTokenDigest authentication (default).
// Use the _at variant so the token's Created (and the password digest
// computed from it) is pinned to the same wsse_now as the Timestamp above.
Debug(2, "ONVIF: Using UsernameTokenDigest authentication");
soap_wsse_add_UsernameTokenDigest(soap, "Auth", username, password);
soap_wsse_add_UsernameTokenDigest_at(soap, "Auth", username, password, wsse_now);
}
}

View File

@@ -19,6 +19,7 @@ set(TEST_SOURCES
zm_font.cpp
zm_image.cpp
zm_onvif_renewal.cpp
zm_onvif_wsse.cpp
zm_pixformat.cpp
zm_poly.cpp
zm_time.cpp

94
tests/zm_onvif_wsse.cpp Normal file
View File

@@ -0,0 +1,94 @@
/*
* This file is part of the ZoneMinder Project. See AUTHORS file for Copyright information
*
* This program 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 2 of the License, or (at your
* option) any later version.
*
* This program 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/>.
*/
#include "zm_catch2.h"
#include <ctime>
#include <string>
#include "soapH.h"
#include "plugin/wsseapi.h"
// Regression test for the WS-Security Created race in ONVIF::set_credentials().
//
// gSOAP's soap_wsse_add_Timestamp() and soap_wsse_add_UsernameTokenDigest()
// each call time(NULL) on their own. When those two calls straddle a one-second
// boundary, the wsu:Timestamp Created and the UsernameToken Created end up one
// second apart, and Hikvision (and some other cameras) reject the request as
// NotAuthorized. set_credentials() avoids this by capturing a single time(NULL)
// and forcing both Created values to it (re-stamping the Timestamp and using the
// _at variant for the token digest).
TEST_CASE("ONVIF WS-Security Created timestamps are pinned together") {
SECTION("Timestamp Created matches UsernameToken Created (digest auth)") {
struct soap *soap = soap_new();
REQUIRE(soap != nullptr);
soap_register_plugin(soap, soap_wsse);
const int validity = 60;
// Reproduce the exact sequence set_credentials() uses for digest auth.
soap_wsse_delete_Security(soap);
time_t wsse_now = time(nullptr);
soap_wsse_add_Timestamp(soap, "Time", validity);
_wsse__Security *security = soap_wsse_add_Security(soap);
REQUIRE(security != nullptr);
REQUIRE(security->wsu__Timestamp != nullptr);
security->wsu__Timestamp->Created = soap_strdup(soap, soap_dateTime2s(soap, wsse_now));
if (security->wsu__Timestamp->Expires) {
security->wsu__Timestamp->Expires =
soap_strdup(soap, soap_dateTime2s(soap, wsse_now + validity));
}
soap_wsse_add_UsernameTokenDigest_at(soap, "Auth", "admin", "password", wsse_now);
REQUIRE(security->wsu__Timestamp->Created != nullptr);
REQUIRE(security->UsernameToken != nullptr);
REQUIRE(security->UsernameToken->wsu__Created != nullptr);
// The whole point of the fix: these two must always be identical, so the
// camera never sees the off-by-one second that triggers NotAuthorized.
REQUIRE(std::string(security->wsu__Timestamp->Created) ==
std::string(security->UsernameToken->wsu__Created));
// And the pinned value is exactly the one we captured.
REQUIRE(std::string(security->wsu__Timestamp->Created) ==
std::string(soap_dateTime2s(soap, wsse_now)));
soap_wsse_delete_Security(soap);
soap_destroy(soap);
soap_end(soap);
soap_free(soap);
}
SECTION("A one-second gap really does change the Created string") {
// Documents why the un-pinned (racy) code fails: two time(NULL) results one
// second apart serialise to different Created values, which is precisely the
// mismatch Hikvision rejected in the captured SOAP logs.
struct soap *soap = soap_new();
REQUIRE(soap != nullptr);
time_t now = time(nullptr);
std::string a(soap_dateTime2s(soap, now));
std::string b(soap_dateTime2s(soap, now + 1));
REQUIRE(a != b);
soap_destroy(soap);
soap_end(soap);
soap_free(soap);
}
}