-
Notifications
You must be signed in to change notification settings - Fork 15
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: Add Tenant Attributes with Token Mapper Support (#56)
* feat: Add tenant attributes with token mapper support - Add support for storing tenant attributes with multi-value capability - Implement token mapper for including tenant attributes in tokens - Support both short (<= 255 chars) and long values with efficient storage - Add attribute search functionality to tenant listing API - Include validation for attribute keys and values - Update documentation in README and OpenAPI spec The tenant attributes feature allows storing and retrieving configuration and metadata at the tenant level, with token mapper support making these attributes available in tokens for applications. * fix: update ApiIntegrationTest to align with listTenants changes - Update test cases to include the new argument in the listTenants function - Ensure tests remain consistent with recent changes made to the function signature These changes were missed in the original commit introducing the new argument.
- Loading branch information
Showing
22 changed files
with
862 additions
and
14 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
122 changes: 122 additions & 0 deletions
122
src/main/java/dev/sultanov/keycloak/multitenancy/model/entity/TenantAttributeEntity.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,122 @@ | ||
package dev.sultanov.keycloak.multitenancy.model.entity; | ||
|
||
import jakarta.persistence.Access; | ||
import jakarta.persistence.AccessType; | ||
import jakarta.persistence.Column; | ||
import jakarta.persistence.Entity; | ||
import jakarta.persistence.FetchType; | ||
import jakarta.persistence.Id; | ||
import jakarta.persistence.JoinColumn; | ||
import jakarta.persistence.ManyToOne; | ||
import jakarta.persistence.NamedQueries; | ||
import jakarta.persistence.NamedQuery; | ||
import jakarta.persistence.Table; | ||
import org.hibernate.annotations.Nationalized; | ||
import org.keycloak.storage.jpa.JpaHashUtils; | ||
|
||
@NamedQueries({ | ||
@NamedQuery(name="deleteTenantAttributesByRealm", | ||
query="delete from TenantAttributeEntity attr where attr.tenant IN (select t from TenantEntity t where t.realmId=:realmId)"), | ||
@NamedQuery(name="deleteTenantAttributesByNameAndTenant", | ||
query="delete from TenantAttributeEntity attr where attr.tenant.id = :tenantId and attr.name = :name"), | ||
@NamedQuery(name="deleteTenantAttributesByNameAndTenantOtherThan", | ||
query="delete from TenantAttributeEntity attr where attr.tenant.id = :tenantId and attr.name = :name and attr.id <> :attrId") | ||
}) | ||
@Table(name="TENANT_ATTRIBUTE") | ||
@Entity | ||
public class TenantAttributeEntity { | ||
|
||
@Id | ||
@Column(name="ID", length = 36) | ||
@Access(AccessType.PROPERTY) | ||
protected String id; | ||
|
||
@ManyToOne(fetch= FetchType.LAZY) | ||
@JoinColumn(name = "TENANT_ID") | ||
protected TenantEntity tenant; | ||
|
||
@Column(name = "NAME") | ||
protected String name; | ||
|
||
@Nationalized | ||
@Column(name = "VALUE") | ||
protected String value; | ||
|
||
@Column(name = "LONG_VALUE_HASH") | ||
private byte[] longValueHash; | ||
|
||
@Column(name = "LONG_VALUE_HASH_LOWER_CASE") | ||
private byte[] longValueHashLowerCase; | ||
|
||
@Nationalized | ||
@Column(name = "LONG_VALUE") | ||
private String longValue; | ||
|
||
public String getId() { | ||
return id; | ||
} | ||
|
||
public void setId(String id) { | ||
this.id = id; | ||
} | ||
|
||
public String getName() { | ||
return name; | ||
} | ||
|
||
public void setName(String name) { | ||
this.name = name; | ||
} | ||
|
||
public String getValue() { | ||
if (value != null && longValue != null) { | ||
throw new IllegalStateException(String.format("Tenant with id %s should not have set both `value` and `longValue` for attribute %s.", tenant.getId(), name)); | ||
} | ||
return value != null ? value : longValue; | ||
} | ||
|
||
public void setValue(String value) { | ||
if (value == null) { | ||
this.value = null; | ||
this.longValue = null; | ||
this.longValueHash = null; | ||
this.longValueHashLowerCase = null; | ||
} else if (value.length() > 255) { | ||
this.value = null; | ||
this.longValue = value; | ||
this.longValueHash = JpaHashUtils.hashForAttributeValue(value); | ||
this.longValueHashLowerCase = JpaHashUtils.hashForAttributeValueLowerCase(value); | ||
} else { | ||
this.value = value; | ||
this.longValue = null; | ||
this.longValueHash = null; | ||
this.longValueHashLowerCase = null; | ||
} | ||
} | ||
|
||
public TenantEntity getTenant() { | ||
return tenant; | ||
} | ||
|
||
public void setTenant(TenantEntity tenant) { | ||
this.tenant = tenant; | ||
} | ||
|
||
@Override | ||
public boolean equals(Object o) { | ||
if (this == o) return true; | ||
if (o == null || getClass() != o.getClass()) return false; | ||
if (!(o instanceof TenantAttributeEntity)) return false; | ||
|
||
TenantAttributeEntity that = (TenantAttributeEntity) o; | ||
|
||
if (!id.equals(that.getId())) return false; | ||
|
||
return true; | ||
} | ||
|
||
@Override | ||
public int hashCode() { | ||
return id.hashCode(); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.