2
0
mirror of https://github.com/xcat2/xcat-core.git synced 2025-10-26 08:55:24 +00:00
Files
xcat-core/xCAT-server/lib/xcat/plugins/rhevm.pm
2017-03-21 12:01:42 -04:00

4037 lines
148 KiB
Perl

# IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html
# This module is used to support the rhev
# The rhev-m is used as the agent to manage the storage domain, network, rhev-h
# There are concepts Datacenter and Cluster for the rhev, SD and network belongs to DC,
# rhev-h (host) belongs to cluster
#
# When installing rhev-h, it will try to register to rhev-m. From xCAT point of view, just approve.
# The fence must be configured for the rhev-h to enable the SD management, otherwise when
# when the SPM failed, the failure take over for the SPM cannot happen automatically.
#
# The SD needs to be created with a host as the SPM (Storage Pool Manager), when the current
# SPM failed, SPM will switch to another available rhev-h.
#
# Add rhevh to the management
# for the rhevh, the adding should be done automatically that specify the rhev-m infor when installing of the rhev-h
# Need to run the approval for the host to be added to the rhevm
#
# The features that are not used
# tag, to catalog the resources with tag
# role, for access permission
# domain, for user/group management
#TODO: handle the functions base on the version
#TODO: add the support of iscsi storage domain
package xCAT_plugin::rhevm;
use strict;
use warnings;
use POSIX qw(WNOHANG nice);
use POSIX qw(WNOHANG setsid :errno_h);
use IO::Select;
require IO::Socket::SSL; IO::Socket::SSL->import('inet4');
use Time::HiRes qw(gettimeofday sleep);
use Fcntl qw/:DEFAULT :flock/;
use File::Path;
use File::Copy;
use Getopt::Long;
Getopt::Long::Configure("bundling");
Getopt::Long::Configure("pass_through");
use HTTP::Headers;
use HTTP::Request;
use XML::LibXML;
use xCAT::Utils;
use xCAT::TableUtils;
use xCAT::ServiceNodeUtils;
use xCAT::Table;
use xCAT::MsgUtils;
use xCAT::Usage;
sub handled_commands {
return {
copycd => 'rhevm',
mkinstall => "nodetype:os=(rhevh.*)",
rpower => 'nodehm:power,mgt',
rsetboot => 'nodehm:mgt',
rmigrate => 'nodehm:mgt',
cfgve => 'rhevm',
lsve => 'rhevm',
lsvm => [ 'hypervisor:type=(rhev.*)', 'nodehm:mgt' ],
mkvm => 'nodehm:mgt',
rmvm => 'nodehm:mgt',
clonevm => 'nodehm:mgt',
#rinv => 'nodehm:mgt',
chvm => 'nodehm:mgt',
#rshutdown => "nodetype:os=(rhev.*)",
rmhypervisor => [ 'hypervisor:type', 'nodetype:os=(rhev.*)' ],
chhypervisor => [ 'hypervisor:type', 'nodetype:os=(rhev.*)' ],
rhevhupdateflag => "nodetype:os=(rhevh.*)",
getrvidparms => 'nodehm:mgt',
};
}
my $verbose;
my $global_callback;
sub preprocess_request {
my $request = shift;
my $callback = shift;
#if already preprocessed, go straight to request
if ((defined($request->{_xcatpreprocessed}->[0]))
&& ($request->{_xcatpreprocessed}->[0] == 1)) {
return [$request];
}
unless ($request and $request->{command} and $request->{command}->[0]) { return; }
if ($request->{command}->[0] eq 'copycd') {
return [$request];
}
my $nodes = $request->{node};
my $command = $request->{command}->[0];
my $extraargs = $request->{arg};
if ($extraargs) {
@ARGV = @{$extraargs};
my $help;
my $ver;
GetOptions("V" => \$verbose, 'h|help' => \$help, 'v|version' => \$ver);
$global_callback = $callback;
if ($help) {
my $usage_string = xCAT::Usage->getUsage($command);
my $rsp;
push @{ $rsp->{data} }, $usage_string;
xCAT::MsgUtils->message("I", $rsp, $callback);
return ();
}
if ($ver) {
my $ver_string = xCAT::Usage->getVersion($command);
my $rsp;
push @{ $rsp->{data} }, $ver_string;
xCAT::MsgUtils->message("I", $rsp, $callback);
return ();
}
}
#pdu commands will be handled in the pdu plugin
if (($extraargs->[0] eq 'pdustat') || ($extraargs->[0] eq 'pdureset') || ($extraargs->[0] eq 'pduon') || ($extraargs->[0] eq 'pduoff')) {
return;
}
# Read the user password for the rhevm
# Only support the general password in passwd table
my $passtab = xCAT::Table->new('passwd');
my ($rhevmadminpw, $rhevhrootpw, $rhevhadminpw);
if ($passtab) {
my $pw = $passtab->getAttribs({ 'key' => 'rhevm', 'username' => 'admin' }, 'password');
if (defined($pw)) {
$rhevmadminpw = $pw->{password};
}
# The $rhevmadminpw must be unencrypted, since http need this to generate the authorized key
if (!$rhevmadminpw) {
my $rsp;
push @{ $rsp->{data} }, "The unencrypted password of \'admin\' for the rhevm much be set in the passwd table.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return ();
}
$pw = $passtab->getAttribs({ 'key' => 'rhevh', 'username' => 'admin' }, 'password');
if (defined($pw)) {
$rhevhadminpw = $pw->{password};
}
if (!$rhevhadminpw) {
my $rsp;
push @{ $rsp->{data} }, "The password of \'admin\' for the rhevh much be set in the passwd table.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return ();
} else {
$rhevhadminpw = authpw($rhevhadminpw);
}
$pw = $passtab->getAttribs({ 'key' => 'rhevh', 'username' => 'root' }, 'password');
if (defined($pw)) {
$rhevhrootpw = $pw->{password};
if ($rhevhrootpw) {
$rhevhrootpw = authpw($rhevhrootpw);
}
}
}
# Get the host for the nodes
my $vmtab = xCAT::Table->new("vm");
my $vmtabhash = $vmtab->getNodesAttribs($nodes, [ 'host', 'mgr' ]);
my $hyptab = xCAT::Table->new("hypervisor");
my $hyptabhash = $hyptab->getNodesAttribs($nodes, [ 'type', 'mgr' ]);
# The hash that use hyp as key
# In following example: rhevm1 is rhev-m; rhevh1 and rhevh2 are rhev-h
# vm11 is vm which located at rhevh1, vm12 is vm which located at rhevh2
# vm1 has not specific host to bind
# The user and password for rhev-m and rhev-h also specified
# |0 'rhevm1'
# |1 HASH(0x99066e0)
# | 'host' => HASH(0x9906230)
# | 'rhevh1' => HASH(0x99064a0)
# | 'adminpw' => '$1$k343MVXm$tZrjCk5GUJgRguNxdyIrT0'
# | 'node' => ARRAY(0x98f91a8)
# | 0 'vm11'
# | 'rootpw' => '$1$k3Yvim3b$9NLOSVlIiQY3ZYluT.CqP/'
# | 'rhevh2' => HASH(0x99064e8)
# | 'adminpw' => '$1$k343MVXm$tZrjCk5GUJgRguNxdyIrT0'
# | 'node' => ARRAY(0x9905c60)
# | 0 'vm12'
# | 'rootpw' => '$1$k3Yvim3b$9NLOSVlIiQY3ZYluT.CqP/'
# | 'node' => ARRAY(0x98ff378)
# | 0 'vm1'
# | 'pw' => '$1$9IXfmatc$Vcoy23AF5q0BcBE0cB3Uq/'
# | 'user' => 'admin'
my %rhevm_hash;
foreach my $node (@$nodes) {
my $vment = $vmtabhash->{$node}->[0];
my $hypent = $hyptabhash->{$node}->[0];
if (defined($vment->{'mgr'})) { # is a vm and has rhevm info
$rhevm_hash{ $vment->{'mgr'} }->{'user'} = "admin";
$rhevm_hash{ $vment->{'mgr'} }->{'pw'} = $rhevmadminpw;
if (defined($vment->{'host'})) { # is a vm and has rhevm, host info
push @{ $rhevm_hash{ $vment->{'mgr'} }->{'host'}->{ $vment->{'host'} }->{'node'} }, $node;
$rhevm_hash{ $vment->{'mgr'} }->{'host'}->{ $vment->{'host'} }->{'adminpw'} = $rhevhadminpw;
$rhevm_hash{ $vment->{'mgr'} }->{'host'}->{ $vment->{'host'} }->{'rootpw'} = $rhevhrootpw;
} else {
push @{ $rhevm_hash{ $vment->{'mgr'} }->{'node'} }, $node;
}
} elsif (defined($hypent->{'mgr'})) { # is a rhevh
$rhevm_hash{ $hypent->{'mgr'} }->{'user'} = "admin";
$rhevm_hash{ $hypent->{'mgr'} }->{'pw'} = $rhevmadminpw;
push @{ $rhevm_hash{ $hypent->{'mgr'} }->{'host'}->{$node}->{'node'} }, $node;
$rhevm_hash{ $hypent->{'mgr'} }->{'host'}->{$node}->{'adminpw'} = $rhevhadminpw;
$rhevm_hash{ $hypent->{'mgr'} }->{'host'}->{$node}->{'rootpw'} = $rhevhrootpw;
} elsif ($command eq 'lsvm') { # hope the node is a rhevm, that only the lsvm can be run to display
} else {
my $rsp;
push @{ $rsp->{data} }, "$node: Missing the management point in \'vm.mgr\' or \'hypervisor.mgr\'.";
xCAT::MsgUtils->message("E", $rsp, $callback);
}
}
# For the lsve or cfgve command, get the object from argument
if ($command =~ /^(cfgve|lsve)$/) {
# -t type -o obj -m mgr
my $mgr;
unless ($extraargs) {
return ();
}
@ARGV = @{$extraargs};
GetOptions("m=s" => \$mgr); #use mgr to know where to dispatch this request
if ($mgr) {
$rhevm_hash{$mgr}->{'user'} = "admin";
$rhevm_hash{$mgr}->{'pw'} = $rhevmadminpw;
} else {
my $rsp;
push @{ $rsp->{data} }, "The flag -m is necessary to perform the command.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return ();
}
}
# Prepare the request for each service node
# The dispatch depends on the rhevm. Since the operation is in serial, so no need to use the service node.
my @requests;
my @rhevms = keys(%rhevm_hash);
my $sn = xCAT::ServiceNodeUtils->get_ServiceNode(\@rhevms, 'xcat', "MN");
foreach my $snkey (keys %$sn) {
my $reqcopy = {%$request};
$reqcopy->{'_xcatdest'} = $snkey;
$reqcopy->{_xcatpreprocessed}->[0] = 1;
# Search the node base on the hypervisor
my $rhevms = $sn->{$snkey};
my @moreinfo = ();
my @nodes = ();
foreach my $rhevm (@$rhevms) {
#[rhevm][user_rhevm][pw_rhevm][rhevh][user_rhevh][pw_rhevh][nodes]
my $data_rhevm = "[$rhevm][$rhevm_hash{$rhevm}->{'user'}][$rhevm_hash{$rhevm}->{'pw'}]";
if (defined($rhevm_hash{$rhevm}->{'host'})) {
foreach my $host (keys %{ $rhevm_hash{$rhevm}->{'host'} }) {
my $data_rhevh = $data_rhevm . "[$host][$rhevm_hash{$rhevm}->{'host'}->{$host}->{'adminpw'}][$rhevm_hash{$rhevm}->{'host'}->{$host}->{'rootpw'}][" . join(',', @{ $rhevm_hash{$rhevm}->{'host'}->{$host}->{'node'} }) . "]";
push @moreinfo, $data_rhevh;
push @nodes, @{ $rhevm_hash{$rhevm}->{'host'}->{$host}->{'node'} };
}
}
if (defined($rhevm_hash{$rhevm}->{'node'})) {
my $data_node = $data_rhevm . "[][][][" . join(',', @{ $rhevm_hash{$rhevm}->{'node'} }) . "]";
push @moreinfo, $data_node;
push @nodes, @{ $rhevm_hash{$rhevm}->{'node'} };
}
unless (defined($rhevm_hash{$rhevm}->{'host'}) || defined($rhevm_hash{$rhevm}->{'node'})) {
push @moreinfo, $data_rhevm . "[][][][]";
}
}
if (scalar @nodes) {
$reqcopy->{node} = \@nodes;
}
$reqcopy->{moreinfo} = \@moreinfo;
if ($verbose) {
$reqcopy->{verbose} = 1;
}
push @requests, $reqcopy;
}
return \@requests;
}
sub process_request {
my $request = shift;
my $callback = shift;
my $subreq = shift;
my $command = $request->{command}->[0];
my $args = $request->{arg};
my $nodes = $request->{node};
$verbose = $request->{verbose};
if ($command eq 'copycd') {
return copycd($request, $callback);
} elsif ($command eq 'rhevhupdateflag') {
# handle the command to update bootloader configuration file for rhevh installation
return rhevhupdateflag($request, $callback, $subreq);
}
my $moreinfo;
if ($request->{moreinfo}) {
$moreinfo = $request->{moreinfo};
} else {
my $rsp;
push @{ $rsp->{data} }, "";
xCAT::MsgUtils->message("I", $rsp, $callback);
}
my %rhevm_hash;
foreach my $info (@$moreinfo) {
$info =~ /^\[(.*?)\]\[(.*?)\]\[(.*?)\]\[(.*?)\]\[(.*?)\]\[(.*?)\]\[(.*?)\]/;
my $rhevm = $1;
my $rhevmuser = $2;
my $rhevmpw = $3;
my $rhevh = $4;
my $rhevhadminpw = $5;
my $rhevhrootpw = $6;
my @nodes;
if ($7) { @nodes = split(',', $7) }
$rhevm_hash{$rhevm}->{name} = $rhevm;
$rhevm_hash{$rhevm}->{user} = $rhevmuser;
$rhevm_hash{$rhevm}->{pw} = $rhevmpw;
if ($rhevh) {
$rhevm_hash{$rhevm}->{host}->{$rhevh}->{name} = $rhevh;
$rhevm_hash{$rhevm}->{host}->{$rhevh}->{adminpw} = $rhevhadminpw;
$rhevm_hash{$rhevm}->{host}->{$rhevh}->{rootpw} = $rhevhrootpw;
push @{ $rhevm_hash{$rhevm}->{host}->{$rhevh}->{node} }, @nodes;
} elsif (@nodes) {
push @{ $rhevm_hash{$rhevm}->{node} }, @nodes;
}
}
# TODO: Plan to make long http connection to the every rhevm, but it does not work well
foreach my $rhevm (keys %rhevm_hash) {
# TODO fork process for each rhev-m
# get the ca.crt for the specific rhevm
if (!-r "/etc/xcat/rhevm/$rhevm/ca.crt") {
if (!-d "/etc/xcat/rhevm/$rhevm") {
mkpath("/etc/xcat/rhevm/$rhevm");
}
my $cmd = "cd /etc/xcat/rhevm/$rhevm/; wget -q http://$rhevm:8080/ca.crt";
xCAT::Utils->runcmd($cmd, -1);
if ($::RUNCMD_RC != 0) {
my $rsp;
push @{ $rsp->{data} }, "Could not get the CA certificate from http://$rhevm:8080/ca.crt.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return undef;
}
}
}
if ($command eq 'mkinstall') {
mkinstall($request, $callback, \%rhevm_hash);
} elsif ($command eq "rsetboot") {
rsetboot($callback, \%rhevm_hash, $args);
} elsif ($command eq "addhost") {
addhost($callback, \%rhevm_hash);
} elsif ($command eq "chhypervisor") {
cfghost($callback, \%rhevm_hash, $nodes, $args);
} elsif ($command eq "rmhypervisor") {
push @$args, "-r";
cfghost($callback, \%rhevm_hash, $nodes, $args);
} elsif ($command eq "cfgve") {
cfgve($callback, \%rhevm_hash, $args);
} elsif ($command eq "lsve") {
lsve($callback, \%rhevm_hash, $args);
} elsif ($command eq "rmhost") {
rmhost();
} elsif ($command eq "lsvm") {
lsvm($callback, \%rhevm_hash, $args);
} elsif ($command eq "chvm") {
chvm($callback, \%rhevm_hash, $nodes, $args);
} elsif ($command eq "mkvm") {
mkvm($callback, \%rhevm_hash, $nodes, $args);
} elsif ($command eq "rmvm") {
rmvm($callback, \%rhevm_hash, $args);
} elsif ($command eq "clonevm") {
clonevm($callback, \%rhevm_hash, $args);
} elsif ($command eq "rmigrate") {
rmigrate($callback, \%rhevm_hash, $args);
} elsif ($command eq "rpower") {
power($callback, \%rhevm_hash, $args);
} elsif ($command eq "getrvidparms") {
getrvidparms($callback, \%rhevm_hash, $nodes);
}
}
my @cpiopid;
# Perform the copycds for rhev-h to a specific dirtory
sub copycd {
my $request = shift;
my $callback = shift;
my $distname;
my $arch;
my $path;
my $mntpath;
my $file; # the iso source file must be passed to generate the initrd
@ARGV = @{ $request->{arg} };
GetOptions('n=s' => \$distname,
'a=s' => \$arch,
'p=s' => \$path,
'm=s' => \$mntpath,
'f=s' => \$file,
);
unless ($distname && $arch && $mntpath) {
return;
}
if ($distname && $distname !~ /^rhev/) {
return;
} elsif (!$file) {
$callback->({ error => "Only support to use the iso file for rhev" });
return;
}
my $installroot = "/install";
my @entries = xCAT::TableUtils->get_site_attribute("installdir");
my $t_entry = $entries[0];
if (defined($t_entry)) {
$installroot = $t_entry;
}
my $rsp;
push @{ $rsp->{data} }, "Copying media to $installroot/$distname/$arch/";
xCAT::MsgUtils->message("I", $rsp, $callback);
unless ($path) {
$path = "$installroot/$distname/$arch";
}
my $omask = umask 0022;
if (-l $path)
{
unlink($path);
}
mkpath("$path");
umask $omask;
my $rc;
my $reaped = 0;
$SIG{INT} = $SIG{TERM} = sub {
foreach (@cpiopid) {
kill 2, $_;
}
if ($mntpath) {
chdir("/");
system("umount $mntpath");
}
};
my $KID;
chdir $mntpath;
my $numFiles = `find . -print | wc -l`;
my $child = open($KID, "|-");
unless (defined $child)
{
$callback->({ error => "Media copy operation fork failure" });
return;
}
if ($child)
{
push @cpiopid, $child;
my @finddata = `find .`;
for (@finddata)
{
print $KID $_;
}
close($KID);
$rc = $?;
}
else
{
nice 10;
my $c = "nice -n 20 cpio -vdump $path";
my $k2 = open(PIPE, "$c 2>&1 |") ||
$callback->({ error => "Media copy operation fork failure" });
push @cpiopid, $k2;
my $copied = 0;
my ($percent, $fout);
while (<PIPE>) {
next if /^cpio:/;
$percent = $copied / $numFiles;
$fout = sprintf "%0.2f%%", $percent * 100;
$callback->({ sinfo => "$fout" });
++$copied;
}
exit;
}
# copy the iso to the source dir to generate the initrd by nodeset
copy($file, "$installroot/$distname/$arch/rhevh.iso");
}
# Perform the install preparation for the installation of rhev-h
sub mkinstall {
my $request = shift;
my $callback = shift;
my $rhevm_hash = shift;
my @nodes = @{ $request->{node} };
my %doneimgs;
my $installdir = "/install";
my @ents = xCAT::TableUtils->get_site_attribute("installdir");
my $site_ent = $ents[0];
if (defined($site_ent))
{
$installdir = $site_ent;
}
my $tftpdir = "/tftpboot";
@ents = xCAT::TableUtils->get_site_attribute("tftpdir");
$site_ent = $ents[0];
if (defined($site_ent))
{
$tftpdir = $site_ent;
}
my $nttab = xCAT::Table->new('nodetype');
my %ntents = %{ $nttab->getNodesAttribs(\@nodes, [ 'profile', 'os', 'arch', 'provmethod' ]) };
my $bptab = xCAT::Table->new('bootparams', -create => 1);
my $restab = xCAT::Table->new('noderes', -create => 1);
my %resents = %{ $restab->getNodesAttribs(\@nodes, [ 'xcatmaster', 'tftpdir', 'primarynic', 'installnic' ]) };
foreach my $rhevm (keys %{$rhevm_hash}) {
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $node (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if ($node eq $rhevm_hash->{$rhevm}->{host}->{$node}->{node}->[0]) { #this is a rhev-h
my $ent = $ntents{$node}->[0];
my $os = $ent->{os};
my $arch = $ent->{arch};
my $profile = $ent->{profile};
my ($kcmdline, $k, $i);
if ($arch =~ /x86/
&& -r "$installdir/$os/$arch/isolinux/vmlinuz0"
&& -r "$installdir/$os/$arch/isolinux/initrd0.img"
&& -r "$installdir/$os/$arch/rhevh.iso") {
my $tftppath = "$tftpdir/xcat/$os/$arch";
mkpath($tftppath);
copy("$installdir/$os/$arch/isolinux/vmlinuz0", $tftppath);
# append the full iso to the initrd. It will be downloaded to the node and as the installation source to install rhev-h
unless ($doneimgs{"$os|$arch"}) {
my $cmd = "cd $installdir/$os/$arch/; echo rhevh.iso | cpio -H newc --quiet -L -o | gzip -9 | cat $installdir/$os/$arch/isolinux/initrd0.img - > $tftppath/initrd0.img";
`$cmd`;
$doneimgs{"$os|$arch"} = 1;
}
$k = "xcat/$os/$arch/vmlinuz0";
$i = "xcat/$os/$arch/initrd0.img";
} else {
my $rsp;
push @{ $rsp->{data} }, "Cannot find vmlinux";
xCAT::MsgUtils->message("E", $rsp, $callback);
}
$kcmdline = " rootflags=loop";
$kcmdline .= " root=live:/rhevh.iso";
$kcmdline .= " rootfstype=auto ro liveimg nomodeset check rootflags=ro crashkernel=512M-2G:64M,2G-:128M elevator=deadline processor.max_cstate=1 install reinstall quiet rd_NO_LVM rhgb rd_NO_LUKS rd_NO_MD rd_NO_DM";
# set the boot device
my $ksdev = "";
$ent = $resents{$node}->[0];
if ($ent->{installnic}) {
if ($ent->{installnic} eq "mac") {
my $mactab = xCAT::Table->new("mac");
my $macref = $mactab->getNodeAttribs($node, ['mac']);
$ksdev = xCAT::Utils->parseMacTabEntry($macref->{mac}, $node);
} else {
$ksdev = $ent->{installnic};
}
} elsif ($ent->{primarynic}) {
if ($ent->{primarynic} eq "mac") {
my $mactab = xCAT::Table->new("mac");
my $macref = $mactab->getNodeAttribs($node, ['mac']);
$ksdev = xCAT::Utils->parseMacTabEntry($macref->{mac}, $node);
} else {
$ksdev = $ent->{primarynic};
}
}
# set the storage parameters
$kcmdline .= " storage_init";
# set the bootif
if ($ksdev) {
$kcmdline .= " BOOTIF=$ksdev ip=dhcp";
} else { #let boot firmware fill it in
$kcmdline .= " ip=dhcp";
}
# set the passwd for admin and root
$kcmdline .= " adminpw=$rhevm_hash->{$rhevm}->{host}->{$node}->{adminpw} rootpw=$rhevm_hash->{$rhevm}->{host}->{$node}->{rootpw} ssh_pwauth=1";
# set the hostname and password of the management server for the node so that node could register to the rhevm automatically.
$kcmdline .= " management_server=$rhevm_hash->{$rhevm}->{name} rhevm_admin_password=$rhevm_hash->{$rhevm}->{host}->{$node}->{rootpw}";
# set the flag update trigger, after installing of rhev-h, this url will be 'wget', xCAT MN will handle this event to run the upfateflag for this rhev-h
my $xcatmaster;
if ($ent and $ent->{xcatmaster}) {
$xcatmaster = $ent->{xcatmaster};
} else {
$xcatmaster = '!myipfn!';
}
$kcmdline .= " local_boot_trigger=http://$xcatmaster/xcatrhevh/rhevh_finish_install/\@HOSTNAME\@";
$bptab->setNodeAttribs($node,
{ kernel => $k,
initrd => $i,
kcmdline => $kcmdline });
}
}
}
}
}
# Generate the REST API http request
# $method: GET, PUT, POST, DELETE
# $api: the url of rest api
# $content: an xml section which including the data to perform the rest api
sub genreq {
my $rhevm = shift;
my $method = shift;
my $api = shift;
my $content = shift;
if (!defined($content)) { $content = ""; }
my $header = HTTP::Headers->new('content-type' => 'application/xml',
'Accept' => 'application/xml',
#'Connection' => 'keep-alive',
'Host' => $rhevm->{name} . ':8443');
$header->authorization_basic($rhevm->{user} . '@internal', $rhevm->{pw});
my $ctlen = length($content);
$header->push_header('Content-Length' => $ctlen);
my $url = "https://" . $rhevm->{name} . ":8443" . $api;
my $request = HTTP::Request->new($method, $url, $header, $content);
$request->protocol('HTTP/1.1');
return $request;
}
# Make connection to rhev-m
# Send REST api request to rhev-m
# Receive the response from rhev-m
# Handle the error cases
#
# return 1-ssl connection error;
# 2-http response error;
# 3-return a http error message;
# 5-operation failed
sub send_req {
my $ref_rhevm = shift;
my $request = shift;
my $rhevm = $ref_rhevm->{name};
my $rc = 0;
my $response;
my $connect;
my $socket = IO::Socket::INET->new(PeerAddr => $rhevm,
PeerPort => '8443',
Timeout => 15);
if ($socket) {
$connect = IO::Socket::SSL->start_SSL($socket, SSL_ca_file => "/etc/xcat/rhevm/$rhevm/ca.crt", Timeout => 0);
if ($connect) {
my $flags = fcntl($connect, F_GETFL, 0);
$flags |= O_NONBLOCK;
fcntl($connect, F_SETFL, $flags);
} else {
$rc = 1;
$response = "Could not make ssl connection to $rhevm:8443.";
}
} else {
$rc = 1;
$response = "Could not create socket to $rhevm:8443.";
}
if ($rc) {
return ($rc, $response);
}
my $IOsel = new IO::Select;
$IOsel->add($connect);
if ($verbose) {
my $rsp;
push @{ $rsp->{data} }, "\n===================================================\n$request----------------";
xCAT::MsgUtils->message("I", $rsp, $global_callback);
}
print $connect $request;
$response = "";
my $retry;
my $ischunked;
my $firstnum;
while ($retry++ < 10) {
unless ($IOsel->can_read(2)) {
next;
}
my $readbytes;
my $res = "";
do { $readbytes = sysread($connect, $res, 65535, length($res)); } while ($readbytes);
if ($res) {
my @part = split(/\r\n/, $res);
for my $data (@part) {
# for chunk formated data, check the last chunk to finish
if ($data =~ /Transfer-Encoding: (\S+)/) {
if ($1 eq "chunked") {
$ischunked = 1;
}
}
if ($ischunked && $data =~ /^([\dabcdefABCDEF]+)$/) {
if ($1 eq 0) {
# last chunk
goto FINISH;
} else {
# continue to get the rest chunks
$retry = 0;
next;
}
} else {
# put all data together
$response .= $data;
}
}
}
unless ($ischunked) {
# for non chunk data, just read once
if ($response) {
last;
} else {
if (not defined $readbytes and $! == EAGAIN) { next; }
$rc = 2;
last;
}
}
}
FINISH:
if ($retry >= 10) { $rc = 3; }
if ($verbose) {
my $rsp;
push @{ $rsp->{data} }, "$response===================================================\n";
xCAT::MsgUtils->message("I", $rsp, $global_callback);
}
$IOsel->remove($connect);
close($connect);
if ($response) {
if (grep (/<html>/, $response)) { # get a error message in the html
$rc = 3;
} elsif (grep (/<\?xml/, $response)) {
$response =~ s/.*?</</ms;
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my $attr;
if ($attr = getAttr($doc, "/fault/detail")) {
$response = $attr;
$rc = 5;
} elsif ($attr = getAttr($doc, "/action/fault/detail")) {
if ($attr eq "[]") {
if ($attr = getAttr($doc, "/action/fault/reason")) {
$response = $attr;
} else {
$response = "failed";
}
} else {
$response = $attr;
}
$rc = 5;
}
}
}
}
return ($rc, $response);
}
# Add the rhels host since it cannot register automatically.
sub addhost {
my $callback = shift;
my $rhevm_hash = shift;
my @domain = xCAT::TableUtils->get_site_attribute("domain");
if (!$domain[0]) {
my $rsp;
push @{ $rsp->{data} }, "The site.domain must be set to enable the rhev support.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return;
}
# Create the xml data
my $doc = XML::LibXML->createDocument();
my $root = $doc->createElement("host");
$doc->setDocumentElement($root);
my $name_ele = $doc->createElement("name");
$root->appendChild($name_ele);
my $name_t = XML::LibXML::Text->new("");
$name_ele->appendChild($name_t);
my $add_ele = $doc->createElement("address");
$root->appendChild($add_ele);
my $add_t = XML::LibXML::Text->new("");
$add_ele->appendChild($add_t);
my $rootpw_ele = $doc->createElement("root_password");
$root->appendChild($rootpw_ele);
my $rootpw_t = XML::LibXML::Text->new("");
$rootpw_ele->appendChild($rootpw_t);
foreach my $rhevm (keys %{$rhevm_hash}) {
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if ($rhevh eq $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}->[0]) {
# Create the host first
my $api = "/api/hosts";
my $method = "POST";
# Generate the content
$name_t->setData($rhevh);
my $addofrhevh = $rhevh . "." . $domain[0];
$add_t->setData($addofrhevh);
$rootpw_t->setData($rhevm_hash->{$rhevm}->{host}->{$rhevh}->{pw});
$rootpw_t->setData('$1$c5TJgKlJ$CuO6rR5B3d5mZc3Etu9HZ1');
my $content = $doc->toString();
my $request = genreq($ref_rhevm, $method, $api, $content);
my ($rc, $response) = send_req($ref_rhevm, $request->as_string());
my $rsp;
if ($rc) {
push @{ $rsp->{data} }, "$rhevh: $response";
next;
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my $attr;
if ($attr = getAttr($doc, "/vms/hosts/status/state")) {
push @{ $rsp->{data} }, "$rhevh: state: $attr";
}
}
}
}
}
}
}
# Adding the host to rhevm
}
# name -> path mapping for the resource display
my $display = {
'datacenters' => {
'description' => ["description"],
'storagetype' => ['storage_type'],
'storageformat' => ['storage_format'],
'state' => ['status/state'],
},
'clusters' => {
'description' => ["description"],
'cpu' => [ "cpu", "id" ],
'memory_overcommit' => [ "memory_policy/overcommit", "percent" ],
'memory_hugepage' => ["memory_policy/transparent_hugepages/enabled"],
},
'storagedomains' => {
'type' => ["type"],
'ismaster' => ["master"],
'storage_type' => ["storage/type"],
'storage_add' => ["storage/address"],
'storage_path' => ["storage/path"],
'available' => ["available"],
'used' => ["used"],
'committed' => ["committed"],
'storage_format' => ["storage_format"],
'status' => ["status/state"],
},
'networks' => {
'description' => ["description"],
'vlan' => [ "vlan", "id" ],
'stp' => ["stp"],
'state' => ["status/state"],
},
'hosts' => {
'address' => ["address"],
'state' => ["status/state"],
'type' => ["type"],
'storage_manager' => ["storage_manager"],
'powermgt' => ["power_management/enabled"],
'powermgt_type' => [ "power_management", "type" ],
'powermgt_addr' => ["power_management/address"],
'powermgt_user' => ["power_management/username"],
'ksm' => ["ksm/enabled"],
'hugepages' => ["transparent_hugepages/enabled"],
'iscsi' => ["iscsi/initiator"],
'cpu' => ["cpu/name"],
'cpuspeed' => ["cpu/speed"],
'summary_active' => ["summary/active"],
'summary_migrating' => ["summary/migrating"],
'summary_total' => ["summary/total"],
},
'host_nics' => {
'network' => [ "network", "id", "networks", "/network/name" ],
'mac' => [ "mac", "address" ],
'ip' => [ "ip", "address" ],
'netmask' => [ "ip", "netmask" ],
'gateway' => [ "ip", "gateway" ],
'speed' => ["speed"],
'boot_protocol' => ["boot_protocol"],
'state' => ["status/state"],
},
'vms' => {
'memory' => ["memory"],
'state' => ["status/state"],
'type' => ["type"],
'cpusocket' => [ "cpu/topology", "sockets" ],
'cpucore' => [ "cpu/topology", "cores" ],
'bootorder' => [ "os/boot", "dev" ],
'display' => ["display/type"],
'start_time' => ["start_time"],
'creation_time' => ["creation_time"],
'stateless' => ["stateless"],
'placement_policy' => ["placement_policy/affinity"],
'memory_guaranteed' => ["memory_policy/guaranteed"],
'host' => [ "host", "id", "hosts", "/host/name" ],
},
'templates' => {
'memory' => ["memory"],
'state' => ["status/state"],
'type' => ["type"],
'cpusocket' => [ "cpu/topology", "sockets" ],
'cpucore' => [ "cpu/topology", "cores" ],
'bootorder' => [ "os/boot", "dev" ],
'display' => ["display/type"],
'creation_time' => ["creation_time"],
'stateless' => ["stateless"],
},
'disks' => {
'size' => ["size"],
'type' => ["type"],
'state' => ["status/state"],
'iftype' => ["interface"],
'format' => ["format"],
'bootable' => ["bootable"],
'storage_domains' => [ "storage_domains/storage_domain", "id", "storagedomains", "/storage_domain/name" ],
},
'nics' => {
'iftype' => ["interface"],
'mac' => [ "mac", "address" ],
'network' => [ "network", "id", "networks", "/network/name" ],
},
};
# Display the resource, it's called by lsvm and lsve
# $reponse - xml response return from send_req
# $type -datacenters, clusters, storagedomains ...
# $prelead - space that will be displayed pre the real message
# $criteria - 'dc=<name>' or 'name=<name>', only display when matchs
sub displaysrc {
my $callback = shift;
my $ref_rhevm = shift;
my $response = shift;
my $type = shift;
my $prelead = shift;
my $criteria = shift;
my $individual = shift;
my @output;
my @displayed;
my $prefix;
if ($type eq "datacenters") {
$prefix = "/data_centers/data_center";
} elsif ($type eq "clusters") {
$prefix = "/clusters/cluster";
} elsif ($type eq "storagedomains") {
if ($individual) {
$prefix = "/storage_domain";
} else {
$prefix = "/storage_domains/storage_domain";
}
} elsif ($type eq "networks") {
$prefix = "/networks/network";
} elsif ($type eq "hosts") {
$prefix = "/hosts/host";
} elsif ($type eq "vms") {
$prefix = "/vms/vm";
} elsif ($type eq "templates") {
$prefix = "/templates/template";
} elsif ($type eq "disks") {
$prefix = "/disks/disk";
} elsif ($type eq "nics") {
$prefix = "/nics/nic";
} elsif ($type eq "host_nics") {
$prefix = "/host_nics/host_nic";
} else {
return ();
}
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my @nodes = $doc->findnodes($prefix);
foreach my $node (@nodes) {
# when crteria specified, the dc or name need to be checked
if ($criteria) {
my ($name, $value) = split('=', $criteria);
my $curval;
if ($name eq "dc") {
my $cnode = $node->findnodes("data_center");
if (defined($cnode->[0])) {
$curval = $cnode->[0]->getAttribute("id");
} else {
next;
}
} elsif ($name eq "name") {
my $cnode = $node->findnodes("name");
if (defined($cnode->[0])) {
$curval = $cnode->[0]->textContent();
} else {
next;
}
}
unless ($curval eq $value) {
next;
}
}
# Get the resource name first and display
my $objname = getAttr($node, "name");
push @displayed, getAttr($node, "", "id");
push @output, $prelead . $type . ": [" . $objname . "]";
# Display each item for the specific type
foreach my $name (sort (keys %{ $display->{$type} })) {
if (defined($display->{$type}->{$name}->[2])) { # search the resource from the id
# If the [3] and [4] params are specified, use the [0] and [1] to get the target resouce id,
# Then search the resource of this id, [3] is type and [4] is the path to get the end message
my $id = getAttr($node, $display->{$type}->{$name}->[0], $display->{$type}->{$name}->[1]);
my $srctype = $display->{$type}->{$name}->[2];
my $srcpath = $display->{$type}->{$name}->[3];
my ($rc, $newid, $stat, $response) = search_src($ref_rhevm, $srctype, "/" . $id, 1);
unless ($rc) {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my $attr;
if ($attr = getAttr($doc, $srcpath) || defined($attr)) {
push @output, $prelead . " " . $name . ": " . $attr;
}
}
}
} else {
my $value = getAttr($node, $display->{$type}->{$name}->[0], $display->{$type}->{$name}->[1]);
if ($value) {
push @output, $prelead . " " . $name . ": " . $value;
}
}
}
}
}
my $rsp;
if (@output) {
push @{ $rsp->{data} }, @output;
}
xCAT::MsgUtils->message("I", $rsp, $callback);
return @displayed;
}
# Display the virtual environment
# -t - type of resouce: dc - datacenter; cl - cluster; sd - storage domain; nw - network; tpl - template
# -o - the object that needs to be displayed. It could be multiple objs separated with ','
# -m - the rhevm that manage the resources
sub lsve {
my $callback = shift;
my $rhevm_hash = shift;
my $args = shift;
my $nodes = shift;
my @output;
my ($type, $objs, $mgr, $approve, $create, $update, $active, $network, $power, $remove);
if ($args) {
@ARGV = @{$args};
GetOptions('t=s' => \$type,
'o=s' => \$objs,
'm=s' => \$mgr);
}
my $rhevm = (keys %{$rhevm_hash})[0];
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
my @objs;
if ($objs) {
@objs = split(',', $objs);
} else {
push @objs, 'xxxxxx_all_objs';
}
foreach my $obj (@objs) {
if ($type eq "dc") {
my ($rc, $id, $stat, $response) = search_src($ref_rhevm, "datacenters", $obj);
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "datacenters", "");
my $dcid = $id;
if ($obj ne 'xxxxxx_all_objs') {
# Display the cluster, storagedomain, network if requiring to display datacenter
($rc, $id, $stat, $response) = search_src($ref_rhevm, "clusters", "datacenter%3D$obj");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "clusters", " ");
}
#($rc, $id, $stat, $response) = search_src($ref_rhevm, "storagedomains", "datacenter%3D$obj");
($rc, $id, $stat, $response) = search_src($ref_rhevm, "datacenters/$dcid/storagedomains:storagedomains");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "storagedomains", " ");
}
($rc, $id, $stat, $response) = search_src($ref_rhevm, "networks");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "networks", " ", "dc=$dcid");
}
($rc, $id, $stat, $response) = search_src($ref_rhevm, "templates", "datacenter%3D$obj");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "templates", " ");
}
}
}
} elsif ($type eq "cl") {
my ($rc, $id, $stat, $response) = search_src($ref_rhevm, "clusters", "$obj");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "clusters", "");
}
} elsif ($type eq "sd") {
my ($rc, $id, $stat, $response) = search_src($ref_rhevm, "storagedomains", "$obj");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "storagedomains", "");
}
} elsif ($type eq "nw") {
my ($rc, $id, $stat, $response) = search_src($ref_rhevm, "networks");
unless ($rc) {
if ($obj eq 'xxxxxx_all_objs') {
displaysrc($callback, $ref_rhevm, $response, "networks", " ");
} else {
displaysrc($callback, $ref_rhevm, $response, "networks", " ", "name=$obj");
}
}
} elsif ($type eq "tpl") {
my ($rc, $id, $stat, $response) = search_src($ref_rhevm, "templates", "$obj");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "templates", "");
}
} else {
my $rsp;
push @{ $rsp->{data} }, "The type: $type is not supported.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return;
}
}
return;
}
# Configure the rhev virtual environment
# -t - type of resouce: dc - datacenter; cl - cluster; sd - storage domain; nw - network
# -o - the object that needs to be configured. It could be multiple objs separated with ','
# -m - the rhevm that manage the resources
# -d - datacenter name that needs by creating
# -c - creating a resource
# -u - updating a resource
# -g - activate a resource
# -s - deactivate a resource
# -a - attach a resource
# -b - detach a resource
# -r - delete a resource
# Working format:
# cfgve -t sd -m <mgr> -o <name> -c
# cfgve -t sd -m <mgr> -o <name> -a/-g/-s
# cfgve -t nw -m <mgr> -o < name> -c
# cfgve -t tpl -m <mgr> -o <name> -r
#
sub cfgve {
my $callback = shift;
my $rhevm_hash = shift;
my $args = shift;
my $nodes = shift;
my ($type, $objlist, $mgr, $datacenter, $cluster, $create, $update, $remove, $activate, $deactivate, $attach, $detach, $force, $stype, $cputype, $vlan);
if ($args) {
@ARGV = @{$args};
GetOptions('t=s' => \$type,
'o=s' => \$objlist,
'm=s' => \$mgr,
'd=s' => \$datacenter,
'l=s' => \$cluster,
'c' => \$create,
'u' => \$update,
'g' => \$activate,
's' => \$deactivate,
'a' => \$attach,
'b' => \$detach,
'r' => \$remove,
'f' => \$force,
'k=s' => \$stype,
'p=s' => \$cputype,
'n=s' => \$vlan);
}
my $rhevm = (keys %{$rhevm_hash})[0];
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
my @objs;
if ($objlist) {
@objs = split(',', $objlist);
}
foreach my $obj (@objs) {
if ($type eq "sd") {
if ($create) {
if (mkSD($callback, $ref_rhevm, $obj)) {
my $rsp;
push @{ $rsp->{data} }, "$obj: create storage domain succeeded.";
xCAT::MsgUtils->message("I", $rsp, $callback);
return;
}
} elsif ($activate || $deactivate || $attach || $detach || $remove) {
# get the name of datacenter
my $vsdtab = xCAT::Table->new('virtsd', -create => 0);
my $vsdent = $vsdtab->getAttribs({ 'node' => $obj }, ['datacenter']);
my $datacenter = $vsdent->{datacenter};
unless ($datacenter) {
$datacenter = "Default";
}
my ($rc, $dcid, $sdid, $stat);
($rc, $dcid, $stat) = search_src($ref_rhevm, "datacenters", $datacenter);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: failed to get datacenter: $datacenter.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return;
}
($rc, $sdid, $stat) = search_src($ref_rhevm, "storagedomains", $obj);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: failed to get storagedomains: $obj.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return;
}
if ($activate || $deactivate) {
my $rsp;
if (activate($callback, $ref_rhevm, "/api/datacenters/$dcid/storagedomains/$sdid", $obj, $deactivate)) {
push @{ $rsp->{data} }, "$obj: failed.";
xCAT::MsgUtils->message("E", $rsp, $callback);
} else {
push @{ $rsp->{data} }, "$obj: succeeded.";
xCAT::MsgUtils->message("I", $rsp, $callback);
}
} elsif ($attach || $detach) {
my $rsp;
if (attach($callback, $ref_rhevm, "/api/datacenters/$dcid/storagedomains", "storage_domain", $sdid, $detach)) {
push @{ $rsp->{data} }, "$obj: failed.";
xCAT::MsgUtils->message("E", $rsp, $callback);
} else {
push @{ $rsp->{data} }, "$obj: succeeded.";
xCAT::MsgUtils->message("I", $rsp, $callback);
}
} elsif ($remove) {
if ($force) {
# deactivate the storage domain
activate($callback, $ref_rhevm, "/api/datacenters/$dcid/storagedomains/$sdid", $obj, 1);
# detach the storage domain to the datacenter
attach($callback, $ref_rhevm, "/api/datacenters/$dcid/storagedomains", "storage_domain", $sdid, 1);
}
if (!deleteSD($callback, $ref_rhevm, "/api/storagedomains/$sdid", $obj)) {
my $rsp;
push @{ $rsp->{data} }, "$obj: delete storage domain succeeded.";
xCAT::MsgUtils->message("I", $rsp, $callback);
return;
}
}
}
} elsif ($type eq "tpl") {
if ($remove) {
my ($rc, $tplid, $stat, $response) = search_src($ref_rhevm, "templates", "$obj");
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: cannot find the template: $obj.";
xCAT::MsgUtils->message("E", $rsp, $callback);
}
generalaction($callback, $ref_rhevm, "/api/templates/$tplid", "DELETE", 1);
}
} elsif ($type eq "nw") {
if ($create) {
# serach datacenter
unless ($datacenter) {
$datacenter = "Default";
}
my ($rc, $dcid, $stat) = search_src($ref_rhevm, "datacenters", $datacenter);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: failed to get datacenter: $datacenter.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
# create the network
my $api = "/api/networks";
my $method = "POST";
my $content = "<network><name>$obj</name><data_center id=\"$dcid\"/></network>";
if ($vlan) {
$content = "<network><name>$obj</name><data_center id=\"$dcid\"/><vlan id=\"$vlan\"/></network>";
}
my $request = genreq($ref_rhevm, $method, $api, $content);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
} else {
my $rsp;
push @{ $rsp->{data} }, "$obj: succeeded";
xCAT::MsgUtils->message("I", $rsp, $callback);
next;
}
} elsif ($remove) {
my ($rc, $nwid, $stat) = search_src($ref_rhevm, "networks", $obj);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: failed to get networks: $obj.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
generalaction($callback, $ref_rhevm, "/api/networks/$nwid", "DELETE", 1);
} elsif ($attach || $detach) {
unless ($cluster) {
$cluster = "Default";
}
my ($rc, $clid, $stat, $response) = search_src($ref_rhevm, "clusters", "$cluster");
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: cannot find the cluster:$cluster.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
my $nwid;
($rc, $nwid, $stat) = search_src($ref_rhevm, "networks", "$obj");
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: cannot find the network.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
if ($attach) {
my $api = "/api/clusters/$clid/networks";
my $method = "POST";
my $content = "<network id=\"$nwid\"><name>$obj</name></network>";
my $request = genreq($ref_rhevm, $method, $api, $content);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
} else {
my $rsp;
push @{ $rsp->{data} }, "$obj: succeeded";
xCAT::MsgUtils->message("I", $rsp, $callback);
next;
}
} elsif ($detach) {
generalaction($callback, $ref_rhevm, "/api/clusters/$clid/networks/$nwid", "DELETE", 1);
}
}
} elsif ($type eq "dc") {
my ($rc, $dcid, $stat, $response) = search_src($ref_rhevm, "datacenters", "$obj");
if ($create) {
if (!$rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: data center has been created.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
unless ($stype && $stype =~ /^(nfs|localfs)$/) {
my $rsp;
push @{ $rsp->{data} }, "$obj: the storage type needs to be specified by -k.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
# create the datacenter
my $api = "/api/datacenters";
my $method = "POST";
my $content = "<data_center><name>$obj</name><storage_type>$stype</storage_type><version minor=\"0\" major=\"3\"/></data_center>";
my $request = genreq($ref_rhevm, $method, $api, $content);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
} else {
my $rsp;
push @{ $rsp->{data} }, "$obj: succeeded";
xCAT::MsgUtils->message("I", $rsp, $callback);
next;
}
} elsif ($remove) {
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: cannot find the data center.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
generalaction($callback, $ref_rhevm, "/api/datacenters/$dcid", "DELETE", 1);
}
} elsif ($type eq "cl") {
my ($rc, $clid) = search_src($ref_rhevm, "clusters", "$obj");
if ($create) {
if (!$rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: cluster has been created.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
unless ($datacenter) {
my $rsp;
push @{ $rsp->{data} }, "$obj: the datacenter for the cluster must be specified.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
my $dcid;
($rc, $dcid) = search_src($ref_rhevm, "datacenters", "$datacenter");
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: failed to get the datacenter: $datacenter.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
unless ($cputype) {
$cputype = "Intel Penryn Family";
}
# create the datacenter
my $api = "/api/clusters";
my $method = "POST";
my $content = "<cluster><name>$obj</name><data_center id=\"$dcid\"/><cpu id=\"$cputype\"/></cluster>";
my $request = genreq($ref_rhevm, $method, $api, $content);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
} else {
my $rsp;
push @{ $rsp->{data} }, "$obj: succeeded";
xCAT::MsgUtils->message("I", $rsp, $callback);
next;
}
} elsif ($remove) {
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$obj: cannot find the cluster.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
generalaction($callback, $ref_rhevm, "/api/clusters/$clid", "DELETE", 1, $force);
}
} else {
my $rsp;
push @{ $rsp->{data} }, "The type: $type is not supported.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return;
}
}
}
# configure host
# -a: approve the host that can be managed by rhev-m
# -n: configure the network for host
# -p: configure the power management for host.
# This will be used for rhev-m to check the power status of host, so that when SPM (Storage Pool Manager) host
# down, rhev-m could switch the SPM role to another host automatically.
# For rack mounted server, the ipmilan is used to do the power management. The IP of bmc and user:passwd are
# neccessary to be configured for power management.
# -e: activate the host
# -d: deactivate a host to maintanance mode
sub cfghost {
my $callback = shift;
my $rhevm_hash = shift;
my $nodes = shift;
my $args = shift;
my ($approve, $network, $power, $activate, $deactivate, $remove, $force);
if ($args) {
@ARGV = @{$args};
GetOptions('a' => \$approve,
'n' => \$network,
'p' => \$power,
'e' => \$activate,
'd' => \$deactivate,
'r' => \$remove,
'f' => \$force);
}
# Set the default user:pw for ipmi
my ($ipmiuser, $ipmipw) = ('USERID', 'PASSW0RD');
my ($hment, $ipmient);
my %hyper;
# get the IP, user, passwd for the bmc of the host if requiring to configure power management
if ($power) {
my $hmtab = xCAT::Table->new('nodehm', -create => 0);
$hment = $hmtab->getNodesAttribs($nodes, ['mgt']);
my $ipmitab = xCAT::Table->new('ipmi', -create => 0);
$ipmient = $ipmitab->getNodesAttribs($nodes, [ 'bmc', 'username', 'password' ]);
#get the default password for bmc
my $pwtab = xCAT::Table->new('passwd', -create => 0);
my $pwent = $pwtab->getAttribs({ 'key' => 'ipmi' }, [ 'username', 'password' ]);
if ($pwent) {
$ipmiuser = $pwent->{'username'};
$ipmipw = $pwent->{'password'};
}
}
# get the network parameters for the host if requiring to configure the network for host
if ($network || $approve) {
# get the network interface for host
my $hyptab = xCAT::Table->new('hypervisor', -create => 0);
my $hypent = $hyptab->getNodesAttribs($nodes, [ 'interface', 'datacenter', 'cluster' ]);
foreach my $node (@$nodes) {
if (defined($hypent->{$node}->[0])) {
$hyper{$node}{interface} = $hypent->{$node}->[0]->{interface};
$hyper{$node}{datacenter} = $hypent->{$node}->[0]->{datacenter};
$hyper{$node}{cluster} = $hypent->{$node}->[0]->{cluster};
}
if (!$hyper{$node}{datacenter}) {
$hyper{$node}{datacenter} = "Default";
}
if (!$hyper{$node}{cluster}) {
$hyper{$node}{cluster} = "Default";
}
}
}
foreach my $rhevm (keys %{$rhevm_hash}) {
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if ($rhevh eq $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}->[0]) {
# get the host
my ($rc, $hostid, $hoststat) = search_src($ref_rhevm, "hosts", $rhevh);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$rhevh: host was not created.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
if ($approve) {
if ($hoststat eq "pending_approval") {
# get the id of cluster
my ($rc, $clusterid, $clusterstat) = search_src($ref_rhevm, "clusters", $hyper{$rhevh}{cluster});
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$rhevh: failed to get cluster: $hyper{$rhevh}{cluster}.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
my $approved = 0;
# Create the host first
my $api = "/api/hosts/$hostid/approve";
my $method = "POST";
# Generate the content
my $content = "<action><cluster id=\"$clusterid\"/></action>";
my $request = genreq($ref_rhevm, $method, $api, $content);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$rhevh: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my $attr;
if ($attr = getAttr($doc, "/action/status/state")) {
if ($attr eq "complete") {
$approved = 1;
}
}
}
}
my $rsp;
if ($approved) {
push @{ $rsp->{data} }, "$rhevh: approved.";
xCAT::MsgUtils->message("I", $rsp, $callback);
} else {
push @{ $rsp->{data} }, "$rhevh: failed to approve.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
} else {
my $rsp;
push @{ $rsp->{data} }, "$rhevh: the state of node is not correct for approve. Current state: $hoststat";
xCAT::MsgUtils->message("E", $rsp, $callback);
}
}
if ($activate || $deactivate) {
my $rsp;
if (activate($callback, $ref_rhevm, "/api/hosts/$hostid", $rhevh, $deactivate)) {
push @{ $rsp->{data} }, "$rhevh: failed.";
xCAT::MsgUtils->message("E", $rsp, $callback);
} else {
push @{ $rsp->{data} }, "$rhevh: succeeded.";
xCAT::MsgUtils->message("I", $rsp, $callback);
}
}
# configure the network interface for a host
if ($network) {
unless ($hyper{$rhevh}{interface}) {
my $rsp;
push @{ $rsp->{data} }, "$rhevh: the hypervisor.interface needs to be configured to configure the network for a host.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
if ($hoststat eq "maintenance") {
cfghypnw($callback, $ref_rhevm, $rhevh, $hyper{$rhevh}{interface}, $hyper{$rhevh}{datacenter});
} else {
my $rsp;
push @{ $rsp->{data} }, "$rhevh: the hypervisor needs to be deactivated to maintenance state for the network configuring.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
}
if ($power) {
# Configure the power management for a host
# for rack mounted machine, use the 'ipmilan' type of power management
if (defined($hment->{$rhevh}->[0]) && $hment->{$rhevh}->[0]->{'mgt'}) {
if ($hment->{$rhevh}->[0]->{'mgt'} eq "ipmi") {
# get the bmc IP, user, password for the bmc
my ($user, $pw, $addr) = ($ipmiuser, $ipmipw);
if (defined($ipmient->{$rhevh}->[0]) && $ipmient->{$rhevh}->[0]->{bmc}) {
$addr = $ipmient->{$rhevh}->[0]->{bmc};
} else {
my $rsp;
push @{ $rsp->{data} }, "$rhevh: the ipmi.bmc was not set to know the hardware control point.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
if (defined($ipmient->{$rhevh}->[0]) && $ipmient->{$rhevh}->[0]->{username}) {
$user = $ipmient->{$rhevh}->[0]->{username};
}
if (defined($ipmient->{$rhevh}->[0]) && $ipmient->{$rhevh}->[0]->{password}) {
$pw = $ipmient->{$rhevh}->[0]->{password};
}
my $doc = XML::LibXML->createDocument();
my $root = $doc->createElement("host");
$doc->setDocumentElement($root);
my $pm_ele = $doc->createElement("power_management");
$root->appendChild($pm_ele);
$pm_ele->setAttribute("type", "ipmilan");
$pm_ele->appendTextChild("enabled", "true");
$pm_ele->appendTextChild("address", $addr);
$pm_ele->appendTextChild("username", $user);
$pm_ele->appendTextChild("password", $pw);
my $api = "/api/hosts/$hostid";
my $method = "PUT";
my $request = genreq($ref_rhevm, $method, $api, $doc->toString);
my ($rc, $response) = send_req($ref_rhevm, $request->as_string());
my $rsp;
if ($rc) {
push @{ $rsp->{data} }, "$rhevh: $response";
next;
} else {
push @{ $rsp->{data} }, "$rhevh: Setting power management: $addr";
}
xCAT::MsgUtils->message("I", $rsp, $callback);
} else {
my $rsp;
push @{ $rsp->{data} }, "$rhevh: the supported power management method: ipmi.";
xCAT::MsgUtils->message("E", $rsp, $callback);
}
} else {
my $rsp;
push @{ $rsp->{data} }, "$rhevh: the nodehm.mgt was not set to know the management method.";
xCAT::MsgUtils->message("E", $rsp, $callback);
}
} # end of power management configure
if ($remove) {
if ($force && ($hoststat ne "maintenance")) {
# deactivate the host anyway
activate($callback, $ref_rhevm, "/api/hosts/$hostid", $rhevh, 1);
if (waitforcomplete($ref_rhevm, "/api/hosts/$hostid", "/host/status/state=maintenance", 30)) {
my $rsp;
push @{ $rsp->{data} }, "$rhevh: failed to waiting the host gets to \"maintenance\" state.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
}
generalaction($callback, $ref_rhevm, "/api/hosts/$hostid", "DELETE", 1);
}
}
} # end of for each host
}
}
}
sub rmhost {
}
# List the host and virtual machine
# -s short
# -v display virtual machines which belongs to the host
sub lsvm {
my $callback = shift;
my $rhevm_hash = shift;
my $args = shift;
my $nodes = shift;
my ($short, $vm4host);
if ($args) {
@ARGV = @{$args};
GetOptions('s' => \$short,
'v' => \$vm4host);
}
foreach my $rhevm (keys %{$rhevm_hash}) {
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
# Get the node that will be handled
my @vms;
my @hyps;
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if (defined $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}) {
foreach my $node (@{ $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node} }) {
if ($rhevh eq $node) {
push @hyps, $rhevh;
} else {
push @vms, $node;
}
}
}
}
}
if (defined $rhevm_hash->{$rhevm}->{node}) {
push @vms, @{ $rhevm_hash->{$rhevm}->{node} };
}
foreach my $hyp (@hyps) {
# Get the host
my ($rc, $id, $stat, $response) = search_src($ref_rhevm, "hosts", "$hyp");
unless ($rc) {
my @hostids = displaysrc($callback, $ref_rhevm, $response, "hosts", "");
# display the nics for the vm
my $hostid = $hostids[0];
($rc, $id, $stat, $response) = search_src($ref_rhevm, "hosts:host_nics", "/$hostid/nics");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "host_nics", " ");
}
# TODO, display the vm for host always?
if (1 || $vm4host) {
my ($rc, $id, $stat, $response) = search_src($ref_rhevm, "vms", "Host.name%3D$hyp");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "vms", " ");
}
}
}
}
# Display virtual machines
foreach my $vm (@vms) {
# Get vm
my ($rc, $id, $stat, $response) = search_src($ref_rhevm, "vms", $vm);
unless ($rc) {
my @vmids = displaysrc($callback, $ref_rhevm, $response, "vms", "");
# display the disks for the vm
my $vmid = $vmids[0];
my ($rc, $id, $stat, $response) = search_src($ref_rhevm, "vms:disks", "/$vmid/disks");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "disks", " ");
}
# display the nics for the vm
($rc, $id, $stat, $response) = search_src($ref_rhevm, "vms:nics", "/$vmid/nics");
unless ($rc) {
displaysrc($callback, $ref_rhevm, $response, "nics", " ");
}
}
}
}
}
# Create virtual machine
# Since the configuration for a vm is complicated, all the parameters will be gotten from vm table
sub mkvm {
my $callback = shift;
my $rhevm_hash = shift;
my $nodes = shift;
my $upmac; # used to update the mac table
my $mactab = new xCAT::Table('mac', -create => 1);
# Get the attributes for the node from the vm table
my $vmtab = xCAT::Table->new('vm', -create => 0);
my $vment = $vmtab->getNodesAttribs($nodes, [ 'master', 'host', 'cluster', 'virtflags', 'storage', 'storagemodel', 'memory', 'cpus', 'nics', 'nicmodel', 'bootorder', 'vidproto' ]);
# Generate the xml content for add the storage
# Note: this is an independent action after the vm creating
my $adds = XML::LibXML->createDocument();
my $asroot = $adds->createElement("disk");
$adds->setDocumentElement($asroot);
# set the disk type: system and data
my $disktype_ele = $adds->createElement("type");
$asroot->appendChild($disktype_ele);
my $disktype_t = XML::LibXML::Text->new("system");
$disktype_ele->appendChild($disktype_t);
# set the bootable
my $diskboot_ele = $adds->createElement("bootable");
$asroot->appendChild($diskboot_ele);
my $diskboot_t = XML::LibXML::Text->new("true");
$diskboot_ele->appendChild($diskboot_t);
my $sd_ele = $adds->createElement("storage_domains");
$asroot->appendChild($sd_ele);
my $sdid_ele = $adds->createElement("storage_domain");
$sd_ele->appendChild($sdid_ele);
# add size of disk
my $sdsize_ele = $adds->createElement("size");
$asroot->appendChild($sdsize_ele);
my $sdsize_t = XML::LibXML::Text->new("");
$sdsize_ele->appendChild($sdsize_t);
# add the element for type of disk interface
my $sdif_ele = $adds->createElement("interface");
$asroot->appendChild($sdif_ele);
my $sdif_t = XML::LibXML::Text->new("virtio");
$sdif_ele->appendChild($sdif_t);
# add the disk format element
my $sdft_ele = $adds->createElement("format");
$asroot->appendChild($sdft_ele);
my $sdfm_t = XML::LibXML::Text->new("cow");
$sdft_ele->appendChild($sdfm_t);
# Generate the xml content for add network interface
# Note: this is an independent action after the vm creating
my $addnw = XML::LibXML->createDocument();
my $anwroot = $addnw->createElement("nic");
$addnw->setDocumentElement($anwroot);
# add the interface type element
my $nwif_ele = $addnw->createElement("interface");
$anwroot->appendChild($nwif_ele);
my $nwif_t = XML::LibXML::Text->new("virtio");
$nwif_ele->appendChild($nwif_t);
# add the name element
my $nwname_ele = $addnw->createElement("name");
$anwroot->appendChild($nwname_ele);
my $nwname_t = XML::LibXML::Text->new("nic1");
$nwname_ele->appendChild($nwname_t);
# add the network element which specify which network this nic
# will be added to
my $nwnw_ele = $addnw->createElement("network");
$anwroot->appendChild($nwnw_ele);
my $nwnwname_ele = $addnw->createElement("name");
$nwnw_ele->appendChild($nwnwname_ele);
my $nwnwname_t = XML::LibXML::Text->new("rhevm");
$nwnwname_ele->appendChild($nwnwname_t);
# create a mac element
my $nwmac_ele = $addnw->createElement("mac");
foreach my $rhevm (keys %{$rhevm_hash}) {
my %node_hyp;
my %hostid;
my $success = 0;
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
# generate the node that will be handled
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if (defined $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}) {
foreach (@{ $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node} }) {
$node_hyp{$_}{hyp} = $rhevh;
$hostid{$rhevh} = 1;
}
}
}
}
if (defined $rhevm_hash->{$rhevm}->{node}) {
foreach (@{ $rhevm_hash->{$rhevm}->{node} }) {
$node_hyp{$_}{hyp} = "";
}
}
# get the host id
# this is used for the case that needs locate vm to a spcific host
foreach my $host (keys %hostid) {
my ($rc, $id, $stat) = search_src($ref_rhevm, "hosts", $host);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "Cannot find $host in the rhevm.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return;
}
$hostid{$host} = $id;
}
my @nodes = (keys %node_hyp);
my $macmac = $mactab->getNodesAttribs(\@nodes, ['mac']);
foreach my $node (@nodes) {
my $myvment = $vment->{$node}->[0];
unless ($myvment) {
my $rsp;
push @{ $rsp->{data} }, "$node: has NOT entry in vm table.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
# Check the existence of the node
my ($rc, $id, $stat) = search_src($ref_rhevm, "vms", $node);
if (!$rc) {
my $rsp;
push @{ $rsp->{data} }, "$node: virtual machine has been created.";
xCAT::MsgUtils->message("I", $rsp, $callback);
next;
}
#Create the virtual machine first
my $api = "/api/vms";
my $method = "POST";
# generate the content
# configure the template
my $hastpl = 0;
my $tplele;
if ($myvment->{master}) {
$tplele = "<template><name>$myvment->{master}</name></template>";
$hastpl = 1;
} else {
$tplele = "<template><name>Blank</name></template>";
}
# configure memory
my $memele;
if ($myvment->{memory}) {
my $memsize = $myvment->{memory};
$memsize =~ s/g/000000000/i;
$memsize =~ s/m/000000/i;
$memele = "<memory>$memsize</memory>";
} elsif (!$hastpl) {
$memele = "<memory>2000000000</memory>";
}
# set the cpu
my $cpuele;
if ($myvment->{cpus}) {
my ($socketnum, $corenum) = split(':', $myvment->{cpus});
unless ($corenum) { $corenum = 1; }
$cpuele = "<cpu><topology cores=\"$corenum\" sockets=\"$socketnum\"/></cpu>";
} elsif (!$hastpl) {
$cpuele = "<cpu><topology cores=\"1\" sockets=\"1\"/></cpu>"
}
# configure bootorder
# there's a bug that sequence is not correct to set two order, so currently just set one
my $boele;
if ($myvment->{bootorder}) {
my ($firstbr, $secbr) = split(',', $myvment->{bootorder});
if ($secbr) {
$boele = "<os><boot dev=\"$firstbr\"/><boot dev=\"$secbr\"/><boot/></os>";
} else {
$boele = "<os><boot dev=\"$firstbr\"/><boot/></os>";
}
} elsif (!$hastpl) {
$boele = "<os><boot dev=\"network\"/><boot/></os>";
}
my $disele;
if ($myvment->{vidproto}) {
$disele = "<display><type>$myvment->{vidproto}</type></display>";
} else {
$disele = "<display><type>vnc</type></display>";
}
my $affinity;
if ($myvment->{virtflags}) {
# parse the specific parameters from vm.virtflags
my @pairs = split(':', $myvment->{virtflags});
foreach my $pair (@pairs) {
my ($name, $value) = split('=', $pair);
if ($name eq "placement_affinity") {
# set the affinity for placement_policy
$affinity = "<affinity>$value</affinity>"
}
}
}
if (!$affinity && !$hastpl) {
$affinity = "<affinity>migratable</affinity>"
}
my $hostele;
if ($myvment->{host}) {
$hostele = "<host id=\"$hostid{$myvment->{host}}\"/>";
}
my $placement_policy;
if ($affinity || $hostele) {
$placement_policy = "<placement_policy>$hostele$affinity</placement_policy>";
}
# set the cluster for the vm
my $clusterele;
if ($myvment->{cluster}) {
$clusterele = "<cluster><name>$myvment->{cluster}</name></cluster>";
} else {
$clusterele = "<cluster><name>Default</name></cluster>";
}
my $content = "<vm><type>server</type><name>$node</name>$clusterele$tplele$memele$cpuele$boele$placement_policy$disele</vm>";
my $request = genreq($ref_rhevm,
$method,
$api,
$content);
my $response;
my $vmid;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if (!$rc) {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
my $newvm;
if (defined($doc->findnodes("/vm/name")->[0])) {
$newvm = $doc->findnodes("/vm/name")->[0]->textContent();
}
if ($newvm ne $node) {
my $rsp;
push @{ $rsp->{data} }, "$node: create virtual machine failed.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
my $vm = $doc->findnodes("/vm")->[0];
$vmid = $vm->getAttribute('id');
$success = 1;
} else {
my $rsp;
push @{ $rsp->{data} }, $response;
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
#Add the disk for the vm from storage domain
my @disklist = split('\|', $myvment->{storage});
foreach (@disklist) {
my ($sdname, $disksize, $disktype) = split(':', $_);
if ($sdname) {
if (waitforcomplete($ref_rhevm, "/api/vms/$vmid", "/vm/status/state=down", 30)) {
my $rsp;
push @{ $rsp->{data} }, "$node: failed to waiting the vm gets to \"down\" state.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
$success = 0;
#Get the storage domain by name
my $sdid;
($rc, $sdid, $stat) = search_src($ref_rhevm, "storagedomains", $sdname);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "Could not get the storage domain $sdname.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
if ($sdid) {
$api = "/api/vms/$vmid/disks";
$method = "POST";
# generate the content
if ($disktype) {
$disktype_t->setData($disktype);
if ($disktype eq "system") {
$diskboot_t->setData("true");
} else {
$diskboot_t->setData("false");
}
} else {
$disktype_t->setData("system");
$diskboot_t->setData("true");
}
# set the size of disk
if ($disksize) {
$disksize =~ s/g/000000000/i;
$disksize =~ s/m/000000/i;
} else {
$disksize = "5000000000"; #5G is default
}
$sdid_ele->setAttribute("id", $sdid);
$sdsize_t->setData($disksize);
# set the interface type and format for disk
if ($myvment->{storagemodel}) {
my ($iftype, $iffmt) = split(':', $myvment->{storagemodel});
$sdif_t->setData($iftype);
$sdfm_t->setData($iffmt);
} else {
$sdif_t->setData("virtio");
$sdfm_t->setData("cow");
}
$request = genreq($ref_rhevm, $method, $api, $adds->toString());
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if (!$rc) {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if (defined($doc->findnodes("/fault")->[0])) {
my $rsp;
push @{ $rsp->{data} }, "$node: Add disk failed for virtual machine";
if ($doc->findnodes("/fault/detail")->[0]) {
push @{ $rsp->{data} }, $doc->findnodes("/fault/detail")->[0]->textContent();
}
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
my $state;
if (defined($doc->findnodes("/disk/creation_status/state")->[0])) {
$state = $doc->findnodes("/disk/creation_status/state")->[0]->textContent();
}
if ($state =~ /fail/i) {
my $rsp;
push @{ $rsp->{data} }, "$node: Add disk failed for virtual machine";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
$success = 1;
} else {
my $rsp;
push @{ $rsp->{data} }, $response;
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
}
}
}
# Add the network interface
#Get the network by name
my @nics;
if ($myvment->{nics}) {
@nics = split(/\|/, $myvment->{nics});
}
if (!@nics && !$hastpl) {
# default is to add nic1 to manament network 'rhevm'
push @nics, "rhevm:eth0:yes";
}
if (@nics) {
if (waitforcomplete($ref_rhevm, "/api/vms/$vmid", "/vm/status/state=down", 30)) {
my $rsp;
push @{ $rsp->{data} }, "$node: failed to waiting the vm gets to \"down\" state.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
}
# if no installnic is specified, set the firstmac to mac.mac
my $firstmac;
# Search the nic
my %oldmac;
($rc, undef, $stat, $response) = search_src($ref_rhevm, "vms:nics", "/$vmid/nics");
unless ($rc) {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
my @nicnodes = $doc->findnodes("/nics/nic");
foreach my $nic (@nicnodes) {
if (defined($nic->findnodes("name"))) {
my $ethname = getAttr($nic, "name");
my $mac = getAttr($nic, "mac", "address");
$oldmac{$ethname} = $mac;
unless ($firstmac) {
$firstmac = $mac;
}
}
}
}
foreach my $nic (@nics) {
# format of nic: [networkname:ifname:installnic]
my ($nwname, $ifname, $instnic) = split(':', $nic);
if (defined($oldmac{$ifname})) {
# The nic has been defined, mostly by clone
if ($instnic) {
$upmac->{$node}->{mac} = $oldmac{$ifname};
}
next;
}
# start the configuring
$success = 0;
my $nwid;
($rc, $nwid, $stat) = search_src($ref_rhevm, "networks", "$nwname");
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "Could not get the network $nwname.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
$api = "/api/vms/$vmid/nics";
$method = "POST";
# generate the content
# set the nic interface type
if ($myvment->{nicmodel}) {
$nwif_t->setData($myvment->{nicmodel});
} else {
$nwif_t->setData("virtio");
}
$nwname_t->setData($ifname);
$nwnwname_t->setData($nwname);
# set the mac address element
# if no entry in mac.mac, and this is install nic, THEN use the existed mac
# otherwise create a new mac automatically by rhev-m
my $orgmac;
if ($instnic && defined($macmac->{$node}->[0]) && defined($macmac->{$node}->[0]->{'mac'})) {
$orgmac = $macmac->{$node}->[0]->{'mac'};
$anwroot->appendChild($nwmac_ele);
$nwmac_ele->setAttribute("address", $orgmac);
} else {
$anwroot->removeChild($nwmac_ele);
}
$content = $addnw->toString();
$request = genreq($ref_rhevm,
$method,
$api,
$content);
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if (!$rc) {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if (defined($doc->findnodes("/nic/mac")->[0])) {
my $realmac = $doc->findnodes("/nic/mac")->[0]->getAttribute("address");
unless ($firstmac) {
$firstmac = $realmac;
}
if ($instnic) {
$upmac->{$node}->{mac} = $realmac;
}
$success = 1;
next;
} else {
my $rsp;
push @{ $rsp->{data} }, "$node: failed to create virtual machine.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
} else {
my $rsp;
push @{ $rsp->{data} }, $response;
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
}
if (!$upmac->{$node}->{mac} && $firstmac) {
$upmac->{$node}->{mac} = $firstmac;
}
if ($success) {
my $rsp;
push @{ $rsp->{data} }, "$node: Succeeded";
xCAT::MsgUtils->message("I", $rsp, $callback);
}
}
}
$mactab->setNodesAttribs($upmac);
}
# Remove a virtual machine
sub rmvm {
my $callback = shift;
my $rhevm_hash = shift;
my $args = shift;
my $force;
if ($args) {
@ARGV = @{$args};
GetOptions('f' => \$force);
}
foreach my $rhevm (keys %{$rhevm_hash}) {
my @nodes;
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
# generate the node that will be handled
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if (defined $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node} };
}
}
}
if (defined $rhevm_hash->{$rhevm}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{node} };
}
# perform the action against the node
foreach my $node (@nodes) {
# Get the ID of node
my ($rc, $id, $state) = search_src($ref_rhevm, "vms", $node);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$node: node was not defined in the rhevm.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
} elsif (!defined($id)) {
my $rsp;
push @{ $rsp->{data} }, "$node: node was not defined in the rhevm.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
# Remove the vm
my $api = "/api/vms/$id";
my $method = "DELETE";
my $content = "<action/>";
if ($force) {
$content = "<action><force>true</force></action>";
}
my $request = genreq($ref_rhevm,
$method,
$api,
$content);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc == 2) {
my $rsp;
push @{ $rsp->{data} }, "$node: succeeded.";
xCAT::MsgUtils->message("I", $rsp, $callback);
next;
} else {
my $rsp;
push @{ $rsp->{data} }, $response;
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
}
}
}
# Change virtual machine
sub chvm {
my $callback = shift;
my $rhevm_hash = shift;
my $nodes = shift;
# Get the mac address for the nodes from the mac table
my $mactab = new xCAT::Table('mac', -create => 1);
# Get the attributes for the nodes from the vm table
my $vmtab = xCAT::Table->new('vm', -create => 0);
my $vment = $vmtab->getNodesAttribs($nodes, [ 'master', 'host', 'cluster', 'virtflags', 'storage', 'storagemodel', 'memory', 'cpus', 'nics', 'nicmodel', 'bootorder', 'vidproto' ]);
foreach my $rhevm (keys %{$rhevm_hash}) {
my %node_hyp;
my %hostid;
my $success = 0;
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
# generate the node that will be handled
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if (defined $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}) {
foreach (@{ $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node} }) {
$node_hyp{$_}{hyp} = $rhevh;
$hostid{$rhevh} = 1;
}
}
}
}
if (defined $rhevm_hash->{$rhevm}->{node}) {
foreach (@{ $rhevm_hash->{$rhevm}->{node} }) {
$node_hyp{$_}{hyp} = "";
}
}
# get the host id
# this is used for the case that needs locate vm to a spcific host
foreach my $host (keys %hostid) {
my ($rc, $id, $stat) = search_src($ref_rhevm, "hosts", $host);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "Cannot find $host in the rhevm.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return;
}
$hostid{$host} = $id;
}
my @nodes = (keys %node_hyp);
my $macmac = $mactab->getNodesAttribs(\@nodes, ['mac']);
foreach my $node (@nodes) {
my $myvment = $vment->{$node}->[0];
unless ($myvment) {
my $rsp;
push @{ $rsp->{data} }, "$node: has NOT entry in vm table.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
# Check the existence of the node
my ($rc, $vmid, $stat) = search_src($ref_rhevm, "vms", $node);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$node: virtual machine was not created.";
xCAT::MsgUtils->message("I", $rsp, $callback);
next;
}
# generate the content
my $tplele;
if ($myvment->{master}) {
$tplele = "<template><name>$myvment->{master}</name></template>";
}
# configure memory
my $memele;
if ($myvment->{memory}) {
my $memsize = $myvment->{memory};
$memsize =~ s/g/000000000/i;
$memsize =~ s/m/000000/i;
$memele = "<memory>$memsize</memory>";
}
# set the cpu
my $cpuele;
if ($myvment->{cpus}) {
my ($socketnum, $corenum) = split(':', $myvment->{cpus});
unless ($corenum) { $corenum = 1; }
$cpuele = "<cpu><topology cores=\"$corenum\" sockets=\"$socketnum\"/></cpu>";
}
# configure bootorder
# there's a bug that sequence is not correct to set two order, so currently just set one
my $boele;
if ($myvment->{bootorder}) {
my ($firstbr, $secbr) = split(',', $myvment->{bootorder});
if ($secbr) {
$boele = "<os><boot dev=\"$firstbr\"/><boot dev=\"$secbr\"/><boot/></os>";
} else {
$boele = "<os><boot dev=\"$firstbr\"/><boot/></os>";
}
}
my $disele;
if ($myvment->{vidproto}) {
$disele = "<display><type>$myvment->{vidproto}</type></display>";
}
my $affinity;
if ($myvment->{virtflags}) {
# parse the specific parameters from vm.virtflags
my @pairs = split(':', $myvment->{virtflags});
foreach my $pair (@pairs) {
my ($name, $value) = split('=', $pair);
if ($name eq "placement_affinity") {
# set the affinity for placement_policy
$affinity = "<affinity>$value</affinity>"
}
}
}
my $hostele;
if ($myvment->{host}) {
$hostele = "<host id=\"$hostid{$myvment->{host}}\"/>";
}
my $placement_policy;
if ($affinity) {
$placement_policy = "<placement_policy>$hostele$affinity</placement_policy>";
} elsif ($hostele) {
$affinity = "<affinity>migratable</affinity>";
$placement_policy = "<placement_policy>$hostele$affinity</placement_policy>";
}
# set the cluster for the vm
my $clusterele;
if ($myvment->{cluster}) {
$clusterele = "<cluster><name>$myvment->{cluster}</name></cluster>";
}
my $api = "/api/vms/$vmid";
my $method = "PUT";
my $content = "<vm><type>server</type><name>$node</name>$clusterele$tplele$memele$cpuele$boele$placement_policy$disele</vm>";
my $request = genreq($ref_rhevm, $method, $api, $content);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$node: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
my $state;
if ($node eq getAttr($doc, "/vm/name")) {
my $rsp;
push @{ $rsp->{data} }, "$node: change vm completed.";
xCAT::MsgUtils->message("I", $rsp, $callback);
next;
} else {
my $rsp;
push @{ $rsp->{data} }, "$node: change vm failed.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
}
}
}
}
# Clone the virtual machine
# create template first
sub clonevm {
my $callback = shift;
my $rhevm_hash = shift;
my $args = shift;
my ($template, $basemaster);
if ($args) {
@ARGV = @{$args};
GetOptions('t=s' => \$template,
'b' => \$basemaster);
}
my $ref_rhevm;
my @nodes;
foreach my $rhevm (keys %{$rhevm_hash}) {
# generate the hash of rhevm which will be used for the action functions
$ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
# generate the node that will be handled
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if (defined $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node} };
}
}
}
if (defined $rhevm_hash->{$rhevm}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{node} };
}
}
my $node = $nodes[0];
# create a template from a vm
if ($template) {
# Get the ID of node
my ($rc, $vmid, $state) = search_src($ref_rhevm, "vms", $node);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$node: node was not defined in the rhevm.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return;
}
if ($state ne "down") {
my $rsp;
push @{ $rsp->{data} }, "$node: vm needs to be shutdown to run the clone.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return;
}
my $api = "/api/templates";
my $method = "POST";
my $content = "<template><name>$template</name><vm id=\"$vmid\"/></template>";
my $request = genreq($ref_rhevm, $method, $api, $content);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
return (1, $response);
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc->findnodes("/template/status/state")->[0]) {
my $state = $doc->findnodes("/template/status/state")->[0]->textContent();
my $rsp;
push @{ $rsp->{data} }, "$template: $state.";
xCAT::MsgUtils->message("I", $rsp, $callback);
} else {
my $rsp;
push @{ $rsp->{data} }, "$template: failed to get the status.";
xCAT::MsgUtils->message("I", $rsp, $callback);
}
return;
}
}
}
# Set the boot sequence for the vm
sub rsetboot {
my $callback = shift;
my $rhevm_hash = shift;
my $args = shift;
my ($showstat, $bootdev);
if ($args) {
my $arg = $args->[0];
if ($arg =~ /^stat/) {
$showstat = 1;
} else {
$bootdev = $arg;
}
} else {
$showstat = 1;
}
my ($firstbr, $secbr);
if ($bootdev) {
($firstbr, $secbr) = split(',', $bootdev);
if (($firstbr && $firstbr !~ /^(network|hd)$/) || ($secbr && $secbr !~ /^(network|hd)$/)) {
my $rsp;
push @{ $rsp->{data} }, "Supported boot device: network, hd";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 1;
}
}
foreach my $rhevm (keys %{$rhevm_hash}) {
my @nodes;
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
# generate the node that will be handled
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if (defined $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node} };
}
}
}
if (defined $rhevm_hash->{$rhevm}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{node} };
}
foreach my $node (@nodes) {
# Get the ID of vm
my ($rc, $vmid, $state, $response) = search_src($ref_rhevm, "vms", $node);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$node: node was not defined in the rhevm.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
if ($showstat) {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
my @bootdevs = getAttr($doc, "/vms/vm/os/boot", "dev");
my $bootlist = join(',', @bootdevs);
my $rsp;
push @{ $rsp->{data} }, "$node: $bootlist";
xCAT::MsgUtils->message("I", $rsp, $callback);
next;
}
# configure bootorder
my $boele;
if ($secbr) {
$boele = "<os><boot dev=\"$firstbr\"/><boot dev=\"$secbr\"/><boot/></os>";
} else {
$boele = "<os><boot dev=\"$firstbr\"/><boot/></os>";
}
my $api = "/api/vms/$vmid";
my $method = "PUT";
my $content = "<vm>$boele</vm>";
my $request = genreq($ref_rhevm, $method, $api, $content);
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$node: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
} else {
my $rsp;
push @{ $rsp->{data} }, "$node: set boot order completed.";
xCAT::MsgUtils->message("I", $rsp, $callback);
next;
}
}
}
}
#Migrate the virtual machine
sub rmigrate {
my $callback = shift;
my $rhevm_hash = shift;
my $args = shift;
my ($template, $basemaster);
unless ($args) {
my $rsp;
push @{ $rsp->{data} }, "Needs a target host.";
xCAT::MsgUtils->message("E", $rsp, $callback);
}
my $host = $args->[0];
foreach my $rhevm (keys %{$rhevm_hash}) {
my @nodes;
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
# generate the node that will be handled
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if (defined $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node} };
}
}
}
if (defined $rhevm_hash->{$rhevm}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{node} };
}
foreach my $node (@nodes) {
# Get the ID of vm
my ($rc, $vmid, $state) = search_src($ref_rhevm, "vms", $node);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$node: node was not defined in the rhevm.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
my $hostid;
($rc, $hostid, $state) = search_src($ref_rhevm, "hosts", $host);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$host: host was not defined in the rhevm.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
# Remove the vm
my $api = "/api/vms/$vmid/migrate";
my $method = "POST";
my $content = "<action><host id=\"$hostid\"/><force>true</force></action>";
my $request = genreq($ref_rhevm, $method, $api, $content);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$node: $response.";
xCAT::MsgUtils->message("E", $rsp, $callback);
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc->findnodes("/action/status/state")->[0]) {
my $state = $doc->findnodes("/action/status/state")->[0]->textContent();
my $rsp;
push @{ $rsp->{data} }, "$node: migrated to $host: $state.";
xCAT::MsgUtils->message("I", $rsp, $callback);
} else {
my $rsp;
push @{ $rsp->{data} }, "$node: failed to migrate to $host.";
xCAT::MsgUtils->message("E", $rsp, $callback);
}
}
}
}
}
# Hardware control
# rpower <vm> on/off/reset
sub power {
my $callback = shift;
my $rhevm_hash = shift;
my $args = shift;
foreach my $rhevm (keys %{$rhevm_hash}) {
my @nodes;
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
# generate the node that will be handled
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if (defined $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node} };
}
}
}
if (defined $rhevm_hash->{$rhevm}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{node} };
}
# perform the action against the node
foreach my $node (@nodes) {
# Get the ID of node
my ($rc, $id, $state) = search_src($ref_rhevm, "vms", $node);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$node: node was not defined in the rhevm.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
my $output;
if ($args->[0] eq 'on') {
if ($state eq "up" || $state eq "powering_up") {
$output = "$node: on";
} else {
my ($rc, $msg) = power_action($ref_rhevm, $id, 'start');
if (!$rc) {
$output = "$node: on";
} else {
$output = "$node: $msg";
}
}
} elsif ($args->[0] eq 'off') {
if ($state eq "down" || $state eq "powering_down" || $state eq "powered_down") {
$output = "$node: off";
} else {
my ($rc, $msg) = power_action($ref_rhevm, $id, 'stop');
if (!$rc) {
$output = "$node: off";
} else {
$output = "$node: $msg";
}
}
} elsif ($args->[0] eq 'reset' || $args->[0] eq 'boot') {
if ($state eq "up" || $state eq "powering_up") {
my ($rc, $msg) = power_action($ref_rhevm, $id, 'stop');
if (!$rc) {
if (waitforcomplete($ref_rhevm, "/api/vms/$id", "/vm/status/state=down", 30)) {
my $rsp;
push @{ $rsp->{data} }, "$node: failed to waiting the vm gets to \"down\" state.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
($rc, $msg) = power_action($ref_rhevm, $id, 'start');
if (!$rc) {
$output = "$node: $args->[0]";
} else {
$output = "$node: $msg";
}
} else {
$output = "$node: $msg";
}
} else {
my ($rc, $msg) = power_action($ref_rhevm, $id, 'start');
if (!$rc) {
$output = "$node: $args->[0]";
} else {
$output = "$node: $msg";
}
}
} elsif ($args->[0] eq 'softoff') {
if ($state eq "down" || $state eq "powering_down" || $state eq "powered_down") {
$output = "$node: softoff";
} else {
my ($rc, $msg) = power_action($ref_rhevm, $id, 'shutdown');
if (!$rc) {
$output = "$node: softoff";
} else {
$output = "$node: $msg";
}
}
} elsif ($args->[0] eq 'suspend') {
if ($state eq "suspended") {
$output = "$node: suspended";
} else {
my ($rc, $msg) = power_action($ref_rhevm, $id, 'suspend');
if (!$rc) {
$output = "$node: suspended";
} else {
$output = "$node: $msg";
}
}
} elsif ($args->[0] =~ /^stat/) {
if ($state eq "down") {
$output = "$node: off";
} elsif ($state eq "up") {
$output = "$node: on";
} else {
$output = "$node: $state";
}
}
my $rsp;
push @{ $rsp->{data} }, $output;
xCAT::MsgUtils->message("I", $rsp, $callback);
}
}
}
# Do the power control
sub power_action {
my $rhevm = shift;
my $id = shift;
my $action = shift;
my $api = "/api/vms/$id/$action";
my $method = "POST";
my $content = "<action/>";
my $request = genreq($rhevm,
$method,
$api,
$content);
my ($rc, $response) = send_req($rhevm, $request->as_string());
if ($rc) {
return (1, $response);
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
my $state = $doc->findnodes("/action/status/state")->[0]->textContent();
return (0, $state);
}
return (1);
}
# Search resource inside rhev-m
# $orgtype: format: [type] or [container:type]
# [type] could be: datacenter, cluster ...
# [container:type could be: 'vms:nics' ('vms' is the container of nic, 'nics' is the real type), vms:disk
# $node: could be name for a resource; or the path of resource when start with '/'
# if no $node specified, search all resource with the '$type'
# $individual: do search for a resource individually
#
# return -1-parameter error; 11-nosuch id;
sub search_src {
my $rhevm = shift;
my $orgtype = shift;
my $node = shift;
my $individual = shift;
my $api;
my ($container, $type);
if ($orgtype =~ /:/) {
($container, $type) = split(/:/, $orgtype);
} else {
$container = $type = $orgtype;
}
my $ispath;
if ($node) {
if ($node =~ /^\//) {
# is a path
$api = "/api/$container" . $node;
$ispath = 1;
} elsif ($node =~ /\%3D/) {
$api = "/api/$container?search=$node";
} elsif ($node eq "xxxxxx_all_objs") {
$api = "/api/$container";
} else {
$api = "/api/$container?search=name%3D$node";
if ($type eq "hosts") {
#append the domain for the hypervisor
$api .= "*";
}
}
} else {
$api = "/api/$container";
}
my $method = "GET";
my $content = "";
my $request = genreq($rhevm, $method, $api, $content);
my ($rc, $response) = send_req($rhevm, $request->as_string());
if ($rc) {
return ($rc, $response);
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my ($id, $state, $idstr, $ststr);
if ($type eq "vms") {
$idstr = "/vms/vm";
} elsif ($type eq "hosts") {
if ($individual) {
$idstr = "/host";
} else {
$idstr = "/hosts/host";
}
} elsif ($type eq "templates") {
$idstr = "/templates/template";
} elsif ($type eq "storagedomains") {
if ($individual) {
$idstr = "/storage_domain";
} else {
$idstr = "/storage_domains/storage_domain";
}
} elsif ($type eq "networks") {
if ($individual) {
$idstr = "/network";
} else {
$idstr = "/networks/network";
}
} elsif ($type eq "datacenters") {
$idstr = "/data_centers/data_center";
} elsif ($type eq "clusters") {
$idstr = "/clusters/cluster";
} elsif ($type eq "disks") {
$idstr = "/disks/disk";
} elsif ($type eq "nics") {
$idstr = "/nics/nic";
} elsif ($type eq "host_nics") {
$idstr = "/host_nics/host_nic";
} else {
return (-1);
}
my $idnode;
# network is special that does not support to serach a specific resource,
# so have to do the search by code from all the output
if ($type eq "networks" && $node && (!$ispath)) {
my @nodes = $doc->findnodes($idstr);
foreach my $n (@nodes) {
if ($node eq getAttr($n, "name")) {
$idnode = $n;
last;
}
}
} else {
$idnode = $doc->findnodes($idstr)->[0];
}
if (defined $idnode) {
$id = $idnode->getAttribute('id');
if ($type eq "vms") {
$ststr = "/vms/vm/status/state";
} elsif ($type eq "hosts") {
$ststr = "/hosts/host/status/state";
} elsif ($type eq "templates") {
$ststr = "/templates/template/status/state";
} elsif ($type eq "storagedomains") {
if ($individual) {
$ststr = "/storage_domain/storage/type";
} else {
$ststr = "/storage_domains/storage_domain/storage/type";
}
} elsif ($type eq "networks") {
$ststr = "status/state";
$doc = $idnode;
} elsif ($type eq "datacenters") {
$ststr = "/data_centers/data_center/status/state";
} elsif ($type eq "clusters") {
$ststr = "/clusters/cluster/name";
} elsif ($type eq "disks") {
$ststr = "/disks/disk/status/state";
} elsif ($type eq "nics") {
$ststr = "/nics/nic/name";
} elsif ($type eq "host_nics") {
$ststr = "/host_nics/host_nic/status/state";
} else {
return (-1);
}
my $statenode = $doc->findnodes($ststr)->[0];
if (defined $statenode) {
$state = $statenode->textContent();
}
# no id was found
if (!$id) {
return (11);
}
return (0, $id, $state, $response);
} else {
return (11);
}
}
}
return (1);
}
# Get the value for a element from the xml of rest api response
sub getAttr {
my $doc = shift;
my $path = shift;
my $att = shift;
my @nodes;
if ($path) {
# handle the cases that has multiple entries for one atributes like boot order
@nodes = $doc->findnodes($path);
} else {
push @nodes, $doc;
}
if (@nodes) {
my @value;
foreach my $node (@nodes) {
if ($att) {
push @value, $node->getAttribute($att);
} else {
push @value, $node->textContent();
}
}
return join(',', @value);
} else {
return "";
}
}
# It's a command that will be triggered from the httpd when the installation of rhev-h has finished
# Run nodeset and updatenodestat to update the status for the rhev-h
sub rhevhupdateflag {
my $request = shift;
my $callback = shift;
my $subreq = shift;
my $node = $request->{node};
# run the nodeset xx next
$subreq->({ command => ['nodeset'], node => $node, arg => ['next'] }, $callback);
# run the 'updatenodestat <node> booted'
$subreq->({ command => ['updatenodestat'], node => $node, arg => ['booted'] }, $callback);
}
# use the md5 to authorize the passwd
sub authpw {
my $passwd = shift;
if ($passwd =~ /^\$1\$/) {
return $passwd;
} else {
my $cmd = "openssl passwd -1 $passwd";
return xCAT::Utils->runcmd($cmd, -1);
}
}
# Configure the network for host
# create the network for cluster if it does not exist
# configure the interface and add them to the corresponding network
sub cfghypnw {
my $callback = shift;
my $ref_rhevm = shift;
my $host = shift;
my $interface = shift;
my $datacenter = shift;
# the format of the interface attirbute
# networkname:interfacename:bootpro:IP:netmask:gateway
my @if = split(/\|/, $interface);
foreach (@if) {
my ($netname, $ifname, $bprotocol, $ip, $nm, $gw) = split(':', $_);
unless ($netname && $ifname) {
my $rsp;
push @{ $rsp->{data} }, "$host: Missing network name or interface name: $_.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
# get host id
my ($rc, $hostid, $stat) = search_src($ref_rhevm, "hosts", $host);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$host: host was not created.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
# get network interface and configure it
my $nicid;
my $api = "/api/hosts/$hostid/nics";
my $method = "GET";
my $request = genreq($ref_rhevm, $method, $api, "");
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$host: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
my @hostnics = $doc->findnodes("/host_nics/host_nic");
foreach my $nicnode (@hostnics) {
if ($ifname eq getAttr($nicnode, "name")) {
$doc = $nicnode;
last;
}
}
if ($doc) {
my $attr;
if ($attr = getAttr($doc, "", "id")) {
$nicid = $attr;
} else {
my $rsp;
push @{ $rsp->{data} }, "$host: does not have interface $ifname.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
# get the network
my $oldnetname;
my $oldnetid;
if ($attr = getAttr($doc, "network", "id")) {
$oldnetid = $attr;
}
if ($attr = getAttr($doc, "network/name")) {
$oldnetname = $attr;
}
# attach the nic to the network if needed
# search the network
my $newnetid;
($rc, $newnetid, $stat) = search_src($ref_rhevm, "networks", $netname);
if ($rc) {
if ($rc == 11) {
my $rsp;
push @{ $rsp->{data} }, "$host: network: $netname does not exist.";
xCAT::MsgUtils->message("E", $rsp, $callback);
} else {
my $rsp;
push @{ $rsp->{data} }, "$host: failed to get the network: $netname.";
xCAT::MsgUtils->message("E", $rsp, $callback);
}
next;
}
# detach the nic from current network if old != new
if (($oldnetname && ($oldnetname ne $netname))
|| ($oldnetid && ($oldnetid ne $newnetid))) {
unless ($oldnetid) {
($rc, $oldnetid, $stat) = search_src($ref_rhevm, "networks", $oldnetname);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$host: failed to get the network: $oldnetname.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
}
#detach the interface to the network
if (attach($callback, $ref_rhevm, "/api/hosts/$hostid/nics/$nicid", "network", $oldnetid, 1)) {
my $rsp;
push @{ $rsp->{data} }, "$host: failed to detach $ifname from $netname.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
}
# attach the interface to the network
if ((!$oldnetname || ($oldnetname ne $netname))
&& (!$oldnetid || ($oldnetid ne $newnetid))) {
if (attach($callback, $ref_rhevm, "/api/hosts/$hostid/nics/$nicid", "network", $newnetid)) {
my $rsp;
push @{ $rsp->{data} }, "$host: failed to attach $ifname to $netname.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
generalaction($callback, $ref_rhevm, "/api/hosts/$hostid/commitnetconfig");
}
# check the bootprotocol and network parameters, and configure if needed
if (defined($bprotocol) && $bprotocol =~ /^(dhcp|static)$/) {
my $newpro;
if ($attr = getAttr($doc, "boot_protocol")) {
if ($attr eq "dhcp") {
if ($bprotocol eq "static") {
$newpro = "static";
}
} elsif ($attr eq "static") {
if ($bprotocol eq "dhcp") {
$newpro = "dhcp";
} else {
my ($curip, $curnm, $curgw);
if ($attr = getAttr($doc, "ip", "address")) {
$curip = $attr;
}
if ($attr = getAttr($doc, "ip", "netmask")) {
$curnm = $attr;
}
if ($attr = getAttr($doc, "ip", "gateway")) {
$curgw = $attr;
}
if ($ip ne $curip || $nm ne $curnm || $gw ne $curgw) {
$newpro = "static";
}
}
}
} else {
$newpro = $bprotocol;
}
# Set the attributes for the nic
$api = "/api/hosts/$hostid/nics/$nicid";
$method = "PUT";
my $content;
if (defined($newpro) && $newpro eq "dhcp") {
$content = "<host_nic><boot_protocol>dhcp</boot_protocol></host_nic>";
} elsif (defined($newpro) && $newpro eq "static") {
$content = "<host_nic><boot_protocol>static</boot_protocol><ip address=\"$ip\" netmask=\"$nm\" gateway=\"$gw\"/></host_nic>";
}
if (defined($newpro)) {
my $request = genreq($ref_rhevm, $method, $api, $content);
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$host: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
}
} else {
my $rsp;
push @{ $rsp->{data} }, "$host: the boot procotol was not set or invalid.";
xCAT::MsgUtils->message("E", $rsp, $callback);
next;
}
generalaction($callback, $ref_rhevm, "/api/hosts/$hostid/commitnetconfig");
}
}
}
return 0;
}
# Create a Storage Domain
# The parameters will be gotten from virtsd table
sub mkSD {
my $callback = shift;
my $ref_rhevm = shift;
my $sd = shift;
# get the informage for the SD
my ($rc, $sdid, $state) = search_src($ref_rhevm, "storagedomains", $sd);
if (!$rc) {
my $rsp;
push @{ $rsp->{data} }, "$sd: storagedomains has been defined in the rhevm.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 0;
}
# get the attributes for the SD
my $vsdtab = xCAT::Table->new('virtsd', -create => 0);
my $vsdent = $vsdtab->getAttribs({ 'node' => $sd }, [ 'sdtype', 'stype', 'location', 'host', 'datacenter' ]);
unless ($vsdent) {
my $rsp;
push @{ $rsp->{data} }, "$sd: cannot find the definition for $sd in the virtsd table.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 0;
}
unless ($vsdent->{host}) {
my $rsp;
push @{ $rsp->{data} }, "$sd: a SPM host needs to be specified.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 0;
}
unless ($vsdent->{stype} && (($vsdent->{stype} eq "localfs") || $vsdent->{location})) {
my $rsp;
push @{ $rsp->{data} }, "$sd: the sdtype and location need to be specified.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 0;
}
unless ($vsdent->{stype} =~ /^(nfs|localfs)$/) {
my $rsp;
push @{ $rsp->{data} }, "$sd: supported storage type: nfs, localfs.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 0;
}
# get the host as SPM
my $hostid;
($rc, $hostid, $state) = search_src($ref_rhevm, "hosts", $vsdent->{host});
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$sd: cannot find the host $vsdent->{host}.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 0;
}
# To create the SD
my $api = "/api/storagedomains";
my $method = "POST";
# Create the xml data
my $doc = XML::LibXML->createDocument();
my $root = $doc->createElement("storage_domain");
$doc->setDocumentElement($root);
$root->appendTextChild("name", $sd);
# set the host will be the SPM
my $host_ele = $doc->createElement("host");
$root->appendChild($host_ele);
$host_ele->setAttribute("id", $hostid);
# set the location of storage
my $storage_ele = $doc->createElement("storage");
$root->appendChild($storage_ele);
$storage_ele->appendTextChild("type", $vsdent->{stype});
my ($address, $path) = split(':', $vsdent->{location});
if ($vsdent->{stype} eq "nfs") {
$storage_ele->appendTextChild("address", $address);
$storage_ele->appendTextChild("path", $path);
} elsif ($vsdent->{stype} eq "localfs") {
$storage_ele->appendTextChild("path", "/data/images/rhev");
}
if ($vsdent->{sdtype}) {
$root->appendTextChild("type", $vsdent->{sdtype});
} else {
$root->appendTextChild("type", "data");
}
my $request = genreq($ref_rhevm, $method, $api, $doc->toString);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$sd: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 0;
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my $sdid;
if ($sdid = getAttr($doc, "/storage_domain", "id")) {
if ($vsdent->{stype} eq "localfs") {
#return directly
return $sdid;
}
# attach the storage domain to the datacenter
my $dc = $vsdent->{datacenter};
unless ($dc) { $dc = "default" }
my $dcid;
($rc, $dcid, $state) = search_src($ref_rhevm, "datacenters", $dc);
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$sd: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 0;
}
# attach the storage domain to the datacenter
if (attach($callback, $ref_rhevm, "/api/datacenters/$dcid/storagedomains", "storage_domain", $sdid)) {
my $rsp;
push @{ $rsp->{data} }, "$sd: failed to attach to datacenter:$dc.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 0;
}
# Check the state of the storage domain
if (checkstat($callback, $ref_rhevm, "storage_domain", "/api/datacenters/$dcid/storagedomains/$sdid") ne "active") {
# active the storage domain
if (activate($callback, $ref_rhevm, "/api/datacenters/$dcid/storagedomains/$sdid", $sd)) {
my $rsp;
push @{ $rsp->{data} }, "$sd: failed to activate the storage domain.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 0;
}
}
return $sdid;
}
}
}
return 0;
}
# Activate or Deactive a resource
# 0 - suc; 1 - failed
sub activate {
my $callback = shift;
my $ref_rhevm = shift;
my $path = shift;
my $name = shift;
my $deactivate = shift;
my $api;
if ($deactivate) {
$api = $path . "/deactivate";
} else {
$api = $path . "/activate";
}
my $method = "POST";
my $content = "<action/>";
my $request = genreq($ref_rhevm, $method, $api, $content);
my ($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$name: $response";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 1;
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my $attr;
if ($attr = getAttr($doc, "/action/status/state")) {
if ($attr ne "complete") {
if (waitforcomplte()) {
return 1;
} else {
return 0;
}
} else {
return 0;
}
}
}
}
return 1;
}
# Attach or Detach a resource
# type: network (for host nic), storage_domain (for sd)
# 0 - suc; 1 - failed
sub attach {
my $callback = shift;
my $ref_rhevm = shift;
my $path = shift;
my $type = shift;
my $id = shift;
my $detach = shift;
my $method = "POST";
my $api;
my $content;
if ($type eq "storage_domain") {
if ($detach) {
$api = "$path/$id";
$method = "DELETE";
$content = "";
} else {
$api = $path;
$content = "<$type id=\"$id\"/>";
}
} else {
if ($detach) {
$api = $path . "/detach";
} else {
$api = $path . "/attach";
}
$content = "<action><$type id=\"$id\"/></action>";
}
my $request = genreq($ref_rhevm, $method, $api, $content);
my ($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
# no output for detaching sd from datacenter
if ($rc == 2 && $type eq "storage_domain" && $detach) {
return 0;
}
my $rsp;
push @{ $rsp->{data} }, "$response:$rc";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 1;
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my $attr;
if ($type eq "storage_domain") {
if (getAttr($doc, "/storage_domain/status/state") =~ /(inactive|active)/) {
return 0;
} else {
return 1;
}
} else {
if ("complete" eq getAttr($doc, "/action/status/state")) {
return 0;
} else {
return 1;
}
}
}
}
return 1;
}
# Common subroutine for general action of rest api
sub generalaction {
my $callback = shift;
my $ref_rhevm = shift;
my $api = shift;
my $method = shift;
my $norsp = shift;
my $force = shift;
unless ($method) {
$method = "POST";
}
my $content = "<action/>";
if ($force) {
$content = "<action><force>true</force></action>";
}
my $request = genreq($ref_rhevm, $method, $api, $content);
my ($rc, $response) = send_req($ref_rhevm, $request->as_string());
# no need to handle response for DELETE
if ($norsp && !$response) {
return;
}
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$response";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 1;
}
}
# Check the state of a object
sub checkstat {
my $callback = shift;
my $ref_rhevm = shift;
my $type = shift;
my $api = shift;
my $request = genreq($ref_rhevm, "GET", $api, "");
my ($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
return "";
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
if ($type eq "storage_domain") {
return getAttr($doc, "/storage_domain/status/state")
}
}
}
return "";
}
# delete storage domain
sub deleteSD {
my $callback = shift;
my $ref_rhevm = shift;
my $path = shift;
my $sd = shift;
# get the attributes for the SD
my $vsdtab = xCAT::Table->new('virtsd', -create => 0);
my $vsdent = $vsdtab->getAttribs({ 'node' => $sd }, ['host']);
unless ($vsdent) {
my $rsp;
push @{ $rsp->{data} }, "$sd: cannot find the definition for $sd in the virtsd table.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 1;
}
unless ($vsdent->{host}) {
my $rsp;
push @{ $rsp->{data} }, "$sd: a SPM host needs to be specified.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 1;
}
# get the id of host
my ($rc, $hostid, $stat) = search_src($ref_rhevm, "hosts", $vsdent->{host});
if ($rc) {
my $rsp;
push @{ $rsp->{data} }, "$sd: Cannot find the host: $vsdent->{host} for the storag domain.";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 1;
}
my $method = "DELETE";
my $api = $path;
my $content;
$content = "<storage_domain><host id=\"$hostid\"/><format>true</format></storage_domain>";
my $request = genreq($ref_rhevm, $method, $api, $content);
my $response;
($rc, $response) = send_req($ref_rhevm, $request->as_string());
# no need to handle response for DELETE
if ($rc) {
# no output for detaching sd from datacenter
if ($rc == 2) {
return 0;
}
my $rsp;
push @{ $rsp->{data} }, "$response";
xCAT::MsgUtils->message("E", $rsp, $callback);
return 1;
}
}
sub waitforcomplete {
my $ref_rhevm = shift;
my $api = shift;
my $criteria = shift;
my $timeout = shift;
unless ($timeout) {
$timeout = 10;
}
my ($path, $target) = split('=', $criteria);
my $method = "GET";
my $content = "";
my $start = Time::HiRes::gettimeofday();
while (1) {
my $request = genreq($ref_rhevm, $method, $api, $content);
my ($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
return ($rc, $response);
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
if ($target eq getAttr($doc, $path)) {
return 0;
}
}
}
my $now = Time::HiRes::gettimeofday();
if (($now - $start) > $timeout) {
return 2;
} else {
sleep(0.5);
}
}
return 1;
}
# Get the vid prameters for external video console program to display console
sub getrvidparms {
my $callback = shift;
my $rhevm_hash = shift;
my $nodes = shift;
foreach my $rhevm (keys %{$rhevm_hash}) {
my @nodes;
# generate the hash of rhevm which will be used for the action functions
my $ref_rhevm = { 'name' => $rhevm,
'user' => $rhevm_hash->{$rhevm}->{user},
'pw' => $rhevm_hash->{$rhevm}->{pw} };
# generate the node that will be handled
if (defined $rhevm_hash->{$rhevm}->{host}) {
foreach my $rhevh (keys %{ $rhevm_hash->{$rhevm}->{host} }) {
if (defined $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{host}->{$rhevh}->{node} };
}
}
}
if (defined $rhevm_hash->{$rhevm}->{node}) {
push @nodes, @{ $rhevm_hash->{$rhevm}->{node} };
}
# perform the action against the node
foreach my $node (@nodes) {
my $node = $nodes->[0];
my %consparam;
$consparam{method} = 'kvm';
# get the attributes for vm
my ($rc, undef, undef, $response) = search_src($ref_rhevm, "vms", "$node");
my $vmid;
my $rsp;
if ($rc) {
$rsp->{node}->[0]->{errorcode} = $rc;
$rsp->{node}->[0]->{name}->[0] = $node;
$rsp->{node}->[0]->{error} = $response;
$callback->($rsp);
next;
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my $attr;
if ($attr = getAttr($doc, "/vms/vm", "id")) {
$vmid = $attr;
}
if ($attr = getAttr($doc, "/vms/vm/display/type")) {
$consparam{vidproto} = $attr;
}
if ($attr = getAttr($doc, "/vms/vm/display/address")) {
$consparam{server} = $attr;
}
if ($attr = getAttr($doc, "/vms/vm/display/port")) {
$consparam{vidport} = $attr;
}
}
}
# get the password ticket for the external program to accesss the VNC
my $api = "/api/vms/$vmid/ticket";
my $method = "POST";
my $content = "<action><ticket><expiry>120</expiry></ticket></action>";
my $request = genreq($ref_rhevm, $method, $api, $content);
($rc, $response) = send_req($ref_rhevm, $request->as_string());
if ($rc) {
$rsp->{node}->[0]->{errorcode} = $rc;
$rsp->{node}->[0]->{name}->[0] = $node;
$rsp->{node}->[0]->{error} = $response;
$callback->($rsp);
next;
} else {
my $parser = XML::LibXML->new();
my $doc = $parser->parse_string($response);
if ($doc) {
my $attr;
if ($attr = getAttr($doc, "/action/ticket/value")) {
$consparam{password} = $attr;
}
}
}
$rsp = ();
$rsp->{node}->[0]->{name}->[0] = $node;
foreach (keys %consparam) {
$rsp->{node}->[0]->{data}->[0]->{desc}->[0] = $_;
$rsp->{node}->[0]->{data}->[0]->{contents}->[0] = $consparam{$_};
$callback->($rsp);
}
}
}
return;
}
1;