nvf/modules/plugins/languages/java.nix
2026-06-05 15:22:32 +02:00

232 lines
6.7 KiB
Nix

{
config,
pkgs,
lib,
inputs,
...
}: let
inherit (lib.modules) mkIf mkMerge;
inherit (lib.options) literalExpression mkEnableOption mkOption;
inherit (lib.types) listOf str enum package;
inherit (lib.attrsets) attrNames genAttrs;
inherit (lib.meta) getExe getExe';
inherit (lib.nvim.types) mkGrammarOption mkPluginSetupOption enumWithRename;
cfg = config.vim.languages.java;
defaultServers = ["jdt-language-server"];
servers = ["jdt-language-server" "jls"];
defaultDebugger = "jls";
debuggers = {
jls = let
pkg = inputs.self.packages.${pkgs.stdenv.hostPlatform.system}.jls;
in {
package = pkg;
config =
# Lua
''
dap.adapters.jls= {
type = 'executable',
command = '${getExe' pkg "jls-dap"}',
}
dap.configurations.java = {
{
type = "jls",
request = "attach",
name = "Attach",
hostName = "localhost",
port = 5005,
sourceRoots = vim.fn.glob("**/src/main/java", true, true),
},
}
'';
};
};
in {
options.vim.languages.java = {
enable = mkEnableOption "Java language support";
treesitter = {
enable =
mkEnableOption "Java treesitter"
// {
default = config.vim.languages.enableTreesitter;
defaultText = literalExpression "config.vim.languages.enableTreesitter";
};
package = mkGrammarOption pkgs "java";
};
lsp = {
enable =
mkEnableOption "Java LSP support"
// {
default = config.vim.lsp.enable;
defaultText = literalExpression "config.vim.lsp.enable";
};
servers = mkOption {
type = listOf (enumWithRename
"vim.languages.java.lsp.servers"
servers
{
jdtls = "jdt-language-server";
});
default = defaultServers;
description = "Java LSP server to use";
};
};
dap = {
enable =
mkEnableOption "Java Debug Adapter"
// {
default = config.vim.languages.enableDAP;
defaultText = literalExpression "config.vim.languages.enableDAP";
};
debugger = mkOption {
type = enum (attrNames debuggers);
default = defaultDebugger;
description = ''
Java debugger to use.
**JLS**
For `jls` to work, you need to run your application with debug symbols and networking.
The `jls` configuration is hardcoded to listen on port `5005`.
This matches the configuration described [upstream](https://github.com/idelice/jls#usage).
You can change this by modifying `vim.debugger.nvim-dap.sources.java-debugger`.
```nix
vim.debugger.nvim-dap.sources.java-debugger = /* lua */ '''
dap.adapters.jls= {
type = 'executable',
command = ''\'''${getExe' inputs.self.packages.''${pkgs.stdenv.hostPlatform.system}.jls "jls-dap"}',
}
dap.configurations.java = {
{
type = "jls",
request = "attach",
name = "Attach",
hostName = "localhost",
port = 6969, -- your custom port
sourceRoots = vim.fn.glob("**/src/main/java", true, true),
},
}
''';
```
*Examples:*
- Manual:
1. Build with debug symbols.
```sh
javac -g ...
```
1. Run with debug socket.
```sh
java -agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=*:5005 -jar your.jar
```
- Springboot Maven:
For Springboot you can just pass the JVM args directly into the `spring-boot:run`.
```sh
mvn spring-boot:run -Dspring-boot.run.jvmArguments="-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=*:5005"
```
'';
};
package = mkOption {
type = package;
default = debuggers.${cfg.dap.debugger}.package;
description = "Java debugger package.";
};
};
extensions = {
maven-nvim = {
enable = mkEnableOption "maven integration";
setupOpts = mkPluginSetupOption "maven-nvim" {
mvn_executable = mkOption {
type = str;
default = getExe pkgs.maven;
defaultText = literalExpression "getExe pkgs.maven";
description = ''
The maven executable to use.
'';
example = ''
- `"mvn"`: to use the maven from the `PATH`.
- `"./mvnw"`: to use the projects maven.
- `"$${getExe pkgs.maven}"`: to use maven from a nix package.
'';
};
};
};
gradle-nvim = {
enable = mkEnableOption "gradle integration";
setupOpts = mkPluginSetupOption "gradle-nvim" {
gadle_executable = mkOption {
type = str;
default = getExe pkgs.gradle;
defaultText = literalExpression "getExe pkgs.gradle";
description = ''
The gradle executable to use.
'';
example = ''
- `"gradle"`: to use the gradle from the `PATH`.
- `"$${getExe pkgs.gradle}"`: to use gradle from a nix package.
'';
};
};
};
};
};
config = mkIf cfg.enable (mkMerge [
(mkIf cfg.treesitter.enable {
vim.treesitter.enable = true;
vim.treesitter.grammars = [cfg.treesitter.package];
})
(mkIf cfg.lsp.enable {
vim.lsp = {
presets = genAttrs cfg.lsp.servers (_: {enable = true;});
servers = genAttrs cfg.lsp.servers (_: {
filetypes = ["java"];
});
};
})
(mkIf cfg.dap.enable {
vim.debugger.nvim-dap = {
enable = true;
sources.java-debugger = debuggers.${cfg.dap.debugger}.config;
};
})
(mkIf cfg.extensions.maven-nvim.enable {
vim = mkMerge [
{
startPlugins = ["nui-nvim" "plenary-nvim"];
lazy.plugins.maven-nvim = {
package = "maven-nvim";
setupModule = "maven";
setupOpts = cfg.extensions.maven-nvim.setupOpts;
};
}
];
})
(mkIf cfg.extensions.gradle-nvim.enable {
vim = mkMerge [
{
startPlugins = ["nui-nvim" "plenary-nvim"];
lazy.plugins.gradle-nvim = {
package = "gradle-nvim";
setupModule = "gradle";
setupOpts = cfg.extensions.gradle-nvim.setupOpts;
};
}
];
})
]);
}