# # Copyright 2022 Centreon (http://www.centreon.com/) # # Centreon is a full-fledged industry-strength solution that meets # the needs in IT infrastructure and application monitoring for # service performance. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # package os::windows::wsman::mode::filesdate; use base qw(centreon::plugins::mode); use strict; use warnings; use DateTime; use centreon::plugins::misc; sub new { my ($class, %options) = @_; my $self = $class->SUPER::new(package => __PACKAGE__, %options, force_new_perfdata => 1); bless $self, $class; $options{options}->add_options(arguments => { 'warning:s' => { name => 'warning' }, 'critical:s' => { name => 'critical' }, 'filter-filename:s' => { name => 'filter_filename' }, 'folder:s' => { name => 'folder' } }); return $self; } sub check_options { my ($self, %options) = @_; $self->SUPER::init(%options); if (($self->{perfdata}->threshold_validate(label => 'warning', value => $self->{option_results}->{warning})) == 0) { $self->{output}->add_option_msg(short_msg => "Wrong warning threshold '" . $self->{option_results}->{warning} . "'."); $self->{output}->option_exit(); } if (($self->{perfdata}->threshold_validate(label => 'critical', value => $self->{option_results}->{critical})) == 0) { $self->{output}->add_option_msg(short_msg => "Wrong critical threshold '" . $self->{option_results}->{critical} . "'."); $self->{output}->option_exit(); } if (!defined($self->{option_results}->{folder}) || $self->{option_results}->{folder} eq '') { $self->{output}->add_option_msg(short_msg => "Need to specify folder option."); $self->{output}->option_exit(); } #### Create file path $self->{option_results}->{folder} =~ s/\//\\\\/g; } sub wmi_to_seconds { my ($self, %options) = @_; # pass in a WMI Timestamp like 2021-11-04T21:24:11.871719Z my $sec = ''; my $age_sec = ''; my $current_dt = ''; my $current_sec = ''; my $tz = ''; my $timezone_direction = '+'; my $timezone_offset = 0; # 1 2 3 4 5 6 7 8 9 if ($options{ts} =~ /^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2}).(\d*)(\S+)$/) { my %ts_info= ( year => $1, month => $2, day => $3, hour => $4, minute => $5, second => $6, nanosecond => $7, time_zone => $8 # set later ); my $dt = DateTime->new(%ts_info); $sec = $dt->epoch(); # force the current time into the same timezone as the queried system $current_dt = DateTime->now(time_zone => $dt->time_zone()); $current_sec = $current_dt->epoch(); $age_sec = $current_sec - $sec; } else { $self->{output}->add_option_msg(short_msg => 'Wrong time format'); $self->{output}->option_exit(); } return ($sec, $age_sec); } sub run { my ($self, %options) = @_; my $total_size = 0; my $current_time = time(); $self->{option_results}->{folder} =~ /^(..)(.*)$/; my ($drive, $path) = ($1, $2); my $WQL = 'Select name, lastmodified from CIM_DataFile where drive = "' . $drive . '" AND path = "' . $path . '"'; my $results = $options{wsman}->request( uri => 'http://schemas.microsoft.com/wbem/wsman/1/wmi/root/cimv2/*', wql_filter => $WQL, result_type => 'array' ); # #$VAR1 = 'CLASS: CIM_DataFile #LastModified;Name #20211224000036.262291+000;C:\\Users\\Administrator\\NTUSER.DAT'; # $self->{output}->output_add( severity => 'OK', short_msg => 'All files times are ok.' ); if (!defined($results) || scalar(@$results) <= 0) { $self->{output}->output_add( severity => 'UNKNOWN', short_msg => 'No file found.' ); } foreach (@$results) { my $last_modified = $_->{LastModified}; my $name = centreon::plugins::misc::trim($_->{Name}); my ($time, $diff_time) = $self->wmi_to_seconds(ts => $last_modified); next if (defined($self->{option_results}->{filter_filename}) && $self->{option_results}->{filter_filename} ne '' && $name !~ /$self->{option_results}->{filter_filename}/); my $exit_code = $self->{perfdata}->threshold_check( value => $diff_time, threshold => [ { label => 'critical', exit_litteral => 'critical' }, { label => 'warning', exit_litteral => 'warning' } ] ); $self->{output}->output_add(long_msg => sprintf("%s: %s seconds (time: %s)", $name, $diff_time, scalar(localtime($time)))); if (!$self->{output}->is_status(litteral => 1, value => $exit_code, compare => 'ok')) { $self->{output}->output_add( severity => $exit_code, short_msg => sprintf('%s: %s seconds (time: %s)', $name, $diff_time, scalar(localtime($time))) ); } $self->{output}->perfdata_add( label => $name, nlabel => 'file.mtime.last.seconds', unit => 's', instances => $name, value => $diff_time, warning => $self->{perfdata}->get_perfdata_for_output(label => 'warning'), critical => $self->{perfdata}->get_perfdata_for_output(label => 'critical') ); } $self->{output}->display(); $self->{output}->exit(); } 1; __END__ =head1 MODE Check files modified time. =over 8 =item B<--folder> Folder to check. (No WQL wildcard allowed) Ex: 'C:/Users/Administrator/'. =item B<--filter-filename> Filter files by name. =item B<--warning> Threshold warning in seconds for each files (diff time). =item B<--critical> Threshold critical in seconds for each files (diff time). =back =cut