Разработка Edge Functions для сайта (Vercel Edge)
Vercel Edge Functions выполняются на Vercel Edge Network — на 100+ узлах по всему миру, в миллисекундах от пользователя. Холодный старт менее 1 мс. Это отличает их от обычных Vercel Functions, которые работают в одном регионе AWS.
Когда Edge, когда обычные Functions
Edge Functions оптимальны для: персонализации (A/B тест, геолокация), редиректов и rewrite на основе условий, middleware-аутентификации, трансформации заголовков и ответов, кэш-валидации по cookie.
Обычные Node.js Functions нужны когда: требуется Node.js API (fs, crypto, нативные модули), соединение с PostgreSQL через TCP, время выполнения > 30 с, объём памяти > 128 MB.
Edge Runtime использует Web API (как в браузере), а не Node.js API.
Middleware для A/B тестирования
Файл middleware.ts в корне проекта Next.js:
import { NextRequest, NextResponse } from "next/server";
export function middleware(request: NextRequest) {
const url = request.nextUrl.clone();
// A/B тест для главной страницы
if (url.pathname === "/") {
const bucket = request.cookies.get("ab-bucket")?.value;
if (!bucket) {
const newBucket = Math.random() < 0.5 ? "a" : "b";
const response = NextResponse.rewrite(
new URL(newBucket === "b" ? "/home-variant" : "/", request.url)
);
response.cookies.set("ab-bucket", newBucket, { maxAge: 86400 * 30 });
return response;
}
if (bucket === "b") {
url.pathname = "/home-variant";
return NextResponse.rewrite(url);
}
}
return NextResponse.next();
}
export const config = {
matcher: ["/", "/pricing", "/features"],
};
Геолокация и персонализация
import { NextRequest, NextResponse } from "next/server";
import { geolocation } from "@vercel/functions";
export function middleware(request: NextRequest) {
const { country, city } = geolocation(request);
// Редирект на локализованную версию
if (country === "RU" && !request.nextUrl.pathname.startsWith("/ru")) {
return NextResponse.redirect(
new URL(`/ru${request.nextUrl.pathname}`, request.url)
);
}
// Добавляем геоданные в заголовки для компонентов
const response = NextResponse.next();
response.headers.set("x-user-country", country || "unknown");
response.headers.set("x-user-city", city || "unknown");
return response;
}
Edge API Route
// app/api/edge-data/route.ts
import { NextRequest, NextResponse } from "next/server";
export const runtime = "edge";
export async function GET(request: NextRequest) {
const { searchParams } = new URL(request.url);
const id = searchParams.get("id");
// Fetch работает нативно в Edge Runtime
const data = await fetch(`https://api.external.com/data/${id}`, {
headers: { Authorization: `Bearer ${process.env.API_KEY}` },
// next.js cache: кэш на 60 секунд
next: { revalidate: 60 },
}).then(r => r.json());
return NextResponse.json(data, {
headers: { "Cache-Control": "s-maxage=60, stale-while-revalidate=120" }
});
}
Защита маршрутов через JWT
import { NextRequest, NextResponse } from "next/server";
import { jwtVerify } from "jose";
const JWT_SECRET = new TextEncoder().encode(process.env.JWT_SECRET);
export async function middleware(request: NextRequest) {
if (request.nextUrl.pathname.startsWith("/dashboard")) {
const token = request.cookies.get("auth-token")?.value;
if (!token) {
return NextResponse.redirect(new URL("/login", request.url));
}
try {
await jwtVerify(token, JWT_SECRET);
return NextResponse.next();
} catch {
const response = NextResponse.redirect(new URL("/login", request.url));
response.cookies.delete("auth-token");
return response;
}
}
}
jose — единственная JWT-библиотека, совместимая с Edge Runtime (использует Web Crypto API вместо node:crypto).
Ограничения Edge Runtime
- Нет
node:fs,node:path,node:crypto(есть Web Crypto) - Нет нативных npm-пакетов
- Память: 128 MB
- CPU time: 30 мс (Hobby), без ограничений на Pro
- Нет прямых соединений к PostgreSQL (Neon и PlanetScale поддерживают HTTP API)
Сроки
Middleware с геолокацией и A/B тестом — 1–2 дня. Edge API Routes с кэшированием и JWT-защитой — 2–3 дня.







