2018-10-26 05:20:08 -04:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
2020-06-28 16:36:43 -04:00
|
|
|
|
;;; Copyright © 2018, 2019, 2020 Ludovic Courtès <ludo@gnu.org>
|
2020-11-13 00:16:48 -05:00
|
|
|
|
;;; Copyright © 2020 John Soo <jsoo1@asu.edu>
|
2018-10-26 05:20:08 -04:00
|
|
|
|
;;;
|
|
|
|
|
;;; This file is part of GNU Guix.
|
|
|
|
|
;;;
|
|
|
|
|
;;; GNU Guix is free software; you can redistribute it and/or modify it
|
|
|
|
|
;;; under the terms of the GNU General Public License as published by
|
|
|
|
|
;;; the Free Software Foundation; either version 3 of the License, or (at
|
|
|
|
|
;;; your option) any later version.
|
|
|
|
|
;;;
|
|
|
|
|
;;; GNU Guix is distributed in the hope that it will be useful, but
|
|
|
|
|
;;; WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
;;; GNU General Public License for more details.
|
|
|
|
|
;;;
|
|
|
|
|
;;; You should have received a copy of the GNU General Public License
|
|
|
|
|
;;; along with GNU Guix. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
(define-module (guix scripts processes)
|
|
|
|
|
#:use-module ((guix store) #:select (%store-prefix))
|
|
|
|
|
#:use-module (guix scripts)
|
|
|
|
|
#:use-module (guix ui)
|
|
|
|
|
#:use-module (srfi srfi-1)
|
|
|
|
|
#:use-module (srfi srfi-9)
|
|
|
|
|
#:use-module (srfi srfi-9 gnu)
|
|
|
|
|
#:use-module (srfi srfi-37)
|
|
|
|
|
#:use-module (ice-9 ftw)
|
|
|
|
|
#:use-module (ice-9 match)
|
|
|
|
|
#:use-module (ice-9 rdelim)
|
|
|
|
|
#:use-module (ice-9 format)
|
|
|
|
|
#:export (process?
|
|
|
|
|
process-id
|
|
|
|
|
process-parent-id
|
|
|
|
|
process-command
|
|
|
|
|
processes
|
|
|
|
|
|
|
|
|
|
daemon-session?
|
|
|
|
|
daemon-session-process
|
|
|
|
|
daemon-session-client
|
|
|
|
|
daemon-session-children
|
|
|
|
|
daemon-session-locks-held
|
|
|
|
|
daemon-sessions
|
|
|
|
|
|
|
|
|
|
guix-processes))
|
|
|
|
|
|
|
|
|
|
;; Process as can be found in /proc on GNU/Linux.
|
|
|
|
|
(define-record-type <process>
|
|
|
|
|
(process id parent command)
|
|
|
|
|
process?
|
|
|
|
|
(id process-id) ;integer
|
|
|
|
|
(parent process-parent-id) ;integer | #f
|
|
|
|
|
(command process-command)) ;list of strings
|
|
|
|
|
|
|
|
|
|
(define (write-process process port)
|
|
|
|
|
(format port "#<process ~a>" (process-id process)))
|
|
|
|
|
|
|
|
|
|
(set-record-type-printer! <process> write-process)
|
|
|
|
|
|
|
|
|
|
(define (read-status-ppid port)
|
|
|
|
|
"Read the PPID from PORT, an input port on a /proc/PID/status file. Return
|
|
|
|
|
#f for PID 1 and kernel pseudo-processes."
|
|
|
|
|
(let loop ()
|
|
|
|
|
(match (read-line port)
|
|
|
|
|
((? eof-object?) #f)
|
|
|
|
|
(line
|
|
|
|
|
(if (string-prefix? "PPid:" line)
|
|
|
|
|
(string->number (string-trim-both (string-drop line 5)))
|
|
|
|
|
(loop))))))
|
|
|
|
|
|
|
|
|
|
(define %not-nul
|
|
|
|
|
(char-set-complement (char-set #\nul)))
|
|
|
|
|
|
|
|
|
|
(define (read-command-line port)
|
|
|
|
|
"Read the zero-split command line from PORT, a /proc/PID/cmdline file, and
|
|
|
|
|
return it as a list."
|
|
|
|
|
(string-tokenize (read-string port) %not-nul))
|
|
|
|
|
|
|
|
|
|
(define (processes)
|
|
|
|
|
"Return a list of process records representing the currently alive
|
|
|
|
|
processes."
|
|
|
|
|
;; This assumes a Linux-compatible /proc file system. There exists one for
|
|
|
|
|
;; GNU/Hurd.
|
|
|
|
|
(filter-map (lambda (pid)
|
|
|
|
|
;; There's a TOCTTOU race here. If we get ENOENT, simply
|
|
|
|
|
;; ignore PID.
|
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(define ppid
|
|
|
|
|
(call-with-input-file (string-append "/proc/" pid "/status")
|
|
|
|
|
read-status-ppid))
|
|
|
|
|
(define command
|
|
|
|
|
(call-with-input-file (string-append "/proc/" pid "/cmdline")
|
|
|
|
|
read-command-line))
|
|
|
|
|
(process (string->number pid) ppid command))
|
|
|
|
|
(lambda args
|
|
|
|
|
(if (= ENOENT (system-error-errno args))
|
|
|
|
|
#f
|
|
|
|
|
(apply throw args)))))
|
|
|
|
|
(scandir "/proc" string->number)))
|
|
|
|
|
|
|
|
|
|
(define (process-open-files process)
|
|
|
|
|
"Return the list of files currently open by PROCESS."
|
|
|
|
|
(let ((directory (string-append "/proc/"
|
|
|
|
|
(number->string (process-id process))
|
|
|
|
|
"/fd")))
|
2019-04-29 15:39:38 -04:00
|
|
|
|
(filter-map (lambda (fd)
|
|
|
|
|
;; There's a TOCTTOU race here, hence the 'catch'.
|
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(readlink (string-append directory "/" fd)))
|
|
|
|
|
(lambda args
|
|
|
|
|
(if (= ENOENT (system-error-errno args))
|
|
|
|
|
#f
|
|
|
|
|
(apply throw args)))))
|
|
|
|
|
(or (scandir directory string->number) '()))))
|
2018-10-26 05:20:08 -04:00
|
|
|
|
|
|
|
|
|
;; Daemon session.
|
|
|
|
|
(define-record-type <daemon-session>
|
|
|
|
|
(daemon-session process client children locks)
|
|
|
|
|
daemon-session?
|
|
|
|
|
(process daemon-session-process) ;<process>
|
|
|
|
|
(client daemon-session-client) ;<process>
|
|
|
|
|
(children daemon-session-children) ;list of <process>
|
|
|
|
|
(locks daemon-session-locks-held)) ;list of strings
|
|
|
|
|
|
|
|
|
|
(define (daemon-sessions)
|
|
|
|
|
"Return two values: the list of <daemon-session> denoting the currently
|
|
|
|
|
active sessions, and the master 'guix-daemon' process."
|
|
|
|
|
(define (lock-file? file)
|
|
|
|
|
(and (string-prefix? (%store-prefix) file)
|
|
|
|
|
(string-suffix? ".lock" file)))
|
|
|
|
|
|
|
|
|
|
(let* ((processes (processes))
|
|
|
|
|
(daemons (filter (lambda (process)
|
|
|
|
|
(match (process-command process)
|
|
|
|
|
((argv0 _ ...)
|
|
|
|
|
(string=? (basename argv0) "guix-daemon"))
|
|
|
|
|
(_ #f)))
|
|
|
|
|
processes))
|
|
|
|
|
(children (filter (lambda (process)
|
|
|
|
|
(match (process-command process)
|
|
|
|
|
((argv0 (= string->number argv1) _ ...)
|
|
|
|
|
(integer? argv1))
|
|
|
|
|
(_ #f)))
|
|
|
|
|
daemons))
|
|
|
|
|
(master (remove (lambda (process)
|
|
|
|
|
(memq process children))
|
|
|
|
|
daemons)))
|
|
|
|
|
(define (lookup-process pid)
|
|
|
|
|
(find (lambda (process)
|
|
|
|
|
(and (process-id process)
|
|
|
|
|
(= pid (process-id process))))
|
|
|
|
|
processes))
|
|
|
|
|
|
|
|
|
|
(define (lookup-children pid)
|
|
|
|
|
(filter (lambda (process)
|
|
|
|
|
(and (process-parent-id process)
|
|
|
|
|
(= pid (process-parent-id process))))
|
|
|
|
|
processes))
|
|
|
|
|
|
2019-04-29 15:42:04 -04:00
|
|
|
|
(define (child-process->session process)
|
|
|
|
|
(match (process-command process)
|
|
|
|
|
((argv0 (= string->number client) _ ...)
|
|
|
|
|
(let ((files (process-open-files process))
|
|
|
|
|
(client (lookup-process client)))
|
|
|
|
|
;; After a client has died, there's a window during which its
|
|
|
|
|
;; corresponding 'guix-daemon' process is still alive, in which
|
|
|
|
|
;; case 'lookup-process' returns #f. In that case ignore the
|
|
|
|
|
;; session.
|
|
|
|
|
(and client
|
|
|
|
|
(daemon-session process client
|
|
|
|
|
(lookup-children
|
|
|
|
|
(process-id process))
|
|
|
|
|
(filter lock-file? files)))))))
|
|
|
|
|
|
|
|
|
|
(values (filter-map child-process->session children)
|
2018-10-26 05:20:08 -04:00
|
|
|
|
master)))
|
|
|
|
|
|
2020-11-04 10:51:52 -05:00
|
|
|
|
(define (lock->recutils lock port)
|
|
|
|
|
(format port "LockHeld: ~a~%" lock))
|
|
|
|
|
|
2018-10-26 05:20:08 -04:00
|
|
|
|
(define (daemon-session->recutils session port)
|
|
|
|
|
"Display SESSION information in recutils format on PORT."
|
|
|
|
|
(format port "SessionPID: ~a~%"
|
|
|
|
|
(process-id (daemon-session-process session)))
|
|
|
|
|
(format port "ClientPID: ~a~%"
|
|
|
|
|
(process-id (daemon-session-client session)))
|
|
|
|
|
(format port "ClientCommand:~{ ~a~}~%"
|
|
|
|
|
(process-command (daemon-session-client session)))
|
2020-11-04 10:51:52 -05:00
|
|
|
|
(for-each (lambda (lock) (lock->recutils lock port))
|
2018-10-26 05:20:08 -04:00
|
|
|
|
(daemon-session-locks-held session))
|
|
|
|
|
(for-each (lambda (process)
|
2020-11-13 00:16:48 -05:00
|
|
|
|
(format port "ChildPID: ~a~%"
|
|
|
|
|
(process-id process))
|
|
|
|
|
(format port "ChildCommand: :~{ ~a~}~%"
|
2018-10-26 05:20:08 -04:00
|
|
|
|
(process-command process)))
|
|
|
|
|
(daemon-session-children session)))
|
|
|
|
|
|
2020-11-04 10:51:52 -05:00
|
|
|
|
(define (daemon-sessions->recutils port sessions)
|
|
|
|
|
"Display denormalized SESSIONS information to PORT."
|
|
|
|
|
(for-each (lambda (session)
|
|
|
|
|
(daemon-session->recutils session port)
|
|
|
|
|
(newline port))
|
|
|
|
|
sessions))
|
|
|
|
|
|
|
|
|
|
(define session-rec-type
|
|
|
|
|
"%rec: Session
|
|
|
|
|
%type: PID int
|
|
|
|
|
%type: ClientPID int
|
|
|
|
|
%key: PID
|
|
|
|
|
%mandatory: ClientPID ClientCommand")
|
|
|
|
|
|
|
|
|
|
(define lock-rec-type
|
|
|
|
|
"%rec: Lock
|
|
|
|
|
%mandatory: LockHeld
|
|
|
|
|
%type: Session rec Session")
|
|
|
|
|
|
|
|
|
|
(define child-process-rec-type
|
|
|
|
|
"%rec: ChildProcess
|
|
|
|
|
%type: PID int
|
|
|
|
|
%type: Session rec Session
|
|
|
|
|
%key: PID
|
|
|
|
|
%mandatory: Command")
|
|
|
|
|
|
|
|
|
|
(define (session-key->recutils session port)
|
|
|
|
|
"Display SESSION PID as a recutils field on PORT."
|
|
|
|
|
(format
|
|
|
|
|
port "Session: ~a"
|
|
|
|
|
(process-id (daemon-session-process session))))
|
|
|
|
|
|
|
|
|
|
(define (session-scalars->normalized-record session port)
|
|
|
|
|
"Display SESSION scalar fields to PORT in normalized form."
|
|
|
|
|
(format port "PID: ~a~%"
|
|
|
|
|
(process-id (daemon-session-process session)))
|
|
|
|
|
(format port "ClientPID: ~a~%"
|
|
|
|
|
(process-id (daemon-session-client session)))
|
|
|
|
|
(format port "ClientCommand:~{ ~a~}~%"
|
|
|
|
|
(process-command (daemon-session-client session))))
|
|
|
|
|
|
|
|
|
|
(define (child-process->normalized-record process port)
|
|
|
|
|
"Display PROCESS record on PORT in normalized form"
|
|
|
|
|
(format port "PID: ~a" (process-id process))
|
|
|
|
|
(newline port)
|
|
|
|
|
(format port "Command:~{ ~a~}" (process-command process)))
|
|
|
|
|
|
|
|
|
|
(define (daemon-sessions->normalized-record port sessions)
|
|
|
|
|
"Display SESSIONS recutils on PORT in normalized form"
|
|
|
|
|
(display session-rec-type port)
|
|
|
|
|
(newline port)
|
|
|
|
|
(newline port)
|
|
|
|
|
(for-each (lambda (session)
|
|
|
|
|
(session-scalars->normalized-record session port)
|
|
|
|
|
(newline port))
|
|
|
|
|
sessions)
|
|
|
|
|
|
|
|
|
|
(display lock-rec-type port)
|
|
|
|
|
(newline port)
|
|
|
|
|
(newline port)
|
|
|
|
|
(for-each (lambda (session)
|
|
|
|
|
(for-each (lambda (lock)
|
|
|
|
|
(lock->recutils "testing testing" port)
|
|
|
|
|
(session-key->recutils session port)
|
|
|
|
|
(newline port)
|
|
|
|
|
(newline port))
|
|
|
|
|
(daemon-session-locks-held session)))
|
|
|
|
|
sessions)
|
|
|
|
|
|
|
|
|
|
(display child-process-rec-type port)
|
|
|
|
|
(newline port)
|
|
|
|
|
(newline port)
|
|
|
|
|
(for-each (lambda (session)
|
|
|
|
|
(for-each (lambda (process)
|
|
|
|
|
(child-process->normalized-record process port)
|
|
|
|
|
(newline port)
|
|
|
|
|
(session-key->recutils session port)
|
|
|
|
|
(newline port)
|
|
|
|
|
(newline port))
|
|
|
|
|
(daemon-session-children session)))
|
|
|
|
|
sessions))
|
|
|
|
|
|
2018-10-26 05:20:08 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Options.
|
|
|
|
|
;;;
|
|
|
|
|
|
2020-11-04 10:51:52 -05:00
|
|
|
|
(define %available-formats
|
|
|
|
|
'("recutils" "normalized"))
|
|
|
|
|
|
|
|
|
|
(define (list-formats)
|
|
|
|
|
(display (G_ "The available formats are:\n"))
|
|
|
|
|
(newline)
|
|
|
|
|
(for-each (lambda (f)
|
|
|
|
|
(format #t " - ~a~%" f))
|
|
|
|
|
%available-formats))
|
2018-10-26 05:20:08 -04:00
|
|
|
|
|
|
|
|
|
(define (show-help)
|
|
|
|
|
(display (G_ "Usage: guix processes
|
|
|
|
|
List the current Guix sessions and their processes."))
|
|
|
|
|
(newline)
|
|
|
|
|
(display (G_ "
|
|
|
|
|
-h, --help display this help and exit"))
|
|
|
|
|
(display (G_ "
|
|
|
|
|
-V, --version display version information and exit"))
|
|
|
|
|
(newline)
|
2020-11-04 10:51:52 -05:00
|
|
|
|
(display (G_ "
|
|
|
|
|
-f, --format=FORMAT display results as normalized record sets"))
|
|
|
|
|
(display (G_ "
|
|
|
|
|
--list-formats display available formats"))
|
|
|
|
|
(newline)
|
2018-10-26 05:20:08 -04:00
|
|
|
|
(show-bug-report-information))
|
|
|
|
|
|
2020-11-04 10:51:52 -05:00
|
|
|
|
(define %options
|
|
|
|
|
(list (option '(#\h "help") #f #f
|
|
|
|
|
(lambda args
|
|
|
|
|
(show-help)
|
|
|
|
|
(exit 0)))
|
|
|
|
|
(option '(#\V "version") #f #f
|
|
|
|
|
(lambda args
|
|
|
|
|
(show-version-and-exit "guix processes")))
|
|
|
|
|
(option '(#\f "format") #t #f
|
|
|
|
|
(lambda (opt name arg result)
|
|
|
|
|
(unless (member arg %available-formats)
|
|
|
|
|
(leave (G_ "~a: unsupported output format~%") arg))
|
|
|
|
|
(alist-cons 'format (string->symbol arg) result)))
|
|
|
|
|
(option '("list-formats") #f #f
|
|
|
|
|
(lambda (opt name arg result)
|
|
|
|
|
(list-formats)
|
|
|
|
|
(exit 0)))))
|
|
|
|
|
|
|
|
|
|
(define %default-options '((format . recutils)))
|
|
|
|
|
|
2018-10-26 05:20:08 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Entry point.
|
|
|
|
|
;;;
|
|
|
|
|
|
2020-09-01 16:13:11 -04:00
|
|
|
|
(define-command (guix-processes . args)
|
|
|
|
|
(category plumbing)
|
|
|
|
|
(synopsis "list currently running sessions")
|
2020-11-04 10:51:52 -05:00
|
|
|
|
|
2018-10-26 05:20:08 -04:00
|
|
|
|
(define options
|
2020-11-04 10:51:52 -05:00
|
|
|
|
(parse-command-line args %options (list %default-options)
|
|
|
|
|
#:build-options? #f))
|
2018-10-26 05:20:08 -04:00
|
|
|
|
|
2020-06-28 16:36:43 -04:00
|
|
|
|
(with-paginated-output-port port
|
2020-11-04 10:51:52 -05:00
|
|
|
|
(match (assoc-ref options 'format)
|
|
|
|
|
('normalized
|
|
|
|
|
(daemon-sessions->normalized-record port (daemon-sessions)))
|
|
|
|
|
(_ (daemon-sessions->recutils port (daemon-sessions))))
|
2020-07-26 10:45:42 -04:00
|
|
|
|
|
|
|
|
|
;; Pass 'R' (instead of 'r') so 'less' correctly estimates line length.
|
|
|
|
|
#:less-options "FRX"))
|