Skip to content

Commit

Permalink
[ACS-5678] create folder from template tests (#3456)
Browse files Browse the repository at this point in the history
* [ACS-5678] create folder from template tests

* taking out only flag

* fixes for linter, import, method name and locator

* fix import and utils strings
  • Loading branch information
azakrzewski-hy authored Oct 2, 2023
1 parent 740b9b3 commit 33c50bd
Show file tree
Hide file tree
Showing 17 changed files with 665 additions and 13 deletions.
416 changes: 416 additions & 0 deletions e2e/playwright/actions/src/tests/create-folder-from-template.spec.ts

Large diffs are not rendered by default.

24 changes: 23 additions & 1 deletion e2e/protractor/protractor.excludes.json
Original file line number Diff line number Diff line change
Expand Up @@ -17,5 +17,27 @@
"C279220": "will be fixed after protractor to playwright migration, see https://alfresco.atlassian.net/browse/ACS-5007",
"C279221": "will be fixed after protractor to playwright migration, see https://alfresco.atlassian.net/browse/ACS-5007",
"C325006": "will be fixed after protractor to playwright migration, see https://alfresco.atlassiana.net/browse/ACS-5007",
"C213097": "https://alfresco.atlassian.net/browse/ACS-5479"
"C213097": "https://alfresco.atlassian.net/browse/ACS-5479",
"C325147" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325148" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325149" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325150" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325153" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325151" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325139" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325142" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325143" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325144" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325145" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325146" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325141" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325140" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325157" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325154" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325156" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325155" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325158" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325161" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325162" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678",
"C325163" : "test migrated to playwright https://alfresco.atlassian.net/browse/ACS-5678"
}
Original file line number Diff line number Diff line change
Expand Up @@ -42,14 +42,15 @@ <h2 mat-dialog-title [innerHTML]="title"></h2>
</form>
</div>
<div mat-dialog-actions>
<button mat-button mat-dialog-close>
<button mat-button mat-dialog-close data-automation-id="cancel-folder-template-button">
{{ 'NODE_FROM_TEMPLATE.CANCEL' | translate }}
</button>
<button
class="create"
[disabled]="form.invalid"
mat-button
(click)="onSubmit()"
data-automation-id="create-folder-template-button"
>
{{ 'NODE_FROM_TEMPLATE.CREATE' | translate }}
</button>
Expand Down
82 changes: 82 additions & 0 deletions projects/aca-playwright-shared/src/api/nodes-api.ts
Original file line number Diff line number Diff line change
Expand Up @@ -154,6 +154,15 @@ export class NodesApi {
}
}

async getNodeById(id: string): Promise<NodeEntry | null> {
try {
return await this.apiService.nodes.getNode(id);
} catch (error) {
logger.error(`${this.constructor.name} ${this.getNodeById.name}`, error);
return null;
}
}

async getNodeIdFromParent(name: string, parentId: string): Promise<string> {
try {
const children = (await this.getNodeChildren(parentId)).list.entries;
Expand Down Expand Up @@ -214,4 +223,77 @@ export class NodesApi {
}
}

async removeUserAccessOnSpaceTemplate(nodeName: string): Promise<NodeEntry> {
try {
const templatesRootFolderId = await this.getSpaceTemplatesFolderId();
const nodeId: string = await this.getNodeIdFromParent(nodeName, templatesRootFolderId);

return this.setInheritPermissions(nodeId, false);
} catch (error) {
logger.error('Admin Actions - removeUserAccessOnSpaceTemplate failed : ', error);
return null;
}
}

async setInheritPermissions(nodeId: string, inheritPermissions: boolean): Promise<NodeEntry | null> {
const data = {
permissions: {
isInheritanceEnabled: inheritPermissions
}
};

try {
return await this.apiService.nodes.updateNode(nodeId, data);
} catch (error) {
logger.error(`${this.constructor.name} ${this.setInheritPermissions.name}`, error);
return null;
}
}

private async addAspects(nodeId: string, aspectNames: string[]): Promise<NodeEntry> {
try {
return this.apiService.nodes.updateNode(nodeId, { aspectNames });
} catch (error) {
logger.error(`${this.constructor.name} ${this.addAspects.name}`, error);
return null;
}
}

async createFolderLink(originalNodeId: string, destinationId: string): Promise<NodeEntry | null> {
const name = (await this.getNodeById(originalNodeId)).entry.name;
const nodeBody = {
name: `Link to ${name}.url`,
nodeType: 'app:folderlink',
properties: {
'cm:title': `Link to ${name}.url`,
'cm:destination': originalNodeId,
'cm:description': `Link to ${name}.url`,
'app:icon': 'space-icon-link'
}
};

try {
const link = await this.apiService.nodes.createNode(destinationId, nodeBody);
await this.addAspects(originalNodeId, ['app:linked']);
return link;
} catch (error) {
logger.error(`${this.constructor.name} ${this.createFolderLink.name}`, error);
return null;
}
}

async createLinkToFolderName(originalFolderName: string, originalFolderParentId: string, destinationParentId?: string): Promise<NodeEntry> {
if (!destinationParentId) {
destinationParentId = originalFolderParentId;
}

try {
const nodeId = await this.getNodeIdFromParent(originalFolderName, originalFolderParentId);
return this.createFolderLink(nodeId, destinationParentId);
} catch (error) {
logger.error('Admin Actions - createLinkToFolderName failed : ', error);
return null;
}
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,9 @@
* from Hyland Software. If not, see <http://www.gnu.org/licenses/>.
*/

import { BaseComponent } from './base.component';
import { Page } from '@playwright/test';
import { MatMenuComponent } from './dataTable';
import { BaseComponent } from './base.component';
export class AcaHeader extends BaseComponent {
private static rootElement = 'aca-toolbar';
private moreActionsButton = this.getChild('button[id="app.viewer.toolbar.more"]');
Expand All @@ -38,9 +39,15 @@ export class AcaHeader extends BaseComponent {
constructor(page: Page) {
super(page, AcaHeader.rootElement);
}
public matMenu = new MatMenuComponent(this.page);

async clickViewerMoreActions(): Promise<void> {
await this.moreActionsButton.waitFor({ state: 'attached' });
await this.moreActionsButton.click();
}

async clickCreateFolderFromTemplate(): Promise<void> {
await this.createButton.click();
await this.matMenu.createFolderFromTemplate.click();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,7 @@ export abstract class BaseComponent extends PlaywrightBase {

async spinnerWaitForReload(): Promise<void> {
try {
await this.page.locator('mat-progress-spinner').waitFor({ state: 'attached', timeout: timeouts.normal });
await this.page.locator('mat-progress-spinner').waitFor({ state: 'attached', timeout: timeouts.medium });
await this.page.locator('mat-progress-spinner').waitFor({ state: 'detached', timeout: timeouts.normal });
} catch (e) {
this.logger.info('Spinner was not present');
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -39,10 +39,14 @@ export class DataTableComponent extends BaseComponent {
getEmptyFolderLocator = this.getChild('.adf-empty-folder');
getEmptyContentTitleLocator = this.getChild('adf-empty-content .adf-empty-content__title');
getEmptyContentSubTitleLocator = this.getChild('adf-empty-content .adf-empty-content__subtitle');
getSelectedRow = this.getChild('.adf-datatable-row.adf-is-selected');

/** Locator for row (or rows) */
getRowLocator = this.getChild(`adf-datatable-row`);

/** Locator to get "No results found" message */
getNoResultsFoundMessage = this.getChild('adf-custom-empty-content-template', { hasText: "No results found" });

/**
* Method used in cases where we want to check that some record is visible in the datatable. It will consider whole row
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ export class MatMenuComponent extends BaseComponent {
public getMenuItemsLocator = this.getChild('button');
public getMenuItemTextLocator = this.getChild('[data-automation-id="menu-item-title"]');
public createFolder = this.getChild('[id="app.create.folder"]');
public createFolderFromTemplate = this.getChild('[id="app.create.folderFromTemplate"]');
public createLibrary = this.getChild('[id="app.create.library"]');
public getButtonByText = (text: string) => this.getChild('button', { hasText: text });

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,16 +28,20 @@ import { BaseComponent } from '../base.component';
export class ContentNodeSelectorDialog extends BaseComponent {
private static rootElement = 'adf-content-node-selector';

constructor(page: Page) {
super(page, ContentNodeSelectorDialog.rootElement);
}

public cancelButton = this.getChild('[data-automation-id="content-node-selector-actions-cancel"]');
public actionButton = this.getChild('[data-automation-id="content-node-selector-actions-choose"]');
public locationDropDown = this.getChild('[id="site-dropdown-container"]');
private selectedRow = this.getChild('.adf-is-selected');
private getOptionLocator = (optionName: string): Locator => this.page.locator('.mat-select-panel .mat-option-text', { hasText: optionName });
getOptionLocator = (optionName: string): Locator => this.page.locator('.mat-select-panel .mat-option-text', { hasText: optionName });
private getRowByName = (name: string | number): Locator => this.getChild(`adf-datatable-row`, { hasText: name.toString() });
getDialogTitle = (text: string) => this.getChild('.mat-dialog-title', { hasText: text });
getBreadcrumb = (text: string) => this.getChild('[data-automation-id="current-folder"]', { hasText: text });
getFolderIcon = this.getChild('mat-icon[role="img"]', { hasText: "folder" });

constructor(page: Page) {
super(page, ContentNodeSelectorDialog.rootElement);
}

async selectLocation(location: string): Promise<void> {
await this.locationDropDown.click();
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
/*!
* Copyright © 2005-2023 Hyland Software, Inc. and its affiliates. All rights reserved.
*
* Alfresco Example Content Application
*
* This file is part of the Alfresco Example Content Application.
* If the software was purchased under a paid Alfresco license, the terms of
* the paid license agreement will prevail. Otherwise, the software is
* provided under the following open source license terms:
*
* The Alfresco Example Content Application is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* The Alfresco Example Content Application is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
*/

import { Locator, Page } from '@playwright/test';
import { BaseComponent } from '../base.component';
import { timeouts } from '../../../utils';

export class CreateFromTemplateDialogComponent extends BaseComponent {
private static rootElement = 'ng-component';

constructor(page: Page) {
super(page, CreateFromTemplateDialogComponent.rootElement);
}

cancelButton = this.getChild('[data-automation-id="cancel-folder-template-button"]');
createButton = this.getChild('[data-automation-id="create-folder-template-button"]');
getDialogTitle = (text: string) => this.getChild('.mat-dialog-title', { hasText: text });
getDialogLabel = (text: string) => this.getChild('label', { hasText: text });
getErrorByText = (text: string): Locator => this.page.locator('mat-error', {hasText: text});


async isErrorMessageDisplayed(errorText: string): Promise<boolean> {
await this.getErrorByText(errorText).waitFor({ state: 'visible', timeout: timeouts.large });
return await this.getErrorByText(errorText).isVisible();
}

/**
* This method is used when we want to fill in Create new folder from template dialog and choose Create button
*/
async createNewFolderFromTemplate( nameInput: string, titleInput?: string, descriptionInput?: string): Promise<void> {
await this.getDialogLabel('Name *').fill(nameInput);
if (titleInput) { await this.getDialogLabel('Title').fill(titleInput); }
if (descriptionInput) { await this.getDialogLabel('Description').fill(descriptionInput); }
await this.createButton.click();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -27,3 +27,4 @@ export * from './adf-library-dialog.component';
export * from './password-overlay-dialog.component';
export * from './viewer-overlay-dialog.component';
export * from './content-node-selector-dialog';
export * from './create-from-template-dialog-component';
Original file line number Diff line number Diff line change
Expand Up @@ -37,3 +37,4 @@ export * from './search/search-input.component';
export * from './search/search-overlay.components';
export * from './breadcrumb/breadcrumb.component';
export * from './sidenav.component';
export * from './aca-header.component';
Original file line number Diff line number Diff line change
Expand Up @@ -24,8 +24,8 @@

import { Page } from '@playwright/test';
import { BasePage } from './base.page';
import { AcaHeader } from '../components/aca-header.component';
import {
AcaHeader,
AdfInfoDrawerComponent,
AdfLibraryDialogComponent,
DataTableComponent,
Expand All @@ -34,7 +34,8 @@ import {
ViewerOverlayDialogComponent,
ContentNodeSelectorDialog,
Breadcrumb,
SidenavComponent
SidenavComponent,
CreateFromTemplateDialogComponent
} from '../components';

export class MyLibrariesPage extends BasePage {
Expand All @@ -53,6 +54,8 @@ export class MyLibrariesPage extends BasePage {
public copyMoveDialog = new ContentNodeSelectorDialog(this.page);
public breadcrumb = new Breadcrumb(this.page);
public sidenav = new SidenavComponent(this.page);
public contentNodeSelector = new ContentNodeSelectorDialog(this.page);
public createFromTemplateDialogComponent = new CreateFromTemplateDialogComponent(this.page);

async selectCreateLibrary(): Promise<void> {
await this.acaHeader.createButton.click();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,9 +24,19 @@

import { Page } from '@playwright/test';
import { BasePage } from './base.page';
import { Breadcrumb, DataTableComponent, MatMenuComponent, ViewerComponent, SidenavComponent } from '../components';
import { AcaHeader } from '../components/aca-header.component';
import { AdfFolderDialogComponent, PasswordOverlayDialogComponent, ViewerOverlayDialogComponent } from '../components/dialogs';
import {
AcaHeader,
AdfFolderDialogComponent,
ContentNodeSelectorDialog,
CreateFromTemplateDialogComponent,
PasswordOverlayDialogComponent,
ViewerOverlayDialogComponent,
Breadcrumb,
DataTableComponent,
MatMenuComponent,
ViewerComponent,
SidenavComponent
} from '../components';

export class PersonalFilesPage extends BasePage {
private static pageUrl = 'personal-files';
Expand All @@ -38,12 +48,14 @@ export class PersonalFilesPage extends BasePage {
public acaHeader = new AcaHeader(this.page);
public matMenu = new MatMenuComponent(this.page);
public folderDialog = new AdfFolderDialogComponent(this.page);
public contentNodeSelector = new ContentNodeSelectorDialog(this.page);
public dataTable = new DataTableComponent(this.page);
public viewer = new ViewerComponent(this.page);
public passwordDialog = new PasswordOverlayDialogComponent(this.page);
public viewerDialog = new ViewerOverlayDialogComponent(this.page);
public breadcrumb = new Breadcrumb(this.page);
public sidenav = new SidenavComponent(this.page);
public createFromTemplateDialogComponent = new CreateFromTemplateDialogComponent(this.page);

async selectCreateFolder(): Promise<void> {
await this.acaHeader.createButton.click();
Expand Down
36 changes: 36 additions & 0 deletions projects/aca-playwright-shared/src/utils/error-strings.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
/*!
* Copyright © 2005-2023 Hyland Software, Inc. and its affiliates. All rights reserved.
*
* Alfresco Example Content Application
*
* This file is part of the Alfresco Example Content Application.
* If the software was purchased under a paid Alfresco license, the terms of
* the paid license agreement will prevail. Otherwise, the software is
* provided under the following open source license terms:
*
* The Alfresco Example Content Application is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* The Alfresco Example Content Application is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* from Hyland Software. If not, see <http://www.gnu.org/licenses/>.
*/

export const errorStrings = {

errorMessageNotPresent: 'Error message is not displayed',
nameIsRequiredError: 'Name is required',
nameWithSpecialCharactersError: `Name can't contain these characters * " < > \\ / ? : |`,
nameEndWithDotError: `Name can't end with a period .`,
nameContainOnlySpacesError: `Name can't contain only spaces`,
titleLengthLimitError: 'Use 256 characters or less for title',
descriptionLengthLimitError: 'Use 512 characters or less for description',
nameAlreadyUsedError: 'This name is already in use, try a different name.'

}
Loading

0 comments on commit 33c50bd

Please sign in to comment.