Smithery Logo
MCPsSkillsDocsPricing
Login
Smithery Logo

Accelerating the Agent Economy

Resources

DocumentationPrivacy PolicySystem Status

Company

PricingAboutBlog

Connect

© 2026 Smithery. All rights reserved.

    0xBigBoss

    typescript-best-practices

    0xBigBoss/typescript-best-practices
    Coding
    35
    3 installs

    About

    SKILL.md

    Install

    Install via Skills CLI

    or add to your agent
    • Claude Code
      Claude Code
    • Codex
      Codex
    • OpenClaw
      OpenClaw
    • Cursor
      Cursor
    • Amp
      Amp
    • GitHub Copilot
      GitHub Copilot
    • Gemini CLI
      Gemini CLI
    • Kilo Code
      Kilo Code
    • Junie
      Junie
    • Replit
      Replit
    • Windsurf
      Windsurf
    • Cline
      Cline
    • Continue
      Continue
    • OpenCode
      OpenCode
    • OpenHands
      OpenHands
    • Roo Code
      Roo Code
    • Augment
      Augment
    • Goose
      Goose
    • Trae
      Trae
    • Zencoder
      Zencoder
    • Antigravity
      Antigravity
    ├─
    ├─
    └─

    About

    Provides TypeScript patterns for type-first development, making illegal states unrepresentable, exhaustive handling, and runtime validation...

    SKILL.md

    TypeScript Best Practices

    Follows type-first, functional, and error handling patterns from CLAUDE.md. This skill covers language-specific idioms only.

    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.

    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> { /* ... */ }
    

    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);
    }
    

    Exhaustive switch with never check:

    type Status = "active" | "inactive";
    
    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}`);
        }
      }
    }
    

    Runtime Validation with Zod

    • Define schemas as single source of truth; infer TypeScript types with z.infer<>. Avoid duplicating types and schemas.
    • Use safeParse for user input where failure is expected; use parse at 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).
    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>;
    
    // Strict parsing at trust boundaries — throws if API contract violated
    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}`);
      }
      return UserSchema.parse(await response.json());
    }
    
    // Caller handles both success and error from user input
    const result = UserSchema.safeParse(formData);
    if (!result.success) {
      setErrors(result.error.flatten().fieldErrors);
      return;
    }
    

    Optional: type-fest

    For advanced type utilities beyond TypeScript builtins, consider type-fest:

    • Opaque<T, Token> - cleaner branded types than manual & { __brand } pattern
    • PartialDeep<T> - recursive partial for nested objects
    • ReadonlyDeep<T> - recursive readonly for immutable data
    • SetRequired<T, K> / SetOptional<T, K> - targeted field modifications
    • Simplify<T> - flatten complex intersection types in IDE tooltips
    import type { Opaque, PartialDeep } from 'type-fest';
    
    type UserId = Opaque<string, 'UserId'>;
    type UserPatch = PartialDeep<User>;
    
    Recommended Servers
    Vercel Grep
    Vercel Grep
    Prisma
    Prisma
    Svelte
    Svelte
    Repository
    0xbigboss/claude-code
    Files