;;; pcmpl-rpm.el --- functions for dealing with rpm completions -*- lexical-binding: t -*-
;; Copyright (C) 1999-2022 Free Software Foundation, Inc.
;; Package: pcomplete
;; This file is part of GNU Emacs.
;; GNU Emacs 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 Emacs 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 Emacs. If not, see .
;;; Commentary:
;; These functions provide completion rules for the `rpm' command and
;; related tools.
;;; Code:
(require 'pcomplete)
(defgroup pcmpl-rpm nil
"Options for rpm completion."
:group 'pcomplete
:prefix "pcmpl-rpm-")
;; rpm -qa can be slow. Adding --nodigest --nosignature is MUCH faster.
(defcustom pcmpl-rpm-query-options
(let (opts)
(with-temp-buffer
(when (ignore-errors (call-process "rpm" nil t nil "--help"))
(if (search-backward "--nodigest " nil 'move)
(setq opts '("--nodigest")))
(goto-char (point-min))
(if (search-forward "--nosignature " nil t)
(push "--nosignature" opts))))
opts)
"String, or list of strings, with extra options for an rpm query command."
:version "24.3"
:type '(choice (const :tag "No options" nil)
(string :tag "Single option")
(repeat :tag "List of options" string)))
(defcustom pcmpl-rpm-cache t
"Whether to cache the list of installed packages."
:version "24.3"
:type 'boolean)
(defconst pcmpl-rpm-cache-stamp-file "/var/lib/rpm/Packages"
"File used to check that the list of installed packages is up-to-date.")
(defvar pcmpl-rpm-cache-time nil
"Time at which the list of installed packages was updated.")
(defvar pcmpl-rpm-packages nil
"List of installed packages.")
;; Functions:
(defun pcmpl-rpm-packages ()
"Return a list of all installed rpm packages."
(if (and pcmpl-rpm-cache
pcmpl-rpm-cache-time
(let ((mtime (file-attribute-modification-time
(file-attributes pcmpl-rpm-cache-stamp-file))))
(and mtime (not (time-less-p pcmpl-rpm-cache-time mtime)))))
pcmpl-rpm-packages
(message "Getting list of installed rpms...")
(setq pcmpl-rpm-cache-time (current-time)
pcmpl-rpm-packages
(split-string (apply #'pcomplete-process-result "rpm"
(append '("-q" "-a")
(if (stringp pcmpl-rpm-query-options)
(list pcmpl-rpm-query-options)
pcmpl-rpm-query-options)))))
(message "Getting list of installed rpms...done")
pcmpl-rpm-packages))
;; Should this use pcmpl-rpm-query-options?
;; I don't think it would speed it up at all (?).
(defun pcmpl-rpm-all-query (flag)
(message "Querying all packages with `%s'..." flag)
(let ((pkgs (pcmpl-rpm-packages))
(provs (list t)))
(while pkgs
(nconc provs (split-string
(pcomplete-process-result
"rpm" "-q" (car pkgs) flag)))
(setq pkgs (cdr pkgs)))
(pcomplete-uniquify-list (cdr provs))))
(defsubst pcmpl-rpm-files ()
(pcomplete-dirs-or-entries "\\.rpm\\'"))
;;;###autoload
(defun pcomplete/rpm ()
"Completion for the `rpm' command."
;; Originally taken from the output of `rpm --help' on a Red Hat 6.1 system.
(let (mode)
(while (<= pcomplete-index pcomplete-last)
(unless mode
(if (pcomplete-match "^--\\(.*\\)" 0)
(pcomplete-here*
'("--addsign"
"--checksig"
"--erase"
"--help"
"--initdb"
"--install"
"--pipe"
"--querytags"
"--rebuild"
"--rebuilddb"
"--recompile"
"--resign"
"--rmsource"
"--setperms"
"--setugids"
"--upgrade"
"--verify"
"--version"))
(pcomplete-opt "vqVyiUebtK")))
; -b
; -t - build package, where is one of:
; p - prep (unpack sources and apply patches)
; l - list check (do some cursory checks on %files)
; c - compile (prep and compile)
; i - install (prep, compile, install)
; b - binary package (prep, compile, install, package)
; a - bin/src package (prep, compile, install, package)
(cond
((or (eq mode 'query)
(pcomplete-match "-[^-]*q"))
(setq mode 'query)
(if (pcomplete-match "^--\\(.*\\)" 0)
(progn
(pcomplete-here*
'("--changelog"
"--dbpath"
"--dump"
"--file"
"--ftpport" ;nyi for the next four
"--ftpproxy"
"--httpport"
"--httpproxy"
"--provides"
"--queryformat"
"--rcfile"
"--requires"
"--root"
"--scripts"
"--triggeredby"
"--whatprovides"
"--whatrequires"))
(cond
((pcomplete-test "--dbpath")
(pcomplete-here* (pcomplete-dirs)))
((pcomplete-test "--queryformat")
(pcomplete-here*))
((pcomplete-test "--rcfile")
(pcomplete-here* (pcomplete-entries)))
((pcomplete-test "--file")
(pcomplete-here* (pcomplete-entries)))
((pcomplete-test "--root")
(pcomplete-here* (pcomplete-dirs)))
((pcomplete-test "--scripts")
(if (pcomplete-match "^--\\(.*\\)" 0)
(pcomplete-here* '("--triggers"))))
((pcomplete-test "--triggeredby")
(pcomplete-here* (pcmpl-rpm-packages)))
((pcomplete-test "--whatprovides")
(pcomplete-here*
(pcmpl-rpm-all-query "--provides")))
((pcomplete-test "--whatrequires")
(pcomplete-here*
(pcmpl-rpm-all-query "--requires")))))
(if (pcomplete-match "^-" 0)
(pcomplete-opt "af.p(pcmpl-rpm-files)ilsdcvR")
(if (pcomplete-test "-[^-]*p" 'first 1)
(pcomplete-here (pcmpl-rpm-files))
(if (pcomplete-test "-[^-]*f" 'first 1)
(pcomplete-here* (pcomplete-entries))
(pcomplete-here (pcmpl-rpm-packages)))))))
((pcomplete-test "--pipe")
(pcomplete-here* (funcall pcomplete-command-completion-function)))
((pcomplete-test "--rmsource")
(pcomplete-here* (pcomplete-entries))
(throw 'pcomplete-completions nil))
((pcomplete-match "\\`--re\\(build\\|compile\\)\\'")
(pcomplete-here (pcmpl-rpm-files))
(throw 'pcomplete-completions nil))
((pcomplete-match "\\`--\\(resign\\|addsign\\)\\'")
(while (pcomplete-here (pcmpl-rpm-files))))
((or (eq mode 'checksig)
(pcomplete-test "--checksig"))
(setq mode 'checksig)
(if (pcomplete-match "^--\\(.*\\)" 0)
(progn
(pcomplete-here*
'("--nopgp"
"--nogpg"
"--nomd5"
"--rcfile"))
(cond
((pcomplete-test "--rcfile")
(pcomplete-here* (pcomplete-entries)))))
(if (pcomplete-match "^-" 0)
(pcomplete-opt "v")
(pcomplete-here (pcmpl-rpm-files)))))
((or (eq mode 'rebuilddb)
(pcomplete-test "--rebuilddb"))
(setq mode 'rebuilddb)
(if (pcomplete-match "^--\\(.*\\)" 0)
(progn
(pcomplete-here*
'("--dbpath"
"--root"
"--rcfile"))
(cond
((pcomplete-test "--dbpath")
(pcomplete-here* (pcomplete-dirs)))
((pcomplete-test "--root")
(pcomplete-here* (pcomplete-dirs)))
((pcomplete-test "--rcfile")
(pcomplete-here* (pcomplete-entries)))))
(if (pcomplete-match "^-" 0)
(pcomplete-opt "v")
(pcomplete-here))))
((memq mode '(install upgrade))
(if (pcomplete-match "^--\\(.*\\)" 0)
(progn
(pcomplete-here*
(append
'("--allfiles"
"--badreloc"
"--dbpath"
"--excludedocs"
"--excludepath"
"--force"
"--hash"
"--ignorearch"
"--ignoreos"
"--ignoresize"
"--includedocs"
"--justdb"
"--nodeps"
"--noorder"
"--noscripts"
"--notriggers")
(if (eq mode 'upgrade)
'("--oldpackage"))
'("--percent"
"--prefix"
"--rcfile"
"--relocate"
"--replacefiles"
"--replacepkgs"
"--root")))
(cond
((pcomplete-test "--dbpath")
(pcomplete-here* (pcomplete-dirs)))
((pcomplete-test "--relocate")
(pcomplete-here*))
((pcomplete-test "--rcfile")
(pcomplete-here* (pcomplete-entries)))
((pcomplete-test "--excludepath")
(pcomplete-here* (pcomplete-entries)))
((pcomplete-test "--root")
(pcomplete-here* (pcomplete-dirs)))
((pcomplete-test "--prefix")
(pcomplete-here* (pcomplete-dirs)))))
(if (pcomplete-match "^-" 0)
(pcomplete-opt "vh")
(pcomplete-here (pcmpl-rpm-files)))))
((or (pcomplete-test "--install")
(pcomplete-match "-[^-]*i"))
(setq mode 'install))
((or (pcomplete-test "--upgrade")
(pcomplete-match "-[^-]*U"))
(setq mode 'upgrade))
((or (eq mode 'erase)
(pcomplete-test "--erase")
(pcomplete-match "-[^-]*e"))
(setq mode 'erase)
(if (pcomplete-match "^--\\(.*\\)" 0)
(progn
(pcomplete-here*
'("--allmatches"
"--dbpath"
"--justdb"
"--nodeps"
"--noorder"
"--noscripts"
"--notriggers"
"--rcfile"
"--root"))
(cond
((pcomplete-test "--dbpath")
(pcomplete-here* (pcomplete-dirs)))
((pcomplete-test "--rcfile")
(pcomplete-here* (pcomplete-entries)))
((pcomplete-test "--root")
(pcomplete-here* (pcomplete-dirs)))))
(if (pcomplete-match "^-" 0)
(pcomplete-opt "v")
(pcomplete-here (pcmpl-rpm-packages)))))
((or (eq mode 'verify)
(pcomplete-test "--verify"))
(setq mode 'verify)
(if (pcomplete-match "^--\\(.*\\)" 0)
(progn
(pcomplete-here*
'("--dbpath"
"--nodeps"
"--nofiles"
"--nomd5"
"--rcfile"
"--root"
"--triggeredby"
"--whatprovides"
"--whatrequires"))
(cond
((pcomplete-test "--dbpath")
(pcomplete-here* (pcomplete-dirs)))
((pcomplete-test "--rcfile")
(pcomplete-here* (pcomplete-entries)))
((pcomplete-test "--root")
(pcomplete-here* (pcomplete-dirs)))
((pcomplete-test "--triggeredby")
(pcomplete-here* (pcmpl-rpm-packages)))
((pcomplete-test "--whatprovides")
(pcomplete-here*
(pcmpl-rpm-all-query "--provides")))
((pcomplete-test "--whatrequires")
(pcomplete-here*
(pcmpl-rpm-all-query "--requires")))))
(if (pcomplete-match "^-" 0)
(pcomplete-opt "af.p(pcmpl-rpm-files)v")
(pcomplete-here (pcmpl-rpm-packages)))))
((or (memq mode '(build test))
(pcomplete-match "\\`-[bt]"))
(setq mode (if (pcomplete-match "\\`-b")
'build
'test))
(if (pcomplete-match "^--\\(.*\\)" 0)
(progn
(pcomplete-here*
'("--buildroot"
"--clean"
"--nobuild"
"--rcfile"
"--rmsource"
"--short-circuit"
"--sign"
"--target"
"--timecheck"))
(cond
((pcomplete-test "--buildroot")
(pcomplete-here* (pcomplete-dirs)))
((pcomplete-test "--rcfile")
(pcomplete-here* (pcomplete-entries)))
((pcomplete-test "--timecheck")
(pcomplete-here*))))
(if (pcomplete-match "^-" 0)
(pcomplete-opt "v")
(pcomplete-here
(pcomplete-dirs-or-entries (if (eq mode 'test)
"\\.tar\\'"
"\\.spec\\'"))))))
(t
(error "You must select a mode: -q, -i, -U, --verify, etc"))))))
;;; DNF
(defvar pcmpl-rpm-dnf-cache-file "/var/cache/dnf/packages.db"
"Location of the DNF cache.")
(defun pcmpl-rpm--dnf-packages (status)
(when (and (file-exists-p pcmpl-rpm-dnf-cache-file)
(executable-find "sqlite3"))
(with-temp-message
"Getting list of packages..."
(process-lines "sqlite3" "-batch" "-init" "/dev/null"
pcmpl-rpm-dnf-cache-file
(pcase-exhaustive status
('available "select pkg from available")
('installed "select pkg from installed")
('not-installed "\
select pkg from available where pkg not in (select pkg from installed)"))))))
;;;###autoload
(defun pcomplete/dnf ()
"Completion for the `dnf' command."
(let ((subcmds (pcomplete-from-help "dnf help"
:margin "^\\(\\)[a-z-]+ "
:argument "[a-z-]+")))
(while (not (member (pcomplete-arg 1) subcmds))
(pcomplete-here (completion-table-merge
subcmds
(pcomplete-from-help "dnf help"))))
(let ((subcmd (pcomplete-arg 1)))
(while (pcase subcmd
((guard (pcomplete-match "\\`-" 0))
(pcomplete-here
(pcomplete-from-help `("dnf" "help" ,subcmd))))
((or "downgrade" "reinstall" "remove")
(pcomplete-here (pcmpl-rpm--dnf-packages 'installed)))
((or "install" "mark" "reinstall" "upgrade")
(pcomplete-here (pcmpl-rpm--dnf-packages 'not-installed)))
((or "builddep" "changelog" "info" "list" "repoquery" "updateinfo")
(pcomplete-here (pcmpl-rpm--dnf-packages 'available))))))))
(provide 'pcmpl-rpm)
;;; pcmpl-rpm.el ends here