mirror of
https://github.com/plankanban/planka.git
synced 2025-07-18 20:59:44 +02:00
161 lines
3.8 KiB
JavaScript
161 lines
3.8 KiB
JavaScript
const path = require('path');
|
|
const bcrypt = require('bcrypt');
|
|
const rimraf = require('rimraf');
|
|
const { v4: uuid } = require('uuid');
|
|
|
|
const valuesValidator = (value) => {
|
|
if (!_.isPlainObject(value)) {
|
|
return false;
|
|
}
|
|
|
|
if (!_.isUndefined(value.email) && !_.isString(value.email)) {
|
|
return false;
|
|
}
|
|
|
|
if (!_.isUndefined(value.password) && !_.isString(value.password)) {
|
|
return false;
|
|
}
|
|
|
|
if (!_.isNil(value.username) && !_.isString(value.username)) {
|
|
return false;
|
|
}
|
|
|
|
if (!_.isNil(value.avatar) && !_.isPlainObject(value.avatar)) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
module.exports = {
|
|
inputs: {
|
|
record: {
|
|
type: 'ref',
|
|
required: true,
|
|
},
|
|
values: {
|
|
type: 'json',
|
|
custom: valuesValidator,
|
|
required: true,
|
|
},
|
|
user: {
|
|
type: 'ref',
|
|
required: true,
|
|
},
|
|
request: {
|
|
type: 'ref',
|
|
},
|
|
},
|
|
|
|
exits: {
|
|
emailAlreadyInUse: {},
|
|
usernameAlreadyInUse: {},
|
|
},
|
|
|
|
async fn(inputs) {
|
|
const { values } = inputs;
|
|
|
|
if (!_.isUndefined(values.email)) {
|
|
values.email = values.email.toLowerCase();
|
|
}
|
|
|
|
let isOnlyPasswordChange = false;
|
|
|
|
if (!_.isUndefined(values.password)) {
|
|
Object.assign(values, {
|
|
password: bcrypt.hashSync(values.password, 10),
|
|
passwordChangedAt: new Date().toUTCString(),
|
|
});
|
|
|
|
if (Object.keys(values).length === 1) {
|
|
isOnlyPasswordChange = true;
|
|
}
|
|
}
|
|
|
|
if (values.username) {
|
|
values.username = values.username.toLowerCase();
|
|
}
|
|
|
|
const user = await User.updateOne({
|
|
id: inputs.record.id,
|
|
deletedAt: null,
|
|
})
|
|
.set({ ...values })
|
|
.intercept(
|
|
{
|
|
message:
|
|
'Unexpected error from database adapter: conflicting key value violates exclusion constraint "user_email_unique"',
|
|
},
|
|
'emailAlreadyInUse',
|
|
)
|
|
.intercept(
|
|
{
|
|
message:
|
|
'Unexpected error from database adapter: conflicting key value violates exclusion constraint "user_username_unique"',
|
|
},
|
|
'usernameAlreadyInUse',
|
|
);
|
|
|
|
if (user) {
|
|
if (
|
|
inputs.record.avatar &&
|
|
(!user.avatar || user.avatar.dirname !== inputs.record.avatar.dirname)
|
|
) {
|
|
try {
|
|
rimraf.sync(path.join(sails.config.custom.userAvatarsPath, inputs.record.avatar.dirname));
|
|
} catch (error) {
|
|
console.warn(error.stack); // eslint-disable-line no-console
|
|
}
|
|
}
|
|
|
|
if (!_.isUndefined(values.password)) {
|
|
sails.sockets.broadcast(
|
|
`user:${user.id}`,
|
|
'userDelete', // TODO: introduce separate event
|
|
{
|
|
item: user,
|
|
},
|
|
inputs.request,
|
|
);
|
|
|
|
if (user.id === inputs.user.id && inputs.request && inputs.request.isSocket) {
|
|
const tempRoom = uuid();
|
|
|
|
sails.sockets.addRoomMembersToRooms(`@user:${user.id}`, tempRoom, () => {
|
|
sails.sockets.leave(inputs.request, tempRoom, () => {
|
|
sails.sockets.leaveAll(tempRoom);
|
|
});
|
|
});
|
|
} else {
|
|
sails.sockets.leaveAll(`@user:${user.id}`);
|
|
}
|
|
}
|
|
|
|
if (!isOnlyPasswordChange) {
|
|
/* const projectIds = await sails.helpers.users.getManagerProjectIds(user.id);
|
|
|
|
const userIds = _.union(
|
|
[user.id],
|
|
await sails.helpers.users.getAdminIds(),
|
|
await sails.helpers.projects.getManagerAndBoardMemberUserIds(projectIds),
|
|
); */
|
|
|
|
const users = await sails.helpers.users.getMany();
|
|
const userIds = sails.helpers.utils.mapRecords(users);
|
|
|
|
userIds.forEach((userId) => {
|
|
sails.sockets.broadcast(
|
|
`user:${userId}`,
|
|
'userUpdate',
|
|
{
|
|
item: user,
|
|
},
|
|
inputs.request,
|
|
);
|
|
});
|
|
}
|
|
}
|
|
|
|
return user;
|
|
},
|
|
};
|