Skip to main content

Open Graph Image Generator v2

The Open Graph Image Generator is a function that generates an Open Graph image using HTML template.

It uses Puppeteer and Chromium to launch a headless browser, set to use HTML template as a content for the page, to take a screenshot, and return the image as a base64-encoded string. It handles HTTP GET requests and returns appropriate error messages for invalid or missing parameters. The function is compatible with Netlify and has a maximum execution time of 10 seconds before Netlify closes the connection.

The source code for the function:

const puppeteer = require("puppeteer-core");
const chromium = require("@sparticuz/chromium");

exports.handler = async (event, context) => {

  console.log({event}, {context});

  const { httpMethod, queryStringParameters } = event;

  if (httpMethod !== "GET")
    return {
      statusCode: 405,
      body: JSON.stringify({
        status: "error",
        message: `${httpMethod} method not allowed. Use GET.`,
      }),
    };

  let { title, subtitle } = queryStringParameters;

  title = decodeURIComponent(title || `Hello World!`);
  subtitle = decodeURIComponent(subtitle || `serverless-gems.com`);

  let template = `
    <!doctype html>
    <html lang="en" style="width:1200px;height:630px">
    <head>
      <meta charset="UTF-8">
      <meta name="robots" content="noindex,nofollow">
      <meta name="viewport" content="width=device-width,initial-scale=1">
      <title>${title}</title>
      <style>
        h1{overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:3}
        .card{background:#111;background-image:linear-gradient(43deg,#4158d0 0,#c850c0 46%,#ffcc70 100%)}
        .shadow{text-shadow:0 0 10px rgba(0,0,0,.5)}
        .img-shadow{box-shadow:0 0 10px rgba(0,0,0,.5)}
      </style>
    </head>
    <body style="width:1200px;height:630px;padding:0;margin:0;font-family:sans-serif">
      <div class="card" style="width:1200px;height:630px;box-sizing:border-box;color:#fff;padding:70px 70px">
        <img class="img-shadow" style="border-radius:999px;border:4px solid #404040"
             width="120" height="120"
             src=""
             alt="Serverless Gems">
        <h1 class="shadow" style="font-size:72px;font-weight:700;margin:20px 0 10px 40px">${title}</h1>
        <p class="shadow" style="margin:20px 0 0 40px;font-size:24px;font-weight:700">${subtitle.toUpperCase()}</p>
      </div>
    </body>
    </html>
  `;

  try {
    const browser = await puppeteer.launch({
      args: chromium.args,
      defaultViewport: { height: 630, width: 1200 },
      executablePath:
        process.env.CHROME_EXECUTABLE_PATH ||
        (await chromium.executablePath(
          "/var/task/node_modules/@sparticuz/chromium/bin"
        )),
    });

    const page = await browser.newPage();

    await page.setContent(template);

    const buffer = await page.screenshot();

    return {
      statusCode: 200,
      headers: {
        "Content-Type": "image/png",
      },
      body: buffer.toString("base64"),
      isBase64Encoded: true,
    };
  } catch (error) {
    return {
      statusCode: 500,
      body: JSON.stringify({
        status: "error",
        message: error.message,
      }),
    };
  }
};

Netlify requires setup config like this:

[functions]
  node_bundler = "esbuild"
  external_node_modules = ["@sparticuz/chromium"]

Check the source code on GitHub.

Test how it works

Execution Duration: 0ms

Open image in new tab

Netlify limits

Netlify serverless function executing time limited by 10 seconds.

If you reach out 10 seconds limit, Netlify will close connection for your request.

Edit this page