* feat: add agent platform showcase and blog post Add a complete platform example (Users, Posts, Comments, Mail) that mirrors micro/blog, demonstrating how existing microservices become AI-accessible through MCP with zero code changes. Includes blog post "Your Microservices Are Already an AI Platform" walking through real agent workflows: signup, content creation, commenting, tagging, and cross-service messaging. https://claude.ai/code/session_01GkduEhcrqcG45rdfYh8dAc * refactor: rename handler types to drop redundant Service suffix UserService → Users, PostService → Posts, CommentService → Comments, MailService → Mail. Matches micro/blog naming convention. https://claude.ai/code/session_01GkduEhcrqcG45rdfYh8dAc * refactor: consolidate top-level directories, reduce framework bloat Move internal/non-public packages behind internal/ or into their parent packages where they belong: - deploy/ → gateway/mcp/deploy/ (Helm charts belong with the gateway) - profile/ → service/profile/ (preset plugin profiles are a service concern) - scripts/ → internal/scripts/ (install script is not public API) - test/ → internal/test/ (test harness is not public API) - util/ → internal/util/ (internal helpers shouldn't be imported externally) Also fixes CLAUDE.md merge conflict markers and updates project structure documentation. All import paths updated. Build and tests pass. https://claude.ai/code/session_01GkduEhcrqcG45rdfYh8dAc * refactor: redesign model package to match framework conventions Rename model.Database interface to model.Model (consistent with client.Client, server.Server, store.Store). Remove generics in favor of interface{}-based API with reflection. Key changes: - model.Model interface: Register once, CRUD infers table from type - DefaultModel + NewModel() + package-level convenience functions - Schema registered via Register(&User{}), no per-call schema passing - Memory implementation as default (in model package, like store) - memory/sqlite/postgres backends updated for new interface - protoc-gen-micro generates RegisterXModel() instead of generic factory - All docs, blog, and README updated https://claude.ai/code/session_01GkduEhcrqcG45rdfYh8dAc --------- Co-authored-by: Claude <noreply@anthropic.com>
Model Package
The model package provides a structured data storage interface with CRUD operations, query filtering, and multiple database backends.
Unlike the store package (which is a raw KV abstraction), model provides structured data access with schema awareness, WHERE queries, ordering, pagination, and indexes.
Quick Start
import (
"context"
"go-micro.dev/v5/model"
)
// Define your model with struct tags
type User struct {
ID string `json:"id" model:"key"`
Name string `json:"name" model:"index"`
Email string `json:"email"`
Age int `json:"age"`
}
// Create a model and register your type
db := model.NewModel()
db.Register(&User{})
ctx := context.Background()
// Create
db.Create(ctx, &User{ID: "1", Name: "Alice", Email: "alice@example.com", Age: 30})
// Read
user := &User{}
db.Read(ctx, "1", user)
fmt.Println(user.Name) // "Alice"
// Update
user.Name = "Alice Smith"
db.Update(ctx, user)
// Delete
db.Delete(ctx, "1", &User{})
Struct Tags
| Tag | Description | Example |
|---|---|---|
model:"key" |
Primary key field | ID string \model:"key"`` |
model:"index" |
Create an index on this field | Name string \model:"index"`` |
json:"name" |
Column name in the database | Name string \json:"name"`` |
If no model:"key" tag is found, the package defaults to a field with json:"id" or column name id.
Querying
// Filter by field value
var users []*User
db.List(ctx, &users, model.Where("name", "Alice"))
// Comparison operators
db.List(ctx, &users, model.WhereOp("age", ">", 25))
db.List(ctx, &users, model.WhereOp("name", "LIKE", "Ali%"))
// Ordering
db.List(ctx, &users, model.OrderAsc("name"))
db.List(ctx, &users, model.OrderDesc("age"))
// Pagination
db.List(ctx, &users, model.Limit(10), model.Offset(20))
// Combine
db.List(ctx, &users,
model.Where("status", "active"),
model.WhereOp("age", ">=", 18),
model.OrderDesc("created_at"),
model.Limit(25),
)
// Count
total, _ := db.Count(ctx, &User{})
active, _ := db.Count(ctx, &User{}, model.Where("status", "active"))
Backends
Memory (Development & Testing)
import "go-micro.dev/v5/model"
db := model.NewModel()
In-memory storage. No persistence. Fast. Good for tests and prototyping.
SQLite (Development & Single-Node Production)
import "go-micro.dev/v5/model/sqlite"
db := sqlite.New("app.db") // File-based
db := sqlite.New(":memory:") // In-memory (testing)
Embedded SQL database. Zero external dependencies. Supports WHERE, indexes, ordering natively.
Postgres (Production)
import "go-micro.dev/v5/model/postgres"
db := postgres.New("postgres://user:pass@localhost/mydb?sslmode=disable")
Full PostgreSQL support. Best for production with rich query capabilities.
Table Names
By default, the table name is the lowercase struct name + "s" (e.g., User → users). Override with model.WithTable:
db.Register(&User{}, model.WithTable("app_users"))
Model Interface
All backends implement the model.Model interface:
type Model interface {
Init(...Option) error
Register(v interface{}, opts ...RegisterOption) error
Create(ctx context.Context, v interface{}) error
Read(ctx context.Context, key string, v interface{}) error
Update(ctx context.Context, v interface{}) error
Delete(ctx context.Context, key string, v interface{}) error
List(ctx context.Context, result interface{}, opts ...QueryOption) error
Count(ctx context.Context, v interface{}, opts ...QueryOption) (int64, error)
Close() error
String() string
}
Model vs Store
| Feature | store |
model |
|---|---|---|
| Data format | Raw []byte |
Go structs |
| Queries | Key prefix/suffix only | WHERE, operators, LIKE |
| Ordering | None | ORDER BY field ASC/DESC |
| Pagination | Limit/Offset on keys | Limit/Offset on results |
| Indexes | None | Via model:"index" tag |
| Schema | None (schemaless KV) | Auto-created from struct |
| Backends | Memory, File, MySQL, Postgres, NATS | Memory, SQLite, Postgres |
| Use case | Config, sessions, cache | Application data, entities |
Testing
go test ./model/...