diff --git a/app/cdap/components/PipelineConfigurations/index.tsx b/app/cdap/components/PipelineConfigurations/index.tsx index 84e08e28d02..c39891a5ef8 100644 --- a/app/cdap/components/PipelineConfigurations/index.tsx +++ b/app/cdap/components/PipelineConfigurations/index.tsx @@ -177,7 +177,7 @@ export default class PipelineConfigurations extends Component class="btn btn-primary" ng-click="PipelineUpgradeController.fixAll()" data-cy="fix-all-btn" + data-testid="fix-all-btn" > Fix All Proceed diff --git a/src/e2e-test/features/pipeline.edit.feature b/src/e2e-test/features/pipeline.edit.feature index 737ebf8801f..89726eccfd7 100644 --- a/src/e2e-test/features/pipeline.edit.feature +++ b/src/e2e-test/features/pipeline.edit.feature @@ -78,5 +78,14 @@ Feature: Pipeline Edit Then Click Continue draft Then Verify changes were saved - + @PIPELINE_EDIT_TEST + Scenario: Editing an orphaned pipeline draft should not break top panel + When Open HttpExecutor Page + Then Create an orphan pipeline draft + When Open pipeline draft list page + Then Go to pipeline "test-orphan-pipeline-6-7-4" draft + Then Click on FixAll Button + Then Verify Studio TopPanel is visible + When Open pipeline draft list page + Then Delete Draft Pipeline "test-orphan-pipeline-6-7-4" diff --git a/src/e2e-test/fixtures/test-orphan-pipeline-6-7-4.json b/src/e2e-test/fixtures/test-orphan-pipeline-6-7-4.json new file mode 100644 index 00000000000..825bfe837e4 --- /dev/null +++ b/src/e2e-test/fixtures/test-orphan-pipeline-6-7-4.json @@ -0,0 +1,65 @@ +{ + "artifact": { + "name": "cdap-data-pipeline", + "version": "6.7.4", + "scope": "SYSTEM", + "label": "Data Pipeline - Batch" + }, + "description": "", + "name": "test-orphan-pipeline-6-7-4", + "change": { + "description": "" + }, + "parentVersion": "this-does-not-exist", + "config": { + "resources": { + "memoryMB": 2048, + "virtualCores": 1 + }, + "driverResources": { + "memoryMB": 2048, + "virtualCores": 1 + }, + "connections": [], + "comments": [], + "postActions": [], + "properties": {}, + "processTimingEnabled": true, + "stageLoggingEnabled": false, + "stages": [ + { + "name": "File", + "plugin": { + "name": "File", + "type": "batchsource", + "label": "File", + "artifact": { + "name": "core-plugins", + "version": "2.9.3", + "scope": "SYSTEM" + }, + "properties": { + "sampleSize": "1000", + "enableQuotedValues": "false", + "skipHeader": "false", + "filenameOnly": "false", + "recursive": "false", + "ignoreNonExistingFolders": "false", + "fileEncoding": "UTF-8", + "schema": "{\"name\":\"fileRecord\",\"type\":\"record\",\"fields\":[{\"name\":\"offset\",\"type\":\"long\"},{\"name\":\"body\",\"type\":\"string\"}]}" + } + }, + "outputSchema": "{\"name\":\"fileRecord\",\"type\":\"record\",\"fields\":[{\"name\":\"offset\",\"type\":\"long\"},{\"name\":\"body\",\"type\":\"string\"}]}", + "id": "File" + } + ], + "schedule": "0 1 */1 * *", + "engine": "spark", + "numOfRecordsPreview": 100, + "rangeRecordsPreview": { + "min": 1, + "max": "5000" + }, + "maxConcurrentRuns": 1 + } +} diff --git a/src/e2e-test/java/io/cdap/cdap/ui/stepsdesign/CommonSteps.java b/src/e2e-test/java/io/cdap/cdap/ui/stepsdesign/CommonSteps.java index e4ee9d6eb9b..a60601a8f83 100644 --- a/src/e2e-test/java/io/cdap/cdap/ui/stepsdesign/CommonSteps.java +++ b/src/e2e-test/java/io/cdap/cdap/ui/stepsdesign/CommonSteps.java @@ -32,7 +32,12 @@ import org.openqa.selenium.By; import org.openqa.selenium.NoAlertPresentException; import org.openqa.selenium.UnhandledAlertException; +import org.openqa.selenium.WebElement; import org.openqa.selenium.WindowType; +import org.openqa.selenium.support.ui.Select; + +import java.io.IOException; +import java.util.UUID; public class CommonSteps { @@ -82,6 +87,12 @@ public void openConfigurationPage() { WaitHelper.waitForPageToLoad(); } + @When("Open HttpExecutor Page") + public void openHttpExecutorPage() { + SeleniumDriver.openPage(Constants.HTTP_EXECUTOR_URL); + WaitHelper.waitForPageToLoad(); + } + @When("Open Source Control Management Page") public void openSourceControlManagementPage() { SeleniumDriver.openPage(Constants.SOURCE_CONTROL_MANAGEMENT_URL); @@ -287,6 +298,11 @@ public void openPipelineDetails(String pipelineName) { ElementHelper.clickOnElement(Helper.locateElementByTestId("deployed-" + pipelineName)); } + @Then("Go to pipeline {string} draft") + public void openPipelineDraft(String pipelineName) { + ElementHelper.clickOnElement(Helper.locateElementByTestId("draft-" + pipelineName)); + } + @Then("Add Projection node to canvas") public void addProjectionNodeToCanvas() { NodeInfo projectionNode = new NodeInfo("Projection", "transform", "0"); @@ -303,4 +319,56 @@ public void openProjectionNodeProperties() { public void closeProjectionNodeProperties() { Commands.closeConfigPopover(); } + + @Then("Create an orphan pipeline draft") + public void createOrphanPipelineDraft() throws IOException { + WebElement requestMethodDropdown = Helper.locateElementByTestId("request-method-selector"); + Select requestMethodSelect = new Select(requestMethodDropdown); + requestMethodSelect.selectByValue("PUT"); + + WebElement requestPathInput = Helper.locateElementByTestId("request-path-input"); + String saveDraftPath = "namespaces/system/apps/pipeline/services/studio/methods/v1/contexts/default/drafts/" + + UUID.randomUUID(); + ElementHelper.sendKeys(requestPathInput, saveDraftPath); + + WebElement requestBodyInput = Helper.locateElementByTestId("request-body"); + String requestBody = Helper.readPipelineFixtureFile("test-orphan-pipeline-6-7-4.json"); + ElementHelper.sendKeys(requestBodyInput, requestBody); + + WebElement callApiButton = Helper.locateElementByTestId("send-btn"); + ElementHelper.clickOnElement(callApiButton); + + String callHistoryEntryXpath = "//*[@data-testid=\"request-path\"][text()=\"" + saveDraftPath + "\"]"; + WaitHelper.waitForElementToBePresent(By.xpath(callHistoryEntryXpath)); + } + + @Then("Click on FixAll Button") + public void clickOnFixAllButton() { + String fixAllBtnXpath = "//*[@data-testid=\"fix-all-btn\"]"; + WaitHelper.waitForElementToBePresent( + By.xpath(fixAllBtnXpath)); + ElementHelper.clickOnElement(Helper.locateElementByXPath(fixAllBtnXpath)); + } + + @Then("Verify Studio TopPanel is visible") + public void verifyStudioTopPanelVisible() { + String orphanedPipelineXpath = "//*[@data-testid=\"pipeline-edit-status\"]"; + WaitHelper.waitForElementToBePresent( + By.xpath(orphanedPipelineXpath)); + Assert.assertTrue(Helper.isElementExists(By.xpath(orphanedPipelineXpath))); + } + + @Then("Delete Draft Pipeline {string}") + public void deletePipelineDraft(String pipelineName) { + String actionsPopoverXpath = + "//*[@data-testid=\"draft-" + pipelineName + "\"]//*[@data-testid=\"actions-popover\"]"; + WebElement actionsPopover = Helper.locateElementByXPath(actionsPopoverXpath); + ElementHelper.clickOnElement(actionsPopover); + String deleteActionXpath = + "//*[@data-testid=\"draft-" + pipelineName + "\"]//*[@data-testid=\"Delete-on-popover\"]"; + ElementHelper.clickOnElement(Helper.locateElementByXPath(deleteActionXpath)); + ElementHelper.clickOnElement(Helper.locateElementByTestId("Delete")); + Helper.waitSeconds(); + Assert.assertFalse(Helper.isElementExists(Helper.getCssSelectorByDataTestId("draft-" + pipelineName))); + } } diff --git a/src/e2e-test/java/io/cdap/cdap/ui/utils/Constants.java b/src/e2e-test/java/io/cdap/cdap/ui/utils/Constants.java index aec8c33c2c4..387aa0c506d 100644 --- a/src/e2e-test/java/io/cdap/cdap/ui/utils/Constants.java +++ b/src/e2e-test/java/io/cdap/cdap/ui/utils/Constants.java @@ -32,6 +32,7 @@ private static double getRandomArbitrary(int min, int max) { public static final String BASE_PIPELINES_URL = BASE_URL + "/pipelines/ns/default"; public static final String CDAP_URL = BASE_URL + "/cdap"; public static final String CONFIGURATION_URL = BASE_URL + "/cdap/administration/configuration"; + public static final String HTTP_EXECUTOR_URL = BASE_URL + "/cdap/httpexecutor"; public static final String SOURCE_CONTROL_MANAGEMENT_URL = BASE_URL + "/cdap/ns/default/details/scm"; public static final String SOURCE_CONTROL_SYNC_URL = BASE_URL + "/cdap/ns/default/scm/sync"; public static final String NAMESPACE_URL = BASE_URL + "/cdap/ns"; diff --git a/src/e2e-test/java/io/cdap/cdap/ui/utils/Helper.java b/src/e2e-test/java/io/cdap/cdap/ui/utils/Helper.java index c67065955c2..2c289bac6a8 100644 --- a/src/e2e-test/java/io/cdap/cdap/ui/utils/Helper.java +++ b/src/e2e-test/java/io/cdap/cdap/ui/utils/Helper.java @@ -33,6 +33,7 @@ import io.cdap.e2e.utils.SeleniumDriver; import io.cdap.e2e.utils.WaitHelper; import org.apache.commons.io.FileUtils; +import org.apache.commons.io.IOUtils; import org.apache.http.NameValuePair; import org.apache.http.client.utils.URIBuilder; import org.eclipse.jgit.api.Git; @@ -60,6 +61,7 @@ import org.slf4j.LoggerFactory; import java.io.File; +import java.io.FileInputStream; import java.io.IOException; import java.io.Reader; import java.net.URISyntaxException; @@ -226,6 +228,14 @@ public static void uploadPipelineFromFile(String filename) { .stalenessOf(locateElementByLocator(By.xpath(pipelineNameXPathSelector)))); } + public static String readPipelineFixtureFile(String filename) throws IOException { + File pipelineJSONFile = new File(Constants.FIXTURES_DIR + filename); + try (FileInputStream inputStream = new FileInputStream(pipelineJSONFile)) { + String contents = IOUtils.toString(inputStream); + return contents; + } + } + public static void deployAndTestPipeline(String filename, String pipelineName) { SeleniumDriver.openPage(Constants.BASE_STUDIO_URL + "pipelines"); WaitHelper.waitForPageToLoad();