ben szabo

Getting the Vercel Preview URL in Github Actions

Using the Github API.

Did you know that you can get the unique Vercel Preview URL via the Github API for every commit? Me neither, but the other day I was forced to figure it out, and bring our rate limited Actions back to life. Read on to find out how!

Quick and dirty

At Lottie, we have some chunky end-to-end tests, that we only run once a PR has been approved. The broad setup is something like this:

  1. PR Approved
  2. Github Actions triggered
  3. Build and deploy with the Vercel CLI (which will give us the url once the deployment is complete)
  4. Invoke the E2E with the deployment_url from the previous step

The above has been working alright for us until a few days ago, when we started getting failing jobs, due to rate limiting on the Vercel API.

Is there a better way?

I jumped on a call with my colleague, Dario, to figure out what to do. Our goal was to take advantage of the preview deployment done by the Vercel App.

Although this sounded easy, it required a bit of digging, mainly reading through Github and Vercel documentations and API specs, then quickly validating our ideas just by making HTTP requests on the command line against the Github API.

This is what we came up with.

  1. Get the deployment(s) relating to the current commit SHA
  2. Use the deployment id, to get the status
  3. Grab the url, set it as output
Gotcha: listDeployments returns empty, until the Vercel deployment is complete.
flowchart

About 40 commits – half of which were typos – later, I had a working example.

How I love working with Github Actions 😀.

Note to self: use nektos/act even if you think it will be just a few commits.

Commit statuses

Right, so we figured out how to get the correct URL in our action, all that was left to do is making sure, that we can still block PRs without an end-to-end test on the latest commit.

Since we are no longer using a pull request event to trigger our action, we needed to manually connect the workflow run to the status check that was a blocker on the PRs.

To do this, we reached for our trusted Github API again, writing things with JS thanks to the github-script action.

This is the helper:

async function createCommitStatus(github, context, options) {
if (!github || !context || !options || !options.status || !options.statusCheckName || !options.description) return;
const { status, description, statusCheckName } = options;
await github.rest.repos.createCommitStatus({
  owner: context.repo.owner,
  repo: context.repo.repo,
  sha: context.sha,
  state: status,
  target_url: `https://github.com/${context.repo.owner}/${context.repo.repo}/actions/runs/${context.runId}`,
  description,
  context: statusCheckName,
});
}
 
module.exports = {
createCommitStatus: async (github, context, options) => createCommitStatus(github, context, options),
};
 

and how to use it:

set_pending_status:
  runs-on: ubuntu-latest
  steps:
    - uses: actions/checkout@v4
    - uses: actions/github-script@v7
      with:
        retries: 3
        script: |
          const { createCommitStatus } = require('./.github/scripts/e2eHelpers.js');
          const options = {
            status: 'pending',
            description: '⌛ Waiting for E2E tests to run',
            statusCheckName: 'pr_e2e_outcome',
          };
          await createCommitStatus(github, context, options)

And the slightly abstracted version of the final action, retaining the interesting bits only, like

name: E2E test
run-name: 'E2E test for: ${{ github.ref_name }}'
 
on:
workflow_dispatch:
 
jobs:
set_pending_status:
 
get_vercel_url:
  needs: set_pending_status
  timeout-minutes: 10
  runs-on: ubuntu-latest
  outputs:
    deployment_url: ${{ steps.get_url.outputs.result }}
  steps:
    - uses: actions/checkout@v4
    - name: get_url
      id: get_url
      uses: actions/github-script@v7
      with:
        result-encoding: string
        script: |
          const getVercelUrl = require('./.github/scripts/fetchVercelNonprodUrl.js');
 
          const vercelUrl = await getVercelUrl(github, context.sha, context.repo.repo, context.repo.owner)
          return vercelUrl
 
run_e2e_tests_against_pr:
 
set_success_status:
 
set_failed_status: