2020-05-11 12:16:29 +02:00
|
|
|
package model
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2020-08-24 10:06:55 +02:00
|
|
|
"time"
|
|
|
|
|
2020-05-11 12:16:29 +02:00
|
|
|
"github.com/caos/logging"
|
2020-08-24 10:06:55 +02:00
|
|
|
"github.com/lib/pq"
|
|
|
|
|
|
|
|
http_util "github.com/caos/zitadel/internal/api/http"
|
2020-05-11 12:16:29 +02:00
|
|
|
caos_errs "github.com/caos/zitadel/internal/errors"
|
|
|
|
"github.com/caos/zitadel/internal/eventstore/models"
|
|
|
|
"github.com/caos/zitadel/internal/project/model"
|
|
|
|
es_model "github.com/caos/zitadel/internal/project/repository/eventsourcing/model"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
ApplicationKeyID = "id"
|
|
|
|
ApplicationKeyProjectID = "project_id"
|
|
|
|
ApplicationKeyResourceOwner = "resource_owner"
|
|
|
|
ApplicationKeyOIDCClientID = "oidc_client_id"
|
2020-06-05 07:50:04 +02:00
|
|
|
ApplicationKeyName = "app_name"
|
2020-05-11 12:16:29 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type ApplicationView struct {
|
2020-10-16 07:49:38 +02:00
|
|
|
ID string `json:"appId" gorm:"column:id;primary_key"`
|
|
|
|
ProjectID string `json:"-" gorm:"column:project_id"`
|
|
|
|
Name string `json:"name" gorm:"column:app_name"`
|
|
|
|
CreationDate time.Time `json:"-" gorm:"column:creation_date"`
|
|
|
|
ChangeDate time.Time `json:"-" gorm:"column:change_date"`
|
|
|
|
State int32 `json:"-" gorm:"column:app_state"`
|
|
|
|
ProjectRoleAssertion bool `json:"projectRoleAssertion" gorm:"column:project_role_assertion"`
|
|
|
|
ProjectRoleCheck bool `json:"projectRoleCheck" gorm:"column:project_role_check"`
|
2020-05-11 12:16:29 +02:00
|
|
|
|
|
|
|
IsOIDC bool `json:"-" gorm:"column:is_oidc"`
|
2020-08-10 09:34:56 +02:00
|
|
|
OIDCVersion int32 `json:"oidcVersion" gorm:"column:oidc_version"`
|
2020-05-11 12:16:29 +02:00
|
|
|
OIDCClientID string `json:"clientId" gorm:"column:oidc_client_id"`
|
|
|
|
OIDCRedirectUris pq.StringArray `json:"redirectUris" gorm:"column:oidc_redirect_uris"`
|
|
|
|
OIDCResponseTypes pq.Int64Array `json:"responseTypes" gorm:"column:oidc_response_types"`
|
|
|
|
OIDCGrantTypes pq.Int64Array `json:"grantTypes" gorm:"column:oidc_grant_types"`
|
|
|
|
OIDCApplicationType int32 `json:"applicationType" gorm:"column:oidc_application_type"`
|
|
|
|
OIDCAuthMethodType int32 `json:"authMethodType" gorm:"column:oidc_auth_method_type"`
|
|
|
|
OIDCPostLogoutRedirectUris pq.StringArray `json:"postLogoutRedirectUris" gorm:"column:oidc_post_logout_redirect_uris"`
|
2020-08-10 09:34:56 +02:00
|
|
|
NoneCompliant bool `json:"-" gorm:"column:none_compliant"`
|
|
|
|
ComplianceProblems pq.StringArray `json:"-" gorm:"column:compliance_problems"`
|
|
|
|
DevMode bool `json:"devMode" gorm:"column:dev_mode"`
|
2020-08-24 10:06:55 +02:00
|
|
|
OriginAllowList pq.StringArray `json:"-" gorm:"column:origin_allow_list"`
|
2020-10-16 07:49:38 +02:00
|
|
|
AccessTokenType int32 `json:"accessTokenType" gorm:"column:access_token_type"`
|
|
|
|
AccessTokenRoleAssertion bool `json:"accessTokenRoleAssertion" gorm:"column:access_token_role_assertion"`
|
|
|
|
IDTokenRoleAssertion bool `json:"idTokenRoleAssertion" gorm:"column:id_token_role_assertion"`
|
2020-05-11 12:16:29 +02:00
|
|
|
|
|
|
|
Sequence uint64 `json:"-" gorm:"sequence"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func ApplicationViewToModel(app *ApplicationView) *model.ApplicationView {
|
|
|
|
return &model.ApplicationView{
|
2020-10-16 07:49:38 +02:00
|
|
|
ID: app.ID,
|
|
|
|
ProjectID: app.ProjectID,
|
|
|
|
Name: app.Name,
|
|
|
|
State: model.AppState(app.State),
|
|
|
|
Sequence: app.Sequence,
|
|
|
|
CreationDate: app.CreationDate,
|
|
|
|
ChangeDate: app.ChangeDate,
|
|
|
|
ProjectRoleAssertion: app.ProjectRoleAssertion,
|
|
|
|
ProjectRoleCheck: app.ProjectRoleCheck,
|
2020-05-11 12:16:29 +02:00
|
|
|
|
|
|
|
IsOIDC: app.IsOIDC,
|
2020-08-10 09:34:56 +02:00
|
|
|
OIDCVersion: model.OIDCVersion(app.OIDCVersion),
|
2020-05-11 12:16:29 +02:00
|
|
|
OIDCClientID: app.OIDCClientID,
|
|
|
|
OIDCRedirectUris: app.OIDCRedirectUris,
|
|
|
|
OIDCResponseTypes: OIDCResponseTypesToModel(app.OIDCResponseTypes),
|
|
|
|
OIDCGrantTypes: OIDCGrantTypesToModel(app.OIDCGrantTypes),
|
|
|
|
OIDCApplicationType: model.OIDCApplicationType(app.OIDCApplicationType),
|
|
|
|
OIDCAuthMethodType: model.OIDCAuthMethodType(app.OIDCAuthMethodType),
|
|
|
|
OIDCPostLogoutRedirectUris: app.OIDCPostLogoutRedirectUris,
|
2020-08-10 09:34:56 +02:00
|
|
|
NoneCompliant: app.NoneCompliant,
|
|
|
|
ComplianceProblems: app.ComplianceProblems,
|
|
|
|
DevMode: app.DevMode,
|
2020-08-24 10:06:55 +02:00
|
|
|
OriginAllowList: app.OriginAllowList,
|
2020-10-16 07:49:38 +02:00
|
|
|
AccessTokenType: model.OIDCTokenType(app.AccessTokenType),
|
|
|
|
AccessTokenRoleAssertion: app.AccessTokenRoleAssertion,
|
|
|
|
IDTokenRoleAssertion: app.IDTokenRoleAssertion,
|
2020-05-11 12:16:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func OIDCResponseTypesToModel(oidctypes []int64) []model.OIDCResponseType {
|
|
|
|
result := make([]model.OIDCResponseType, len(oidctypes))
|
|
|
|
for i, t := range oidctypes {
|
|
|
|
result[i] = model.OIDCResponseType(t)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func OIDCGrantTypesToModel(granttypes []int64) []model.OIDCGrantType {
|
|
|
|
result := make([]model.OIDCGrantType, len(granttypes))
|
|
|
|
for i, t := range granttypes {
|
|
|
|
result[i] = model.OIDCGrantType(t)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func ApplicationViewsToModel(roles []*ApplicationView) []*model.ApplicationView {
|
|
|
|
result := make([]*model.ApplicationView, len(roles))
|
|
|
|
for i, r := range roles {
|
|
|
|
result[i] = ApplicationViewToModel(r)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2020-08-13 08:28:18 +02:00
|
|
|
func (a *ApplicationView) AppendEventIfMyApp(event *models.Event) (err error) {
|
|
|
|
view := new(ApplicationView)
|
|
|
|
switch event.Type {
|
|
|
|
case es_model.ApplicationAdded:
|
|
|
|
err = view.SetData(event)
|
|
|
|
case es_model.ApplicationChanged,
|
|
|
|
es_model.OIDCConfigAdded,
|
|
|
|
es_model.OIDCConfigChanged,
|
|
|
|
es_model.ApplicationDeactivated,
|
|
|
|
es_model.ApplicationReactivated:
|
|
|
|
err := view.SetData(event)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case es_model.ApplicationRemoved:
|
|
|
|
return view.SetData(event)
|
2020-10-16 07:49:38 +02:00
|
|
|
case es_model.ProjectChanged:
|
|
|
|
return a.AppendEvent(event)
|
2020-08-13 08:28:18 +02:00
|
|
|
case es_model.ProjectRemoved:
|
|
|
|
return a.AppendEvent(event)
|
|
|
|
default:
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if view.ID == a.ID {
|
|
|
|
return a.AppendEvent(event)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2020-10-15 13:52:41 +02:00
|
|
|
|
2020-05-11 12:16:29 +02:00
|
|
|
func (a *ApplicationView) AppendEvent(event *models.Event) (err error) {
|
|
|
|
a.Sequence = event.Sequence
|
|
|
|
a.ChangeDate = event.CreationDate
|
|
|
|
switch event.Type {
|
|
|
|
case es_model.ApplicationAdded:
|
|
|
|
a.setRootData(event)
|
|
|
|
a.CreationDate = event.CreationDate
|
|
|
|
err = a.SetData(event)
|
|
|
|
case es_model.OIDCConfigAdded:
|
|
|
|
a.IsOIDC = true
|
|
|
|
err = a.SetData(event)
|
2020-08-10 09:34:56 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
a.setCompliance()
|
2020-08-24 10:06:55 +02:00
|
|
|
return a.setOriginAllowList()
|
2020-05-11 12:16:29 +02:00
|
|
|
case es_model.OIDCConfigChanged,
|
|
|
|
es_model.ApplicationChanged:
|
|
|
|
err = a.SetData(event)
|
2020-08-10 09:34:56 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
a.setCompliance()
|
2020-08-24 10:06:55 +02:00
|
|
|
return a.setOriginAllowList()
|
2020-10-16 07:49:38 +02:00
|
|
|
case es_model.ProjectChanged:
|
|
|
|
return a.SetData(event)
|
2020-05-11 12:16:29 +02:00
|
|
|
case es_model.ApplicationDeactivated:
|
2020-06-23 14:47:47 +02:00
|
|
|
a.State = int32(model.AppStateInactive)
|
2020-05-11 12:16:29 +02:00
|
|
|
case es_model.ApplicationReactivated:
|
2020-06-23 14:47:47 +02:00
|
|
|
a.State = int32(model.AppStateActive)
|
2020-08-13 08:28:18 +02:00
|
|
|
case es_model.ApplicationRemoved, es_model.ProjectRemoved:
|
|
|
|
a.State = int32(model.AppStateRemoved)
|
2020-05-11 12:16:29 +02:00
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *ApplicationView) setRootData(event *models.Event) {
|
|
|
|
a.ProjectID = event.AggregateID
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *ApplicationView) SetData(event *models.Event) error {
|
|
|
|
if err := json.Unmarshal(event.Data, a); err != nil {
|
|
|
|
logging.Log("EVEN-lo9ds").WithError(err).Error("could not unmarshal event data")
|
|
|
|
return caos_errs.ThrowInternal(err, "MODEL-8suie", "Could not unmarshal data")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2020-08-10 09:34:56 +02:00
|
|
|
|
2020-08-24 10:06:55 +02:00
|
|
|
func (a *ApplicationView) setOriginAllowList() error {
|
|
|
|
allowList := make([]string, 0)
|
|
|
|
for _, redirect := range a.OIDCRedirectUris {
|
|
|
|
origin, err := http_util.GetOriginFromURLString(redirect)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !http_util.IsOriginAllowed(allowList, origin) {
|
|
|
|
allowList = append(allowList, origin)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
a.OriginAllowList = allowList
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-10 09:34:56 +02:00
|
|
|
func (a *ApplicationView) setCompliance() {
|
2020-08-19 09:56:05 +02:00
|
|
|
compliance := model.GetOIDCCompliance(model.OIDCVersion(a.OIDCVersion), model.OIDCApplicationType(a.OIDCApplicationType), OIDCGrantTypesToModel(a.OIDCGrantTypes), OIDCResponseTypesToModel(a.OIDCResponseTypes), model.OIDCAuthMethodType(a.OIDCAuthMethodType), a.OIDCRedirectUris)
|
2020-08-10 09:34:56 +02:00
|
|
|
a.NoneCompliant = compliance.NoneCompliant
|
|
|
|
a.ComplianceProblems = compliance.Problems
|
|
|
|
}
|