2023-01-04 15:39:27 +08:00
|
|
|
import type { User, CreateUser, UserWithRoleNames } from '@logto/schemas';
|
|
|
|
import { SearchJointMode, Users } from '@logto/schemas';
|
2022-10-21 13:14:17 +08:00
|
|
|
import type { OmitAutoSetFields } from '@logto/shared';
|
|
|
|
import { conditionalSql, convertToIdentifiers } from '@logto/shared';
|
2021-07-02 21:14:18 +08:00
|
|
|
import { sql } from 'slonik';
|
2021-08-30 11:30:54 +08:00
|
|
|
|
2022-11-21 16:38:24 +08:00
|
|
|
import { buildUpdateWhere } from '#src/database/update-where.js';
|
|
|
|
import envSet from '#src/env-set/index.js';
|
|
|
|
import { DeletionError } from '#src/errors/SlonikError/index.js';
|
2022-12-14 16:36:57 +08:00
|
|
|
import type { Search } from '#src/utils/search.js';
|
|
|
|
import { buildConditionsFromSearch } from '#src/utils/search.js';
|
2021-07-02 21:14:18 +08:00
|
|
|
|
2023-01-04 15:39:27 +08:00
|
|
|
import { findRoleByRoleName, findRolesByRoleIds } from './roles.js';
|
|
|
|
import { findUsersRolesByRoleId, findUsersRolesByUserId } from './users-roles.js';
|
|
|
|
|
2021-07-02 21:14:18 +08:00
|
|
|
const { table, fields } = convertToIdentifiers(Users);
|
|
|
|
|
2021-07-26 22:32:18 +08:00
|
|
|
export const findUserByUsername = async (username: string) =>
|
2022-06-02 16:29:17 +08:00
|
|
|
envSet.pool.maybeOne<User>(sql`
|
2021-12-20 14:20:23 +08:00
|
|
|
select ${sql.join(Object.values(fields), sql`,`)}
|
|
|
|
from ${table}
|
|
|
|
where ${fields.username}=${username}
|
|
|
|
`);
|
2021-07-26 22:32:18 +08:00
|
|
|
|
2022-01-31 11:04:55 +08:00
|
|
|
export const findUserByEmail = async (email: string) =>
|
2022-11-11 14:54:46 +08:00
|
|
|
envSet.pool.maybeOne<User>(sql`
|
2022-01-31 11:04:55 +08:00
|
|
|
select ${sql.join(Object.values(fields), sql`,`)}
|
|
|
|
from ${table}
|
2022-11-09 15:33:07 +08:00
|
|
|
where lower(${fields.primaryEmail})=lower(${email})
|
2022-01-31 11:04:55 +08:00
|
|
|
`);
|
|
|
|
|
2022-02-09 16:14:42 +08:00
|
|
|
export const findUserByPhone = async (phone: string) =>
|
2022-11-11 14:54:46 +08:00
|
|
|
envSet.pool.maybeOne<User>(sql`
|
2022-02-09 16:14:42 +08:00
|
|
|
select ${sql.join(Object.values(fields), sql`,`)}
|
|
|
|
from ${table}
|
|
|
|
where ${fields.primaryPhone}=${phone}
|
|
|
|
`);
|
|
|
|
|
2023-01-04 15:39:27 +08:00
|
|
|
export const findUserById = async (id: string): Promise<UserWithRoleNames> => {
|
|
|
|
const user = await envSet.pool.one<User>(sql`
|
2021-12-20 14:20:23 +08:00
|
|
|
select ${sql.join(Object.values(fields), sql`,`)}
|
|
|
|
from ${table}
|
|
|
|
where ${fields.id}=${id}
|
|
|
|
`);
|
2023-01-04 15:39:27 +08:00
|
|
|
const userRoles = await findUsersRolesByUserId(user.id);
|
|
|
|
|
|
|
|
const roles =
|
|
|
|
userRoles.length > 0 ? await findRolesByRoleIds(userRoles.map(({ roleId }) => roleId)) : [];
|
|
|
|
|
|
|
|
return {
|
|
|
|
...user,
|
|
|
|
roleNames: roles.map(({ name }) => name),
|
|
|
|
};
|
|
|
|
};
|
2021-07-04 17:41:46 +08:00
|
|
|
|
2022-06-17 11:00:02 +08:00
|
|
|
export const findUserByIdentity = async (target: string, userId: string) =>
|
2022-11-28 10:37:25 +08:00
|
|
|
envSet.pool.maybeOne<User>(
|
2022-02-11 15:19:18 +08:00
|
|
|
sql`
|
|
|
|
select ${sql.join(Object.values(fields), sql`,`)}
|
|
|
|
from ${table}
|
2022-06-17 11:00:02 +08:00
|
|
|
where ${fields.identities}::json#>>'{${sql.identifier([target])},userId}' = ${userId}
|
2022-02-11 15:19:18 +08:00
|
|
|
`
|
|
|
|
);
|
|
|
|
|
2022-11-10 19:27:53 +08:00
|
|
|
export const hasUser = async (username: string, excludeUserId?: string) =>
|
2022-04-19 21:49:20 +08:00
|
|
|
envSet.pool.exists(sql`
|
2021-12-20 14:20:23 +08:00
|
|
|
select ${fields.id}
|
|
|
|
from ${table}
|
|
|
|
where ${fields.username}=${username}
|
2022-11-10 19:27:53 +08:00
|
|
|
${conditionalSql(excludeUserId, (id) => sql`and ${fields.id}<>${id}`)}
|
2021-12-20 14:20:23 +08:00
|
|
|
`);
|
2021-07-04 17:41:46 +08:00
|
|
|
|
|
|
|
export const hasUserWithId = async (id: string) =>
|
2022-04-19 21:49:20 +08:00
|
|
|
envSet.pool.exists(sql`
|
2021-12-20 14:20:23 +08:00
|
|
|
select ${fields.id}
|
|
|
|
from ${table}
|
|
|
|
where ${fields.id}=${id}
|
|
|
|
`);
|
2021-07-04 17:41:46 +08:00
|
|
|
|
2022-11-10 19:27:53 +08:00
|
|
|
export const hasUserWithEmail = async (email: string, excludeUserId?: string) =>
|
2022-04-19 21:49:20 +08:00
|
|
|
envSet.pool.exists(sql`
|
2022-02-07 14:14:42 +08:00
|
|
|
select ${fields.primaryEmail}
|
|
|
|
from ${table}
|
2022-11-09 15:33:07 +08:00
|
|
|
where lower(${fields.primaryEmail})=lower(${email})
|
2022-11-10 19:27:53 +08:00
|
|
|
${conditionalSql(excludeUserId, (id) => sql`and ${fields.id}<>${id}`)}
|
2022-02-07 14:14:42 +08:00
|
|
|
`);
|
|
|
|
|
2022-11-10 19:27:53 +08:00
|
|
|
export const hasUserWithPhone = async (phone: string, excludeUserId?: string) =>
|
2022-04-19 21:49:20 +08:00
|
|
|
envSet.pool.exists(sql`
|
2022-02-09 16:14:42 +08:00
|
|
|
select ${fields.primaryPhone}
|
|
|
|
from ${table}
|
|
|
|
where ${fields.primaryPhone}=${phone}
|
2022-11-10 19:27:53 +08:00
|
|
|
${conditionalSql(excludeUserId, (id) => sql`and ${fields.id}<>${id}`)}
|
2022-02-09 16:14:42 +08:00
|
|
|
`);
|
|
|
|
|
2022-06-17 11:00:02 +08:00
|
|
|
export const hasUserWithIdentity = async (target: string, userId: string) =>
|
2022-04-19 21:49:20 +08:00
|
|
|
envSet.pool.exists(
|
2022-02-11 15:19:18 +08:00
|
|
|
sql`
|
|
|
|
select ${fields.id}
|
|
|
|
from ${table}
|
2022-06-17 11:00:02 +08:00
|
|
|
where ${fields.identities}::json#>>'{${sql.identifier([target])},userId}' = ${userId}
|
2022-02-11 15:19:18 +08:00
|
|
|
`
|
|
|
|
);
|
|
|
|
|
2023-01-04 15:39:27 +08:00
|
|
|
const buildUserConditions = (search: Search, excludeUserIds: string[]) => {
|
2022-12-14 16:36:57 +08:00
|
|
|
const hasSearch = search.matches.length > 0;
|
|
|
|
const searchFields = [
|
|
|
|
Users.fields.id,
|
|
|
|
Users.fields.primaryEmail,
|
|
|
|
Users.fields.primaryPhone,
|
|
|
|
Users.fields.username,
|
|
|
|
Users.fields.name,
|
|
|
|
];
|
2022-02-16 15:55:08 +08:00
|
|
|
|
2023-01-04 15:39:27 +08:00
|
|
|
if (excludeUserIds.length > 0) {
|
|
|
|
// FIXME @sijie temp solution to filter out admin users,
|
|
|
|
// It is too complex to use join
|
2022-06-22 10:22:15 +08:00
|
|
|
return sql`
|
2023-01-04 15:39:27 +08:00
|
|
|
where ${fields.id} not in (${sql.join(excludeUserIds, sql`, `)})
|
2022-11-04 05:47:54 +03:00
|
|
|
${conditionalSql(
|
2022-12-14 16:36:57 +08:00
|
|
|
hasSearch,
|
|
|
|
() => sql`and (${buildConditionsFromSearch(search, searchFields)})`
|
2022-11-04 05:47:54 +03:00
|
|
|
)}
|
2022-06-22 10:22:15 +08:00
|
|
|
`;
|
|
|
|
}
|
|
|
|
|
2022-12-14 16:36:57 +08:00
|
|
|
return conditionalSql(
|
|
|
|
hasSearch,
|
|
|
|
() => sql`where ${buildConditionsFromSearch(search, searchFields)}`
|
|
|
|
);
|
2022-06-22 10:22:15 +08:00
|
|
|
};
|
|
|
|
|
2022-12-14 16:36:57 +08:00
|
|
|
export const defaultUserSearch = { matches: [], isCaseSensitive: false, joint: SearchJointMode.Or };
|
|
|
|
|
2023-01-04 15:39:27 +08:00
|
|
|
export const countUsers = async (
|
|
|
|
search: Search = defaultUserSearch,
|
|
|
|
excludeUserIds: string[] = []
|
|
|
|
) =>
|
2022-04-19 21:49:20 +08:00
|
|
|
envSet.pool.one<{ count: number }>(sql`
|
2022-02-24 12:29:34 +08:00
|
|
|
select count(*)
|
|
|
|
from ${table}
|
2023-01-04 15:39:27 +08:00
|
|
|
${buildUserConditions(search, excludeUserIds)}
|
2022-02-24 12:29:34 +08:00
|
|
|
`);
|
|
|
|
|
2022-06-22 10:22:15 +08:00
|
|
|
export const findUsers = async (
|
|
|
|
limit: number,
|
|
|
|
offset: number,
|
2022-12-14 16:36:57 +08:00
|
|
|
search: Search,
|
2023-01-04 15:39:27 +08:00
|
|
|
excludeUserIds: string[] = []
|
2022-06-22 10:22:15 +08:00
|
|
|
) =>
|
2022-04-19 21:49:20 +08:00
|
|
|
envSet.pool.any<User>(
|
2022-02-24 12:29:34 +08:00
|
|
|
sql`
|
2023-01-04 15:39:27 +08:00
|
|
|
select ${sql.join(
|
|
|
|
Object.values(fields).map((field) => sql`${table}.${field}`),
|
|
|
|
sql`,`
|
|
|
|
)}
|
2022-02-24 12:29:34 +08:00
|
|
|
from ${table}
|
2023-01-04 15:39:27 +08:00
|
|
|
${buildUserConditions(search, excludeUserIds)}
|
2022-02-24 12:29:34 +08:00
|
|
|
limit ${limit}
|
|
|
|
offset ${offset}
|
|
|
|
`
|
|
|
|
);
|
2021-11-18 11:26:34 +08:00
|
|
|
|
2023-01-04 15:39:27 +08:00
|
|
|
export const findUsersByIds = async (userIds: string[]) =>
|
2023-01-07 12:08:03 +08:00
|
|
|
userIds.length > 0
|
|
|
|
? envSet.pool.any<User>(sql`
|
|
|
|
select ${sql.join(Object.values(fields), sql`, `)}
|
|
|
|
from ${table}
|
|
|
|
where ${fields.id} in (${sql.join(userIds, sql`, `)})
|
|
|
|
`)
|
|
|
|
: [];
|
2023-01-04 15:39:27 +08:00
|
|
|
|
2022-04-19 21:49:20 +08:00
|
|
|
const updateUser = buildUpdateWhere<CreateUser, User>(Users, true);
|
2021-11-18 11:26:34 +08:00
|
|
|
|
2022-06-14 21:38:10 +08:00
|
|
|
export const updateUserById = async (
|
|
|
|
id: string,
|
|
|
|
set: Partial<OmitAutoSetFields<CreateUser>>,
|
|
|
|
jsonbMode: 'replace' | 'merge' = 'merge'
|
|
|
|
) => updateUser({ set, where: { id }, jsonbMode });
|
2021-11-18 11:26:34 +08:00
|
|
|
|
|
|
|
export const deleteUserById = async (id: string) => {
|
2022-04-19 21:49:20 +08:00
|
|
|
const { rowCount } = await envSet.pool.query(sql`
|
2021-12-20 14:20:23 +08:00
|
|
|
delete from ${table}
|
2022-02-28 14:30:27 +08:00
|
|
|
where ${fields.id}=${id}
|
2021-12-20 14:20:23 +08:00
|
|
|
`);
|
2022-01-27 19:26:34 +08:00
|
|
|
|
2021-11-18 11:26:34 +08:00
|
|
|
if (rowCount < 1) {
|
2022-02-14 11:50:47 +08:00
|
|
|
throw new DeletionError(Users.table, id);
|
2021-11-18 11:26:34 +08:00
|
|
|
}
|
|
|
|
};
|
2022-02-17 14:10:26 +08:00
|
|
|
|
2022-06-17 11:00:02 +08:00
|
|
|
export const deleteUserIdentity = async (userId: string, target: string) =>
|
2022-04-19 21:49:20 +08:00
|
|
|
envSet.pool.one<User>(sql`
|
2022-03-25 15:48:53 +08:00
|
|
|
update ${table}
|
2022-06-17 11:00:02 +08:00
|
|
|
set ${fields.identities}=${fields.identities}::jsonb-${target}
|
2022-03-25 15:48:53 +08:00
|
|
|
where ${fields.id}=${userId}
|
|
|
|
returning *
|
|
|
|
`);
|
2022-06-09 11:42:52 +08:00
|
|
|
|
|
|
|
export const hasActiveUsers = async () =>
|
|
|
|
envSet.pool.exists(sql`
|
|
|
|
select ${fields.id}
|
|
|
|
from ${table}
|
|
|
|
limit 1
|
|
|
|
`);
|
2022-09-29 15:32:43 +08:00
|
|
|
|
|
|
|
export const getDailyNewUserCountsByTimeInterval = async (
|
|
|
|
startTimeExclusive: number,
|
|
|
|
endTimeInclusive: number
|
|
|
|
) =>
|
|
|
|
envSet.pool.any<{ date: string; count: number }>(sql`
|
|
|
|
select date(${fields.createdAt}), count(*)
|
|
|
|
from ${table}
|
|
|
|
where ${fields.createdAt} > to_timestamp(${startTimeExclusive}::double precision / 1000)
|
|
|
|
and ${fields.createdAt} <= to_timestamp(${endTimeInclusive}::double precision / 1000)
|
|
|
|
group by date(${fields.createdAt})
|
|
|
|
`);
|
2023-01-04 15:39:27 +08:00
|
|
|
|
|
|
|
export const findUsersByRoleName = async (roleName: string) => {
|
|
|
|
const role = await findRoleByRoleName(roleName);
|
|
|
|
|
|
|
|
if (!role) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
const usersRoles = await findUsersRolesByRoleId(role.id);
|
|
|
|
|
|
|
|
return findUsersByIds(usersRoles.map(({ userId }) => userId));
|
|
|
|
};
|