2016-08-06 19:59:22 +00:00
|
|
|
package internet
|
|
|
|
|
|
|
|
import (
|
2016-09-21 12:39:07 +00:00
|
|
|
"errors"
|
|
|
|
|
2016-08-20 18:55:45 +00:00
|
|
|
"v2ray.com/core/common"
|
|
|
|
"v2ray.com/core/common/alloc"
|
2016-09-21 12:39:07 +00:00
|
|
|
"v2ray.com/core/common/loader"
|
|
|
|
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
|
|
"github.com/golang/protobuf/ptypes"
|
2016-08-06 19:59:22 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type Authenticator interface {
|
|
|
|
Seal(*alloc.Buffer)
|
|
|
|
Open(*alloc.Buffer) bool
|
|
|
|
Overhead() int
|
|
|
|
}
|
|
|
|
|
|
|
|
type AuthenticatorFactory interface {
|
2016-09-21 12:39:07 +00:00
|
|
|
Create(interface{}) Authenticator
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *AuthenticatorConfig) GetInternalConfig() (interface{}, error) {
|
|
|
|
return configCache.CreateConfig(this.Name)
|
2016-08-06 19:59:22 +00:00
|
|
|
}
|
|
|
|
|
2016-09-21 12:39:07 +00:00
|
|
|
func NewAuthenticatorConfig(name string, config interface{}) (*AuthenticatorConfig, error) {
|
|
|
|
pbMsg, ok := config.(proto.Message)
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("Internet|Authenticator: Failed to convert config into proto message.")
|
|
|
|
}
|
|
|
|
anyConfig, err := ptypes.MarshalAny(pbMsg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &AuthenticatorConfig{
|
|
|
|
Name: name,
|
|
|
|
Settings: anyConfig,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *AuthenticatorConfig) CreateAuthenticator() (Authenticator, error) {
|
|
|
|
config, err := this.GetInternalConfig()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return CreateAuthenticator(this.Name, config)
|
2016-08-06 19:59:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
authenticatorCache = make(map[string]AuthenticatorFactory)
|
2016-09-21 12:39:07 +00:00
|
|
|
configCache = loader.ConfigCreatorCache{}
|
2016-08-06 19:59:22 +00:00
|
|
|
)
|
|
|
|
|
2016-08-25 11:25:59 +00:00
|
|
|
func RegisterAuthenticator(name string, factory AuthenticatorFactory) error {
|
2016-08-06 19:59:22 +00:00
|
|
|
if _, found := authenticatorCache[name]; found {
|
2016-08-18 06:34:21 +00:00
|
|
|
return common.ErrDuplicatedName
|
2016-08-06 19:59:22 +00:00
|
|
|
}
|
|
|
|
authenticatorCache[name] = factory
|
2016-08-25 11:25:59 +00:00
|
|
|
return nil
|
2016-08-06 19:59:22 +00:00
|
|
|
}
|
|
|
|
|
2016-09-21 12:39:07 +00:00
|
|
|
func RegisterAuthenticatorConfig(name string, configCreator loader.ConfigCreator) error {
|
|
|
|
return configCache.RegisterCreator(name, configCreator)
|
|
|
|
}
|
|
|
|
|
|
|
|
func CreateAuthenticator(name string, config interface{}) (Authenticator, error) {
|
2016-08-06 19:59:22 +00:00
|
|
|
factory, found := authenticatorCache[name]
|
|
|
|
if !found {
|
2016-08-18 06:34:21 +00:00
|
|
|
return nil, common.ErrObjectNotFound
|
2016-08-06 19:59:22 +00:00
|
|
|
}
|
2016-08-08 20:47:59 +00:00
|
|
|
return factory.Create(config), nil
|
2016-08-06 19:59:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type AuthenticatorChain struct {
|
|
|
|
authenticators []Authenticator
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewAuthenticatorChain(auths ...Authenticator) Authenticator {
|
|
|
|
return &AuthenticatorChain{
|
|
|
|
authenticators: auths,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *AuthenticatorChain) Overhead() int {
|
|
|
|
total := 0
|
|
|
|
for _, auth := range this.authenticators {
|
|
|
|
total += auth.Overhead()
|
|
|
|
}
|
|
|
|
return total
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *AuthenticatorChain) Open(payload *alloc.Buffer) bool {
|
|
|
|
for _, auth := range this.authenticators {
|
|
|
|
if !auth.Open(payload) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *AuthenticatorChain) Seal(payload *alloc.Buffer) {
|
|
|
|
for i := len(this.authenticators) - 1; i >= 0; i-- {
|
|
|
|
auth := this.authenticators[i]
|
|
|
|
auth.Seal(payload)
|
|
|
|
}
|
|
|
|
}
|