|
|
@@ -13,12 +13,17 @@ let parentForRename2;
|
|
|
let parentForRename3;
|
|
|
let parentForRename4;
|
|
|
|
|
|
+let childForRename1;
|
|
|
+let childForRename2;
|
|
|
+let childForRename3;
|
|
|
+
|
|
|
let parentForDuplicate;
|
|
|
let parentForDelete;
|
|
|
let parentForDeleteCompletely;
|
|
|
-let parentForRevert;
|
|
|
|
|
|
-let childForRename;
|
|
|
+let parentForRevert1;
|
|
|
+let parentForRevert2;
|
|
|
+
|
|
|
let childForDuplicate;
|
|
|
let childForDelete;
|
|
|
let childForDeleteCompletely;
|
|
|
@@ -82,6 +87,18 @@ describe('PageService', () => {
|
|
|
creator: testUser1,
|
|
|
lastUpdateUser: testUser1,
|
|
|
},
|
|
|
+ {
|
|
|
+ path: '/parentForRename2/child',
|
|
|
+ grant: Page.GRANT_PUBLIC,
|
|
|
+ creator: testUser1,
|
|
|
+ lastUpdateUser: testUser1,
|
|
|
+ },
|
|
|
+ {
|
|
|
+ path: '/parentForRename3/child',
|
|
|
+ grant: Page.GRANT_PUBLIC,
|
|
|
+ creator: testUser1,
|
|
|
+ lastUpdateUser: testUser1,
|
|
|
+ },
|
|
|
{
|
|
|
path: '/parentForDuplicate',
|
|
|
grant: Page.GRANT_PUBLIC,
|
|
|
@@ -120,13 +137,22 @@ describe('PageService', () => {
|
|
|
lastUpdateUser: testUser1,
|
|
|
},
|
|
|
{
|
|
|
- path: '/parentForRevert',
|
|
|
+ path: '/trash/parentForRevert1',
|
|
|
+ status: Page.STATUS_DELETED,
|
|
|
grant: Page.GRANT_PUBLIC,
|
|
|
creator: testUser1,
|
|
|
lastUpdateUser: testUser1,
|
|
|
},
|
|
|
{
|
|
|
- path: '/parentForRevert/child',
|
|
|
+ path: '/trash/parentForRevert2',
|
|
|
+ status: Page.STATUS_DELETED,
|
|
|
+ grant: Page.GRANT_PUBLIC,
|
|
|
+ creator: testUser1,
|
|
|
+ lastUpdateUser: testUser1,
|
|
|
+ },
|
|
|
+ {
|
|
|
+ path: '/trash/parentForRevert/child',
|
|
|
+ status: Page.STATUS_DELETED,
|
|
|
grant: Page.GRANT_PUBLIC,
|
|
|
creator: testUser1,
|
|
|
lastUpdateUser: testUser1,
|
|
|
@@ -137,16 +163,21 @@ describe('PageService', () => {
|
|
|
parentForRename2 = await Page.findOne({ path: '/parentForRename2' });
|
|
|
parentForRename3 = await Page.findOne({ path: '/parentForRename3' });
|
|
|
parentForRename4 = await Page.findOne({ path: '/parentForRename4' });
|
|
|
+
|
|
|
parentForDuplicate = await Page.findOne({ path: '/parentForDuplicate' });
|
|
|
parentForDelete = await Page.findOne({ path: '/parentForDelete' });
|
|
|
parentForDeleteCompletely = await Page.findOne({ path: '/parentForDeleteCompletely' });
|
|
|
- parentForRevert = await Page.findOne({ path: '/parentForRevert' });
|
|
|
+ parentForRevert1 = await Page.findOne({ path: '/trash/parentForRevert1' });
|
|
|
+ parentForRevert2 = await Page.findOne({ path: '/trash/parentForRevert2' });
|
|
|
+
|
|
|
+ childForRename1 = await Page.findOne({ path: '/parentForRename1/child' });
|
|
|
+ childForRename2 = await Page.findOne({ path: '/parentForRename2/child' });
|
|
|
+ childForRename3 = await Page.findOne({ path: '/parentForRename3/child' });
|
|
|
|
|
|
- childForRename = await Page.findOne({ path: '/parentForRename1/child' });
|
|
|
childForDuplicate = await Page.findOne({ path: '/parentForDuplicate/child' });
|
|
|
childForDelete = await Page.findOne({ path: '/parentForDelete/child' });
|
|
|
childForDeleteCompletely = await Page.findOne({ path: '/parentForDeleteCompletely/child' });
|
|
|
- childForRevert = await Page.findOne({ path: '/parentForRevert/child' });
|
|
|
+ childForRevert = await Page.findOne({ path: '/trash/parentForRevert/child' });
|
|
|
|
|
|
|
|
|
await Tag.insertMany([
|
|
|
@@ -274,8 +305,69 @@ describe('PageService', () => {
|
|
|
|
|
|
});
|
|
|
|
|
|
- test('renameDescendants()', () => {
|
|
|
- expect(3).toBe(3);
|
|
|
+ test('renameDescendants without options', async() => {
|
|
|
+ const oldPagePathPrefix = new RegExp('^/parentForRename1', 'i');
|
|
|
+ const newPagePathPrefix = '/renamed1';
|
|
|
+
|
|
|
+ await crowi.pageService.renameDescendants([childForRename1], testUser2, {}, oldPagePathPrefix, newPagePathPrefix);
|
|
|
+ const resultPage = await Page.findOne({ path: '/renamed1/child' });
|
|
|
+ const redirectedFromPage = await Page.findOne({ path: '/parentForRename1/child' });
|
|
|
+ const redirectedFromPageRevision = await Revision.findOne({ path: '/parentForRename1/child' });
|
|
|
+
|
|
|
+ expect(resultPage).not.toBeNull();
|
|
|
+ expect(pageEventSpy).toHaveBeenCalledWith('updateMany', [childForRename1], testUser2);
|
|
|
+
|
|
|
+ expect(resultPage.path).toBe('/renamed1/child');
|
|
|
+ expect(resultPage.updatedAt).toEqual(childForRename1.updatedAt);
|
|
|
+ expect(resultPage.lastUpdateUser).toEqual(testUser1._id);
|
|
|
+
|
|
|
+ expect(redirectedFromPage).toBeNull();
|
|
|
+ expect(redirectedFromPageRevision).toBeNull();
|
|
|
+ });
|
|
|
+
|
|
|
+ test('renameDescendants with updateMetadata option', async() => {
|
|
|
+ const oldPagePathPrefix = new RegExp('^/parentForRename2', 'i');
|
|
|
+ const newPagePathPrefix = '/renamed2';
|
|
|
+
|
|
|
+ await crowi.pageService.renameDescendants([childForRename2], testUser2, { updateMetadata: true }, oldPagePathPrefix, newPagePathPrefix);
|
|
|
+ const resultPage = await Page.findOne({ path: '/renamed2/child' });
|
|
|
+ const redirectedFromPage = await Page.findOne({ path: '/parentForRename2/child' });
|
|
|
+ const redirectedFromPageRevision = await Revision.findOne({ path: '/parentForRename2/child' });
|
|
|
+
|
|
|
+ expect(resultPage).not.toBeNull();
|
|
|
+ expect(pageEventSpy).toHaveBeenCalledWith('updateMany', [childForRename2], testUser2);
|
|
|
+
|
|
|
+ expect(resultPage.path).toBe('/renamed2/child');
|
|
|
+ expect(resultPage.updatedAt).toEqual(dateToUse);
|
|
|
+ expect(resultPage.lastUpdateUser).toEqual(testUser2._id);
|
|
|
+
|
|
|
+ expect(redirectedFromPage).toBeNull();
|
|
|
+ expect(redirectedFromPageRevision).toBeNull();
|
|
|
+ });
|
|
|
+
|
|
|
+ test('renameDescendants with createRedirectPage option', async() => {
|
|
|
+ const oldPagePathPrefix = new RegExp('^/parentForRename3', 'i');
|
|
|
+ const newPagePathPrefix = '/renamed3';
|
|
|
+
|
|
|
+ await crowi.pageService.renameDescendants([childForRename3], testUser2, { createRedirectPage: true }, oldPagePathPrefix, newPagePathPrefix);
|
|
|
+ const resultPage = await Page.findOne({ path: '/renamed3/child' });
|
|
|
+ const redirectedFromPage = await Page.findOne({ path: '/parentForRename3/child' });
|
|
|
+ const redirectedFromPageRevision = await Revision.findOne({ path: '/parentForRename3/child' });
|
|
|
+
|
|
|
+ expect(resultPage).not.toBeNull();
|
|
|
+ expect(pageEventSpy).toHaveBeenCalledWith('updateMany', [childForRename3], testUser2);
|
|
|
+
|
|
|
+ expect(resultPage.path).toBe('/renamed3/child');
|
|
|
+ expect(resultPage.updatedAt).toEqual(childForRename3.updatedAt);
|
|
|
+ expect(resultPage.lastUpdateUser).toEqual(testUser1._id);
|
|
|
+
|
|
|
+ expect(redirectedFromPage).not.toBeNull();
|
|
|
+ expect(redirectedFromPage.path).toBe('/parentForRename3/child');
|
|
|
+ expect(redirectedFromPage.redirectTo).toBe('/renamed3/child');
|
|
|
+
|
|
|
+ expect(redirectedFromPageRevision).not.toBeNull();
|
|
|
+ expect(redirectedFromPageRevision.path).toBe('/parentForRename3/child');
|
|
|
+ expect(redirectedFromPageRevision.body).toBe('redirect /renamed3/child');
|
|
|
});
|
|
|
});
|
|
|
|
|
|
@@ -383,8 +475,56 @@ describe('PageService', () => {
|
|
|
});
|
|
|
|
|
|
describe('revert page', () => {
|
|
|
- test('revertDeletedPage()', () => {
|
|
|
- expect(3).toBe(3);
|
|
|
+ let getRevertDeletedPageNameSpy;
|
|
|
+ let findByPathSpy;
|
|
|
+ let deleteCompletelySpy;
|
|
|
+ let revertDeletedDescendantsWithStreamSpy;
|
|
|
+
|
|
|
+ beforeEach(async(done) => {
|
|
|
+ getRevertDeletedPageNameSpy = jest.spyOn(Page, 'getRevertDeletedPageName');
|
|
|
+ deleteCompletelySpy = jest.spyOn(crowi.pageService, 'deleteCompletely').mockImplementation();
|
|
|
+ revertDeletedDescendantsWithStreamSpy = jest.spyOn(crowi.pageService, 'revertDeletedDescendantsWithStream').mockImplementation();
|
|
|
+ done();
|
|
|
+ });
|
|
|
+
|
|
|
+ test('revert deleted page when the redirect from page exists', async() => {
|
|
|
+
|
|
|
+ findByPathSpy = jest.spyOn(Page, 'findByPath').mockImplementation(() => {
|
|
|
+ return { redirectTo: '/trash/parentForRevert1' };
|
|
|
+ });
|
|
|
+
|
|
|
+ const resultPage = await crowi.pageService.revertDeletedPage(parentForRevert1, testUser2);
|
|
|
+
|
|
|
+ expect(getRevertDeletedPageNameSpy).toHaveBeenCalledWith(parentForRevert1.path);
|
|
|
+ expect(findByPathSpy).toHaveBeenCalledWith('/parentForRevert1');
|
|
|
+ expect(deleteCompletelySpy).toHaveBeenCalled();
|
|
|
+ expect(revertDeletedDescendantsWithStreamSpy).not.toHaveBeenCalled();
|
|
|
+
|
|
|
+ expect(resultPage.path).toBe('/parentForRevert1');
|
|
|
+ expect(resultPage.lastUpdateUser._id).toEqual(testUser2._id);
|
|
|
+ expect(resultPage.status).toBe(Page.STATUS_PUBLISHED);
|
|
|
+ expect(resultPage.deleteUser).toBeNull();
|
|
|
+ expect(resultPage.deletedAt).toBeNull();
|
|
|
+ });
|
|
|
+
|
|
|
+ test('revert deleted page when the redirect from page does not exist', async() => {
|
|
|
+
|
|
|
+ findByPathSpy = jest.spyOn(Page, 'findByPath').mockImplementation(() => {
|
|
|
+ return null;
|
|
|
+ });
|
|
|
+
|
|
|
+ const resultPage = await crowi.pageService.revertDeletedPage(parentForRevert2, testUser2, {}, true);
|
|
|
+
|
|
|
+ expect(getRevertDeletedPageNameSpy).toHaveBeenCalledWith(parentForRevert2.path);
|
|
|
+ expect(findByPathSpy).toHaveBeenCalledWith('/parentForRevert2');
|
|
|
+ expect(deleteCompletelySpy).not.toHaveBeenCalled();
|
|
|
+ expect(revertDeletedDescendantsWithStreamSpy).toHaveBeenCalled();
|
|
|
+
|
|
|
+ expect(resultPage.path).toBe('/parentForRevert2');
|
|
|
+ expect(resultPage.lastUpdateUser._id).toEqual(testUser2._id);
|
|
|
+ expect(resultPage.status).toBe(Page.STATUS_PUBLISHED);
|
|
|
+ expect(resultPage.deleteUser).toBeNull();
|
|
|
+ expect(resultPage.deletedAt).toBeNull();
|
|
|
});
|
|
|
|
|
|
test('revertDeletedPages()', () => {
|