Skip to content

dlants/magenta.nvim

Repository files navigation

magenta.nvim

   ________
  ╱        ╲
 ╱         ╱
╱         ╱
╲__╱__╱__╱
Magenta is for agents.

Jan 2025 update

video of Jan 2025 update

  • inline mode
  • context management
  • prompt caching
  • port to node

Plugin overview (Dec 2024)

video demo of the plugin

  • chat window
  • tools
  • context pinning
  • architecture overview

magenta.nvim is a plugin for leveraging LLM agents in neovim. It provides a chat window where you can talk to your AI coding assistant, as well as tools to populate context and perform inline edits. In functionality, it's similar to cursor-compose, cody or windsurf.

Rather than writing complex code to compress your repo and send it to the LLM (like a repomap in aider, etc...), magenta is built around the idea that the AI agent can choose which context to gather via tools.

Flagship models will continue to get better at tools use, and as this happens, the gap between tools like magenta and other editors that try to be clever about context management will grow smaller.

Installation

Make sure you have node installed, at least v20:

node --version

The plugin will look for configuration for providers in the following env variables:

  • anthropic: ANTHROPIC_API_KEY
  • openai: OPENAI_API_KEY, OPENAI_BASE_URL

Using lazy.nvim

{
    "dlants/magenta.nvim",
    lazy = false, -- you could also bind to <leader>mt
    build = "npm install --frozen-lockfile",
    opts = {},
},

Using vim-plug

local vim = vim
local Plug = vim.fn['plug#']

vim.call('plug#begin')
Plug('dlants/magenta.vim', {
  ['do'] = 'npm install --frozen-lockfile',
})
vim.call('plug#end')

require('magenta').setup()

Usage

keymaps

Global keymaps are set here.

Input and display buffer keymaps are set here.

Commands are all nested under :Magenta <cmd>, and can be found here.

TLDR:

  • <leader>mt is for :Magenta toggle, will toggle the sidebar on and off.
  • <leader>mp is for :Magenta paste-selection. In visual mode it will take the current selection and paste it into the input buffer.
  • <leader>mb is for :Magenta context-files with your current file. It will pin the current file to your context.
  • <leader>mf is for :Magenta context-files it allows you to select files via fzf-lua, and will pin those files to your context. This requires that fzf-lua is installed.
  • <leader>mc is for :Magenta clear, which will clear the current chat.
  • <leader>ma is for :Magenta abort, which will abort the current in-flight request.

Inline edit

  • <leader>mi is for :Magenta start-inline-edit, or start-inline-edit-selection in visual mode. This will bring up a new split where you can write a prompt to edit the current buffer. Magenta will force a find-and-replace tool use for normal mode, or force a replace tool use for the selection in visual mode.

Inline edit uses your chat history so far, so a great workflow is to build up context in the chat panel, and then use it to perform inline edits in a buffer.

display buffer

The display buffer is not modifiable, however you can interact with some parts of the display buffer by pressing <CR>. For example, you can expand the tool request and responses to see their details, and you can trigger a diff to appear on file edits.

  • hit enter on a [review] message to pull up the diff to try and edit init
  • hit enter on a tool to see the details of the request & result. Enter again on any part of the expanded view to collapse it.
  • hit enter on a piece of context to remove it

providers

The command :Magenta provider <provider> will set the current provider. Currently supported values are openai (defaults to ChatGPT 4o) and anthropic (defaults to Claude Sonnet 3.5). You can also provide configuration to setup that will choose the default provider and model. code

Any provider that has a node sdk and supports tool use should be easy to add. Contributions are welcome.

tools available to the LLM

See the most up-to-date list of implemented tools here.

  • list a directory (only in cwd, excluding hidden and gitignored files)
  • list current buffers (only buffers in cwd, excluding hidden and gitignored files)
  • get the contents of a file (requires user approval if not in cwd or hidden/gitignored)
  • get lsp diagnostics
  • get lsp references for a symbol in a buffer
  • get lsp "hover" info for a symbol in a buffer
  • insert or replace in a file (the user can then review the changes via neovim's diff mode)

Why it's cool

  • It uses the new rpc-pased remote plugin setup. This means more flexible plugin development (can easily use both lua and typescript), and no need for :UpdateRemotePlugins! (h/t wallpants).
  • The state of the plugin is managed via an elm-inspired architecture (The Elm Architecture or TEA) code. I think this makes it fairly easy to understand and lays out a clear pattern for extending the feature set, as well as eases testing. It also unlocks some cool future features (like the ability to persist a structured chat state into a file).
  • I spent a considerable amount of time figuring out a full end-to-end testing setup. Combined with typescript's async/await, it makes writing tests fairly easy and readable. The plugin is already fairly well-tested code.
  • In order to use TEA, I had to build a VDOM-like system for rendering text into a buffer. This makes writing view code declarative. code example defining a tool view
  • we can leverage existing sdks to communicate with LLMs, and async/await to manage side-effect chains, which greatly speeds up development. For example, streaming responses was pretty easy to implement, and I think is typically one of the trickier parts of other LLM plugins. code
  • smart prompt caching. Pinned files only move up in the message history when they change, which means the plugin is more likely to be able to use caching. I also implemented anthropic's prompt caching pr using an cache breakpoints.
  • I made an effort to expose the raw tool use requests and responses, as well as the stop reasons and usage info from interactions with each model. This should make debugging your workflows a lot more straightforward.

How is this different from other coding assistant plugins?

I think the closest plugins to this one are avante.nvim and codecompanion.nvim

compared to codecompanion:

Codecompanion has a single buffer, while magenta.nvim has separate input & display buffers. This makes it easier to add some interactivity to the display buffer (since it's not directly editable). I think this makes it nicer for situations when the LLM uses multiple tools at once. So for example, in codecompanion when the LLM needs permission to open a file, or proposes a change, this takes over your editor, which isn't a nice workflow when the tool needs to edit multiple files.

compared to avante:

I think it's fairly similar. However, magenta.nvim is written in typescript and uses the sdks to implement streaming, which I think makes it more stable. I think the main advantage is the architecture is very clean so it should be easy to extend the functionality. Between typescript, sdks and the architecture, I think my velocity is pretty high. I haven't used avante in a while so I'm not sure how close I got feature-wise, but it should be fairly close, and only after a couple of weeks of development time.

compared to both:

AFAIK both avante and codecompanion roll their own tool system, so the tools are defined in-prompt, and they do the parsing of the tool use themselves. I'm instead using the providers tool capabilities, like the one in anthropic. In practice I think this makes the tool use a lot more robust.

I'm not doing any treesitter analysis of symbols, dependencies, or repository summarization / repomap construction. As I mentioned in the intro, I'm opting instead to rely on the agent to explore the repo using the tools available to it. Right now that's occasionally worse than the repomap approach, but I think with time it will matter less and less.

Another thing that's probably glaringly missing is model selection and customization of keymappings, etc... I'll probably do some of this eventually, but if you use a different picker / completion plugin, or you would like to make something configurable that is not currently, I would welcome contributions.

Contributions

See the contributions guide

About

A tool-use-focused LLM plugin for neovim.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •  

Languages