mirror of
https://github.com/logto-io/logto.git
synced 2024-12-30 20:33:54 -05:00
feat(schemas): init organization tables
This commit is contained in:
parent
059a06ea7e
commit
b81c573583
8 changed files with 262 additions and 7 deletions
|
@ -0,0 +1,150 @@
|
|||
import { type CommonQueryMethods, sql } from 'slonik';
|
||||
|
||||
import type { AlterationScript } from '../lib/types/alteration.js';
|
||||
|
||||
const getDatabaseName = async (pool: CommonQueryMethods) => {
|
||||
const { currentDatabase } = await pool.one<{ currentDatabase: string }>(sql`
|
||||
select current_database();
|
||||
`);
|
||||
|
||||
return currentDatabase.replaceAll('-', '_');
|
||||
};
|
||||
|
||||
const enableRls = async (pool: CommonQueryMethods, database: string, table: string) => {
|
||||
const baseRoleId = sql.identifier([`logto_tenant_${database}`]);
|
||||
|
||||
await pool.query(sql`
|
||||
create trigger set_tenant_id before insert on ${sql.identifier([table])}
|
||||
for each row execute procedure set_tenant_id();
|
||||
|
||||
alter table ${sql.identifier([table])} enable row level security;
|
||||
|
||||
create policy ${sql.identifier([`${table}_tenant_id`])} on ${sql.identifier([table])}
|
||||
as restrictive
|
||||
using (tenant_id = (select id from tenants where db_user = current_user));
|
||||
|
||||
create policy ${sql.identifier([`${table}_modification`])} on ${sql.identifier([table])}
|
||||
using (true);
|
||||
|
||||
grant select, insert, update, delete on ${sql.identifier([table])} to ${baseRoleId};
|
||||
`);
|
||||
};
|
||||
|
||||
const alteration: AlterationScript = {
|
||||
up: async (pool) => {
|
||||
const database = await getDatabaseName(pool);
|
||||
|
||||
await pool.query(sql`
|
||||
create table organizations (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
/** The globally unique identifier of the organization. */
|
||||
id varchar(21) not null,
|
||||
/** The organization's name for display. */
|
||||
name varchar(128) not null,
|
||||
/** A brief description of the organization. */
|
||||
description varchar(256) not null,
|
||||
/** When the organization was created. */
|
||||
created_at timestamptz not null default(now()),
|
||||
primary key (id)
|
||||
);
|
||||
|
||||
create index organizations__id
|
||||
on organizations (tenant_id, id);
|
||||
`);
|
||||
await enableRls(pool, database, 'organizations');
|
||||
|
||||
await pool.query(sql`
|
||||
create table organization_roles (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
/** The globally unique identifier of the organization role. */
|
||||
id varchar(21) not null,
|
||||
/** The organization role's name, unique within the organization template. */
|
||||
name varchar(128) not null,
|
||||
/** A brief description of the organization role. */
|
||||
description varchar(256) not null,
|
||||
primary key (id),
|
||||
constraint organization_roles__name
|
||||
unique (tenant_id, name)
|
||||
);
|
||||
|
||||
create index organization_roles__id
|
||||
on organization_roles (tenant_id, id);
|
||||
`);
|
||||
await enableRls(pool, database, 'organization_roles');
|
||||
|
||||
await pool.query(sql`
|
||||
create table organization_scopes (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
/** The globally unique identifier of the organization scope. */
|
||||
id varchar(21) not null,
|
||||
/** The organization scope's name, unique within the organization template. */
|
||||
name varchar(128) not null,
|
||||
/** A brief description of the organization scope. */
|
||||
description varchar(256) not null,
|
||||
primary key (id),
|
||||
constraint organization_scopes__name
|
||||
unique (tenant_id, name)
|
||||
);
|
||||
|
||||
create index organization_scopes__id
|
||||
on organization_scopes (tenant_id, id);
|
||||
`);
|
||||
await enableRls(pool, database, 'organization_scopes');
|
||||
|
||||
await pool.query(sql`
|
||||
create table organization_role_user_relations (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
organization_id varchar(21) not null
|
||||
references organizations (id) on update cascade on delete cascade,
|
||||
organization_role_id varchar(21) not null
|
||||
references organization_roles (id) on update cascade on delete cascade,
|
||||
user_id varchar(21) not null
|
||||
references users (id) on update cascade on delete cascade,
|
||||
primary key (tenant_id, organization_id, organization_role_id, user_id)
|
||||
);
|
||||
`);
|
||||
await enableRls(pool, database, 'organization_role_user_relations');
|
||||
|
||||
await pool.query(sql`
|
||||
create table organization_role_scope_relations (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
organization_role_id varchar(21) not null
|
||||
references organization_roles (id) on update cascade on delete cascade,
|
||||
organization_scope_id varchar(21) not null
|
||||
references organization_scopes (id) on update cascade on delete cascade,
|
||||
primary key (tenant_id, organization_role_id, organization_scope_id)
|
||||
);
|
||||
`);
|
||||
await enableRls(pool, database, 'organization_role_scope_relations');
|
||||
|
||||
await pool.query(sql`
|
||||
create table organization_user_relations (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
organization_id varchar(21) not null
|
||||
references organizations (id) on update cascade on delete cascade,
|
||||
user_id varchar(21) not null
|
||||
references users (id) on update cascade on delete cascade,
|
||||
primary key (tenant_id, organization_id, user_id)
|
||||
);
|
||||
`);
|
||||
await enableRls(pool, database, 'organization_user_relations');
|
||||
},
|
||||
down: async (pool) => {
|
||||
await pool.query(sql`
|
||||
drop table organization_role_scope_relations;
|
||||
drop table organization_role_user_relations;
|
||||
drop table organization_scopes;
|
||||
drop table organization_roles;
|
||||
drop table organization_user_relations;
|
||||
drop table organizations;
|
||||
`);
|
||||
},
|
||||
};
|
||||
|
||||
export default alteration;
|
|
@ -1,5 +1,5 @@
|
|||
import { type SchemaLike } from '@logto/shared/universal';
|
||||
import type { ZodObject, ZodType, ZodOptional } from 'zod';
|
||||
import type { ZodObject, ZodType, ZodOptional, ZodTypeAny } from 'zod';
|
||||
|
||||
export type { SchemaLike, SchemaValue, SchemaValuePrimitive } from '@logto/shared/universal';
|
||||
|
||||
|
@ -7,13 +7,23 @@ type ParseOptional<K> = undefined extends K
|
|||
? ZodOptional<ZodType<Exclude<K, undefined>>>
|
||||
: ZodType<K>;
|
||||
|
||||
export type CreateGuard<T extends Record<string, unknown>> = ZodObject<{
|
||||
[key in keyof T]-?: ParseOptional<T[key]>;
|
||||
}>;
|
||||
export type CreateGuard<T extends Record<string, unknown>> = ZodObject<
|
||||
{
|
||||
[key in keyof T]-?: ParseOptional<T[key]>;
|
||||
},
|
||||
'strip',
|
||||
ZodTypeAny,
|
||||
T
|
||||
>;
|
||||
|
||||
export type Guard<T extends Record<string, unknown>> = ZodObject<{
|
||||
[key in keyof T]: ZodType<T[key]>;
|
||||
}>;
|
||||
export type Guard<T extends Record<string, unknown>> = ZodObject<
|
||||
{
|
||||
[key in keyof T]: ZodType<T[key]>;
|
||||
},
|
||||
'strip',
|
||||
ZodTypeAny,
|
||||
T
|
||||
>;
|
||||
|
||||
export type GeneratedSchema<
|
||||
CreateSchema extends SchemaLike,
|
||||
|
|
|
@ -0,0 +1,12 @@
|
|||
/* init_order = 2 */
|
||||
|
||||
/** The relations between organization roles and organization scopes. It indicates which organization scopes are available to which organization roles. */
|
||||
create table organization_role_scope_relations (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
organization_role_id varchar(21) not null
|
||||
references organization_roles (id) on update cascade on delete cascade,
|
||||
organization_scope_id varchar(21) not null
|
||||
references organization_scopes (id) on update cascade on delete cascade,
|
||||
primary key (tenant_id, organization_role_id, organization_scope_id)
|
||||
);
|
14
packages/schemas/tables/organization_role_user_relations.sql
Normal file
14
packages/schemas/tables/organization_role_user_relations.sql
Normal file
|
@ -0,0 +1,14 @@
|
|||
/* init_order = 2 */
|
||||
|
||||
/** The relations between organizations, organization roles, and users. A relation means that a user has a role in an organization. */
|
||||
create table organization_role_user_relations (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
organization_id varchar(21) not null
|
||||
references organizations (id) on update cascade on delete cascade,
|
||||
organization_role_id varchar(21) not null
|
||||
references organization_roles (id) on update cascade on delete cascade,
|
||||
user_id varchar(21) not null
|
||||
references users (id) on update cascade on delete cascade,
|
||||
primary key (tenant_id, organization_id, organization_role_id, user_id)
|
||||
);
|
19
packages/schemas/tables/organization_roles.sql
Normal file
19
packages/schemas/tables/organization_roles.sql
Normal file
|
@ -0,0 +1,19 @@
|
|||
/* init_order = 1 */
|
||||
|
||||
/** The roles defined by the organization template. */
|
||||
create table organization_roles (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
/** The globally unique identifier of the organization role. */
|
||||
id varchar(21) not null,
|
||||
/** The organization role's name, unique within the organization template. */
|
||||
name varchar(128) not null,
|
||||
/** A brief description of the organization role. */
|
||||
description varchar(256) not null,
|
||||
primary key (id),
|
||||
constraint organization_roles__name
|
||||
unique (tenant_id, name)
|
||||
);
|
||||
|
||||
create index organization_roles__id
|
||||
on organization_roles (tenant_id, id);
|
19
packages/schemas/tables/organization_scopes.sql
Normal file
19
packages/schemas/tables/organization_scopes.sql
Normal file
|
@ -0,0 +1,19 @@
|
|||
/* init_order = 1 */
|
||||
|
||||
/** The scopes (permissions) defined by the organization template. */
|
||||
create table organization_scopes (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
/** The globally unique identifier of the organization scope. */
|
||||
id varchar(21) not null,
|
||||
/** The organization scope's name, unique within the organization template. */
|
||||
name varchar(128) not null,
|
||||
/** A brief description of the organization scope. */
|
||||
description varchar(256) not null,
|
||||
primary key (id),
|
||||
constraint organization_scopes__name
|
||||
unique (tenant_id, name)
|
||||
);
|
||||
|
||||
create index organization_scopes__id
|
||||
on organization_scopes (tenant_id, id);
|
12
packages/schemas/tables/organization_user_relations.sql
Normal file
12
packages/schemas/tables/organization_user_relations.sql
Normal file
|
@ -0,0 +1,12 @@
|
|||
/* init_order = 2 */
|
||||
|
||||
/** The relations between organizations and users. It indicates membership of users in organizations. */
|
||||
create table organization_user_relations (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
organization_id varchar(21) not null
|
||||
references organizations (id) on update cascade on delete cascade,
|
||||
user_id varchar(21) not null
|
||||
references users (id) on update cascade on delete cascade,
|
||||
primary key (tenant_id, organization_id, user_id)
|
||||
);
|
19
packages/schemas/tables/organizations.sql
Normal file
19
packages/schemas/tables/organizations.sql
Normal file
|
@ -0,0 +1,19 @@
|
|||
/* init_order = 1 */
|
||||
|
||||
/** Organizations defined by [RFC 0001](https://github.com/logto-io/rfcs/blob/HEAD/active/0001-organization.md). */
|
||||
create table organizations (
|
||||
tenant_id varchar(21) not null
|
||||
references tenants (id) on update cascade on delete cascade,
|
||||
/** The globally unique identifier of the organization. */
|
||||
id varchar(21) not null,
|
||||
/** The organization's name for display. */
|
||||
name varchar(128) not null,
|
||||
/** A brief description of the organization. */
|
||||
description varchar(256) not null,
|
||||
/** When the organization was created. */
|
||||
created_at timestamptz not null default(now()),
|
||||
primary key (id)
|
||||
);
|
||||
|
||||
create index organizations__id
|
||||
on organizations (tenant_id, id);
|
Loading…
Reference in a new issue