niceload: Fixed --battery.

niceload: Implemented --net/--nethops.
This commit is contained in:
Ole Tange 2016-05-08 14:18:10 +02:00
parent f8d1474943
commit 9d9016b88f
5 changed files with 204 additions and 17 deletions

View file

@ -85,12 +85,16 @@ if(@opt::prg) {
} elsif(@substr_name_pids) { } elsif(@substr_name_pids) {
warning("@opt::prg no exact matches. Using substrings."); warning("@opt::prg no exact matches. Using substrings.");
my %name_pids; my %name_pids;
for(@substr_name_pids) { for(sort @substr_name_pids) {
# If the process has run for long, then time column will
# enter the name, so remove leading digits
$name_of->{$_} =~ s/^\d+ //;
# Remove arguments
$name_of->{$_} =~ s/ .*//; $name_of->{$_} =~ s/ .*//;
push @{$name_pids{$name_of->{$_}}},$_; push @{$name_pids{$name_of->{$_}}},$_;
} }
warning("Niceloading", warning("Niceloading",
map { "$_ (".(join",",@{$name_pids{$_}}).")" } keys %name_pids map { "$_ (".(join" ",sort @{$name_pids{$_}}).")" } keys %name_pids
); );
@pids = @substr_name_pids; @pids = @substr_name_pids;
} else { } else {
@ -292,8 +296,9 @@ sub get_options_from_array {
"rn|runnoswap|run-noswap|run-no-swap" => \$opt::run_noswap, "rn|runnoswap|run-noswap|run-no-swap" => \$opt::run_noswap,
"noswap|N" => \$opt::noswap, "noswap|N" => \$opt::noswap,
# niceload -l -1 --sensor 'cat /sys/class/power_supply/BAT0/status /proc/acpi/battery/BAT0/state 2>/dev/null |grep -i -q discharging; echo $?'
"battery|B" => \$opt::battery, "battery|B" => \$opt::battery,
"net" => \$opt::net,
"nethops=i" => \$opt::nethops,
"nice|n=i" => \$opt::nice, "nice|n=i" => \$opt::nice,
"program|prg=s" => \@opt::prg, "program|prg=s" => \@opt::prg,
@ -305,6 +310,24 @@ sub get_options_from_array {
"verbose|v" => \$opt::verbose, "verbose|v" => \$opt::verbose,
"version|V" => \$opt::version, "version|V" => \$opt::version,
); );
if($opt::battery) {
# niceload -l -1 --sensor \
# 'cat /sys/class/power_supply/BAT0/status \
# /proc/acpi/battery/BAT0/state 2>/dev/null |
# grep -i -q discharging; echo $?'
$opt::sensor = ('cat /sys/class/power_supply/BAT0/status '.
'/proc/acpi/battery/BAT0/state 2>/dev/null | '.
'grep -i -q discharging; echo $?');
$opt::load = -1;
}
if($opt::net) {
$opt::nethops ||= 2;
}
if($opt::nethops) {
# niceload -l 0.01 --sensor 'netsensor_script'
$opt::sensor = netsensor_script($opt::nethops);
$opt::load ||= 0.01;
}
if(not $this_is_ARGV) { if(not $this_is_ARGV) {
@{$array_ref} = @::ARGV; @{$array_ref} = @::ARGV;
@::ARGV = @save_argv; @::ARGV = @save_argv;
@ -312,6 +335,28 @@ sub get_options_from_array {
return @retval; return @retval;
} }
sub shell_quote_scalar {
# Quote for other shells
my $a = $_[0];
if(defined $a) {
# zsh wants '=' quoted
# Solaris sh wants ^ quoted.
# $a =~ s/([\002-\011\013-\032\\\#\?\`\(\)\{\}\[\]\^\*\>\<\~\|\; \"\!\$\&\'\202-\377])/\\$1/g;
# This is 1% faster than the above
if(($a =~ s/[\002-\011\013-\032\\\#\?\`\(\)\{\}\[\]\^\*\<\=\>\~\|\; \"\!\$\&\'\202-\377]/\\$&/go)
+
# quote newline as '\n'
($a =~ s/[\n]/'\n'/go)) {
# A string was replaced
# No need to test for "" or \0
} elsif($a eq "") {
$a = "''";
} elsif($a eq "\0") {
$a = "";
}
}
return $a;
}
sub die_usage { sub die_usage {
help(); help();
@ -343,15 +388,38 @@ sub die_bug {
exit(255); exit(255);
} }
sub now {
# Returns time since epoch as in seconds with 3 decimals
# Uses:
# @Global::use
# Returns:
# $time = time now with millisecond accuracy
if(not $Global::use{"Time::HiRes"}) {
if(eval "use Time::HiRes qw ( time );") {
eval "sub TimeHiRestime { return Time::HiRes::time };";
} else {
eval "sub TimeHiRestime { return time() };";
}
$Global::use{"Time::HiRes"} = 1;
}
return (int(TimeHiRestime()*1000))/1000;
}
sub usleep { sub usleep {
# Sleep this many milliseconds. # Sleep this many milliseconds.
my $ms = shift; my $ms = shift;
::debug("Sleeping ",$ms," millisecs\n"); ::debug("Sleeping ",$ms," millisecs\n");
select(undef, undef, undef, $ms/1000); my $start = now();
my $now;
do {
# Something makes 'select' wake up too early
# when using --sensor
select(undef, undef, undef, $ms/1000);
$now = now();
} while($now < $start + $ms/1000);
} }
sub debug { sub debug {
if($opt::debug) { if($opt::debug) {
print STDERR @_; print STDERR @_;
@ -424,6 +492,97 @@ sub min {
return $min; return $min;
} }
sub netsensor_script {
# Script for --sensor when using --net
my $hops = shift;
my $perlscript = q{
use Net::Traceroute;
use Net::Ping;
my $medtrc = MedianTraceroute->new(shift);
$medtrc->ping();
$medtrc->ping();
while(1) {
my $ms = $medtrc->ping();
my $m = $medtrc->remedian();
# printf("%.2f %.2f med*1.5 %f\n",$m*1000,$ms*1000,$m*1.5 < $ms);
# 1 = median*1.5 < current latency
# 0 = median*1.5 > current latency
printf("%d\n",$m*1.5 < $ms);
sleep(1);
}
package MedianTraceroute;
sub new {
my $class = shift;
my $hop = shift;
# Find router
my $tr = Net::Traceroute->new(host => "8.8.8.8",
max_ttl => $hop);
if($tr->found) {
$host = $tr->hop_query_host($hop, 0);
} else {
# ns1.censurfridns.dk
$tr = Net::Traceroute->new(host => "89.233.43.71",
max_ttl => $hop);
if($tr->found) {
$host = $tr->hop_query_host($hop, 0);
} else {
die("Cannot traceroute to 8.8.8.8 and 89.233.43.71");
}
}
my $p = Net::Ping->new("external");
$p->hires();
return bless {
'hop' => $hop,
'host' => $host,
'pinger' => $p,
'remedian_idx' => 0,
'remedian_arr' => [],
'remedian' => undef,
}, ref($class) || $class;
}
sub ping {
my $self = shift;
for(1..3) {
# Ping should never take longer than 5.5 sec
my ($ret, $duration, $ip) =
$self->{'pinger'}->ping($self->{'host'}, 5.5);
if($ret) {
$self->set_remedian($duration);
return $duration;
}
}
warn("Ping failed 3 times.");
}
sub remedian {
my $self = shift;
return $self->{'remedian'};
}
sub set_remedian {
# Set median of the last 999^3 (=997002999) values using Remedian
#
# Rousseeuw, Peter J., and Gilbert W. Bassett Jr. "The remedian: A
# robust averaging method for large data sets." Journal of the
# American Statistical Association 85.409 (1990): 97-104.
my $self = shift;
my $val = shift;
my $i = $self->{'remedian_idx'}++;
my $rref = $self->{'remedian_arr'};
$rref->[0][$i%999] = $val;
$rref->[1][$i/999%999] = (sort @{$rref->[0]})[$#{$rref->[0]}/2];
$rref->[2][$i/999/999%999] = (sort @{$rref->[1]})[$#{$rref->[1]}/2];
$self->{'remedian'} = (sort @{$rref->[2]})[$#{$rref->[2]}/2];
}
};
return "perl -e ".shell_quote_scalar($perlscript)." $hops";
}
package Process; package Process;
@ -768,11 +927,13 @@ sub nonblockGetLines {
sub read_sensor { sub read_sensor {
my $self = shift; my $self = shift;
::debug("read_sensor"); ::debug("read_sensor: ");
my $fh = $self->{'sensor_fh'}; my $fh = $self->{'sensor_fh'};
if(not $fh) { if(not $fh) {
# Start the sensor # Start the sensor
open($fh, "-|", $opt::sensor) || ::die_bug("Cannot open: $opt::sensor"); $self->{'sensor_pid'} =
open($fh, "-|", $opt::sensor) ||
::die_bug("Cannot open: $opt::sensor");
$self->{'sensor_fh'} = $fh; $self->{'sensor_fh'} = $fh;
} }
# Read as much as we can (non_block) # Read as much as we can (non_block)
@ -781,14 +942,17 @@ sub read_sensor {
# new load = last full line # new load = last full line
foreach my $line (@lines) { foreach my $line (@lines) {
if(defined $line) { if(defined $line) {
::debug("Pipe saw: $eof [$line]\n"); ::debug("Pipe saw: [$line] eof=$eof\n");
$Global::last_sensor_reading = $line; $Global::last_sensor_reading = $line;
} }
} }
if($eof) { if($eof) {
# End of file => Restart the sensor # End of file => Restart the sensor
close $fh; close $fh;
open($fh, "-|", $opt::sensor) || ::die_bug("Cannot open: $opt::sensor"); # waitpid($self->{'sensor_pid'},0);
$self->{'sensor_pid'} =
open($fh, "-|", $opt::sensor) ||
::die_bug("Cannot open: $opt::sensor");
$self->{'sensor_fh'} = $fh; $self->{'sensor_fh'} = $fh;
} }
@ -991,4 +1155,4 @@ sub io_status_darwin {
return ::min($io, 10); return ::min($io, 10);
} }
$::exitsignal = $::exitstatus = $opt::battery = 0; # Dummy $::exitsignal = $::exitstatus = 0; # Dummy

View file

@ -32,11 +32,11 @@ run 1 second, suspend (3.00-1.00) seconds, run 1 second, suspend
=over 9 =over 9
=item B<-B> =item B<-B> (alpha testing)
=item B<--battery> =item B<--battery> (alpha testing)
Suspend if the system is running on battery. Short hand for: -l -1 --sensor 'cat /sys/class/power_supply/BAT0/status /proc/acpi/battery/BAT0/state 2>/dev/null |grep -i -q discharging; echo $?' Suspend if the system is running on battery. Shorthand for: -l -1 --sensor 'cat /sys/class/power_supply/BAT0/status /proc/acpi/battery/BAT0/state 2>/dev/null |grep -i -q discharging; echo $?'
=item B<-f> I<FACTOR> =item B<-f> I<FACTOR>
@ -102,6 +102,29 @@ B<--noswap> is over limit if the system is swapping both in and out.
B<--noswap> will set both B<--start-noswap> and B<run-noswap>. B<--noswap> will set both B<--start-noswap> and B<run-noswap>.
=item B<--net> (alpha testing)
Shorthand for B<--nethops 2>.
=item B<--nethops> I<h> (alpha testing)
Network nice. Pause if the internet connection is overloaded.
B<niceload> finds a router I<h> hops closer to the internet. It
B<ping>s this every second. If the latency is more than 50% bigger
than the median, it is regarded as being over the limit.
B<--nethops> can be combined with B<--hard>. Without B<--hard> the
program may be able to queue up so much traffic that it will take
longer than the B<--suspend> time to clear it. B<--hard> is useful for
traffic that does not break by being suspended for a longer time.
B<--nethops> can be combined with a high B<--suspend>. This way a
program can be allowed to do a bit of traffic now and then. This is
useful to keep the connection alive.
=item B<-n> I<niceness> =item B<-n> I<niceness>
=item B<--nice> I<niceness> =item B<--nice> I<niceness>

View file

@ -7566,7 +7566,7 @@ sub start {
open ERR, '>&', $stderr_fh or ::die_bug("Can't dup STDERR: $!"); open ERR, '>&', $stderr_fh or ::die_bug("Can't dup STDERR: $!");
$pid = ::open3("<&IN", ">&OUT", ">&ERR", $pid = ::open3("<&IN", ">&OUT", ">&ERR",
"exec $Global::shell -c ". "exec $Global::shell -c ".
::shell_quote_scalar_default($command)) || ::shell_quote_scalar($command)) ||
::die_bug("open3-/dev/tty"); ::die_bug("open3-/dev/tty");
$Global::tty_taken = $pid; $Global::tty_taken = $pid;
close $devtty_fh; close $devtty_fh;

View file

@ -2,12 +2,12 @@ testsuite: 3
true true
3: ../src/parallel tests-to-run/* wanted-results/* startdb prereqlocal prereqremote 3: ../src/parallel tests-to-run/* wanted-results/* startdb prereqlocal prereqremote
TRIES=3 time bash Start.sh - mem || true TRIES=3 time bash Start.sh '' mem || true
touch ~/.parallel/will-cite touch ~/.parallel/will-cite
make stopvm make stopvm
1: ../src/parallel tests-to-run/* wanted-results/* prereqlocal startdb prereqremote 1: ../src/parallel tests-to-run/* wanted-results/* prereqlocal startdb prereqremote
TRIES=1 time bash Start.sh - 'mem|polarhome' || true TRIES=1 time bash Start.sh '' 'mem|polarhome' || true
touch ~/.parallel/will-cite touch ~/.parallel/will-cite
make stopvm make stopvm

View file

@ -19,5 +19,5 @@ echo '### multiple -p'
/tmp/mysleep 2 & /tmp/mysleep 2 &
/tmp/mysleep 2 & /tmp/mysleep 2 &
echo '### --prg' echo '### --prg'
stdout /usr/bin/time -f %e niceload -l 8 -H --prg mysleep | perl -ne '$_ > 5 and print "--prg OK\n"' stdout /usr/bin/time -f %e niceload -l 8 -H --prg /tmp/mysleep | perl -ne '$_ > 5 and print "--prg OK\n"'
rm /tmp/mysleep rm /tmp/mysleep