apps/docs/content/guides/functions/quickstart.mdx
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.
<Admonition type="tip">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.
</Admonition>If you don't have a project yet, initialize a new Supabase project in your current directory.
mkdir my-edge-functions-project
cd my-edge-functions-project
supabase init
If 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.
cd your-existing-project
supabase init # Initialize Supabase, if you haven't already
After this step, you should have a project directory with a supabase folder containing a config.toml file.
Within your project, generate a new Edge Function with a basic template:
supabase functions new hello-world
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:
export default {
fetch: withSupabase({ auth: ['publishable', 'secret'] }, async (req, ctx) => {
const { name } = await req.json()
return Response.json({
message: `Hello ${name}!`,
})
}),
}
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.
After starting Docker, start the local development server to test your function:
supabase start # Start all Supabase services
supabase functions serve hello-world
On 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.
supabase stop then supabase start to restart servicessupabase statussupabase stopOpen 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.
curl -i --location --request POST 'http://127.0.0.1:54321/functions/v1/hello-world' \
--header 'apiKey: <SUPABASE_PUBLISHABLE_KEY>' \
--data '{"name":"Functions"}'
After running this curl command, you should see:
{ "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.
</Admonition>To deploy your function globally, you need to connect your local project to a Supabase project.
<Admonition type="tip" label="Need to create a new Supabase project?">Create one at database.new.
</Admonition>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.
supabase login
Next, list your Supabase projects to find your project ID:
supabase projects list
Next, 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.
supabase 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.
Deploy your function to Supabase's global edge network:
supabase functions deploy hello-world
If you want to deploy all functions, run the deploy command without specifying a function name:
supabase functions deploy
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:
supabase functions deploy hello-world --use-api
When the deployment is successful, your function is automatically distributed to edge locations worldwide.
<Admonition type="note">Now, you should have your Edge Function deployed and running globally at https://[YOUR_PROJECT_ID].supabase.co/functions/v1/hello-world.
š 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:
curl --request POST 'https://[YOUR_PROJECT_ID].supabase.co/functions/v1/hello-world' \
--header 'apikey: <SUPABASE_PUBLISHABLE_KEY>' \
--header 'Content-Type: application/json' \
--data '{"name":"Production"}'
Expected response:
{ "message": "Hello Production!" }
Now that your function is deployed, you can invoke it from within an app:
<Tabs scrollable size="small" type="underlined" defaultActiveId="supabase-js"
<TabPanel id="supabase-js" label="Supabase Client">
import { createClient } from '@supabase/supabase-js'
const supabase = createClient('https://[YOUR_PROJECT_ID].supabase.co', 'YOUR_PUBLISHABLE_KEY')
const { data, error } = await supabase.functions.invoke('hello-world', {
body: { name: 'JavaScript' },
})
console.log(data) // { message: "Hello JavaScript!" }
const response = await fetch('https://[YOUR_PROJECT_ID].supabase.co/functions/v1/hello-world', {
method: 'POST',
headers: {
apikey: '<SUPABASE_PUBLISHABLE_KEY>',
'Content-Type': 'application/json',
},
body: JSON.stringify({ name: 'Fetch' }),
})
const data = await response.json()
console.log(data)