1 # Girocco::CLIUtil.pm -- Command Line Interface Utility Functions
2 # Copyright (C) 2016 Kyle J. McKay. All rights reserved.
4 # This program is free software; you can redistribute it and/or
5 # modify it under the terms of the GNU General Public License
6 # as published by the Free Software Foundation; either version 2
7 # of the License, or (at your option) any later version.
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
14 # You should have received a copy of the GNU General Public License
15 # along with this program; if not, write to the Free Software
16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
21 ## This package MUST NOT be used by any CGI script as it cancels
22 ## the effect of CGI::Carp::fatalsToBrowser which could result in the
23 ## output of a CGI script becoming unparseable by the web server!
26 package Girocco
::CLIUtil
;
31 use base
qw(Exporter);
32 our ($VERSION, @EXPORT, @EXPORT_OK);
36 diename recreate_file strict_bool
37 is_yes is_no is_yesno valid_bool clean_bool
39 ynprompt ynprompt_or_die
40 prompt_noecho prompt_noecho_or_die
41 prompt_noecho_nl prompt_noecho_nl_or_die
42 yes_to_continue yes_to_continue_or_die
43 get_all_users get_user get_all_projects get_project
44 get_full_users nice_me setup_pager setup_pager_stdout
48 _parse_options _prompt_rl _prompt_rl_or_die
49 check_passwd_match _which
56 use POSIX qw(:fcntl_h);
59 use Girocco
::HashUtil
;
61 BEGIN {noFatalsToBrowser
}
65 require Term
::ReadLine
;
70 package Girocco
::CLIUtil
::NoEcho
;
73 my $class = shift; # ignored
77 $self->{ios
} = POSIX
::Termios
->new;
78 $self->{ios
}->getattr($self->{fd
});
79 my $noecho = POSIX
::Termios
->new;
80 $noecho->getattr($fd);
81 $noecho->setlflag($noecho->getlflag & ~(&POSIX
::ECHO
));
82 $noecho->setattr($fd, &POSIX
::TCSANOW
);
88 $self->{ios
}->setattr($self->{fd
}, &POSIX
::TCSANOW
);
93 package Girocco
::CLIUtil
::Progress
;
95 use Scalar
::Util
qw(looks_like_number);
96 use Time
::HiRes
qw(gettimeofday);
98 sub fractime
() { return scalar(gettimeofday
) }
101 BEGIN { $_init = sub {
104 looks_like_number
($max) && $max >= 0 or $max = 100;
106 defined($title) or $title = "";
108 $title ne "" or $title = "Progress";
109 my $lastupd = $self->{lastupd
};
110 my $shown1 = $self->{shown1
};
111 looks_like_number
($lastupd) or $lastupd = fractime
+ 2;
119 defined($shown1) and $self->{shown1
} = $shown1;
124 my $class = shift || __PACKAGE__
;
125 my $self = bless {}, $class;
127 select((select(STDERR
),$|=1)[0]);
128 select((select(STDOUT
),$|=1)[0]);
135 my $wasvis = $self->{wasvis
};
137 $wasvis and $self->show;
141 sub val
{ $_[0]->{cur
} }
142 sub done
{ $_[0]->{cur
} >= $_[0]->{max
} }
146 !$self->{max
} and return;
147 my $last = $self->{cur
};
149 looks_like_number
($newcur) or $newcur = $last + 1;
150 $newcur >= $last or $newcur = $last;
151 $newcur > $self->{max
} and $newcur = $self->{max
};
152 $self->{cur
} = $newcur unless $newcur == $last;
154 if ($self->{shown1
} && $newcur > $last && $newcur >= $self->{max
} ||
155 $now >= $self->{lastupd
} + 1) {
156 $self->{lastupd
} = $now;
157 !$self->{len
} || $newcur != $last and $self->show;
163 delete $self->{wasvis
};
164 !$self->{max
} and return;
165 my $p = int((100 * $self->{cur
} / $self->{max
}) + 0.5);
166 $p > 100 and $p = 100;
167 $p == 100 && $self->{cur
} < $self->{max
} and $p = 99;
168 my $status = sprintf("%s: %3d%% (%d/%d)", $self->{title
},
169 $p, $self->{cur
}, $self->{max
});
170 my $newlen = length($status);
171 $self->{len
} > $newlen and $status .= " " x
($self->{len
} - $newlen);
172 printf STDERR
"%s\r", $status;
173 $self->{len
} = $newlen;
180 printf STDERR
"%s\r", " " x
$self->{len
};
188 $self->{wasvis
} and $self->show;
194 my $msg = join(' ', @_);
195 defined($msg) or return;
197 $msg ne '' or return;
199 printf $fh "%s\n", $msg;
205 $self->emitfh(\
*STDOUT
, @_);
210 $self->emitfh(\
*STDERR
, @_);
218 }# END package Girocco::CLIUtil::Progress
221 BEGIN {$diename = ""}
223 my $result = $diename;
224 $diename = join(" ", @_) if @_;
230 # Remove any leading options matching the given specs from the @ARGV array and
231 # store them as indicated. Parsing stops when an unknown option is encountered,
232 # "--" is encountered (in which case it's removed) or a non-option is encountered.
233 # Note that "-" by itself is considered a non-option argument.
235 # Option bundling for single-letter options is NOT supported.
237 # Optional first arg is CODE ref:
238 # sub {my ($err, $opt) = @_; ...}
239 # with $err of '?' meaning $opt is unknown
240 # with $err of ':' meaning $opt is missing its argument
241 # $opt is the full option as given on the command line (including leading - etc.)
242 # the default if omitted dies with an error
243 # If the sub returns, _parse_options exits immediately with 0
245 # The rest of the arguments form pairs:
247 # where ref must be either a SCALAR ref or a CODE ref, if it's neither
248 # then the "spec" => ref pair is silently ignored.
250 # "name" -- an incrementing flag (matches -name and --name)
251 # ":name" -- an option with a value (matches -name=val and --name=val)
252 # Using option "--name" matches spec "name" if given otherwise matches spec
253 # ":name" if given and there's at least one more argument (if not the ':' error
255 # Using option "--name=val" only matches spec ":name" (but "val" can be "").
256 # For flags, a SCALAR ref is incremented, a CODE ref is called with no arguments.
257 # For values (":name" specs) a SCALAR ref is assigned the value a CODE ref is
258 # called with the value as its single argument.
260 # _parse_options returns 1 as long as there were no errors
263 my $failsub = sub {die((($_[0]eq'?')?
"unrecognized":"missing argument for")." option \"$_[1]\"\n")};
264 $failsub = shift if @_ && ref($_[0]) eq "CODE";
267 if (defined($_[0]) && $_[0] =~ /^:?[^-:\s]/ &&
268 defined($_[1]) && (ref($_[1]) eq "SCALAR" || ref($_[1]) eq "CODE")) {
269 $opts{$_[0]} = $_[1];
274 while (@ARGV && $ARGV[0] =~ /^--?[^-:\s]/) {
275 my $opt = shift @ARGV;
278 if ($sopt =~ /^([^=]+)=(.*)$/) {
279 my ($name, $val) = ($1, $2);
280 if ($opts{":$name"}) {
281 ${$opts{":$name"}} = $val if ref($opts{":$name"}) eq "SCALAR";
282 &{$opts{":$name"}}($val) if ref($opts{":$name"}) eq "CODE";
284 &$failsub('?', $opt);
287 } elsif ($opts{$sopt}) {
288 ++${$opts{$sopt}} if ref($opts{$sopt}) eq "SCALAR";
289 &{$opts{$sopt}}() if ref($opts{$sopt}) eq "CODE";
290 } elsif ($opts{":$sopt"}) {
291 &$failsub(':', $opt),return(0) unless @ARGV;
292 my $val = shift @ARGV;
293 ${$opts{":$sopt"}} = $val if ref($opts{":$sopt"} eq "SCALAR");
294 &{$opts{":$sopt"}}($val) if ref($opts{":$sopt"} eq "CODE");
296 &$failsub('?', $opt);
300 if (@ARGV && $ARGV[0] eq "--") {
304 if (@ARGV && $ARGV[0] =~ /^-./) {
305 &$failsub('?', $ARGV[0]);
312 open F
, '>', $_[0] or die "failed to create $_[0]: $!\n";
319 defined ($b) or $b = "";
320 return lc($b) eq "yes" || (!$strict && lc($b) eq "y");
326 defined ($b) or $b = "";
327 return lc($b) eq "no" || (!$strict && lc($b) eq "n");
331 return is_yes
(@_) || is_no
(@_);
352 exists($boolvals{lc($_[0])});
357 return $boolvals{lc($b)} || 0;
361 my ($norl, $prompt, $default, $promptsfx) = @_;
362 ! -t STDIN
and $norl = 1;
363 defined($promptsfx) or $promptsfx = ': ';
364 defined($prompt) or $prompt = '';
366 $ds = " [" . $default . "]" if defined($default);
367 if ($have_rl && !$norl) {
368 my $rl = Term
::ReadLine
->new(basename
($0), \
*STDIN
, \
*STDOUT
);
370 $_ = $rl->readline($prompt . $ds . $promptsfx);
371 $rl->addhistory($_) if defined($_) && $_ =~ /\S/;
373 print $prompt, $ds, $promptsfx;
376 return undef unless defined($_);
378 return $_ eq '' && defined($default) ?
$default : $_;
382 return _prompt_rl
(undef, @_);
388 $args[2] = "? " unless defined$args[2];
390 $result = prompt
(@args);
391 return undef unless defined($result);
392 redo unless is_yesno
($result);
394 return clean_bool
($result);
397 sub _prompt_rl_or_die
{
398 my $result = _prompt_rl
(@_);
399 unless (defined($result)) {
401 defined($nm) or $nm = "";
402 $nm eq "" or $nm .= " ";
403 die "\n${nm}aborted\n";
409 return _prompt_rl_or_die
(undef, @_);
412 sub ynprompt_or_die
{
413 my $result = ynprompt
(@_);
414 unless (defined($result)) {
416 defined($nm) or $nm = "";
417 $nm eq "" or $nm .= " ";
418 die "\n${nm}aborted\n";
424 my $ne = Girocco
::CLIUtil
::NoEcho
->new;
428 sub prompt_noecho_or_die
{
429 my $ne = Girocco
::CLIUtil
::NoEcho
->new;
430 _prompt_rl_or_die
(1, @_);
433 sub prompt_noecho_nl
{
434 my $result = prompt_noecho
(@_);
439 sub prompt_noecho_nl_or_die
{
440 my $result = prompt_noecho_or_die
(@_);
445 sub yes_to_continue
{
446 return !!ynprompt
(($_[0]||"Continue (enter \"yes\" to continue)"), "no");
449 sub yes_to_continue_or_die
{
450 unless (ynprompt_or_die
(($_[0]||"Continue (enter \"yes\" to continue)"), "no")) {
452 defined($nm) or $nm = "";
453 $nm .= " " if $nm ne "";
454 die "${nm}aborted\n";
460 my $user_list_loaded;
462 my $full_user_list_loaded;
464 # If single argument is true, return ALL passwd entries not just "...@..." ones
465 sub _get_all_users_internal
{
466 my $full = shift || 0;
468 return @full_user_list if $full_user_list_loaded;
470 return @user_list if $user_list_loaded;
472 my $passwd_file = jailed_file
("/etc/passwd");
473 open my $fd, '<', $passwd_file or die "could not open \"$passwd_file\": $!\n";
477 @users = map {/^([^:\s#][^:\s]*):[^:]*:(-?\d+):(-?\d+)(:|$)/
478 ?
[++$line,split(':',$_,-1)] : ()} <$fd>;
480 @users = map {/^([^:_\s#][^:\s#]*):[^:]+:(\d{5,}):(\d+):([^:,][^:]*)/
481 ?
[++$line,$1,$2,$3,split(',',$4)] : ()} <$fd>;
485 $$_[5] = [split(',', $$_[5])] foreach @users;
486 @full_user_list = @users;
487 $full_user_list_loaded = 1;
489 @users = grep({$$_[4] =~ /\@/} @users);
491 $user_list_loaded = 1;
496 # Return array of arrayref where each arrayref has:
497 # [0] = ordering ordinal from $chroot/etc/passwd
499 # [2] = user id number
500 # [3] = user group number
502 # [5] = user UUID (text as 8x-4x-4x-4x-12x) or undef if none
503 # [6] = user creation date as YYYYMMDD_HHMMSS (UTC) or undef if none
504 sub get_all_users
{ return _get_all_users_internal
; }
506 # Return array of arrayref where each arrayref has:
507 # [0] = ordering ordinal from $chroot/etc/passwd
509 # [2] = user password field (usually "x")
510 # [3] = user id number
511 # [4] = user group number
512 # [5] = [info fields] from passwd line (usually email,uuid,creation)
513 # [6] = home dir field
515 # [...] possibly more, but [7] is usually max
516 sub get_full_users
{ return _get_all_users_internal
(1); }
518 # Result of Girocco::User->load or fatal die if that fails
519 # Returns undef if passed undef or ""
521 my $username = shift;
522 defined($username) && $username ne "" or return undef;
523 Girocco
::User
::does_exist
($username, 1) or die "No such user: \"$username\"\n";
526 $user = Girocco
::User
->load($username);
528 } && $user->{uid
} or die "Could not load user \"$username\"\n";
533 my $project_list_loaded;
535 # Return array of arrayref where each arrayref has:
536 # [0] = ordering ordinal from $chroot/etc/group
538 # [2] = group password hash
539 # [3] = group id number
540 # [4] = owner from gitproj.list
541 # [5] = list of comma-separated push user names (can be "") or ":" if mirror
542 sub get_all_projects
{
543 return @project_list if $project_list_loaded;
545 my $projlist_file = $Girocco::Config
::projlist_cache_dir
."/gitproj.list";
546 open $fd, '<', $projlist_file or die "could not open \"$projlist_file\": $!\n";
547 my $chomper = sub {chomp(my $x = shift); $x;};
548 my %owners = map {(split(/\s+/, &$chomper($_), 3))[0,2]} <$fd>;
550 my $group_file = jailed_file
("/etc/group");
551 open $fd, '<', $group_file or die "could not open \"$group_file\": $!\n";
555 return ':' if $list =~ /^:/;
559 my $defu = sub {defined($_[0])?
$_[0]:""};
560 my @projects = map {/^([^:_\s#][^:\s#]*):([^:]*):(\d{5,}):(.*)$/
561 ?
[++$line,$1,$2,$3,&$defu($owners{$1}),&$trimu($4)] : ()} <$fd>;
563 @project_list = @projects;
564 $project_list_loaded = 1;
568 # Result of Girocco::Project->load or fatal die if that fails
569 # Returns undef if passed undef or ""
571 my $projname = shift;
572 $projname =~ s/\.git$//i if defined($projname);
573 defined($projname) && $projname ne "" or return undef;
574 Girocco
::Project
::does_exist
($projname, 1) or die "No such project: \"$projname\"\n";
577 $project = Girocco
::Project
->load($projname);
579 } && $project->{loaded
} or die "Could not load project \"$projname\"\n";
583 # return true if $enc_passwd is a match for $plain_passwd
584 sub check_passwd_match
{
585 my ($enc_passwd, $plain_passwd) = @_;
586 defined($enc_passwd) or $enc_passwd = '';
587 defined($plain_passwd) or $plain_passwd = '';
588 # $enc_passwd may be crypt or crypt_sha1
589 if ($enc_passwd =~ m
(^\
$sha1\
$(\d
+)\
$([./0-9A-Za-z]{1,64})\$[./0-9A
-Za
-z
]{28}$)) {
590 # It's using sha1-crypt
591 return $enc_passwd eq crypt_sha1
($plain_passwd, $2, -(0+$1));
594 return $enc_passwd eq crypt($plain_passwd, $enc_passwd);
600 foreach (File
::Spec
->path()) {
601 my $p = File
::Spec
->catfile($_, $cmd);
602 no warnings
'newline';
603 return $p if -x
$p && -f _
;
608 # apply maximum nice and ionice
612 if (defined($niceval) && $niceval =~ /^\d+$/ && 0 + $niceval >= 1) {
613 my $oldval = POSIX
::nice
(0);
614 POSIX
::nice
($niceval - $oldval) if $oldval && $niceval > $oldval;
618 defined($ionice) or $ionice = _which
("ionice");
619 defined($ionice) or $ionice = "";
621 my $devnullfd = POSIX
::open(File
::Spec
->devnull, O_RDWR
);
622 defined($devnullfd) && $devnullfd >= 0 or die "cannot open /dev/null: $!";
623 my ($dupin, $dupout, $duperr);
624 open $dupin, '<&0' or die "cannot dup STDIN_FILENO: $!";
625 open $dupout, '>&1' or die "cannot dup STDOUT_FILENO: $!";
626 open $duperr, '>&2' or die "cannot dup STDERR_FILENO: $!";
627 POSIX
::dup2
($devnullfd, 0) or die "cannot dup2 STDIN_FILENO: $!";
628 POSIX
::dup2
($devnullfd, 1) or die "cannot dup2 STDOUT_FILENO: $!";
629 POSIX
::dup2
($devnullfd, 2) or POSIX
::dup2
(fileno($duperr), 2), die "cannot dup2 STDERR_FILENO: $!";
630 POSIX
::close($devnullfd);
631 system $ionice, "-c", "3", "-p", $$;
632 POSIX
::dup2
(fileno($duperr), 2) or die "cannot dup2 STDERR_FILENO: $!";
633 POSIX
::dup2
(fileno($dupout), 1) or die "cannot dup2 STDOUT_FILENO: $!";
634 POSIX
::dup2
(fileno($dupin), 0) or die "cannot dup2 STDIN_FILENO: $!";
641 # spawn a pager and return the write side of
642 # a pipe to its input. Does not check to see
643 # if STDOUT is a terminal or anything else like
644 # that. Caller is responsible for those checks.
645 # Pager will be chosen as follows:
646 # 1. $ENV{PAGER} if non-empty (eval'd by shell)
647 # 2. less if found in $ENV{PATH}
648 # 3. more if found in $ENV{PATH}
649 # Returns undef if no pager can be found or
650 # setup fails. If return context is wantarray
651 # and pager is created, will return list of
652 # new output handle and pid of child.
653 # As a special case to facilitate paging of STDOUT,
654 # if the first argument is the string "become child",
655 # then, if a pager is created, the child will return
656 # to the caller and the parent will exec the pager!
657 # (The returned pid in that case is the parent's pid.)
660 defined($magic) && lc($magic) eq "become child" or
663 if (defined($ENV{PAGER
}) && $ENV{PAGER
} ne "") {
664 my $cmd = $ENV{PAGER
};
665 $cmd =~ /^(.+)$/ and $cmd = $1;
668 no warnings
'newline';
669 -x
$cmd && -f
$cmd and $pgbin = $cmd;
671 defined($pgbin) && $pgbin ne "" or $pgbin = _which
($cmd);
672 if (defined($pgbin) && $pgbin ne "") {
673 $pgbin =~ /^(.+)$/ and push(@cmd, $1);
675 $cmd =~ /\s/ || is_shellish
($cmd) or
677 my $sh = $Girocco::Config
::posix_sh_bin
;
678 defined($sh) && $sh ne "" or $sh = '/bin/sh';
679 push(@cmd, $sh, "-c", $cmd, $sh);
683 my $pgbin = _which
("less");
684 $pgbin or $pgbin = _which
("more");
685 defined($pgbin) && $pgbin ne "" or return undef;
686 $pgbin =~ /^(.+)$/ and push(@cmd, $1);
688 local $ENV{LESS
} = "-FRX" unless exists($ENV{LESS
});
689 local $ENV{LV
} = "-c" unless exists($ENV{LV
});
692 my ($rfd, $wfd) = POSIX
::pipe();
693 defined($rfd) && defined($wfd) && $rfd >= 0 && $wfd >= 0 or
694 die "POSIX::pipe failed: $!\n";
697 die "fork failed: $!\n";
698 if (!$magic && !$pid || $magic && $pid) {
700 POSIX
::dup2
($rfd, 0);
702 $magic and $SIG{CHLD
} = 'IGNORE';
703 exec {$cmd[0]} @cmd or
704 die "exec \"$cmd[0]\" failed: $!\n";
706 $magic and $pid = getppid();
708 open $pghnd, '>&=', $wfd or
709 die "fdopen of pipe write end failed: $!\n";
710 defined($pid) && defined($pghnd) or return undef;
711 return wantarray ?
($pghnd, $pid) : $pghnd;
714 # return true if any of the known PAGER_IN_USE environment
717 return $ENV{GIT_PAGER_IN_USE
} || $ENV{TG_PAGER_IN_USE
};
720 # possibly set STDOUT to flow through a pager
722 # defined and false -> return without doing anything
723 # defined and true -> set STDOUT to setup_pager result
725 # ! -t STDOUT -> return without doing anything
727 # $_[1] is false -> set STDOUT to setup_pager result
728 # $_[1] is true -> return without doing anything
729 # $[1] means do NOT enable paging by default on -t STDOUT
730 # Most clients can simply call this function without arguments
731 # which will add a pager only if STDOUT is a terminal
732 # If pager_in_use, returns without doing anything.
733 # If pager is activated, sets known pager in use env vars.
734 sub setup_pager_stdout
{
735 pager_in_use
() and return;
736 my $want_pager = $_[0];
737 defined($want_pager) or
738 $want_pager = (-t STDOUT
) ?
!$_[1] : 0;
739 return unless $want_pager;
740 my $pghnd = setup_pager
('become child');
741 defined($pghnd) or return;
742 if (open(STDOUT
, '>&=', $pghnd)) {
743 $ENV{GIT_PAGER_IN_USE
} = 1;
744 $ENV{TG_PAGER_IN_USE
} = 1;
746 die "failed to set STDOUT to pager: $!\n";