2016-11-04 14:55:54 +01:00
|
|
|
#
|
2017-01-09 17:12:12 +01:00
|
|
|
# Copyright 2017 Centreon (http://www.centreon.com/)
|
2016-11-04 14:55:54 +01:00
|
|
|
#
|
|
|
|
# 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 hardware::server::hp::ilo::xmlapi::custom::api;
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use warnings;
|
|
|
|
use IO::Socket::SSL;
|
|
|
|
use LWP::UserAgent;
|
|
|
|
use XML::Simple;
|
|
|
|
|
|
|
|
sub new {
|
|
|
|
my ($class, %options) = @_;
|
|
|
|
my $self = {};
|
|
|
|
bless $self, $class;
|
|
|
|
|
|
|
|
if (!defined($options{output})) {
|
|
|
|
print "Class Custom: Need to specify 'output' argument.\n";
|
|
|
|
exit 3;
|
|
|
|
}
|
|
|
|
if (!defined($options{options})) {
|
|
|
|
$options{output}->add_option_msg(short_msg => "Class Custom: Need to specify 'options' argument.");
|
|
|
|
$options{output}->option_exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!defined($options{noptions})) {
|
|
|
|
$options{options}->add_options(arguments =>
|
|
|
|
{
|
|
|
|
"hostname:s" => { name => 'hostname' },
|
|
|
|
"timeout:s" => { name => 'timeout', default => 30 },
|
|
|
|
"port:s" => { name => 'port', default => 443 },
|
|
|
|
"username:s" => { name => 'username' },
|
|
|
|
"password:s" => { name => 'password' },
|
|
|
|
'ssl-opt:s%' => { name => 'ssl_opt' },
|
|
|
|
"force-ilo3" => { name => 'force_ilo3' },
|
|
|
|
});
|
|
|
|
}
|
|
|
|
$options{options}->add_help(package => __PACKAGE__, sections => 'XML API OPTIONS', once => 1);
|
|
|
|
|
|
|
|
$self->{output} = $options{output};
|
|
|
|
$self->{mode} = $options{mode};
|
|
|
|
|
|
|
|
return $self;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub set_options {
|
|
|
|
my ($self, %options) = @_;
|
|
|
|
|
|
|
|
$self->{option_results} = $options{option_results};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub set_defaults {
|
|
|
|
my ($self, %options) = @_;
|
|
|
|
|
|
|
|
foreach (keys %{$options{default}}) {
|
|
|
|
if ($_ eq $self->{mode}) {
|
|
|
|
for (my $i = 0; $i < scalar(@{$options{default}->{$_}}); $i++) {
|
|
|
|
foreach my $opt (keys %{$options{default}->{$_}[$i]}) {
|
|
|
|
if (!defined($self->{option_results}->{$opt}[$i])) {
|
|
|
|
$self->{option_results}->{$opt}[$i] = $options{default}->{$_}[$i]->{$opt};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub check_options {
|
|
|
|
my ($self, %options) = @_;
|
|
|
|
|
|
|
|
if (!defined($self->{option_results}->{hostname}) || $self->{option_results}->{hostname} eq '') {
|
|
|
|
$self->{output}->add_option_msg(short_msg => "Need to set hostname option.");
|
|
|
|
$self->{output}->option_exit();
|
|
|
|
}
|
|
|
|
if (!defined($self->{option_results}->{username}) || $self->{option_results}->{username} eq '') {
|
|
|
|
$self->{output}->add_option_msg(short_msg => "Need to set username option.");
|
|
|
|
$self->{output}->option_exit();
|
|
|
|
}
|
|
|
|
if (!defined($self->{option_results}->{password})) {
|
|
|
|
$self->{output}->add_option_msg(short_msg => "Need to set password option.");
|
|
|
|
$self->{output}->option_exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
$self->{ssl_opts} = '';
|
|
|
|
if (!defined($self->{option_results}->{ssl_opt})) {
|
|
|
|
$self->{ssl_opts} = 'SSL_verify_mode => SSL_VERIFY_NONE';
|
|
|
|
} else {
|
|
|
|
foreach (keys %{$self->{option_results}->{ssl_opt}}) {
|
|
|
|
$self->{ssl_opts} .= "$_ => " . $self->{option_results}->{ssl_opt}->{$_} . ", ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub find_ilo_version {
|
|
|
|
my ($self, %options) = @_;
|
|
|
|
|
|
|
|
($self->{ilo2}, $self->{ilo3}) = (0, 0);
|
|
|
|
my $client = new IO::Socket::SSL->new(PeerAddr => $self->{option_results}->{hostname} . ':' . $self->{option_results}->{port},
|
|
|
|
eval $self->{ssl_opts}, Timeout => $self->{option_results}->{timeout});
|
|
|
|
if (!$client) {
|
|
|
|
$self->{output}->add_option_msg(short_msg => "Failed to establish SSL connection: $!, ssl_error=$SSL_ERROR");
|
|
|
|
$self->{output}->option_exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
print $client 'POST /ribcl HTTP/1.1' . "\r\n";
|
|
|
|
print $client "HOST: me" . "\r\n"; # Mandatory for http 1.1
|
|
|
|
print $client "User-Agent: locfg-Perl-script/3.0\r\n";
|
|
|
|
print $client "Content-length: 30" . "\r\n"; # Mandatory for http 1.1
|
|
|
|
print $client 'Connection: Close' . "\r\n"; # Required
|
|
|
|
print $client "\r\n"; # End of http header
|
|
|
|
print $client "<RIBCL VERSION=\"2.0\"></RIBCL>\r\n"; # Used by Content-length
|
|
|
|
my $ln = <$client>;
|
|
|
|
if ($ln =~ m/HTTP.1.1 200 OK/) {
|
|
|
|
$self->{ilo3} = 1;
|
|
|
|
} else {
|
|
|
|
$self->{ilo2} = 1;
|
|
|
|
}
|
|
|
|
close $client;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub get_ilo2_data {
|
|
|
|
my ($self, %options) = @_;
|
|
|
|
|
|
|
|
my $client = new IO::Socket::SSL->new(PeerAddr => $self->{option_results}->{hostname} . ':' . $self->{option_results}->{port},
|
|
|
|
eval $self->{ssl_opts}, Timeout => $self->{option_results}->{timeout});
|
|
|
|
if (!$client) {
|
|
|
|
$self->{output}->add_option_msg(short_msg => "Failed to establish SSL connection: $!, ssl_error=$SSL_ERROR");
|
|
|
|
$self->{output}->option_exit();
|
|
|
|
}
|
|
|
|
print $client '<?xml version="1.0"?>' . "\r\n";
|
|
|
|
print $client '<RIBCL VERSION="2.21">' . "\r\n";
|
|
|
|
print $client '<LOGIN USER_LOGIN="' . $self->{option_results}->{username} . '" PASSWORD="' . $self->{option_results}->{password} . '">' . "\r\n";
|
|
|
|
print $client '<SERVER_INFO MODE="read">' . "\r\n";
|
|
|
|
print $client '<GET_EMBEDDED_HEALTH />' . "\r\n";
|
|
|
|
print $client '</SERVER_INFO>' . "\r\n";
|
|
|
|
print $client '</LOGIN>' . "\r\n";
|
|
|
|
print $client '</RIBCL>' . "\r\n";
|
|
|
|
|
|
|
|
while (my $line = <$client>) {
|
|
|
|
$self->{content} .= $line;
|
|
|
|
}
|
|
|
|
close $client;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub get_ilo3_data {
|
|
|
|
my ($self, %options) = @_;
|
|
|
|
|
|
|
|
my $xml_script = "<RIBCL VERSION=\"2.21\">
|
|
|
|
<LOGIN USER_LOGIN=\"$self->{option_results}->{username}\" PASSWORD=\"$self->{option_results}->{password}\">
|
|
|
|
<SERVER_INFO MODE=\"read\">
|
|
|
|
<GET_EMBEDDED_HEALTH />
|
|
|
|
</SERVER_INFO>
|
|
|
|
</LOGIN>
|
|
|
|
</RIBCL>
|
|
|
|
";
|
|
|
|
|
|
|
|
my $ua = LWP::UserAgent->new(keep_alive => 0, protocols_allowed => ['http', 'https'], timeout => $self->{option_results}->{timeout});
|
|
|
|
my $req = HTTP::Request->new(POST => "https://" . $self->{option_results}->{hostname} . '/ribcl');
|
|
|
|
$req->content_length(length($xml_script));
|
|
|
|
$req->content($xml_script);
|
|
|
|
$req->header(TE => 'chunked');
|
|
|
|
$req->header(Connection => 'Close');
|
|
|
|
|
|
|
|
my $context = new IO::Socket::SSL::SSL_Context(eval $self->{ssl_opts});
|
|
|
|
IO::Socket::SSL::set_default_context($context);
|
|
|
|
|
|
|
|
my $response = $ua->request($req);
|
|
|
|
$self->{content} = $response->content;
|
|
|
|
if (!$response->is_success) {
|
|
|
|
$self->{output}->add_option_msg(short_msg => "Cannot get data: $response->status_line");
|
|
|
|
$self->{output}->option_exit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub check_ilo_error {
|
|
|
|
my ($self, %options) = @_;
|
|
|
|
|
|
|
|
# Looking for:
|
|
|
|
# <RESPONSE
|
|
|
|
# STATUS="0x005F"
|
|
|
|
# MESSAGE='Login credentials rejected.'
|
|
|
|
# />
|
|
|
|
while ($self->{content} =~ /<response[^>]*?status="0x(.*?)"[^>]*?message='(.*?)'/msig) {
|
|
|
|
my ($status_code, $message) = ($1, $2);
|
|
|
|
if ($status_code !~ /^0+$/) {
|
|
|
|
$self->{output}->add_option_msg(short_msg => "Cannot get data: $2");
|
|
|
|
$self->{output}->option_exit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub change_shitty_xml {
|
|
|
|
my ($self, %options) = @_;
|
|
|
|
|
|
|
|
# Can be like that the root <RIBCL VERSION="2.22" /> ???!!
|
|
|
|
$options{response} =~ s/<RIBCL VERSION="(.*?)"\s*\/>/<RIBCL VERSION="$1">/mg;
|
|
|
|
# ILO2 can send:
|
|
|
|
# <DRIVES>
|
|
|
|
# <Backplane firmware version="Unknown", enclosure addr="224"/>
|
|
|
|
# <Drive Bay: "1"; status: "Ok"; uid led: "Off">
|
|
|
|
# <Drive Bay: "2"; status: "Ok"; uid led: "Off">
|
|
|
|
# <Drive Bay: "3"; status: "Not Installed"; uid led: "Off">
|
|
|
|
# <Drive Bay: "4"; status: "Not Installed"; uid led: "Off">
|
|
|
|
# <Backplane firmware version="1.16own", enclosure addr="226"/>
|
|
|
|
# <Drive Bay: "5"; status: "Not Installed"; uid led: "Off">
|
|
|
|
# <Drive Bay: "6"; status: "Not Installed"; uid led: "Off">
|
|
|
|
# <Drive Bay: "7"; status: "Not Installed"; uid led: "Off">
|
|
|
|
# <Drive Bay: "8"; status: "Not Installed"; uid led: "Off">
|
|
|
|
# </DRIVES>
|
|
|
|
$options{response} =~ s/<Backplane firmware version="(.*?)", enclosure addr="(.*?)"/<BACKPLANE FIRMWARE_VERSION="$1" ENCLOSURE_ADDR="$2"/mg;
|
|
|
|
$options{response} =~ s/<Drive Bay: "(.*?)"; status: "(.*?)"; uid led: "(.*?)">/<DRIVE_BAY NUM="$1" STATUS="$2" UID_LED="$3" \/>/mg;
|
|
|
|
|
|
|
|
return $options{response};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub get_ilo_response {
|
|
|
|
my ($self, %options) = @_;
|
|
|
|
|
|
|
|
# ilo result is so shitty. We get the good result from size...
|
|
|
|
my ($length, $response) = (0, '');
|
|
|
|
foreach (split /<\?xml.*?\?>/, $self->{content}) {
|
|
|
|
if (length($_) > $length) {
|
|
|
|
$response = $_;
|
|
|
|
$length = length($_);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$response = $self->change_shitty_xml(response => $response);
|
|
|
|
my $xml_result;
|
|
|
|
eval {
|
|
|
|
$xml_result = XMLin($response,
|
|
|
|
ForceArray => ['FAN', 'TEMP', 'MODULE', 'SUPPLY', 'PROCESSOR', 'NIC',
|
|
|
|
'SMART_STORAGE_BATTERY', 'CONTROLLER', 'DRIVE_ENCLOSURE',
|
|
|
|
'LOGICAL_DRIVE', 'PHYSICAL_DRIVE', 'DRIVE_BAY']);
|
|
|
|
};
|
|
|
|
if ($@) {
|
|
|
|
$self->{output}->add_option_msg(short_msg => "Cannot decode xml response: $@");
|
|
|
|
$self->{output}->option_exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $xml_result;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub get_ilo_data {
|
|
|
|
my ($self, %options) = @_;
|
|
|
|
|
|
|
|
$self->{content} = '';
|
|
|
|
if (!defined($self->{option_results}->{force_ilo3})) {
|
|
|
|
$self->find_ilo_version();
|
|
|
|
} else {
|
|
|
|
$self->{ilo3} = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($self->{ilo3} == 1) {
|
|
|
|
$self->get_ilo3_data();
|
|
|
|
} else {
|
|
|
|
$self->get_ilo2_data();
|
|
|
|
}
|
|
|
|
|
|
|
|
$self->{content} =~ s/\r//sg;
|
|
|
|
$self->{output}->output_add(long_msg => $self->{content}, debug => 1);
|
|
|
|
|
|
|
|
$self->check_ilo_error();
|
|
|
|
return $self->get_ilo_response();
|
|
|
|
}
|
|
|
|
|
|
|
|
1;
|
|
|
|
|
|
|
|
__END__
|
|
|
|
|
|
|
|
=head1 NAME
|
|
|
|
|
|
|
|
ILO API
|
|
|
|
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
|
|
|
|
ilo api
|
|
|
|
|
|
|
|
=head1 XML API OPTIONS
|
|
|
|
|
|
|
|
=over 8
|
|
|
|
|
|
|
|
=item B<--hostname>
|
|
|
|
|
|
|
|
Hostname to query.
|
|
|
|
|
|
|
|
=item B<--username>
|
|
|
|
|
|
|
|
ILO username.
|
|
|
|
|
|
|
|
=item B<--password>
|
|
|
|
|
|
|
|
ILO password.
|
|
|
|
|
|
|
|
=item B<--port>
|
|
|
|
|
|
|
|
ILO Port (Default: 443).
|
|
|
|
|
|
|
|
=item B<--timeout>
|
|
|
|
|
|
|
|
Set timeout (Default: 30).
|
|
|
|
|
|
|
|
=item B<--force-ilo3>
|
|
|
|
|
|
|
|
Don't try to find ILO version.
|
|
|
|
|
|
|
|
=item B<--ssl-opt>
|
|
|
|
|
|
|
|
Set SSL Options (--ssl-opt="SSL_version=SSLv3").
|
2017-06-28 15:28:24 +02:00
|
|
|
Default: --ssl-opt="SSL_verify_mode=SSL_VERIFY_NONE"
|
2016-11-04 14:55:54 +01:00
|
|
|
|
|
|
|
=back
|
|
|
|
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
|
|
|
|
B<custom>.
|
|
|
|
|
|
|
|
=cut
|