rembrembdocs

Supabase for Platforms

Use Supabase as a platform for your own business and tools.


Supabase is a Platform as a Service (PaaS) that can be managed programmatically. You can use it to offer the key primitives to your own users, such as Database, Auth, Edge Functions, Storage, and Realtime. Supabase is commonly used as a platform by AI Builders and frameworks needing a backend.

This document will guide you on best practices when using Supabase for your own platform and assumes that Supabase projects are in a Supabase organization that you own. If you want to instead interact with projects that your users own, navigate to OAuth integration for more details.

Platform as a Service

Overview#

All features of Supabase can be managed through the Management API or the remote MCP Server.

Launching projects#

Management API endpoints:

We recommend:

1curl https://api.supabase.com/v1/projects \2  --request POST \3  --header "Content-Type: application/json" \4  --header "Authorization: Bearer YOUR_SECRET_TOKEN" \5  --data '{6    "name": "Todo App",7    "organization_slug": "aaaabbbbccccddddeeee",8    "db_pass": "SUPER_SECURE_PASSWORD",9    "region_selection": {10      "type": "smartGroup",11      "code": "americas"12    },13    "desired_instance_size": "micro"14  }'

Pico compute instance#

Contact us for access to Pico instances

Only select customers have access to Pico instances. Submit this form to get access.

Pico instance is our newest compute instance and the only one that scales to zero when not used.

Recommended API keys#

Management API endpoints:

We are in the process of migrating away from our legacy API keys anon and service_role and towards API keys publishable and secret.

You can learn more by navigating to Upcoming changes to Supabase API Keys #29260.

Get the API keys by making a GET /v1/projects/{ref}/api-keys request.

1curl 'https://api.supabase.com/v1/projects/{ref}/api-keys?reveal=true' \2  --header 'Authorization: Bearer YOUR_SECRET_TOKEN'

If the response includes "publishable" and "secret" keys then you're all set and you should only use those from now on.

Otherwise, enable the API keys by making two POST /v1/projects/{ref}/api-keys requests, one for publishable and another for secret.

1curl 'https://api.supabase.com/v1/projects/{ref}/api-keys' \2  --request POST \3  --header 'Content-Type: application/json' \4  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \5  --data '{6  "type": "publishable",7  "name": "default"8}'
1curl 'https://api.supabase.com/v1/projects/{ref}/api-keys?reveal=true' \2  --request POST \3  --header 'Content-Type: application/json' \4  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \5  --data '{6  "type": "secret",7  "name": "default",8  "secret_jwt_template": {9    "role": "service_role"10  }11}'

Changing compute sizes#

Management API endpoint: PATCH /v1/projects/{ref}/billing/addons

Once you have access to Pico instances, you will no longer be able to upgrade or downgrade to Nano instances.

You can upgrade and downgrade compute sizes by making requests to PATCH /v1/projects/{ref}/billing/addons.

1curl 'https://api.supabase.com/v1/projects/{ref}/billing/addons' \2  --request PATCH \3  --header 'Content-Type: application/json' \4  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \5  --data '{6    "addon_type": "compute_instance",7    "addon_variant": "ci_small"8  }'

Configuration changes#

Management API endpoints:

You can manage the configuration of all services using the Management API.

Development workflow#

Supabase is a stateful service: we store data. If anything breaks in production, you can't "roll back" to a point in time because doing so might cause your users to lose any data that their production environment received since the last checkpoint.

Because of this, it's important that you adopt a development workflow on behalf of your users:

Change flow

Creating a DEV branch#

Management API endpoint: POST /v1/projects/{ref}/branches

After launching a project, it's important that all changes happen on a development branch. Branches can be treated like ephemeral servers: if anything goes wrong you can either revert the changes or destroy the branch and create a new one based off of production.

1curl 'https://api.supabase.com/v1/projects/{ref}/branches' \2  --request POST \3  --header 'Content-Type: application/json' \4  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \5  --data '{6    "branch_name": "DEV",7    "secrets": {8      "STRIPE_SECRET_KEY":"sk_test_123..."9      "STRIPE_PUBLISHABLE_KEY":"pk_test_123..."10    }11  }'

Make database changes#

Management API endpoint: POST /v1/projects/{ref}/database/migrations

Contact us for access to migrations endpoint

Only select customers have access to database migrations endpoint. Submit this form to get access.

For this example we will create a todos table using the POST /v1/projects/{ref}/database/migrations endpoint.

1create table public.todos (2  id serial primary key,3  task text not null4);56alter table public.todos7enable row level security;

This endpoint will automatically create a migration inside the supabase_migrations schema and run the migration. If the schema migration fails, the changes will be rolled back.

1curl https://api.supabase.com/v1/projects/{ref}/database/migrations \2  --request POST \3  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \4  --header 'Content-Type: application/json' \5  --data '{6    "query": "create table public.todos (id serial primary key, task text not null); alter table public.todos enable row level security;",7    "name": "Create a todos table"8  }'

Create a restore point#

Contact us for access to restore points

Only select customers have access to restore points. Submit this form to get access.

After every change you make to the database, it's a good idea to create a restore point. This will allow you to roll back the database if you decide to go in a different direction.

Beware that only database changes are captured when creating a restore point.

1curl https://api.supabase.com/v1/projects/{ref}/database/backups/restore-point \2  --request POST \3  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \4  --header 'Content-Type: application/json' \5  --data '{6    "name": "abcdefg"7  }'

Reverting changes#

Revert

Contact us for access to restore points

Only select customers have access to restore points. Submit this form to get access.

After creating restore points, you are able to revert back to any restore point that you want.

1curl https://api.supabase.com/v1/projects/{ref}/database/backups/undo \2  --request POST \3  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \4  --header 'Content-Type: application/json' \5  --data '{6  "name": "abcdefg"7}'

When you revert changes, you are undo-ing all database changes since the specified restore point, including:

It will not affect:

Add seed data#

Do not use production data in branches!

It's important that the data in development branches is NOT production data, especially for non-developers who don't understand the implications of working with data. Security and side-effects (e.g. emailing all production users) are two reasons why this is important.

It's common in DEV branches to "seed" data. This is basically test data for users. Let's insert the following seed to our new todos table:

1insert into todos (task)2values3  ('Task 1'),4  ('Task 2'),5  ('Task 3');

You can use the POST /database/query endpoint to add data:

1curl https://api.supabase.com/v1/projects/{branch_ref}/database/query \2  --request POST \3  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \4  --header 'Content-Type: application/json' \5  --data-binary @- <<EOF6{7  "query": "insert into todos (task) values ('Task 1'), ('Task 2'), ('Task 3');"8}9EOF

Deploying Edge Functions#

You can create and deploy Edge Functions on a branch and then merge them into Production.

Deploy Edge Functions

1curl https://api.supabase.com/v1/projects/functions/deploy \2  --request POST \3  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \4  --header 'Content-Type: multipart/form-data' \5  --form 'file=@path/to/function.zip' \6  --form 'metadata={7    "name": "my-function",8    "entrypoint_path": "index.ts",9    "import_map_path": "import_map.json",10    "static_patterns": ["assets/*", "public/*"],11    "verify_jwt": true12  }'

Merge all changes#

Management API endpoint: POST /v1/branches/{branch_id_or_ref}/merge

Once the changes have been made, you can merge the changes into the production project:

1curl https://api.supabase.com/v1/branches/{ref}/merge \2  --request POST \3  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \4  --header 'Content-Type: application/json' \5  --data '{}'

Merging changes will automatically merge only the following:

Security checks for production#

Management API endpoint: GET /v1/projects/{ref}/advisors/security

Prior to deploying to production we strongly recommend running the security advisor on the project to make sure the latest changes are secure.

1curl https://api.supabase.com/v1/projects/{ref}/advisors/security \2  --request GET \3  --header  'Authorization: Bearer {PAT_OR_USER_TOKEN}' \4  --header 'Content-Type: application/json'

Disaster recovery for production#

Management API endpoint: POST /v1/projects/{ref}/database/backups/restore-pitr

Rollback

1curl https://api.supabase.com/v1/projects/database/backups/restore-pitr \2  --request POST \3  --header 'Authorization: Bearer {PAT_OR_USER_TOKEN}' \4  --header 'Content-Type: application/json' \5  --data '{6  "recovery_time_target_unix": 17}'

Claim flow#

Management API endpoint: GET /v1/oauth/authorize/project-claim

Contact us for access to claim flow

Only select customers have access to claim flow. Submit this form to get access.

Your users may want to claim the project that currently lives in your org so that they can have more control over it.

We've enabled transferring the project from your org to your user's org while you continue to retain access to interact with the project through an OAuth integration.

1curl -L "https://api.supabase.com/v1/oauth/authorize/project-claim?project_ref={ref}&client_id={oauth_client_id}&response_type=code&redirect_uri={redirect_uri}" \2  --request GET \3  --header 'Authorization: Bearer {PERSONAL_ACCESS_TOKEN}'

The user is redirected to a Supabase UI:

  1. Create a new Supabase account or sign in to an existing one
  2. Create a new Supabase Organization or select an existing one
  3. Review your OAuth integration's permission scopes
  4. Review the Project transfer details
  5. Confirm the OAuth integration and the Project transfer for the selected Organization
Remove custom configuration before transferring project

Before transferring the project to your user's org, make sure to remove any custom configuration that you do not want your user's project to retain.

Platform kit#

Docs: https://supabase.com/ui/docs/platform/platform-kit

We've created Platform Kit, a collection of UI components that interact with Management API, as a lightweight version of Supabase Dashboard that you can embed directly in your app so your users never have to leave in order to interact with the project.

Debugging projects#

Management API endpoint: GET /v1/projects/{ref}/analytics/endpoints/logs.all

When you need to debug a project, you can query the project's logs to see if there are any errors and address them accordingly.

1curl 'https://api.supabase.com/v1/projects/{ref}/analytics/endpoints/logs.all' \2  --get \3  --header 'Authorization: Bearer YOUR_SECRET_TOKEN' \4  --data-urlencode 'sql=SELECT datetime(timestamp), status_code, path, event_message5    FROM edge_logs6    CROSS JOIN UNNEST(metadata) AS metadata7    CROSS JOIN UNNEST(response) AS response8    WHERE status_code >= 4009    ORDER BY timestamp DESC10    LIMIT 100' \11  --data-urlencode 'iso_timestamp_start=2025-03-23T00:00:00Z' \12  --data-urlencode 'iso_timestamp_end=2025-03-23T01:00:00Z'