typescript-best-practices
Scannednpx machina-cli add skill aiskillstore/marketplace/typescript-best-practices --openclawTypeScript Best Practices
Pair with React Best Practices
When working with React components (.tsx, .jsx files or @react imports), always load react-best-practices alongside this skill. This skill covers TypeScript fundamentals; React-specific patterns (effects, hooks, refs, component design) are in the dedicated React skill.
Type-First Development
Types define the contract before implementation. Follow this workflow:
- Define the data model - types, interfaces, and schemas first
- Define function signatures - input/output types before logic
- Implement to satisfy types - let the compiler guide completeness
- Validate at boundaries - runtime checks where data enters the system
Make Illegal States Unrepresentable
Use the type system to prevent invalid states at compile time.
Discriminated unions for mutually exclusive states:
// Good: only valid combinations possible
type RequestState<T> =
| { status: 'idle' }
| { status: 'loading' }
| { status: 'success'; data: T }
| { status: 'error'; error: Error };
// Bad: allows invalid combinations like { loading: true, error: Error }
type RequestState<T> = {
loading: boolean;
data?: T;
error?: Error;
};
Branded types for domain primitives:
type UserId = string & { readonly __brand: 'UserId' };
type OrderId = string & { readonly __brand: 'OrderId' };
// Compiler prevents passing OrderId where UserId expected
function getUser(id: UserId): Promise<User> { /* ... */ }
function createUserId(id: string): UserId {
return id as UserId;
}
Const assertions for literal unions:
const ROLES = ['admin', 'user', 'guest'] as const;
type Role = typeof ROLES[number]; // 'admin' | 'user' | 'guest'
// Array and type stay in sync automatically
function isValidRole(role: string): role is Role {
return ROLES.includes(role as Role);
}
Required vs optional fields - be explicit:
// Creation: some fields required
type CreateUser = {
email: string;
name: string;
};
// Update: all fields optional
type UpdateUser = Partial<CreateUser>;
// Database row: all fields present
type User = CreateUser & {
id: UserId;
createdAt: Date;
};
Module Structure
Prefer smaller, focused files: one component, hook, or utility per file. Split when a file handles multiple concerns or exceeds ~200 lines. Colocate tests with implementation (foo.test.ts alongside foo.ts). Group related files by feature rather than by type.
Functional Patterns
- Prefer
constoverlet; usereadonlyandReadonly<T>for immutable data. - Use
array.map/filter/reduceoverforloops; chain transformations in pipelines. - Write pure functions for business logic; isolate side effects in dedicated modules.
- Avoid mutating function parameters; return new objects/arrays instead.
Instructions
- Enable
strictmode; model data with interfaces and types. Strong typing catches bugs at compile time. - Every code path returns a value or throws; use exhaustive
switchwithneverchecks in default. Unhandled cases become compile errors. - Propagate errors with context; catching requires re-throwing or returning a meaningful result. Hidden failures delay debugging.
- Handle edge cases explicitly: empty arrays, null/undefined inputs, boundary values. Defensive checks prevent runtime surprises.
- Use
awaitfor async calls; wrap external calls with contextual error messages. Unhandled rejections crash Node processes. - Add or update focused tests when changing logic; test behavior, not implementation details.
Examples
Explicit failure for unimplemented logic:
export function buildWidget(widgetType: string): never {
throw new Error(`buildWidget not implemented for type: ${widgetType}`);
}
Exhaustive switch with never check:
type Status = "active" | "inactive";
export function processStatus(status: Status): string {
switch (status) {
case "active":
return "processing";
case "inactive":
return "skipped";
default: {
const _exhaustive: never = status;
throw new Error(`unhandled status: ${_exhaustive}`);
}
}
}
Wrap external calls with context:
export async function fetchWidget(id: string): Promise<Widget> {
const response = await fetch(`/api/widgets/${id}`);
if (!response.ok) {
throw new Error(`fetch widget ${id} failed: ${response.status}`);
}
return response.json();
}
Debug logging with namespaced logger:
import debug from "debug";
const log = debug("myapp:widgets");
export function createWidget(name: string): Widget {
log("creating widget: %s", name);
const widget = { id: crypto.randomUUID(), name };
log("created widget: %s", widget.id);
return widget;
}
Runtime Validation with Zod
- Define schemas as single source of truth; infer TypeScript types with
z.infer<>. Avoid duplicating types and schemas. - Use
safeParsefor user input where failure is expected; useparseat trust boundaries where invalid data is a bug. - Compose schemas with
.extend(),.pick(),.omit(),.merge()for DRY definitions. - Add
.transform()for data normalization at parse time (trim strings, parse dates). - Include descriptive error messages; use
.refine()for custom validation logic.
Examples
Schema as source of truth with type inference:
import { z } from "zod";
const UserSchema = z.object({
id: z.string().uuid(),
email: z.string().email(),
name: z.string().min(1),
createdAt: z.string().transform((s) => new Date(s)),
});
type User = z.infer<typeof UserSchema>;
Return parse results to callers (never swallow errors):
import { z, SafeParseReturnType } from "zod";
export function parseUserInput(raw: unknown): SafeParseReturnType<unknown, User> {
return UserSchema.safeParse(raw);
}
// Caller handles both success and error:
const result = parseUserInput(formData);
if (!result.success) {
setErrors(result.error.flatten().fieldErrors);
return;
}
await submitUser(result.data);
Strict parsing at trust boundaries:
export async function fetchUser(id: string): Promise<User> {
const response = await fetch(`/api/users/${id}`);
if (!response.ok) {
throw new Error(`fetch user ${id} failed: ${response.status}`);
}
const data = await response.json();
return UserSchema.parse(data); // throws if API contract violated
}
Schema composition:
const CreateUserSchema = UserSchema.omit({ id: true, createdAt: true });
const UpdateUserSchema = CreateUserSchema.partial();
const UserWithPostsSchema = UserSchema.extend({
posts: z.array(PostSchema),
});
Configuration
- Load config from environment variables at startup; validate with Zod before use. Invalid config should crash immediately.
- Define a typed config object as single source of truth; avoid accessing
process.envthroughout the codebase. - Use sensible defaults for development; require explicit values for production secrets.
Examples
Typed config with Zod validation:
import { z } from "zod";
const ConfigSchema = z.object({
PORT: z.coerce.number().default(3000),
DATABASE_URL: z.string().url(),
API_KEY: z.string().min(1),
NODE_ENV: z.enum(["development", "production", "test"]).default("development"),
});
export const config = ConfigSchema.parse(process.env);
Access config values (not process.env directly):
import { config } from "./config";
const server = app.listen(config.PORT);
const db = connect(config.DATABASE_URL);
Optional: type-fest
For advanced type utilities beyond TypeScript builtins, consider type-fest:
Opaque<T, Token>- cleaner branded types than manual& { __brand }patternPartialDeep<T>- recursive partial for nested objectsReadonlyDeep<T>- recursive readonly for immutable dataLiteralUnion<Literals, Fallback>- literals with autocomplete + string fallbackSetRequired<T, K>/SetOptional<T, K>- targeted field modificationsSimplify<T>- flatten complex intersection types in IDE tooltips
import type { Opaque, PartialDeep, SetRequired } from 'type-fest';
// Branded type (cleaner than manual approach)
type UserId = Opaque<string, 'UserId'>;
// Deep partial for patch operations
type UserPatch = PartialDeep<User>;
// Make specific fields required
type UserWithEmail = SetRequired<Partial<User>, 'email'>;
Source
git clone https://github.com/aiskillstore/marketplace/blob/main/skills/0xbigboss/typescript-best-practices/SKILL.mdView on GitHub Overview
TypeScript Best Practices teaches type-first development patterns to make illegal states unrepresentable, enable exhaustive handling, and perform runtime validation. It guides you to model data with interfaces and schemas before implementing logic, ensuring safer, more maintainable code across TS/JS projects.
How This Skill Works
It promotes designing types, interfaces, and discriminated unions first, then adding implementation that satisfies those contracts. It uses branded types for domain primitives, const assertions for literal unions, and explicit required vs optional fields to keep data shapes predictable; runtime checks validate boundaries where data enters the system.
When to Use It
- Model data with type-first design before coding complex logic
- Represent mutually exclusive states with discriminated unions to force exhaustive handling
- Enforce domain rules with branded types so APIs can’t mix IDs
- Keep codebase small and cohesive: one component, hook, or utility per file with colocated tests
- Guard against runtime surprises by validating inputs at boundaries and writing focused tests
Quick Start
- Step 1: Define type-first data models (interfaces, unions, and branded types) before writing logic
- Step 2: Implement functions against those types and add exhaustive switches (never) for all cases
- Step 3: Enable strict mode in tsconfig and add runtime boundary validations; colocate tests
Best Practices
- Define data models with interfaces and types first, then implement logic to satisfy them
- Use discriminated unions to represent mutually exclusive states for exhaustive switches
- Apply branded types to domain primitives to prevent mixing IDs across APIs
- Favor const assertions and readonly/Readonly<T> for immutable data
- Enable strict mode and write exhaustive code with never, plus contextual error handling and focused tests
Example Use Cases
- Discriminated union for RequestState<T> to represent idle/loading/success/error
- Branded types for UserId vs OrderId with createUserId(id) helper
- Const assertion for role union and type-safe isValidRole function
- CreateUser vs UpdateUser types showing required vs optional fields
- Small, feature-scoped modules with colocated tests and focused APIs