mirror of
				https://github.com/xcat2/xcat-core.git
				synced 2025-11-04 13:22:36 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			296 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			296 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
# 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;
 | 
						|
require xCAT::Utils;
 | 
						|
use xCAT::TableUtils;
 | 
						|
use xCAT::SvrUtils;
 | 
						|
my $output_handler;
 | 
						|
my $newiqns;
 | 
						|
my $domain;
 | 
						|
my %nodedomains;
 | 
						|
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 process_request {
 | 
						|
    my $request = shift;
 | 
						|
    $output_handler = shift;
 | 
						|
    $iscsitab       = xCAT::Table->new('iscsi');
 | 
						|
    unless ($iscsitab) {
 | 
						|
        xCAT::SvrUtils::sendmsg([ 1, "iSCSI configuration lacking from the iscsi table" ], $output_handler);
 | 
						|
        return;
 | 
						|
    }
 | 
						|
    my @nodes = @{ $request->{node} };
 | 
						|
 | 
						|
    my $nd = xCAT::NetworkUtils->getNodeDomains(\@nodes);
 | 
						|
    %nodedomains = %{$nd};
 | 
						|
 | 
						|
    my $nodetype = xCAT::Table->new('nodetype', -create => 0);
 | 
						|
    unless ($nodetype) {
 | 
						|
        xCAT::SvrUtils::sendmsg([ 1, "ONTAP plugin requires nodetype table to be populated" ], $output_handler);
 | 
						|
        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};
 | 
						|
 | 
						|
        $domain = $nodedomains{$node};
 | 
						|
        if ($domain) {
 | 
						|
            $domain = join(".", reverse(split(/\./, $domain)));
 | 
						|
        } else {
 | 
						|
            xCAT::SvrUtils::sendmsg([ 1, "Cannot determine domain name for iqn generation" ], $output_handler);
 | 
						|
            next;    # ????
 | 
						|
        }
 | 
						|
 | 
						|
        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}) {
 | 
						|
        xCAT::SvrUtils::sendmsg([ 1, "nodetype.os must be set for ONTAP plugin to create a lun" ], $output_handler);
 | 
						|
    }
 | 
						|
    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
 | 
						|
        xCAT::SvrUtils::sendmsg([ 1, "Insufficient data" ], $output_handler);
 | 
						|
    }
 | 
						|
    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;
 | 
						|
 | 
						|
 |