1 Star 0 Fork 0

13683679291/fabric

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
registry.go 7.62 KB
一键复制 编辑 原始数据 按行查看 历史
Will Lahti 提交于 2018-10-29 12:07 . Replace / in logger names with dots
/*
Copyright IBM Corp, SecureKey Technologies Inc. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0
*/
package library
import (
"fmt"
"os"
"plugin"
"reflect"
"sync"
"github.com/hyperledger/fabric/common/flogging"
"github.com/hyperledger/fabric/core/handlers/auth"
"github.com/hyperledger/fabric/core/handlers/decoration"
endorsement2 "github.com/hyperledger/fabric/core/handlers/endorsement/api"
"github.com/hyperledger/fabric/core/handlers/validation/api"
)
var logger = flogging.MustGetLogger("core.handlers")
// Registry defines an object that looks up
// handlers by name
type Registry interface {
// Lookup returns a handler with a given
// registered name, or nil if does not exist
Lookup(HandlerType) interface{}
}
// HandlerType defines custom handlers that can filter and mutate
// objects passing within the peer
type HandlerType int
const (
// Auth handler - reject or forward proposals from clients
Auth HandlerType = iota
// Decoration handler - append or mutate the chaincode input
// passed to the chaincode
Decoration
Endorsement
Validation
authPluginFactory = "NewFilter"
decoratorPluginFactory = "NewDecorator"
pluginFactory = "NewPluginFactory"
)
type registry struct {
filters []auth.Filter
decorators []decoration.Decorator
endorsers map[string]endorsement2.PluginFactory
validators map[string]validation.PluginFactory
}
var once sync.Once
var reg registry
// Config configures the factory methods
// and plugins for the registry
type Config struct {
AuthFilters []*HandlerConfig `mapstructure:"authFilters" yaml:"authFilters"`
Decorators []*HandlerConfig `mapstructure:"decorators" yaml:"decorators"`
Endorsers PluginMapping `mapstructure:"endorsers" yaml:"endorsers"`
Validators PluginMapping `mapstructure:"validators" yaml:"validators"`
}
type PluginMapping map[string]*HandlerConfig
// HandlerConfig defines configuration for a plugin or compiled handler
type HandlerConfig struct {
Name string `mapstructure:"name" yaml:"name"`
Library string `mapstructure:"library" yaml:"library"`
}
// InitRegistry creates the (only) instance
// of the registry
func InitRegistry(c Config) Registry {
once.Do(func() {
reg = registry{
endorsers: make(map[string]endorsement2.PluginFactory),
validators: make(map[string]validation.PluginFactory),
}
reg.loadHandlers(c)
})
return &reg
}
// loadHandlers loads the configured handlers
func (r *registry) loadHandlers(c Config) {
for _, config := range c.AuthFilters {
r.evaluateModeAndLoad(config, Auth)
}
for _, config := range c.Decorators {
r.evaluateModeAndLoad(config, Decoration)
}
for chaincodeID, config := range c.Endorsers {
r.evaluateModeAndLoad(config, Endorsement, chaincodeID)
}
for chaincodeID, config := range c.Validators {
r.evaluateModeAndLoad(config, Validation, chaincodeID)
}
}
// evaluateModeAndLoad if a library path is provided, load the shared object
func (r *registry) evaluateModeAndLoad(c *HandlerConfig, handlerType HandlerType, extraArgs ...string) {
if c.Library != "" {
r.loadPlugin(c.Library, handlerType, extraArgs...)
} else {
r.loadCompiled(c.Name, handlerType, extraArgs...)
}
}
// loadCompiled loads a statically compiled handler
func (r *registry) loadCompiled(handlerFactory string, handlerType HandlerType, extraArgs ...string) {
registryMD := reflect.ValueOf(&HandlerLibrary{})
o := registryMD.MethodByName(handlerFactory)
if !o.IsValid() {
logger.Panicf(fmt.Sprintf("Method %s isn't a method of HandlerLibrary", handlerFactory))
}
inst := o.Call(nil)[0].Interface()
if handlerType == Auth {
r.filters = append(r.filters, inst.(auth.Filter))
} else if handlerType == Decoration {
r.decorators = append(r.decorators, inst.(decoration.Decorator))
} else if handlerType == Endorsement {
if len(extraArgs) != 1 {
logger.Panicf("expected 1 argument in extraArgs")
}
r.endorsers[extraArgs[0]] = inst.(endorsement2.PluginFactory)
} else if handlerType == Validation {
if len(extraArgs) != 1 {
logger.Panicf("expected 1 argument in extraArgs")
}
r.validators[extraArgs[0]] = inst.(validation.PluginFactory)
}
}
// loadPlugin loads a pluggagle handler
func (r *registry) loadPlugin(pluginPath string, handlerType HandlerType, extraArgs ...string) {
if _, err := os.Stat(pluginPath); err != nil {
logger.Panicf(fmt.Sprintf("Could not find plugin at path %s: %s", pluginPath, err))
}
p, err := plugin.Open(pluginPath)
if err != nil {
logger.Panicf(fmt.Sprintf("Error opening plugin at path %s: %s", pluginPath, err))
}
if handlerType == Auth {
r.initAuthPlugin(p)
} else if handlerType == Decoration {
r.initDecoratorPlugin(p)
} else if handlerType == Endorsement {
r.initEndorsementPlugin(p, extraArgs...)
} else if handlerType == Validation {
r.initValidationPlugin(p, extraArgs...)
}
}
// initAuthPlugin constructs an auth filter from the given plugin
func (r *registry) initAuthPlugin(p *plugin.Plugin) {
constructorSymbol, err := p.Lookup(authPluginFactory)
if err != nil {
panicWithLookupError(authPluginFactory, err)
}
constructor, ok := constructorSymbol.(func() auth.Filter)
if !ok {
panicWithDefinitionError(authPluginFactory)
}
filter := constructor()
if filter != nil {
r.filters = append(r.filters, filter)
}
}
// initDecoratorPlugin constructs a decorator from the given plugin
func (r *registry) initDecoratorPlugin(p *plugin.Plugin) {
constructorSymbol, err := p.Lookup(decoratorPluginFactory)
if err != nil {
panicWithLookupError(decoratorPluginFactory, err)
}
constructor, ok := constructorSymbol.(func() decoration.Decorator)
if !ok {
panicWithDefinitionError(decoratorPluginFactory)
}
decorator := constructor()
if decorator != nil {
r.decorators = append(r.decorators, constructor())
}
}
func (r *registry) initEndorsementPlugin(p *plugin.Plugin, extraArgs ...string) {
if len(extraArgs) != 1 {
logger.Panicf("expected 1 argument in extraArgs")
}
factorySymbol, err := p.Lookup(pluginFactory)
if err != nil {
panicWithLookupError(pluginFactory, err)
}
constructor, ok := factorySymbol.(func() endorsement2.PluginFactory)
if !ok {
panicWithDefinitionError(pluginFactory)
}
factory := constructor()
if factory == nil {
logger.Panicf("factory instance returned nil")
}
r.endorsers[extraArgs[0]] = factory
}
func (r *registry) initValidationPlugin(p *plugin.Plugin, extraArgs ...string) {
if len(extraArgs) != 1 {
logger.Panicf("expected 1 argument in extraArgs")
}
factorySymbol, err := p.Lookup(pluginFactory)
if err != nil {
panicWithLookupError(pluginFactory, err)
}
constructor, ok := factorySymbol.(func() validation.PluginFactory)
if !ok {
panicWithDefinitionError(pluginFactory)
}
factory := constructor()
if factory == nil {
logger.Panicf("factory instance returned nil")
}
r.validators[extraArgs[0]] = factory
}
// panicWithLookupError panics when a handler constructor lookup fails
func panicWithLookupError(factory string, err error) {
logger.Panicf(fmt.Sprintf("Plugin must contain constructor with name %s. Error from lookup: %s",
factory, err))
}
// panicWithDefinitionError panics when a handler constructor does not match
// the expected function definition
func panicWithDefinitionError(factory string) {
logger.Panicf(fmt.Sprintf("Constructor method %s does not match expected definition",
factory))
}
// Lookup returns a list of handlers with the given
// given type, or nil if none exist
func (r *registry) Lookup(handlerType HandlerType) interface{} {
if handlerType == Auth {
return r.filters
} else if handlerType == Decoration {
return r.decorators
} else if handlerType == Endorsement {
return r.endorsers
} else if handlerType == Validation {
return r.validators
}
return nil
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/mmcro/fabric.git
git@gitee.com:mmcro/fabric.git
mmcro
fabric
fabric
v1.4.2

搜索帮助