2018-08-04 16:08:54 +01:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
let
|
|
|
|
|
2021-10-03 17:06:03 +01:00
|
|
|
inherit (lib.options) literalExpression mkEnableOption mkOption;
|
2022-02-05 10:30:50 +00:00
|
|
|
inherit (lib.types) bool enum ints lines attrsOf nonEmptyStr nullOr path str submodule;
|
2018-08-04 16:08:54 +01:00
|
|
|
inherit (lib.modules) mkDefault mkIf mkMerge;
|
|
|
|
|
|
|
|
commonDescr = ''
|
|
|
|
Values can be either strings or integers
|
|
|
|
(which will be added to the config file verbatimly)
|
|
|
|
or lists thereof
|
|
|
|
(which will be translated to multiple
|
|
|
|
lines with the same configuration key).
|
|
|
|
Boolean values are translated to "Yes" or "No".
|
|
|
|
The default contains some reasonable
|
|
|
|
configuration to yield an operational system.
|
|
|
|
'';
|
|
|
|
|
|
|
|
configAttrType =
|
|
|
|
# Options in HylaFAX configuration files can be
|
|
|
|
# booleans, strings, integers, or list thereof
|
|
|
|
# representing multiple config directives with the same key.
|
|
|
|
# This type definition resolves all
|
|
|
|
# those types into a list of strings.
|
|
|
|
let
|
2021-05-08 19:44:32 +01:00
|
|
|
inherit (lib.types) attrsOf coercedTo int listOf;
|
2018-08-04 16:08:54 +01:00
|
|
|
innerType = coercedTo bool (x: if x then "Yes" else "No")
|
|
|
|
(coercedTo int (toString) str);
|
|
|
|
in
|
|
|
|
attrsOf (coercedTo innerType lib.singleton (listOf innerType));
|
|
|
|
|
|
|
|
cfg = config.services.hylafax;
|
|
|
|
|
|
|
|
modemConfigOptions = { name, config, ... }: {
|
|
|
|
options = {
|
|
|
|
name = mkOption {
|
2022-02-05 10:30:50 +00:00
|
|
|
type = nonEmptyStr;
|
2018-08-04 16:08:54 +01:00
|
|
|
example = "ttyS1";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Name of modem device,
|
2022-07-28 22:19:15 +01:00
|
|
|
will be searched for in {file}`/dev`.
|
2018-08-04 16:08:54 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
type = mkOption {
|
2022-02-05 10:30:50 +00:00
|
|
|
type = nonEmptyStr;
|
2018-08-04 16:08:54 +01:00
|
|
|
example = "cirrus";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Name of modem configuration file,
|
2022-07-28 22:19:15 +01:00
|
|
|
will be searched for in {file}`config`
|
2018-08-04 16:08:54 +01:00
|
|
|
in the spooling area directory.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
config = mkOption {
|
|
|
|
type = configAttrType;
|
|
|
|
example = {
|
|
|
|
AreaCode = "49";
|
|
|
|
LocalCode = "30";
|
|
|
|
FAXNumber = "123456";
|
|
|
|
LocalIdentifier = "LostInBerlin";
|
|
|
|
};
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Attribute set of values for the given modem.
|
|
|
|
${commonDescr}
|
|
|
|
Options defined here override options in
|
2022-07-28 22:19:15 +01:00
|
|
|
{option}`commonModemConfig` for this modem.
|
2018-08-04 16:08:54 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
config.name = mkDefault name;
|
|
|
|
config.config.Include = [ "config/${config.type}" ];
|
|
|
|
};
|
|
|
|
|
|
|
|
defaultConfig =
|
|
|
|
let
|
|
|
|
inherit (config.security) wrapperDir;
|
|
|
|
inherit (config.services.mail.sendmailSetuidWrapper) program;
|
|
|
|
mkIfDefault = cond: value: mkIf cond (mkDefault value);
|
|
|
|
noWrapper = config.services.mail.sendmailSetuidWrapper==null;
|
|
|
|
# If a sendmail setuid wrapper exists,
|
|
|
|
# we add the path to the default configuration file.
|
|
|
|
# Otherwise, we use `false` to provoke
|
|
|
|
# an error if hylafax tries to use it.
|
|
|
|
c.sendmailPath = mkMerge [
|
2021-01-24 09:19:10 +00:00
|
|
|
(mkIfDefault noWrapper "${pkgs.coreutils}/bin/false")
|
|
|
|
(mkIfDefault (!noWrapper) "${wrapperDir}/${program}")
|
2018-08-04 16:08:54 +01:00
|
|
|
];
|
|
|
|
importDefaultConfig = file:
|
|
|
|
lib.attrsets.mapAttrs
|
|
|
|
(lib.trivial.const mkDefault)
|
|
|
|
(import file { inherit pkgs; });
|
|
|
|
c.commonModemConfig = importDefaultConfig ./modem-default.nix;
|
|
|
|
c.faxqConfig = importDefaultConfig ./faxq-default.nix;
|
|
|
|
c.hfaxdConfig = importDefaultConfig ./hfaxd-default.nix;
|
|
|
|
in
|
|
|
|
c;
|
|
|
|
|
|
|
|
localConfig =
|
|
|
|
let
|
|
|
|
c.hfaxdConfig.UserAccessFile = cfg.userAccessFile;
|
|
|
|
c.faxqConfig = lib.attrsets.mapAttrs
|
|
|
|
(lib.trivial.const (v: mkIf (v!=null) v))
|
|
|
|
{
|
|
|
|
AreaCode = cfg.areaCode;
|
|
|
|
CountryCode = cfg.countryCode;
|
|
|
|
LongDistancePrefix = cfg.longDistancePrefix;
|
|
|
|
InternationalPrefix = cfg.internationalPrefix;
|
|
|
|
};
|
|
|
|
c.commonModemConfig = c.faxqConfig;
|
|
|
|
in
|
|
|
|
c;
|
|
|
|
|
|
|
|
in
|
|
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
options.services.hylafax = {
|
|
|
|
|
2022-08-28 20:18:44 +01:00
|
|
|
enable = mkEnableOption (lib.mdDoc "HylaFAX server");
|
2018-08-04 16:08:54 +01:00
|
|
|
|
|
|
|
autostart = mkOption {
|
|
|
|
type = bool;
|
|
|
|
default = true;
|
|
|
|
example = false;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Autostart the HylaFAX queue manager at system start.
|
2022-07-28 22:19:15 +01:00
|
|
|
If this is `false`, the queue manager
|
2018-08-04 16:08:54 +01:00
|
|
|
will still be started if there are pending
|
|
|
|
jobs or if a user tries to connect to it.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
countryCode = mkOption {
|
2022-02-05 10:30:50 +00:00
|
|
|
type = nullOr nonEmptyStr;
|
2018-08-04 16:08:54 +01:00
|
|
|
default = null;
|
|
|
|
example = "49";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Country code for server and all modems.";
|
2018-08-04 16:08:54 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
areaCode = mkOption {
|
2022-02-05 10:30:50 +00:00
|
|
|
type = nullOr nonEmptyStr;
|
2018-08-04 16:08:54 +01:00
|
|
|
default = null;
|
|
|
|
example = "30";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Area code for server and all modems.";
|
2018-08-04 16:08:54 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
longDistancePrefix = mkOption {
|
|
|
|
type = nullOr str;
|
|
|
|
default = null;
|
|
|
|
example = "0";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Long distance prefix for server and all modems.";
|
2018-08-04 16:08:54 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
internationalPrefix = mkOption {
|
|
|
|
type = nullOr str;
|
|
|
|
default = null;
|
|
|
|
example = "00";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "International prefix for server and all modems.";
|
2018-08-04 16:08:54 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
spoolAreaPath = mkOption {
|
|
|
|
type = path;
|
|
|
|
default = "/var/spool/fax";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
The spooling area will be created/maintained
|
|
|
|
at the location given here.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
userAccessFile = mkOption {
|
|
|
|
type = path;
|
|
|
|
default = "/etc/hosts.hfaxd";
|
|
|
|
description = ''
|
|
|
|
The <filename>hosts.hfaxd</filename>
|
|
|
|
file entry in the spooling area
|
|
|
|
will be symlinked to the location given here.
|
|
|
|
This file must exist and be
|
|
|
|
readable only by the <literal>uucp</literal> user.
|
|
|
|
See hosts.hfaxd(5) for details.
|
|
|
|
This configuration permits access for all users:
|
2022-08-30 12:08:40 +01:00
|
|
|
<programlisting>
|
2018-08-04 16:08:54 +01:00
|
|
|
environment.etc."hosts.hfaxd" = {
|
|
|
|
mode = "0600";
|
|
|
|
user = "uucp";
|
|
|
|
text = ".*";
|
|
|
|
};
|
2022-08-30 12:08:40 +01:00
|
|
|
</programlisting>
|
2018-08-04 16:08:54 +01:00
|
|
|
Note that host-based access can be controlled with
|
|
|
|
<option>config.systemd.sockets.hylafax-hfaxd.listenStreams</option>;
|
|
|
|
by default, only 127.0.0.1 is permitted to connect.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
sendmailPath = mkOption {
|
|
|
|
type = path;
|
2021-10-03 17:06:03 +01:00
|
|
|
example = literalExpression ''"''${pkgs.postfix}/bin/sendmail"'';
|
2018-08-04 16:08:54 +01:00
|
|
|
# '' ; # fix vim
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
|
|
|
Path to {file}`sendmail` program.
|
2018-08-04 16:08:54 +01:00
|
|
|
The default uses the local sendmail wrapper
|
2022-07-28 22:19:15 +01:00
|
|
|
(see {option}`config.services.mail.sendmailSetuidWrapper`),
|
|
|
|
otherwise the {file}`false`
|
2018-08-04 16:08:54 +01:00
|
|
|
binary to cause an error if used.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
hfaxdConfig = mkOption {
|
|
|
|
type = configAttrType;
|
|
|
|
example.RecvqProtection = "0400";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Attribute set of lines for the global
|
2022-07-28 22:19:15 +01:00
|
|
|
hfaxd config file {file}`etc/hfaxd.conf`.
|
2018-08-04 16:08:54 +01:00
|
|
|
${commonDescr}
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
faxqConfig = mkOption {
|
|
|
|
type = configAttrType;
|
|
|
|
example = {
|
|
|
|
InternationalPrefix = "00";
|
|
|
|
LongDistancePrefix = "0";
|
|
|
|
};
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Attribute set of lines for the global
|
2022-07-28 22:19:15 +01:00
|
|
|
faxq config file {file}`etc/config`.
|
2018-08-04 16:08:54 +01:00
|
|
|
${commonDescr}
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
commonModemConfig = mkOption {
|
|
|
|
type = configAttrType;
|
|
|
|
example = {
|
|
|
|
InternationalPrefix = "00";
|
|
|
|
LongDistancePrefix = "0";
|
|
|
|
};
|
2022-08-15 06:16:25 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Attribute set of default values for
|
2022-08-15 06:16:25 +01:00
|
|
|
modem config files {file}`etc/config.*`.
|
2018-08-04 16:08:54 +01:00
|
|
|
${commonDescr}
|
|
|
|
Think twice before changing
|
|
|
|
paths of fax-processing scripts.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
modems = mkOption {
|
2020-08-23 00:28:45 +01:00
|
|
|
type = attrsOf (submodule [ modemConfigOptions ]);
|
2018-08-04 16:08:54 +01:00
|
|
|
default = {};
|
|
|
|
example.ttyS1 = {
|
|
|
|
type = "cirrus";
|
|
|
|
config = {
|
|
|
|
FAXNumber = "123456";
|
|
|
|
LocalIdentifier = "Smith";
|
|
|
|
};
|
|
|
|
};
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Description of installed modems.
|
|
|
|
At least on modem must be defined
|
|
|
|
to enable the HylaFAX server.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
spoolExtraInit = mkOption {
|
|
|
|
type = lines;
|
|
|
|
default = "";
|
2021-01-24 09:19:10 +00:00
|
|
|
example = "chmod 0755 . # everyone may read my faxes";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Additional shell code that is executed within the
|
|
|
|
spooling area directory right after its setup.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2022-08-28 20:18:44 +01:00
|
|
|
faxcron.enable.spoolInit = mkEnableOption (lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Purge old files from the spooling area with
|
2022-08-28 20:18:44 +01:00
|
|
|
{file}`faxcron`
|
2018-08-04 16:08:54 +01:00
|
|
|
each time the spooling area is initialized.
|
2022-08-28 20:18:44 +01:00
|
|
|
'');
|
2018-08-04 16:08:54 +01:00
|
|
|
faxcron.enable.frequency = mkOption {
|
2022-02-05 10:30:50 +00:00
|
|
|
type = nullOr nonEmptyStr;
|
2018-08-04 16:08:54 +01:00
|
|
|
default = null;
|
|
|
|
example = "daily";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Purge old files from the spooling area with
|
2022-07-28 22:19:15 +01:00
|
|
|
{file}`faxcron` with the given frequency
|
2018-08-04 16:08:54 +01:00
|
|
|
(see systemd.time(7)).
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
faxcron.infoDays = mkOption {
|
2021-05-08 19:44:32 +01:00
|
|
|
type = ints.positive;
|
2018-08-04 16:08:54 +01:00
|
|
|
default = 30;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Set the expiration time for data in the
|
|
|
|
remote machine information directory in days.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
faxcron.logDays = mkOption {
|
2021-05-08 19:44:32 +01:00
|
|
|
type = ints.positive;
|
2018-08-04 16:08:54 +01:00
|
|
|
default = 30;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Set the expiration time for
|
|
|
|
session trace log files in days.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
faxcron.rcvDays = mkOption {
|
2021-05-08 19:44:32 +01:00
|
|
|
type = ints.positive;
|
2018-08-04 16:08:54 +01:00
|
|
|
default = 7;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Set the expiration time for files in
|
|
|
|
the received facsimile queue in days.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2022-08-28 20:18:44 +01:00
|
|
|
faxqclean.enable.spoolInit = mkEnableOption (lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Purge old files from the spooling area with
|
2022-08-28 20:18:44 +01:00
|
|
|
{file}`faxqclean`
|
2018-08-04 16:08:54 +01:00
|
|
|
each time the spooling area is initialized.
|
2022-08-28 20:18:44 +01:00
|
|
|
'');
|
2018-08-04 16:08:54 +01:00
|
|
|
faxqclean.enable.frequency = mkOption {
|
2022-02-05 10:30:50 +00:00
|
|
|
type = nullOr nonEmptyStr;
|
2018-08-04 16:08:54 +01:00
|
|
|
default = null;
|
|
|
|
example = "daily";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Purge old files from the spooling area with
|
2022-07-28 22:19:15 +01:00
|
|
|
{file}`faxcron` with the given frequency
|
2018-08-04 16:08:54 +01:00
|
|
|
(see systemd.time(7)).
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
faxqclean.archiving = mkOption {
|
|
|
|
type = enum [ "never" "as-flagged" "always" ];
|
|
|
|
default = "as-flagged";
|
|
|
|
example = "always";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Enable or suppress job archiving:
|
2022-07-28 22:19:15 +01:00
|
|
|
`never` disables job archiving,
|
|
|
|
`as-flagged` archives jobs that
|
2018-08-04 16:08:54 +01:00
|
|
|
have been flagged for archiving by sendfax,
|
2022-07-28 22:19:15 +01:00
|
|
|
`always` forces archiving of all jobs.
|
2018-08-04 16:08:54 +01:00
|
|
|
See also sendfax(1) and faxqclean(8).
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
faxqclean.doneqMinutes = mkOption {
|
2021-05-08 19:44:32 +01:00
|
|
|
type = ints.positive;
|
2018-08-04 16:08:54 +01:00
|
|
|
default = 15;
|
2021-10-03 17:06:03 +01:00
|
|
|
example = literalExpression "24*60";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Set the job
|
|
|
|
age threshold (in minutes) that controls how long
|
|
|
|
jobs may reside in the doneq directory.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
faxqclean.docqMinutes = mkOption {
|
2021-05-08 19:44:32 +01:00
|
|
|
type = ints.positive;
|
2018-08-04 16:08:54 +01:00
|
|
|
default = 60;
|
2021-10-03 17:06:03 +01:00
|
|
|
example = literalExpression "24*60";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2018-08-04 16:08:54 +01:00
|
|
|
Set the document
|
|
|
|
age threshold (in minutes) that controls how long
|
|
|
|
unreferenced files may reside in the docq directory.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
config.services.hylafax =
|
|
|
|
mkIf
|
|
|
|
(config.services.hylafax.enable)
|
|
|
|
(mkMerge [ defaultConfig localConfig ])
|
|
|
|
;
|
|
|
|
|
|
|
|
}
|