Chris D. MacRae

1 days ago · 4 min read · Web Development

Generating Social Images with Astro

Write HTML and CSS to generate custom social images for all of your content.

We all want our content to be the best possible experience for our visitors.

When a web page is shared on social media networks like Twitter or Mastodon, or in tools like Slack or Notion, a rich preview is displayed if the web page has the necessary metadata to show the image.

I wanted to ensure all of my pages had the best possible preview, so I built a URL-based generator to do just that.

My social images look like this:

Picture of my OG image generation

My social images are generated by React components. 🤯

Here’s the code:

export type OgImageColor = 'primary' | 'success' | 'info' | 'black'

export type OgImageProps = {
  title: string
  color: OgImageColor
}

export const OgImage: React.FC<OgImageProps> = ({title, color = 'black'}) => {
  const headingStyles = getColor(color)

  return (
    <div style={{ 
      display: 'flex', 
      flexDirection: 'column',
      width: '100%', 
      height: '100%', 
      padding: '54px 74px',
      backgroundImage: `url(${dotGrid})`,
      backgroundSize: '100% 100%'
    }}>
      <h1 style={{
        fontSize: '84px',
        fontWeight: 900,
        ...headingStyles
      }}
      >
        {title}
      </h1>
      <h2 style={{ 
        fontSize: '64px',
        marginTop: 'auto', 
        color: 'transparent',
        backgroundImage: 'linear-gradient(to right, rgb(250, 204, 21), rgb(217, 119, 6))',
        backgroundClip: 'text'
      }} 
      >
        Chris D. MacRae
      </h2>
    </div>
  )
}

function getColor(color: OgImageColor) {
  switch (color) {
    case 'primary':
      return {
        color: 'transparent',
        backgroundImage: 'linear-gradient(to right, rgb(192, 132, 252), rgb(219, 39, 119))',
        backgroundClip: 'text'
      }
    case 'success':
      return {
        color: 'transparent',
        backgroundImage: 'linear-gradient(to right, rgb(74, 222, 128), rgb(8, 145, 178))',
        backgroundClip: 'text'
      }
    case 'info':
      return {
        color: 'transparent',
        backgroundImage: 'linear-gradient(to right, rgb(59, 130, 246), rgb(79, 70, 229))',
        backgroundClip: 'text'
      }
    default:
      return {
        color: 'transparent',
        backgroundImage: 'linear-gradient(to right, rgb(250, 204, 21), rgb(217, 119, 6))',
        backgroundClip: 'text'
      }
  }
}

const dotGrid = ``

Enabling SSR

The first step to generating social images is enabling SSR (server-side rendering) for your Astro site.

We do this because generating these images at build time would be slow and, honestly, very frustrating to support.

I will be posting a follow-up on how to generate social images with SSG, however!

We enable SSR by editing our Astro config, found at astro.config.{js,cjs,mjs,ts}:

import { defineConfig } from 'astro/config';

export default defineConfig({
  output: 'server'
})

We’ll also need to enable an adaptor (essentially a plugin that builds our site for a specific hosting provider) for our hosting provider.

Astro supports a wide variety, so do your homework and pick the one that best suits you.

Then, for all of your existing pages, you need to opt back into SSG (static site generation). You can do this by adding the following to the frontmatter of your page:

---
export const prerender = true
---
<!-- Your page content >

Setting Up The Image Generator

To set up the image generator, we need to do 3 things:

  1. We need to setup the React component (or raw HTML) for rendering the image content

  2. We need to setup an API endpoint to do the image generation

  3. We need to add the OG image meta tag to our pages

The result will look like this:

jkhfdsjhdfhjk

Generating the Social Image

To generate the social image, we can either:

I’ll take you through doing both.

Using an Astro component

To generate a component with Astro, we’re going to need to install the dependency satori-html:

npm install satori-html
yarn add satori-html

Now, we create the component. We’ll create a simple component that renders a white background with your page’s title:

---
interface Props {
  title: string
}

const { title } = Astro.props
---
<div class="container">
  <h1>{title}</h1>
</div>
<style>
  .container {
    display: flex;
    justify-content: center;
    align-items: center;
    width: 100%;
    height: 100%;
  }

  h1 {
    color: black;
    font-size: 48px;
    font-weight: 500;
  }
</style>
Using a React or Preact component

To generate a component with React or Preact, we’re going to need to add the relevant Astro integration:

npx astro add react
npx astro add preact

Now, we create the component. We’ll create a simple component that renders a white background with your page’s title:

export interface OgImageProps {
  title: string
}

export const OgImage = ({ title }: OgImageProps) => (
  <div style={{
    display: 'flex',
    justifyContent: 'center',
    alignItems: 'center',
    width: '100%',
    height: '100%'
  }}>
    <h1 style={{
      color: 'black',
      fontSize: '48px',
      fontweight: '500'
    }}>
      {title}
    </h1>
  </div>
)

Setting up the API endpoint

Now that we have a basic component that you can customize to generate your social images, we need to setup an API endpoint that will convert the component to a PNG and serve it to the browser.

We’ll do this in a few steps. To start, we’ll create a basic API endpoint that takes title as an argument via query params and returns a 404 if it’s not provided.

import type { APIRoute } from 'astro'

export const get: APIRoute = async () => {
  const title = url.searchParams.get('title')

  if (!title) {
    return new Response(null, {
      status: 500,
      statusText: "Title missing"
    })
  }

  return new Response(null, {
    status: 200,
    headers: {
      'Content-Type': 'image/png',
    },
  })
}

Next, we’ll create a function to generate our image! To do so, we’ll need to install a few dependencies:

  • satori: the image generation library that converts our HTML to a valid SVG

  • sharp: the image generation library that converts the aforementioned SVG to a PNG

  • yoga-wasm-esm: a library that allows loading WASM (web assembly) in Nodejs

npm install satori sharp yoga-wasm-esm
yarn add satori sharp yoga-wasm-esm

Now, let’s create our function by adding the following to our API endpoint:

import sharp from 'sharp'
// @ts-ignore: no types
import initYoga from 'yoga-wasm-web/asm'
// @ts-ignore: no types
import satori, { init as initSatori } from 'satori/wasm'

const YOGA = initYoga()
initSatori(YOGA)

type ImageOptions = {
  site: string,
  width: number
  height: number,
  debug?: boolean
}

async function generateImage(jsx: any, { width, height, debug }: ImageOptions) {
  const roboto500 = await fetch("https://fonts.gstatic.com/s/inter/v12/UcCO3FwrK3iLTeHuS_fvQtMwCp50KnMw2boKoduKmMEVuI6fAZ9hjp-Ek-_EeA.woff").then(
    (res) => res.arrayBuffer()
  )
  const svg = await satori(
    jsx,
    {
      debug: debug,
      width: width,
      height: height,
      fonts: [
        {
          name: 'Roboto',
          data: roboto400,
          weight: 500,
          style: 'normal',
        }
      ]
    }
  )

  return await sharp(Buffer.from(svg)).png().toBuffer()
}

Now, we can use the generateImage function with our component to return an image from our endpoint:

import sharp from 'sharp'
// @ts-ignore: no types
import initYoga from 'yoga-wasm-web/asm'
// @ts-ignore: no types
import satori, { init as initSatori } from 'satori/wasm'
import Component from '../../components/OgImage.tsx'

const YOGA = initYoga()
initSatori(YOGA)

export const get: APIRoute = async () => {
  const title = url.searchParams.get('title')

  if (!title) {
    return new Response(null, {
      status: 500,
      statusText: "Title missing"
    })
  }

  const args = Object.fromEntries(url.searchParams)
  const imageOptions = { site: site.href, width, height, debug }
  const jsx = Component(args)
  const buffer = await generateImage(jsx, imageOptions)

  return new Response(buffer, {
    status: 200,
    headers: {
      'Content-Type': 'image/png',
      'Cache-Control': 'max-age=31536000, immutable',
    },
  })
}

type ImageOptions = {
  site: string,
  width: number
  height: number,
  debug?: boolean
}

async function generateImage(jsx: any, { width, height, debug }: ImageOptions) {
  const roboto500 = await fetch("https://fonts.gstatic.com/s/inter/v12/UcCO3FwrK3iLTeHuS_fvQtMwCp50KnMw2boKoduKmMEVuI6fAZ9hjp-Ek-_EeA.woff").then(
    (res) => res.arrayBuffer()
  )
  const svg = await satori(
    jsx,
    {
      debug: debug,
      width: width,
      height: height,
      fonts: [
        {
          name: 'Roboto',
          data: roboto400,
          weight: 500,
          style: 'normal',
        }
      ]
    }
  )

  return await sharp(Buffer.from(svg)).png().toBuffer()
}
import sharp from 'sharp'
// @ts-ignore: no types
import initYoga from 'yoga-wasm-web/asm'
// @ts-ignore: no types
import satori, { init as initSatori } from 'satori/wasm'
import Component from '../../components/OgImage.tsx'

const YOGA = initYoga()
initSatori(YOGA)

export const get: APIRoute = async () => {
  const title = url.searchParams.get('title')

  if (!title) {
    return new Response(null, {
      status: 500,
      statusText: "Title missing"
    })
  }

  const args = Object.fromEntries(url.searchParams)
  const imageOptions = { site: site.href, width, height, debug }
  const jsx = Component(args)
  const buffer = await generateImage(jsx, imageOptions)

  return new Response(buffer, {
    status: 200,
    headers: {
      'Content-Type': 'image/png',
      'Cache-Control': 'max-age=31536000, immutable',
    },
  })
}

type ImageOptions = {
  site: string,
  width: number
  height: number,
  debug?: boolean
}

async function generateImage(jsx: any, { width, height, debug }: ImageOptions) {
  const roboto500 = await fetch("https://fonts.gstatic.com/s/inter/v12/UcCO3FwrK3iLTeHuS_fvQtMwCp50KnMw2boKoduKmMEVuI6fAZ9hjp-Ek-_EeA.woff").then(
    (res) => res.arrayBuffer()
  )
  const svg = await satori(
    jsx,
    {
      debug: debug,
      width: width,
      height: height,
      fonts: [
        {
          name: 'Roboto',
          data: roboto400,
          weight: 500,
          style: 'normal',
        }
      ]
    }
  )

  return await sharp(Buffer.from(svg)).png().toBuffer()
}

type ImageOptions = {
  site: string,
  width: number
  height: number,
  debug?: boolean
}

async function generateImage(jsx: any, { width, height, debug }: ImageOptions) {
  const roboto500 = await fetch("https://fonts.gstatic.com/s/inter/v12/UcCO3FwrK3iLTeHuS_fvQtMwCp50KnMw2boKoduKmMEVuI6fAZ9hjp-Ek-_EeA.woff").then(
    (res) => res.arrayBuffer()
  )
  const svg = await satori(
    jsx,
    {
      debug: debug,
      width: width,
      height: height,
      fonts: [
        {
          name: 'Roboto',
          data: roboto400,
          weight: 500,
          style: 'normal',
        }
      ]
    }
  )

  return await sharp(Buffer.from(svg)).png().toBuffer()
}
import sharp from 'sharp'
// @ts-ignore: no types
import initYoga from 'yoga-wasm-web/asm'
// @ts-ignore: no types
import satori, { init as initSatori } from 'satori/wasm'
import { html } from 'satori-html'
import Component from '../../components/OgImage.astro'

const YOGA = initYoga()
initSatori(YOGA)

export const get: APIRoute = async () => {
  const title = url.searchParams.get('title')

  if (!title) {
    return new Response(null, {
      status: 500,
      statusText: "Title missing"
    })
  }

  const args = Object.fromEntries(url.searchParams)
  const imageOptions = { site: site.href, width, height, debug }
  const jsx = html(Component(args))
  const buffer = await generateImage(jsx, imageOptions)

  return new Response(buffer, {
    status: 200,
    headers: {
      'Content-Type': 'image/png',
      'Cache-Control': 'max-age=31536000, immutable',
    },
  })
}

type ImageOptions = {
  site: string,
  width: number
  height: number,
  debug?: boolean
}

async function generateImage(jsx: any, { width, height, debug }: ImageOptions) {
  const roboto500 = await fetch("https://fonts.gstatic.com/s/inter/v12/UcCO3FwrK3iLTeHuS_fvQtMwCp50KnMw2boKoduKmMEVuI6fAZ9hjp-Ek-_EeA.woff").then(
    (res) => res.arrayBuffer()
  )
  const svg = await satori(
    jsx,
    {
      debug: debug,
      width: width,
      height: height,
      fonts: [
        {
          name: 'Roboto',
          data: roboto400,
          weight: 500,
          style: 'normal',
        }
      ]
    }
  )

  return await sharp(Buffer.from(svg)).png().toBuffer()
}

Now go forth, customize your component, and start generating your own social images!

Include social images in your pages

Now that your endpoint is working, you need to actually put the images on your pages.

In the <head> of your page, add the following meta tags:

<meta property="og:image" content={/api/og.png?title=${title}`}/>
<meta property="og:image:width" content="1200"/>
<meta property="og:image:height" content="630"/>