2025-05-10 02:09:06 +02:00
|
|
|
/*!
|
|
|
|
|
* Copyright (c) 2024 PLANKA Software GmbH
|
|
|
|
|
* Licensed under the Fair Use License: https://github.com/plankanban/planka/blob/master/LICENSE.md
|
|
|
|
|
*/
|
|
|
|
|
|
2025-08-23 00:03:20 +02:00
|
|
|
const { makeWhereQueryBuilder } = require('../helpers');
|
|
|
|
|
|
|
|
|
|
const hasAvatarChanged = (avatar, prevAvatar) => {
|
|
|
|
|
if (!avatar && !prevAvatar) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!avatar || !prevAvatar) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return avatar.uploadedFileId !== prevAvatar.uploadedFileId;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const buildWhereQuery = makeWhereQueryBuilder(User);
|
|
|
|
|
|
2025-05-10 02:09:06 +02:00
|
|
|
const defaultFind = (criteria) => User.find(criteria).sort('id');
|
|
|
|
|
|
|
|
|
|
/* Query methods */
|
|
|
|
|
|
|
|
|
|
const createOne = (values) => {
|
2025-08-23 00:03:20 +02:00
|
|
|
if (!_.isNil(sails.config.custom.activeUsersLimit)) {
|
2025-05-10 02:09:06 +02:00
|
|
|
return sails.getDatastore().transaction(async (db) => {
|
|
|
|
|
const queryResult = await sails
|
|
|
|
|
.sendNativeQuery('SELECT NULL FROM user_account WHERE is_deactivated = $1 FOR UPDATE', [
|
|
|
|
|
false,
|
|
|
|
|
])
|
|
|
|
|
.usingConnection(db);
|
|
|
|
|
|
|
|
|
|
if (queryResult.rowCount >= sails.config.custom.activeUsersLimit) {
|
|
|
|
|
throw 'activeLimitReached';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return User.create({ ...values })
|
|
|
|
|
.fetch()
|
|
|
|
|
.usingConnection(db);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return User.create({ ...values }).fetch();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const getByIds = (ids) => defaultFind(ids);
|
|
|
|
|
|
|
|
|
|
const getAll = ({ roleOrRoles } = {}) =>
|
|
|
|
|
defaultFind({
|
|
|
|
|
role: roleOrRoles,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const getOneById = (id, { withDeactivated = true } = {}) => {
|
|
|
|
|
const criteria = {
|
|
|
|
|
id,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (!withDeactivated) {
|
|
|
|
|
criteria.isDeactivated = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return User.findOne(criteria);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const getOneByEmail = (email) =>
|
|
|
|
|
User.findOne({
|
|
|
|
|
email: email.toLowerCase(),
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const getOneActiveByEmailOrUsername = (emailOrUsername) => {
|
|
|
|
|
const fieldName = emailOrUsername.includes('@') ? 'email' : 'username';
|
|
|
|
|
|
|
|
|
|
return User.findOne({
|
|
|
|
|
[fieldName]: emailOrUsername.toLowerCase(),
|
|
|
|
|
isDeactivated: false,
|
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
|
2025-08-23 00:03:20 +02:00
|
|
|
const updateOne = async (criteria, values) => {
|
|
|
|
|
const enforceActiveLimit =
|
|
|
|
|
values.isDeactivated === false && !_.isNil(sails.config.custom.activeUsersLimit);
|
|
|
|
|
|
|
|
|
|
if (!_.isUndefined(values.avatar) || enforceActiveLimit) {
|
2025-05-10 02:09:06 +02:00
|
|
|
return sails.getDatastore().transaction(async (db) => {
|
2025-08-23 00:03:20 +02:00
|
|
|
if (enforceActiveLimit) {
|
|
|
|
|
const queryResult = await sails
|
|
|
|
|
.sendNativeQuery('SELECT NULL FROM user_account WHERE is_deactivated = $1 FOR UPDATE', [
|
|
|
|
|
false,
|
|
|
|
|
])
|
|
|
|
|
.usingConnection(db);
|
|
|
|
|
|
|
|
|
|
if (queryResult.rowCount >= sails.config.custom.activeUsersLimit) {
|
|
|
|
|
throw 'activeLimitReached';
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-05-10 02:09:06 +02:00
|
|
|
|
2025-08-25 17:23:59 +02:00
|
|
|
let prev;
|
2025-08-23 00:03:20 +02:00
|
|
|
if (!_.isUndefined(values.avatar)) {
|
|
|
|
|
const [whereQuery, whereQueryValues] = buildWhereQuery(criteria);
|
|
|
|
|
|
|
|
|
|
const queryResult = await sails
|
|
|
|
|
.sendNativeQuery(
|
2025-08-25 17:23:59 +02:00
|
|
|
`SELECT avatar FROM user_account WHERE ${whereQuery} LIMIT 1 FOR UPDATE`,
|
2025-08-23 00:03:20 +02:00
|
|
|
whereQueryValues,
|
|
|
|
|
)
|
|
|
|
|
.usingConnection(db);
|
|
|
|
|
|
|
|
|
|
if (queryResult.rowCount === 0) {
|
|
|
|
|
return { user: null };
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-25 17:23:59 +02:00
|
|
|
prev = {
|
|
|
|
|
avatar: queryResult.rows[0].avatar,
|
|
|
|
|
};
|
2025-05-10 02:09:06 +02:00
|
|
|
}
|
|
|
|
|
|
2025-08-23 00:03:20 +02:00
|
|
|
const user = await User.updateOne(criteria)
|
2025-05-10 02:09:06 +02:00
|
|
|
.set({ ...values })
|
|
|
|
|
.usingConnection(db);
|
2025-08-23 00:03:20 +02:00
|
|
|
|
|
|
|
|
let uploadedFile;
|
2025-08-25 17:23:59 +02:00
|
|
|
if (!_.isUndefined(values.avatar) && hasAvatarChanged(user.avatar, prev.avatar)) {
|
|
|
|
|
if (prev.avatar) {
|
2025-08-23 00:03:20 +02:00
|
|
|
const queryResult = await sails
|
|
|
|
|
.sendNativeQuery(
|
|
|
|
|
'UPDATE uploaded_file SET references_total = CASE WHEN references_total > 1 THEN references_total - 1 END, updated_at = $1 WHERE id = $2 RETURNING *',
|
2025-08-25 17:23:59 +02:00
|
|
|
[new Date().toISOString(), prev.avatar.uploadedFileId],
|
2025-08-23 00:03:20 +02:00
|
|
|
)
|
|
|
|
|
.usingConnection(db);
|
|
|
|
|
|
|
|
|
|
const [row] = queryResult.rows;
|
|
|
|
|
|
|
|
|
|
uploadedFile = {
|
|
|
|
|
id: row.id,
|
|
|
|
|
type: row.type,
|
|
|
|
|
mimeType: row.mime_type,
|
|
|
|
|
size: row.size,
|
|
|
|
|
referencesTotal: row.references_total,
|
|
|
|
|
createdAt: row.created_at,
|
|
|
|
|
updatedAt: row.updated_at,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (user.avatar) {
|
|
|
|
|
const queryResult = await sails
|
|
|
|
|
.sendNativeQuery(
|
|
|
|
|
'UPDATE uploaded_file SET references_total = references_total + 1, updated_at = $1 WHERE id = $2 AND references_total IS NOT NULL',
|
|
|
|
|
[new Date().toISOString(), user.avatar.uploadedFileId],
|
|
|
|
|
)
|
|
|
|
|
.usingConnection(db);
|
|
|
|
|
|
|
|
|
|
if (queryResult.rowCount === 0) {
|
|
|
|
|
throw 'uploadedFileNotFound';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return { user, uploadedFile };
|
2025-05-10 02:09:06 +02:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-23 00:03:20 +02:00
|
|
|
const user = await User.updateOne(criteria).set({ ...values });
|
|
|
|
|
return { user };
|
2025-05-10 02:09:06 +02:00
|
|
|
};
|
|
|
|
|
|
2025-08-23 00:03:20 +02:00
|
|
|
const deleteOne = (criteria) =>
|
|
|
|
|
sails.getDatastore().transaction(async (db) => {
|
|
|
|
|
const user = await User.destroyOne(criteria).usingConnection(db);
|
|
|
|
|
|
|
|
|
|
let uploadedFile;
|
|
|
|
|
if (user.avatar) {
|
|
|
|
|
const queryResult = await sails
|
|
|
|
|
.sendNativeQuery(
|
|
|
|
|
'UPDATE uploaded_file SET references_total = CASE WHEN references_total > 1 THEN references_total - 1 END, updated_at = $1 WHERE id = $2 RETURNING *',
|
|
|
|
|
[new Date().toISOString(), user.avatar.uploadedFileId],
|
|
|
|
|
)
|
|
|
|
|
.usingConnection(db);
|
|
|
|
|
|
|
|
|
|
const [row] = queryResult.rows;
|
|
|
|
|
|
|
|
|
|
uploadedFile = {
|
|
|
|
|
id: row.id,
|
|
|
|
|
type: row.type,
|
|
|
|
|
mimeType: row.mime_type,
|
|
|
|
|
size: row.size,
|
|
|
|
|
referencesTotal: row.references_total,
|
|
|
|
|
createdAt: row.created_at,
|
|
|
|
|
updatedAt: row.updated_at,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return { user, uploadedFile };
|
|
|
|
|
});
|
2025-05-10 02:09:06 +02:00
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
|
createOne,
|
|
|
|
|
getByIds,
|
|
|
|
|
getAll,
|
|
|
|
|
getOneById,
|
|
|
|
|
getOneByEmail,
|
|
|
|
|
getOneActiveByEmailOrUsername,
|
|
|
|
|
updateOne,
|
|
|
|
|
deleteOne,
|
|
|
|
|
};
|