nix-community.nixvim/plugins/lsp/language-servers/_mk-lsp.nix
Matt Sturgeon b72ba2e4e2
plugins/lsp/servers: replace rootDir with rootMarkers
nvim-lspconfig historically used `root_dir`, along with util functions
like `root_pattern`.

Now that neovim's own LSP API is used, `root_dir` appears to be subtly
different and `root_markers` is introduced to replace `util.root_pattern`.

Since we cannot easily warn about the `root_dir` differences, it can
usually be replaced with `root_markers`, and can still be manually
configured via `extraOptions` if needed; the simplest approach here is
to remove the `rootDir` option.
2025-04-25 16:04:38 +01:00

215 lines
6.4 KiB
Nix

#########
# mkLsp #
#########
#
# Helper function that returns the Nix module `plugins.lsp.servers.<name>`.
#
{
name,
description ? "Enable ${name}.",
serverName ? name,
package ? null,
url ? null,
cmd ? null,
cmdText ? throw "cmdText is required when cmd is a function",
settings ? (cfg: cfg),
settingsOptions ? { },
extraConfig ? cfg: { },
extraOptions ? { },
...
}@args:
# returns a module
{
pkgs,
config,
options,
lib,
...
}:
let
cfg = config.plugins.lsp.servers.${name};
opts = options.plugins.lsp.servers.${name};
enabled = config.plugins.lsp.enable && cfg.enable;
inherit (lib) mkOption types;
in
{
meta.nixvimInfo = {
# TODO: description
# The package default can throw when the server is unpackaged, so use tryEval
url = args.url or (builtins.tryEval opts.package.default).value.meta.homepage or null;
path = [
"plugins"
"lsp"
"servers"
name
];
};
options = {
plugins.lsp.servers.${name} = {
enable = lib.mkEnableOption description;
package =
lib.nixvim.mkMaybeUnpackagedOption "plugins.lsp.servers.${name}.package" pkgs name
package;
cmd = mkOption {
type = with types; nullOr (listOf str);
default =
# TODO: do we really only want the default `cmd` when `package` is non-null?
if !(opts.package.isDefined or false) then
null
else if cfg.package == null then
null
else if builtins.isFunction cmd then
cmd cfg
else
cmd;
defaultText = lib.literalMD ''
null when `package` is null, otherwise ${
if args ? cmdText || builtins.isFunction cmd then
let
literal = lib.options.renderOptionValue cmdText;
inherit (literal) text;
in
if literal._type == "literalMD" then
text
else if lib.hasInfix "\n" text || lib.hasInfix "``" text then
"\n\n```\n${text}\n```"
else
"`` ${text} ``"
else if cmd == null then
"null"
else
"`[ ${lib.concatMapStringsSep " " builtins.toJSON cmd} ]`"
}
'';
description = ''
A list where each entry corresponds to the blankspace delimited part of the command that
launches the server.
The first entry is the binary used to run the language server.
Additional entries are passed as arguments.
'';
};
filetypes = lib.nixvim.mkNullOrOption (types.listOf types.str) ''
Set of filetypes for which to attempt to resolve {root_dir}.
May be empty, or server may specify a default value.
'';
autostart = lib.nixvim.defaultNullOpts.mkBool true ''
Controls if the `FileType` autocommand that launches a language server is created.
If `false`, allows for deferring language servers until manually launched with
`:LspStart` (|lspconfig-commands|).
'';
rootMarkers = lib.nixvim.defaultNullOpts.mkListOf types.str null ''
A list of files that mark the root of the project/workspace.
Vim's LSP will try to share the same language server instance for all
buffers matching `filetypes` within the same project.
A new server instance is only spawned when opening a buffer with a
different project root.
See `:h lsp-config` and `:h vim.fs.root()`.
'';
onAttach = lib.nixvim.mkCompositeOption "Server specific on_attach behavior." {
override = mkOption {
type = types.bool;
default = false;
description = "Override the global `plugins.lsp.onAttach` function.";
};
function = mkOption {
type = types.lines;
description = ''
Body of the on_attach function.
The argument `client` and `bufnr` is provided.
'';
};
};
settings = lib.nixvim.mkSettingsOption {
description = "The settings for this LSP.";
options = settingsOptions;
};
extraOptions = mkOption {
default = { };
type = types.attrsOf types.anything;
description = "Extra options for the ${name} language server.";
};
} // extraOptions;
};
config = lib.mkIf enabled {
extraPackages = [ cfg.package ];
plugins.lsp.enabledServers = [
{
name = serverName;
extraOptions = {
inherit (cfg) cmd filetypes autostart;
root_markers = cfg.rootMarkers;
on_attach = lib.nixvim.ifNonNull' cfg.onAttach (
lib.nixvim.mkRaw ''
function(client, bufnr)
${lib.optionalString (!cfg.onAttach.override) config.plugins.lsp.onAttach}
${cfg.onAttach.function}
end
''
);
settings = lib.nixvim.plugins.utils.applyExtraConfig {
extraConfig = settings;
cfg = cfg.settings;
opts = opts.settings;
enabled = true;
};
} // cfg.extraOptions;
}
];
};
imports =
let
basePath = [
"plugins"
"lsp"
"servers"
];
basePluginPath = basePath ++ [ name ];
basePluginPathString = builtins.concatStringsSep "." basePluginPath;
in
[
(lib.mkRemovedOptionModule (
basePluginPath ++ [ "extraSettings" ]
) "You can use `${basePluginPathString}.extraOptions.settings` instead.")
(lib.mkRemovedOptionModule (basePluginPath ++ [ "rootDir" ]) ''
nvim-lspconfig has switched from its own `root_dir` implementation to using neovim's built-in LSP API.
In most cases you can use `${opts.rootMarkers}` instead. It should be a list of files that mark the root of the project.
In more complex cases you can still use `${opts.extraOptions}.root_dir`.
'')
]
++ lib.optional (args ? extraConfig) (
lib.nixvim.plugins.utils.applyExtraConfig {
inherit
extraConfig
cfg
opts
enabled
;
}
)
# Add an alias (with warning) for the lspconfig server name, if different to `name`.
# Note: users may use lspconfig's docs to guess the `plugins.lsp.servers.*` name
++ (lib.optional (name != serverName) (
lib.mkRenamedOptionModule (basePath ++ [ serverName ]) basePluginPath
));
}