Prisma ORM: полное руководство для начинающих (и не только). Часть 2
- понедельник, 7 марта 2022 г. в 00:35:12

Привет, друзья!
В этой серии из 2 статей я хочу поделиться с вами своими заметками о Prisma.
Prisma — это современное (продвинутое) объектно-реляционное отображение (Object-Relational Mapping, ORM) для Node.js и TypeScript. Проще говоря, Prisma — это инструмент, позволяющий работать с реляционными (PostgreSQL, MySQL, SQL Server, SQLite) и нереляционной (MongoDB) базами данных с помощью JavaScript или TypeScript без использования SQL (хотя такая возможность имеется).
Если вам это интересно, прошу под кат.
select определяет, какие поля включаются в возвращаемый объект.
const user = await prisma.user.findUnique({
where: { email },
select: {
id: true,
email: true,
first_name: true,
last_name: true,
age: true
}
})
// or
const usersWithPosts = await prisma.user.findMany({
select: {
id: true,
email: true,
posts: {
select: {
id: true,
title: true,
content: true,
author_id: true,
created_at: true
}
}
}
})
// or
const usersWithPostsAndComments = await prisma.user.findMany({
select: {
id: true,
email: true,
posts: {
include: {
comments: true
}
}
}
})include определяет, какие отношения (связанные записи) включаются в возвращаемый объект.
const userWithPostsAndComments = await prisma.user.findUnique({
where: { email },
include: {
posts: true,
comments: true
}
})where определяет один или более фильтр (о фильтрах мы поговорим отдельно), применяемый к свойствам записи или связанных записей:
const admins = await prisma.user.findMany({
where: {
email: {
contains: 'admin'
}
}
})orderBy определяет поля и порядок сортировки. Возможными значениями orderBy являются asc и desc.
const usersByPostCount = await prisma.user.findMany({
orderBy: {
posts: {
count: 'desc'
}
}
})distinct определяет поля, которые должны быть уникальными в возвращаемом объекте.
const distinctCities = await prisma.user.findMany({
select: {
city: true,
country: true
},
distinct: ['city']
})create: { data } | [{ data1 }, { data2 }, ...{ dataN }] — добавляет новую связанную запись или набор записей в родительскую запись. create доступен при создании (create) новой родительской записи или обновлении (update) существующей родительской записиconst user = await prisma.user.create({
data: {
email,
profile: {
// вложенный запрос
create: {
first_name,
last_name
}
}
}
})createMany: [{ data1 }, { data2 }, ...{ dataN }] — добавляет набор новых связанных записей в родительскую запись. createMany доступен при создании (create) новой родительской записи или обновлении (update) существующей родительской записиconst userWithPosts = await prisma.user.create({
data: {
email,
posts: {
// !
createMany: {
data: posts
}
}
}
})update: { data } | [{ data1 }, { data2 }, ...{ dataN }] — обновляет одну или более связанных записейconst user = await prisma.user.update({
where: { email },
data: {
profile: {
// !
update: { age }
}
}
})updateMany: { data } | [{ data1 }, { data2 }, ...{ dataN }] — обновляет массив связанных записей. Поддерживается фильтрацияconst result = await prisma.user.update({
where: { id },
data: {
posts: {
// !
updateMany: {
where: {
published: false
},
data: {
like_count: 0
}
}
}
}
})upsert: { data } | [{ data1 }, { data2 }, ...{ dataN }] — обновляет существующую связанную запись или создает новуюconst user = await prisma.user.update({
where: { email },
data: {
profile: {
// !
upsert: {
create: { age },
update: { age }
}
}
}
})delete: boolean | { data } | [{ data1 }, { data2 }, ...{ dataN }] — удаляет связанную запись. Родительская запись при этом не удаляетсяconst user = await prisma.user.update({
where: { email },
data: {
profile: {
delete: true
}
}
})deleteMany: { data } | [{ data1 }, { data2 }, ...{ dataN }] — удаляет связанные записи. Поддерживается фильтрацияconst user = await prisma.user.update({
where: { id },
data: {
age,
posts: {
// !
deleteMany: {}
}
}
})set: { data } | [{ data1 }, { data2 }, ...{ dataN }] — перезаписывает значение связанной записиconst userWithPosts = await prisma.user.update({
where: { email },
data: {
posts: {
// !
set: newPosts
}
}
})connect — подключает запись к существующей связанной записи по идентификатору или уникальному полюconst user = await prisma.post.create({
data: {
title,
content,
author: {
connect: { email }
}
}
})connectOrCreate — подключает запись к существующей связанной записи по идентификатору или уникальному полю либо создает связанную запись при отсутствии таковой;disconnect — отключает родительскую запись от связанной без удаления последней. disconnect доступен только если отношение является опциональным.equals — значение равняется nconst usersWithNameHarry = await prisma.user.findMany({
where: {
name: {
equals: 'Harry'
}
}
})
// `equals` может быть опущено
const usersWithNameHarry = await prisma.user.findMany({
where: {
name: 'Harry'
}
})not — значение не равняется n;in — значение n содержится в списке (массиве)const usersWithNameAliceOrBob = await prisma.user.findMany({
where: {
user_name: {
// !
in: ['Alice', 'Bob']
}
}
})notIn — n не содержится в списке;lt — n меньше xconst notPopularPosts = await prisma.post.findMany({
where: {
likeCount: {
lt: 100
}
}
})lte — n меньше или равно x;gt — n больше x;gte — n больше или равно x;contains — n содержит xconst admins = await prisma.user.findMany({
where: {
email: {
contains: 'admin'
}
}
})startsWith — n начинается с xconst usersWithNameStartsWithA = await prisma.user.findMany({
where: {
user_name: {
startsWith: 'A'
}
}
})endsWith — n заканчивается x.AND — все условия должны возвращать trueconst notPublishedPostsAboutTypeScript = await prisma.post.findMany({
where: {
AND: [
{
title: {
contains: 'TypeScript'
}
},
{
published: false
}
]
}
})Обратите внимание: оператор указывается до названия поля (снаружи поля), а фильтр после (внутри).
OR — хотя бы одно условие должно возвращать true;NOT — все условия должны возвращать false.some — возвращает все связанные записи, соответствующие одному или более критерию фильтрацииconst usersWithPostsAboutTypeScript = await prisma.user.findMany({
where: {
posts: {
some: {
title: {
contains: 'TypeScript'
}
}
}
}
})every — возвращает все связанные записи, соответствующие всем критериям;none — возвращает все связанные записи, не соответствующие ни одному критерию;is — возвращает все связанные записи, соответствующие критерию;notIs — возвращает все связанные записи, не соответствующие критерию.$disconnect — закрывает соединение с БД, которое было установлено после вызова метода $connect (данный метод чаще всего не требуется вызывать явно), и останавливает движок запросов (query engine) Prismaimport { PrismaClient } from '@prisma/client'
const prisma = new PrismaClient()
async function seedDb() {
try {
await prisma.model.create(data)
} catch (e) {
onError(e)
} finally {
// !
await prisma.$disconnect()
}
}$use — добавляет посредника (middleware)prisma.$use(async (params, next) => {
console.log('Это посредник')
// работаем с `params`
return next(params)
})next — представляет "следующий уровень" в стеке посредников. Таким уровнем может быть следующий посредник или движок запросов Prisma;params — объект со следующими свойствами:action — тип запроса, например, create или findMany;args — аргументы, переданные в запрос, например, where или data;model — модель, например, User или Post;runInTransaction — возвращает true, если запрос был запущен в контексте транзакции;$queryRaw, $executeRaw и $runCommandRaw предназначены для работы с SQL. Почитать о них можно здесь;$transaction — выполняет запросы в контексте транзакции (см. ниже).Подробнее о клиенте можно почитать здесь.
Транзакция — это последовательность операций чтения/записи, которые обрабатываются как единое целое, т.е. либо все операции завершаются успешно, либо все операции отклоняются с ошибкой.
Prisma позволяет использовать транзакции тремя способами:
const newUserWithProfile = await prisma.user.create({
data: {
email,
profile: {
// !
create: {
first_name,
last_name
}
}
}
})createMany, updateMany и deleteManyconst removedUser = await prisma.user.delete({
where: {
email
}
})
// !
await prisma.post.deleteMany({
where: {
author_id: removedUser.id
}
})$transaction.Интерфейс $transaction может быть использован в двух формах:
$transaction([ query1, query2, ...queryN ]) — принимает массив последовательно выполняемых запросов;$transaction(fn) — принимает функцию, которая может включать запросы и другой код.Пример транзакции, возвращающей посты, в заголовке которых встречается слово TypeScript и общее количество постов:
const [postsAboutTypeScript, totalPostCount] = await prisma.$transaction([
prisma.post.findMany({ where: { title: { contains: 'TypeScript' } } }),
prisma.post.count()
])В $transaction допускается использование SQL:
const [userNames, updatedUser] = await prisma.$transaction([
prisma.$queryRaw`SELECT 'user_name' FROM users`,
prisma.$executeRaw`UPDATE users SET user_name = 'Harry' WHERE id = 42`
])Интерактивные транзакции предоставляют разработчикам больший контроль над выполняемыми в контексте транзакции операциями. В данный момент они имеют статус экспериментальной возможности, которую можно включить следующим образом:
generator client {
provider = "prisma-client-js"
previewFeatures = ["interactiveTransactions"]
}Рассмотрим пример совершения платежа.
Предположим, что у Alice и Bob имеется по 100$ на счетах (account), и Alice хочет отправить Bob свои 100$.
import { PrismaClient } from '@prisma/client'
const prisma = new PrismaClient()
async function transfer(from, to, amount) {
try {
await prisma.$transaction(async (prisma) => {
// 1. Уменьшаем баланс отправителя
const sender = await prisma.account.update({
data: {
balance: {
decrement: amount
}
},
where: {
email: from
}
})
// 2. Проверяем, что баланс отправителя после уменьшения >= 0
if (sender.balance < 0) {
throw new Error(`${from} имеет недостаточно средств для отправки ${amount}`)
}
// 3. Увеличиваем баланс получателя
const recipient = await prisma.account.update({
data: {
balance: {
increment: amount
}
},
where: {
email: to
}
})
return recipient
})
} catch(e) {
// обрабатываем ошибку
}
}
async function main() {
// эта транзакция разрешится
await transfer('alice@mail.com', 'bob@mail.com', 100)
// а эта провалится
await transfer('alice@mail.com', 'bob@mail.com', 100)
}
main().finally(() => {
prisma.$disconnect()
})Подробнее о транзакциях можно почитать здесь.
Пожалуй, это все, что я хотел рассказать вам о Prisma.
Благодарю за внимание и happy coding!