mujmap: add module
mujmap is a tool that synchronizes mail between a mail server and notmuch via JMAP. It's very similar to lieer, so I heavily based the implementation of the notmuch module on lieer's. I did not include an equivalent to lieer's periodic synchronization service, however, because I plan to soon introduce a daemon mode to mujmap. https://github.com/elizagamedev/mujmap
This commit is contained in:
parent
467617947d
commit
d059b9448a
9 changed files with 397 additions and 0 deletions
3
.github/CODEOWNERS
vendored
3
.github/CODEOWNERS
vendored
|
@ -163,6 +163,9 @@
|
|||
|
||||
/modules/programs/mu.nix @KarlJoad
|
||||
|
||||
/modules/programs/mujmap.nix @elizagamedev
|
||||
/tests/modules/programs/mujmap @elizagamedev
|
||||
|
||||
/modules/programs/navi.nix @marsam
|
||||
|
||||
/modules/programs/ncmpcpp.nix @olmokramer
|
||||
|
|
|
@ -554,6 +554,13 @@ in
|
|||
A new module is available: 'services.mopidy'.
|
||||
'';
|
||||
}
|
||||
|
||||
{
|
||||
time = "2022-06-21T22:29:37+00:00";
|
||||
message = ''
|
||||
A new module is available: 'programs.mujmap'.
|
||||
'';
|
||||
}
|
||||
];
|
||||
};
|
||||
}
|
||||
|
|
|
@ -109,6 +109,7 @@ let
|
|||
./programs/mpv.nix
|
||||
./programs/msmtp.nix
|
||||
./programs/mu.nix
|
||||
./programs/mujmap.nix
|
||||
./programs/navi.nix
|
||||
./programs/ncmpcpp.nix
|
||||
./programs/ncspot.nix
|
||||
|
|
315
modules/programs/mujmap.nix
Normal file
315
modules/programs/mujmap.nix
Normal file
|
@ -0,0 +1,315 @@
|
|||
{ config, lib, pkgs, ... }:
|
||||
|
||||
with lib;
|
||||
|
||||
let
|
||||
cfg = config.programs.mujmap;
|
||||
|
||||
mujmapAccounts =
|
||||
filter (a: a.mujmap.enable) (attrValues config.accounts.email.accounts);
|
||||
|
||||
missingNotmuchAccounts = map (a: a.name)
|
||||
(filter (a: !a.notmuch.enable && a.mujmap.notmuchSetupWarning)
|
||||
mujmapAccounts);
|
||||
|
||||
notmuchConfigHelp =
|
||||
map (name: "accounts.email.accounts.${name}.notmuch.enable = true;")
|
||||
missingNotmuchAccounts;
|
||||
|
||||
settingsFormat = pkgs.formats.toml { };
|
||||
|
||||
filterNull = attrs: attrsets.filterAttrs (n: v: v != null) attrs;
|
||||
|
||||
configFile = account:
|
||||
let
|
||||
settings'' = if (account.jmap == null) then
|
||||
{ }
|
||||
else
|
||||
filterNull {
|
||||
fqdn = account.jmap.host;
|
||||
session_url = account.jmap.sessionUrl;
|
||||
};
|
||||
|
||||
settings' = settings'' // {
|
||||
username = account.userName;
|
||||
password_command = escapeShellArgs account.passwordCommand;
|
||||
} // filterNull account.mujmap.settings;
|
||||
|
||||
settings = if (hasAttr "fqdn" settings') then
|
||||
(removeAttrs settings' [ "session_url" ])
|
||||
else
|
||||
settings';
|
||||
in {
|
||||
name = "${account.maildir.absPath}/mujmap.toml";
|
||||
value.source = settingsFormat.generate
|
||||
"mujmap-${lib.replaceStrings [ "@" ] [ "_at_" ] account.address}.toml"
|
||||
settings;
|
||||
};
|
||||
|
||||
tagsOpts = {
|
||||
lowercase = mkOption {
|
||||
type = types.bool;
|
||||
default = false;
|
||||
description = ''
|
||||
If true, translate all mailboxes to lowercase names when mapping to notmuch
|
||||
tags.
|
||||
'';
|
||||
};
|
||||
|
||||
directory_separator = mkOption {
|
||||
type = types.str;
|
||||
default = "/";
|
||||
example = ".";
|
||||
description = ''
|
||||
Directory separator for mapping notmuch tags to maildirs.
|
||||
'';
|
||||
};
|
||||
|
||||
inbox = mkOption {
|
||||
type = types.str;
|
||||
default = "inbox";
|
||||
description = ''
|
||||
Tag for notmuch to use for messages stored in the mailbox labeled with the
|
||||
<code>Inbox</code> name attribute.
|
||||
</para><para>
|
||||
If set to an empty string, this mailbox <emphasis>and its child
|
||||
mailboxes</emphasis> are not synchronized with a tag.
|
||||
'';
|
||||
};
|
||||
|
||||
deleted = mkOption {
|
||||
type = types.str;
|
||||
default = "deleted";
|
||||
description = ''
|
||||
Tag for notmuch to use for messages stored in the mailbox labeled with the
|
||||
<code>Trash</code> name attribute.
|
||||
</para><para>
|
||||
If set to an empty string, this mailbox <emphasis>and its child
|
||||
mailboxes</emphasis> are not synchronized with a tag.
|
||||
'';
|
||||
};
|
||||
|
||||
sent = mkOption {
|
||||
type = types.str;
|
||||
default = "sent";
|
||||
description = ''
|
||||
Tag for notmuch to use for messages stored in the mailbox labeled with the
|
||||
<code>Sent</code> name attribute.
|
||||
</para><para>
|
||||
If set to an empty string, this mailbox <emphasis>and its child
|
||||
mailboxes</emphasis> are not synchronized with a tag.
|
||||
'';
|
||||
};
|
||||
|
||||
spam = mkOption {
|
||||
type = types.str;
|
||||
default = "spam";
|
||||
description = ''
|
||||
Tag for notmuch to use for messages stored in the mailbox labeled with the
|
||||
<code>Junk</code> name attribute and/or with the <code>$Junk</code> keyword,
|
||||
<emphasis>except</emphasis> for messages with the <code>$NotJunk</code> keyword.
|
||||
</para><para>
|
||||
If set to an empty string, this mailbox, <emphasis>its child
|
||||
mailboxes</emphasis>, and these keywords are not synchronized with a tag.
|
||||
'';
|
||||
};
|
||||
|
||||
important = mkOption {
|
||||
type = types.str;
|
||||
default = "important";
|
||||
description = ''
|
||||
Tag for notmuch to use for messages stored in the mailbox labeled with the
|
||||
<code>Important</code> name attribute and/or with the <code>$Important</code>
|
||||
keyword.
|
||||
</para><para>
|
||||
If set to an empty string, this mailbox, <emphasis>its child
|
||||
mailboxes</emphasis>, and these keywords are not synchronized with a tag.
|
||||
'';
|
||||
};
|
||||
|
||||
phishing = mkOption {
|
||||
type = types.str;
|
||||
default = "phishing";
|
||||
description = ''
|
||||
Tag for notmuch to use for the IANA <code>$Phishing</code> keyword.
|
||||
</para><para>
|
||||
If set to an empty string, this keyword is not synchronized with a tag.
|
||||
'';
|
||||
};
|
||||
};
|
||||
|
||||
rootOpts = {
|
||||
username = mkOption {
|
||||
type = types.nullOr types.str;
|
||||
default = null;
|
||||
example = "alice@example.com";
|
||||
description = ''
|
||||
Username for basic HTTP authentication.
|
||||
</para><para>
|
||||
If <literal>null</literal>, defaults to
|
||||
<xref linkend="opt-accounts.email.accounts._name_.userName"/>.
|
||||
'';
|
||||
};
|
||||
|
||||
password_command = mkOption {
|
||||
type = types.nullOr (types.either types.str (types.listOf types.str));
|
||||
default = null;
|
||||
apply = p: if isList p then escapeShellArgs p else p;
|
||||
example = "pass alice@example.com";
|
||||
description = ''
|
||||
Shell command which will print a password to stdout for basic HTTP
|
||||
authentication.
|
||||
</para><para>
|
||||
If <literal>null</literal>, defaults to
|
||||
<xref linkend="opt-accounts.email.accounts._name_.passwordCommand"/>.
|
||||
'';
|
||||
};
|
||||
|
||||
fqdn = mkOption {
|
||||
type = types.nullOr types.str;
|
||||
default = null;
|
||||
example = "example.com";
|
||||
description = ''
|
||||
Fully qualified domain name of the JMAP service.
|
||||
</para><para>
|
||||
mujmap looks up the JMAP SRV record for this host to determine the JMAP session
|
||||
URL. Mutually exclusive with
|
||||
<xref linkend="opt-accounts.email.accounts._name_.mujmap.settings.session_url"/>.
|
||||
</para><para>
|
||||
If <literal>null</literal>, defaults to
|
||||
<xref linkend="opt-accounts.email.accounts._name_.jmap.host"/>.
|
||||
'';
|
||||
};
|
||||
|
||||
session_url = mkOption {
|
||||
type = types.nullOr types.str;
|
||||
default = null;
|
||||
example = "https://jmap.example.com/.well-known/jmap";
|
||||
description = ''
|
||||
Sesion URL to connect to.
|
||||
</para><para>
|
||||
Mutually exclusive with
|
||||
<xref linkend="opt-accounts.email.accounts._name_.mujmap.settings.fqdn"/>.
|
||||
</para><para>
|
||||
If <literal>null</literal>, defaults to
|
||||
<xref linkend="opt-accounts.email.accounts._name_.jmap.sessionUrl"/>.
|
||||
'';
|
||||
};
|
||||
|
||||
auto_create_new_mailboxes = mkOption {
|
||||
type = types.bool;
|
||||
default = true;
|
||||
description = ''
|
||||
Whether to create new mailboxes automatically on the server from notmuch
|
||||
tags.
|
||||
'';
|
||||
};
|
||||
|
||||
cache_dir = mkOption {
|
||||
type = types.nullOr types.str;
|
||||
default = null;
|
||||
description = ''
|
||||
The cache directory in which to store mail files while they are being
|
||||
downloaded. The default is operating-system specific.
|
||||
'';
|
||||
};
|
||||
|
||||
tags = mkOption {
|
||||
type = types.submodule {
|
||||
freeformType = settingsFormat.type;
|
||||
options = tagsOpts;
|
||||
};
|
||||
default = { };
|
||||
description = ''
|
||||
Tag configuration.
|
||||
</para><para>
|
||||
Beware that there are quirks that require manual consideration if changing the
|
||||
values of these files; please see
|
||||
<link xlink:href="https://github.com/elizagamedev/mujmap/blob/main/mujmap.toml.example"/>
|
||||
for more details.
|
||||
'';
|
||||
};
|
||||
};
|
||||
|
||||
mujmapOpts = {
|
||||
enable = mkEnableOption "mujmap JMAP synchronization for notmuch";
|
||||
|
||||
notmuchSetupWarning = mkOption {
|
||||
type = types.bool;
|
||||
default = true;
|
||||
description = ''
|
||||
Warn if Notmuch is not also enabled for this account.
|
||||
</para><para>
|
||||
This can safely be disabled if <filename>mujmap.toml</filename> is managed
|
||||
outside of Home Manager.
|
||||
'';
|
||||
};
|
||||
|
||||
settings = mkOption {
|
||||
type = types.submodule {
|
||||
freeformType = settingsFormat.type;
|
||||
options = rootOpts;
|
||||
};
|
||||
default = { };
|
||||
description = ''
|
||||
Settings which are applied to <filename>mujmap.toml</filename>
|
||||
for the account.
|
||||
</para><para>
|
||||
See the <link xlink:href="https://github.com/elizagamedev/mujmap">mujmap project</link>
|
||||
for documentation of settings not explicitly covered by this module.
|
||||
'';
|
||||
};
|
||||
};
|
||||
|
||||
mujmapModule = types.submodule { options = { mujmap = mujmapOpts; }; };
|
||||
in {
|
||||
meta.maintainers = with maintainers; [ elizagamedev ];
|
||||
|
||||
options = {
|
||||
programs.mujmap = {
|
||||
enable = mkEnableOption "mujmap Gmail synchronization for notmuch";
|
||||
|
||||
package = mkOption {
|
||||
type = types.package;
|
||||
default = pkgs.mujmap;
|
||||
defaultText = "pkgs.mujmap";
|
||||
description = ''
|
||||
mujmap package to use.
|
||||
'';
|
||||
};
|
||||
};
|
||||
|
||||
accounts.email.accounts =
|
||||
mkOption { type = with types; attrsOf mujmapModule; };
|
||||
};
|
||||
|
||||
config = mkIf cfg.enable (mkMerge [
|
||||
(mkIf (missingNotmuchAccounts != [ ]) {
|
||||
warnings = [''
|
||||
mujmap is enabled for the following email accounts, but notmuch is not:
|
||||
|
||||
${concatStringsSep "\n " missingNotmuchAccounts}
|
||||
|
||||
Notmuch can be enabled with:
|
||||
|
||||
${concatStringsSep "\n " notmuchConfigHelp}
|
||||
|
||||
If you have configured notmuch outside of Home Manager, you can suppress this
|
||||
warning with:
|
||||
|
||||
programs.mujmap.notmuchSetupWarning = false;
|
||||
''];
|
||||
})
|
||||
|
||||
{
|
||||
warnings = flatten (map (account: account.warnings) mujmapAccounts);
|
||||
|
||||
home.packages = [ cfg.package ];
|
||||
|
||||
# Notmuch should ignore non-mail files created by mujmap.
|
||||
programs.notmuch.new.ignore = [ "/.*[.](toml|json|lock)$/" ];
|
||||
|
||||
home.file = listToAttrs (map configFile mujmapAccounts);
|
||||
}
|
||||
]);
|
||||
}
|
|
@ -77,6 +77,7 @@ import nmt {
|
|||
./modules/programs/mbsync
|
||||
./modules/programs/mpv
|
||||
./modules/programs/mu
|
||||
./modules/programs/mujmap
|
||||
./modules/programs/ncmpcpp
|
||||
./modules/programs/ne
|
||||
./modules/programs/neomutt
|
||||
|
|
5
tests/modules/programs/mujmap/default.nix
Normal file
5
tests/modules/programs/mujmap/default.nix
Normal file
|
@ -0,0 +1,5 @@
|
|||
{
|
||||
mujmap-defaults = ./mujmap-defaults.nix;
|
||||
mujmap-fqdn-and-session-url-specified =
|
||||
./mujmap-fqdn-and-session-url-specified.nix;
|
||||
}
|
14
tests/modules/programs/mujmap/mujmap-defaults-expected.toml
Normal file
14
tests/modules/programs/mujmap/mujmap-defaults-expected.toml
Normal file
|
@ -0,0 +1,14 @@
|
|||
auto_create_new_mailboxes = true
|
||||
fqdn = "example.com"
|
||||
password_command = "'password-command'"
|
||||
username = "home.manager"
|
||||
|
||||
[tags]
|
||||
deleted = "deleted"
|
||||
directory_separator = "/"
|
||||
important = "important"
|
||||
inbox = "inbox"
|
||||
lowercase = false
|
||||
phishing = "phishing"
|
||||
sent = "sent"
|
||||
spam = "spam"
|
25
tests/modules/programs/mujmap/mujmap-defaults.nix
Normal file
25
tests/modules/programs/mujmap/mujmap-defaults.nix
Normal file
|
@ -0,0 +1,25 @@
|
|||
{ config, lib, pkgs, ... }:
|
||||
|
||||
with lib;
|
||||
|
||||
{
|
||||
imports = [ ../../accounts/email-test-accounts.nix ];
|
||||
|
||||
config = {
|
||||
programs.mujmap.enable = true;
|
||||
programs.mujmap.package = config.lib.test.mkStubPackage { };
|
||||
|
||||
accounts.email.accounts."hm@example.com" = {
|
||||
jmap.host = "example.com";
|
||||
mujmap.enable = true;
|
||||
notmuch.enable = true;
|
||||
};
|
||||
|
||||
nmt.script = ''
|
||||
assertFileExists home-files/Mail/hm@example.com/mujmap.toml
|
||||
assertFileContent home-files/Mail/hm@example.com/mujmap.toml ${
|
||||
./mujmap-defaults-expected.toml
|
||||
}
|
||||
'';
|
||||
};
|
||||
}
|
|
@ -0,0 +1,26 @@
|
|||
{ config, lib, pkgs, ... }:
|
||||
|
||||
with lib;
|
||||
|
||||
{
|
||||
imports = [ ../../accounts/email-test-accounts.nix ];
|
||||
|
||||
config = {
|
||||
programs.mujmap.enable = true;
|
||||
programs.mujmap.package = config.lib.test.mkStubPackage { };
|
||||
|
||||
accounts.email.accounts."hm@example.com" = {
|
||||
jmap.host = "example.com";
|
||||
jmap.sessionUrl = "https://jmap.example.com/";
|
||||
mujmap.enable = true;
|
||||
notmuch.enable = true;
|
||||
};
|
||||
|
||||
nmt.script = ''
|
||||
assertFileExists home-files/Mail/hm@example.com/mujmap.toml
|
||||
assertFileContent home-files/Mail/hm@example.com/mujmap.toml ${
|
||||
./mujmap-defaults-expected.toml
|
||||
}
|
||||
'';
|
||||
};
|
||||
}
|
Loading…
Reference in a new issue