aboutsummaryrefslogtreecommitdiff
;;; GNU Guix --- Functional package management for GNU
;;; Copyright © 2020 Ludovic Courtès <ludo@gnu.org>
;;; Copyright © 2020 Mathieu Othacehe <m.othacehe@gmail.com>
;;; Copyright © 2024 Janneke Nieuwenhuizen <janneke@gnu.org>
;;;
;;; 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 (gnu installer tests)
  #:use-module (srfi srfi-1)
  #:use-module (srfi srfi-34)
  #:use-module (srfi srfi-35)
  #:use-module (ice-9 match)
  #:use-module (ice-9 regex)
  #:use-module (ice-9 pretty-print)
  #:export (&pattern-not-matched
            pattern-not-matched?

            %installer-socket-file
            open-installer-socket

            converse
            conversation-log-port

            choose-locale+keyboard
            enter-host-name+passwords
            choose-kernel
            choose-services
            choose-partitioning
            start-installation
            complete-installation

            edit-configuration-file))

;;; Commentary:
;;;
;;; This module provides tools to test the guided "graphical" installer in a
;;; non-interactive fashion.  The core of it is 'converse': it allows you to
;;; state Expect-style dialogues, which happen over the Unix-domain socket the
;;; installer listens to.  Higher-level procedures such as
;;; 'choose-locale+keyboard' are provided to perform specific parts of the
;;; dialogue.
;;;
;;; Code:

(define %installer-socket-file
  ;; Socket the installer listens to.
  "/var/guix/installer-socket")

(define* (open-installer-socket #:optional (file %installer-socket-file))
  "Return a socket connected to the installer."
  (let ((sock (socket AF_UNIX SOCK_STREAM 0)))
    (connect sock AF_UNIX file)
    sock))

(define-condition-type &pattern-not-matched &error
  pattern-not-matched?
  (pattern pattern-not-matched-pattern)
  (sexp    pattern-not-matched-sexp))

(define (pattern-error pattern sexp)
  (raise (condition
          (&pattern-not-matched
           (pattern pattern) (sexp sexp)))))

(define conversation-log-port
  ;; Port where debugging info is logged
  (make-parameter (current-error-port)))

(define (converse-debug pattern)
  (format (conversation-log-port)
          "conversation expecting pattern ~s~%"
          pattern))

(define-syntax converse
  (lambda (s)
    "Convert over PORT: read sexps from there, match them against each
PATTERN, and send the corresponding REPLY.  Raise to '&pattern-not-matched'
when one of the PATTERNs is not matched."

    ;; XXX: Strings that appear in PATTERNs must be in the language the
    ;; installer is running in.  In the future, we should add support to allow
    ;; writing English strings in PATTERNs and have the pattern matcher
    ;; automatically translate them.

    ;; Here we emulate 'pmatch' syntax on top of 'match'.  This is ridiculous
    ;; but that's because 'pmatch' compares objects with 'eq?', making it
    ;; pretty useless, and it doesn't support ellipses and such.

    (define (quote-pattern s)
      ;; Rewrite the pattern S from pmatch style (a ,b) to match style like
      ;; ('a b).
      (with-ellipsis :::
        (syntax-case s (unquote _ ...)
          ((unquote id) #'id)
          (_ #'_)
          (... #'...)
          (id
           (identifier? #'id)
           #''id)
          ((lst :::) (map quote-pattern #'(lst :::)))
          (pattern #'pattern))))

    (define (match-pattern s)
      ;; Match one pattern without a guard.
      (syntax-case s ()
        ((port (pattern reply) continuation)
         (with-syntax ((pattern (quote-pattern #'pattern)))
           #'(let ((pat 'pattern))
               (converse-debug pat)
               (match (read port)
                 (pattern
                  (let ((data (call-with-values (lambda () reply)
                                list)))
                    (for-each (lambda (obj)
                                (write obj port)
                                (newline port))
                              data)
                    (force-output port)
                    (continuation port)))
                 (sexp
                  (pattern-error pat sexp))))))))

    (syntax-case s ()
      ((_ port (pattern reply) rest ...)
       (match-pattern #'(port (pattern reply)
                              (lambda (port)
                                (converse port rest ...)))))
      ((_ port (pattern guard reply) rest ...)
       #`(let ((skip? (not guard))
               (next  (lambda (p)
                        (converse p rest ...))))
           (if skip?
               (next port)
               #,(match-pattern #'(port (pattern reply) next)))))
      ((_ port)
       #t))))

(define* (choose-locale+keyboard port
                                 #:key
                                 (language "English")
                                 (location "Hong Kong")
                                 (timezone '("Europe" "Zagreb"))
                                 (keyboard
                                  '("English (US)"
                                    "English (intl., with AltGr dead keys)")))
  "Converse over PORT with the guided installer to choose the specified
LANGUAGE, LOCATION, TIMEZONE, and KEYBOARD."
  (converse port
    ((list-selection (title "Locale language")
                     (multiple-choices? #f)
                     (items _))
     language)
    ((list-selection (title "Locale location")
                     (multiple-choices? #f)
                     (items _))
     location)
    ((menu (title "GNU Guix install")
           (text _)
           (items (,guided _ ...)))           ;"Guided graphical installation"
     guided)
    ((list-selection (title "Timezone")
                     (multiple-choices? #f)
                     (items _))
     (first timezone))
    ((list-selection (title "Timezone")
                     (multiple-choices? #f)
                     (items _))
     (second timezone))
    ((list-selection (title "Layout")
                     (multiple-choices? #f)
                     (items _))
     (first keyboard))
    ((list-selection (title "Variant")
                     (multiple-choices? #f)
                     (items _))
     (second keyboard))))

(define* (enter-host-name+passwords port
                                    #:key
                                    (host-name "guix")
                                    (root-password "foo")
                                    (users '(("alice" "pass1")
                                             ("bob" "pass2")
                                             ("charlie" "pass3"))))
  "Converse over PORT with the guided installer to choose HOST-NAME,
ROOT-PASSWORD, and USERS."
  (converse port
    ((input (title "Hostname") (text _) (default _))
     host-name)
    ((input (title "System administrator password") (text _) (default _))
     root-password)
    ((input (title "Password confirmation required") (text _) (default _))
     root-password)
    ((add-users)
     (match users
       (((names passwords) ...)
        (map (lambda (name password)
               `(user (name ,name) (real-name ,(string-titlecase name))
                      (home-directory ,(string-append "/home/" name))
                      (password ,password)))
             names passwords))))))

(define* (choose-kernel port #:key (kernel "Linux Libre"))
  "Converse over PORT with the guided installer to choose the specified
KERNEL."
  (converse port
    ((list-selection (title "Kernel")
                     (multiple-choices? #f)
                     (items _))
     kernel)))

(define* (choose-services port
                          #:key
                          (choose-desktop-environment? (const #f))
                          (choose-network-service?
                           (lambda (service)
                             (or (string-contains service "SSH")
                                 (string-contains service "NSS"))))
                          (choose-network-management-tool?
                           (lambda (service)
                             (string-contains service "DHCP")))
                          (choose-misc-service?
                           (lambda (service)
                             (string-contains service "NTP")))
                          (choose-other-service? (const #f)))

  "Converse over PORT to choose services."
  (define desktop-environments '())

  (converse port
    ((checkbox-list (title "Desktop environment") (text _)
                    (items ,services))
     (let ((desktops (filter choose-desktop-environment? services)))
       (set! desktop-environments desktops)
       desktops))
    ((checkbox-list (title "Network service") (text _)
                    (items ,services))
     (filter choose-network-service? services))

    ;; The "Network management" dialog shows up only when no desktop
    ;; environments have been selected, hence the guard.
    ((list-selection (title "Network management")
                     (multiple-choices? #f)
                     (items ,services))
     (null? desktop-environments)
     (find choose-network-management-tool? services))

    ((checkbox-list (title "Console services") (text _)
                    (items ,services))
     (null? desktop-environments)
     (filter choose-misc-service? services))

    ((checkbox-list (title "Printing and document services") (text _)
                    (items ,services))
     (filter choose-other-service? services))))

(define (edit-configuration-file file)
  "Edit FILE, an operating system configuration file generated by the
installer, by adding a marionette service such that the installed OS is
instrumented for further testing."
  (define (read-expressions port)
    (let loop ((result '()))
      (match (read port)
        ((? eof-object?)
         (reverse result))
        (exp
         (loop (cons exp result))))))

  (define (edit exp)
    (match exp
      (('operating-system _ ...)
       `(marionette-operating-system ,exp
                                     #:imported-modules
                                     '((gnu services herd)
                                       (guix build utils)
                                       (guix combinators))))
      (_
       exp)))

  (let ((content (call-with-input-file file read-expressions)))
    ;; XXX: Remove the file before re-writing it, to be sure there are no
    ;; leftovers.  We shouldn't have to do that as CALL-WITH-OUTPUT-FILE uses
    ;; the O_TRUNC flag by default.
    (delete-file file)
    (call-with-output-file file
      (lambda (port)
        (format port "\
;; Operating system configuration edited for automated testing.~%~%")

        (pretty-print '(use-modules (gnu tests)) port)
        (for-each (lambda (exp)
                    (pretty-print (edit exp) port)
                    (newline port))
                  content)))

    #t))

(define* (choose-partitioning port
                              #:key
                              (encrypted? #t)
                              (uefi-support? #f)
                              (passphrase "thepassphrase")
                              (edit-configuration-file
                               edit-configuration-file))
  "Converse over PORT to choose the partitioning method.  When ENCRYPTED? is
true, choose full-disk encryption with PASSPHRASE as the LUKS passphrase.

When UEFI-SUPPORT? is true, assume that we are running the installation tests
on an UEFI capable machine.

This conversation stops when the user partitions have been formatted, right
before the installer generates the configuration file and shows it in a dialog
box. "
  (converse port
    ((list-selection (title "Partitioning method")
                     (multiple-choices? #f)
                     (items (,not-encrypted ,encrypted _ ...)))
     (if encrypted?
         encrypted
         not-encrypted))
    ((list-selection (title "Disk") (multiple-choices? #f)
                     (items (,disks ...)))
     ;; When running the installation from an ISO image, the CD/DVD drive
     ;; shows up in the list.  Avoid it.
     (find (lambda (disk)
             (not (or (string-contains disk "DVD")
                      (string-contains disk "CD-ROM"))))
           disks))

    ;; The "Partition table" dialog pops up only if there's not already a
    ;; partition table and if the system does not support UEFI.
    ((list-selection (title "Partition table")
                     (multiple-choices? #f)
                     (items _))
     ;; When UEFI is supported, the partition is forced to GPT by the
     ;; installer.
     (not uefi-support?)
     "gpt")

    ((list-selection (title "Partition scheme")
                     (multiple-choices? #f)
                     (items (,one-partition _ ...)))
     one-partition)
    ((list-selection (title "Guided partitioning")
                     (multiple-choices? #f)
                     (items (,disk _ ...)))
     disk)
    ((input (title "Password required")
            (text _) (default #f))
     encrypted?                                   ;only when ENCRYPTED?
     passphrase)
    ((input (title "Password confirmation required")
            (text _) (default #f))
     encrypted?
     passphrase)
    ((confirmation (title "Format disk?") (text _))
     #t)
    ((info (title "Preparing partitions") _ ...)
     (values))                                    ;nothing to return
    ((starting-final-step)
     ;; Do not return anything.  The reply will be sent by
     ;; 'conclude-installation' and in the meantime the installer just waits
     ;; for us, giving us a chance to do things such as changing partition
     ;; UUIDs before it generates the configuration file.
     (values))))

(define (start-installation port)
  "Start the installation by checking over PORT that we get the generated
configuration file, accepting it and starting the installation, and then
receiving the pause message once the 'guix system init' process has
completed."
  ;; Assume the previous message received was 'starting-final-step'; here we
  ;; send the reply to that message, which lets the installer continue.
  (write #t port)
  (newline port)
  (force-output port)

  (converse port
    ((file-dialog (title "Configuration file")
                  (text _)
                  (file ,configuration-file))
     (edit-configuration-file configuration-file))
    ((pause)                                      ;"Press Enter to continue."
     (values))))

(define (complete-installation port)
  "Complete the installation by replying to the installer pause message and
waiting for the installation-complete message."
  ;; Assume the previous message received was 'pause'; here we send the reply
  ;; to that message, which lets the installer continue.
  (write #t port)
  (newline port)
  (force-output port)

  (converse port
    ((installation-complete)
     (values))))

;;; Local Variables:
;;; eval: (put 'converse 'scheme-indent-function 1)
;;; eval: (put 'with-ellipsis 'scheme-indent-function 1)
;;; End:
(origin (method git-fetch) (uri (git-reference (url "https://github.com/gnotclub/xst") (commit (string-append "v" version)))) (file-name (git-file-name name version)) (sha256 (base32 "1q64x7czpbcg0v509qchn5v96zdnx7jmvy0zxhjmkk3d10x5rqlw")))) (home-page "https://github.com/gnotclub/xst") (synopsis "Fork of st that uses Xresources") (description "@command{xst} uses Xresources and applies the following patches to @command{st}: @itemize @item @uref{https://st.suckless.org/patches/alpha/, alpha} @item @uref{https://st.suckless.org/patches/boxdraw/, boxdraw} @item @uref{https://st.suckless.org/patches/clipboard/, clipboard} @item @uref{https://st.suckless.org/patches/disable_bold_italic_fonts/, disable_bold_italic_fonts} @item @uref{https://st.suckless.org/patches/externalpipe/, externalpipe} @item @uref{https://st.suckless.org/patches/scrollback/, scrollback} @item @uref{https://st.suckless.org/patches/spoiler/, spoiler} @item @uref{https://st.suckless.org/patches/vertcenter/, vertcenter} @end itemize") (license (list license:x11 license:expat)))) (define-public lukesmithxyz-st (let ((commit "e053bd6036331cc7d14f155614aebc20f5371d3a") (revision "0")) (package (inherit st) (name "lukesmithxyz-st") (version "0.8.4") (source (origin (method git-fetch) (uri (git-reference (url "https://github.com/LukeSmithxyz/st") (commit commit))) (file-name (git-file-name name version)) (sha256 (base32 "12avzzapkkj4mvd00zh8b6gynk6jysh84jcwlkliyyd82lvyw22v")))) (arguments (substitute-keyword-arguments (package-arguments st) ((#:phases phases) `(modify-phases ,phases (add-after 'unpack 'remove-calls-to-git (lambda _ (substitute* "Makefile" (("git submodule init") "") (("git submodule update") "")))))))) (inputs (modify-inputs (package-inputs st) (prepend libxext harfbuzz))) (home-page "https://github.com/LukeSmithxyz/st") (synopsis "Luke Smith's fork of st") (description "This package is Luke's fork of the suckless simple terminal (st) with Vim bindings and Xresource compatibility.") (license license:expat)))) (define-public sxmo-st (package (inherit st) (name "sxmo-st") (version "0.8.4.1") (source (origin (method git-fetch) (uri (git-reference (url "https://git.sr.ht/~mil/sxmo-st") (commit version))) (file-name (git-file-name name version)) (sha256 (base32 "1nl40q1pxf46hpbibz9m9d0giiy1p3lrhr9agw0fkyba2vzbbafa")))) (home-page "https://git.sr.ht/~mil/sxmo-st") (synopsis "St terminal emulator for the Simple X Mobile PinePhone environment") (license license:expat))) (define-public surf (package (name "surf") (version "2.1") (source (origin (method url-fetch) (uri (string-append "https://dl.suckless.org/surf/surf-" version ".tar.gz")) (sha256 (base32 "0mrj0kp01bwrgrn4v298g81h6zyq64ijsg790di68nm21f985rbj")))) (build-system glib-or-gtk-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure) ;; Use the right file name for dmenu and xprop. (add-before 'build 'set-dmenu-and-xprop-file-name (lambda* (#:key inputs #:allow-other-keys) (substitute* "config.def.h" (("dmenu") (search-input-file inputs "/bin/dmenu")) (("xprop") (search-input-file inputs "/bin/xprop"))) #t))))) (inputs `(("dmenu" ,dmenu) ("gcr" ,gcr-3) ("glib-networking" ,glib-networking) ("gsettings-desktop-schemas" ,gsettings-desktop-schemas) ("webkitgtk" ,webkitgtk-with-libsoup2) ("xprop" ,xprop))) (native-inputs (list pkg-config)) (home-page "https://surf.suckless.org/") (synopsis "Simple web browser") (description "Surf is a simple web browser based on WebKit/GTK+. It is able to display websites and follow links. It supports the XEmbed protocol which makes it possible to embed it in another application. Furthermore, one can point surf to another URI by setting its XProperties.") (license (list license:expat license:x11)))) (define-public sent (package (name "sent") (version "1") (source (origin (method url-fetch/tarbomb) (uri (string-append "https://dl.suckless.org/tools/sent-" version ".tar.gz")) (sha256 (base32 "0cxysz5lp25mgww73jl0mgip68x7iyvialyzdbriyaff269xxwvv")))) (build-system gnu-build-system) (arguments `(#:phases (modify-phases %standard-phases (delete 'configure) ;no configuration (add-before 'build 'patch-farbfeld (lambda* (#:key inputs #:allow-other-keys) (substitute* "config.def.h" (("2ff") (search-input-file inputs "/bin/2ff"))))) (add-after 'install 'install-doc (lambda* (#:key outputs #:allow-other-keys) (let* ((out (assoc-ref outputs "out")) (doc (string-append out "/share/doc/" ,name "-" ,(package-version this-package)))) (install-file "README.md" doc))))) #:tests? #f ;no test suite #:make-flags (let ((pkg-config (lambda (flag) (string-append "$(shell pkg-config " flag " " "xft fontconfig x11 libpng)")))) (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output) (string-append "INCS=-I. " (pkg-config "--cflags")) (string-append "LIBS=" (pkg-config "--libs") " -lm"))))) (native-inputs (list pkg-config)) (inputs (list farbfeld libpng libx11 libxft fontconfig)) (synopsis "Plain-text presentation tool") (description "Sent uses plain-text files and PNG images to create slideshow presentations. Each paragraph represents a slide in the presentation. Especially for presentations using the Takahashi method this is very nice and allows you to write down the presentation for a quick lightning talk within a few minutes.") (home-page "https://tools.suckless.org/sent/") (license license:isc))) (define-public wmname (package (name "wmname") (version "0.1") (source (origin (method url-fetch) (uri (string-append "https://dl.suckless.org/tools/wmname-" version ".tar.gz")) (sha256 (base32 "1i82ilhbk36hspc2j0fbpg27wjj7xnvzpv1ppgf6fccina4d36jm")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases ; no tests (modify-phases %standard-phases (delete 'configure)))) ; no configure script (inputs (list libx11)) (home-page "https://tools.suckless.org/x/wmname/") (synopsis "Print or set the window manager name") (description "@command{wmname} prints/sets the window manager name property of the root window similar to how @command{hostname} behaves. It is useful for fixing problems with JDK versions and other broken programs assuming a reparenting window manager for instance.") (license (list license:x11 license:expat)))) (define-public xbattmon (package (name "xbattmon") (version "1.1") (source (origin (method url-fetch) (uri (string-append "https://dl.2f30.org/releases/" "xbattmon-" version ".tar.gz")) (sha256 (base32 "1zr6y8lml9xkx0a3dbbsds2qz1bjxvskp7wsckkf8mlsqrbb3xsg")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)))) (inputs (list libx11)) (home-page "https://git.2f30.org/xbattmon/") (synopsis "Simple battery monitor for X") (description "Xbattmon is a simple battery monitor for X.") (license license:isc))) (define-public wificurse (package (name "wificurse") (version "0.3.9") (source (origin (method url-fetch) (uri (string-append "https://dl.2f30.org/releases/" "wificurse-" version ".tar.gz")) (sha256 (base32 "067ghr1xly5ca41kc83xila1p5hpq0bxfcmc8jvxi2ggm6wrhavn")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; No tests #:make-flags (list (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) ; No configure script (home-page "https://git.2f30.org/wificurse/") (synopsis "Wifi DoS attack tool") (description "Wificurses listens for beacons sent from wireless access points in the range of your wireless station. Once received the program extracts the BSSID of the AP and transmits deauthentication packets using the broadcast MAC address. This results to the disconnection of all clients connected to the AP at the time of the attack. This is essencially a WiFi DoS attack tool created for educational purposes only. It works only in Linux and requires wireless card drivers capable of injecting packets in wireless networks.") (license license:gpl3+))) (define-public skroll (package (name "skroll") (version "0.6") (source (origin (method url-fetch) (uri (string-append "https://dl.2f30.org/releases/" "skroll-" version ".tar.gz")) (sha256 (base32 "0km6bjfz4ssb1z0xwld6iiixnn7d255ax8yjs3zkdm42z8q9yl0f")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) ; no configure script (home-page "https://2f30.org/") (synopsis "Commandline utility which scrolls text") (description "Skroll is a small utility that you can use to make a text scroll. Pipe text to it, and it will scroll a given number of letters from right to left.") (license license:wtfpl2))) (define-public sbm (package (name "sbm") (version "0.9") (source (origin (method url-fetch) (uri (string-append "https://dl.2f30.org/releases/" "sbm-" version ".tar.gz")) (sha256 (base32 "1nks5mkh5wn30kyjzlkjlgi31bv1wq52kbp0r6nzbyfnvfdlywik")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) ; no configure script (home-page "https://git.2f30.org/sbm/") (synopsis "Simple bandwidth monitor") (description "Sbm is a simple bandwidth monitor.") (license license:isc))) (define-public prout (package (name "prout") (version "0.2") (source (origin (method url-fetch) (uri (string-append "https://dl.2f30.org/releases/" "prout-" version ".tar.gz")) (sha256 (base32 "1s6c3ygg1h1fyxkh8gd7nzjk6qhnwsb4535d2k780kxnwns5fzas")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) ; no configure script (inputs (list cups-minimal zlib)) (home-page "https://git.2f30.org/prout/") (synopsis "Smaller lp command") (description "Prout (PRint OUT) is a small utility one can use to send documents to a printer. It has no feature, and does nothing else. Just set your default printer in client.conf(5) and start printing. No need for a local cups server to be installed.") (license license:wtfpl2))) (define-public noice (package (name "noice") (version "0.8") (source (origin (method url-fetch) (uri (string-append "https://dl.2f30.org/releases/" "noice-" version ".tar.gz")) (sha256 (base32 "0g01iwzscdv27c1idv93gd74kjzy3n9kazgm6qz08rygp96qx4xw")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure) ; no configure script (add-before 'build 'curses (lambda _ (substitute* "Makefile" (("lcurses") "lncurses"))))))) (inputs (list ncurses)) (home-page "https://git.2f30.org/noice/") (synopsis "Small file browser") (description "Noice is a small curses-based file browser.") (license license:bsd-2))) (define-public human (package (name "human") (version "0.3") (source (origin (method git-fetch) (uri (git-reference (url "git://git.2f30.org/human.git") (commit version))) (file-name (git-file-name name version)) (sha256 (base32 "0y0bsmvpwfwb2lwspi6a799y34h1faxc6yfanyw6hygxc8661mga")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) ; no configure script (home-page "https://git.2f30.org/human/") (synopsis "Convert bytes to human readable formats") (description "Human is a small program which translate numbers into a human readable format. By default, it tries to detect the best factorisation, but you can force its output. You can adjust the number of decimals with the @code{SCALE} environment variable.") (license license:wtfpl2))) (define-public fortify-headers (package (name "fortify-headers") (version "1.1") (source (origin (method url-fetch) (uri (string-append "https://dl.2f30.org/releases/" "fortify-headers-" version ".tar.gz")) (sha256 (base32 "1dhz41jq1azcf7rbvga8w6pnx19l1j9r6jwj8qrlrfnjl9hdi9bb")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) ; no configure script (home-page "https://git.2f30.org/fortify-headers/") (synopsis "Standalone fortify-source implementation") (description "This is a standalone implementation of fortify source. It provides compile time buffer checks. It is libc-agnostic and simply overlays the system headers by using the @code{#include_next} extension found in GCC. It was initially intended to be used on musl-based Linux distributions. @itemize @item It is portable, works on *BSD, Linux, Solaris and possibly others. @item It will only trap non-conformant programs. This means that fortify level 2 is treated in the same way as level 1. @item Avoids making function calls when undefined behaviour has already been invoked. This is handled by using @code{__builtin_trap()}. @item Support for out-of-bounds read interfaces, such as @code{send()}, @code{write()}, @code{fwrite()}, etc. @item No ABI is enforced. All of the fortify check functions are inlined into the resulting binary. @end itemize\n") (license license:isc))) (define-public colors (package (name "colors") (version "0.3") (source (origin (method url-fetch) (uri (string-append "https://dl.2f30.org/releases/" "colors-" version ".tar.gz")) (sha256 (base32 "1lckmqpgj89841splng0sszbls2ag71ggkgr1wsv9y3v6y87589z")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) ; no configure script (inputs (list libpng)) (home-page "https://git.2f30.org/colors/") (synopsis "Extract colors from pictures") (description "Extract colors from PNG files. It is similar to strings(1) but for pictures. For a given input file it outputs a colormap to stdout.") (license license:isc))) (define-public libgrapheme (package (name "libgrapheme") (version "2.0.2") (source (origin (method url-fetch) (uri (string-append "https://dl.suckless.org/libgrapheme/libgrapheme-" version ".tar.gz")) (sha256 (base32 "099i2jm9c25nkbg5420wr12z0gd189gcw5j1ssjmpmbbwzfvv2x6")))) (build-system gnu-build-system) (arguments (list #:test-target "test" #:phases #~(modify-phases %standard-phases (add-after 'configure 'post-configure (lambda _ ;; Remove ldconfing invocation in Makefile, which ;; is not needed in Guix. (substitute* "config.mk" (("ldconfig") ""))))) #:make-flags #~(list (string-append "CC=" #$(cc-for-target)) (string-append "PREFIX=" #$output)))) (home-page "https://libs.suckless.org/libgrapheme/") (synopsis "C99 library for Unicode strings") (description "Libgrapheme is a simple freestanding C99 library providing utilities to handle strings according to the Unicode standard.") (license license:isc))) ;; No new releases were made at github, this repository is more active than ;; the one at http://git.suckless.org/libutf/ and it is ;; done by the same developer. (define-public libutf (let ((revision "1") (commit "ff4c60635e1f455b0a0b4200f8183fbd5a88225b")) (package (name "libutf") (version (git-version "0.0.0" revision commit)) (source (origin (method git-fetch) (uri (git-reference (url "https://github.com/cls/libutf") (commit commit))) (file-name (git-file-name name version)) (sha256 (base32 "1ih5vjavilzggyr1j1z6w1z12c2fs5fg77cfnv7ami5ivsy3kg3d")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) ; no configure script (inputs (list gawk)) (home-page "https://github.com/cls/libutf") (synopsis "Plan 9 compatible UTF-8 library") (description "This is a C89 UTF-8 library, with an API compatible with that of Plan 9's libutf, but with a number of improvements: @itemize @item Support for runes beyond the Basic Multilingual Plane. @item utflen and utfnlen cannot overflow on 32- or 64-bit machines. @item chartorune treats all invalid codepoints as though Runeerror. @item fullrune, utfecpy, and utfnlen do not overestimate the length of malformed runes. @item An extra function, charntorune(p,s,n), equivalent to fullrune(s,n) ? chartorune(p,s): 0. @item Runeerror may be set to an alternative replacement value, such as -1, to be used instead of U+FFFD. @end itemize\n") (license license:expat)))) ;; No release tarballs so far. (define-public lchat (let ((revision "4") (commit "e3b64e67b9b9d832462382246474ce1e7d92217c")) (package (name "lchat") (version (git-version "0.0.0" revision commit)) (source (origin (method git-fetch) (uri (git-reference (url "https://github.com/younix/lchat") (commit commit))) (file-name (git-file-name name version)) (sha256 (base32 "1qcjqbgmsskc04j2r6xl3amkwj05n520sq1wv2mqyqncz42qrxm0")))) (build-system gnu-build-system) (arguments `(#:test-target "test" #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure) ; no configure script (add-before 'build 'libbsd (lambda _ (substitute* "Makefile" (("-lutf") "-lutf -lbsd")))) (replace 'install (lambda* (#:key outputs #:allow-other-keys) (let* ((out (assoc-ref outputs "out")) (bin (string-append out "/bin")) (man1 (string-append out "/share/man/man1"))) (install-file "lchat" bin) (install-file "lchat.1" man1) #t)))))) (inputs (list grep ncurses libutf libbsd)) (home-page "https://github.com/younix/lchat") (synopsis "Line chat is a frontend for the irc client ii from suckless") (description "Lchat (line chat) is the little and small brother of cii. It is a front end for ii-like chat programs. It uses @code{tail -f} to get the chat output in the background.") (license license:isc)))) (define-public snooze (package (name "snooze") (version "0.5") (source (origin (method git-fetch) (uri (git-reference (url "https://github.com/leahneukirchen/snooze") (commit (string-append "v" version)))) (file-name (git-file-name name version)) (sha256 (base32 "02ng3r1gzgpyjia4b60i11dj5bhn3xjsdcbwmxaam6dzb33dmgib")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; There are no tests. #:make-flags (list (string-append "CC=" ,(cc-for-target)) ;; Set prefix path to root of package path in store instead ;; of /usr/local. (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) (home-page "https://github.com/leahneukirchen/snooze") (synopsis "Run a command at a particular time") (description "@command{snooze} is a tool for waiting until a particular time and then running a command.") (license license:cc0))) (define-public sbase ;; There are no tagged releases. (let ((commit "2c2a7f54ab55a022a617e510b6e00c3e2736fabd") (revision "0")) (package (name "sbase") (version (git-version "0" revision commit)) (source (origin (method git-fetch) (uri (git-reference (url "https://git.suckless.org/sbase/") (commit commit))) (file-name (git-file-name name version)) (sha256 (base32 "119v1lpgsx8bx9h57wg454ddhzz2awqavl3wrn35a704vifg28g0")))) (build-system gnu-build-system) (arguments (list #:tests? #f ;no test suite #:make-flags #~(list (string-append "CC=" #$(cc-for-target)) (string-append "PREFIX=" #$output)) #:phases #~(modify-phases %standard-phases (delete 'configure)))) (home-page "https://core.suckless.org/sbase/") (synopsis "Collection of UNIX tools") (description "@command{sbase} is a collection of UNIX tools similar to those of GNU Coreutils, containing utilities commands such as @command{grep}, @command{cp}, @command{rm}, etc.") (license license:expat)))) (define-public scron (package (name "scron") (version "0.4") (source (origin (method url-fetch) (uri (string-append "https://dl.2f30.org/releases/" "scron-" version ".tar.gz")) (sha256 (base32 "066fwa55kqcgfrsqgxh94sqbkxfsr691360xg4ljxr4i75d25s2a")))) (build-system gnu-build-system) (arguments `(#:tests? #f ; no tests #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) ; no configure script (home-page "https://git.2f30.org/scron/") (synopsis "Simple cron daemon") (description "Schedule commands to be run at specified dates and times. Single daemon and configuration file. Log to stdout or syslog. No mail support.") (license license:expat))) (define-public sfm (package (name "sfm") (version "0.4") (source (origin (method git-fetch) (uri (git-reference (url "git://git.afify.dev/sfm.git") (commit (string-append "v" version)))) (file-name (git-file-name name version)) (sha256 (base32 "0g6k884mggryld0k054sjcj6kpkbca9cvr50w98klszym73yw0sp")))) (build-system gnu-build-system) (arguments `(#:tests? #f ;no check target #:make-flags (list (string-append "CC=" ,(cc-for-target)) (string-append "PREFIX=" %output)) #:phases (modify-phases %standard-phases (delete 'configure)))) ;no configure script (home-page "https://github.com/afify/sfm") (synopsis "Simple file manager") (description "sfm is a simple file manager.") (license license:isc))) (define-public sfeed (package (name "sfeed") (version "2.0") (source (origin (method git-fetch) (uri (git-reference (url "git://git.codemadness.org/sfeed") (commit version))) (file-name (git-file-name name version)) (sha256 (base32 "0zmgkvq75a7h4rmk7izpsaxswlqh9zg446wxv7qcw0bh5xdckg0d")))) (build-system gnu-build-system) (arguments (list #:tests? #f ;no check target #:make-flags #~(list (string-append "CC=" #$(cc-for-target)) (string-append "PREFIX=" #$output)) #:phases '(modify-phases %standard-phases (add-after 'unpack 'fix-ncurses (lambda _ (substitute* "Makefile" (("-lcurses") "-lncurses")))) (delete 'configure)))) ;no configure script (inputs (list ncurses)) (home-page "https://git.codemadness.org/sfeed") (synopsis "RSS and Atom parser") (description "@code{sfeed} converts RSS or Atom feeds from XML to a TAB-separated file. There are formatting programs included to convert this TAB-separated format to various other formats. There are also some programs and scripts included to import and export OPML and to fetch, filter, merge and order feed items.") (license license:isc))) (define-public farbfeld (let ((commit "ab5e3dfc9cdb476218538c6687df9f44826d8f11") (revision "0")) (package (name "farbfeld") (version (git-version "4" revision commit)) (source (origin (method git-fetch) (uri (git-reference (url "git://git.suckless.org/farbfeld") (commit commit))) (file-name (git-file-name name version)) (sha256 (base32 "0pkmkvv5ggpzqwqdchd19442x8gh152xy5z1z13ipfznhspsf870")))) (build-system gnu-build-system) (inputs (list libpng libjpeg-turbo imagemagick)) (arguments (list #:tests? #f ;no check target #:make-flags #~(list (string-append "PREFIX=" #$output) (string-append "CC=" #$(cc-for-target))) #:phases #~(modify-phases %standard-phases (add-before 'configure 'patch-2ff (lambda* (#:key inputs outputs #:allow-other-keys) (substitute* "2ff" (("png2ff") (string-append #$output "/bin/png2ff")) (("jpg2ff") (string-append #$output "/bin/jpg2ff")) (("convert") (search-input-file inputs "/bin/convert"))))) (delete 'configure)))) (synopsis "Image format and conversion tools") (description "farbfeld is a lossless image format which is easy to parse, pipe and compress.") (home-page "https://git.suckless.org/farbfeld/") (license license:isc)))) (define-public snafu (let ((commit "e436fb4f61ca93a4ec85122506b2c2d4fec30eb6") (revision "0")) (package (name "snafu") (version (git-version "0.0.0" revision commit)) (source (origin (method git-fetch) (uri (git-reference (url "https://github.com/jsbmg/snafu") (commit commit))) (file-name (git-file-name name version)) (sha256 (base32 "1c6ahxw0qz0703my28k2z0kgi0am5bp5d02l4rgyphgvjk1jfv8h")))) (arguments `(#:cargo-inputs (("rust-chrono" ,rust-chrono-0.4)) #:tests? #f)) ; There are no tests. (build-system cargo-build-system) (home-page "https://github.com/jsbmg/snafu") (synopsis "Status text for DWM window manager") (description "@code{snafu} provides status text for DWM's builtin bar. It shows battery status, battery capacity, current WiFi connection, and the time in a nice format.") (license license:isc)))) (define-public svkbd (package (name "svkbd") (version "0.4.1") (source (origin (method url-fetch) (uri (string-append "https://dl.suckless.org/tools/svkbd-" version ".tar.gz")) (sha256 (base32 "0nhgmr38pk1a8zrcrxd1ygh0m843a3bdchkv8phl508x7vy63hpv")))) (build-system gnu-build-system) (arguments (list #:tests? #f #:make-flags #~(list (string-append "CC=" #$(cc-for-target)) (string-append "PREFIX=" #$output)) #:phases #~(modify-phases %standard-phases (add-after 'unpack 'patch (lambda* (#:key inputs outputs #:allow-other-keys) (substitute* "config.mk" (("/usr/local") #$output) (("/usr/X11R6") #$libx11) (("/usr/include/freetype2") (string-append #$freetype "/include/freetype2"))))) (delete 'configure)))) ;no configure script (native-inputs (list pkg-config)) (inputs (list freetype libx11 libxft libxtst libxinerama)) (propagated-inputs (list (libc-utf8-locales-for-target))) (home-page "https://tools.suckless.org/x/svkbd/") (synopsis "Virtual on-screen keyboard") (description "svkbd is a simple virtual keyboard, intended to be used in environments, where no keyboard is available.") (license license:expat))) (define-public lib9 ;; no release since 2010 (let ((commit "63916da7bd6d73d9a405ce83fc4ca34845667cce") (revision "0")) (package (name "lib9") (version (git-version "7" revision commit)) (source (origin (method git-fetch) (uri (git-reference (url "https://git.suckless.org/9base/") (commit commit))) (file-name (git-file-name name version)) (sha256 (base32 "04j8js4s3jmlzi3m46q81bq76rn41an58ffhkbj9p5wwr5hvplh8")))) (build-system gnu-build-system) (arguments (list #:tests? #f ;no tests #:phases #~(modify-phases %standard-phases (add-after 'unpack 'patch (lambda _ (substitute* "config.mk" (("^PREFIX.*=.*$") (string-append "PREFIX=\"" #$output "\"\n")) (("^OBJTYPE.*=.*$") #$@(cond ((target-x86-64?) #~(string-append "OBJTYPE=x86_64\n")) ((target-x86?) #~(string-append "OBJTYPE=386\n")) ((target-powerpc?) #~(string-append "OBJTYPE=power\n")) ((target-arm32?) #~(string-append "OBJTYPE=arm\n")))) (("^CFLAGS.*+=.*$") (string-append "CFLAGS+=-O2 -g -I. -fPIC -c -DPLAN9PORT " "-DPREFIX=\\\"" #$output "\\\"\n")) (("^LDFLAGS.*+=.*$") "LDFLAGS+=\n") (("^CC.*=.*$") (string-append "CC=" #$(cc-for-target) "\n"))) (substitute* "lib9/libc.h" (("#define main.*p9main") "")) (substitute* "lib9/main.c" (("p9main\\(argc, argv\\);") "") (("extern void p9main\\(int, char\\*\\*\\);") "")) (substitute* "lib9/Makefile" (("lib9.a") "lib9.so") (("LIB9OFILES=\\\\") "LIB9OFILES=convM2D.o \\") (("@\\$\\{AR\\} \\$\\{LIB\\} \\$\\{OFILES\\}") "gcc -shared ${OFILES} -o lib9.so")))) (add-after 'patch 'chdir (lambda _ (chdir "lib9"))) (delete 'configure)))) ;no configure script (home-page "https://tools.suckless.org/9base/") (supported-systems '("x86_64-linux" "i686-linux" "armhf-linux" "powerpc-linux")) (synopsis "Unix port of Plan 9's formatted I/O C library") (description "This package provides a ported version of the Plan 9 lib9 C library. It also contains the Plan 9 libbio, libregexp, libfmt and libutf libraries.") (license (list license:expat ;modifications license:lpl1.02))))) ;original plan9 code (define-public xssstate (package (name "xssstate") (version "1.1") (source (origin (method url-fetch) (uri (string-append "https://dl.suckless.org/tools/xssstate-" version ".tar.gz")) (sha256 (base32 "04b03jz38pn5qhddg8a9hh01qqzrrdjvsq09qrxj9sx8lq2gbdn4")))) (build-system gnu-build-system) (arguments (list #:tests? #f ; no tests #:make-flags #~(list (string-append "CC=" #$(cc-for-target)) (string-append "PREFIX=" #$output)) #:phases #~(modify-phases %standard-phases (delete 'configure)))) (inputs (list libxscrnsaver)) (home-page "https://tools.suckless.org/x/xssstate/") (synopsis "Simple tool to retrieve the X screensaver state") (description "A utility to retrieve the state of the X screensaver. These states include the idle time, the screensaver state and the length of time until the screensaver should be activated.") (license license:x11))) (define-public 9yacc (package (inherit lib9) (name "9yacc") (arguments (substitute-keyword-arguments (package-arguments lib9) ((#:phases phases) #~(modify-phases #$phases (add-after 'patch 'patch-for-9yacc (lambda _ (substitute* "yacc/yacc.c" (("#9/yacc") (string-append #$output "/lib"))) (substitute* "config.mk" (("^CFLAGS.*+=.*$") (string-append "CFLAGS+=-O2 -g -c -DPLAN9PORT " "-DPREFIX=\\\"" #$output "\\\"\n"))))) (replace 'chdir (lambda _ (chdir "yacc"))) (delete 'install-include) (add-after 'install 'install-yaccpar (lambda _ (install-file "yaccpar" (string-append #$output "/lib")) (install-file "yaccpars" (string-append #$output "/lib")))))))) (inputs (list lib9)) (synopsis "Port of Plan 9's yacc parser generator for Unix") (description "This package provides a ported version of the Plan 9 yacc parser generator."))) (define-public 9base (package (inherit 9yacc) (name "9base") (arguments (substitute-keyword-arguments (package-arguments 9yacc) ((#:phases phases) #~(modify-phases #$phases (add-after 'patch-for-9yacc 'patch-for-9base (lambda _ (substitute* "Makefile" (("SUBDIRS = lib9\\\\") "SUBDIRS = \\") (("@chmod 755 yacc/9yacc") "")) (for-each (lambda (x) (substitute* "Makefile" (((string-append x "\\\\")) "\\"))) '("yacc" "diff" "hoc" "rc")) (substitute* "sam/Makefile" (("\\$\\{CFLAGS\\}") "${CFLAGS} -I.")) (substitute* "config.mk" (("^YACC.*=.*$") (string-append "YACC=" #$(this-package-native-input "9yacc") "/bin/yacc\n"))))) (delete 'chdir) (delete 'install-yaccpar))))) (native-inputs (list 9yacc)) (inputs (list lib9)) (propagated-inputs (list rc)) (synopsis "Port of various Plan 9 tools for Unix") (description "This package provides ported versions of various Plan 9 userland tools for Unix.")))