{ stdenv, symlinkJoin, lib, makeWrapper, writeText, nodePackages, python3, python3Packages, callPackage, neovimUtils, vimUtils, vimPlugins, perl, lndir, lazyUtils, }: neovim-unwrapped: let wrapper = { extraName ? "", # should contain all args but the binary. Can be either a string or list wrapperArgs ? [ ], # a limited RC script used only to generate the manifest for remote plugins manifestRc ? null, withPython2 ? false, withPython3 ? true, python3Env ? python3, withNodeJs ? false, withPerl ? false, rubyEnv ? null, vimAlias ? false, viAlias ? false, # vimL code that should be sourced as part of the generated init.lua file neovimRcContent ? null, # lua code to put into the generated init.lua file luaRcContent ? "", lazyPlugins ? [ ], lazyConfig ? { }, vPlug ? vimPlugins, ... }: assert withPython2 -> throw "Python2 support has been removed from the neovim wrapper, please remove withPython2 and python2Env."; stdenv.mkDerivation ( finalAttrs: let lazyUtilsOverride = lazyUtils.override { inherit vPlug; }; rcContent = # lua '' ${luaRcContent} vim.opt.rtp:prepend("${lazyUtilsOverride.lazyPluginsJoin { inherit lazyConfig lazyPlugins; }}") require("lazyWrapper") '' + lib.optionalString (!isNull neovimRcContent) '' vim.cmd.source "${writeText "init.vim" neovimRcContent}" ''; wrapperArgsStr = if lib.isString wrapperArgs then wrapperArgs else lib.escapeShellArgs wrapperArgs; generatedWrapperArgs = # vim accepts a limited number of commands so we join them all [ "--add-flags" ''--cmd "lua ${providerLuaRc}"'' # (lib.intersperse "|" hostProviderViml) ]; providerLuaRc = neovimUtils.generateProviderRc { inherit withPython3 withNodeJs withPerl; withRuby = rubyEnv != null; }; # If configure != {}, we can't generate the rplugin.vim file with e.g # NVIM_SYSTEM_RPLUGIN_MANIFEST *and* NVIM_RPLUGIN_MANIFEST env vars set in # the wrapper. That's why only when configure != {} (tested both here and # when postBuild is evaluated), we call makeWrapper once to generate a # wrapper with most arguments we need, excluding those that cause problems to # generate rplugin.vim, but still required for the final wrapper. finalMakeWrapperArgs = [ "${neovim-unwrapped}/bin/nvim" "${placeholder "out"}/bin/nvim" ] ++ [ "--set" "NVIM_SYSTEM_RPLUGIN_MANIFEST" "${placeholder "out"}/rplugin.vim" ] ++ [ "--add-flags" "-u ${writeText "init.lua" rcContent}" ] ++ finalAttrs.generatedWrapperArgs; perlEnv = perl.withPackages (p: [ p.NeovimExt p.Appcpanminus ]); pname = "neovim"; version = lib.getVersion neovim-unwrapped; in { name = "${pname}-${version}${extraName}"; inherit pname version; __structuredAttrs = true; dontUnpack = true; inherit viAlias vimAlias withNodeJs withPython3 withPerl ; inherit providerLuaRc lazyConfig; inherit python3Env rubyEnv; withRuby = rubyEnv != null; inherit wrapperArgs generatedWrapperArgs; luaRcContent = rcContent; # Remove the symlinks created by symlinkJoin which we need to perform # extra actions upon postBuild = lib.optionalString stdenv.isLinux '' rm $out/share/applications/nvim.desktop substitute ${neovim-unwrapped}/share/applications/nvim.desktop $out/share/applications/nvim.desktop \ --replace 'Name=Neovim' 'Name=Neovim wrapper' '' + lib.optionalString finalAttrs.withPython3 '' makeWrapper ${python3Env.interpreter} $out/bin/nvim-python3 --unset PYTHONPATH --unset PYTHONSAFEPATH '' + lib.optionalString (finalAttrs.rubyEnv != null) '' ln -s ${finalAttrs.rubyEnv}/bin/neovim-ruby-host $out/bin/nvim-ruby '' + lib.optionalString finalAttrs.withNodeJs '' ln -s ${nodePackages.neovim}/bin/neovim-node-host $out/bin/nvim-node '' + lib.optionalString finalAttrs.withPerl '' ln -s ${perlEnv}/bin/perl $out/bin/nvim-perl '' + lib.optionalString finalAttrs.vimAlias '' ln -s $out/bin/nvim $out/bin/vim '' + lib.optionalString finalAttrs.viAlias '' ln -s $out/bin/nvim $out/bin/vi '' + lib.optionalString (manifestRc != null) ( let manifestWrapperArgs = [ "${neovim-unwrapped}/bin/nvim" "${placeholder "out"}/bin/nvim-wrapper" ] ++ finalAttrs.generatedWrapperArgs; in '' echo "Generating remote plugin manifest" export NVIM_RPLUGIN_MANIFEST=$out/rplugin.vim makeWrapper ${lib.escapeShellArgs manifestWrapperArgs} ${wrapperArgsStr} # Some plugins assume that the home directory is accessible for # initializing caches, temporary files, etc. Even if the plugin isn't # actively used, it may throw an error as soon as Neovim is launched # (e.g., inside an autoload script), causing manifest generation to # fail. Therefore, let's create a fake home directory before generating # the manifest, just to satisfy the needs of these plugins. # # See https://github.com/Yggdroot/LeaderF/blob/v1.21/autoload/lfMru.vim#L10 # for an example of this behavior. export HOME="$(mktemp -d)" # Launch neovim with a vimrc file containing only the generated plugin # code. Pass various flags to disable temp file generation # (swap/viminfo) and redirect errors to stderr. # Only display the log on error since it will contain a few normally # irrelevant messages. if ! $out/bin/nvim-wrapper \ -u ${writeText "manifest.vim" manifestRc} \ -i NONE -n \ -V1rplugins.log \ +UpdateRemotePlugins +quit! > outfile 2>&1; then cat outfile echo -e "\nGenerating rplugin.vim failed!" exit 1 fi rm "${placeholder "out"}/bin/nvim-wrapper" '' ) + '' rm $out/bin/nvim touch $out/rplugin.vim makeWrapper ${lib.escapeShellArgs finalMakeWrapperArgs} ${wrapperArgsStr} ''; buildPhase = '' runHook preBuild mkdir -p $out for i in ${neovim-unwrapped}; do lndir -silent $i $out done runHook postBuild ''; preferLocalBuild = true; nativeBuildInputs = [ makeWrapper lndir ]; passthru = { inherit providerLuaRc lazyConfig; unwrapped = neovim-unwrapped; initRc = neovimRcContent; # tests = callPackage ./tests {}; }; meta = neovim-unwrapped.meta // { # To prevent builds on hydra hydraPlatforms = [ ]; # prefer wrapper over the package priority = (neovim-unwrapped.meta.priority or 0) - 1; }; } ); in lib.makeOverridable wrapper