1
0
Fork 0
mirror of https://github.com/plankanban/planka.git synced 2025-07-18 20:59:44 +02:00
planka/server/api/helpers/cards/update-one.js

545 lines
15 KiB
JavaScript
Raw Normal View History

/*!
* Copyright (c) 2024 PLANKA Software GmbH
* Licensed under the Fair Use License: https://github.com/plankanban/planka/blob/master/LICENSE.md
*/
2022-12-26 21:10:50 +01:00
const { POSITION_GAP } = require('../../../constants');
2022-12-26 21:10:50 +01:00
2019-08-31 04:07:25 +05:00
module.exports = {
inputs: {
record: {
type: 'ref',
required: true,
2019-08-31 04:07:25 +05:00
},
values: {
type: 'json',
2022-12-26 21:10:50 +01:00
required: true,
},
project: {
2019-08-31 04:07:25 +05:00
type: 'ref',
required: true,
2019-08-31 04:07:25 +05:00
},
board: {
type: 'ref',
required: true,
},
list: {
type: 'ref',
required: true,
},
actorUser: {
type: 'ref',
required: true,
},
2019-08-31 04:07:25 +05:00
request: {
type: 'ref',
},
2019-08-31 04:07:25 +05:00
},
2020-04-23 03:02:53 +05:00
exits: {
positionMustBeInValues: {},
boardInValuesMustBelongToProject: {},
2022-12-26 21:10:50 +01:00
listMustBeInValues: {},
listInValuesMustBelongToBoard: {},
coverAttachmentInValuesMustContainImage: {},
2020-04-23 03:02:53 +05:00
},
// TODO: use normalizeValues and refactor
async fn(inputs) {
2019-08-31 04:07:25 +05:00
const { isSubscribed, ...values } = inputs.values;
if (values.project && values.project.id === inputs.project.id) {
delete values.project;
}
2019-08-31 04:07:25 +05:00
const project = values.project || inputs.project;
if (values.board) {
if (values.board.projectId !== project.id) {
throw 'boardInValuesMustBelongToProject';
}
if (values.board.id === inputs.board.id) {
delete values.board;
} else {
values.boardId = values.board.id;
}
}
const board = values.board || inputs.board;
if (values.list) {
if (values.list.boardId !== board.id) {
throw 'listInValuesMustBelongToBoard';
}
if (values.list.id === inputs.list.id) {
delete values.list;
} else {
values.listId = values.list.id;
2019-08-31 04:07:25 +05:00
}
} else if (values.board) {
throw 'listMustBeInValues';
2020-04-23 03:02:53 +05:00
}
const list = values.list || inputs.list;
if (sails.helpers.lists.isFinite(list)) {
if (values.list && _.isUndefined(values.position)) {
throw 'positionMustBeInValues';
}
} else {
values.position = null;
2019-08-31 04:07:25 +05:00
}
if (values.coverAttachment) {
if (!values.coverAttachment.data.image) {
throw 'coverAttachmentInValuesMustContainImage';
}
2019-08-31 04:07:25 +05:00
values.coverAttachmentId = values.coverAttachment.id;
}
2022-12-26 21:10:50 +01:00
let card;
if (_.isEmpty(values)) {
card = inputs.record;
} else {
if (!_.isNil(values.position)) {
const cards = await Card.qm.getByListId(list.id, {
exceptIdOrIds: inputs.record.id,
2019-08-31 04:07:25 +05:00
});
const { position, repositions } = sails.helpers.utils.insertToPositionables(
values.position,
cards,
);
values.position = position;
2019-08-31 04:07:25 +05:00
if (repositions.length > 0) {
// eslint-disable-next-line no-restricted-syntax
for (const reposition of repositions) {
// eslint-disable-next-line no-await-in-loop
await Card.qm.updateOne(
{
id: reposition.record.id,
listId: reposition.record.listId,
},
{
position: reposition.position,
},
);
2024-08-12 23:17:17 +02:00
sails.sockets.broadcast(`board:${board.id}`, 'cardUpdate', {
item: {
id: reposition.record.id,
position: reposition.position,
},
});
2024-08-12 23:17:17 +02:00
// TODO: send webhooks
}
}
2024-08-12 23:17:17 +02:00
}
let prevLabels;
2022-12-26 21:10:50 +01:00
if (values.board) {
prevLabels = await sails.helpers.cards.getLabels(inputs.record.id);
2022-12-26 21:10:50 +01:00
const boardMemberUserIds = await sails.helpers.boards.getMemberUserIds(values.board.id);
await CardSubscription.qm.delete({
2022-12-26 21:10:50 +01:00
cardId: inputs.record.id,
userId: {
'!=': boardMemberUserIds,
},
});
await CardMembership.qm.delete({
2022-12-26 21:10:50 +01:00
cardId: inputs.record.id,
userId: {
'!=': boardMemberUserIds,
},
});
await CardLabel.qm.delete({
cardId: inputs.record.id,
});
const taskLists = await TaskList.qm.getByCardId(inputs.record.id);
const taskListIds = sails.helpers.utils.mapRecords(taskLists);
await Task.qm.update(
{
taskListId: taskListIds,
assigneeUserId: {
'!=': boardMemberUserIds,
},
},
{
assigneeUserId: null,
},
);
const boardCustomFieldGroups = await CustomFieldGroup.qm.getByBoardId(inputs.board.id);
const boardCustomFieldGroupIds = sails.helpers.utils.mapRecords(boardCustomFieldGroups);
const boardCustomFields =
await CustomField.qm.getByCustomFieldGroupIds(boardCustomFieldGroupIds);
const cardCustomFieldGroups = await CustomFieldGroup.qm.getByCardId(inputs.record.id);
let basedCardCustomFieldGroups;
let basedCustomFieldGroups;
let baseCustomFieldGroupById;
let customFieldsByBaseCustomFieldGroupId;
if (values.project) {
const basedBoardCustomFieldGroups = boardCustomFieldGroups.filter(
({ baseCustomFieldGroupId }) => baseCustomFieldGroupId,
);
basedCardCustomFieldGroups = cardCustomFieldGroups.filter(
({ baseCustomFieldGroupId }) => baseCustomFieldGroupId,
);
basedCustomFieldGroups = [...basedBoardCustomFieldGroups, ...basedCardCustomFieldGroups];
const baseCustomFieldGroupIds = sails.helpers.utils.mapRecords(
basedCustomFieldGroups,
'baseCustomFieldGroupId',
true,
);
const baseCustomFieldGroups =
await BaseCustomFieldGroup.qm.getByIds(baseCustomFieldGroupIds);
baseCustomFieldGroupById = _.keyBy(baseCustomFieldGroups, 'id');
const baseCustomFields = await CustomField.qm.getByBaseCustomFieldGroupIds(
Object.keys(baseCustomFieldGroupById),
);
customFieldsByBaseCustomFieldGroupId = _.groupBy(
baseCustomFields,
'baseCustomFieldGroupId',
);
}
let idsTotal = boardCustomFieldGroups.length + boardCustomFields.length;
if (values.project) {
idsTotal += basedCustomFieldGroups.reduce((result, customFieldGroup) => {
const customFieldsItem =
customFieldsByBaseCustomFieldGroupId[customFieldGroup.baseCustomFieldGroupId];
return result + (customFieldsItem ? customFieldsItem.length : 0);
}, 0);
}
const ids = await sails.helpers.utils.generateIds(idsTotal);
const nextCustomFieldGroupIdByCustomFieldGroupId = {};
const nextCustomFieldGroupsValues = boardCustomFieldGroups.map(
(customFieldGroup, index) => {
const id = ids.shift();
nextCustomFieldGroupIdByCustomFieldGroupId[customFieldGroup.id] = id;
const nextValues = {
..._.pick(customFieldGroup, ['baseCustomFieldGroupId', 'name']),
id,
cardId: inputs.record.id,
position: POSITION_GAP * (index + 1),
};
if (values.project && customFieldGroup.baseCustomFieldGroupId) {
nextValues.baseCustomFieldGroupId = null;
if (!customFieldGroup.name) {
nextValues.name =
baseCustomFieldGroupById[customFieldGroup.baseCustomFieldGroupId].name;
}
}
return nextValues;
},
);
if (nextCustomFieldGroupsValues.length > 0) {
const { position } = nextCustomFieldGroupsValues[nextCustomFieldGroupsValues.length - 1];
await Promise.all(
cardCustomFieldGroups.map((customFieldGroup) =>
CustomFieldGroup.qm.updateOne(customFieldGroup.id, {
position: customFieldGroup.position + position,
}),
),
);
}
await CustomFieldGroup.qm.create(nextCustomFieldGroupsValues);
if (values.project) {
await CustomFieldGroup.qm.update(
{
cardId: inputs.record.id,
baseCustomFieldGroupId: {
'!=': null,
},
},
{
baseCustomFieldGroupId: null,
},
);
const unnamedCustomFieldGroups = basedCardCustomFieldGroups.filter(({ name }) => !name);
await Promise.all(
unnamedCustomFieldGroups.map((customFieldGroup) =>
CustomFieldGroup.qm.updateOne(customFieldGroup.id, {
name: baseCustomFieldGroupById[customFieldGroup.baseCustomFieldGroupId].name,
}),
),
);
}
const nextCustomFieldIdByCustomFieldId = {};
const nextCustomFieldsValues = boardCustomFields.map((customField) => {
const id = ids.shift();
nextCustomFieldIdByCustomFieldId[customField.id] = id;
return {
..._.pick(customField, ['name', 'showOnFrontOfCard', 'position']),
id,
customFieldGroupId:
nextCustomFieldGroupIdByCustomFieldGroupId[customField.customFieldGroupId],
};
});
if (values.project) {
basedCustomFieldGroups.forEach((customFieldGroup) => {
const customFieldsItem =
customFieldsByBaseCustomFieldGroupId[customFieldGroup.baseCustomFieldGroupId];
if (!customFieldsItem) {
return;
}
customFieldsItem.forEach((customField) => {
const id = ids.shift();
nextCustomFieldIdByCustomFieldId[`${customFieldGroup.id}:${customField.id}`] = id;
nextCustomFieldsValues.push({
..._.pick(customField, ['name', 'showOnFrontOfCard', 'position']),
id,
customFieldGroupId:
nextCustomFieldGroupIdByCustomFieldGroupId[customFieldGroup.id] ||
customFieldGroup.id,
});
});
});
}
await CustomField.qm.create(nextCustomFieldsValues);
const customFieldGroupIds = boardCustomFieldGroupIds;
if (values.project) {
customFieldGroupIds.push(...sails.helpers.utils.mapRecords(basedCardCustomFieldGroups));
}
const customFieldValues = await CustomFieldValue.qm.getByCardId(inputs.record.id, {
customFieldGroupIdOrIds: customFieldGroupIds,
});
await Promise.all(
customFieldValues.map((customFieldValue) => {
const updateValues = {
customFieldGroupId:
nextCustomFieldGroupIdByCustomFieldGroupId[customFieldValue.customFieldGroupId],
};
const nextCustomFieldId =
nextCustomFieldIdByCustomFieldId[
`${customFieldValue.customFieldGroupId}:${customFieldValue.customFieldId}`
] || nextCustomFieldIdByCustomFieldId[customFieldValue.customFieldId];
if (nextCustomFieldId) {
updateValues.customFieldId = nextCustomFieldId;
}
return CustomFieldValue.qm.updateOne(customFieldValue.id, updateValues);
}),
);
}
if (values.list) {
values.listChangedAt = new Date().toISOString();
if (values.board || inputs.list.type === List.Types.TRASH) {
values.prevListId = null;
} else if (sails.helpers.lists.isArchiveOrTrash(values.list)) {
values.prevListId = inputs.list.id;
} else if (inputs.list.type === List.Types.ARCHIVE) {
values.prevListId = null;
}
}
card = await Card.qm.updateOne(inputs.record.id, values);
2019-08-31 04:07:25 +05:00
if (!card) {
return card;
2019-08-31 04:07:25 +05:00
}
2022-12-26 21:10:50 +01:00
if (values.board) {
const labels = await Label.qm.getByBoardId(card.boardId);
2022-12-26 21:10:50 +01:00
const labelByName = _.keyBy(labels, 'name');
const labelIds = await Promise.all(
2022-12-26 21:10:50 +01:00
prevLabels.map(async (label) => {
if (labelByName[label.name]) {
return labelByName[label.name].id;
}
2022-12-26 21:10:50 +01:00
const { id } = await sails.helpers.labels.createOne.with({
project,
2022-12-26 21:10:50 +01:00
values: {
..._.omit(label, ['id', 'boardId', 'createdAt', 'updatedAt']),
board,
2022-12-26 21:10:50 +01:00
},
actorUser: inputs.actorUser,
2022-12-26 21:10:50 +01:00
});
return id;
}),
);
await Promise.all(
labelIds.map((labelId) => {
try {
return CardLabel.qm.createOne({
labelId,
cardId: card.id,
});
} catch (error) {
if (error.code !== 'E_UNIQUE') {
throw error;
}
}
return Promise.resolve();
}),
);
sails.sockets.broadcast(
`board:${inputs.board.id}`,
'cardUpdate',
{
item: {
id: card.id,
boardId: null,
},
},
inputs.request,
);
sails.sockets.broadcast(`board:${card.boardId}`, 'cardUpdate', {
item: card,
});
// TODO: add transfer action
} else {
sails.sockets.broadcast(
`board:${card.boardId}`,
'cardUpdate',
{
item: card,
},
inputs.request,
);
if (values.list) {
await sails.helpers.actions.createOne.with({
values: {
card,
type: Action.Types.MOVE_CARD,
data: {
2025-05-22 23:14:46 +02:00
card: _.pick(card, ['name']),
fromList: _.pick(inputs.list, ['id', 'type', 'name']),
toList: _.pick(values.list, ['id', 'type', 'name']),
},
user: inputs.actorUser,
},
project: inputs.project,
board: inputs.board,
list: values.list,
});
}
}
2019-08-31 04:07:25 +05:00
sails.helpers.utils.sendWebhooks.with({
event: 'cardUpdate',
buildData: () => ({
item: card,
included: {
projects: [project],
boards: [board],
lists: [list],
},
}),
buildPrevData: () => ({
item: inputs.record,
included: {
projects: [inputs.project],
boards: [inputs.board],
lists: [inputs.list],
},
}),
user: inputs.actorUser,
});
2019-08-31 04:07:25 +05:00
}
if (!_.isUndefined(isSubscribed)) {
const wasSubscribed = await sails.helpers.users.isCardSubscriber(
inputs.actorUser.id,
card.id,
);
2019-08-31 04:07:25 +05:00
if (isSubscribed !== wasSubscribed) {
2019-08-31 04:07:25 +05:00
if (isSubscribed) {
try {
await CardSubscription.qm.createOne({
cardId: card.id,
userId: inputs.actorUser.id,
});
} catch (error) {
if (error.code !== 'E_UNIQUE') {
throw error;
}
}
2019-08-31 04:07:25 +05:00
} else {
await CardSubscription.qm.deleteOne({
2019-08-31 04:07:25 +05:00
cardId: card.id,
userId: inputs.actorUser.id,
2019-08-31 04:07:25 +05:00
});
}
sails.sockets.broadcast(
`user:${inputs.actorUser.id}`,
2019-08-31 04:07:25 +05:00
'cardUpdate',
{
item: {
isSubscribed,
id: card.id,
},
2019-08-31 04:07:25 +05:00
},
inputs.request,
2019-08-31 04:07:25 +05:00
);
// TODO: send webhooks
2019-08-31 04:07:25 +05:00
}
}
return card;
},
2019-08-31 04:07:25 +05:00
};