Skip to content

Commit

Permalink
feat: format injected languages (stevearc#83)
Browse files Browse the repository at this point in the history
  • Loading branch information
stevearc authored Sep 29, 2023
1 parent 388d6e2 commit a5526fb
Show file tree
Hide file tree
Showing 6 changed files with 253 additions and 27 deletions.
2 changes: 2 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ Lightweight yet powerful formatter plugin for Neovim
- **Fixes bad-behaving LSP formatters** - Some LSP servers are lazy and simply replace the entire buffer, leading to the problems mentioned above. Conform hooks into the LSP handler and turns these responses into proper piecewise changes.
- **Enables range formatting for all formatters** - Since conform calculates minimal diffs, it can perform range formatting even if the underlying formatter doesn't support it.
- **Simple API** - Conform exposes a simple, imperative API modeled after `vim.lsp.buf.format()`.
- **Formats embedded code blocks** - Use the `injected` formatter to format code blocks e.g. in markdown files.

## Installation

Expand Down Expand Up @@ -192,6 +193,7 @@ To view configured and available formatters, as well as to see the log file, run
- [golines](https://github.com/segmentio/golines) - A golang formatter that fixes long lines
- [htmlbeautifier](https://github.com/threedaymonk/htmlbeautifier) - A normaliser/beautifier for HTML that also understands embedded Ruby. Ideal for tidying up Rails templates.
- [indent](https://www.gnu.org/software/indent/) - GNU Indent
- [injected](lua/conform/formatters/injected.lua) - Format treesitter injected languages.
- [isort](https://github.com/PyCQA/isort) - Python utility / library to sort imports alphabetically and automatically separate them into sections and by type.
- [jq](https://github.com/stedolan/jq) - Command-line JSON processor.
- [latexindent](https://github.com/cmhughes/latexindent.pl) - A perl script for formatting LaTeX files that is generally included in major TeX distributions.
Expand Down
1 change: 1 addition & 0 deletions doc/conform.txt
Original file line number Diff line number Diff line change
Expand Up @@ -196,6 +196,7 @@ FORMATTERS *conform-formatter
`htmlbeautifier` - A normaliser/beautifier for HTML that also understands
embedded Ruby. Ideal for tidying up Rails templates.
`indent` - GNU Indent
`injected` - Format treesitter injected languages.
`isort` - Python utility / library to sort imports alphabetically and
automatically separate them into sections and by type.
`jq` - Command-line JSON processor.
Expand Down
15 changes: 8 additions & 7 deletions doc/recipes.md
Original file line number Diff line number Diff line change
Expand Up @@ -41,11 +41,12 @@ require("conform.formatters.yamlfix").env = {
}

-- Or create your own formatter that overrides certain values
require("conform").formatters.yamlfix = vim.tbl_deep_extend("force", require("conform.formatters.yamlfix"), {
env = {
YAMLFIX_SEQUENCE_STYLE = "block_style",
},
})
require("conform").formatters.yamlfix =
vim.tbl_deep_extend("force", require("conform.formatters.yamlfix"), {
env = {
YAMLFIX_SEQUENCE_STYLE = "block_style",
},
})

-- Here is an example that modifies the command arguments for prettier to add
-- a custom config file, if it is present
Expand Down Expand Up @@ -188,8 +189,8 @@ require("conform").formatters.prettier = vim.tbl_deep_extend("force", prettier,

-- Pass append=true to append the extra arguments to the end
local deno_fmt = require("conform.formatters.deno_fmt")
require("conform").formatters.deno_fmt = vim.tbl_deep_extend('force', deno_fmt, {
args = util.extend_args(deno_fmt.args, { "--use-tabs" }, { append = true })
require("conform").formatters.deno_fmt = vim.tbl_deep_extend("force", deno_fmt, {
args = util.extend_args(deno_fmt.args, { "--use-tabs" }, { append = true }),
})

-- There is also a utility to modify a formatter in-place
Expand Down
98 changes: 98 additions & 0 deletions lua/conform/formatters/injected.lua
Original file line number Diff line number Diff line change
@@ -0,0 +1,98 @@
---@param range? conform.Range
---@param start_lnum integer
---@param end_lnum integer
---@return boolean
local function in_range(range, start_lnum, end_lnum)
return not range or (start_lnum <= range["end"][1] and range["start"][1] <= end_lnum)
end

---@type conform.FileLuaFormatterConfig
return {
meta = {
url = "lua/conform/formatters/injected.lua",
description = "Format treesitter injected languages.",
},
condition = function(ctx)
local ok = pcall(vim.treesitter.get_parser, ctx.buf)
return ok
end,
format = function(ctx, lines, callback)
local conform = require("conform")
local util = require("conform.util")
local ok, parser = pcall(vim.treesitter.get_parser, ctx.buf)
if not ok then
callback("No treesitter parser for buffer")
return
end
local root_lang = parser:lang()
local regions = {}
for lang, child_lang in pairs(parser:children()) do
local formatter_names = conform.formatters_by_ft[lang]
if formatter_names and lang ~= root_lang then
for _, tree in ipairs(child_lang:trees()) do
local root = tree:root()
local start_lnum = root:start() + 1
local end_lnum = root:end_()
if start_lnum <= end_lnum and in_range(ctx.range, start_lnum, end_lnum) then
table.insert(regions, { lang, start_lnum, end_lnum })
end
end
end
end
-- Sort from largest start_lnum to smallest
table.sort(regions, function(a, b)
return a[2] > b[2]
end)

local replacements = {}
local format_error = nil

local function apply_format_results()
if format_error then
callback(format_error)
return
end

local formatted_lines = vim.deepcopy(lines)
for _, replacement in ipairs(replacements) do
local start_lnum, end_lnum, new_lines = unpack(replacement)
for _ = start_lnum, end_lnum do
table.remove(formatted_lines, start_lnum)
end
for i = #new_lines, 1, -1 do
table.insert(formatted_lines, start_lnum, new_lines[i])
end
end
callback(nil, formatted_lines)
end

local num_format = 0
local formatter_cb = function(err, idx, start_lnum, end_lnum, new_lines)
if err then
format_error = err
else
replacements[idx] = { start_lnum, end_lnum, new_lines }
end
num_format = num_format - 1
if num_format == 0 then
apply_format_results()
end
end
local last_start_lnum = #lines + 1
for _, region in ipairs(regions) do
local lang, start_lnum, end_lnum = unpack(region)
-- Ignore regions that overlap (contain) other regions
if end_lnum < last_start_lnum then
num_format = num_format + 1
last_start_lnum = start_lnum
local input_lines = util.tbl_slice(lines, start_lnum, end_lnum)
local formatter_names = conform.formatters_by_ft[lang]
local format_opts = { async = true, bufnr = ctx.buf, quiet = true }
local idx = num_format
conform.format_lines(formatter_names, input_lines, format_opts, function(err, new_lines)
formatter_cb(err, idx, start_lnum, end_lnum, new_lines)
end)
end
end
end,
}
80 changes: 76 additions & 4 deletions lua/conform/init.lua
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ local M = {}
---@field available boolean
---@field available_msg? string

---@class (exact) conform.FormatterConfig
---@class (exact) conform.JobFormatterConfig
---@field command string|fun(ctx: conform.Context): string
---@field args? string|string[]|fun(ctx: conform.Context): string|string[]
---@field range_args? fun(ctx: conform.RangeContext): string|string[]
Expand All @@ -18,9 +18,18 @@ local M = {}
---@field exit_codes? integer[] Exit codes that indicate success (default {0})
---@field env? table<string, any>|fun(ctx: conform.Context): table<string, any>

---@class (exact) conform.FileFormatterConfig : conform.FormatterConfig
---@class (exact) conform.LuaFormatterConfig
---@field format fun(ctx: conform.Context, lines: string[], callback: fun(err: nil|string, new_lines: nil|string[]))
---@field condition? fun(ctx: conform.Context): boolean

---@class (exact) conform.FileLuaFormatterConfig : conform.LuaFormatterConfig
---@field meta conform.FormatterMeta

---@class (exact) conform.FileFormatterConfig : conform.JobFormatterConfig
---@field meta conform.FormatterMeta

---@alias conform.FormatterConfig conform.JobFormatterConfig|conform.LuaFormatterConfig

---@class (exact) conform.FormatterMeta
---@field url string
---@field description string
Expand Down Expand Up @@ -415,6 +424,56 @@ M.format = function(opts, callback)
end
end

---Process lines with formatters
---@private
---@param formatter_names string[]
---@param lines string[]
---@param opts? table
--- timeout_ms nil|integer Time in milliseconds to block for formatting. Defaults to 1000. No effect if async = true.
--- bufnr nil|integer use this as the working buffer (default 0)
--- async nil|boolean If true the method won't block. Defaults to false. If the buffer is modified before the formatter completes, the formatting will be discarded.
--- quiet nil|boolean Don't show any notifications for warnings or failures. Defaults to false.
---@param callback? fun(err: nil|string, lines: nil|string[]) Called once formatting has completed
---@return nil|string error Only present if async = false
---@return nil|string[] new_lines Only present if async = false
M.format_lines = function(formatter_names, lines, opts, callback)
---@type {timeout_ms: integer, bufnr: integer, async: boolean, quiet: boolean}
opts = vim.tbl_extend("keep", opts or {}, {
timeout_ms = 1000,
bufnr = 0,
async = false,
quiet = false,
})
callback = callback or function(_err, _lines) end
local log = require("conform.log")
local runner = require("conform.runner")
local formatters = resolve_formatters(formatter_names, opts.bufnr, not opts.quiet)
if vim.tbl_isempty(formatters) then
callback(nil, lines)
return
end

---@param err? conform.Error
---@param new_lines? string[]
local function handle_err(err, new_lines)
if err then
local level = runner.level_for_code(err.code)
log.log(level, err.message)
end
local err_message = err and err.message
callback(err_message, new_lines)
end

if opts.async then
runner.format_lines_async(opts.bufnr, formatters, nil, lines, handle_err)
else
local err, new_lines =
runner.format_lines_sync(opts.bufnr, formatters, opts.timeout_ms, nil, lines)
handle_err(err, new_lines)
return err and err.message, new_lines
end
end

---Retrieve the available formatters for a buffer
---@param bufnr? integer
---@return conform.FormatterInfo[]
Expand Down Expand Up @@ -508,13 +567,26 @@ M.get_formatter_info = function(formatter, bufnr)

local ctx = require("conform.runner").build_context(bufnr, config)

local available = true
local available_msg = nil
if config.format then
if config.condition and not config.condition(ctx) then
available = false
available_msg = "Condition failed"
end
return {
name = formatter,
command = formatter,
available = available,
available_msg = available_msg,
}
end

local command = config.command
if type(command) == "function" then
command = command(ctx)
end

local available = true
local available_msg = nil
if vim.fn.executable(command) == 0 then
available = false
available_msg = "Command not found"
Expand Down
Loading

0 comments on commit a5526fb

Please sign in to comment.