Skills biome
Lint and format frontend code with Biome 2.4. Covers type-aware linting, GritQL custom rules, domains, import organizer, and migration from ESLint/Prettier. Use when configuring linting rules, formatting code, writing custom lint rules, or setting up CI checks. Triggers on biome, biome config, biome lint, biome format, biome check, biome ci, gritql, migrate from eslint, migrate from prettier, import sorting, code formatting, lint rules, type-aware linting, noFloatingPromises.
git clone https://github.com/tenequm/skills
T=$(mktemp -d) && git clone --depth=1 https://github.com/tenequm/skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/biome" ~/.claude/skills/tenequm-skills-biome && rm -rf "$T"
skills/biome/SKILL.mdBiome
Fast, unified linting, formatting, and import organization for JavaScript, TypeScript, JSX, CSS, and GraphQL. Biome 2.4 provides type-aware linting without the TypeScript compiler, GritQL plugins for custom rules, and domain-based rule grouping. Single binary, zero config by default, 97% Prettier compatibility.
Critical Rules
files.ignore
DOES NOT EXIST - use files.includes
with negation
files.ignorefiles.includesBiome 2.x only supports
files.includes (with an s). There is NO files.ignore, NO files.include (without s), NO files.exclude. Using any of these will throw Found an unknown key errors.
The only valid keys under
files are: includes, maxSize, ignoreUnknown, experimentalScannerIgnores.
To exclude files (generated code, vendored files, etc.), use negation patterns in
files.includes:
{ "files": { "includes": ["**", "!**/routeTree.gen.ts", "!**/generated/**"] } }
Do NOT use
overrides with linter/formatter/assists: { enabled: false } to skip generated files - that approach is fragile (easy to miss a subsystem like assists/import organizer) and unnecessarily complex. Just exclude via files.includes.
Always use biome check
, not separate lint + format
biome checkbiome check runs formatter, linter, and import organizer in one pass. Never call biome lint and biome format separately in CI - use biome check (or biome ci for CI mode).
biome.json lives at project root
Every project needs one
biome.json at the root. Monorepo packages use nested configs with "extends": "//" to inherit from root. Never use relative paths like "extends": ["../../biome.json"].
Use --write
to apply fixes, not --fix
--write--fixbiome check --write . # Apply safe fixes biome check --write --unsafe . # Apply all fixes (review changes)
Pin exact versions and migrate after upgrades
pnpm add --save-dev --save-exact @biomejs/biome@latest pnpm biome migrate --write
Quick Start
pnpm add --save-dev --save-exact @biomejs/biome pnpm biome init # Creates default biome.json with recommended rules
IDE Setup
VS Code - Install
biomejs.biome extension:
{ "editor.defaultFormatter": "biomejs.biome", "editor.formatOnSave": true, "editor.codeActionsOnSave": { "source.organizeImports.biome": "explicit" } }
Zed - Biome extension available natively. The
inline_config feature (v2.4) lets editors override rules without affecting biome.json:
{ "formatter": { "language_server": { "name": "biome" } }, "lsp": { "biome": { "settings": { "inline_config": { "linter": { "rules": { "suspicious": { "noConsole": "off" } } } } } } } }
CI Integration
pnpm biome ci . # No writes, non-zero exit on errors pnpm biome ci --reporter=default --reporter=github . # GitHub Actions annotations
Configuration (biome.json)
Recommended config for React/TypeScript projects
{ "$schema": "./node_modules/@biomejs/biome/configuration_schema.json", "vcs": { "enabled": true, "clientKind": "git", "useIgnoreFile": true }, "files": { "includes": [ "src/**/*.ts", "src/**/*.tsx", "tests/**/*.ts", "**/*.config.ts", "**/*.json", "!**/generated", "!**/components/ui" ] }, "formatter": { "enabled": true, "indentStyle": "space", "indentWidth": 2, "lineWidth": 120 }, "linter": { "enabled": true, "rules": { "recommended": true }, "domains": { "react": "recommended" } }, "javascript": { "formatter": { "quoteStyle": "double" } }, "assist": { "enabled": true, "actions": { "source": { "organizeImports": "on" } } } }
Formatter options
Key options:
indentStyle ("space"/"tab"), indentWidth, lineWidth, lineEnding ("lf"/"crlf"), trailingNewline. JS-specific: quoteStyle, trailingCommas, semicolons, arrowParentheses, bracketSpacing.
Linter rule configuration
Rules use severity levels
"error", "warn", "info", or "off". Some accept options:
{ "linter": { "rules": { "recommended": true, "style": { "noRestrictedGlobals": { "level": "error", "options": { "deniedGlobals": { "Buffer": "Use Uint8Array for browser compatibility." } } }, "useComponentExportOnlyModules": "off" } } } }
Import organizer
The import organizer (Biome Assist) merges duplicates, sorts by distance, and supports custom grouping:
{ "assist": { "enabled": true, "actions": { "source": { "organizeImports": { "level": "on", "options": { "groups": [ { "source": "builtin" }, { "source": "external" }, { "source": "internal", "match": "@company/*" }, { "source": "relative" } ] } } } } } }
Per-subsystem includes
Each subsystem (
linter, formatter, assist) has its own includes for fine-grained scoping. Applied after files.includes - can only narrow, not widen.
{ "files": { "includes": ["**", "!**/dist"] }, "linter": { "includes": ["**", "!**/components/ui"] }, "formatter": { "includes": ["**", "!**/components/ui"] } }
This lints and formats everything except
dist/ and components/ui, while assists (import organizer) still run on components/ui.
Overrides
Overrides apply different settings to specific file patterns. Use for per-file rule tweaks (e.g., relaxing rules for vendored/shadcn components). The field is
includes (with s).
{ "overrides": [ { "includes": ["**/components/ui/**"], "linter": { "rules": { "suspicious": { "noDocumentCookie": "off" }, "style": { "useComponentExportOnlyModules": "off" } } } }, { "includes": ["**/*.test.ts"], "linter": { "rules": { "suspicious": { "noConsole": "off" } } } } ] }
Monorepo configuration
Root
biome.json holds shared config. Package configs inherit with "extends": "//":
{ "$schema": "../../node_modules/@biomejs/biome/configuration_schema.json", "extends": "//" }
Override specific rules per package by adding a
linter.rules section alongside "extends": "//".
Configuration file discovery (v2.4)
Search order:
biome.json -> biome.jsonc -> .biome.json -> .biome.jsonc -> platform config home (~/.config/biome on Linux, ~/Library/Application Support/biome on macOS).
Domains
Domains group lint rules by technology. Enable only what your stack needs:
{ "linter": { "domains": { "react": "recommended", "next": "recommended", "test": "recommended", "types": "all" } } }
| Domain | Purpose | Auto-detected |
|---|---|---|
| React hooks, JSX patterns | dependency |
| Next.js-specific rules | |
| Solid.js rules | dependency |
| Testing best practices (any framework) | - |
| Playwright test rules | |
| Cross-file analysis (noImportCycles, noUnresolvedImports) | - |
| Type inference rules (noFloatingPromises, noMisusedPromises) | - |
Activation levels:
"recommended" (stable rules only), "all" (includes nursery), "none" (disable).
The
project domain enables rules needing the module graph. The types domain (v2.4) enables rules requiring type inference. Both trigger a file scan that adds a small overhead.
Type-Aware Linting
Biome 2.0 introduced type-aware linting without the TypeScript compiler. Biome has its own type inference engine in Rust - no
typescript dependency needed.
How it works
Enable the
types domain to activate file scanning and type inference. Performance impact is minimal compared to typescript-eslint because inference runs natively.
Key rules
| Rule | What it catches |
|---|---|
| Unhandled promises (missing await/return/void) |
| Promises in conditionals, array callbacks |
| Awaiting non-thenable values |
| Conditions that are always true/false |
| where is better |
| instead of |
| without compare function |
noFloatingPromises
The most impactful type-aware rule. Detects unhandled promises:
// ERROR: floating promise async function loadData() { fetch("/api/data"); } // VALID: awaited async function loadData() { await fetch("/api/data"); } // VALID: explicitly voided (fire-and-forget) async function loadData() { void fetch("/api/data"); }
Detects ~75% of cases compared to typescript-eslint, improving each release.
Limitations vs typescript-eslint
- Complex generic type inference may miss some cases
- Not a full type checker - handles common patterns, not every edge case
- Rules still in nursery - expect improvements with each release
- Major performance advantage: fraction of tsc-based linting time
GritQL Custom Rules
GritQL is a declarative pattern-matching language for custom lint rules. Create
.grit files and register them as plugins.
{ "plugins": ["./lint-rules/no-object-assign.grit"] }
Examples
Ban
:Object.assign
`$fn($args)` where { $fn <: `Object.assign`, register_diagnostic( span = $fn, message = "Prefer object spread instead of `Object.assign()`" ) }
CSS - enforce color classes:
language css; `$selector { $props }` where { $props <: contains `color: $color` as $rule, not $selector <: r"\.color-.*", register_diagnostic( span = $rule, message = "Don't set explicit colors. Use `.color-*` classes instead." ) }
Plugin API
register_diagnostic() arguments:
-severity
,"hint"
,"info"
,"warn"
(default:"error"
)"error"
(required) - diagnostic messagemessage
(required) - syntax node to highlightspan
Supported target languages: JavaScript (default), CSS, JSON (v2.4). Profile with
biome lint --profile-rules ..
Suppression Patterns
Single-line
// biome-ignore lint/suspicious/noConsole: needed for debugging console.log("debug info");
File-level
// biome-ignore-all lint/suspicious/noConsole: logger module
Range
// biome-ignore-start lint/style/useConst: legacy code let x = 1; let y = 2; // biome-ignore-end lint/style/useConst const a = 4; // this line IS checked
biome-ignore-end is optional - omit to suppress until end of file. Biome requires explanation text after the colon.
Migration
From ESLint
pnpm biome migrate eslint --write pnpm biome migrate eslint --write --include-inspired # Include non-identical rules
Supports legacy and flat configs,
extends resolution, plugins (typescript-eslint, react, jsx-a11y, unicorn), .eslintignore.
From Prettier
pnpm biome migrate prettier --write
Maps
tabWidth -> indentWidth, useTabs -> indentStyle, singleQuote -> quoteStyle, trailingComma -> trailingCommas.
From ESLint + Prettier combo
pnpm biome migrate eslint --write pnpm biome migrate prettier --write pnpm remove eslint prettier eslint-config-prettier eslint-plugin-prettier \ @typescript-eslint/parser @typescript-eslint/eslint-plugin rm .eslintrc* .prettierrc* .eslintignore .prettierignore
Enable VCS integration since ESLint respects gitignore by default:
{ "vcs": { "enabled": true, "clientKind": "git", "useIgnoreFile": true } }
CLI Reference
biome check (primary command)
biome check . # Check all files biome check --write . # Apply safe fixes biome check --write --unsafe . # Apply all fixes biome check --changed . # Only VCS-changed files biome check --staged . # Only staged files
biome ci (CI mode)
biome ci . # No writes, exit code on errors biome ci --reporter=github . # GitHub annotations biome ci --reporter=sarif --reporter-file=report.sarif . # SARIF output
biome lint
biome lint --only=suspicious/noDebugger . # Single rule biome lint --skip=project . # Skip domain biome lint --only=types . # Only type-aware rules biome lint --error-on-warnings . # Warnings become errors
Other commands
biome format --write . # Format only biome search '`console.$method($args)`' . # GritQL pattern search biome rage # Debug info for bug reports biome explain noFloatingPromises # Explain a rule
Best Practices
- Use
as your single command - combines format, lint, and import organizationbiome check - Start with
- disable individual rules as neededrecommended: true - Enable relevant domains -
,react
,next
,test
based on your stacktypes - Enable VCS integration - respects
, enables.gitignore
/--changed--staged - Use
in pipelines - never writes files, clear exit codesbiome ci - Pin exact versions - avoid surprise rule changes between releases
- Run
after every upgradebiome migrate --write - Use
in pre-commit hooks:--stagedbiome check --staged --write --no-errors-on-unmatched . - Profile slow rules with
(v2.4)biome lint --profile-rules - Use GritQL plugins for project-specific patterns instead of disabling rules globally
Gotchas
These are real mistakes that have caused broken configs, dirty working trees, and wasted debugging time. Read before writing any Biome config.
-
,files.ignore
,files.include
do not exist. Onlyfiles.exclude
(withfiles.includes
). Biome will throws
for anything else. See the first critical rule above.Found an unknown key -
is NOT a top-level config key. In Biome 2.x it moved underorganizeImports
. Using it at the top level is a config error.assist.actions.source.organizeImports -
that disableoverrides
+linter
still runformatter
. If you use overrides to skip a generated file, the import organizer (an assist action) will still rewrite it. This silently dirties your working tree. Useassist
negation to fully exclude a file instead.files.includes -
field isoverrides
(withincludes
), nots
. Same naming asinclude
.files.includes -
runs formatter + linter + assists in one pass. Any of these three can modify files. If a generated file keeps getting dirtied afterbiome check --write
, check which subsystem is touching it - it's often the import organizer (assist), not the formatter or linter.check --write -
does not exist. The flag is--fix
.--write
will silently do nothing or error.--fix
Resources
- Biome Docs: https://biomejs.dev/
- Biome GitHub: https://github.com/biomejs/biome
- GritQL Docs: https://docs.grit.io/
- Biome v2 Blog: https://biomejs.dev/blog/biome-v2/
- Biome v2.4 Blog: https://biomejs.dev/blog/biome-v2-4/
- Migration Guide: https://biomejs.dev/guides/migrate-eslint-prettier/
- Rules Reference: https://biomejs.dev/linter/javascript/rules/
- Domains: https://biomejs.dev/linter/domains/
- Plugins: https://biomejs.dev/linter/plugins/
For detailed lint rules by category with code examples, see rules-reference.md.