mikro-orm-markdown
Generate Mermaid ERD + Markdown documentation from your MikroORM entities.
Heavily inspired by prisma-markdown by @samchon. Thank you for the great idea.
Features
- Mermaid ERD diagrams generated from MikroORM entity metadata
- Markdown schema documentation with per-entity column tables, actual DB column names, keys, nullability, descriptions, indexes, and constraints
- JSDoc-driven grouping and visibility via
@namespace,@erd,@describe, and@hidden - No live database connection required — uses MikroORM metadata discovery from your config
- Works across common SQL drivers — covered by smoke tests for SQLite, PostgreSQL, MySQL, and MariaDB
MikroORM-specific concepts
Beyond what Prisma-based tools can express, mikro-orm-markdown also visualizes concepts unique to MikroORM:
- Embeddable — a value object stored inside the owning entity's table, either as flattened columns (e.g.
address_street,address_city) or as a JSON column depending on the@Embeddedoptions. No separate table is created. - Single Table Inheritance (STI) — subclasses like
DogandCatshare oneanimalstable. A discriminator column (e.g.type) distinguishes which subclass each row belongs to. - @Formula — a virtual column with no physical DB column. Its value is computed by a SQL expression at SELECT time (e.g.
LENGTH(name)).
These are first-class MikroORM features, though not every project uses all of them.
Embeddableis especially useful for value objects, such as anAddressstored asaddress_*columns or as JSON, and can also reduce duplication when several entities share the same column group.
Requirements
- Node.js >= 18
- MikroORM >= 6 —
@mikro-orm/coreis a peer dependency. - A MikroORM config file — the CLI expects a default export of a plain MikroORM options object.
- The matching MikroORM driver package — for example
@mikro-orm/postgresql,@mikro-orm/mysql,@mikro-orm/mariadb, or@mikro-orm/sqlite. A live database connection is not required, but MikroORM still needs the driver to discover metadata. - Decorator-based entities — entities must be
@Entity()classes.EntitySchema-defined entities are not currently supported. - Resolvable property types — each entity property's type must be known during MikroORM discovery. Use explicit decorator options such as
type:/entity:, or install@mikro-orm/reflectionso the CLI can auto-useTsMorphMetadataProviderfor TypeScript sources. tsxfor TypeScript config files — required only when loading a.tsMikroORM config through the CLI..jsconfig files do not need it.
If you install
@mikro-orm/reflection, keep it at the same exact version as@mikro-orm/core. MikroORM expects official@mikro-orm/*packages to share one version, and mismatches can fail discovery.
Installation
npm install -D mikro-orm-markdown
# or
pnpm add -D mikro-orm-markdownQuick Start
Add a script to your package.json, pointing --config at your MikroORM config file:
{
"scripts": {
"erd": "mikro-orm-markdown --config ./mikro-orm.config.ts --out ./ERD.md --title 'My Database'"
}
}.tsconfig — installtsxas a dev dependency (npm install -D tsx); the CLI loads it automatically and defaults MikroORM discovery toentitiesTsunless you explicitly setpreferTs..jsconfig — no extra packages needed. Can be hand-written, or your own build output (e.g../dist/mikro-orm.config.js).
Then run:
npm run erdCLI Options
| Option | Default | Description |
|---|---|---|
-c, --config <path> |
(required) | Path to MikroORM config file |
-o, --out <path> |
./ERD.md |
Output Markdown file path |
-t, --title <string> |
Database Schema |
H1 heading of the generated document |
-d, --description <string> |
— | Optional description paragraph shown below the title |
--tsconfig <path> |
— | tsconfig.json used when loading a .ts config; defaults to the nearest one beside the config file |
--src <paths...> |
— | Original TypeScript entity source paths/globs; only needed when MikroORM discovers entities from compiled JavaScript |
--mermaid-layout <layout> |
— | Mermaid layout engine (dagre|elk|elk.stress). Omit to use the viewer's default. |
--mermaid-theme <theme> |
— | Mermaid theme (default|neutral|dark|forest|base). Omit to use the viewer's default. |
For a long or multiline description, use the programmatic API instead — it accepts any string directly, without shell quoting limits.
JSDoc Tags
Annotate your entity classes to control sections and visibility in the generated document. JSDoc comments are read from TypeScript entity source files.
Recommended setup: Use a
.tsMikroORM config withentitiesTspointing at your source entities. In this setup, JSDoc is read from the original TypeScript files and--srcis not needed.
/**
* Blog post authored by a registered user.
* @namespace Blog
*/
@Entity()
export class Post {
/** Post title */
@Property()
title!: string;
}Plain JSDoc text (no tag) becomes a description: text above a class describes the entity, and text above a property describes its column. When a property has no JSDoc, its @Property({ comment }) value (the DDL column comment) is used as the column description instead.
| Tag | Description |
|---|---|
@namespace <Name> |
Include entity in section Name (ERD + text table) |
@erd <Name> |
Include in section Name's ERD diagram only |
@describe <Name> |
Include in section Name's text table only |
@hidden |
Exclude entity from the entire document |
Entities with no tag are placed in the default section.
An entity can carry multiple tags to appear in more than one section.
Compiled JavaScript Builds
If your MikroORM config discovers entities from compiled .js files, such as entities: ['./dist/**/*.js'], entity structure can still be discovered, but JSDoc comments may have been stripped.
That means descriptions and tags such as @namespace and @hidden cannot be read from those .js files.
Use --src only in this case:
mikro-orm-markdown \
--config ./dist/mikro-orm.config.js \
--src "src/**/*.entity.ts"If --src matches no files or omits discovered entity declarations, generation fails instead of silently producing incomplete documentation.
Relation Cardinality: @atLeastOne
@atLeastOne is a JSDoc tag, not a TypeScript decorator.
A collection relation (1:N or M:N) renders as zero-or-more by default. Tag the collection property with @atLeastOne to render that collection side as one-or-more instead:
@Entity()
export class Author {
/** @atLeastOne */
@OneToMany(() => Post, (post) => post.author)
posts = new Collection<Post>(this);
}This turns the ERD edge Post }o--|| Author into Post }|--|| Author. It is a documentation hint only — MikroORM has no schema-level minimum, and the count is not enforced. (Mermaid distinguishes only zero-or-more vs. one-or-more, so no larger minimum can be expressed.)
A relation edge has two ends, set independently:
- Singular side (
@ManyToOne, or the owning@OneToOne) — read from your schema automatically, no tag needed: exactly-one (||) by default, or zero-or-one (o|) whennullable: true. - Collection side (
@OneToMany/@ManyToMany) — zero-or-more by default;@atLeastOneraises that side to one-or-more. Mermaid uses}o→}|oro{→|{depending on which side of the edge the collection is rendered on.
The four combinations (Post Author):
Post }o--|| Author → author 0+ posts, post exactly 1 author (default)
Post }o--o| Author → author 0+ posts, post 0-or-1 author (nullable: true)
Post }|--|| Author → author 1+ posts, post exactly 1 author (@atLeastOne)
Post }|--o| Author → author 1+ posts, post 0-or-1 author (both)
NestJS Swagger Plugin:
@namespace,@erd,@describe,@hidden, and@atLeastOneare custom tags formikro-orm-markdown. NestJS Swagger does not use these tags for OpenAPI metadata. If you use entity classes directly as DTOs and enable Swagger comment introspection, the plain JSDoc description may also appear in your Swagger docs, but these custom tags do not create a functional conflict.
Output Example
Given these entities:
/**
* Blog post authored by a registered user.
* @namespace Blog
*/
@Entity()
export class Post {
@PrimaryKey({ type: 'integer' })
id!: number;
/** Post title */
@Property({ type: 'string' })
title!: string;
@Property({ type: 'text', nullable: true })
body?: string;
@ManyToOne({ entity: () => Author })
author!: Author;
}
/** @namespace Blog */
@Entity()
export class Author {
@PrimaryKey({ type: 'integer' })
id!: number;
@Property({ type: 'string' })
name!: string;
@Property({ type: 'string', unique: true })
email!: string;
/** @atLeastOne */
@OneToMany({ entity: () => Post, mappedBy: 'author' })
posts = new Collection<Post>(this);
}Example notes: Imports are omitted for brevity. This example uses explicit
type:options so it works without extra reflection setup. If@mikro-orm/reflectionis installed, MikroORM can also discover simple scalar types from@Property() title!: string.
Both entities share the @namespace Blog tag, so they land in one ## Blog section. With MikroORM's default naming strategy, the generated ERD.md contains an ERD like this:
erDiagram
Post {
integer id PK
string title
text body
integer author_id FK
}
Author {
integer id PK
string name
string email UK
}
Post }|--|| Author : "author"
How the code maps to the output:
@namespace Blog→ both entities are grouped under the## Blogsection@ManyToOne({ entity: () => Author })→ thePosttoAuthorrelation line and theauthor_id FKcolumn@atLeastOneonAuthor.posts→ the collection side is rendered as one-or-more:Post }|--|| Authorunique: trueonemail→emailis markedUK(unique key)@Property({ nullable: true })onbody→ theNullablecell is markedY/** Post title */→ fills the Description cell fortitle
Key legend:
| Marker | Meaning |
|---|---|
PK |
Primary key |
FK |
Foreign key |
UK |
Unique key |
Y in Nullable |
Nullable column |
Each entity also gets a column table. For example, the generated Post section looks like this:
### Post
*Table: `post`*
> Blog post authored by a registered user.
| Column | Type | Key | Nullable | Description |
| --------- | ------- | ----------- | -------- | ----------- |
| id | integer | PK | | |
| title | string | | | Post title |
| body | text | | Y | |
| author_id | integer | FK (author) | | |MikroORM-specific annotations in the generated output:
| Annotation | Meaning |
|---|---|
formula: <expr> |
Mermaid comment for an @Formula computed column |
[EmbeddableType] |
Flat column inlined from an @Embedded value object |
discriminator |
STI discriminator column |
Notes
Single Table Inheritance (STI)
STI is a pattern where multiple entity classes share a single database table, using a discriminator column to tell rows apart.
@Entity({ discriminatorColumn: 'type', abstract: true })
export class Animal {
@PrimaryKey({ type: 'integer' })
id!: number;
@Property({ type: 'string' })
name!: string;
}
@Entity({ discriminatorValue: 'dog' })
export class Dog extends Animal {
@Property({ type: 'string', nullable: true })
breed?: string;
}When an entity uses discriminatorColumn, mikro-orm-markdown detects it automatically. Even though the subclasses share one physical table, each class is drawn as its own box so the diagram shows the effective shape of every subclass:
erDiagram
Animal {
integer id PK
string name
string type "discriminator"
}
Dog {
integer id PK
string name
string type
string breed
}
The root (Animal) lists only the shared columns and marks the discriminator (type); each subclass (Dog) repeats the inherited columns and adds its own.
The generated Markdown table also includes STI notes, such as STI root — discriminator column: type on the root and Extends Animal (Single Table Inheritance, discriminator value: dog) on each subclass.
Trade-off: STI keeps several entity types in one table, but can increase query complexity and produce sparse nullable columns. Use it when sharing one table is an intentional part of your model.
Troubleshooting
"No entities were discovered"
Your MikroORM config found zero entities. This usually means the entity path doesn't match how the CLI is loading your config:
- If you're using a
.tsconfig (the CLI loadstsxautomatically and defaults topreferTs: true), make sureentitiesTspoints to your TypeScript source files. - If you're using a compiled
.jsconfig, make sureentitiespoints to the built output (e.g../dist/**/*.entity.js) and that you've run your build first. - MikroORM uses
entitiesTswhen running in TypeScript mode andentitiesotherwise — if you use folder/file-based discovery, specify both.
"Please provide either 'type' or 'entity' attribute"
MikroORM could not resolve a property type during metadata discovery. The CLI loads .ts configs through tsx, so enabling emitDecoratorMetadata alone will not fix this path.
Fix it in one of these ways:
- Add explicit decorator options, such as
@Property({ type: 'string' })or@ManyToOne({ entity: () => User }). - Install
@mikro-orm/reflectionat the same exact version as@mikro-orm/coreso the CLI can auto-useTsMorphMetadataProvider.
"Cannot find module '@/...'" (path aliases)
If your config or entities use tsconfig path aliases (e.g. @/entities/user), tsx may fail to resolve them when it cannot find the right tsconfig.json. Keeping the config file at your project root (next to tsconfig.json) avoids this. If your config lives elsewhere, pass the right file explicitly:
mikro-orm-markdown --config ./packages/api/mikro-orm.config.ts --tsconfig ./packages/api/tsconfig.jsonJSDoc tags are missing, or @hidden entities appear
Your entities were probably discovered from compiled JavaScript. Build tools may strip comments from .js files, so descriptions, @namespace, and @hidden cannot be read there.
Prefer a .ts config with entitiesTs pointing at your source files. If you must run from compiled .js, pass the original TypeScript sources:
mikro-orm-markdown --config ./dist/mikro-orm.config.js --src "src/**/*.entity.ts"Config file requirements
The config file must have a default export of a plain configuration object:
export default defineConfig({ ... }); // ✅
export const config = defineConfig({ ... }); // ❌ named export not supported
export default async () => defineConfig({ ... }); // ❌ functions/Promises not supportedIf you need to resolve the config asynchronously, use the programmatic API instead (see below).
Advanced Usage
Programmatic API
If you need to integrate ERD generation into a custom build script or process the output programmatically:
import { writeFile } from 'node:fs/promises';
import { generateMarkdown } from 'mikro-orm-markdown';
import ormConfig from './mikro-orm.config.js';
const markdown = await generateMarkdown({
orm: ormConfig,
title: 'My Database',
description: 'Schema documentation generated from MikroORM metadata.',
});
await writeFile('./ERD.md', markdown, 'utf-8');Programmatic options:
| Option | Description |
|---|---|
orm |
MikroORM options object. Required. |
title |
H1 title. Defaults to Database Schema. |
description |
Optional paragraph below the title. Unlike the CLI flag, this can be any string without shell quoting concerns. |
src |
Original TypeScript entity source paths/globs. Only needed when orm.entities discovers compiled JavaScript. |
onWarn |
Callback for non-fatal warnings, such as compiled JavaScript JSDoc loss. |
mermaid |
Optional Mermaid rendering options. See Mermaid rendering options below. |
If your MikroORM config is asynchronous, resolve it yourself and pass the resulting options object:
const ormConfig = await createOrmConfig();
const markdown = await generateMarkdown({ orm: ormConfig });Mermaid rendering options
By default, mikro-orm-markdown does not emit Mermaid frontmatter config. This preserves each Markdown viewer's default Mermaid rendering behavior.
You can opt into Mermaid rendering options via the CLI:
mikro-orm-markdown --config ./mikro-orm.config.ts --mermaid-layout elk
mikro-orm-markdown --config ./mikro-orm.config.ts --mermaid-theme forest
mikro-orm-markdown --config ./mikro-orm.config.ts --mermaid-layout elk --mermaid-theme neutralOr via the programmatic API:
const markdown = await generateMarkdown({
orm: ormConfig,
mermaid: {
layout: 'elk',
theme: 'forest',
},
});When a layout or theme is set, a YAML frontmatter block is prepended to each erDiagram fence:
```mermaid
---
config:
layout: elk
theme: forest
---
erDiagram
...
```Available layout values:
| Value | Description |
|---|---|
dagre |
Mermaid's default layered layout. |
elk |
Alternative layout engine; can improve line routing on larger or denser ERDs. |
elk.stress |
ELK stress layout variant. Support varies by Mermaid version and viewer. |
Available theme values: default, neutral, dark, forest, base.
Viewer support for
elkand theme values varies. If no--mermaid-layoutor--mermaid-themeis provided, no frontmatter is emitted.
License
MIT