2025-09-05 10:54:34 +02:00
|
|
|
package repository
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"context"
|
|
|
|
|
|
|
|
|
|
"github.com/zitadel/zitadel/backend/v3/domain"
|
|
|
|
|
"github.com/zitadel/zitadel/backend/v3/storage/database"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
// repository
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
var _ domain.OrganizationRepository = (*org)(nil)
|
|
|
|
|
|
|
|
|
|
type org struct {
|
|
|
|
|
repository
|
|
|
|
|
shouldLoadDomains bool
|
|
|
|
|
domainRepo domain.OrganizationDomainRepository
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func OrganizationRepository(client database.QueryExecutor) domain.OrganizationRepository {
|
|
|
|
|
return &org{
|
|
|
|
|
repository: repository{
|
|
|
|
|
client: client,
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const queryOrganizationStmt = `SELECT organizations.id, organizations.name, organizations.instance_id, organizations.state, organizations.created_at, organizations.updated_at` +
|
2025-09-08 09:35:31 +02:00
|
|
|
` , jsonb_agg(json_build_object('domain', org_domains.domain, 'isVerified', org_domains.is_verified, 'isPrimary', org_domains.is_primary, 'validationType', org_domains.validation_type, 'createdAt', org_domains.created_at, 'updatedAt', org_domains.updated_at)) FILTER (WHERE org_domains.org_id IS NOT NULL) AS domains` +
|
2025-09-05 10:54:34 +02:00
|
|
|
` FROM zitadel.organizations`
|
|
|
|
|
|
|
|
|
|
// Get implements [domain.OrganizationRepository].
|
|
|
|
|
func (o *org) Get(ctx context.Context, opts ...database.QueryOption) (*domain.Organization, error) {
|
|
|
|
|
opts = append(opts,
|
|
|
|
|
o.joinDomains(),
|
|
|
|
|
database.WithGroupBy(o.InstanceIDColumn(), o.IDColumn()),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
options := new(database.QueryOpts)
|
|
|
|
|
for _, opt := range opts {
|
|
|
|
|
opt(options)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var builder database.StatementBuilder
|
|
|
|
|
builder.WriteString(queryOrganizationStmt)
|
|
|
|
|
options.Write(&builder)
|
|
|
|
|
|
|
|
|
|
return scanOrganization(ctx, o.client, &builder)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// List implements [domain.OrganizationRepository].
|
|
|
|
|
func (o *org) List(ctx context.Context, opts ...database.QueryOption) ([]*domain.Organization, error) {
|
|
|
|
|
opts = append(opts,
|
|
|
|
|
o.joinDomains(),
|
|
|
|
|
database.WithGroupBy(o.InstanceIDColumn(), o.IDColumn()),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
options := new(database.QueryOpts)
|
|
|
|
|
for _, opt := range opts {
|
|
|
|
|
opt(options)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var builder database.StatementBuilder
|
|
|
|
|
builder.WriteString(queryOrganizationStmt)
|
|
|
|
|
options.Write(&builder)
|
|
|
|
|
|
|
|
|
|
return scanOrganizations(ctx, o.client, &builder)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (o *org) joinDomains() database.QueryOption {
|
|
|
|
|
columns := make([]database.Condition, 0, 3)
|
|
|
|
|
columns = append(columns,
|
|
|
|
|
database.NewColumnCondition(o.InstanceIDColumn(), o.Domains(false).InstanceIDColumn()),
|
|
|
|
|
database.NewColumnCondition(o.IDColumn(), o.Domains(false).OrgIDColumn()),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// If domains should not be joined, we make sure to return null for the domain columns
|
|
|
|
|
// the query optimizer of the dialect should optimize this away if no domains are requested
|
|
|
|
|
if !o.shouldLoadDomains {
|
|
|
|
|
columns = append(columns, database.IsNull(o.domainRepo.OrgIDColumn()))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return database.WithLeftJoin(
|
|
|
|
|
"zitadel.org_domains",
|
|
|
|
|
database.And(columns...),
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const createOrganizationStmt = `INSERT INTO zitadel.organizations (id, name, instance_id, state)` +
|
|
|
|
|
` VALUES ($1, $2, $3, $4)` +
|
|
|
|
|
` RETURNING created_at, updated_at`
|
|
|
|
|
|
|
|
|
|
// Create implements [domain.OrganizationRepository].
|
|
|
|
|
func (o *org) Create(ctx context.Context, organization *domain.Organization) error {
|
|
|
|
|
builder := database.StatementBuilder{}
|
|
|
|
|
builder.AppendArgs(organization.ID, organization.Name, organization.InstanceID, organization.State)
|
|
|
|
|
builder.WriteString(createOrganizationStmt)
|
|
|
|
|
|
|
|
|
|
return o.client.QueryRow(ctx, builder.String(), builder.Args()...).Scan(&organization.CreatedAt, &organization.UpdatedAt)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Update implements [domain.OrganizationRepository].
|
|
|
|
|
func (o *org) Update(ctx context.Context, id domain.OrgIdentifierCondition, instanceID string, changes ...database.Change) (int64, error) {
|
|
|
|
|
if len(changes) == 0 {
|
|
|
|
|
return 0, database.ErrNoChanges
|
|
|
|
|
}
|
|
|
|
|
builder := database.StatementBuilder{}
|
|
|
|
|
builder.WriteString(`UPDATE zitadel.organizations SET `)
|
|
|
|
|
|
|
|
|
|
instanceIDCondition := o.InstanceIDCondition(instanceID)
|
|
|
|
|
|
|
|
|
|
conditions := []database.Condition{id, instanceIDCondition}
|
|
|
|
|
database.Changes(changes).Write(&builder)
|
|
|
|
|
writeCondition(&builder, database.And(conditions...))
|
|
|
|
|
|
|
|
|
|
stmt := builder.String()
|
|
|
|
|
|
|
|
|
|
rowsAffected, err := o.client.Exec(ctx, stmt, builder.Args()...)
|
|
|
|
|
return rowsAffected, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Delete implements [domain.OrganizationRepository].
|
|
|
|
|
func (o *org) Delete(ctx context.Context, id domain.OrgIdentifierCondition, instanceID string) (int64, error) {
|
|
|
|
|
builder := database.StatementBuilder{}
|
|
|
|
|
|
|
|
|
|
builder.WriteString(`DELETE FROM zitadel.organizations`)
|
|
|
|
|
|
|
|
|
|
instanceIDCondition := o.InstanceIDCondition(instanceID)
|
|
|
|
|
|
|
|
|
|
conditions := []database.Condition{id, instanceIDCondition}
|
|
|
|
|
writeCondition(&builder, database.And(conditions...))
|
|
|
|
|
|
|
|
|
|
return o.client.Exec(ctx, builder.String(), builder.Args()...)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
// changes
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
// SetName implements [domain.organizationChanges].
|
|
|
|
|
func (o org) SetName(name string) database.Change {
|
|
|
|
|
return database.NewChange(o.NameColumn(), name)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// SetState implements [domain.organizationChanges].
|
|
|
|
|
func (o org) SetState(state domain.OrgState) database.Change {
|
|
|
|
|
return database.NewChange(o.StateColumn(), state)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
// conditions
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
// IDCondition implements [domain.organizationConditions].
|
|
|
|
|
func (o org) IDCondition(id string) domain.OrgIdentifierCondition {
|
|
|
|
|
return database.NewTextCondition(o.IDColumn(), database.TextOperationEqual, id)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NameCondition implements [domain.organizationConditions].
|
|
|
|
|
func (o org) NameCondition(name string) domain.OrgIdentifierCondition {
|
|
|
|
|
return database.NewTextCondition(o.NameColumn(), database.TextOperationEqual, name)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// InstanceIDCondition implements [domain.organizationConditions].
|
|
|
|
|
func (o org) InstanceIDCondition(instanceID string) database.Condition {
|
|
|
|
|
return database.NewTextCondition(o.InstanceIDColumn(), database.TextOperationEqual, instanceID)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// StateCondition implements [domain.organizationConditions].
|
|
|
|
|
func (o org) StateCondition(state domain.OrgState) database.Condition {
|
|
|
|
|
return database.NewTextCondition(o.StateColumn(), database.TextOperationEqual, state.String())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
// columns
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
// IDColumn implements [domain.organizationColumns].
|
|
|
|
|
func (org) IDColumn() database.Column {
|
|
|
|
|
return database.NewColumn("organizations", "id")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NameColumn implements [domain.organizationColumns].
|
|
|
|
|
func (org) NameColumn() database.Column {
|
|
|
|
|
return database.NewColumn("organizations", "name")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// InstanceIDColumn implements [domain.organizationColumns].
|
|
|
|
|
func (org) InstanceIDColumn() database.Column {
|
|
|
|
|
return database.NewColumn("organizations", "instance_id")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// StateColumn implements [domain.organizationColumns].
|
|
|
|
|
func (org) StateColumn() database.Column {
|
|
|
|
|
return database.NewColumn("organizations", "state")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// CreatedAtColumn implements [domain.organizationColumns].
|
|
|
|
|
func (org) CreatedAtColumn() database.Column {
|
|
|
|
|
return database.NewColumn("organizations", "created_at")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// UpdatedAtColumn implements [domain.organizationColumns].
|
|
|
|
|
func (org) UpdatedAtColumn() database.Column {
|
|
|
|
|
return database.NewColumn("organizations", "updated_at")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
// scanners
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
|
2025-09-08 09:35:31 +02:00
|
|
|
type rawOrg struct {
|
2025-09-05 10:54:34 +02:00
|
|
|
*domain.Organization
|
2025-09-08 09:35:31 +02:00
|
|
|
Domains JSONArray[domain.OrganizationDomain] `json:"domains,omitempty" db:"domains"`
|
2025-09-05 10:54:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func scanOrganization(ctx context.Context, querier database.Querier, builder *database.StatementBuilder) (*domain.Organization, error) {
|
|
|
|
|
rows, err := querier.Query(ctx, builder.String(), builder.Args()...)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-08 09:35:31 +02:00
|
|
|
var org rawOrg
|
2025-09-05 10:54:34 +02:00
|
|
|
if err := rows.(database.CollectableRows).CollectExactlyOneRow(&org); err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2025-09-08 09:35:31 +02:00
|
|
|
org.Organization.Domains = org.Domains
|
2025-09-05 10:54:34 +02:00
|
|
|
|
|
|
|
|
return org.Organization, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func scanOrganizations(ctx context.Context, querier database.Querier, builder *database.StatementBuilder) ([]*domain.Organization, error) {
|
|
|
|
|
rows, err := querier.Query(ctx, builder.String(), builder.Args()...)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-08 09:35:31 +02:00
|
|
|
var orgs []*rawOrg
|
|
|
|
|
if err := rows.(database.CollectableRows).Collect(&orgs); err != nil {
|
2025-09-05 10:54:34 +02:00
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-08 09:35:31 +02:00
|
|
|
result := make([]*domain.Organization, len(orgs))
|
|
|
|
|
for i, org := range orgs {
|
|
|
|
|
result[i] = org.Organization
|
|
|
|
|
result[i].Domains = org.Domains
|
2025-09-05 10:54:34 +02:00
|
|
|
}
|
2025-09-08 09:35:31 +02:00
|
|
|
|
|
|
|
|
return result, nil
|
2025-09-05 10:54:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
// sub repositories
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
// Domains implements [domain.OrganizationRepository].
|
|
|
|
|
func (o *org) Domains(shouldLoad bool) domain.OrganizationDomainRepository {
|
|
|
|
|
if !o.shouldLoadDomains {
|
|
|
|
|
o.shouldLoadDomains = shouldLoad
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if o.domainRepo != nil {
|
|
|
|
|
return o.domainRepo
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
o.domainRepo = &orgDomain{
|
|
|
|
|
repository: o.repository,
|
|
|
|
|
org: o,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return o.domainRepo
|
|
|
|
|
}
|