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:
// 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 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:
// 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 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:
// 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 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:
// 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 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
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 }
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:
// 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.
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