Nest.js 实战 (二):如何使用 Prisma 和连接 PostgreSQL 数据库

本文涉及的产品
云原生数据库 PolarDB MySQL 版,通用型 2核4GB 50GB
云原生数据库 PolarDB PostgreSQL 版,标准版 2核4GB 50GB
简介: 这篇文章介绍了什么是Prisma以及如何在Node.js和TypeScript后端应用中使用它。Prisma是一个开源的下一代ORM,包含PrismaClient、PrismaMigrate、PrismaStudio等部分。文章详细叙述了安装PrismaCLI和依赖包、初始化Prisma、连接数据库、定义Prisma模型、创建Prisma模块的过程,并对比了Prisma和Sequelize在Nest.js中的使用体验,认为Prisma更加便捷高效,没有繁琐的配置。

什么是 Prisma?

Prisma 是一个开源的下一代 ORM。它包含了以下部分:

Prisma 客户端可以被用在 任何 Node.js(支持的版本)或 TypeScript 后端应用中(包括 Serverless 应用和微服务)。可以是一个 REST API,一个 GraphQL API,一个 gRPC API,或任何其他需要数据库的东西。

安装 Prisma

  1. 安装 Prisma CLI 和依赖包
    pnpm add @prisma/client
    pnpm add prisma -D
    
  2. 初始化 Prisma
    npx prisma init
    
    此命令将生成一个基础的 prisma 文件夹和 schema.prisma 文件,文件目录结构如下:
    nest-project
    ├── prisma
    │   ├── schema.prisma // 指定数据库连接并包含数据库 schema
    └── src
    

连接数据库

  1. schema.prisma 文件中配置数据库:

    generator client {
    provider = "prisma-client-js"
    }
    
    datasource db {
    provider = "postgresql" // 这里使用 postgreSQL
    url      = env("DATABASE_URL")
    }
    
  2. .env 文件中配置数据库连接:

    DATABASE_URL="postgresql://postgres:123456@localhost:5432/vue3-admin?schema=public"
    

定义 Prisma 模型

假设现在我们需要开发一个组织管理模块,而且组织是树形层级结构。

  1. schema.prisma 文件中定义 Organization 模型
    model Organization {
    id        String     @id @default(uuid()) // 主键
    name      String     @unique // 组织名称
    code      String     @unique // 组织编码
    parentId    String?  
    parent      Organization? @relation(name: "OrgHierarchy", fields: [parentId], references: [id])
    children Organization[] @relation(name: "OrgHierarchy")
    sort       Int // 排序
    description String? // 组织描述
    icon       String? // 组织图标
    createdAt DateTime @default(now()) // 创建时间
    updatedAt DateTime @updatedAt // 更新时间
    }
    
  2. 迁移数据库,将上述模型同步到数据库中:
    npx prisma migrate dev --name init
    
    此命令会生成 prisma/migrations 目录,文件目录结构如下:
    nest-project
    ├── prisma
    │  ├── migrations
    │   └── 220240715074827_init
    │       └── migration.sql
    │   ├── schema.prisma // 指定数据库连接并包含数据库 schema
    └── src
    
  3. 每次修改 schema.prisma 后,都需要重新生成 Prisma 客户端:
    npx prisma generate
    

创建 Prisma 模块

  1. src/modules/prisma 目录中创建 prisma.service.ts 文件:

    import {
         
          Injectable, OnModuleDestroy, OnModuleInit } from '@nestjs/common';
    import {
         
          PrismaClient } from '@prisma/client';
    
    @Injectable()
    export class PrismaService extends PrismaClient implements OnModuleInit, OnModuleDestroy {
         
         
    constructor() {
         
         
      super();
    }
    async onModuleInit() {
         
         
      await this.$connect(); // 在模块初始化时连接到数据库
    }
    
    async onModuleDestroy() {
         
         
      await this.$disconnect(); // 在应用程序关闭时断开与数据库的连
    }
    }
    
  2. src/modules/prisma 目录中创建 prisma.module.ts 文件:

    import {
         
          Module } from '@nestjs/common';
    
    import {
         
          PrismaService } from './prisma.service';
    
    @Module({
         
         
    providers: [PrismaService],
    exports: [PrismaService],
    })
    export class PrismaModule {
         
          }
    

使用 Prisma 客户端

  1. 创建 organazation 模块,新建 organazation.service.ts 文件:

    import {
         
          Injectable } from '@nestjs/common';
    
    import {
         
          PrismaService } from '@/modules/prisma/prisma.service';
    import {
         
          responseMessage } from '@/utils';
    
    @Injectable()
    export class OrganazationService {
         
         
    constructor(private prisma: PrismaService) {
         
          }
    
    /**
    * @description: 查询组织列表
    */
    async findAll() {
         
         
     const result = await this.prisma.organization.findMany();
     return responseMessage({
         
         
        records: result,
     });
    }
    }
    
  2. organazation.controller.ts 中使用 OrganazationService :

    import {
         
          Controller, Get} from '@nestjs/common';
    import {
         
          OrganazationService } from './organazation.service';
    
    @Controller('organazation')
    export class OrganazationController {
         
         
    constructor(private readonly organazationService: OrganazationService) {
         
          }
    /**
    @description: 查询组织列表
    */
    @Get()
    findAll() {
         
         
    return this.organazationService.findAll();
    }
    }
    
  3. organazation.module.ts 文件:

    import {
         
          Module } from '@nestjs/common';
    
    import {
         
          PrismaModule } from '@/modules/prisma/prisma.module';
    
    import {
         
          OrganazationController } from './organazation.controller';
    import {
         
          OrganazationService } from './organazation.service';
    
    @Module({
         
         
    imports: [PrismaModule],
    controllers: [OrganazationController],
    providers: [OrganazationService],
    exports: [OrganazationService],
    })
    export class OrganazationModule {
         
          }
    
  4. src/app.module.ts 中注册:

    import {
         
          Module } from '@nestjs/common';
    
    import {
         
          OrganazationModule } from '@/modules/administrative/organazation/organazation.module';
    
    @Module({
         
         
    imports: [OrganazationModule],
    })
    export class AppModule {
         
          }
    
  5. 如果 PrismaModule 需要全局注册,修改 prisma.module.ts,并在 AppModule 中导入 :

    import {
         
          Global, Module } from '@nestjs/common';
    
    import {
         
          PrismaService } from './prisma.service';
    
    @Global() // 添加这个装饰器表明这个模块的提供商应该是全局的
    @Module({
         
         
    providers: [PrismaService],
    exports: [PrismaService],
    })
    export class PrismaModule {
         
          }
    

最终效果

cvkq7xv8ay9ujsczss3w135vl5ud83e0.png

总结

说一下我的使用感受,相比之前用的 Sequelize,感觉在 Nest.js 中使用 Prisma 更加便捷高效,没有繁琐的配置。

Github 仓库Vue3-Admin

相关实践学习
使用PolarDB和ECS搭建门户网站
本场景主要介绍基于PolarDB和ECS实现搭建门户网站。
阿里云数据库产品家族及特性
阿里云智能数据库产品团队一直致力于不断健全产品体系,提升产品性能,打磨产品功能,从而帮助客户实现更加极致的弹性能力、具备更强的扩展能力、并利用云设施进一步降低企业成本。以云原生+分布式为核心技术抓手,打造以自研的在线事务型(OLTP)数据库Polar DB和在线分析型(OLAP)数据库Analytic DB为代表的新一代企业级云原生数据库产品体系, 结合NoSQL数据库、数据库生态工具、云原生智能化数据库管控平台,为阿里巴巴经济体以及各个行业的企业客户和开发者提供从公共云到混合云再到私有云的完整解决方案,提供基于云基础设施进行数据从处理、到存储、再到计算与分析的一体化解决方案。本节课带你了解阿里云数据库产品家族及特性。
相关文章
|
8天前
|
设计模式 数据安全/隐私保护
Next.js 实战 (七):浅谈 Layout 布局的嵌套设计模式
这篇文章介绍了在Next.js框架下,如何处理中后台管理系统中特殊页面(如登录页)不包裹根布局(RootLayout)的问题。作者指出Next.js的设计理念是通过布局的嵌套来创建复杂的页面结构,这虽然保持了代码的整洁和可维护性,但对于特殊页面来说,却造成了不必要的布局包裹。文章提出了一个解决方案,即通过判断页面的skipGlobalLayout属性来决定是否包含RootLayout,从而实现特殊页面不包裹根布局的目标。
58 33
|
16天前
|
前端开发 API 开发者
Next.js 实战 (五):添加路由 Transition 过渡效果和 Loading 动画
这篇文章介绍了Framer Motion,一个为React设计的动画库,提供了声明式API处理动画和页面转换,适合创建响应式用户界面。文章包括首屏加载动画、路由加载Loading、路由进场和退场动画等主题,并提供了使用Framer Motion和next.js实现这些动画的示例代码。最后,文章总结了这些效果,并邀请读者探讨更好的实现方案。
|
13天前
|
JavaScript 前端开发 API
Next.js 实战 (六):如何实现文件本地上传
这篇文章介绍了在Next.js中如何实现文件上传到本地的方法。文章首先提到Next.js官方文档中没有提供文件上传的实例代码,因此开发者需要自行实现,通常有两种思路:使用Node.js原生上传或使用第三方插件如multer。接着,文章选择了使用Node.js原生上传的方式来讲解实现过程,包括如何通过哈希值命名文件、上传到指定目录以及如何分类文件夹。然后,文章展示了具体的实现步骤,包括编写代码来处理文件上传,并给出了代码示例。最后,文章通过一个效果演示说明了如何通过postman模拟上传文件,并展示了上传后的文件夹结构。
|
1月前
Next.js 实战 (二):搭建 Layouts 基础排版布局
本文介绍了作者在Next.js v15.x版本发布后,对一个旧项目的重构过程。文章详细说明了项目开发规范配置、UI组件库选择(最终选择了Ant-Design)、以及使用Ant Design的Layout组件实现中后台布局的方法。文末展示了布局的初步效果,并提供了GitHub仓库链接供读者参考学习。
Next.js 实战 (二):搭建 Layouts 基础排版布局
|
1月前
|
存储 网络架构
Next.js 实战 (四):i18n 国际化的最优方案实践
这篇文章介绍了Next.js国际化方案,作者对比了网上常见的方案并提出了自己的需求:不破坏应用程序的目录结构和路由。文章推荐使用next-intl库来实现国际化,并提供了详细的安装步骤和代码示例。作者实现了国际化切换时不改变路由,并把当前语言的key存储到浏览器cookie中,使得刷新浏览器后语言不会失效。最后,文章总结了这种国际化方案的优势,并提供Github仓库链接供读者参考。
|
1月前
Next.js 实战 (三):优雅的实现暗黑主题模式
这篇文章介绍了在Next.js中实现暗黑模式的具体步骤。首先,需要安装next-themes库。然后,在/components/ThemeProvider/index.tsx文件中新增ThemeProvider组件,并在/app/layout.tsx文件中注入该组件。如果想要加入过渡动画,可以修改代码实现主题切换时的动画效果。最后,需要在需要的位置引入ThemeModeButton组件,实现暗黑模式的切换。
|
2月前
|
数据库连接 Linux Shell
Linux下ODBC与 南大通用GBase 8s数据库的无缝连接配置指南
本文详细介绍在Linux系统下配置GBase 8s数据库ODBC的过程,涵盖环境变量设置、ODBC配置文件编辑及连接测试等步骤。首先配置数据库环境变量如GBASEDBTDIR、PATH等,接着修改odbcinst.ini和odbc.ini文件,指定驱动路径、数据库名称等信息,最后通过catalog.c工具或isql命令验证ODBC连接是否成功。
|
2月前
|
JSON JavaScript 关系型数据库
node.js连接GBase 8a 数据库 并进行查询代码示例
node.js连接GBase 8a 数据库 并进行查询代码示例
|
24天前
|
存储 Oracle 关系型数据库
数据库传奇:MySQL创世之父的两千金My、Maria
《数据库传奇:MySQL创世之父的两千金My、Maria》介绍了MySQL的发展历程及其分支MariaDB。MySQL由Michael Widenius等人于1994年创建,现归Oracle所有,广泛应用于阿里巴巴、腾讯等企业。2009年,Widenius因担心Oracle收购影响MySQL的开源性,创建了MariaDB,提供额外功能和改进。维基百科、Google等已逐步替换为MariaDB,以确保更好的性能和社区支持。掌握MariaDB作为备用方案,对未来发展至关重要。
53 3
|
24天前
|
安全 关系型数据库 MySQL
MySQL崩溃保险箱:探秘Redo/Undo日志确保数据库安全无忧!
《MySQL崩溃保险箱:探秘Redo/Undo日志确保数据库安全无忧!》介绍了MySQL中的三种关键日志:二进制日志(Binary Log)、重做日志(Redo Log)和撤销日志(Undo Log)。这些日志确保了数据库的ACID特性,即原子性、一致性、隔离性和持久性。Redo Log记录数据页的物理修改,保证事务持久性;Undo Log记录事务的逆操作,支持回滚和多版本并发控制(MVCC)。文章还详细对比了InnoDB和MyISAM存储引擎在事务支持、锁定机制、并发性等方面的差异,强调了InnoDB在高并发和事务处理中的优势。通过这些机制,MySQL能够在事务执行、崩溃和恢复过程中保持
59 3