home / skills / tenequm / claude-plugins / biome
/frontend-dev/skills/biome
This skill lints and formats frontend code using Biome 2.4, enabling type-aware checks, import organization, and CI-ready rules.
npx playbooks add skill tenequm/claude-plugins --skill biomeReview the files below or copy the command above to add this skill to your agents.
---
name: biome
description: 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.
---
# Biome
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
### Always use `biome check`, not separate lint + format
`biome 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`
```bash
biome check --write . # Apply safe fixes
biome check --write --unsafe . # Apply all fixes (review changes)
```
### Pin exact versions and migrate after upgrades
```bash
pnpm add --save-dev --save-exact @biomejs/biome@latest
pnpm biome migrate --write
```
## Quick Start
```bash
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:
```json
{
"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`:
```json
{
"formatter": { "language_server": { "name": "biome" } },
"lsp": {
"biome": {
"settings": {
"inline_config": {
"linter": { "rules": { "suspicious": { "noConsole": "off" } } }
}
}
}
}
}
```
### CI Integration
```bash
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
```json
{
"$schema": "./node_modules/@biomejs/biome/configuration_schema.json",
"vcs": {
"enabled": true,
"clientKind": "git",
"useIgnoreFile": true
},
"files": {
"include": [
"src/**/*.ts", "src/**/*.tsx",
"tests/**/*.ts", "**/*.config.ts", "**/*.json"
],
"ignore": [
"*.d.ts", "**/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:
```json
{
"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:
```json
{
"assist": {
"enabled": true,
"actions": {
"source": {
"organizeImports": {
"level": "on",
"options": {
"groups": [
{ "source": "builtin" },
{ "source": "external" },
{ "source": "internal", "match": "@company/*" },
{ "source": "relative" }
]
}
}
}
}
}
}
```
### Monorepo configuration
Root `biome.json` holds shared config. Package configs inherit with `"extends": "//"`:
```json
{
"$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:
```json
{
"linter": {
"domains": {
"react": "recommended",
"next": "recommended",
"test": "recommended",
"types": "all"
}
}
}
```
| Domain | Purpose | Auto-detected |
|--------|---------|---------------|
| `react` | React hooks, JSX patterns | `react` dependency |
| `next` | Next.js-specific rules | `next >= 14.0.0` |
| `solid` | Solid.js rules | `solid-js` dependency |
| `test` | Testing best practices (any framework) | - |
| `playwright` | Playwright test rules | `@playwright/test` |
| `project` | Cross-file analysis (noImportCycles, noUnresolvedImports) | - |
| `types` | 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 |
|------|----------------|
| `noFloatingPromises` | Unhandled promises (missing await/return/void) |
| `noMisusedPromises` | Promises in conditionals, array callbacks |
| `useAwaitThenable` | Awaiting non-thenable values |
| `noUnnecessaryConditions` | Conditions that are always true/false |
| `useRegexpExec` | `string.match()` where `regexp.exec()` is better |
| `useFind` | `array.filter()[0]` instead of `array.find()` |
| `useArraySortCompare` | `Array.sort()` without compare function |
### noFloatingPromises
The most impactful type-aware rule. Detects unhandled promises:
```ts
// 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.
```json
{ "plugins": ["./lint-rules/no-object-assign.grit"] }
```
### Examples
**Ban `Object.assign`:**
```grit
`$fn($args)` where {
$fn <: `Object.assign`,
register_diagnostic(
span = $fn,
message = "Prefer object spread instead of `Object.assign()`"
)
}
```
**CSS - enforce color classes:**
```grit
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"`, `"error"` (default: `"error"`)
- `message` (required) - diagnostic message
- `span` (required) - syntax node to highlight
Supported target languages: JavaScript (default), CSS, JSON (v2.4). Profile with `biome lint --profile-rules .`.
## Suppression Patterns
### Single-line
```ts
// biome-ignore lint/suspicious/noConsole: needed for debugging
console.log("debug info");
```
### File-level
```ts
// biome-ignore-all lint/suspicious/noConsole: logger module
```
### Range
```ts
// 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
```bash
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
```bash
pnpm biome migrate prettier --write
```
Maps `tabWidth` -> `indentWidth`, `useTabs` -> `indentStyle`, `singleQuote` -> `quoteStyle`, `trailingComma` -> `trailingCommas`.
### From ESLint + Prettier combo
```bash
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:
```json
{ "vcs": { "enabled": true, "clientKind": "git", "useIgnoreFile": true } }
```
## CLI Reference
### biome check (primary command)
```bash
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)
```bash
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
```bash
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
```bash
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
1. **Use `biome check` as your single command** - combines format, lint, and import organization
2. **Start with `recommended: true`** - disable individual rules as needed
3. **Enable relevant domains** - `react`, `next`, `test`, `types` based on your stack
4. **Enable VCS integration** - respects `.gitignore`, enables `--changed`/`--staged`
5. **Use `biome ci` in pipelines** - never writes files, clear exit codes
6. **Pin exact versions** - avoid surprise rule changes between releases
7. **Run `biome migrate --write` after every upgrade**
8. **Use `--staged` in pre-commit hooks**: `biome check --staged --write --no-errors-on-unmatched .`
9. **Profile slow rules** with `biome lint --profile-rules` (v2.4)
10. **Use GritQL plugins** for project-specific patterns instead of disabling rules globally
## 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](references/rules-reference.md).
This skill helps teams lint, format, and organize frontend code using Biome 2.4. It guides configuration, type-aware linting, GritQL custom rules, import organization, and migration from ESLint/Prettier. Use it to set up CI checks, editor integration, and domain-based rule sets for JavaScript, TypeScript, JSX, CSS, and GraphQL.
The skill inspects and explains Biome configuration (biome.json), recommended CLI usage (biome check / biome ci), and key features like the type inference engine and domains. It describes how to enable type-aware rules (types domain), create and register GritQL rules, and configure the import organizer. It also covers migration commands and best practices for monorepos, IDE setups, and CI integration.
Should I use biome lint and biome format separately in CI?
No. Use biome check or biome ci. They run formatter, linter, and import organizer in one pass and avoid inconsistent results.
Where should biome.json live in a monorepo?
Place a single biome.json at the repository root. Individual packages may add nested configs that inherit using "extends": "//"; avoid relative extends like ../../biome.json.