Headers up: the Vercel edition

After a wait of nearly eighteen months, I finally have a Vercel Edge Function running on my non‑Next.js project.


Vercel announced in October, 2021, that it would be adding Edge Functions to its stable of products and features. However, at first, Vercel Edge Functions would work only on Vercel’s Next.js framework, with non-Next.js support — including code examples to help developers — slated for sometime in 2022.

Well, it’s no longer 2022 but, at least for me, the wait has finally ended. So, if you, too, have been waiting impatiently to make use of Vercel Edge Functions on your non-Next.js project, this post may lift your spirits.


Edge functions aren’t just on Vercel but, rather, exist on multiple platforms under different names. For example, Cloudflare Workers were introduced in 2017.

Like web content, a platform’s edge functions exist on the “edges” — i.e., the worldwide points of presence (PoPs) — of that platform’s global content delivery network (CDN). This imposes far less latency than if the functions were on only one server that might be halfway around the world from a user. But, unlike content, edge functions also run code on the CDN’s PoPs, providing a new run with each load or refresh of a web page. This can enable more dynamic capabilities for otherwise completely static websites.

Even after Vercel made it possible for non-Next.js projects to use Edge Functions, I was unsatisfied with the lack of relevant code examples. And I wasn’t alone. Others had begun asking for such examples as early as a month after Vercel’s original announcement of Edge Functions’ availability.

By mid-2022, a small number of non-Next.js examples had trickled into Vercel’s GitHub repository. However, I had no luck with adapting them to anything remotely similar to the Cloudflare Worker I’d begun using on my site months before the Vercel announcement, as explained in the first “Headers up” post. While that Worker’s original purpose had been only to control HTTP headers for caching purposes, it had evolved over time so that it could also enable me to inject nonces for an air-tight Content Security Policy (CSP).

In multiple sessions over a period of months, I struggled in vain to duplicate the Worker’s functionality in Vercel Edge Middleware. What drove me especially crazy was that, despite knowing this was simply “white-labeled” Cloudflare Workers technology, I couldn’t manage to use my Worker’s code without crashing the resulting Edge Function, thus taking down whatever project I was attempting to run on Vercel.1

I knew, of course, that the code couldn’t be exactly the same. For example, while my Worker began with:

addEventListener('fetch', event => {

Functions to generate the nonce
on each page load

async function handleRequest(request) {

. . . the middleware.js2 file for Vercel would have to begin with:

Functions to generate the nonce
on each page load

export default async function handleRequest(request) {

Try, try, try, try again

Then, a few nights ago, I decided to give the whole thing yet another go. If I chose at some point to move this site to Vercel, I’d want to do so without weakening (or abandoning) the site’s CSP, and the only way to that end would be via an Edge Function.

I began cautiously, adapting one of the non-Next.js examples to inject simple headers such as:

x-test-from-middleware: true

. . . and, to my surprise and delight, everything worked. The Edge Function didn’t crash and did show each header I specified. Even this simple result was more than I’d been able to accomplish in my earlier tests.

Gradually, I added some innocuous pieces of my original Worker code, such as for caching-related headers. It all still worked.

Then, emboldened by my apparent success after so many past failures, I tried dumping nearly the entire Worker code, including the nonces stuff, into middleware.js — and the Edge Function crashed, taking down my project with it. (Obviously, the real site was still on Cloudflare Pages during these tests; I wasn’t that reckless, after all.)

I reverted middleware.js to the point where its Edge Function had worked, and began adding Worker code one or two lines at a time, trying to figure out what was causing the crash. Finally, late in the night, I discovered the culprit was this single line from the Worker code:

  let cache = caches.default

. . . which understandably flummoxed Vercel, since caches.default refers to a Cloudflare Workers API. And, anyway, I saw this line didn’t even serve a purpose in my Worker, much less in middleware.js! Thus, I simply commented-out the problematic line and, whammo, my Vercel project had an Edge Function that performed the same tasks as had the Worker for my real site.

Note: To be safe, I later wrapped most of the code inside a trycatch statement so, if any future edit included a typo or otherwise introduced a glitch, the Edge Function wouldn’t crash.

Subsequent tests confirmed that the site, if moved to Vercel, would retain all the functionality I’d had on Cloudflare Pages, thanks primarily3 to my Edge Function.

So, if you’ve considered using Vercel Edge Functions with your non-Next.js project, go ahead and give it your best shot. It may require some hassles before you can make it work, but take heart: it can be done.

Note from the future: Still more tests later indicated that otherwise identical sites load a bit more quickly, file-by-file, on Cloudflare Pages than on Vercel. One could guess this is because, in the case of the Vercel version, the middleware.js file is (as noted earlier) going “outside” Vercel to use Cloudflare Workers — vs. what, for obvious reasons, is likely a more seamless integration between Cloudflare Pages and Cloudflare Workers. Staying in-platform can pay off, all other things being equal.

  1. This is at least one way that the implementation of Cloudflare Workers is better than that of Vercel Edge Functions since, if my Worker crashes for some reason, it usually won’t clobber my project but, rather, will only cease providing add-ons. ↩︎

  2. In the early days, Vercel required the file’s name to be _middleware.js/.ts, but that’s not indicated in the current documentation↩︎

  3. Over and above what the Edge Function allowed, I also wanted to duplicate on Vercel the email address obfuscation Cloudflare had provided in every instance on the site where I gave my address — namely, on the contact page and in the “Reply via email” button at the bottom of each post. After finding a related discussion on the Hugo Discourse forum and reviewing a reader’s emailed comment on the same topic, I implemented the suggested solution. ↩︎