Merge branch 'storageareas' of github.com:ConnorTechnology/ZoneMinder into storageareas
This commit is contained in:
commit
1773b00aa7
|
@ -31,12 +31,16 @@ use warnings;
|
||||||
require ZoneMinder::Base;
|
require ZoneMinder::Base;
|
||||||
require ZoneMinder::Object;
|
require ZoneMinder::Object;
|
||||||
require ZoneMinder::Storage;
|
require ZoneMinder::Storage;
|
||||||
|
require ZoneMinder::Frame;
|
||||||
require Date::Manip;
|
require Date::Manip;
|
||||||
require File::Find;
|
require File::Find;
|
||||||
require File::Path;
|
require File::Path;
|
||||||
require File::Copy;
|
require File::Copy;
|
||||||
require File::Basename;
|
require File::Basename;
|
||||||
require Number::Bytes::Human;
|
require Number::Bytes::Human;
|
||||||
|
require Date::Parse;
|
||||||
|
require POSIX;
|
||||||
|
use Date::Format qw(time2str);
|
||||||
|
|
||||||
#our @ISA = qw(ZoneMinder::Object);
|
#our @ISA = qw(ZoneMinder::Object);
|
||||||
use parent qw(ZoneMinder::Object);
|
use parent qw(ZoneMinder::Object);
|
||||||
|
@ -50,7 +54,6 @@ use parent qw(ZoneMinder::Object);
|
||||||
use ZoneMinder::Config qw(:all);
|
use ZoneMinder::Config qw(:all);
|
||||||
use ZoneMinder::Logger qw(:all);
|
use ZoneMinder::Logger qw(:all);
|
||||||
use ZoneMinder::Database qw(:all);
|
use ZoneMinder::Database qw(:all);
|
||||||
require Date::Parse;
|
|
||||||
|
|
||||||
use vars qw/ $table $primary_key %fields $serial @identified_by %defaults/;
|
use vars qw/ $table $primary_key %fields $serial @identified_by %defaults/;
|
||||||
$table = 'Events';
|
$table = 'Events';
|
||||||
|
@ -96,7 +99,6 @@ $serial = $primary_key = 'Id';
|
||||||
Executed => '0',
|
Executed => '0',
|
||||||
);
|
);
|
||||||
|
|
||||||
use POSIX;
|
|
||||||
|
|
||||||
sub Time {
|
sub Time {
|
||||||
if ( @_ > 1 ) {
|
if ( @_ > 1 ) {
|
||||||
|
@ -158,7 +160,7 @@ sub RelativePath {
|
||||||
if ( $event->Time() ) {
|
if ( $event->Time() ) {
|
||||||
$$event{RelativePath} = join('/',
|
$$event{RelativePath} = join('/',
|
||||||
$event->{MonitorId},
|
$event->{MonitorId},
|
||||||
strftime( '%y/%m/%d/%H/%M/%S',
|
POSIX::strftime( '%y/%m/%d/%H/%M/%S',
|
||||||
localtime($event->Time())
|
localtime($event->Time())
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
@ -170,7 +172,7 @@ sub RelativePath {
|
||||||
if ( $event->Time() ) {
|
if ( $event->Time() ) {
|
||||||
$$event{RelativePath} = join('/',
|
$$event{RelativePath} = join('/',
|
||||||
$event->{MonitorId},
|
$event->{MonitorId},
|
||||||
strftime('%Y-%m-%d', localtime($event->Time())),
|
POSIX::strftime('%Y-%m-%d', localtime($event->Time())),
|
||||||
$event->{Id},
|
$event->{Id},
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
@ -198,7 +200,7 @@ sub LinkPath {
|
||||||
if ( $event->Time() ) {
|
if ( $event->Time() ) {
|
||||||
$$event{LinkPath} = join('/',
|
$$event{LinkPath} = join('/',
|
||||||
$event->{MonitorId},
|
$event->{MonitorId},
|
||||||
strftime( '%y/%m/%d',
|
POSIX::strftime( '%y/%m/%d',
|
||||||
localtime($event->Time())
|
localtime($event->Time())
|
||||||
),
|
),
|
||||||
'.'.$$event{Id}
|
'.'.$$event{Id}
|
||||||
|
@ -678,9 +680,19 @@ sub recover_timestamps {
|
||||||
closedir(DIR);
|
closedir(DIR);
|
||||||
|
|
||||||
my @mp4_files = grep( /^\d+\-video\.mp4$/, @contents);
|
my @mp4_files = grep( /^\d+\-video\.mp4$/, @contents);
|
||||||
my @capture_jpgs = grep( /^\d+\-capture\.jpg$/, @contents);
|
if ( @mp4_files ) {
|
||||||
|
$$Event{DefaultVideo} = $mp4_files[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
my @analyse_jpgs = grep( /^\d+\-analyse\.jpg$/, @contents);
|
||||||
|
if ( @analyse_jpgs ) {
|
||||||
|
$$Event{Save_JPEGs} |= 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
my @capture_jpgs = grep( /^\d+\-capture\.jpg$/, @contents);
|
||||||
if ( @capture_jpgs ) {
|
if ( @capture_jpgs ) {
|
||||||
|
$$Event{Frames} = scalar @capture_jpgs;
|
||||||
|
$$Event{Save_JPEGs} |= 1;
|
||||||
# can get start and end times from stat'ing first and last jpg
|
# can get start and end times from stat'ing first and last jpg
|
||||||
@capture_jpgs = sort { $a cmp $b } @capture_jpgs;
|
@capture_jpgs = sort { $a cmp $b } @capture_jpgs;
|
||||||
my $first_file = "$path/$capture_jpgs[0]";
|
my $first_file = "$path/$capture_jpgs[0]";
|
||||||
|
@ -696,6 +708,25 @@ sub recover_timestamps {
|
||||||
$Event->StartTime( Date::Format::time2str('%Y-%m-%d %H:%M:%S', $first_timestamp) );
|
$Event->StartTime( Date::Format::time2str('%Y-%m-%d %H:%M:%S', $first_timestamp) );
|
||||||
$Event->EndTime( Date::Format::time2str('%Y-%m-%d %H:%M:%S', $last_timestamp) );
|
$Event->EndTime( Date::Format::time2str('%Y-%m-%d %H:%M:%S', $last_timestamp) );
|
||||||
Debug("From capture Jpegs have duration $duration = $last_timestamp - $first_timestamp : $$Event{StartTime} to $$Event{EndTime}");
|
Debug("From capture Jpegs have duration $duration = $last_timestamp - $first_timestamp : $$Event{StartTime} to $$Event{EndTime}");
|
||||||
|
$ZoneMinder::Database::dbh->begin_work();
|
||||||
|
foreach my $jpg ( @capture_jpgs ) {
|
||||||
|
my ( $id ) = $jpg =~ /^(\d+)\-capture\.jpg$/;
|
||||||
|
|
||||||
|
if ( ! ZoneMinder::Frame->find_one( EventId=>$$Event{Id}, FrameId=>$id ) ) {
|
||||||
|
my $file = "$path/$jpg";
|
||||||
|
( $file ) = $file =~ /^(.*)$/;
|
||||||
|
my $timestamp = (stat($file))[9];
|
||||||
|
my $Frame = new ZoneMinder::Frame();
|
||||||
|
$Frame->save({
|
||||||
|
EventId=>$$Event{Id}, FrameId=>$id,
|
||||||
|
TimeStamp=>Date::Format::time2str('%Y-%m-%d %H:%M:%S',$timestamp),
|
||||||
|
Delta => $timestamp - $first_timestamp,
|
||||||
|
Type=>'Normal',
|
||||||
|
Score=>0,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$ZoneMinder::Database::dbh->commit();
|
||||||
} elsif ( @mp4_files ) {
|
} elsif ( @mp4_files ) {
|
||||||
my $file = "$path/$mp4_files[0]";
|
my $file = "$path/$mp4_files[0]";
|
||||||
( $file ) = $file =~ /^(.*)$/;
|
( $file ) = $file =~ /^(.*)$/;
|
||||||
|
|
|
@ -30,6 +30,7 @@ use warnings;
|
||||||
|
|
||||||
require ZoneMinder::Base;
|
require ZoneMinder::Base;
|
||||||
require Date::Manip;
|
require Date::Manip;
|
||||||
|
require POSIX;
|
||||||
|
|
||||||
use parent qw(ZoneMinder::Object);
|
use parent qw(ZoneMinder::Object);
|
||||||
|
|
||||||
|
@ -48,8 +49,6 @@ use ZoneMinder::Database qw(:all);
|
||||||
require ZoneMinder::Storage;
|
require ZoneMinder::Storage;
|
||||||
require ZoneMinder::Server;
|
require ZoneMinder::Server;
|
||||||
|
|
||||||
use POSIX;
|
|
||||||
|
|
||||||
sub Name {
|
sub Name {
|
||||||
if ( @_ > 1 ) {
|
if ( @_ > 1 ) {
|
||||||
$_[0]{Name} = $_[1];
|
$_[0]{Name} = $_[1];
|
||||||
|
@ -435,7 +434,7 @@ sub DateTimeToSQL {
|
||||||
Error( "Unable to parse date string '$dt_str'\n" );
|
Error( "Unable to parse date string '$dt_str'\n" );
|
||||||
return( undef );
|
return( undef );
|
||||||
}
|
}
|
||||||
return( strftime( "%Y-%m-%d %H:%M:%S", localtime( $dt_val ) ) );
|
return( POSIX::strftime( "%Y-%m-%d %H:%M:%S", localtime( $dt_val ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
1;
|
1;
|
||||||
|
|
|
@ -91,7 +91,7 @@ use ZoneMinder::Config qw(:all);
|
||||||
|
|
||||||
use DBI;
|
use DBI;
|
||||||
use Carp;
|
use Carp;
|
||||||
use POSIX;
|
require POSIX;
|
||||||
use IO::Handle;
|
use IO::Handle;
|
||||||
use Data::Dumper;
|
use Data::Dumper;
|
||||||
use Time::HiRes qw/gettimeofday/;
|
use Time::HiRes qw/gettimeofday/;
|
||||||
|
@ -539,7 +539,7 @@ sub logPrint {
|
||||||
if ( $level <= $this->{fileLevel} or $level <= $this->{termLevel} ) {
|
if ( $level <= $this->{fileLevel} or $level <= $this->{termLevel} ) {
|
||||||
my $message = sprintf(
|
my $message = sprintf(
|
||||||
'%s.%06d %s[%d].%s [%s:%d] [%s]'
|
'%s.%06d %s[%d].%s [%s:%d] [%s]'
|
||||||
, strftime('%x %H:%M:%S', localtime($seconds))
|
, POSIX::strftime('%x %H:%M:%S', localtime($seconds))
|
||||||
, $microseconds
|
, $microseconds
|
||||||
, $this->{id}
|
, $this->{id}
|
||||||
, $$
|
, $$
|
||||||
|
|
|
@ -341,7 +341,7 @@ $log->debug("No serial") if $debug;
|
||||||
} # end if
|
} # end if
|
||||||
} # end if
|
} # end if
|
||||||
ZoneMinder::Database::end_transaction( $local_dbh, $ac );
|
ZoneMinder::Database::end_transaction( $local_dbh, $ac );
|
||||||
$self->load();
|
#$self->load();
|
||||||
#if ( $$fields{id} ) {
|
#if ( $$fields{id} ) {
|
||||||
#if ( ! $ZoneMinder::Object::cache{$type}{$$self{id}} ) {
|
#if ( ! $ZoneMinder::Object::cache{$type}{$$self{id}} ) {
|
||||||
#$ZoneMinder::Object::cache{$type}{$$self{id}} = $self;
|
#$ZoneMinder::Object::cache{$type}{$$self{id}} = $self;
|
||||||
|
|
|
@ -25,7 +25,6 @@ use POSIX;
|
||||||
use File::Find;
|
use File::Find;
|
||||||
use Time::HiRes qw/gettimeofday/;
|
use Time::HiRes qw/gettimeofday/;
|
||||||
use Getopt::Long;
|
use Getopt::Long;
|
||||||
use Date::Format;
|
|
||||||
use autouse 'Pod::Usage'=>qw(pod2usage);
|
use autouse 'Pod::Usage'=>qw(pod2usage);
|
||||||
|
|
||||||
use constant ZM_RECOVER_PID => '@ZM_RUNDIR@/zmrecover.pid';
|
use constant ZM_RECOVER_PID => '@ZM_RUNDIR@/zmrecover.pid';
|
||||||
|
@ -231,7 +230,6 @@ Debug("@Monitors");
|
||||||
$Event->Width( $Monitor->Width() );
|
$Event->Width( $Monitor->Width() );
|
||||||
$Event->Height( $Monitor->Height() );
|
$Event->Height( $Monitor->Height() );
|
||||||
$Event->Orientation( $Monitor->Orientation() );
|
$Event->Orientation( $Monitor->Orientation() );
|
||||||
|
|
||||||
$Event->recover_timestamps();
|
$Event->recover_timestamps();
|
||||||
|
|
||||||
$Event->save({}, 1);
|
$Event->save({}, 1);
|
||||||
|
@ -343,6 +341,7 @@ Debug("@Monitors");
|
||||||
$$Event{Id} = $event_id;
|
$$Event{Id} = $event_id;
|
||||||
$$Event{Path} = join('/', $Storage->Path(), $event_dir );
|
$$Event{Path} = join('/', $Storage->Path(), $event_dir );
|
||||||
Debug("Have event $$Event{Id} at $$Event{Path}");
|
Debug("Have event $$Event{Id} at $$Event{Path}");
|
||||||
|
if ( confirm() ) {
|
||||||
$$Event{Scheme} = 'Medium';
|
$$Event{Scheme} = 'Medium';
|
||||||
$$Event{RelativePath} = $event_dir;
|
$$Event{RelativePath} = $event_dir;
|
||||||
$$Event{Name} = "Event $event_id recovered";
|
$$Event{Name} = "Event $event_id recovered";
|
||||||
|
@ -352,7 +351,6 @@ Debug("@Monitors");
|
||||||
$Event->Orientation( $Monitor->Orientation() );
|
$Event->Orientation( $Monitor->Orientation() );
|
||||||
$Event->StorageId( $Storage->Id() );
|
$Event->StorageId( $Storage->Id() );
|
||||||
$Event->recover_timestamps();
|
$Event->recover_timestamps();
|
||||||
if ( confirm() ) {
|
|
||||||
$Event->save({}, 1);
|
$Event->save({}, 1);
|
||||||
Debug("Event resurrected as " . $Event->to_string() );
|
Debug("Event resurrected as " . $Event->to_string() );
|
||||||
}
|
}
|
||||||
|
@ -380,6 +378,7 @@ Debug("@Monitors");
|
||||||
$$Event{Id} = $event;
|
$$Event{Id} = $event;
|
||||||
$$Event{Path} = join('/', $Storage->Path(), $event );
|
$$Event{Path} = join('/', $Storage->Path(), $event );
|
||||||
Debug("Have event $$Event{Id} at $$Event{Path}");
|
Debug("Have event $$Event{Id} at $$Event{Path}");
|
||||||
|
if ( confirm() ) {
|
||||||
$$Event{Scheme} = 'Shallow';
|
$$Event{Scheme} = 'Shallow';
|
||||||
$$Event{Name} = "Event $event recovered";
|
$$Event{Name} = "Event $event recovered";
|
||||||
#$$Event{Path} = $event_path;
|
#$$Event{Path} = $event_path;
|
||||||
|
@ -391,6 +390,7 @@ Debug("@Monitors");
|
||||||
$Event->recover_timestamps();
|
$Event->recover_timestamps();
|
||||||
$Event->save({}, 1);
|
$Event->save({}, 1);
|
||||||
Debug("Event resurrected as " . $Event->to_string() );
|
Debug("Event resurrected as " . $Event->to_string() );
|
||||||
|
}
|
||||||
} # end foreach event
|
} # end foreach event
|
||||||
chdir( $Storage->Path() );
|
chdir( $Storage->Path() );
|
||||||
} # end foreach monitor
|
} # end foreach monitor
|
||||||
|
|
|
@ -109,7 +109,10 @@ bool EventStream::loadInitialEventData( uint64_t init_event_id, unsigned int ini
|
||||||
bool EventStream::loadEventData(uint64_t event_id) {
|
bool EventStream::loadEventData(uint64_t event_id) {
|
||||||
static char sql[ZM_SQL_MED_BUFSIZ];
|
static char sql[ZM_SQL_MED_BUFSIZ];
|
||||||
|
|
||||||
snprintf(sql, sizeof(sql), "SELECT MonitorId, StorageId, Frames, unix_timestamp( StartTime ) AS StartTimestamp, (SELECT max(Delta)-min(Delta) FROM Frames WHERE EventId=Events.Id) AS Duration, DefaultVideo, Scheme FROM Events WHERE Id = %" PRIu64, event_id);
|
snprintf(sql, sizeof(sql),
|
||||||
|
"SELECT MonitorId, StorageId, Frames, unix_timestamp( StartTime ) AS StartTimestamp, "
|
||||||
|
"(SELECT max(Delta)-min(Delta) FROM Frames WHERE EventId=Events.Id) AS Duration, "
|
||||||
|
"DefaultVideo, Scheme, SaveJPEGs FROM Events WHERE Id = %" PRIu64, event_id);
|
||||||
|
|
||||||
if ( mysql_query(&dbconn, sql) ) {
|
if ( mysql_query(&dbconn, sql) ) {
|
||||||
Error("Can't run query: %s", mysql_error(&dbconn));
|
Error("Can't run query: %s", mysql_error(&dbconn));
|
||||||
|
@ -151,6 +154,7 @@ bool EventStream::loadEventData(uint64_t event_id) {
|
||||||
} else {
|
} else {
|
||||||
event_data->scheme = Storage::SHALLOW;
|
event_data->scheme = Storage::SHALLOW;
|
||||||
}
|
}
|
||||||
|
event_data->SaveJPEGs = dbrow[7] == NULL ? 0 : atoi(dbrow[7]);
|
||||||
mysql_free_result( result );
|
mysql_free_result( result );
|
||||||
|
|
||||||
Storage * storage = new Storage(event_data->storage_id);
|
Storage * storage = new Storage(event_data->storage_id);
|
||||||
|
@ -241,6 +245,7 @@ bool EventStream::loadEventData(uint64_t event_id) {
|
||||||
if ( event_data->video_file[0] ) {
|
if ( event_data->video_file[0] ) {
|
||||||
char filepath[PATH_MAX];
|
char filepath[PATH_MAX];
|
||||||
snprintf(filepath, sizeof(filepath), "%s/%s", event_data->path, event_data->video_file);
|
snprintf(filepath, sizeof(filepath), "%s/%s", event_data->path, event_data->video_file);
|
||||||
|
Debug(1, "Loading video file from %s", filepath);
|
||||||
ffmpeg_input = new FFmpeg_Input();
|
ffmpeg_input = new FFmpeg_Input();
|
||||||
if ( 0 > ffmpeg_input->Open( filepath ) ) {
|
if ( 0 > ffmpeg_input->Open( filepath ) ) {
|
||||||
Warning("Unable to open ffmpeg_input %s/%s", event_data->path, event_data->video_file);
|
Warning("Unable to open ffmpeg_input %s/%s", event_data->path, event_data->video_file);
|
||||||
|
@ -570,15 +575,18 @@ bool EventStream::sendFrame( int delta_us ) {
|
||||||
|
|
||||||
// This needs to be abstracted. If we are saving jpgs, then load the capture file. If we are only saving analysis frames, then send that.
|
// This needs to be abstracted. If we are saving jpgs, then load the capture file. If we are only saving analysis frames, then send that.
|
||||||
// // This is also wrong, need to have this info stored in the event! FIXME
|
// // This is also wrong, need to have this info stored in the event! FIXME
|
||||||
if ( monitor->GetOptSaveJPEGs() & 1 ) {
|
if ( event_data->SaveJPEGs & 1 ) {
|
||||||
snprintf( filepath, sizeof(filepath), staticConfig.capture_file_format, event_data->path, curr_frame_id );
|
snprintf( filepath, sizeof(filepath), staticConfig.capture_file_format, event_data->path, curr_frame_id );
|
||||||
} else if ( monitor->GetOptSaveJPEGs() & 2 ) {
|
} else if ( event_data->SaveJPEGs & 2 ) {
|
||||||
snprintf( filepath, sizeof(filepath), staticConfig.analyse_file_format, event_data->path, curr_frame_id );
|
snprintf( filepath, sizeof(filepath), staticConfig.analyse_file_format, event_data->path, curr_frame_id );
|
||||||
if ( stat( filepath, &filestat ) < 0 ) {
|
if ( stat( filepath, &filestat ) < 0 ) {
|
||||||
Debug(1, "analyze file %s not found will try to stream from other", filepath);
|
Debug(1, "analyze file %s not found will try to stream from other", filepath);
|
||||||
snprintf( filepath, sizeof(filepath), staticConfig.capture_file_format, event_data->path, curr_frame_id );
|
snprintf( filepath, sizeof(filepath), staticConfig.capture_file_format, event_data->path, curr_frame_id );
|
||||||
|
if ( stat( filepath, &filestat ) < 0 ) {
|
||||||
|
Debug(1, "capture file %s not found either", filepath);
|
||||||
filepath[0] = 0;
|
filepath[0] = 0;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
} else if ( ! ffmpeg_input ) {
|
} else if ( ! ffmpeg_input ) {
|
||||||
Fatal("JPEGS not saved.zms is not capable of streaming jpegs from mp4 yet");
|
Fatal("JPEGS not saved.zms is not capable of streaming jpegs from mp4 yet");
|
||||||
|
|
|
@ -65,6 +65,7 @@ class EventStream : public StreamBase {
|
||||||
FrameData *frames;
|
FrameData *frames;
|
||||||
char video_file[PATH_MAX];
|
char video_file[PATH_MAX];
|
||||||
Storage::Schemes scheme;
|
Storage::Schemes scheme;
|
||||||
|
int SaveJPEGs;
|
||||||
};
|
};
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
Loading…
Reference in New Issue