Getting Started with Edge Functions
Learn how to create, test, and deploy your first Edge Function using the Supabase CLI.
This guide walks you through creating, testing locally, deploying, and invoking a Supabase Edge Function using the CLI. By the end, you'll have a working function running on Supabase's global edge network.
You can also create and deploy functions directly from the Supabase Dashboard. Read the Dashboard Quickstart guide for more information.
Supabase Edge Functions only supports creating functions in TypeScript with the Deno runtime. This is because Deno was designed with extensibility in mind and its Rust codebase offers a modern developer experience, memory safety, and other features ideal for running edge functions.
Prerequisites#
- Make sure you have the Supabase CLI installed and configured. Read the CLI installation guide for installation methods and troubleshooting.
- Running and testing Supabase Edge Functions locally requires Docker or a Docker-compatible runtime.
Step 1: Create or configure your project#
If you don't have a project yet, initialize a new Supabase project in your current directory.
1mkdir my-edge-functions-project2cd my-edge-functions-project3supabase initIf you already have a project locally, navigate to your project directory. If you haven't configured the project for Supabase yet, make sure to run the supabase init command.
1cd your-existing-project2supabase init # Initialize Supabase, if you haven't alreadyAfter this step, you should have a project directory with a supabase folder containing a config.toml file.
Step 2: Create your first function#
Within your project, generate a new Edge Function with a basic template:
1supabase functions new hello-worldSecure your function with Supabase Auth
When an HTTP request is sent to Edge Functions, you can use Supabase Auth to secure endpoints. By default, the supabase functions new command adds handling a valid publishable or secret key to the basic template. However, you can change this behavior with the --auth flag when creating a new function.
This creates a new function at supabase/functions/hello-world/index.ts with this starter code:
1export default {2 fetch: withSupabase({ allow: ['public', 'secret'] }, async (req, ctx) => {3 const { name } = await req.json()45 return Response.json({6 message: `Hello ${name}!`,7 })8 }),9}This function accepts a JSON payload with a name field and returns a greeting message.
The supabase functions new command also optionally creates Deno configuration for VSCode.
Step 3: Test your function locally#
After starting Docker, start the local development server to test your function:
1supabase start # Start all Supabase services2supabase functions serve hello-worldOn first use, the supabase start command downloads Docker images, and starts all Supabase services locally, which can take a few minutes.
Your function is now running at http://localhost:54321/functions/v1/hello-world. Hot reloading is enabled, which means that the server automatically reloads when you save changes to your function code. Keep this terminal window open.
Function not starting locally?#
- Make sure Docker is running
- Run
supabase stopthensupabase startto restart services
Port already in use?#
- Check what's running with
supabase status - Stop other Supabase instances with
supabase stop
Step 4: Send a test request#
Open a new terminal and test your function with curl. You can find your local Publishable key, by running supabase status, or you can find the complete curl command already in functions/hello-world/index.ts.
1curl -i --location --request POST 'http://127.0.0.1:54321/functions/v1/hello-world' \2 --header 'apiKey: <SUPABASE_PUBLISHABLE_KEY>' \3 --data '{"name":"Functions"}'After running this curl command, you should see:
1{ "message": "Hello Functions!" }You can also try different inputs. Change "Functions" to "World" in the curl command and run it again to see the response change.
After this step, you should have successfully tested your Edge Function locally and received a JSON response with your greeting message.
Step 5: Connect to your Supabase project#
To deploy your function globally, you need to connect your local project to a Supabase project.
Need to create a new Supabase project?
Create one at database.new.
First, login to the CLI if you haven't already, and authenticate with Supabase. This opens your browser to authenticate with Supabase; complete the login process in your browser.
1supabase loginNext, list your Supabase projects to find your project ID:
1supabase projects listNext, copy your project ID from the output, then connect your local project to your remote Supabase project. Replace YOUR_PROJECT_ID with the ID from the previous step.
1supabase link --project-ref [YOUR_PROJECT_ID]After this step, you should have your local project authenticated and linked to your remote Supabase project. You can verify this by running supabase status.
Step 6: Deploy to production#
Deploy your function to Supabase's global edge network:
1supabase functions deploy hello-worldIf you want to deploy all functions, run the deploy command without specifying a function name:
1supabase functions deployDocker not required
The CLI automatically falls back to API-based deployment if Docker isn't available. You can also explicitly use API deployment with the --use-api flag:
1supabase functions deploy hello-world --use-apiWhen the deployment is successful, your function is automatically distributed to edge locations worldwide.
Now, you should have your Edge Function deployed and running globally at https://[YOUR_PROJECT_ID].supabase.co/functions/v1/hello-world.
Step 7: Test your live function#
🎉 Your function is now live! Test it with your project's publishable key that you can find in the Settings > API Keys section of the Dashboard:
1curl --request POST 'https://[YOUR_PROJECT_ID].supabase.co/functions/v1/hello-world' \2 --header 'apikey: <SUPABASE_PUBLISHABLE_KEY>' \3 --header 'Content-Type: application/json' \4 --data '{"name":"Production"}'Expected response:
1{ "message": "Hello Production!" }Usage#
Now that your function is deployed, you can invoke it from within an app:
1import { createClient } from '@supabase/supabase-js'23const supabase = createClient('https://[YOUR_PROJECT_ID].supabase.co', 'YOUR_PUBLISHABLE_KEY')45const { data, error } = await supabase.functions.invoke('hello-world', {6 body: { name: 'JavaScript' },7})89console.log(data) // { message: "Hello JavaScript!" }