apps/www/_blog/2024-01-31-nosql-mongodb-compatibility-with-ferretdb-and-flydotio.mdx
FerretDB is an open source document database that adds MongoDB compatibility to other database backends, such as Postgres and SQLite. By using FerretDB, developers can access familiar MongoDB features and tools using the same syntax and commands for many of their use cases.
In this post, we'll start from scratch, running FerretDB locally via Docker, trying out the connection with mongosh and the MongoDB Node.js client, and finally deploy FerretDB to Fly.io for a production ready set up.
If you prefer video guide, you can follow along below. And make sure to subscribe to the Supabase YouTube channel!
<div className="video-container"> <iframe className="w-full" src="https://www.youtube-nocookie.com/embed/8EKi2dV51TY" title="YouTube video player" allow="accelerometer; autoplay; clipboard-write; encrypted-media; fullscreen; gyroscope; picture-in-picture; web-share" allowfullscreen /> </div>FerretDB provides a Docker image allowing us to run it locally, for example via Orbstack, with a couple simple commands.
FerretDB only requires the Postgres database URI to be provided as the FERRETDB_POSTGRESQL_URL environment variable. Every Supabase project comes with a full Postgres database. You can find the connection URI string in your Supabase Dashboard.
Make sure Use connection pooling is checked and Session mode is selected. Then copy the URI. Replace the password placeholder with your saved database password.
<Admonition type="tip">If your network supports IPv6 connections, you can also use the direct connection string. Uncheck Use connection pooling and copy the new URI.
</Admonition># Set the required environment variables
export DB_USER=postgres
export DB_PASSWORD=<your db password>
export SUPA_PROJECT_REF=<your Supabase project ref>
export SUPA_REGION=<your project region>
export DB_URL=postgres://$DB_USER.$SUPA_PROJECT_REF:$DB_PASSWORD@$SUPA_REGION.pooler.supabase.com:5432/postgres
# Run FerretDB in docker container
docker run -p 27017:27017 -p 8080:8080 -e FERRETDB_POSTGRESQL_URL=$DB_URL ghcr.io/ferretdb/ferretdb
FerretDB runs on the default MongoDB port 27017 and also spins up some monitoring tools on port 8080. Once up and running you can access these at localhost:8080.
Once up and running, constructing the MongoDB URI is easil:
export MONGODB_URL=mongodb://$DB_USER.$SUPA_PROJECT_REF:$DB_PASSWORD@127.0.0.1:27017/ferretdb?authMechanism=PLAIN
mongoshIf you have MongoDB installed locally on your machine, you can test via mongosh, the MongoDB shell.
mongosh '$MONGODB_URL'
If you don't have MongoDB installed locally, you can run the shell via a Docker container:
docker run --rm -it --entrypoint=mongosh mongo \
"$MONGODB_URL"
With mongosh running, let's try to insert some documents into our FerretDB instance.
You are going to insert two footballer data into a players collection.
db.players.insertMany([
{
futbin_id: 3,
player_name: "Giggs",
player_extended_name: "Ryan Giggs",
quality: "Gold - Rare",
overall: 92,
nationality: "Wales",
position: "LM",
pace: 90,
dribbling: 91,
shooting: 80,
passing: 90,
defending: 44,
physicality: 67
},
{
futbin_id: 4,
player_name: "Scholes",
player_extended_name: "Paul Scholes",
quality: "Gold - Rare",
overall: 91,
nationality: "England",
position: "CM",
pace: 72,
dribbling: 80,
shooting: 87,
passing: 91,
defending: 64,
physicality: 82,
base_id: 246
}
]);
Great!
Now when you run db.players.find(), it should return all the documents stored in the collection.
Next, you need to update "Giggs" record to reflect his current position as a CM.
To do this, we can just run an updateOne command to target just that particular player:
db.players.updateOne(
{ player_name: "Giggs" },
{ $set: { position: "CM" } }
);
Let's query the collection to see if the changes have been made:
db.players.find({player_name: "Giggs"})
You can run many MongoDB operations on FerretDB. See the list of supported commands in the FerretDB documentation for more.
FerretDB stores each collection in a table on the ferretdb schema, each document represented by a JSONB entry. You can inspect this in the Table Editor in your Supabase Dashboard.
For production use cases, you can easily deploy FerretDB on Fly. Simply create a fly.toml file (make sure to update primary_region)
app = "supa-ferretdb-<your-supabase-project-ref>"
primary_region = "bos"
[build]
image = "ghcr.io/ferretdb/ferretdb"
[[services]]
internal_port = 27017
protocol = "tcp"
[[services.ports]]
port = "27017"
[[vm]]
cpu_kind = "shared"
cpus = 1
memory_mb = 1024
And follow these flyctl commands:
Now simply replace 127.0.0.1 in the MONGODB_URL with your dedicated IPv4 address and you're ready to roll!
FerretDB allows you to run MongoDB workloads on Postgres and SQLite. This flexibility means you can easily add MongoDB compatibility to your Supabase projects, while avoiding vendor lock-in and retaining control of your data architecture.
To get started with FerretDB, check out the FerretDB quickstart guide.