go-kratos v2 + gorm 实现增删改查demo
由于正在学习Bilibili的go开源框架 go-kratos,简单学习的过程中有一些摸索过程,现在根据go-kratos v2 + gorm 实现增删改查demo实现用户的增删改查
mysql用户表
#sql
CREATE TABLE `user` (
`id` int unsigned NOT NULL AUTO_INCREMENT,
`name` char(20) COLLATE utf8mb4_unicode_ci NOT NULL COMMENT '名称',
PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='用户表';
项目初始化
0.文档地址
1.创建项目模板
kratos new user
cd user
2.拉取项目依赖
go mod download
3.生成proto模板
kratos proto add api/user/user.proto
这个时候我们去修改api/user/user.proto, 先仅仅实现用户新增功能,代码如下
syntax = "proto3";
package api.user.v1;
option go_package = "users/api/user/v1;v1";
option java_multiple_files = true;
option java_package = "api.user.v1";
import "google/api/annotations.proto";
// the validate rules:
// https://github.com/envoyproxy/protoc-gen-validate
import "validate/validate.proto";
service user {
// 用户创建
rpc CreateUser(CreateUserRequest) returns (CreateUserReply){
option (google.api.http) = {
post: "/v1/user"
body: "*"
};
}
message User {
int64 id = 1;
string name = 2;
}
message CreateUserRequest {
string name = 1 [(validate.rules).string = {min_len : 5, max_len: 50}];
}
message CreateUserReply {
User User = 1;
}
proro文件中有参数校验,使用 proto-gen-validate 文档地址
4.生成proto源码 文档地址
kratos proto client api/user/user.proto
5.生成server模板
kratos proto server api/user/user.proto -t internal/service
coding阶段
打开 internal/service/user.go,代码如下:
package service
import (
"context"
"github.com/go-kratos/kratos/v2/log"
"users/internal/biz"
pb "users/api/user/v1"
)
type UserService struct {
pb.UnimplementedUserServer
uc *biz.UserUseCase
log *log.Helper
}
func NewUserService(uc *biz.UserUseCase, logger log.Logger) *UserService {
return &UserService{uc:uc, log: log.NewHelper(logger)}
}
// 用户创建
func (s *UserService) CreateUser(ctx context.Context, req *pb.CreateUserRequest) (*pb.CreateUserReply, error) {
s.log.WithContext(ctx).Infof("CreateUser Received: %v", req)
user := biz.User{
Id: 0,
Name: req.Name,
}
return &pb.CreateUserReply{}, s.uc.Create(ctx, &user)
}
打开 internal/service/service.go,代码如下:
package service
import "github.com/google/wire"
// ProviderSet is service providers.
var ProviderSet = wire.NewSet(NewUserService)
代码中引入了biz.User, 建立 ./internal/biz/user.go
package biz
import (
"context"
"github.com/go-kratos/kratos/v2/log"
)
type User struct {
Id int64
Name string
}
type UserRepo interface{
CreateUser(ctx context.Context, user *User) error
}
type UserUseCase struct{
repo UserRepo
log *log.Helper
}
func NewUserUseCase(repo UserRepo, logger log.Logger) *UserUseCase {
return &UserUseCase{repo:repo, log: log.NewHelper(logger)}
}
// 创建
func (uc *UserUseCase) Create(ctx context.Context, user *User) error {
return uc.repo.CreateUser(ctx, user)
}
修改 ./internal/biz/biz.go
package biz
import "github.com/google/wire"
// ProviderSet is biz providers.
var ProviderSet = wire.NewSet(NewUserUseCase)
打开 ./internal/data/data.go
package data
import (
"users/internal/conf"
"github.com/go-kratos/kratos/v2/log"
"github.com/google/wire"
"gorm.io/driver/mysql"
"gorm.io/gorm"
)
// ProviderSet is data providers.
var ProviderSet = wire.NewSet(NewData, NewUserRepo)
// Data .
type Data struct {
db *gorm.DB
}
// NewData .
func NewData(conf *conf.Data, logger log.Logger) (*Data, func(), error) {
log := log.NewHelper(logger)
// mysql数据库连接
db, err := gorm.Open(mysql.Open(conf.Database.Source), &gorm.Config{});
if err != nil {
return nil, nil, err
}
d := &Data{
db: db,
}
return d, func() {
log.Info("message", "closing the data resources")
}, nil
}
建立 ./internal/data/user.go
package data
import (
"context"
"errors"
"github.com/go-kratos/kratos/v2/log"
"gorm.io/gorm"
"users/internal/biz"
"users/internal/data/model"
)
type userRepo struct {
data *Data
log *log.Helper
}
func (u userRepo) CreateUser(ctx context.Context, user *biz.User) error {
// 判断名称是否存在,存在则返回错误
record := u.GetUserByName(user.Name)
if record.Id != 0 {
return errors.New("用户名已存在")
}
return u.data.db.Model(&model.User{}).Create(&model.User{
Id: 0,
Name: user.Name,
}).Error
}
func NewUserRepo(data *Data, logger log.Logger) biz.UserRepo {
return &userRepo{
data: data,
log: log.NewHelper(logger),
}
}
建立 ./internal/data/model/user.go
package model
type User struct{
Id int64 `json:"id"`
Name string `json:"name"`
}
func (User) TableName() string {
return "user"
}
修改./internal/server/http.go
package server
import (
"github.com/go-kratos/kratos/v2/log"
"github.com/go-kratos/kratos/v2/middleware/logging"
"github.com/go-kratos/kratos/v2/middleware/metrics"
"github.com/go-kratos/kratos/v2/middleware/recovery"
"github.com/go-kratos/kratos/v2/middleware/tracing"
"github.com/go-kratos/kratos/v2/middleware/validate"
"github.com/go-kratos/kratos/v2/transport/http"
v1 "users/api/user/v1"
"users/internal/conf"
"users/internal/service"
)
// NewHTTPServer new a HTTP server.
func NewHTTPServer(c *conf.Server, user *service.UserService, logger log.Logger) *http.Server {
var opts = []http.ServerOption{
http.Middleware(
recovery.Recovery(),
tracing.Server(),
logging.Server(logger),
metrics.Server(),
validate.Validator(),
),
}
if c.Http.Network != "" {
opts = append(opts, http.Network(c.Http.Network))
}
if c.Http.Addr != "" {
opts = append(opts, http.Address(c.Http.Addr))
}
if c.Http.Timeout != nil {
opts = append(opts, http.Timeout(c.Http.Timeout.AsDuration()))
}
srv := http.NewServer(opts...)
v1.RegisterUserHTTPServer(srv, user)
return srv
}
修改./internal/server/grpc.go
package server
import (
"github.com/go-kratos/kratos/v2/log"
"github.com/go-kratos/kratos/v2/middleware/logging"
"github.com/go-kratos/kratos/v2/middleware/metrics"
"github.com/go-kratos/kratos/v2/middleware/recovery"
"github.com/go-kratos/kratos/v2/middleware/tracing"
"github.com/go-kratos/kratos/v2/middleware/validate"
"github.com/go-kratos/kratos/v2/transport/grpc"
v1 "users/api/user/v1"
"users/internal/conf"
"users/internal/service"
)
// NewGRPCServer new a gRPC server.
func NewGRPCServer(c *conf.Server, user *service.UserService, logger log.Logger) *grpc.Server {
var opts = []grpc.ServerOption{
grpc.Middleware(
recovery.Recovery(),
tracing.Server(),
logging.Server(logger),
metrics.Server(),
validate.Validator(),
),
}
if c.Grpc.Network != "" {
opts = append(opts, grpc.Network(c.Grpc.Network))
}
if c.Grpc.Addr != "" {
opts = append(opts, grpc.Address(c.Grpc.Addr))
}
if c.Grpc.Timeout != nil {
opts = append(opts, grpc.Timeout(c.Grpc.Timeout.AsDuration()))
}
srv := grpc.NewServer(opts...)
v1.RegisterUserServer(srv, user)
return srv
}
修改配置文件
打开 ./configs/config.yaml,修改成自己的数据库
server:
http:
addr: 0.0.0.0:8000
timeout: 1s
grpc:
addr: 0.0.0.0:9000
timeout: 1s
data:
database:
driver: mysql
source: root: root@tcp(127.0.0.1:3306)/kratos_user?charset=utf8&parseTime=True&loc=Local
redis:
addr: 127.0.0.1:6379
read_timeout: 0.2s
write_timeout: 0.2s
项目编译
生成所有proto源码、wire等等
go generate ./...
运行项目
kratos run
发起请求 (postman)
POST localhost:8000/v1/user
{
"name":"周周周周周"
}
以上是单个http接口请求,完整的增删改查自行下载
链接: https://pan.baidu.com/s/1mfUGAFr4YA3GtQUJ9Gd6TQ 密码: 2cma