1
0
Fork 0
mirror of https://github.com/nix-community/home-manager.git synced 2025-11-08 11:36:05 +01:00
home-manager/modules/programs/thunderbird.nix
Adam Dinwoodie dbfcd3292d accounts.email: add option to disable an account
Allow a user to disable an email account by setting
`accounts.email.accounts.<name>.enable = false`.  This is useful if
someone wants to configure email accounts globally but only use them in
certain circumstances.

Everywhere email account configuration is used, check if the account is
enabled before checking any attributes of the account.
2025-08-07 14:08:46 -05:00

1034 lines
36 KiB
Nix

{
config,
lib,
pkgs,
...
}:
let
inherit (lib)
attrValues
concatStringsSep
filter
length
literalExpression
mapAttrsToList
mkIf
mkOption
mkOptionDefault
optionalAttrs
optionalString
types
;
inherit (pkgs.stdenv.hostPlatform) isDarwin;
cfg = config.programs.thunderbird;
thunderbirdJson = types.attrsOf (pkgs.formats.json { }).type // {
description = "Thunderbird preference (int, bool, string, and also attrs, list, float as a JSON string)";
};
# The extensions path shared by all profiles.
extensionPath = "extensions/{ec8030f7-c20a-464f-9b0e-13a3a9e97384}";
moduleName = "programs.thunderbird";
addId = map (a: a // { id = builtins.hashString "sha256" a.name; });
enabledEmailAccounts = filter (a: a.enable && a.thunderbird.enable) (
attrValues config.accounts.email.accounts
);
enabledEmailAccountsWithId = addId enabledEmailAccounts;
enabledCalendarAccounts = filter (a: a.thunderbird.enable) (
attrValues config.accounts.calendar.accounts
);
enabledCalendarAccountsWithId = addId enabledCalendarAccounts;
enabledContactAccounts = filter (a: a.thunderbird.enable) (
attrValues config.accounts.contact.accounts
);
enabledContactAccountsWithId = addId enabledContactAccounts;
thunderbirdConfigPath = if isDarwin then "Library/Thunderbird" else ".thunderbird";
thunderbirdProfilesPath =
if isDarwin then "${thunderbirdConfigPath}/Profiles" else thunderbirdConfigPath;
profilesWithId = lib.imap0 (i: v: v // { id = toString i; }) (attrValues cfg.profiles);
profilesIni =
lib.foldl lib.recursiveUpdate
{
General = {
StartWithLastProfile = 1;
}
// lib.optionalAttrs (cfg.profileVersion != null) {
Version = cfg.profileVersion;
};
}
(
lib.flip map profilesWithId (profile: {
"Profile${profile.id}" = {
Name = profile.name;
Path = if isDarwin then "Profiles/${profile.name}" else profile.name;
IsRelative = 1;
Default = if profile.isDefault then 1 else 0;
};
})
);
getId =
account: address:
if address == account.address then
account.id
else
(builtins.hashString "sha256" (
if (builtins.isString address) then address else (address.address + address.realName)
));
toThunderbirdIdentity =
account: address:
# For backwards compatibility, the primary address reuses the account ID.
let
id = getId account address;
addressIsString = builtins.isString address;
identity = if addressIsString then account else address // { inherit id; };
in
{
"mail.identity.id_${id}.fullName" = identity.realName;
"mail.identity.id_${id}.useremail" = if addressIsString then address else address.address;
"mail.identity.id_${id}.valid" = true;
"mail.identity.id_${id}.htmlSigText" =
if identity.signature.showSignature == "none" then "" else identity.signature.text;
}
// optionalAttrs (identity.gpg != null) {
"mail.identity.id_${id}.attachPgpKey" = false;
"mail.identity.id_${id}.autoEncryptDrafts" = true;
"mail.identity.id_${id}.e2etechpref" = 0;
"mail.identity.id_${id}.encryptionpolicy" = if identity.gpg.encryptByDefault then 2 else 0;
"mail.identity.id_${id}.is_gnupg_key_id" = true;
"mail.identity.id_${id}.last_entered_external_gnupg_key_id" = identity.gpg.key;
"mail.identity.id_${id}.openpgp_key_id" = identity.gpg.key;
"mail.identity.id_${id}.protectSubject" = true;
"mail.identity.id_${id}.sign_mail" = identity.gpg.signByDefault;
}
// optionalAttrs (identity.smtp != null) {
"mail.identity.id_${id}.smtpServer" = "smtp_${identity.id}";
}
// account.thunderbird.perIdentitySettings id;
toThunderbirdSMTP =
account: address:
let
id = getId account address;
addressIsString = builtins.isString address;
in
optionalAttrs (!addressIsString && address.smtp != null) {
"mail.smtpserver.smtp_${id}.authMethod" = 3;
"mail.smtpserver.smtp_${id}.hostname" = address.smtp.host;
"mail.smtpserver.smtp_${id}.port" = if (address.smtp.port != null) then address.smtp.port else 587;
"mail.smtpserver.smtp_${id}.try_ssl" =
if !address.smtp.tls.enable then
0
else if address.smtp.tls.useStartTls then
2
else
3;
"mail.smtpserver.smtp_${id}.username" = address.userName;
};
toThunderbirdAccount =
account: profile:
let
id = account.id;
addresses = [ account.address ] ++ account.aliases;
in
{
"mail.account.account_${id}.identities" = concatStringsSep "," (
map (address: "id_${getId account address}") addresses
);
"mail.account.account_${id}.server" = "server_${id}";
}
// optionalAttrs account.primary {
"mail.accountmanager.defaultaccount" = "account_${id}";
}
// optionalAttrs (account.imap != null) {
"mail.server.server_${id}.directory" = "${thunderbirdProfilesPath}/${profile.name}/ImapMail/${id}";
"mail.server.server_${id}.directory-rel" = "[ProfD]ImapMail/${id}";
"mail.server.server_${id}.hostname" = account.imap.host;
"mail.server.server_${id}.login_at_startup" = true;
"mail.server.server_${id}.name" = account.name;
"mail.server.server_${id}.port" = if (account.imap.port != null) then account.imap.port else 143;
"mail.server.server_${id}.socketType" =
if !account.imap.tls.enable then
0
else if account.imap.tls.useStartTls then
2
else
3;
"mail.server.server_${id}.type" = "imap";
"mail.server.server_${id}.userName" = account.userName;
}
// optionalAttrs (account.smtp != null) {
"mail.smtpserver.smtp_${id}.authMethod" = 3;
"mail.smtpserver.smtp_${id}.hostname" = account.smtp.host;
"mail.smtpserver.smtp_${id}.port" = if (account.smtp.port != null) then account.smtp.port else 587;
"mail.smtpserver.smtp_${id}.try_ssl" =
if !account.smtp.tls.enable then
0
else if account.smtp.tls.useStartTls then
2
else
3;
"mail.smtpserver.smtp_${id}.username" = account.userName;
}
// builtins.foldl' (a: b: a // b) { } (
builtins.map (address: toThunderbirdSMTP account address) addresses
)
// optionalAttrs (account.smtp != null && account.primary) {
"mail.smtp.defaultserver" = "smtp_${id}";
}
// builtins.foldl' (a: b: a // b) { } (
builtins.map (address: toThunderbirdIdentity account address) addresses
)
// account.thunderbird.settings id;
toThunderbirdCalendar =
calendar: _:
let
inherit (calendar) id;
in
{
"calendar.registry.calendar_${id}.name" = calendar.name;
"calendar.registry.calendar_${id}.calendar-main-in-composite" = true;
"calendar.registry.calendar_${id}.cache.enabled" = true;
}
// optionalAttrs (calendar.remote == null) {
"calendar.registry.calendar_${id}.type" = "storage";
"calendar.registry.calendar_${id}.uri" = "moz-storage-calendar://";
}
// optionalAttrs (calendar.remote != null) {
"calendar.registry.calendar_${id}.type" =
if (calendar.remote.type == "http") then "ics" else calendar.remote.type;
"calendar.registry.calendar_${id}.uri" = calendar.remote.url;
"calendar.registry.calendar_${id}.username" = calendar.remote.userName;
}
// optionalAttrs calendar.primary {
"calendar.registry.calendar_${id}.calendar-main-default" = true;
}
// optionalAttrs calendar.thunderbird.readOnly {
"calendar.registry.calendar_${id}.readOnly" = true;
}
// optionalAttrs (calendar.thunderbird.color != "") {
"calendar.registry.calendar_${id}.color" = calendar.thunderbird.color;
};
toThunderbirdContact =
contact: _:
let
inherit (contact) id;
in
lib.filterAttrs (n: v: v != null) (
{
"ldap_2.servers.contact_${id}.description" = contact.name;
"ldap_2.servers.contact_${id}.filename" = "contact_${id}.sqlite"; # this is needed for carddav to work
}
// optionalAttrs (contact.remote == null) {
"ldap_2.servers.contact_${id}.dirType" = 101; # dirType 101 for local address book
}
// optionalAttrs (contact.remote != null && contact.remote.type == "carddav") {
"ldap_2.servers.contact_${id}.dirType" = 102; # dirType 102 for CardDAV
"ldap_2.servers.contact_${id}.carddav.url" = contact.remote.url;
"ldap_2.servers.contact_${id}.carddav.username" = contact.remote.userName;
"ldap_2.servers.contact_${id}.carddav.token" = contact.thunderbird.token;
}
);
toThunderbirdFeed =
feed: profile:
let
id = feed.id;
in
{
"mail.account.account_${id}.server" = "server_${id}";
"mail.server.server_${id}.name" = feed.name;
"mail.server.server_${id}.type" = "rss";
"mail.server.server_${id}.directory" =
"${thunderbirdProfilesPath}/${profile.name}/Mail/Feeds-${id}";
"mail.server.server_${id}.directory-rel" = "[ProfD]Mail/Feeds-${id}";
"mail.server.server_${id}.hostname" = "Feeds-${id}";
};
mkUserJs = prefs: extraPrefs: ''
// Generated by Home Manager.
${lib.concatStrings (
mapAttrsToList (name: value: ''
user_pref("${name}", ${builtins.toJSON value});
'') prefs
)}
${extraPrefs}
'';
mkFilterToIniString =
f:
if f.text == null then
''
name="${f.name}"
enabled="${if f.enabled then "yes" else "no"}"
type="${f.type}"
action="${f.action}"
''
+ optionalString (f.actionValue != null) ''
actionValue="${f.actionValue}"
''
+ ''
condition="${f.condition}"
''
+ optionalString (f.extraConfig != null) f.extraConfig
else
f.text;
mkFilterListToIni =
filters:
''
version="9"
logging="no"
''
+ lib.concatStrings (map (f: mkFilterToIniString f) filters);
getAccountsForProfile =
profileName: accounts:
(filter (
a: a.thunderbird.profiles == [ ] || lib.any (p: p == profileName) a.thunderbird.profiles
) accounts);
in
{
meta.maintainers = with lib.hm.maintainers; [
d-dervishi
jkarlson
];
options = {
programs.thunderbird = {
enable = lib.mkEnableOption "Thunderbird";
package = lib.mkPackageOption pkgs "thunderbird" {
example = "pkgs.thunderbird-91";
};
profileVersion = mkOption {
internal = true;
type = types.nullOr types.ints.unsigned;
default = if isDarwin then null else 2;
description = "profile version, set null for nix-darwin";
};
nativeMessagingHosts = mkOption {
visible = true;
type = types.listOf types.package;
default = [ ];
description = ''
Additional packages containing native messaging hosts that should be
made available to Thunderbird extensions.
'';
};
profiles = mkOption {
type = types.attrsOf (
types.submodule (
{ config, name, ... }:
{
options = {
name = mkOption {
type = types.str;
default = name;
readOnly = true;
description = "This profile's name.";
};
isDefault = mkOption {
type = types.bool;
default = false;
example = true;
description = ''
Whether this is a default profile. There must be exactly one
default profile.
'';
};
feedAccounts = mkOption {
type = types.attrsOf (
types.submodule (
{ name, ... }:
{
options = {
name = mkOption {
type = types.str;
default = name;
readOnly = true;
description = "This feed account's name.";
};
};
}
)
);
default = { };
description = ''
Attribute set of feed accounts. Feeds themselves have to be
managed through Thunderbird's settings. This option allows
feeds to coexist with declaratively managed email accounts.
'';
};
settings = mkOption {
type = thunderbirdJson;
default = { };
example = literalExpression ''
{
"mail.spellcheck.inline" = false;
"mailnews.database.global.views.global.columns" = {
selectCol = {
visible = false;
ordinal = 1;
};
threadCol = {
visible = true;
ordinal = 2;
};
};
}
'';
description = ''
Preferences to add to this profile's
{file}`user.js`.
'';
};
accountsOrder = mkOption {
type = types.listOf types.str;
default = [ ];
description = ''
Custom ordering of accounts and local folders in
Thunderbird's folder pane. The accounts are specified
by their name. For declarative accounts, it must be the name
of their attribute in `config.accounts.email.accounts` (or
`config.programs.thunderbird.profiles.<name>.feedAccounts`
for feed accounts). The local folders name can be found in
the `mail.accountmanager.accounts` Thunderbird preference,
for example with Settings > Config Editor ("account1" by
default). Enabled accounts and local folders that aren't
listed here appear in an arbitrary order after the ordered
accounts.
'';
example = ''
[
"my-awesome-account"
"private"
"work"
"rss"
/* Other accounts in arbitrary order */
]
'';
};
calendarAccountsOrder = mkOption {
type = types.listOf types.str;
default = [ ];
description = ''
Custom ordering of calendar accounts. The accounts are specified
by their name. For declarative accounts, it must be the name
of their attribute in `config.accounts.calendar.accounts`.
Enabled accounts that aren't listed here appear in an arbitrary
order after the ordered accounts.
'';
example = ''
[
"my-awesome-account"
"private"
"work"
"holidays"
/* Other accounts in arbitrary order */
]
'';
};
withExternalGnupg = mkOption {
type = types.bool;
default = false;
example = true;
description = "Allow using external GPG keys with GPGME.";
};
userChrome = mkOption {
type = types.lines;
default = "";
description = "Custom Thunderbird user chrome CSS.";
example = ''
/* Hide tab bar in Thunderbird */
#tabs-toolbar {
visibility: collapse !important;
}
'';
};
userContent = mkOption {
type = types.lines;
default = "";
description = "Custom Thunderbird user content CSS.";
example = ''
/* Hide scrollbar on Thunderbird pages */
*{scrollbar-width:none !important}
'';
};
extraConfig = mkOption {
type = types.lines;
default = "";
description = ''
Extra preferences to add to {file}`user.js`.
'';
};
search = mkOption {
type = types.submodule (
args:
import ./firefox/profiles/search.nix {
inherit (args) config;
inherit lib pkgs;
appName = "Thunderbird";
package = cfg.package;
modulePath = [
"programs"
"thunderbird"
"profiles"
name
"search"
];
profilePath = name;
}
);
default = { };
description = "Declarative search engine configuration.";
};
extensions = mkOption {
type = types.listOf types.package;
default = [ ];
example = literalExpression ''
[
pkgs.some-thunderbird-extension
]
'';
description = ''
List of ${name} add-on packages to install for this profile.
Note that it is necessary to manually enable extensions
inside ${name} after the first installation.
To automatically enable extensions add
`"extensions.autoDisableScopes" = 0;`
to
[{option}`${moduleName}.profiles.<profile>.settings`](#opt-${moduleName}.profiles._name_.settings)
'';
};
};
}
)
);
description = "Attribute set of Thunderbird profiles.";
};
settings = mkOption {
type = thunderbirdJson;
default = { };
example = literalExpression ''
{
"general.useragent.override" = "";
"privacy.donottrackheader.enabled" = true;
}
'';
description = ''
Attribute set of Thunderbird preferences to be added to
all profiles.
'';
};
darwinSetupWarning = mkOption {
type = types.bool;
default = true;
example = false;
visible = false;
readOnly = !isDarwin;
description = ''
Using programs.thunderbird.darwinSetupWarning is deprecated. The
module is compatible with all Thunderbird installations.
'';
};
};
accounts.email.accounts = mkOption {
type = types.attrsOf (
types.submodule (
{ config, ... }:
{
config.thunderbird = {
settings = lib.mkIf (config.flavor == "gmail.com") (id: {
"mail.smtpserver.smtp_${id}.authMethod" = mkOptionDefault 10; # 10 = OAuth2
"mail.server.server_${id}.authMethod" = mkOptionDefault 10; # 10 = OAuth2
"mail.server.server_${id}.socketType" = mkOptionDefault 3; # SSL/TLS
"mail.server.server_${id}.is_gmail" = mkOptionDefault true; # handle labels, trash, etc
});
};
options.thunderbird = {
enable = lib.mkEnableOption "the Thunderbird mail client for this account";
profiles = mkOption {
type = with types; listOf str;
default = [ ];
example = literalExpression ''
[ "profile1" "profile2" ]
'';
description = ''
List of Thunderbird profiles for which this account should be
enabled. If this list is empty (the default), this account will
be enabled for all declared profiles.
'';
};
settings = mkOption {
type =
with types;
functionTo (
attrsOf (oneOf [
bool
int
str
])
);
default = _: { };
defaultText = literalExpression "_: { }";
example = literalExpression ''
id: {
"mail.server.server_''${id}.check_new_mail" = false;
};
'';
description = ''
Extra settings to add to this Thunderbird account configuration.
The {var}`id` given as argument is an automatically
generated account identifier.
'';
};
perIdentitySettings = mkOption {
type =
with types;
functionTo (
attrsOf (oneOf [
bool
int
str
])
);
default = _: { };
defaultText = literalExpression "_: { }";
example = literalExpression ''
id: {
"mail.identity.id_''${id}.protectSubject" = false;
"mail.identity.id_''${id}.autoEncryptDrafts" = false;
};
'';
description = ''
Extra settings to add to each identity of this Thunderbird
account configuration. The {var}`id` given as
argument is an automatically generated identifier.
'';
};
messageFilters = mkOption {
type =
with types;
listOf (submodule {
options = {
name = mkOption {
type = str;
description = "Name for the filter.";
};
enabled = mkOption {
type = bool;
default = true;
description = "Whether this filter is currently active.";
};
type = mkOption {
type = str;
description = "Type for this filter.";
};
action = mkOption {
type = str;
description = "Action to perform on matched messages.";
};
actionValue = mkOption {
type = nullOr str;
default = null;
description = "Argument passed to the filter action, e.g. a folder path.";
};
condition = mkOption {
type = str;
description = "Condition to match messages against.";
};
extraConfig = mkOption {
type = nullOr str;
default = null;
description = "Extra settings to apply to the filter";
};
text = mkOption {
type = nullOr str;
default = null;
description = ''
The raw text of the filter.
Note that this will override all other options.
'';
};
};
});
default = [ ];
defaultText = literalExpression "[ ]";
example = literalExpression ''
[
{
name = "Mark as Read on Archive";
enabled = true;
type = "128";
action = "Mark read";
condition = "ALL";
}
]
'';
description = ''
List of message filters to add to this Thunderbird account
configuration.
'';
};
};
}
)
);
};
accounts.calendar.accounts = mkOption {
type =
with types;
attrsOf (submodule {
options.thunderbird = {
enable = lib.mkEnableOption "the Thunderbird mail client for this account";
profiles = mkOption {
type = with types; listOf str;
default = [ ];
example = literalExpression ''
[ "profile1" "profile2" ]
'';
description = ''
List of Thunderbird profiles for which this account should be
enabled. If this list is empty (the default), this account will
be enabled for all declared profiles.
'';
};
readOnly = mkOption {
type = bool;
default = false;
description = "Mark calendar as read only";
};
color = mkOption {
type = str;
default = "";
example = "#dc8add";
description = "Display color of the calendar in hex";
};
};
});
};
accounts.contact.accounts = mkOption {
type =
with types;
attrsOf (submodule {
options.thunderbird = {
enable = lib.mkEnableOption "the Thunderbird mail client for this account";
profiles = mkOption {
type = with types; listOf str;
default = [ ];
example = literalExpression ''
[ "profile1" "profile2" ]
'';
description = ''
List of Thunderbird profiles for which this account should be
enabled. If this list is empty (the default), this account will
be enabled for all declared profiles.
'';
};
token = mkOption {
type = nullOr str;
default = null;
example = "secret_token";
description = ''
A token is generated when adding an address book manually to Thunderbird, this can be entered here.
'';
};
};
});
};
};
config = mkIf cfg.enable {
warnings = lib.optionals (!cfg.darwinSetupWarning) [
''
Using programs.thunderbird.darwinSetupWarning is deprecated and will be
removed in the future. Thunderbird is now supported on Darwin.
''
];
assertions = [
(
let
defaults = lib.catAttrs "name" (filter (a: a.isDefault) profilesWithId);
in
{
assertion = cfg.profiles == { } || length defaults == 1;
message =
"Must have exactly one default Thunderbird profile but found "
+ toString (length defaults)
+ optionalString (length defaults > 1) (", namely " + concatStringsSep "," defaults);
}
)
(
let
profiles = lib.catAttrs "name" profilesWithId;
selectedProfiles = lib.concatMap (a: a.thunderbird.profiles) (
enabledEmailAccounts ++ enabledCalendarAccounts
);
in
{
assertion = (lib.intersectLists profiles selectedProfiles) == selectedProfiles;
message =
"Cannot enable an account for a non-declared profile. "
+ "The declared profiles are "
+ (concatStringsSep "," profiles)
+ ", but the used profiles are "
+ (concatStringsSep "," selectedProfiles);
}
)
(
let
foundCalendars = filter (
a: a.remote != null && a.remote.type == "google_calendar"
) enabledCalendarAccounts;
in
{
assertion = length foundCalendars == 0;
message =
'''accounts.calendar.accounts.<name>.remote.type = "google_calendar";' is not directly supported by Thunderbird, ''
+ "but declared for these calendars: "
+ (concatStringsSep ", " (lib.catAttrs "name" foundCalendars))
+ "\n"
+ ''
To use google calendars in Thunderbird choose 'type = "caldav"' instead.
The 'url' will be "https://apidata.googleusercontent.com/caldav/v2/ID/events/", replace ID with the "Calendar ID".
The ID can be found in the Google Calendar web app: Settings > Settings for my calendars > scroll to "Integrate calendar" > copy the "Calendar ID".
'';
}
)
(
let
foundContacts = filter (
a: a.remote != null && a.remote.type == "google_contacts"
) enabledContactAccounts;
in
{
assertion = (length foundContacts == 0);
message =
'''accounts.contact.accounts.<name>.remote.type = "google_contacts";' is not directly supported by Thunderbird, ''
+ "but declared for these address books: "
+ (concatStringsSep ", " (lib.catAttrs "name" foundContacts))
+ "\n"
+ ''
To use google address books in Thunderbird choose 'type = "caldav"' instead.
The 'url' will be something like "https://www.googleapis.com/carddav/v1/principals/[YOUR-MAIL-ADDRESS]/lists/default/".
To get the exact URL, add the address book to Thunderbird manually and copy the URL from the "Advanced Preferences" section.
'';
}
)
(
let
foundContacts = filter (a: a.remote != null && a.remote.type == "http") enabledContactAccounts;
in
{
assertion = (length foundContacts == 0);
message =
'''accounts.contact.accounts.<name>.remote.type = "http";' is not supported by Thunderbird, ''
+ "but declared for these address books: "
+ (concatStringsSep ", " (lib.catAttrs "name" foundContacts))
+ "\n"
+ ''
Use a calendar of 'type = "caldav"' instead.
'';
}
)
];
home.packages = [
cfg.package
]
++ lib.optional (lib.any (p: p.withExternalGnupg) (attrValues cfg.profiles)) pkgs.gpgme;
mozilla.thunderbirdNativeMessagingHosts = [
cfg.package # package configured native messaging hosts (entire mail app actually)
]
++ cfg.nativeMessagingHosts; # user configured native messaging hosts
home.file = lib.mkMerge (
[
{
"${thunderbirdConfigPath}/profiles.ini" = mkIf (cfg.profiles != { }) {
text = lib.generators.toINI { } profilesIni;
};
}
]
++ lib.flip mapAttrsToList cfg.profiles (
name: profile: {
"${thunderbirdProfilesPath}/${name}/chrome/userChrome.css" = mkIf (profile.userChrome != "") {
text = profile.userChrome;
};
"${thunderbirdProfilesPath}/${name}/chrome/userContent.css" = mkIf (profile.userContent != "") {
text = profile.userContent;
};
"${thunderbirdProfilesPath}/${name}/user.js" =
let
emailAccounts = getAccountsForProfile name enabledEmailAccountsWithId;
calendarAccounts = getAccountsForProfile name enabledCalendarAccountsWithId;
contactAccounts = getAccountsForProfile name enabledContactAccountsWithId;
smtp = filter (a: a.smtp != null) emailAccounts;
feedAccounts = addId (attrValues profile.feedAccounts);
# NOTE: `calendarAccounts` not added here as calendars are not part of the 'Mail' view
accounts = emailAccounts ++ feedAccounts;
orderedAccounts =
let
accountNameToId = builtins.listToAttrs (
map (a: {
name = a.name;
value = "account_${a.id}";
}) accounts
);
accountsOrderIds = map (a: accountNameToId."${a}" or a) profile.accountsOrder;
# Append the default local folder name "account1".
# See https://github.com/nix-community/home-manager/issues/5031.
enabledAccountsIds = (lib.attrsets.mapAttrsToList (name: value: value) accountNameToId) ++ [
"account1"
];
in
lib.optionals (accounts != [ ]) (
accountsOrderIds ++ (lib.lists.subtractLists accountsOrderIds enabledAccountsIds)
);
orderedCalendarAccounts =
let
accountNameToId = builtins.listToAttrs (
map (a: {
name = a.name;
value = "calendar_${a.id}";
}) calendarAccounts
);
accountsOrderIds = map (a: accountNameToId."${a}" or a) profile.calendarAccountsOrder;
enabledAccountsIds = (lib.attrsets.mapAttrsToList (name: value: value) accountNameToId);
in
lib.optionals (calendarAccounts != [ ]) (
accountsOrderIds ++ (lib.lists.subtractLists accountsOrderIds enabledAccountsIds)
);
in
{
text = mkUserJs (builtins.foldl' (a: b: a // b) { } (
[
cfg.settings
(optionalAttrs (length orderedAccounts != 0) {
"mail.accountmanager.accounts" = concatStringsSep "," orderedAccounts;
})
(optionalAttrs (length orderedCalendarAccounts != 0) {
"calendar.list.sortOrder" = concatStringsSep " " orderedCalendarAccounts;
})
(optionalAttrs (length smtp != 0) {
"mail.smtpservers" = concatStringsSep "," (map (a: "smtp_${a.id}") smtp);
})
{ "mail.openpgp.allow_external_gnupg" = profile.withExternalGnupg; }
profile.settings
]
++ (map (a: toThunderbirdAccount a profile) emailAccounts)
++ (map (calendar: toThunderbirdCalendar calendar profile) calendarAccounts)
++ (map (contact: toThunderbirdContact contact profile) contactAccounts)
++ (map (feed: toThunderbirdFeed feed profile) feedAccounts)
)) profile.extraConfig;
};
"${thunderbirdProfilesPath}/${name}/search.json.mozlz4" = mkIf (profile.search.enable) {
enable = profile.search.enable;
force = profile.search.force;
source = profile.search.file;
};
"${thunderbirdProfilesPath}/${name}/extensions" = mkIf (profile.extensions != [ ]) {
source =
let
extensionsEnvPkg = pkgs.buildEnv {
name = "hm-thunderbird-extensions";
paths = profile.extensions;
};
in
"${extensionsEnvPkg}/share/mozilla/${extensionPath}";
recursive = true;
force = true;
};
}
)
++ (mapAttrsToList (
name: profile:
let
emailAccountsWithFilters = (
filter (a: a.thunderbird.messageFilters != [ ]) (
getAccountsForProfile name enabledEmailAccountsWithId
)
);
in
(builtins.listToAttrs (
map (a: {
name = "${thunderbirdProfilesPath}/${name}/ImapMail/${a.id}/msgFilterRules.dat";
value = {
text = mkFilterListToIni a.thunderbird.messageFilters;
};
}) emailAccountsWithFilters
))
) cfg.profiles)
);
};
}