【Express.js】AWS SDKを使用したS3へのファイルアップロードと署名URLの取得

この記事では、Express.jsを使用してAWS SDKを利用し、S3バケットにファイルをアップロードする方法と、署名付きURL(Presigned URL)を生成する方法を紹介する。

全体像

実装の流れ

  • パッケージインストール
  • 環境変数設定
  • S3に接続するためのクラスを作成する
  • MediaFileモデルの実装
  • MediaFileを操作するためのコントローラーを実装
  • ルーティングを定義

実装

パッケージインストール

以下のパッケージをインストールする。

npm install @aws-sdk/client-s3 @aws-sdk/s3-request-presigner multer uuid
  • @aws-sdk/client-s3:
    Amazon S3 (Simple Storage Service) を操作するためのクライアントライブラリ
  • @aws-sdk/s3-request-presigner:
    Amazon S3のリクエストに対して事前署名URL (Presigned URL) を生成するためのライブラリ
  • multer:
    Node.jsとExpressのためのミドルウェアで、フォームデータ(特にファイルのアップロード)の処理を行う。ファイルのアップロード、アップロードされたファイルの一時保存などをサポートする。

環境変数を設定

バケット名、アクセスキー、シークレットアクセスキー、リージョンを環境変数で登録する。

# AWS S3 設定情報
AWS_S3_BUCKET_NAME=xxxxxxx
AWS_S3_ACCESS_KEY_ID=xxxxxxxxxx
AWS_S3_SECRET_ACCESS_KEY=xxxxxxxxxx
AWS_REGION=ap-northeast-1

S3に接続するためのクラスを作成する

以下のコードでS3クライアントの設定とファイルのアップロード、および署名付きURLを生成するクラスを作成する。

src/config/aws/s3.ts

import {
  S3Client,
  PutObjectCommand,
  GetObjectCommand,
} from "@aws-sdk/client-s3";
import { getSignedUrl } from "@aws-sdk/s3-request-presigner";
import dotenv from "dotenv";
import { v4 as uuidv4 } from "uuid";

dotenv.config();

// Presigned URLの有効期限: 1時間有効で設定
const SIGNED_URL_EXPIRES_IN = 3600;

class AWSS3Config {
  private static instance: AWSS3Config;
  private s3Client: S3Client;

  private constructor() {
    // S3クライアントの初期化
    this.s3Client = new S3Client({
      region: process.env.AWS_REGION,
      credentials: {
        accessKeyId: process.env.AWS_S3_ACCESS_KEY_ID!,
        secretAccessKey: process.env.AWS_S3_SECRET_ACCESS_KEY!,
      },
    });
  }

  public static getInstance(): AWSS3Config {
    if (!AWSS3Config.instance) {
      AWSS3Config.instance = new AWSS3Config();
    }
    return AWSS3Config.instance;
  }

  /**
   * S3にファイルをアップロードする
   * @param file ファイル
   * @param directory ディレクトリパス(例: 'post/{postId}')
   * @returns アップロードされたファイルのURL
   */
  public async uploadToS3(file: Express.Multer.File, directory: string) {
    const params = {
      Bucket: process.env.AWS_S3_BUCKET_NAME,
      Key: `${directory}/${uuidv4()}-${file.originalname}`, // ディレクトリパスを含むユニークなファイル名を生成
      Body: file.buffer,
      ContentType: file.mimetype,
    };

    try {
      const command = new PutObjectCommand(params);
      await this.s3Client.send(command);
      return `https://${process.env.AWS_S3_BUCKET_NAME}.s3.${process.env.AWS_REGION}.amazonaws.com/${params.Key}`;
    } catch (error) {
      console.error("Error uploading to s3: ", error);
      throw error;
    }
  }

  /**
   * Presigned URLを生成する関数(GET操作用)
   * @param key S3オブジェクトのキー(ファイル名)
   * @returns Presigned URL
   */
  public async generatePresignedUrlForViewing(key: string): Promise<string> {
    const command = new GetObjectCommand({
      Bucket: process.env.AWS_S3_BUCKET_NAME,
      Key: key,
    });

    try {
      const url = await getSignedUrl(this.s3Client, command, {
        expiresIn: SIGNED_URL_EXPIRES_IN,
      });
      return url;
    } catch (error) {
      console.error("Error generating presigned URL: ", error);
      throw error;
    }
  }
}

export default AWSS3Config.getInstance();

MediaFileモデルの実装

次に、MediaFileモデルを定義する。このモデルは、アップロードされたファイルのデータを保存する。

src/model/MediaFile.ts

import mongoose, { Document, Schema } from "mongoose";

export interface IMediaFile extends Document {
  url: string;
  title?: string;
  description?: string;
  alt_text?: string;
  file_type: string; // 画像、動画、PDFなど
  uploaded_by: string; // Cognito Sub
  created_at: Date;
  updated_at: Date;
}

const mediaFileSchema = new Schema(
  {
    url: { type: String, required: true },
    title: { type: String },
    description: { type: String },
    alt_text: { type: String },
    file_type: { type: String, required: true }, // 画像、動画、PDFなど
    uploaded_by: { type: String, required: true }, // Cognito Sub
  },
  { timestamps: { createdAt: "created_at", updatedAt: "updated_at" } }
);

const MediaFile = mongoose.model<IMediaFile>("MediaFile", mediaFileSchema);

export default MediaFile;

MediaFileを操作するためのコントローラーを実装

次に、ファイルのアップロードや署名付きURLの生成を行うコントローラーを実装する。

src/controllers/MediaFileController.ts

import { NextFunction, Request, Response } from "express";
import MediaFile from "../model/MediaFile";
import AWSS3Config from "../config/aws/s3";

// メディアファイルの一覧取得
export const getFiles = async (
  req: Request,
  res: Response,
  next: NextFunction
) => {
  try {
    const results = await MediaFile.find();
    return res.status(200).json({ status: "success", data: results });
  } catch (error) {
    next(error);
  }
};

/**
 * 画像をS3にアップロードして、その情報をImageモデルに保存する
 */
export const uploadFile = async (
  req: Request,
  res: Response,
  next: NextFunction
) => {
  try {
    const file = req.file;
    if (!file) {
      return res.status(400).json({ message: "No file uploaded" });
    }

    const directory = req.body.directory || "others";
    // アップロードしてURLを受け取る
    const s3Url = await AWSS3Config.uploadToS3(file, directory);

    // アップロード失敗の場合はエラーにする
    if (!s3Url) {
      throw new Error("S3 upload failed");
    }

    const { title, description, alt_text, uploaded_by, file_type } = req.body;
    // MediaFileモデルに保存
    const newMediaFile = new MediaFile({
      url: s3Url,
      title,
      description,
      alt_text,
      file_type,
      uploaded_by,
    });
    await newMediaFile.save();

    res.status(201).json({ status: "success", data: newMediaFile });
  } catch (error) {
    next(error);
  }
};

/**
 * Presigned URLを生成するコントローラー関数
 */
export const generatePresignedUrlForViewing = async (
  req: Request,
  res: Response,
  next: NextFunction
) => {
  const { url } = req.body;

  // URL未指定の場合は400エラーを返す
  if (!url) {
    return res
      .status(400)
      .json({ status: "error", message: "URL is required" });
  }

  try {
    // URLからバケット名とキーを抽出
    const bucketName = process.env.AWS_S3_BUCKET_NAME;
    const key = url.split(
      `${bucketName}.s3.${process.env.AWS_REGION}.amazonaws.com/`
    )[1];

    // keyが抽出できない場合は400エラーを返す
    if (!key) {
      return res.status(400).json({ status: "error", message: "Invalid URL" });
    }

    // Presigned URLを生成
    const presignedUrl = await AWSS3Config.generatePresignedUrlForViewing(key);
    res.status(200).json({ status: "success", data: presignedUrl });
  } catch (error) {
    next(error);
  }
};

ルーティングを定義

最後に、ルーティングを定義する。ここでは、ファイルのアップロードや署名付きURLの生成を行うエンドポイントを定義。

src/routes/mediaFile.ts

import express from "express";
import {
  generatePresignedUrlForViewing,
  uploadFile,
  getFiles,
} from "../controllers/MediaFileController";
import { checkAuth } from "../middleware/authMiddleware";
import multer from "multer";

const router = express.Router();
const upload = multer();

router.get("/list", getFiles);

router.post("/upload", checkAuth, upload.single("file"), uploadFile);

router.post(
  "/generate-presigned-url",
  checkAuth,
  generatePresignedUrlForViewing
);

export default router;

これで、AWS SDKを使用してS3へのファイルアップロードと署名付きURLの取得ができるようになる。

関連記事

コメント

この記事へのコメントはありません。