typescript-patterns
This skill provides TypeScript-specific implementation patterns for type-safe, maintainable code.
When to Invoke This Skill
Automatically activate for:
- TypeScript/JavaScript project implementation
- Type system design and refinement
- Generic patterns and utility types
- Error handling with type safety
- API type definitions
Strict Typing Patterns
Branded Types for Domain Safety
// Prevent mixing IDs of different entities
type UserId = string & { readonly brand: unique symbol };
type OrderId = string & { readonly brand: unique symbol };
// Type-safe ID creation
function createUserId(id: string): UserId {
return id as UserId;
}
// Compiler prevents: processOrder(userId) ✗
function processOrder(orderId: OrderId): void { /* ... */ }
Discriminated Unions for State
// Result type for error handling
type Result<T, E = Error> =
| { success: true; data: T }
| { success: false; error: E };
// Usage with exhaustive checking
function handleResult<T>(result: Result<T>): T {
if (result.success) {
return result.data;
}
throw result.error;
}
// State machines
type RequestState<T> =
| { status: 'idle' }
| { status: 'loading' }
| { status: 'success'; data: T }
| { status: 'error'; error: Error };
Type Guards
// Runtime type validation
function isUser(value: unknown): value is User {
return (
typeof value === 'object' &&
value !== null &&
'id' in value &&
'email' in value &&
typeof (value as User).id === 'string' &&
typeof (value as User).email === 'string'
);
}
// Array type guard
function isArrayOf<T>(
arr: unknown,
guard: (item: unknown) => item is T
): arr is T[] {
return Array.isArray(arr) && arr.every(guard);
}
// Assertion function
function assertNonNull<T>(
value: T | null | undefined,
message?: string
): asserts value is T {
if (value === null || value === undefined) {
throw new Error(message ?? 'Value is null or undefined');
}
}
Utility Types
// Deep partial for nested objects
type DeepPartial<T> = {
[P in keyof T]?: T[P] extends object ? DeepPartial<T[P]> : T[P];
};
// Make specific properties required
type RequireFields<T, K extends keyof T> = T & Required<Pick<T, K>>;
// Extract function return type with error handling
type SafeReturn<T extends (...args: any[]) => any> =
ReturnType<T> extends Promise<infer U> ? U : ReturnType<T>;
// Strict omit (errors on invalid keys)
type StrictOmit<T, K extends keyof T> = Pick<T, Exclude<keyof T, K>>;
Error Handling Patterns
Custom Error Hierarchy
// Base application error
class AppError extends Error {
constructor(
message: string,
public readonly code: string,
public readonly statusCode: number = 500,
public readonly isOperational: boolean = true
) {
super(message);
this.name = this.constructor.name;
Error.captureStackTrace(this, this.constructor);
}
}
// Specific error types
class ValidationError extends AppError {
constructor(
message: string,
public readonly fields: Record<string, string>
) {
super(message, 'VALIDATION_ERROR', 400);
}
}
class NotFoundError extends AppError {
constructor(resource: string, id: string) {
super(`${resource} with id ${id} not found`, 'NOT_FOUND', 404);
}
}
class UnauthorizedError extends AppError {
constructor(message = 'Unauthorized') {
super(message, 'UNAUTHORIZED', 401);
}
}
Type-Safe Error Handling
// Global error handler with type narrowing
function handleError(error: unknown): { code: string; message: string } {
if (error instanceof AppError && error.isOperational) {
return { code: error.code, message: error.message };
}
if (error instanceof Error) {
console.error('Unexpected error:', error);
return { code: 'INTERNAL_ERROR', message: 'Something went wrong' };
}
console.error('Unknown error:', error);
return { code: 'UNKNOWN_ERROR', message: 'An unknown error occurred' };
}
// Try-catch wrapper with typed errors
async function tryCatch<T, E = Error>(
fn: () => Promise<T>
): Promise<Result<T, E>> {
try {
const data = await fn();
return { success: true, data };
} catch (error) {
return { success: false, error: error as E };
}
}
Generic Patterns
Repository Pattern
interface Repository<T, ID = string> {
findById(id: ID): Promise<T | null>;
findAll(options?: FindOptions): Promise<T[]>;
create(data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>): Promise<T>;
update(id: ID, data: Partial<T>): Promise<T>;
delete(id: ID): Promise<void>;
}
interface FindOptions {
limit?: number;
offset?: number;
orderBy?: string;
orderDir?: 'asc' | 'desc';
}
Builder Pattern
class QueryBuilder<T> {
private filters: Array<(item: T) => boolean> = [];
private sortFn?: (a: T, b: T) => number;
private limitCount?: number;
where(predicate: (item: T) => boolean): this {
this.filters.push(predicate);
return this;
}
orderBy<K extends keyof T>(key: K, dir: 'asc' | 'desc' = 'asc'): this {
this.sortFn = (a, b) => {
const result = a[key] < b[key] ? -1 : a[key] > b[key] ? 1 : 0;
return dir === 'asc' ? result : -result;
};
return this;
}
limit(count: number): this {
this.limitCount = count;
return this;
}
execute(data: T[]): T[] {
let result = data.filter(item =>
this.filters.every(f => f(item))
);
if (this.sortFn) result = result.sort(this.sortFn);
if (this.limitCount) result = result.slice(0, this.limitCount);
return result;
}
}
Module Organization
Barrel Exports
// types/index.ts - Export all types
export type { User, UserCreate, UserUpdate } from './user';
export type { Order, OrderCreate, OrderStatus } from './order';
export type { ApiResponse, PaginatedResponse } from './api';
// services/index.ts - Export services
export { UserService } from './user.service';
export { OrderService } from './order.service';
Dependency Injection
// Container pattern
interface Container {
get<T>(token: symbol): T;
register<T>(token: symbol, factory: () => T): void;
}
// Service with injected dependencies
class UserService {
constructor(
private readonly db: Database,
private readonly cache: Cache,
private readonly logger: Logger
) {}
}
// Factory function
function createUserService(container: Container): UserService {
return new UserService(
container.get<Database>(DatabaseToken),
container.get<Cache>(CacheToken),
container.get<Logger>(LoggerToken)
);
}
Configuration Patterns
Environment Variables
// Type-safe config
interface Config {
readonly port: number;
readonly nodeEnv: 'development' | 'production' | 'test';
readonly database: {
readonly url: string;
readonly maxConnections: number;
};
}
function loadConfig(): Config {
const port = parseInt(process.env.PORT ?? '3000', 10);
const nodeEnv = process.env.NODE_ENV as Config['nodeEnv'] ?? 'development';
if (!['development', 'production', 'test'].includes(nodeEnv)) {
throw new Error(`Invalid NODE_ENV: ${nodeEnv}`);
}
const dbUrl = process.env.DATABASE_URL;
if (!dbUrl) {
throw new Error('DATABASE_URL is required');
}
return {
port,
nodeEnv,
database: {
url: dbUrl,
maxConnections: parseInt(process.env.DB_MAX_CONN ?? '10', 10),
},
};
}
// Freeze for immutability
export const config: Config = Object.freeze(loadConfig());
Best Practices Checklist
- Use
strict: truein tsconfig.json - Avoid
any- useunknownand narrow with type guards - Prefer
interfacefor object shapes,typefor unions/intersections - Use
readonlyfor immutable properties - Leverage discriminated unions for state management
- Create branded types for domain-specific identifiers
- Implement proper error class hierarchy
- Use assertion functions for runtime validation
- Export types separately from implementations
- Document complex types with JSDoc comments
More from duyet/claude-plugins
react-nextjs-patterns
React and Next.js implementation patterns for performance and maintainability. Use when building frontend components, pages, and applications with React ecosystem.
131backend-api-patterns
Backend and API implementation patterns for scalability, security, and maintainability. Use when building APIs, services, and backend infrastructure.
40quality-gates
Systematic quality verification procedures for code review and delivery. Use when validating completed work, conducting code reviews, or ensuring production readiness.
36gemini-prompting
Prompt engineering guidance for Gemini (Google) model. Use when crafting prompts for Gemini to leverage system instructions, multimodal capabilities, ultra-long context, and strong reasoning features.
26frontend-design
Create distinctive, production-grade frontend interfaces with high design quality. Use this skill when the user asks to build web components, pages, or applications. Generates creative, polished code that avoids generic AI aesthetics.
21orchestration
Orchestrate complex work through parallel agent coordination. Decompose requests into task graphs, spawn background workers, and synthesize results elegantly. Use for multi-component features, large investigations, or any work benefiting from parallelization.
19