# IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html
package xCAT_plugin::boottarget;

BEGIN
{
    $::XCATROOT = $ENV{'XCATROOT'} ? $ENV{'XCATROOT'} : '/opt/xcat';
}
use lib "$::XCATROOT/lib/perl";
use Storable qw(dclone);
use Sys::Syslog;
use Thread qw(yield);
use POSIX qw(WNOHANG nice);
use xCAT::Table;
use xCAT::Utils;
use xCAT::TableUtils;
use xCAT::MsgUtils;
use xCAT::Template;

#use xCAT::Postage;
use Data::Dumper;
use Getopt::Long;
Getopt::Long::Configure("bundling");
Getopt::Long::Configure("pass_through");
use File::Path;
use File::Copy;



sub handled_commands
{
    return {
        mknetboot => "nodetype:os=(boottarget)|(target)|(bt)",
        mkinstall => "nodetype:os=(boottarget)|(target)|(bt)"
    };
}

sub preprocess_request
{
    my $req      = shift;
    my $callback = shift;

    #if already preprocessed, go straight to request
    if ($req->{_xcatpreprocessed}->[0] == 1) { return [$req]; }

    if ($req->{command}->[0] eq 'copycd')
    {    #don't farm out copycd
        return [$req];
    }

    #my $stab = xCAT::Table->new('site');
    #my $sent;
    #($sent) = $stab->getAttribs({key => 'sharedtftp'}, 'value');
    my @entries    = xCAT::TableUtils->get_site_attribute("sharedtftp");
    my $site_entry = $entries[0];
    unless (defined($site_entry)
        and ($site_entry =~ /no/i or $site_entry =~ /0/))
    {

        #unless requesting no sharedtftp, don't make hierarchical call
        return [$req];
    }

    my %localnodehash;
    my %dispatchhash;
    my $nrtab = xCAT::Table->new('noderes');
    my $nrents = $nrtab->getNodesAttribs($req->{node}, [qw(tftpserver servicenode)]);
    foreach my $node (@{ $req->{node} })
    {
        my $nodeserver;
        my $tent = $nrents->{$node}->[0]; #$nrtab->getNodeAttribs($node, ['tftpserver']);
        if ($tent) { $nodeserver = $tent->{tftpserver} }
        unless ($tent and $tent->{tftpserver})
        {
            $tent = $nrents->{$node}->[0]; #$nrtab->getNodeAttribs($node, ['servicenode']);
            if ($tent) { $nodeserver = $tent->{servicenode} }
        }
        if ($nodeserver)
        {
            $dispatchhash{$nodeserver}->{$node} = 1;
        }
        else
        {
            $localnodehash{$node} = 1;
        }
    }
    my @requests;
    my $reqc = {%$req};
    $reqc->{node} = [ keys %localnodehash ];
    if (scalar(@{ $reqc->{node} })) { push @requests, $reqc }

    foreach my $dtarg (keys %dispatchhash)
    {    #iterate dispatch targets
        my $reqcopy = {%$req};    #deep copy
        $reqcopy->{'_xcatdest'} = $dtarg;
        $reqcopy->{_xcatpreprocessed}->[0] = 1;
        $reqcopy->{node} = [ keys %{ $dispatchhash{$dtarg} } ];
        push @requests, $reqcopy;
    }
    return \@requests;
}

sub process_request
{
    my $request  = shift;
    my $callback = shift;
    my $doreq    = shift;
    my $distname = undef;
    my $arch     = undef;
    my $path     = undef;
    return mknetboot($request, $callback, $doreq);
}

sub mknetboot
{
    my $req      = shift;
    my $callback = shift;
    my $doreq    = shift;
    my $nodes    = @{ $request->{node} };
    my @args     = @{ $req->{arg} };
    my @nodes    = @{ $req->{node} };
    my $ostab    = xCAT::Table->new('nodetype');

    my $installroot = xCAT::TableUtils->getInstallDir();
    my $tftpdir     = xCAT::TableUtils->getTftpDir();

    my %donetftp = ();
    my %oents    = %{ $ostab->getNodesAttribs(\@nodes, [qw(os arch profile)]) };
    my $restab   = xCAT::Table->new('noderes');
    my $bptab    = xCAT::Table->new('bootparams', -create => 1);
    my $hmtab    = xCAT::Table->new('nodehm');
    my $ttab     = xCAT::Table->new('boottarget');

    foreach $node (@nodes)
    {
        my $ent = $oents{$node}->[0]; #ostab->getNodeAttribs($node, ['os', 'arch', 'profile']);
        unless ($ent->{os} and $ent->{profile})
        {
            $callback->(
                {
                    error     => ["Insufficient nodetype entry for $node"],
                    errorcode => [1]
                }
            );
            next;
        }

        my $profile = $ent->{profile};
        ($tent) = $ttab->getAttribs({ 'bprofile' => $profile }, 'kernel', 'initrd', 'kcmdline'); #TODO: coalesce these queries into one Table query, speed it up
        if (!defined($tent)) {
            my $msg = "$profile in nodetype table was not defined in boottarget table";
            $callback->({
                    error     => ["$msg"],
                    errorcode => [1]
            });
        }
        $kernel   = $tent->{kernel};
        $initrd   = $tent->{initrd};
        $kcmdline = $tent->{kcmdline};

        #TODO: big todo, cheap and rapid, but should be more like esx.pm implementation, i.e. more scalable
        while ($kcmdline =~ /#NODEATTRIB:([^:#]+):([^:#]+)#/) {
            my $natab = xCAT::Table->new($1);
            my $naent = $natab->getNodeAttribs($node, [$2]);
            my $naval = $naent->{$2};
            $kcmdline =~ s/#NODEATTRIB:([^:#]+):([^:#]+)#/$naval/;
        }
        while ($kcmdline =~ /#TABLE:([^:#]+):([^:#]+):([^:#]+)#/) {
            my $tabname = $1;
            my $keyname = $2;
            my $colname = $3;
            if ($2 =~ /THISNODE/ or $2 =~ /\$NODE/) {
                my $natab = xCAT::Table->new($tabname);
                my $naent = $natab->getNodeAttribs($node, [$colname]);
                my $naval = $naent->{$colname};
                $kcmdline =~ s/#TABLE:([^:#]+):([^:#]+):([^:#]+)#/$naval/;
            } else {
                my $msg = "Table key of $2 not yet supported by boottarget mini-template";
                $callback->({
                        error     => ["$msg"],
                        errorcode => [1]
                });

            }
        }

        if ($initrd eq '') {
            $bptab->setNodeAttribs(
                $node,
                {
                    kernel   => $kernel,
                    initrd   => '',
                    kcmdline => $kcmdline
                }
            );
        } else {
            $bptab->setNodeAttribs(
                $node,
                {
                    kernel   => $kernel,
                    initrd   => $initrd,
                    kcmdline => $kcmdline
                }
            );
        }
    }

}

1;