describe.scm 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. ;;; GNU Guix --- Functional package management for GNU
  2. ;;; Copyright © 2018, 2019, 2020, 2021 Ludovic Courtès <ludo@gnu.org>
  3. ;;;
  4. ;;; This file is part of GNU Guix.
  5. ;;;
  6. ;;; GNU Guix is free software; you can redistribute it and/or modify it
  7. ;;; under the terms of the GNU General Public License as published by
  8. ;;; the Free Software Foundation; either version 3 of the License, or (at
  9. ;;; your option) any later version.
  10. ;;;
  11. ;;; GNU Guix is distributed in the hope that it will be useful, but
  12. ;;; WITHOUT ANY WARRANTY; without even the implied warranty of
  13. ;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. ;;; GNU General Public License for more details.
  15. ;;;
  16. ;;; You should have received a copy of the GNU General Public License
  17. ;;; along with GNU Guix. If not, see <http://www.gnu.org/licenses/>.
  18. (define-module (guix describe)
  19. #:use-module (guix memoization)
  20. #:use-module (guix profiles)
  21. #:use-module (guix packages)
  22. #:use-module ((guix utils) #:select (location-file))
  23. #:use-module ((guix store) #:select (%store-prefix store-path?))
  24. #:use-module ((guix config) #:select (%state-directory))
  25. #:autoload (guix channels) (channel-name
  26. sexp->channel
  27. manifest-entry-channel)
  28. #:use-module (srfi srfi-1)
  29. #:use-module (ice-9 match)
  30. #:export (current-profile
  31. current-profile-date
  32. current-profile-entries
  33. current-channels
  34. package-path-entries
  35. package-provenance
  36. package-channels
  37. manifest-entry-with-provenance
  38. manifest-entry-provenance))
  39. ;;; Commentary:
  40. ;;;
  41. ;;; This module provides supporting code to allow a Guix instance to find, at
  42. ;;; run time, which profile it's in (profiles created by 'guix pull'). That
  43. ;;; allows it to read meta-information about itself (e.g., repository URL and
  44. ;;; commit ID) and to find other channels available in the same profile. It's
  45. ;;; a bit like ELPA's pkg-info.el.
  46. ;;;
  47. ;;; Code:
  48. (define initial-program-arguments
  49. ;; Save the initial program arguments. This allows us to see the "real"
  50. ;; 'guix' program, even if 'guix repl -s' calls 'set-program-arguments'
  51. ;; later on.
  52. (program-arguments))
  53. (define current-profile
  54. (mlambda ()
  55. "Return the profile (created by 'guix pull') the calling process lives in,
  56. or #f if this is not applicable."
  57. (match initial-program-arguments
  58. ((program . _)
  59. (and (string-suffix? "/bin/guix" program)
  60. ;; Note: We want to do _lexical dot-dot resolution_. Using ".."
  61. ;; for real would instead take us into the /gnu/store directory
  62. ;; that ~/.config/guix/current/bin points to, whereas we want to
  63. ;; obtain ~/.config/guix/current.
  64. (let ((candidate (dirname (dirname program))))
  65. (and (file-exists? (string-append candidate "/manifest"))
  66. candidate)))))))
  67. (define (current-profile-date)
  68. "Return the creation date of the current profile (produced by 'guix pull'),
  69. as a number of seconds since the Epoch, or #f if it could not be determined."
  70. ;; Normally 'current-profile' will return ~/.config/guix/current. We need
  71. ;; to 'readlink' once to get '/var/guix/…/guix-profile', whose mtime is the
  72. ;; piece of information we're looking for.
  73. (let loop ((profile (current-profile)))
  74. (match profile
  75. (#f #f)
  76. ((? store-path?) #f)
  77. (file
  78. (if (string-prefix? %state-directory file)
  79. (and=> (lstat file) stat:mtime)
  80. (catch 'system-error
  81. (lambda ()
  82. (let ((target (readlink file)))
  83. (loop (if (string-prefix? "/" target)
  84. target
  85. (string-append (dirname file) "/" target)))))
  86. (const #f)))))))
  87. (define (channel-metadata)
  88. "Return the 'guix' channel metadata sexp from (guix config) if available;
  89. otherwise return #f."
  90. ;; Older 'build-self.scm' would create a (guix config) file without the
  91. ;; '%channel-metadata' variable. Thus, properly deal with a lack of
  92. ;; information.
  93. (let ((module (resolve-interface '(guix config))))
  94. (and=> (module-variable module '%channel-metadata) variable-ref)))
  95. (define current-profile-entries
  96. (mlambda ()
  97. "Return the list of entries in the 'guix pull' profile the calling process
  98. lives in, or the empty list if this is not applicable."
  99. (match (current-profile)
  100. (#f '())
  101. (profile
  102. (let ((manifest (profile-manifest profile)))
  103. (manifest-entries manifest))))))
  104. (define current-channel-entries
  105. (mlambda ()
  106. "Return manifest entries corresponding to extra channels--i.e., not the
  107. 'guix' channel."
  108. (remove (lambda (entry)
  109. (string=? (manifest-entry-name entry) "guix"))
  110. (current-profile-entries))))
  111. (define current-channels
  112. (mlambda ()
  113. "Return the list of channels currently available, including the 'guix'
  114. channel. Return the empty list if this information is missing."
  115. (match (current-profile-entries)
  116. (()
  117. ;; As a fallback, if we're not running from a profile, use 'guix'
  118. ;; channel metadata from (guix config).
  119. (match (channel-metadata)
  120. (#f '())
  121. (sexp (or (and=> (sexp->channel sexp 'guix) list) '()))))
  122. (entries
  123. (filter-map manifest-entry-channel entries)))))
  124. (define (package-path-entries)
  125. "Return two values: the list of package path entries to be added to the
  126. package search path, and the list to be added to %LOAD-COMPILED-PATH. These
  127. entries are taken from the 'guix pull' profile the calling process lives in,
  128. when applicable."
  129. ;; Filter out Guix itself.
  130. (unzip2 (map (lambda (entry)
  131. (list (string-append (manifest-entry-item entry)
  132. "/share/guile/site/"
  133. (effective-version))
  134. (string-append (manifest-entry-item entry)
  135. "/lib/guile/" (effective-version)
  136. "/site-ccache")))
  137. (current-channel-entries))))
  138. (define (package-channels package)
  139. "Return the list of channels providing PACKAGE or an empty list if it could
  140. not be determined."
  141. (match (and=> (package-location package) location-file)
  142. (#f '())
  143. (file
  144. (let ((file (if (string-prefix? "/" file)
  145. file
  146. (search-path %load-path file))))
  147. (if (and file
  148. (string-prefix? (%store-prefix) file))
  149. (filter-map
  150. (lambda (entry)
  151. (let ((item (manifest-entry-item entry)))
  152. (and (or (string-prefix? item file)
  153. (string=? "guix" (manifest-entry-name entry)))
  154. (manifest-entry-channel entry))))
  155. (current-profile-entries))
  156. '())))))
  157. (define (package-provenance package)
  158. "Return the provenance of PACKAGE as an sexp for use as the 'provenance'
  159. property of manifest entries, or #f if it could not be determined."
  160. (define (entry-source entry)
  161. (match (assq 'source
  162. (manifest-entry-properties entry))
  163. (('source value) value)
  164. (_ #f)))
  165. (let* ((channels (package-channels package))
  166. (names (map (compose symbol->string channel-name) channels)))
  167. ;; Always store information about the 'guix' channel and
  168. ;; optionally about the specific channel FILE comes from.
  169. (or (let ((main (and=> (find (lambda (entry)
  170. (string=? "guix"
  171. (manifest-entry-name entry)))
  172. (current-profile-entries))
  173. entry-source))
  174. (extra (any (lambda (entry)
  175. (let ((item (manifest-entry-item entry))
  176. (name (manifest-entry-name entry)))
  177. (and (member name names)
  178. (not (string=? name "guix"))
  179. (entry-source entry))))
  180. (current-profile-entries))))
  181. (and main
  182. `(,main
  183. ,@(if extra (list extra) '())))))))
  184. (define (manifest-entry-with-provenance entry)
  185. "Return ENTRY with an additional 'provenance' property if it's not already
  186. there."
  187. (let ((properties (manifest-entry-properties entry)))
  188. (if (assq 'provenance properties)
  189. entry
  190. (let ((item (manifest-entry-item entry)))
  191. (manifest-entry
  192. (inherit entry)
  193. (properties
  194. (match (and (package? item) (package-provenance item))
  195. (#f properties)
  196. (sexp `((provenance ,@sexp)
  197. ,@properties)))))))))
  198. (define (manifest-entry-provenance entry)
  199. "Return the list of channels ENTRY comes from. Return the empty list if
  200. that information is missing."
  201. (match (assq-ref (manifest-entry-properties entry) 'provenance)
  202. ((main extras ...)
  203. ;; XXX: Until recently, channel sexps lacked the channel name. For
  204. ;; entries created by 'manifest-entry-with-provenance', the first sexp
  205. ;; is known to be the 'guix channel, and for the other ones, invent a
  206. ;; fallback name (it's OK as the name is just a "pet name").
  207. (match (sexp->channel main 'guix)
  208. (#f '())
  209. (channel
  210. (let loop ((extras extras)
  211. (counter 1)
  212. (channels (list channel)))
  213. (match extras
  214. (()
  215. (reverse channels))
  216. ((head . tail)
  217. (let* ((name (string->symbol
  218. (format #f "channel~a" counter)))
  219. (extra (sexp->channel head name)))
  220. (if extra
  221. (loop tail (+ 1 counter) (cons extra channels))
  222. (loop tail counter channels)))))))))
  223. (_
  224. '())))