mirror of
https://github.com/nix-community/home-manager
synced 2024-12-26 19:59:47 +01:00
235 lines
8 KiB
Nix
235 lines
8 KiB
Nix
{ config, lib, pkgs, ... }:
|
|
|
|
with lib;
|
|
|
|
let
|
|
|
|
cfg = config.services.emacs;
|
|
emacsCfg = config.programs.emacs;
|
|
emacsBinPath = "${cfg.package}/bin";
|
|
emacsVersion = getVersion cfg.package;
|
|
|
|
clientWMClass =
|
|
if versionAtLeast emacsVersion "28" then "Emacsd" else "Emacs";
|
|
|
|
# Workaround for https://debbugs.gnu.org/47511
|
|
needsSocketWorkaround = versionOlder emacsVersion "28"
|
|
&& cfg.socketActivation.enable;
|
|
|
|
# Adapted from upstream emacs.desktop
|
|
clientDesktopItem = pkgs.writeTextDir "share/applications/emacsclient.desktop"
|
|
(generators.toINI { } {
|
|
"Desktop Entry" = {
|
|
Type = "Application";
|
|
Exec = "${emacsBinPath}/emacsclient ${
|
|
concatStringsSep " " cfg.client.arguments
|
|
} %F";
|
|
Terminal = false;
|
|
Name = "Emacs Client";
|
|
Icon = "emacs";
|
|
Comment = "Edit text";
|
|
GenericName = "Text Editor";
|
|
MimeType =
|
|
"text/english;text/plain;text/x-makefile;text/x-c++hdr;text/x-c++src;text/x-chdr;text/x-csrc;text/x-java;text/x-moc;text/x-pascal;text/x-tcl;text/x-tex;application/x-shellscript;text/x-c;text/x-c++;";
|
|
Categories = "Development;TextEditor;";
|
|
Keywords = "Text;Editor;";
|
|
StartupWMClass = clientWMClass;
|
|
};
|
|
});
|
|
|
|
# Match the default socket path for the Emacs version so emacsclient continues
|
|
# to work without wrapping it.
|
|
socketDir = "%t/emacs";
|
|
socketPath = "${socketDir}/server";
|
|
in {
|
|
meta.maintainers = [ maintainers.tadfisher ];
|
|
|
|
options.services.emacs = {
|
|
enable = mkEnableOption "the Emacs daemon";
|
|
|
|
package = mkOption {
|
|
type = types.package;
|
|
default = if emacsCfg.enable then emacsCfg.finalPackage else pkgs.emacs;
|
|
defaultText = literalExpression ''
|
|
if config.programs.emacs.enable then config.programs.emacs.finalPackage
|
|
else pkgs.emacs
|
|
'';
|
|
description = "The Emacs package to use.";
|
|
};
|
|
|
|
extraOptions = mkOption {
|
|
type = with types; listOf str;
|
|
default = [ ];
|
|
example = [ "-f" "exwm-enable" ];
|
|
description = ''
|
|
Extra command-line arguments to pass to {command}`emacs`.
|
|
'';
|
|
};
|
|
|
|
client = {
|
|
enable = mkEnableOption "generation of Emacs client desktop file";
|
|
arguments = mkOption {
|
|
type = with types; listOf str;
|
|
default = [ "-c" ];
|
|
description = ''
|
|
Command-line arguments to pass to {command}`emacsclient`.
|
|
'';
|
|
};
|
|
};
|
|
|
|
# Attrset for forward-compatibility; there may be a need to customize the
|
|
# socket path, though allowing for such is not easy to do as systemd socket
|
|
# units don't perform variable expansion for 'ListenStream'.
|
|
socketActivation = {
|
|
enable = mkEnableOption "systemd socket activation for the Emacs service";
|
|
};
|
|
|
|
startWithUserSession = mkOption {
|
|
type = with types; either bool (enum [ "graphical" ]);
|
|
default = !cfg.socketActivation.enable;
|
|
defaultText =
|
|
literalExpression "!config.services.emacs.socketActivation.enable";
|
|
example = "graphical";
|
|
description = ''
|
|
Whether to launch Emacs service with the systemd user session. If it is
|
|
`true`, Emacs service is started by
|
|
`default.target`. If it is
|
|
`"graphical"`, Emacs service is started by
|
|
`graphical-session.target`.
|
|
'';
|
|
};
|
|
|
|
defaultEditor = mkOption rec {
|
|
type = types.bool;
|
|
default = false;
|
|
example = !default;
|
|
description = ''
|
|
Whether to configure {command}`emacsclient` as the default
|
|
editor using the {env}`EDITOR` environment variable.
|
|
'';
|
|
};
|
|
};
|
|
|
|
config = mkIf cfg.enable (mkMerge [
|
|
(mkIf pkgs.stdenv.isLinux {
|
|
systemd.user.services.emacs = {
|
|
Unit = {
|
|
Description = "Emacs text editor";
|
|
Documentation =
|
|
"info:emacs man:emacs(1) https://gnu.org/software/emacs/";
|
|
|
|
After = optional (cfg.startWithUserSession == "graphical")
|
|
"graphical-session.target";
|
|
PartOf = optional (cfg.startWithUserSession == "graphical")
|
|
"graphical-session.target";
|
|
|
|
# Avoid killing the Emacs session, which may be full of
|
|
# unsaved buffers.
|
|
X-RestartIfChanged = false;
|
|
} // optionalAttrs needsSocketWorkaround {
|
|
# Emacs deletes its socket when shutting down, which systemd doesn't
|
|
# handle, resulting in a server without a socket.
|
|
# See https://github.com/nix-community/home-manager/issues/2018
|
|
RefuseManualStart = true;
|
|
};
|
|
|
|
Service = {
|
|
Type = "notify";
|
|
|
|
# We wrap ExecStart in a login shell so Emacs starts with the user's
|
|
# environment, most importantly $PATH and $NIX_PROFILES. It may be
|
|
# worth investigating a more targeted approach for user services to
|
|
# import the user environment.
|
|
ExecStart = ''
|
|
${pkgs.runtimeShell} -l -c "${emacsBinPath}/emacs --fg-daemon${
|
|
# In case the user sets 'server-directory' or 'server-name' in
|
|
# their Emacs config, we want to specify the socket path explicitly
|
|
# so launching 'emacs.service' manually doesn't break emacsclient
|
|
# when using socket activation.
|
|
optionalString cfg.socketActivation.enable
|
|
"=${escapeShellArg socketPath}"
|
|
} ${escapeShellArgs cfg.extraOptions}"'';
|
|
|
|
# Emacs will exit with status 15 after having received SIGTERM, which
|
|
# is the default "KillSignal" value systemd uses to stop services.
|
|
SuccessExitStatus = 15;
|
|
|
|
Restart = "on-failure";
|
|
} // optionalAttrs needsSocketWorkaround {
|
|
# Use read-only directory permissions to prevent emacs from
|
|
# deleting systemd's socket file before exiting.
|
|
ExecStartPost =
|
|
"${pkgs.coreutils}/bin/chmod --changes -w ${socketDir}";
|
|
ExecStopPost =
|
|
"${pkgs.coreutils}/bin/chmod --changes +w ${socketDir}";
|
|
};
|
|
} // optionalAttrs (cfg.startWithUserSession != false) {
|
|
Install = {
|
|
WantedBy = [
|
|
(if cfg.startWithUserSession == true then
|
|
"default.target"
|
|
else
|
|
"graphical-session.target")
|
|
];
|
|
};
|
|
};
|
|
|
|
home = {
|
|
packages = optional cfg.client.enable (hiPrio clientDesktopItem);
|
|
|
|
sessionVariables = mkIf cfg.defaultEditor {
|
|
EDITOR = getBin (pkgs.writeShellScript "editor" ''
|
|
exec ${
|
|
getBin cfg.package
|
|
}/bin/emacsclient "''${@:---create-frame}"'');
|
|
};
|
|
};
|
|
})
|
|
|
|
(mkIf (cfg.socketActivation.enable && pkgs.stdenv.isLinux) {
|
|
systemd.user.sockets.emacs = {
|
|
Unit = {
|
|
Description = "Emacs text editor";
|
|
Documentation =
|
|
"info:emacs man:emacs(1) https://gnu.org/software/emacs/";
|
|
};
|
|
|
|
Socket = {
|
|
ListenStream = socketPath;
|
|
FileDescriptorName = "server";
|
|
SocketMode = "0600";
|
|
DirectoryMode = "0700";
|
|
# This prevents the service from immediately starting again
|
|
# after being stopped, due to the function
|
|
# `server-force-stop' present in `kill-emacs-hook', which
|
|
# calls `server-running-p', which opens the socket file.
|
|
FlushPending = true;
|
|
};
|
|
|
|
Install = {
|
|
WantedBy = [ "sockets.target" ];
|
|
# Adding this Requires= dependency ensures that systemd
|
|
# manages the socket file, in the case where the service is
|
|
# started when the socket is stopped.
|
|
# The socket unit is implicitly ordered before the service.
|
|
RequiredBy = [ "emacs.service" ];
|
|
};
|
|
};
|
|
})
|
|
|
|
(mkIf pkgs.stdenv.isDarwin {
|
|
launchd.agents.emacs = {
|
|
enable = true;
|
|
config = {
|
|
ProgramArguments = [ "${cfg.package}/bin/emacs" "--fg-daemon" ]
|
|
++ cfg.extraOptions;
|
|
RunAtLoad = true;
|
|
KeepAlive = {
|
|
Crashed = true;
|
|
SuccessfulExit = false;
|
|
};
|
|
};
|
|
};
|
|
})
|
|
]);
|
|
}
|