0
Fork 0
mirror of https://github.com/logto-io/logto.git synced 2024-12-30 20:33:54 -05:00
logto/packages/schemas/alterations/1.0.0_rc.0-1674032095.5-multi-tenancy.ts

229 lines
6.6 KiB
TypeScript
Raw Permalink Normal View History

2024-03-16 06:04:55 -05:00
import { sql } from '@silverhand/slonik';
2023-01-19 07:27:01 -05:00
import type { AlterationScript } from '../lib/types/alteration.js';
const getId = (value: string) => sql.identifier([value]);
const tenantId = sql.identifier(['tenant_id']);
2023-01-27 23:29:47 -05:00
const tables: string[] = [
'applications',
'applications_roles',
2023-01-27 23:29:47 -05:00
'connectors',
'custom_phrases',
'logs',
'oidc_model_instances',
'passcodes',
'resources',
'roles_scopes',
'roles',
'scopes',
'settings',
'sign_in_experiences',
'users_roles',
'users',
2023-01-19 07:27:01 -05:00
];
type IndexInfo = {
table: string;
2023-01-27 23:29:47 -05:00
indexes: Array<{ name?: string; columns: string[]; strategy?: 'drop-only' }>;
2023-01-19 07:27:01 -05:00
};
const indexes: IndexInfo[] = [
{
table: 'logs',
indexes: [
{ columns: ['key'] },
{ columns: ['created_at'], strategy: 'drop-only' },
{ name: 'user_id', columns: ["(payload->>'user_id') nulls last"] },
{ name: 'application_id', columns: ["(payload->>'application_id') nulls last"] },
],
},
{
table: 'oidc_model_instances',
indexes: [
{ name: 'model_name_payload_user_code', columns: ['model_name', "(payload->>'userCode')"] },
{ name: 'model_name_payload_uid', columns: ['model_name', "(payload->>'uid')"] },
{ name: 'model_name_payload_grant_id', columns: ['model_name', "(payload->>'grantId')"] },
],
},
{
table: 'passcodes',
indexes: [
{ columns: ['interaction_jti', 'type'] },
{ columns: ['email', 'type'] },
{ columns: ['phone', 'type'] },
],
},
{
table: 'users',
indexes: [{ columns: ['name'] }, { columns: ['created_at'], strategy: 'drop-only' }],
},
];
2023-01-27 23:29:47 -05:00
type ConstraintInfo = {
table: string;
columns: string[];
original?: 'index';
};
const constraints: ConstraintInfo[] = [
{ table: 'applications_roles', columns: ['application_id', 'role_id'] },
{ table: 'custom_phrases', columns: ['language_tag'] },
{ table: 'oidc_model_instances', columns: ['model_name', 'id'] },
2023-01-27 23:29:47 -05:00
{ table: 'roles_scopes', columns: ['role_id', 'scope_id'] },
{ table: 'users_roles', columns: ['user_id', 'role_id'] },
{ table: 'resources', columns: ['indicator'], original: 'index' },
{ table: 'roles', columns: ['name'], original: 'index' },
{ table: 'scopes', columns: ['resource_id', 'name'], original: 'index' },
];
2023-01-19 07:27:01 -05:00
const alteration: AlterationScript = {
up: async (pool) => {
2023-01-27 23:29:47 -05:00
// Add `tenant_id` column and create index accordingly
2023-01-19 07:27:01 -05:00
await Promise.all(
2023-01-27 23:29:47 -05:00
tables.map(async (tableName) => {
// Add `tenant_id` column and set existing data to the default tenant
2023-01-19 07:27:01 -05:00
await pool.query(sql`
2023-01-27 23:29:47 -05:00
alter table ${getId(tableName)}
2023-01-19 07:27:01 -05:00
add column ${tenantId} varchar(21) not null default 'default'
2023-01-27 23:29:47 -05:00
references tenants (id) on update cascade on delete cascade;
2023-01-19 07:27:01 -05:00
`);
2023-01-27 23:29:47 -05:00
// Column should not have a default tenant ID, it should be always assigned manually or by a trigger
2023-01-19 07:27:01 -05:00
await pool.query(sql`
2023-01-27 23:29:47 -05:00
alter table ${getId(tableName)}
alter column ${tenantId} drop default;
2023-01-19 07:27:01 -05:00
`);
2023-01-27 23:29:47 -05:00
// Skip OIDC model instances since we always query them with a model name
if (tableName !== 'oidc_model_instances') {
// Add ID index for better RLS query performance
await pool.query(sql`
create index ${getId(`${tableName}__id`)}
on ${getId(tableName)} (${tenantId}, id);
`);
}
2023-01-19 07:27:01 -05:00
})
);
// Update indexes
await Promise.all(
indexes.flatMap(({ table, indexes }) =>
2023-01-27 23:29:47 -05:00
indexes.map(async ({ name, columns, strategy }) => {
2023-01-19 07:27:01 -05:00
const indexName = getId(`${table}__${name ?? columns.join('_')}`);
await pool.query(sql`drop index ${indexName}`);
if (strategy !== 'drop-only') {
await pool.query(
sql`
2023-01-27 23:29:47 -05:00
create index ${indexName}
on ${getId(table)} (
2023-01-19 07:27:01 -05:00
${tenantId},
${sql.join(
2024-03-17 20:59:38 -05:00
columns.map((column) => sql.raw(column)),
2023-01-19 07:27:01 -05:00
sql`, `
)}
);
`
);
}
})
)
);
2023-01-27 23:29:47 -05:00
// Update constraints
await Promise.all(
constraints.map(async ({ table, columns, original }) => {
const indexName = getId(`${table}__${columns.join('_')}`);
if (original === 'index') {
await pool.query(sql`drop index ${indexName}`);
}
await pool.query(sql`
alter table ${getId(table)}
${original === 'index' ? sql`` : sql`drop constraint ${indexName},`}
add constraint ${indexName} unique (
${tenantId},
${sql.join(
2024-03-17 20:59:38 -05:00
columns.map((column) => sql.raw(column)),
2023-01-27 23:29:47 -05:00
sql`, `
)}
);
`);
})
);
2023-01-19 07:27:01 -05:00
},
down: async (pool) => {
2023-01-27 23:29:47 -05:00
// Restore constraints
await Promise.all(
constraints.map(async ({ table, columns, original }) => {
const indexName = getId(`${table}__${columns.join('_')}`);
await pool.query(sql`
alter table ${getId(table)}
drop constraint ${indexName};
`);
await (original === 'index'
? pool.query(sql`
create unique index ${indexName}
on ${getId(table)} (
${sql.join(
2024-03-17 20:59:38 -05:00
columns.map((column) => sql.raw(column)),
2023-01-27 23:29:47 -05:00
sql`, `
)}
)
`)
: pool.query(sql`
alter table ${getId(table)}
add constraint ${indexName} unique (
${sql.join(
2024-03-17 20:59:38 -05:00
columns.map((column) => sql.raw(column)),
2023-01-27 23:29:47 -05:00
sql`, `
)}
);
`));
})
);
// Restore indexes
await Promise.all(
indexes.flatMap(({ table, indexes }) =>
indexes.map(async ({ name, columns, strategy }) => {
const indexName = getId(`${table}__${name ?? columns.join('_')}`);
if (strategy !== 'drop-only') {
await pool.query(sql`drop index ${indexName}`);
}
await pool.query(
sql`
create index ${indexName}
on ${getId(table)} (
${sql.join(
2024-03-17 20:59:38 -05:00
columns.map((column) => sql.raw(column)),
2023-01-27 23:29:47 -05:00
sql`, `
)}
);
`
);
})
)
);
// Drop `tenant_id` column cascade
await Promise.all(
tables.map(async (tableName) => {
// Add `tenant_id` column and set existing data to the default tenant
await pool.query(sql`
alter table ${getId(tableName)}
drop column ${tenantId} cascade;
`);
})
);
2023-01-19 07:27:01 -05:00
},
};
export default alteration;