Pinned Biome, Oxlint/Oxfmt, Ultracite, and React Doctor presets for Howells projects.
The goal is not to invent a second lint philosophy. The goal is to:
- pin a single
@biomejs/biomeversion - pin a single
oxlintversion - pin a single
oxfmtversion - pin a single
ultraciteversion - pin React Doctor's Oxlint plugin for React and Next.js projects on the Oxlint/Oxfmt lane
- pin a single
@manypkg/cliversion for monorepo consistency checks - give every consumer the same small preset matrix
- discourage repo-local overrides unless the project has a genuinely unique constraint
Oxlint/Oxfmt is the preferred toolchain for Howells JavaScript and TypeScript projects. The Biome lane is retained for projects that need Biome compatibility or are not ready to adopt the preferred Oxlint/Oxfmt lane.
When configuring a project, do this in order:
- Require Node 22.18.0+ and pnpm in the root
package.json, and pin.node-versionto22.18.0. - Install only
@howells/lintas the direct lint dependency. - Add
oxlint.config.tsandoxfmt.config.tsthat extend the closest Oxlint/Oxfmt presets. - Add read-only
lintand mutatinglint:fixscripts. - If the project is a monorepo, add root workspace scripts that run
howells-workspace-check. - Verify with
pnpm lint.
All projects using this package should declare the runtime and package manager explicitly:
{
"packageManager": "pnpm@10.23.0",
"engines": {
"node": ">=22.18.0"
}
}Also add a root .node-version file:
22.18.0
Install the shared tooling:
pnpm add -D @howells/lintDo not add @biomejs/biome, oxlint, oxfmt, oxlint-tsgolint, ultracite, oxlint-plugin-react-doctor, eslint-plugin-playwright, oxc-parser, or @manypkg/cli directly unless you are developing this package itself. They are pinned transitively here.
The Biome lane is a frozen compatibility lane. It is retained for projects that need Biome presets, and it receives dependency, breakage, and ecosystem-compatibility updates, but new Howells policy work should target Oxlint/Oxfmt first.
Choose the closest preset instead of starting from a generic base and patching it locally:
@howells/lint/biome/corefor Node or non-React TypeScript packages@howells/lint/biome/reactfor React packages@howells/lint/biome/nextfor Next.js apps
These presets already pin Biome and Ultracite, enable VCS ignore file support, ignore common build output directories, keep ignoreUnknown on for mixed repos, enforce 2-space indentation, and enable Tailwind CSS directives on DOM-oriented presets.
The shared presets exclude generated and output folders seen across Howells projects: node_modules, .next, .turbo, .vercel, dist, build, coverage, out, storybook-static, playwright-report, test-results, .source, .cache, .expo, .output, .wrangler, .svelte-kit, .nuxt, .vite, .vinxi, dev-dist, tmp, and temp. Keep repo-local excludes only for genuinely project-specific generated files or data directories.
Node or non-React TypeScript package:
{
"$schema": "https://biomejs.dev/schemas/2.4.16/schema.json",
"extends": ["@howells/lint/biome/core"],
"root": true
}React package:
{
"$schema": "https://biomejs.dev/schemas/2.4.16/schema.json",
"extends": ["@howells/lint/biome/core", "@howells/lint/biome/react"],
"root": true
}Next.js app:
{
"$schema": "https://biomejs.dev/schemas/2.4.16/schema.json",
"extends": ["@howells/lint/biome/core", "@howells/lint/biome/react", "@howells/lint/biome/next"],
"root": true
}Use this lane for new Howells JavaScript and TypeScript projects. React and Next presets stack the relevant Ultracite Ox rules with React Doctor rules in one config.
React Doctor severities are passed through as published by React Doctor. Native Oxlint Next.js severities come from Oxlint's official nextjs plugin via Ultracite's Next preset. @howells/lint adds canonical Howells policy on top for file naming, barrel files, env access, workspace boundaries, file size, function size, complexity, and tests.
The core Oxlint preset enables type-aware linting and native Oxlint rules that keep code files navigable: max-lines errors above 600 non-comment, non-blank lines; max-lines-per-function warns above 120 non-comment, non-blank lines; max-statements warns above 45 statements per function; and complexity warns above cyclomatic complexity 15. It also rejects runtime import() expressions, including literal specifiers, so package loading stays statically traceable. React projects promote React Doctor's react-doctor/no-giant-component rule to an error, so genuinely oversized components still block CI. Test files keep the file-level max-lines guard but disable function-size, statement-count, and complexity limits, because test framework callbacks naturally wrap many independent cases. Generated files should be ignored at the project level; rare intentional exceptions should use an exact-file override with a short refactor note.
Core, React, Next, and Playwright presets also enforce the default Howells workspace convention: apps live under apps/*, shared packages live under packages/*, packages must not import apps, and apps must not import sibling apps. The rule is intentionally narrow and does not infer boundary meaning from other workspace folder names.
React and Next presets also reject generic component suffixes that tend to hide responsibility: wrapper, client, page, component, container, and manager. The rule checks .jsx and .tsx filenames and PascalCase component declarations. It allows real Next App Router app/**/page.tsx files and their conventional Page export.
Next presets reject App Router pages that only pass through to one imported client component. Route pages should keep server composition, data loading, and route-level structure in the page, then push only the interactive leaves behind a client boundary.
Playwright support adds the recommended eslint-plugin-playwright rules through Oxlint and promotes brittle E2E patterns to errors, including playwright/no-wait-for-timeout, playwright/no-force-option, playwright/no-element-handle, and playwright/prefer-web-first-assertions. Use the Playwright export as an overlay for app-level E2E tests, or as a standalone preset for dedicated E2E packages.
Choose the closest preset:
@howells/lint/oxlint/corefor Node or non-React TypeScript@howells/lint/oxlint/reactfor React (Ultracite React + React Doctor recommended rules)@howells/lint/oxlint/nextfor Next.js (react preset + Next.js rules)@howells/lint/oxlint/playwrightas an overlay for Playwright E2E tests or as a preset for dedicated E2E packages@howells/lint/oxlint/boundariesfor composing only the default workspace boundary rule into custom configs@howells/lint/oxlint/react-doctor-rulesfor composing or disabling React Doctor rules in mixed workspaces
Node or non-React TypeScript:
import { defineConfig } from "oxlint";
import core from "@howells/lint/oxlint/core";
export default defineConfig({
extends: [core],
});React package:
import { defineConfig } from "oxlint";
import react from "@howells/lint/oxlint/react";
export default defineConfig({
extends: [react],
});Next.js app:
import { defineConfig } from "oxlint";
import next from "@howells/lint/oxlint/next";
export default defineConfig({
extends: [next],
});Next.js app with Playwright E2E tests:
import { defineConfig } from "oxlint";
import next from "@howells/lint/oxlint/next";
import { playwrightJsPlugins, playwrightRules } from "@howells/lint/oxlint/playwright";
export default defineConfig({
extends: [next],
jsPlugins: playwrightJsPlugins,
overrides: [
{
files: ["**/*.spec.ts", "**/*.e2e.ts", "tests/**/*.{ts,tsx}"],
rules: playwrightRules,
},
],
});Dedicated Playwright E2E package:
import { defineConfig } from "oxlint";
import playwright from "@howells/lint/oxlint/playwright";
export default defineConfig({
extends: [playwright],
});Custom boundary-only config:
import { defineConfig } from "oxlint";
import {
boundaryJsPlugins,
boundaryRules,
boundarySettings,
} from "@howells/lint/oxlint/boundaries";
export default defineConfig({
jsPlugins: boundaryJsPlugins,
settings: boundarySettings,
rules: boundaryRules,
});Boundary rules are already part of the core, React, Next, and Playwright presets. Use the boundary-only export only when building a custom Oxlint config that cannot extend the standard presets.
Mixed monorepo with a Next.js app and Node-only packages:
import { defineConfig } from "oxlint";
import next from "@howells/lint/oxlint/next";
import { disabledReactDoctorRules } from "@howells/lint/oxlint/react-doctor-rules";
export default defineConfig({
extends: [next],
overrides: [
{
files: ["packages/**/*.ts"],
rules: disabledReactDoctorRules,
},
],
});Create an oxfmt.config.ts:
import { defineConfig } from "oxfmt";
import howells from "@howells/lint/oxfmt";
export default defineConfig({
extends: [howells],
});Oxlint type-aware mode is enabled by the shared core preset through the pinned oxlint-tsgolint dependency. Projects choosing the Oxlint/Oxfmt lane should be ready for Oxlint's TypeScript type-aware constraints.
During migration only, a project may temporarily disable type-aware mode:
export default defineConfig({
extends: [core],
options: {
typeAware: false,
},
});Treat this as a migration exception with a removal path, not as a normal project preference.
Use scripts that match the lane the project has chosen.
Biome lane:
{
"scripts": {
"lint": "howells-biome check .",
"lint:fix": "howells-biome check . --write"
}
}Oxlint/Oxfmt lane:
{
"scripts": {
"lint": "howells-check .",
"lint:fix": "howells-fix ."
}
}The Oxlint/Oxfmt lane does not define a separate lint:strict; React Doctor, type-aware Oxlint, workspace boundaries, and Playwright overlays belong in the normal check.
Keep lint non-mutating. Put all write behavior in lint:fix or format so CI and local checks have the same semantics.
Prefer the package binaries over raw tool commands or long target lists. Use explicit script targets only when the package has a real scope constraint:
{
"scripts": {
"lint": "howells-check apps/web packages/ui",
"lint:fix": "howells-fix apps/web packages/ui"
}
}Use howells-fix --unsafe . only when you deliberately want Oxlint's dangerous fixes.
Use workspace lint only at the monorepo root. Do not add howells-workspace-check to individual packages, and do not add it to single-package apps.
A monorepo root should have:
{
"packageManager": "pnpm@10.23.0",
"engines": {
"node": ">=22.18.0"
},
"scripts": {
"lint": "turbo run lint && howells-workspace-check",
"lint:fix": "turbo run lint:fix && howells-workspace-fix",
"check": "pnpm lint && pnpm typecheck && pnpm test"
},
"devDependencies": {
"@howells/lint": "^0.4.0"
}
}howells-workspace-check validates that the root declares packageManager: "pnpm@...", requires Node 22.18.0+ in engines.node, pins .node-version to 22.18.0, keeps pnpm-workspace.yaml present when workspace package directories exist, and passes manypkg check.
CI should call pnpm lint or pnpm check so root workspace lint is not bypassed by a direct turbo lint command.
Installers only need @howells/lint as a direct dependency. Use these package binaries:
howells-biomeproxies to the pinned Biome binaryhowells-ultraciteproxies to the pinned Ultracite binaryhowells-checkrunsoxfmt --check, thenoxlinthowells-fixrunsoxfmt --write, thenoxlint --fixhowells-oxlintproxies to the pinned Oxlint binaryhowells-oxfmtproxies to the pinned Oxfmt binaryhowells-workspace-checkruns workspace lint, then runsmanypkg checkhowells-workspace-fixrunsmanypkg fix
- Do not add local overrides just to preserve old ESLint behavior.
- Do not create local
base,shared, orcustomBiome wrappers. - Do not mix Biome and Oxlint/Oxfmt scripts in the same package unless the project has a deliberate migration plan.
- If multiple repos need the same exception, add or adjust a preset here.
- If a repo needs framework-specific linting, choose the matching preset instead of layering rules manually.
- Prefer inline
biome-ignorecomments for truly isolated exceptions over broad config overrides. - Keep package
lintscripts read-only; uselint:fixfor formatting and safe writes. - Prefer
howells-check .over raw tool commands or long target lists unless a package has a real scope constraint.
Add this to .claude/settings.json so files are fixed on edit and at session end:
{
"hooks": {
"PostToolUse": [
{
"matcher": "Edit|Write",
"hooks": [
{
"type": "command",
"command": "jq -r '.tool_input.file_path' | { read file_path; case \"$file_path\" in *.js|*.ts|*.jsx|*.tsx|*.json|*.jsonc|*.css|*.graphql) howells-fix \"$file_path\" 2>/dev/null || true ;; esac; }"
}
]
}
],
"Stop": [
{
"hooks": [
{
"type": "command",
"command": "git diff --name-only --diff-filter=d HEAD | grep -E '\\.(js|ts|jsx|tsx|json|jsonc|css|graphql)$' | xargs howells-fix 2>/dev/null || true"
}
]
}
]
}
}This package wraps: