Data ModelRelations

Many-to-many relations

How to define and work with many-to-many relations in Prisma.

Many-to-many (m-n) relations connect zero or more records on one side to zero or more on the other. They can be implicit (Prisma manages the relation table) or explicit (you define the relation table).

Relational databases

Use implicit m-n unless you need to store additional metadata in the relation table.

Explicit many-to-many relations

The relation table is represented as a model in the schema:

model Post {
  id         Int                 @id @default(autoincrement())
  title      String
  categories CategoriesOnPosts[]
}

model Category {
  id    Int                 @id @default(autoincrement())
  name  String
  posts CategoriesOnPosts[]
}

model CategoriesOnPosts {
  post       Post     @relation(fields: [postId], references: [id])
  postId     Int
  category   Category @relation(fields: [categoryId], references: [id])
  categoryId Int
  assignedAt DateTime @default(now())
  assignedBy String
  @@id([postId, categoryId])
}

The relation table can store additional fields like assignedAt and assignedBy.

Querying explicit many-to-many

// Create post with new category
const post = await prisma.post.create({
  data: {
    title: "How to be Bob",
    categories: {
      create: [
        {
          assignedBy: "Bob",
          category: { create: { name: "New category" } },
        },
      ],
    },
  },
});

// Connect to existing categories
await prisma.post.create({
  data: {
    title: "My Post",
    categories: {
      create: [
        { assignedBy: "Bob", category: { connect: { id: 9 } } },
        { assignedBy: "Bob", category: { connect: { id: 22 } } },
      ],
    },
  },
});

// Query posts by category
const posts = await prisma.post.findMany({
  where: { categories: { some: { category: { name: "New Category" } } } },
});

Implicit many-to-many relations

Prisma manages the relation table automatically:

model Post {
  id         Int        @id @default(autoincrement())
  title      String
  categories Category[]
}

model Category {
  id    Int    @id @default(autoincrement())
  name  String
  posts Post[]
}

Querying implicit many-to-many

// Create post with categories
const post = await prisma.post.create({
  data: {
    title: "How to become a butterfly",
    categories: {
      create: [{ name: "Magic" }, { name: "Butterflies" }],
    },
  },
});

// Get posts with categories
const posts = await prisma.post.findMany({
  include: { categories: true },
});

Rules for implicit m-n

  • Both models must have a single @id (no composite IDs or @unique)
  • No @relation attribute needed (unless disambiguating)
  • Cannot use fields, references, onUpdate, or onDelete in @relation

Relation table conventions

For prisma db pull to recognize implicit m-n tables:

  • Table name: _CategoryToPost (underscore + model names alphabetically + To)
  • Columns: A (FK to first model alphabetically) and B (FK to second)
  • Unique index on both columns, non-unique index on B

Configuring relation table name

Use @relation("MyRelationTable") on both sides to customize the table name.

MongoDB

MongoDB requires explicit ID arrays on both sides:

model Post {
  id          String     @id @default(auto()) @map("_id") @db.ObjectId
  categoryIDs String[]   @db.ObjectId
  categories  Category[] @relation(fields: [categoryIDs], references: [id])
}

model Category {
  id      String   @id @default(auto()) @map("_id") @db.ObjectId
  name    String
  postIDs String[] @db.ObjectId
  posts   Post[]   @relation(fields: [postIDs], references: [id])
}

Querying MongoDB m-n

// Find posts by category IDs
const posts = await prisma.post.findMany({
  where: { categoryIDs: { hasSome: [id1, id2] } },
});

// Find posts by category name
const posts = await prisma.post.findMany({
  where: { categories: { some: { name: { contains: "Servers" } } } },
});

On this page