mirror of
				https://github.com/NotAShelf/nvf.git
				synced 2025-11-03 20:22:21 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			285 lines
		
	
	
	
		
			8.9 KiB
		
	
	
	
		
			Nix
		
	
	
	
	
	
			
		
		
	
	
			285 lines
		
	
	
	
		
			8.9 KiB
		
	
	
	
		
			Nix
		
	
	
	
	
	
{
 | 
						|
  config,
 | 
						|
  pkgs,
 | 
						|
  lib,
 | 
						|
  ...
 | 
						|
}: let
 | 
						|
  inherit (builtins) attrNames;
 | 
						|
  inherit (lib.options) mkEnableOption mkOption literalExpression;
 | 
						|
  inherit (lib.meta) getExe;
 | 
						|
  inherit (lib.modules) mkIf mkMerge;
 | 
						|
  inherit (lib.lists) isList toList;
 | 
						|
  inherit (lib.types) enum either listOf package str bool attrsOf;
 | 
						|
  inherit (lib.nvim.lua) expToLua;
 | 
						|
  inherit (lib) genAttrs;
 | 
						|
 | 
						|
  cfg = config.vim.languages.python;
 | 
						|
 | 
						|
  defaultServer = "basedpyright";
 | 
						|
  servers = {
 | 
						|
    pyright = {
 | 
						|
      package = pkgs.pyright;
 | 
						|
      lspConfig = ''
 | 
						|
        lspconfig.pyright.setup{
 | 
						|
          capabilities = capabilities;
 | 
						|
          on_attach = default_on_attach;
 | 
						|
          cmd = ${
 | 
						|
          if isList cfg.lsp.package.pyright
 | 
						|
          then expToLua cfg.lsp.package.pyright
 | 
						|
          else ''{"${cfg.lsp.package.pyright}/bin/pyright-langserver", "--stdio"}''
 | 
						|
        }
 | 
						|
        }
 | 
						|
      '';
 | 
						|
    };
 | 
						|
 | 
						|
    basedpyright = {
 | 
						|
      package = pkgs.basedpyright;
 | 
						|
      lspConfig = ''
 | 
						|
        lspconfig.basedpyright.setup{
 | 
						|
          capabilities = capabilities;
 | 
						|
          on_attach = default_on_attach;
 | 
						|
          cmd = ${
 | 
						|
          if isList cfg.lsp.package.basedpyright
 | 
						|
          then expToLua cfg.lsp.package.basedpyright
 | 
						|
          else ''{"${cfg.lsp.package.basedpyright}/bin/basedpyright-langserver", "--stdio"}''
 | 
						|
        }
 | 
						|
        }
 | 
						|
      '';
 | 
						|
    };
 | 
						|
 | 
						|
    ruff = {
 | 
						|
      package = pkgs.ruff;
 | 
						|
      lspConfig = ''
 | 
						|
        lspconfig.ruff.setup{
 | 
						|
          capabilities = capabilities;
 | 
						|
          on_attach = default_on_attach;
 | 
						|
          cmd = ${
 | 
						|
          if isList cfg.lsp.package.ruff
 | 
						|
          then expToLua cfg.lsp.package.ruff
 | 
						|
          else ''{"${cfg.lsp.package.ruff}/bin/ruff", "server"}''
 | 
						|
        }
 | 
						|
        }
 | 
						|
      '';
 | 
						|
    };
 | 
						|
 | 
						|
    python-lsp-server = {
 | 
						|
      package = pkgs.python3Packages.python-lsp-server;
 | 
						|
      lspConfig = ''
 | 
						|
        lspconfig.pylsp.setup{
 | 
						|
          capabilities = capabilities;
 | 
						|
          on_attach = default_on_attach;
 | 
						|
          cmd = ${
 | 
						|
          if isList cfg.lsp.package.pylsp
 | 
						|
          then expToLua cfg.lsp.package.pylsp
 | 
						|
          else ''{"${cfg.lsp.package.pylsp}/bin/pylsp"}''
 | 
						|
        }
 | 
						|
        }
 | 
						|
      '';
 | 
						|
    };
 | 
						|
  };
 | 
						|
 | 
						|
  defaultFormat = "black";
 | 
						|
  formats = {
 | 
						|
    black = {
 | 
						|
      package = pkgs.black;
 | 
						|
    };
 | 
						|
 | 
						|
    isort = {
 | 
						|
      package = pkgs.isort;
 | 
						|
    };
 | 
						|
 | 
						|
    black-and-isort = {
 | 
						|
      package = pkgs.writeShellApplication {
 | 
						|
        name = "black";
 | 
						|
        runtimeInputs = [pkgs.black pkgs.isort];
 | 
						|
        text = ''
 | 
						|
          black --quiet - "$@" | isort --profile black -
 | 
						|
        '';
 | 
						|
      };
 | 
						|
    };
 | 
						|
 | 
						|
    ruff = {
 | 
						|
      package = pkgs.writeShellApplication {
 | 
						|
        name = "ruff";
 | 
						|
        runtimeInputs = [pkgs.ruff];
 | 
						|
        text = ''
 | 
						|
          ruff format -
 | 
						|
        '';
 | 
						|
      };
 | 
						|
    };
 | 
						|
  };
 | 
						|
 | 
						|
  defaultDebugger = "debugpy";
 | 
						|
  debuggers = {
 | 
						|
    debugpy = {
 | 
						|
      # idk if this is the best way to install/run debugpy
 | 
						|
      package = pkgs.python3.withPackages (ps: with ps; [debugpy]);
 | 
						|
      dapConfig = ''
 | 
						|
        dap.adapters.debugpy = function(cb, config)
 | 
						|
          if config.request == 'attach' then
 | 
						|
            ---@diagnostic disable-next-line: undefined-field
 | 
						|
            local port = (config.connect or config).port
 | 
						|
            ---@diagnostic disable-next-line: undefined-field
 | 
						|
            local host = (config.connect or config).host or '127.0.0.1'
 | 
						|
            cb({
 | 
						|
              type = 'server',
 | 
						|
              port = assert(port, '`connect.port` is required for a python `attach` configuration'),
 | 
						|
              host = host,
 | 
						|
              options = {
 | 
						|
                source_filetype = 'python',
 | 
						|
              },
 | 
						|
            })
 | 
						|
          else
 | 
						|
            cb({
 | 
						|
              type = 'executable',
 | 
						|
              command = '${getExe cfg.dap.package}',
 | 
						|
              args = { '-m', 'debugpy.adapter' },
 | 
						|
              options = {
 | 
						|
                source_filetype = 'python',
 | 
						|
              },
 | 
						|
            })
 | 
						|
          end
 | 
						|
        end
 | 
						|
 | 
						|
        dap.configurations.python = {
 | 
						|
          {
 | 
						|
            -- The first three options are required by nvim-dap
 | 
						|
            type = 'debugpy'; -- the type here established the link to the adapter definition: `dap.adapters.debugpy`
 | 
						|
            request = 'launch';
 | 
						|
            name = "Launch file";
 | 
						|
 | 
						|
            -- Options below are for debugpy, see https://github.com/microsoft/debugpy/wiki/Debug-configuration-settings for supported options
 | 
						|
 | 
						|
            program = "''${file}"; -- This configuration will launch the current file if used.
 | 
						|
            pythonPath = function()
 | 
						|
              -- debugpy supports launching an application with a different interpreter then the one used to launch debugpy itself.
 | 
						|
              -- The code below looks for a `venv` or `.venv` folder in the current directly and uses the python within.
 | 
						|
              -- You could adapt this - to for example use the `VIRTUAL_ENV` environment variable.
 | 
						|
              local cwd = vim.fn.getcwd()
 | 
						|
              if vim.fn.executable(cwd .. '/venv/bin/python') == 1 then
 | 
						|
                return cwd .. '/venv/bin/python'
 | 
						|
              elseif vim.fn.executable(cwd .. '/.venv/bin/python') == 1 then
 | 
						|
                return cwd .. '/.venv/bin/python'
 | 
						|
              elseif vim.fn.executable("python") == 1 then
 | 
						|
                return vim.fn.exepath("python")
 | 
						|
              else -- WARNING cfg.dap.package probably has NO libraries other than builtins and debugpy
 | 
						|
                return '${getExe cfg.dap.package}'
 | 
						|
              end
 | 
						|
            end;
 | 
						|
          },
 | 
						|
        }
 | 
						|
      '';
 | 
						|
    };
 | 
						|
  };
 | 
						|
in {
 | 
						|
  options.vim.languages.python = {
 | 
						|
    enable = mkEnableOption "Python language support";
 | 
						|
 | 
						|
    treesitter = {
 | 
						|
      enable = mkEnableOption "Python treesitter" // {default = config.vim.languages.enableTreesitter;};
 | 
						|
      package = mkOption {
 | 
						|
        description = "Python treesitter grammar to use";
 | 
						|
        type = package;
 | 
						|
        default = pkgs.vimPlugins.nvim-treesitter.builtGrammars.python;
 | 
						|
      };
 | 
						|
    };
 | 
						|
 | 
						|
    lsp = {
 | 
						|
      enable = mkEnableOption "Python LSP support" // {default = config.vim.lsp.enable;};
 | 
						|
 | 
						|
      server = mkOption {
 | 
						|
        description = "Python LSP server to use either as a single server or a list of servers";
 | 
						|
        example = ''
 | 
						|
          server = "basedpyright;
 | 
						|
          server = ["basedpyright" "ruff"];
 | 
						|
        '';
 | 
						|
        type = either (enum (attrNames servers)) (listOf (enum (attrNames servers)));
 | 
						|
        default = defaultServer;
 | 
						|
      };
 | 
						|
 | 
						|
      package = mkOption {
 | 
						|
        description = "python LSP server package, or the command to run as a list of strings";
 | 
						|
        example = ''[lib.getExe pkgs.jdt-language-server "-data" "~/.cache/jdtls/workspace"]'';
 | 
						|
        type = attrsOf (either package (listOf str));
 | 
						|
        default = genAttrs (toList cfg.lsp.server) (name: servers.${name}.package);
 | 
						|
      };
 | 
						|
    };
 | 
						|
 | 
						|
    format = {
 | 
						|
      enable = mkEnableOption "Python formatting" // {default = config.vim.languages.enableFormat;};
 | 
						|
 | 
						|
      type = mkOption {
 | 
						|
        description = "Python formatter to use";
 | 
						|
        type = enum (attrNames formats);
 | 
						|
        default = defaultFormat;
 | 
						|
      };
 | 
						|
 | 
						|
      package = mkOption {
 | 
						|
        description = "Python formatter package";
 | 
						|
        type = package;
 | 
						|
        default = formats.${cfg.format.type}.package;
 | 
						|
      };
 | 
						|
    };
 | 
						|
 | 
						|
    # TODO this implementation is very bare bones, I don't know enough python to implement everything
 | 
						|
    dap = {
 | 
						|
      enable = mkOption {
 | 
						|
        description = "Enable Python Debug Adapter";
 | 
						|
        type = bool;
 | 
						|
        default = config.vim.languages.enableDAP;
 | 
						|
      };
 | 
						|
 | 
						|
      debugger = mkOption {
 | 
						|
        description = "Python debugger to use";
 | 
						|
        type = enum (attrNames debuggers);
 | 
						|
        default = defaultDebugger;
 | 
						|
      };
 | 
						|
 | 
						|
      package = mkOption {
 | 
						|
        type = package;
 | 
						|
        default = debuggers.${cfg.dap.debugger}.package;
 | 
						|
        example = literalExpression "with pkgs; python39.withPackages (ps: with ps; [debugpy])";
 | 
						|
        description = ''
 | 
						|
          Python debugger package.
 | 
						|
          This is a python package with debugpy installed, see https://nixos.wiki/wiki/Python#Install_Python_Packages.
 | 
						|
        '';
 | 
						|
      };
 | 
						|
    };
 | 
						|
  };
 | 
						|
 | 
						|
  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.lspconfig.enable = true;
 | 
						|
      vim.lsp.lspconfig.sources = genAttrs (toList cfg.lsp.server) (name: servers.${name}.lspConfig);
 | 
						|
    })
 | 
						|
 | 
						|
    (mkIf cfg.format.enable {
 | 
						|
      vim.formatter.conform-nvim = {
 | 
						|
        enable = true;
 | 
						|
        # HACK: I'm planning to remove these soon so I just took the easiest way out
 | 
						|
        setupOpts.formatters_by_ft.python =
 | 
						|
          if cfg.format.type == "black-and-isort"
 | 
						|
          then ["black"]
 | 
						|
          else [cfg.format.type];
 | 
						|
        setupOpts.formatters =
 | 
						|
          if (cfg.format.type == "black-and-isort")
 | 
						|
          then {
 | 
						|
            black.command = "${cfg.format.package}/bin/black";
 | 
						|
          }
 | 
						|
          else {
 | 
						|
            ${cfg.format.type}.command = getExe cfg.format.package;
 | 
						|
          };
 | 
						|
      };
 | 
						|
    })
 | 
						|
 | 
						|
    (mkIf cfg.dap.enable {
 | 
						|
      vim.debugger.nvim-dap.enable = true;
 | 
						|
      vim.debugger.nvim-dap.sources.python-debugger = debuggers.${cfg.dap.debugger}.dapConfig;
 | 
						|
    })
 | 
						|
  ]);
 | 
						|
}
 |