;;; GNU Guix --- Functional package management for GNU ;;; Copyright © 2013, 2014 Ludovic Courtès ;;; ;;; 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 (gnu system shadow) #:use-module (guix store) #:use-module (guix records) #:use-module (guix packages) #:use-module (guix gexp) #:use-module (guix monads) #:use-module ((gnu packages admin) #:select (shadow)) #:use-module (gnu packages bash) #:use-module (srfi srfi-1) #:use-module (ice-9 match) #:export (user-account user-account? user-account-name user-account-pass user-account-uid user-account-gid user-account-comment user-account-home-directory user-account-shell user-group user-group? user-group-name user-group-password user-group-id user-group-members passwd-file group-file guix-build-accounts)) ;;; Commentary: ;;; ;;; Utilities for configuring the Shadow tool suite ('login', 'passwd', etc.) ;;; ;;; Code: (define-record-type* user-account make-user-account user-account? (name user-account-name) (password user-account-pass (default "")) (uid user-account-uid) (gid user-account-gid) (comment user-account-comment (default "")) (home-directory user-account-home-directory) (shell user-account-shell ; gexp (default #~(string-append #$bash "/bin/bash")))) (define-record-type* user-group make-user-group user-group? (name user-group-name) (password user-group-password (default #f)) (id user-group-id) (members user-group-members (default '()))) (define (group-file groups) "Return a /etc/group file for GROUPS, a list of objects." (define contents (let loop ((groups groups) (result '())) (match groups ((($ name _ gid (users ...)) rest ...) ;; XXX: Ignore the group password. (loop rest (cons (string-append name "::" (number->string gid) ":" (string-join users ",")) result))) (() (string-join (reverse result) "\n" 'suffix))))) (text-file "group" contents)) (define* (passwd-file accounts #:key shadow?) "Return a password file for ACCOUNTS, a list of objects. If SHADOW? is true, then it is a /etc/shadow file, otherwise it is a /etc/passwd file." ;; XXX: The resulting file is world-readable, so beware when SHADOW? is #t! (define account-exp (match-lambda (($ name pass uid gid comment home-dir shell) (if shadow? ; XXX: use (crypt PASS …)? #~(format #t "~a::::::::~%" #$name) #~(format #t "~a:x:~a:~a:~a:~a:~a~%" #$name #$(number->string uid) #$(number->string gid) #$comment #$home-dir #$shell))))) (define builder #~(begin (with-output-to-file #$output (lambda () #$@(map account-exp accounts) #t)))) (gexp->derivation (if shadow? "shadow" "passwd") builder)) ;;; shadow.scm ends here