Merge branch 'storageareas' of github.com:ConnorTechnology/ZoneMinder into storageareas
This commit is contained in:
commit
2c8735e79b
|
@ -35,6 +35,14 @@
|
||||||
#include <netinet/in.h>
|
#include <netinet/in.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if HAVE_LIBPCRE
|
||||||
|
static RegExpr *header_expr = 0;
|
||||||
|
static RegExpr *status_expr = 0;
|
||||||
|
static RegExpr *connection_expr = 0;
|
||||||
|
static RegExpr *content_length_expr = 0;
|
||||||
|
static RegExpr *content_type_expr = 0;
|
||||||
|
#endif
|
||||||
|
|
||||||
RemoteCameraHttp::RemoteCameraHttp(
|
RemoteCameraHttp::RemoteCameraHttp(
|
||||||
unsigned int p_monitor_id,
|
unsigned int p_monitor_id,
|
||||||
const std::string &p_method,
|
const std::string &p_method,
|
||||||
|
@ -72,9 +80,9 @@ RemoteCameraHttp::RemoteCameraHttp(
|
||||||
|
|
||||||
if ( p_method == "simple" )
|
if ( p_method == "simple" )
|
||||||
method = SIMPLE;
|
method = SIMPLE;
|
||||||
else if ( p_method == "regexp" )
|
else if ( p_method == "regexp" ) {
|
||||||
method = REGEXP;
|
method = REGEXP;
|
||||||
else
|
} else
|
||||||
Fatal( "Unrecognised method '%s' when creating HTTP camera %d", p_method.c_str(), monitor_id );
|
Fatal( "Unrecognised method '%s' when creating HTTP camera %d", p_method.c_str(), monitor_id );
|
||||||
if ( capture ) {
|
if ( capture ) {
|
||||||
Initialise();
|
Initialise();
|
||||||
|
@ -114,7 +122,22 @@ void RemoteCameraHttp::Initialise() {
|
||||||
mode = SINGLE_IMAGE;
|
mode = SINGLE_IMAGE;
|
||||||
format = UNDEF;
|
format = UNDEF;
|
||||||
state = HEADER;
|
state = HEADER;
|
||||||
|
|
||||||
|
#if HAVE_LIBPCRE
|
||||||
|
if ( method == REGEXP ) {
|
||||||
|
if ( !header_expr )
|
||||||
|
header_expr = new RegExpr("^(.+?\r?\n\r?\n)", PCRE_DOTALL);
|
||||||
|
if ( !status_expr )
|
||||||
|
status_expr = new RegExpr("^HTTP/(1\\.[01]) +([0-9]+) +(.+?)\r?\n", PCRE_CASELESS);
|
||||||
|
if ( !connection_expr )
|
||||||
|
connection_expr = new RegExpr("Connection: ?(.+?)\r?\n", PCRE_CASELESS);
|
||||||
|
if ( !content_length_expr )
|
||||||
|
content_length_expr = new RegExpr("Content-length: ?([0-9]+)\r?\n", PCRE_CASELESS);
|
||||||
|
if ( !content_type_expr )
|
||||||
|
content_type_expr = new RegExpr("Content-type: ?(.+?)(?:; ?boundary=\x22?(.+?)\x22?)?\r?\n", PCRE_CASELESS);
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
} // end void RemoteCameraHttp::Initialise()
|
||||||
|
|
||||||
int RemoteCameraHttp::Connect() {
|
int RemoteCameraHttp::Connect() {
|
||||||
struct addrinfo *p = NULL;
|
struct addrinfo *p = NULL;
|
||||||
|
@ -264,9 +287,9 @@ int RemoteCameraHttp::ReadData( Buffer &buffer, unsigned int bytes_expected ) {
|
||||||
total_bytes_to_read -= bytes_read;
|
total_bytes_to_read -= bytes_read;
|
||||||
} while ( total_bytes_to_read );
|
} while ( total_bytes_to_read );
|
||||||
|
|
||||||
Debug( 4, buffer );
|
Debug(4, "%s", buffer);
|
||||||
|
|
||||||
return( total_bytes_read );
|
return total_bytes_read;
|
||||||
}
|
}
|
||||||
|
|
||||||
int RemoteCameraHttp::GetResponse() {
|
int RemoteCameraHttp::GetResponse() {
|
||||||
|
@ -287,36 +310,23 @@ int RemoteCameraHttp::GetResponse() {
|
||||||
//int subcontent_length = 0;
|
//int subcontent_length = 0;
|
||||||
//const char *subcontent_type = "";
|
//const char *subcontent_type = "";
|
||||||
|
|
||||||
while ( true )
|
while ( !zm_terminate ) {
|
||||||
{
|
switch( state ) {
|
||||||
switch( state )
|
|
||||||
{
|
|
||||||
case HEADER :
|
case HEADER :
|
||||||
{
|
{
|
||||||
static RegExpr *header_expr = 0;
|
|
||||||
static RegExpr *status_expr = 0;
|
|
||||||
static RegExpr *connection_expr = 0;
|
|
||||||
static RegExpr *content_length_expr = 0;
|
|
||||||
static RegExpr *content_type_expr = 0;
|
|
||||||
|
|
||||||
while ( !( buffer_len = ReadData(buffer) ) && !zm_terminate ) {
|
while ( !( buffer_len = ReadData(buffer) ) && !zm_terminate ) {
|
||||||
Debug(4, "Timeout waiting for REGEXP HEADER");
|
Debug(4, "Timeout waiting for REGEXP HEADER");
|
||||||
}
|
}
|
||||||
if ( buffer_len < 0 ) {
|
if ( buffer_len < 0 ) {
|
||||||
Error("Unable to read header data");
|
Error("Unable to read header data");
|
||||||
return( -1 );
|
return -1;
|
||||||
}
|
}
|
||||||
bytes += buffer_len;
|
bytes += buffer_len;
|
||||||
if ( !header_expr )
|
if ( header_expr->Match( (char*)buffer, buffer.size() ) == 2 ) {
|
||||||
header_expr = new RegExpr( "^(.+?\r?\n\r?\n)", PCRE_DOTALL );
|
|
||||||
if ( header_expr->Match( (char*)buffer, buffer.size() ) == 2 )
|
|
||||||
{
|
|
||||||
header = header_expr->MatchString( 1 );
|
header = header_expr->MatchString( 1 );
|
||||||
header_len = header_expr->MatchLength( 1 );
|
header_len = header_expr->MatchLength( 1 );
|
||||||
Debug(4, "Captured header (%d bytes):\n'%s'", header_len, header);
|
Debug(4, "Captured header (%d bytes):\n'%s'", header_len, header);
|
||||||
|
|
||||||
if ( !status_expr )
|
|
||||||
status_expr = new RegExpr( "^HTTP/(1\\.[01]) +([0-9]+) +(.+?)\r?\n", PCRE_CASELESS );
|
|
||||||
if ( status_expr->Match( header, header_len ) < 4 )
|
if ( status_expr->Match( header, header_len ) < 4 )
|
||||||
{
|
{
|
||||||
Error( "Unable to extract HTTP status from header" );
|
Error( "Unable to extract HTTP status from header" );
|
||||||
|
@ -355,24 +365,18 @@ int RemoteCameraHttp::GetResponse() {
|
||||||
}
|
}
|
||||||
Debug( 3, "Got status '%d' (%s), http version %s", status_code, status_mesg, http_version );
|
Debug( 3, "Got status '%d' (%s), http version %s", status_code, status_mesg, http_version );
|
||||||
|
|
||||||
if ( !connection_expr )
|
|
||||||
connection_expr = new RegExpr( "Connection: ?(.+?)\r?\n", PCRE_CASELESS );
|
|
||||||
if ( connection_expr->Match( header, header_len ) == 2 )
|
if ( connection_expr->Match( header, header_len ) == 2 )
|
||||||
{
|
{
|
||||||
connection_type = connection_expr->MatchString( 1 );
|
connection_type = connection_expr->MatchString( 1 );
|
||||||
Debug( 3, "Got connection '%s'", connection_type );
|
Debug( 3, "Got connection '%s'", connection_type );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( !content_length_expr )
|
|
||||||
content_length_expr = new RegExpr( "Content-length: ?([0-9]+)\r?\n", PCRE_CASELESS );
|
|
||||||
if ( content_length_expr->Match( header, header_len ) == 2 )
|
if ( content_length_expr->Match( header, header_len ) == 2 )
|
||||||
{
|
{
|
||||||
content_length = atoi( content_length_expr->MatchString( 1 ) );
|
content_length = atoi( content_length_expr->MatchString( 1 ) );
|
||||||
Debug( 3, "Got content length '%d'", content_length );
|
Debug( 3, "Got content length '%d'", content_length );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( !content_type_expr )
|
|
||||||
content_type_expr = new RegExpr( "Content-type: ?(.+?)(?:; ?boundary=\x22?(.+?)\x22?)?\r?\n", PCRE_CASELESS );
|
|
||||||
if ( content_type_expr->Match( header, header_len ) >= 2 )
|
if ( content_type_expr->Match( header, header_len ) >= 2 )
|
||||||
{
|
{
|
||||||
content_type = content_type_expr->MatchString( 1 );
|
content_type = content_type_expr->MatchString( 1 );
|
||||||
|
@ -556,17 +560,14 @@ int RemoteCameraHttp::GetResponse() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ( mode == SINGLE_IMAGE )
|
if ( mode == SINGLE_IMAGE ) {
|
||||||
{
|
|
||||||
state = HEADER;
|
state = HEADER;
|
||||||
Disconnect();
|
Disconnect();
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
state = SUBHEADER;
|
state = SUBHEADER;
|
||||||
}
|
}
|
||||||
Debug( 3, "Returning %d (%d) bytes of captured content", content_length, buffer.size() );
|
Debug( 3, "Returning %d (%d) bytes of captured content", content_length, buffer.size() );
|
||||||
return( content_length );
|
return content_length;
|
||||||
}
|
}
|
||||||
case HEADERCONT :
|
case HEADERCONT :
|
||||||
case SUBHEADERCONT :
|
case SUBHEADERCONT :
|
||||||
|
@ -576,8 +577,7 @@ int RemoteCameraHttp::GetResponse() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
} else
|
||||||
else
|
|
||||||
#endif // HAVE_LIBPCRE
|
#endif // HAVE_LIBPCRE
|
||||||
{
|
{
|
||||||
static const char *http_match = "HTTP/";
|
static const char *http_match = "HTTP/";
|
||||||
|
@ -630,7 +630,7 @@ int RemoteCameraHttp::GetResponse() {
|
||||||
static char content_boundary[64];
|
static char content_boundary[64];
|
||||||
static int content_boundary_len;
|
static int content_boundary_len;
|
||||||
|
|
||||||
while ( true ) {
|
while ( !zm_terminate ) {
|
||||||
switch( state ) {
|
switch( state ) {
|
||||||
case HEADER :
|
case HEADER :
|
||||||
{
|
{
|
||||||
|
@ -653,11 +653,11 @@ int RemoteCameraHttp::GetResponse() {
|
||||||
case HEADERCONT :
|
case HEADERCONT :
|
||||||
{
|
{
|
||||||
while ( !( buffer_len = ReadData(buffer) ) && !zm_terminate ) {
|
while ( !( buffer_len = ReadData(buffer) ) && !zm_terminate ) {
|
||||||
Debug(4, "Timeout waiting for HEADERCONT");
|
Debug(1, "Timeout waiting for HEADERCONT");
|
||||||
}
|
}
|
||||||
if ( buffer_len < 0 ) {
|
if ( buffer_len < 0 ) {
|
||||||
Error("Unable to read header");
|
Error("Unable to read header");
|
||||||
return( -1 );
|
return -1;
|
||||||
}
|
}
|
||||||
bytes += buffer_len;
|
bytes += buffer_len;
|
||||||
|
|
||||||
|
@ -670,6 +670,7 @@ int RemoteCameraHttp::GetResponse() {
|
||||||
int crlf_len = memspn(header_ptr, "\r\n", header_len);
|
int crlf_len = memspn(header_ptr, "\r\n", header_len);
|
||||||
if ( n_headers ) {
|
if ( n_headers ) {
|
||||||
if ( (crlf_len == 2 && !strncmp( header_ptr, "\n\n", crlf_len )) || (crlf_len == 4 && !strncmp( header_ptr, "\r\n\r\n", crlf_len )) ) {
|
if ( (crlf_len == 2 && !strncmp( header_ptr, "\n\n", crlf_len )) || (crlf_len == 4 && !strncmp( header_ptr, "\r\n\r\n", crlf_len )) ) {
|
||||||
|
Debug(3, "Have double linefeed, done headers");
|
||||||
*header_ptr = '\0';
|
*header_ptr = '\0';
|
||||||
header_ptr += crlf_len;
|
header_ptr += crlf_len;
|
||||||
header_len -= buffer.consume( header_ptr-(char *)buffer );
|
header_len -= buffer.consume( header_ptr-(char *)buffer );
|
||||||
|
@ -729,7 +730,7 @@ int RemoteCameraHttp::GetResponse() {
|
||||||
start_ptr = http_header;
|
start_ptr = http_header;
|
||||||
end_ptr = start_ptr+strspn( start_ptr, "10." );
|
end_ptr = start_ptr+strspn( start_ptr, "10." );
|
||||||
|
|
||||||
// FIXME WHy are we memsetting every time? Can we not do it once?
|
// FIXME Why are we memsetting every time? Can we not do it once?
|
||||||
memset( http_version, 0, sizeof(http_version) );
|
memset( http_version, 0, sizeof(http_version) );
|
||||||
strncpy( http_version, start_ptr, end_ptr-start_ptr );
|
strncpy( http_version, start_ptr, end_ptr-start_ptr );
|
||||||
|
|
||||||
|
@ -813,7 +814,7 @@ int RemoteCameraHttp::GetResponse() {
|
||||||
strcpy( content_type, start_ptr );
|
strcpy( content_type, start_ptr );
|
||||||
Debug( 3, "Got content type '%s'", content_type );
|
Debug( 3, "Got content type '%s'", content_type );
|
||||||
}
|
}
|
||||||
}
|
} // end if content_type_header
|
||||||
|
|
||||||
if ( !strcasecmp( content_type, "image/jpeg" ) || !strcasecmp( content_type, "image/jpg" ) ) {
|
if ( !strcasecmp( content_type, "image/jpeg" ) || !strcasecmp( content_type, "image/jpg" ) ) {
|
||||||
// Single image
|
// Single image
|
||||||
|
@ -851,7 +852,7 @@ int RemoteCameraHttp::GetResponse() {
|
||||||
Debug(3, "Unable to extract entire header from stream, continuing");
|
Debug(3, "Unable to extract entire header from stream, continuing");
|
||||||
state = HEADERCONT;
|
state = HEADERCONT;
|
||||||
//return( -1 );
|
//return( -1 );
|
||||||
}
|
} // end if all_headers
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case SUBHEADER :
|
case SUBHEADER :
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit ea90c0cd7f6e24333a90885e563b5d30b793db29
|
Subproject commit 7108489f218c54d36d235d3af91d6da2f8311237
|
|
@ -1 +1 @@
|
||||||
Subproject commit 0bd63fb464957080ead342db58ca9e01532cf1ef
|
Subproject commit c3976f1478c681b0bbc132ec3a3e82c3984eeed5
|
Loading…
Reference in New Issue