open-vault/builtin/logical/database/dbplugin/plugin.go

135 lines
3.8 KiB
Go
Raw Normal View History

2017-04-06 19:20:10 +00:00
package dbplugin
import (
"context"
2017-04-12 23:41:06 +00:00
"fmt"
2017-04-06 19:20:10 +00:00
"net/rpc"
"time"
"google.golang.org/grpc"
2017-04-06 19:20:10 +00:00
"github.com/hashicorp/go-plugin"
"github.com/hashicorp/vault/helper/pluginutil"
log "github.com/mgutz/logxi/v1"
)
2017-04-24 20:59:12 +00:00
// Database is the interface that all database objects must implement.
type Database interface {
2017-04-12 23:41:06 +00:00
Type() (string, error)
CreateUser(ctx context.Context, statements Statements, usernameConfig UsernameConfig, expiration time.Time) (username string, password string, err error)
RenewUser(ctx context.Context, statements Statements, username string, expiration time.Time) error
RevokeUser(ctx context.Context, statements Statements, username string) error
2017-04-06 19:20:10 +00:00
Initialize(ctx context.Context, config map[string]interface{}, verifyConnection bool) error
2017-04-06 19:20:10 +00:00
Close() error
}
// PluginFactory is used to build plugin database types. It wraps the database
// object in a logging and metrics middleware.
func PluginFactory(ctx context.Context, pluginName string, sys pluginutil.LookRunnerUtil, logger log.Logger) (Database, error) {
2017-04-11 18:50:34 +00:00
// Look for plugin in the plugin catalog
pluginRunner, err := sys.LookupPlugin(ctx, pluginName)
2017-04-06 19:20:10 +00:00
if err != nil {
return nil, err
}
var transport string
2017-04-24 20:59:12 +00:00
var db Database
if pluginRunner.Builtin {
// Plugin is builtin so we can retrieve an instance of the interface
2017-04-24 20:59:12 +00:00
// from the pluginRunner. Then cast it to a Database.
dbRaw, err := pluginRunner.BuiltinFactory()
if err != nil {
return nil, fmt.Errorf("error getting plugin type: %s", err)
}
var ok bool
2017-04-24 20:59:12 +00:00
db, ok = dbRaw.(Database)
if !ok {
return nil, fmt.Errorf("unsuported database type: %s", pluginName)
}
transport = "builtin"
} else {
// create a DatabasePluginClient instance
db, err = newPluginClient(ctx, sys, pluginRunner, logger)
if err != nil {
return nil, err
}
// Switch on the underlying database client type to get the transport
// method.
switch db.(*DatabasePluginClient).Database.(type) {
case *gRPCClient:
transport = "gRPC"
case *databasePluginRPCClient:
transport = "netRPC"
}
2017-04-06 19:20:10 +00:00
}
2017-04-12 23:41:06 +00:00
typeStr, err := db.Type()
if err != nil {
return nil, fmt.Errorf("error getting plugin type: %s", err)
}
2017-04-06 19:20:10 +00:00
// Wrap with metrics middleware
db = &databaseMetricsMiddleware{
next: db,
2017-04-12 23:41:06 +00:00
typeStr: typeStr,
2017-04-06 19:20:10 +00:00
}
// Wrap with tracing middleware
if logger.IsTrace() {
db = &databaseTracingMiddleware{
transport: transport,
next: db,
typeStr: typeStr,
logger: logger,
}
2017-04-06 19:20:10 +00:00
}
return db, nil
}
// handshakeConfigs are used to just do a basic handshake between
// a plugin and host. If the handshake fails, a user friendly error is shown.
// This prevents users from executing bad plugins or executing a plugin
// directory. It is a UX feature, not a security feature.
var handshakeConfig = plugin.HandshakeConfig{
2017-08-09 00:01:38 +00:00
ProtocolVersion: 3,
2017-04-06 19:20:10 +00:00
MagicCookieKey: "VAULT_DATABASE_PLUGIN",
MagicCookieValue: "926a0820-aea2-be28-51d6-83cdf00e8edb",
}
var _ plugin.Plugin = &DatabasePlugin{}
var _ plugin.GRPCPlugin = &DatabasePlugin{}
2017-04-11 18:50:34 +00:00
// DatabasePlugin implements go-plugin's Plugin interface. It has methods for
// retrieving a server and a client instance of the plugin.
2017-04-06 19:20:10 +00:00
type DatabasePlugin struct {
2017-04-24 20:59:12 +00:00
impl Database
2017-04-06 19:20:10 +00:00
}
func (d DatabasePlugin) Server(*plugin.MuxBroker) (interface{}, error) {
return &databasePluginRPCServer{impl: d.impl}, nil
}
func (DatabasePlugin) Client(b *plugin.MuxBroker, c *rpc.Client) (interface{}, error) {
return &databasePluginRPCClient{client: c}, nil
}
func (d DatabasePlugin) GRPCServer(_ *plugin.GRPCBroker, s *grpc.Server) error {
RegisterDatabaseServer(s, &gRPCServer{impl: d.impl})
return nil
2017-04-06 19:20:10 +00:00
}
func (DatabasePlugin) GRPCClient(doneCtx context.Context, _ *plugin.GRPCBroker, c *grpc.ClientConn) (interface{}, error) {
return &gRPCClient{
client: NewDatabaseClient(c),
clientConn: c,
doneCtx: doneCtx,
}, nil
2017-04-06 19:20:10 +00:00
}