|
|
@@ -21,7 +21,7 @@ class PageService {
|
|
|
const ShareLink = this.crowi.model('ShareLink');
|
|
|
const Revision = this.crowi.model('Revision');
|
|
|
|
|
|
- await Promise.all([
|
|
|
+ return Promise.all([
|
|
|
Bookmark.find({ page: { $in: pageIds } }).remove({}),
|
|
|
Comment.find({ page: { $in: pageIds } }).remove({}),
|
|
|
PageTagRelation.find({ relatedPage: { $in: pageIds } }).remove({}),
|
|
|
@@ -38,7 +38,7 @@ class PageService {
|
|
|
const { attachmentService } = this.crowi;
|
|
|
const attachments = await Attachment.find({ page: { $in: pageIds } });
|
|
|
|
|
|
- attachmentService.removeAttachment(attachments);
|
|
|
+ return attachmentService.removeAttachment(attachments);
|
|
|
}
|
|
|
|
|
|
async duplicate(page, newPagePath, user) {
|
|
|
@@ -93,8 +93,8 @@ class PageService {
|
|
|
return newParentpage;
|
|
|
}
|
|
|
|
|
|
-
|
|
|
- async completelyDeletePage(pagesData, user, options = {}) {
|
|
|
+ // delete multiple pages
|
|
|
+ async completelyDeletePages(pagesData, user, options = {}) {
|
|
|
this.validateCrowi();
|
|
|
let pageEvent;
|
|
|
// init event
|
|
|
@@ -104,25 +104,33 @@ class PageService {
|
|
|
pageEvent.on('update', pageEvent.onUpdate);
|
|
|
}
|
|
|
|
|
|
- // In case of recursively
|
|
|
- if (pagesData.length != null) {
|
|
|
- const ids = pagesData.map(page => (page._id));
|
|
|
- const paths = pagesData.map(page => (page.path));
|
|
|
- const socketClientId = options.socketClientId || null;
|
|
|
+ const ids = pagesData.map(page => (page._id));
|
|
|
+ const paths = pagesData.map(page => (page.path));
|
|
|
+ const socketClientId = options.socketClientId || null;
|
|
|
|
|
|
- logger.debug('Deleting completely', paths);
|
|
|
+ logger.debug('Deleting completely', paths);
|
|
|
|
|
|
- await this.deleteCompletely(ids, paths);
|
|
|
+ await this.deleteCompletely(ids, paths);
|
|
|
|
|
|
- if (socketClientId != null) {
|
|
|
- pageEvent.emit('delete', pagesData, user, socketClientId); // update as renamed page
|
|
|
- }
|
|
|
- return;
|
|
|
+ if (socketClientId != null) {
|
|
|
+ pageEvent.emit('deleteCompletely', pagesData, user, socketClientId); // update as renamed page
|
|
|
+ }
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ // delete single page completely
|
|
|
+ async completelyDeleteSinglePage(pageData, user, options = {}) {
|
|
|
+ this.validateCrowi();
|
|
|
+ let pageEvent;
|
|
|
+ // init event
|
|
|
+ if (this.crowi != null) {
|
|
|
+ pageEvent = this.crowi.event('page');
|
|
|
+ pageEvent.on('create', pageEvent.onCreate);
|
|
|
+ pageEvent.on('update', pageEvent.onUpdate);
|
|
|
}
|
|
|
|
|
|
- // Simply delete completely a page
|
|
|
- const ids = [pagesData].map(page => (page._id));
|
|
|
- const paths = [pagesData].map(page => (page.path));
|
|
|
+ const ids = [pageData._id];
|
|
|
+ const paths = [pageData.path];
|
|
|
const socketClientId = options.socketClientId || null;
|
|
|
|
|
|
logger.debug('Deleting completely', paths);
|
|
|
@@ -130,8 +138,9 @@ class PageService {
|
|
|
await this.deleteCompletely(ids, paths);
|
|
|
|
|
|
if (socketClientId != null) {
|
|
|
- pageEvent.emit('delete', [pagesData], user, socketClientId); // update as renamed page
|
|
|
+ pageEvent.emit('delete', pageData, user, socketClientId); // update as renamed page
|
|
|
}
|
|
|
+ return;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
@@ -145,28 +154,33 @@ class PageService {
|
|
|
const pages = await Page.findManageableListWithDescendants(targetPage, user, findOpts);
|
|
|
|
|
|
// TODO streaming bellow action
|
|
|
- await this.completelyDeletePage(pages, user, options);
|
|
|
+ return this.completelyDeletePages(pages, user, options);
|
|
|
}
|
|
|
|
|
|
- async revertDeletedPageRecursively(targetPage, user, options = {}) {
|
|
|
- const findOpts = { includeTrashed: true };
|
|
|
+ // revert pages recursively
|
|
|
+ async revertDeletedPages(page, user, options = {}) {
|
|
|
const Page = this.crowi.model('Page');
|
|
|
- const pages = await Page.findManageableListWithDescendants(targetPage, user, findOpts);
|
|
|
-
|
|
|
- let updatedPage = null;
|
|
|
- await Promise.all(pages.map((page) => {
|
|
|
- const isParent = (page.path === targetPage.path);
|
|
|
- const p = this.revertDeletedPage(page, user, options);
|
|
|
- if (isParent) {
|
|
|
- updatedPage = p;
|
|
|
+ const newPath = Page.getRevertDeletedPageName(page.path);
|
|
|
+ const originPage = await Page.findByPath(newPath);
|
|
|
+ if (originPage != null) {
|
|
|
+ // When the page is deleted, it will always be created with "redirectTo" in the path of the original page.
|
|
|
+ // So, it's ok to delete the page
|
|
|
+ // However, If a page exists that is not "redirectTo", something is wrong. (Data correction is needed).
|
|
|
+ if (originPage.redirectTo !== page.path) {
|
|
|
+ throw new Error('The new page of to revert is exists and the redirect path of the page is not the deleted page.');
|
|
|
}
|
|
|
- return p;
|
|
|
- }));
|
|
|
+ // originPage is object.
|
|
|
+ await this.completelyDeletePages([originPage], options);
|
|
|
+ }
|
|
|
|
|
|
+ page.status = STATUS_PUBLISHED;
|
|
|
+ page.lastUpdateUser = user;
|
|
|
+ debug('Revert deleted the page', page, newPath);
|
|
|
+ const updatedPage = await Page.rename(page, newPath, user, {});
|
|
|
return updatedPage;
|
|
|
}
|
|
|
|
|
|
- async revertDeletedPage(page, user, options = {}) {
|
|
|
+ async revertSingleDeletedPage(page, user, options = {}) {
|
|
|
const Page = this.crowi.model('Page');
|
|
|
const newPath = Page.getRevertDeletedPageName(page.path);
|
|
|
const originPage = await Page.findByPath(newPath);
|
|
|
@@ -177,7 +191,7 @@ class PageService {
|
|
|
if (originPage.redirectTo !== page.path) {
|
|
|
throw new Error('The new page of to revert is exists and the redirect path of the page is not the deleted page.');
|
|
|
}
|
|
|
- await this.completelyDeletePage(originPage, options);
|
|
|
+ await this.completelyDeleteSinglePage(originPage, options);
|
|
|
}
|
|
|
|
|
|
page.status = STATUS_PUBLISHED;
|
|
|
@@ -198,10 +212,7 @@ class PageService {
|
|
|
}));
|
|
|
break;
|
|
|
case 'delete':
|
|
|
- await Promise.all(pages.map((page) => {
|
|
|
- return this.completelyDeletePage(page);
|
|
|
- }));
|
|
|
- break;
|
|
|
+ return this.completelyDeletePages(pages);
|
|
|
case 'transfer':
|
|
|
await Promise.all(pages.map((page) => {
|
|
|
return Page.transferPageToGroup(page, transferToUserGroupId);
|