ogp 画像を動的に生成したくてタイトルのような処理が必要になった。
より詳細にやりたいことを書くと、下記のような感じ。
- フロントで canvas を使っていい感じの ogp のレイアウトを生成する
- canvas を画像化してサーバ側で保存したいので、base64 にエンコードして、リクエストの body に持たせる
- サーバ側で受け取った base64 にエンコードされた png をデコードして、保存する。
ひとつずつやっていく。
フロントで canvas を使っていい感じの ogp のレイアウトを生成する
ここでは rect をひとつ描画する
const canvas = document.getElementById('canvas') as HTMLCanvasElement;
if (!canvas) {
return;
}
const ctx = canvas.getContext('2d');
ctx.fillStyle = 'skyblue';
ctx.fillRect(0, 0, 100, 100);
こんな感じで表示される
canvas を画像化してサーバ側で保存したいので、base64 にエンコードして、リクエストの body に持たせる
const canvas = document.getElementById('canvas') as HTMLCanvasElement;
if (!canvas) {
return;
}
const ctx = canvas.getContext('2d');
ctx.fillStyle = 'skyblue';
ctx.fillRect(0, 0, 100, 100);
// base64 にエンコード
const canvasAsBase64 = canvas.toDataURL();
console.log('canvasAsBase64', canvasAsBase64);
このような base64 が出力される
ブラウザの url にこの base64 を入力すると、canvas で描画したのと同じ画像が表示される

リクエストする
fetch('api/hoge', {
method: 'POST',
body: JSON.stringify({ pngData: canvasAsBase64 })
});
サーバ側で受け取った base64 にエンコードされた png をデコードして、保存する。
// リクエストデータ取得
const requestData = await JSON.parse(req.body);
const { pngData: pngDataAsBase64 } = requestData;
// base64 の先頭部分の data:image/png;base64, は不要なので削除する
const fileData = pngDataAsBase64.replace(/^data:\w+\/\w+;base64,/, '');
// base64 でデコード (new Buffer() は非推奨らしいので Buffer.from を使用)
const buffer = Buffer.from(fileData, 'base64');
// ファイル保存
fs.writeFileSync('./hoge.png', buffer);
これでぶじ canvas を画像化してサーバに保存することができた🎉
おわり。