2003-03-26 19:57:29 +08:00
|
|
|
//
|
|
|
|
// ZoneMinder Remote Camera Class Implementation, $Date$, $Revision$
|
2006-01-17 18:56:30 +08:00
|
|
|
// Copyright (C) 2003, 2004, 2005, 2006 Philip Coombes
|
2003-03-26 19:57:29 +08:00
|
|
|
//
|
|
|
|
// 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, write to the Free Software
|
|
|
|
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <syslog.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <netdb.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
|
2003-05-16 18:27:41 +08:00
|
|
|
#include "zm.h"
|
2003-03-26 19:57:29 +08:00
|
|
|
#include "zm_remote_camera.h"
|
|
|
|
|
2008-07-14 22:43:47 +08:00
|
|
|
RemoteCamera::RemoteCamera( const char *p_host, const char *p_port, const char *p_path, int p_width, int p_height, int p_palette, int p_brightness, int p_contrast, int p_hue, int p_colour, bool p_capture ) : Camera( REMOTE_SRC, p_width, p_height, p_palette, p_brightness, p_contrast, p_hue, p_colour, p_capture )
|
2003-03-26 19:57:29 +08:00
|
|
|
{
|
2006-01-22 01:58:46 +08:00
|
|
|
strncpy( host, p_host, sizeof(host) );
|
|
|
|
strncpy( port, p_port, sizeof(port) );
|
|
|
|
strncpy( path, p_path, sizeof(path) );
|
|
|
|
|
|
|
|
auth[0] = '\0';
|
2004-02-16 03:17:38 +08:00
|
|
|
auth64[0] = '\0';
|
|
|
|
|
2003-03-26 19:57:29 +08:00
|
|
|
sd = -1;
|
|
|
|
hp = 0;
|
|
|
|
request[0] = '\0';
|
|
|
|
timeout.tv_sec = 0;
|
|
|
|
timeout.tv_usec = 0;
|
|
|
|
|
|
|
|
if ( capture )
|
|
|
|
{
|
|
|
|
Initialise();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RemoteCamera::~RemoteCamera()
|
|
|
|
{
|
|
|
|
if ( capture )
|
|
|
|
{
|
|
|
|
Terminate();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoteCamera::Initialise()
|
|
|
|
{
|
2004-02-16 03:17:38 +08:00
|
|
|
if( !host )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "No host specified for remote get" );
|
2004-02-16 03:17:38 +08:00
|
|
|
exit( -1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
if( !port )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "No port specified for remote get" );
|
2004-02-16 03:17:38 +08:00
|
|
|
exit( -1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
if( !path )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "No path specified for remote get" );
|
2004-02-16 03:17:38 +08:00
|
|
|
exit( -1 );
|
|
|
|
}
|
|
|
|
|
2003-03-26 19:57:29 +08:00
|
|
|
// Cache as much as we can to speed things up
|
2004-02-16 03:17:38 +08:00
|
|
|
char *auth_ptr = strchr( host, '@' );
|
|
|
|
|
|
|
|
if ( auth_ptr )
|
|
|
|
{
|
|
|
|
*auth_ptr = '\0';
|
2006-01-22 01:58:46 +08:00
|
|
|
strncpy( auth, host, sizeof(auth) );
|
|
|
|
strncpy( host, auth_ptr+1, sizeof(host) );
|
2004-02-16 03:17:38 +08:00
|
|
|
Base64Encode( auth, auth64 );
|
|
|
|
}
|
|
|
|
|
2003-03-26 19:57:29 +08:00
|
|
|
if ( !hp )
|
|
|
|
{
|
|
|
|
if ( !(hp = gethostbyname(host)) )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Can't gethostbyname(%s): %s", host, strerror(h_errno) );
|
2003-03-26 19:57:29 +08:00
|
|
|
exit( -1 );
|
|
|
|
}
|
|
|
|
memcpy((char *)&sa.sin_addr, (char *)hp->h_addr, hp->h_length);
|
|
|
|
sa.sin_family = hp->h_addrtype;
|
|
|
|
sa.sin_port = htons(atoi(port));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !request[0] )
|
|
|
|
{
|
2007-08-30 01:34:33 +08:00
|
|
|
snprintf( request, sizeof(request), "GET %s HTTP/%s\r\n", path, config.http_version );
|
2006-10-18 23:14:10 +08:00
|
|
|
snprintf( &(request[strlen(request)]), sizeof(request)-strlen(request), "User-Agent: %s/%s\r\n", config.http_ua, ZM_VERSION );
|
|
|
|
snprintf( &(request[strlen(request)]), sizeof(request)-strlen(request), "Host: %s\r\n", host );
|
|
|
|
snprintf( &(request[strlen(request)]), sizeof(request)-strlen(request), "Connection: Keep-Alive\r\n" );
|
2006-01-22 01:58:46 +08:00
|
|
|
if ( auth[0] )
|
2004-02-16 03:17:38 +08:00
|
|
|
{
|
2006-10-18 23:14:10 +08:00
|
|
|
snprintf( &(request[strlen(request)]), sizeof(request)-strlen(request), "Authorization: Basic %s\r\n", auth64 );
|
2004-02-16 03:17:38 +08:00
|
|
|
}
|
2006-10-18 23:14:10 +08:00
|
|
|
snprintf( &(request[strlen(request)]), sizeof(request)-strlen(request), "\r\n" );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 2, "Request: %s", request );
|
2003-03-26 19:57:29 +08:00
|
|
|
}
|
|
|
|
if ( !timeout.tv_sec )
|
|
|
|
{
|
2005-05-16 17:27:06 +08:00
|
|
|
timeout.tv_sec = config.http_timeout/1000;
|
|
|
|
timeout.tv_usec = config.http_timeout%1000;
|
2003-03-26 19:57:29 +08:00
|
|
|
}
|
2004-03-13 22:07:57 +08:00
|
|
|
|
|
|
|
int max_size = width*height*colours;
|
|
|
|
|
|
|
|
buffer.Size( max_size );
|
|
|
|
|
2006-01-20 23:20:07 +08:00
|
|
|
mode = SINGLE_IMAGE;
|
|
|
|
format = UNDEF;
|
2004-03-13 22:07:57 +08:00
|
|
|
state = HEADER;
|
2003-03-26 19:57:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int RemoteCamera::Connect()
|
|
|
|
{
|
|
|
|
if ( sd < 0 )
|
|
|
|
{
|
2005-01-17 00:39:31 +08:00
|
|
|
sd = socket( hp->h_addrtype, SOCK_STREAM, 0 );
|
2003-03-26 19:57:29 +08:00
|
|
|
if ( sd < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Can't create socket: %s", strerror(errno) );
|
2003-03-26 19:57:29 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( connect( sd, (struct sockaddr *)&sa, sizeof(sa) ) < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Can't connect to remote camera: %s", strerror(errno) );
|
2005-01-17 00:39:31 +08:00
|
|
|
Disconnect();
|
2003-03-26 19:57:29 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
}
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Connected to host, socket = %d", sd );
|
2003-03-26 19:57:29 +08:00
|
|
|
return( sd );
|
|
|
|
}
|
|
|
|
|
|
|
|
int RemoteCamera::Disconnect()
|
|
|
|
{
|
|
|
|
close( sd );
|
|
|
|
sd = -1;
|
|
|
|
return( 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
int RemoteCamera::SendRequest()
|
|
|
|
{
|
|
|
|
if ( write( sd, request, strlen(request) ) < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Can't write: %s", strerror(errno) );
|
2003-03-26 19:57:29 +08:00
|
|
|
Disconnect();
|
|
|
|
return( -1 );
|
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
format = UNDEF;
|
2004-03-13 22:07:57 +08:00
|
|
|
state = HEADER;
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Request sent" );
|
2003-03-26 19:57:29 +08:00
|
|
|
return( 0 );
|
|
|
|
}
|
|
|
|
|
2005-05-14 22:40:17 +08:00
|
|
|
int RemoteCamera::ReadData( Buffer &buffer, int bytes_expected )
|
2003-03-26 19:57:29 +08:00
|
|
|
{
|
|
|
|
fd_set rfds;
|
|
|
|
FD_ZERO(&rfds);
|
|
|
|
FD_SET(sd, &rfds);
|
|
|
|
|
2004-03-13 22:07:57 +08:00
|
|
|
struct timeval temp_timeout = timeout;
|
2003-03-26 19:57:29 +08:00
|
|
|
|
2004-03-13 22:07:57 +08:00
|
|
|
int n_found = select( sd+1, &rfds, NULL, NULL, &temp_timeout );
|
|
|
|
if( n_found == 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Select timed out" );
|
2004-03-13 22:07:57 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
else if ( n_found < 0)
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Select error: %s", strerror(errno) );
|
2004-03-13 22:07:57 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2003-06-04 05:23:16 +08:00
|
|
|
|
2004-03-13 22:07:57 +08:00
|
|
|
int total_bytes_to_read = 0;
|
2005-05-14 22:40:17 +08:00
|
|
|
|
|
|
|
if ( bytes_expected )
|
2003-03-26 19:57:29 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
total_bytes_to_read = bytes_expected;
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
else
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
if ( ioctl( sd, FIONREAD, &total_bytes_to_read ) < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Can't ioctl(): %s", strerror(errno) );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( total_bytes_to_read == 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Socket closed" );
|
2005-05-14 22:40:17 +08:00
|
|
|
Disconnect();
|
|
|
|
return( 0 );
|
|
|
|
}
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Expecting %d bytes", total_bytes_to_read );
|
2004-03-13 22:07:57 +08:00
|
|
|
|
|
|
|
int total_bytes_read = 0;
|
|
|
|
do
|
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
static unsigned char temp_buffer[5*BUFSIZ];
|
2004-03-13 22:07:57 +08:00
|
|
|
int bytes_to_read = total_bytes_to_read>sizeof(temp_buffer)?sizeof(temp_buffer):total_bytes_to_read;
|
|
|
|
int bytes_read = read( sd, temp_buffer, bytes_to_read );
|
|
|
|
|
|
|
|
if ( bytes_read < 0)
|
2003-03-26 19:57:29 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Read error: %s", strerror(errno) );
|
2003-03-26 19:57:29 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
else if ( bytes_read == 0)
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Socket closed" );
|
2005-05-14 22:40:17 +08:00
|
|
|
Disconnect();
|
|
|
|
return( 0 );
|
|
|
|
}
|
2004-03-13 22:07:57 +08:00
|
|
|
else if ( bytes_read < bytes_to_read )
|
2003-03-26 19:57:29 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Incomplete read, expected %d, got %d", bytes_to_read, bytes_read );
|
2003-03-26 19:57:29 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Read %d bytes", bytes_read );
|
2004-03-13 22:07:57 +08:00
|
|
|
buffer.Append( temp_buffer, bytes_read );
|
|
|
|
total_bytes_read += bytes_read;
|
|
|
|
total_bytes_to_read -= bytes_read;
|
|
|
|
}
|
|
|
|
while ( total_bytes_to_read );
|
2003-03-26 19:57:29 +08:00
|
|
|
|
2004-03-13 22:07:57 +08:00
|
|
|
return( total_bytes_read );
|
|
|
|
}
|
2003-03-26 19:57:29 +08:00
|
|
|
|
2004-03-13 22:07:57 +08:00
|
|
|
int RemoteCamera::GetResponse()
|
|
|
|
{
|
2004-03-15 17:43:51 +08:00
|
|
|
#if HAVE_LIBPCRE
|
2005-05-16 17:27:06 +08:00
|
|
|
if ( config.netcam_regexps )
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
const char *header = 0;
|
|
|
|
int header_len = 0;
|
|
|
|
const char *http_version = 0;
|
|
|
|
int status_code = 0;
|
|
|
|
const char *status_mesg = 0;
|
|
|
|
const char *connection_type = "";
|
|
|
|
int content_length = 0;
|
|
|
|
const char *content_type = "";
|
|
|
|
const char *content_boundary = "";
|
|
|
|
const char *subheader = 0;
|
|
|
|
int subheader_len = 0;
|
|
|
|
//int subcontent_length = 0;
|
|
|
|
//const char *subcontent_type = "";
|
|
|
|
|
|
|
|
while ( true )
|
2003-03-26 19:57:29 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
switch( state )
|
2003-03-26 19:57:29 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
case HEADER :
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
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;
|
2003-03-26 19:57:29 +08:00
|
|
|
|
2005-05-14 22:40:17 +08:00
|
|
|
int buffer_len = ReadData( buffer );
|
|
|
|
if ( buffer_len < 0 )
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to read header data" );
|
2004-03-13 22:07:57 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
if ( !header_expr )
|
|
|
|
header_expr = new RegExpr( "^(.+?\r?\n\r?\n)", PCRE_DOTALL );
|
|
|
|
if ( header_expr->Match( (char*)buffer, buffer.Size() ) == 2 )
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
header = header_expr->MatchString( 1 );
|
|
|
|
header_len = header_expr->MatchLength( 1 );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 4, "Captured header (%d bytes):\n'%s'", header_len, header );
|
2005-05-14 22:40:17 +08:00
|
|
|
|
|
|
|
if ( !status_expr )
|
|
|
|
status_expr = new RegExpr( "^HTTP/(1\\.[01]) +([0-9]+) +(.+?)\r?\n", PCRE_CASELESS );
|
|
|
|
if ( status_expr->Match( header, header_len ) < 4 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to extract HTTP status from header" );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
http_version = status_expr->MatchString( 1 );
|
|
|
|
status_code = atoi( status_expr->MatchString( 2 ) );
|
|
|
|
status_mesg = status_expr->MatchString( 3 );
|
|
|
|
|
|
|
|
if ( status_code < 200 || status_code > 299 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Invalid response status %d: %s", status_code, status_mesg );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got status '%d' (%s), http version %s", status_code, status_mesg, http_version );
|
2003-03-26 19:57:29 +08:00
|
|
|
|
2005-05-14 22:40:17 +08:00
|
|
|
if ( !connection_expr )
|
|
|
|
connection_expr = new RegExpr( "Connection: ?(.+?)\r?\n", PCRE_CASELESS );
|
|
|
|
if ( connection_expr->Match( header, header_len ) == 2 )
|
|
|
|
{
|
|
|
|
connection_type = connection_expr->MatchString( 1 );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got connection '%s'", connection_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
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 )
|
|
|
|
{
|
|
|
|
content_length = atoi( content_length_expr->MatchString( 1 ) );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got content length '%d'", content_length );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( !content_type_expr )
|
|
|
|
content_type_expr = new RegExpr( "Content-type: ?(.+?)(?:; ?boundary=(.+?))?\r?\n", PCRE_CASELESS );
|
|
|
|
if ( content_type_expr->Match( header, header_len ) >= 2 )
|
|
|
|
{
|
|
|
|
content_type = content_type_expr->MatchString( 1 );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got content type '%s'\n", content_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
if ( content_type_expr->MatchCount() > 2 )
|
|
|
|
{
|
|
|
|
content_boundary = content_type_expr->MatchString( 2 );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got content boundary '%s'", content_boundary );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !strcasecmp( content_type, "image/jpeg" ) || !strcasecmp( content_type, "image/jpg" ) )
|
|
|
|
{
|
|
|
|
// Single image
|
2006-01-20 23:20:07 +08:00
|
|
|
mode = SINGLE_IMAGE;
|
|
|
|
format = JPEG;
|
|
|
|
state = CONTENT;
|
|
|
|
}
|
|
|
|
else if ( !strcasecmp( content_type, "image/x-rgb" ) )
|
|
|
|
{
|
|
|
|
// Single image
|
|
|
|
mode = SINGLE_IMAGE;
|
|
|
|
format = X_RGB;
|
2005-05-14 22:40:17 +08:00
|
|
|
state = CONTENT;
|
|
|
|
}
|
2006-01-20 23:27:48 +08:00
|
|
|
else if ( !strcasecmp( content_type, "image/x-rgbz" ) )
|
|
|
|
{
|
|
|
|
// Single image
|
|
|
|
mode = SINGLE_IMAGE;
|
|
|
|
format = X_RGBZ;
|
|
|
|
state = CONTENT;
|
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
else if ( !strcasecmp( content_type, "multipart/x-mixed-replace" ) )
|
|
|
|
{
|
|
|
|
// Image stream, so start processing
|
|
|
|
if ( !content_boundary[0] )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "No content boundary found in header '%s'", header );
|
2006-01-16 01:39:18 +08:00
|
|
|
return( -1 );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
mode = MULTI_IMAGE;
|
2005-05-14 22:40:17 +08:00
|
|
|
state = SUBHEADER;
|
|
|
|
}
|
|
|
|
//else if ( !strcasecmp( content_type, "video/mpeg" ) || !strcasecmp( content_type, "video/mpg" ) )
|
|
|
|
//{
|
|
|
|
//// MPEG stream, coming soon!
|
|
|
|
//}
|
|
|
|
else
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unrecognised content type '%s'", content_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
buffer.Consume( header_len );
|
|
|
|
}
|
|
|
|
else
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Unable to extract header from stream, retrying" );
|
2005-05-14 22:40:17 +08:00
|
|
|
//return( -1 );
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SUBHEADER :
|
|
|
|
{
|
|
|
|
static RegExpr *subheader_expr = 0;
|
|
|
|
static RegExpr *subcontent_length_expr = 0;
|
|
|
|
static RegExpr *subcontent_type_expr = 0;
|
2003-03-26 19:57:29 +08:00
|
|
|
|
2005-05-14 22:40:17 +08:00
|
|
|
if ( !subheader_expr )
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
char subheader_pattern[256] = "";
|
|
|
|
snprintf( subheader_pattern, sizeof(subheader_pattern), "^((?:\r?\n){0,2}?(?:--)?%s\r?\n.+?\r?\n\r?\n)", content_boundary );
|
|
|
|
subheader_expr = new RegExpr( subheader_pattern, PCRE_DOTALL );
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
if ( subheader_expr->Match( (char *)buffer, (int)buffer ) == 2 )
|
|
|
|
{
|
|
|
|
subheader = subheader_expr->MatchString( 1 );
|
|
|
|
subheader_len = subheader_expr->MatchLength( 1 );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 4, "Captured subheader (%d bytes):'%s'", subheader_len, subheader );
|
2003-03-26 19:57:29 +08:00
|
|
|
|
2005-05-14 22:40:17 +08:00
|
|
|
if ( !subcontent_length_expr )
|
|
|
|
subcontent_length_expr = new RegExpr( "Content-length: ?([0-9]+)\r?\n", PCRE_CASELESS );
|
|
|
|
if ( subcontent_length_expr->Match( subheader, subheader_len ) == 2 )
|
|
|
|
{
|
|
|
|
content_length = atoi( subcontent_length_expr->MatchString( 1 ) );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got subcontent length '%d'", content_length );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( !subcontent_type_expr )
|
|
|
|
subcontent_type_expr = new RegExpr( "Content-type: ?(.+?)\r?\n", PCRE_CASELESS );
|
|
|
|
if ( subcontent_type_expr->Match( subheader, subheader_len ) == 2 )
|
|
|
|
{
|
|
|
|
content_type = subcontent_type_expr->MatchString( 1 );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got subcontent type '%s'", content_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
buffer.Consume( subheader_len );
|
|
|
|
state = CONTENT;
|
|
|
|
}
|
|
|
|
else
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Unable to extract subheader from stream, retrying" );
|
2005-05-14 22:40:17 +08:00
|
|
|
int buffer_len = ReadData( buffer );
|
|
|
|
if ( buffer_len < 0 )
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case CONTENT :
|
|
|
|
{
|
2006-01-20 23:20:07 +08:00
|
|
|
if ( !strcasecmp( content_type, "image/jpeg" ) || !strcasecmp( content_type, "image/jpg" ) )
|
|
|
|
{
|
|
|
|
format = JPEG;
|
|
|
|
}
|
|
|
|
else if ( !strcasecmp( content_type, "image/x-rgb" ) )
|
|
|
|
{
|
|
|
|
format = X_RGB;
|
|
|
|
}
|
2006-01-20 23:27:48 +08:00
|
|
|
else if ( !strcasecmp( content_type, "image/x-rgbz" ) )
|
|
|
|
{
|
|
|
|
format = X_RGBZ;
|
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
else
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Found unsupported content type '%s'", content_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2003-06-04 05:23:16 +08:00
|
|
|
|
2005-05-14 22:40:17 +08:00
|
|
|
if ( content_length )
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
while ( buffer.Size() < content_length )
|
|
|
|
{
|
|
|
|
int buffer_len = ReadData( buffer );
|
|
|
|
if ( buffer_len < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to read content" );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
}
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got end of image by length, content-length = %d", content_length );
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
else
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
while ( !content_length )
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
int buffer_len = ReadData( buffer );
|
|
|
|
if ( buffer_len < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to read content" );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
static RegExpr *content_expr = 0;
|
|
|
|
if ( buffer_len )
|
|
|
|
{
|
2006-01-20 23:20:07 +08:00
|
|
|
if ( mode == MULTI_IMAGE )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
|
|
|
if ( !content_expr )
|
|
|
|
{
|
|
|
|
char content_pattern[256] = "";
|
|
|
|
snprintf( content_pattern, sizeof(content_pattern), "^(.+?)(?:\r?\n)*(?:--)?%s\r?\n", content_boundary );
|
|
|
|
content_expr = new RegExpr( content_pattern, PCRE_DOTALL );
|
|
|
|
}
|
|
|
|
if ( content_expr->Match( buffer, buffer.Size() ) == 2 )
|
|
|
|
{
|
|
|
|
content_length = content_expr->MatchLength( 1 );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got end of image by pattern, content-length = %d", content_length );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
content_length = buffer.Size();
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got end of image by closure, content-length = %d", content_length );
|
2006-01-20 23:20:07 +08:00
|
|
|
if ( mode == SINGLE_IMAGE )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
|
|
|
if ( !content_expr )
|
|
|
|
{
|
|
|
|
content_expr = new RegExpr( "^(.+?)(?:\r?\n){1,2}?$", PCRE_DOTALL );
|
|
|
|
}
|
|
|
|
if ( content_expr->Match( buffer, buffer.Size() ) == 2 )
|
|
|
|
{
|
|
|
|
content_length = content_expr->MatchLength( 1 );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Trimmed end of image, new content-length = %d", content_length );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
if ( mode == SINGLE_IMAGE )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
|
|
|
state = HEADER;
|
|
|
|
Disconnect();
|
|
|
|
}
|
2004-03-13 22:07:57 +08:00
|
|
|
else
|
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
state = SUBHEADER;
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Returning %d (%d) bytes of captured content", content_length, buffer.Size() );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( content_length );
|
2003-06-04 05:23:16 +08:00
|
|
|
}
|
2008-03-13 21:36:12 +08:00
|
|
|
default :
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unexpected content parsing state %d", state );
|
2008-03-13 21:36:12 +08:00
|
|
|
break;
|
|
|
|
}
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif // HAVE_LIBPCRE
|
|
|
|
{
|
2005-10-17 05:34:58 +08:00
|
|
|
if ( config.netcam_regexps )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Warning( "Unable to use netcam regexps as not compiled with libpcre" );
|
2005-10-17 05:34:58 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
static const char *http_match = "HTTP/";
|
|
|
|
static const char *connection_match = "Connection:";
|
|
|
|
static const char *content_length_match = "Content-length:";
|
|
|
|
static const char *content_type_match = "Content-type:";
|
|
|
|
static const char *boundary_match = "boundary=";
|
|
|
|
static int http_match_len = 0;
|
|
|
|
static int connection_match_len = 0;
|
|
|
|
static int content_length_match_len = 0;
|
|
|
|
static int content_type_match_len = 0;
|
|
|
|
static int boundary_match_len = 0;
|
|
|
|
|
|
|
|
if ( !http_match_len )
|
|
|
|
http_match_len = strlen( http_match );
|
|
|
|
if ( !connection_match_len )
|
|
|
|
connection_match_len = strlen( connection_match );
|
|
|
|
if ( !content_length_match_len )
|
|
|
|
content_length_match_len = strlen( content_length_match );
|
|
|
|
if ( !content_type_match_len )
|
|
|
|
content_type_match_len = strlen( content_type_match );
|
|
|
|
if ( !boundary_match_len )
|
|
|
|
boundary_match_len = strlen( boundary_match );
|
|
|
|
|
|
|
|
static int n_headers;
|
|
|
|
static char *headers[32];
|
|
|
|
|
|
|
|
static int n_subheaders;
|
|
|
|
static char *subheaders[32];
|
2003-06-04 05:23:16 +08:00
|
|
|
|
2005-05-14 22:40:17 +08:00
|
|
|
static char *http_header;
|
|
|
|
static char *connection_header;
|
|
|
|
static char *content_length_header;
|
|
|
|
static char *content_type_header;
|
|
|
|
static char *boundary_header;
|
2005-06-13 18:20:08 +08:00
|
|
|
static char subcontent_length_header[32];
|
|
|
|
static char subcontent_type_header[64];
|
2005-05-14 22:40:17 +08:00
|
|
|
|
|
|
|
static char http_version[16];
|
|
|
|
static char status_code[16];
|
|
|
|
static int status;
|
|
|
|
static char status_mesg[256];
|
|
|
|
static char connection_type[32];
|
|
|
|
static int content_length;
|
|
|
|
static char content_type[32];
|
|
|
|
static char content_boundary[64];
|
2006-01-17 19:33:35 +08:00
|
|
|
static int content_boundary_len;
|
2005-05-14 22:40:17 +08:00
|
|
|
|
|
|
|
while ( true )
|
|
|
|
{
|
|
|
|
switch( state )
|
|
|
|
{
|
|
|
|
case HEADER :
|
2003-03-26 19:57:29 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
n_headers = 0;
|
|
|
|
http_header = 0;
|
|
|
|
connection_header = 0;
|
|
|
|
content_length_header = 0;
|
|
|
|
content_type_header = 0;
|
|
|
|
|
|
|
|
http_version[0] = '\0';
|
|
|
|
status_code [0]= '\0';
|
|
|
|
status = 0;
|
|
|
|
status_mesg [0]= '\0';
|
|
|
|
connection_type [0]= '\0';
|
|
|
|
content_length = 0;
|
|
|
|
content_type[0] = '\0';
|
|
|
|
content_boundary[0] = '\0';
|
2006-01-17 20:03:30 +08:00
|
|
|
content_boundary_len = 0;
|
2003-03-26 19:57:29 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
case HEADERCONT :
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
int buffer_len = ReadData( buffer );
|
|
|
|
if ( buffer_len < 0 )
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to read header" );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2003-06-04 05:23:16 +08:00
|
|
|
|
2005-05-14 22:40:17 +08:00
|
|
|
char *crlf = 0;
|
|
|
|
char *header_ptr = (char *)buffer;
|
|
|
|
int header_len = buffer.Size();
|
|
|
|
bool all_headers = false;
|
|
|
|
|
|
|
|
while( true )
|
2003-06-04 05:23:16 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
int crlf_len = memspn( header_ptr, "\r\n", header_len );
|
|
|
|
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 )) )
|
|
|
|
{
|
|
|
|
*header_ptr = '\0';
|
|
|
|
header_ptr += crlf_len;
|
|
|
|
header_len -= buffer.Consume( header_ptr-(char *)buffer );
|
|
|
|
all_headers = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( crlf_len )
|
|
|
|
{
|
|
|
|
if ( header_len == crlf_len )
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*header_ptr = '\0';
|
|
|
|
header_ptr += crlf_len;
|
|
|
|
header_len -= buffer.Consume( header_ptr-(char *)buffer );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 6, header_ptr );
|
2008-03-13 21:36:12 +08:00
|
|
|
if ( ( crlf = mempbrk( header_ptr, "\r\n", header_len ) ) )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
|
|
|
headers[n_headers++] = header_ptr;
|
|
|
|
|
|
|
|
if ( !http_header && (strncasecmp( header_ptr, http_match, http_match_len ) == 0) )
|
|
|
|
{
|
|
|
|
http_header = header_ptr+http_match_len;
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 6, "Got http header '%s'", header_ptr );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
else if ( !connection_header && (strncasecmp( header_ptr, connection_match, connection_match_len) == 0) )
|
|
|
|
{
|
|
|
|
connection_header = header_ptr+connection_match_len;
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 6, "Got connection header '%s'", header_ptr );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
else if ( !content_length_header && (strncasecmp( header_ptr, content_length_match, content_length_match_len) == 0) )
|
|
|
|
{
|
|
|
|
content_length_header = header_ptr+content_length_match_len;
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 6, "Got content length header '%s'", header_ptr );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
else if ( !content_type_header && (strncasecmp( header_ptr, content_type_match, content_type_match_len) == 0) )
|
|
|
|
{
|
|
|
|
content_type_header = header_ptr+content_type_match_len;
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 6, "Got content type header '%s'", header_ptr );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 6, "Got ignored header '%s'", header_ptr );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
header_ptr = crlf;
|
|
|
|
header_len -= buffer.Consume( header_ptr-(char *)buffer );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// No end of line found
|
|
|
|
break;
|
|
|
|
}
|
2003-06-04 05:23:16 +08:00
|
|
|
}
|
2004-03-13 22:07:57 +08:00
|
|
|
|
2005-05-14 22:40:17 +08:00
|
|
|
if ( all_headers )
|
2003-06-04 05:23:16 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
char *start_ptr, *end_ptr;
|
|
|
|
|
|
|
|
if ( !http_header )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to extract HTTP status from header" );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
start_ptr = http_header;
|
|
|
|
end_ptr = start_ptr+strspn( start_ptr, "10." );
|
|
|
|
|
|
|
|
memset( http_version, 0, sizeof(http_version) );
|
|
|
|
strncpy( http_version, start_ptr, end_ptr-start_ptr );
|
|
|
|
|
|
|
|
start_ptr = end_ptr;
|
|
|
|
start_ptr += strspn( start_ptr, " " );
|
|
|
|
end_ptr = start_ptr+strspn( start_ptr, "0123456789" );
|
|
|
|
|
|
|
|
memset( status_code, 0, sizeof(status_code) );
|
|
|
|
strncpy( status_code, start_ptr, end_ptr-start_ptr );
|
|
|
|
int status = atoi( status_code );
|
|
|
|
|
|
|
|
start_ptr = end_ptr;
|
|
|
|
start_ptr += strspn( start_ptr, " " );
|
|
|
|
strcpy( status_mesg, start_ptr );
|
|
|
|
|
|
|
|
if ( status < 200 || status > 299 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Invalid response status %s: %s", status_code, status_mesg );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got status '%d' (%s), http version %s", status, status_mesg, http_version );
|
2005-05-14 22:40:17 +08:00
|
|
|
|
|
|
|
if ( connection_header )
|
|
|
|
{
|
|
|
|
memset( connection_type, 0, sizeof(connection_type) );
|
|
|
|
start_ptr = connection_header + strspn( connection_header, " " );
|
|
|
|
strcpy( connection_type, start_ptr );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got connection '%s'", connection_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
if ( content_length_header )
|
|
|
|
{
|
|
|
|
start_ptr = content_length_header + strspn( content_length_header, " " );
|
|
|
|
content_length = atoi( start_ptr );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got content length '%d'", content_length );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
if ( content_type_header )
|
|
|
|
{
|
|
|
|
memset( content_type, 0, sizeof(content_type) );
|
|
|
|
start_ptr = content_type_header + strspn( content_type_header, " " );
|
2008-03-13 21:36:12 +08:00
|
|
|
if ( ( end_ptr = strchr( start_ptr, ';' ) ) )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
|
|
|
strncpy( content_type, start_ptr, end_ptr-start_ptr );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got content type '%s'", content_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
|
|
|
|
start_ptr = end_ptr + strspn( end_ptr, "; " );
|
|
|
|
|
|
|
|
if ( strncasecmp( start_ptr, boundary_match, boundary_match_len ) == 0 )
|
|
|
|
{
|
|
|
|
start_ptr += boundary_match_len;
|
|
|
|
start_ptr += strspn( start_ptr, "-" );
|
|
|
|
content_boundary_len = sprintf( content_boundary, "--%s", start_ptr );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got content boundary '%s'", content_boundary );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "No content boundary found in header '%s'", content_type_header );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
strcpy( content_type, start_ptr );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got content type '%s'", content_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !strcasecmp( content_type, "image/jpeg" ) || !strcasecmp( content_type, "image/jpg" ) )
|
|
|
|
{
|
|
|
|
// Single image
|
2006-01-20 23:20:07 +08:00
|
|
|
mode = SINGLE_IMAGE;
|
|
|
|
format = JPEG;
|
|
|
|
state = CONTENT;
|
|
|
|
}
|
|
|
|
else if ( !strcasecmp( content_type, "image/x-rgb" ) )
|
|
|
|
{
|
|
|
|
// Single image
|
|
|
|
mode = SINGLE_IMAGE;
|
|
|
|
format = X_RGB;
|
2005-05-14 22:40:17 +08:00
|
|
|
state = CONTENT;
|
|
|
|
}
|
2006-01-20 23:27:48 +08:00
|
|
|
else if ( !strcasecmp( content_type, "image/x-rgbz" ) )
|
|
|
|
{
|
|
|
|
// Single image
|
|
|
|
mode = SINGLE_IMAGE;
|
|
|
|
format = X_RGBZ;
|
|
|
|
state = CONTENT;
|
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
else if ( !strcasecmp( content_type, "multipart/x-mixed-replace" ) )
|
|
|
|
{
|
|
|
|
// Image stream, so start processing
|
2006-02-08 20:21:27 +08:00
|
|
|
if ( !content_boundary[0] )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "No content boundary found in header '%s'", content_type_header );
|
2006-01-16 01:39:18 +08:00
|
|
|
return( -1 );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
mode = MULTI_IMAGE;
|
2005-05-14 22:40:17 +08:00
|
|
|
state = SUBHEADER;
|
|
|
|
}
|
|
|
|
//else if ( !strcasecmp( content_type, "video/mpeg" ) || !strcasecmp( content_type, "video/mpg" ) )
|
|
|
|
//{
|
|
|
|
//// MPEG stream, coming soon!
|
|
|
|
//}
|
|
|
|
else
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unrecognised content type '%s'", content_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2003-06-04 05:23:16 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
else
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Unable to extract entire header from stream, continuing" );
|
2005-05-14 22:40:17 +08:00
|
|
|
state = HEADERCONT;
|
|
|
|
//return( -1 );
|
|
|
|
}
|
|
|
|
break;
|
2003-03-26 19:57:29 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
case SUBHEADER :
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
n_subheaders = 0;
|
|
|
|
boundary_header = 0;
|
2005-06-13 18:20:08 +08:00
|
|
|
subcontent_length_header[0] = '\0';
|
|
|
|
subcontent_type_header[0] = '\0';
|
2005-05-14 22:40:17 +08:00
|
|
|
content_length = 0;
|
|
|
|
content_type[0] = '\0';
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
case SUBHEADERCONT :
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
char *crlf = 0;
|
|
|
|
char *subheader_ptr = (char *)buffer;
|
|
|
|
int subheader_len = buffer.Size();
|
|
|
|
bool all_headers = false;
|
|
|
|
|
|
|
|
while( true )
|
|
|
|
{
|
|
|
|
int crlf_len = memspn( subheader_ptr, "\r\n", subheader_len );
|
|
|
|
if ( n_subheaders )
|
|
|
|
{
|
|
|
|
if ( (crlf_len == 2 && !strncmp( subheader_ptr, "\n\n", crlf_len )) || (crlf_len == 4 && !strncmp( subheader_ptr, "\r\n\r\n", crlf_len )) )
|
|
|
|
{
|
|
|
|
*subheader_ptr = '\0';
|
|
|
|
subheader_ptr += crlf_len;
|
|
|
|
subheader_len -= buffer.Consume( subheader_ptr-(char *)buffer );
|
|
|
|
all_headers = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( crlf_len )
|
|
|
|
{
|
|
|
|
if ( subheader_len == crlf_len )
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*subheader_ptr = '\0';
|
|
|
|
subheader_ptr += crlf_len;
|
|
|
|
subheader_len -= buffer.Consume( subheader_ptr-(char *)buffer );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 6, "%d: %s", subheader_len, subheader_ptr );
|
2005-05-14 22:40:17 +08:00
|
|
|
|
2008-03-13 21:36:12 +08:00
|
|
|
if ( ( crlf = mempbrk( subheader_ptr, "\r\n", subheader_len ) ) )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
|
|
|
subheaders[n_subheaders++] = subheader_ptr;
|
|
|
|
|
|
|
|
if ( !boundary_header && (strncasecmp( subheader_ptr, content_boundary, content_boundary_len ) == 0) )
|
|
|
|
{
|
|
|
|
boundary_header = subheader_ptr;
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 4, "Got boundary subheader '%s'", subheader_ptr );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
2005-06-13 18:20:08 +08:00
|
|
|
else if ( !subcontent_length_header[0] && (strncasecmp( subheader_ptr, content_length_match, content_length_match_len) == 0) )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
2005-06-13 18:20:08 +08:00
|
|
|
strncpy( subcontent_length_header, subheader_ptr+content_length_match_len, sizeof(subcontent_length_header) );
|
|
|
|
*(subcontent_length_header+strcspn( subcontent_length_header, "\r\n" )) = '\0';
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 4, "Got content length subheader '%s'", subcontent_length_header );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
2005-06-13 18:20:08 +08:00
|
|
|
else if ( !subcontent_type_header[0] && (strncasecmp( subheader_ptr, content_type_match, content_type_match_len) == 0) )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
2005-06-13 18:20:08 +08:00
|
|
|
strncpy( subcontent_type_header, subheader_ptr+content_type_match_len, sizeof(subcontent_type_header) );
|
|
|
|
*(subcontent_type_header+strcspn( subcontent_type_header, "\r\n" )) = '\0';
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 4, "Got content type subheader '%s'", subcontent_type_header );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 6, "Got ignored subheader '%s' found", subheader_ptr );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
subheader_ptr = crlf;
|
|
|
|
subheader_len -= buffer.Consume( subheader_ptr-(char *)buffer );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// No line end found
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( all_headers && boundary_header )
|
2004-03-13 22:07:57 +08:00
|
|
|
{
|
2008-03-13 21:36:12 +08:00
|
|
|
char *start_ptr;
|
2005-05-14 22:40:17 +08:00
|
|
|
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got boundary '%s'", boundary_header );
|
2005-05-14 22:40:17 +08:00
|
|
|
|
|
|
|
if ( subcontent_length_header )
|
|
|
|
{
|
|
|
|
start_ptr = subcontent_length_header + strspn( subcontent_length_header, " " );
|
|
|
|
content_length = atoi( start_ptr );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got subcontent length '%d'", content_length );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
if ( subcontent_type_header )
|
|
|
|
{
|
|
|
|
memset( content_type, 0, sizeof(content_type) );
|
|
|
|
start_ptr = subcontent_type_header + strspn( subcontent_type_header, " " );
|
|
|
|
strcpy( content_type, start_ptr );
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got subcontent type '%s'", content_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
state = CONTENT;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Unable to extract subheader from stream, retrying" );
|
2004-03-19 18:09:05 +08:00
|
|
|
int buffer_len = ReadData( buffer );
|
|
|
|
if ( buffer_len < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to read subheader" );
|
2004-03-19 18:09:05 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
state = SUBHEADERCONT;
|
2004-03-19 18:09:05 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
break;
|
2004-03-19 18:09:05 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
case CONTENT :
|
2004-03-19 18:09:05 +08:00
|
|
|
{
|
2006-01-20 23:20:07 +08:00
|
|
|
if ( !strcasecmp( content_type, "image/jpeg" ) || !strcasecmp( content_type, "image/jpg" ) )
|
|
|
|
{
|
|
|
|
format = JPEG;
|
|
|
|
}
|
|
|
|
else if ( !strcasecmp( content_type, "image/x-rgb" ) )
|
|
|
|
{
|
|
|
|
format = X_RGB;
|
|
|
|
}
|
2006-01-20 23:27:48 +08:00
|
|
|
else if ( !strcasecmp( content_type, "image/x-rgbz" ) )
|
|
|
|
{
|
|
|
|
format = X_RGBZ;
|
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
else
|
2004-03-19 18:09:05 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Found unsupported content type '%s'", content_type );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
|
2006-01-20 23:20:07 +08:00
|
|
|
if ( format == JPEG && buffer.Size() >= 2 )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
|
|
|
if ( buffer[0] != 0xff || buffer[1] != 0xd8 )
|
2004-03-19 18:09:05 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Found bogus jpeg header '%02x%02x'", buffer[0], buffer[1] );
|
2004-03-19 18:09:05 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( content_length )
|
|
|
|
{
|
|
|
|
while ( buffer.Size() < content_length )
|
2004-03-19 18:09:05 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
//int buffer_len = ReadData( buffer, content_length-buffer.Size() );
|
|
|
|
int buffer_len = ReadData( buffer );
|
|
|
|
if ( buffer_len < 0 )
|
2004-03-20 20:53:27 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to read content" );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
2004-03-20 20:53:27 +08:00
|
|
|
}
|
|
|
|
}
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got end of image by length, content-length = %d", content_length );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-06-13 18:20:08 +08:00
|
|
|
int content_pos = 0;
|
2005-05-14 22:40:17 +08:00
|
|
|
while ( !content_length )
|
2004-03-20 20:53:27 +08:00
|
|
|
{
|
2005-05-14 22:40:17 +08:00
|
|
|
int buffer_len = ReadData( buffer );
|
2005-06-13 18:20:08 +08:00
|
|
|
int buffer_size = buffer.Size();
|
2005-05-14 22:40:17 +08:00
|
|
|
if ( buffer_len < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to read content" );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
if ( buffer_len )
|
2004-08-11 23:24:47 +08:00
|
|
|
{
|
2006-01-20 23:20:07 +08:00
|
|
|
if ( mode == MULTI_IMAGE )
|
2004-08-11 23:24:47 +08:00
|
|
|
{
|
2005-06-13 18:20:08 +08:00
|
|
|
while ( char *start_ptr = (char *)memstr( (char *)buffer+content_pos, "\r\n--", buffer_size-content_pos ) )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
2005-06-13 18:20:08 +08:00
|
|
|
content_length = start_ptr - (char *)buffer;
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got end of image by pattern (crlf--), content-length = %d", content_length );
|
2005-06-13 18:20:08 +08:00
|
|
|
break;
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
2004-08-11 23:24:47 +08:00
|
|
|
}
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-06-13 18:20:08 +08:00
|
|
|
content_length = buffer_size;
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Got end of image by closure, content-length = %d", content_length );
|
2006-01-20 23:20:07 +08:00
|
|
|
if ( mode == SINGLE_IMAGE )
|
2004-08-11 23:24:47 +08:00
|
|
|
{
|
2005-06-13 18:20:08 +08:00
|
|
|
char *end_ptr = (char *)buffer+buffer_size;
|
2005-05-14 22:40:17 +08:00
|
|
|
|
|
|
|
while( *end_ptr == '\r' || *end_ptr == '\n' )
|
|
|
|
{
|
|
|
|
content_length--;
|
|
|
|
end_ptr--;
|
|
|
|
}
|
|
|
|
|
2005-06-13 18:20:08 +08:00
|
|
|
if ( end_ptr != ((char *)buffer+buffer_size) )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Trimmed end of image, new content-length = %d", content_length );
|
2005-05-14 22:40:17 +08:00
|
|
|
}
|
2004-08-11 23:24:47 +08:00
|
|
|
}
|
|
|
|
}
|
2004-03-19 18:09:05 +08:00
|
|
|
}
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
if ( mode == SINGLE_IMAGE )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
|
|
|
state = HEADER;
|
|
|
|
Disconnect();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
state = SUBHEADER;
|
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
|
|
|
|
if ( format == JPEG && buffer.Size() >= 2 )
|
2005-05-14 22:40:17 +08:00
|
|
|
{
|
|
|
|
if ( buffer[0] != 0xff || buffer[1] != 0xd8 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Found bogus jpeg header '%02x%02x'", buffer[0], buffer[1] );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-07-14 22:43:47 +08:00
|
|
|
Debug( 3, "Returning %d (%d) bytes of captured content", content_length, buffer.Size() );
|
2005-05-14 22:40:17 +08:00
|
|
|
return( content_length );
|
2004-03-13 22:07:57 +08:00
|
|
|
}
|
2003-03-26 19:57:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-03-13 22:07:57 +08:00
|
|
|
return( 0 );
|
2003-03-26 19:57:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int RemoteCamera::PreCapture()
|
|
|
|
{
|
|
|
|
if ( sd < 0 )
|
|
|
|
{
|
2004-03-13 22:07:57 +08:00
|
|
|
Connect();
|
|
|
|
if ( sd < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to connect to camera" );
|
2004-03-13 22:07:57 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
mode = SINGLE_IMAGE;
|
2004-03-13 22:07:57 +08:00
|
|
|
buffer.Empty();
|
2003-03-26 19:57:29 +08:00
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
if ( mode == SINGLE_IMAGE )
|
2003-03-26 19:57:29 +08:00
|
|
|
{
|
2004-03-13 22:07:57 +08:00
|
|
|
if ( SendRequest() < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to send request" );
|
2004-03-13 22:07:57 +08:00
|
|
|
Disconnect();
|
|
|
|
return( -1 );
|
|
|
|
}
|
2003-03-26 19:57:29 +08:00
|
|
|
}
|
2003-04-07 18:56:38 +08:00
|
|
|
return( 0 );
|
2003-03-26 19:57:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int RemoteCamera::PostCapture( Image &image )
|
|
|
|
{
|
2004-03-13 22:07:57 +08:00
|
|
|
int content_length = GetResponse();
|
2003-03-26 19:57:29 +08:00
|
|
|
if ( content_length < 0 )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to get response" );
|
2003-03-26 19:57:29 +08:00
|
|
|
Disconnect();
|
|
|
|
return( -1 );
|
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
switch( format )
|
2005-10-18 05:55:02 +08:00
|
|
|
{
|
2006-01-20 23:20:07 +08:00
|
|
|
case JPEG :
|
|
|
|
{
|
|
|
|
if ( !image.DecodeJpeg( buffer.Extract( content_length ), content_length ) )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to decode jpeg" );
|
2006-01-20 23:27:48 +08:00
|
|
|
Disconnect();
|
2006-01-20 23:20:07 +08:00
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case X_RGB :
|
|
|
|
{
|
|
|
|
if ( content_length != image.Size() )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Image length mismatch, expected %d bytes, content length was %d", image.Size(), content_length );
|
2006-01-20 23:20:07 +08:00
|
|
|
Disconnect();
|
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
image.Assign( width, height, colours, buffer );
|
|
|
|
break;
|
|
|
|
}
|
2006-01-20 23:27:48 +08:00
|
|
|
case X_RGBZ :
|
|
|
|
{
|
|
|
|
if ( !image.Unzip( buffer.Extract( content_length ), content_length ) )
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unable to unzip RGB image" );
|
2006-01-20 23:27:48 +08:00
|
|
|
Disconnect();
|
|
|
|
return( -1 );
|
|
|
|
}
|
|
|
|
image.Assign( width, height, colours, buffer );
|
|
|
|
break;
|
|
|
|
}
|
2006-01-20 23:20:07 +08:00
|
|
|
default :
|
|
|
|
{
|
2008-07-14 22:43:47 +08:00
|
|
|
Error( "Unexpected image format encountered" );
|
2006-01-20 23:20:07 +08:00
|
|
|
Disconnect();
|
|
|
|
return( -1 );
|
|
|
|
}
|
2005-10-18 05:55:02 +08:00
|
|
|
}
|
2003-03-26 19:57:29 +08:00
|
|
|
return( 0 );
|
|
|
|
}
|
2004-02-16 03:17:38 +08:00
|
|
|
|
|
|
|
void RemoteCamera::Base64Encode( const char *in_string, char *out_string )
|
|
|
|
{
|
|
|
|
static char base64_table[64] = { '\0' };
|
|
|
|
|
|
|
|
if ( !base64_table[0] )
|
|
|
|
{
|
|
|
|
int i = 0;
|
|
|
|
for ( char c = 'A'; c <= 'Z'; c++ )
|
|
|
|
base64_table[i++] = c;
|
|
|
|
for ( char c = 'a'; c <= 'z'; c++ )
|
|
|
|
base64_table[i++] = c;
|
|
|
|
for ( char c = '0'; c <= '9'; c++ )
|
|
|
|
base64_table[i++] = c;
|
|
|
|
base64_table[i++] = '+';
|
|
|
|
base64_table[i++] = '/';
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *in_ptr = in_string;
|
|
|
|
char *out_ptr = out_string;
|
|
|
|
while( *in_ptr )
|
|
|
|
{
|
|
|
|
unsigned char selection = *in_ptr >> 2;
|
|
|
|
unsigned char remainder = (*in_ptr++ & 0x03) << 4;
|
|
|
|
*out_ptr++ = base64_table[selection];
|
|
|
|
|
|
|
|
if ( *in_ptr )
|
|
|
|
{
|
|
|
|
selection = remainder | (*in_ptr >> 4);
|
|
|
|
remainder = (*in_ptr++ & 0x0f) << 2;
|
|
|
|
*out_ptr++ = base64_table[selection];
|
|
|
|
|
|
|
|
if ( *in_ptr )
|
|
|
|
{
|
|
|
|
selection = remainder | (*in_ptr >> 6);
|
|
|
|
*out_ptr++ = base64_table[selection];
|
|
|
|
selection = (*in_ptr++ & 0x3f);
|
|
|
|
*out_ptr++ = base64_table[selection];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*out_ptr++ = base64_table[remainder];
|
|
|
|
*out_ptr++ = '=';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*out_ptr++ = base64_table[remainder];
|
|
|
|
*out_ptr++ = '=';
|
|
|
|
*out_ptr++ = '=';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*out_ptr = '\0';
|
|
|
|
}
|