Go Micro
March 4, 2026 — By the Go Micro Team
Go Micro has always given you service.Client() to call other services and service.Server() to handle requests. But most services also need to save and query data. Until now, that meant either using the low-level store package (key-value only) or wiring up your own database layer.
Today we’re shipping the model package — a typed data model layer that completes the service trifecta: Client, Server, Model.
The existing store package is great for simple key-value storage, but real services need more. You need to filter by fields, paginate results, count records, and use different databases in dev vs production. Most teams end up writing their own data layer or pulling in an ORM that has nothing to do with Go Micro.
We wanted something that feels native to the framework. Define a Go struct, tag a key, and get type-safe CRUD and queries — with the same pluggable backend pattern Go Micro uses everywhere.
type User struct {
ID string `json:"id" model:"key"`
Name string `json:"name"`
Email string `json:"email" model:"index"`
Age int `json:"age"`
}
The model:"key" tag marks your primary key. The model:"index" tag creates an index for faster queries. Column names come from json tags (or lowercased field names if no tag).
Register your type and use it:
db := service.Model()
db.Register(&User{})
// Create
db.Create(ctx, &User{ID: "1", Name: "Alice", Email: "alice@example.com", Age: 30})
// Read
user := &User{}
db.Read(ctx, "1", user)
// Update
user.Name = "Alice Smith"
db.Update(ctx, user)
// Delete
db.Delete(ctx, "1", &User{})
No migrations. No connection setup. No configuration files. The schema is derived from your struct at startup.
List and count with composable query options:
var active []*User
// Simple equality filter
db.List(ctx, &active, model.Where("email", "alice@example.com"))
// Operators, ordering, pagination
var page []*User
db.List(ctx, &page,
model.WhereOp("age", ">=", 18),
model.OrderDesc("name"),
model.Limit(10),
model.Offset(20),
)
// Count records
total, _ := db.Count(ctx, &User{}, model.Where("age", 30))
Filters support =, !=, <, >, <=, >=, and LIKE. Everything composes — add as many query options as you need.
The model layer follows Go Micro’s pluggable pattern. Same code, different backends:
Memory — the default. Zero config, great for development and testing:
service := micro.New("users")
db := service.Model() // in-memory by default
db.Register(&User{})
SQLite — single-file database for local development or single-node production:
db, _ := sqlite.New(model.WithDSN("file:app.db"))
service := micro.New("users", micro.Model(db))
Postgres — production-grade with connection pooling:
db, _ := postgres.New(model.WithDSN("postgres://localhost/myapp"))
service := micro.New("users", micro.Model(db))
Start with memory in dev, switch to SQLite or Postgres for production. Your application code doesn’t change.
The Service interface now has three core accessors:
type Service interface {
Client() client.Client // Call other services
Server() server.Server // Handle incoming requests
Model() model.Model // Save and query data
// ...
}
This means a typical service has everything it needs in one place:
func main() {
service := micro.New("users", micro.Address(":9001"))
// Data layer
db := service.Model()
db.Register(&User{})
// Handler with data access
service.Handle(&UserService{db: db})
// Run
service.Run()
}
Call services with service.Client(). Handle requests with service.Server(). Save data with service.Model(). That’s the complete picture.
You can create multiple typed models from the same database connection:
db := service.Model()
db.Register(&User{})
db.Register(&Post{})
db.Register(&Comment{})
Each type gets its own table (derived from the struct name). They share the database connection.
The model package is production-ready with memory, SQLite, and Postgres backends. Coming soon:
protoc-gen-micro generates model code from proto definitionsSee the model documentation for the full API reference, or browse the model package source to see the implementation.