| name | biome-developer |
| description | General development best practices and common gotchas when working on Biome. Use for avoiding common mistakes, understanding Biome-specific patterns (AST, syntax nodes, string extraction, embedded languages), and learning technical tips. |
| compatibility | Designed for coding agents working on the Biome codebase (github.com/biomejs/biome). |
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_test to 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
VueDirective and VueV*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_test instead
- Do NOT assume syntax patterns without inspecting the AST first
Example - Inspecting AST:
#[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());
}
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 like HtmlAttributeName to get the text content
- Verify whether values use
HtmlString (quotes) or HtmlTextExpression (curly braces)
DON'T:
- Do NOT use
text_trimmed() when you need inner_string_text() for extracting quoted string contents
Example - String Extraction:
let html_string = value.as_html_string()?;
let content = html_string.value_token()?.text_trimmed();
let html_string = value.as_html_string()?;
let inner_text = html_string.inner_string_text().ok()?;
let content = inner_text.text();
Working with Embedded Languages
DO:
- Verify changes work for different value formats (quoted strings vs text expressions) when handling multiple frameworks
- Use appropriate
EmbeddingKind for context (Vue, Svelte, Astro, etc.)
- Check if embedded content needs
is_source: true (script tags) vs is_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:
let html_string = value.as_html_string()?;
let inner_text = html_string.inner_string_text().ok()?;
let text_expression = value.as_html_attribute_single_text_expression()?;
let expression = text_expression.expression().ok()?;
Borrow Checker and Temporary Values
DO:
- Use intermediate
let bindings 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:
let html_string = value.value().ok()?.as_html_string()?;
let token = html_string.value_token().ok()?;
let value_node = value.value().ok()?;
let html_string = value_node.as_html_string()?;
let token = html_string.value_token().ok()?;
Clippy and Code Style
DO:
- Use
let chains to collapse nested if let statements (cleaner and follows Rust idioms)
- Run
just l before committing to catch clippy warnings
- Fix clippy suggestions unless there's a good reason not to
- Always import types at the top of the file — never inline crate paths in type positions
- Let the compiler infer types when it can — only add explicit type annotations when the compiler cannot infer them
DON'T:
- Do NOT ignore clippy warnings - they often catch real issues or suggest better patterns
- Do NOT inline crate paths in type positions (e.g.
Vec<some_crate::module::Type>) — add a use import instead
- Do NOT annotate types that the compiler can already infer
Example - Collapsible If:
if let Some(directive) = VueDirective::cast_ref(&element) {
if let Some(initializer) = directive.initializer() {
}
}
if let Some(directive) = VueDirective::cast_ref(&element)
&& let Some(initializer) = directive.initializer()
{
}
Example - Import types, don't inline paths:
enum Frame {
Visit(biome_css_semantic::model::RuleId),
}
use biome_css_semantic::model::RuleId;
enum Frame {
Visit(RuleId),
}
Example - Let the compiler infer types:
let mut visited: FxHashSet<RuleId> = FxHashSet::default();
let mut visited = FxHashSet::default();
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 case ties 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:
if is_embedded_script(node) {
return normalize_offset(node);
}
if is_embedded_script(node) {
return normalize_offset(node);
}
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, …):
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 all biome_* dev-dependencies
- Preserve any extra attributes like
features when converting
DON'T:
- Do NOT use
workspace = true for biome_* crates in [dev-dependencies]
Example:
[dev-dependencies]
biome_js_parser = { workspace = true }
biome_formatter = { workspace = true, features = ["countme"] }
[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*ShorthandDirective types)
- 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:
if let Some(directive) = AnySvelteDirective::cast_ref(&element) {
}
if let Some(attribute) = HtmlAttribute::cast_ref(&element) {
if let Ok(name) = attribute.name() {
}
}
Checking Multiple Variant Types
For frameworks with multiple directive syntaxes, handle each type:
if let Some(directive) = VueVOnShorthandDirective::cast_ref(&element) {
}
if let Some(directive) = VueVBindShorthandDirective::cast_ref(&element) {
}
if let Some(directive) = VueVSlotShorthandDirective::cast_ref(&element) {
}
if let Some(directive) = VueDirective::cast_ref(&element) {
}
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 a markup! block. markup! supports interpolation, E.g. markup! { "Hello, "{name}"!" }.
- Calling
.to_string() or .to_string_trimmed() (allocates a string) on a SyntaxToken or SyntaxNode. 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 &str from the token's text without allocating a new string, using token.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