npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

typezer

v0.9.30

Published

Parse Typescript files to get a synthetic description of defined types and variables

Downloads

7

Readme

Typezer

Parse Typescript files and retrieve a synthetic description of defined types and variables

Typezer is a high-level wrapper around the Typescript compiler api that focuses on extracting the types of a typescript file.

This library also come with a subpackage dedicated to validate a value against a type previously extracted.

Why?

This library is similar to projects like ts-json-schema-generator but aims to deal with all cases imaginable. In particular, these types are not handled correctly by ts-json-schema-generator:

  • functions are treated as objects,
  • records and mapped types are ignored.

Also, the schema output of Typezer is cleaner, more readable and more powerful than json schema specifications, that cannot deal with things like circular definitions.

A big difference between Typezer and other libraries is that Typezer does not rely on type name to detect the true type. Concretely, it means that that kind of code:

type string = number
let notAString: string = 12

will result with notAString being a string for some other libraries but a number for Typezer.

Instead of name detection, Typezer uses feature detection to make sure that all variables are assigned to their true types.

When to use it

You can use Typezer when you need to do meta-programming with Typescript. A typical example is when you need to generate Typescript code - or anything else - from Typescript source code.

Some use cases:

  • you create your backend API with Typescript and would like to reuse your type definitions to validate input from the client;
  • you are creating an ORM and would like to use Typescript types to define your database schema;
  • you need to extract the types for writing an automatic documentation;
  • etc...

Features

The following types are handled by Typezer:

✓ Never ✓ Any ✓ Null ✓ Undefined ✓ Void ✓ NumberLiteral ✓ StringLiteral ✓ TemplateLiteral ✓ BooleanLiteral ✓ BigIntegerLiteral ✓ Number ✓ String ✓ Boolean ✓ Symbol ✓ BigInteger ✓ RegularExpression ✓ Date ✓ ArrayBuffer ✓ Promise ✓ Object ✓ Namespace ✓ Class ✓ Array ✓ Set ✓ Tuple ✓ Union ✓ Record ✓ Map ✓ Enumeration ✓ Function

Also, Typezer correctly handles:

✓ Property modifiers (static, readonly, optionals, ...) ✓ Circular references ✓ Generics (but not the special case of circular generics; see limitations) ✓ Generic constraints ✓ Watching type changes

API

This library exports three high-level functions:

import {
  findDeclaration,
  findAllDeclarations,
  watchDeclarations
} from "typezer"

function findDeclaration(
  symbol: string, // name of the declaration
  options: Omit<TypezerOptions, "symbols"> // additional options
): {
  declaration: Declaration
  schema: Record<string, Declaration>
}
 
function findManyDeclarations(
  options: TypezerOptions
): Record<string, Declaration>
 
 
function watchDeclarations(
  options: TypezerOptions & { onChange: WatcherCallback }
): Record<string, Declaration>

See the type definitions of Declaration and TypezerOptions below 👇

Example

Let sample.ts be a Typescript file that we want to parse:

// ./sample.ts
const foo = 123

export type Bar = {
  someNumber: number
  someString: string
}

Then we can retrieve the types of top-level declarations very easily with Typezer:

import { findManyDeclarations } from "typezer"

const schema = findManyDeclarations({
  files: ["./sample.ts"]
})

console.log("schema:", schema)

This should print:

schema: {
  foo: {
    id: 'foo',
    name: 'foo',
    fileName: '/full/path/to/sample.ts',
    declare: 'variable',
    exportedAs: [], // not exported
    typeName: 'NumberLiteral',
    value: 123
  },
  Bar: {
    id: 'Bar',
    name: 'Bar',
    fileName: '/full/path/to/sample.ts',
    declare: 'type',
    exportedAs: [ 'Bar' ],
    typeName: 'Object',
    properties: {
      someNumber: { typeName: 'Number' },
      someString: { typeName: 'String' }
    }
  }
}

Watching

It is possible to watch type changes. Typezer will create a dependency tree and update the schema whenever a dependency changes.

import { watchDeclarations } from "typezer"

watchDeclarations({
  files: ["./sample.ts"],
  onChange: (schema) => {
    // executed at startup then every time a change happens
    console.log("schema:", schema)
  }
})

Validation API

Given a schema and a type inside this schema, Typezer exports a submodule to validate any value against this type.

import { validateType } from "typezer/validate"

function validateType(
  schema: Record<string, Declaration>,
  type: Type, // the type to test the value
  value: unknown // the value to test
): Array<string> // errors raised

Example

import { findManyDeclarations } from "typezer"
import { validateType } from "typezer/validate"

const schema = findManyDeclarations({
  files: ["./sample.ts"]
})

let errors: Array<string>

// no errors - 123 matches the number literal 123
errors = validateType(schema, "foo", 123)
console.log("Errors:", errors)

// errors - 200 does not match the number literal 123
errors = validateType(schema, "foo", 200)
console.log("Errors:", errors)

// no errors - someNumber matches a number and someString matches a string
errors = validateType(schema, "Bar", {
  someNumber: 123,
  someString: "123",
})

// you can validate a subtype by using an array to describe the path
errors = validateType(schema, ["Bar", "someNumber"], 123)
errors = validateType(schema, ["Bar", "someString"], 123)

Validating a signature

It is also possible to validate a function signature.

Example:

// ./sample.ts
function foo(someNumber: number, someString: string): string {
  ...
}
import { findManyDeclarations } from "typezer"
import { validateSignature } from "typezer/validate"

const schema = findManyDeclarations({
  files: ["./sample.ts"]
})

const { errors, returnType } = validateSignature(
  schema,
  "foo",
  [123, "123"] // parameters: one number then one string
)

console.log(errors) // []
console.log(returnType) // { typeName: "String" }

If several signatures are declared for a function, the return type will be the one of the first signature that match the parameters.

Limitations

There is one particular edge case where Typezer will crash: the mighty circular generics. It happens when you have a generic function inside a generic type that has a circular reference to the parent generic type.

A concrete example:

type Foo<T1> = {
  foo: <T2 = T1>() => T2
}

Note that generics must be constraint-less for the bug to happen. If you add a constraint to X or Y, Typezer will be able to guess the right types.

The TS compiler API makes it very hard to deal with this kind of situation, so for now it is left as a known bug, but there will be efforts in the future to address this case.

Types

TypezerOptions

type TypezerOptions = {
  files?: string[] // globs of files to look declarations in
  symbols?: string[] // globs of symbol names to match
  compilerOptions?: ts.CompilerOptions // custom ts compiler options
  tsconfigFile?: string // path to custom tscongif.json file
}

If no tsconfigFile is given, Typezer will look for a default config file depending on the path of the first file.

Declaration

type Declaration = Type & {
  id: string
  declare:
    | "namespace"
    | "enumeration"
    | "class"
    | "interface"
    | "type"
    | "variable"
    | "function"
    | "default"
  fileName: string
  name: string
  exportedAs: string[]
}

Type

type Type =
  // -- primitives --
  | { typeName: "Unknown" }
  | { typeName: "Never" }
  | { typeName: "Void" }
  | { typeName: "Any" }
  | { typeName: "Boolean" }
  | { typeName: "Number" }
  | { typeName: "BigInteger" }
  | { typeName: "String" }
  | { typeName: "Symbol" }
  | { typeName: "RegularExpression" }
  | { typeName: "Date" }
  | { typeName: "ArrayBuffer" }

  // -- literals --
  | { typeName: "Null" }
  | { typeName: "Undefined" }
  | {
      typeName: "StringLiteral"
      value: string
    }
  | {
      typeName: "TemplateLiteral"
      texts: string[]
      types: ("string" | "number" | "bigint")[]
    }
  | {
      typeName: "NumberLiteral"
      value: number
    }
  | {
      typeName: "BigIntegerLiteral"
      value: string
    }
  | {
      typeName: "BooleanLiteral"
      value: boolean
    }

  // -- objects --
  | {
      typeName: "Namespace"
      properties: Properties
    }
  | {
      typeName: "Object"
      properties: Properties
    }
  | {
      typeName: "Promise"
      item: Type
    }
  | {
      typeName: "Record"
      keys: Type
      items: Type
    }
  | {
      typeName: "Map"
      keys: Type
      items: Type
    }
  | {
      typeName: "Array"
      items: Type
    }
  | {
      typeName: "Set"
      items: Type
    }
  | {
      typeName: "Tuple"
      items: Type[]
    }
  | {
      typeName: "Union"
      items: Type[]
    }
  | {
      typeName: "Enumeration"
      items: Record<string, Type>
    }
  | {
      typeName: "Function"
      signatures: Signature[]
    }
  | {
      typeName: "Class"
      staticProperties: Properties
      properties: Properties
      signature?: Constructor
    }

  // -- others --
  | {
      typeName: "CircularReference"
      level: number // level of the parent
    }
  | {
      typeName: "Unresolved" // constraint-less generic
      uniqueId: number // unique id to identify the generic
    }