2022-09-12 17:18:08 +01:00
|
|
|
package saml
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-08-15 17:04:45 +02:00
|
|
|
"encoding/json"
|
2024-12-19 12:11:40 +01:00
|
|
|
"strings"
|
2022-09-12 17:18:08 +01:00
|
|
|
"time"
|
|
|
|
|
2023-08-15 17:04:45 +02:00
|
|
|
"github.com/dop251/goja"
|
|
|
|
"github.com/zitadel/logging"
|
2022-09-12 17:18:08 +01:00
|
|
|
"github.com/zitadel/saml/pkg/provider"
|
|
|
|
"github.com/zitadel/saml/pkg/provider/key"
|
|
|
|
"github.com/zitadel/saml/pkg/provider/models"
|
|
|
|
"github.com/zitadel/saml/pkg/provider/serviceprovider"
|
|
|
|
"github.com/zitadel/saml/pkg/provider/xml/samlp"
|
|
|
|
|
2023-08-15 17:04:45 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/actions"
|
|
|
|
"github.com/zitadel/zitadel/internal/actions/object"
|
2023-10-25 14:09:15 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/activity"
|
2024-12-19 12:11:40 +01:00
|
|
|
http_utils "github.com/zitadel/zitadel/internal/api/http"
|
2022-09-12 17:18:08 +01:00
|
|
|
"github.com/zitadel/zitadel/internal/api/http/middleware"
|
|
|
|
"github.com/zitadel/zitadel/internal/auth/repository"
|
|
|
|
"github.com/zitadel/zitadel/internal/command"
|
|
|
|
"github.com/zitadel/zitadel/internal/crypto"
|
|
|
|
"github.com/zitadel/zitadel/internal/domain"
|
|
|
|
"github.com/zitadel/zitadel/internal/eventstore"
|
|
|
|
"github.com/zitadel/zitadel/internal/eventstore/handler/crdb"
|
|
|
|
"github.com/zitadel/zitadel/internal/query"
|
|
|
|
"github.com/zitadel/zitadel/internal/telemetry/tracing"
|
2023-12-08 16:30:55 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/zerrors"
|
2022-09-12 17:18:08 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var _ provider.EntityStorage = &Storage{}
|
|
|
|
var _ provider.IdentityProviderStorage = &Storage{}
|
|
|
|
var _ provider.AuthStorage = &Storage{}
|
|
|
|
var _ provider.UserStorage = &Storage{}
|
|
|
|
|
2024-12-19 12:11:40 +01:00
|
|
|
const (
|
|
|
|
LoginClientHeader = "x-zitadel-login-client"
|
|
|
|
)
|
|
|
|
|
2022-09-12 17:18:08 +01:00
|
|
|
type Storage struct {
|
|
|
|
certChan <-chan interface{}
|
|
|
|
defaultCertificateLifetime time.Duration
|
|
|
|
|
|
|
|
currentCACertificate query.Certificate
|
|
|
|
currentMetadataCertificate query.Certificate
|
|
|
|
currentResponseCertificate query.Certificate
|
|
|
|
|
|
|
|
locker crdb.Locker
|
|
|
|
certificateAlgorithm string
|
|
|
|
encAlg crypto.EncryptionAlgorithm
|
|
|
|
certEncAlg crypto.EncryptionAlgorithm
|
|
|
|
|
|
|
|
eventstore *eventstore.Eventstore
|
|
|
|
repo repository.Repository
|
|
|
|
command *command.Commands
|
|
|
|
query *query.Queries
|
|
|
|
|
2024-12-19 12:11:40 +01:00
|
|
|
defaultLoginURL string
|
|
|
|
defaultLoginURLv2 string
|
2022-09-12 17:18:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) GetEntityByID(ctx context.Context, entityID string) (*serviceprovider.ServiceProvider, error) {
|
2024-09-17 13:34:14 +02:00
|
|
|
app, err := p.query.ActiveAppBySAMLEntityID(ctx, entityID)
|
2022-09-12 17:18:08 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return serviceprovider.NewServiceProvider(
|
|
|
|
app.ID,
|
|
|
|
&serviceprovider.Config{
|
|
|
|
Metadata: app.SAMLConfig.Metadata,
|
|
|
|
},
|
2024-12-19 12:11:40 +01:00
|
|
|
func(id string) string {
|
|
|
|
if strings.HasPrefix(id, command.IDPrefixV2) {
|
|
|
|
return p.defaultLoginURLv2 + id
|
|
|
|
}
|
|
|
|
return p.defaultLoginURL + id
|
|
|
|
},
|
2022-09-12 17:18:08 +01:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) GetEntityIDByAppID(ctx context.Context, appID string) (string, error) {
|
2024-09-17 13:34:14 +02:00
|
|
|
app, err := p.query.AppByID(ctx, appID, true)
|
2022-09-12 17:18:08 +01:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return app.SAMLConfig.EntityID, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) Health(context.Context) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) GetCA(ctx context.Context) (*key.CertificateAndKey, error) {
|
2024-08-14 17:18:14 +03:00
|
|
|
return p.GetCertificateAndKey(ctx, crypto.KeyUsageSAMLCA)
|
2022-09-12 17:18:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) GetMetadataSigningKey(ctx context.Context) (*key.CertificateAndKey, error) {
|
2024-08-14 17:18:14 +03:00
|
|
|
return p.GetCertificateAndKey(ctx, crypto.KeyUsageSAMLMetadataSigning)
|
2022-09-12 17:18:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) GetResponseSigningKey(ctx context.Context) (*key.CertificateAndKey, error) {
|
2024-08-14 17:18:14 +03:00
|
|
|
return p.GetCertificateAndKey(ctx, crypto.KeyUsageSAMLResponseSinging)
|
2022-09-12 17:18:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) CreateAuthRequest(ctx context.Context, req *samlp.AuthnRequestType, acsUrl, protocolBinding, relayState, applicationID string) (_ models.AuthRequestInt, err error) {
|
|
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
|
|
defer func() { span.EndWithError(err) }()
|
2024-12-19 12:11:40 +01:00
|
|
|
|
|
|
|
headers, _ := http_utils.HeadersFromCtx(ctx)
|
|
|
|
if loginClient := headers.Get(LoginClientHeader); loginClient != "" {
|
|
|
|
return p.createAuthRequestLoginClient(ctx, req, acsUrl, protocolBinding, relayState, applicationID, loginClient)
|
|
|
|
}
|
|
|
|
return p.createAuthRequest(ctx, req, acsUrl, protocolBinding, relayState, applicationID)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) createAuthRequestLoginClient(ctx context.Context, req *samlp.AuthnRequestType, acsUrl, protocolBinding, relayState, applicationID, loginClient string) (_ models.AuthRequestInt, err error) {
|
|
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
|
|
defer func() { span.EndWithError(err) }()
|
|
|
|
samlRequest := &command.SAMLRequest{
|
|
|
|
ApplicationID: applicationID,
|
|
|
|
ACSURL: acsUrl,
|
|
|
|
RelayState: relayState,
|
|
|
|
RequestID: req.Id,
|
|
|
|
Binding: protocolBinding,
|
|
|
|
Issuer: req.Issuer.Text,
|
|
|
|
Destination: req.Destination,
|
|
|
|
LoginClient: loginClient,
|
|
|
|
}
|
|
|
|
|
|
|
|
aar, err := p.command.AddSAMLRequest(ctx, samlRequest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &AuthRequestV2{aar}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) createAuthRequest(ctx context.Context, req *samlp.AuthnRequestType, acsUrl, protocolBinding, relayState, applicationID string) (_ models.AuthRequestInt, err error) {
|
|
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
|
|
defer func() { span.EndWithError(err) }()
|
2022-09-12 17:18:08 +01:00
|
|
|
userAgentID, ok := middleware.UserAgentIDFromCtx(ctx)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowPreconditionFailed(nil, "SAML-sd436", "no user agent id")
|
2022-09-12 17:18:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
authRequest := CreateAuthRequestToBusiness(ctx, req, acsUrl, protocolBinding, applicationID, relayState, userAgentID)
|
|
|
|
|
|
|
|
resp, err := p.repo.CreateAuthRequest(ctx, authRequest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return AuthRequestFromBusiness(resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) AuthRequestByID(ctx context.Context, id string) (_ models.AuthRequestInt, err error) {
|
|
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
|
|
defer func() { span.EndWithError(err) }()
|
2024-12-19 12:11:40 +01:00
|
|
|
|
|
|
|
if strings.HasPrefix(id, command.IDPrefixV2) {
|
|
|
|
req, err := p.command.GetCurrentSAMLRequest(ctx, id)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &AuthRequestV2{req}, nil
|
|
|
|
}
|
|
|
|
|
2022-09-12 17:18:08 +01:00
|
|
|
userAgentID, ok := middleware.UserAgentIDFromCtx(ctx)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowPreconditionFailed(nil, "SAML-D3g21", "no user agent id")
|
2022-09-12 17:18:08 +01:00
|
|
|
}
|
|
|
|
resp, err := p.repo.AuthRequestByIDCheckLoggedIn(ctx, id, userAgentID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return AuthRequestFromBusiness(resp)
|
|
|
|
}
|
|
|
|
|
2023-08-15 17:04:45 +02:00
|
|
|
func (p *Storage) SetUserinfoWithUserID(ctx context.Context, applicationID string, userinfo models.AttributeSetter, userID string, attributes []int) (err error) {
|
2022-09-12 17:18:08 +01:00
|
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
|
|
defer func() { span.EndWithError(err) }()
|
2023-11-21 14:11:38 +02:00
|
|
|
user, err := p.query.GetUserByID(ctx, true, userID)
|
2022-09-12 17:18:08 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-09-17 15:21:49 +02:00
|
|
|
if user.State != domain.UserStateActive {
|
|
|
|
return zerrors.ThrowPreconditionFailed(nil, "SAML-S3gFd", "Errors.User.NotActive")
|
|
|
|
}
|
2022-09-12 17:18:08 +01:00
|
|
|
|
2023-08-15 17:04:45 +02:00
|
|
|
userGrants, err := p.getGrants(ctx, userID, applicationID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
customAttributes, err := p.getCustomAttributes(ctx, user, userGrants)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
setUserinfo(user, userinfo, attributes, customAttributes)
|
2023-10-25 14:09:15 +02:00
|
|
|
|
|
|
|
// trigger activity log for authentication for user
|
2024-03-14 09:49:10 +01:00
|
|
|
activity.Trigger(ctx, user.ResourceOwner, user.ID, activity.SAMLResponse, p.eventstore.FilterToQueryReducer)
|
2022-09-12 17:18:08 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) SetUserinfoWithLoginName(ctx context.Context, userinfo models.AttributeSetter, loginName string, attributes []int) (err error) {
|
|
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
|
|
defer func() { span.EndWithError(err) }()
|
|
|
|
|
2023-12-08 13:14:22 +01:00
|
|
|
user, err := p.query.GetUserByLoginName(ctx, true, loginName)
|
2022-09-12 17:18:08 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-09-17 15:21:49 +02:00
|
|
|
if user.State != domain.UserStateActive {
|
|
|
|
return zerrors.ThrowPreconditionFailed(nil, "SAML-FJ262", "Errors.User.NotActive")
|
|
|
|
}
|
2022-09-12 17:18:08 +01:00
|
|
|
|
2023-08-15 17:04:45 +02:00
|
|
|
setUserinfo(user, userinfo, attributes, map[string]*customAttribute{})
|
2022-09-12 17:18:08 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-08-15 17:04:45 +02:00
|
|
|
func setUserinfo(user *query.User, userinfo models.AttributeSetter, attributes []int, customAttributes map[string]*customAttribute) {
|
|
|
|
for name, attr := range customAttributes {
|
|
|
|
userinfo.SetCustomAttribute(name, "", attr.nameFormat, attr.attributeValue)
|
|
|
|
}
|
2022-09-12 17:18:08 +01:00
|
|
|
if len(attributes) == 0 {
|
|
|
|
userinfo.SetUsername(user.PreferredLoginName)
|
|
|
|
userinfo.SetUserID(user.ID)
|
|
|
|
if user.Human == nil {
|
|
|
|
return
|
|
|
|
}
|
2023-03-14 20:20:38 +01:00
|
|
|
userinfo.SetEmail(string(user.Human.Email))
|
2022-09-12 17:18:08 +01:00
|
|
|
userinfo.SetSurname(user.Human.LastName)
|
|
|
|
userinfo.SetGivenName(user.Human.FirstName)
|
|
|
|
userinfo.SetFullName(user.Human.DisplayName)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for _, attribute := range attributes {
|
|
|
|
switch attribute {
|
|
|
|
case provider.AttributeEmail:
|
|
|
|
if user.Human != nil {
|
2023-03-14 20:20:38 +01:00
|
|
|
userinfo.SetEmail(string(user.Human.Email))
|
2022-09-12 17:18:08 +01:00
|
|
|
}
|
|
|
|
case provider.AttributeSurname:
|
|
|
|
if user.Human != nil {
|
|
|
|
userinfo.SetSurname(user.Human.LastName)
|
|
|
|
}
|
|
|
|
case provider.AttributeFullName:
|
|
|
|
if user.Human != nil {
|
|
|
|
userinfo.SetFullName(user.Human.DisplayName)
|
|
|
|
}
|
|
|
|
case provider.AttributeGivenName:
|
|
|
|
if user.Human != nil {
|
|
|
|
userinfo.SetGivenName(user.Human.FirstName)
|
|
|
|
}
|
|
|
|
case provider.AttributeUsername:
|
|
|
|
userinfo.SetUsername(user.PreferredLoginName)
|
|
|
|
case provider.AttributeUserID:
|
|
|
|
userinfo.SetUserID(user.ID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-08-15 17:04:45 +02:00
|
|
|
|
|
|
|
func (p *Storage) getCustomAttributes(ctx context.Context, user *query.User, userGrants *query.UserGrants) (map[string]*customAttribute, error) {
|
|
|
|
customAttributes := make(map[string]*customAttribute, 0)
|
2023-11-20 16:21:08 +01:00
|
|
|
queriedActions, err := p.query.GetActiveActionsByFlowAndTriggerType(ctx, domain.FlowTypeCustomizeSAMLResponse, domain.TriggerTypePreSAMLResponseCreation, user.ResourceOwner)
|
2023-08-15 17:04:45 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
ctxFields := actions.SetContextFields(
|
|
|
|
actions.SetFields("v1",
|
|
|
|
actions.SetFields("getUser", func(c *actions.FieldConfig) interface{} {
|
|
|
|
return func(call goja.FunctionCall) goja.Value {
|
|
|
|
return object.UserFromQuery(c, user)
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
actions.SetFields("user",
|
|
|
|
actions.SetFields("getMetadata", func(c *actions.FieldConfig) interface{} {
|
|
|
|
return func(goja.FunctionCall) goja.Value {
|
|
|
|
resourceOwnerQuery, err := query.NewUserMetadataResourceOwnerSearchQuery(user.ResourceOwner)
|
|
|
|
if err != nil {
|
|
|
|
logging.WithError(err).Debug("unable to create search query")
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
metadata, err := p.query.SearchUserMetadata(
|
|
|
|
ctx,
|
|
|
|
true,
|
|
|
|
user.ID,
|
|
|
|
&query.UserMetadataSearchQueries{Queries: []query.SearchQuery{resourceOwnerQuery}},
|
|
|
|
false,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
logging.WithError(err).Info("unable to get md in action")
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return object.UserMetadataListFromQuery(c, metadata)
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
actions.SetFields("grants", func(c *actions.FieldConfig) interface{} {
|
2024-04-22 13:34:23 +02:00
|
|
|
return object.UserGrantsFromQuery(ctx, p.query, c, userGrants)
|
2023-08-15 17:04:45 +02:00
|
|
|
}),
|
|
|
|
),
|
2024-01-26 09:56:10 +01:00
|
|
|
actions.SetFields("org",
|
|
|
|
actions.SetFields("getMetadata", func(c *actions.FieldConfig) interface{} {
|
|
|
|
return func(goja.FunctionCall) goja.Value {
|
2024-04-22 13:34:23 +02:00
|
|
|
return object.GetOrganizationMetadata(ctx, p.query, c, user.ResourceOwner)
|
2024-01-26 09:56:10 +01:00
|
|
|
}
|
|
|
|
}),
|
|
|
|
),
|
2023-08-15 17:04:45 +02:00
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
for _, action := range queriedActions {
|
|
|
|
actionCtx, cancel := context.WithTimeout(ctx, action.Timeout())
|
|
|
|
|
|
|
|
apiFields := actions.WithAPIFields(
|
|
|
|
actions.SetFields("v1",
|
|
|
|
actions.SetFields("attributes",
|
|
|
|
actions.SetFields("setCustomAttribute", func(name string, nameFormat string, attributeValue ...string) {
|
|
|
|
if _, ok := customAttributes[name]; !ok {
|
|
|
|
customAttributes = appendCustomAttribute(customAttributes, name, nameFormat, attributeValue)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
),
|
|
|
|
actions.SetFields("user",
|
2023-08-21 14:21:45 +02:00
|
|
|
actions.SetFields("setMetadata", func(call goja.FunctionCall) goja.Value {
|
2023-08-15 17:04:45 +02:00
|
|
|
if len(call.Arguments) != 2 {
|
|
|
|
panic("exactly 2 (key, value) arguments expected")
|
|
|
|
}
|
|
|
|
key := call.Arguments[0].Export().(string)
|
|
|
|
val := call.Arguments[1].Export()
|
|
|
|
|
|
|
|
value, err := json.Marshal(val)
|
|
|
|
if err != nil {
|
|
|
|
logging.WithError(err).Debug("unable to marshal")
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
metadata := &domain.Metadata{
|
|
|
|
Key: key,
|
|
|
|
Value: value,
|
|
|
|
}
|
|
|
|
if _, err = p.command.SetUserMetadata(ctx, metadata, user.ID, user.ResourceOwner); err != nil {
|
|
|
|
logging.WithError(err).Info("unable to set md in action")
|
|
|
|
panic(err)
|
|
|
|
}
|
2023-08-21 14:21:45 +02:00
|
|
|
return nil
|
2023-08-15 17:04:45 +02:00
|
|
|
}),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
err = actions.Run(
|
|
|
|
actionCtx,
|
|
|
|
ctxFields,
|
|
|
|
apiFields,
|
|
|
|
action.Script,
|
|
|
|
action.Name,
|
|
|
|
append(actions.ActionToOptions(action), actions.WithHTTP(actionCtx))...,
|
|
|
|
)
|
|
|
|
cancel()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return customAttributes, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Storage) getGrants(ctx context.Context, userID, applicationID string) (*query.UserGrants, error) {
|
2023-11-21 14:11:38 +02:00
|
|
|
projectID, err := p.query.ProjectIDFromClientID(ctx, applicationID)
|
2023-08-15 17:04:45 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
projectQuery, err := query.NewUserGrantProjectIDSearchQuery(projectID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
userIDQuery, err := query.NewUserGrantUserIDSearchQuery(userID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-09-17 14:18:29 +02:00
|
|
|
activeQuery, err := query.NewUserGrantStateQuery(domain.UserGrantStateActive)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-08-15 17:04:45 +02:00
|
|
|
return p.query.UserGrants(ctx, &query.UserGrantsQueries{
|
|
|
|
Queries: []query.SearchQuery{
|
|
|
|
projectQuery,
|
|
|
|
userIDQuery,
|
2024-09-17 14:18:29 +02:00
|
|
|
activeQuery,
|
2023-08-15 17:04:45 +02:00
|
|
|
},
|
2024-01-08 16:26:30 +01:00
|
|
|
}, true)
|
2023-08-15 17:04:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type customAttribute struct {
|
|
|
|
nameFormat string
|
|
|
|
attributeValue []string
|
|
|
|
}
|
|
|
|
|
|
|
|
func appendCustomAttribute(customAttributes map[string]*customAttribute, name string, nameFormat string, attributeValue []string) map[string]*customAttribute {
|
|
|
|
if customAttributes == nil {
|
|
|
|
customAttributes = make(map[string]*customAttribute)
|
|
|
|
}
|
|
|
|
customAttributes[name] = &customAttribute{
|
|
|
|
nameFormat: nameFormat,
|
|
|
|
attributeValue: attributeValue,
|
|
|
|
}
|
|
|
|
return customAttributes
|
|
|
|
}
|