Plugin that builds on top of Neovim's |exrc| and vim.secure by providing convenient utilities to set up project-local configurations.
Neovim has a feature called |exrc| that can automatically
load .nvim.lua
/.nvimrc
/.exrc
files when starting the editor, which is great for setting up project-local
configuration. For a long time it was considered a security risk because it could lead to arbitrary code execution
(e.g. malicious .nvim.lua
in cloned git repo), but with the addition of vim.secure
this is no longer the case - Neovim manages a database of "trusted" files and, whenever loading a file which has
been modified since last time, you will be asked if you trust the file before executing any code.
While |exrc|
is useful, it is limited to only load the file on Neovim startup and does not fit workflows where
you change current directory at runtime (either manually or by loading a different session) - it assumes that you
will open new Neovim instance. This plugin serves as an alternative to the |exrc|
option, it implements it's
functionality, extends it and adds several utilities that make it easier to write your |exrc|
files in Lua.
Features include:
- Configurable name of exrc files (defaults to
.nvim.lua
) - Auto-detect path to the exrc file that is currently being executed
- Provide
source_up
for loading exrc from directories above :Exrc*
commands for managing exrc files (info, edit, load/unload/reload, ...)- Automatic telescope integration if available
- Auto-trust exrc files changed from withing Neovim
- Automatically load exrc on
VimEnter
- Automatically load exrc when changing directory with
:cd
- Setting up project-local LSP config (even
config.cmd
) when using nvim-lspconfig - Set
on_unload
cleanup hooks called when un-/reloading exrc files - TODO: register variables with UI to modify them and auto-reload exrc
- nvim-config-local unneeded as
vim.secure.trust
is now built into Neovim; does not provide.nvim.lua
helpers - neoconf.nvim limited config as it uses JSON files not Lua; can only update LSP
config.settings.*
- nlsp-settings.nvim like neoconf.nvim
Currently requires Neovim nightly.
Feel free to test it with older versions of Neovim and report issues/create PRs. Maybe it would be relatively easy to support some older releases.
Example using lazy.nvim:
{
{
'jedrzejboczar/exrc.nvim',
dependencies = {'neovim/nvim-lspconfig'}, -- (optional)
config = true,
opts = { --[[ your config ]] } ,
},
}
Run require('exrc').setup { ... }
.
If using lazy.nvim as described above just put these options in the
opts
table (or useconfig
as a function).
Available options (with default values):
require('exrc').setup {
exrc_name = '.nvim.lua', -- Name of exrc files to use
on_vim_enter = true, -- Load exrc from current directory on start
on_dir_changed = { -- Automatically load exrc files on DirChanged autocmd
enabled = true,
-- Wait until CursorHold and use vim.ui.select to confirm files to load, instead of loading unconditionally
use_ui_select = true,
},
trust_on_write = true, -- Automatically trust when saving exrc file
use_telescope = true, -- Use telescope instead of vim.ui.select for picking files (if available)
min_log_level = vim.log.levels.DEBUG, -- Disable notifications below this level (TRACE=most logs)
lsp = {
auto_setup = false, -- Automatically configure lspconfig to register on_new_config
},
commands = {
instant_edit_single = true, -- Do not use vim.ui.select if there is only 1 candidate for ExrcEdit* commands
},
}
IMPORTANT! When you use
on_vim_enter=true
then do not lazy-load this plugin.
exrc
files were originally meant to run only once at startup, so existing exrc
files in your projects
may not be written in a way that will work well when loading in other situations (or more than once).
If you feel so, then you might decide to use some other name then the default name .nvim.lua
(e.g. .nvim.local.lua
). If you do so, then this plugin will become orthogonal to the exrc
option and
there should be no problems with using both.
However, the default for this plugin is to use .nvim.lua
with the assumption that you are in control of
exrc files and you will adjust them if needed. You can disable the exrc
option (:set noexrc
) and let
exrc.nvim handle exrc files (it will set up VimEnter
and DirChanged
autocmds for loading the files).
But if you decide to still keep exrc
enabled then it shouldn't cause problems - the file won't be loaded
twice.
Create an exrc file (.nvim.lua
) in your project directory, e.g. /my/project/.nvim.lua
and initialize exrc.nvim Context, then use provided fields/methods, e.g.
local ctx = require('exrc').init()
local path_to_this_file = ctx.exrc_path
-- to load first exrc from directories above
ctx:source_up()
exrc.nvim
provides some useful commands like ExrcEdit
or ExrcInfo
.
Type :command Exrc
to see all the available commands with descriptions.
To set up local LSP configuration you must be using nvim-lspconfig (current limitation).
If not using lsp.auto_setup = true
then manually configure the on_setup
hook such that
it sets on_new_config
hook from exrc.nvim for each of your LSP setups:
local lsp_util = require('lspconfig.util')
lsp_util.on_setup = lsp_util.add_hook_before(lsp_util.on_setup, function(config, user_config)
config.on_new_config = lsp_util.add_hook_before(config.on_new_config, require('exrc.lsp').on_new_config)
end)
This on_new_config
hook will check all LSP handlers registered in exrc files and apply
updates from matching handlers whenever LSP client is started.
Now you can use lsp_setup
in your exrc files. Here is an example content of .nvim.lua
:
local ctx = require('exrc').init()
-- Pass a table with keys being LSP client names. Handlers will be executed only
-- for given client when root_dir matches (root_dir=exrc_dir or is below exrc_dir).
-- Modify config table in-place.
ctx:lsp_setup {
lua_ls = function(config, root_dir)
-- Example of changing lua_ls settings
config.settings = vim.tbl_deep_extend('force', config.settings, {
Lua = {
diagnostics = {
globals = { "vim" },
unusedLocalExclude = { '_*' },
}
}
}
end,
clangd = function(config, root_dir)
-- Example of changing LSP command to run in Docker, with volume mounts and clangd path mappings
local host_dir = vim.fs.dirname(ctx.exrc_path)
config.cmd = {
'docker', 'run',
'--interactive',
'--rm',
'--user', 'root',
'--volume', string.format('%s/.config/clangd:/root/.config/clangd', vim.env.HOME),
'--volume', string.format('%s:%s', host_dir, '/root/workdir'),
'--workdir', '/root/workdir',
'my_clangd_docker_image',
-- command to run in container
'clangd',
'--clang-tidy',
'--background-index',
'--compile-commands-dir=/root/workdir/build',
'--query-driver=/usr/bin/arm-none-eabi-*', -- ARM bare metal toolchain
-- clangd path mappings to translate between paths on host and in container
string.format('--path-mappings=%s=%s', host_dir, '/root/workdir'),
}
end,
}
You can use $GIT_DIR/info/exclude
file or configure core.excludeFiles
,
see Git documentation.
For $GIT_DIR/info/exclude
this can be symlinked somewhere else.
If you want to track your .nvim.lua in Git in separate repository, use symbolic links, e.g.
> tree ./top-dir
./top-dir
├── my-exrc-repo
│ └── my-project.lua
└── my-project
└── my-project.lua -> ../my-exrc-repo/my-project.lua
Register local tasks for (overseer.nvim)[https://github.com/stevearc/overseer.nvim]:
local ctx = require('exrc').init()
local overseer = require('overseer')
overseer.register_template {
name = 'my local task template',
condition = { dir = ctx.exrc_dir },
builder = function(params)
return {
name = 'my local task',
cwd = ctx.exrc_dir,
cmd = 'echo "running task command"',
}
end,
}
overseer.register_template {
name = 'my local complex task template',
tags = { overseer.TAG.BUILD },
params = {
-- ...
},
condition = { dir = ctx.exrc_dir },
builder = function(params)
local task = {
name = 'my local complex task',
cwd = ctx.exrc_dir,
strategy = {
'orchestrator',
tasks = {
{ 'shell', name = 'stage 1', cmd = 'echo "setting something up in directory $PWD"' },
{ 'shell', name = 'stage 2', cmd = 'echo "running build process"' },
},
},
}
return task
end,
}