1
0
mirror of https://github.com/go-micro/go-micro.git synced 2026-04-30 19:15:24 +02:00
Files
Asim Aslam 1bb25d6e7f Add agent platform showcase and refactor project structure (#2884)
* 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>
2026-03-05 11:21:41 +00:00

216 lines
4.7 KiB
Go

package memory
import (
"context"
"testing"
"go-micro.dev/v5/model"
)
type User struct {
ID string `json:"id" model:"key"`
Name string `json:"name" model:"index"`
Email string `json:"email"`
Age int `json:"age"`
}
func setup(t *testing.T) model.Model {
t.Helper()
db := New()
if err := db.Register(&User{}); err != nil {
t.Fatalf("register: %v", err)
}
return db
}
func TestCRUD(t *testing.T) {
db := setup(t)
ctx := context.Background()
// Create
err := db.Create(ctx, &User{ID: "1", Name: "Alice", Email: "alice@test.com", Age: 30})
if err != nil {
t.Fatalf("create: %v", err)
}
// Read
u := &User{}
err = db.Read(ctx, "1", u)
if err != nil {
t.Fatalf("read: %v", err)
}
if u.Name != "Alice" {
t.Errorf("expected Alice, got %s", u.Name)
}
if u.Age != 30 {
t.Errorf("expected age 30, got %d", u.Age)
}
// Update
u.Name = "Alice Updated"
u.Age = 31
err = db.Update(ctx, u)
if err != nil {
t.Fatalf("update: %v", err)
}
u2 := &User{}
db.Read(ctx, "1", u2)
if u2.Name != "Alice Updated" {
t.Errorf("expected 'Alice Updated', got %s", u2.Name)
}
if u2.Age != 31 {
t.Errorf("expected age 31, got %d", u2.Age)
}
// Delete
err = db.Delete(ctx, "1", &User{})
if err != nil {
t.Fatalf("delete: %v", err)
}
err = db.Read(ctx, "1", &User{})
if err != model.ErrNotFound {
t.Errorf("expected ErrNotFound, got %v", err)
}
}
func TestDuplicateKey(t *testing.T) {
db := setup(t)
ctx := context.Background()
db.Create(ctx, &User{ID: "1", Name: "Alice"})
err := db.Create(ctx, &User{ID: "1", Name: "Bob"})
if err != model.ErrDuplicateKey {
t.Errorf("expected ErrDuplicateKey, got %v", err)
}
}
func TestNotFound(t *testing.T) {
db := setup(t)
ctx := context.Background()
err := db.Read(ctx, "nonexistent", &User{})
if err != model.ErrNotFound {
t.Errorf("expected ErrNotFound, got %v", err)
}
err = db.Update(ctx, &User{ID: "nonexistent"})
if err != model.ErrNotFound {
t.Errorf("expected ErrNotFound on update, got %v", err)
}
err = db.Delete(ctx, "nonexistent", &User{})
if err != model.ErrNotFound {
t.Errorf("expected ErrNotFound on delete, got %v", err)
}
}
func TestList(t *testing.T) {
db := setup(t)
ctx := context.Background()
db.Create(ctx, &User{ID: "1", Name: "Alice", Age: 30})
db.Create(ctx, &User{ID: "2", Name: "Bob", Age: 25})
db.Create(ctx, &User{ID: "3", Name: "Charlie", Age: 35})
var all []*User
err := db.List(ctx, &all)
if err != nil {
t.Fatalf("list: %v", err)
}
if len(all) != 3 {
t.Errorf("expected 3, got %d", len(all))
}
}
func TestListWithFilter(t *testing.T) {
db := setup(t)
ctx := context.Background()
db.Create(ctx, &User{ID: "1", Name: "Alice", Age: 30})
db.Create(ctx, &User{ID: "2", Name: "Bob", Age: 25})
db.Create(ctx, &User{ID: "3", Name: "Alice", Age: 35})
var results []*User
err := db.List(ctx, &results, model.Where("name", "Alice"))
if err != nil {
t.Fatalf("list with filter: %v", err)
}
if len(results) != 2 {
t.Errorf("expected 2 Alices, got %d", len(results))
}
}
func TestListWithLimitOffset(t *testing.T) {
db := setup(t)
ctx := context.Background()
db.Create(ctx, &User{ID: "1", Name: "A", Age: 1})
db.Create(ctx, &User{ID: "2", Name: "B", Age: 2})
db.Create(ctx, &User{ID: "3", Name: "C", Age: 3})
db.Create(ctx, &User{ID: "4", Name: "D", Age: 4})
var results []*User
err := db.List(ctx, &results,
model.OrderAsc("name"),
model.Limit(2),
model.Offset(1),
)
if err != nil {
t.Fatalf("list: %v", err)
}
if len(results) != 2 {
t.Fatalf("expected 2, got %d", len(results))
}
if results[0].Name != "B" {
t.Errorf("expected B, got %s", results[0].Name)
}
if results[1].Name != "C" {
t.Errorf("expected C, got %s", results[1].Name)
}
}
func TestCount(t *testing.T) {
db := setup(t)
ctx := context.Background()
db.Create(ctx, &User{ID: "1", Name: "Alice", Age: 30})
db.Create(ctx, &User{ID: "2", Name: "Bob", Age: 25})
db.Create(ctx, &User{ID: "3", Name: "Alice", Age: 35})
count, err := db.Count(ctx, &User{})
if err != nil {
t.Fatalf("count: %v", err)
}
if count != 3 {
t.Errorf("expected 3, got %d", count)
}
count, err = db.Count(ctx, &User{}, model.Where("name", "Alice"))
if err != nil {
t.Fatalf("count with filter: %v", err)
}
if count != 2 {
t.Errorf("expected 2, got %d", count)
}
}
func TestWhereOp(t *testing.T) {
db := setup(t)
ctx := context.Background()
db.Create(ctx, &User{ID: "1", Name: "Alice", Age: 30})
db.Create(ctx, &User{ID: "2", Name: "Bob", Age: 25})
db.Create(ctx, &User{ID: "3", Name: "Charlie", Age: 35})
var results []*User
err := db.List(ctx, &results, model.WhereOp("age", ">", 28))
if err != nil {
t.Fatalf("list: %v", err)
}
if len(results) != 2 {
t.Errorf("expected 2 (age > 28), got %d", len(results))
}
}