|
@@ -1,16 +1,20 @@
|
|
|
import { ErrorV3 } from '@growi/core';
|
|
import { ErrorV3 } from '@growi/core';
|
|
|
import { body } from 'express-validator';
|
|
import { body } from 'express-validator';
|
|
|
|
|
+import { Types } from 'mongoose';
|
|
|
|
|
|
|
|
|
|
+import { BookmarkFolderItems, BookmarkedPage } from '~/interfaces/bookmark-info';
|
|
|
import { apiV3FormValidator } from '~/server/middlewares/apiv3-form-validator';
|
|
import { apiV3FormValidator } from '~/server/middlewares/apiv3-form-validator';
|
|
|
import { InvalidParentBookmarkFolderError } from '~/server/models/errors';
|
|
import { InvalidParentBookmarkFolderError } from '~/server/models/errors';
|
|
|
|
|
+import { serializePageSecurely } from '~/server/models/serializers/page-serializer';
|
|
|
import loggerFactory from '~/utils/logger';
|
|
import loggerFactory from '~/utils/logger';
|
|
|
|
|
|
|
|
import BookmarkFolder from '../../models/bookmark-folder';
|
|
import BookmarkFolder from '../../models/bookmark-folder';
|
|
|
|
|
|
|
|
const logger = loggerFactory('growi:routes:apiv3:bookmark-folder');
|
|
const logger = loggerFactory('growi:routes:apiv3:bookmark-folder');
|
|
|
-
|
|
|
|
|
const express = require('express');
|
|
const express = require('express');
|
|
|
|
|
|
|
|
|
|
+const { serializeUserSecurely } = require('../../models/serializers/user-serializer');
|
|
|
|
|
+
|
|
|
const router = express.Router();
|
|
const router = express.Router();
|
|
|
|
|
|
|
|
const validator = {
|
|
const validator = {
|
|
@@ -52,6 +56,7 @@ module.exports = (crowi) => {
|
|
|
return res.apiv3({ bookmarkFolder });
|
|
return res.apiv3({ bookmarkFolder });
|
|
|
}
|
|
}
|
|
|
catch (err) {
|
|
catch (err) {
|
|
|
|
|
+ logger.error(err);
|
|
|
if (err instanceof InvalidParentBookmarkFolderError) {
|
|
if (err instanceof InvalidParentBookmarkFolderError) {
|
|
|
return res.apiv3Err(new ErrorV3(err.message, 'failed_to_create_bookmark_folder'));
|
|
return res.apiv3Err(new ErrorV3(err.message, 'failed_to_create_bookmark_folder'));
|
|
|
}
|
|
}
|
|
@@ -63,12 +68,68 @@ module.exports = (crowi) => {
|
|
|
router.get('/list/:userId', accessTokenParser, loginRequiredStrictly, async(req, res) => {
|
|
router.get('/list/:userId', accessTokenParser, loginRequiredStrictly, async(req, res) => {
|
|
|
const { userId } = req.params;
|
|
const { userId } = req.params;
|
|
|
|
|
|
|
|
|
|
+ const getBookmarkFolders = async(
|
|
|
|
|
+ userId: Types.ObjectId | string,
|
|
|
|
|
+ parentFolderId?: Types.ObjectId | string,
|
|
|
|
|
+ ) => {
|
|
|
|
|
+ const Page = crowi.model('Page');
|
|
|
|
|
+ const User = crowi.model('User');
|
|
|
|
|
+
|
|
|
|
|
+ const folders = await BookmarkFolder.find({ owner: userId, parent: parentFolderId })
|
|
|
|
|
+ .populate('children')
|
|
|
|
|
+ .populate({
|
|
|
|
|
+ path: 'bookmarks',
|
|
|
|
|
+ model: 'Bookmark',
|
|
|
|
|
+ populate: {
|
|
|
|
|
+ path: 'page',
|
|
|
|
|
+ model: 'Page',
|
|
|
|
|
+ populate: {
|
|
|
|
|
+ path: 'lastUpdateUser',
|
|
|
|
|
+ model: 'User',
|
|
|
|
|
+ },
|
|
|
|
|
+ },
|
|
|
|
|
+ });
|
|
|
|
|
+
|
|
|
|
|
+ const returnValue: BookmarkFolderItems[] = [];
|
|
|
|
|
+
|
|
|
|
|
+ // serialize page and user
|
|
|
|
|
+ folders.forEach((folder: BookmarkFolderItems) => {
|
|
|
|
|
+ folder.bookmarks.forEach((bookmark: BookmarkedPage) => {
|
|
|
|
|
+ if (bookmark.page != null && bookmark.page instanceof Page) {
|
|
|
|
|
+ bookmark.page = serializePageSecurely(bookmark.page);
|
|
|
|
|
+ }
|
|
|
|
|
+ if (bookmark.page.lastUpdateUser != null && bookmark.page.lastUpdateUser instanceof User) {
|
|
|
|
|
+ bookmark.page.lastUpdateUser = serializeUserSecurely(bookmark.page.lastUpdateUser);
|
|
|
|
|
+ }
|
|
|
|
|
+ });
|
|
|
|
|
+ });
|
|
|
|
|
+
|
|
|
|
|
+ const promises = folders.map(async(folder: BookmarkFolderItems) => {
|
|
|
|
|
+ const children = await getBookmarkFolders(userId, folder._id);
|
|
|
|
|
+
|
|
|
|
|
+ const res = {
|
|
|
|
|
+ _id: folder._id.toString(),
|
|
|
|
|
+ name: folder.name,
|
|
|
|
|
+ owner: folder.owner,
|
|
|
|
|
+ bookmarks: folder.bookmarks,
|
|
|
|
|
+ children,
|
|
|
|
|
+ parent: folder.parent,
|
|
|
|
|
+ };
|
|
|
|
|
+ return res;
|
|
|
|
|
+ });
|
|
|
|
|
+
|
|
|
|
|
+ const results = await Promise.all(promises) as unknown as BookmarkFolderItems[];
|
|
|
|
|
+ returnValue.push(...results);
|
|
|
|
|
+ return returnValue;
|
|
|
|
|
+ };
|
|
|
|
|
+
|
|
|
try {
|
|
try {
|
|
|
- const bookmarkFolderItems = await BookmarkFolder.findFolderAndChildren(userId);
|
|
|
|
|
|
|
+ const bookmarkFolderItems = await getBookmarkFolders(userId, undefined);
|
|
|
|
|
|
|
|
return res.apiv3({ bookmarkFolderItems });
|
|
return res.apiv3({ bookmarkFolderItems });
|
|
|
}
|
|
}
|
|
|
catch (err) {
|
|
catch (err) {
|
|
|
|
|
+ logger.error(err);
|
|
|
return res.apiv3Err(err, 500);
|
|
return res.apiv3Err(err, 500);
|
|
|
}
|
|
}
|
|
|
});
|
|
});
|
|
@@ -94,6 +155,7 @@ module.exports = (crowi) => {
|
|
|
return res.apiv3({ bookmarkFolder });
|
|
return res.apiv3({ bookmarkFolder });
|
|
|
}
|
|
}
|
|
|
catch (err) {
|
|
catch (err) {
|
|
|
|
|
+ logger.error(err);
|
|
|
return res.apiv3Err(err, 500);
|
|
return res.apiv3Err(err, 500);
|
|
|
}
|
|
}
|
|
|
});
|
|
});
|
|
@@ -108,6 +170,7 @@ module.exports = (crowi) => {
|
|
|
return res.apiv3({ bookmarkFolder });
|
|
return res.apiv3({ bookmarkFolder });
|
|
|
}
|
|
}
|
|
|
catch (err) {
|
|
catch (err) {
|
|
|
|
|
+ logger.error(err);
|
|
|
return res.apiv3Err(err, 500);
|
|
return res.apiv3Err(err, 500);
|
|
|
}
|
|
}
|
|
|
});
|
|
});
|
|
@@ -120,6 +183,7 @@ module.exports = (crowi) => {
|
|
|
return res.apiv3({ bookmarkFolder });
|
|
return res.apiv3({ bookmarkFolder });
|
|
|
}
|
|
}
|
|
|
catch (err) {
|
|
catch (err) {
|
|
|
|
|
+ logger.error(err);
|
|
|
return res.apiv3Err(err, 500);
|
|
return res.apiv3Err(err, 500);
|
|
|
}
|
|
}
|
|
|
});
|
|
});
|