mirror of
https://github.com/logto-io/logto.git
synced 2025-01-06 20:40:08 -05:00
229 lines
6.6 KiB
TypeScript
229 lines
6.6 KiB
TypeScript
import { sql } from '@silverhand/slonik';
|
|
import { raw } from 'slonik-sql-tag-raw';
|
|
|
|
import type { AlterationScript } from '../lib/types/alteration.js';
|
|
|
|
const getId = (value: string) => sql.identifier([value]);
|
|
const tenantId = sql.identifier(['tenant_id']);
|
|
|
|
const tables: string[] = [
|
|
'applications',
|
|
'applications_roles',
|
|
'connectors',
|
|
'custom_phrases',
|
|
'logs',
|
|
'oidc_model_instances',
|
|
'passcodes',
|
|
'resources',
|
|
'roles_scopes',
|
|
'roles',
|
|
'scopes',
|
|
'settings',
|
|
'sign_in_experiences',
|
|
'users_roles',
|
|
'users',
|
|
];
|
|
|
|
type IndexInfo = {
|
|
table: string;
|
|
indexes: Array<{ name?: string; columns: string[]; strategy?: 'drop-only' }>;
|
|
};
|
|
|
|
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' }],
|
|
},
|
|
];
|
|
|
|
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'] },
|
|
{ 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' },
|
|
];
|
|
|
|
const alteration: AlterationScript = {
|
|
up: async (pool) => {
|
|
// Add `tenant_id` column and create index accordingly
|
|
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)}
|
|
add column ${tenantId} varchar(21) not null default 'default'
|
|
references tenants (id) on update cascade on delete cascade;
|
|
`);
|
|
|
|
// Column should not have a default tenant ID, it should be always assigned manually or by a trigger
|
|
await pool.query(sql`
|
|
alter table ${getId(tableName)}
|
|
alter column ${tenantId} drop default;
|
|
`);
|
|
|
|
// 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);
|
|
`);
|
|
}
|
|
})
|
|
);
|
|
|
|
// Update indexes
|
|
await Promise.all(
|
|
indexes.flatMap(({ table, indexes }) =>
|
|
indexes.map(async ({ name, columns, strategy }) => {
|
|
const indexName = getId(`${table}__${name ?? columns.join('_')}`);
|
|
await pool.query(sql`drop index ${indexName}`);
|
|
|
|
if (strategy !== 'drop-only') {
|
|
await pool.query(
|
|
sql`
|
|
create index ${indexName}
|
|
on ${getId(table)} (
|
|
${tenantId},
|
|
${sql.join(
|
|
columns.map((column) => raw(column)),
|
|
sql`, `
|
|
)}
|
|
);
|
|
`
|
|
);
|
|
}
|
|
})
|
|
)
|
|
);
|
|
|
|
// 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(
|
|
columns.map((column) => raw(column)),
|
|
sql`, `
|
|
)}
|
|
);
|
|
`);
|
|
})
|
|
);
|
|
},
|
|
down: async (pool) => {
|
|
// 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(
|
|
columns.map((column) => raw(column)),
|
|
sql`, `
|
|
)}
|
|
)
|
|
`)
|
|
: pool.query(sql`
|
|
alter table ${getId(table)}
|
|
add constraint ${indexName} unique (
|
|
${sql.join(
|
|
columns.map((column) => raw(column)),
|
|
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(
|
|
columns.map((column) => raw(column)),
|
|
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;
|
|
`);
|
|
})
|
|
);
|
|
},
|
|
};
|
|
|
|
export default alteration;
|