【解決方法】next.jsで外部ドメインの画像を呼び出した際に発生したCSPエラー回避の対応

この記事は公開されてから1年以上経過しています。情報が古い可能性がありますので、ご注意ください。
next.jsで外部ドメインの画像を呼び出した際に発生したCSPエラーを回避するための対応をメモしました。
経緯
next.jsのバージョンを10にあげたのと合わせて関連ライブラリのアップデートをこなった際に
外部ドメインからの画像呼び出しがCSPエラーとなり画像が表示されなくなってしまったことがきっかけになります。
環境
- next 9.1.1 → 10.2.3
- node 14.15.3
- express 4.17.1
- helmet 3.21.1 → 4.6.0
発生したエラー
CSPに違反しているため、画像 '' のロードを拒否しましたというエラーが発生しています。
CSP(セキュリティポリシーディレクティブ)エラー
vbnetRefused to load the image '<URL>' because it violates the following Content Security Policy directive: "img-src 'self' data:".
画像呼び出しもとのHTMLタグ
imgタグで外部ドメインを指定して呼び出しを行なっています。
ini<img src="http://subdomain.t-cr.jp/smpale.jpg" />
CSPヘッダーへ追記して解決
CSPヘッダーへ指定ドメインの設定を追記することで解決できました。
next.jsの構成
next.jsの構成としてSSR(サーバーサイドレンダリング)をexpressで行ってます。
またexpressのセキュリティー設定としてhelmetを入れているためhelmetの設定で対応しました。
helmetの設定
helmetの設定を追記
今回は'self'とドメインを追加しました。
server/index.jsimport helmet from "helmet";
import express from "express";
import next from "next";
const nextServer = next();
nextServer.prepare().then(() => {
  const app = express();
  app.use(
    helmet.contentSecurityPolicy({
      useDefaults: true,
      directives: {
        "img-src": ["'self'", "static.t-cr.jp"]
      }
    })
  );
})
helmetの設定
ここでimg-srcへ追記しています。
css    helmet.contentSecurityPolicy({
      useDefaults: true,
      directives: {
        "img-src": ["'self'", "static.t-cr.jp"]
      }
    })
必要に応じてhttps:やdata:などをあけてあげてください。
個別でCSPヘッダーを指定
helmetではなく個別でCSPヘッダーを指定する場合はexpress-csp-headerで設定できます。
express-csp-headerの設定
app.useでexpressCspHeaderの設定を渡してあげます。
server/index.jsimport { expressCspHeader, SELF } from 'express-csp-header';
import express from "express";
import next from "next";
const nextServer = next();
nextServer.prepare().then(() => {
  const app = express();
  app.use(expressCspHeader({
    directives: {
        'img-src': [SELF, 'subdomain.t-cr.jp']
    }
  }));
})
 article article- Next.js でドキュメントポータル:MDX/全文検索/バージョン切替の設計例
 article article- Next.js でインフィニットスクロールを実装:Route Handlers +`use` で滑らかデータ読込
 article article- Redis 使い方:Next.js で Cache-Tag と再検証を実装(Edge/Node 両対応)
 article article- Next.js の RSC 境界設計:Client Components を最小化する責務分離戦略
 article article- Next.js ルーティング早見表:セグメント・グループ・オプションの一枚まとめ
 article article- Next.js × pnpm/Turborepo 初期構築:ワークスペース・共有パッケージ・CI 最適化
 article article- Nano Banana のインストール完全ガイド:macOS/Windows/Linux 別の最短手順
 article article- ESLint が遅い時の処方箋:--cache/並列化/ルール絞り込みの実践
 article article- Dify のコール失敗を解決:429/5xx/タイムアウト時の再試行とバックオフ戦略
 article article- MySQL ERROR 1449 対策:DEFINER 不明でビューやトリガーが壊れた時の復旧手順
 article article- Cursor で差分が崩れる/意図しない大量変更が入るときの復旧プレイブック
 article article- Motion(旧 Framer Motion)で exit が発火しない/遅延する問題の原因切り分けガイド
 blog blog- iPhone 17シリーズの発表!全モデルiPhone 16から進化したポイントを見やすく整理
 blog blog- Googleストアから訂正案内!Pixel 10ポイント有効期限「1年」表示は誤りだった
 blog blog- 【2025年8月】Googleストア「ストアポイント」は1年表記はミス?2年ルールとの整合性を検証
 blog blog- Googleストアの注文キャンセルはなぜ起きる?Pixel 10購入前に知るべき注意点
 blog blog- Pixcel 10シリーズの発表!全モデル Pixcel 9 から進化したポイントを見やすく整理
 blog blog- フロントエンドエンジニアの成長戦略:コーチングで最速スキルアップする方法
 review review- 今の自分に満足していますか?『持たざる者の逆襲 まだ何者でもない君へ』溝口勇児
 review review- ついに語られた業界の裏側!『フジテレビの正体』堀江貴文が描くテレビ局の本当の姿
 review review- 愛する勇気を持てば人生が変わる!『幸せになる勇気』岸見一郎・古賀史健のアドラー実践編で真の幸福を手に入れる
 review review- 週末を変えれば年収も変わる!『世界の一流は「休日」に何をしているのか』越川慎司の一流週末メソッド
 review review- 新しい自分に会いに行こう!『自分の変え方』村岡大樹の認知科学コーチングで人生リセット
 review review- 科学革命から AI 時代へ!『サピエンス全史 下巻』ユヴァル・ノア・ハラリが予見する人類の未来