mirror of
https://github.com/hashicorp/vault.git
synced 2025-08-12 17:47:02 +02:00
* Lazy load plugins to avoid setup-unwrap cycle * Remove commented blocks * Refactor NewTestCluster, use single core cluster on basic plugin tests * Set c.pluginDirectory in TestAddTestPlugin for setupPluginCatalog to work properly * Add special path to mock plugin * Move ensureCoresSealed to vault/testing.go * Use same method for EnsureCoresSealed and Cleanup * Bump ensureCoresSealed timeout to 60s * Correctly handle nil opts on NewTestCluster * Add metadata flag to APIClientMeta, use meta-enabled plugin when mounting to bootstrap * Check metadata flag directly on the plugin process * Plumb isMetadataMode down to PluginRunner * Add NOOP shims when running in metadata mode * Remove unused flag from the APIMetadata object * Remove setupSecretPlugins and setupCredentialPlugins functions * Move when we setup rollback manager to after the plugins are initialized * Fix tests * Fix merge issue * start rollback manager after the credential setup * Add guards against running certain client and server functions while in metadata mode * Call initialize once a plugin is loaded on the fly * Add more tests, update basic secret/auth plugin tests to trigger lazy loading * Skip mount if plugin removed from catalog * Fixup * Remove commented line on LookupPlugin * Fail on mount operation if plugin is re-added to catalog and mount is on existing path * Check type and special paths on startBackend * Fix merge conflicts * Refactor PluginRunner run methods to use runCommon, fix TestSystemBackend_Plugin_auth
188 lines
4.4 KiB
Go
188 lines
4.4 KiB
Go
package plugin
|
|
|
|
import (
|
|
"errors"
|
|
"net/rpc"
|
|
"os"
|
|
|
|
"github.com/hashicorp/go-plugin"
|
|
"github.com/hashicorp/vault/helper/pluginutil"
|
|
"github.com/hashicorp/vault/logical"
|
|
)
|
|
|
|
var (
|
|
ErrServerInMetadataMode = errors.New("plugin server can not perform action while in metadata mode")
|
|
)
|
|
|
|
// backendPluginServer is the RPC server that backendPluginClient talks to,
|
|
// it methods conforming to requirements by net/rpc
|
|
type backendPluginServer struct {
|
|
broker *plugin.MuxBroker
|
|
backend logical.Backend
|
|
factory func(*logical.BackendConfig) (logical.Backend, error)
|
|
|
|
loggerClient *rpc.Client
|
|
sysViewClient *rpc.Client
|
|
storageClient *rpc.Client
|
|
}
|
|
|
|
func inMetadataMode() bool {
|
|
return os.Getenv(pluginutil.PluginMetadaModeEnv) == "true"
|
|
}
|
|
|
|
func (b *backendPluginServer) HandleRequest(args *HandleRequestArgs, reply *HandleRequestReply) error {
|
|
if inMetadataMode() {
|
|
return ErrServerInMetadataMode
|
|
}
|
|
|
|
storage := &StorageClient{client: b.storageClient}
|
|
args.Request.Storage = storage
|
|
|
|
resp, err := b.backend.HandleRequest(args.Request)
|
|
*reply = HandleRequestReply{
|
|
Response: resp,
|
|
Error: plugin.NewBasicError(err),
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (b *backendPluginServer) SpecialPaths(_ interface{}, reply *SpecialPathsReply) error {
|
|
*reply = SpecialPathsReply{
|
|
Paths: b.backend.SpecialPaths(),
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (b *backendPluginServer) HandleExistenceCheck(args *HandleExistenceCheckArgs, reply *HandleExistenceCheckReply) error {
|
|
if inMetadataMode() {
|
|
return ErrServerInMetadataMode
|
|
}
|
|
|
|
storage := &StorageClient{client: b.storageClient}
|
|
args.Request.Storage = storage
|
|
|
|
checkFound, exists, err := b.backend.HandleExistenceCheck(args.Request)
|
|
*reply = HandleExistenceCheckReply{
|
|
CheckFound: checkFound,
|
|
Exists: exists,
|
|
Error: plugin.NewBasicError(err),
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (b *backendPluginServer) Cleanup(_ interface{}, _ *struct{}) error {
|
|
b.backend.Cleanup()
|
|
|
|
// Close rpc clients
|
|
b.loggerClient.Close()
|
|
b.sysViewClient.Close()
|
|
b.storageClient.Close()
|
|
return nil
|
|
}
|
|
|
|
func (b *backendPluginServer) Initialize(_ interface{}, _ *struct{}) error {
|
|
if inMetadataMode() {
|
|
return ErrServerInMetadataMode
|
|
}
|
|
|
|
err := b.backend.Initialize()
|
|
return err
|
|
}
|
|
|
|
func (b *backendPluginServer) InvalidateKey(args string, _ *struct{}) error {
|
|
if inMetadataMode() {
|
|
return ErrServerInMetadataMode
|
|
}
|
|
|
|
b.backend.InvalidateKey(args)
|
|
return nil
|
|
}
|
|
|
|
// Setup dials into the plugin's broker to get a shimmed storage, logger, and
|
|
// system view of the backend. This method also instantiates the underlying
|
|
// backend through its factory func for the server side of the plugin.
|
|
func (b *backendPluginServer) Setup(args *SetupArgs, reply *SetupReply) error {
|
|
// Dial for storage
|
|
storageConn, err := b.broker.Dial(args.StorageID)
|
|
if err != nil {
|
|
*reply = SetupReply{
|
|
Error: plugin.NewBasicError(err),
|
|
}
|
|
return nil
|
|
}
|
|
rawStorageClient := rpc.NewClient(storageConn)
|
|
b.storageClient = rawStorageClient
|
|
|
|
storage := &StorageClient{client: rawStorageClient}
|
|
|
|
// Dial for logger
|
|
loggerConn, err := b.broker.Dial(args.LoggerID)
|
|
if err != nil {
|
|
*reply = SetupReply{
|
|
Error: plugin.NewBasicError(err),
|
|
}
|
|
return nil
|
|
}
|
|
rawLoggerClient := rpc.NewClient(loggerConn)
|
|
b.loggerClient = rawLoggerClient
|
|
|
|
logger := &LoggerClient{client: rawLoggerClient}
|
|
|
|
// Dial for sys view
|
|
sysViewConn, err := b.broker.Dial(args.SysViewID)
|
|
if err != nil {
|
|
*reply = SetupReply{
|
|
Error: plugin.NewBasicError(err),
|
|
}
|
|
return nil
|
|
}
|
|
rawSysViewClient := rpc.NewClient(sysViewConn)
|
|
b.sysViewClient = rawSysViewClient
|
|
|
|
sysView := &SystemViewClient{client: rawSysViewClient}
|
|
|
|
config := &logical.BackendConfig{
|
|
StorageView: storage,
|
|
Logger: logger,
|
|
System: sysView,
|
|
Config: args.Config,
|
|
}
|
|
|
|
// Call the underlying backend factory after shims have been created
|
|
// to set b.backend
|
|
backend, err := b.factory(config)
|
|
if err != nil {
|
|
*reply = SetupReply{
|
|
Error: plugin.NewBasicError(err),
|
|
}
|
|
}
|
|
b.backend = backend
|
|
|
|
return nil
|
|
}
|
|
|
|
func (b *backendPluginServer) Type(_ interface{}, reply *TypeReply) error {
|
|
*reply = TypeReply{
|
|
Type: b.backend.Type(),
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (b *backendPluginServer) RegisterLicense(args *RegisterLicenseArgs, reply *RegisterLicenseReply) error {
|
|
if inMetadataMode() {
|
|
return ErrServerInMetadataMode
|
|
}
|
|
|
|
err := b.backend.RegisterLicense(args.License)
|
|
if err != nil {
|
|
*reply = RegisterLicenseReply{
|
|
Error: plugin.NewBasicError(err),
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|