
Linkコンポーネントによる画面遷移
はじめに 前回までにNext.jsのページコンポーネント作成方法を学びました。今回は、作成したページ間をシームレスに移動するための「Linkコンポーネント」につ […]
Next.js アプリケーションでデータベースに接続する方法を、手順を追って詳しく説明します。このガイドでは PostgreSQL を例にしますが、他のデータベースでも同様の原則が適用できます。
brew install postgresql
sudo apt-get install postgresql postgresql-contrib
sudo -u postgres psql
CREATE DATABASE nextjs_demo;
CREATE USER nextjs_user WITH PASSWORD 'securepassword';
GRANT ALL PRIVILEGES ON DATABASE nextjs_demo TO nextjs_user;
\q
Supabase や Neon などのサービスを使用する場合:
npx create-next-app@latest nextjs-database-demo
cd nextjs-database-demo
npm install pg @types/pg dotenv
# または
yarn add pg @types/pg dotenv
.env.local
ファイルを作成: POSTGRES_URL="postgres://nextjs_user:securepassword@localhost:5432/nextjs_demo"
# クラウドサービスの場合は次のような形式になります:
# POSTGRES_URL="postgres://user:password@host:port/database?options"
.gitignore
に .env.local
が含まれていることを確認lib/db.ts
ファイルを作成: import { Pool } from 'pg';
import dotenv from 'dotenv';
dotenv.config();
const pool = new Pool({
connectionString: process.env.POSTGRES_URL,
});
export default {
query: (text: string, params?: any[]) => pool.query(text, params),
};
scripts/initDB.ts
: import db from '../lib/db';
async function initDatabase() {
try {
await db.query(`
CREATE TABLE IF NOT EXISTS users (
id SERIAL PRIMARY KEY,
name VARCHAR(100) NOT NULL,
email VARCHAR(100) UNIQUE NOT NULL,
created_at TIMESTAMP DEFAULT NOW()
);
`);
console.log('Database initialized successfully');
} catch (error) {
console.error('Error initializing database:', error);
}
}
initDatabase();
npx ts-node scripts/initDB.ts
pages/api/users.ts
を作成:import { NextApiRequest, NextApiResponse } from 'next';
import db from '../../lib/db';
export default async function handler(
req: NextApiRequest,
res: NextApiResponse
) {
try {
switch (req.method) {
case 'GET':
const { rows } = await db.query('SELECT * FROM users ORDER BY created_at DESC');
res.status(200).json(rows);
break;
case 'POST':
const { name, email } = req.body;
if (!name || !email) {
return res.status(400).json({ message: 'Name and email are required' });
}
const result = await db.query(
'INSERT INTO users (name, email) VALUES ($1, $2) RETURNING *',
[name, email]
);
res.status(201).json(result.rows[0]);
break;
default:
res.setHeader('Allow', ['GET', 'POST']);
res.status(405).end(`Method ${req.method} Not Allowed`);
}
} catch (error) {
console.error('API Error:', error);
res.status(500).json({ message: 'Internal server error' });
}
}
pages/users/index.tsx
:
import { GetStaticProps } from 'next';
import db from '../../lib/db';
interface User {
id: number;
name: string;
email: string;
created_at: string;
}
export default function UsersPage({ users }: { users: User[] }) {
return (
<div>
<h1>Users</h1>
<ul>
{users.map((user) => (
<li key={user.id}>
{user.name} - {user.email}
</li>
))}
</ul>
</div>
);
}
export const getStaticProps: GetStaticProps = async () => {
const { rows } = await db.query('SELECT * FROM users ORDER BY created_at DESC');
return {
props: {
users: rows,
},
revalidate: 60, // ISR: 60秒ごとに再生成
};
};
pages/users/ssr.tsx
:
import { GetServerSideProps } from 'next';
import db from '../../lib/db';
interface User {
id: number;
name: string;
email: string;
created_at: string;
}
export default function UsersSSRPage({ users }: { users: User[] }) {
return (
<div>
<h1>Users (SSR)</h1>
<ul>
{users.map((user) => (
<li key={user.id}>
{user.name} - {user.email}
</li>
))}
</ul>
</div>
);
}
export const getServerSideProps: GetServerSideProps = async () => {
const { rows } = await db.query('SELECT * FROM users ORDER BY created_at DESC');
return {
props: {
users: rows,
},
};
};
vercel.json
をプロジェクトルートに作成(必要に応じて): {
"version": 2,
"builds": [
{
"src": "package.json",
"use": "@vercel/next"
}
]
}
vercel
# または GitHub と連携して自動デプロイ
本番環境では、接続プーリングを適切に設定することが重要です。lib/db.ts
を更新:
import { Pool } from 'pg';
import dotenv from 'dotenv';
dotenv.config();
const isProduction = process.env.NODE_ENV === 'production';
const connectionString = isProduction
? process.env.POSTGRES_URL
: process.env.POSTGRES_URL;
const pool = new Pool({
connectionString,
ssl: isProduction ? { rejectUnauthorized: false } : false,
max: 20, // 最大接続数
idleTimeoutMillis: 30000, // アイドル状態の接続を閉じるまでの時間
connectionTimeoutMillis: 2000, // 新しい接続のタイムアウト
});
// 接続のリークを防ぐためのクリーンアップ
process.on('exit', () => {
pool.end();
});
export default {
query: (text: string, params?: any[]) => pool.query(text, params),
pool, // 必要に応じてプール自体もエクスポート
};
PostgreSQL 以外にも、Next.js でよく使用されるデータベース:
npm install mongodb
npm install mysql2
npm install better-sqlite3
npm install prisma @prisma/client
npm install sequelize pg pg-hstore
npm install typeorm reflect-metadata pg
このガイドで、Next.js アプリケーションからデータベースに接続する基本的な方法を理解できたはずです。実際のプロジェクトでは、要件に応じてさらに最適化やセキュリティ対策を追加してください。