$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 本体に内包。
→ どこに置いても動く。依存関係ゼロ。サイズは大きい。
→ 再現性が高く、サーバー移動にも強い。
インストール版(動的ビルド)
→ OS の共有ライブラリに依存。
→ OS のバージョンやパッケージ状況で動作が変わる。
→ サイズは小さいが、環境差異で壊れやすい。
インストールできる環境でも、スタンドアロン版の方が安定・安全・再現性が高い。
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
echo $PATH
PHP が実際に使っている PATH
exec('echo $PATH', $out);
var_dump($out);
PHP で絶対パスを使う
which で確認した絶対パスを使う。
0 0 28-31 * * [ "$(date +\%d -d tomorrow)" = "01" ] && 実行したいコマンド
外部コマンドを扱うときは、次の 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のときは、デフォルト引数は適用されない。
$ wget -O /tmp/chrome.rpm https://dl.google.com/linux/direct/google-chrome-stable_current_x86_64.rpm
root で。
# dnf localinstall /tmp/chrome.rpm