You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
boundary/internal/iam/role_grant_scope.go

292 lines
9.3 KiB

// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: MPL-2.0
package iam
import (
"context"
"fmt"
"strings"
"github.com/hashicorp/boundary/globals"
"github.com/hashicorp/boundary/internal/db"
"github.com/hashicorp/boundary/internal/db/timestamp"
"github.com/hashicorp/boundary/internal/errors"
"github.com/hashicorp/boundary/internal/iam/store"
"github.com/hashicorp/boundary/internal/oplog"
"github.com/hashicorp/boundary/internal/types/resource"
"github.com/hashicorp/boundary/internal/types/scope"
"google.golang.org/protobuf/proto"
)
const (
defaultRoleGrantScopeTable = "iam_role_grant_scope"
defaultGlobalRoleIndividualOrgGrantScopeTable = "iam_role_global_individual_org_grant_scope"
defaultGlobalRoleIndividualProjectGrantScopeTable = "iam_role_global_individual_project_grant_scope"
defaultOrgRoleIndividualGrantScopeTable = "iam_role_org_individual_grant_scope"
)
// ensure that RoleGrantScope implements the interfaces of: Cloneable and db.VetForWriter
var (
_ Cloneable = (*RoleGrantScope)(nil)
_ db.VetForWriter = (*RoleGrantScope)(nil)
_ Cloneable = (*globalRoleIndividualOrgGrantScope)(nil)
_ db.VetForWriter = (*globalRoleIndividualOrgGrantScope)(nil)
_ oplog.ReplayableMessage = (*globalRoleIndividualOrgGrantScope)(nil)
_ roleGrantScoper = (*globalRoleIndividualOrgGrantScope)(nil)
_ Cloneable = (*globalRoleIndividualProjectGrantScope)(nil)
_ db.VetForWriter = (*globalRoleIndividualProjectGrantScope)(nil)
_ oplog.ReplayableMessage = (*globalRoleIndividualProjectGrantScope)(nil)
_ roleGrantScoper = (*globalRoleIndividualProjectGrantScope)(nil)
_ Cloneable = (*orgRoleIndividualGrantScope)(nil)
_ db.VetForWriter = (*orgRoleIndividualGrantScope)(nil)
_ oplog.ReplayableMessage = (*orgRoleIndividualGrantScope)(nil)
_ roleGrantScoper = (*orgRoleIndividualGrantScope)(nil)
)
// roleGrantScoper is an interface for converting internal grantScopeTypes to exported RoleGrantScope
type roleGrantScoper interface {
roleGrantScope() *RoleGrantScope
}
// RoleGrantScope defines the grant scopes that are assigned to a role
type RoleGrantScope struct {
CreateTime *timestamp.Timestamp
RoleId string
ScopeIdOrSpecial string
}
func (r *RoleGrantScope) GetCreateTime() *timestamp.Timestamp {
if r == nil {
return nil
}
return r.CreateTime
}
func (r *RoleGrantScope) GetRoleId() string {
if r == nil {
return ""
}
return r.RoleId
}
func (r *RoleGrantScope) GetScopeIdOrSpecial() string {
if r == nil {
return ""
}
return r.ScopeIdOrSpecial
}
// NewRoleGrantScope creates a new in memory role grant scope. No options are
// supported.
func NewRoleGrantScope(ctx context.Context, roleId string, grantScope string, _ ...Option) (*RoleGrantScope, error) {
const op = "iam.NewRoleGrantScope"
switch {
case roleId == "":
return nil, errors.New(ctx, errors.InvalidParameter, op, "missing role id")
case grantScope == "":
return nil, errors.New(ctx, errors.InvalidParameter, op, "missing grant scope")
case grantScope == scope.Global.String(),
grantScope == globals.GrantScopeThis,
grantScope == globals.GrantScopeChildren,
grantScope == globals.GrantScopeDescendants:
case globals.ResourceInfoFromPrefix(grantScope).Type == resource.Scope:
default:
return nil, errors.New(ctx, errors.InvalidParameter, op, fmt.Sprintf("unknown grant scope id %q", grantScope))
}
rgs := &RoleGrantScope{
RoleId: roleId,
ScopeIdOrSpecial: grantScope,
}
return rgs, nil
}
// Clone creates a clone of the RoleGrantScope
func (g *RoleGrantScope) Clone() any {
return &RoleGrantScope{
CreateTime: g.CreateTime,
RoleId: g.RoleId,
ScopeIdOrSpecial: g.ScopeIdOrSpecial,
}
}
// VetForWrite implements db.VetForWrite() interface
func (g *RoleGrantScope) VetForWrite(ctx context.Context, _ db.Reader, _ db.OpType, _ ...db.Option) error {
const op = "iam.(RoleGrantScope).VetForWrite"
if g.RoleId == "" {
return errors.New(ctx, errors.InvalidParameter, op, "missing role id")
}
if g.ScopeIdOrSpecial == "" {
return errors.New(ctx, errors.InvalidParameter, op, "missing scope id")
}
switch {
case g.ScopeIdOrSpecial == scope.Global.String(),
g.ScopeIdOrSpecial == globals.GrantScopeThis,
g.ScopeIdOrSpecial == globals.GrantScopeChildren,
g.ScopeIdOrSpecial == globals.GrantScopeDescendants:
case globals.ResourceInfoFromPrefix(g.ScopeIdOrSpecial).Type == resource.Scope:
default:
return errors.New(ctx, errors.InvalidParameter, op, fmt.Sprintf("unknown grant scope id %q", g.ScopeIdOrSpecial))
}
return nil
}
// globalRoleIndividualOrgGrantScope defines the grant org scopes
// that are assigned to a global role
type globalRoleIndividualOrgGrantScope struct {
*store.GlobalRoleIndividualOrgGrantScope
tableName string `gorm:"-"`
}
func (g *globalRoleIndividualOrgGrantScope) roleGrantScope() *RoleGrantScope {
if g == nil {
return nil
}
return &RoleGrantScope{
CreateTime: g.CreateTime,
RoleId: g.RoleId,
ScopeIdOrSpecial: g.GetScopeId(),
}
}
func (g *globalRoleIndividualOrgGrantScope) TableName() string {
if g.tableName != "" {
return g.tableName
}
return defaultGlobalRoleIndividualOrgGrantScopeTable
}
func (g *globalRoleIndividualOrgGrantScope) SetTableName(name string) {
g.tableName = name
}
func (g *globalRoleIndividualOrgGrantScope) VetForWrite(ctx context.Context, r db.Reader, opType db.OpType, opt ...db.Option) error {
const op = "iam.(GlobalRoleIndividualOrgGrantScope).VetForWrite"
if g.RoleId == "" {
return errors.New(ctx, errors.InvalidParameter, op, "missing role id")
}
if g.ScopeId == "" {
return errors.New(ctx, errors.InvalidParameter, op, "missing scope id")
}
if globals.ResourceInfoFromPrefix(g.ScopeId).Type != resource.Scope &&
strings.HasPrefix(g.String(), globals.OrgPrefix) {
return errors.New(ctx, errors.InvalidParameter, op, fmt.Sprintf("invalid scope ID %s", g.ScopeId))
}
return nil
}
func (g *globalRoleIndividualOrgGrantScope) Clone() any {
cp := proto.Clone(g.GlobalRoleIndividualOrgGrantScope)
return &globalRoleIndividualOrgGrantScope{
GlobalRoleIndividualOrgGrantScope: cp.(*store.GlobalRoleIndividualOrgGrantScope),
}
}
// globalRoleIndividualProjectGrantScope defines the grant project scopes
// that are assigned to a global role
type globalRoleIndividualProjectGrantScope struct {
*store.GlobalRoleIndividualProjectGrantScope
tableName string `gorm:"-"`
}
func (g *globalRoleIndividualProjectGrantScope) roleGrantScope() *RoleGrantScope {
if g == nil {
return nil
}
return &RoleGrantScope{
CreateTime: g.GetCreateTime(),
RoleId: g.GetRoleId(),
ScopeIdOrSpecial: g.GetScopeId(),
}
}
func (g *globalRoleIndividualProjectGrantScope) TableName() string {
if g.tableName != "" {
return g.tableName
}
return defaultGlobalRoleIndividualProjectGrantScopeTable
}
func (g *globalRoleIndividualProjectGrantScope) SetTableName(name string) {
g.tableName = name
}
func (g *globalRoleIndividualProjectGrantScope) VetForWrite(ctx context.Context, r db.Reader, opType db.OpType, opt ...db.Option) error {
const op = "iam.(GlobalRoleIndividualProjectGrantScope).VetForWrite"
if g.RoleId == "" {
return errors.New(ctx, errors.InvalidParameter, op, "missing role id")
}
if g.ScopeId == "" {
return errors.New(ctx, errors.InvalidParameter, op, "missing scope id")
}
if globals.ResourceInfoFromPrefix(g.ScopeId).Type != resource.Scope &&
strings.HasPrefix(g.String(), globals.ProjectPrefix) {
return errors.New(ctx, errors.InvalidParameter, op, fmt.Sprintf("invalid scope ID %s", g.ScopeId))
}
return nil
}
func (g *globalRoleIndividualProjectGrantScope) Clone() any {
cp := proto.Clone(g.GlobalRoleIndividualProjectGrantScope)
return &globalRoleIndividualProjectGrantScope{
GlobalRoleIndividualProjectGrantScope: cp.(*store.GlobalRoleIndividualProjectGrantScope),
}
}
// OrgRoleIndividualGrantScope defines the grant project scopes
// that are assigned to an org role
type orgRoleIndividualGrantScope struct {
*store.OrgRoleIndividualGrantScope
tableName string `gorm:"-"`
}
func (g *orgRoleIndividualGrantScope) roleGrantScope() *RoleGrantScope {
if g == nil {
return nil
}
return &RoleGrantScope{
CreateTime: g.GetCreateTime(),
RoleId: g.GetRoleId(),
ScopeIdOrSpecial: g.GetScopeId(),
}
}
func (g *orgRoleIndividualGrantScope) TableName() string {
if g.tableName != "" {
return g.tableName
}
return defaultOrgRoleIndividualGrantScopeTable
}
func (g *orgRoleIndividualGrantScope) SetTableName(name string) {
g.tableName = name
}
func (g *orgRoleIndividualGrantScope) VetForWrite(ctx context.Context, r db.Reader, opType db.OpType, opt ...db.Option) error {
const op = "iam.(OrgRoleIndividualGrantScope).VetForWrite"
if g.RoleId == "" {
return errors.New(ctx, errors.InvalidParameter, op, "missing role id")
}
if g.ScopeId == "" {
return errors.New(ctx, errors.InvalidParameter, op, "missing scope id")
}
if globals.ResourceInfoFromPrefix(g.ScopeId).Type != resource.Scope &&
strings.HasPrefix(g.String(), globals.ProjectPrefix) {
return errors.New(ctx, errors.InvalidParameter, op, fmt.Sprintf("invalid scope ID %s", g.ScopeId))
}
return nil
}
func (g *orgRoleIndividualGrantScope) Clone() any {
cp := proto.Clone(g.OrgRoleIndividualGrantScope)
return &orgRoleIndividualGrantScope{
OrgRoleIndividualGrantScope: cp.(*store.OrgRoleIndividualGrantScope),
}
}