Run
Point ImageForge at your source directory. It converts files to WebP/AVIF and generates blurDataURL placeholders.
imageforge ./public/images -f webp,avifImageForge CLI converts JPG/PNG assets to WebP and AVIF at build time, generates blurDataURL placeholders, writes imageforge.json, and keeps reruns deterministic with hash-based caching.
npm install -g @imageforge/clinpx @imageforge/cli ./public/imagesUse global install for repeated runs, npx for one-off execution.
Stale outputs · missing placeholders · scripts only one person understands
Point ImageForge at your source directory. It converts files to WebP/AVIF and generates blurDataURL placeholders.
imageforge ./public/images -f webp,avifFuture runs reuse hash-based cache data so only modified images are processed.
imageforge ./public/images --concurrency 4
# example: 3 processed, 47 cachedConsume imageforge.json in app code for dimensions, format outputs, and placeholders.
{
"generated": "2026-02-11T09:30:00.000Z",
"images": {
"hero.jpg": {
"blurDataURL": "...",
"outputs": {
"webp": { "path": "hero.webp", "size": 98765 },
"avif": { "path": "hero.avif", "size": 65432 }
}
}
}
}Convert source images to modern formats with deterministic output paths.
-f webp,avif
Generate blurDataURL values and dimensions for fast progressive rendering in Next.js.
--blur
Content and options hashing skips unchanged images so reruns stay fast and predictable.
--cache
Use --check to exit non-zero when images need processing in CI.
--check
Control parallel processing with --concurrency for stable local and CI resource usage.
--concurrency
Use --json to emit a structured run report to stdout for CI logs and automation.
--json
Actual run (example) - 12 source images, single command
Input
8.4 MB
12 JPEGs + PNGs
Output
1.9 MB
WebP/AVIF derivatives
Saved
77%
smaller on disk
Time
2.1s
example run duration
Every run writes imageforge.json with source dimensions, format outputs, blurDataURL, and cache hash metadata. Example timestamp: 2026-02-11T09:30:00.000Z.
{
"version": "1.0",
"generated": "2026-02-11T09:30:00.000Z",
"images": {
"hero.jpg": {
"width": 1200,
"height": 800,
"aspectRatio": 1.5,
"blurDataURL": "...",
"originalSize": 345678,
"outputs": {
"webp": { "path": "hero.webp", "size": 98765 },
"avif": { "path": "hero.avif", "size": 65432 }
},
"hash": "a1b2c3d4e5f67890"
}
}
}Use --check in CI to fail when source images need processing and keep output deterministic across branches.
Failing run
$ imageforge ./public/images --check
[1/3] cached logo.png
[2/3] needs processing hero.jpg
[3/3] needs processing new-banner.png
2 image(s) need processing.
Run: imageforge ./public/images -f webp,avif
Exit code: 1Passing run
$ imageforge ./public/images --check
[1/3] cached logo.png
[2/3] cached hero.jpg
[3/3] cached new-banner.png
All images up to date.
Exit code: 0Import imageforge.json and feed blurDataURL and dimensions into next/image.
// lib/imageforge.ts
import manifest from "../imageforge.json";
export function getImageMeta(src: string) {
return manifest.images[src];
}
// app/page.tsx
import Image from "next/image";
import { getImageMeta } from "@/lib/imageforge";
const hero = getImageMeta("hero.jpg");
const heroSrc = "/images/hero.jpg";
<Image
src={heroSrc}
alt="Hero"
width={hero.width}
height={hero.height}
placeholder="blur"
blurDataURL={hero.blurDataURL}
/>;