ゆう's Blog
WebPへ変換

$img = new Imagick($filepath); $frames = $img->getNumberImages(); if ($frames > 1) { if ($file_ext === 'gif') { // アニメ GIF s_webp -> ffmpeg webp -> gif2webp o_webp -> gif2webp } // アニメ WebP s_webp -> アニメ WebP → PNG → ffmpeg(静止画) webp = o_webp -> そのままコピー(PHP の copy) } else { if ($file_ext === 'gif') { // 静止 GIF s_webp -> ffmpeg webp -> 静止GIF → PNG → cwebp o_webp -> 静止GIF → PNG → cwebp } // JPEG/PNG/静止 WebP s_webp -> ffmpeg webp -> cwebp o_webp -> cwebp }

※ cwebp が GIF をサポートしていない場合
※ ffmpeg で Animated WebP をデコードできない場合

ffmpeg スタンドアロン版とインストール版

スタンドアロン版(静的ビルド)
→ すべての必要ライブラリを ffmpeg 本体に内包。
→ どこに置いても動く。依存関係ゼロ。サイズは大きい。
→ 再現性が高く、サーバー移動にも強い。

インストール版(動的ビルド)
→ OS の共有ライブラリに依存。
→ OS のバージョンやパッケージ状況で動作が変わる。
→ サイズは小さいが、環境差異で壊れやすい。

インストールできる環境でも、スタンドアロン版の方が安定・安全・再現性が高い。

ffmpeg でアニメGIFを「中央クロップ150px角のアニメWebP」に変換する

ffmpeg の crop は 入力画像より大きいサイズを指定すると必ずエラーになるため scale で短い方を150pxに“拡大”して揃えてから中央150px角を cropする。

アニメGIF → アニメWebP

ffmpeg -i input.gif \ -vf "scale='if(gt(iw,ih),-1,150)':'if(gt(iw,ih),150,-1)',crop=150:150" \ -c:v libwebp -loop 0 output.webp

アニメGIF → 静止サムネイル(PNG)

ffmpeg -i input.gif \ -vf "scale='if(gt(iw,ih),-1,150)':'if(gt(iw,ih),150,-1)',crop=150:150" \ -vframes 1 temp.png

● 1. まず短い方を 必ず150pxに揃える
横長(iw > ih)
→ 高さを150に、幅は自動計算(-1)
縦長(ih > iw)
→ 幅を150に、高さは自動計算(-1)
つまり 短い方が必ず150pxになる
● 2. その後に中央150px角を crop

ffmpeg の “increase” が効かない特殊ケース

Invalid too big or non positive size for width '150' or height '150' Failed to configure input pad on Parsed_crop_1 Error reinitializing filters! Failed to inject frame into filter network: Invalid argument Error while filtering: Invalid argument

内部で「拡大後のサイズが整数にならない」などの理由で crop の初期化に失敗することがある。
220×123 の GIF は、縦が 123px(150px 未満)なので本来は increase で 150px 以上に拡大されるはずですが、 拡大後のサイズが 奇数/偶数の組み合わせで ffmpeg が scale → crop の再初期化に失敗することがあります。

シェルの PATH と PHP の PATH は違う

シェルの PATH

echo $PATH

PHP が実際に使っている PATH

exec('echo $PATH', $out); var_dump($out);

PHP で絶対パスを使う
which で確認した絶対パスを使う。

月末にのみ実行する crontab の定番パターン

0 0 28-31 * * [ "$(date +\%d -d tomorrow)" = "01" ] && 実行したいコマンド

PHP の外部コマンド実行に必要な 3 つの要素

外部コマンドを扱うときは、次の 3 つが基本になります。
sprintf:文字列をフォーマットしてコマンドを組み立てる
escapeshellarg:引数を安全にエスケープする
exec:コマンドを実行して結果を受け取る

🧱 1. sprintf:フォーマットしてコマンドを組み立てる
役割
「%s の部分に変数を埋め込んで、整った文字列を作る」

$cmd = sprintf('cwebp %s -o %s', $input, $output);

🛡 2. escapeshellarg:引数を安全にする
役割
「シェルで危険な文字を無害化して、1つの引数として扱わせる」

$input = escapeshellarg($inputPath); $output = escapeshellarg($outputPath);

🚀 3. exec:コマンドを実行して結果を受け取る

exec($cmd, $output, $ret);

引数の意味
$cmd:実行するコマンド
$output:コマンドの標準出力(配列で返る)
$ret:終了ステータス(0 が成功)

exec($cmd, $output, $ret); if ($ret !== 0) { echo "エラーが発生しました"; }

🧩 3 つを組み合わせた「正しい外部コマンド実行」

$input = escapeshellarg($inputPath); $output = escapeshellarg($outputPath); $cmd = sprintf( 'cwebp %s -o %s', $input, $output ); exec($cmd, $outputLines, $ret); if ($ret !== 0) { // エラー処理 }

画像変換

SVG → PNG/WebP の品質は、
rsvg-convert + cwebp の組み合わせが最強。

SVG → PNG(rsvg-convert)
PNG → WebP(cwebp)

GIF / JPEG / PNG / WebP → WebP の品質は、
cwebp + gif2webp の組み合わせが最強。

🌟 cwebp は何をするツールなのか
✔ 入力:PNG / JPEG / TIFF / BMP / PPM / WebP(再圧縮)
✔ 出力:WebP(lossy または lossless)

GIF と WebP 変換の正しい整理

1. アニメーション GIF
フレームを複数持つ
→ gif2webp が必須
cwebp はアニメーションを扱えない
フレーム保持・最適化・ループ情報などを正しく処理できるのは gif2webp

2. 静止画 GIF(1フレーム)
実体は「256色の PNG に近い静止画像」
→ cwebp で完全に変換可能
透明度もそのまま扱える
画質も PNG と同じ扱いで良い

判定

● GIF のフレーム数をチェック

$img = new Imagick($input);
if ($img->getNumberImages() > 1) {
  $isAnimatedGif = true;
}

または、

$cmd = sprintf('identify -format "%%n" %s', escapeshellarg($tmpImg));
$frames = (int) trim(shell_exec($cmd));

但し、cwebp が GIF をサポートしている必要がある。
GIF デコーダを含まないビルドであるならば、最適な変換フローは、

JPEG / PNG / WebP → cwebp
静止 GIF → PNG(Imagick)→ WebP(cwebp)
アニメ GIF → gif2webp

また、cwebpでリサイズはできるが、gif2webpではリサイズはできない。アニメgifのリサイズ+WebPへの変換は ffmpeg が使える。

配列をスペース区切りの文字列にしたい

const list = ["one", "two", "three"]; const result = list.join(" "); console.log(result); // "one two three"

デフォルト引数

引数がundefinedのときデフォルト値が使われる。
nullのときは、デフォルト引数は適用されない。

CentOS Stream に Google Chrome をインストールする

$ wget -O /tmp/chrome.rpm https://dl.google.com/linux/direct/google-chrome-stable_current_x86_64.rpm

root で。

# dnf localinstall /tmp/chrome.rpm