Render videos programmatically from a dataset
You can use Remotion to do a batch render to create many videos based on a dataset. In the following example, we are going to turn a JSON dataset into a series of videos.
We'll start by creating a blank Remotion project:
- npm
- pnpm
- bun
- yarn
bash
bash
bash
bash
bash
bash
bash
bash
Sample dataset
JSON is the most convenient format to import in Remotion. If your dataset is in a different format, you can convert it using one of many available libraries on NPM.
my-data.tsts
my-data.tsts
Sample component
This component will animate a title, subtitle and image using Remotion. Replace the contents of the src/Composition.tsx file with the following:
Composition.tsxtsx
Composition.tsxtsx
Writing the script
In order to render our videos, we'll first need to bundle our project using Webpack and prepare it for rendering.
This can be done by using the bundle() function from the @remotion/bundler package. Make sure to include the webpack override in the bundle if you have one.
tsbundle } from "@remotion/bundler";import {webpackOverride } from "./webpack-override";constbundleLocation = awaitbundle ({entryPoint : "./src/index.ts",// If you have a webpack override, don't forget to add itwebpackOverride :webpackOverride ,});
tsbundle } from "@remotion/bundler";import {webpackOverride } from "./webpack-override";constbundleLocation = awaitbundle ({entryPoint : "./src/index.ts",// If you have a webpack override, don't forget to add itwebpackOverride :webpackOverride ,});
Rendering videos
Import the dataset and loop over each entry.
Fetch the composition metadata for each render using selectComposition(). You have the opportunity to parametrize the duration, width and height of the composition based on the data entry with the calculateMetadata() function.
Trigger a render using renderMedia() and pass the data entry as inputProps. This will pass the object as React props to the component above.
tsrenderMedia ,selectComposition } from "@remotion/renderer";import {data } from "./dataset";for (constentry ofdata ) {constcomposition = awaitselectComposition ({serveUrl :bundleLocation ,id :compositionId ,inputProps :entry ,});awaitrenderMedia ({composition ,serveUrl :bundleLocation ,codec : "h264",outputLocation : `out/${entry .name }.mp4`,inputProps :entry ,});}
tsrenderMedia ,selectComposition } from "@remotion/renderer";import {data } from "./dataset";for (constentry ofdata ) {constcomposition = awaitselectComposition ({serveUrl :bundleLocation ,id :compositionId ,inputProps :entry ,});awaitrenderMedia ({composition ,serveUrl :bundleLocation ,codec : "h264",outputLocation : `out/${entry .name }.mp4`,inputProps :entry ,});}
It is not recommended to render more than one video at once.
Full script
render.mjstsselectComposition ,renderMedia } from "@remotion/renderer";import {webpackOverride } from "./webpack-override";import {bundle } from "@remotion/bundler";import {data } from "./dataset";constcompositionId = "MyComp";constbundleLocation = awaitbundle ({entryPoint : "./src/index.ts",// If you have a webpack override in remotion.config.ts, pass it here as well.webpackOverride :webpackOverride ,});for (constentry ofdata ) {constcomposition = awaitselectComposition ({serveUrl :bundleLocation ,id :compositionId ,inputProps :entry ,});awaitrenderMedia ({composition ,serveUrl :bundleLocation ,codec : "h264",outputLocation : `out/${entry .name }.mp4`,inputProps :entry ,});}
render.mjstsselectComposition ,renderMedia } from "@remotion/renderer";import {webpackOverride } from "./webpack-override";import {bundle } from "@remotion/bundler";import {data } from "./dataset";constcompositionId = "MyComp";constbundleLocation = awaitbundle ({entryPoint : "./src/index.ts",// If you have a webpack override in remotion.config.ts, pass it here as well.webpackOverride :webpackOverride ,});for (constentry ofdata ) {constcomposition = awaitselectComposition ({serveUrl :bundleLocation ,id :compositionId ,inputProps :entry ,});awaitrenderMedia ({composition ,serveUrl :bundleLocation ,codec : "h264",outputLocation : `out/${entry .name }.mp4`,inputProps :entry ,});}
Running the script
- Node
- Bun
bash
bash
To use TypeScript, rename the file to render.ts, install ts-node from npm and run ts-node render.ts. If you get errors, wrap the asynchronous code in an async function and call it.
bash
bash
TypeScript should work out of the box if you rename your file to render.ts.
Rendering videos from a CSV dataset
Use a package like csv2json to convert your dataset into a JSON.



