VibORM
Relations

One-to-One

Define one-to-one relationships connecting a single record to exactly one other record

Basic Example

import { s } from "viborm";

// User has one profile
const user = s.model({
  id: s.string().id().ulid(),
  email: s.string().unique(),
  profile: s.oneToOne(() => profile).optional(),
});

// Profile belongs to one user
const profile = s.model({
  id: s.string().id().ulid(),
  bio: s.string(),
  userId: s.string().unique(),  // FK field, unique for 1:1
  user: s.oneToOne(() => user)
    .fields("userId")
    .references("id"),
});

Which Side Owns the FK?

In a one-to-one relation, one side must have the foreign key:

SideHas FKConfiguration
Owner (Profile)Yes.fields("userId").references("id")
Non-owner (User)No.optional() if nullable, or nothing

Chainable Methods

.fields(...fieldNames)

Specifies the FK field(s) on the current model:

s.oneToOne(() => user).fields("userId")

.references(...fieldNames)

Specifies the referenced field(s) on the target model:

s.oneToOne(() => user)
  .fields("userId")
  .references("id")

.optional()

Makes the relation optional (can be null):

// User may or may not have a profile
profile: s.oneToOne(() => profile).optional()

.onDelete(action)

Referential action when the related record is deleted:

s.oneToOne(() => user)
  .fields("userId")
  .references("id")
  .onDelete("cascade")  // Delete profile when user is deleted
ActionEffect
cascadeDelete this record too
setNullSet FK to null (requires .optional())
restrictPrevent deletion
noActionDatabase default

.onUpdate(action)

Referential action when the referenced field is updated:

s.oneToOne(() => user)
  .fields("userId")
  .references("id")
  .onUpdate("cascade")

.name(relationName)

Set a custom relation name:

s.oneToOne(() => user).name("owner")

Complete Example

const user = s.model({
  id: s.string().id().ulid(),
  email: s.string().unique(),
  // Non-owning side - no FK, can be optional
  profile: s.oneToOne(() => profile).optional(),
});

const profile = s.model({
  id: s.string().id().ulid(),
  bio: s.string().nullable(),
  avatar: s.string().nullable(),
  // FK field - unique for 1:1
  userId: s.string().unique(),
  // Owning side - has FK configuration
  user: s.oneToOne(() => user)
    .fields("userId")
    .references("id")
    .onDelete("cascade"),
});

Querying One-to-One

// Include profile when fetching user
const user = await client.user.findUnique({
  where: { id: "user_123" },
  include: { profile: true },
});
// user.profile: Profile | null

// Filter users by profile data
const users = await client.user.findMany({
  where: {
    profile: {
      is: { bio: { contains: "developer" } }
    }
  }
});

// Create user with profile
const user = await client.user.create({
  data: {
    email: "alice@example.com",
    profile: {
      create: { bio: "Software developer" }
    }
  },
  include: { profile: true },
});

Common Patterns

Required vs Optional

// Required: Every user MUST have a profile
// (Create profile in same transaction)
profile: s.oneToOne(() => profile)

// Optional: User MAY have a profile
profile: s.oneToOne(() => profile).optional()

Self-Referential

const employee = s.model({
  id: s.string().id().ulid(),
  name: s.string(),
  managerId: s.string().nullable(),
  manager: s.oneToOne(() => employee)
    .fields("managerId")
    .references("id")
    .optional(),
});

On this page