Docker Backend
This guide explains how to run splat-transform as a containerised backend service on a Linux host with an NVIDIA GPU.
When you need this
splat-transform is a Node.js CLI and the published npm package runs anywhere Node 22+ runs — Docker is not strictly required.
A few features are GPU-only and will not run without WebGPU:
--filter-clusterand--filter-floaters..voxel.jsonoutput and-K/--collision-mesh(see the Collision Mesh guide).
A few are GPU-accelerated but optional:
- SOG /
meta.json/lod-meta.json/.htmlviewer output (see SOG Compression Options). The only step inside this writer that uses the GPU is k-means clustering of spherical-harmonic coefficients, so:- Inputs without SH bands (e.g.
.splat, SH-stripped PLYs, or anything piped through-H 0/--filter-harmonics 0) write SOG fully on the CPU. - Inputs with SH bands work on the CPU too via
-g cpu, but SH clustering is roughly 5-10x slower without a GPU.
- Inputs without SH bands (e.g.
All GPU paths go through WebGPU, which on Linux is implemented over Vulkan. That sets the host and container requirements below.
If you don't need the GPU-only features and you're happy paying the CPU cost for SH compression (or your inputs have no SH), skip to the CPU-only variant.
Host prerequisites (outside Docker)
These must be configured on the host machine — they cannot be installed from inside a container, because the container reuses the host's NVIDIA driver and Vulkan ICD.
- NVIDIA GPU with a driver that exposes a Vulkan ICD (most consumer drivers do; some headless cloud drivers do not by default).
- Docker Engine with the NVIDIA Container Toolkit installed and configured. This is what makes
--gpus allwork and what projects the host's NVIDIA driver libraries (including the Vulkan ICD JSON) into the container at runtime. - (Optional)
vulkan-toolson the host — useful for sanity-checking the driver from outside Docker (e.g.vulkaninfo --summaryon the host before launching containers). The container brings its own Vulkan loader via thevulkan-toolsapt package installed in the Dockerfile below.
AWS GPU instances
On AWS GPU instances (g4dn, g5, g6, etc.) running Amazon Linux 2023 the default driver is the compute-only Tesla driver and does not include Vulkan. To use any of the WebGPU paths you need to install the NVIDIA GRID Driver v19.2 or newer on the host — the GRID driver ships the Vulkan ICD that the Tesla driver lacks.
NVIDIA hosts the GRID installer in an S3 bucket, so the EC2 instance role must have the AmazonS3ReadOnlyAccess policy attached.
Update and reboot first:
sudo dnf update -y
sudo reboot
After the reboot, install the driver and reboot again:
sudo aws s3 cp --recursive s3://ec2-linux-nvidia-drivers/grid-19.2/ . # installs v580.x.x
sudo chmod +x NVIDIA-Linux-x86_64*.run
sudo ./NVIDIA-Linux-x86_64*.run
sudo reboot
Install the Vulkan loader, tools, and libXext (which vulkaninfo links against):
sudo dnf install vulkan vulkan-tools libXext
Verify the driver and Vulkan ICD from the host:
nvidia-smi # driver version, GPU model
vulkaninfo --summary # NVIDIA Vulkan ICD with a physical device
This is host/OS level — it cannot be done from inside the container. The Dockerfile in the next section assumes the driver and Vulkan packages are already installed on the host.
Minimal Dockerfile
A minimal multi-stage Dockerfile that installs the published CLI on top of an NVIDIA CUDA + Vulkan base image:
# syntax=docker/dockerfile:1.4
# Pull Node.js from the official image rather than piping a remote
# install script to bash.
FROM node:22-slim AS node
FROM nvidia/cuda:12.8.1-devel-ubuntu22.04
# Expose compute + graphics + utility caps so the NVIDIA Container Toolkit
# projects the host's NVIDIA driver libraries and Vulkan ICD into the
# container at runtime.
ENV NVIDIA_DRIVER_CAPABILITIES=compute,graphics,utility
RUN apt-get update \
&& apt-get install -y --no-install-recommends \
ca-certificates \
vulkan-tools \
libgl1 libglvnd0 libglx0 libegl1 libxext6 \
&& rm -rf /var/lib/apt/lists/*
# Bring in Node.js and npm from the official image.
COPY --from=node /usr/local/bin/ /usr/local/bin/
COPY --from=node /usr/local/lib/node_modules/ /usr/local/lib/node_modules/
RUN npm install -g @playcanvas/splat-transform
RUN useradd --create-home --uid 2000 splat
USER splat
WORKDIR /work
ENTRYPOINT ["splat-transform"]
A few notes on what each part is doing:
nvidia/cuda:...-devel-ubuntu22.04— gives you a CUDA-aware base image that the NVIDIA Container Toolkit recognises. The CUDA toolkit itself is not used bysplat-transform; we just need a base where--gpus allprojects the host's NVIDIA driver and Vulkan ICD correctly.NVIDIA_DRIVER_CAPABILITIES=compute,graphics,utility—graphicsis the important one; without it the toolkit will not mount the Vulkan ICD.vulkan-tools— providesvulkaninfo(useful for debugging) and pulls in the Vulkan loader the container needs.libgl1,libglvnd0,libglx0,libegl1,libxext6— runtime libraries Vulkan/EGL drivers depend on.node:22-slimstage +COPY --from=node— provides Node.js and npm without piping a third-party install script tobash.npm install -g @playcanvas/splat-transform— installs the published CLI on thePATH.- The non-root
splatuser is a good default for batch workers and means files written into a host-mounted directory aren't owned by root.
Build and run
Build the image:
docker build -t splat-transform .
Run a conversion, mounting the current directory as /work:
docker run --rm --gpus all -v "$PWD":/work splat-transform \
input.ply output.sog
The image runs as a non-root user (UID 2000), so any files written into a host-mounted directory will be owned by UID 2000 rather than your host user, and the container will fail to write at all if the host directory isn't writable by UID 2000. The portable fix is to pass --user "$(id -u):$(id -g)":
docker run --rm --gpus all --user "$(id -u):$(id -g)" \
-v "$PWD":/work splat-transform input.ply output.sog
The remaining examples omit --user for brevity — add it to any invocation that mounts a host directory.
Any normal splat-transform invocation works — see the CLI reference for the full CLI surface. For example:
# GPU SOG compression
docker run --rm --gpus all -v "$PWD":/work splat-transform \
-i 20 input.ply output.sog
# Voxel + collision mesh
docker run --rm --gpus all -v "$PWD":/work splat-transform \
room.ply --filter-cluster --seed-pos 0,1,0 \
--voxel-external-fill --voxel-carve \
-K room.voxel.json
Verifying GPU access
Two quick checks confirm the GPU is visible.
List GPU adapters via the CLI:
docker run --rm --gpus all splat-transform --list-gpus
You should see at least one Vulkan adapter named after your GPU (e.g. NVIDIA L4, Tesla T4).
Sanity-check Vulkan directly:
docker run --rm --gpus all --entrypoint vulkaninfo splat-transform --summary
vulkaninfo should print a GPU0 section with an NVIDIA driverID. If this fails, the host setup (driver / NVIDIA Container Toolkit / Vulkan ICD) is the problem, not the image.
CPU-only variant
If you don't need the GPU-only features (--filter-cluster, --filter-floaters, .voxel.json, -K) you can skip the GPU setup entirely and use a much smaller base image. SOG / meta.json / lod-meta.json / .html outputs still work in this image — the SH compression step falls back to CPU:
FROM node:22-slim
RUN npm install -g @playcanvas/splat-transform
WORKDIR /work
ENTRYPOINT ["splat-transform"]
Run it without --gpus:
docker run --rm -v "$PWD":/work splat-transform input.ply output.ply
Unlike the GPU image, node:22-slim runs as root by default, so without --user the files written into a bind-mounted host directory will be owned by root on the host. Pass --user "$(id -u):$(id -g)" (as described in Build and run), or add a non-root USER line to the Dockerfile if this image is for shared use.
If you keep the GPU image but want to force CPU mode for a single run, pass -g cpu:
docker run --rm -v "$PWD":/work splat-transform -g cpu input.ply output.sog
Note that -g cpu is incompatible with the GPU-only features (--filter-cluster, --filter-floaters, .voxel.json, -K). CPU-side SH compression also runs roughly 5-10x slower than GPU.
Troubleshooting
--list-gpusreports no adapters. The Vulkan ICD is not reaching the container. Check that you ran with--gpus all, thatNVIDIA_DRIVER_CAPABILITIESincludesgraphics, and that the NVIDIA Container Toolkit is installed and configured on the host.vulkaninforeportsCould not find a Vulkan ICD. The host NVIDIA driver does not expose Vulkan. On AWS, switch from the Tesla driver to the GRID driver (v19.2+) and reinstall, then installvulkan vulkan-tools libXexton the host.vulkaninfosegfaults or fails to loadlibXext.so.6.libxext6is missing from the image — make sure theapt-get installline in the Dockerfile includes it (and on AWS, thatlibXextis also installed on the host).docker: Error response from daemon: could not select device driver "" with capabilities: [[gpu]]. The NVIDIA Container Toolkit is not installed or the Docker daemon has not picked up its config. Reinstall the toolkit and restart Docker.- Outputs in the bind-mounted directory aren't owned by my host user. The container's default user (UID 2000 in the GPU image, root in the slim image) doesn't match your host UID. Pass
--user "$(id -u):$(id -g)"todocker runto align them, orchownthe output directory to the container UID before mounting. Permission deniedwriting to the bind-mounted directory. The container's default user has no write access to the host directory. Use--user "$(id -u):$(id -g)", orchownthe host directory so the container user can write.splat-transform: command not foundinside the container. Make surenpm install -g @playcanvas/splat-transformran successfully during the build. The default entrypoint already invokes the binary, sodocker run <image> --helpshould work without specifyingsplat-transform.