// Copyright (c) HashiCorp, Inc. // SPDX-License-Identifier: BUSL-1.1 package database import ( "context" "fmt" "strings" "time" "github.com/hashicorp/go-multierror" "github.com/hashicorp/go-secure-stdlib/strutil" v4 "github.com/hashicorp/vault/sdk/database/dbplugin" v5 "github.com/hashicorp/vault/sdk/database/dbplugin/v5" "github.com/hashicorp/vault/sdk/framework" "github.com/hashicorp/vault/sdk/helper/locksutil" "github.com/hashicorp/vault/sdk/logical" "github.com/hashicorp/vault/sdk/queue" "github.com/robfig/cron/v3" ) func pathListRoles(b *databaseBackend) []*framework.Path { return []*framework.Path{ { Pattern: "roles/?$", DisplayAttrs: &framework.DisplayAttributes{ OperationPrefix: operationPrefixDatabase, OperationVerb: "list", OperationSuffix: "roles", }, Callbacks: map[logical.Operation]framework.OperationFunc{ logical.ListOperation: b.pathRoleList, }, HelpSynopsis: pathRoleHelpSyn, HelpDescription: pathRoleHelpDesc, }, { Pattern: "static-roles/?$", DisplayAttrs: &framework.DisplayAttributes{ OperationPrefix: operationPrefixDatabase, OperationVerb: "list", OperationSuffix: "static-roles", }, Callbacks: map[logical.Operation]framework.OperationFunc{ logical.ListOperation: b.pathRoleList, }, HelpSynopsis: pathStaticRoleHelpSyn, HelpDescription: pathStaticRoleHelpDesc, }, } } func pathRoles(b *databaseBackend) []*framework.Path { return []*framework.Path{ { Pattern: "roles/" + framework.GenericNameRegex("name"), DisplayAttrs: &framework.DisplayAttributes{ OperationPrefix: operationPrefixDatabase, OperationSuffix: "role", }, Fields: fieldsForType(databaseRolePath), ExistenceCheck: b.pathRoleExistenceCheck, Callbacks: map[logical.Operation]framework.OperationFunc{ logical.ReadOperation: b.pathRoleRead, logical.CreateOperation: b.pathRoleCreateUpdate, logical.UpdateOperation: b.pathRoleCreateUpdate, logical.DeleteOperation: b.pathRoleDelete, }, HelpSynopsis: pathRoleHelpSyn, HelpDescription: pathRoleHelpDesc, }, { Pattern: "static-roles/" + framework.GenericNameRegex("name"), DisplayAttrs: &framework.DisplayAttributes{ OperationPrefix: operationPrefixDatabase, OperationSuffix: "static-role", }, Fields: fieldsForType(databaseStaticRolePath), ExistenceCheck: b.pathStaticRoleExistenceCheck, Callbacks: map[logical.Operation]framework.OperationFunc{ logical.ReadOperation: b.pathStaticRoleRead, logical.CreateOperation: b.pathStaticRoleCreateUpdate, logical.UpdateOperation: b.pathStaticRoleCreateUpdate, logical.DeleteOperation: b.pathStaticRoleDelete, }, HelpSynopsis: pathStaticRoleHelpSyn, HelpDescription: pathStaticRoleHelpDesc, }, } } // fieldsForType returns a map of string/FieldSchema items for the given role // type. The purpose is to keep the shared fields between dynamic and static // roles consistent, and allow for each type to override or provide their own // specific fields func fieldsForType(roleType string) map[string]*framework.FieldSchema { fields := map[string]*framework.FieldSchema{ "name": { Type: framework.TypeString, Description: "Name of the role.", }, "db_name": { Type: framework.TypeString, Description: "Name of the database this role acts on.", }, "credential_type": { Type: framework.TypeString, Description: "The type of credential to manage. Options include: " + "'password', 'rsa_private_key'. Defaults to 'password'.", Default: "password", }, "credential_config": { Type: framework.TypeKVPairs, Description: "The configuration for the given credential_type.", }, } // Get the fields that are specific to the type of role, and add them to the // common fields var typeFields map[string]*framework.FieldSchema switch roleType { case databaseStaticRolePath: typeFields = staticFields() default: typeFields = dynamicFields() } for k, v := range typeFields { fields[k] = v } return fields } // dynamicFields returns a map of key and field schema items that are specific // only to dynamic roles func dynamicFields() map[string]*framework.FieldSchema { fields := map[string]*framework.FieldSchema{ "default_ttl": { Type: framework.TypeDurationSecond, Description: "Default ttl for role.", }, "max_ttl": { Type: framework.TypeDurationSecond, Description: "Maximum time a credential is valid for", }, "creation_statements": { Type: framework.TypeStringSlice, Description: `Specifies the database statements executed to create and configure a user. See the plugin's API page for more information on support and formatting for this parameter.`, }, "revocation_statements": { Type: framework.TypeStringSlice, Description: `Specifies the database statements to be executed to revoke a user. See the plugin's API page for more information on support and formatting for this parameter.`, }, "renew_statements": { Type: framework.TypeStringSlice, Description: `Specifies the database statements to be executed to renew a user. Not every plugin type will support this functionality. See the plugin's API page for more information on support and formatting for this parameter. `, }, "rollback_statements": { Type: framework.TypeStringSlice, Description: `Specifies the database statements to be executed rollback a create operation in the event of an error. Not every plugin type will support this functionality. See the plugin's API page for more information on support and formatting for this parameter.`, }, } return fields } // staticFields returns a map of key and field schema items that are specific // only to static roles func staticFields() map[string]*framework.FieldSchema { fields := map[string]*framework.FieldSchema{ "username": { Type: framework.TypeString, Description: `Name of the static user account for Vault to manage. Requires "rotation_period" to be specified`, }, "rotation_period": { Type: framework.TypeDurationSecond, Description: `Period for automatic credential rotation of the given username. Not valid unless used with "username". Mutually exclusive with "rotation_schedule."`, }, "rotation_schedule": { Type: framework.TypeString, Description: `Schedule for automatic credential rotation of the given username. Mutually exclusive with "rotation_period."`, }, "rotation_window": { Type: framework.TypeDurationSecond, Description: `The window of time in which rotations are allowed to occur starting from a given "rotation_schedule". Requires "rotation_schedule" to be specified`, }, "rotation_statements": { Type: framework.TypeStringSlice, Description: `Specifies the database statements to be executed to rotate the accounts credentials. Not every plugin type will support this functionality. See the plugin's API page for more information on support and formatting for this parameter.`, }, "self_managed_password": { Type: framework.TypeString, Description: `Used to connect to a self-managed static account. Must be provided by the user when root credentials are not provided.`, }, } return fields } func (b *databaseBackend) pathRoleExistenceCheck(ctx context.Context, req *logical.Request, data *framework.FieldData) (bool, error) { role, err := b.Role(ctx, req.Storage, data.Get("name").(string)) if err != nil { return false, err } return role != nil, nil } func (b *databaseBackend) pathStaticRoleExistenceCheck(ctx context.Context, req *logical.Request, data *framework.FieldData) (bool, error) { role, err := b.StaticRole(ctx, req.Storage, data.Get("name").(string)) if err != nil { return false, err } return role != nil, nil } func (b *databaseBackend) pathRoleDelete(ctx context.Context, req *logical.Request, data *framework.FieldData) (*logical.Response, error) { name := data.Get("name").(string) err := req.Storage.Delete(ctx, databaseRolePath+name) if err != nil { return nil, err } b.dbEvent(ctx, "role-delete", req.Path, name, true) return nil, nil } func (b *databaseBackend) pathStaticRoleDelete(ctx context.Context, req *logical.Request, data *framework.FieldData) (*logical.Response, error) { name := data.Get("name").(string) // Grab the exclusive lock lock := locksutil.LockForKey(b.roleLocks, name) lock.Lock() defer lock.Unlock() // Remove the item from the queue _, _ = b.popFromRotationQueueByKey(name) err := req.Storage.Delete(ctx, databaseStaticRolePath+name) if err != nil { return nil, err } walIDs, err := framework.ListWAL(ctx, req.Storage) if err != nil { return nil, err } var merr *multierror.Error for _, walID := range walIDs { wal, err := b.findStaticWAL(ctx, req.Storage, walID) if err != nil { merr = multierror.Append(merr, err) continue } if wal != nil && name == wal.RoleName { b.Logger().Debug("deleting WAL for deleted role", "WAL ID", walID, "role", name) err = framework.DeleteWAL(ctx, req.Storage, walID) if err != nil { b.Logger().Debug("failed to delete WAL for deleted role", "WAL ID", walID, "error", err) merr = multierror.Append(merr, err) } } } b.dbEvent(ctx, "static-role-delete", req.Path, name, true) return nil, merr.ErrorOrNil() } func (b *databaseBackend) pathStaticRoleRead(ctx context.Context, req *logical.Request, d *framework.FieldData) (*logical.Response, error) { role, err := b.StaticRole(ctx, req.Storage, d.Get("name").(string)) if err != nil { return nil, err } if role == nil { return nil, nil } data := map[string]interface{}{ "db_name": role.DBName, "rotation_statements": role.Statements.Rotation, "credential_type": role.CredentialType.String(), } // guard against nil StaticAccount; shouldn't happen but we'll be safe if role.StaticAccount != nil { data["username"] = role.StaticAccount.Username data["rotation_statements"] = role.Statements.Rotation if !role.StaticAccount.LastVaultRotation.IsZero() { data["last_vault_rotation"] = role.StaticAccount.LastVaultRotation } // only return one of the mutually exclusive fields in the response if role.StaticAccount.UsesRotationPeriod() { data["rotation_period"] = role.StaticAccount.RotationPeriod.Seconds() } else if role.StaticAccount.UsesRotationSchedule() { data["rotation_schedule"] = role.StaticAccount.RotationSchedule // rotation_window is only valid with rotation_schedule if role.StaticAccount.RotationWindow != 0 { data["rotation_window"] = role.StaticAccount.RotationWindow.Seconds() } } } if len(role.CredentialConfig) > 0 { data["credential_config"] = role.CredentialConfig } if len(role.Statements.Rotation) == 0 { data["rotation_statements"] = []string{} } return &logical.Response{ Data: data, }, nil } func (b *databaseBackend) pathRoleRead(ctx context.Context, req *logical.Request, d *framework.FieldData) (*logical.Response, error) { role, err := b.Role(ctx, req.Storage, d.Get("name").(string)) if err != nil { return nil, err } if role == nil { return nil, nil } data := map[string]interface{}{ "db_name": role.DBName, "creation_statements": role.Statements.Creation, "revocation_statements": role.Statements.Revocation, "rollback_statements": role.Statements.Rollback, "renew_statements": role.Statements.Renewal, "default_ttl": role.DefaultTTL.Seconds(), "max_ttl": role.MaxTTL.Seconds(), "credential_type": role.CredentialType.String(), } if len(role.CredentialConfig) > 0 { data["credential_config"] = role.CredentialConfig } if len(role.Statements.Creation) == 0 { data["creation_statements"] = []string{} } if len(role.Statements.Revocation) == 0 { data["revocation_statements"] = []string{} } if len(role.Statements.Rollback) == 0 { data["rollback_statements"] = []string{} } if len(role.Statements.Renewal) == 0 { data["renew_statements"] = []string{} } return &logical.Response{ Data: data, }, nil } func (b *databaseBackend) pathRoleList(ctx context.Context, req *logical.Request, data *framework.FieldData) (*logical.Response, error) { path := databaseRolePath if strings.HasPrefix(req.Path, "static-roles") { path = databaseStaticRolePath } entries, err := req.Storage.List(ctx, path) if err != nil { return nil, err } return logical.ListResponse(entries), nil } func (b *databaseBackend) pathRoleCreateUpdate(ctx context.Context, req *logical.Request, data *framework.FieldData) (*logical.Response, error) { name := data.Get("name").(string) if name == "" { return logical.ErrorResponse("empty role name attribute given"), nil } exists, err := b.pathStaticRoleExistenceCheck(ctx, req, data) if err != nil { return nil, err } if exists { return logical.ErrorResponse("Role and Static Role names must be unique"), nil } role, err := b.Role(ctx, req.Storage, name) if err != nil { return nil, err } if role == nil { role = &roleEntry{} } createOperation := (req.Operation == logical.CreateOperation) // DB Attributes { if dbNameRaw, ok := data.GetOk("db_name"); ok { role.DBName = dbNameRaw.(string) } else if createOperation { role.DBName = data.Get("db_name").(string) } if role.DBName == "" { return logical.ErrorResponse("database name is required"), nil } if credentialTypeRaw, ok := data.GetOk("credential_type"); ok { credentialType := credentialTypeRaw.(string) if err := role.setCredentialType(credentialType); err != nil { return logical.ErrorResponse(err.Error()), nil } } var credentialConfig map[string]string if raw, ok := data.GetOk("credential_config"); ok { credentialConfig = raw.(map[string]string) } else if req.Operation == logical.CreateOperation { credentialConfig = data.Get("credential_config").(map[string]string) } if err := role.setCredentialConfig(credentialConfig); err != nil { return logical.ErrorResponse("credential_config validation failed: %s", err), nil } } // Statements { if creationStmtsRaw, ok := data.GetOk("creation_statements"); ok { role.Statements.Creation = creationStmtsRaw.([]string) } else if createOperation { role.Statements.Creation = data.Get("creation_statements").([]string) } if revocationStmtsRaw, ok := data.GetOk("revocation_statements"); ok { role.Statements.Revocation = revocationStmtsRaw.([]string) } else if createOperation { role.Statements.Revocation = data.Get("revocation_statements").([]string) } if rollbackStmtsRaw, ok := data.GetOk("rollback_statements"); ok { role.Statements.Rollback = rollbackStmtsRaw.([]string) } else if createOperation { role.Statements.Rollback = data.Get("rollback_statements").([]string) } if renewStmtsRaw, ok := data.GetOk("renew_statements"); ok { role.Statements.Renewal = renewStmtsRaw.([]string) } else if createOperation { role.Statements.Renewal = data.Get("renew_statements").([]string) } // Do not persist deprecated statements that are populated on role read role.Statements.CreationStatements = "" role.Statements.RevocationStatements = "" role.Statements.RenewStatements = "" role.Statements.RollbackStatements = "" } role.Statements.Revocation = strutil.RemoveEmpty(role.Statements.Revocation) // TTLs { if defaultTTLRaw, ok := data.GetOk("default_ttl"); ok { role.DefaultTTL = time.Duration(defaultTTLRaw.(int)) * time.Second } else if createOperation { role.DefaultTTL = time.Duration(data.Get("default_ttl").(int)) * time.Second } if maxTTLRaw, ok := data.GetOk("max_ttl"); ok { role.MaxTTL = time.Duration(maxTTLRaw.(int)) * time.Second } else if createOperation { role.MaxTTL = time.Duration(data.Get("max_ttl").(int)) * time.Second } } // Store it entry, err := logical.StorageEntryJSON(databaseRolePath+name, role) if err != nil { return nil, err } if err := req.Storage.Put(ctx, entry); err != nil { return nil, err } b.dbEvent(ctx, fmt.Sprintf("role-%s", req.Operation), req.Path, name, true) return nil, nil } func (b *databaseBackend) pathStaticRoleCreateUpdate(ctx context.Context, req *logical.Request, data *framework.FieldData) (*logical.Response, error) { response := &logical.Response{} name := data.Get("name").(string) if name == "" { return logical.ErrorResponse("empty role name attribute given"), nil } // Grab the exclusive lock as well potentially pop and re-push the queue item // for this role lock := locksutil.LockForKey(b.roleLocks, name) lock.Lock() defer lock.Unlock() exists, err := b.pathRoleExistenceCheck(ctx, req, data) if err != nil { return nil, err } if exists { return logical.ErrorResponse("Role and Static Role names must be unique"), nil } role, err := b.StaticRole(ctx, req.Storage, data.Get("name").(string)) if err != nil { return nil, err } // createRole is a boolean to indicate if this is a new role creation. This is // can be used later by database plugins that distinguish between creating and // updating roles, and may use separate statements depending on the context. createRole := (req.Operation == logical.CreateOperation) if role == nil { role = &roleEntry{ StaticAccount: &staticAccount{}, } createRole = true } // DB Attributes if dbNameRaw, ok := data.GetOk("db_name"); ok { role.DBName = dbNameRaw.(string) } else if createRole { role.DBName = data.Get("db_name").(string) } if role.DBName == "" { return logical.ErrorResponse("database name is a required field"), nil } username := data.Get("username").(string) if username == "" && createRole { return logical.ErrorResponse("username is a required field to create a static account"), nil } if role.StaticAccount.Username != "" && role.StaticAccount.Username != username { return logical.ErrorResponse("cannot update static account username"), nil } role.StaticAccount.Username = username rotationPeriodSecondsRaw, rotationPeriodOk := data.GetOk("rotation_period") rotationScheduleRaw, rotationScheduleOk := data.GetOk("rotation_schedule") rotationWindowSecondsRaw, rotationWindowOk := data.GetOk("rotation_window") if rotationScheduleOk && rotationPeriodOk { return logical.ErrorResponse("mutually exclusive fields rotation_period and rotation_schedule were both specified; only one of them can be provided"), nil } else if createRole && (!rotationScheduleOk && !rotationPeriodOk) { return logical.ErrorResponse("one of rotation_schedule or rotation_period must be provided to create a static account"), nil } if rotationPeriodOk { rotationPeriodSeconds := rotationPeriodSecondsRaw.(int) if rotationPeriodSeconds < defaultQueueTickSeconds { // If rotation frequency is specified, and this is an update, the value // must be at least that of the queue tick interval (5 seconds at // time of writing), otherwise we wont be able to rotate in time return logical.ErrorResponse(fmt.Sprintf("rotation_period must be %d seconds or more", defaultQueueTickSeconds)), nil } role.StaticAccount.RotationPeriod = time.Duration(rotationPeriodSeconds) * time.Second if rotationWindowOk { return logical.ErrorResponse("rotation_window is invalid with use of rotation_period"), nil } // Unset rotation schedule and window if rotation period is set since // these are mutually exclusive role.StaticAccount.RotationSchedule = "" role.StaticAccount.RotationWindow = 0 } if rotationScheduleOk { rotationSchedule := rotationScheduleRaw.(string) parsedSchedule, err := b.schedule.Parse(rotationSchedule) if err != nil { return logical.ErrorResponse("could not parse rotation_schedule", "error", err), nil } role.StaticAccount.RotationSchedule = rotationSchedule role.StaticAccount.Schedule = *parsedSchedule if rotationWindowOk { rotationWindowSeconds := rotationWindowSecondsRaw.(int) err := b.schedule.ValidateRotationWindow(rotationWindowSeconds) if err != nil { return logical.ErrorResponse("rotation_window is invalid", "error", err), nil } role.StaticAccount.RotationWindow = time.Duration(rotationWindowSeconds) * time.Second } // Unset rotation period if rotation schedule is set since these are // mutually exclusive role.StaticAccount.RotationPeriod = 0 } if rotationStmtsRaw, ok := data.GetOk("rotation_statements"); ok { role.Statements.Rotation = rotationStmtsRaw.([]string) } else if req.Operation == logical.CreateOperation { role.Statements.Rotation = data.Get("rotation_statements").([]string) } if credentialTypeRaw, ok := data.GetOk("credential_type"); ok { credentialType := credentialTypeRaw.(string) if err := role.setCredentialType(credentialType); err != nil { return logical.ErrorResponse(err.Error()), nil } } if smPasswordRaw, ok := data.GetOk("self_managed_password"); ok && createRole { role.StaticAccount.SelfManagedPassword = smPasswordRaw.(string) } var credentialConfig map[string]string if raw, ok := data.GetOk("credential_config"); ok { credentialConfig = raw.(map[string]string) } else if req.Operation == logical.CreateOperation { credentialConfig = data.Get("credential_config").(map[string]string) } if err := role.setCredentialConfig(credentialConfig); err != nil { return logical.ErrorResponse("credential_config validation failed: %s", err), nil } // lvr represents the roles' LastVaultRotation lvr := role.StaticAccount.LastVaultRotation // Only call setStaticAccount if we're creating the role for the // first time var item *queue.Item switch req.Operation { case logical.CreateOperation: // setStaticAccount calls Storage.Put and saves the role to storage resp, err := b.setStaticAccount(ctx, req.Storage, &setStaticAccountInput{ RoleName: name, Role: role, }) if err != nil { if resp != nil && resp.WALID != "" { b.Logger().Debug("deleting WAL for failed role creation", "WAL ID", resp.WALID, "role", name) walDeleteErr := framework.DeleteWAL(ctx, req.Storage, resp.WALID) if walDeleteErr != nil { b.Logger().Debug("failed to delete WAL for failed role creation", "WAL ID", resp.WALID, "error", walDeleteErr) var merr *multierror.Error merr = multierror.Append(merr, err) merr = multierror.Append(merr, fmt.Errorf("failed to clean up WAL from failed role creation: %w", walDeleteErr)) err = merr.ErrorOrNil() } } return nil, err } // guard against RotationTime not being set or zero-value lvr = resp.RotationTime item = &queue.Item{ Key: name, } case logical.UpdateOperation: // store updated Role entry, err := logical.StorageEntryJSON(databaseStaticRolePath+name, role) if err != nil { return nil, err } if err := req.Storage.Put(ctx, entry); err != nil { return nil, err } item, err = b.popFromRotationQueueByKey(name) if err != nil { return nil, err } } if rotationPeriodOk { b.logger.Debug("init priority for RotationPeriod", "lvr", lvr, "next", lvr.Add(role.StaticAccount.RotationPeriod)) item.Priority = lvr.Add(role.StaticAccount.RotationPeriod).Unix() } else if rotationScheduleOk { next := role.StaticAccount.Schedule.Next(lvr) b.logger.Debug("init priority for Schedule", "lvr", lvr, "next", next) item.Priority = next.Unix() } // Add their rotation to the queue if err := b.pushItem(item); err != nil { return nil, err } b.dbEvent(ctx, fmt.Sprintf("static-role-%s", req.Operation), req.Path, name, true) return response, nil } type roleEntry struct { DBName string `json:"db_name"` Statements v4.Statements `json:"statements"` DefaultTTL time.Duration `json:"default_ttl"` MaxTTL time.Duration `json:"max_ttl"` CredentialType v5.CredentialType `json:"credential_type"` CredentialConfig map[string]interface{} `json:"credential_config"` StaticAccount *staticAccount `json:"static_account" mapstructure:"static_account"` } // setCredentialType sets the credential type for the role given its string form. // Returns an error if the given credential type string is unknown. func (r *roleEntry) setCredentialType(credentialType string) error { switch credentialType { case v5.CredentialTypePassword.String(): r.CredentialType = v5.CredentialTypePassword case v5.CredentialTypeRSAPrivateKey.String(): r.CredentialType = v5.CredentialTypeRSAPrivateKey case v5.CredentialTypeClientCertificate.String(): r.CredentialType = v5.CredentialTypeClientCertificate default: return fmt.Errorf("invalid credential_type %q", credentialType) } return nil } // setCredentialConfig validates and sets the credential configuration // for the role using the role's credential type. It will also populate // all default values. Returns an error if the configuration is invalid. func (r *roleEntry) setCredentialConfig(config map[string]string) error { c := make(map[string]interface{}) for k, v := range config { c[k] = v } switch r.CredentialType { case v5.CredentialTypePassword: generator, err := newPasswordGenerator(c) if err != nil { return err } cm, err := generator.configMap() if err != nil { return err } if len(cm) > 0 { r.CredentialConfig = cm } case v5.CredentialTypeRSAPrivateKey: generator, err := newRSAKeyGenerator(c) if err != nil { return err } cm, err := generator.configMap() if err != nil { return err } if len(cm) > 0 { r.CredentialConfig = cm } case v5.CredentialTypeClientCertificate: generator, err := newClientCertificateGenerator(c) if err != nil { return err } cm, err := generator.configMap() if err != nil { return err } if len(cm) > 0 { r.CredentialConfig = cm } } return nil } type staticAccount struct { // Username to create or assume management for static accounts Username string `json:"username"` // SelfManagedPassword is used to make a dedicated connection to the DB // user specified by Username. The credentials will leverage the existing // static role mechanisms to handle password rotations. Required when root // credentials are not provided. SelfManagedPassword string `json:"self_managed_password"` // Password is the current password credential for static accounts. As an input, // this is used/required when trying to assume management of an existing static // account. Returned on credential request if the role's credential type is // CredentialTypePassword. Password string `json:"password"` // PrivateKey is the current private key credential for static accounts. As an input, // this is used/required when trying to assume management of an existing static // account. Returned on credential request if the role's credential type is // CredentialTypeRSAPrivateKey. PrivateKey []byte `json:"private_key"` // LastVaultRotation represents the last time Vault rotated the password LastVaultRotation time.Time `json:"last_vault_rotation"` // NextVaultRotation represents the next time Vault is expected to rotate // the password NextVaultRotation time.Time `json:"next_vault_rotation"` // RotationPeriod is number in seconds between each rotation, effectively a // "time to live". This value is compared to the LastVaultRotation to // determine if a password needs to be rotated RotationPeriod time.Duration `json:"rotation_period"` // RotationSchedule is a "chron style" string representing the allowed // schedule for each rotation. // e.g. "1 0 * * *" would rotate at one minute past midnight (00:01) every // day. RotationSchedule string `json:"rotation_schedule"` // RotationWindow is number in seconds in which rotations are allowed to // occur starting from a given rotation_schedule. RotationWindow time.Duration `json:"rotation_window"` // Schedule holds the parsed "chron style" string representing the allowed // schedule for each rotation. Schedule cron.SpecSchedule `json:"schedule"` // RevokeUser is a boolean flag to indicate if Vault should revoke the // database user when the role is deleted RevokeUserOnDelete bool `json:"revoke_user_on_delete"` } // NextRotationTime calculates the next rotation for period and schedule-based // rotations. // // Period-based expiries are calculated by adding the Rotation Period to the // last known vault rotation. Schedule-based expiries are calculated by // querying for the next schedule expiry since the last known vault rotation. func (s *staticAccount) NextRotationTime() time.Time { if s.UsesRotationPeriod() { return s.LastVaultRotation.Add(s.RotationPeriod) } return s.Schedule.Next(time.Now()) } // NextRotationTimeFromInput calculates the next rotation time for period and // schedule-based roles based on the input. func (s *staticAccount) NextRotationTimeFromInput(input time.Time) time.Time { if s.UsesRotationPeriod() { return input.Add(s.RotationPeriod) } return s.Schedule.Next(input) } // UsesRotationSchedule returns true if the given static account has been // configured to rotate credentials on a schedule (i.e. NOT on a rotation period). func (s *staticAccount) UsesRotationSchedule() bool { return s.RotationSchedule != "" && s.RotationPeriod == 0 } // UsesRotationPeriod returns true if the given static account has been // configured to rotate credentials on a period (i.e. NOT on a rotation schedule). func (s *staticAccount) UsesRotationPeriod() bool { return s.RotationPeriod != 0 && s.RotationSchedule == "" } // IsInsideRotationWindow returns true if the current time t is within a given // static account's rotation window. // // Returns true if the rotation window is not set. In this case, the rotation // window is effectively the span of time between two consecutive rotation // schedules and we should not prevent rotation. func (s *staticAccount) IsInsideRotationWindow(t time.Time) bool { if s.UsesRotationSchedule() && s.RotationWindow != 0 { return t.Before(s.NextVaultRotation.Add(s.RotationWindow)) } return true } // ShouldRotate returns true if a given static account should have its // credentials rotated. // // This will return true when the priority <= the current Unix time. If this // static account is schedule-based with a rotation window, this method will // return false if t is outside the rotation window. func (s *staticAccount) ShouldRotate(priority int64, t time.Time) bool { return priority <= t.Unix() && s.IsInsideRotationWindow(t) } // SetNextVaultRotation func (s *staticAccount) SetNextVaultRotation(t time.Time) { if s.UsesRotationPeriod() { s.NextVaultRotation = t.Add(s.RotationPeriod) } else { s.NextVaultRotation = s.Schedule.Next(t) } } // CredentialTTL calculates the approximate time remaining until the credential is // no longer valid. This is approximate because the rotation expiry is only // checked approximately every 5 seconds, and each rotation can take a small // amount of time to process. This can result in a negative TTL time while the // rotation function processes the Static Role and performs the rotation. If the // TTL is negative, zero is returned. Users should not trust passwords with a // Zero TTL, as they are likely in the process of being rotated and will quickly // be invalidated. func (s *staticAccount) CredentialTTL() time.Duration { next := s.NextRotationTime() ttl := next.Sub(time.Now()).Round(time.Second) if ttl < 0 { ttl = time.Duration(0) } return ttl } const pathRoleHelpSyn = ` Manage the roles that can be created with this backend. ` const pathStaticRoleHelpSyn = ` Manage the static roles that can be created with this backend. ` const pathRoleHelpDesc = ` This path lets you manage the roles that can be created with this backend. The "db_name" parameter is required and configures the name of the database connection to use. The "creation_statements" parameter customizes the string used to create the credentials. This can be a sequence of SQL queries, or other statement formats for a particular database type. Some substitution will be done to the statement strings for certain keys. The names of the variables must be surrounded by "{{" and "}}" to be replaced. * "name" - The random username generated for the DB user. * "password" - The random password generated for the DB user. * "expiration" - The timestamp when this user will expire. Example of a decent creation_statements for a postgresql database plugin: CREATE ROLE "{{name}}" WITH LOGIN PASSWORD '{{password}}' VALID UNTIL '{{expiration}}'; GRANT ALL PRIVILEGES ON ALL TABLES IN SCHEMA public TO "{{name}}"; The "revocation_statements" parameter customizes the statement string used to revoke a user. Example of a decent revocation_statements for a postgresql database plugin: REVOKE ALL PRIVILEGES ON ALL TABLES IN SCHEMA public FROM {{name}}; REVOKE ALL PRIVILEGES ON ALL SEQUENCES IN SCHEMA public FROM {{name}}; REVOKE USAGE ON SCHEMA public FROM {{name}}; DROP ROLE IF EXISTS {{name}}; The "renew_statements" parameter customizes the statement string used to renew a user. The "rollback_statements' parameter customizes the statement string used to rollback a change if needed. ` const pathStaticRoleHelpDesc = ` This path lets you manage the static roles that can be created with this backend. Static Roles are associated with a single database user, and manage the credential based on a rotation period, automatically rotating the credential. The "db_name" parameter is required and configures the name of the database connection to use. The "creation_statements" parameter customizes the string used to create the credentials. This can be a sequence of SQL queries, or other statement formats for a particular database type. Some substitution will be done to the statement strings for certain keys. The names of the variables must be surrounded by "{{" and "}}" to be replaced. * "name" - The random username generated for the DB user. * "password" - The random password generated for the DB user. Populated if the static role's credential_type is 'password'. * "public_key" - The public key generated for the DB user. Populated if the static role's credential_type is 'rsa_private_key'. Example of a decent creation_statements for a postgresql database plugin: CREATE ROLE "{{name}}" WITH LOGIN PASSWORD '{{password}}' GRANT ALL PRIVILEGES ON ALL TABLES IN SCHEMA public TO "{{name}}"; The "revocation_statements" parameter customizes the statement string used to revoke a user. Example of a decent revocation_statements for a postgresql database plugin: REVOKE ALL PRIVILEGES ON ALL TABLES IN SCHEMA public FROM {{name}}; REVOKE ALL PRIVILEGES ON ALL SEQUENCES IN SCHEMA public FROM {{name}}; REVOKE USAGE ON SCHEMA public FROM {{name}}; DROP ROLE IF EXISTS {{name}}; The "renew_statements" parameter customizes the statement string used to renew a user. The "rollback_statements' parameter customizes the statement string used to rollback a change if needed. `