Multi-Tenant Architecture

Hello good folks. The purpose of this question is to get insight from the more experienced community. I'm trying to build a B2B system with data isolation concerns. For now, I chose to go with a schema-per-tenant data partition strategy. All tenants will share a common set of tables attached to their schema. But, the problem with this approach is the schema managment. To change even a single column across all schemas when you have many tenants is a challenge. I've tried an approach where I use drizzleORM to generate the "template" migration scripts. Then when a request comes in, I read the sql content from the file and replace the placeholder with the tenant's schema name. This works fine for creating new tenants. But the real problem now is global updates. I've not found a reliable way to do this so far. It's a greenfield project, so I'm trying to avoid tech debt to the best of my ability. How would you go about building such a system?
31 Replies
Neto
Neto•2mo ago
you can check how atlas does it
Deezy
DeezyOP•2mo ago
Thanks @Neto for this. I am kind of glad you dropped a link to the atlas docs because in fact it's atlas I have been experimenting with all this while for this purpose
Neto
Neto•2mo ago
as long as you aren't deeep deeep deeep into messing with atlas should work just fine
Deezy
DeezyOP•2mo ago
In fact, deep sums it up 😅 I have actually tried it
Neto
Neto•2mo ago
deep as in, using it as a plain lib
Deezy
DeezyOP•2mo ago
It hasn't been great to say the least
Neto
Neto•2mo ago
docs are rough....
Deezy
DeezyOP•2mo ago
I'm sorry I'm not sure I caught this
Neto
Neto•2mo ago
atlas is both a cli tool and a library the cli part is quite good but the library part is awful you most likely want to build something like this
Neto
Neto•2mo ago
GitHub
examples/app-turso-crm/app/lib/utils.ts at master · tursodatabase/e...
A compilation of quickstart & sample projects involving Turso and popular frameworks - tursodatabase/examples
Neto
Neto•2mo ago
ignore the drizzle part
Deezy
DeezyOP•2mo ago
Yes I reckon. That's what I've been using. I tried the library and it just doesn't work Checking it out now!
Neto
Neto•2mo ago
in the end i had two options
env "dev" {
url = "..."
dev = "docker://postgres/17/dev"
}

data "sql" "tenants" {
url = "..."
query = <<EOS
SELECT ref FROM companies
EOS
args = []
}

env "node" {
for_each = toset(data.sql.tenants.values)
url = "sqlite://tenants/${each.value}.db"
dev = "sqlite://file?mode=memory"
}

variable "ref" {
type = string
description = "Database to be created"
default = "default value"
}

env "newnode" {
url = "sqlite://tenants/${var.ref}.db"
dev = "sqlite://file?mode=memory"
}
env "dev" {
url = "..."
dev = "docker://postgres/17/dev"
}

data "sql" "tenants" {
url = "..."
query = <<EOS
SELECT ref FROM companies
EOS
args = []
}

env "node" {
for_each = toset(data.sql.tenants.values)
url = "sqlite://tenants/${each.value}.db"
dev = "sqlite://file?mode=memory"
}

variable "ref" {
type = string
description = "Database to be created"
default = "default value"
}

env "newnode" {
url = "sqlite://tenants/${var.ref}.db"
dev = "sqlite://file?mode=memory"
}
node is for general migration newnode is to create a new tenant and calling via bash cmd := exec.Command("sh", "-c", "atlas schema apply --env newnode --file ./db/node-schema.sql --auto-approve --var ref="+_uuid.String())
Deezy
DeezyOP•2mo ago
Okay so you then run this with the CLI?
Neto
Neto•2mo ago
this config is via the hcl file a solution i found was calling the cli via os commands probably not the best, but it's an option
Neto
Neto•2mo ago
GitHub
GitHub - golang-migrate/migrate: Database migrations. CLI and Golan...
Database migrations. CLI and Golang library. Contribute to golang-migrate/migrate development by creating an account on GitHub.
Neto
Neto•2mo ago
multi-tenant is a mess and people avoid for this exact reason its a shit show
Deezy
DeezyOP•2mo ago
My word! I cannot tell you how comforting that sounds to me 😅 I am kind of behind time with the schedule of the project which is why I'm just all out looking for any solution now. I've been trying to get it working with atlas but it's been like I've been fighting the library all this while
Neto
Neto•2mo ago
drizzle is quite laid back in terms of multi-tenancy
Deezy
DeezyOP•2mo ago
I totally agree. It's been feeling as if I'm trying to hack the capability into it the past few days. I also looked into TypeORM
Neto
Neto•2mo ago
avoid typeorm for any reason
Deezy
DeezyOP•2mo ago
But I don't know if I like TypeORM Gotcha 😭 From what I saw from the docs, you don't get typesafe queries like drizzle. For all the decorators it will add to the codebase, I don't know man
Neto
Neto•2mo ago
/migrations
/master
/[...]
/[...]
/[...]
/tenants
/[...]
/migrations
/master
/[...]
/[...]
/[...]
/tenants
/[...]
when creating a new tenant, run drizzle migrate on migrations/tenants for the schema create the script for migrating all tenants you will need like 2? drizzle config files one for the master and one for the tenants basically "two" drizzle projects in the same repo
Deezy
DeezyOP•2mo ago
Got it! master would be for the global tables yeah?
Neto
Neto•2mo ago
yeah
Deezy
DeezyOP•2mo ago
Thank you! I actually have a similar setup as it stands. But the script for migrating the tenants, I'd have to opt-out for drizzle for that one So it seems Thanks for the insight nyx! Do you mind if I drop a text should I have questions on the topic?
Neto
Neto•2mo ago
sure if i can help
Deezy
DeezyOP•2mo ago
Copy that! Thank you! I'll try to make it work!
rotemtam
rotemtam•3w ago
hey all atlas maintainer here (cto for ariga.io, company behind it) a friend tagged me here, happy to help if you need anything
Neto
Neto•3w ago
hello, i really like atlas overall, just wish it had better docs for using it as a library and not as a cli other than that, it's super solid for most cases

Did you find this page helpful?