modules/lsp/server: declare package defaults

Convert the `attrsOf (servers.nix)` option to a freeform submodule.

Declare a `servers.nix` option for each lsp server listed in
`lsp-packages.nix` that has a known nixpkgs package.
This commit is contained in:
Matt Sturgeon 2025-04-29 02:44:19 +01:00
parent 276abde288
commit e34eaf8395
No known key found for this signature in database
GPG key ID: 4F91844CED1A8299
3 changed files with 98 additions and 29 deletions

View file

@ -2,6 +2,7 @@
lib, lib,
config, config,
options, options,
pkgs,
... ...
}: }:
let let
@ -9,6 +10,55 @@ let
inherit (lib.nixvim) toLuaObject; inherit (lib.nixvim) toLuaObject;
cfg = config.lsp; cfg = config.lsp;
# Import `server.nix` and apply args
# For convenience, we set a default here for args.pkgs
mkServerModule = args: lib.modules.importApply ./server.nix ({ inherit pkgs; } // args);
# Create a submodule type from `server.nix`
# Used as the type for both the freeform `lsp.servers.<name>`
# and the explicitly declared `lsp.servers.*` options
mkServerType = args: types.submodule (mkServerModule args);
# Create a server option
# Used below for the `lsp.servers.*` options
mkServerOption =
name: args:
let
homepage = lib.pipe options.lsp.servers [
# Get suboptions of `lsp.servers`
(opt: opt.type.getSubOptions opt.loc)
# Get suboptions of `lsp.servers.<name>`
(opts: opts.${name}.type.getSubOptions opts.${name}.loc)
# Get package option's homepage
(opts: opts.package.default.meta.homepage or null)
];
# If there's a known homepage for this language server,
# we'll link to it in the option description
nameLink = if homepage == null then name else "[${name}](${homepage})";
in
lib.mkOption {
type = mkServerType args;
description = ''
The ${nameLink} language server.
'';
default = { };
};
# Combine `packages` and `customCmd` sets from `lsp-packages.nix`
# We use this set to generate the package-option defaults
serverPackages =
let
inherit (import ../../plugins/lsp/lsp-packages.nix)
packages
customCmd
;
in
builtins.mapAttrs (name: v: {
inherit name;
package = v.package or v;
}) (packages // customCmd);
in in
{ {
options.lsp = { options.lsp = {
@ -25,7 +75,11 @@ in
}; };
servers = lib.mkOption { servers = lib.mkOption {
type = types.attrsOf (types.submodule ./server.nix); type = types.submodule {
freeformType = types.attrsOf (mkServerType { });
options = builtins.mapAttrs mkServerOption serverPackages;
};
description = '' description = ''
LSP servers to enable and/or configure. LSP servers to enable and/or configure.
@ -35,7 +89,7 @@ in
This can be installed using [`${options.plugins.lspconfig.enable}`][`plugins.lspconfig`]. This can be installed using [`${options.plugins.lspconfig.enable}`][`plugins.lspconfig`].
[nvim-lspconfig]: ${options.plugins.lspconfig.package.default.meta.homepage} [nvim-lspconfig]: ${options.plugins.lspconfig.package.default.meta.homepage}
[`plugins.lspconfig`]: ../plugins/lspconfig/index.md [`plugins.lspconfig`]: ../../plugins/lspconfig/index.md
''; '';
default = { }; default = { };
example = { example = {

View file

@ -1,59 +1,74 @@
# Usage: lib.importApply ./server.nix { /*args*/ }
{
name ? null,
package ? null,
# Avoid naming conflict with the `config` module arg
# TODO: consider renaming the `config` option to something like `settings`?
configOption ? null,
pkgs ? { },
}@args:
{ lib, name, ... }: { lib, name, ... }:
let let
inherit (lib) types; inherit (lib) types;
displayName = args.name or "the language server";
packageName = package.name or (lib.strings.removePrefix "the " displayName);
in in
{ {
options = { options = {
enable = lib.mkEnableOption "the language server"; enable = lib.mkEnableOption displayName;
name = lib.mkOption { name = lib.mkOption {
type = types.maybeRaw types.str; type = types.maybeRaw types.str;
description = '' description = ''
The name of the language server, supplied to functions like `vim.lsp.enable()`. The name to use for ${displayName}.
Supplied to functions like `vim.lsp.enable()`.
''; '';
default = name; # Use the supplied attr name, or fallback to the name module-arg
defaultText = lib.literalMD "the attribute name"; default = args.name or name;
defaultText = args.name or (lib.literalMD "the attribute name");
}; };
activate = lib.mkOption { activate = lib.mkOption {
type = types.bool; type = types.bool;
description = '' description = ''
Whether to call `vim.lsp.enable()` for this server. Whether to call `vim.lsp.enable()` for ${displayName}.
''; '';
default = true; default = true;
example = false; example = false;
}; };
package = lib.mkOption { package = lib.mkPackageOption pkgs packageName {
type = with types; nullOr package; nullable = true;
default = null; default = package.default or package;
description = '' example = package.example or null;
Package to use for this language server. extraDescription = ''
${package.extraDescription or ""}
Alternatively, the language server should be available on your `$PATH`. Alternatively, ${displayName} should be installed on your `$PATH`.
''; '';
}; };
config = lib.mkOption { config = lib.mkOption {
type = with types; attrsOf anything; type = with types; attrsOf anything;
description = '' description = ''
Configurations for each language server. Configurations for ${displayName}. ${configOption.extraDescription or ""}
''; '';
default = { }; default = { };
example = { example =
cmd = [ configOption.example or {
"clangd" cmd = [
"--background-index" "clangd"
]; "--background-index"
root_markers = [ ];
"compile_commands.json" root_markers = [
"compile_flags.txt" "compile_commands.json"
]; "compile_flags.txt"
filetypes = [ ];
"c" filetypes = [
"cpp" "c"
]; "cpp"
}; ];
};
}; };
}; };
} }

View file

@ -35,7 +35,7 @@ lib.nixvim.plugins.mkNeovimPlugin {
> setup that relate to neovim's builtin LSP and are now being moved to the > setup that relate to neovim's builtin LSP and are now being moved to the
> new [`lsp`] module. > new [`lsp`] module.
[`lsp`]: ../../lsp/servers.md [`lsp`]: ../../lsp/servers/index.md
[`plugins.lsp`]: ../lsp/index.md [`plugins.lsp`]: ../lsp/index.md
[nvim-lspconfig]: ${opts.package.default.meta.homepage} [nvim-lspconfig]: ${opts.package.default.meta.homepage}
''; '';