319 lines
9.3 KiB
Perl
Raw Normal View History

# IBM(c) 2009 EPL license http://www.eclipse.org/legal/epl-v10.html
#This plugin handles 'setupiscsidev' operation for ONTAP platform devices,
#such as the IBM N-series storage subsystems.
#It requires that the storage enclosure have root's public key as an
#authorized key for passwordless
package xCAT_plugin::ontap;
#In ONTAP world, the entire controller is a particular target iqn, that shows
#different lun numbers depending on client iqn.
#iscsi.target, iscsi.server, and iscsi.lun may therefore look identital
#for multiple nodes, but mean entirely different things
#This plugin will populate the lun and targetname
use strict;
BEGIN
{
$::XCATROOT = $ENV{'XCATROOT'} ? $ENV{'XCATROOT'} : '/opt/xcat';
}
use lib "$::XCATROOT/lib/perl";
use warnings "all";
use xCAT::Table;
my $output_handler;
my $newiqns;
my $domain;
my $iscsitab;
my $nodetypeinfo;
sub handled_commands {
return {
testontap => 'ontap',
# setupiscsidev => iscsi:method
#Mayhaps need an iscsiserver table? iscsi table column has a pretty weak
#association, but on the other hand, iscsi servers aren't necessarily a node
#So two options:
# Add field to iscsi table to declare the mechanism to setup device:
# Pros:
# -No extra 'nodes' required to be defined
# Cons:
# -Could be awkward if the enviornment has hybrid iscsi storage providers
# Add new table
# Pros: allows the management method of the target to be abstracted from the node
# cons: requires that every iscsi entity be defined as a node
}
}
my %iscsicfg;
sub sendmsg { #TODO: common code
my $callback = $output_handler;
my $text = shift;
my $node = shift;
my $descr;
my $rc;
if (ref $text eq 'HASH') {
return $callback->($text);
} elsif (ref $text eq 'ARRAY') {
$rc = $text->[0];
$text = $text->[1];
}
if ($text =~ /:/) {
($descr,$text) = split /:/,$text,2;
}
$text =~ s/^ *//;
$text =~ s/ *$//;
my $msg;
my $curptr;
if ($node) {
$msg->{node}=[{name => [$node]}];
$curptr=$msg->{node}->[0];
} else {
$msg = {};
$curptr = $msg;
}
if ($rc) {
$curptr->{errorcode}=[$rc];
$curptr->{error}=[$text];
$curptr=$curptr->{error}->[0];
} else {
$curptr->{data}=[{contents=>[$text]}];
$curptr=$curptr->{data}->[0];
if ($descr) { $curptr->{desc}=[$descr]; }
}
$callback->($msg);
}
sub process_request {
my $request = shift;
$output_handler = shift;
$iscsitab = xCAT::Table->new('iscsi');
unless ($iscsitab) {
sendmsg([1,"iSCSI configuration lacking from the iscsi table"]);
return;
}
my @nodes = @{$request->{node}};
my $sitetab = xCAT::Table->new('site');
(my $dent) = $sitetab->getAttribs({key=>'domain'},'value');
if ($dent and $dent->{value}) {
$domain = $dent->{value};
$domain = join(".",reverse(split(/\./,$domain)));
} else {
sendmsg([1,"Cannot determine domain name for iqn generation from site table"]);
return;
}
my $nodetype =xCAT::Table->new('nodetype',-create=>0);
unless ($nodetype) {
sendmsg([1,"ONTAP plugin requires nodetype table to be populated"]);
return;
}
$nodetypeinfo = $nodetype->getNodesAttribs(\@nodes,['os']);
my $iscsitabdata = $iscsitab->getNodesAttribs(\@nodes,[qw/server lun iname file/]);
my $node;
foreach $node (keys %$iscsitabdata) { #Re-layout the data so we can iterate target-wise rather than node wise
$iscsicfg{$iscsitabdata->{$node}->[0]->{server}}->{$node}=$iscsitabdata->{$node}->[0];
}
my $controller;
foreach $controller (keys %iscsicfg) { #TODO: we need to forkify this
#Develop serially first, then put fork semantics in place
handle_targets($controller,$request);
}
use Data::Dumper;
};
sub get_controller_iqn {
my $controller = shift;
my $output = `ssh $controller iscsi nodename`;
$output =~ s/^[^:]*://;
chomp $output;
$output =~ s/^\s*//;
return $output;
}
sub build_lunmap {
my $controller = shift;
my @nodes = @_;
my $lunmap;
my @groupoutput = `ssh $controller igroup show `;
my @mapoutput = `ssh $controller lun show -m`;
shift @mapoutput; #Get rid of header
shift @mapoutput;
my $groupname;
my $tgr;
my $node;
my $iqn;
my @time = localtime;
my $year = 1900+$time[5];
my $month = $time[4]+1;
my %returns;
foreach $node (@nodes) {
$tgr = undef;
$iqn = $iscsicfg{$controller}->{$node}->{iname};
unless ($iqn) { #We must control client iqn, ONTAP acls require it
$newiqns->{$node} = sprintf("iqn.%d-%02d.%s:%s-initiator",$year,$month,$domain,$node);
$iqn = $newiqns->{$node};
$iscsicfg{$controller}->{$node}->{iname} = $iqn;
}
foreach (@groupoutput) {
if (/^ ([^ ]+)/) { #This is a new group
$tgr = $1;
} elsif (/^ $iqn/) {
$groupname = $tgr;
}
}
unless ($groupname) {
next;
}
foreach (@mapoutput) {
unless (/iSCSI/) { next; }
my $backing;
my $igr;
my $lunid;
my $method;
($backing,$igr,$lunid,$method) = split /\s+/,$_,4;
if ($igr eq $groupname) {
$returns{$node}->{$backing}=$lunid;
}
}
}
foreach $node (keys %$newiqns) { #setNodesAttribs won't work since the values are unique per node
$iscsitab->setNodeAttribs($node,{iname=>$newiqns->{$node}});
}
return \%returns;
}
sub handle_targets {
my $controller = shift;
my $request = shift;
my $node;
my $target = get_controller_iqn($controller);
my @nodes = keys %{$iscsicfg{$controller}};
my @upnodes;
foreach $node (@nodes) { #though traversing this is marginally more expensive than just setting,
#do this to allow group level definitions to look sane when manually done
if ($iscsicfg{$controller}->{$node}->{target} ne $target) {
push @upnodes,$node;
}
}
$iscsitab->setNodesAttribs(\@upnodes,{target=>$target});
my $lunmap = build_lunmap($controller,keys %{$iscsicfg{$controller}});
foreach $node (@nodes) {
configure_node($controller,$node,$lunmap,$request);
}
}
sub getUnits {
my $amount = shift;
my $defunit = shift;
my $divisor=shift;
unless ($divisor) {
$divisor = 1;
}
if ($amount =~ /(\D)$/) { #If unitless, add unit
$defunit=$1;
chop $amount;
}
if ($defunit =~ /k/i) {
return $amount*1024/$divisor;
} elsif ($defunit =~ /m/i) {
return $amount*1048576/$divisor;
} elsif ($defunit =~ /g/i) {
return $amount*1073741824/$divisor;
}
}
sub create_new_lun {
print Dumper(@_);
my $controller = shift;
my $gname = shift;
my $cfg = shift;
my %args = @_;
my $lunsize;
my $mspec;
if ($args{mspec}) {
$mspec = $args{mspec}
} elsif ($args{lsize}) {
$lunsize = $args{lsize};
}
my %osmap = (
'rh.*' => 'linux',
'centos.*' => 'linux',
'sles.*' => 'linux',
'win2k8' => 'windows_2008',
'win2k3' => 'windows',
imagex => 'windows'
);
unless ($nodetypeinfo->{$gname}->[0]->{os}) {
sendmsg([1,"nodetype.os must be set for ONTAP plugin to create a lun"]);
}
my $ltype;
my $ost=$nodetypeinfo->{$gname}->[0]->{os};
foreach (keys %osmap) {
if ($ost =~ /$_/) {
$ltype = $osmap{$_};
last;
}
}
my $gtype = $ltype;
$gtype =~ s/_2008//; #The group types don't include a 2k8 specific type
my $file = $cfg->{file};
my $iname = $cfg->{iname};
my $output;
unless (($lunsize or $mspec) and $ltype and $file and $gtype) { #TODO etc
sendmsg([1,"Insufficient data"]);
}
if ($lunsize) {
my $size = getUnits($lunsize,'g',1048576);
$size .= "m";
$output = `ssh $controller lun create -s $size -t $ltype $file`;
} elsif ($mspec) {
my $mlun;
my $msnap;
($mlun,$msnap) = split /@/,$mspec,2;
my $output = `ssh $controller lun clone create $file -b $mlun $msnap`;
}
$output = `ssh $controller igroup create -i -t $gtype $gname`;
$output = `ssh $controller igroup add $gname $iname`;
$output = `ssh $controller lun map $file $gname`;
}
sub configure_node {
my $controller = shift;
my $node = shift;
my $cfg = $iscsicfg{$controller}->{$node};
my $lunmap = shift;
my $request = shift;
my $lunsize;
my $masterspec;
if ($request->{arg}) {
use Getopt::Long;
@ARGV=@{$request->{arg}};
GetOptions(
"size|s=i" => \$lunsize,
"master|m=s" => \$masterspec,
);
}
unless (defined $lunmap->{$node}->{$cfg->{file}}) {
if ($lunsize) {
create_new_lun($controller,$node,$cfg,lsize=>$lunsize);
} elsif ($masterspec) {
create_new_lun($controller,$node,$cfg,mspec=>$masterspec);
} else {
die "IMPLEMENT MAKING NEW LUN";
}
}
if ($cfg->{lun} ne $lunmap->{$node}->{$cfg->{file}}) {
$iscsitab->setNodeAttribs($node,{lun=>$lunmap->{$node}->{$cfg->{file}}});
}
}
1;