SQL query builder and ORM/Factory generator for Go with support for PostgreSQL, MySQL and SQLite
Bob is a set of Go packages and tools to work with SQL databases.
Bob’s philosophy centres around the following:
Bob can be progressively adopted from raw SQL query strings, to fully typed queries with models and factories generated for your database.
Bob consists of several components that build on each other for the full experience.
sqlc
.Check out the documentation for more information.
Queries | Models | ORM Gen | Factory Gen | Query Gen | |
---|---|---|---|---|---|
Postgres | ✅ | ✅ | ✅ | ✅ | ✅ |
MySQL/MariaDB | ✅ | ✅ | ✅ | ✅ | |
SQLite | ✅ | ✅ | ✅ | ✅ | ✅ |
This is just a fluent query builder that has no concept of your DB, and by extension cannot offer any type-safety.
The main reason, I consider it better than most alternatives is that since each dialect is hand-crafted, it can support building ANY query for that dialect.
However, each dialect is also independent, so you don’t have to worry about creating an invalid query.
IMPORTANT: Queries are built using “Query Mods”
psql.Select(
sm.From("users"), // This is a query mod
sm.Where(psql.Quote("age").GTE(psql.Arg(21))), // This is also a mod
)
This is where the type safety comes.
A full ORM, and query mods that is based on the database schema. If you use the generated query mods, these will ensure correct type safety.
Here is the above query using generated query-mods.
models.Users.Query(
models.SelectWhere.Users.Age.GTE(21), // This is type-safe
)
Factories make testing much much easier. Especially when the test depends on a database entry that depends on relations in other tables (e.g. testing comments that rely on posts which in turn rely on users).
With knowledge of the database schema, Bob can generate factories for each table.
// Quickly create a 10 comments (posts and users are created appropriately)
comments, err := f.NewComment().CreateMany(ctx, db, 10)
I believe this is the final peice of the puzzle, and extends the type-safety to hand-crafted SQL queries.
For example, you could generate code for the query:
-- UserPosts
SELECT * FROM posts WHERE user_id = $1
This will generate a function UserPosts
that takes an int32
.
// UserPosts
userPosts, err := queries.UserPosts(1).All(ctx, db)
Then, if you need to, you can add an extra filter to get only published posts.
However whether it is type safe or not depends on if you use the generated mods or not:
// Get only published posts
query := psql.Select(
UserPosts(1),
models.PostWhere.Status.EQ("published"), // type-safe
sm.Where(psql.Quote("posts", "status").Eq(psql.Arg("published"))), // not type-safe
)
Thanks to all the people who have contributed to Bob!