0

0

Prisma Client 扩展类型提取与模块化管理教程

霞舞

霞舞

发布时间:2025-11-10 18:16:01

|

563人浏览过

|

来源于php中文网

原创

Prisma Client 扩展类型提取与模块化管理教程

本文旨在解决prisma client扩展在模块化组织时遇到的类型复杂性问题。通过深入分析prisma `$extends` 方法的类型结构,我们将学习如何利用typescript的 `extract` 和 `parameters` 工具类型,从基础prisma客户端中精确提取出扩展配置的类型定义。这种方法能有效实现扩展逻辑的分离,同时确保完整的类型安全和代码可维护性。

理解 Prisma Client 扩展及其类型挑战

Prisma Client 扩展(Extensions)是Prisma提供的一项强大功能,允许开发者在不修改生成客户端代码的情况下,为Prisma Client添加自定义逻辑、修改查询行为或引入新的模型方法。这对于实现业务逻辑、审计日志、多租户等场景非常有用。

例如,我们可能需要为 Company 模型的 update 操作添加特定逻辑,如当公司状态变为 DECLINED 时,自动锁定关联用户的账户。

import { PrismaClient, CompanyStatus, AccountLockedReason } from '@prisma/client';

const _prismaClient = new PrismaClient(); // 基础 PrismaClient 实例

const prismaClient = _prismaClient.$extends({
  query: {
    company: {
      update: async ({ args, query }) => {
        if (args.data?.status === CompanyStatus.DECLINED) {
          args.data.user = {
            update: {
              accountLocked: AccountLockedReason.COMPANY_DECLINED,
            },
          };
        }
        return query(args);
      },
    },
  },
});

export type ExtendedPrismaClient = typeof prismaClient;

虽然上述代码在单个文件中运行良好,但随着项目规模的扩大和扩展逻辑的增多,将所有扩展集中在一个文件中会导致代码臃肿且难以维护。理想情况下,我们希望将不同模型的扩展逻辑分离到各自的文件中,例如将 company 相关的扩展逻辑放入 companyExtensions.ts。

模块化扩展的挑战:类型安全

当尝试将扩展逻辑分离时,最大的挑战在于如何为分离出的对象提供准确的TypeScript类型定义,以确保在外部文件中编写扩展时依然能获得完整的类型提示和检查。

考虑以下分离尝试:

// companyExtensions.ts
// 我们需要为 companyExtensions 提供一个类型定义
export const companyExtensions = {
  update: async ({ args, query }) => {
    if (args.data?.status === CompanyStatus.DECLINED) {
      args.data.user = {
        update: {
          accountLocked: AccountLockedReason.COMPANY_DECLINED,
        },
      };
    }
    return query(args);
  },
};

// prismaClient.ts
import { PrismaClient } from '@prisma/client';
import { companyExtensions } from './companyExtensions';

const _prismaClient = new PrismaClient();

const prismaClient = _prismaClient.$extends({
  query: {
    company: companyExtensions, // 这里需要 companyExtensions 具有正确的类型
  },
});

直接将 companyExtensions 定义为一个匿名对象会导致其内部的 args 和 query 参数失去类型信息。简单的尝试,如 type CompanyExtensions = Parameters[0],也往往不能奏效,原因在于 prismaClient 已经是扩展过的实例,其 $extends 方法的类型可能更复杂,且我们通常需要的是未扩展的基客户端的扩展参数类型。此外,Parameters[0] 得到的类型可能是一个非常宽泛的联合类型,难以直接用于局部定义。

Prisma 提供了 defineExtension 函数,但它主要用于发布通用的、可分发的客户端扩展,并且通常不包含针对特定模型参数的详细类型信息,这与我们为特定应用场景定义细粒度扩展的需求不完全匹配。

解决方案:利用 TypeScript 工具类型精确提取

解决这个问题的关键在于利用 TypeScript 的高级工具类型 Parameters 和 Extract,从基础的 _prismaClient 实例中精确地提取出 $extends 方法的配置对象类型。

成新网络商城购物系统
成新网络商城购物系统

使用模板与程序分离的方式构建,依靠专门设计的数据库操作类实现数据库存取,具有专有错误处理模块,通过 Email 实时报告数据库错误,除具有满足购物需要的全部功能外,成新商城购物系统还对购物系统体系做了丰富的扩展,全新设计的搜索功能,自定义成新商城购物系统代码功能代码已经全面优化,杜绝SQL注入漏洞前台测试用户名:admin密码:admin888后台管理员名:admin密码:admin888

下载
  1. 获取 $extends 方法的参数类型Parameters[0] 会返回 _prismaClient.$extends 方法的第一个参数的类型。这个参数是一个包含 query、model、client 等属性的配置对象,其类型通常是一个复杂的联合类型。

  2. 使用 Extract 缩小类型范围Extract 工具类型用于从 Type 中提取所有可赋值给 Union 的成员。在这里,我们可以利用 $extends 配置对象的一个共同特征——它通常包含一个可选的 name 属性(尽管我们不一定使用它),来帮助 TypeScript 准确识别出我们需要的配置对象结构。

结合以上两点,我们可以定义一个通用的 ExtensionArgs 类型:

// types/prisma-extensions.d.ts 或任何 .ts 文件
import { PrismaClient } from '@prisma/client';

// 假设 _prismaClient 是未扩展的 PrismaClient 实例
// 这里我们使用 typeof new PrismaClient() 来获取基础客户端的类型
// 避免循环依赖,或者从一个已知的基础客户端实例获取类型
type BasePrismaClient = PrismaClient; // 也可以直接使用 PrismaClient

/**
 * 提取 Prisma Client $extends 方法的配置参数类型。
 * 通过 Extract<{ name?: string }> 来筛选出符合扩展配置对象特征的类型。
 * 这有助于从复杂的联合类型中精确地获取所需的类型结构。
 */
export type ExtensionArgs = Extract<
  Parameters[0],
  { name?: string }
>;

现在,我们可以使用 ExtensionArgs 来定义我们的模块化扩展:

// companyExtensions.ts
import { CompanyStatus, AccountLockedReason } from '@prisma/client';
import { ExtensionArgs } from './types/prisma-extensions'; // 导入我们定义的类型

// 声明 companyExtensions 为 ExtensionArgs 类型的一部分
// 这里我们只关注 query.company 部分,所以可以进一步缩小类型
export const companyExtensions: ExtensionArgs['query']['company'] = {
  update: async ({ args, query }) => {
    if (args.data?.status === CompanyStatus.DECLINED) {
      args.data.user = {
        update: {
          accountLocked: AccountLockedReason.COMPANY_DECLINED,
        },
      };
    }
    return query(args);
  },
};
// prismaClient.ts
import { PrismaClient } from '@prisma/client';
import { companyExtensions } from './companyExtensions';

const _prismaClient = new PrismaClient(); // 基础 PrismaClient 实例

export const prismaClient = _prismaClient.$extends({
  query: {
    company: companyExtensions,
  },
});

export type ExtendedPrismaClient = typeof prismaClient;

通过这种方式,companyExtensions 对象现在拥有了完整的类型信息。在编写 update 方法时,args 和 query 参数将获得 Prisma 提供的精确类型提示,包括 args.data 的结构、args.where 等。

总结与注意事项

  1. 使用未扩展的客户端类型: 在提取 ExtensionArgs 时,务必使用 _prismaClient(即未扩展的 PrismaClient 实例)的类型,或者直接使用 PrismaClient 类型本身。如果使用已经扩展过的客户端,其 $extends 方法的参数类型可能已经发生变化,导致提取的类型不准确。
  2. { name?: string } 的作用: Extract 结合 { name?: string } 是一种巧妙的方法,用于从 $extends 方法参数的复杂联合类型中“挑选”出代表整个扩展配置对象的那个成员。这是因为 Prisma 内部的扩展配置对象通常会包含一个可选的 name 属性。
  3. 模块化优势: 这种方法极大地提高了代码的可维护性和可读性。你可以为每个模型或每个功能领域创建独立的扩展文件,每个文件都享有完整的类型安全。
  4. 通用性: 这种类型提取模式不仅适用于 query 扩展,也适用于 model 和 client 扩展,只需根据需要调整 ExtensionArgs['query']['company'] 到 ExtensionArgs['model']['YourModel'] 或 ExtensionArgs['client'] 即可。

通过上述方法,开发者可以自信地将 Prisma Client 扩展拆分到独立的模块中,同时享受 TypeScript 带来的强大类型检查和开发体验,从而构建出更加健壮和易于管理的应用。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

443

2023.08.02

c语言union的用法
c语言union的用法

c语言union的用法是一种特殊的数据类型,它允许在相同的内存位置存储不同的数据类型,union的使用可以帮助我们节省内存空间,并且可以方便地在不同的数据类型之间进行转换。使用union时需要注意对应的成员是有效的,并且只能同时访问一个成员。本专题为大家提供union相关的文章、下载、课程内容,供大家免费下载体验。

126

2023.09.27

JavaScript中的typeof用法
JavaScript中的typeof用法

在JavaScript中,typeof是一个用来确定给定变量的数据类型的操作符。可以用来确定一个变量是字符串、数字、布尔值、函数、对象或undefined的数据类型。更多关于typeof用法相关文章,详情请看本专题下面的文章,php中文网欢迎大家前来学习。

752

2023.11.23

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

165

2026.01.28

包子漫画在线官方入口大全
包子漫画在线官方入口大全

本合集汇总了包子漫画2026最新官方在线观看入口,涵盖备用域名、正版无广告链接及多端适配地址,助你畅享12700+高清漫画资源。阅读专题下面的文章了解更多详细内容。

34

2026.01.28

ao3中文版官网地址大全
ao3中文版官网地址大全

AO3最新中文版官网入口合集,汇总2026年主站及国内优化镜像链接,支持简体中文界面、无广告阅读与多设备同步。阅读专题下面的文章了解更多详细内容。

73

2026.01.28

php怎么写接口教程
php怎么写接口教程

本合集涵盖PHP接口开发基础、RESTful API设计、数据交互与安全处理等实用教程,助你快速掌握PHP接口编写技巧。阅读专题下面的文章了解更多详细内容。

2

2026.01.28

php中文乱码如何解决
php中文乱码如何解决

本文整理了php中文乱码如何解决及解决方法,阅读节专题下面的文章了解更多详细内容。

4

2026.01.28

Java 消息队列与异步架构实战
Java 消息队列与异步架构实战

本专题系统讲解 Java 在消息队列与异步系统架构中的核心应用,涵盖消息队列基本原理、Kafka 与 RabbitMQ 的使用场景对比、生产者与消费者模型、消息可靠性与顺序性保障、重复消费与幂等处理,以及在高并发系统中的异步解耦设计。通过实战案例,帮助学习者掌握 使用 Java 构建高吞吐、高可靠异步消息系统的完整思路。

8

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
TypeScript 教程
TypeScript 教程

共19课时 | 2.5万人学习

TypeScript——十天技能课堂
TypeScript——十天技能课堂

共21课时 | 1.1万人学习

TypeScript-45分钟入门
TypeScript-45分钟入门

共6课时 | 0.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号