{ config, pkgs, ... }: with pkgs.lib; let inherit (pkgs) writeText openssh; cfg = config.services.nfsKernel; exports = if builtins.pathExists cfg.exports then cfg.exports else pkgs.writeText "exports" cfg.exports; in { ###### interface options = { services.nfsKernel = { enable = mkOption { default = false; description = '' Whether to enable the kernel's NFS server. ''; }; # !!! Why is this a file? Why not a list of export entries? exports = mkOption { check = v: v != "/etc/exports"; # this won't work description = '' The file listing the directories to be exported. See exports 5 for the format. ''; }; hostName = mkOption { default = null; description = '' Hostname or address on which NFS requests will be accepted. Default is all. See the option in nfsd 8. ''; }; nproc = mkOption { default = 8; description = '' Number of NFS server threads. Defaults to the recommended value of 8. ''; }; createMountPoints = mkOption { default = false; description = "Whether to create the mount points in the exports file at startup time."; }; }; }; ###### implementation config = mkIf config.services.nfsKernel.enable { environment.etc = singleton { source = exports; target = "exports"; }; jobs.nfs_kernel_exports = { name = "nfs-kernel-exports"; description = "Kernel NFS server"; startOn = "network-interfaces/started"; stopOn = "network-interfaces/stop"; preStart = '' export PATH=${pkgs.nfsUtils}/sbin:$PATH mkdir -p /var/lib/nfs ${config.system.sbin.modprobe}/sbin/modprobe nfsd || true ${optionalString cfg.createMountPoints '' # create export directories: # skip comments, take first col which may either be a quoted # "foo bar" or just foo (-> man export) sed '/^#.*/d;s/^"\([^"]*\)".*/\1/;t;s/[ ].*//' ${exports} \ | xargs -d '\n' mkdir -p '' } # exports file is ${exports} # keep this comment so that this job is restarted whenever exports changes! exportfs -ra ''; }; jobs.nfs_kernel_nfsd = { name = "nfs-kernel-nfsd"; description = "Kernel NFS server"; startOn = "nfs-kernel-exports/started"; stopOn = "nfs-kernel-exports/stop"; exec = "${pkgs.nfsUtils}/sbin/rpc.nfsd ${if cfg.hostName != null then "-H ${cfg.hostName}" else ""} ${builtins.toString cfg.nproc}"; }; jobs.nfs_kernel_mountd = { name = "nfs-kernel-mountd"; description = "Kernel NFS server - mount daemon"; startOn = "nfs-kernel-nfsd/started"; stopOn = "nfs-kernel-exports/stop"; exec = "${pkgs.nfsUtils}/sbin/rpc.mountd -F -f ${exports}"; }; jobs.nfs_kernel_statd = { name = "nfs-kernel-statd"; description = "Kernel NFS server - Network Status Monitor"; startOn = "nfs-kernel-nfsd/started"; stopOn = "nfs-kernel-exports/stop"; preStart = '' mkdir -p /var/lib/nfs ''; exec = "${pkgs.nfsUtils}/sbin/rpc.statd -F"; }; }; }