;;; GNU Guix --- Functional package management for GNU ;;; Copyright © 2014 Federico Beffa ;;; Copyright © 2014, 2015 Ludovic Courtès ;;; Copyright © 2018 Mark H Weaver ;;; Copyright © 2019, 2020 Hartmut Goebel ;;; Copyright © 2020 Jakub Kądziołka ;;; ;;; 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 . (define-module (guix build qt-build-system) #:use-module ((guix build cmake-build-system) #:prefix cmake:) #:use-module (guix build utils) #:use-module (ice-9 match) #:use-module (ice-9 regex) #:use-module (ice-9 ftw) #:use-module (srfi srfi-1) #:use-module (srfi srfi-26) #:export (%standard-phases qt-build)) ;; Commentary: ;; ;; Builder-side code of the standard Qt build procedure. ;; ;; Code: (define* (check-setup #:rest args) ;; Make Qt render "offscreen". In many cases this allows to run tests ;; without starting a X11 server. (setenv "QT_QPA_PLATFORM" "offscreen") ;; Qt/KDE tests often need dbus (`dbus-launch …`) which is not fully ;; set-up the the build container. (setenv "DBUS_FATAL_WARNINGS" "0") ;; Set here to ease overwriting 'check (even if set there, too) (setenv "CTEST_OUTPUT_ON_FAILURE" "1") #t) ;; NOTE: Apart from standard subdirectories of /share, Qt also provides facilities ;; for per-application data directories, such as /share/quassel. Thus, we include ;; the output directory even if it doesn't contain any of the standard subdirectories. (define (variables-for-wrapping base-directories output-directory) (define (collect-sub-dirs base-directories subdir-spec) (filter-map (lambda (dir) (and (match subdir-spec ((subdir) (directory-exists? (string-append dir subdir))) ((subdir children) (or (or-map (lambda (child) (directory-exists? (string-append dir subdir child))) children) (and (eq? dir output-directory) (directory-exists? (string-append dir subdir)))))) (string-append dir (car subdir-spec)))) base-directories)) (filter (lambda (var-to-wrap) (not (null? (last var-to-wrap)))) (map (match-lambda ((var . subdir-spec) `(,var = ,(collect-sub-dirs base-directories subdir-spec)))) (list ;; these shall match the search-path-specification for Qt and KDE ;; libraries '("XDG_DATA_DIRS" "/share" ("/applications" "/fonts" "/icons" "/mime")) '("XDG_CONFIG_DIRS" "/etc/xdg") '("QT_PLUGIN_PATH" "/lib/qt5/plugins") '("QML2_IMPORT_PATH" "/lib/qt5/qml"))))) (define* (wrap-all-programs #:key inputs outputs (qt-wrap-excluded-inputs '("qttools")) (qt-wrap-excluded-outputs '()) #:allow-other-keys) "Implement phase \"qt-wrap\": look for GSettings schemas and gtk+-v.0 libraries and create wrappers with suitably set environment variables if found. Wrapping is not applied to outputs whose name is listed in QT-WRAP-EXCLUDED-OUTPUTS. This is useful when an output is known not to contain any Qt binaries, and where wrapping would gratuitously add a dependency of that output on Qt." (define (find-files-to-wrap directory) (append-map (lambda (dir) (if (directory-exists? dir) (find-files dir ".*") (list))) (list (string-append directory "/bin") (string-append directory "/sbin") (string-append directory "/libexec") (string-append directory "/lib/libexec")))) (define input-directories (filter-map (match-lambda ((name . dir) (if (member name qt-wrap-excluded-inputs) #f dir))) inputs)) (define handle-output (match-lambda ((output . directory) (unless (member output qt-wrap-excluded-outputs) (let ((bin-list (find-files-to-wrap directory)) (vars-to-wrap (variables-for-wrapping (cons directory input-directories) directory))) (when (not (null? vars-to-wrap)) (for-each (cut apply wrap-program <> vars-to-wrap) bin-list))))))) (for-each handle-output outputs) #t) (define %standard-phases (modify-phases cmake:%standard-phases (add-before 'check 'check-setup check-setup) (add-after 'install 'qt-wrap wrap-all-programs))) (define* (qt-build #:key inputs (phases %standard-phases) #:allow-other-keys #:rest args) "Build the given package, applying all of PHASES in order." (apply cmake:cmake-build #:inputs inputs #:phases phases args))