Sponsoring The Perl Toolchain Summit 2025: Help make this important event another success Learn more

use Dancer qw/:syntax :script/;
use IPC::Run 'harness';
use MIME::Base64 'decode_base64';
use File::ShareDir 'dist_dir';
use File::Slurper qw/read_text write_text/;
use File::Temp ();
use JSON::PP ();
use YAML::XS ();
=head1 NAME
App::Netdisco::Transport::Python
=head1 DESCRIPTION
Not really a transport, but has similar behaviour to a Transport.
Returns an object which has a live Python subprocess expecting
instruction to run worklets.
my $runsub = App::Netdisco::Transport::Python->py_worklet();
=cut
__PACKAGE__->attributes(qw/ runner stdin stdout context /);
sub init {
my ( $class, $instance ) = @_;
my ( $stdin, $stdout );
$instance->stdin( \$stdin );
$instance->stdout( \$stdout );
$instance->context( File::Temp->new() );
my $cmd = [ py_cmd('run_worklet'), $instance->context->filename ];
debug "\N{SNAKE} starting persistent Python worklet subprocess";
$instance->runner( harness(
($ENV{ND2_PYTHON_HARNESS_DEBUG} ? (debug => 1) : ()),
$cmd,
'<', \$stdin,
'1>', \$stdout,
'2>', sub { debug $_[0] },
) );
debug $instance->context if $ENV{ND2_PYTHON_HARNESS_DEBUG};
return $instance;
}
=head1 py_worklet( )
Returns an object instance with a live Python subprocess.
Will die if Python cannot be run.
=cut
sub py_worklet {
my ($self, $job, $workerconf) = @_;
my $action = $workerconf->{action};
my $coder = JSON::PP->new->utf8(1)
->allow_nonref(1)
->allow_unknown(1)
->allow_blessed(1)
->allow_bignum(1);
# this is only really used the first time (pump calls start)
$ENV{'ND2_JOB_METADATA'} = $coder->encode( { %$job, device => (($job->device || '') .'') } );
$ENV{'ND2_CONFIGURATION'} = $coder->encode( config() );
$ENV{'ND2_FSM_TEMPLATES'} = Path::Class::Dir->new( dist_dir('App-Netdisco') )
->subdir('python')->subdir('tfsm')->stringify;
my $inref = $self->stdin;
my $outref = $self->stdout;
# copy latest vars to the worklet
write_text($self->context->filename, $coder->encode( { vars => vars() } ));
# necessary before running, but do first (instead of after) to aid debugging
$$outref = '';
$$inref = $workerconf->{pyworklet} ."\n";
$self->runner->pump until ($$outref and $$outref =~ /^\.\Z/m);
my $context = read_text($self->context->filename);
truncate($self->context, 0); # do not leave things lying around on disk
my $retdata = try { YAML::XS::Load(decode_base64($context)) }; # might explode
$retdata = {} if not ref $retdata or 'HASH' ne ref $retdata;
# use DDP;
# p $$outref;
# p $retdata;
my $status = $retdata->{status} || '';
my $log = $retdata->{log}
|| ($status eq 'done' ? (sprintf '%s exit OK', $action)
: (sprintf '%s exit with status "%s"', $action, $status));
# TODO support merging more deeply
var($_ => $retdata->{stash}->{$_}) for keys %{ $retdata->{stash} || {} };
return Status->$status($log);
}
true;