0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

【ハンズオン】Next.jsでPrismaを使ってみた

Last updated at Posted at 2025-04-24

はじめに

趣味でNext.jsの公式チュートリアルをやっていた時の話。
「やたらとSQLを記述することが多いから、インターンで利用経験があったPrismaを使って書き直してみよう!」と思ったのですが、最新の記事があまり見当たらなかったので導入方法をまとめてみます。

Next.js公式チュートリアル(App Router編)

Prismaってなに?

そもそもPrismaってなんぞや?となる人もいるかと思います。
PrismaはNode.jsで開発する際に使われるオープンソースのORM(Object-Relational Mapping)ツールです。
ここではORMについて詳しく解説しませんが、ざっくり言うとJavaScriptやTypeScriptのオブジェクトを扱うのと同じようにデータベースを操作することができます!(ざっくりしすぎたかもしれません...)

導入方法

Next.jsのプロジェクトをデプロイする

今回は公式チュートリアルに則ってVelcelを利用しました。もし、Velcelアカウントがなければ作成してgithubアカウントと連携しましょう。

  • 連携が終わったら以下の画面から作成したリポジトリをimport
    01_deploy_1.png
  • プロジェクトに名前をつけてdeployをクリック
    02_deploy_2.png

データベースを作成する

今回はデータベースにPostgresを利用しました。2025年4月現在、多くのプロバイダーがありますがその中でもNeonを選択しました。
NeonはPostgresをサーバーレスで提供しているDBaaS(DataBase as a Service)になります。1アカウントあたり1データベースまで無料で利用できます!

  • プロジェクトダッシュボードからstorageを選択(私は作成済みのため以下のような表示になっています)
    03_db_1.png
  • 一覧からNeonを選択
    04_db_2.png
  • RegionはWashington, D.C.、PlanはFreeを選択してcontinue
    スクリーンショット 2025-04-24 21.09.06.png
  • Next.jsプロジェクト内の.envにDATABASE_URLから始まる記述を貼り付け
    スクリーンショット 2025-04-24 21.14.11.png

Prismaをインストールする

データベースの作成が完了したので本題であるPrismaをインストールしていきます。

  • prisma, prisma clientのインストール
    $ npm install prisma @prisma/client
    
  • スキーマ定義の生成
    $ npx prisma init
    
  • prismaディレクトリ配下に生成されたschema.prismaにモデルを定義
    // This is your Prisma schema file,
    // learn more about it in the docs: https://pris.ly/d/prisma-schema
    
    // Looking for ways to speed up your queries, or scale easily with your serverless or edge functions?
    // Try Prisma Accelerate: https://pris.ly/cli/accelerate-init
    
    generator client {
      provider = "prisma-client-js"
    }
    
    datasource db {
      provider = "postgresql"
      url      = env("DATABASE_URL")
    }
    
    model User {
      id        String      @id @default(dbgenerated("uuid_generate_v4()")) @db.Uuid
      name      String
      email     String      @unique
      password  String
    }
    
    model Invoices {
      id          String      @id @default(dbgenerated("uuid_generate_v4()")) @db.Uuid
      customerId  String      @db.Uuid
      amount      Int
      status      String
      date        DateTime 
    
      customers   Customers   @relation(fields: [customerId], references: [id])
    }
    
    model Customers {
      id          String      @id @default(dbgenerated("uuid_generate_v4()")) @db.Uuid
      name        String
      email       String      @unique
      image_url   String
    
      invoices    Invoices[]
    }
    
    model Revenue {
      month      String       @unique
      revenue    Int
    }
    
  • prismaクライアントを生成してデータベースにスキーマ定義を反映
    $ npx prisma generate
    $ npx prisma db push
    
  • シードスクリプトの書き換え
    • app/lib/prisma.ts
      import { PrismaClient } from "@prisma/client";
      
      export const prisma = new PrismaClient();
      
      export default prisma;
      
    • app/seed/route.ts
      import bcrypt from 'bcryptjs';
      import { PrismaClient } from '@prisma/client';
      import { invoices, customers, revenue, users } from '../lib/placeholder-data';
      
      const prisma = new PrismaClient();
      
      async function seedUsers() {
        const insertedUsers = await Promise.all(
          users.map(async (user) => {
            const hashedPassword = await bcrypt.hash(user.password, 10);
            const seedUser = await prisma.user.create({
              data: {
                id: user.id,
                name: user.name,
                email: user.email,
                password: hashedPassword,
              },
            });
            return seedUser;
          }),
        );
      
        return insertedUsers;
      }
      
      async function seedInvoices() {
        const insertedInvoices = await Promise.all(
          invoices.map(async (invoice) => {
            const seedInvoices = await prisma.invoices.create({
              data: {
                customerId: invoice.customer_id,
                amount: invoice.amount,
                status: invoice.status,
                date: new Date(invoice.date),
              },
            });
            return seedInvoices;
          }),
        );
      
        return insertedInvoices;
      }
      
      (途中略)
      
      export async function GET() {
        try {
          await prisma.$transaction(async () => {
            await seedUsers();
            await seedCustomers();
            await seedInvoices();
            await seedRevenue();
        });
      
          return new Response(JSON.stringify({ message: 'Database seeded successfully' }), {
            status: 200,
          });
        } catch (error) {
          const errorMessage = error instanceof Error ? error.message : 'Unknown error occurred';
          return new Response(JSON.stringify({ error: errorMessage }), { status: 500 });
        }
      }
      
  • データベースにシード
    開発サーバを立ち上げて http://localhost:3000/seed にアクセス
    "Database seeded successfully"と表示されたら成功です!

おわりに

今回はNext.jsにORMであるPrismaを導入する方法についてご紹介しました。ORMに頼り切りになるのは良くないとの記事を目にしたこともありますが、私自身も上手な付き合い方を探していければと思います!

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?