-
Notifications
You must be signed in to change notification settings - Fork 227
Multi tenancy
Support for multi-tenancy build on top of the concepts of a VIEW in Phoenix. Users create a logical tenant-specific table as a VIEW and query and update it just like with regular Phoenix tables. Data in these tenant-specific tables resides in a shared, regular Phoenix table (and thus in a shared HBase table) that is declared at table creation time to support multi-tenancy. All tenant-specific Phoenix tables whose data resides in the same physical HBase table have the same primary key structure but each tenant’s table can contain any number of non-PK columns unique to it. The main advantages afforded by this feature are:
- It implements physical tenant data isolation, including automatically constraining tenants to only work with data that “belongs” to the each tenant.
- It prevents a proliferation of HBase tables, minimizing operational complexity.
The first primary key column of the physical multi-tenant table must identify the tenant. For example:
CREATE TABLE base.event (tenant_id VARCHAR, event_type SMALLINT, created_date DATE, event_id BIGINT)
MULTI_TENANT=true;
In this case, the tenant_id column identifies the tenant and the table is declared to be multi-tenant.
Tenants are identified by the presence or absence of a TenantId property at JDBC connection-time. A connection with a non-null TenantId is considered a tenant-specific connection. A connection an unspecified or null TenantId is a regular connection. A tenant’s connection may only query:
- their own schema, which is to say it only sees tenant-specific views that were created by that tenant.
- non multi-tenant global tables, that is tables created with a regular connection without the MULTI_TENANT=TRUE declaration.
Tenant-specific views may only be created using a tenant-specific connection and the base table must be a multi-tenant table. Regular connections are used to create global tables, including those that can be used as base tables for tenant-specific tables.
For example, if a tenant-specific connection is established like this: Properties props = new Properties(); props.setProperty("TenantId", "Acme"); Connection conn = DriverManager.getConnection("localhost", props);
through which a tenant-specific table would be defined like this:
CREATE VIEW acme.event AS
SELECT * FROM base.event;