2025-06-17 09:46:01 +02:00
|
|
|
package repository
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2025-07-22 19:09:56 +02:00
|
|
|
"encoding/json"
|
2025-06-17 09:46:01 +02:00
|
|
|
|
|
|
|
"github.com/zitadel/zitadel/backend/v3/domain"
|
|
|
|
"github.com/zitadel/zitadel/backend/v3/storage/database"
|
|
|
|
)
|
|
|
|
|
|
|
|
var _ domain.InstanceRepository = (*instance)(nil)
|
|
|
|
|
|
|
|
type instance struct {
|
|
|
|
repository
|
2025-07-22 19:09:56 +02:00
|
|
|
shouldLoadDomains bool
|
|
|
|
domainRepo *instanceDomain
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func InstanceRepository(client database.QueryExecutor) domain.InstanceRepository {
|
|
|
|
return &instance{
|
|
|
|
repository: repository{
|
|
|
|
client: client,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// repository
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
const (
|
|
|
|
queryInstanceStmt = `SELECT instances.id, instances.name, instances.default_org_id, instances.iam_project_id, instances.console_client_id, instances.console_app_id, instances.default_language, instances.created_at, instances.updated_at` +
|
2025-07-23 09:47:04 +02:00
|
|
|
` , CASE WHEN count(instance_domains.domain) > 0 THEN jsonb_agg(json_build_object('domain', instance_domains.domain, 'isVerified', instance_domains.is_verified, 'isPrimary', instance_domains.is_primary, 'isGenerated', instance_domains.is_generated, 'validationType', instance_domains.validation_type, 'createdAt', instance_domains.created_at, 'updatedAt', instance_domains.updated_at)) ELSE NULL::JSONB END domains` +
|
|
|
|
` FROM zitadel.instances`
|
2025-07-22 19:09:56 +02:00
|
|
|
)
|
2025-06-17 09:46:01 +02:00
|
|
|
|
|
|
|
// Get implements [domain.InstanceRepository].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (i *instance) Get(ctx context.Context, opts ...database.QueryOption) (*domain.Instance, error) {
|
2025-07-23 09:47:04 +02:00
|
|
|
opts = append(opts,
|
|
|
|
i.joinDomains(),
|
2025-07-22 19:09:56 +02:00
|
|
|
database.WithGroupBy(i.IDColumn(true)),
|
|
|
|
)
|
2025-07-23 09:47:04 +02:00
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
options := new(database.QueryOpts)
|
|
|
|
for _, opt := range opts {
|
|
|
|
opt(options)
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
var builder database.StatementBuilder
|
2025-06-17 09:46:01 +02:00
|
|
|
builder.WriteString(queryInstanceStmt)
|
2025-07-22 19:09:56 +02:00
|
|
|
options.Write(&builder)
|
2025-06-17 09:46:01 +02:00
|
|
|
|
2025-07-14 21:27:14 +02:00
|
|
|
return scanInstance(ctx, i.client, &builder)
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// List implements [domain.InstanceRepository].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (i *instance) List(ctx context.Context, opts ...database.QueryOption) ([]*domain.Instance, error) {
|
2025-07-23 09:47:04 +02:00
|
|
|
opts = append(opts,
|
|
|
|
i.joinDomains(),
|
2025-07-22 19:09:56 +02:00
|
|
|
database.WithGroupBy(i.IDColumn(true)),
|
|
|
|
)
|
2025-07-23 09:47:04 +02:00
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
options := new(database.QueryOpts)
|
|
|
|
for _, opt := range opts {
|
|
|
|
opt(options)
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
var builder database.StatementBuilder
|
2025-06-17 09:46:01 +02:00
|
|
|
builder.WriteString(queryInstanceStmt)
|
2025-07-22 19:09:56 +02:00
|
|
|
options.Write(&builder)
|
|
|
|
|
|
|
|
return scanInstances(ctx, i.client, &builder)
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
func (i *instance) joinDomains() database.QueryOption {
|
|
|
|
columns := make([]database.Condition, 0, 2)
|
|
|
|
columns = append(columns, database.NewColumnCondition(i.IDColumn(true), i.Domains(false).InstanceIDColumn(true)))
|
|
|
|
|
|
|
|
// 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 !i.shouldLoadDomains {
|
|
|
|
columns = append(columns, database.IsNull(i.Domains(false).InstanceIDColumn(true)))
|
2025-07-15 20:20:53 +02:00
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
return database.WithLeftJoin(
|
|
|
|
"zitadel.instance_domains",
|
|
|
|
database.And(columns...),
|
|
|
|
)
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const createInstanceStmt = `INSERT INTO zitadel.instances (id, name, default_org_id, iam_project_id, console_client_id, console_app_id, default_language)` +
|
|
|
|
` VALUES ($1, $2, $3, $4, $5, $6, $7)` +
|
|
|
|
` RETURNING created_at, updated_at`
|
|
|
|
|
|
|
|
// Create implements [domain.InstanceRepository].
|
|
|
|
func (i *instance) Create(ctx context.Context, instance *domain.Instance) error {
|
|
|
|
var builder database.StatementBuilder
|
|
|
|
|
|
|
|
builder.AppendArgs(instance.ID, instance.Name, instance.DefaultOrgID, instance.IAMProjectID, instance.ConsoleClientID, instance.ConsoleAppID, instance.DefaultLanguage)
|
|
|
|
builder.WriteString(createInstanceStmt)
|
|
|
|
|
2025-07-17 00:54:21 +02:00
|
|
|
return i.client.QueryRow(ctx, builder.String(), builder.Args()...).Scan(&instance.CreatedAt, &instance.UpdatedAt)
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Update implements [domain.InstanceRepository].
|
2025-07-14 21:27:14 +02:00
|
|
|
func (i instance) Update(ctx context.Context, id string, changes ...database.Change) (int64, error) {
|
2025-07-22 19:09:56 +02:00
|
|
|
if len(changes) == 0 {
|
|
|
|
return 0, database.NoChangesError
|
2025-07-14 21:27:14 +02:00
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
var builder database.StatementBuilder
|
|
|
|
|
|
|
|
builder.WriteString(`UPDATE zitadel.instances SET `)
|
|
|
|
|
|
|
|
database.Changes(changes).Write(&builder)
|
2025-07-14 21:27:14 +02:00
|
|
|
|
|
|
|
idCondition := i.IDCondition(id)
|
2025-07-15 20:20:53 +02:00
|
|
|
writeCondition(&builder, idCondition)
|
2025-06-17 09:46:01 +02:00
|
|
|
|
|
|
|
stmt := builder.String()
|
|
|
|
|
2025-07-17 00:54:21 +02:00
|
|
|
return i.client.Exec(ctx, stmt, builder.Args()...)
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Delete implements [domain.InstanceRepository].
|
2025-07-14 21:27:14 +02:00
|
|
|
func (i instance) Delete(ctx context.Context, id string) (int64, error) {
|
2025-06-17 09:46:01 +02:00
|
|
|
var builder database.StatementBuilder
|
|
|
|
|
2025-07-15 20:20:53 +02:00
|
|
|
builder.WriteString(`DELETE FROM zitadel.instances`)
|
2025-06-17 09:46:01 +02:00
|
|
|
|
2025-07-14 21:27:14 +02:00
|
|
|
idCondition := i.IDCondition(id)
|
2025-07-15 20:20:53 +02:00
|
|
|
writeCondition(&builder, idCondition)
|
2025-07-14 21:27:14 +02:00
|
|
|
|
|
|
|
return i.client.Exec(ctx, builder.String(), builder.Args()...)
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// changes
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
|
|
|
// SetName implements [domain.instanceChanges].
|
|
|
|
func (i instance) SetName(name string) database.Change {
|
2025-07-22 19:09:56 +02:00
|
|
|
return database.NewChange(i.NameColumn(false), name)
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// conditions
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
|
|
|
// IDCondition implements [domain.instanceConditions].
|
|
|
|
func (i instance) IDCondition(id string) database.Condition {
|
2025-07-22 19:09:56 +02:00
|
|
|
return database.NewTextCondition(i.IDColumn(true), database.TextOperationEqual, id)
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NameCondition implements [domain.instanceConditions].
|
|
|
|
func (i instance) NameCondition(op database.TextOperation, name string) database.Condition {
|
2025-07-22 19:09:56 +02:00
|
|
|
return database.NewTextCondition(i.NameColumn(true), op, name)
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// columns
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
|
|
|
// IDColumn implements [domain.instanceColumns].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (instance) IDColumn(qualified bool) database.Column {
|
|
|
|
if qualified {
|
|
|
|
return database.NewColumn("instances.id")
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
return database.NewColumn("id")
|
|
|
|
}
|
|
|
|
|
|
|
|
// NameColumn implements [domain.instanceColumns].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (instance) NameColumn(qualified bool) database.Column {
|
|
|
|
if qualified {
|
|
|
|
return database.NewColumn("instances.name")
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
return database.NewColumn("name")
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreatedAtColumn implements [domain.instanceColumns].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (instance) CreatedAtColumn(qualified bool) database.Column {
|
|
|
|
if qualified {
|
|
|
|
return database.NewColumn("instances.created_at")
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
return database.NewColumn("created_at")
|
|
|
|
}
|
|
|
|
|
|
|
|
// DefaultOrgIdColumn implements [domain.instanceColumns].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (instance) DefaultOrgIDColumn(qualified bool) database.Column {
|
|
|
|
if qualified {
|
|
|
|
return database.NewColumn("instances.default_org_id")
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
return database.NewColumn("default_org_id")
|
|
|
|
}
|
|
|
|
|
|
|
|
// IAMProjectIDColumn implements [domain.instanceColumns].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (instance) IAMProjectIDColumn(qualified bool) database.Column {
|
|
|
|
if qualified {
|
|
|
|
return database.NewColumn("instances.iam_project_id")
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
return database.NewColumn("iam_project_id")
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConsoleClientIDColumn implements [domain.instanceColumns].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (instance) ConsoleClientIDColumn(qualified bool) database.Column {
|
|
|
|
if qualified {
|
|
|
|
return database.NewColumn("instances.console_client_id")
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
return database.NewColumn("console_client_id")
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConsoleAppIDColumn implements [domain.instanceColumns].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (instance) ConsoleAppIDColumn(qualified bool) database.Column {
|
|
|
|
if qualified {
|
|
|
|
return database.NewColumn("instances.console_app_id")
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
return database.NewColumn("console_app_id")
|
|
|
|
}
|
|
|
|
|
|
|
|
// DefaultLanguageColumn implements [domain.instanceColumns].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (instance) DefaultLanguageColumn(qualified bool) database.Column {
|
|
|
|
if qualified {
|
|
|
|
return database.NewColumn("instances.default_language")
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
return database.NewColumn("default_language")
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdatedAtColumn implements [domain.instanceColumns].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (instance) UpdatedAtColumn(qualified bool) database.Column {
|
|
|
|
if qualified {
|
|
|
|
return database.NewColumn("instances.updated_at")
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
return database.NewColumn("updated_at")
|
|
|
|
}
|
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
type rawInstance struct {
|
|
|
|
*domain.Instance
|
|
|
|
RawDomains json.RawMessage `json:"domains,omitempty" db:"domains"`
|
|
|
|
}
|
|
|
|
|
2025-07-14 21:27:14 +02:00
|
|
|
func scanInstance(ctx context.Context, querier database.Querier, builder *database.StatementBuilder) (*domain.Instance, error) {
|
|
|
|
rows, err := querier.Query(ctx, builder.String(), builder.Args()...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
2025-07-23 09:47:04 +02:00
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
var instance rawInstance
|
|
|
|
if err := rows.(database.CollectableRows).CollectExactlyOneRow(&instance); err != nil {
|
2025-07-14 21:27:14 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
if len(instance.RawDomains) > 0 {
|
|
|
|
if err := json.Unmarshal(instance.RawDomains, &instance.Domains); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return instance.Instance, nil
|
2025-06-17 09:46:01 +02:00
|
|
|
}
|
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
func scanInstances(ctx context.Context, querier database.Querier, builder *database.StatementBuilder) ([]*domain.Instance, error) {
|
2025-07-14 21:27:14 +02:00
|
|
|
rows, err := querier.Query(ctx, builder.String(), builder.Args()...)
|
2025-06-17 09:46:01 +02:00
|
|
|
if err != nil {
|
2025-07-14 21:27:14 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
var rawInstances []*rawInstance
|
|
|
|
if err := rows.(database.CollectableRows).Collect(&rawInstances); err != nil {
|
2025-06-17 09:46:01 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2025-07-22 19:09:56 +02:00
|
|
|
instances := make([]*domain.Instance, len(rawInstances))
|
|
|
|
for i, instance := range rawInstances {
|
|
|
|
if len(instance.RawDomains) > 0 {
|
|
|
|
if err := json.Unmarshal(instance.RawDomains, &instance.Domains); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
instances[i] = instance.Instance
|
|
|
|
}
|
2025-06-17 09:46:01 +02:00
|
|
|
return instances, nil
|
|
|
|
}
|
2025-07-16 09:26:46 +02:00
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// sub repositories
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
|
|
|
|
// Domains implements [domain.InstanceRepository].
|
2025-07-22 19:09:56 +02:00
|
|
|
func (i *instance) Domains(shouldLoad bool) domain.InstanceDomainRepository {
|
|
|
|
if !i.shouldLoadDomains {
|
|
|
|
i.shouldLoadDomains = shouldLoad
|
|
|
|
}
|
2025-07-16 09:26:46 +02:00
|
|
|
|
|
|
|
if i.domainRepo != nil {
|
|
|
|
return i.domainRepo
|
|
|
|
}
|
|
|
|
|
|
|
|
i.domainRepo = &instanceDomain{
|
|
|
|
repository: i.repository,
|
|
|
|
instance: i,
|
|
|
|
}
|
|
|
|
return i.domainRepo
|
|
|
|
}
|