/usr/bin/pegasus-submit-dag is in pegasus-wms 4.4.0+dfsg-7.
This file is owned by root:root, with mode 0o755.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 | #!/usr/bin/env perl
#
# wrapper around condor_submit_dag, activates basic throttles
#
# This file or a portion of this file is licensed under the terms of
# the Globus Toolkit Public License, found in file GTPL, or at
# http://www.globus.org/toolkit/download/license.html. This notice must
# appear in redistributions of this file, with or without modification.
#
# Redistributions of this Software, with or without modification, must
# reproduce the GTPL in: (1) the Software, or (2) the Documentation or
# some other similar material which is provided with the Software (if
# any).
#
# Copyright 1999-2004 University of Chicago and The University of
# Southern California. All rights reserved.
#
# Author: Jens-S. Vöckler voeckler@cs.uchicago.edu
# Author: Gaurang Mehta gmehta at isi dot edu
# Revision : $Revision$
#
use 5.006;
use strict;
use File::Spec;
use File::Copy;
use File::Basename qw(basename dirname);
use Getopt::Long qw(:config bundling no_ignore_case);
use Data::Dumper;
BEGIN {
my $pegasus_config = File::Spec->catfile( dirname($0), 'pegasus-config' );
eval `$pegasus_config --perl-dump`;
die("Unable to eval pegasus-config output. $@") if $@;
}
# load our own local modules, see PERL5LIB settings
use Pegasus::Properties; # parses -Dprop=val from @ARGV
use Pegasus::Common;
# some reasonable defaults
my $maxpre = 20;
my $maxpost = 20;
my $maxjobs = 0;
my $maxidle = 0;
my $dagman;
my $submit=1;
my $notify = 'NEVER';
my $verbose;
my $conffile;
my %props = ();
my $grid=0; #Grid checks enabled
$main::DEBUG = 0; # for now
$main::revision = 'unknown';
$_ = '$Revision$'; # don't edit, automatically updated by CVS
$main::revision=$1 if /Revision:\s+([0-9.]+)/o;
sub myversion() {
my $version = version();
print "Pegasus $version, @{[basename($0)]} $main::revision\n";
exit 0;
}
sub usage(;$) {
my $msg = shift;
print "ERROR: $msg\n" if defined $msg && lc($msg) ne 'help';
print << "EOF";
Usage: @{[basename($0)]} [-Dprops] [options] dagfile
-Dprop=val Commandline overwrite for properties, must be initial args!
-c|--conf fn Read properties from given filename instead of rundir.
-d|--debug lv Initializes the level lv of verbosity, default $main::DEBUG
-e|--dagman fn Specify an alternative dagman binary to use.
-P|--maxpre N Maximum number of pre-scripts to run, default $maxpre
-p|--maxpost N Maximum number of post-scripts to run, default $maxpost
-j|--maxjobs N Maximum number of jobs to submit to Condor, default $maxjobs
-i|--maxidle N Maximum number of idle jobs, default $maxidle
-n|--notify x When to notify: Never, Error, Complete; default $notify
-v|--verbose Enter DAGMan verbose mode, default is not
-V|--version Print version number and exit.
--grid | --nogrid Enable checks for grid proxy and GLOBUS LOCATION (Default is enabled)
A maximum number of 0 means unlimited.
EOF
exit(1);
}
sub proxy_duration {
# purpose: determine remaining time on grid user proxy
# returns: undef in case of error, or remaining time.
#
my $gpi = File::Spec->catfile( $ENV{'GLOBUS_LOCATION'}, 'bin',
'grid-proxy-info' );
die "ERROR: Unable to find GLOBUS_LOCATION, please check your setup\n"
unless exists $ENV{'GLOBUS_LOCATION'} && $ENV{'GLOBUS_LOCATION'};
die "ERROR: Unable to execute grid-proxy-info\n"
unless -x $gpi;
my $left = 0;
chomp($left=`$gpi -timeleft`);
$? == 0 ? $left + 0 : undef;
}
sub salvage_logfile($) {
# purpose: salvage Condor common log file from truncation
# paramtr: $dagfn (IN): Name of dag filename
# returns: -
#
my $dagfn = shift;
my $result = undef;
local(*DAG,*SUB,*LOG);
if ( open( DAG, "<$dagfn" ) ) {
# read to to figure out submit files
my @x;
my %submit = ();
while ( <DAG> ) {
next unless /^\s*job/i;
s/[\r\n]+$//; # safe chomp
@x = split;
$submit{$x[1]} = $x[2]; # dagjobid -> subfn
}
close DAG;
if ( $main::DEBUG > 2 ) {
print STDERR "# found the following associations:\n";
local $Data::Dumper::Indent = 1;
local $Data::Dumper::Pad = "# ";
print STDERR Data::Dumper->Dump( [\%submit], [qw(config)] );
}
# read two submit files to figure out condor common log file
foreach my $subfn ( values %submit ) {
if ( open( SUB, "<$subfn" ) ) {
my $logfile = undef;
while ( <SUB> ) {
next unless /^log(=|\s)/i;
s/[\r\n]+$//; # safe chomp
@x = split /\s*=\s*/, $_, 2;
$logfile = ( substr( $x[1], 0, 1 ) =~ /[''""]/ ?
substr( $x[1], 1, -1 ) : $x[1] );
last;
}
close SUB;
print STDERR "# $subfn points to $logfile\n"
if ( $main::DEBUG > 1 );
if ( ! defined $result ) {
$result = $logfile;
} else {
last if $result eq $logfile;
warn "# Using distinct, different log files, skipping preservation.\n";
return undef;
}
} else {
warn "Unable to read sub file $subfn: $!\n";
}
}
# try to preserve log file
if ( defined $result && -s $result ) {
my $newfn;
print STDERR "# log $result exists, rescuing from DAGMan.\n"
if $main::DEBUG;
for ( my $i=0; $i<1000; ++$i ) {
$newfn = sprintf "%s.%03d", $result, $i;
if ( open( LOG, "<$newfn" ) ) {
# file exists
close LOG;
} else {
# file does not exist, use that
my $newresult=$result;
#check if the file is a smylink then dereference it.
if ( -l $result ) {
$newresult=readlink($result);
}
print STDOUT "Rescued $result as $newfn\n"
if copy( $newresult, $newfn ) or warn "Could not rescue the log file $newresult to $newfn\n $! \nTrying to continue\n";
last;
}
}
} else {
print STDERR "# log $result does not yet exist (good)\n"
if ( $main::DEBUG );
}
} else {
die "ERROR: Unable to read dag file $dagfn: $!\n";
}
$result;
}
GetOptions( "debug|d=i" => \$main::DEBUG,
"maxpre|P=i" => \$maxpre,
"maxpost|p=i" => \$maxpost,
'dagman|e=s' => \$dagman,
"submit!" => \$submit,
'conf|c=s' => \$conffile,
"maxjob|maxjobs|j=i" => \$maxjobs,
"maxidle|i=i" => \$maxidle,
"notify|n=s" => \$notify,
"version|V" => \&myversion,
"verbose|v" => \$verbose,
"grid!"=>\$grid,
"help|h|?" => \&usage );
#check grid stuff only if $grid enabled
if($grid){
# check Globus proxy lifetime?
my $left = proxy_duration() ||
die "ERROR: Problems with grid-proxy-info. Check your user proxy\n";
if ( $left <= 0 ) {
die "ERROR: Your grid user proxy has expired, please refresh now.\n";
} elsif ( $left < 7200 ) {
warn "Warning: There is little time remaining on your proxy. You need to refresh soon!\n";
}
}
my $dag = shift || usage("Need the name of a .dag file\n");
my $c_s_d = find_exec( 'condor_submit_dag' ) ||
die "Unable to locate condor_submit_dag\n";
my $c_s = find_exec('condor_submit') || die "Unable to locate condor_submit\n";
salvage_logfile($dag);
my $run=dirname($dag);
my %config = slurp_braindb( $run ) or die "ERROR: open braindb: $!\n";
# pre-condition: The planner writes all properties per workflow into the DAG dir.
my $props = Pegasus::Properties->new( $conffile, File::Spec->catfile($run,$config{properties} ));
# set true defaults from properties
$maxpre = $props->property('dagman.maxpre') || $maxpre;
$maxpost = $props->property('dagman.maxpost') || $maxpost;
$maxjobs = $props->property('dagman.maxjobs') || $maxjobs;
$maxidle = $props->property('dagman.maxidle') || $maxidle;
$notify = $props->property('dagman.notify') || $notify;
$verbose = 1 if lc($props->property('dagman.verbose')) =~ /(true|on|1)/ || $verbose;
# construct commandline
my @arg = ( $c_s_d );
#push( @arg, '-dagman', $dagman ) if $dagman;
push( @arg, '-MaxPre', $maxpre ) if $maxpre > 0;
push( @arg, '-MaxPost', $maxpost ) if $maxpost > 0;
push( @arg, '-maxjobs', $maxjobs ) if $maxjobs > 0;
push( @arg, '-maxidle', $maxidle ) if $maxidle > 0;
push( @arg, '-notification', $notify );
push( @arg, '-verbose' ) if $verbose;
push( @arg, '-append', 'executable='.$dagman ) if $dagman;
push( @arg, '-append', '+pegasus_wf_uuid="'.$config{'wf_uuid'}.'"' );
push( @arg, '-append', '+pegasus_root_wf_uuid="'.$config{'root_wf_uuid'}.'"' );
push( @arg, '-append', '+pegasus_wf_name="'.$config{'pegasus_wf_name'}.'"' );
push( @arg, '-append', '+pegasus_wf_time="'.$config{timestamp}.'"' );
push( @arg, '-append', '+pegasus_version="'.$config{'planner_version'}.'"' );
push( @arg, '-append', '+pegasus_job_class=11' );
push( @arg, '-append', '+pegasus_cluster_size=1' );
push( @arg, '-append', '+pegasus_site="local"' );
push( @arg, '-append', '+pegasus_wf_xformation="pegasus::dagman"' );
#push( @arg, '-no_submit') if $submit==0;
push( @arg, $dag );
print STDERR "# @arg\n" if $main::DEBUG;
#my $csdresult=`@arg`;
exec { $arg[0] } @arg or die "Cannot execute @arg: $! \n";
exit 127;
|