cards.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813
  1. const escapeForRegex = require('escape-string-regexp');
  2. Meteor.publish('card', cardId => {
  3. check(cardId, String);
  4. return Cards.find({ _id: cardId });
  5. });
  6. Meteor.publish('myCards', function() {
  7. const userId = Meteor.userId();
  8. const archivedBoards = [];
  9. Boards.find({ archived: true }).forEach(board => {
  10. archivedBoards.push(board._id);
  11. });
  12. const archivedSwimlanes = [];
  13. Swimlanes.find({ archived: true }).forEach(swimlane => {
  14. archivedSwimlanes.push(swimlane._id);
  15. });
  16. const archivedLists = [];
  17. Lists.find({ archived: true }).forEach(list => {
  18. archivedLists.push(list._id);
  19. });
  20. selector = {
  21. archived: false,
  22. boardId: { $nin: archivedBoards },
  23. swimlaneId: { $nin: archivedSwimlanes },
  24. listId: { $nin: archivedLists },
  25. $or: [{ members: userId }, { assignees: userId }],
  26. };
  27. const cards = Cards.find(selector, {
  28. fields: {
  29. _id: 1,
  30. archived: 1,
  31. boardId: 1,
  32. swimlaneId: 1,
  33. listId: 1,
  34. title: 1,
  35. type: 1,
  36. sort: 1,
  37. members: 1,
  38. assignees: 1,
  39. colors: 1,
  40. dueAt: 1,
  41. },
  42. });
  43. const boards = [];
  44. const swimlanes = [];
  45. const lists = [];
  46. const users = [];
  47. cards.forEach(card => {
  48. if (card.boardId) boards.push(card.boardId);
  49. if (card.swimlaneId) swimlanes.push(card.swimlaneId);
  50. if (card.listId) lists.push(card.listId);
  51. if (card.members) {
  52. card.members.forEach(userId => {
  53. users.push(userId);
  54. });
  55. }
  56. if (card.assignees) {
  57. card.assignees.forEach(userId => {
  58. users.push(userId);
  59. });
  60. }
  61. });
  62. return [
  63. cards,
  64. Boards.find({ _id: { $in: boards } }),
  65. Swimlanes.find({ _id: { $in: swimlanes } }),
  66. Lists.find({ _id: { $in: lists } }),
  67. Users.find({ _id: { $in: users } }, { fields: Users.safeFields }),
  68. ];
  69. });
  70. Meteor.publish('dueCards', function(allUsers = false) {
  71. check(allUsers, Boolean);
  72. // eslint-disable-next-line no-console
  73. // console.log('all users:', allUsers);
  74. const user = Users.findOne({ _id: this.userId });
  75. const archivedBoards = [];
  76. Boards.find({ archived: true }).forEach(board => {
  77. archivedBoards.push(board._id);
  78. });
  79. const permiitedBoards = [];
  80. let selector = {
  81. archived: false,
  82. };
  83. selector.$or = [
  84. { permission: 'public' },
  85. { members: { $elemMatch: { userId: user._id, isActive: true } } },
  86. ];
  87. Boards.find(selector).forEach(board => {
  88. permiitedBoards.push(board._id);
  89. });
  90. const archivedSwimlanes = [];
  91. Swimlanes.find({ archived: true }).forEach(swimlane => {
  92. archivedSwimlanes.push(swimlane._id);
  93. });
  94. const archivedLists = [];
  95. Lists.find({ archived: true }).forEach(list => {
  96. archivedLists.push(list._id);
  97. });
  98. selector = {
  99. archived: false,
  100. boardId: { $nin: archivedBoards, $in: permiitedBoards },
  101. swimlaneId: { $nin: archivedSwimlanes },
  102. listId: { $nin: archivedLists },
  103. dueAt: { $ne: null },
  104. endAt: null,
  105. };
  106. if (!allUsers) {
  107. selector.$or = [{ members: user._id }, { assignees: user._id }];
  108. }
  109. const cards = Cards.find(selector, {
  110. fields: {
  111. _id: 1,
  112. archived: 1,
  113. boardId: 1,
  114. swimlaneId: 1,
  115. listId: 1,
  116. title: 1,
  117. type: 1,
  118. sort: 1,
  119. members: 1,
  120. assignees: 1,
  121. colors: 1,
  122. dueAt: 1,
  123. },
  124. });
  125. const boards = [];
  126. const swimlanes = [];
  127. const lists = [];
  128. const users = [];
  129. cards.forEach(card => {
  130. if (card.boardId) boards.push(card.boardId);
  131. if (card.swimlaneId) swimlanes.push(card.swimlaneId);
  132. if (card.listId) lists.push(card.listId);
  133. if (card.members) {
  134. card.members.forEach(userId => {
  135. users.push(userId);
  136. });
  137. }
  138. if (card.assignees) {
  139. card.assignees.forEach(userId => {
  140. users.push(userId);
  141. });
  142. }
  143. });
  144. return [
  145. cards,
  146. Boards.find({ _id: { $in: boards } }),
  147. Swimlanes.find({ _id: { $in: swimlanes } }),
  148. Lists.find({ _id: { $in: lists } }),
  149. Users.find({ _id: { $in: users } }, { fields: Users.safeFields }),
  150. ];
  151. });
  152. Meteor.publish('globalSearch', function(sessionId, queryParams) {
  153. check(sessionId, String);
  154. check(queryParams, Object);
  155. // eslint-disable-next-line no-console
  156. // console.log('queryParams:', queryParams);
  157. const userId = Meteor.userId();
  158. // eslint-disable-next-line no-console
  159. // console.log('userId:', userId);
  160. const errors = new (class {
  161. constructor() {
  162. this.notFound = {
  163. boards: [],
  164. swimlanes: [],
  165. lists: [],
  166. labels: [],
  167. users: [],
  168. members: [],
  169. assignees: [],
  170. status: [],
  171. comments: [],
  172. };
  173. this.colorMap = Boards.colorMap();
  174. }
  175. hasErrors() {
  176. for (const value of Object.values(this.notFound)) {
  177. if (value.length) {
  178. return true;
  179. }
  180. }
  181. return false;
  182. }
  183. errorMessages() {
  184. const messages = [];
  185. this.notFound.boards.forEach(board => {
  186. messages.push({ tag: 'board-title-not-found', value: board });
  187. });
  188. this.notFound.swimlanes.forEach(swim => {
  189. messages.push({ tag: 'swimlane-title-not-found', value: swim });
  190. });
  191. this.notFound.lists.forEach(list => {
  192. messages.push({ tag: 'list-title-not-found', value: list });
  193. });
  194. this.notFound.comments.forEach(comments => {
  195. comments.forEach(text => {
  196. messages.push({ tag: 'comment-not-found', value: text });
  197. });
  198. });
  199. this.notFound.labels.forEach(label => {
  200. messages.push({
  201. tag: 'label-not-found',
  202. value: label,
  203. color: Boards.labelColors().includes(label),
  204. });
  205. });
  206. this.notFound.users.forEach(user => {
  207. messages.push({ tag: 'user-username-not-found', value: user });
  208. });
  209. this.notFound.members.forEach(user => {
  210. messages.push({ tag: 'user-username-not-found', value: user });
  211. });
  212. this.notFound.assignees.forEach(user => {
  213. messages.push({ tag: 'user-username-not-found', value: user });
  214. });
  215. return messages;
  216. }
  217. })();
  218. let selector = {};
  219. let skip = 0;
  220. if (queryParams.skip) {
  221. skip = queryParams.skip;
  222. }
  223. let limit = 25;
  224. if (queryParams.limit) {
  225. limit = queryParams.limit;
  226. }
  227. if (queryParams.selector) {
  228. selector = queryParams.selector;
  229. } else {
  230. const boardsSelector = {};
  231. let archived = false;
  232. let endAt = null;
  233. if (queryParams.status.length) {
  234. queryParams.status.forEach(status => {
  235. if (status === 'archived') {
  236. archived = true;
  237. } else if (status === 'all') {
  238. archived = null;
  239. } else if (status === 'ended') {
  240. endAt = { $nin: [null, ''] };
  241. } else if (['private', 'public'].includes(status)) {
  242. boardsSelector.permission = status;
  243. }
  244. });
  245. }
  246. selector = {
  247. type: 'cardType-card',
  248. // boardId: { $in: Boards.userBoardIds(userId) },
  249. $and: [],
  250. };
  251. if (archived !== null) {
  252. if (archived) {
  253. selector.boardId = {
  254. $in: Boards.userBoardIds(userId, null, boardsSelector),
  255. };
  256. selector.$and.push({
  257. $or: [
  258. {
  259. boardId: {
  260. $in: Boards.userBoardIds(userId, archived, boardsSelector),
  261. },
  262. },
  263. { swimlaneId: { $in: Swimlanes.archivedSwimlaneIds() } },
  264. { listId: { $in: Lists.archivedListIds() } },
  265. { archived: true },
  266. ],
  267. });
  268. } else {
  269. selector.boardId = {
  270. $in: Boards.userBoardIds(userId, false, boardsSelector),
  271. };
  272. selector.swimlaneId = { $nin: Swimlanes.archivedSwimlaneIds() };
  273. selector.listId = { $nin: Lists.archivedListIds() };
  274. selector.archived = false;
  275. }
  276. } else {
  277. selector.boardId = {
  278. $in: Boards.userBoardIds(userId, null, boardsSelector),
  279. };
  280. }
  281. if (endAt !== null) {
  282. selector.endAt = endAt;
  283. }
  284. if (queryParams.boards.length) {
  285. const queryBoards = [];
  286. queryParams.boards.forEach(query => {
  287. const boards = Boards.userSearch(userId, {
  288. title: new RegExp(escapeForRegex(query), 'i'),
  289. });
  290. if (boards.count()) {
  291. boards.forEach(board => {
  292. queryBoards.push(board._id);
  293. });
  294. } else {
  295. errors.notFound.boards.push(query);
  296. }
  297. });
  298. selector.boardId.$in = queryBoards;
  299. }
  300. if (queryParams.swimlanes.length) {
  301. const querySwimlanes = [];
  302. queryParams.swimlanes.forEach(query => {
  303. const swimlanes = Swimlanes.find({
  304. title: new RegExp(escapeForRegex(query), 'i'),
  305. });
  306. if (swimlanes.count()) {
  307. swimlanes.forEach(swim => {
  308. querySwimlanes.push(swim._id);
  309. });
  310. } else {
  311. errors.notFound.swimlanes.push(query);
  312. }
  313. });
  314. // eslint-disable-next-line no-prototype-builtins
  315. if (!selector.swimlaneId.hasOwnProperty('swimlaneId')) {
  316. selector.swimlaneId = { $in: [] };
  317. }
  318. selector.swimlaneId.$in = querySwimlanes;
  319. }
  320. if (queryParams.lists.length) {
  321. const queryLists = [];
  322. queryParams.lists.forEach(query => {
  323. const lists = Lists.find({
  324. title: new RegExp(escapeForRegex(query), 'i'),
  325. });
  326. if (lists.count()) {
  327. lists.forEach(list => {
  328. queryLists.push(list._id);
  329. });
  330. } else {
  331. errors.notFound.lists.push(query);
  332. }
  333. });
  334. // eslint-disable-next-line no-prototype-builtins
  335. if (!selector.hasOwnProperty('listId')) {
  336. selector.listId = { $in: [] };
  337. }
  338. selector.listId.$in = queryLists;
  339. }
  340. if (queryParams.comments.length) {
  341. const cardIds = CardComments.textSearch(userId, queryParams.comments).map(
  342. com => {
  343. return com.cardId;
  344. },
  345. );
  346. if (cardIds.length) {
  347. selector._id = { $in: cardIds };
  348. } else {
  349. errors.notFound.comments.push(queryParams.comments);
  350. }
  351. }
  352. if (queryParams.dueAt !== null) {
  353. selector.dueAt = { $lte: new Date(queryParams.dueAt) };
  354. }
  355. if (queryParams.createdAt !== null) {
  356. selector.createdAt = { $gte: new Date(queryParams.createdAt) };
  357. }
  358. if (queryParams.modifiedAt !== null) {
  359. selector.modifiedAt = { $gte: new Date(queryParams.modifiedAt) };
  360. }
  361. const queryMembers = [];
  362. const queryAssignees = [];
  363. if (queryParams.users.length) {
  364. queryParams.users.forEach(query => {
  365. const users = Users.find({
  366. username: query,
  367. });
  368. if (users.count()) {
  369. users.forEach(user => {
  370. queryMembers.push(user._id);
  371. queryAssignees.push(user._id);
  372. });
  373. } else {
  374. errors.notFound.users.push(query);
  375. }
  376. });
  377. }
  378. if (queryParams.members.length) {
  379. queryParams.members.forEach(query => {
  380. const users = Users.find({
  381. username: query,
  382. });
  383. if (users.count()) {
  384. users.forEach(user => {
  385. queryMembers.push(user._id);
  386. });
  387. } else {
  388. errors.notFound.members.push(query);
  389. }
  390. });
  391. }
  392. if (queryParams.assignees.length) {
  393. queryParams.assignees.forEach(query => {
  394. const users = Users.find({
  395. username: query,
  396. });
  397. if (users.count()) {
  398. users.forEach(user => {
  399. queryAssignees.push(user._id);
  400. });
  401. } else {
  402. errors.notFound.assignees.push(query);
  403. }
  404. });
  405. }
  406. if (queryMembers.length && queryAssignees.length) {
  407. selector.$and.push({
  408. $or: [
  409. { members: { $in: queryMembers } },
  410. { assignees: { $in: queryAssignees } },
  411. ],
  412. });
  413. } else if (queryMembers.length) {
  414. selector.members = { $in: queryMembers };
  415. } else if (queryAssignees.length) {
  416. selector.assignees = { $in: queryAssignees };
  417. }
  418. if (queryParams.labels.length) {
  419. queryParams.labels.forEach(label => {
  420. const queryLabels = [];
  421. let boards = Boards.userSearch(userId, {
  422. labels: { $elemMatch: { color: label.toLowerCase() } },
  423. });
  424. if (boards.count()) {
  425. boards.forEach(board => {
  426. // eslint-disable-next-line no-console
  427. // console.log('board:', board);
  428. // eslint-disable-next-line no-console
  429. // console.log('board.labels:', board.labels);
  430. board.labels
  431. .filter(boardLabel => {
  432. return boardLabel.color === label.toLowerCase();
  433. })
  434. .forEach(boardLabel => {
  435. queryLabels.push(boardLabel._id);
  436. });
  437. });
  438. } else {
  439. // eslint-disable-next-line no-console
  440. // console.log('label:', label);
  441. const reLabel = new RegExp(escapeForRegex(label), 'i');
  442. // eslint-disable-next-line no-console
  443. // console.log('reLabel:', reLabel);
  444. boards = Boards.userSearch(userId, {
  445. labels: { $elemMatch: { name: reLabel } },
  446. });
  447. if (boards.count()) {
  448. boards.forEach(board => {
  449. board.labels
  450. .filter(boardLabel => {
  451. return boardLabel.name.match(reLabel);
  452. })
  453. .forEach(boardLabel => {
  454. queryLabels.push(boardLabel._id);
  455. });
  456. });
  457. } else {
  458. errors.notFound.labels.push(label);
  459. }
  460. }
  461. selector.labelIds = { $in: queryLabels };
  462. });
  463. }
  464. if (queryParams.has.length) {
  465. queryParams.has.forEach(has => {
  466. if (has === 'description') {
  467. selector.description = { $exists: true, $nin: [null, ''] };
  468. } else if (has === 'attachment') {
  469. const attachments = Attachments.find({}, { fields: { cardId: 1 } });
  470. selector.$and.push({ _id: { $in: attachments.map(a => a.cardId) } });
  471. } else if (has === 'checklist') {
  472. const checklists = Checklists.find({}, { fields: { cardId: 1 } });
  473. selector.$and.push({ _id: { $in: checklists.map(a => a.cardId) } });
  474. }
  475. });
  476. }
  477. if (queryParams.text) {
  478. const regex = new RegExp(escapeForRegex(queryParams.text), 'i');
  479. const items = ChecklistItems.find(
  480. { title: regex },
  481. { fields: { cardId: 1 } },
  482. );
  483. const checklists = Checklists.find(
  484. {
  485. $or: [
  486. { title: regex },
  487. { _id: { $in: items.map(item => item.checklistId) } },
  488. ],
  489. },
  490. { fields: { cardId: 1 } },
  491. );
  492. const attachments = Attachments.find({ 'original.name': regex });
  493. selector.$and.push({
  494. $or: [
  495. { title: regex },
  496. { description: regex },
  497. { customFields: { $elemMatch: { value: regex } } },
  498. {
  499. _id: {
  500. $in: CardComments.textSearch(userId, [queryParams.text]).map(
  501. com => com.cardId,
  502. ),
  503. },
  504. },
  505. { _id: { $in: checklists.map(list => list.cardId) } },
  506. { _id: { $in: attachments.map(attach => attach.cardId) } },
  507. ],
  508. });
  509. }
  510. if (selector.$and.length === 0) {
  511. delete selector.$and;
  512. }
  513. }
  514. // eslint-disable-next-line no-console
  515. // console.log('selector:', selector);
  516. // eslint-disable-next-line no-console
  517. // console.log('selector.$and:', selector.$and);
  518. let cards = null;
  519. if (!errors.hasErrors()) {
  520. const projection = {
  521. fields: {
  522. _id: 1,
  523. archived: 1,
  524. boardId: 1,
  525. swimlaneId: 1,
  526. listId: 1,
  527. title: 1,
  528. type: 1,
  529. sort: 1,
  530. members: 1,
  531. assignees: 1,
  532. colors: 1,
  533. dueAt: 1,
  534. createdAt: 1,
  535. modifiedAt: 1,
  536. labelIds: 1,
  537. customFields: 1,
  538. },
  539. skip,
  540. limit,
  541. };
  542. if (queryParams.sort === 'due') {
  543. projection.sort = {
  544. dueAt: 1,
  545. boardId: 1,
  546. swimlaneId: 1,
  547. listId: 1,
  548. sort: 1,
  549. };
  550. } else if (queryParams.sort === 'modified') {
  551. projection.sort = {
  552. modifiedAt: -1,
  553. boardId: 1,
  554. swimlaneId: 1,
  555. listId: 1,
  556. sort: 1,
  557. };
  558. } else if (queryParams.sort === 'created') {
  559. projection.sort = {
  560. createdAt: -1,
  561. boardId: 1,
  562. swimlaneId: 1,
  563. listId: 1,
  564. sort: 1,
  565. };
  566. } else if (queryParams.sort === 'system') {
  567. projection.sort = {
  568. boardId: 1,
  569. swimlaneId: 1,
  570. listId: 1,
  571. modifiedAt: 1,
  572. sort: 1,
  573. };
  574. }
  575. // eslint-disable-next-line no-console
  576. // console.log('projection:', projection);
  577. cards = Cards.find(selector, projection);
  578. // eslint-disable-next-line no-console
  579. // console.log('count:', cards.count());
  580. }
  581. const update = {
  582. $set: {
  583. totalHits: 0,
  584. lastHit: 0,
  585. resultsCount: 0,
  586. cards: [],
  587. errors: errors.errorMessages(),
  588. selector: SessionData.pickle(selector),
  589. },
  590. };
  591. if (cards) {
  592. update.$set.totalHits = cards.count();
  593. update.$set.lastHit =
  594. skip + limit < cards.count() ? skip + limit : cards.count();
  595. update.$set.cards = cards.map(card => {
  596. return card._id;
  597. });
  598. update.$set.resultsCount = update.$set.cards.length;
  599. }
  600. SessionData.upsert({ userId, sessionId }, update);
  601. // remove old session data
  602. SessionData.remove({
  603. userId,
  604. modifiedAt: {
  605. $lt: new Date(
  606. moment()
  607. .subtract(1, 'day')
  608. .format(),
  609. ),
  610. },
  611. });
  612. if (cards) {
  613. const boards = [];
  614. const swimlanes = [];
  615. const lists = [];
  616. const customFieldIds = [];
  617. const users = [this.userId];
  618. cards.forEach(card => {
  619. if (card.boardId) boards.push(card.boardId);
  620. if (card.swimlaneId) swimlanes.push(card.swimlaneId);
  621. if (card.listId) lists.push(card.listId);
  622. if (card.members) {
  623. card.members.forEach(userId => {
  624. users.push(userId);
  625. });
  626. }
  627. if (card.assignees) {
  628. card.assignees.forEach(userId => {
  629. users.push(userId);
  630. });
  631. }
  632. if (card.customFields) {
  633. card.customFields.forEach(field => {
  634. customFieldIds.push(field._id);
  635. });
  636. }
  637. });
  638. const fields = {
  639. _id: 1,
  640. title: 1,
  641. archived: 1,
  642. sort: 1,
  643. type: 1,
  644. };
  645. return [
  646. cards,
  647. Boards.find(
  648. { _id: { $in: boards } },
  649. { fields: { ...fields, labels: 1, color: 1 } },
  650. ),
  651. Swimlanes.find(
  652. { _id: { $in: swimlanes } },
  653. { fields: { ...fields, color: 1 } },
  654. ),
  655. Lists.find({ _id: { $in: lists } }, { fields }),
  656. CustomFields.find({ _id: { $in: customFieldIds } }),
  657. Users.find({ _id: { $in: users } }, { fields: Users.safeFields }),
  658. Checklists.find({ cardId: { $in: cards.map(c => c._id) } }),
  659. Attachments.find({ cardId: { $in: cards.map(c => c._id) } }),
  660. CardComments.find({ cardId: { $in: cards.map(c => c._id) } }),
  661. SessionData.find({ userId: this.userId, sessionId }),
  662. ];
  663. }
  664. return [SessionData.find({ userId: this.userId, sessionId })];
  665. });
  666. Meteor.publish('brokenCards', function() {
  667. const user = Users.findOne({ _id: this.userId });
  668. const permiitedBoards = [null];
  669. let selector = {};
  670. selector.$or = [
  671. { permission: 'public' },
  672. { members: { $elemMatch: { userId: user._id, isActive: true } } },
  673. ];
  674. Boards.find(selector).forEach(board => {
  675. permiitedBoards.push(board._id);
  676. });
  677. selector = {
  678. boardId: { $in: permiitedBoards },
  679. $or: [
  680. { boardId: { $in: [null, ''] } },
  681. { swimlaneId: { $in: [null, ''] } },
  682. { listId: { $in: [null, ''] } },
  683. ],
  684. };
  685. const cards = Cards.find(selector, {
  686. fields: {
  687. _id: 1,
  688. archived: 1,
  689. boardId: 1,
  690. swimlaneId: 1,
  691. listId: 1,
  692. title: 1,
  693. type: 1,
  694. sort: 1,
  695. members: 1,
  696. assignees: 1,
  697. colors: 1,
  698. dueAt: 1,
  699. },
  700. });
  701. const boards = [];
  702. const swimlanes = [];
  703. const lists = [];
  704. const users = [];
  705. cards.forEach(card => {
  706. if (card.boardId) boards.push(card.boardId);
  707. if (card.swimlaneId) swimlanes.push(card.swimlaneId);
  708. if (card.listId) lists.push(card.listId);
  709. if (card.members) {
  710. card.members.forEach(userId => {
  711. users.push(userId);
  712. });
  713. }
  714. if (card.assignees) {
  715. card.assignees.forEach(userId => {
  716. users.push(userId);
  717. });
  718. }
  719. });
  720. return [
  721. cards,
  722. Boards.find({ _id: { $in: boards } }),
  723. Swimlanes.find({ _id: { $in: swimlanes } }),
  724. Lists.find({ _id: { $in: lists } }),
  725. Users.find({ _id: { $in: users } }, { fields: Users.safeFields }),
  726. ];
  727. });