|
@@ -0,0 +1,214 @@
|
|
|
+package mqtt_binding
|
|
|
+
|
|
|
+import (
|
|
|
+ "git.sxidc.com/go-framework/baize/framework/core/domain"
|
|
|
+ "git.sxidc.com/go-framework/baize/framework/core/infrastructure"
|
|
|
+ "git.sxidc.com/go-framework/baize/framework/core/infrastructure/logger"
|
|
|
+ "git.sxidc.com/go-framework/baize/framework/core/mqtt_api"
|
|
|
+ "git.sxidc.com/go-framework/baize/framework/core/mqtt_api/request"
|
|
|
+ "git.sxidc.com/go-framework/baize/framework/core/mqtt_api/response"
|
|
|
+ "git.sxidc.com/go-tools/utils/reflectutils"
|
|
|
+ "git.sxidc.com/go-tools/utils/strutils"
|
|
|
+ "github.com/pkg/errors"
|
|
|
+ "reflect"
|
|
|
+ "strings"
|
|
|
+)
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+type ResponseIdentifierFunc func(c *mqtt_api.Context, params any) (string, error)
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+type FormDomainObjectsFunc func(c *mqtt_api.Context, params any) ([]domain.Object, error)
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+type ServiceFunc[O any] func(c *mqtt_api.Context, params any, objects []domain.Object, i *infrastructure.Infrastructure) (O, error)
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+func Bind[O any](binder *Binder, item *BindItem[O], responseIdentifier string, middlewares ...Middleware) {
|
|
|
+ item.bind(binder, responseIdentifier, middlewares...)
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+type BindItem[O any] struct {
|
|
|
+
|
|
|
+ Topic string
|
|
|
+
|
|
|
+
|
|
|
+ SendResponseFunc response.SendResponseFunc[O]
|
|
|
+
|
|
|
+
|
|
|
+ RequestParams any
|
|
|
+
|
|
|
+
|
|
|
+ ResponseIdentifierFunc ResponseIdentifierFunc
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ FormDomainObjectsFunc FormDomainObjectsFunc
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ Objects []domain.Object
|
|
|
+
|
|
|
+
|
|
|
+ ServiceFunc ServiceFunc[O]
|
|
|
+}
|
|
|
+
|
|
|
+func (item *BindItem[O]) bind(binder *Binder, responseIdentifier string, middlewares ...Middleware) {
|
|
|
+ if strutils.IsStringEmpty(item.Topic) {
|
|
|
+ panic("需要指定主题")
|
|
|
+ }
|
|
|
+
|
|
|
+ if item.SendResponseFunc == nil {
|
|
|
+ panic("需要指定响应函数")
|
|
|
+ }
|
|
|
+
|
|
|
+ if item.ServiceFunc == nil {
|
|
|
+ panic("需要指定应用服务函数")
|
|
|
+ }
|
|
|
+
|
|
|
+ var outputZero O
|
|
|
+ outputZeroValue := reflect.ValueOf(outputZero)
|
|
|
+ if outputZeroValue.IsValid() && outputZeroValue.Kind() == reflect.Pointer {
|
|
|
+ panic("bind的输出类型不能使用指针类型")
|
|
|
+ }
|
|
|
+
|
|
|
+ if outputZeroValue.IsValid() && strings.Contains(outputZeroValue.String(), "response.InfosData") {
|
|
|
+ infosField := outputZeroValue.FieldByName("Infos")
|
|
|
+ if infosField.IsValid() && infosField.Type().Elem().Kind() == reflect.Pointer {
|
|
|
+ panic("bind的输出类型不能使用指针类型")
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ apiMiddlewares := make([]mqtt_api.Handler, len(middlewares))
|
|
|
+ for i, middleware := range middlewares {
|
|
|
+ innerMiddleware := middleware
|
|
|
+ apiMiddlewares[i] = func(c *mqtt_api.Context) {
|
|
|
+ innerMiddleware(c, binder.i)
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ handlers := append(apiMiddlewares, func(c *mqtt_api.Context) {
|
|
|
+ var params any
|
|
|
+
|
|
|
+
|
|
|
+ if item.RequestParams != nil {
|
|
|
+ requestParamsType := reflect.TypeOf(item.RequestParams)
|
|
|
+ if !reflectutils.IsTypeStructOrStructPointer(requestParamsType) {
|
|
|
+ err := errors.New("请求参数不是结构或结构指针")
|
|
|
+ logger.GetInstance().Error(err)
|
|
|
+ item.SendResponseFunc(c, responseIdentifier, outputZero, err)
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ if requestParamsType.Kind() == reflect.Pointer {
|
|
|
+ params = reflect.New(requestParamsType.Elem()).Interface()
|
|
|
+ } else {
|
|
|
+ params = reflect.New(requestParamsType).Interface()
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ err := request.BindingJson(c, params)
|
|
|
+ if err != nil {
|
|
|
+ logger.GetInstance().Error(err)
|
|
|
+ item.SendResponseFunc(c, responseIdentifier, outputZero, err)
|
|
|
+ return
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if item.ResponseIdentifierFunc != nil {
|
|
|
+ newResponseIdentifier, err := item.ResponseIdentifierFunc(c, params)
|
|
|
+ if err != nil {
|
|
|
+ logger.GetInstance().Error(err)
|
|
|
+ item.SendResponseFunc(c, responseIdentifier, outputZero, err)
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ responseIdentifier = newResponseIdentifier
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ var domainObjects []domain.Object
|
|
|
+ if item.FormDomainObjectsFunc != nil {
|
|
|
+ innerDomainObjects, err := item.FormDomainObjectsFunc(c, params)
|
|
|
+ if err != nil {
|
|
|
+ item.SendResponseFunc(c, responseIdentifier, outputZero, err)
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ domainObjects = innerDomainObjects
|
|
|
+ } else {
|
|
|
+ if item.Objects != nil && len(item.Objects) != 0 {
|
|
|
+ for _, object := range item.Objects {
|
|
|
+ if object == nil {
|
|
|
+ continue
|
|
|
+ }
|
|
|
+
|
|
|
+ objectType := reflect.TypeOf(object)
|
|
|
+ if !reflectutils.IsTypeStructOrStructPointer(objectType) {
|
|
|
+ item.SendResponseFunc(c, responseIdentifier, outputZero, errors.New("领域对象不是结构或结构指针"))
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ obj := reflect.New(reflectutils.PointerTypeElem(objectType)).Interface().(domain.Object)
|
|
|
+
|
|
|
+ if params != nil {
|
|
|
+ err := request.AssignRequestParamsToDomainObject(params, obj)
|
|
|
+ if err != nil {
|
|
|
+ item.SendResponseFunc(c, responseIdentifier, outputZero, err)
|
|
|
+ return
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ domainObjects = append(domainObjects, obj)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ outputModel, err := item.ServiceFunc(c, params, domainObjects, binder.i)
|
|
|
+
|
|
|
+
|
|
|
+ item.SendResponseFunc(c, responseIdentifier, outputModel, err)
|
|
|
+ return
|
|
|
+ })
|
|
|
+
|
|
|
+
|
|
|
+ err := binder.router.AddTopic(item.Topic, handlers...)
|
|
|
+ if err != nil {
|
|
|
+ panic("添加主题" + item.Topic + "失败")
|
|
|
+ }
|
|
|
+}
|