effective-typescript
Effective TypeScript Skill
Apply the 62 items from Dan Vanderkam's "Effective TypeScript" to review existing code and write new TypeScript. This skill operates in two modes: Review Mode (analyze code for violations) and Write Mode (produce idiomatic, well-typed TypeScript from scratch).
Reference Files
This skill includes categorized reference files covering all 62 items:
ref-01-getting-to-know-ts.md— Items 1-5: TS/JS relationship, compiler options, code generation, structural typing, anyref-02-type-system.md— Items 6-18: editor, sets, type vs value space, declarations vs assertions, object wrappers, excess property checking, generics, readonly, mapped typesref-03-type-inference.md— Items 19-27: inferable types, widening, narrowing, objects at once, aliases, async/await, context, functional constructsref-04-type-design.md— Items 28-37: valid states, Postel's Law, documentation, null perimeter, unions of interfaces, string types, branded typesref-05-working-with-any.md— Items 38-44: narrowest scope, precise any variants, unsafe assertions, evolving any, unknown, monkey patching, type coverageref-06-type-declarations.md— Items 45-52: devDependencies, three versions, export types, TSDoc, this in callbacks, conditional types, mirror types, testing typesref-07-writing-running-code.md— Items 53-57: ECMAScript features, iterating objects, DOM hierarchy, private, source mapsref-08-migrating.md— Items 58-62: modern JS, @ts-check, allowJs, module-by-module, noImplicitAny
How to Use This Skill
Before responding, read the relevant reference files based on the code's topic. For a general review, read all files. For targeted work (e.g., type design), read the specific reference (e.g., ref-04-type-design.md).
Mode 1: Code Review
When the user asks you to review existing TypeScript code, follow this process:
Step 1: Read Relevant References
Determine which chapters apply to the code under review and read those reference files. If unsure, read all of them.
Step 2: Analyze the Code
Before listing issues, first ask: Is this code already applying Effective TypeScript principles? Look for positive signals:
- Tagged unions with a discriminant field (Item 28/32)
- Branded types for nominal typing (Item 37)
unknownfor external data, narrowed before use (Item 42)- Type assertions scoped inside well-typed wrapper functions (Item 40)
readonlyon fields/parameters (Item 17)async/awaitwith typed return types (Item 25)- TSDoc comments on public functions (Item 48)
Key rule — Item 40 interaction with Item 9: A type assertion (as T) inside a function that has a fully-typed signature is NOT a violation of Item 9. Item 40 explicitly endorses hiding unsafe assertions inside well-typed wrappers. Only flag as when it appears at a call-site or as an escape hatch on a public-facing value.
For each relevant item from the book, check whether the code follows or violates the guideline. Focus on:
- TypeScript Fundamentals (Items 1-5): Is
strictmode enabled? Isanyused carelessly? Does structural typing cause surprises? - Type System Usage (Items 6-18): Are type declarations preferred over assertions? Are object wrapper types avoided? Are
readonlyand mapped types used appropriately? - Type Inference (Items 19-27): Is inference relied upon where possible? Are
async/awaitused over callbacks? Are aliases consistent? - Type Design (Items 28-37): Do types represent only valid states? Are string types replaced with literal unions? Are null values pushed to the perimeter?
- Working with any (Items 38-44): Is
anyscoped as narrowly as possible? Isunknownused for truly unknown values? Are unsafe assertions hidden in well-typed wrappers? - Type Declarations (Items 45-52): Are
@typesin devDependencies? Are public API types exported? Is TSDoc used for comments? - Code Execution (Items 53-57): Are ECMAScript features preferred over TypeScript-only equivalents? Is object iteration done safely?
- Migration (Items 58-62): Is modern JavaScript used as a baseline? Is migration done module-by-module?
Step 3: Calibrate Your Response
If the code is already well-typed:
- Open with acknowledgment of what is correct and which Items are applied
- Only note genuine issues; do not manufacture problems
- Any remaining observations must be clearly labeled as "optional polish" or "minor suggestion"
- Do NOT escalate a narrowly scoped assertion inside a well-typed function to Critical/Important
If the code has real issues: For each issue found, report:
- Item number and name (e.g., "Item 9: Prefer Type Declarations to Type Assertions")
- Location in the code
- What's wrong (the anti-pattern)
- How to fix it (the TypeScript-idiomatic way)
- Priority: Critical (bugs/correctness), Important (maintainability), Suggestion (style)
Step 4: Provide Fixed Code (only when needed)
If there are real issues, offer a corrected version with comments explaining each change. If the code is already correct, you may offer a brief "what's great here" summary instead of a rewrite.
Mode 2: Writing New Code
When the user asks you to write new TypeScript code, apply these core practices:
<core_principles>
Always Apply These Core Practices
-
Enable strict mode (Item 2). Never write TypeScript without
"strict": truein tsconfig.json. -
Prefer type declarations over assertions (Item 9). Use
const x: MyType = valuenotconst x = value as MyType. -
Avoid object wrapper types (Item 10). Use
string,number,boolean— neverString,Number,Boolean. -
Use types that represent only valid states (Item 28). Eliminate impossible states at the type level with tagged unions.
-
Push null to the perimeter (Item 31). Don't scatter
T | nullthroughout — handle nullability at boundaries. -
Prefer unions of interfaces to interfaces of unions (Item 32). Model tagged unions instead of interfaces with optional fields that have implicit relationships.
-
Replace plain string types with string literal unions (Item 33).
type Direction = 'north' | 'south' | 'east' | 'west'notstring. -
Generate types from APIs and specs, not data (Item 35). Use
quicktypeor OpenAPI code generation — don't hand-write types for external data. -
Use
unknowninstead ofanyfor values with unknown type (Item 42).unknownforces callers to narrow before use. -
Scope
anyas narrowly as possible (Item 38). Apply it to a single value, never a whole object or module. -
Use
readonlyto prevent mutation bugs (Item 17). Preferreadonlyon function parameters accepting arrays, and on class fields that should not be reassigned. -
Use
async/awaitover raw Promises and callbacks (Item 25). It produces cleaner inferred types and clearer code. -
Use type aliases to avoid repeating yourself (Item 14). DRY applies to types too — extract shared structure with
Pick,Omit, mapped types. -
Export all types that appear in public APIs (Item 47). Don't force users to reconstruct types with
ReturnType<>orParameters<>. -
Use TSDoc for API comments (Item 48).
/** */comments appear in editor tooltips;@param,@returns,@deprecatedare recognized by tooling.
</core_principles>
Type Structure Template
// Prefer interfaces for object shapes (extendable); type aliases for unions/intersections
interface User {
readonly id: UserId; // Item 17: readonly on fields that shouldn't change
name: string;
email: string;
}
// Branded type for nominal typing (Item 37)
type UserId = string & { readonly __brand: 'UserId' };
// Tagged union — only valid states representable (Item 28, 32)
type RequestState<T> =
| { status: 'loading' }
| { status: 'success'; data: T }
| { status: 'error'; message: string };
// unknown, not any, for values from external sources (Item 42)
function parseResponse(json: string): unknown {
return JSON.parse(json);
}
// async/await over callbacks (Item 25)
async function fetchUser(id: UserId): Promise<User> {
const response = await fetch(`/api/users/${id}`);
if (!response.ok) throw new Error(`HTTP ${response.status}`);
return response.json() as User; // narrowly scoped assertion inside well-typed function (Item 40)
}
any Guidelines
- If
anyis unavoidable, apply it to the smallest possible scope (Item 38) - Prefer
unknownfor values received from external sources (Item 42) - Hide unsafe assertions inside well-typed wrapper functions (Item 40)
- Track type coverage with
type-coverageCLI to prevent regressions (Item 44)
Priority of Items by Impact
<core_principles>
Critical (Correctness & Bugs)
- Item 2: Enable
strictmode —noImplicitAnyandstrictNullChecksprevent whole classes of bugs - Item 9: Prefer declarations to assertions — but see Item 40: assertions inside well-typed wrappers are fine
- Item 28: Types that always represent valid states — impossible states cause runtime errors
- Item 31: Push null to the perimeter — scattered nullability causes null dereferences
- Item 42: Use
unknowninstead ofany—anysilently disables type checking
Item 40 exception:
raw as SomeTypeinside a function with a fully-typed signature is explicitly endorsed by Item 40. It is acceptable and should NOT be flagged as a Critical or Important violation. At most, note it as a minor optional polish item (suggest a runtime validator like zod as a complement).
</core_principles>
Important (Maintainability)
- Item 13: Know the differences between
typeandinterface - Item 14: Use type operations and generics to avoid repetition
- Item 17: Use
readonlyto prevent mutation bugs - Item 25: Use
async/awaitover callbacks - Item 32: Prefer unions of interfaces to interfaces of unions
- Item 33: Prefer string literal unions over plain
string - Item 47: Export all types that appear in public APIs
- Item 48: Use TSDoc for API comments
Suggestions (Polish & Optimization)
- Item 19: Omit inferable types to reduce clutter
- Item 35: Generate types from APIs and specs
- Item 37: Consider brands for nominal typing
- Item 44: Track type coverage
- Item 53: Prefer ECMAScript features over TypeScript-only equivalents
<anti_patterns>
Common Anti-Patterns to Flag
any overuse (Items 38, 42)
- Return type
Promise<any>— usePromise<unknown>or a concrete typed interface as anycast at call-site or on a public-facing value — narrowest-scope rule violated- Interface fields typed as
any(e.g.,result?: any) — useunknownor a typed union
Interface of unions (Items 28, 32)
- An interface with
boolean+ optional fields that only make sense together:{ isLoading: boolean; data?: T; error?: string }is an interface-of-unions anti-pattern- Invalid combinations are representable (e.g.,
{ isLoading: true, data: [...] }or{ isLoading: false, data: undefined }) - Fix: replace with a tagged union using a
statusdiscriminant field - Non-null assertions (
!) inside render/display logic are a strong symptom of this pattern
- Invalid combinations are representable (e.g.,
Plain string parameters (Item 33)
- Function parameters typed as
stringwhen only a small, known set of values is valid- Fix:
type Theme = 'light' | 'dark'or similar literal union
- Fix:
Type assertions at call-sites (Item 9)
value as SomeTypeappearing in application code (not inside a well-typed wrapper function)- Fix: use a type declaration
const x: SomeType = valueor a validator function that returns the typed value
- Fix: use a type declaration
Missing strict mode (Item 2)
- TypeScript files without
"strict": truein tsconfig — whole classes of bugs (null dereferences, implicit any) go undetected
Callbacks over async/await (Item 25)
.then()chains instead ofasync/await— harder to read, worse type inference
</anti_patterns>
<strengths_to_praise>
Strengths to Recognize in Good Code
When reviewing code that applies these patterns correctly, explicitly acknowledge:
- Branded types (Item 37):
type FooId = string & { readonly __brand: 'FooId' }— prevents mixing up identifiers of different domain entities - Tagged/discriminated unions (Item 28, 32):
type Result = { status: 'success'; data: T } | { status: 'error'; message: string }— only valid states representable unknownfor external data (Item 42):const raw: unknown = await response.json()— forces explicit narrowing before use- Assertion inside well-typed wrapper (Item 40):
as Tinside a function with a fully-typed signature — safe encapsulation of unsafe boundary readonlyfields (Item 17):readonly id: UserIdon domain entity interfaces — communicates immutability intentasync/await(Item 25): cleaner than.then()chains, better type inference- TSDoc on public functions (Item 48):
/** @param ... @returns ... */— shows up in editor tooltips - String literal unions (Item 33):
type Status = 'pending' | 'processing' | 'shipped'instead ofstring
</strengths_to_praise>
More from booklib-ai/skills
effective-java
>
21lean-startup
>
19clean-code-reviewer
Reviews code against Robert C. Martin's Clean Code principles. Use when users share code for review, ask for refactoring suggestions, or want to improve code quality. Produces actionable feedback organized by Clean Code principles with concrete before/after examples.
17domain-driven-design
>
16design-patterns
>
14refactoring-ui
>
14