聊聊dubbo-go-proxy的Client

本文主要研究一下dubbo-go-proxy的Client

Client

dubbo-go-proxy/pkg/client/client.go

// Client represents the interface of http/dubbo clients
type Client interface {
	Init() error
	Close() error

	// Call invoke the downstream service.
	Call(req *Request) (res interface{}, err error)

	// MapParams mapping param, uri, query, body ...
	MapParams(req *Request) (reqData interface{}, err error)
}

Client接口定義了Init、Close、Call、MapParams方法

Client

dubbo-go-proxy/pkg/client/dubbo/dubbo.go

// Client client to generic invoke dubbo
type Client struct {
	lock               sync.RWMutex
	GenericServicePool map[string]*dg.GenericService
}

Client定義了lock、GenericServicePool屬性,它實現了Client接口

Init

dubbo-go-proxy/pkg/client/dubbo/dubbo.go

// Init init dubbo, config mapping can do here
func (dc *Client) Init() error {
	dc.GenericServicePool = make(map[string]*dg.GenericService, 4)

	cls := config.GetBootstrap().StaticResources.Clusters

	// dubbogo comsumer config
	dgCfg = dg.ConsumerConfig{
		Check:      new(bool),
		Registries: make(map[string]*dg.RegistryConfig, 4),
	}
	dgCfg.ApplicationConfig = defaultApplication
	for i := range cls {
		c := cls[i]
		dgCfg.Request_Timeout = c.RequestTimeoutStr
		dgCfg.Connect_Timeout = c.ConnectTimeoutStr
		for k, v := range c.Registries {
			if len(v.Protocol) == 0 {
				logger.Warnf("can not find registry protocol config, use default type 'zookeeper'")
				v.Protocol = defaultDubboProtocol
			}
			dgCfg.Registries[k] = &dg.RegistryConfig{
				Protocol:   v.Protocol,
				Address:    v.Address,
				TimeoutStr: v.Timeout,
				Username:   v.Username,
				Password:   v.Password,
			}
		}
	}

	initDubbogo()

	return nil
}

func initDubbogo() {
	dg.SetConsumerConfig(dgCfg)
	dg.Load()
}

Init方法主要是構建ConsumerConfig,然後執行initDubbogo

Close

dubbo-go-proxy/pkg/client/dubbo/dubbo.go

// Close clear GenericServicePool.
func (dc *Client) Close() error {
	dc.lock.Lock()
	defer dc.lock.Unlock()
	for k := range dc.GenericServicePool {
		delete(dc.GenericServicePool, k)
	}
	return nil
}

Close方法通過加鎖遍歷dc.GenericServicePool執行delete操作

Call

dubbo-go-proxy/pkg/client/dubbo/dubbo.go

// Call invoke service
func (dc *Client) Call(req *client.Request) (res interface{}, err error) {
	types, values, err := dc.genericArgs(req)
	if err != nil {
		return nil, err
	}

	dm := req.API.Method.IntegrationRequest
	method := dm.Method

	logger.Debugf("[dubbo-go-proxy] dubbo invoke, method:%s, types:%s, reqData:%v", method, types, values)

	gs := dc.Get(dm)

	rst, err := gs.Invoke(req.Context, []interface{}{method, types, values})

	if err != nil {
		return nil, err
	}

	logger.Debugf("[dubbo-go-proxy] dubbo client resp:%v", rst)

	return rst, nil
}

Call方法通過dc.Get(dm)獲取GenericService,然後通過GenericService.Invoke進行請求

MapParams

dubbo-go-proxy/pkg/client/dubbo/dubbo.go

// MapParams params mapping to api.
func (dc *Client) MapParams(req *client.Request) (interface{}, error) {
	r := req.API.Method.IntegrationRequest
	if len(r.ParamTypes) != len(r.MappingParams) {
		return nil, errors.New("Numbers of param types and paramMappings are not the same")
	}
	var values []interface{}
	for _, mappingParam := range r.MappingParams {
		source, _, err := client.ParseMapSource(mappingParam.Name)
		if err != nil {
			return nil, err
		}
		if mapper, ok := mappers[source]; ok {
			if err := mapper.Map(mappingParam, req, &values, buildOption(mappingParam)); err != nil {
				return nil, err
			}
		}
	}
	return values, nil
}

MapParams方法遍歷MappingParams,通過client.ParseMapSource獲取source,再通過mappers[source]獲取mapper,最後通過mapper的Map方法進行轉換

小結

dubbo-go-proxy的client.Client接口定義了Init、Close、Call、MapParams方法;其dubbo.Client實現了client.Client接口;其主要是通過mapper進行參數轉換,然後通過GenericService.Invoke進行請求。

doc

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章