Skip to content

A neovim plugin written in lua to automatically detect your current workspace's build system and set makeprg accordingly

License

Notifications You must be signed in to change notification settings

cyuria/build.nvim

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

30 Commits
 
 
 
 
 
 
 
 

Repository files navigation

build.nvim

A neovim plugin written in lua which automatically detects the build system of your current project and sets makeprg accordingly.

Requirements

  • neovim >= 0.8

Features

  • Automatically detects your build system based on the first build file found, currently supports:
    • CMake (CMakeLists.txt)
    • Make (Makefile)
    • Meson (meson.build)
    • Zig (build.zig)
    • Cargo (Cargo.toml)
    • Setuptools (setup.py)
  • Automatically detects project root based on the presence of a file
    • Git, mercurial, svn and a few other version control systems
    • A package.json file
    • Failing all else, build.nvim searches recursively downwards for known build system files
  • Highly Customisable

Installation

Install as you would any other plugin with your least favourite package manager (optional).

vim-plug

" Vimscript
Plug "cyuria/build.nvim"

lua << EOF
require('build').setup {
    -- put your configuration here
    -- or don't, see the config section for
    -- available options and defaults
}
EOF

lazy

{
    "cyuria/build.nvim",
    opts = {}
}

packer

-- lua
use {
    "cyuria/build.nvim",
    config = function()
        require('build').setup {
            -- put your configuration here
            -- or don't, see the config section for
            -- available options and defaults
        }
    end
}

Usage

To use build.nvim, just run :make as you would with any Makefile based build system. Even if you aren't in the project root directory, build.nvim should automatically find the project root and add extra arguments for you. I.e. instead of running make build.nvim will run make -C ../PATH/TO/YOUR/PROJECT/ROOT when in a makefile based build system.

A Note for Some Build Systems

Some build systems, i.e. meson and cargo will need an extra argument, such as :make compile with meson or :make build with cargo. This allows you more control, like running :make test or :make check instead.

This can of course be changed by simply creating your own config which overrides the relevant systems. The same applies if you want to do the inverse with some other build system.

Lua API

build.nvim exposes the following lua functions and vim commands.

local build = require('build')
  • :SetMakeprg/build.set_makeprg()
    • Updates and sets the makeprg variable. Nothing will happen if no build system has been detected.
  • :SetBuildDir directory/build.set_build_dir(directory, ?root)
    • Manually overrides the build directory for the current project. Does nothing if directory is nil. Does NOT check if the build directory exists. Optionally specify a project root for the given build directory, the existence of the root directory is not checked either.

Configuration

build.nvim comes with the following options and defaults.

local opts = {
    -- Set the makeprg variable during the setup call
    set_makeprg_immediately = true,

    -- A list of autocommand events upon which to update/set the makeprg
    -- varaible
    update_on_event = { "DirChanged", },
    -- A list of files used for detecting the project root. If any one of these
    -- files or directories is detected, that directory is selected as the root
    -- directory. If none of the files are found, recursively descend the
    -- directory tree until one of them is found
    root_files = {
        ".git",
        "package.json",
        "_darcs",
        ".hg",
        ".bzr",
        ".svn",
    },
    -- A list of directories to search for which will be set as the build
    -- directory for the project
    build_dirs = {
        "build", "builddir", "bin"
    },
    -- Custom build system indicators and programs. Can also be used to
    -- overwrite existing indicators and/or programs
    extra_indicators = {},
    extra_programs = {},
    -- The path of the JSON file used to store individually set build
    -- directories on a case by case basis using the project root directory
    build_dirs_file = vim.fn.stdpath('data') .. '/build.nvim/build_directories.json',
}

Indicators

Indicators are a table mapping of file names to build system names, for example the built in configuration is:

{
    ["CMakeLists.txt"] = "cmake",
    ["Makefile"] = "make",
    ["meson.build"] = "meson",
    ["Cargo.toml"] = "cargo",
    ["build.zig"] = "zig",
    ["setup.py"] = "setuptools",
}

To add a new command, you might configure your options as follows:

{
    extra_indicators = {
        ["package.json"] = "npm",
    },
}

This would also require a program called npm.

Programs

Programs are a table mapping of build system names to lua functions similar to indicators.

For example, you might have the following configuration to add an npm build system.

{
    extra_programs = {
        npm = function (root, build)
            return "npm run $*"
        end,
    }
}

Programs are called with two arguments

root The project root directory
build The project's build directory

For example, a file hierarchy for a project might look something like this:

myapp
├── LICENSE
├── Makefile
├── README.md
├── build
│   ├── gui.o
│   ├── myapp
│   ├── main.o
│   ├── mylib.so
│   └── utils.o
├── docs
│   ├── index.rst
│   └── installation.rst
├── examples
│   ├── a_big_box.c
│   ├── hello_world.c
│   └── README.md
├── src
│   ├── gui.c
│   ├── main.c
│   └── utils.c
├── h
│   ├── gui.h
│   └── utils.h
└── tests
    ├── test_hashmap.c
    └── test_fuzz.c

In this case, the build system would be "make", the "root" directory myapp/ and the "build" directory myapp/build/.

Note that build.nvim uses absolute paths wherever possible.

Contributing

Please feel free to submit any PRs or issues, especially if they add support for more build systems. It would be great if we could support every build system under the sun.

To add a build system just add the relevant entries from the extra_indicators and extra_programs sections of your own configuration to lua/build/systems.lua

Some currently unsupported systems which would be nice to support are

  • build2
  • Maven or Gradle or something java I don't know
  • Bazel
  • Better support for the different variations of Make (i.e. GNU make vs BSD make vs NMake etc)
  • Ninja

Another cool feature I've been thinking about but been too lazy to implement is adding a default make command for meson/cargo etc, so that you can run :make and it will default to :make build or whatever equivalent. If you want to open a PR for this that would be incredible.

About

A neovim plugin written in lua to automatically detect your current workspace's build system and set makeprg accordingly

Topics

Resources

License

Stars

Watchers

Forks

Languages