In this article, we're comparing commonly used Go ORMs to help you answer which one is suitable for your project. But first, let's start with what an ORM is and when you might want to use one.
An ORM (Object-Relational Mapper) is a tool that simplifies interaction with databases. In Go development, it basically maps Go structs to database tables, making it more intuitive to work with relational data.
For example, we might have this table in our database:
CREATE TABLE users (
id INT AUTO_INCREMENT PRIMARY KEY,
first_name VARCHAR(100),
last_name VARCHAR(100),
);
And you want to use this SQL query:
-- Fetch all users with the last name "Smith"
SELECT * FROM users WHERE last_name = 'Smith';
Which using an ORM (GORM in the example below) can be written like this in your Go program:
// First we create a Go struct that maps to our users table
type User struct {
ID uint
FirstName string
LastName string
}
// Using GORM, the equivalent query looks like this
var users []User
result := db.Where("last_name = ?", "Smith").Find(&users)
In this example, db
is your GORM database connection.
The Where
method adds a condition to the query, and Find
executes the query, populating the users
slice with the results.
This GORM query abstracts away the raw SQL, providing a more idiomatic Go way to interact with the database. It's particularly useful for more complex queries and can significantly improve code readability and maintainability.
Typically you'd use an ORM when you need to speed up database-related development, ensure type safety, and make code easier to read and maintain. It's particularly handy for applications with complex data models or when you prefer not to deal with the boilerplate of raw SQL queries.
It's wise to be cautious about using ORMs in performance-critical systems or where you need direct control over database interactions, as ORMs can add overhead and sometimes obscure what's happening at the database level.
Now without further ado, let's take a look at the best Go ORMs.
GORM is a comprehensive ORM tool in Go, offering a code-first approach which allows defining database schemas using struct tags in Go. It's known for its developer-friendly nature, making it suitable for both beginners and experienced users. GORM supports a variety of SQL databases like MySQL, PostgreSQL, and SQLite. It's designed to be flexible, allowing developers to drop down to raw SQL when necessary. However, it's important to be cautious about its performance implications in large-scale applications.
An easy path to try out GORM is by using Encore.go. It's a backend framework for Go that automatically provisions and migrates PostgreSQL databases in your local development environment. Learn more in the docs.
sqlc is not strictly a conventional ORM. It offers a unique approach by generating Go code from SQL queries. This allows developers to write SQL, which sqlc then converts into type-safe Go code, reducing the boilerplate significantly. It ensures that your queries are syntactically correct and type-safe. sqlc is ideal for those who prefer writing SQL and are looking for an efficient way to integrate it into a Go application. Here at Encore we really like sqlc, so much so that we wrote a whole blog post about it.
Encore is designed to work well with sqlc and is a simple method of trying it out, as Encore will take care of provisioning your databases automatically. Check out this hello-world example app using Encore and sqlc on GitHub.
ent is a fairly recent ORM that uses a code-first approach where you define your schema in Go code. Ent is popular thanks to its ability to handle complex data models and relationships elegantly. It's statically typed, which can help catch errors at compile time. However, the learning curve might be steeper compared to more straightforward ORMs like GORM. It's a good fit for applications where complex data models and type safety are priorities.
Try out ent in a simple way by leveraging Encore's capability to automatically provision and migrate PostgreSQL databases. Learn more in the docs.
SQLBoiler takes a database-first approach, generating Go code from your database schema. This means it creates highly optimized and custom-tailored code for your specific database schema. SQLBoiler is great for applications where the database schema is well-defined and changes infrequently. However, like sqlc, it requires regenerating the code when the database schema changes. It's well-suited for projects where performance is a key concern and the database design is stable.
Each ORM has its strengths and ideal use cases. GORM and ent are more code-first and user-friendly, suitable for a broad range of applications. sqlc and SQLBoiler, on the other hand, are more specialized, excelling in scenarios where direct SQL interaction or database-first design is preferred. The choice largely depends on the specific needs and preferences of your project.
We hope this brief article helps inform your choice about which Go ORM to use in your next project. Related to picking an ORM, it's common to want to use a framework to make development more efficient. There are many frameworks to choose from, all with different characteristics. To help inform your decision, we've recently published a guide to the best Go frameworks.