ui: Handle multiword and empty $PAGER values.

* guix/ui.scm (call-with-paginated-output-port): Empty PAGER values
disable paging.  Non-empty ones are split into command arguments.

Reported by Daniel Brooks <db48x@db48x.net>.
This commit is contained in:
Tobias Geerinckx-Rice 2020-11-15 19:25:00 +01:00
parent 4d0b61a1f6
commit a81258c124
No known key found for this signature in database
GPG key ID: 0DB0FF884F556D79

View file

@ -12,7 +12,7 @@
;;; Copyright © 2018 Kyle Meyer <kyle@kyleam.com> ;;; Copyright © 2018 Kyle Meyer <kyle@kyleam.com>
;;; Copyright © 2018 Ricardo Wurmus <rekado@elephly.net> ;;; Copyright © 2018 Ricardo Wurmus <rekado@elephly.net>
;;; Copyright © 2019 Chris Marusich <cmmarusich@gmail.com> ;;; Copyright © 2019 Chris Marusich <cmmarusich@gmail.com>
;;; Copyright © 2019 Tobias Geerinckx-Rice <me@tobias.gr> ;;; Copyright © 2019, 2020 Tobias Geerinckx-Rice <me@tobias.gr>
;;; Copyright © 2019 Simon Tournier <zimon.toutoune@gmail.com> ;;; Copyright © 2019 Simon Tournier <zimon.toutoune@gmail.com>
;;; Copyright © 2020 Arun Isaac <arunisaac@systemreboot.net> ;;; Copyright © 2020 Arun Isaac <arunisaac@systemreboot.net>
;;; Copyright © 2020 Maxim Cournoyer <maxim.cournoyer@gmail.com> ;;; Copyright © 2020 Maxim Cournoyer <maxim.cournoyer@gmail.com>
@ -1664,24 +1664,33 @@ (define (package-relevance package regexps)
(define* (call-with-paginated-output-port proc (define* (call-with-paginated-output-port proc
#:key (less-options "FrX")) #:key (less-options "FrX"))
(if (isatty?* (current-output-port)) (let ((pager-command-line (or (getenv "GUIX_PAGER")
;; Set 'LESS' so that 'less' exits if everything fits on the screen (F), (getenv "PAGER")
;; lets ANSI escapes through (r), does not send the termcap "less")))
;; initialization string (X). Set it unconditionally because some ;; Setting PAGER to the empty string conventionally disables paging.
;; distros set it to something that doesn't work here. (if (and (not (string-null? pager-command-line))
;; (isatty?* (current-output-port)))
;; For things that produce long lines, such as 'guix processes', use 'R' ;; Set 'LESS' so that 'less' exits if everything fits on the screen
;; instead of 'r': this strips hyperlinks but allows 'less' to make a ;; (F), lets ANSI escapes through (r), does not send the termcap
;; good estimate of the line length. ;; initialization string (X). Set it unconditionally because some
(let ((pager (with-environment-variables `(("LESS" ,less-options)) ;; distros set it to something that doesn't work here.
(open-pipe* OPEN_WRITE ;;
(or (getenv "GUIX_PAGER") (getenv "PAGER") ;; For things that produce long lines, such as 'guix processes', use
"less"))))) ;; 'R' instead of 'r': this strips hyperlinks but allows 'less' to
(dynamic-wind ;; make a good estimate of the line length.
(const #t) (let* ((pager (with-environment-variables `(("LESS" ,less-options))
(lambda () (proc pager)) (apply open-pipe* OPEN_WRITE
(lambda () (close-pipe pager)))) ;; Split into arguments. Treat runs of multiple
(proc (current-output-port)))) ;; whitespace characters as one. libpipeline-
;; style "cmd one\ arg" escaping is unsupported.
(remove (lambda (s) (string-null? s))
(string-split pager-command-line
char-set:whitespace))))))
(dynamic-wind
(const #t)
(lambda () (proc pager))
(lambda () (close-pipe pager))))
(proc (current-output-port)))))
(define-syntax with-paginated-output-port (define-syntax with-paginated-output-port
(syntax-rules () (syntax-rules ()