panache 2.19.0

An LSP, formatter, and linter for Pandoc markdown, Quarto, and RMarkdown
# Panache <img src='https://raw.githubusercontent.com/jolars/panache/refs/heads/main/images/logo.png' align="right" width="139" />

[![Build and
Test](https://github.com/jolars/panache/actions/workflows/build-and-test.yml/badge.svg)](https://github.com/jolars/panache/actions/workflows/build-and-test.yml)
[![Crates.io](https://img.shields.io/crates/v/panache.svg)](https://crates.io/crates/panache)
[![Open
VSX](https://img.shields.io/open-vsx/v/jolars/panache)](https://open-vsx.org/extension/jolars/panache)
[![VS
Code](https://img.shields.io/visual-studio-marketplace/v/jolars.panache)](https://marketplace.visualstudio.com/items?itemName=jolars.panache)
[![codecov](https://codecov.io/gh/jolars/panache/graph/badge.svg?token=uaBVOBfILv)](https://codecov.io/gh/jolars/panache)

A language server, formatter, and linter for Pandoc, Quarto, and R Markdown.

## Work in Progress

This project is in early development. Expect bugs, missing features, and
breaking changes.

## Installation

### From crates.io

If you have Rust installed, the easiest way is likely to install from
[crates.io](https://crates.io/crates/panache):

```bash
cargo install panache
```

### Pre-built Binaries

Alternatively, you can install pre-built binary packages from the [releases
page](https://github.com/jolars/panache/releases) for Linux, macOS, and Windows.
For Linux, packages are available for generic distributions (tarballs) as well
as Debian/Ubuntu (`.deb`) and Fedora/RHEL/openSUSE (`.rpm`).

### Arch Linux

There is also an Arch Linux package available in the AUR:
[panache-bin](https://aur.archlinux.org/packages/panache-bin/).

```bash
yay -S panache-bin # or your preferred AUR helper
```

### VS Code Extension

If you are running VS Code or an editor that supports VS
Code extensions (like Positron), you can install the [Panache
extension](https://marketplace.visualstudio.com/items?itemName=jolars.panache)
from the VS Code Marketplace or the [Open VSX
extension](https://open-vsx.org/extension/jolars/panache), which will
automatically also install the `panache` CLI and start the language server when
editing supported files.

## Usage

Panache provides a single CLI interface for formatting, linting, and running the
LSP server.

### Formatting

To format a file in place, simply run:

```bash
panache format document.qmd
```

You can also format from stdin by piping content into `panache format`:

```bash
cat <file> | panache format
```

`panache format` supports glob patterns and recursive directory formatting:

```bash
panache format **/*.{qmd,md}
```

You can use Panache as a linter via the `--check` flag to check if files are
already formatted without making changes:

```bash
panache format --check document.qmd
```

#### External Code Formatters

Panache supports external formatters for code blocks. For example, you can
configure it to run `air` on R code blocks and `ruff` on Python code blocks:

```toml
[formatters]
r = "air"
python = "ruff"
javascript = "prettier"
typescript = "prettier" # Reuse same formatter
```

You can setup custom formatters or modify built-in presets with additional
arguments:

```toml
[formatters]
python = ["isort", "black"]
javascript = "foobar"

[formatters.isort]
args = ["--profile=black"]

[formatters.myformatters]
cmd = "foobar"
args = ["--print-width=100"]
stdin = true
```

### Linting

Panache also features a linter that can report formatting issues and optionally
auto-fix them. To run the linter, use:

```bash
panache lint document.qmd
```

As with `panache format`, you can use glob patterns and recursive formatting:

```bash
panache lint **/*.{qmd,md}
```

#### External Linters

As with formatting, Panache supports external linters for code blocks. These
are configured in the `[linters]` section of the configuration, but due to the
complexity of linting, including dealing with auto-fixing, external linters
cannot be customized and only support presets and at the moment only support R
via the `jarl` linter:

```toml
# Enable R linting
[linters]
r = "jarl" # R linter with JSON output
```

### Language Server

Panache implements the language server protocol (LSP) to provide editor features
like formatting, diagnostics, code actions, and more.

To start the language server, run:

```bash
panache lsp
```

Most users will want to configure their editor to start the
language server automatically when editing. See [the language
server documentation](https://jolars.github.io/panache/lsp)
for guides on configuring your editor. For VS Code and related
editors such as Positron, there is a [VS Code Marketplace
extension](https://marketplace.visualstudio.com/items?itemName=jolars.panache)
and [Open VSX extension](https://open-vsx.org/extension/jolars/panache) that
provides an LSP client and additional features.

The server communicates over stdin/stdout and provides document formatting
capabilities. For Quarto projects, the LSP also reads `_quarto.yml`,
per-directory `_metadata.yml`, and `metadata-files` includes to supply
project-level metadata to bibliography-aware features. For Bookdown,
`_bookdown.yml`, `_output.yml`, and `index.Rmd` frontmatter are also considered.

The list of LSP features supported by Panache includes, among others:

- Document formatting (full document, incremental and range)
- Diagnostics with quick fixes
- Code actions for refactoring
  - Convert between loose/compact lists
  - Convert between inline/reference footnotes
- Document symbols/outline
- Folding ranges
- Go to definition for references and footnotes

## Configuration

Panache looks for a configuration in:

1. `.panache.toml` or `panache.toml` in current directory or parent directories
2. `$XDG_CONFIG_HOME/panache/config.toml` (usually
   `~/.config/panache/config.toml`)

### Example

```toml
# Markdown flavor and line width
flavor = "quarto"
line-width = 80
line-ending = "auto"

# Formatting style
[style]
wrap = "reflow"

# External code formatters (opt-in)
[formatters]
python = ["isort", "black"] # Sequential formatting
r = "air"                   # Built-in preset
javascript = "prettier"     # Reusable definitions
typescript = "prettier"
yaml = "yamlfmt"            # Formats both code blocks AND frontmatter

# Customize formatters
[formatters.prettier]
prepend-args = ["--print-width=100"]

# External code linters
[linters]
r = "jarl" # Enable R linting
```

See `.panache.toml.example` for a complete configuration reference.

### Pre-commit Hooks

Panache integrates with [pre-commit](https://pre-commit.com/) to automatically
format and lint your files before committing.

**Installation:**

First, install pre-commit if you haven't already:

```bash
pip install pre-commit
# or
brew install pre-commit
```

Then add Panache to your `.pre-commit-config.yaml`:

```yaml
repos:
  - repo: https://github.com/jolars/panache
    rev: v2.16.0 # Use the latest version
    hooks:
      - id: panache-format # Format files
      - id: panache-lint # Lint and auto-fix issues
```

Install the hooks:

```bash
pre-commit install
```

Panache will now automatically run on your staged `.qmd`, `.md`, and `.Rmd`
files before each commit.

See [examples/pre-commit-config.yaml](examples/pre-commit-config.yaml) for more
configuration options.

## Motivation

I wanted a formatter that understands Quarto and Pandoc syntax. I have tried to
use Prettier as well as mdformat, but both fail to handle some of the particular
syntax used in Quarto documents, such as fenced divs and some of the table
syntax.

## Design Goals and Scope

- Full LSP implementation with formatting, diagnostics, code actions, and more
- Standalone CLI for both formatting and linting
- Support for Quarto, Pandoc, and R Markdown syntax
- Lossless CST-based parsing
- Idempotent formatting
- Semi-opinionated defaults with configurable style options for common
  formatting decisions
- Support for running external formatters and linters on code blocks, with
  built-in presets for popular languages and tools

## Acknowledgements

The development of Panache has simplified considerably thanks to the extensive
documentation, well-structured code, and testing infrastructure provided by
Pandoc. We also owe significant debt to the rust-analyzer project, on which
Panche is heavily inspired.