biome-developer
Purpose
This skill provides general development best practices, common gotchas, and Biome-specific patterns that apply across different areas of the codebase. Use this as a reference when you encounter unfamiliar APIs or need to avoid common mistakes.
Prerequisites
- Basic familiarity with Rust
- Understanding of Biome's architecture (parser, analyzer, formatter)
- Development environment set up (see CONTRIBUTING.md)
Common Gotchas and Best Practices
Working with AST and Syntax Nodes
DO:
- Use parser crate's
quick_testto inspect AST structure before implementing - Understand the node hierarchy and parent-child relationships
- Check both general cases AND specific types (e.g., Vue has both
VueDirectiveandVueV*ShorthandDirective) - Verify your solution works for all relevant variant types, not just the first one you find
DON'T:
- Do NOT build the full Biome binary just to inspect syntax (expensive) - use parser crate's
quick_testinstead - Do NOT assume syntax patterns without inspecting the AST first
Example - Inspecting AST:
// In crates/biome_html_parser/tests/quick_test.rs
// Modify the quick_test function:
#[test]
pub fn quick_test() {
let code = r#"<button on:click={handleClick}>Click</button>"#;
let source_type = HtmlFileSource::svelte();
let options = HtmlParserOptions::from(&source_type);
let root = parse_html(code, options);
dbg!(&root.syntax()); // Shows full AST structure
}
Run: just qt biome_html_parser
String Extraction and Text Handling
DO:
- Use
inner_string_text()when extracting content from quoted strings (removes quotes) - Use
text_trimmed()when you need the full token text without leading/trailing whitespace - Use
token_text_trimmed()on nodes likeHtmlAttributeNameto get the text content - Verify whether values use
HtmlString(quotes) orHtmlTextExpression(curly braces)
DON'T:
- Do NOT use
text_trimmed()when you needinner_string_text()for extracting quoted string contents
Example - String Extraction:
// WRONG: text_trimmed() includes quotes
let html_string = value.as_html_string()?;
let content = html_string.value_token()?.text_trimmed(); // Returns: "\"handler\""
// CORRECT: inner_string_text() removes quotes
let html_string = value.as_html_string()?;
let inner_text = html_string.inner_string_text().ok()?;
let content = inner_text.text(); // Returns: "handler"
Working with Embedded Languages
DO:
- Verify changes work for different value formats (quoted strings vs text expressions) when handling multiple frameworks
- Use appropriate
EmbeddingKindfor context (Vue, Svelte, Astro, etc.) - Check if embedded content needs
is_source: true(script tags) vsis_source: false(template expressions) - Calculate offsets correctly: token start + 1 for opening quote, or use
text_range().start()for text expressions
DON'T:
- Do NOT assume all frameworks use the same syntax (Vue uses quotes, Svelte uses curly braces)
- Do NOT implement features for "widely used" patterns without evidence - ask the user first
Example - Different Value Formats:
// Vue directives use quoted strings: @click="handler"
let html_string = value.as_html_string()?;
let inner_text = html_string.inner_string_text().ok()?;
// Svelte directives use text expressions: on:click={handler}
let text_expression = value.as_html_attribute_single_text_expression()?;
let expression = text_expression.expression().ok()?;
Borrow Checker and Temporary Values
DO:
- Use intermediate
letbindings to avoid temporary value borrows that get dropped - Store method results that return owned values before calling methods on them
DON'T:
- Do NOT create temporary value borrows that get dropped before use
Example - Avoiding Borrow Issues:
// WRONG: Temporary borrow gets dropped
let html_string = value.value().ok()?.as_html_string()?;
let token = html_string.value_token().ok()?; // ERROR: html_string dropped
// CORRECT: Store intermediate result
let value_node = value.value().ok()?;
let html_string = value_node.as_html_string()?;
let token = html_string.value_token().ok()?; // OK
Clippy and Code Style
DO:
- Use
letchains to collapse nestedif letstatements (cleaner and follows Rust idioms) - Run
just lbefore committing to catch clippy warnings - Fix clippy suggestions unless there's a good reason not to
DON'T:
- Do NOT ignore clippy warnings - they often catch real issues or suggest better patterns
Example - Collapsible If:
// WRONG: Nested if let (clippy::collapsible_if warning)
if let Some(directive) = VueDirective::cast_ref(&element) {
if let Some(initializer) = directive.initializer() {
// ... do something
}
}
// CORRECT: Use let chains
if let Some(directive) = VueDirective::cast_ref(&element)
&& let Some(initializer) = directive.initializer()
{
// ... do something
}
Code Comments
Comments exist for the next developer who reads this code, not for the developer currently writing it. Write them like you are explaining the code to a colleague who walked into the room ten minutes ago — not to a reviewer on this specific PR.
DO:
- Explain code that is hard to read, or document exceptions and edge cases
- Provide context when names alone are not descriptive enough
- Describe the business logic a function implements
- Clarify contextual words like "normalize" — e.g., "normalize a file path" and "normalize a URL" mean different things; spell out what normalization means here
- Strike a balance between plain English and technical precision. Prefer concrete nouns ("the HTML file", "the
<style>block") over abstract ones ("the host CST", "the delegated pipeline") when both convey the same idea - Add comments only where they are needed, for example, docstrings, or code paths that are particular and require a special explanation. Most of the code (even new that you write) doesn't need a comment if it follows the business logic.
- Write comments using proper English grammar and punctuation.
DON'T:
- Do NOT embed the context of the current work into comments. A comment like
// As per issue #1234, we skip this caseties the code to a transient artifact. Instead, explain why the case is skipped in terms any future reader would understand. - Do NOT scope comments to the specific trigger that prompted the change. For example, if a bug was reported for Astro but the fix applies broadly, do NOT write
// Fix for Astro embedding. Write a comment that describes the general condition being handled. - Do NOT scope comments narrower than the code itself. If the function is generic across all embedded languages, the comment should not name "CSS" or "
<style>" — describe the contract the code enforces for any embed, and use a concrete example only as illustration. - Do NOT lead with formal-methods / math jargon like
// Invariant:,// Precondition:,// Lemma:unless the surrounding code genuinely uses those terms. For most Biome code, plain prose ("When X happens, Y must hold, otherwise …") reads better and is just as precise. - Do NOT pile technical terms on top of each other ("delegated format pipeline", "canonical embed IR", "host CST token text") when one plain-English sentence would do. Jargon density should be low; a reader should not need a glossary to understand a comment.
- Do NOT just paraphrase the function name or the next line of code. If a comment can be deleted without losing information, delete it.
Think big picture, not current task. Before writing a comment, ask three things:
- If someone reads this a year from now with no knowledge of the issue or PR, does this comment give them the context they need?
- Is my comment describing the code at the same level of abstraction as the code? (A generic helper deserves a generic explanation; a specific branch deserves a specific one.)
- Could I swap any technical term for a plainer word without losing meaning? If yes, swap it.
Example 1 — issue/task context and over-specificity:
// WRONG: Carries issue/task context
// Fix for #5678: Astro files need special handling here
if is_embedded_script(node) {
return normalize_offset(node);
}
// WRONG: Describes what the code does (the code already says that)
// Check if the node is an embedded script and normalize the offset
if is_embedded_script(node) {
return normalize_offset(node);
}
// CORRECT: Explains why and clarifies "normalize"
// Embedded script blocks (e.g. <script> inside .vue/.svelte/.astro files)
// report offsets relative to the embedding document, not the script itself.
// Normalize here means: subtract the script block's start position so the
// offset is relative to the script content.
if is_embedded_script(node) {
return normalize_offset(node);
}
Example 2 — jargon, narrow scope, and abstraction mismatch. This is a real example from a generic helper that replaces an embedded snippet inside any host document (HTML, Vue, Svelte, Astro, …):
// WRONG: starts with formal-methods jargon, names a specific case
// (`<style>`) even though the function handles any embed, and stacks
// technical terms ("host CST token text", "delegated pipeline") that a
// new reader has to decode before they can understand the point.
// Invariant: for a file that required no fix actions, `fix_file` and
// `format_file` must produce byte-identical output. For `<style>`
// blocks, `fix_all`'s final format pass prints embedded content
// verbatim from the host CST token text, while `format_file` routes
// through the delegated `format_embedded` pipeline and re-wraps the
// result with the host's indent. …
// CORRECT: plain language, stays at the generic level of the function,
// uses `<style>` only as a parenthetical example, and is understandable
// without prior context.
// The embedded formatter (e.g. the CSS formatter for a <style> block)
// doesn't know how deeply its code is nested inside the HTML file, so
// it always returns the code indented from column zero. If we pasted
// that code back as-is, only the first line would get the HTML
// indentation (from the leading whitespace we already captured); every
// other line would end up too far to the left. Add the same indentation
// to every line so the embed lines up with its surroundings.
The corrected comment names one concrete example (<style> / CSS) to make the reader's mental picture vivid, but the rest of the sentence is generic enough to cover any host/embed pair. That is the balance to aim for.
Cargo Dependencies: workspace = true vs path = "..."
Internal biome_* crates listed under [dev-dependencies] MUST use path = "../<crate_name>", not workspace = true. Using workspace = true for dev-dependencies can cause Cargo to resolve the crate from the registry instead of the local workspace, which is incorrect.
Regular [dependencies] still use workspace = true as normal — this rule only applies to [dev-dependencies].
DO:
- Use
path = "../biome_foo"for allbiome_*dev-dependencies - Preserve any extra attributes like
featureswhen converting
DON'T:
- Do NOT use
workspace = trueforbiome_*crates in[dev-dependencies]
Example:
# WRONG: may resolve from registry
[dev-dependencies]
biome_js_parser = { workspace = true }
biome_formatter = { workspace = true, features = ["countme"] }
# CORRECT: always resolves locally
[dev-dependencies]
biome_js_parser = { path = "../biome_js_parser" }
biome_formatter = { path = "../biome_formatter", features = ["countme"] }
All crates live as siblings under crates/, so the relative path is always ../biome_<name>.
Legacy and Deprecated Syntax
DO:
- Ask users before implementing deprecated/legacy syntax support
- Wait for user demand before spending time on legacy features
- Document when features are intentionally not supported due to being legacy
DON'T:
- Do NOT implement legacy/deprecated syntax without checking with the user first
- Do NOT claim patterns are "widely used" or "common" without evidence
Example:
Svelte's on:click event handler syntax is legacy (Svelte 3/4). Modern Svelte 5 runes mode uses regular attributes. Unless users specifically request it, don't implement legacy syntax support.
Testing and Development
For testing commands, snapshot workflows, and code generation, see the testing-codegen skill. Key reminders specific to Biome development patterns:
- Test with multiple variants when working with enums (e.g., all
VueV*ShorthandDirectivetypes) - Use CLI tests for testing embedded languages (Vue/Svelte directives, etc.)
- Do NOT try to test embedded languages in analyzer packages (they don't have embedding capabilities)
Pattern Matching Tips
Working with Node Variants
When working with enum variants (like AnySvelteDirective), check if there are also non-enum types that need handling:
// Check AnySvelteDirective enum (bind:, class:, style:, etc.)
if let Some(directive) = AnySvelteDirective::cast_ref(&element) {
// Handle special Svelte directives
}
// But also check regular HTML attributes with specific prefixes
if let Some(attribute) = HtmlAttribute::cast_ref(&element) {
if let Ok(name) = attribute.name() {
// Some directives might be parsed as regular attributes
}
}
Checking Multiple Variant Types
For frameworks with multiple directive syntaxes, handle each type:
// Vue has multiple shorthand types
if let Some(directive) = VueVOnShorthandDirective::cast_ref(&element) {
// Handle @click
}
if let Some(directive) = VueVBindShorthandDirective::cast_ref(&element) {
// Handle :prop
}
if let Some(directive) = VueVSlotShorthandDirective::cast_ref(&element) {
// Handle #slot
}
if let Some(directive) = VueDirective::cast_ref(&element) {
// Handle v-if, v-show, etc.
}
Common API Confusion
String/Text Methods
| Method | Use When | Returns |
|---|---|---|
inner_string_text() |
Extracting content from quoted strings | Content without quotes |
text_trimmed() |
Getting token text without whitespace | Full token text |
token_text_trimmed() |
Getting text from nodes like HtmlAttributeName |
Node text content |
text() |
Getting raw text | Exact text as written |
Value Extraction Methods
| Type | Method | Framework |
|---|---|---|
HtmlString |
inner_string_text() |
Vue (quotes) |
HtmlAttributeSingleTextExpression |
expression() |
Svelte (curly braces) |
HtmlTextExpression |
html_literal_token() |
Template expressions |
References
- Main contributing guide:
../../CONTRIBUTING.md - Testing workflows:
../testing-codegen/SKILL.md - Parser development:
../parser-development/SKILL.md - Biome internals docs: https://biomejs.dev/internals
Documentation and Markdown Formatting
DO:
- Use spaces around table separators:
| --- | --- | --- |(not|---|---|---|) - Ensure all Markdown tables follow "compact" style with proper spacing
- Test documentation changes with markdown linters before committing
DON'T:
- Do NOT use compact table separators without spaces (causes CI linting failures)
Example - Table Formatting:
<!-- WRONG: No spaces around separators -->
| Method | Use When | Returns |
|--------|----------|---------|
<!-- CORRECT: Spaces around separators -->
| Method | Use When | Returns |
| --- | --- | --- |
The CI uses markdownlint-cli2 which enforces the "compact" style requiring spaces.
Common Mistakes to Avoid
- Calling
format!()(allocates a string) when formatting strings in amarkup!block.markup!supports interpolation, E.g.markup! { "Hello, "{name}"!" }. - Calling
.to_string()or.to_string_trimmed()(allocates a string) on aSyntaxTokenorSyntaxNode. It's highly unlikely that you actually need to call these methods on a syntax node. As for syntax tokens, you can easily borrow a&strfrom the token's text without allocating a new string, usingtoken.text().
When to Use This Skill
Load this skill when:
- Working with unfamiliar Biome APIs
- Getting borrow checker errors with temporary values
- Extracting strings or text from syntax nodes
- Implementing support for embedded languages (Vue, Svelte, etc.)
- Wondering why your AST inspection doesn't match expectations
- Making decisions about legacy/deprecated syntax support
- Writing or updating markdown documentation
More from biomejs/biome
parser-development
Guide for implementing parsers with error recovery for new languages in Biome. Use when adding parsing support for a new language, implementing error recovery in a parser, or writing grammar definitions in .ungram format for JavaScript, CSS, JSON, HTML, GraphQL, or other languages.
80lint-rule-development
Step-by-step guide for creating and implementing lint rules in Biome's analyzer. Use when implementing rules like noVar, useConst, or any custom lint/assist rule, adding code actions to fix diagnostics, implementing semantic analysis for binding references, or adding configurable options to rules.
74formatter-development
Guide for implementing formatting rules using Biome's IR-based formatter infrastructure. Use when implementing formatting for new syntax nodes, handling comments in formatted output, writing or debugging formatter snapshot tests, diagnosing idempotency failures, or comparing Biome's formatting against Prettier for JavaScript, CSS, JSON, HTML, Markdown, or other languages.
71testing-codegen
Guide for testing workflows and code generation commands in Biome. Use when running snapshot tests for lint rules, managing insta snapshots, or regenerating analyzer/parser/formatter code after changes.
69type-inference
Guide for working with Biome's module graph and type inference system. Use when implementing type-aware lint rules, understanding type resolution, working on the module graph infrastructure, or implementing type inference for new features.
67diagnostics-development
Guide for creating high-quality, user-friendly diagnostics in Biome. Use when creating diagnostics for lint rules, adding helpful advice to error messages, implementing code frame displays, or improving diagnostic quality.
67