# Asset Pipeline

URL: /bengal/docs/reference/architecture/rendering/assets-pipeline/
Section: rendering
Description: How Bengal processes static assets

---

> For a complete page index, fetch /bengal/llms.txt.

Bengal processes static assets (CSS, JS, images, fonts) through two systems:

1. **AssetOrchestrator** (Python) — Copying, fingerprinting, manifest generation
2. **NodePipeline** (optional) — SCSS compilation, PostCSS, JS bundling

## AssetOrchestrator

The core asset processor handles:

| Task | Description |
|------|-------------|
| **Copying** | Copy assets from `assets/` and theme to output |
| **Fingerprinting** | Hash-based filenames for cache-busting (`main.abc123.css`) |
| **Manifest** | JSON mapping of logical → fingerprinted paths |
| **Cleanup** | Remove stale fingerprinted files |

### Processing Flow

```mermaid
flowchart LR
    subgraph "Input"
        A[Site Assets]
        B[Theme Assets]
        C[Pipeline Output]
    end

    subgraph "AssetOrchestrator"
        D[Discover]
        E[Filter Changed]
        F[Process/Copy]
        G[Fingerprint]
        H[Generate Manifest]
    end

    A --> D
    B --> D
    C --> D
    D --> E --> F --> G --> H
```

### Parallel Processing

Assets process in parallel above a threshold:

```python
from bengal.orchestration.asset import AssetOrchestrator

orchestrator = AssetOrchestrator(site)
orchestrator.process(
    assets=site.assets,
    parallel=True,  # Auto-enables ThreadPoolExecutor
    progress_manager=progress,
)
```

### Asset Manifest

Templates access fingerprinted URLs via the manifest:

```kida
<link rel="stylesheet" href="{{ asset_url('css/main.css') }}">
{# Outputs: /assets/css/main.abc123.css #}
```

```python
# Module: bengal/assets/manifest.py
manifest = AssetManifest()
manifest.set_entry(
    "css/main.css",
    "assets/css/main.abc123.css",
    fingerprint="abc123",
    size_bytes=4096,
    updated_at=time.time(),
)
manifest.write(output_dir / "asset-manifest.json")
```

## NodePipeline (Optional)

For modern frontend tooling, enable the Node-based pipeline:

```yaml
# bengal.yaml
assets:
  pipeline: true
  scss: true
  postcss: true
  bundle_js: true
  esbuild_target: es2018
  sourcemaps: true
```

### Requirements

```bash
npm install -D sass postcss postcss-cli autoprefixer esbuild
```

### Supported Transformations

| Stage | Tool | Input | Output |
|-------|------|-------|--------|
| SCSS | `sass` CLI | `.scss` | `.css` |
| PostCSS | `postcss` CLI | `.css` | `.css` (prefixed) |
| JS/TS | `esbuild` | `.js`, `.ts` | Bundled `.js` |

### Pipeline Flow

```mermaid
flowchart LR
    subgraph "Source"
        A[assets/*.scss]
        B[assets/js/*.ts]
    end

    subgraph "NodePipeline"
        C[sass compile]
        D[postcss transform]
        E[esbuild bundle]
    end

    subgraph "Output"
        F[.bengal/pipeline/]
        G[AssetOrchestrator]
    end

    A --> C --> D --> F
    B --> E --> F
    F --> G
```

### Usage

```python
from bengal.assets.pipeline import from_site

pipeline = from_site(site)
compiled_files = pipeline.build()  # Returns list of compiled paths
# Files are written to .bengal/pipeline/ for AssetOrchestrator to pick up
```

## Incremental Processing

During incremental builds, only changed assets are processed:

1. **Hash comparison**: Compare SHA256 of source files
2. **CSS bundling**: Inline `@import` statements into entry points
3. **Selective copy**: Only copy/transform changed files

## Key Modules

| Module | Purpose |
|--------|---------|
| `bengal/orchestration/asset.py` | AssetOrchestrator class |
| `bengal/assets/manifest.py` | Fingerprint manifest generation |
| `bengal/assets/pipeline.py` | NodePipeline (SCSS, PostCSS, esbuild) |
| `bengal/core/asset/` | Asset model and processing logic |

:::{seealso}
- [Assets](/docs/theming/assets/) — User-facing asset documentation
- [Configuration](/docs/building/configuration/) — Asset pipeline settings
:::
