local run_formatter = function(text) local split = vim.split(text, "\n") local result = table.concat(vim.list_slice(split, 2, #split - 1), "\n") -- Finds sql-format-via-python somewhere in your nvim config path local bin = vim.api.nvim_get_runtime_file("bin/sql-format-via-python.py", false)[1] local j = require("plenary.job"):new({ command = "python", args = { bin }, writer = { result }, }) return j:sync() end local embedded_sql = vim.treesitter.query.parse( "rust", [[ ([ (string_literal) (raw_string_literal) ] @sql (#match? @sql "(SELECT|select|INSERT|insert|UPDATE|update|DELETE|delete).+(FROM|from|INTO|into|VALUES|values|SET|set).*(WHERE|where|GROUP BY|group by)?") (#offset! @sql 0 1 0 -1)) ]] ) local get_root = function(bufnr) local parser = vim.treesitter.get_parser(bufnr, "rust", {}) local tree = parser:parse()[1] return tree:root() end local format_dat_sql = function(bufnr) bufnr = bufnr or vim.api.nvim_get_current_buf() if vim.bo[bufnr].filetype ~= "rust" then vim.notify("can only be used in rust") return end local root = get_root(bufnr) local changes = {} for id, node in embedded_sql:iter_captures(root, bufnr, 0, -1) do local name = embedded_sql.captures[id] if name == "sql" then -- { start row, start col, end row, end col } local range = { node:range() } local indentation = string.rep(" ", range[2]) -- Run the formatter, based on the node text local formatted = run_formatter(vim.treesitter.get_node_text(node, bufnr)) -- Add some indentation (can be anything you like!) for idx, line in ipairs(formatted) do formatted[idx] = indentation .. line end -- Keep track of changes -- But insert them in reverse order of the file, -- so that when we make modifications, we don't have -- any out of date line numbers table.insert(changes, 1, { start = range[1] + 1, final = range[3], formatted = formatted, }) end end for _, change in ipairs(changes) do vim.api.nvim_buf_set_lines(bufnr, change.start, change.final, false, change.formatted) end end vim.api.nvim_create_user_command("SqlMagic", function() format_dat_sql() end, {}) local group = vim.api.nvim_create_augroup("rust-sql-magic", { clear = true }) vim.api.nvim_create_autocmd("BufWritePre", { group = group, pattern = "*.rs", callback = function() format_dat_sql() end, })