
Linkコンポーネントによる画面遷移
はじめに 前回までにNext.jsのページコンポーネント作成方法を学びました。今回は、作成したページ間をシームレスに移動するための「Linkコンポーネント」につ […]
PostgreSQL に続いて、Next.js アプリケーションで MySQL データベースに接続する方法を詳しく説明します。Vercel デプロイも考慮した手順です。
brew install mysql
sudo apt-get install mysql-server
mysql -u root -p
CREATE DATABASE nextjs_mysql_demo;
CREATE USER 'nextjs_user'@'localhost' IDENTIFIED BY 'securepassword';
GRANT ALL PRIVILEGES ON nextjs_mysql_demo.* TO 'nextjs_user'@'localhost';
FLUSH PRIVILEGES;
EXIT;
PlanetScale や AWS RDS などのサービスを使用する場合:
npx create-next-app@latest nextjs-mysql-demo
cd nextjs-mysql-demo
npm install mysql2
# または
yarn add mysql2
.env.local
ファイルを作成: MYSQL_HOST="localhost"
MYSQL_PORT="3306"
MYSQL_DATABASE="nextjs_mysql_demo"
MYSQL_USER="nextjs_user"
MYSQL_PASSWORD="securepassword"
# クラウドサービスの場合は次のような形式になります:
# DATABASE_URL="mysql://username:password@host:port/database?options"
.gitignore
に .env.local
が含まれていることを確認lib/db.ts
ファイルを作成: import { createPool, Pool } from 'mysql2/promise';
import dotenv from 'dotenv';
dotenv.config();
let pool: Pool;
export async function initializePool() {
pool = createPool({
host: process.env.MYSQL_HOST,
port: parseInt(process.env.MYSQL_PORT || '3306'),
database: process.env.MYSQL_DATABASE,
user: process.env.MYSQL_USER,
password: process.env.MYSQL_PASSWORD,
waitForConnections: true,
connectionLimit: 10,
queueLimit: 0,
});
}
export async function query(sql: string, values?: any[]) {
if (!pool) await initializePool();
const [rows] = await pool.query(sql, values);
return rows;
}
scripts/initDB.ts
:import { query } from '../lib/db';
async function initDatabase() {
try {
await query(`
CREATE TABLE IF NOT EXISTS users (
id INT AUTO_INCREMENT PRIMARY KEY,
name VARCHAR(100) NOT NULL,
email VARCHAR(100) UNIQUE NOT NULL,
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
);
`);
console.log('MySQL database initialized successfully');
} catch (error) {
console.error('Error initializing MySQL database:', error);
}
}
initDatabase();
npx ts-node scripts/initDB.ts
pages/api/users.ts
を作成:import { NextApiRequest, NextApiResponse } from 'next';
import { query } from '../../lib/db';
export default async function handler(
req: NextApiRequest,
res: NextApiResponse
) {
try {
switch (req.method) {
case 'GET':
const users = await query('SELECT * FROM users ORDER BY created_at DESC');
res.status(200).json(users);
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 query(
'INSERT INTO users (name, email) VALUES (?, ?)',
[name, email]
);
// MySQLの場合は挿入されたIDを別途取得
const [newUser] = await query('SELECT * FROM users WHERE id = LAST_INSERT_ID()');
res.status(201).json(newUser);
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 { query } 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 users = await query('SELECT * FROM users ORDER BY created_at DESC');
return {
props: {
users: JSON.parse(JSON.stringify(users)), // Dateオブジェクトをシリアライズ
},
revalidate: 60, // ISR: 60秒ごとに再生成
};
};
pages/users/ssr.tsx
:
import { GetServerSideProps } from 'next';
import { query } 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 users = await query('SELECT * FROM users ORDER BY created_at DESC');
return {
props: {
users: JSON.parse(JSON.stringify(users)),
},
};
};
vercel.json
をプロジェクトルートに作成(必要に応じて): {
"version": 2,
"builds": [
{
"src": "package.json",
"use": "@vercel/next"
}
],
"env": {
"MYSQL_HOST": "@mysql_host",
"MYSQL_USER": "@mysql_user",
"MYSQL_PASSWORD": "@mysql_password",
"MYSQL_DATABASE": "@mysql_database"
}
}
vercel
# または GitHub と連携して自動デプロイ
createPool
を使用して接続を効率的に管理JSON.parse(JSON.stringify())
でシリアライズLAST_INSERT_ID()
関数を使用ssl: { rejectUnauthorized: true }
が必要lib/db.ts
を更新して本番環境用に最適化:
import { createPool, Pool } from 'mysql2/promise';
import dotenv from 'dotenv';
dotenv.config();
const isProduction = process.env.NODE_ENV === 'production';
let pool: Pool;
export async function initializePool() {
pool = createPool({
host: process.env.MYSQL_HOST,
port: parseInt(process.env.MYSQL_PORT || '3306'),
database: process.env.MYSQL_DATABASE,
user: process.env.MYSQL_USER,
password: process.env.MYSQL_PASSWORD,
ssl: isProduction ? { rejectUnauthorized: true } : undefined,
waitForConnections: true,
connectionLimit: isProduction ? 20 : 10,
queueLimit: 0,
idleTimeout: 60000, // アイドル接続のタイムアウト(ms)
enableKeepAlive: true, // 接続を維持
keepAliveInitialDelay: 0,
});
}
export async function query(sql: string, values?: any[]) {
if (!pool) await initializePool();
try {
const [rows] = await pool.query(sql, values);
return rows;
} catch (error) {
console.error('MySQL query error:', error);
throw error;
}
}
// アプリケーション終了時に接続プールをクリーンアップ
process.on('SIGINT', async () => {
if (pool) {
await pool.end();
console.log('MySQL connection pool closed');
}
process.exit(0);
});
ssl: { rejectUnauthorized: true }
await query('SELECT * FROM users WHERE id = ?', [userId]);
MySQLでORMを使用する場合:
npm install prisma @prisma/client
npx prisma init
schema.prisma
でMySQLを指定:
datasource db {
provider = "mysql"
url = env("DATABASE_URL")
}
npm install sequelize mysql2
npm install typeorm reflect-metadata mysql2
ER_NOT_SUPPORTED_AUTH_MODE
ALTER USER 'username'@'localhost' IDENTIFIED WITH mysql_native_password BY 'password';
// 接続設定に追加
timezone: '+09:00' // 日本時間の場合
// プール設定に追加
connectTimeout: 10000, // 10秒
CREATE INDEX idx_users_email ON users(email);
const explain = await query('EXPLAIN SELECT * FROM users WHERE email = ?', [email]);
console.log(explain);
このガイドで、Next.js アプリケーションから MySQL データベースに接続する方法を理解できたはずです。実際のプロジェクトでは、要件に応じてさらに最適化を加えてください。