依葫芦画瓢理解一个小型Go框架

最近在开发Go程序,同事( github.com/WiFeng/go-sky )参考go-kit框架封装了一个简易的轮子,包含了Api和Task任务,已经能满足大部分Web需求,依葫芦画瓢,自己理解了下,参考下图:

1:cmd/service.go

package main

import (
    "github.com/WiFeng/go-sky"
    "pkg/config"
    "pkg/endpoint"
    "pkg/service"
    "pkg/task"
    "pkg/transport/http"
)

func main() {

    var (
        service     = service.New()
        endpoints   = endpoint.New(service)
        httpHandler = http.NewHandler(endpoints)
    )

    sky.LoadAppConfig(&config.GlobalAppConfig)
    sky.RegisterTask(task.Start, nil, true)
    sky.Run(httpHandler)
}

初始化service、endpoint,NewHandler注册路由作为web服务,再注册Task运行后台任务。

2:pkg\endpoint\endpoint.go:

package endpoint
import "pkg/service"

type Endpoints struct {
    Article ArticleEndpoints
}

func New(s service.Service) Endpoints {
    return Endpoints{
        Article: NewArticleEndpoints(s),
    }
}

返回一个大的Endpoints,其中包含子的Endpoints,会将service.Service结构体传递给Endpoints。

3:pkg\endpoint\article.go:

package endpoint

import (
    "context"
    kitendpoint "github.com/go-kit/kit/endpoint"
    . "pkg/entity"
    "pkg/service"
)

type ArticleEndpoints struct {
    MGet kitendpoint.Endpoint
}

func NewArticleEndpoints(s service.Service) ArticleEndpoints {
    return ArticleEndpoints{
        MGet: MakeArticleMGetEndpoint(s),
    }
}

func MakeArticleMGetEndpoint(s service.Service) kitendpoint.Endpoint {
    return func(ctx context.Context, request interface{}) (response interface{}, err error) {
        req := request.(ArticleInfoMGetRequest)
        return s.Article.MGet(ctx, req)
    }
}

每一个子的Endpoint应该包含同一种类型的服务,最终调用对应的service服务方法。

不过ArticleInfoMGetRequest也可以在service\article.go(例子中注册在entity) 中定义。MakeArticleMGetEndpoint返回一个闭包,注册了一个路由。

4:pkg\entity\article.go

package entity

type ArticleInfo struct {
    ArticleId int64 `json:"aid"`
    Uid       int64 `json:"uid"`
}

type ArticleInfoMGetRequest struct {
    BaseRequest
    ArticleIds   []int64 `json:"aids"`
    ForceNoCache bool    `json:"force_no_cache"`
}

type ArticleInfoMGetRespData struct {
    Infos []ArticleInfo `json:"infos"`
}

type ArticleInfoMGetResponse struct {
    BaseResponse
    Data ArticleInfoMGetRespData `json:"data"`
}

entity包含特定的工具方法。

5:pkg\service\service.go:

package service

type Service struct {
    Article ArticleService
}

func New() Service {
    return Service{
        Article: ArticleService{},
    }
}

service大结构体初始化,包括子service初始化。

6:pkg\service\article.go:

package service

import (
    "context"
      "pkg/dao"
    . "github.com/xiwujie/article/pkg/entity"
)

type ArticleSyncJobRequest struct {
    BaseRequest
    Limit   int    `json:"limit"`
    JobName string `json:"job_name"`
}

type ArticleSyncJobResponse struct {
    BaseResponse
}

type ArticleService struct {
}

func (s *ArticleService) MGet(ctx context.Context, req ArticleInfoMGetRequest) (interface{}, error) {
    var resp ArticleInfoMGetResponse

    if req.ArticleIds == nil || len(req.ArticleIds) < 1 {
        return resp, nil
    }
    sdao = dao.NewSearchActivityTable(ctx)
    sdao.FetchById()

    return resp, nil
}

具体的service服务,包含req,response的定义,也可以定义到 entry 目录下。

7:pkg/dao/article.go

package dao

import (
    "context"
    "database/sql"
    "fmt"

    skydb "github.com/WiFeng/go-sky/database"
)

const (
    searchActivityTableName = ""
)

type SearchActivityTable struct {
    db *sql.DB
}

func NewSearchActivityTable(ctx context.Context) (*SearchActivityTable, error) {
}

func (t *SearchActivityTable) FetchById(ctx context.Context, id int)  {
}

dao方法,主要进行数据库等资源的操作。

8:pkg\transport\http\handler.go

func NewHandler(endpoints endpoint.Endpoints) http.Handler {
    r := skyhttp.NewRouter()

    genericOptions := []kithttp.ServerOption{
        kithttp.ServerErrorEncoder(genericErrorEncoder),
    }

    r.Methods(http.MethodPost).Path(ArticleInfoMgetURI).Handler(skyhttp.NewServer(
        endpoints.Article.MGet,
        decodeHTTPArticleInfoMgetRequest,
        encodeHTTPGenericResponse,
        genericOptions...,
    ))

    return r
}

注册http路由,endpoint作为参数传递给handler。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值