home-manager/modules/programs/bash.nix

242 lines
6.6 KiB
Nix
Raw Normal View History

2017-01-07 19:16:26 +01:00
{ config, lib, pkgs, ... }:
with lib;
let
cfg = config.programs.bash;
2023-01-21 11:27:05 +01:00
writeBashScript = name: text:
pkgs.writeTextFile {
inherit name text;
checkPhase = ''
${pkgs.stdenv.shellDryRun} "$target"
'';
};
2017-01-07 19:16:26 +01:00
2023-01-21 11:27:05 +01:00
in {
meta.maintainers = [ maintainers.rycee ];
imports = [
(mkRenamedOptionModule [ "programs" "bash" "enableAutojump" ] [
"programs"
"autojump"
"enable"
])
];
2017-01-07 19:16:26 +01:00
options = {
programs.bash = {
enable = mkEnableOption "GNU Bourne-Again SHell";
2017-01-07 19:16:26 +01:00
2024-08-18 15:15:31 +02:00
package = mkPackageOption pkgs "bash" { default = "bashInteractive"; };
2022-08-07 13:10:13 +02:00
enableCompletion = mkOption {
type = types.bool;
default = true;
description = ''
2022-08-07 13:10:13 +02:00
Whether to enable Bash completion for all interactive Bash shells.
Note, if you use NixOS or nix-darwin and do not have Bash completion
enabled in the system configuration, then make sure to add
```nix
2022-08-07 13:10:13 +02:00
environment.pathsToLink = [ "/share/bash-completion" ];
```
2022-08-07 13:10:13 +02:00
to your system configuration to get completion for system packages.
Note, the legacy {file}`/etc/bash_completion.d` path is
2022-08-07 13:10:13 +02:00
not supported by Home Manager.
'';
};
2017-01-07 19:16:26 +01:00
historySize = mkOption {
type = types.int;
default = 10000;
description = "Number of history lines to keep in memory.";
2017-01-07 19:16:26 +01:00
};
2018-01-06 10:10:20 +01:00
historyFile = mkOption {
type = types.nullOr types.str;
default = null;
description = "Location of the bash history file.";
2018-01-06 10:10:20 +01:00
};
2017-01-07 19:16:26 +01:00
historyFileSize = mkOption {
type = types.int;
default = 100000;
description = "Number of history lines to keep on file.";
2017-01-07 19:16:26 +01:00
};
historyControl = mkOption {
type = types.listOf
(types.enum [ "erasedups" "ignoredups" "ignorespace" "ignoreboth" ]);
2023-01-21 11:27:05 +01:00
default = [ ];
description = "Controlling how commands are saved on the history list.";
2017-01-07 19:16:26 +01:00
};
historyIgnore = mkOption {
type = types.listOf types.str;
2023-01-21 11:27:05 +01:00
default = [ ];
2017-01-15 20:03:55 +01:00
example = [ "ls" "cd" "exit" ];
description =
2023-01-21 11:27:05 +01:00
"List of commands that should not be saved to the history list.";
2017-01-07 19:16:26 +01:00
};
shellOptions = mkOption {
type = types.listOf types.str;
default = [
# Append to history file rather than replacing it.
"histappend"
# check the window size after each command and, if
# necessary, update the values of LINES and COLUMNS.
"checkwinsize"
# Extended globbing.
"extglob"
"globstar"
# Warn if closing shell with running jobs.
"checkjobs"
];
2023-01-21 11:27:05 +01:00
example = [ "extglob" "-cdspell" ];
description = ''
2021-08-23 07:28:21 +02:00
Shell options to set. Prefix an option with
"`-`" to unset.
2021-08-23 07:28:21 +02:00
'';
2017-01-07 19:16:26 +01:00
};
2017-10-12 15:06:51 +02:00
sessionVariables = mkOption {
2023-01-21 11:27:05 +01:00
default = { };
type = types.attrs;
2017-10-12 15:06:51 +02:00
example = { MAILCHECK = 30; };
description = ''
2017-10-12 15:06:51 +02:00
Environment variables that will be set for the Bash session.
'';
};
2017-01-07 19:16:26 +01:00
shellAliases = mkOption {
2023-01-21 11:27:05 +01:00
default = { };
2019-03-31 13:00:02 +02:00
type = types.attrsOf types.str;
example = literalExpression ''
{
ll = "ls -l";
".." = "cd ..";
}
'';
description = ''
2017-01-07 19:16:26 +01:00
An attribute set that maps aliases (the top level attribute names in
this option) to command strings or directly to build outputs.
2017-01-07 19:16:26 +01:00
'';
};
profileExtra = mkOption {
default = "";
type = types.lines;
description = ''
2018-01-07 15:04:57 +01:00
Extra commands that should be run when initializing a login
shell.
'';
2017-01-07 19:16:26 +01:00
};
initExtra = mkOption {
default = "";
type = types.lines;
description = ''
Extra commands that should be run when initializing an
interactive shell.
'';
};
bashrcExtra = mkOption {
2017-01-07 19:16:26 +01:00
default = "";
type = types.lines;
description = ''
Extra commands that should be placed in {file}`~/.bashrc`.
Note that these commands will be run even in non-interactive shells.
2018-01-07 15:04:57 +01:00
'';
2017-01-07 19:16:26 +01:00
};
2019-07-23 23:55:41 +02:00
logoutExtra = mkOption {
default = "";
type = types.lines;
description = ''
2019-07-23 23:55:41 +02:00
Extra commands that should be run when logging out of an
interactive shell.
'';
};
2017-01-07 19:16:26 +01:00
};
};
2023-01-21 11:27:05 +01:00
config = let
aliasesStr = concatStringsSep "\n"
(mapAttrsToList (k: v: "alias ${k}=${escapeShellArg v}")
cfg.shellAliases);
shoptsStr = let switch = v: if hasPrefix "-" v then "-u" else "-s";
in concatStringsSep "\n"
(map (v: "shopt ${switch v} ${removePrefix "-" v}") cfg.shellOptions);
sessionVarsStr = config.lib.shell.exportAll cfg.sessionVariables;
historyControlStr = concatStringsSep "\n"
(mapAttrsToList (n: v: "${n}=${v}") ({
HISTFILESIZE = toString cfg.historyFileSize;
HISTSIZE = toString cfg.historySize;
} // optionalAttrs (cfg.historyFile != null) {
HISTFILE = ''"${cfg.historyFile}"'';
} // optionalAttrs (cfg.historyControl != [ ]) {
HISTCONTROL = concatStringsSep ":" cfg.historyControl;
} // optionalAttrs (cfg.historyIgnore != [ ]) {
HISTIGNORE = escapeShellArg (concatStringsSep ":" cfg.historyIgnore);
}));
in mkIf cfg.enable {
2024-08-18 15:15:31 +02:00
home.packages = [ cfg.package ];
2023-01-21 11:27:05 +01:00
home.file.".bash_profile".source = writeBashScript "bash_profile" ''
# include .profile if it exists
[[ -f ~/.profile ]] && . ~/.profile
# include .bashrc if it exists
[[ -f ~/.bashrc ]] && . ~/.bashrc
'';
2017-01-07 19:16:26 +01:00
2023-01-21 11:27:05 +01:00
# If completion is enabled then make sure it is sourced very early. This
# is to avoid problems if any other initialization code attempts to set up
# completion.
programs.bash.initExtra = mkIf cfg.enableCompletion (mkOrder 100 ''
if [[ ! -v BASH_COMPLETION_VERSINFO ]]; then
. "${pkgs.bash-completion}/etc/profile.d/bash_completion.sh"
fi
'');
2022-08-07 13:10:13 +02:00
2023-01-21 11:27:05 +01:00
home.file.".profile".source = writeBashScript "profile" ''
. "${config.home.profileDirectory}/etc/profile.d/hm-session-vars.sh"
2023-01-21 11:27:05 +01:00
${sessionVarsStr}
2017-01-07 19:16:26 +01:00
2023-01-21 11:27:05 +01:00
${cfg.profileExtra}
'';
2017-01-07 19:16:26 +01:00
2023-01-21 11:27:05 +01:00
home.file.".bashrc".source = writeBashScript "bashrc" ''
${cfg.bashrcExtra}
2023-01-21 11:27:05 +01:00
# Commands that should be applied only for interactive shells.
[[ $- == *i* ]] || return
2023-01-21 11:27:05 +01:00
${historyControlStr}
2023-01-21 11:27:05 +01:00
${shoptsStr}
2023-01-21 11:27:05 +01:00
${aliasesStr}
2023-01-21 11:27:05 +01:00
${cfg.initExtra}
'';
2017-01-07 19:16:26 +01:00
2023-01-21 11:27:05 +01:00
home.file.".bash_logout" = mkIf (cfg.logoutExtra != "") {
source = writeBashScript "bash_logout" cfg.logoutExtra;
};
};
2017-01-07 19:16:26 +01:00
}