kickstart.nvim/lua/kickstart/plugins/debug.lua

315 lines
9.3 KiB
Lua

-- debug.lua
--
-- Shows how to use the DAP plugin to debug your code.
--
-- Primarily focused on configuring the debugger for Go, but can
-- be extended to other languages as well. That's why it's called
-- kickstart.nvim and not kitchen-sink.nvim ;)
return {
-- NOTE: Yes, you can install new plugins here!
'mfussenegger/nvim-dap',
-- NOTE: And you can specify dependencies as well
dependencies = {
-- Creates a beautiful debugger UI
'rcarriga/nvim-dap-ui',
-- Required dependency for nvim-dap-ui
'nvim-neotest/nvim-nio',
-- Installs the debug adapters for you
'mason-org/mason.nvim',
'jay-babu/mason-nvim-dap.nvim',
-- Add your own debuggers here
'leoluz/nvim-dap-go',
-- Virtual text
'theHamsta/nvim-dap-virtual-text',
-- JSON5 parser for .vscode/launch.json (supports comments and trailing commas)
'Joakker/lua-json5',
},
keys = {
-- Basic debugging keymaps, feel free to change to your liking!
{
'<F5>',
function()
require('dap').continue()
end,
desc = 'Debug: Start/Continue',
},
{
'<S-F5>',
function()
local dap = require('dap')
dap.terminate()
vim.defer_fn(function()
if dap.session() then
dap.close()
end
end, 500)
end,
desc = 'Debug: Stop/Terminate',
},
{
'<F1>',
function()
require('dap').step_into()
end,
desc = 'Debug: Step Into',
},
{
'<F2>',
function()
require('dap').step_over()
end,
desc = 'Debug: Step Over',
},
{
'<F3>',
function()
require('dap').step_out()
end,
desc = 'Debug: Step Out',
},
{
'<leader>b',
function()
require('dap').toggle_breakpoint()
end,
desc = 'Debug: Toggle Breakpoint',
},
{
'<leader>B',
function()
require('dap').set_breakpoint(vim.fn.input 'Breakpoint condition: ')
end,
desc = 'Debug: Set Breakpoint',
},
-- Toggle to see last session result. Without this, you can't see session output in case of unhandled exception.
{
'<F7>',
function()
require('dapui').toggle()
end,
desc = 'Debug: See last session result.',
},
{
'<leader>dt',
function()
require('dap').terminate()
vim.defer_fn(function()
if require('dap').session() then
require('dap').close()
end
end, 500)
end,
desc = 'Debug: Terminate Session',
},
},
config = function()
local dap = require 'dap'
local dapui = require 'dapui'
require('mason-nvim-dap').setup {
-- Makes a best effort to setup the various debuggers with
-- reasonable debug configurations
automatic_installation = true,
-- You can provide additional configuration to the handlers,
-- see mason-nvim-dap README for more information
handlers = {},
-- You'll need to check that you have the required things installed
-- online, please don't ask me how to install them :)
ensure_installed = {
-- Update this to ensure that you have the debuggers for the langs you want
'delve',
'js',
},
}
-- Enable JSON5 parser for .vscode/launch.json to support comments and trailing commas
-- nvim-dap will automatically load .vscode/launch.json configurations via dap.launch.json provider
-- This allows you to use VSCode-style launch.json files with JavaScript-style comments
local ok, json5 = pcall(require, 'json5')
if ok then
require('dap.ext.vscode').json_decode = json5.parse
end
-- Basic debugging keymaps, feel free to change to your liking!
vim.keymap.set('n', '<F5>', dap.continue, { desc = 'Debug: Start/Continue' })
vim.keymap.set('n', '<F1>', dap.step_into, { desc = 'Debug: Step Into' })
vim.keymap.set('n', '<F2>', dap.step_over, { desc = 'Debug: Step Over' })
vim.keymap.set('n', '<F3>', dap.step_out, { desc = 'Debug: Step Out' })
vim.keymap.set('n', '<leader>b', dap.toggle_breakpoint, { desc = 'Debug: Toggle Breakpoint' })
vim.keymap.set('n', '<leader>B', function()
dap.set_breakpoint(vim.fn.input 'Breakpoint condition: ')
end, { desc = 'Debug: Set Breakpoint' })
-- Terminate session (with force close after timeout)
vim.keymap.set('n', '<leader>dt', function()
dap.terminate()
vim.defer_fn(function()
if dap.session() then
dap.close()
end
end, 500)
end, { desc = 'Debug: Terminate Session' })
-- Dap virtual text setup
require('nvim-dap-virtual-text').setup()
-- Dap UI setup
-- For more information, see |:help nvim-dap-ui|
dapui.setup {
-- Set icons to characters that are more likely to work in every terminal.
-- Feel free to remove or use ones that you like more! :)
-- Don't feel like these are good choices.
icons = { expanded = '', collapsed = '', current_frame = '*' },
controls = {
icons = {
pause = '',
play = '',
step_into = '',
step_over = '',
step_out = '',
step_back = 'b',
run_last = '▶▶',
terminate = '',
disconnect = '',
},
},
layouts = {
{
elements = {
{
id = 'scopes',
size = 0.25,
},
{
id = 'breakpoints',
size = 0.25,
},
{
id = 'stacks',
size = 0.25,
},
{
id = 'watches',
size = 0.25,
},
},
position = 'left',
size = 40,
},
{
elements = {
-- {
-- id = 'repl',
-- size = 1,
-- },
{
id = 'console',
size = 1,
},
},
position = 'bottom',
size = 10,
},
},
}
-- Change breakpoint icons
-- vim.api.nvim_set_hl(0, 'DapBreak', { fg = '#e51400' })
-- vim.api.nvim_set_hl(0, 'DapStop', { fg = '#ffcc00' })
-- local breakpoint_icons = vim.g.have_nerd_font
-- and { Breakpoint = '', BreakpointCondition = '', BreakpointRejected = '', LogPoint = '', Stopped = '' }
-- or { Breakpoint = '●', BreakpointCondition = '⊜', BreakpointRejected = '⊘', LogPoint = '◆', Stopped = '⭔' }
-- for type, icon in pairs(breakpoint_icons) do
-- local tp = 'Dap' .. type
-- local hl = (type == 'Stopped') and 'DapStop' or 'DapBreak'
-- vim.fn.sign_define(tp, { text = icon, texthl = hl, numhl = hl })
-- end
dap.listeners.after.event_initialized['dapui_config'] = dapui.open
dap.listeners.after.event_terminated['dapui_config'] = dapui.close
dap.listeners.after.event_exited['dapui_config'] = dapui.close
-- Configure vscode-js-debug adapter (installed via Mason as js-debug-adapter)
-- NOTE: We configure this manually instead of using nvim-dap-vscode-js plugin
-- because that plugin is unmaintained. Direct configuration is the recommended
-- approach in 2026 and provides better control.
-- This adapter supports Node.js debugging
dap.adapters['pwa-node'] = {
type = 'server',
host = 'localhost',
port = '${port}',
executable = {
command = vim.fn.stdpath 'data' .. '/mason/bin/js-debug-adapter',
args = { '${port}' },
},
options = {
disconnect_on_terminate = true,
},
}
-- Configure debugging for JavaScript/TypeScript
local js_based_languages = { 'typescript', 'javascript', 'typescriptreact', 'javascriptreact' }
for _, language in ipairs(js_based_languages) do
dap.configurations[language] = {
{
type = 'pwa-node',
request = 'launch',
name = 'Launch Current File (Node)',
program = '${file}',
cwd = '${workspaceFolder}',
sourceMaps = true,
protocol = 'inspector',
console = 'integratedTerminal',
},
{
type = 'pwa-node',
request = 'launch',
name = 'Launch Program',
program = '${workspaceFolder}/dist/index.js',
cwd = '${workspaceFolder}',
sourceMaps = true,
protocol = 'inspector',
console = 'integratedTerminal',
preLaunchTask = 'npm: build',
},
{
type = 'pwa-node',
request = 'attach',
name = 'Attach to Process',
processId = require('dap.utils').pick_process,
cwd = '${workspaceFolder}',
sourceMaps = true,
protocol = 'inspector',
},
{
type = 'pwa-node',
request = 'launch',
name = 'Debug Jest Tests',
runtimeExecutable = 'node',
runtimeArgs = {
'./node_modules/jest/bin/jest.js',
'--runInBand',
'--no-coverage',
'--no-cache',
'--watchAll=false',
},
rootPath = '${workspaceFolder}',
cwd = '${workspaceFolder}',
console = 'integratedTerminal',
internalConsoleOptions = 'neverOpen',
sourceMaps = true,
},
}
end
end,
}