From 737467b3b868fd101b0db8d3379d67ebac29c65a Mon Sep 17 00:00:00 2001 From: Uladzislau Date: Fri, 19 May 2023 15:18:24 +0200 Subject: [PATCH 01/34] Release/v1.2.0-221 created --- build.gradle.kts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle.kts b/build.gradle.kts index a1839465e..21d015f51 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -22,7 +22,7 @@ apply(plugin = "kotlin") apply(plugin = "org.jetbrains.intellij") group = "eu.ibagroup" -version = "1.1.0-221" +version = "1.2.0-221" val remoteRobotVersion = "0.11.18" val okHttp3Version = "4.10.0" val kotestVersion = "5.5.5" From beb0466bd0305036b01a1cf777f1fd04a5662ddb Mon Sep 17 00:00:00 2001 From: Uladzislau Date: Fri, 19 May 2023 15:22:18 +0200 Subject: [PATCH 02/34] Release/v1.2.0-223 created --- build.gradle.kts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle.kts b/build.gradle.kts index 21d015f51..a0a2da891 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -22,7 +22,7 @@ apply(plugin = "kotlin") apply(plugin = "org.jetbrains.intellij") group = "eu.ibagroup" -version = "1.2.0-221" +version = "1.2.0-223" val remoteRobotVersion = "0.11.18" val okHttp3Version = "4.10.0" val kotestVersion = "5.5.5" From 9035a46f343850da7d5d07fa2d65d185fba649d4 Mon Sep 17 00:00:00 2001 From: Uladzislau Date: Fri, 19 May 2023 15:24:07 +0200 Subject: [PATCH 03/34] Release/v1.2.0-231 created --- build.gradle.kts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle.kts b/build.gradle.kts index b60a73f88..42a54731f 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -22,7 +22,7 @@ apply(plugin = "kotlin") apply(plugin = "org.jetbrains.intellij") group = "eu.ibagroup" -version = "1.2.0-223" +version = "1.2.0-231" val remoteRobotVersion = "0.11.18" val okHttp3Version = "4.10.0" val kotestVersion = "5.5.5" From c154e027f19e9d41a9509e5582427df48e1630d1 Mon Sep 17 00:00:00 2001 From: Tsikhamirau Date: Mon, 29 May 2023 17:10:33 +0300 Subject: [PATCH 04/34] IJMP-1072-USS-Sorting-feature 1) implemented sorting + refresh --- .../dataops/UnitRemoteQueryImpl.kt | 20 + .../fetch/RemoteFileFetchProviderBase.kt | 8 +- .../explorer/actions/UssSortActionHolder.kt | 451 ++++++++++ .../formainframe/explorer/ui/FileFetchNode.kt | 4 +- .../formainframe/explorer/ui/UssDirNode.kt | 94 +- .../formainframe/explorer/ui/UssNode.kt | 8 +- .../explorer/ui/UssSortableNode.kt | 21 + src/main/resources/META-INF/plugin.xml | 37 + .../actions/UssSortActionHolderTestSpec.kt | 827 ++++++++++++++++++ .../explorer/ui/UssDirNodeTestSpec.kt | 202 +++++ 10 files changed, 1658 insertions(+), 14 deletions(-) create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssSortableNode.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNodeTestSpec.kt diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt index fad70d36e..66c617634 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt @@ -11,6 +11,7 @@ package eu.ibagroup.formainframe.dataops import eu.ibagroup.formainframe.config.connect.ConnectionConfigBase +import eu.ibagroup.formainframe.explorer.ui.UssNode import eu.ibagroup.formainframe.utils.UNIT_CLASS /** @@ -20,6 +21,25 @@ data class UnitRemoteQueryImpl( override val request: R, override val connectionConfig: Connection ) : RemoteQuery { + val sortKeys = mutableListOf(SortQueryKeys.DATE, SortQueryKeys.ASCENDING) + var requester: UssNode? = null override val resultClass: Class get() = UNIT_CLASS +} + +/** + * Enum class represents the sorting keys which is currently enabled for particular Node + */ +enum class SortQueryKeys(private val sortType: String) { + NAME("name"), + TYPE("type"), + DATE("date"), + NONE("none"), + ASCENDING("ascending"), + DESCENDING("descending"); + + override fun toString(): String { + return sortType + } + } \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt index 3588970e1..be6535fa8 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt @@ -15,9 +15,7 @@ import com.intellij.openapi.progress.ProcessCanceledException import com.intellij.openapi.progress.ProgressIndicator import com.intellij.openapi.vfs.VirtualFile import eu.ibagroup.formainframe.config.connect.ConnectionConfigBase -import eu.ibagroup.formainframe.dataops.DataOpsManager -import eu.ibagroup.formainframe.dataops.Query -import eu.ibagroup.formainframe.dataops.RemoteQuery +import eu.ibagroup.formainframe.dataops.* import eu.ibagroup.formainframe.dataops.exceptions.CallException import eu.ibagroup.formainframe.dataops.services.ErrorSeparatorService import eu.ibagroup.formainframe.utils.castOrNull @@ -164,7 +162,8 @@ abstract class RemoteFileFetchProviderBase = getFetchedFiles(query, progressIndicator) // Cleans up attributes of invalid files cache[query] @@ -231,6 +230,7 @@ abstract class RemoteFileFetchProviderBase, sendTopic: Boolean) { cacheState.remove(query) + cache.remove(query) if (sendTopic) { sendTopic(FileFetchProvider.CACHE_CHANGES, dataOpsManager.componentManager).onCacheCleaned(query) } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt new file mode 100644 index 000000000..04105d002 --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt @@ -0,0 +1,451 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions + +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DefaultActionGroup +import com.intellij.openapi.actionSystem.ToggleAction +import com.intellij.openapi.components.service +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.RemoteQuery +import eu.ibagroup.formainframe.dataops.SortQueryKeys +import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.fetch.UssQuery +import eu.ibagroup.formainframe.explorer.ui.FileExplorerView +import eu.ibagroup.formainframe.explorer.ui.UssDirNode +import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.vfs.MFVirtualFile + +/** + * Represents internal USS fetch provider to be able to update the query for each USS dir node whose sorting is enabled + */ +internal val fetchProvider = service().getFileFetchProvider(UssQuery::class.java, RemoteQuery::class.java, MFVirtualFile::class.java) + +/** + * Represents the custom sort action group in the FileExplorerView context menu + */ +class SortActionGroup : DefaultActionGroup() { + + /** + * Update method to determine if sorting is possible for particular item in the tree + */ + override fun update(e: AnActionEvent) { + val view = e.getExplorerView() + view ?: let { + e.presentation.isEnabledAndVisible = false + return@let + } + val selectedNode = view?.mySelectedNodesData ?: return + val treePathFromModel = view.myTree.selectionPath + e.presentation.isEnabledAndVisible = selectedNode.any { + it.node is UssDirNode && view.myTree.isExpanded(treePathFromModel) + } + } +} + +/** + * Custom action for sorting by item Name (Dir or File name in case of USS) + */ +class SortByNameAction : ToggleAction() { + + /** + * Update method to determine if sorting by Name is possible for particular item in the tree + */ + override fun update(e: AnActionEvent) { + super.update(e) + e.presentation.isEnabledAndVisible = true + } + + /** + * Action performed method to register the custom behavior when By Name sorting is clicked + */ + // TODO: Consider using below commented code in v1.*.*-223 and greater: +// override fun actionPerformed(e: AnActionEvent) { +// val view = e.getExplorerView() ?: return +// val selectedNode = view.mySelectedNodesData[0].node +// if (selectedNode is UssDirNode) { +// val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) +// if (selectedNode.currentSortQueryKeysList.isEmpty()) { +// selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) +// selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) +// queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) +// } else { +// selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.TYPE) +// selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DATE) +// if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.NAME)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.NAME) +// queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) +// } +// queryToUpdate.requester = selectedNode +// selectedNode.cleanCache(false) +// fetchProvider.reload(queryToUpdate) +// } +// } + + /** + * Custom isSelected method determines if the sorting By Name is currently enabled or not. Updates UI by 'tick' mark + */ + override fun isSelected(e: AnActionEvent): Boolean { + val view = e.getExplorerView() ?: return false + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + return selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.NAME) + } + return false + } + + /** + * Always set or unset the selected 'tick' in UI. Called by actionPerformed final method in ToggleAction + */ + // TODO: Consider comment out below code and switch to actionPerformed(e) in v1.*.*-223 and greater: + override fun setSelected(e: AnActionEvent, state: Boolean) { + val view = e.getExplorerView() ?: return + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) + if (selectedNode.currentSortQueryKeysList.isEmpty()) { + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) + queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) + } else { + selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.TYPE) + selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DATE) + if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.NAME)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.NAME) + queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) + } + queryToUpdate.requester = selectedNode + selectedNode.cleanCache(false) + fetchProvider.reload(queryToUpdate) + } + } + + /** + * If action is dumb aware or not + */ + override fun isDumbAware(): Boolean { + return true + } +} + +/** + * Custom action for sorting by item Type (Dir or File type+name in case of USS) + */ +class SortByTypeAction : ToggleAction() { + + override fun update(e: AnActionEvent) { + super.update(e) + e.presentation.isEnabledAndVisible = true + } + + /** + * Action performed method to register the custom behavior when By Type sorting is clicked + */ + // TODO: Consider using below commented code in v1.*.*-223 and greater: +// override fun actionPerformed(e: AnActionEvent) { +// val view = e.getExplorerView() ?: return +// val selectedNode = view.mySelectedNodesData[0].node +// if (selectedNode is UssDirNode) { +// val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) +// if (selectedNode.currentSortQueryKeysList.isEmpty()) { +// selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) +// selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) +// queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) +// } else { +// selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.NAME) +// selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DATE) +// if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.TYPE)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.TYPE) +// queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) +// } +// queryToUpdate.requester = selectedNode +// selectedNode.cleanCache(false) +// fetchProvider.reload(queryToUpdate) +// } +// } + + /** + * Custom isSelected method determines if the sorting By Type is currently enabled or not. Updates UI by 'tick' mark + */ + override fun isSelected(e: AnActionEvent): Boolean { + val view = e.getExplorerView() ?: return false + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + return selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.TYPE) + } + return false + } + + /** + * Always set or unset the selected 'tick' in UI. Called by actionPerformed final method in ToggleAction + */ + // TODO: Consider comment out below code and switch to actionPerformed(e) in v1.*.*-223 and greater: + override fun setSelected(e: AnActionEvent, state: Boolean) { + val view = e.getExplorerView() ?: return + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) + if (selectedNode.currentSortQueryKeysList.isEmpty()) { + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) + queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) + } else { + selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.NAME) + selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DATE) + if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.TYPE)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.TYPE) + queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) + } + queryToUpdate.requester = selectedNode + selectedNode.cleanCache(false) + fetchProvider.reload(queryToUpdate) + } + } + + /** + * If action is dumb aware or not + */ + override fun isDumbAware(): Boolean { + return true + } +} + +/** + * Custom action for sorting by item Modification Date (Dir or File modify date in case of USS) + */ +class SortByModificationDateAction : ToggleAction() { + + override fun update(e: AnActionEvent) { + super.update(e) + e.presentation.isEnabledAndVisible = true + } + + /** + * Action performed method to register the custom behavior when By Modification Date sorting is clicked + */ + // TODO: Consider using below commented code in v1.*.*-223 and greater: +// override fun actionPerformed(e: AnActionEvent) { +// val view = e.getExplorerView() ?: return +// val selectedNode = view.mySelectedNodesData[0].node +// if (selectedNode is UssDirNode) { +// val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) +// if (selectedNode.currentSortQueryKeysList.isEmpty()) { +// selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) +// selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) +// queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) +// } else { +// selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.NAME) +// selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.TYPE) +// if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DATE)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) +// queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) +// } +// queryToUpdate.requester = selectedNode +// selectedNode.cleanCache(false) +// fetchProvider.reload(queryToUpdate) +// } +// } + + /** + * Custom isSelected method determines if the sorting By Modification Date is currently enabled or not. Updates UI by 'tick' mark + */ + override fun isSelected(e: AnActionEvent): Boolean { + val view = e.getExplorerView() ?: return false + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + return selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DATE) + } + return false + } + + /** + * Always set or unset the selected 'tick' in UI. Called by actionPerformed final method in ToggleAction + */ + // TODO: Consider comment out below code and switch to actionPerformed(e) in v1.*.*-223 and greater: + override fun setSelected(e: AnActionEvent, state: Boolean) { + val view = e.getExplorerView() ?: return + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) + if (selectedNode.currentSortQueryKeysList.isEmpty()) { + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) + queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) + } else { + selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.NAME) + selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.TYPE) + if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DATE)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) + queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) + } + queryToUpdate.requester = selectedNode + selectedNode.cleanCache(false) + fetchProvider.reload(queryToUpdate) + } + } + + /** + * If action is dumb aware or not + */ + override fun isDumbAware(): Boolean { + return true + } +} + +/** + * Custom action for sorting by item Name/Type/Modify Date in ascending order (Dir or File name in case of USS) + */ +class SortByAscendingOrderAction : ToggleAction() { + + override fun update(e: AnActionEvent) { + super.update(e) + e.presentation.isEnabledAndVisible = true + } + + /** + * Action performed method to register the custom behavior when Ascending sorting is clicked + */ + // TODO: Consider using below commented code in v1.*.*-223 and greater: +// override fun actionPerformed(e: AnActionEvent) { +// val view = e.getExplorerView() ?: return +// val selectedNode = view.mySelectedNodesData[0].node +// if (selectedNode is UssDirNode) { +// val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) +// if (selectedNode.currentSortQueryKeysList.isEmpty()) { +// selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) +// selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) +// queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) +// } else { +// selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DESCENDING) +// if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.ASCENDING)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) +// queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) +// } +// queryToUpdate.requester = selectedNode +// selectedNode.cleanCache(false) +// fetchProvider.reload(queryToUpdate) +// } +// } + + /** + * Custom isSelected method determines if the ascending sorting is currently enabled or not. Updates UI by 'tick' mark + */ + override fun isSelected(e: AnActionEvent): Boolean { + val view = e.getExplorerView() ?: return false + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + return selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.ASCENDING) + } + return false + } + + /** + * Always set or unset the selected 'tick' in UI. Called by actionPerformed final method in ToggleAction + */ + // TODO: Consider comment out below code and switch to actionPerformed(e) in v1.*.*-223 and greater: + override fun setSelected(e: AnActionEvent, state: Boolean) { + val view = e.getExplorerView() ?: return + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) + if (selectedNode.currentSortQueryKeysList.isEmpty()) { + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) + queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) + } else { + selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DESCENDING) + if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.ASCENDING)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) + queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) + } + queryToUpdate.requester = selectedNode + selectedNode.cleanCache(false) + fetchProvider.reload(queryToUpdate) + } + } + + /** + * If action is dumb aware or not + */ + override fun isDumbAware(): Boolean { + return true + } +} + +/** + * Custom action for sorting by item Name/Type/Modify Date in descending order (Dir or File name in case of USS) + */ +class SortByDescendingOrderAction : ToggleAction() { + + override fun update(e: AnActionEvent) { + super.update(e) + e.presentation.isEnabledAndVisible = true + } + + /** + * Action performed method to register the custom behavior when Descending sorting is clicked + */ + // TODO: Consider using below commented code in v1.*.*-223 and greater: +// override fun actionPerformed(e: AnActionEvent) { +// val view = e.getExplorerView() ?: return +// val selectedNode = view.mySelectedNodesData[0].node +// if (selectedNode is UssDirNode) { +// val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) +// if (selectedNode.currentSortQueryKeysList.isEmpty()) { +// selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) +// selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DESCENDING) +// queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) +// } else { +// selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.ASCENDING) +// if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DESCENDING)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DESCENDING) +// queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) +// } +// queryToUpdate.requester = selectedNode +// selectedNode.cleanCache(false) +// fetchProvider.reload(queryToUpdate) +// } +// } + + /** + * Custom isSelected method determines if the descending sorting is currently enabled or not. Updates UI by 'tick' mark + */ + override fun isSelected(e: AnActionEvent): Boolean { + val view = e.getExplorerView() ?: return false + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + return selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DESCENDING) + } + return false + } + + /** + * Always set or unset the selected 'tick' in UI. Called by actionPerformed final method in ToggleAction + */ + // TODO: Consider comment out below code and switch to actionPerformed(e) in v1.*.*-223 and greater: + override fun setSelected(e: AnActionEvent, state: Boolean) { + val view = e.getExplorerView() ?: return + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) + if (selectedNode.currentSortQueryKeysList.isEmpty()) { + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DESCENDING) + queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) + } else { + selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.ASCENDING) + if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DESCENDING)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DESCENDING) + queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) + } + queryToUpdate.requester = selectedNode + selectedNode.cleanCache(false) + fetchProvider.reload(queryToUpdate) + } + } + + /** + * If action is dumb aware or not + */ + override fun isDumbAware(): Boolean { + return true + } +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNode.kt index 555035b49..748ba91cc 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNode.kt @@ -18,9 +18,7 @@ import com.intellij.ui.tree.LeafState import com.intellij.util.containers.toMutableSmartList import eu.ibagroup.formainframe.common.message import eu.ibagroup.formainframe.config.connect.ConnectionConfigBase -import eu.ibagroup.formainframe.dataops.BatchedRemoteQuery -import eu.ibagroup.formainframe.dataops.DataOpsManager -import eu.ibagroup.formainframe.dataops.Query +import eu.ibagroup.formainframe.dataops.* import eu.ibagroup.formainframe.explorer.ExplorerUnit import eu.ibagroup.formainframe.utils.castOrNull import eu.ibagroup.formainframe.utils.locked diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNode.kt index 34f10c65c..1b55bb7c4 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNode.kt @@ -17,12 +17,9 @@ import com.intellij.openapi.project.Project import com.intellij.util.IconUtil import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.ws.UssPath -import eu.ibagroup.formainframe.dataops.DataOpsManager -import eu.ibagroup.formainframe.dataops.RemoteQuery -import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.* import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes import eu.ibagroup.formainframe.dataops.fetch.UssQuery -import eu.ibagroup.formainframe.dataops.getAttributesService import eu.ibagroup.formainframe.explorer.FilesWorkingSet import eu.ibagroup.formainframe.utils.service import eu.ibagroup.formainframe.vfs.MFVirtualFile @@ -58,6 +55,12 @@ class UssDirNode( super.init() } + /** Field which holds and identifies the current sort keys for particular Node, be default Nodes are sorted by Data in Ascending order */ + var currentSortQueryKeysList = mutableListOf(SortQueryKeys.DATE, SortQueryKeys.ASCENDING) + + /** Stores the sorted nodes for particular Node */ + private var sortedNodes: List> = mutableListOf() + val isConfigUssPath = vFile == null override val query: RemoteQuery? @@ -93,7 +96,7 @@ class UssDirNode( } else { UssFileNode(it, notNullProject, this@UssDirNode, unit, treeStructure) } - } ?: listOf() + }?.sortChildrenNodes(currentSortQueryKeysList) ?: listOf() } override val requestClass = UssQuery::class.java @@ -141,4 +144,85 @@ class UssDirNode( return vFile } + /** + * Method sorts the children nodes regarding the sort keys are currently enabled + * @param sortKeys - Sort keys to check + * @return list of sorted nodes + */ + override fun List>.sortChildrenNodes(sortKeys: List): List> { + if (sortKeys.contains(SortQueryKeys.NAME)) { + if (sortKeys.contains(SortQueryKeys.ASCENDING)) { + return this.sortedBy { + when (it) { + is UssDirNode -> it.vFile?.filenameInternal + is UssFileNode -> it.virtualFile.filenameInternal + else -> null + } + }.also { sortedNodes = it } + } else { + return this.sortedByDescending { + when (it) { + is UssDirNode -> it.vFile?.filenameInternal + is UssFileNode -> it.virtualFile.filenameInternal + else -> null + } + }.also { sortedNodes = it } + } + } else if (sortKeys.contains(SortQueryKeys.TYPE)) { + val listToReturn = mutableListOf>() + val dirs = mutableListOf() + val files = mutableListOf() + val sortedDirs: List + val sortedFiles: List + this.forEach { + when (it) { + is UssDirNode -> dirs.add(it) + is UssFileNode -> files.add(it) + } + } + return if (sortKeys.contains(SortQueryKeys.ASCENDING)) { + sortedDirs = dirs.sortedBy { it.vFile?.filenameInternal } + sortedFiles = files.sortedBy { it.virtualFile.filenameInternal } + listToReturn.addAll(sortedDirs) + listToReturn.addAll(sortedFiles) + listToReturn.also { sortedNodes = it } + } else { + sortedDirs = dirs.sortedByDescending { it.vFile?.filenameInternal } + sortedFiles = files.sortedByDescending { it.virtualFile.filenameInternal } + listToReturn.addAll(sortedDirs) + listToReturn.addAll(sortedFiles) + listToReturn.also { sortedNodes = it } + } + } else if (sortKeys.contains(SortQueryKeys.DATE)) { + + fun select(node: AbstractTreeNode<*>) : String? { + return when (node) { + is UssDirNode -> { + node.virtualFile?.let { vFile -> attributesService.getAttributes(vFile) }?.modificationTime + } + is UssFileNode -> { + attributesService.getAttributes(node.virtualFile)?.modificationTime + } + else -> null + } + } + + return if (sortKeys.contains(SortQueryKeys.ASCENDING)) { + this.sortedBy { + return@sortedBy select(it) + }.also { sortedNodes = it } + } else { + this.sortedByDescending { + return@sortedByDescending select(it) + }.also { sortedNodes = it } + } + } else { + return this + } + } + + fun sortChildrenForTestInternal(sortKeys: List): (List>) -> List> = { it.sortChildrenNodes(sortKeys) } } + +fun sortChildrenForTest(listToSort: List>, sortKeys: List): UssDirNode.() -> List> + = { sortChildrenForTestInternal(sortKeys).invoke(listToSort) } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssNode.kt index 6718f8eca..7bb375d66 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssNode.kt @@ -10,5 +10,9 @@ package eu.ibagroup.formainframe.explorer.ui -interface UssNode { -} \ No newline at end of file +import com.intellij.ide.util.treeView.AbstractTreeNode + +/** + * interface which represents any USS Node. Extents USS sortable Node which implements children nodes sorting method + */ +interface UssNode : UssSortableNode> \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssSortableNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssSortableNode.kt new file mode 100644 index 000000000..a7d90e6ca --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssSortableNode.kt @@ -0,0 +1,21 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.ui + +import eu.ibagroup.formainframe.dataops.SortQueryKeys + +/** + * Interface which represents any USS sortable Node + * @param Node - Nodes type to sort + */ +interface UssSortableNode { + fun List.sortChildrenNodes(sortKeys: List): List = mutableListOf() +} \ No newline at end of file diff --git a/src/main/resources/META-INF/plugin.xml b/src/main/resources/META-INF/plugin.xml index 8fd6f4f96..f6f4288b2 100755 --- a/src/main/resources/META-INF/plugin.xml +++ b/src/main/resources/META-INF/plugin.xml @@ -423,6 +423,28 @@ Example of how to see the output:
class="eu.ibagroup.formainframe.explorer.actions.CreateUssDirectoryAction" text="Directory" icon="AllIcons.Nodes.Folder"/> + + + + + + + + + + + + + + + + + + + @@ -593,6 +628,8 @@ Example of how to see the output:
+ + diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt new file mode 100644 index 000000000..7805539c6 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt @@ -0,0 +1,827 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions + +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.application.ApplicationManager +import com.intellij.openapi.vfs.VirtualFile +import com.intellij.testFramework.LightProjectDescriptor +import com.intellij.testFramework.fixtures.IdeaTestFixtureFactory +import com.intellij.testFramework.fixtures.impl.LightTempDirTestFixtureImpl +import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.Query +import eu.ibagroup.formainframe.dataops.SortQueryKeys +import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes +import eu.ibagroup.formainframe.dataops.fetch.FileFetchProvider +import eu.ibagroup.formainframe.dataops.fetch.UssFileFetchProvider +import eu.ibagroup.formainframe.dataops.fetch.UssQuery +import eu.ibagroup.formainframe.explorer.ui.* +import eu.ibagroup.formainframe.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.utils.service +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.core.spec.style.ShouldSpec +import io.kotest.matchers.shouldBe +import io.mockk.* +import javax.swing.tree.TreePath + +class UssSortActionHolderTestSpec : ShouldSpec({ + beforeSpec { + // FIXTURE SETUP TO HAVE ACCESS TO APPLICATION INSTANCE + val factory = IdeaTestFixtureFactory.getFixtureFactory() + val projectDescriptor = LightProjectDescriptor.EMPTY_PROJECT_DESCRIPTOR + val fixtureBuilder = factory.createLightFixtureBuilder(projectDescriptor, "for-mainframe") + val fixture = fixtureBuilder.fixture + val myFixture = IdeaTestFixtureFactory.getFixtureFactory().createCodeInsightFixture( + fixture, + LightTempDirTestFixtureImpl(true) + ) + myFixture.setUp() + } + + afterSpec { + clearAllMocks() + } + + context("sort actions") { + + val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + val mockedFileFetchProvider = mockk() + dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(ApplicationManager.getApplication()) { + @Suppress("UNCHECKED_CAST") + override fun , File : VirtualFile> getFileFetchProvider( + requestClass: Class, + queryClass: Class>, + vFileClass: Class + ): FileFetchProvider { + return mockedFileFetchProvider as FileFetchProvider + } + + } + val mockedActionEvent = mockk() + every { mockedActionEvent.presentation } returns mockk() + every { mockedActionEvent.presentation.putClientProperty(any(), any()) } just Runs + // isFromContextMenu is marked as deprecated + every { mockedActionEvent.isFromContextMenu } returns false + every { mockedActionEvent.presentation.isEnabledAndVisible = true } just Runs + every { mockedActionEvent.presentation.isEnabledAndVisible = false } just Runs + + context("common spec") { + + val mockedFileExplorerView = mockk() + + // Target UssDirNode + Query for test + val mockedMFVirtualFile = mockk() + val mockedUssDirNode = mockk() + val mockedUssRemoteAttributes = mockk() + val mockedUssQuery = mockk>() + every { mockedUssDirNode.virtualFile } returns mockedMFVirtualFile + every { mockedUssDirNode.query } returns mockedUssQuery + every { mockedUssDirNode.cleanCache(false) } just Runs + every { mockedUssQuery.setProperty("requester").value(mockedUssDirNode) } just Runs + every { mockedUssQuery.sortKeys } returns mutableListOf() + + // NodeData for test + val mockedNodeDataForTest = NodeData(mockedUssDirNode, mockedMFVirtualFile, mockedUssRemoteAttributes) + mockkObject(mockedNodeDataForTest) + + // Common config for test + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + + context("sort action group spec") { + + // group action to spy + val mockedActionGroup = spyk(SortActionGroup()) + + val selectionPath = mockk() + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { mockedFileExplorerView.myTree } returns mockk() + every { mockedFileExplorerView.myTree.selectionPath } returns selectionPath + + should("is visible from context menu if file explorer view is null") { + var isVisible = true + every { mockedActionEvent.getExplorerView() } answers { + isVisible = false + null + } + + mockedActionGroup.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + + should("is visible from context menu if file explorer view is not null and selected node is not UssDirNode") { + var isVisible = true + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + isVisible = false + listOf(mockedNodeDataNotUssForTest) + } + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + + mockedActionGroup.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + + should("is visible from context menu if file explorer view is not null and selected node is UssDirNode and path is expanded") { + var isVisible = false + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedFileExplorerView.myTree.isExpanded(selectionPath) } answers { + isVisible = true + true + } + + mockedActionGroup.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe true + } + } + + should("is visible from context menu if file explorer view is not null and selected node is UssDirNode and path is not expanded") { + var isVisible = true + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedFileExplorerView.myTree.isExpanded(selectionPath) } answers { + isVisible = false + false + } + + mockedActionGroup.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + } + + context("sort by name action spec") { + + // action to spy + val mockedSortActionByName = spyk(SortByNameAction()) + + should("sort by name action performed if file explorer is null") { + var actionPerformed = false + every { mockedActionEvent.getExplorerView() } answers { + actionPerformed = true + null + } + mockedSortActionByName.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by name action isSelected if file explorer is null") { + var isSelected = true + every { mockedActionEvent.getExplorerView() } answers { + isSelected = false + null + } + mockedSortActionByName.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort by name action performed if selected node is not UssDirNode") { + var actionPerformed = false + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + actionPerformed = true + listOf(mockedNodeDataNotUssForTest) + } + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + mockedSortActionByName.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by name action isSelected if selected node is not UssDirNode") { + var isSelected = true + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + isSelected = false + listOf(mockedNodeDataNotUssForTest) + } + mockedSortActionByName.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort by name action performed if sort query keys is empty") { + var actionPerformed = false + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf() + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionByName.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by name action isSelected if current sort query keys does not contain NAME key") { + var isSelected = true + every { mockedUssDirNode.currentSortQueryKeysList } answers { + isSelected = false + mutableListOf() + } + mockedSortActionByName.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort by name action performed if sort query keys is not empty") { + var actionPerformed = false + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DATE, SortQueryKeys.ASCENDING) + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionByName.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by name action performed if sort query keys is not empty and contains desired key") { + var actionPerformed = false + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DATE, SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionByName.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by name action isSelected if current sort query keys contain NAME key") { + var isSelected = false + every { mockedUssDirNode.currentSortQueryKeysList } answers { + isSelected = true + mutableListOf(SortQueryKeys.NAME) + } + mockedSortActionByName.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe true + } + } + + should("sort by name action miscellaneous") { + mockedSortActionByName.setSelected(mockedActionEvent, true) + mockedSortActionByName.setSelected(mockedActionEvent, false) + mockedSortActionByName.update(mockedActionEvent) + val aware = mockedSortActionByName.isDumbAware + assertSoftly { + aware shouldBe true + } + } + } + + context("sort by type action spec") { + + // action to spy + val mockedSortActionByType = spyk(SortByTypeAction()) + + should("sort by type action performed if file explorer is null") { + var actionPerformed = false + every { mockedActionEvent.getExplorerView() } answers { + actionPerformed = true + null + } + mockedSortActionByType.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by type action isSelected if file explorer is null") { + var isSelected = true + every { mockedActionEvent.getExplorerView() } answers { + isSelected = false + null + } + mockedSortActionByType.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort by type action performed if selected node is not UssDirNode") { + var actionPerformed = false + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + actionPerformed = true + listOf(mockedNodeDataNotUssForTest) + } + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + mockedSortActionByType.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by type action isSelected if selected node is not UssDirNode") { + var isSelected = true + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + isSelected = false + listOf(mockedNodeDataNotUssForTest) + } + mockedSortActionByType.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort by type action performed if sort query keys is empty") { + var actionPerformed = false + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf() + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionByType.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by type action isSelected if current sort query keys does not contain TYPE key") { + var isSelected = true + every { mockedUssDirNode.currentSortQueryKeysList } answers { + isSelected = false + mutableListOf() + } + mockedSortActionByType.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort by type action performed if sort query keys is not empty") { + var actionPerformed = false + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DATE, SortQueryKeys.ASCENDING) + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionByType.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by type action performed if sort query keys is not empty and contains desired key") { + var actionPerformed = false + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DATE, SortQueryKeys.TYPE, SortQueryKeys.ASCENDING) + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionByType.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by type action isSelected if current sort query keys contain TYPE key") { + var isSelected = false + every { mockedUssDirNode.currentSortQueryKeysList } answers { + isSelected = true + mutableListOf(SortQueryKeys.TYPE) + } + mockedSortActionByType.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe true + } + } + + should("sort by type action miscellaneous") { + mockedSortActionByType.setSelected(mockedActionEvent, true) + mockedSortActionByType.setSelected(mockedActionEvent, false) + mockedSortActionByType.update(mockedActionEvent) + val aware = mockedSortActionByType.isDumbAware + assertSoftly { + aware shouldBe true + } + } + } + + context("sort by date action spec") { + + // action to spy + val mockedSortActionByDate = spyk(SortByModificationDateAction()) + + should("sort by date action performed if file explorer is null") { + var actionPerformed = false + every { mockedActionEvent.getExplorerView() } answers { + actionPerformed = true + null + } + mockedSortActionByDate.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by date action isSelected if file explorer is null") { + var isSelected = true + every { mockedActionEvent.getExplorerView() } answers { + isSelected = false + null + } + mockedSortActionByDate.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort by date action performed if selected node is not UssDirNode") { + var actionPerformed = false + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + actionPerformed = true + listOf(mockedNodeDataNotUssForTest) + } + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + mockedSortActionByDate.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by date action isSelected if selected node is not UssDirNode") { + var isSelected = true + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + isSelected = false + listOf(mockedNodeDataNotUssForTest) + } + mockedSortActionByDate.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort by date action performed if sort query keys is empty") { + var actionPerformed = false + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf() + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionByDate.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by date action isSelected if current sort query keys does not contain DATE key") { + var isSelected = true + every { mockedUssDirNode.currentSortQueryKeysList } answers { + isSelected = false + mutableListOf() + } + mockedSortActionByDate.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort by date action performed if sort query keys is not empty") { + var actionPerformed = false + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionByDate.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by date action performed if sort query keys is not empty and contains desired key") { + var actionPerformed = false + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DATE, SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionByDate.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort by date action isSelected if current sort query keys contain DATE key") { + var isSelected = false + every { mockedUssDirNode.currentSortQueryKeysList } answers { + isSelected = true + mutableListOf(SortQueryKeys.DATE) + } + mockedSortActionByDate.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe true + } + } + + should("sort by date action miscellaneous") { + mockedSortActionByDate.setSelected(mockedActionEvent, true) + mockedSortActionByDate.setSelected(mockedActionEvent, false) + mockedSortActionByDate.update(mockedActionEvent) + val aware = mockedSortActionByDate.isDumbAware + assertSoftly { + aware shouldBe true + } + } + } + + context("sort ascending action spec") { + + // action to spy + val mockedSortActionAscending = spyk(SortByAscendingOrderAction()) + + should("sort Ascending action performed if file explorer is null") { + var actionPerformed = false + every { mockedActionEvent.getExplorerView() } answers { + actionPerformed = true + null + } + mockedSortActionAscending.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort Ascending action isSelected if file explorer is null") { + var isSelected = true + every { mockedActionEvent.getExplorerView() } answers { + isSelected = false + null + } + mockedSortActionAscending.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort Ascending action performed if selected node is not UssDirNode") { + var actionPerformed = false + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + actionPerformed = true + listOf(mockedNodeDataNotUssForTest) + } + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + mockedSortActionAscending.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort Ascending action isSelected if selected node is not UssDirNode") { + var isSelected = true + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + isSelected = false + listOf(mockedNodeDataNotUssForTest) + } + mockedSortActionAscending.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort Ascending action performed if sort query keys is empty") { + var actionPerformed = false + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf() + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionAscending.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort Ascending action isSelected if current sort query keys does not contain ASCENDING key") { + var isSelected = true + every { mockedUssDirNode.currentSortQueryKeysList } answers { + isSelected = false + mutableListOf() + } + mockedSortActionAscending.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort Ascending action performed if sort query keys is not empty") { + var actionPerformed = false + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.NAME, SortQueryKeys.DESCENDING) + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionAscending.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort Ascending action performed if sort query keys is not empty and contains desired key") { + var actionPerformed = false + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DESCENDING, SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionAscending.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort Ascending action isSelected if current sort query keys contain ASCENDING key") { + var isSelected = false + every { mockedUssDirNode.currentSortQueryKeysList } answers { + isSelected = true + mutableListOf(SortQueryKeys.ASCENDING) + } + mockedSortActionAscending.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe true + } + } + + should("sort Ascending action miscellaneous") { + mockedSortActionAscending.setSelected(mockedActionEvent, true) + mockedSortActionAscending.setSelected(mockedActionEvent, false) + mockedSortActionAscending.update(mockedActionEvent) + val aware = mockedSortActionAscending.isDumbAware + assertSoftly { + aware shouldBe true + } + } + } + + context("sort descending action spec") { + + // action to spy + val mockedSortActionDescending = spyk(SortByDescendingOrderAction()) + + should("sort Descending action performed if file explorer is null") { + var actionPerformed = false + every { mockedActionEvent.getExplorerView() } answers { + actionPerformed = true + null + } + mockedSortActionDescending.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort Descending action isSelected if file explorer is null") { + var isSelected = true + every { mockedActionEvent.getExplorerView() } answers { + isSelected = false + null + } + mockedSortActionDescending.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort Descending action performed if selected node is not UssDirNode") { + var actionPerformed = false + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + actionPerformed = true + listOf(mockedNodeDataNotUssForTest) + } + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + mockedSortActionDescending.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort Descending action isSelected if selected node is not UssDirNode") { + var isSelected = true + val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + isSelected = false + listOf(mockedNodeDataNotUssForTest) + } + mockedSortActionDescending.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort Descending action performed if sort query keys is empty") { + var actionPerformed = false + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf() + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionDescending.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort Descending action isSelected if current sort query keys does not contain DESCENDING key") { + var isSelected = true + every { mockedUssDirNode.currentSortQueryKeysList } answers { + isSelected = false + mutableListOf() + } + mockedSortActionDescending.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("sort Descending action performed if sort query keys is not empty") { + var actionPerformed = false + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionDescending.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort Descending action performed if sort query keys is not empty and contains desired key") { + var actionPerformed = false + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DESCENDING, SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { + actionPerformed = true + } + + mockedSortActionDescending.actionPerformed(mockedActionEvent) + assertSoftly { + actionPerformed shouldBe true + } + } + + should("sort Descending action isSelected if current sort query keys contain DESCENDING key") { + var isSelected = false + every { mockedUssDirNode.currentSortQueryKeysList } answers { + isSelected = true + mutableListOf(SortQueryKeys.DESCENDING) + } + mockedSortActionDescending.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe true + } + } + + should("sort Descending action miscellaneous") { + mockedSortActionDescending.setSelected(mockedActionEvent, true) + mockedSortActionDescending.setSelected(mockedActionEvent, false) + mockedSortActionDescending.update(mockedActionEvent) + val aware = mockedSortActionDescending.isDumbAware + assertSoftly { + aware shouldBe true + } + } + } + } + } + +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNodeTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNodeTestSpec.kt new file mode 100644 index 000000000..7348eefb9 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNodeTestSpec.kt @@ -0,0 +1,202 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.ui + +import com.intellij.ide.util.treeView.AbstractTreeNode +import com.intellij.openapi.project.Project +import com.intellij.testFramework.LightProjectDescriptor +import com.intellij.testFramework.fixtures.IdeaTestFixtureFactory +import com.intellij.testFramework.fixtures.impl.LightTempDirTestFixtureImpl +import eu.ibagroup.formainframe.config.ws.UssPath +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.SortQueryKeys +import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributesService +import eu.ibagroup.formainframe.dataops.getAttributesService +import eu.ibagroup.formainframe.explorer.FileExplorer +import eu.ibagroup.formainframe.explorer.FilesWorkingSetImpl +import eu.ibagroup.formainframe.utils.service +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.core.spec.style.ShouldSpec +import io.kotest.matchers.shouldBe +import io.mockk.* + +class UssDirNodeTestSpec : ShouldSpec({ + beforeSpec { + // FIXTURE SETUP TO HAVE ACCESS TO APPLICATION INSTANCE + val factory = IdeaTestFixtureFactory.getFixtureFactory() + val projectDescriptor = LightProjectDescriptor.EMPTY_PROJECT_DESCRIPTOR + val fixtureBuilder = factory.createLightFixtureBuilder(projectDescriptor, "for-mainframe") + val fixture = fixtureBuilder.fixture + val myFixture = IdeaTestFixtureFactory.getFixtureFactory().createCodeInsightFixture( + fixture, + LightTempDirTestFixtureImpl(true) + ) + myFixture.setUp() + } + afterSpec { + clearAllMocks() + } + context("explorer module: ui/UssDirNode") { + val mockedPath = mockk("USS//test_path/ZOSMFAD/test_dir") + val mockedProject = mockk() + val mockedExplorerTreeNodeParent = mockk() + val mockedWorkingSet = mockk() + val mockedExplorer = mockk() + val mockedExplorerTreeStructure = mockk() + val isRootNode = false + + val mockedUssAttributesService = mockk() + + every { mockedWorkingSet.explorer } returns mockedExplorer + every { mockedExplorer.componentManager } returns mockk() + every { mockedExplorer.componentManager.service() } returns mockk() + every { mockedExplorer.componentManager.service().getAttributesService() } returns mockedUssAttributesService + every { mockedExplorerTreeStructure.registerNode(any()) } just Runs + + context("sort children nodes") { + val mockedVFile = mockk() + val childNode3 = mockk() + val childNode4 = mockk() + val unexpectedNode = mockk() + val mockedVFileChild1 = mockk() + val mockedVFileChild2 = mockk() + val mockedAttributes1 = mockk() + val mockedAttributes2 = mockk() + val mockedAttributes3 = mockk() + val mockedAttributes4 = mockk() + + beforeEach { + every { mockedVFile.filenameInternal } returns "test_dir" + every { mockedVFileChild1.filenameInternal } returns "a_dir" + every { mockedVFileChild2.filenameInternal } returns "b_dir" + every { childNode3.virtualFile } returns mockk() + every { childNode3.virtualFile.filenameInternal } returns "a_file" + every { childNode4.virtualFile } returns mockk() + every { childNode4.virtualFile.filenameInternal } returns "b_file" + + every { mockedUssAttributesService.getAttributes(mockedVFileChild1) } returns mockedAttributes1 + every { mockedUssAttributesService.getAttributes(mockedVFileChild2) } returns mockedAttributes2 + every { mockedUssAttributesService.getAttributes(childNode3.virtualFile) } returns mockedAttributes3 + every { mockedUssAttributesService.getAttributes(childNode4.virtualFile) } returns mockedAttributes4 + + every { mockedAttributes1.modificationTime } returns "2022/12/11" + every { mockedAttributes2.modificationTime } returns "2022/12/10" + every { mockedAttributes3.modificationTime } returns "2022/12/09" + every { mockedAttributes4.modificationTime } returns "2022/12/08" + } + + val mockedUssDirNodeChild1 = spyk( + UssDirNode(mockedPath, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure, mockedVFileChild1, isRootNode) + ) + val mockedUssDirNodeChild2 = spyk( + UssDirNode(mockedPath, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure, mockedVFileChild2, isRootNode) + ) + val mockedUssDirNode = spyk( + UssDirNode(mockedPath, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure, mockedVFile, isRootNode) + ) + + val mockedChildrenNodes = listOf>(mockedUssDirNodeChild1, mockedUssDirNodeChild2, childNode3, childNode4, unexpectedNode) + + should("sort by name ascending") { + + val sortQueryKeys = listOf(SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + mockkObject(sortQueryKeys) + + val expected = listOf(unexpectedNode, mockedUssDirNodeChild1, childNode3, mockedUssDirNodeChild2, childNode4) + val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by name descending") { + + val sortQueryKeys = listOf(SortQueryKeys.NAME, SortQueryKeys.DESCENDING) + mockkObject(sortQueryKeys) + + val expected = listOf(childNode4, mockedUssDirNodeChild2, childNode3, mockedUssDirNodeChild1, unexpectedNode) + val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by type ascending") { + + val sortQueryKeys = listOf(SortQueryKeys.TYPE, SortQueryKeys.ASCENDING) + mockkObject(sortQueryKeys) + + val expected = listOf(mockedUssDirNodeChild1, mockedUssDirNodeChild2, childNode3, childNode4) + val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by type descending") { + + val sortQueryKeys = listOf(SortQueryKeys.TYPE, SortQueryKeys.DESCENDING) + mockkObject(sortQueryKeys) + + val expected = listOf(mockedUssDirNodeChild2, mockedUssDirNodeChild1, childNode4, childNode3) + val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by date ascending") { + + val sortQueryKeys = listOf(SortQueryKeys.DATE, SortQueryKeys.ASCENDING) + mockkObject(sortQueryKeys) + + val expected = listOf(unexpectedNode, childNode4, childNode3, mockedUssDirNodeChild2, mockedUssDirNodeChild1) + val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by date descending") { + + val sortQueryKeys = listOf(SortQueryKeys.DATE, SortQueryKeys.DESCENDING) + mockkObject(sortQueryKeys) + + val expected = listOf(mockedUssDirNodeChild1, mockedUssDirNodeChild2, childNode3, childNode4, unexpectedNode) + val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by none key") { + + val sortQueryKeys = listOf(SortQueryKeys.NONE) + mockkObject(sortQueryKeys) + + val expected = listOf(mockedUssDirNodeChild1, mockedUssDirNodeChild2, childNode3, childNode4, unexpectedNode) + val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + + assertSoftly { + actual shouldBe expected + } + } + } + } +}) \ No newline at end of file From 68c5e7de4b67ce648d19243f83b65dea50c4c7d2 Mon Sep 17 00:00:00 2001 From: Nikita Date: Fri, 2 Jun 2023 08:43:08 +0300 Subject: [PATCH 05/34] IJMP-1089: reworked error notifications --- .../dataops/exceptions/CallException.kt | 2 +- .../formainframe/explorer/Explorer.kt | 31 +++++++++++++++++-- 2 files changed, 29 insertions(+), 4 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/exceptions/CallException.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/exceptions/CallException.kt index 257c8b07c..720f42715 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/exceptions/CallException.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/exceptions/CallException.kt @@ -43,7 +43,7 @@ private fun formatMessage(code: Int, message: String, errorParams: Map<*, *>?): */ class CallException( val code: Int, - headMessage: String, + val headMessage: String, val errorParams: Map<*, *>? = null, override val cause: Throwable? = null ) : Exception(formatMessage(code, headMessage, errorParams)) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt index d8b8418dd..1bc8ed12d 100755 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt @@ -12,6 +12,7 @@ package eu.ibagroup.formainframe.explorer import com.intellij.ide.BrowserUtil import com.intellij.notification.Notification +import com.intellij.notification.NotificationAction import com.intellij.notification.NotificationType import com.intellij.notification.Notifications import com.intellij.openapi.Disposable @@ -24,6 +25,7 @@ import com.intellij.openapi.extensions.ExtensionPointName import com.intellij.openapi.progress.ProcessCanceledException import com.intellij.openapi.project.DumbAwareAction import com.intellij.openapi.project.Project +import com.intellij.openapi.ui.Messages import com.intellij.openapi.util.Disposer import com.intellij.util.messages.Topic import eu.ibagroup.formainframe.config.CONFIGS_CHANGED @@ -33,6 +35,7 @@ import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.connect.ConnectionConfigBase import eu.ibagroup.formainframe.config.connect.CredentialsListener import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.exceptions.CallException import eu.ibagroup.formainframe.editor.ChangeContentService import eu.ibagroup.formainframe.utils.* import eu.ibagroup.formainframe.utils.crudable.EntityWithUuid @@ -196,12 +199,34 @@ abstract class AbstractExplorerBase>()?.joinToString("\n") ?: "Unknown error" + } else { + title = t.message ?: t.toString() + details = "Unknown error" + } + Notification( EXPLORER_NOTIFICATION_GROUP_ID, - "Error in plugin For Mainframe", - t.message ?: t.toString(), + title, + details, NotificationType.ERROR - ).let { + ).addAction(object : NotificationAction("More") { + override fun actionPerformed(e: AnActionEvent, notification: Notification) { + Messages.showErrorDialog( + project, + t.message ?: t.toString(), + title + ) + } + }).let { Notifications.Bus.notify(it) } } From 3b83005d29e5e0b8499ff5b35bdc8f3315b609c4 Mon Sep 17 00:00:00 2001 From: Tsikhamirau Date: Fri, 23 Jun 2023 14:25:59 +0300 Subject: [PATCH 06/34] IJMP-1114-data-ops-unit-tests-pt1 1) added JobFetcherHelper tests --- .../dataops/fetch/JobFetchHelper.kt | 5 +- .../DatasetFileFetchProviderTestSpec.kt | 187 +++++++++++++ .../formainframe/dataops/FetchTestSpec.kt | 23 -- .../dataops/JobFetchHelperTestSpec.kt | 223 ++++++++++++++++ .../dataops/JobFetchProviderTestSpec.kt | 252 ++++++++++++++++++ .../actions/UssSortActionHolderTestSpec.kt | 2 +- 6 files changed, 664 insertions(+), 28 deletions(-) create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/dataops/DatasetFileFetchProviderTestSpec.kt delete mode 100644 src/test/kotlin/eu/ibagroup/formainframe/dataops/FetchTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchHelperTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchProviderTestSpec.kt diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/JobFetchHelper.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/JobFetchHelper.kt index 1f231e441..49078cea6 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/JobFetchHelper.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/JobFetchHelper.kt @@ -42,10 +42,7 @@ class JobFetchHelper(private val query: RemoteQuery() + every { mockedConnectionConfig.authToken } returns "auth_token" + every { mockedConnectionConfig.url } returns "test_url" + val mockedQuery = mockk>() + val mockedRequest = mockk() + every { mockedRequest.mask } returns "TESTMASK" + every { mockedRequest.volser } returns "TESTVOL" + + val progressMockk = mockk() + every { progressMockk.fraction = any() as Double } just Runs + + every { mockedQuery.request } returns mockedRequest + every { mockedQuery.connectionConfig } returns mockedConnectionConfig + + val dataset1 = mockk() + val dataset2 = mockk() + val dataset3 = mockk() + every { dataset1.name } returns "DATASET1" + every { dataset2.name } returns "DATASET2" + every { dataset3.name } returns "DATASET3" + every { dataset1.migrated } returns HasMigrated.NO + every { dataset2.migrated } returns HasMigrated.NO + every { dataset3.migrated } returns HasMigrated.NO + every { dataset1.datasetOrganization } returns DatasetOrganization.PS + every { dataset2.datasetOrganization } returns DatasetOrganization.PS + every { dataset3.datasetOrganization } returns DatasetOrganization.PO + val dataSetsList = mockk() + every { dataSetsList.items } returns mutableListOf(dataset1, dataset2, dataset3) + every { dataSetsList.totalRows } returns 3 + val mockedCall = mockk>() + val mockedResponse = mockk>() + every { mockedCall.execute() } returns mockedResponse + every { mockedResponse.body() } returns dataSetsList + + val zosmfApi = ApplicationManager.getApplication().service() as TestZosmfApiImpl + zosmfApi.testInstance = mockk() + val mockedApi = mockk() + every { zosmfApi.testInstance.getApi(DataAPI::class.java, mockedConnectionConfig) } returns mockedApi + every { mockedApi.listDataSets(authorizationToken = any() as String, dsLevel = any() as String, volser = any() as String, xIBMAttr = any() as XIBMAttr, xIBMMaxItems = any() as Int, start = any() as String) } returns mockedCall + every { mockedApi.listDataSets(authorizationToken = any() as String, dsLevel = any() as String, volser = any() as String, xIBMAttr = any() as XIBMAttr, xIBMMaxItems = any() as Int, start = any() as String).cancelByIndicator(progressMockk) } returns mockedCall + + val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + val componentManager = dataOpsManagerService.componentManager + + // needed for cleanupUnusedFile test + val mockedVirtualFile = mockk() + val mockedFileAttributes = mockk() + val mockedAttributesService = mockk() + val mockedMaskedRequester = mockk() + val requesters = mutableListOf(mockedMaskedRequester) + + afterEach { unmockkAll() } + + val datasetFileFetchProviderForTest = spyk(DatasetFileFetchProvider(dataOpsManagerService), recordPrivateCalls = true) + + should("dataset fetch provider fetchResponse test") { + val fetchResponseMethodRef = datasetFileFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + fetchResponseMethodRef.trySetAccessible() + every { mockedResponse.isSuccessful } returns true + + val datasetAttributes = (fetchResponseMethodRef.invoke(datasetFileFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) + + assertSoftly { + datasetAttributes shouldHaveSize dataSetsList.items.size + } + } + + should("cleanup unused file if connection config of the query is the same as for dataset file") { + var cleanupPerformed = false + val cleanupUnusedFileMethodRef = datasetFileFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } + cleanupUnusedFileMethodRef.trySetAccessible() + dataOpsManagerService.testInstance = object: TestDataOpsManagerImpl(componentManager) { + override fun getAttributesService( + attributesClass: Class, + vFileClass: Class + ): AttributesService { + return mockedAttributesService as AttributesService + } + } + + every { mockedAttributesService.getAttributes(mockedVirtualFile) } returns mockedFileAttributes + every { mockedAttributesService.clearAttributes(mockedVirtualFile) } just Runs + + every { mockedMaskedRequester.connectionConfig } returns mockedConnectionConfig + every { mockedMaskedRequester.queryVolser } returns "TESTVOL" + every { mockedFileAttributes.requesters } returns requesters + + every { mockedVirtualFile.delete(any() as DatasetFileFetchProvider) } answers { + cleanupPerformed = true + } + + cleanupUnusedFileMethodRef.invoke(datasetFileFetchProviderForTest, mockedVirtualFile, mockedQuery) + + assertSoftly { + cleanupPerformed shouldBe true + } + } + + should("cleanup unused file if connection config of the query is the same as for dataset file") { + var cleanupPerformed = false + val cleanupUnusedFileMethodRef = datasetFileFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } + cleanupUnusedFileMethodRef.trySetAccessible() + + every { mockedAttributesService.getAttributes(mockedVirtualFile) } returns mockedFileAttributes + every { mockedAttributesService.clearAttributes(mockedVirtualFile) } just Runs + + every { mockedMaskedRequester.connectionConfig } returns mockedConnectionConfig + every { mockedMaskedRequester.queryVolser } returns "ANOTHER" + every { mockedFileAttributes.requesters } returns requesters + every { mockedAttributesService.updateAttributes(mockedVirtualFile, any() as RemoteDatasetAttributes.() -> Unit) } answers { + cleanupPerformed = true + secondArg Unit>().invoke(mockedFileAttributes) + } + + every { mockedVirtualFile.delete(any() as DatasetFileFetchProvider) } answers { + cleanupPerformed = true + } + + cleanupUnusedFileMethodRef.invoke(datasetFileFetchProviderForTest, mockedVirtualFile, mockedQuery) + + assertSoftly { + cleanupPerformed shouldBe true + } + } + } + } +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/dataops/FetchTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/dataops/FetchTestSpec.kt deleted file mode 100644 index 594103587..000000000 --- a/src/test/kotlin/eu/ibagroup/formainframe/dataops/FetchTestSpec.kt +++ /dev/null @@ -1,23 +0,0 @@ -/* - * This program and the accompanying materials are made available under the terms of the - * Eclipse Public License v2.0 which accompanies this distribution, and is available at - * https://www.eclipse.org/legal/epl-v20.html - * - * SPDX-License-Identifier: EPL-2.0 - * - * Copyright IBA Group 2020 - */ - -package eu.ibagroup.formainframe.dataops - -import io.kotest.core.spec.style.ShouldSpec - -class FetchTestSpec : ShouldSpec({ - context("dataops module: fetch") { - // SpoolFileFetchProvider.reload - should("reload files cache") {} - should("reload files cache with failure") {} - // DatasetFileFetchProvider.cleanupUnusedFile - should("clean up files attributes") {} - } -}) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchHelperTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchHelperTestSpec.kt new file mode 100644 index 000000000..6150dea28 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchHelperTestSpec.kt @@ -0,0 +1,223 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.dataops + +import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.config.ws.JobsFilter +import eu.ibagroup.formainframe.dataops.attributes.JobsRequester +import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes +import eu.ibagroup.formainframe.dataops.fetch.JobFetchHelper +import eu.ibagroup.formainframe.dataops.log.JobLogFetcher +import eu.ibagroup.formainframe.dataops.log.JobProcessInfo +import eu.ibagroup.formainframe.utils.asMutableList +import io.kotest.assertions.assertSoftly +import io.kotest.core.spec.style.ShouldSpec +import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe +import io.mockk.* +import org.zowe.kotlinsdk.Job + +class JobFetchHelperTestSpec : ShouldSpec({ + context("dataops module: fetch") { + + context("JobFetchHelper common") { + + val mockedConnectionConfig = mockk() + + val mockedJobFilter = mockk() + every { mockedJobFilter.jobId } returns "TEST" + every { mockedJobFilter.owner } returns "ZOSMFAD" + every { mockedJobFilter.prefix } returns "TEST*" + + val jobQuery = mockk>() + every { jobQuery.connectionConfig } returns mockedConnectionConfig + every { jobQuery.request } returns mockedJobFilter + + val jobInfoForTest = Job("TSU06062", "TESTJOB", "TEST1", + "ZOSMFAD", Job.Status.OUTPUT, Job.JobType.JOB, null, "CC=00", + "test_url", "test_url", null, 1, "phase", + emptyList(), null, null, null, null, null, null + ) + val mockedJobAttributesForTest = mockk() + every { mockedJobAttributesForTest.jobInfo } returns jobInfoForTest + every { mockedJobAttributesForTest.url } returns jobInfoForTest.url + every { mockedConnectionConfig.url } returns jobInfoForTest.url + every { mockedJobAttributesForTest.requesters } returns JobsRequester(jobQuery.connectionConfig, jobQuery.request).asMutableList() + + var updatedJobAttributes: RemoteJobAttributes? + + beforeEach { updatedJobAttributes = null } + afterEach { unmockkAll() } + + context("JobFetchHelper create instance and perform tests") { + + should("run job fetch helper if fetch spool file returns empty logs") { + + val jobFetchHelperForTest = spyk(JobFetchHelper(jobQuery, mockedJobAttributesForTest), recordPrivateCalls = true) + val jobFetcherField = jobFetchHelperForTest::class.java.getDeclaredField("jobLogFetcher") + jobFetcherField.isAccessible = true + val jobFetcher = jobFetcherField.get(jobFetchHelperForTest) as JobLogFetcher + mockkObject(jobFetcher) + + val emptySpoolContent = emptyArray() + every { jobFetcher.fetchLogsBySpoolId(any() as JobProcessInfo, any() as Int) } returns emptySpoolContent + + jobFetchHelperForTest.start() + while(true) { + if (!jobFetchHelperForTest.isAlive) { + updatedJobAttributes = jobFetchHelperForTest.getUpdatedJobAttributes() + break + } + } + assertSoftly { + updatedJobAttributes?.jobInfo?.execStarted?.trim() shouldBe "" + updatedJobAttributes?.jobInfo?.execEnded?.trim() shouldBe "JCL NOT AVAILABLE" + } + } + + should("run job fetch helper if fetch spool file returns started/ended date and time") { + + val jobFetchHelperForTest = spyk(JobFetchHelper(jobQuery, mockedJobAttributesForTest), recordPrivateCalls = true) + val jobFetcherField = jobFetchHelperForTest::class.java.getDeclaredField("jobLogFetcher") + jobFetcherField.isAccessible = true + val jobFetcher = jobFetcherField.get(jobFetchHelperForTest) as JobLogFetcher + mockkObject(jobFetcher) + + val spoolContent = arrayOf( + "J E S 2 J O B L O G -- S Y S T E M S 0 W 1 -- N O D E S 0 W 1\n" + + "0\n" + + "19.45.23 TSU06062 ---- TUESDAY, 20 JUN 2023 ----\n" + + "19.45.23 TSU06062 HASP373 ZOSMFAD STARTED\n" + + "19.45.23 TSU06062 IEF125I ZOSMFAD - LOGGED ON - TIME=19.45.23\n" + + "20.09.28 TSU06062 BPXP018I THREAD 2037180000000000, IN PROCESS 83952040, ENDED 856\n" + + "856 WITHOUT BEING UNDUBBED WITH COMPLETION CODE 40222000\n" + + "856 , AND REASON CODE 00000000.\n" + + "20.09.29 TSU06062 IEF450I ZOSMFAD IZUFPROC IZUFPROC - ABEND=S222 U0000 REASON=00000000 858\n" + + "858 TIME=20.09.29\n" + + "20.09.29 TSU06062 HASP395 TESTJOB ENDED - ABEND=S222\n" + + "0------ JES2 JOB STATISTICS ------\n" + + "- 20 JUN 2023 JOB EXECUTION DATE\n" + + " - 3 CARDS READ\n" + + " - 148 SYSOUT PRINT RECORDS\n" + + " - 0 SYSOUT PUNCH RECORDS\n" + + " - 13 SYSOUT SPOOL KBYTES\n" + + " - 24.09 MINUTES EXECUTION TIME" + ) + every { jobFetcher.fetchLogsBySpoolId(any() as JobProcessInfo, any() as Int) } returns spoolContent + + jobFetchHelperForTest.start() + while(true) { + if (!jobFetchHelperForTest.isAlive) { + updatedJobAttributes = jobFetchHelperForTest.getUpdatedJobAttributes() + break + } + } + assertSoftly { + updatedJobAttributes?.jobInfo?.execStarted?.trim() shouldBe "20 JUN 2023 19.45.23" + updatedJobAttributes?.jobInfo?.execEnded?.trim() shouldBe "20 JUN 2023 20.09.29" + } + } + + should("run job fetch helper if fetch spool file returns started/ended date and time if ended date was not found") { + + val jobFetchHelperForTest = spyk(JobFetchHelper(jobQuery, mockedJobAttributesForTest), recordPrivateCalls = true) + val jobFetcherField = jobFetchHelperForTest::class.java.getDeclaredField("jobLogFetcher") + jobFetcherField.isAccessible = true + val jobFetcher = jobFetcherField.get(jobFetchHelperForTest) as JobLogFetcher + mockkObject(jobFetcher) + + val spoolContent = arrayOf( + "J E S 2 J O B L O G -- S Y S T E M S 0 W 1 -- N O D E S 0 W 1\n" + + "0\n" + + "19.45.23 TSU06061 ---- TUESDAY, 20 JUN 2023 ----\n" + + "19.45.23 TSU06062 HASP373 TESTJOB STARTED\n" + + "19.45.23 TSU06062 IEF125I ZOSMFAD - LOGGED ON - TIME=19.45.23\n" + + "20.09.28 TSU06062 BPXP018I THREAD 2037180000000000, IN PROCESS 83952040, ENDED 856\n" + + "856 WITHOUT BEING UNDUBBED WITH COMPLETION CODE 40222000\n" + + "856 , AND REASON CODE 00000000.\n" + + "20.09.29 TSU06062 IEF450I ZOSMFAD IZUFPROC IZUFPROC - ABEND=S222 U0000 REASON=00000000 858\n" + + "858 TIME=20.09.29\n" + + "20.09.29 TSU06062 HASP395 TESTJOB ENDED - ABEND=S222\n" + + "0------ JES2 JOB STATISTICS ------\n" + + "- 20 JUN 2023 JOB\n" + + " - 3 CARDS READ\n" + + " - 148 SYSOUT PRINT RECORDS\n" + + " - 0 SYSOUT PUNCH RECORDS\n" + + " - 13 SYSOUT SPOOL KBYTES\n" + + " - 24.09 MINUTES EXECUTION TIME" + ) + every { jobFetcher.fetchLogsBySpoolId(any() as JobProcessInfo, any() as Int) } returns spoolContent + + jobFetchHelperForTest.start() + while(true) { + if (!jobFetchHelperForTest.isAlive) { + updatedJobAttributes = jobFetchHelperForTest.getUpdatedJobAttributes() + break + } + } + assertSoftly { + updatedJobAttributes?.jobInfo?.execStarted?.trim() shouldBe "20 JUN 2023 19.45.23" + updatedJobAttributes?.jobInfo?.execEnded?.trim() shouldBe "20 JUN 2023 20.09.29" + } + } + + should("run job fetch helper if fetch spool file returns no date and time in it") { + + val jobFetchHelperForTest = spyk(JobFetchHelper(jobQuery, mockedJobAttributesForTest), recordPrivateCalls = true) + val jobFetcherField = jobFetchHelperForTest::class.java.getDeclaredField("jobLogFetcher") + jobFetcherField.isAccessible = true + val jobFetcher = jobFetcherField.get(jobFetchHelperForTest) as JobLogFetcher + mockkObject(jobFetcher) + + val spoolContent = arrayOf("J E S 2 J O B L O G -- S Y S T E M S 0 W 1 -- N O D E S 0 W 1") + every { jobFetcher.fetchLogsBySpoolId(any() as JobProcessInfo, any() as Int) } returns spoolContent + + jobFetchHelperForTest.start() + while(true) { + if (!jobFetchHelperForTest.isAlive) { + updatedJobAttributes = jobFetchHelperForTest.getUpdatedJobAttributes() + break + } + } + assertSoftly { + updatedJobAttributes?.jobInfo?.execStarted?.trim() shouldBe "" + updatedJobAttributes?.jobInfo?.execEnded?.trim() shouldBe "" + } + } + + should("run job fetch helper if fetch spool file throws an exception") { + + val exception: Throwable? + val jobFetchHelperForTest = spyk(JobFetchHelper(jobQuery, mockedJobAttributesForTest), recordPrivateCalls = true) + val jobFetcherField = jobFetchHelperForTest::class.java.getDeclaredField("jobLogFetcher") + jobFetcherField.isAccessible = true + val jobFetcher = jobFetcherField.get(jobFetchHelperForTest) as JobLogFetcher + mockkObject(jobFetcher) + + every { jobFetcher.fetchLogsBySpoolId(any() as JobProcessInfo, any() as Int) } answers { + throw IllegalArgumentException("TEST FAILED") + } + + jobFetchHelperForTest.start() + while(true) { + if (!jobFetchHelperForTest.isAlive) { + exception = jobFetchHelperForTest.getException() + break + } + } + assertSoftly { + exception shouldNotBe null + } + } + } + } + } +}) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchProviderTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchProviderTestSpec.kt new file mode 100644 index 000000000..1b8eb0a02 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchProviderTestSpec.kt @@ -0,0 +1,252 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.dataops + +import com.intellij.openapi.application.ApplicationManager +import com.intellij.openapi.progress.ProgressIndicator +import com.intellij.openapi.vfs.VirtualFile +import com.intellij.testFramework.LightProjectDescriptor +import com.intellij.testFramework.fixtures.IdeaTestFixtureFactory +import com.intellij.testFramework.fixtures.impl.LightTempDirTestFixtureImpl +import eu.ibagroup.formainframe.api.ZosmfApi +import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.config.connect.authToken +import eu.ibagroup.formainframe.config.ws.JobsFilter +import eu.ibagroup.formainframe.dataops.attributes.* +import eu.ibagroup.formainframe.dataops.fetch.JobFetchProvider +import eu.ibagroup.formainframe.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.testServiceImpl.TestZosmfApiImpl +import eu.ibagroup.formainframe.utils.cancelByIndicator +import eu.ibagroup.formainframe.utils.service +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.assertions.throwables.shouldThrow +import io.kotest.core.spec.style.ShouldSpec +import io.kotest.matchers.collections.shouldHaveSize +import io.kotest.matchers.shouldBe +import io.kotest.matchers.throwable.shouldHaveMessage +import io.mockk.* +import org.zowe.kotlinsdk.ExecData +import org.zowe.kotlinsdk.JESApi +import org.zowe.kotlinsdk.Job +import retrofit2.Call +import retrofit2.Response +import java.lang.reflect.InvocationTargetException + +class JobFetchProviderTestSpec : ShouldSpec({ + + beforeSpec { + // FIXTURE SETUP TO HAVE ACCESS TO APPLICATION INSTANCE + val factory = IdeaTestFixtureFactory.getFixtureFactory() + val projectDescriptor = LightProjectDescriptor.EMPTY_PROJECT_DESCRIPTOR + val fixtureBuilder = factory.createLightFixtureBuilder(projectDescriptor, "for-mainframe") + val fixture = fixtureBuilder.fixture + val myFixture = IdeaTestFixtureFactory.getFixtureFactory().createCodeInsightFixture( + fixture, + LightTempDirTestFixtureImpl(true) + ) + myFixture.setUp() + } + + afterSpec { + clearAllMocks() + } + + context("dataops module: fetch") { + + context("JobFetchProvider common") { + val mockedConnectionConfig = mockk() + every { mockedConnectionConfig.authToken } returns "auth_token" + every { mockedConnectionConfig.url } returns "test_url" + val mockedQuery = mockk>() + val mockedRequest = mockk() + every { mockedRequest.owner } returns "TESTOWNER" + every { mockedRequest.prefix } returns "TESTPREFIX" + every { mockedRequest.userCorrelatorFilter } returns "TESTFILTER" + val progressMockk = mockk() + every { mockedQuery.request } returns mockedRequest + every { mockedQuery.connectionConfig } returns mockedConnectionConfig + + val job1 = mockk() + val job2 = mockk() + every { job1.execStarted } returns null + every { job1.execEnded } returns null + every { job1.execSubmitted } returns null + every { job2.execStarted } returns null + every { job2.execEnded } returns null + every { job2.execSubmitted } returns null + every { job1.jobId } returns "TSUTEST1" + every { job2.jobId } returns "TSUTEST2" + every { job1.jobName } returns "TESTJOB1" + every {job2.jobName } returns "TESTJOB2" + + val jobs = mutableListOf(job1, job2) + val mockedCall = mockk>>() + val mockedResponse = mockk>>() + every { mockedCall.execute() } returns mockedResponse + every { mockedResponse.body() } returns jobs + + val zosmfApi = ApplicationManager.getApplication().service() as TestZosmfApiImpl + zosmfApi.testInstance = mockk() + val mockedApi = mockk() + every { zosmfApi.testInstance.getApi(JESApi::class.java, mockedConnectionConfig) } returns mockedApi + every { mockedApi.getFilteredJobs(basicCredentials = any() as String, owner = any() as String, prefix = any() as String, userCorrelator = any() as String, execData = any() as ExecData) } returns mockedCall + every { mockedApi.getFilteredJobs(basicCredentials = any() as String, owner = any() as String, prefix = any() as String, userCorrelator = any() as String, execData = any() as ExecData).cancelByIndicator(progressMockk) } returns mockedCall + every { mockedApi.getFilteredJobs(basicCredentials = any() as String, jobId = any() as String, execData = any() as ExecData) } returns mockedCall + every { mockedApi.getFilteredJobs(basicCredentials = any() as String, jobId = any() as String, execData = any() as ExecData).cancelByIndicator(progressMockk) } returns mockedCall + + val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + val componentManager = dataOpsManagerService.componentManager + + // needed for cleanupUnusedFile test + val mockedVirtualFile = mockk() + val mockedFileAttributes = mockk() + val mockedAttributesService = mockk() + val mockedJobsRequester = mockk() + val requesters = mutableListOf(mockedJobsRequester) + + afterEach { unmockkAll() } + + val jobFetchProviderForTest = spyk(JobFetchProvider(dataOpsManagerService), recordPrivateCalls = true) + + should("fetchResponse get job attributes if job ID is not null and response is successful and exec dates/times are null") { + + val fetchResponseMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + fetchResponseMethodRef.trySetAccessible() + every { mockedRequest.jobId } returns "TSUTEST" + every { mockedResponse.isSuccessful } returns true + + val jobAttributes = (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) + + assertSoftly { + jobAttributes shouldHaveSize jobs.size + } + } + + should("fetchResponse get job attributes if job ID is null and response is successful") { + + val fetchResponseMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + fetchResponseMethodRef.trySetAccessible() + every { mockedRequest.jobId } returns "" + every { mockedResponse.isSuccessful } returns true + + val jobAttributes = (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) + + assertSoftly { + jobAttributes shouldHaveSize jobs.size + } + } + + should("fetchResponse get job attributes if exec dates/times are not null") { + + val fetchResponseMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + fetchResponseMethodRef.trySetAccessible() + every { mockedRequest.jobId } returns "TSUTEST" + every { mockedResponse.isSuccessful } returns true + + every { job1.execStarted } returns "" + every { job1.execEnded } returns "" + every { job1.execSubmitted } returns "" + every { job2.execStarted } returns "" + every { job2.execEnded } returns "" + every { job2.execSubmitted } returns "" + + val jobAttributes = (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) + + assertSoftly { + jobAttributes shouldHaveSize jobs.size + } + } + + should("fetchResponse get job attributes if response does not return any job") { + + val fetchResponseMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + fetchResponseMethodRef.trySetAccessible() + every { mockedRequest.jobId } returns "TSUTEST" + every { mockedResponse.isSuccessful } returns true + every { mockedResponse.body() } returns emptyList() + + val jobAttributes = (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) + + assertSoftly { + jobAttributes shouldHaveSize 0 + } + } + + should("fetchResponse get job attributes if response was not successful") { + val fetchResponseMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + fetchResponseMethodRef.trySetAccessible() + every { mockedRequest.jobId } returns "TSUTEST" + every { mockedResponse.isSuccessful } returns false + every { mockedResponse.code() } returns 404 + every { mockedResponse.body() } returns emptyList() + + val exception = shouldThrow { + fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) + } + + assertSoftly { + exception.cause!! shouldHaveMessage "Cannot retrieve Job files list\nCode: 404" + } + } + + should("cleanup unused file if connection config of the query is the same as for job file") { + var cleanupPerformed = false + val cleanupUnusedFileMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } + cleanupUnusedFileMethodRef.trySetAccessible() + dataOpsManagerService.testInstance = object: TestDataOpsManagerImpl(componentManager) { + override fun getAttributesService( + attributesClass: Class, + vFileClass: Class + ): AttributesService { + return mockedAttributesService as AttributesService + } + } + + every { mockedAttributesService.getAttributes(mockedVirtualFile) } returns mockedFileAttributes + every { mockedAttributesService.clearAttributes(mockedVirtualFile) } just Runs + + every { mockedJobsRequester.connectionConfig } returns mockedConnectionConfig + every { mockedFileAttributes.requesters } returns requesters + + every { mockedVirtualFile.delete(any() as JobFetchProvider) } answers { + cleanupPerformed = true + } + + cleanupUnusedFileMethodRef.invoke(jobFetchProviderForTest, mockedVirtualFile, mockedQuery) + + assertSoftly { + cleanupPerformed shouldBe true + } + } + + should("cleanup file if connections are not the same") { + var cleanupPerformed = false + val cleanupUnusedFileMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } + cleanupUnusedFileMethodRef.trySetAccessible() + + every { mockedJobsRequester.connectionConfig } returns mockk() + every { mockedFileAttributes.requesters } returns requesters + every { mockedAttributesService.updateAttributes(mockedVirtualFile, any() as RemoteJobAttributes.() -> Unit) } answers { + cleanupPerformed = true + secondArg Unit>().invoke(mockedFileAttributes) + } + + cleanupUnusedFileMethodRef.invoke(jobFetchProviderForTest, mockedVirtualFile, mockedQuery) + + assertSoftly { + cleanupPerformed shouldBe true + } + } + + } + } +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt index 7805539c6..7bf5d268c 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt @@ -71,7 +71,7 @@ class UssSortActionHolderTestSpec : ShouldSpec({ } val mockedActionEvent = mockk() every { mockedActionEvent.presentation } returns mockk() - every { mockedActionEvent.presentation.putClientProperty(any(), any()) } just Runs + every { mockedActionEvent.presentation.putClientProperty(any() as String, any() as Boolean) } just Runs // isFromContextMenu is marked as deprecated every { mockedActionEvent.isFromContextMenu } returns false every { mockedActionEvent.presentation.isEnabledAndVisible = true } just Runs From c85efda1cd2acfc2bdbdbbd0d2a7ab4320119835 Mon Sep 17 00:00:00 2001 From: Dzianis Lisiankou Date: Tue, 18 Jul 2023 15:30:06 +0300 Subject: [PATCH 07/34] IJMP-1115: added unit tests --- .../dataops/OperationsTestSpec.kt | 81 ++++++++++++++++--- 1 file changed, 72 insertions(+), 9 deletions(-) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/dataops/OperationsTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/dataops/OperationsTestSpec.kt index f43da9a2f..75d271e32 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/dataops/OperationsTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/dataops/OperationsTestSpec.kt @@ -25,16 +25,16 @@ import eu.ibagroup.formainframe.dataops.attributes.* import eu.ibagroup.formainframe.dataops.content.synchronizer.ContentSynchronizer import eu.ibagroup.formainframe.dataops.content.synchronizer.DocumentedSyncProvider import eu.ibagroup.formainframe.dataops.content.synchronizer.LF_LINE_SEPARATOR +import eu.ibagroup.formainframe.dataops.exceptions.CallException import eu.ibagroup.formainframe.dataops.operations.DeleteOperation +import eu.ibagroup.formainframe.dataops.operations.ZOSInfoOperation +import eu.ibagroup.formainframe.dataops.operations.ZOSInfoOperationRunner import eu.ibagroup.formainframe.dataops.operations.mover.CrossSystemMemberOrUssFileOrSequentialToUssDirMover import eu.ibagroup.formainframe.dataops.operations.mover.MoveCopyOperation import eu.ibagroup.formainframe.dataops.operations.mover.RemoteToLocalFileMover import eu.ibagroup.formainframe.testServiceImpl.TestDataOpsManagerImpl import eu.ibagroup.formainframe.testServiceImpl.TestZosmfApiImpl -import eu.ibagroup.formainframe.utils.castOrNull -import eu.ibagroup.formainframe.utils.changeFileEncodingTo -import eu.ibagroup.formainframe.utils.service -import eu.ibagroup.formainframe.utils.setUssFileTag +import eu.ibagroup.formainframe.utils.* import eu.ibagroup.formainframe.vfs.MFVirtualFile import io.kotest.assertions.assertSoftly import io.kotest.assertions.throwables.shouldThrow @@ -42,15 +42,14 @@ import io.kotest.core.spec.style.ShouldSpec import io.kotest.matchers.shouldBe import io.kotest.matchers.string.shouldContain import io.mockk.* -import org.zowe.kotlinsdk.DataAPI -import org.zowe.kotlinsdk.FilePath -import org.zowe.kotlinsdk.XIBMDataType +import org.zowe.kotlinsdk.* import org.zowe.kotlinsdk.annotations.ZVersion import retrofit2.Call import retrofit2.Response import java.io.File import java.nio.charset.Charset import java.nio.charset.StandardCharsets +import kotlin.reflect.KFunction inline fun mockkRequesters( attributes: MFRemoteFileAttributes<*, *>, @@ -100,8 +99,72 @@ class OperationsTestSpec : ShouldSpec({ clearAllMocks() } context("dataops module: operations/ZOSIntoOperationRunner") { - context("run") { - should("perform Info operation") {} + val zosInfoOperationRunner = ZOSInfoOperationRunner() + + val zosmfApi = ApplicationManager.getApplication().service() as TestZosmfApiImpl + zosmfApi.testInstance = mockk() + + val progressIndicatorMockk = mockk() + val responseMockk = mockk>() + every { + zosmfApi.testInstance.getApi(InfoAPI::class.java, any()) + .getSystemInfo() + .cancelByIndicator(progressIndicatorMockk) + .execute() + } returns responseMockk + + val operationMockk = mockk() + every { operationMockk.connectionConfig } returns mockk() + + val callExceptionRef: (Response<*>, String) -> CallException = ::CallException + beforeEach { + mockkStatic(callExceptionRef as KFunction<*>) + } + afterEach { + unmockkAll() + } + + // ZOSIntoOperationRunner.run + should("perform Info operation with successful response") { + val expected = InfoResponse(zosVersion = "2.3", zosmfHostname = "host", zosmfPort = "port") + + every { responseMockk.isSuccessful } returns true + every { responseMockk.body() } returns expected + + val actual = zosInfoOperationRunner.run(operationMockk, progressIndicatorMockk) + + assertSoftly { actual shouldBe expected } + } + should("perform Info operation with unsuccessful response") { + val expected = CallException(500, "An internal error has occurred") + + every { responseMockk.isSuccessful } returns false + every { CallException(responseMockk, any()) } returns expected + + var actual: Throwable? = null + runCatching { + zosInfoOperationRunner.run(operationMockk, progressIndicatorMockk) + }.onFailure { + actual = it + } + + assertSoftly { actual shouldBe expected } + } + should("perform Info operation when response body is null") { + val expected = CallException(500, "Cannot parse z/OSMF info request body") + + every { responseMockk.isSuccessful } returns true + every { responseMockk.body() } returns null + every { CallException(responseMockk, any()) } returns expected + + var actual: Throwable? = null + runCatching { + zosInfoOperationRunner.run(operationMockk, progressIndicatorMockk) + }.onFailure { + actual = it + } + + assertSoftly { actual shouldBe expected } } } context("dataops module: operations/RenameOperationRunner") { From 15aae2ee4214a926501682703e8ecad215b1e6d3 Mon Sep 17 00:00:00 2001 From: Nikita Date: Mon, 31 Jul 2023 16:47:06 +0300 Subject: [PATCH 08/34] IJMP-1154: removed redundant code from another branch --- .../connect/ui/zosmf/ConnectionDialog.kt | 4 +--- .../ui/build/jobs/JobBuildTreeView.kt | 23 ++----------------- 2 files changed, 3 insertions(+), 24 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt index 2b3ef380d..0496816ec 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt @@ -96,6 +96,7 @@ class ConnectionDialog( runCatching { service().performOperation(InfoOperation(newTestedConnConfig), it) }.onSuccess { + state.owner = whoAmI(newTestedConnConfig) ?: "" val systemInfo = service().performOperation(ZOSInfoOperation(newTestedConnConfig)) state.zVersion = when (systemInfo.zosVersion) { "04.25.00" -> ZVersion.ZOS_2_2 @@ -142,9 +143,6 @@ class ConnectionDialog( } addAnyway } else { - runTask(title = "Retrieving user information", project = project) { - state.owner = whoAmI(newTestedConnConfig) ?: "" - } true } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/jobs/JobBuildTreeView.kt b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/jobs/JobBuildTreeView.kt index 5e3126f52..25d02a8d7 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/jobs/JobBuildTreeView.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/jobs/JobBuildTreeView.kt @@ -41,9 +41,6 @@ import javax.swing.tree.DefaultMutableTreeNode val JOBS_LOG_VIEW = DataKey.create("jobsLogView") const val JOBS_LOG_NOTIFICATION_GROUP_ID = "eu.ibagroup.formainframe.explorer.ExplorerNotificationGroup" -const val SUCCESSFUL_JOB_COMPLETION_CODE = 0 -const val SUCCESSFUL_JOB_COMPLETION_CODE_WITH_WARNING = 4 - /** * Console with BuildTree for display job execution process and results. * @param jobLogInfo job process information necessary to get log and status. @@ -123,22 +120,8 @@ class JobBuildTreeView( .getCachedJobStatus() ?.returnedCode ?.uppercase() - var codeWithWarning = false - val result = if (rc == null || rc.contains(Regex("ERR|ABEND|CANCEL|FAIL"))) FailureResultImpl() - else if (rc.contains("CC")) { // result code can be in format "CC nnnn" - val completionCode = rc.split(" ")[1].toInt() - when (completionCode) { - SUCCESSFUL_JOB_COMPLETION_CODE -> SuccessResultImpl() - - SUCCESSFUL_JOB_COMPLETION_CODE_WITH_WARNING -> { - codeWithWarning = true - SuccessResultImpl() - } - - else -> FailureResultImpl() - } - } else SuccessResultImpl() - + val result = if (rc == null || rc.contains(Regex("ERR|ABEND|CANCEL"))) FailureResultImpl() + else SuccessResultImpl() jobLogger.logFetcher.getCachedLog() .forEach { treeConsoleView.onEvent(buildId, FinishEventImpl(it.key.id, buildId, Date().time, it.key.ddName, result)) @@ -148,8 +131,6 @@ class JobBuildTreeView( val buildExecutionNode = (buildNode as DefaultMutableTreeNode).userObject as ExecutionNode if (result is FailureResultImpl) { buildExecutionNode.setIconProvider { AllIcons.General.BalloonError } - } else if (codeWithWarning) { - buildExecutionNode.setIconProvider { AllIcons.General.BalloonWarning } } else { buildExecutionNode.setIconProvider { AllIcons.General.InspectionsOK } } From 451f3528545d596b820a5d7b043f76f22a9b5836 Mon Sep 17 00:00:00 2001 From: Dzianis Lisiankou Date: Tue, 1 Aug 2023 17:27:00 +0300 Subject: [PATCH 09/34] IJMP-722: added notification for allocation units that z/OSMF does not support --- build.gradle.kts | 2 +- .../explorer/actions/AllocateDatasetAction.kt | 29 +++++++++---------- 2 files changed, 15 insertions(+), 16 deletions(-) diff --git a/build.gradle.kts b/build.gradle.kts index e93b22864..dbd94b11d 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -59,7 +59,7 @@ dependencies { implementation("org.jetbrains.kotlin:kotlin-reflect:1.6.20") implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.4") implementation("org.jgrapht:jgrapht-core:1.5.1") - implementation("org.zowe.sdk:zowe-kotlin-sdk:0.4.0") + implementation("org.zowe.sdk:zowe-kotlin-sdk:0.5.0-rc.1") implementation("com.segment.analytics.java:analytics:3.3.1") implementation("com.ibm.mq:com.ibm.mq.allclient:9.3.0.0") testImplementation("io.mockk:mockk:1.13.5") diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetAction.kt index f5996c55e..0e352157d 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetAction.kt @@ -237,7 +237,7 @@ class AllocateLikeAction : AnAction() { val datasetInfo = (selected[0].attributes as RemoteDatasetAttributes).datasetInfo val initialState = DatasetAllocationParams().apply { allocationParameters.datasetOrganization = datasetInfo.datasetOrganization ?: DatasetOrganization.PS - allocationParameters.allocationUnit = spaceUnitsToAllocationUnits(datasetInfo.spaceUnits) ?: AllocationUnit.TRK + allocationParameters.allocationUnit = datasetInfo.spaceUnits?.let { spaceUnitsToAllocationUnits(it) } ?: AllocationUnit.TRK allocationParameters.blockSize = datasetInfo.blockSize allocationParameters.recordLength = datasetInfo.recordLength allocationParameters.recordFormat = datasetInfo.recordFormat ?: RecordFormat.FB @@ -257,24 +257,23 @@ class AllocateLikeAction : AnAction() { } /** - * Transforms info about space units of existing datasets to format that is suitable for allocation operation + * Transforms info about space units of existing datasets to format that is suitable for allocation operation. + * For allocation, z/OSMF provides only TRACKS and CYLINDERS, other units will be changed to TRACKS * @param spaceUnits space units info of existing dataset * @return processed info about space units */ - private fun spaceUnitsToAllocationUnits(spaceUnits: SpaceUnits?): AllocationUnit? { - if (spaceUnits == SpaceUnits.TRACKS) { - return AllocationUnit.TRK - } - if (spaceUnits == SpaceUnits.CYLINDERS) { - return AllocationUnit.CYL - } - if (spaceUnits == SpaceUnits.BLOCKS) { - Messages.showWarningDialog( - "Allocation unit BLK is not supported. It will be changed to TRK.", - "Allocation Unit Will Be Changed" - ) + private fun spaceUnitsToAllocationUnits(spaceUnits: SpaceUnits): AllocationUnit { + return when (spaceUnits) { + SpaceUnits.TRACKS -> AllocationUnit.TRK + SpaceUnits.CYLINDERS -> AllocationUnit.CYL + else -> { + Messages.showWarningDialog( + "Allocation unit ${spaceUnits.name} is not supported by z/OSMF. It will be changed to TRACKS.", + "Allocation Unit Will Be Changed" + ) + AllocationUnit.TRK + } } - return null } /** From 6195cbaefdf94df9ce30149a75091182b165f350 Mon Sep 17 00:00:00 2001 From: Arseni Tsikhamirau Date: Tue, 8 Aug 2023 16:58:07 +0300 Subject: [PATCH 10/34] IJMP-822-Bulk-jobs-purge 1) implemented --- .../explorer/actions/PurgeJobAction.kt | 270 +++++++++++++----- 1 file changed, 205 insertions(+), 65 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt index f353ce6b1..c1f86f6b7 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt @@ -3,8 +3,10 @@ package eu.ibagroup.formainframe.explorer.actions import com.intellij.notification.NotificationType import com.intellij.openapi.actionSystem.AnAction import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.application.runInEdt import com.intellij.openapi.components.service import com.intellij.openapi.progress.runBackgroundableTask +import com.intellij.openapi.progress.runModalTask import eu.ibagroup.formainframe.api.api import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.connect.authToken @@ -18,7 +20,8 @@ import eu.ibagroup.formainframe.ui.build.jobs.JobBuildTreeView import eu.ibagroup.formainframe.utils.service // TODO: remove in v1.*.*-223 and greater import org.zowe.kotlinsdk.ExecData import org.zowe.kotlinsdk.JESApi -import org.zowe.kotlinsdk.Job +import retrofit2.Response +import java.util.concurrent.ConcurrentHashMap /** An action to purge a job */ class PurgeJobAction : AnAction() { @@ -36,68 +39,37 @@ class PurgeJobAction : AnAction() { e.presentation.isEnabledAndVisible = false return } - var jobStatus: Job? = null - var connectionConfig: ConnectionConfig? = null if (view is JesExplorerView) { - val node = view.mySelectedNodesData.getOrNull(0)?.node - if (node is ExplorerTreeNode<*, *>) { - val virtualFile = node.virtualFile - if (virtualFile != null) { - val dataOpsManager = node.explorer.componentManager.service() - val attributes: RemoteJobAttributes = - dataOpsManager.tryToGetAttributes(virtualFile)?.clone() as RemoteJobAttributes - jobStatus = attributes.jobInfo - connectionConfig = attributes.requesters[0].connectionConfig - } - } + val jobNodesToPurge = view.mySelectedNodesData + val filtersToNodesMap = selectJobNodesByWSFilters(jobNodesToPurge) + runMassivePurgeAndRefreshByFilter(filtersToNodesMap, jobNodesToPurge, e, view) + } else if (view is JobBuildTreeView) { - jobStatus = view.getJobLogger().logFetcher.getCachedJobStatus() - connectionConfig = view.getConnectionConfig() - } - val dataOpsManager = service() - if (jobStatus != null && connectionConfig != null) { - runBackgroundableTask( - title = "Purging ${jobStatus.jobName}: ${jobStatus.jobId}", - project = e.project, - cancellable = true - ) { - runCatching { - dataOpsManager.performOperation( - operation = PurgeJobOperation( - request = BasicPurgeJobParams(jobStatus.jobName, jobStatus.jobId), - connectionConfig = connectionConfig - ), - progressIndicator = it - ) - }.onFailure { - if (view is JesExplorerView) { - view.explorer.showNotification( - "Error purging ${jobStatus.jobName}: ${jobStatus.jobId}", - "${it.message}", - NotificationType.ERROR, - e.project + val jobStatus = view.getJobLogger().logFetcher.getCachedJobStatus() + val connectionConfig: ConnectionConfig = view.getConnectionConfig() + val dataOpsManager = service() + if (jobStatus != null) { + runBackgroundableTask( + title = "Purging ${jobStatus.jobName}: ${jobStatus.jobId}", + project = e.project, + cancellable = true + ) { + runCatching { + dataOpsManager.performOperation( + operation = PurgeJobOperation( + request = BasicPurgeJobParams(jobStatus.jobName, jobStatus.jobId), + connectionConfig = connectionConfig + ), + progressIndicator = it ) - } else if (view is JobBuildTreeView) { + }.onFailure { view.showNotification( "Error purging ${jobStatus.jobName}: ${jobStatus.jobId}", "${it.message}", e.project, NotificationType.ERROR ) - } - }.onSuccess { - if (view is JesExplorerView) { - view.explorer.showNotification( - "${jobStatus.jobName}: ${jobStatus.jobId} has been purged", - "$it", - NotificationType.INFORMATION, - e.project - ) - val jobFilterNode = view.mySelectedNodesData.getOrNull(0)?.node?.parent - if (jobFilterNode is FetchNode) { - waitJobReleasedAndRefresh(jobFilterNode, jobStatus) - } - } else if (view is JobBuildTreeView) { + }.onSuccess { view.showNotification( "${jobStatus.jobName}: ${jobStatus.jobId} has been purged", "$it", @@ -111,14 +83,18 @@ class PurgeJobAction : AnAction() { } /** - * Function which checks if job is already purged from JES2. If not, repeats the scenario, cleans cache otherwise - * @param jobParentNode - parent node for particular job - * @param jobInfo - job info for particular job + * After performing the massive purge of the jobs scope (several filters) z/OSMF returns immediate success in the response, + * but actual purge was not performed yet on the mainframe which causes the problems during fetching the new job list during refresh. + * The purpose of this function is to perform the refresh on each filter only when actual purge was performed + * Note: This function performs refresh only on those filters from which purge action is requested + * @param jobsByFilterWaitingPurgeMap - concurrent map of the filter-to-jobs relationship waiting to purge * @return Void */ - private fun waitJobReleasedAndRefresh(jobParentNode: ExplorerTreeNode, jobInfo: Job) { - if (jobParentNode is JesFilterNode) { - val query = jobParentNode.query + private fun waitJobsReleasedAndRefresh(jobsByFilterWaitingPurgeMap: ConcurrentHashMap>>) { + val foundJobsWaitingInPurgeQueue: MutableList = mutableListOf() + val filtersWithRefreshErrors: MutableMap> = mutableMapOf() + jobsByFilterWaitingPurgeMap.keys.forEach { filterNode -> + val query = filterNode.query if (query != null) { val response = api(query.connectionConfig).getFilteredJobs( basicCredentials = query.connectionConfig.authToken, @@ -129,12 +105,174 @@ class PurgeJobAction : AnAction() { ).execute() val result = response.body() if (response.isSuccessful && result != null) { - val job = result.find { it.jobId == jobInfo.jobId } - if (job != null) waitJobReleasedAndRefresh(jobParentNode, jobInfo) else jobParentNode.cleanCache() + jobsByFilterWaitingPurgeMap[filterNode]?.forEach { data -> + val nodeToFind = data.node as JobNode + val jobAttributes = data.attributes as RemoteJobAttributes + val jobInfo = jobAttributes.jobInfo + val foundJob = result.find { it.jobId == jobInfo.jobId } + if (foundJob != null) foundJobsWaitingInPurgeQueue.add(nodeToFind) + } + if (foundJobsWaitingInPurgeQueue.isNotEmpty()) { + foundJobsWaitingInPurgeQueue.clear() + val filterRefreshSize = jobsByFilterWaitingPurgeMap[filterNode]!!.size + runRefreshByFilter(filterNode, if (filterRefreshSize == 1) (filterRefreshSize.toLong() * 1000) else (filterRefreshSize / 2).toLong() * 1000) + } else { + filterNode.cleanCache() + } + } else { + filtersWithRefreshErrors[filterNode] = response + } + } + } + if (filtersWithRefreshErrors.isNotEmpty()) { + var errorFilterNames = "" + for (entry in filtersWithRefreshErrors) { + errorFilterNames += entry.key.unit.name + "\n" + } + throw RuntimeException("Refresh error. Failed filters are: $errorFilterNames") + } + } + + /** + * Function triggers refresh by filter + * @param filter - jes filter + * @param timeWait - time to wait before refresh + */ + private fun runRefreshByFilter(filter: JesFilterNode, timeWait: Long) { + runInEdt { + Thread.sleep(timeWait) + filter.cleanCache() + } + } + + /** + * Function needs to determine when to show "purge job/jobs" action. Currently, purge is only possible within 1 Working Set + * @param jobs - the list of selected job nodes in JES view + * @return true if jobs were selected within 1 WS, false otherwise + */ + private fun isSelectedJobNodesFromSameWS(jobs: List>): Boolean { + val firstJob = jobs[0].node as JobNode + val firstConnection = firstJob.query?.connectionConfig + + // Could be different connections + val diffConnectionJobNode = jobs.filter { + (it.node is JobNode && it.node.unit.connectionConfig != firstConnection) + } + if (diffConnectionJobNode.isNotEmpty()) return false + + // Could be the same connections but different Working Sets + val firstJobWS = if (firstJob.parent is JesFilterNode) firstJob.parent.parent as? JesWsNode else null + if (firstJobWS != null) { + val diffWS = jobs.filter { + val jobWS = it.node.parent?.parent as JesWsNode + firstJobWS.unit.name != jobWS.unit.name + } + if (diffWS.isNotEmpty()) return false + } + return true + } + + /** + * Function builds the filter-to-jobs dependency and puts it to the concurrent map + * @param nodes - list of the selected job nodes in JES view + * @return concurrent map of the filter-to-jobs dependency + */ + private fun selectJobNodesByWSFilters(nodes: List>): ConcurrentHashMap>> { + val jobFilters: MutableList = mutableListOf() + val filtersToJobNodesMap: ConcurrentHashMap>> = ConcurrentHashMap() + nodes.forEach { + val jobNode = it.node as JobNode + val filterNode = findFilter(jobNode) + if (filterNode != null && !jobFilters.contains(filterNode)) { + jobFilters.add(filterNode) + filtersToJobNodesMap[filterNode] = findJobsByFilter(filterNode, nodes) + } + } + return filtersToJobNodesMap + } + + /** + * Function finds the corresponding filter or null if filter was not found + * @param node - job node to find its filter + * @return the JesFilterNode object corresponding to this job node or null if it was not found + */ + private fun findFilter(node: JobNode): JesFilterNode? { + return node.parent as? JesFilterNode + } + + /** + * Function finds all selected job nodes which correspond to the specified filter + * @param filter - filter to find the corresponding job nodes + * @param nodes - the list of all selected job nodes in JES view + * @return the list of job nodes which correspond to the specified filter + */ + private fun findJobsByFilter( + filter: JesFilterNode, + nodes: List> + ): List> { + val jobsByFilter: MutableList> = mutableListOf() + nodes.forEach { + val jobNode = it.node as JobNode + val jobFilter = findFilter(jobNode) + if (filter == jobFilter) { + jobsByFilter.add(it) + } + } + return jobsByFilter + } + + /** + * The main function which triggers the jobs purge on the mainframe and refreshes each affected filter in the JES view tree + * @param jobsByFilterMap - the concurrent map of the filter-to-jobs dependency + * @param nodes - the list of all selected job nodes to purge + * @param e - purge action event + * @param view - an instance of current JES view + */ + private fun runMassivePurgeAndRefreshByFilter( + jobsByFilterMap: ConcurrentHashMap>>, + nodes: List>, + e: AnActionEvent, + view: JesExplorerView + ) { + val dataOpsManager = service() + nodes.forEach { nodeData -> + val jobAttributes = nodeData.attributes as RemoteJobAttributes + val jobStatus = jobAttributes.jobInfo + val connectionConfig = jobAttributes.requesters[0].connectionConfig + runModalTask( + title = "Purging ${jobStatus.jobName}: ${jobStatus.jobId}", + project = e.project, + cancellable = true + ) { + runCatching { + dataOpsManager.performOperation( + operation = PurgeJobOperation( + request = BasicPurgeJobParams(jobStatus.jobName, jobStatus.jobId), + connectionConfig = connectionConfig + ), + progressIndicator = it + ) + }.onFailure { throwable -> + (jobsByFilterMap.values.find { it.contains(nodeData) } as MutableList).remove(nodeData) + view.explorer.showNotification( + "Error purging ${jobStatus.jobName}: ${jobStatus.jobId}", + "${throwable.message}", + NotificationType.ERROR, + e.project + ) + }.onSuccess { + view.explorer.showNotification( + "${jobStatus.jobName}: ${jobStatus.jobId} has been purged", + "$it", + NotificationType.INFORMATION, + e.project + ) } } } + waitJobsReleasedAndRefresh(jobsByFilterMap) } + /** * A job can be purged from the Jobs Tool Window * or from the JES Explorer by clicking on the corresponding job @@ -146,9 +284,11 @@ class PurgeJobAction : AnAction() { } if (view is JesExplorerView) { val selected = view.mySelectedNodesData - val node = selected.getOrNull(0)?.node - e.presentation.isVisible = selected.size == 1 - && node is JobNode + val wrongNode = selected.find { it.node !is JobNode } + e.presentation.apply { + isEnabledAndVisible = wrongNode == null && isSelectedJobNodesFromSameWS(selected) + text = if (isEnabledAndVisible && selected.size > 1) "Purge Jobs" else "Purge Job" + } } else if (view is JobBuildTreeView) { val jobStatus = view.getJobLogger().logFetcher.getCachedJobStatus()?.status if (jobStatus == null) { From 83375d2502a07b802ec6e060b1d3e003b5b8f181 Mon Sep 17 00:00:00 2001 From: Nikita Date: Fri, 18 Aug 2023 06:07:13 +0300 Subject: [PATCH 11/34] IJMP-1154: finalized error notifications --- .../synchronizer/DocumentedSyncProvider.kt | 46 +++++++++++++++---- .../formainframe/explorer/Explorer.kt | 3 ++ .../formainframe/utils/ussFileTagUtils.kt | 38 +++++++++++---- 3 files changed, 70 insertions(+), 17 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/content/synchronizer/DocumentedSyncProvider.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/content/synchronizer/DocumentedSyncProvider.kt index 8fd5850d9..000cacea5 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/content/synchronizer/DocumentedSyncProvider.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/content/synchronizer/DocumentedSyncProvider.kt @@ -11,16 +11,20 @@ package eu.ibagroup.formainframe.dataops.content.synchronizer import com.intellij.notification.Notification +import com.intellij.notification.NotificationAction import com.intellij.notification.NotificationType import com.intellij.notification.Notifications +import com.intellij.openapi.actionSystem.AnActionEvent import com.intellij.openapi.components.service import com.intellij.openapi.editor.Document import com.intellij.openapi.editor.impl.DocumentImpl import com.intellij.openapi.fileEditor.FileDocumentManager +import com.intellij.openapi.ui.Messages import com.intellij.openapi.util.text.StringUtilRt import com.intellij.openapi.vfs.VirtualFile import eu.ibagroup.formainframe.dataops.DataOpsManager import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes +import eu.ibagroup.formainframe.dataops.exceptions.CallException import eu.ibagroup.formainframe.editor.DocumentChangeListener import eu.ibagroup.formainframe.utils.castOrNull import eu.ibagroup.formainframe.vfs.MFVirtualFile @@ -49,15 +53,39 @@ class DocumentedSyncProvider( * param file - the file to get the name to display * param th - the throwable to show the error message */ - val defaultOnThrowableHandler: (VirtualFile, Throwable) -> Unit = { file, th -> - Notifications.Bus.notify( - Notification( - SYNC_NOTIFICATION_GROUP_ID, - "Cannot synchronize file \"${file.name}\" with mainframe", - th.message ?: "", - NotificationType.ERROR - ) - ) + val defaultOnThrowableHandler: (VirtualFile, Throwable) -> Unit = { _, th -> + lateinit var title: String + lateinit var details: String + + if (th is CallException) { + title = (th.errorParams?.getOrDefault("message", th.headMessage) as String).replaceFirstChar { it.uppercase() } + if (title.contains(".")) { + title = title.split(".")[0] + } + details = th.errorParams["details"]?.castOrNull>()?.joinToString("\n") ?: "Unknown error" + if (details.contains(":")) { + details = details.split(":").last() + } + } else { + title = th.message ?: th.toString() + details = "Unknown error" + } + Notification( + SYNC_NOTIFICATION_GROUP_ID, + title, + details, + NotificationType.ERROR + ).addAction(object : NotificationAction("More") { + override fun actionPerformed(e: AnActionEvent, notification: Notification) { + Messages.showErrorDialog( + e.project, + th.message ?: th.toString(), + title + ) + } + }).let { + Notifications.Bus.notify(it) + } } /** Default sync success handler. Won't do anything after the sync action is completed until redefined */ diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt index 1bc8ed12d..9933daae6 100755 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt @@ -208,6 +208,9 @@ abstract class AbstractExplorerBase>()?.joinToString("\n") ?: "Unknown error" + if (details.contains(":")) { + details = details.split(":").last() + } } else { title = t.message ?: t.toString() details = "Unknown error" diff --git a/src/main/kotlin/eu/ibagroup/formainframe/utils/ussFileTagUtils.kt b/src/main/kotlin/eu/ibagroup/formainframe/utils/ussFileTagUtils.kt index dfffe1d71..f9012492d 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/utils/ussFileTagUtils.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/utils/ussFileTagUtils.kt @@ -12,13 +12,17 @@ package eu.ibagroup.formainframe.utils import com.ibm.mq.headers.CCSID import com.intellij.notification.Notification +import com.intellij.notification.NotificationAction import com.intellij.notification.NotificationType import com.intellij.notification.Notifications +import com.intellij.openapi.actionSystem.AnActionEvent import com.intellij.openapi.components.service +import com.intellij.openapi.ui.Messages import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.dataops.DataOpsManager import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes import eu.ibagroup.formainframe.dataops.content.synchronizer.DEFAULT_BINARY_CHARSET +import eu.ibagroup.formainframe.dataops.exceptions.CallException import eu.ibagroup.formainframe.dataops.operations.uss.ChangeFileTagOperation import eu.ibagroup.formainframe.dataops.operations.uss.ChangeFileTagOperationParams import org.zowe.kotlinsdk.FileTagList @@ -177,14 +181,32 @@ fun removeUssFileTag(attributes: RemoteUssAttributes) { * @param title error text. */ private fun notifyError(th: Throwable, title: String) { - Notifications.Bus.notify( - Notification( - FILE_TAG_NOTIFICATION_GROUP_ID, - title, - th.message ?: "", - NotificationType.ERROR - ) - ) + + var details: String = if (th is CallException) { + th.errorParams?.get("details")?.castOrNull>()?.joinToString("\n") ?: "Unknown error" + } else { + "Unknown error" + } + if (details.contains(":")) { + details = details.split(":").last() + } + + Notification( + FILE_TAG_NOTIFICATION_GROUP_ID, + title, + details, + NotificationType.ERROR + ).addAction(object : NotificationAction("More") { + override fun actionPerformed(e: AnActionEvent, notification: Notification) { + Messages.showErrorDialog( + e.project, + th.message ?: th.toString(), + title + ) + } + }).let { + Notifications.Bus.notify(it) + } } private val unsupportedEncodings = listOf( From 6afa32654b28de3ffe55628219742b3b4822f809 Mon Sep 17 00:00:00 2001 From: Arseni Tsikhamirau Date: Fri, 18 Aug 2023 12:14:13 +0300 Subject: [PATCH 12/34] IJMP-822-Bulk-jobs-purge 1) added tests --- .../explorer/actions/PurgeJobAction.kt | 10 + .../actions/PurgeJobActionTestSpec.kt | 329 ++++++++++++++++-- 2 files changed, 312 insertions(+), 27 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt index c1f86f6b7..dc44536b4 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt @@ -1,3 +1,13 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + package eu.ibagroup.formainframe.explorer.actions import com.intellij.notification.NotificationType diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt index 3331aacf3..eea83d512 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt @@ -1,5 +1,46 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + package eu.ibagroup.formainframe.explorer.actions +import com.intellij.notification.NotificationType +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.application.ApplicationManager +import com.intellij.openapi.progress.ProgressIndicator +import com.intellij.openapi.project.Project +import com.intellij.testFramework.LightProjectDescriptor +import com.intellij.testFramework.fixtures.IdeaTestFixtureFactory +import com.intellij.testFramework.fixtures.impl.LightTempDirTestFixtureImpl +import eu.ibagroup.formainframe.api.ZosmfApi +import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.config.connect.authToken +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.attributes.JobsRequester +import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes +import eu.ibagroup.formainframe.dataops.operations.jobs.PurgeJobOperation +import eu.ibagroup.formainframe.explorer.Explorer +import eu.ibagroup.formainframe.explorer.JesWorkingSetImpl +import eu.ibagroup.formainframe.explorer.ui.* +import eu.ibagroup.formainframe.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.testServiceImpl.TestZosmfApiImpl +import eu.ibagroup.formainframe.utils.cancelByIndicator +import eu.ibagroup.formainframe.utils.service +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.core.spec.style.ShouldSpec +import io.kotest.matchers.shouldBe +import io.mockk.* +import org.junit.jupiter.api.Assertions.assertThrows +import org.zowe.kotlinsdk.* +import retrofit2.Response + // TODO: this test suite needs to be reworked (?) as it gives StackOverflowError. May be the problem with MockK/Kotest // import com.intellij.notification.NotificationType @@ -52,33 +93,267 @@ package eu.ibagroup.formainframe.explorer.actions // import retrofit2.Call // import retrofit2.Response -// class PurgeJobActionTestSpec : ShouldSpec({ -// beforeSpec { -// // FIXTURE SETUP TO HAVE ACCESS TO APPLICATION INSTANCE -// val factory = IdeaTestFixtureFactory.getFixtureFactory() -// val projectDescriptor = LightProjectDescriptor.EMPTY_PROJECT_DESCRIPTOR -// val fixtureBuilder = factory.createLightFixtureBuilder(projectDescriptor, "for-mainframe") -// val fixture = fixtureBuilder.fixture -// val myFixture = IdeaTestFixtureFactory.getFixtureFactory().createCodeInsightFixture( -// fixture, -// LightTempDirTestFixtureImpl(true) -// ) -// myFixture.setUp() -// } -// afterSpec { -// clearAllMocks() -// } -// context("explorer module: actions/PurgeJobAction") { -// context("actionPerformed") { -// val purgeAction = PurgeJobAction() -// val mockActionEventForJesEx = mockk() -// val jesExplorerView = mockk() - -// val job = mockk() -// every { job.jobName } returns "name" -// every { job.jobId } returns "id" -// val connectionConfig = mockk() -// every { connectionConfig.uuid } returns "uuid" + class PurgeJobActionTestSpec : ShouldSpec({ + beforeSpec { + // FIXTURE SETUP TO HAVE ACCESS TO APPLICATION INSTANCE + val factory = IdeaTestFixtureFactory.getFixtureFactory() + val projectDescriptor = LightProjectDescriptor.EMPTY_PROJECT_DESCRIPTOR + val fixtureBuilder = factory.createLightFixtureBuilder(projectDescriptor, "for-mainframe") + val fixture = fixtureBuilder.fixture + val myFixture = IdeaTestFixtureFactory.getFixtureFactory().createCodeInsightFixture( + fixture, + LightTempDirTestFixtureImpl(true) + ) + myFixture.setUp() + } + afterSpec { + clearAllMocks() + } + context("explorer module: actions/PurgeJobAction") { + context("api spec") { + + val zosmfApi = ApplicationManager.getApplication().service() as TestZosmfApiImpl + zosmfApi.testInstance = mockk() + + val responseMockk = mockk>>() + every { + zosmfApi.testInstance.getApi(JESApi::class.java, any()) + .getFilteredJobs(any(), any(), any(), any(), any(), any(), any(), any()) + .execute() + } returns responseMockk + + val purgeAction = spyk(PurgeJobAction()) + val projectMock = mockk() + val mockActionEventForJesEx = mockk() + every { mockActionEventForJesEx.project } returns projectMock + every { mockActionEventForJesEx.presentation.text = any() } just Runs + val jesExplorerView = mockk() + val explorerMock = mockk>() + every { mockActionEventForJesEx.getData(EXPLORER_VIEW) } returns jesExplorerView + every { jesExplorerView.explorer } returns explorerMock + val connectionConfig = mockk() + every { connectionConfig.uuid } returns "uuid" + every { connectionConfig.authToken } returns "auth_token" + val diffConnectionConfig = mockk() + every { diffConnectionConfig.uuid } returns "diffUuid" + + context("common test spec") { + + val jobParent1 = mockk() + val jobParentParent1 = mockk() + every { jobParent1.parent } returns jobParentParent1 + every { jobParent1.query } returns mockk() + every { jobParent1.query?.connectionConfig } returns connectionConfig + every { jobParent1.query?.request } returns mockk() + every { jobParent1.query?.request?.prefix } returns "prefix_1" + every { jobParent1.query?.request?.owner } returns "owner" + every { jobParent1.query?.request?.userCorrelatorFilter } returns "filter_1" + + every { jobParentParent1.unit } returns mockk() + every { jobParentParent1.unit.name } returns "firstWS" + val jobParent2 = mockk() + val jobParentParent2 = mockk() + every { jobParent2.parent } returns jobParentParent2 + every { jobParent2.query } returns mockk() + every { jobParent2.query?.connectionConfig } returns connectionConfig + every { jobParent2.query?.request } returns mockk() + every { jobParent2.query?.request?.prefix } returns "prefix_2" + every { jobParent2.query?.request?.owner } returns "owner" + every { jobParent2.query?.request?.userCorrelatorFilter } returns "filter_2" + every { jobParentParent2.unit } returns mockk() + every { jobParentParent2.unit.name } returns "secondWS" + + context("isSelectedJobNodesFromSameWS") { + + val jesWorkingSet1 = mockk() + every { jesWorkingSet1.connectionConfig } returns connectionConfig + val jesWorkingSet2 = mockk() + every { jesWorkingSet2.connectionConfig } returns connectionConfig + val jesWorkingSet3 = mockk() + every { jesWorkingSet3.connectionConfig } returns diffConnectionConfig + + val jobNode1 = mockk() + every { jobNode1.unit } returns jesWorkingSet1 + every { jobNode1.query } returns mockk() + every { jobNode1.query?.connectionConfig } returns connectionConfig + every { jobNode1.parent } returns jobParent1 + val jobNode2 = mockk() + every { jobNode2.unit } returns jesWorkingSet2 + every { jobNode2.query } returns mockk() + every { jobNode2.query?.connectionConfig } returns connectionConfig + every { jobNode2.parent } returns jobParent2 + val jobNode3 = mockk() + every { jobNode3.unit } returns jesWorkingSet3 + every { jobNode3.query } returns mockk() + every { jobNode3.query?.connectionConfig } returns diffConnectionConfig + val wrongJobNode4 = mockk() + every { wrongJobNode4.unit } returns jesWorkingSet3 + + val virtualFileMock = mockk() + val jobInfo1 = mockk() + val jobInfo2 = mockk() + every { jobInfo1.jobId } returns "TSU01" + every { jobInfo2.jobId } returns "TSU02" + every { jobInfo1.jobName } returns "test1" + every { jobInfo2.jobName } returns "test2" + val requester1 = mockk() + val requester2 = mockk() + every { requester1.connectionConfig } returns connectionConfig + every { requester2.connectionConfig } returns connectionConfig + + val attributes1 = mockk() + every { attributes1.jobInfo } returns jobInfo1 + every { attributes1.requesters[0] } returns requester1 + val attributes2 = mockk() + every { attributes2.jobInfo } returns jobInfo2 + every { attributes2.requesters[0] } returns requester2 + val attributes3 = mockk() + val attributes4 = mockk() + + var mySelectedData: List> + var isEnableAndVisibleAction: Boolean + + every { mockActionEventForJesEx.presentation.isEnabledAndVisible = false } answers { + isEnableAndVisibleAction = false + } + every { mockActionEventForJesEx.presentation.isEnabledAndVisible = true } answers { + isEnableAndVisibleAction = true + } + + should("action is not visible when selected job nodes contains wrong node") { + isEnableAndVisibleAction = true + val nodeData1 = NodeData(jobNode1, virtualFileMock, attributes1) + mockkObject(nodeData1) + val nodeData2 = NodeData(jobNode2, virtualFileMock, attributes2) + mockkObject(nodeData2) + val nodeData3 = NodeData(jobNode3, virtualFileMock, attributes3) + mockkObject(nodeData3) + val nodeData4 = NodeData(wrongJobNode4, virtualFileMock, attributes4) + mockkObject(nodeData4) + mySelectedData = listOf(nodeData1, nodeData2, nodeData3, nodeData4) + every { jesExplorerView.mySelectedNodesData } returns mySelectedData + every { mockActionEventForJesEx.presentation.isEnabledAndVisible } returns false + purgeAction.update(mockActionEventForJesEx) + + assertSoftly { + isEnableAndVisibleAction shouldBe false + } + unmockkObject(nodeData1, nodeData2, nodeData3, nodeData4) + } + should("action is not visible when selected job nodes from different connections") { + isEnableAndVisibleAction = true + val nodeData1 = NodeData(jobNode1, virtualFileMock, attributes1) + mockkObject(nodeData1) + val nodeData2 = NodeData(jobNode2, virtualFileMock, attributes2) + mockkObject(nodeData2) + val nodeData3 = NodeData(jobNode3, virtualFileMock, attributes3) + mockkObject(nodeData3) + mySelectedData = listOf(nodeData1, nodeData2, nodeData3) + every { jesExplorerView.mySelectedNodesData } returns mySelectedData + purgeAction.update(mockActionEventForJesEx) + + assertSoftly { + isEnableAndVisibleAction shouldBe false + } + unmockkObject(nodeData1, nodeData2, nodeData3) + } + should("action is not visible when selected job nodes from different working sets") { + isEnableAndVisibleAction = true + val nodeData1 = NodeData(jobNode1, virtualFileMock, attributes1) + mockkObject(nodeData1) + val nodeData2 = NodeData(jobNode2, virtualFileMock, attributes2) + mockkObject(nodeData2) + mySelectedData = listOf(nodeData1, nodeData2) + every { jesExplorerView.mySelectedNodesData } returns mySelectedData + purgeAction.update(mockActionEventForJesEx) + + assertSoftly { + isEnableAndVisibleAction shouldBe false + } + unmockkObject(nodeData1, nodeData2) + } + should("action is visible when selected job nodes from the same connections and same working set") { + isEnableAndVisibleAction = false + every { jobParentParent1.unit.name } returns "sameWS" + every { jobParentParent2.unit.name } returns "sameWS" + val nodeData1 = NodeData(jobNode1, virtualFileMock, attributes1) + mockkObject(nodeData1) + val nodeData2 = NodeData(jobNode2, virtualFileMock, attributes2) + mockkObject(nodeData2) + mySelectedData = listOf(nodeData1, nodeData2) + every { jesExplorerView.mySelectedNodesData } returns mySelectedData + purgeAction.update(mockActionEventForJesEx) + + assertSoftly { + isEnableAndVisibleAction shouldBe true + } + unmockkObject(nodeData1, nodeData2) + } + should("purge actionPerformed when jobs haven't been purged due to error") { + var isJobsPurged = true + val nodeData1 = NodeData(jobNode1, virtualFileMock, attributes1) + mockkObject(nodeData1) + val nodeData2 = NodeData(jobNode2, virtualFileMock, attributes2) + mockkObject(nodeData2) + mySelectedData = listOf(nodeData1, nodeData2) + every { jesExplorerView.mySelectedNodesData } returns mySelectedData + every { + explorerMock.showNotification( + any() as String, + any() as String, + any() as NotificationType, + any() as Project + ) + } just Runs + every { responseMockk.isSuccessful } returns true + every { responseMockk.body() } returns listOf(jobInfo1, jobInfo2) + every { jobParent1.cleanCache() } answers { + isJobsPurged = false + } + every { jobParent2.cleanCache() } answers { + isJobsPurged = false + } + purgeAction.actionPerformed(mockActionEventForJesEx) + + assertSoftly { + isJobsPurged shouldBe false + } + } + should("purge actionPerformed when jobs have been purged successfully") { + var isJobsPurged = false + val dataOpsManager = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + dataOpsManager.testInstance = mockk() + every { dataOpsManager.testInstance.performOperation(any() as PurgeJobOperation, any() as ProgressIndicator) } answers { + isJobsPurged = true + CancelJobPurgeOutRequest() + } + every { responseMockk.isSuccessful } returns true + every { responseMockk.body() } returns listOf(jobInfo1, jobInfo2) + + purgeAction.actionPerformed(mockActionEventForJesEx) + + assertSoftly { + isJobsPurged shouldBe true + } + } + should("purge actionPerformed when jobs have been purged successfully, but refresh by filter fails") { + every { responseMockk.isSuccessful } returns false + every { responseMockk.body() } returns listOf() + every { jobParent1.unit } returns mockk() + every { jobParent1.unit.name } returns "firstFilter" + every { jobParent2.unit } returns mockk() + every { jobParent2.unit.name } returns "secondFilter" + + assertThrows(RuntimeException::class.java) { + purgeAction.actionPerformed(mockActionEventForJesEx) + } + } + } + } + unmockkAll() + } + } + }) // val jobsFilter = spyk( // JobsFilter( // "owner", From 82763f582aa16976eef01005d8cd64b763d96481 Mon Sep 17 00:00:00 2001 From: Uladzislau Date: Tue, 22 Aug 2023 18:59:17 +0200 Subject: [PATCH 13/34] Changes after merge with release branch --- .../DatasetFileFetchProviderTestSpec.kt | 76 ++++++++++--- .../dataops/JobFetchProviderTestSpec.kt | 103 +++++++++++++---- .../dataops/OperationsTestSpec.kt | 3 + .../actions/PurgeJobActionTestSpec.kt | 53 +++++++-- .../actions/UssSortActionHolderTestSpec.kt | 104 ++++++++++++++---- 5 files changed, 264 insertions(+), 75 deletions(-) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/dataops/DatasetFileFetchProviderTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/dataops/DatasetFileFetchProviderTestSpec.kt index 3a5609cc4..cd62ae6bb 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/dataops/DatasetFileFetchProviderTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/dataops/DatasetFileFetchProviderTestSpec.kt @@ -20,10 +20,14 @@ import eu.ibagroup.formainframe.api.ZosmfApi import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.connect.authToken import eu.ibagroup.formainframe.config.ws.DSMask -import eu.ibagroup.formainframe.dataops.attributes.* +import eu.ibagroup.formainframe.dataops.attributes.AttributesService +import eu.ibagroup.formainframe.dataops.attributes.FileAttributes +import eu.ibagroup.formainframe.dataops.attributes.MaskedRequester +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributesService import eu.ibagroup.formainframe.dataops.fetch.DatasetFileFetchProvider -import eu.ibagroup.formainframe.testServiceImpl.TestDataOpsManagerImpl -import eu.ibagroup.formainframe.testServiceImpl.TestZosmfApiImpl +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestZosmfApiImpl import eu.ibagroup.formainframe.utils.cancelByIndicator import eu.ibagroup.formainframe.utils.service import eu.ibagroup.formainframe.vfs.MFVirtualFile @@ -31,8 +35,19 @@ import io.kotest.assertions.assertSoftly import io.kotest.core.spec.style.ShouldSpec import io.kotest.matchers.collections.shouldHaveSize import io.kotest.matchers.shouldBe -import io.mockk.* -import org.zowe.kotlinsdk.* +import io.mockk.Runs +import io.mockk.clearAllMocks +import io.mockk.every +import io.mockk.just +import io.mockk.mockk +import io.mockk.spyk +import io.mockk.unmockkAll +import org.zowe.kotlinsdk.DataAPI +import org.zowe.kotlinsdk.DataSetsList +import org.zowe.kotlinsdk.Dataset +import org.zowe.kotlinsdk.DatasetOrganization +import org.zowe.kotlinsdk.HasMigrated +import org.zowe.kotlinsdk.XIBMAttr import retrofit2.Call import retrofit2.Response @@ -96,10 +111,29 @@ class DatasetFileFetchProviderTestSpec : ShouldSpec({ zosmfApi.testInstance = mockk() val mockedApi = mockk() every { zosmfApi.testInstance.getApi(DataAPI::class.java, mockedConnectionConfig) } returns mockedApi - every { mockedApi.listDataSets(authorizationToken = any() as String, dsLevel = any() as String, volser = any() as String, xIBMAttr = any() as XIBMAttr, xIBMMaxItems = any() as Int, start = any() as String) } returns mockedCall - every { mockedApi.listDataSets(authorizationToken = any() as String, dsLevel = any() as String, volser = any() as String, xIBMAttr = any() as XIBMAttr, xIBMMaxItems = any() as Int, start = any() as String).cancelByIndicator(progressMockk) } returns mockedCall - - val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + every { + mockedApi.listDataSets( + authorizationToken = any() as String, + dsLevel = any() as String, + volser = any() as String, + xIBMAttr = any() as XIBMAttr, + xIBMMaxItems = any() as Int, + start = any() as String + ) + } returns mockedCall + every { + mockedApi.listDataSets( + authorizationToken = any() as String, + dsLevel = any() as String, + volser = any() as String, + xIBMAttr = any() as XIBMAttr, + xIBMMaxItems = any() as Int, + start = any() as String + ).cancelByIndicator(progressMockk) + } returns mockedCall + + val dataOpsManagerService = + ApplicationManager.getApplication().service() as TestDataOpsManagerImpl val componentManager = dataOpsManagerService.componentManager // needed for cleanupUnusedFile test @@ -111,14 +145,17 @@ class DatasetFileFetchProviderTestSpec : ShouldSpec({ afterEach { unmockkAll() } - val datasetFileFetchProviderForTest = spyk(DatasetFileFetchProvider(dataOpsManagerService), recordPrivateCalls = true) + val datasetFileFetchProviderForTest = + spyk(DatasetFileFetchProvider(dataOpsManagerService), recordPrivateCalls = true) should("dataset fetch provider fetchResponse test") { - val fetchResponseMethodRef = datasetFileFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + val fetchResponseMethodRef = + datasetFileFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } fetchResponseMethodRef.trySetAccessible() every { mockedResponse.isSuccessful } returns true - val datasetAttributes = (fetchResponseMethodRef.invoke(datasetFileFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) + val datasetAttributes = + (fetchResponseMethodRef.invoke(datasetFileFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) assertSoftly { datasetAttributes shouldHaveSize dataSetsList.items.size @@ -127,9 +164,10 @@ class DatasetFileFetchProviderTestSpec : ShouldSpec({ should("cleanup unused file if connection config of the query is the same as for dataset file") { var cleanupPerformed = false - val cleanupUnusedFileMethodRef = datasetFileFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } + val cleanupUnusedFileMethodRef = + datasetFileFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } cleanupUnusedFileMethodRef.trySetAccessible() - dataOpsManagerService.testInstance = object: TestDataOpsManagerImpl(componentManager) { + dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(componentManager) { override fun getAttributesService( attributesClass: Class, vFileClass: Class @@ -158,7 +196,8 @@ class DatasetFileFetchProviderTestSpec : ShouldSpec({ should("cleanup unused file if connection config of the query is the same as for dataset file") { var cleanupPerformed = false - val cleanupUnusedFileMethodRef = datasetFileFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } + val cleanupUnusedFileMethodRef = + datasetFileFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } cleanupUnusedFileMethodRef.trySetAccessible() every { mockedAttributesService.getAttributes(mockedVirtualFile) } returns mockedFileAttributes @@ -167,7 +206,12 @@ class DatasetFileFetchProviderTestSpec : ShouldSpec({ every { mockedMaskedRequester.connectionConfig } returns mockedConnectionConfig every { mockedMaskedRequester.queryVolser } returns "ANOTHER" every { mockedFileAttributes.requesters } returns requesters - every { mockedAttributesService.updateAttributes(mockedVirtualFile, any() as RemoteDatasetAttributes.() -> Unit) } answers { + every { + mockedAttributesService.updateAttributes( + mockedVirtualFile, + any() as RemoteDatasetAttributes.() -> Unit + ) + } answers { cleanupPerformed = true secondArg Unit>().invoke(mockedFileAttributes) } diff --git a/src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchProviderTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchProviderTestSpec.kt index 1b8eb0a02..6d83e413d 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchProviderTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/dataops/JobFetchProviderTestSpec.kt @@ -20,10 +20,14 @@ import eu.ibagroup.formainframe.api.ZosmfApi import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.connect.authToken import eu.ibagroup.formainframe.config.ws.JobsFilter -import eu.ibagroup.formainframe.dataops.attributes.* +import eu.ibagroup.formainframe.dataops.attributes.AttributesService +import eu.ibagroup.formainframe.dataops.attributes.FileAttributes +import eu.ibagroup.formainframe.dataops.attributes.JobsRequester +import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributesService import eu.ibagroup.formainframe.dataops.fetch.JobFetchProvider -import eu.ibagroup.formainframe.testServiceImpl.TestDataOpsManagerImpl -import eu.ibagroup.formainframe.testServiceImpl.TestZosmfApiImpl +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestZosmfApiImpl import eu.ibagroup.formainframe.utils.cancelByIndicator import eu.ibagroup.formainframe.utils.service import eu.ibagroup.formainframe.vfs.MFVirtualFile @@ -33,7 +37,13 @@ import io.kotest.core.spec.style.ShouldSpec import io.kotest.matchers.collections.shouldHaveSize import io.kotest.matchers.shouldBe import io.kotest.matchers.throwable.shouldHaveMessage -import io.mockk.* +import io.mockk.Runs +import io.mockk.clearAllMocks +import io.mockk.every +import io.mockk.just +import io.mockk.mockk +import io.mockk.spyk +import io.mockk.unmockkAll import org.zowe.kotlinsdk.ExecData import org.zowe.kotlinsdk.JESApi import org.zowe.kotlinsdk.Job @@ -86,7 +96,7 @@ class JobFetchProviderTestSpec : ShouldSpec({ every { job1.jobId } returns "TSUTEST1" every { job2.jobId } returns "TSUTEST2" every { job1.jobName } returns "TESTJOB1" - every {job2.jobName } returns "TESTJOB2" + every { job2.jobName } returns "TESTJOB2" val jobs = mutableListOf(job1, job2) val mockedCall = mockk>>() @@ -98,12 +108,41 @@ class JobFetchProviderTestSpec : ShouldSpec({ zosmfApi.testInstance = mockk() val mockedApi = mockk() every { zosmfApi.testInstance.getApi(JESApi::class.java, mockedConnectionConfig) } returns mockedApi - every { mockedApi.getFilteredJobs(basicCredentials = any() as String, owner = any() as String, prefix = any() as String, userCorrelator = any() as String, execData = any() as ExecData) } returns mockedCall - every { mockedApi.getFilteredJobs(basicCredentials = any() as String, owner = any() as String, prefix = any() as String, userCorrelator = any() as String, execData = any() as ExecData).cancelByIndicator(progressMockk) } returns mockedCall - every { mockedApi.getFilteredJobs(basicCredentials = any() as String, jobId = any() as String, execData = any() as ExecData) } returns mockedCall - every { mockedApi.getFilteredJobs(basicCredentials = any() as String, jobId = any() as String, execData = any() as ExecData).cancelByIndicator(progressMockk) } returns mockedCall - - val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + every { + mockedApi.getFilteredJobs( + basicCredentials = any() as String, + owner = any() as String, + prefix = any() as String, + userCorrelator = any() as String, + execData = any() as ExecData + ) + } returns mockedCall + every { + mockedApi.getFilteredJobs( + basicCredentials = any() as String, + owner = any() as String, + prefix = any() as String, + userCorrelator = any() as String, + execData = any() as ExecData + ).cancelByIndicator(progressMockk) + } returns mockedCall + every { + mockedApi.getFilteredJobs( + basicCredentials = any() as String, + jobId = any() as String, + execData = any() as ExecData + ) + } returns mockedCall + every { + mockedApi.getFilteredJobs( + basicCredentials = any() as String, + jobId = any() as String, + execData = any() as ExecData + ).cancelByIndicator(progressMockk) + } returns mockedCall + + val dataOpsManagerService = + ApplicationManager.getApplication().service() as TestDataOpsManagerImpl val componentManager = dataOpsManagerService.componentManager // needed for cleanupUnusedFile test @@ -119,12 +158,14 @@ class JobFetchProviderTestSpec : ShouldSpec({ should("fetchResponse get job attributes if job ID is not null and response is successful and exec dates/times are null") { - val fetchResponseMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + val fetchResponseMethodRef = + jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } fetchResponseMethodRef.trySetAccessible() every { mockedRequest.jobId } returns "TSUTEST" every { mockedResponse.isSuccessful } returns true - val jobAttributes = (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) + val jobAttributes = + (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) assertSoftly { jobAttributes shouldHaveSize jobs.size @@ -133,12 +174,14 @@ class JobFetchProviderTestSpec : ShouldSpec({ should("fetchResponse get job attributes if job ID is null and response is successful") { - val fetchResponseMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + val fetchResponseMethodRef = + jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } fetchResponseMethodRef.trySetAccessible() every { mockedRequest.jobId } returns "" every { mockedResponse.isSuccessful } returns true - val jobAttributes = (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) + val jobAttributes = + (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) assertSoftly { jobAttributes shouldHaveSize jobs.size @@ -147,7 +190,8 @@ class JobFetchProviderTestSpec : ShouldSpec({ should("fetchResponse get job attributes if exec dates/times are not null") { - val fetchResponseMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + val fetchResponseMethodRef = + jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } fetchResponseMethodRef.trySetAccessible() every { mockedRequest.jobId } returns "TSUTEST" every { mockedResponse.isSuccessful } returns true @@ -159,7 +203,8 @@ class JobFetchProviderTestSpec : ShouldSpec({ every { job2.execEnded } returns "" every { job2.execSubmitted } returns "" - val jobAttributes = (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) + val jobAttributes = + (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) assertSoftly { jobAttributes shouldHaveSize jobs.size @@ -168,13 +213,15 @@ class JobFetchProviderTestSpec : ShouldSpec({ should("fetchResponse get job attributes if response does not return any job") { - val fetchResponseMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + val fetchResponseMethodRef = + jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } fetchResponseMethodRef.trySetAccessible() every { mockedRequest.jobId } returns "TSUTEST" every { mockedResponse.isSuccessful } returns true every { mockedResponse.body() } returns emptyList() - val jobAttributes = (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) + val jobAttributes = + (fetchResponseMethodRef.invoke(jobFetchProviderForTest, mockedQuery, progressMockk) as Collection<*>) assertSoftly { jobAttributes shouldHaveSize 0 @@ -182,7 +229,8 @@ class JobFetchProviderTestSpec : ShouldSpec({ } should("fetchResponse get job attributes if response was not successful") { - val fetchResponseMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } + val fetchResponseMethodRef = + jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "fetchResponse" } fetchResponseMethodRef.trySetAccessible() every { mockedRequest.jobId } returns "TSUTEST" every { mockedResponse.isSuccessful } returns false @@ -200,9 +248,10 @@ class JobFetchProviderTestSpec : ShouldSpec({ should("cleanup unused file if connection config of the query is the same as for job file") { var cleanupPerformed = false - val cleanupUnusedFileMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } + val cleanupUnusedFileMethodRef = + jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } cleanupUnusedFileMethodRef.trySetAccessible() - dataOpsManagerService.testInstance = object: TestDataOpsManagerImpl(componentManager) { + dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(componentManager) { override fun getAttributesService( attributesClass: Class, vFileClass: Class @@ -230,12 +279,18 @@ class JobFetchProviderTestSpec : ShouldSpec({ should("cleanup file if connections are not the same") { var cleanupPerformed = false - val cleanupUnusedFileMethodRef = jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } + val cleanupUnusedFileMethodRef = + jobFetchProviderForTest::class.java.declaredMethods.first { it.name == "cleanupUnusedFile" } cleanupUnusedFileMethodRef.trySetAccessible() every { mockedJobsRequester.connectionConfig } returns mockk() every { mockedFileAttributes.requesters } returns requesters - every { mockedAttributesService.updateAttributes(mockedVirtualFile, any() as RemoteJobAttributes.() -> Unit) } answers { + every { + mockedAttributesService.updateAttributes( + mockedVirtualFile, + any() as RemoteJobAttributes.() -> Unit + ) + } answers { cleanupPerformed = true secondArg Unit>().invoke(mockedFileAttributes) } diff --git a/src/test/kotlin/eu/ibagroup/formainframe/dataops/OperationsTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/dataops/OperationsTestSpec.kt index dee3612ba..293ab8686 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/dataops/OperationsTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/dataops/OperationsTestSpec.kt @@ -40,6 +40,7 @@ import eu.ibagroup.formainframe.dataops.operations.mover.RemoteToLocalFileMover import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl import eu.ibagroup.formainframe.testutils.testServiceImpl.TestZosmfApiImpl +import eu.ibagroup.formainframe.utils.cancelByIndicator import eu.ibagroup.formainframe.utils.castOrNull import eu.ibagroup.formainframe.utils.changeFileEncodingTo import eu.ibagroup.formainframe.utils.service @@ -57,6 +58,8 @@ import io.mockk.spyk import io.mockk.unmockkAll import org.zowe.kotlinsdk.DataAPI import org.zowe.kotlinsdk.FilePath +import org.zowe.kotlinsdk.InfoAPI +import org.zowe.kotlinsdk.InfoResponse import org.zowe.kotlinsdk.XIBMDataType import org.zowe.kotlinsdk.annotations.ZVersion import retrofit2.Call diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt index 8bf8e6f85..c0ebffd2d 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt @@ -12,33 +12,59 @@ package eu.ibagroup.formainframe.explorer.actions import com.intellij.notification.NotificationType import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DataKey import com.intellij.openapi.application.ApplicationManager import com.intellij.openapi.progress.ProgressIndicator import com.intellij.openapi.project.Project -import com.intellij.testFramework.LightProjectDescriptor -import com.intellij.testFramework.fixtures.IdeaTestFixtureFactory -import com.intellij.testFramework.fixtures.impl.LightTempDirTestFixtureImpl +import com.intellij.openapi.vfs.VirtualFile import eu.ibagroup.formainframe.api.ZosmfApi import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.config.connect.CredentialService import eu.ibagroup.formainframe.config.connect.authToken +import eu.ibagroup.formainframe.config.ws.JobsFilter import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.Operation +import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.attributes.FileAttributes import eu.ibagroup.formainframe.dataops.attributes.JobsRequester import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes +import eu.ibagroup.formainframe.dataops.log.JobLogFetcher +import eu.ibagroup.formainframe.dataops.log.MFLogger import eu.ibagroup.formainframe.dataops.operations.jobs.PurgeJobOperation import eu.ibagroup.formainframe.explorer.Explorer import eu.ibagroup.formainframe.explorer.JesWorkingSetImpl -import eu.ibagroup.formainframe.explorer.ui.* -import eu.ibagroup.formainframe.testServiceImpl.TestDataOpsManagerImpl -import eu.ibagroup.formainframe.testServiceImpl.TestZosmfApiImpl -import eu.ibagroup.formainframe.utils.cancelByIndicator +import eu.ibagroup.formainframe.explorer.ui.EXPLORER_VIEW +import eu.ibagroup.formainframe.explorer.ui.ErrorNode +import eu.ibagroup.formainframe.explorer.ui.JesExplorerView +import eu.ibagroup.formainframe.explorer.ui.JesFilterNode +import eu.ibagroup.formainframe.explorer.ui.JesWsNode +import eu.ibagroup.formainframe.explorer.ui.JobNode +import eu.ibagroup.formainframe.explorer.ui.NodeData +import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestZosmfApiImpl +import eu.ibagroup.formainframe.ui.build.jobs.JobBuildTreeView import eu.ibagroup.formainframe.utils.service import eu.ibagroup.formainframe.vfs.MFVirtualFile import io.kotest.assertions.assertSoftly -import io.kotest.core.spec.style.ShouldSpec import io.kotest.matchers.shouldBe -import io.mockk.* +import io.mockk.Runs +import io.mockk.clearAllMocks +import io.mockk.every +import io.mockk.just +import io.mockk.justRun +import io.mockk.mockk +import io.mockk.mockkObject +import io.mockk.spyk +import io.mockk.unmockkAll +import io.mockk.unmockkObject import org.junit.jupiter.api.Assertions.assertThrows -import org.zowe.kotlinsdk.* +import org.zowe.kotlinsdk.CancelJobPurgeOutRequest +import org.zowe.kotlinsdk.JESApi +import org.zowe.kotlinsdk.Job +import org.zowe.kotlinsdk.gson +import retrofit2.Call import retrofit2.Response class PurgeJobActionTestSpec : WithApplicationShouldSpec({ @@ -501,7 +527,12 @@ class PurgeJobActionTestSpec : WithApplicationShouldSpec({ var isJobsPurged = false val dataOpsManager = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl dataOpsManager.testInstance = mockk() - every { dataOpsManager.testInstance.performOperation(any() as PurgeJobOperation, any() as ProgressIndicator) } answers { + every { + dataOpsManager.testInstance.performOperation( + any() as PurgeJobOperation, + any() as ProgressIndicator + ) + } answers { isJobsPurged = true CancelJobPurgeOutRequest() } diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt index 7bf5d268c..db17cde3a 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt @@ -26,14 +26,24 @@ import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes import eu.ibagroup.formainframe.dataops.fetch.FileFetchProvider import eu.ibagroup.formainframe.dataops.fetch.UssFileFetchProvider import eu.ibagroup.formainframe.dataops.fetch.UssQuery -import eu.ibagroup.formainframe.explorer.ui.* -import eu.ibagroup.formainframe.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.explorer.ui.FileExplorerView +import eu.ibagroup.formainframe.explorer.ui.LibraryNode +import eu.ibagroup.formainframe.explorer.ui.NodeData +import eu.ibagroup.formainframe.explorer.ui.UssDirNode +import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl import eu.ibagroup.formainframe.utils.service import eu.ibagroup.formainframe.vfs.MFVirtualFile import io.kotest.assertions.assertSoftly import io.kotest.core.spec.style.ShouldSpec import io.kotest.matchers.shouldBe -import io.mockk.* +import io.mockk.Runs +import io.mockk.clearAllMocks +import io.mockk.every +import io.mockk.just +import io.mockk.mockk +import io.mockk.mockkObject +import io.mockk.spyk import javax.swing.tree.TreePath class UssSortActionHolderTestSpec : ShouldSpec({ @@ -125,7 +135,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("is visible from context menu if file explorer view is not null and selected node is not UssDirNode") { var isVisible = true - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { isVisible = false listOf(mockedNodeDataNotUssForTest) @@ -198,7 +209,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by name action performed if selected node is not UssDirNode") { var actionPerformed = false - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { actionPerformed = true listOf(mockedNodeDataNotUssForTest) @@ -212,7 +224,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by name action isSelected if selected node is not UssDirNode") { var isSelected = true - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { isSelected = false listOf(mockedNodeDataNotUssForTest) @@ -251,7 +264,10 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by name action performed if sort query keys is not empty") { var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DATE, SortQueryKeys.ASCENDING) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( + SortQueryKeys.DATE, + SortQueryKeys.ASCENDING + ) every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { actionPerformed = true } @@ -264,7 +280,11 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by name action performed if sort query keys is not empty and contains desired key") { var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DATE, SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( + SortQueryKeys.DATE, + SortQueryKeys.NAME, + SortQueryKeys.ASCENDING + ) every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { actionPerformed = true } @@ -329,7 +349,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by type action performed if selected node is not UssDirNode") { var actionPerformed = false - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { actionPerformed = true listOf(mockedNodeDataNotUssForTest) @@ -343,7 +364,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by type action isSelected if selected node is not UssDirNode") { var isSelected = true - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { isSelected = false listOf(mockedNodeDataNotUssForTest) @@ -382,7 +404,10 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by type action performed if sort query keys is not empty") { var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DATE, SortQueryKeys.ASCENDING) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( + SortQueryKeys.DATE, + SortQueryKeys.ASCENDING + ) every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { actionPerformed = true } @@ -395,7 +420,11 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by type action performed if sort query keys is not empty and contains desired key") { var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DATE, SortQueryKeys.TYPE, SortQueryKeys.ASCENDING) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( + SortQueryKeys.DATE, + SortQueryKeys.TYPE, + SortQueryKeys.ASCENDING + ) every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { actionPerformed = true } @@ -460,7 +489,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by date action performed if selected node is not UssDirNode") { var actionPerformed = false - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { actionPerformed = true listOf(mockedNodeDataNotUssForTest) @@ -474,7 +504,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by date action isSelected if selected node is not UssDirNode") { var isSelected = true - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { isSelected = false listOf(mockedNodeDataNotUssForTest) @@ -513,7 +544,10 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by date action performed if sort query keys is not empty") { var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( + SortQueryKeys.NAME, + SortQueryKeys.ASCENDING + ) every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { actionPerformed = true } @@ -526,7 +560,11 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort by date action performed if sort query keys is not empty and contains desired key") { var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DATE, SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( + SortQueryKeys.DATE, + SortQueryKeys.NAME, + SortQueryKeys.ASCENDING + ) every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { actionPerformed = true } @@ -591,7 +629,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort Ascending action performed if selected node is not UssDirNode") { var actionPerformed = false - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { actionPerformed = true listOf(mockedNodeDataNotUssForTest) @@ -605,7 +644,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort Ascending action isSelected if selected node is not UssDirNode") { var isSelected = true - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { isSelected = false listOf(mockedNodeDataNotUssForTest) @@ -644,7 +684,10 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort Ascending action performed if sort query keys is not empty") { var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.NAME, SortQueryKeys.DESCENDING) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( + SortQueryKeys.NAME, + SortQueryKeys.DESCENDING + ) every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { actionPerformed = true } @@ -657,7 +700,11 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort Ascending action performed if sort query keys is not empty and contains desired key") { var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DESCENDING, SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( + SortQueryKeys.DESCENDING, + SortQueryKeys.NAME, + SortQueryKeys.ASCENDING + ) every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { actionPerformed = true } @@ -722,7 +769,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort Descending action performed if selected node is not UssDirNode") { var actionPerformed = false - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { actionPerformed = true listOf(mockedNodeDataNotUssForTest) @@ -736,7 +784,8 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort Descending action isSelected if selected node is not UssDirNode") { var isSelected = true - val mockedNodeDataNotUssForTest = NodeData(mockk(), mockk(), mockk()) + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) every { mockedFileExplorerView.mySelectedNodesData } answers { isSelected = false listOf(mockedNodeDataNotUssForTest) @@ -775,7 +824,10 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort Descending action performed if sort query keys is not empty") { var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( + SortQueryKeys.NAME, + SortQueryKeys.ASCENDING + ) every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { actionPerformed = true } @@ -788,7 +840,11 @@ class UssSortActionHolderTestSpec : ShouldSpec({ should("sort Descending action performed if sort query keys is not empty and contains desired key") { var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.DESCENDING, SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( + SortQueryKeys.DESCENDING, + SortQueryKeys.NAME, + SortQueryKeys.ASCENDING + ) every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { actionPerformed = true } From f801a4cac57de6274bcc92f7e74825c75cff3ce1 Mon Sep 17 00:00:00 2001 From: Arseni Tsikhamirau Date: Wed, 23 Aug 2023 12:37:40 +0300 Subject: [PATCH 14/34] IJMP-822-Bulk-jobs-purge 1) fixed unit tests after merge from release/v1.1.0 --- .../actions/PurgeJobActionTestSpec.kt | 53 +------------------ 1 file changed, 2 insertions(+), 51 deletions(-) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt index c0ebffd2d..b2cd035d4 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobActionTestSpec.kt @@ -34,7 +34,6 @@ import eu.ibagroup.formainframe.dataops.operations.jobs.PurgeJobOperation import eu.ibagroup.formainframe.explorer.Explorer import eu.ibagroup.formainframe.explorer.JesWorkingSetImpl import eu.ibagroup.formainframe.explorer.ui.EXPLORER_VIEW -import eu.ibagroup.formainframe.explorer.ui.ErrorNode import eu.ibagroup.formainframe.explorer.ui.JesExplorerView import eu.ibagroup.formainframe.explorer.ui.JesFilterNode import eu.ibagroup.formainframe.explorer.ui.JesWsNode @@ -201,13 +200,6 @@ class PurgeJobActionTestSpec : WithApplicationShouldSpec({ } - var isOperationSucceededForJesEx = false - every { - explorer.showNotification(any(), any(), NotificationType.INFORMATION, any()) - } answers { - isOperationSucceededForJesEx = true - } - var isOperationSucceededForJobsLog = false every { jobsLogView.showNotification(any(), any(), any(), NotificationType.INFORMATION) @@ -215,22 +207,15 @@ class PurgeJobActionTestSpec : WithApplicationShouldSpec({ isOperationSucceededForJobsLog = true } - var isVisibleInJes = false - every { mockActionEventForJesEx.presentation.isVisible = true } answers { isVisibleInJes = true } - every { job.status } returns mockk() var isEnabledInJobsLog = true every { mockActionEventForJobsLog.presentation.isEnabled = false } answers { isEnabledInJobsLog = false } - purgeAction.actionPerformed(mockActionEventForJesEx) - purgeAction.update(mockActionEventForJesEx) purgeAction.actionPerformed(mockActionEventForJobsLog) purgeAction.update(mockActionEventForJobsLog) assertSoftly { - isOperationSucceededForJesEx shouldBe true isOperationSucceededForJobsLog shouldBe true - isVisibleInJes shouldBe true isEnabledInJobsLog shouldBe true purgeAction.isDumbAware shouldBe true } @@ -238,23 +223,6 @@ class PurgeJobActionTestSpec : WithApplicationShouldSpec({ } should("perform purge on job with error") { - val updateJesAction = mockk() - val jesViewForUpdate = mockk() - every { updateJesAction.getExplorerView() } returns jesViewForUpdate - every { jesViewForUpdate.mySelectedNodesData } returns listOf() - - val updateJesAction2 = mockk() - val jesViewForUpdate2 = mockk() - every { updateJesAction2.getExplorerView() } returns jesViewForUpdate2 - val errorNode = mockk>() - every { jesViewForUpdate2.mySelectedNodesData } returns listOf( - NodeData( - errorNode, - virtualFile, - null - ) - ) - dataOpsManager = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl dataOpsManager.testInstance = object : TestDataOpsManagerImpl(explorer.componentManager) { override fun tryToGetAttributes(file: VirtualFile): FileAttributes { @@ -266,13 +234,6 @@ class PurgeJobActionTestSpec : WithApplicationShouldSpec({ } } - var isOperationFailedForJesEx = false - every { - explorer.showNotification(any(), any(), NotificationType.ERROR, any()) - } answers { - isOperationFailedForJesEx = true - } - var isOperationFailedForJobsLog = false every { jobsLogView.showNotification(any(), any(), any(), NotificationType.ERROR) @@ -291,28 +252,17 @@ class PurgeJobActionTestSpec : WithApplicationShouldSpec({ var isEnabledInJobsLog = true every { mockActionEventForJobsLog.presentation.isEnabled = false } answers { isEnabledInJobsLog = false } - var isVisibleForJes = true - every { updateJesAction.presentation.isVisible = false } answers { isVisibleForJes = false } - var isVisibleForJes2 = true - every { updateJesAction2.presentation.isVisible = false } answers { isVisibleForJes2 = false } - - purgeAction.actionPerformed(mockActionEventForJesEx) purgeAction.actionPerformed(mockActionEventForJobsLog) purgeAction.actionPerformed(mockActionEventWithoutDataContext) purgeAction.update(mockActionEventWithoutDataContext) purgeAction.update(mockActionEventForJobsLog) - purgeAction.update(updateJesAction) - purgeAction.update(updateJesAction2) assertSoftly { - isOperationFailedForJesEx shouldBe true isOperationFailedForJobsLog shouldBe true isEnabledInJobsLog shouldBe false - isVisibleForJes shouldBe false - isVisibleForJes2 shouldBe false - isOperationFailedForNoContextAction shouldBe true } } + unmockkAll() } context("api spec") { @@ -355,6 +305,7 @@ class PurgeJobActionTestSpec : WithApplicationShouldSpec({ every { jobParentParent1.unit } returns mockk() every { jobParentParent1.unit.name } returns "firstWS" + val jobParent2 = mockk() val jobParentParent2 = mockk() every { jobParent2.parent } returns jobParentParent2 From 1ff218e4bdb6b9074f264ba1cdc436498827f753 Mon Sep 17 00:00:00 2001 From: Dzianis Lisiankou Date: Fri, 25 Aug 2023 14:57:53 +0300 Subject: [PATCH 15/34] IJMP-1266: description added --- .../eu/ibagroup/formainframe/explorer/ui/AllocationDialog.kt | 5 +++++ src/main/resources/messages/FMBundle.properties | 2 ++ 2 files changed, 7 insertions(+) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AllocationDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AllocationDialog.kt index f5261a9e0..526d44519 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AllocationDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AllocationDialog.kt @@ -18,6 +18,7 @@ import com.intellij.ui.components.JBScrollPane import com.intellij.ui.dsl.builder.* import com.intellij.ui.dsl.gridLayout.HorizontalAlign import com.intellij.ui.layout.selectedValueMatches +import eu.ibagroup.formainframe.common.message import eu.ibagroup.formainframe.common.ui.StatefulDialog import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.connect.getUsername @@ -111,6 +112,10 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var .bindItem(state.allocationParameters::allocationUnit.toNullableProperty()) .also { spaceUnitBox = it.component } .widthGroup(sameWidthComboBoxGroup) + contextHelp( + description = message("allocation.dialog.unit.size.hint.description"), + title = message("allocation.dialog.unit.size.hint.title") + ) } row { label("Primary allocation: ") diff --git a/src/main/resources/messages/FMBundle.properties b/src/main/resources/messages/FMBundle.properties index cc0ac2007..afe363b32 100755 --- a/src/main/resources/messages/FMBundle.properties +++ b/src/main/resources/messages/FMBundle.properties @@ -25,3 +25,5 @@ encoding.reload.or.convert.dialog.title={0}: Reload or Convert to {1} encoding.reload.or.convert.dialog.message=The encoding you've chosen ('{1}') may change the contents of '{0}'.
Do you want to
1. Reload the file from remote in the new encoding '{1}' and overwrite contents (may not display correctly) or
2. Convert the text and overwrite file in the new encoding?
encoding.reload.dialog.title={0}: Reload to {1} encoding.reload.dialog.message=The encoding you've chosen ('{1}') may change the contents of '{0}'.
Do you want to Reload the file from remote in the new encoding '{1}' and overwrite contents (may not display correctly).
+allocation.dialog.unit.size.hint.description=For IBM 3390 direct access storage device:
1 CYLINDER = 15 TRACKS
1 TRACK = 56664 BYTES +allocation.dialog.unit.size.hint.title=Allocation unit Size From 22e882b8381fc104fc206a8cbe744bb34bcf12d9 Mon Sep 17 00:00:00 2001 From: Dzianis Lisiankou Date: Wed, 13 Sep 2023 18:03:18 +0300 Subject: [PATCH 16/34] IJMP-1363: dialog window after dataset is created changed to suggestion --- .../explorer/actions/AllocateActionBase.kt | 67 +++++++++++-------- 1 file changed, 40 insertions(+), 27 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateActionBase.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateActionBase.kt index 79fd503c7..3ddb05781 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateActionBase.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateActionBase.kt @@ -10,12 +10,14 @@ package eu.ibagroup.formainframe.explorer.actions +import com.intellij.notification.Notification +import com.intellij.notification.NotificationAction +import com.intellij.notification.NotificationType +import com.intellij.notification.Notifications import com.intellij.openapi.actionSystem.AnAction import com.intellij.openapi.actionSystem.AnActionEvent -import com.intellij.openapi.application.runInEdt import com.intellij.openapi.components.service import com.intellij.openapi.progress.runModalTask -import com.intellij.openapi.ui.showOkNoDialog import eu.ibagroup.formainframe.analytics.AnalyticsService import eu.ibagroup.formainframe.analytics.events.FileAction import eu.ibagroup.formainframe.analytics.events.FileEvent @@ -28,6 +30,7 @@ import eu.ibagroup.formainframe.config.ws.FilesWorkingSetConfig import eu.ibagroup.formainframe.dataops.DataOpsManager import eu.ibagroup.formainframe.dataops.operations.DatasetAllocationOperation import eu.ibagroup.formainframe.dataops.operations.DatasetAllocationParams +import eu.ibagroup.formainframe.explorer.ExplorerUnit import eu.ibagroup.formainframe.explorer.FilesWorkingSet import eu.ibagroup.formainframe.explorer.ui.AllocationDialog import eu.ibagroup.formainframe.explorer.ui.DSMaskNode @@ -44,6 +47,8 @@ import org.zowe.kotlinsdk.Dataset import org.zowe.kotlinsdk.DatasetOrganization import org.zowe.kotlinsdk.DsnameType +const val ALLOCATE_ACTION_NOTIFICATION_GROUP_ID = "eu.ibagroup.formainframe.explorer.AllocateActionNotificationGroup" + abstract class AllocateActionBase : AnAction() { /** @@ -140,33 +145,10 @@ abstract class AllocateActionBase : AnAction() { } val nodeToClean = parentProbablyDSMaskNode?.castOrNull>() nodeToClean?.let { cleanInvalidateOnExpand(nodeToClean, view) } + nodeToClean?.cleanCache(recursively = false, cleanBatchedQuery = true) - var nodeCleaned = false - runInEdt { - if ( - showOkNoDialog( - title = "Dataset ${state.datasetName} Has Been Created", - message = "Would you like to add mask \"${state.datasetName}\" to ${workingSet.name}", - project = e.project, - okText = "Yes", - noText = "No" - ) - ) { - val filesWorkingSetConfig = - configCrudable.getByUniqueKey(workingSet.uuid)?.clone() - if (filesWorkingSetConfig != null) { - nodeToClean?.cleanCache(recursively = false, cleanBatchedQuery = true, sendTopic = false) - nodeCleaned = true - - filesWorkingSetConfig.dsMasks.add(DSMask().apply { mask = state.datasetName }) - configCrudable.update(filesWorkingSetConfig) - } - } + showNotification(state, workingSet) - if (!nodeCleaned) { - nodeToClean?.cleanCache(recursively = false, cleanBatchedQuery = true) - } - } initialState.errorMessage = "" } .onFailure { t -> @@ -187,4 +169,35 @@ abstract class AllocateActionBase : AnAction() { override fun isDumbAware(): Boolean { return true } + + + /** + * Shows a notification about successful allocation and suggest adding a mask to the working set + */ + private fun showNotification( + state: DatasetAllocationParams, + workingSet: ExplorerUnit<*> + ) { + val notification = Notification( + ALLOCATE_ACTION_NOTIFICATION_GROUP_ID, + "Dataset ${state.datasetName} has been created", + "Would you like to add mask \"${state.datasetName}\" to ${workingSet.name}?", + NotificationType.INFORMATION + ) + notification.addActions( + setOf( + NotificationAction.createSimpleExpiring("Add mask") { + val filesWorkingSetConfig = + configCrudable.getByUniqueKey(workingSet.uuid)?.clone() + if (filesWorkingSetConfig != null) { + filesWorkingSetConfig.dsMasks.add(DSMask().apply { mask = state.datasetName }) + configCrudable.update(filesWorkingSetConfig) + } + }, + NotificationAction.createSimpleExpiring("Skip") { } + ) + ) + notification.setSuggestionType(true) + Notifications.Bus.notify(notification) + } } From c130ea5202be2b7981d7f7e2065d619a4a7a07df Mon Sep 17 00:00:00 2001 From: Dzianis Lisiankou Date: Wed, 27 Sep 2023 19:01:27 +0300 Subject: [PATCH 17/34] IJMP-1274: query is not removed from cache when cache changes --- .../formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt | 1 - 1 file changed, 1 deletion(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt index 770b27e12..53407a979 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt @@ -233,7 +233,6 @@ abstract class RemoteFileFetchProviderBase, sendTopic: Boolean) { cacheState.remove(query) - cache.remove(query) if (sendTopic) { sendTopic(FileFetchProvider.CACHE_CHANGES, dataOpsManager.componentManager).onCacheCleaned(query) } From 8b3c9c05b5e026dd6885b8c468f13358d36da6db Mon Sep 17 00:00:00 2001 From: Mikita Belabotski Date: Thu, 2 Nov 2023 07:43:16 +0100 Subject: [PATCH 18/34] Feature/ijmp 1217 do not send info request on connection name change --- .../config/connect/ui/zosmf/ConnectionDialog.kt | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt index 9800e4988..cfa004e81 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt @@ -56,6 +56,17 @@ class ConnectionDialog( private val lastSuccessfulState: ConnectionDialogState = if(state.mode == DialogMode.UPDATE) state.connectionConfig.toDialogState(crudable) else ConnectionDialogState() companion object { + /** + * Check if only the connection name has changed in the connection dialog + */ + private fun isOnlyConnectionNameChanged(initialState: ConnectionDialogState, state: ConnectionDialogState): Boolean { + return initialState.connectionName != state.connectionName && + initialState.connectionUrl == state.connectionUrl && + initialState.username == state.username && + initialState.password == state.password && + initialState.isAllowSsl == state.isAllowSsl + } + /** Show Test connection dialog and test the connection regarding the dialog state. * First the method checks whether connection succeeds for specified user/password. * If connection succeeds then the method automatically fill in z/OS version for this connection. @@ -68,12 +79,16 @@ class ConnectionDialog( project: Project? = null, initialState: ConnectionDialogState ): ConnectionDialogState? { + val initState = initialState.clone() return showUntilDone( initialState = initialState, factory = { ConnectionDialog(crudable, initialState, project) }, test = { state -> val newTestedConnConfig : ConnectionConfig if (initialState.mode == DialogMode.UPDATE) { + if (isOnlyConnectionNameChanged(initState, state)) { + return@showUntilDone true + } val newState = state.clone() newState.initEmptyUuids(crudable) newTestedConnConfig = ConnectionConfig(newState.connectionUuid, newState.connectionName, newState.connectionUrl, newState.isAllowSsl, newState.zVersion) From 71410d0aa177f8040796f1c348c824d4dc56e3b9 Mon Sep 17 00:00:00 2001 From: Dzianis Lisiankou Date: Mon, 13 Nov 2023 22:35:34 +0300 Subject: [PATCH 19/34] IJMP-1363: fix tests --- .../actions/AllocateDatasetActionTestSpec.kt | 343 ++++-------------- 1 file changed, 76 insertions(+), 267 deletions(-) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetActionTestSpec.kt index 6beebd61b..b2c52fb4c 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetActionTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetActionTestSpec.kt @@ -10,13 +10,15 @@ package eu.ibagroup.formainframe.explorer.actions +import com.intellij.notification.Notification +import com.intellij.notification.Notifications +import com.intellij.openapi.actionSystem.AnAction import com.intellij.openapi.actionSystem.AnActionEvent import com.intellij.openapi.actionSystem.Presentation import com.intellij.openapi.application.ApplicationManager import com.intellij.openapi.components.ComponentManager import com.intellij.openapi.progress.ProgressIndicator import com.intellij.openapi.project.Project -import com.intellij.openapi.ui.showOkNoDialog import eu.ibagroup.formainframe.analytics.AnalyticsService import eu.ibagroup.formainframe.analytics.events.AnalyticsEvent import eu.ibagroup.formainframe.common.ui.StatefulDialog @@ -53,13 +55,12 @@ import io.mockk.mockk import io.mockk.mockkObject import io.mockk.mockkStatic import io.mockk.unmockkAll -import org.junit.jupiter.api.fail import org.zowe.kotlinsdk.DatasetOrganization import org.zowe.kotlinsdk.DsnameType import java.util.* import javax.swing.Icon -import javax.swing.SwingUtilities import kotlin.reflect.KFunction +import kotlin.reflect.full.declaredFunctions class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ afterSpec { @@ -78,6 +79,7 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ val dataOpsManagerMock = mockk() val componentManagerMock = mockk() val explorerMock = mockk>() + lateinit var addMaskActionInst: AnAction val analyticsService = ApplicationManager.getApplication().service() as TestAnalyticsServiceImpl @@ -118,6 +120,17 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ } answers { isCleanInvalidateOnExpandTriggered = true } + + val notifyRef = Notifications.Bus::class.declaredFunctions + .filter { it.name == "notify" } + .first { it.parameters.size == 1 } + mockkStatic(notifyRef) + mockkStatic(Notification::get) + every { Notifications.Bus.notify(any()) } answers { + val notification = firstArg() + every { Notification.get(any()) } returns notification + addMaskActionInst = notification.actions.first { it.templateText == "Add mask" } + } } afterEach { @@ -133,7 +146,6 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ val dsMaskNodeMock = mockk() lateinit var initState: DatasetAllocationParams var isOperationPerformed = false - var isCleanCacheTriggered = false var isUpdateOnConfigCrudableCalled = false var isShowUntilDoneSucceeded = false @@ -162,15 +174,7 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ configCrudable.getByUniqueKey(any(), any()) } returns Optional.of(filesWorkingSetConfigMock) - every { - dsMaskNodeMock.cleanCache(any(), any(), any(), any()) - } answers { - isCleanCacheTriggered = true - val isSendTopic = lastArg() - if (isSendTopic) { - fail("cleanCache should not send topic in this testcase") - } - } + every { dsMaskNodeMock.cleanCache(any(), any(), any(), any()) } returns Unit every { nodeMock.parent } returns dsMaskNodeMock every { nodeMock.hint(FilesWorkingSet::class).unit } returns workingSetMock every { viewMock.mySelectedNodesData } returns selectedNodesData @@ -189,34 +193,17 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ Optional.of(mockk()) } - val showOkNoDialogMock: ( - String, - String, - Project?, - String, - String, - Icon? - ) -> Boolean = ::showOkNoDialog - mockkStatic(showOkNoDialogMock as KFunction<*>) - every { - hint(Boolean::class) - showOkNoDialogMock(any(), any(), any(), any(), any(), any()) - } answers { - true - } - allocateDsActionInst.actionPerformed(anActionEventMock) + addMaskActionInst.actionPerformed(anActionEventMock) - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe true } - assertSoftly { isShowUntilDoneSucceeded shouldBe true } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isOperationPerformed shouldBe true } - assertSoftly { isCleanCacheTriggered shouldBe true } - assertSoftly { isUpdateOnConfigCrudableCalled shouldBe true } - assertSoftly { isThrowableReported shouldBe false } - assertSoftly { initState.errorMessage shouldBe "" } + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe true + isShowUntilDoneSucceeded shouldBe true + isAnalitycsTracked shouldBe true + isOperationPerformed shouldBe true + isUpdateOnConfigCrudableCalled shouldBe true + isThrowableReported shouldBe false + initState.errorMessage shouldBe "" } } should("perform allocate PS dataset action creating a new dataset mask") { @@ -227,7 +214,6 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ val dsMaskNodeMock = mockk() lateinit var initState: DatasetAllocationParams var isOperationPerformed = false - var isCleanCacheTriggered = false var isUpdateOnConfigCrudableCalled = false var isShowUntilDoneSucceeded = false @@ -263,15 +249,7 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ configCrudable.getByUniqueKey(any(), any()) } returns Optional.of(filesWorkingSetConfigMock) - every { - dsMaskNodeMock.cleanCache(any(), any(), any(), any()) - } answers { - isCleanCacheTriggered = true - val isSendTopic = lastArg() - if (isSendTopic) { - fail("cleanCache should not send topic in this testcase") - } - } + every { dsMaskNodeMock.cleanCache(any(), any(), any(), any()) } returns Unit every { nodeMock.parent } returns dsMaskNodeMock every { nodeMock.hint(FilesWorkingSet::class).unit } returns workingSetMock every { viewMock.mySelectedNodesData } returns selectedNodesData @@ -290,35 +268,18 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ Optional.of(mockk()) } - val showOkNoDialogMock: ( - String, - String, - Project?, - String, - String, - Icon? - ) -> Boolean = ::showOkNoDialog - mockkStatic(showOkNoDialogMock as KFunction<*>) - every { - hint(Boolean::class) - showOkNoDialogMock(any(), any(), any(), any(), any(), any()) - } answers { - true - } - allocateDsActionInst.actionPerformed(anActionEventMock) - - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe true } - assertSoftly { isShowUntilDoneSucceeded shouldBe true } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isOperationPerformed shouldBe true } - assertSoftly { isCleanCacheTriggered shouldBe true } - assertSoftly { isUpdateOnConfigCrudableCalled shouldBe true } - assertSoftly { isThrowableReported shouldBe false } - assertSoftly { initState.errorMessage shouldBe "" } - assertSoftly { initState.allocationParameters.directoryBlocks shouldBe null } + addMaskActionInst.actionPerformed(anActionEventMock) + + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe true + isShowUntilDoneSucceeded shouldBe true + isAnalitycsTracked shouldBe true + isOperationPerformed shouldBe true + isUpdateOnConfigCrudableCalled shouldBe true + isThrowableReported shouldBe false + initState.errorMessage shouldBe "" + initState.allocationParameters.directoryBlocks shouldBe null } } should("perform allocate PO-E dataset action creating a new dataset mask") { @@ -329,7 +290,6 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ val dsMaskNodeMock = mockk() lateinit var initState: DatasetAllocationParams var isOperationPerformed = false - var isCleanCacheTriggered = false var isUpdateOnConfigCrudableCalled = false var isShowUntilDoneSucceeded = false @@ -362,15 +322,7 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ configCrudable.getByUniqueKey(any(), any()) } returns Optional.of(filesWorkingSetConfigMock) - every { - dsMaskNodeMock.cleanCache(any(), any(), any(), any()) - } answers { - isCleanCacheTriggered = true - val isSendTopic = lastArg() - if (isSendTopic) { - fail("cleanCache should not send topic in this testcase") - } - } + every { dsMaskNodeMock.cleanCache(any(), any(), any(), any()) } returns Unit every { nodeMock.parent } returns dsMaskNodeMock every { nodeMock.hint(FilesWorkingSet::class).unit } returns workingSetMock every { viewMock.mySelectedNodesData } returns selectedNodesData @@ -389,36 +341,19 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ Optional.of(mockk()) } - val showOkNoDialogMock: ( - String, - String, - Project?, - String, - String, - Icon? - ) -> Boolean = ::showOkNoDialog - mockkStatic(showOkNoDialogMock as KFunction<*>) - every { - hint(Boolean::class) - showOkNoDialogMock(any(), any(), any(), any(), any(), any()) - } answers { - true - } - allocateDsActionInst.actionPerformed(anActionEventMock) - - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe true } - assertSoftly { isShowUntilDoneSucceeded shouldBe true } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isOperationPerformed shouldBe true } - assertSoftly { isCleanCacheTriggered shouldBe true } - assertSoftly { isUpdateOnConfigCrudableCalled shouldBe true } - assertSoftly { isThrowableReported shouldBe false } - assertSoftly { initState.errorMessage shouldBe "" } - assertSoftly { initState.allocationParameters.datasetOrganization shouldBe DatasetOrganization.PO } - assertSoftly { initState.allocationParameters.dsnType shouldBe DsnameType.LIBRARY } + addMaskActionInst.actionPerformed(anActionEventMock) + + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe true + isShowUntilDoneSucceeded shouldBe true + isAnalitycsTracked shouldBe true + isOperationPerformed shouldBe true + isUpdateOnConfigCrudableCalled shouldBe true + isThrowableReported shouldBe false + initState.errorMessage shouldBe "" + initState.allocationParameters.datasetOrganization shouldBe DatasetOrganization.PO + initState.allocationParameters.dsnType shouldBe DsnameType.LIBRARY } } should("perform allocate dataset action without creating a new dataset mask") { @@ -429,7 +364,6 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ val dsMaskNodeMock = mockk() lateinit var initState: DatasetAllocationParams var isOperationPerformed = false - var isCleanCacheTriggered = false var isUpdateOnConfigCrudableCalled = false var isShowUntilDoneSucceeded = false @@ -458,15 +392,7 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ configCrudable.getByUniqueKey(any(), any()) } returns Optional.of(filesWorkingSetConfigMock) - every { - dsMaskNodeMock.cleanCache(any(), any(), any(), any()) - } answers { - isCleanCacheTriggered = true - val isSendTopic = lastArg() - if (!isSendTopic) { - fail("cleanCache should send topic in this testcase") - } - } + every { dsMaskNodeMock.cleanCache(any(), any(), any(), any()) } returns Unit every { nodeMock.parent } returns dsMaskNodeMock every { nodeMock.hint(FilesWorkingSet::class).unit } returns workingSetMock every { viewMock.mySelectedNodesData } returns selectedNodesData @@ -485,119 +411,19 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ Optional.of(mockk()) } - val showOkNoDialogMock: ( - String, - String, - Project?, - String, - String, - Icon? - ) -> Boolean = ::showOkNoDialog - mockkStatic(showOkNoDialogMock as KFunction<*>) - every { - hint(Boolean::class) - showOkNoDialogMock(any(), any(), any(), any(), any(), any()) - } answers { - false - } - allocateDsActionInst.actionPerformed(anActionEventMock) - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe true } - assertSoftly { isShowUntilDoneSucceeded shouldBe true } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isOperationPerformed shouldBe true } - assertSoftly { isCleanCacheTriggered shouldBe true } - assertSoftly { isUpdateOnConfigCrudableCalled shouldBe false } - assertSoftly { isThrowableReported shouldBe false } - assertSoftly { initState.errorMessage shouldBe "" } + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe true + isShowUntilDoneSucceeded shouldBe true + isAnalitycsTracked shouldBe true + isOperationPerformed shouldBe true + isUpdateOnConfigCrudableCalled shouldBe false + isThrowableReported shouldBe false + initState.errorMessage shouldBe "" } } - should("perform allocate dataset action without refreshing dataset mask as the selected node is a working set") { - val workingSetMock = mockk() - val nodeMock = mockk() - val nodeDataMock = NodeData(nodeMock, null, null) - val selectedNodesData = listOf(nodeDataMock) - lateinit var initState: DatasetAllocationParams - var isOperationPerformed = false - var isUpdateOnConfigCrudableCalled = false - var isShowUntilDoneSucceeded = false - - val showUntilDoneMockk: ( - DatasetAllocationParams, - (DatasetAllocationParams) -> StatefulDialog, - (DatasetAllocationParams) -> Boolean - ) -> DatasetAllocationParams? = ::showUntilDone - mockkStatic(showUntilDoneMockk as KFunction<*>) - every { - hint(DatasetAllocationParams::class) - showUntilDoneMockk( - any(), - any<(DatasetAllocationParams) -> StatefulDialog>(), - any<(DatasetAllocationParams) -> Boolean>() - ) - } answers { - initState = firstArg() - val thirdBlockResult = thirdArg<(DatasetAllocationParams) -> Boolean>() - isShowUntilDoneSucceeded = thirdBlockResult(initState) - initState - } - - mockkObject(configCrudable) - every { - configCrudable.getByUniqueKey(any(), any()) - } returns Optional.of(filesWorkingSetConfigMock) - - every { nodeMock.parent } returns null - every { nodeMock.hint(FilesWorkingSet::class).unit } returns workingSetMock - every { viewMock.mySelectedNodesData } returns selectedNodesData - every { workingSetMock.name } returns "test" - every { workingSetMock.uuid } returns "test" - every { workingSetMock.hint(ConnectionConfig::class).connectionConfig } returns mockk() - every { - dataOpsManagerMock.hint(Boolean::class).performOperation(any>(), any()) - } answers { - isOperationPerformed = true - true - } - every { workingSetMock.explorer } returns explorerMock - every { configCrudable.update(any(), any()) } answers { - isUpdateOnConfigCrudableCalled = true - Optional.of(mockk()) - } - - val showOkNoDialogMock: ( - String, - String, - Project?, - String, - String, - Icon? - ) -> Boolean = ::showOkNoDialog - mockkStatic(showOkNoDialogMock as KFunction<*>) - every { - hint(Boolean::class) - showOkNoDialogMock(any(), any(), any(), any(), any(), any()) - } answers { - true - } - - allocateDsActionInst.actionPerformed(anActionEventMock) - - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe false } - assertSoftly { isShowUntilDoneSucceeded shouldBe true } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isOperationPerformed shouldBe true } - assertSoftly { isUpdateOnConfigCrudableCalled shouldBe true } - assertSoftly { isThrowableReported shouldBe false } - assertSoftly { initState.errorMessage shouldBe "" } - } - } - should("perform allocate dataset action creating new dataset mask without refreshing the existing on as the connection config is not found") { + should("perform allocate dataset action creating new dataset mask without adding as the connection config is not found") { val workingSetMock = mockk() val nodeMock = mockk() val nodeDataMock = NodeData(nodeMock, null, null) @@ -650,33 +476,17 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ Optional.of(mockk()) } - val showOkNoDialogMock: ( - String, - String, - Project?, - String, - String, - Icon? - ) -> Boolean = ::showOkNoDialog - mockkStatic(showOkNoDialogMock as KFunction<*>) - every { - hint(Boolean::class) - showOkNoDialogMock(any(), any(), any(), any(), any(), any()) - } answers { - true - } - allocateDsActionInst.actionPerformed(anActionEventMock) + addMaskActionInst.actionPerformed(anActionEventMock) - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe false } - assertSoftly { isShowUntilDoneSucceeded shouldBe true } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isOperationPerformed shouldBe true } - assertSoftly { isUpdateOnConfigCrudableCalled shouldBe false } - assertSoftly { isThrowableReported shouldBe false } - assertSoftly { initState.errorMessage shouldBe "" } + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe false + isShowUntilDoneSucceeded shouldBe true + isAnalitycsTracked shouldBe true + isOperationPerformed shouldBe true + isUpdateOnConfigCrudableCalled shouldBe false + isThrowableReported shouldBe false + initState.errorMessage shouldBe "" } } should("perform allocate dataset action with failure on operation performing") { @@ -720,13 +530,12 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ allocateDsActionInst.actionPerformed(anActionEventMock) - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe false } - assertSoftly { isShowUntilDoneSucceeded shouldBe false } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isThrowableReported shouldBe true } - assertSoftly { initState.errorMessage shouldBe exceptionMsg } + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe false + isShowUntilDoneSucceeded shouldBe false + isAnalitycsTracked shouldBe true + isThrowableReported shouldBe true + initState.errorMessage shouldBe exceptionMsg } } } From c01926de94b228f09e84f020aadb75fdec7cf30f Mon Sep 17 00:00:00 2001 From: Mikita Belabotski Date: Tue, 14 Nov 2023 14:34:02 +0100 Subject: [PATCH 20/34] IJMP-1367: added: bubble notification when the cancel is clicked during the operation --- .../connect/ui/zosmf/ConnectionDialog.kt | 18 ++++++++---- .../formainframe/explorer/Explorer.kt | 29 +++++++++++++------ .../explorer/actions/AllocateActionBase.kt | 7 +++-- .../explorer/actions/CreateUssEntityAction.kt | 7 +++-- .../explorer/actions/EditJclAction.kt | 9 +++--- .../actions/GetFilePropertiesAction.kt | 11 +++---- .../explorer/actions/MigrationActions.kt | 6 ++-- .../explorer/actions/SubmitJobAction.kt | 5 ++-- .../actions/SubmitJobToolbarAction.kt | 9 +++--- .../actions/TsoSessionCreateAction.kt | 9 +++++- .../resources/messages/FMBundle.properties | 1 + 11 files changed, 72 insertions(+), 39 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt index cfa004e81..4f5a88ea6 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt @@ -12,6 +12,7 @@ package eu.ibagroup.formainframe.config.connect.ui.zosmf import com.intellij.icons.AllIcons import com.intellij.openapi.components.service +import com.intellij.openapi.progress.ProcessCanceledException import com.intellij.openapi.project.Project import com.intellij.openapi.ui.MessageDialogBuilder import com.intellij.openapi.ui.MessageType @@ -20,6 +21,7 @@ import com.intellij.openapi.ui.popup.JBPopupFactory import com.intellij.ui.awt.RelativePoint import com.intellij.ui.dsl.builder.* import com.intellij.ui.dsl.gridLayout.HorizontalAlign +import eu.ibagroup.formainframe.common.message import eu.ibagroup.formainframe.common.ui.DialogMode import eu.ibagroup.formainframe.common.ui.StatefulDialog import eu.ibagroup.formainframe.common.ui.showUntilDone @@ -130,12 +132,16 @@ class ConnectionDialog( if (throwable != null) { state.mode = DialogMode.UPDATE val confirmMessage = "Do you want to add it anyway?" - val tMessage = throwable.message?.let { - if (it.contains("Exception")) { - it.substring(it.lastIndexOf(":") + 2) - .replaceFirstChar { c -> if (c.isLowerCase()) c.titlecase(Locale.getDefault()) else c.toString() } - } else { - it + val tMessage = if (throwable is ProcessCanceledException) { + message("explorer.cancel.by.user.error") + } else { + throwable.message?.let { + if (it.contains("Exception")) { + it.substring(it.lastIndexOf(":") + 2) + .replaceFirstChar { c -> if (c.isLowerCase()) c.titlecase(Locale.getDefault()) else c.toString() } + } else { + it + } } } val message = if (tMessage != null) { diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt index 9933daae6..2d27430cf 100755 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/Explorer.kt @@ -28,6 +28,7 @@ import com.intellij.openapi.project.Project import com.intellij.openapi.ui.Messages import com.intellij.openapi.util.Disposer import com.intellij.util.messages.Topic +import eu.ibagroup.formainframe.common.message import eu.ibagroup.formainframe.config.CONFIGS_CHANGED import eu.ibagroup.formainframe.config.configCrudable import eu.ibagroup.formainframe.config.connect.CREDENTIALS_CHANGED @@ -196,13 +197,15 @@ abstract class AbstractExplorerBase() ?: return val parentNode = view.mySelectedNodesData[0].node + val project = e.project if (parentNode is ExplorerUnitTreeNodeBase<*, *, *> && parentNode.unit is FilesWorkingSet) { val workingSet = parentNode.unit val explorer = workingSet.explorer @@ -120,13 +121,13 @@ abstract class AllocateActionBase : AnAction() { val initialState = preProcessState(datasetInfo) showUntilDone( initialState, - { initState -> AllocationDialog(project = e.project, config, initState) } + { initState -> AllocationDialog(project = project, config, initState) } ) { val state = postProcessState(it) var res = false runModalTask( title = "Allocating Data Set ${state.datasetName}", - project = e.project, + project = project, cancellable = true ) { progressIndicator -> runCatching { @@ -152,7 +153,7 @@ abstract class AllocateActionBase : AnAction() { initialState.errorMessage = "" } .onFailure { t -> - explorer.reportThrowable(t, e.project) + explorer.reportThrowable(t, project) initialState.errorMessage = t.message ?: t.toString() } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/CreateUssEntityAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/CreateUssEntityAction.kt index c1b01764f..be639d8f7 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/CreateUssEntityAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/CreateUssEntityAction.kt @@ -61,6 +61,7 @@ abstract class CreateUssEntityAction : AnAction() { val view = e.getExplorerView() ?: return val selected = view.mySelectedNodesData[0] val selectedNode = selected.node + val project = e.project val node = if (selectedNode is UssFileNode) { selectedNode.parent?.takeIf { it is UssDirNode } } else { @@ -80,7 +81,7 @@ abstract class CreateUssEntityAction : AnAction() { if (filePath != null) { showUntilDone( initialState = fileType.apply { path = filePath }, - { initState -> CreateFileDialog(e.project, state = initState, filePath = filePath) } + { initState -> CreateFileDialog(project, state = initState, filePath = filePath) } ) { var res = false val allocationParams = it.toAllocationParams() @@ -91,7 +92,7 @@ abstract class CreateUssEntityAction : AnAction() { } runModalTask( title = "Creating $fileType ${allocationParams.fileName}", - project = e.project, + project = project, cancellable = true ) { indicator -> val ussDirNode = node.castOrNull() @@ -122,7 +123,7 @@ abstract class CreateUssEntityAction : AnAction() { ussDirNode?.cleanCache(false) res = true }.onFailure { t -> - view.explorer.reportThrowable(t, e.project) + view.explorer.reportThrowable(t, project) } } res diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/EditJclAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/EditJclAction.kt index a140b185a..d78baf3d6 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/EditJclAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/EditJclAction.kt @@ -47,6 +47,7 @@ class EditJclAction : AnAction() { val view = e.getExplorerView() ?: return val selected = view.mySelectedNodesData.getOrNull(0) val node = selected?.node + val project = e.project if (node is JobNode) { val connectionConfig = node.unit.connectionConfig if (connectionConfig != null) { @@ -54,7 +55,7 @@ class EditJclAction : AnAction() { val dataOpsManager = service() runBackgroundableTask( title = "Get JCL records for job ${attributes.jobInfo.jobName}", - project = e.project, + project = project, cancellable = true ) { runCatching { @@ -78,10 +79,10 @@ class EditJclAction : AnAction() { cachedFile = createdFile wasCreatedBefore = false } - val descriptor = e.project?.let { pr -> OpenFileDescriptor(pr, cachedFile) } + val descriptor = project?.let { pr -> OpenFileDescriptor(pr, cachedFile) } descriptor?.let { val syncProvider = - DocumentedSyncProvider(file = cachedFile, saveStrategy = SaveStrategy.default(e.project)) + DocumentedSyncProvider(file = cachedFile, saveStrategy = SaveStrategy.default(project)) if (!wasCreatedBefore) { syncProvider.putInitialContent(jclContentBytes) changeFileEncodingTo(cachedFile, DEFAULT_TEXT_CHARSET) @@ -97,7 +98,7 @@ class EditJclAction : AnAction() { } } }.onFailure { - node.explorer.reportThrowable(it, e.project) + node.explorer.reportThrowable(it, project) } } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/GetFilePropertiesAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/GetFilePropertiesAction.kt index 39aded898..1931626c8 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/GetFilePropertiesAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/GetFilePropertiesAction.kt @@ -35,6 +35,7 @@ class GetFilePropertiesAction : AnAction() { override fun actionPerformed(e: AnActionEvent) { val view = e.getExplorerView() ?: return val node = view.mySelectedNodesData.getOrNull(0)?.node ?: return + val project = e.project if (node is ExplorerUnitTreeNodeBase>) { val virtualFile = node.virtualFile val connectionConfig = node.unit.connectionConfig ?: return @@ -42,19 +43,19 @@ class GetFilePropertiesAction : AnAction() { val dataOpsManager = node.explorer.componentManager.service() when (val attributes = dataOpsManager.tryToGetAttributes(virtualFile)) { is RemoteDatasetAttributes -> { - val dialog = DatasetPropertiesDialog(e.project, DatasetState(attributes)) + val dialog = DatasetPropertiesDialog(project, DatasetState(attributes)) dialog.showAndGet() } is RemoteUssAttributes -> { val oldCharset = attributes.charset val initFileMode = attributes.fileMode?.clone() - val dialog = UssFilePropertiesDialog(e.project, UssFileState(attributes, virtualFile.isBeingEditingNow())) + val dialog = UssFilePropertiesDialog(project, UssFileState(attributes, virtualFile.isBeingEditingNow())) if (dialog.showAndGet()) { if (attributes.fileMode?.owner != initFileMode?.owner || attributes.fileMode?.group != initFileMode?.group || attributes.fileMode?.all != initFileMode?.all) { runBackgroundableTask( title = "Changing file mode on ${attributes.path}", - project = e.project, + project = project, cancellable = true ) { if (attributes.fileMode != null) { @@ -67,7 +68,7 @@ class GetFilePropertiesAction : AnAction() { progressIndicator = it ) }.onFailure { t -> - view.explorer.reportThrowable(t, e.project) + view.explorer.reportThrowable(t, project) } node.parent?.cleanCacheIfPossible(cleanBatchedQuery = false) } @@ -86,7 +87,7 @@ class GetFilePropertiesAction : AnAction() { } is RemoteMemberAttributes -> { - val dialog = MemberPropertiesDialog(e.project, MemberState(attributes)) + val dialog = MemberPropertiesDialog(project, MemberState(attributes)) dialog.showAndGet() } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/MigrationActions.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/MigrationActions.kt index fa062d54b..2d9fa87a2 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/MigrationActions.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/MigrationActions.kt @@ -66,6 +66,7 @@ class RecallAction : DumbAwareAction() { */ override fun actionPerformed(e: AnActionEvent) { val view = e.getExplorerView() + val project = e.project if (view != null) { val triples = view.mySelectedNodesData.mapNotNull { getRequestDataForNode(it.node) } val operations: List = triples.map { @@ -85,7 +86,7 @@ class RecallAction : DumbAwareAction() { ) } }.onFailure { - view.explorer.reportThrowable(it, e.project) + view.explorer.reportThrowable(it, project) } } makeUniqueCacheClean(view.mySelectedNodesData.map { it.node }) @@ -122,6 +123,7 @@ class MigrateAction : DumbAwareAction() { */ override fun actionPerformed(e: AnActionEvent) { val view = e.getExplorerView() + val project = e.project if (view != null) { val triples = view.mySelectedNodesData.mapNotNull { getRequestDataForNode(it.node) } val operations: List = triples.map { @@ -141,7 +143,7 @@ class MigrateAction : DumbAwareAction() { ) } }.onFailure { - view.explorer.reportThrowable(it, e.project) + view.explorer.reportThrowable(it, project) } } makeUniqueCacheClean(view.mySelectedNodesData.map { it.node }) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/SubmitJobAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/SubmitJobAction.kt index 2ddd4fb6c..dbe80c748 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/SubmitJobAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/SubmitJobAction.kt @@ -45,6 +45,7 @@ class SubmitJobAction : AnAction() { e.presentation.isEnabledAndVisible = false return } + val project = e.project val selected = view.mySelectedNodesData val node = selected.getOrNull(0)?.node ?: return @@ -77,9 +78,9 @@ class SubmitJobAction : AnAction() { } } }.onSuccess { - view.explorer.showNotification("Job ${it.jobname} has been submitted", "$it", project = e.project) + view.explorer.showNotification("Job ${it.jobname} has been submitted", "$it", project = project) }.onFailure { - view.explorer.reportThrowable(it, e.project) + view.explorer.reportThrowable(it, project) } } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/SubmitJobToolbarAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/SubmitJobToolbarAction.kt index 9ea7bf704..cfb7221bc 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/SubmitJobToolbarAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/SubmitJobToolbarAction.kt @@ -32,7 +32,8 @@ class SubmitJobToolbarAction: AnAction() { * Opens job log */ override fun actionPerformed(e: AnActionEvent) { - val explorerView = e.project?.let { FileExplorerContentProvider.getInstance().getExplorerView(it) } + val project = e.project + val explorerView = project?.let { FileExplorerContentProvider.getInstance().getExplorerView(it) } val editor = e.getData(CommonDataKeys.EDITOR) ?: return val file = e.getData(CommonDataKeys.VIRTUAL_FILE) ?: return val jclContent = editor.document.text @@ -42,7 +43,7 @@ class SubmitJobToolbarAction: AnAction() { val connectionConfig = parentAttributes.requesters[0].connectionConfig runBackgroundableTask( title = "Submitting job from file ${file.name}", - project = e.project, + project = project, cancellable = true ) { runCatching { @@ -53,12 +54,12 @@ class SubmitJobToolbarAction: AnAction() { ), progressIndicator = it ).also { result -> - e.project?.let { project -> + project?.let { project -> sendTopic(JOB_ADDED_TOPIC).submitted(project, connectionConfig, file.parent.path, result) } } }.onFailure { - explorerView?.explorer?.reportThrowable(it, e.project) + explorerView?.explorer?.reportThrowable(it, project) } } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/TsoSessionCreateAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/TsoSessionCreateAction.kt index 5e9c38974..470000efc 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/TsoSessionCreateAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/TsoSessionCreateAction.kt @@ -13,9 +13,11 @@ package eu.ibagroup.formainframe.explorer.actions import com.intellij.openapi.actionSystem.AnAction import com.intellij.openapi.actionSystem.AnActionEvent import com.intellij.openapi.components.service +import com.intellij.openapi.progress.ProcessCanceledException import com.intellij.openapi.project.ProjectManager import com.intellij.openapi.ui.Messages import com.intellij.util.containers.isEmpty +import eu.ibagroup.formainframe.common.message import eu.ibagroup.formainframe.common.ui.showUntilDone import eu.ibagroup.formainframe.config.configCrudable import eu.ibagroup.formainframe.config.connect.ConnectionConfig @@ -59,7 +61,12 @@ class TsoSessionCreateAction : AnAction() { } } if (throwable != null) { - val errorTemplate = "An error occurred. See details below:\n $throwable" + val tMessage = if (throwable is ProcessCanceledException) { + message("explorer.cancel.by.user.error") + } else { + "${throwable.message}" + } + val errorTemplate = "An error occurred. See details below:\n $tMessage" Messages.showErrorDialog( project, errorTemplate, diff --git a/src/main/resources/messages/FMBundle.properties b/src/main/resources/messages/FMBundle.properties index afe363b32..bbf7cda37 100755 --- a/src/main/resources/messages/FMBundle.properties +++ b/src/main/resources/messages/FMBundle.properties @@ -21,6 +21,7 @@ configurable.ws.tables.actions.connection.selected.fix.url=Fix z/OSMF connection analytics.agreement.text=By {0} “I agree” the User agrees to collection of statistical information via analytics-java – a Java client for Segment. The terms and conditions of such collection are described in Privacy Policy set out above configurable.ws.dialog.title.add="Add Working Set" title.error="Error" +explorer.cancel.by.user.error=Error during operation execution: operation cancelled by user encoding.reload.or.convert.dialog.title={0}: Reload or Convert to {1} encoding.reload.or.convert.dialog.message=The encoding you've chosen ('{1}') may change the contents of '{0}'.
Do you want to
1. Reload the file from remote in the new encoding '{1}' and overwrite contents (may not display correctly) or
2. Convert the text and overwrite file in the new encoding?
encoding.reload.dialog.title={0}: Reload to {1} From 4421699727d5dcae6fbeeb03459f3857a1ab2493 Mon Sep 17 00:00:00 2001 From: Dzianis Lisiankou Date: Thu, 16 Nov 2023 21:19:59 +0300 Subject: [PATCH 21/34] IJMP-1363: improve tests --- .../formainframe/editor/EditorTestSpec.kt | 19 +- .../actions/AllocateDatasetActionTestSpec.kt | 7 +- .../actions/AllocateLikeActionTestSpec.kt | 204 +++++------------- .../ui/ChangeEncodingDialogTestSpec.kt | 7 +- .../utils/EncodingUtilsTestSpec.kt | 7 +- 5 files changed, 67 insertions(+), 177 deletions(-) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/editor/EditorTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/editor/EditorTestSpec.kt index cc01e4d68..1c64264f9 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/editor/EditorTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/editor/EditorTestSpec.kt @@ -75,7 +75,6 @@ import java.nio.charset.CoderResult import javax.swing.JComponent import javax.swing.SwingUtilities import kotlin.reflect.KFunction -import kotlin.reflect.full.declaredFunctions class EditorTestSpec : WithApplicationShouldSpec({ afterSpec { @@ -300,10 +299,8 @@ class EditorTestSpec : WithApplicationShouldSpec({ val decoderResultMock = mockk() every { decoderResultMock.length() } returns 1 - val getLastItemRef = ContainerUtil::class.declaredFunctions - .filter { it.name == "getLastItem" } - .first { it.parameters.size == 2 } - mockkStatic(getLastItemRef) + val getLastItemRef: (MutableList) -> Any = ContainerUtil::getLastItem + mockkStatic(getLastItemRef as KFunction<*>) every { ContainerUtil.getLastItem(any>()) } answers { val descriptors = firstArg>() if (descriptors.isNotEmpty()) descriptors.last() else null @@ -502,10 +499,8 @@ class EditorTestSpec : WithApplicationShouldSpec({ decoderResultMock } - val commonPrefixLengthRef = StringUtil::class.declaredFunctions - .filter { it.name == "commonPrefixLength" } - .first { it.parameters.size == 2 } - mockkStatic(commonPrefixLengthRef) + val commonPrefixLengthRef: (CharSequence, CharSequence) -> Int = StringUtil::commonPrefixLength + mockkStatic(commonPrefixLengthRef as KFunction<*>) every { StringUtil.commonPrefixLength(any(), any()) } returns text.length val descriptors = lossyEncodingInspection.checkFile(psiFileMock, inspectionManagerMock, isOnTheFly) @@ -621,10 +616,8 @@ class EditorTestSpec : WithApplicationShouldSpec({ decoderResultMock } - val commonPrefixLengthRef = StringUtil::class.declaredFunctions - .filter { it.name == "commonPrefixLength" } - .first { it.parameters.size == 2 } - mockkStatic(commonPrefixLengthRef) + val commonPrefixLengthRef: (CharSequence, CharSequence) -> Int = StringUtil::commonPrefixLength + mockkStatic(commonPrefixLengthRef as KFunction<*>) every { StringUtil.commonPrefixLength(any(), any()) } returns text.length - 1 val descriptors = lossyEncodingInspection.checkFile(psiFileMock, inspectionManagerMock, isOnTheFly) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetActionTestSpec.kt index b2c52fb4c..b98b68d17 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetActionTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateDatasetActionTestSpec.kt @@ -60,7 +60,6 @@ import org.zowe.kotlinsdk.DsnameType import java.util.* import javax.swing.Icon import kotlin.reflect.KFunction -import kotlin.reflect.full.declaredFunctions class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ afterSpec { @@ -121,10 +120,8 @@ class AllocateDatasetActionTestSpec : WithApplicationShouldSpec({ isCleanInvalidateOnExpandTriggered = true } - val notifyRef = Notifications.Bus::class.declaredFunctions - .filter { it.name == "notify" } - .first { it.parameters.size == 1 } - mockkStatic(notifyRef) + val notifyRef: (Notification) -> Unit = Notifications.Bus::notify + mockkStatic(notifyRef as KFunction<*>) mockkStatic(Notification::get) every { Notifications.Bus.notify(any()) } answers { val notification = firstArg() diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateLikeActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateLikeActionTestSpec.kt index a92bb88d5..c0bc04312 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateLikeActionTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/AllocateLikeActionTestSpec.kt @@ -10,6 +10,9 @@ package eu.ibagroup.formainframe.explorer.actions +import com.intellij.notification.Notification +import com.intellij.notification.Notifications +import com.intellij.openapi.actionSystem.AnAction import com.intellij.openapi.actionSystem.AnActionEvent import com.intellij.openapi.actionSystem.Presentation import com.intellij.openapi.application.ApplicationManager @@ -17,7 +20,6 @@ import com.intellij.openapi.components.ComponentManager import com.intellij.openapi.progress.ProgressIndicator import com.intellij.openapi.project.Project import com.intellij.openapi.ui.Messages.showWarningDialog -import com.intellij.openapi.ui.showOkNoDialog import eu.ibagroup.formainframe.analytics.AnalyticsService import eu.ibagroup.formainframe.analytics.events.AnalyticsEvent import eu.ibagroup.formainframe.common.ui.StatefulDialog @@ -54,14 +56,12 @@ import io.mockk.mockk import io.mockk.mockkObject import io.mockk.mockkStatic import io.mockk.unmockkAll -import org.junit.jupiter.api.fail import org.zowe.kotlinsdk.Dataset import org.zowe.kotlinsdk.DatasetOrganization import org.zowe.kotlinsdk.RecordFormat import org.zowe.kotlinsdk.SpaceUnits import java.util.* import javax.swing.Icon -import javax.swing.SwingUtilities import kotlin.reflect.KFunction class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ @@ -80,6 +80,7 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ val dataOpsManagerMock = mockk() val componentManagerMock = mockk() val explorerMock = mockk>() + lateinit var addMaskActionInst: AnAction val analyticsService = ApplicationManager.getApplication().service() as TestAnalyticsServiceImpl @@ -119,6 +120,15 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ } answers { isCleanInvalidateOnExpandTriggered = true } + + val notifyRef: (Notification) -> Unit = Notifications.Bus::notify + mockkStatic(notifyRef as KFunction<*>) + mockkStatic(Notification::get) + every { Notifications.Bus.notify(any()) } answers { + val notification = firstArg() + every { Notification.get(any()) } returns notification + addMaskActionInst = notification.actions.first { it.templateText == "Add mask" } + } } afterEach { @@ -137,7 +147,6 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ val dsMaskNodeMock = mockk() lateinit var initState: DatasetAllocationParams var isOperationPerformed = false - var isCleanCacheTriggered = false var isShowUntilDoneSucceeded = false every { anActionEventMock.getExplorerView() } returns viewMock @@ -168,15 +177,7 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ configCrudable.getByUniqueKey(any(), any()) } returns Optional.of(filesWorkingSetConfigMock) - every { - dsMaskNodeMock.cleanCache(any(), any(), any(), any()) - } answers { - isCleanCacheTriggered = true - val isSendTopic = lastArg() - if (!isSendTopic) { - fail("cleanCache should send topic in this testcase") - } - } + every { dsMaskNodeMock.cleanCache(any(), any(), any(), any()) } returns Unit every { nodeMock.parent } returns dsMaskNodeMock every { nodeMock.hint(FilesWorkingSet::class).unit } returns workingSetMock every { viewMock.mySelectedNodesData } returns selectedNodesData @@ -191,33 +192,16 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ } every { workingSetMock.explorer } returns explorerMock - val showOkNoDialogMock: ( - String, - String, - Project?, - String, - String, - Icon? - ) -> Boolean = ::showOkNoDialog - mockkStatic(showOkNoDialogMock as KFunction<*>) - every { - hint(Boolean::class) - showOkNoDialogMock(any(), any(), any(), any(), any(), any()) - } answers { - false - } - allocateDsActionInst.actionPerformed(anActionEventMock) - - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe true } - assertSoftly { isShowUntilDoneSucceeded shouldBe true } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isOperationPerformed shouldBe true } - assertSoftly { isCleanCacheTriggered shouldBe true } - assertSoftly { isThrowableReported shouldBe false } - assertSoftly { initState.errorMessage shouldBe "" } + addMaskActionInst.actionPerformed(anActionEventMock) + + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe true + isShowUntilDoneSucceeded shouldBe true + isAnalitycsTracked shouldBe true + isOperationPerformed shouldBe true + isThrowableReported shouldBe false + initState.errorMessage shouldBe "" } } should("perform allocate PDS dataset with TRACKS action without creating a new dataset mask") { @@ -236,7 +220,6 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ val dsMaskNodeMock = mockk() lateinit var initState: DatasetAllocationParams var isOperationPerformed = false - var isCleanCacheTriggered = false var isShowUntilDoneSucceeded = false every { anActionEventMock.getExplorerView() } returns viewMock @@ -267,15 +250,7 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ configCrudable.getByUniqueKey(any(), any()) } returns Optional.of(filesWorkingSetConfigMock) - every { - dsMaskNodeMock.cleanCache(any(), any(), any(), any()) - } answers { - isCleanCacheTriggered = true - val isSendTopic = lastArg() - if (!isSendTopic) { - fail("cleanCache should send topic in this testcase") - } - } + every { dsMaskNodeMock.cleanCache(any(), any(), any(), any()) } returns Unit every { nodeMock.parent } returns dsMaskNodeMock every { nodeMock.hint(FilesWorkingSet::class).unit } returns workingSetMock every { viewMock.mySelectedNodesData } returns selectedNodesData @@ -290,33 +265,15 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ } every { workingSetMock.explorer } returns explorerMock - val showOkNoDialogMock: ( - String, - String, - Project?, - String, - String, - Icon? - ) -> Boolean = ::showOkNoDialog - mockkStatic(showOkNoDialogMock as KFunction<*>) - every { - hint(Boolean::class) - showOkNoDialogMock(any(), any(), any(), any(), any(), any()) - } answers { - false - } - allocateDsActionInst.actionPerformed(anActionEventMock) - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe true } - assertSoftly { isShowUntilDoneSucceeded shouldBe true } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isOperationPerformed shouldBe true } - assertSoftly { isCleanCacheTriggered shouldBe true } - assertSoftly { isThrowableReported shouldBe false } - assertSoftly { initState.errorMessage shouldBe "" } + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe true + isShowUntilDoneSucceeded shouldBe true + isAnalitycsTracked shouldBe true + isOperationPerformed shouldBe true + isThrowableReported shouldBe false + initState.errorMessage shouldBe "" } } should("perform allocate PDS/E dataset with CYLINDERS action without creating a new dataset mask") { @@ -335,7 +292,6 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ val dsMaskNodeMock = mockk() lateinit var initState: DatasetAllocationParams var isOperationPerformed = false - var isCleanCacheTriggered = false var isShowUntilDoneSucceeded = false every { anActionEventMock.getExplorerView() } returns viewMock @@ -366,15 +322,7 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ configCrudable.getByUniqueKey(any(), any()) } returns Optional.of(filesWorkingSetConfigMock) - every { - dsMaskNodeMock.cleanCache(any(), any(), any(), any()) - } answers { - isCleanCacheTriggered = true - val isSendTopic = lastArg() - if (!isSendTopic) { - fail("cleanCache should send topic in this testcase") - } - } + every { dsMaskNodeMock.cleanCache(any(), any(), any(), any()) } returns Unit every { nodeMock.parent } returns dsMaskNodeMock every { nodeMock.hint(FilesWorkingSet::class).unit } returns workingSetMock every { viewMock.mySelectedNodesData } returns selectedNodesData @@ -389,33 +337,15 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ } every { workingSetMock.explorer } returns explorerMock - val showOkNoDialogMock: ( - String, - String, - Project?, - String, - String, - Icon? - ) -> Boolean = ::showOkNoDialog - mockkStatic(showOkNoDialogMock as KFunction<*>) - every { - hint(Boolean::class) - showOkNoDialogMock(any(), any(), any(), any(), any(), any()) - } answers { - false - } - allocateDsActionInst.actionPerformed(anActionEventMock) - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe true } - assertSoftly { isShowUntilDoneSucceeded shouldBe true } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isOperationPerformed shouldBe true } - assertSoftly { isCleanCacheTriggered shouldBe true } - assertSoftly { isThrowableReported shouldBe false } - assertSoftly { initState.errorMessage shouldBe "" } + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe true + isShowUntilDoneSucceeded shouldBe true + isAnalitycsTracked shouldBe true + isOperationPerformed shouldBe true + isThrowableReported shouldBe false + initState.errorMessage shouldBe "" } } should("perform allocate PS dataset with BLOCKS action without creating a new dataset mask, changing BLOCKS to TRACKS") { @@ -429,7 +359,6 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ val dsMaskNodeMock = mockk() lateinit var initState: DatasetAllocationParams var isOperationPerformed = false - var isCleanCacheTriggered = false var isShowUntilDoneSucceeded = false var isBlocksChangedToTracks = false @@ -470,15 +399,7 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ configCrudable.getByUniqueKey(any(), any()) } returns Optional.of(filesWorkingSetConfigMock) - every { - dsMaskNodeMock.cleanCache(any(), any(), any(), any()) - } answers { - isCleanCacheTriggered = true - val isSendTopic = lastArg() - if (!isSendTopic) { - fail("cleanCache should send topic in this testcase") - } - } + every { dsMaskNodeMock.cleanCache(any(), any(), any(), any()) } returns Unit every { nodeMock.parent } returns dsMaskNodeMock every { nodeMock.hint(FilesWorkingSet::class).unit } returns workingSetMock every { viewMock.mySelectedNodesData } returns selectedNodesData @@ -493,34 +414,16 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ } every { workingSetMock.explorer } returns explorerMock - val showOkNoDialogMock: ( - String, - String, - Project?, - String, - String, - Icon? - ) -> Boolean = ::showOkNoDialog - mockkStatic(showOkNoDialogMock as KFunction<*>) - every { - hint(Boolean::class) - showOkNoDialogMock(any(), any(), any(), any(), any(), any()) - } answers { - false - } - allocateDsActionInst.actionPerformed(anActionEventMock) - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe true } - assertSoftly { isShowUntilDoneSucceeded shouldBe true } - assertSoftly { isAnalitycsTracked shouldBe true } - assertSoftly { isOperationPerformed shouldBe true } - assertSoftly { isCleanCacheTriggered shouldBe true } - assertSoftly { isBlocksChangedToTracks shouldBe true } - assertSoftly { isThrowableReported shouldBe false } - assertSoftly { initState.errorMessage shouldBe "" } + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe true + isShowUntilDoneSucceeded shouldBe true + isAnalitycsTracked shouldBe true + isOperationPerformed shouldBe true + isBlocksChangedToTracks shouldBe true + isThrowableReported shouldBe false + initState.errorMessage shouldBe "" } } should("not perform 'allocate like' action as the file explorer view is not found") { @@ -537,12 +440,11 @@ class AllocateLikeActionTestSpec : WithApplicationShouldSpec({ allocateDsActionInst.actionPerformed(anActionEventMock) - // Pause to wait until all EDT events are finished - SwingUtilities.invokeAndWait { - assertSoftly { isCleanInvalidateOnExpandTriggered shouldBe false } - assertSoftly { isAnalitycsTracked shouldBe false } - assertSoftly { isOperationPerformed shouldBe false } - assertSoftly { isThrowableReported shouldBe false } + assertSoftly { + isCleanInvalidateOnExpandTriggered shouldBe false + isAnalitycsTracked shouldBe false + isOperationPerformed shouldBe false + isThrowableReported shouldBe false } } } diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/ChangeEncodingDialogTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/ChangeEncodingDialogTestSpec.kt index 16504f004..4b7bc0a4e 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/ChangeEncodingDialogTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/ChangeEncodingDialogTestSpec.kt @@ -47,7 +47,7 @@ import java.awt.event.ActionEvent import java.nio.charset.Charset import javax.swing.Action import javax.swing.Icon -import kotlin.reflect.full.declaredFunctions +import kotlin.reflect.KFunction class ChangeEncodingDialogTestSpec : WithApplicationShouldSpec({ afterSpec { @@ -91,9 +91,8 @@ class ChangeEncodingDialogTestSpec : WithApplicationShouldSpec({ val actionEventMock = mockk() - val showDialogRef = Messages::class.declaredFunctions - .first { it.name == "showDialog" && it.parameters.size == 5 } - mockkStatic(showDialogRef) + val showDialogRef: (String, String, Array, Int, Icon) -> Int = Messages::showDialog + mockkStatic(showDialogRef as KFunction<*>) every { contentSynchronizerMock.synchronizeWithRemote(any()) } returns Unit diff --git a/src/test/kotlin/eu/ibagroup/formainframe/utils/EncodingUtilsTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/utils/EncodingUtilsTestSpec.kt index 8ddc53ed7..70884ba83 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/utils/EncodingUtilsTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/utils/EncodingUtilsTestSpec.kt @@ -53,7 +53,7 @@ import java.nio.charset.CharsetDecoder import java.nio.charset.CharsetEncoder import java.nio.charset.UnsupportedCharsetException import javax.swing.Icon -import kotlin.reflect.full.declaredFunctions +import kotlin.reflect.KFunction class EncodingUtilsTestSpec : WithApplicationShouldSpec({ context("utils module: encodingUtils") { @@ -140,9 +140,8 @@ class EncodingUtilsTestSpec : WithApplicationShouldSpec({ mockkStatic(InspectionEngine::runInspectionOnFile) - val showDialogRef = Messages::class.declaredFunctions - .first { it.name == "showDialog" && it.parameters.size == 5 } - mockkStatic(showDialogRef) + val showDialogRef: (String, String, Array, Int, Icon) -> Int = Messages::showDialog + mockkStatic(showDialogRef as KFunction<*>) beforeEach { dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(explorerMock.componentManager) { From 48834cacef0626fd3d36039247b00aeba83105e2 Mon Sep 17 00:00:00 2001 From: Dzianis Lisiankou Date: Fri, 17 Nov 2023 11:40:14 +0100 Subject: [PATCH 22/34] IJMP-1428: implement cancel TSO command --- .../common/SettingsPropertyManager.kt | 31 ++++++++++ .../dataops/operations/TsoOperationRunner.kt | 57 ++++++++++++++++--- .../ui/build/TerminalCommandReceiver.kt | 37 +++++++++++- .../ui/build/tso/TSOWindowFactory.kt | 2 +- .../ui/build/tso/ui/TSOConsoleView.kt | 54 +++++++++++++++--- src/main/resources/settings.properties | 1 + .../formainframe/common/CommonTestSpec.kt | 52 ++++++++++++++++- 7 files changed, 213 insertions(+), 21 deletions(-) create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/common/SettingsPropertyManager.kt create mode 100644 src/main/resources/settings.properties diff --git a/src/main/kotlin/eu/ibagroup/formainframe/common/SettingsPropertyManager.kt b/src/main/kotlin/eu/ibagroup/formainframe/common/SettingsPropertyManager.kt new file mode 100644 index 000000000..9aced3e81 --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/common/SettingsPropertyManager.kt @@ -0,0 +1,31 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.common + +import java.util.* + +class SettingsPropertyManager + +/** + * Properties from the settings.properties file + */ +internal val settingsProperties by lazy { + Properties().apply { + load(SettingsPropertyManager::class.java.classLoader.getResourceAsStream("settings.properties")) + } +} + +/** + * Check if the debug mode is enabled + */ +fun isDebugModeEnabled(): Boolean { + return settingsProperties.getProperty("debug.mode")?.toBoolean() ?: false +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/operations/TsoOperationRunner.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/operations/TsoOperationRunner.kt index 8c5c64edd..b6a7d8803 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/operations/TsoOperationRunner.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/operations/TsoOperationRunner.kt @@ -19,14 +19,13 @@ import eu.ibagroup.formainframe.ui.build.tso.config.TSOConfigWrapper import eu.ibagroup.formainframe.ui.build.tso.ui.TSOSessionParams import eu.ibagroup.formainframe.utils.cancelByIndicator import eu.ibagroup.formainframe.utils.log +import eu.ibagroup.formainframe.dataops.operations.MessageType as MessageTypeEnum import org.zowe.kotlinsdk.MessageType import org.zowe.kotlinsdk.TsoApi import org.zowe.kotlinsdk.TsoData import org.zowe.kotlinsdk.TsoResponse import io.ktor.util.* import retrofit2.Response -import java.nio.charset.Charset -import java.util.* /** * Factory class which represents a TSO operation runner. Defined in plugin.xml @@ -85,12 +84,7 @@ class TsoOperationRunner : OperationRunner { response = api(state.getConnectionConfig()) .sendMessageToTso( state.getConnectionConfig().authToken, - body = TsoData( - tsoResponse = MessageType( - version = "0100", - data = operation.message - ) - ), + body = createTsoData(operation), servletKey = servletKey ) .cancelByIndicator(progressIndicator) @@ -140,4 +134,51 @@ class TsoOperationRunner : OperationRunner { return response?.body() ?: throw Exception("Cannot retrieve response from server.") } + /** + * Create TsoData object depending on the specified message type + * @throws Exception if message type not specified + */ + private fun createTsoData(operation: TsoOperation): TsoData { + return when (operation.messageType) { + MessageTypeEnum.TSO_MESSAGE -> TsoData( + tsoMessage = createMessageType(operation) + ) + + MessageTypeEnum.TSO_PROMPT -> TsoData( + tsoPrompt = createMessageType(operation) + ) + + MessageTypeEnum.TSO_RESPONSE -> TsoData( + tsoResponse = createMessageType(operation) + ) + + null -> throw Exception("Message type not specified") + } + } + + /** + * Create MessageType object depending on the specified message data + * @throws Exception if message data not specified + */ + private fun createMessageType(operation: TsoOperation): MessageType { + return when (operation.messageData) { + MessageData.DATA_DATA -> MessageType( + version = "0100", + data = operation.message + ) + + MessageData.DATA_HIDDEN -> MessageType( + version = "0100", + hidden = operation.message + ) + + MessageData.DATA_ACTION -> MessageType( + version = "0100", + action = operation.message + ) + + null -> throw Exception("Message data not specified") + } + } + } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/TerminalCommandReceiver.kt b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/TerminalCommandReceiver.kt index 3d44aea58..de07b5b9d 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/TerminalCommandReceiver.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/TerminalCommandReceiver.kt @@ -13,6 +13,7 @@ package eu.ibagroup.formainframe.ui.build import com.intellij.execution.process.NopProcessHandler import com.intellij.execution.process.ProcessHandler import com.intellij.execution.process.ProcessOutputType +import com.intellij.openapi.util.Key import com.intellij.terminal.TerminalExecutionConsole import com.jediterm.terminal.TerminalKeyEncoder import eu.ibagroup.formainframe.ui.build.tso.utils.InputRecognizer @@ -33,6 +34,7 @@ class TerminalCommandReceiver(terminalConsole: TerminalExecutionConsole) { private var needToWaitForCommandInput = false private var commandsInQueue: Queue = LinkedList() private var expectParameters = false + private var prevCommandEndsWithReady = false var initialized = false private var onCommandEntered: (String) -> Unit = {} @@ -146,15 +148,34 @@ class TerminalCommandReceiver(terminalConsole: TerminalExecutionConsole) { override fun getProcessInput(): OutputStream { return this@TerminalCommandReceiver.processInput } + + /** + * Override notifyTextAvailable() method to check what the command ends with + */ + override fun notifyTextAvailable(text: String, outputType: Key<*>) { + if (text != "\n" && text.endsWith("\n")) { + prevCommandEndsWithReady = isTextEndsWithReady(text) + } + super.notifyTextAvailable(text, outputType) + } } terminalConsole.withConvertLfToCrlfForNonPtyProcess(true) terminalConsole.attachToProcess(processHandler) } + /** + * Check if the text ends with "READY" - successful completion of TSO command + */ + private fun isTextEndsWithReady(text: String): Boolean { + val successfulEnding = "READY" + val trimmedText = text.trimEnd() + return trimmedText.endsWith(successfulEnding, true) + } + /** * Called when command is submitted. Clean up the entered command for follow up user input */ - private fun cleanCommand() { + fun cleanCommand() { this.typedCommand = "" this.textAfterCursor = "" this.cursorPosition = 0 @@ -193,6 +214,20 @@ class TerminalCommandReceiver(terminalConsole: TerminalExecutionConsole) { } } + /** + * Return true if console is waiting for command input or else otherwise + */ + fun isNeedToWaitForCommandInput(): Boolean { + return needToWaitForCommandInput + } + + /** + * Return true if previous command ends with "READY" or else otherwise + */ + fun isPrevCommandEndsWithReady(): Boolean { + return prevCommandEndsWithReady + } + /** * Called when user finished typing the command and pressed Enter */ diff --git a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/TSOWindowFactory.kt b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/TSOWindowFactory.kt index 4efc5d045..148b6c96b 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/TSOWindowFactory.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/TSOWindowFactory.kt @@ -336,7 +336,7 @@ class TSOWindowFactory : ToolWindowFactory { processHandler.notifyTextAvailable("Attempting to reconnect...\n", ProcessOutputType.STDOUT) try { sendTopic(SESSION_RECONNECT_TOPIC).reconnect(project, console, session) - processHandler.notifyTextAvailable("Successfully reconnected to the TSO session.\n", ProcessOutputType.STDOUT) + processHandler.notifyTextAvailable("Successfully reconnected to the TSO session.\nREADY\n", ProcessOutputType.STDOUT) } catch (e : Exception) { throw e } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt index ee6865d9a..5e66fa889 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt @@ -11,15 +11,20 @@ package eu.ibagroup.formainframe.ui.build.tso.ui //import com.intellij.ui.layout.cellPanel +import com.intellij.execution.process.ProcessEvent import com.intellij.execution.process.ProcessHandler +import com.intellij.execution.process.ProcessListener import com.intellij.execution.ui.ExecutionConsole import com.intellij.openapi.project.Project import com.intellij.openapi.util.Disposer +import com.intellij.openapi.util.Key import com.intellij.terminal.TerminalExecutionConsole import com.intellij.ui.CollectionComboBoxModel import com.intellij.ui.SimpleListCellRenderer import com.intellij.ui.components.JBPanel import com.intellij.ui.dsl.builder.panel +import com.intellij.util.ui.JBEmptyBorder +import eu.ibagroup.formainframe.common.isDebugModeEnabled import eu.ibagroup.formainframe.dataops.operations.MessageData import eu.ibagroup.formainframe.dataops.operations.MessageType import eu.ibagroup.formainframe.ui.build.TerminalCommandReceiver @@ -29,7 +34,7 @@ import eu.ibagroup.formainframe.ui.build.tso.utils.InputRecognizer import eu.ibagroup.formainframe.utils.log import eu.ibagroup.formainframe.utils.sendTopic import java.awt.BorderLayout -import javax.swing.JComboBox +import javax.swing.JButton import javax.swing.JComponent /** @@ -42,12 +47,13 @@ class TSOConsoleView( private var tsoSession: TSOConfigWrapper ) : ExecutionConsole, JBPanel() { - private lateinit var tsoMessageType: JComboBox - private lateinit var tsoDataType: JComboBox + private lateinit var tsoMessageType: MessageType + private lateinit var tsoDataType: MessageData + private lateinit var cancelCommandButton: JButton private val tsoWidthGroup: String = "TSO_WIDTH_GROUP" private val tsoMessageTypes: List = - listOf(MessageType.TSO_MESSAGE, MessageType.TSO_PROMPT, MessageType.TSO_RESPONSE) + listOf(MessageType.TSO_RESPONSE, MessageType.TSO_MESSAGE, MessageType.TSO_PROMPT) private val tsoDataTypes: List = listOf(MessageData.DATA_DATA, MessageData.DATA_HIDDEN, MessageData.DATA_ACTION) @@ -61,6 +67,8 @@ class TSOConsoleView( private val log = log() + private val debugMode = isDebugModeEnabled() + /** * UI panel which contains 2 combo boxes of TSO message type and message data type */ @@ -73,18 +81,35 @@ class TSOConsoleView( model = tsoMessageTypeComboBoxModel, renderer = SimpleListCellRenderer.create("") { it.type } ).also { - tsoMessageType = it.component + tsoMessageType = it.component.item } - } + }.visible(debugMode) row { label("TSO data type").widthGroup(tsoWidthGroup) comboBox( model = tsoDataTypeComboBoxModel, renderer = SimpleListCellRenderer.create("") { it.data } ).also { - tsoDataType = it.component + tsoDataType = it.component.item + } + }.visible(debugMode) + row { + button("Cancel Command (PA1)") { + log.info("CANCEL COMMAND (PA1)") + val prevTsoMessageType = tsoMessageType + val prevTsoDataType = tsoDataType + tsoMessageType = MessageType.TSO_RESPONSE + tsoDataType = MessageData.DATA_ACTION + terminalCommandReceiver.cleanCommand() + processHandler.processInput?.write(("\r").toByteArray()) + tsoMessageType = prevTsoMessageType + tsoDataType = prevTsoDataType + }.also { + cancelCommandButton = it.component } } + }.also { + it.border = JBEmptyBorder(10, 15, 10, 15) } } @@ -102,8 +127,8 @@ class TSOConsoleView( this, tsoSession, enteredCommand.trim(), - tsoMessageType.selectedItem as MessageType, - tsoDataType.selectedItem as MessageData, + tsoMessageType, + tsoDataType, processHandler ) terminalCommandReceiver.waitForCommandInput() @@ -111,6 +136,17 @@ class TSOConsoleView( terminalCommandReceiver.initialized = true + processHandler.addProcessListener(object : ProcessListener { + override fun startNotified(event: ProcessEvent) {} + + override fun processTerminated(event: ProcessEvent) {} + + override fun onTextAvailable(event: ProcessEvent, outputType: Key<*>) { + cancelCommandButton.isEnabled = + !terminalCommandReceiver.isPrevCommandEndsWithReady() && terminalCommandReceiver.isNeedToWaitForCommandInput() + } + }) + Disposer.register(this, consoleView) layout = BorderLayout() add(this.component, BorderLayout.WEST) diff --git a/src/main/resources/settings.properties b/src/main/resources/settings.properties new file mode 100644 index 000000000..ccee7c8ba --- /dev/null +++ b/src/main/resources/settings.properties @@ -0,0 +1 @@ +debug.mode=false \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/common/CommonTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/common/CommonTestSpec.kt index 07b14dc46..5985c97ec 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/common/CommonTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/common/CommonTestSpec.kt @@ -10,9 +10,16 @@ package eu.ibagroup.formainframe.common -import io.kotest.core.spec.style.ShouldSpec +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import io.kotest.assertions.assertSoftly +import io.kotest.matchers.shouldBe +import io.mockk.* +import java.util.Properties -class CommonTestSpec : ShouldSpec({ +class CommonTestSpec : WithApplicationShouldSpec({ + afterSpec { + clearAllMocks() + } context("common module: ui") { // ValidatingCellRenderer.getTableCellRendererComponent should("get table cell renderer") {} @@ -26,4 +33,45 @@ class CommonTestSpec : ShouldSpec({ // StatefulDialog.showUntilDone should("show dialog until it is fulfilled") {} } + context("common module: SettingsPropertyManager") { + val propertyName = "debug.mode" + + mockkConstructor(Properties::class) + + // isDebugModeEnabled + should("debug mode enabled") { + every { anyConstructed().getProperty(propertyName) } returns "true" + val debugMode = isDebugModeEnabled() + + assertSoftly { + debugMode shouldBe true + } + } + should("debug mode disabled") { + every { anyConstructed().getProperty(propertyName) } returns "false" + val debugMode = isDebugModeEnabled() + + assertSoftly { + debugMode shouldBe false + } + } + should("debug mode property not found") { + every { anyConstructed().getProperty(propertyName) } returns null + val debugMode = isDebugModeEnabled() + + assertSoftly { + debugMode shouldBe false + } + } + should("debug mode property contains a non-boolean value") { + every { anyConstructed().getProperty(propertyName) } returns "123" + val debugMode = isDebugModeEnabled() + + assertSoftly { + debugMode shouldBe false + } + } + + unmockkAll() + } }) From 3e683dfd1d98d581cd8c865dc12381de0f23db64 Mon Sep 17 00:00:00 2001 From: Uladzislau Date: Tue, 9 Jan 2024 12:16:57 +0100 Subject: [PATCH 23/34] Release/v2.0.0 is created Signed-off-by: Uladzislau --- build.gradle.kts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle.kts b/build.gradle.kts index 1b8fbcc36..02acd07d4 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -22,7 +22,7 @@ apply(plugin = "kotlin") apply(plugin = "org.jetbrains.intellij") group = "eu.ibagroup" -version = "1.2.0-231" +version = "2.0.0-231" val remoteRobotVersion = "0.11.19" val okHttp3Version = "4.10.0" val kotestVersion = "5.6.2" From aeed4f46ae2e0a9ff99a79ab30e1dda32b93f49c Mon Sep 17 00:00:00 2001 From: Uladzislau Date: Tue, 9 Jan 2024 16:16:43 +0100 Subject: [PATCH 24/34] Fix for sorting Signed-off-by: Uladzislau --- .../explorer/actions/UssSortActionHolder.kt | 32 ++++++++++++------- 1 file changed, 21 insertions(+), 11 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt index efbd77a14..a256ae750 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt @@ -28,7 +28,12 @@ import eu.ibagroup.formainframe.vfs.MFVirtualFile /** * Represents internal USS fetch provider to be able to update the query for each USS dir node whose sorting is enabled */ -internal val fetchProvider = service().getFileFetchProvider(UssQuery::class.java, RemoteQuery::class.java, MFVirtualFile::class.java) +internal val fetchProvider = service() + .getFileFetchProvider( + UssQuery::class.java, + RemoteQuery::class.java, + MFVirtualFile::class.java + ) /** * Represents the custom sort action group in the FileExplorerView context menu @@ -68,7 +73,7 @@ class SortByNameAction : ToggleAction() { /** * Action performed method to register the custom behavior when By Name sorting is clicked */ - override fun actionPerformed(e: AnActionEvent) { + override fun setSelected(e: AnActionEvent, state: Boolean) { val view = e.getExplorerView() ?: return val selectedNode = view.mySelectedNodesData[0].node if (selectedNode is UssDirNode) { @@ -80,7 +85,8 @@ class SortByNameAction : ToggleAction() { } else { selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.TYPE) selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DATE) - if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.NAME)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.NAME) + if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.NAME)) + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.NAME) queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) } queryToUpdate.requester = selectedNode @@ -122,7 +128,7 @@ class SortByTypeAction : ToggleAction() { /** * Action performed method to register the custom behavior when By Type sorting is clicked */ - override fun actionPerformed(e: AnActionEvent) { + override fun setSelected(e: AnActionEvent, state: Boolean) { val view = e.getExplorerView() ?: return val selectedNode = view.mySelectedNodesData[0].node if (selectedNode is UssDirNode) { @@ -134,7 +140,8 @@ class SortByTypeAction : ToggleAction() { } else { selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.NAME) selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DATE) - if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.TYPE)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.TYPE) + if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.TYPE)) + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.TYPE) queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) } queryToUpdate.requester = selectedNode @@ -176,7 +183,7 @@ class SortByModificationDateAction : ToggleAction() { /** * Action performed method to register the custom behavior when By Modification Date sorting is clicked */ - override fun actionPerformed(e: AnActionEvent) { + override fun setSelected(e: AnActionEvent, state: Boolean) { val view = e.getExplorerView() ?: return val selectedNode = view.mySelectedNodesData[0].node if (selectedNode is UssDirNode) { @@ -188,7 +195,8 @@ class SortByModificationDateAction : ToggleAction() { } else { selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.NAME) selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.TYPE) - if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DATE)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) + if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DATE)) + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) } queryToUpdate.requester = selectedNode @@ -230,7 +238,7 @@ class SortByAscendingOrderAction : ToggleAction() { /** * Action performed method to register the custom behavior when Ascending sorting is clicked */ - override fun actionPerformed(e: AnActionEvent) { + override fun setSelected(e: AnActionEvent, state: Boolean) { val view = e.getExplorerView() ?: return val selectedNode = view.mySelectedNodesData[0].node if (selectedNode is UssDirNode) { @@ -241,7 +249,8 @@ class SortByAscendingOrderAction : ToggleAction() { queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) } else { selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DESCENDING) - if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.ASCENDING)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) + if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.ASCENDING)) + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) } queryToUpdate.requester = selectedNode @@ -283,7 +292,7 @@ class SortByDescendingOrderAction : ToggleAction() { /** * Action performed method to register the custom behavior when Descending sorting is clicked */ - override fun actionPerformed(e: AnActionEvent) { + override fun setSelected(e: AnActionEvent, state: Boolean) { val view = e.getExplorerView() ?: return val selectedNode = view.mySelectedNodesData[0].node if (selectedNode is UssDirNode) { @@ -294,7 +303,8 @@ class SortByDescendingOrderAction : ToggleAction() { queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) } else { selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.ASCENDING) - if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DESCENDING)) selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DESCENDING) + if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DESCENDING)) + selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DESCENDING) queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) } queryToUpdate.requester = selectedNode From 27d8a118d1f78bff4847f028e216c91384d0e9ba Mon Sep 17 00:00:00 2001 From: Ivan Shyshkou Date: Tue, 16 Jan 2024 08:27:37 +0100 Subject: [PATCH 25/34] Feature/ijmp 1387 tests refactoring make ui tests independent 231 --- CHANGELOG.md | 1 + build.gradle.kts | 1 + src/uiTest/kotlin/Data.kt | 246 +++ src/uiTest/kotlin/Locators.kt | 35 + src/uiTest/kotlin/auxiliary/FirstTime.kt | 6 +- src/uiTest/kotlin/auxiliary/SmokeTest.kt | 20 +- .../dialogs/AddWorkingSetSubDialog.kt | 68 + .../components/dialogs/CreateMaskSubDialog.kt | 31 + .../components/dialogs/DeletionOfDSMask.kt | 29 + .../dialogs/DeletionOfUssPathRoot.kt | 23 + .../auxiliary/components/dialogs/Dialog.kt | 62 + .../dialogs/EditWorkingSetSubDialog.kt | 33 + .../dialogs/RenameDatasetMaskDialog.kt | 40 + .../components/elements/ButtonElement.kt | 46 + .../containers/AddWorkingSetDialog.kt | 3 +- .../containers/AllocateDatasetDialog.kt | 11 +- .../containers/EditWorkingSetDialog.kt | 3 +- .../auxiliary/containers/IdeFrameImpl.kt | 3 +- .../containers/MemberPropertiesDialog.kt | 27 +- .../auxiliary/containers/WelcomeFrame.kt | 5 +- src/uiTest/kotlin/auxiliary/utils.kt | 1368 +++++++++-------- .../kotlin/jes/CancelHoldReleaseJobTest.kt | 35 +- .../jes/JesWorkingSetViaActionButtonTest.kt | 35 +- .../jes/JesWorkingSetViaContextMenuTest.kt | 115 +- .../jes/JesWorkingSetViaSettingsTest.kt | 89 +- src/uiTest/kotlin/jes/PurgeJobTest.kt | 46 +- src/uiTest/kotlin/jes/SubmitJobTest.kt | 34 +- src/uiTest/kotlin/jes/ViewJwsPropertyTest.kt | 18 +- .../settings/connection/ConnectionManager.kt | 34 +- .../kotlin/testutils/InjectDispatcher.kt.kt | 209 +++ .../testutils/MockResponseDispatcher.kt | 155 +- .../kotlin/workingset/AllocateDatasetTest.kt | 337 ++-- .../workingset/CreateUssFileAndDirTest.kt | 47 +- src/uiTest/kotlin/workingset/Data.kt | 102 -- .../kotlin/workingset/DeleteDatasetTest.kt | 160 +- src/uiTest/kotlin/workingset/Locators.kt | 13 - .../kotlin/workingset/MigrateDatasetTest.kt | 124 +- .../kotlin/workingset/RenameDatasetTest.kt | 572 ++----- .../kotlin/workingset/ViewDatasetTest.kt | 225 +-- .../kotlin/workingset/ViewWsPropertyTest.kt | 320 ++-- .../kotlin/workingset/WorkingSetBase.kt | 505 +++++- .../WorkingSetViaActionButtonTest.kt | 346 ++--- .../WorkingSetViaContextMenuTest.kt | 961 +++++------- .../workingset/WorkingSetViaSettingsTest.kt | 77 +- 44 files changed, 3514 insertions(+), 3106 deletions(-) create mode 100644 src/uiTest/kotlin/Data.kt create mode 100644 src/uiTest/kotlin/Locators.kt create mode 100644 src/uiTest/kotlin/auxiliary/components/dialogs/AddWorkingSetSubDialog.kt create mode 100644 src/uiTest/kotlin/auxiliary/components/dialogs/CreateMaskSubDialog.kt create mode 100644 src/uiTest/kotlin/auxiliary/components/dialogs/DeletionOfDSMask.kt create mode 100644 src/uiTest/kotlin/auxiliary/components/dialogs/DeletionOfUssPathRoot.kt create mode 100644 src/uiTest/kotlin/auxiliary/components/dialogs/Dialog.kt create mode 100644 src/uiTest/kotlin/auxiliary/components/dialogs/EditWorkingSetSubDialog.kt create mode 100644 src/uiTest/kotlin/auxiliary/components/dialogs/RenameDatasetMaskDialog.kt create mode 100644 src/uiTest/kotlin/auxiliary/components/elements/ButtonElement.kt create mode 100644 src/uiTest/kotlin/testutils/InjectDispatcher.kt.kt delete mode 100644 src/uiTest/kotlin/workingset/Data.kt delete mode 100644 src/uiTest/kotlin/workingset/Locators.kt diff --git a/CHANGELOG.md b/CHANGELOG.md index 6bbb195f7..963b973d2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -175,6 +175,7 @@ All notable changes to the Zowe IntelliJ Plugin will be documented in this file. PDS ([ec94b39e](https://github.com/zowe/zowe-explorer-intellij/commit/ec94b39e)) * Bugfix: USS file empty after rename ([71c49b24](https://github.com/zowe/zowe-explorer-intellij/commit/71c49b24)) + ## `1.0.0 (2023-03-13)` * Breaking: Java 17 usage introduced. Plugin requires to use it with IntelliJ version >= 2022.3 diff --git a/build.gradle.kts b/build.gradle.kts index 02acd07d4..d2297f116 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -64,6 +64,7 @@ dependencies { implementation("com.ibm.mq:com.ibm.mq.allclient:9.3.3.0") testImplementation("io.mockk:mockk:1.13.5") testImplementation("org.junit.jupiter:junit-jupiter-api:5.9.2") + implementation("org.junit.jupiter:junit-jupiter-params:5.9.2") testImplementation("io.kotest:kotest-assertions-core:$kotestVersion") testImplementation("io.kotest:kotest-runner-junit5:$kotestVersion") testImplementation("com.intellij.remoterobot:remote-robot:$remoteRobotVersion") diff --git a/src/uiTest/kotlin/Data.kt b/src/uiTest/kotlin/Data.kt new file mode 100644 index 000000000..920bbc732 --- /dev/null +++ b/src/uiTest/kotlin/Data.kt @@ -0,0 +1,246 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset + +import auxiliary.* + +//Global constants +const val PROJECT_NAME = "untitled" +const val REMOTE_ROBOT_URL = "http://127.0.0.1" + +//Allocation unit shorts +const val TRACKS_ALLOCATION_UNIT_SHORT = "TRK" + +//Button text +const val NO_TEXT = "No" +const val OK_TEXT = "OK" +const val YES_TEXT = "Yes" +const val DELETE_TEXT = "Delete" +const val CANCEL_TEXT = "Cancel" + +// Datasets types +const val PO_ORG_FULL = "Partitioned (PO)" +const val PO_ORG_SHORT = "PO" +const val SEQUENTIAL_ORG_FULL = "Sequential (PS)" +const val SEQUENTIAL_ORG_SHORT = "PS" +const val POE_ORG_FULL = "Partitioned Extended (PO-E)" +const val POE_ORG_SHORT = "POE" +const val PDS_TYPE = "PDS" + +//rename dataset +const val DATASET_FOR_RENAME_PROPERTY = "{\"dsorg\":\"PO\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":2,\"recfm\":\"VB\",\"blksize\":6120,\"lrecl\":255, \"migr\":false}" + +//record formats +const val F_RECORD_FORMAT_SHORT = "F" +const val FB_RECORD_FORMAT_SHORT = "FB" +const val V_RECORD_FORMAT_SHORT = "V" +const val VA_RECORD_FORMAT_SHORT = "VA" +const val VB_RECORD_FORMAT_SHORT = "VB" + +//action menu points +const val REFRESH_POINT_TEXT = "Refresh" +const val NEW_POINT_TEXT = "New" +const val WORKING_SET = "Working Set" +const val DATASET_POINT_TEXT = "Dataset" +const val MIGRATE_POINT_TEXT = "Migrate" +const val RECALL_POINT_TEXT = "Recall" +const val RENAME_POINT_TEXT = "Rename" +const val EDIT_POINT_TEXT = "Edit" +const val PROPERTIES_POINT_TEXT = "Properties" +const val MASK_POINT_TEXT = "Mask" + +//Errors messages +const val invalidDatasetNameConstant = "Each name segment (qualifier) is 1 to 8 characters, the first of which must be alphabetic (A to Z) or " + + "national (# @ $). The remaining seven characters are either alphabetic, numeric (0 - 9), national, " + + "a hyphen (-). Name segments are separated by a period (.)" +const val numberGreaterThanOneMsg = "Enter a number greater than or equal to 1" +const val enterValueInCorrectRangeFromOneMsg = "Please enter a number from 1 to 2,147,483,646" +const val enterValueInCorrectRangeFromZeroMsg = "Please enter a number from 0 to 2,147,483,646" +const val enterPositiveNumberMsg = "Enter a positive number" +const val MEMBER_ALREADY_EXISTS = "Member already exists" +const val RENAME_MEMBER_FAILED = "Rename member failed" +const val MEMBER_IN_USE = "Member in use" +const val INVALID_MEMBER_NAME_MESSAGE = "Member name should contain only A-Z a-z 0-9 or national characters" +const val INVALID_MEMBER_NAME_BEGINNING_MESSAGE = "Member name should start with A-Z a-z or national characters" +const val MEMBER_EMPTY_NAME_MESSAGE = "This field must not be blank" +const val DATASET_INVALID_SECTION_MESSAGE = + "Each name segment (qualifier) is 1 to 8 characters, the first of which must be alphabetic (A to Z) or national (# @ \$). The remaining seven characters are either alphabetic, numeric (0 - 9), national, a hyphen (-). Name segments are separated by a period (.)" +const val DATASET_NAME_LENGTH_MESSAGE = "Dataset name cannot exceed 44 characters" +const val ERROR_IN_PLUGIN = "Error in plugin For Mainframe" +const val DATA_SET_RENAME_FAILED_MSG = "Data set rename failed" +const val NO_ITEMS_FOUND_MSG = "No items found" +const val IDENTICAL_MASKS_MESSAGE = "You cannot add several identical masks to table" +const val EMPTY_DATASET_MESSAGE = "You are going to create a Working Set that doesn't fetch anything" +const val INVALID_URL_PORT = "Invalid URL port: \"%s\"" +const val UNIQUE_WORKING_SET_NAME = "You must provide unique working set name. Working Set %s already exists." +const val UNIQUE_MASK = "You must provide unique mask in working set. Working Set \"%s\" already has mask - %s" + +//Migrate options: +const val HMIGRATE_MIGRATE_OPTIONS = "hmigrate" +const val HRECALL_MIGRATE_OPTIONS = "hrecall" + + +//dialog text\patterns + +const val datasetHasBeenCreated = "Dataset %s Has Been Created " + + +//bad alloc params cases +data class InvalidAllocate( + val wsName: String, + val datasetName: String, + val datasetOrganization: String, + val allocationUnit: String, + val primaryAllocation: Int, + val secondaryAllocation: Int, + val directory: Int, + val recordFormat: String, + val recordLength: Int, + val blockSize: Int, + val averageBlockLength: Int, + val message: String +) + +val invalidDatasetNameParams = InvalidAllocate( + "", "A23456789.A", PO_ORG_FULL, "TRK", 10, 1, + 1, "FB", 80, 3200, 0, invalidDatasetNameConstant +) +val invalidPrimaryAllocationParams = InvalidAllocate( + "", "A23.A23", PO_ORG_FULL, "TRK", -2, 0, 1, + "FB", 80, 3200, 0, enterValueInCorrectRangeFromOneMsg) + +val invalidDirectoryParams = InvalidAllocate( + "", "A23.A23", PO_ORG_FULL, "TRK", 10, 0, 0, + "FB", 80, 3200, 0,enterValueInCorrectRangeFromOneMsg + ) +val invalidRecordLengthParams = InvalidAllocate( + "", "A23.A23", PO_ORG_FULL, "TRK", 10, 0, 1, + "FB", 0, 3200, 0,enterValueInCorrectRangeFromOneMsg + ) +val invalidSecondaryAllocationParams = InvalidAllocate( + "", "A23.A23", PO_ORG_FULL, "TRK", 10, -10, 1, + "FB", 80, 3200, 0, enterValueInCorrectRangeFromZeroMsg + ) +val invalidBlockSizeParams = InvalidAllocate( + "", "A23.A23", PO_ORG_FULL, "TRK", 10, 0, 1, + "FB", 80, -1, 0, enterValueInCorrectRangeFromZeroMsg + ) +val invalidAverageBlockLengthParams = InvalidAllocate( + "", "A23.A23", PO_ORG_FULL, "TRK", 10, 0, 1, + "FB", 80, 3200, -1, enterValueInCorrectRangeFromZeroMsg + ) + +val invalidAllocateScenarios = mapOf( + Pair("invalidDatasetNameParams", invalidDatasetNameParams), + Pair("invalidPrimaryAllocationParams", invalidPrimaryAllocationParams), + Pair("invalidDirectoryParams", invalidDirectoryParams), + Pair("invalidRecordLengthParams", invalidRecordLengthParams), + Pair("invalidSecondaryAllocationParams", invalidSecondaryAllocationParams), + Pair("invalidBlockSizeParams", invalidBlockSizeParams), + Pair("invalidAverageBlockLengthParams", invalidAverageBlockLengthParams), +) + +//rename members constants + +const val TOO_LONG_MEMBER_NAME = "123456789" +const val INVALID_NAME_VIA_CONTEXT_MENU = "@*" +const val INVALID_INVALID_FIRST_SYMBOL = "**" +const val EMPTY_STRING = "" + + +//rename dataset name +const val DATA_SET_RENAME_FAILED = "Data set rename failed" + + +val incorrectRenameMember = mapOf( + Pair(TOO_LONG_MEMBER_NAME, MEMBER_NAME_LENGTH_MESSAGE), + Pair(INVALID_NAME_VIA_CONTEXT_MENU, INVALID_MEMBER_NAME_MESSAGE), + Pair(INVALID_INVALID_FIRST_SYMBOL, INVALID_MEMBER_NAME_BEGINNING_MESSAGE), + Pair(EMPTY_STRING, MEMBER_EMPTY_NAME_MESSAGE), +) + +//member's property +const val NO_MEMBERS = "{\"items\":[],\"returnedRows\":0,\"totalRows\":0,\"moreRows\":null,\"JSONversion\":1}" + +//dialog names +const val RENAME_MEMBER_NAME = "Rename Member" +const val RENAME_DATASET_NAME = "Rename Dataset" +const val DELETION_OF_DS_MASK = "Deletion of DS Mask %s" +const val DELETION_OF_USS_PATH_ROOT = "Deletion of Uss Path Root %s" + + +//RC STRINGS + +const val RC_8 = "8.0" +const val RC_8_TEXT = "EDC5051I An error occurred when renaming a file." + + +//ws names +const val WS_NAME_WS_1 = "WS1" +const val WS_NAME_WS_2 = "WS2" +const val WS_NAME_WS_3 = "WS3" +const val WS_NAME_WS_4 = "WS4" +const val WS_NAME_WS_5 = "WS5" +const val WS_NAME_WS_6 = "WS6" +const val WS_NAME_WS_7 = "WS7" +const val WS_NAME_WS_8 = "WS8" +const val WS_NAME_WS_9 = "WS9" +const val WS_NAME_WS_10 = "WS10" + + +//members content +const val EMPTY_MEMBER_CONTENT = "" +const val SHORT_MEMBER_CONTENT = "content" + +//members name +const val MEMBER_NAME_PATTERN = "MEMBER" +const val MEMBER_NAME_1 = "MEMBER"+"1" +const val MEMBER_NAME_2 = "MEMBER"+"2" + +//mask types: +const val ZOS_MASK = "z/OS" +const val USS_MASK = "USS" + + +//masks/mask types combo + +val zosUserDatasetMask = "$ZOS_USERID.*".uppercase() +val zosUserDatasetMaskDoubleStar = "$ZOS_USERID.**".uppercase() +const val ussMask = "/u/$ZOS_USERID" +const val defaultNewUssMask = "/etc/ssh" +val singleMask = Pair("$ZOS_USERID.*", "z/OS") +val singleUssMask = Pair(ussMask, "USS") +val validZOSMasks = listOf( + "$ZOS_USERID.*", "$ZOS_USERID.**", "$ZOS_USERID.@#%", "$ZOS_USERID.@#%.*", "Q.*", "WWW.*", maskWithLength44, + ZOS_USERID +) +val validUSSMasks = listOf("/u", "/etc/ssh", ussMask) + +//todo add mask *.* when bug is fixed +val maskMessageMap = mapOf( + "1$ZOS_USERID.*" to ENTER_VALID_DS_MASK_MESSAGE, + "$ZOS_USERID.{!" to ENTER_VALID_DS_MASK_MESSAGE, + "$ZOS_USERID.A23456789.*" to QUALIFIER_ONE_TO_EIGHT, + "$ZOS_USERID." to ENTER_VALID_DS_MASK_MESSAGE, + maskWithLength45 to "Dataset mask length must not exceed 44 characters", + "$ZOS_USERID.***" to "Invalid asterisks in the qualifier" +) + +//ports +const val PORT_104431 = "104431" +const val PORT_104431_AND_1 = "1044311" + + +//dialogs +const val EDIT_WORKING_SET = "Edit Working Set" +const val CREATE_MASK_DIALOG = "Create Mask" +const val ADD_WORKING_SET_DIALOG = "Add Working Set" +const val RENAME_DATASET_MASK_DIALOG = "Edit Mask" +const val ALLOCATE_DATASET_DIALOG = "Allocate Dataset" diff --git a/src/uiTest/kotlin/Locators.kt b/src/uiTest/kotlin/Locators.kt new file mode 100644 index 000000000..d3b87bd3c --- /dev/null +++ b/src/uiTest/kotlin/Locators.kt @@ -0,0 +1,35 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset + +import com.intellij.remoterobot.search.locators.byXpath + +//common dialogs locators +val myDialogXpathLoc = byXpath("//div[@class='MyDialog']") +val removeButtonLoc = byXpath("//div[contains(@myvisibleactions, 'Down')]//div[@myaction.key='button.text.remove']") +val removeEMaskButtonLoc = byXpath("//div[contains(@myaction.key, 'dom.action.remove')]") +val linkLoc = byXpath("//div[@class='LinkLabel']") +val closeNotificationLoc = byXpath("//div[@tooltiptext.key='tooltip.close.notification']") + +//allocate dialog locators +val datasetNameInputLoc = byXpath("//div[@class='JBTextField']") +val datasetOrgDropDownLoc = byXpath("//div[@class='ComboBox']") +val inputFieldLoc = byXpath("//div[@class='JBTextField']") +val dropdownsLoc = byXpath("//div[@class='ComboBox']") +val messageLoc = byXpath("//div[@class='HeavyWeightWindow']") +val helpLoc = byXpath("//div[@class='HeavyWeightWindow'][.//div[@class='Header']]") +val errorDetailHeaderLoc = byXpath("//div[@javaclass='javax.swing.JEditorPane']") +val errorDetailBodyLoc = byXpath("//div[contains(@visible_text, 'Code:')]") +val closeDialogLoc = byXpath("//div[@class='InplaceButton']") + +// property tab +val dataTabLoc = byXpath("//div[@text='Data']") + +val treesLoc = byXpath("//div[@class='DnDAwareTree']") diff --git a/src/uiTest/kotlin/auxiliary/FirstTime.kt b/src/uiTest/kotlin/auxiliary/FirstTime.kt index 6d5f688c7..3f0f29c52 100644 --- a/src/uiTest/kotlin/auxiliary/FirstTime.kt +++ b/src/uiTest/kotlin/auxiliary/FirstTime.kt @@ -16,6 +16,7 @@ import org.junit.jupiter.api.extension.ExtendWith import auxiliary.containers.* import com.intellij.remoterobot.search.locators.Locator import org.junit.jupiter.api.Tag +import workingset.PROJECT_NAME /** * When adding UI tests to GitHub Actions pipeline, there is a need to first run dummy test, which @@ -25,15 +26,14 @@ import org.junit.jupiter.api.Tag @ExtendWith(RemoteRobotExtension::class) class ConnectionManager { private var stack = mutableListOf() - private val projectName = "untitled" @Test fun firstTime(remoteRobot: RemoteRobot) = with(remoteRobot) { welcomeFrame { - open(projectName) + open() } Thread.sleep(60000) - ideFrameImpl(projectName, stack) { + ideFrameImpl(PROJECT_NAME, stack) { dialog("For Mainframe Plugin Privacy Policy and Terms and Conditions") { clickButton("I Agree") } diff --git a/src/uiTest/kotlin/auxiliary/SmokeTest.kt b/src/uiTest/kotlin/auxiliary/SmokeTest.kt index 484620823..fd1685254 100644 --- a/src/uiTest/kotlin/auxiliary/SmokeTest.kt +++ b/src/uiTest/kotlin/auxiliary/SmokeTest.kt @@ -18,6 +18,7 @@ import org.junit.jupiter.api.extension.ExtendWith import auxiliary.containers.* import com.intellij.remoterobot.search.locators.Locator import org.junit.jupiter.api.* +import workingset.PROJECT_NAME /** * When adding UI tests to GitHub Actions pipeline, there is a need to first run dummy test, which @@ -27,7 +28,6 @@ import org.junit.jupiter.api.* @TestInstance(TestInstance.Lifecycle.PER_CLASS) @ExtendWith(RemoteRobotExtension::class) class SmokeTest { - private val projectName = "untitled" private val wsName = "ws1" private val jwsName = "jws1" private val connectionName = "conName" @@ -43,7 +43,7 @@ class SmokeTest { */ @BeforeAll fun setUpAll(remoteRobot: RemoteRobot) { - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) } /** @@ -51,17 +51,17 @@ class SmokeTest { */ @AfterAll fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @Test fun testBasics(remoteRobot: RemoteRobot) = with(remoteRobot) { - createConnection(projectName, fixtureStack, closableFixtureCollector, connectionName, true, remoteRobot) - createWsAndMask(projectName, wsName, listOf(defaultZosMask, defaultUssMask), connectionName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + createConnection(fixtureStack, closableFixtureCollector, connectionName, true, remoteRobot) + createWsAndMask(wsName, listOf(defaultZosMask, defaultUssMask), connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJesWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName, ZOS_USERID, defaultJobFilter) @@ -70,8 +70,8 @@ class SmokeTest { } closableFixtureCollector.closeOnceIfExists(AddJesWorkingSetDialog.name) } - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - openMaskInExplorer(defaultUssMask.first, "", projectName, fixtureStack, remoteRobot) - openMaskInExplorer(defaultZosMask.first, "", projectName, fixtureStack, remoteRobot) + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) + openMaskInExplorer(defaultUssMask.first, "", fixtureStack, remoteRobot) + openMaskInExplorer(defaultZosMask.first, "",fixtureStack, remoteRobot) } } diff --git a/src/uiTest/kotlin/auxiliary/components/dialogs/AddWorkingSetSubDialog.kt b/src/uiTest/kotlin/auxiliary/components/dialogs/AddWorkingSetSubDialog.kt new file mode 100644 index 000000000..ee3f41d92 --- /dev/null +++ b/src/uiTest/kotlin/auxiliary/components/dialogs/AddWorkingSetSubDialog.kt @@ -0,0 +1,68 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset.auxiliary.components.dialogs + +import auxiliary.containers.addWorkingSetDialog +import auxiliary.containers.ideFrameImpl +import com.intellij.remoterobot.RemoteRobot +import com.intellij.remoterobot.search.locators.Locator +import workingset.* +import workingset.auxiliary.components.elements.ButtonElement + +class AddWorkingSetSubDialog(fixtureStack: MutableList, remoteRobot: RemoteRobot) :AbstractDialog(fixtureStack, remoteRobot) { + + override val dialogTitle: String = ADD_WORKING_SET_DIALOG + var okButton = ButtonElement() + var cancelButton = ButtonElement() + + init { + okButton = ButtonElement(OK_TEXT, fixtureStack, remoteRobot) + cancelButton = ButtonElement(CANCEL_TEXT, fixtureStack, remoteRobot) + } + + constructor() : this(mutableListOf(), RemoteRobot(REMOTE_ROBOT_URL)){} + /* + fill all fields in add working set dialog, mask - Pair + */ + fun fillAddWorkingSet(connectionName:String, wsName:String, mask: Pair, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + waitTitle() + ideFrameImpl(PROJECT_NAME, fixtureStack) { + addWorkingSetDialog(fixtureStack) { + addWorkingSet(wsName, connectionName, mask) + } + } + } + + /* + fill all fields in add working set dialog, mask - String + */ + fun fillAddWorkingSet(connectionName:String, wsName:String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + waitTitle() + ideFrameImpl(PROJECT_NAME, fixtureStack) { + + addWorkingSetDialog(fixtureStack) { + addWorkingSet(wsName, connectionName) + } + } + } + + /* + fill all fields in add working set dialog, mask - ArrayList> + */ + fun fillAddWorkingSet(connectionName:String, wsName:String, mask: ArrayList>, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + waitTitle() + ideFrameImpl(PROJECT_NAME, fixtureStack) { + + addWorkingSetDialog(fixtureStack) { + addWorkingSet(wsName, connectionName, mask) + } + } + } +} \ No newline at end of file diff --git a/src/uiTest/kotlin/auxiliary/components/dialogs/CreateMaskSubDialog.kt b/src/uiTest/kotlin/auxiliary/components/dialogs/CreateMaskSubDialog.kt new file mode 100644 index 000000000..385b39525 --- /dev/null +++ b/src/uiTest/kotlin/auxiliary/components/dialogs/CreateMaskSubDialog.kt @@ -0,0 +1,31 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset.auxiliary.components.dialogs + +import auxiliary.containers.createMaskDialog +import auxiliary.containers.ideFrameImpl +import com.intellij.remoterobot.RemoteRobot +import com.intellij.remoterobot.search.locators.Locator +import workingset.CREATE_MASK_DIALOG +import workingset.PROJECT_NAME +import workingset.REMOTE_ROBOT_URL + +class CreateMaskSubDialog(fixtureStack: MutableList, remoteRobot: RemoteRobot) :AbstractDialog(fixtureStack, remoteRobot) { + override val dialogTitle: String = CREATE_MASK_DIALOG + constructor() : this(mutableListOf(), RemoteRobot(REMOTE_ROBOT_URL)) + + fun setMask(mask: Pair) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + createMaskDialog(fixtureStack) { + createMask(mask) + } + } + } +} \ No newline at end of file diff --git a/src/uiTest/kotlin/auxiliary/components/dialogs/DeletionOfDSMask.kt b/src/uiTest/kotlin/auxiliary/components/dialogs/DeletionOfDSMask.kt new file mode 100644 index 000000000..903a6f289 --- /dev/null +++ b/src/uiTest/kotlin/auxiliary/components/dialogs/DeletionOfDSMask.kt @@ -0,0 +1,29 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset.auxiliary.components.dialogs + +import com.intellij.remoterobot.RemoteRobot +import com.intellij.remoterobot.search.locators.Locator +import workingset.DELETION_OF_DS_MASK +import workingset.REMOTE_ROBOT_URL +import workingset.YES_TEXT +import workingset.auxiliary.components.elements.ButtonElement + +class DeletionOfDSMask (fixtureStack: MutableList, remoteRobot: RemoteRobot) :AbstractDialog(fixtureStack, remoteRobot) { + + override var dialogTitle: String = DELETION_OF_DS_MASK + + var yes_button = ButtonElement() + + constructor() : this(mutableListOf(), RemoteRobot(REMOTE_ROBOT_URL)){ + yes_button = ButtonElement(YES_TEXT, fixtureStack, remoteRobot) + } + +} \ No newline at end of file diff --git a/src/uiTest/kotlin/auxiliary/components/dialogs/DeletionOfUssPathRoot.kt b/src/uiTest/kotlin/auxiliary/components/dialogs/DeletionOfUssPathRoot.kt new file mode 100644 index 000000000..a0b0cd51d --- /dev/null +++ b/src/uiTest/kotlin/auxiliary/components/dialogs/DeletionOfUssPathRoot.kt @@ -0,0 +1,23 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset.auxiliary.components.dialogs + +import com.intellij.remoterobot.RemoteRobot +import com.intellij.remoterobot.search.locators.Locator +import workingset.DELETION_OF_USS_PATH_ROOT +import workingset.REMOTE_ROBOT_URL + +class DeletionOfUssPathRoot (fixtureStack: MutableList, remoteRobot: RemoteRobot) :AbstractDialog(fixtureStack, remoteRobot) { + + override var dialogTitle: String = DELETION_OF_USS_PATH_ROOT + + constructor() : this(mutableListOf(), RemoteRobot(REMOTE_ROBOT_URL)) + +} \ No newline at end of file diff --git a/src/uiTest/kotlin/auxiliary/components/dialogs/Dialog.kt b/src/uiTest/kotlin/auxiliary/components/dialogs/Dialog.kt new file mode 100644 index 000000000..b69e754e4 --- /dev/null +++ b/src/uiTest/kotlin/auxiliary/components/dialogs/Dialog.kt @@ -0,0 +1,62 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset.auxiliary.components.dialogs + +import auxiliary.RemoteRobotExtension +import auxiliary.clickButton +import auxiliary.containers.dialog +import auxiliary.containers.ideFrameImpl +import com.intellij.remoterobot.RemoteRobot +import com.intellij.remoterobot.fixtures.HeavyWeightWindowFixture +import com.intellij.remoterobot.fixtures.JTextFieldFixture +import com.intellij.remoterobot.search.locators.Locator +import org.junit.jupiter.api.TestInstance +import org.junit.jupiter.api.extension.ExtendWith +import workingset.* +import workingset.auxiliary.components.elements.ButtonElement +import java.time.Duration + +@TestInstance(TestInstance.Lifecycle.PER_CLASS) +@ExtendWith(RemoteRobotExtension::class) +abstract class AbstractDialog(internal var fixtureStack: MutableList, internal var remoteRobot: RemoteRobot){ + + abstract val dialogTitle: String + + constructor() : this(mutableListOf(), RemoteRobot(REMOTE_ROBOT_URL)) + + internal fun isShown(): Boolean = with(remoteRobot) { + var status = false + ideFrameImpl(PROJECT_NAME, fixtureStack) { + status = dialog(dialogTitle).isShowing + } + return status + } + + internal fun fillFirstFilld(fild_text: String) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + dialog(dialogTitle) { + find(datasetNameInputLoc).text = fild_text + } + } + } + + internal fun clickButtonByName(button_name: String) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + dialog(dialogTitle) { + ButtonElement(button_name, fixtureStack, remoteRobot).click() + } + } + } + + internal fun waitTitle(duration: Long=3) = with(remoteRobot){ + find(myDialogXpathLoc, Duration.ofSeconds(duration)).findText(dialogTitle) + } + +} \ No newline at end of file diff --git a/src/uiTest/kotlin/auxiliary/components/dialogs/EditWorkingSetSubDialog.kt b/src/uiTest/kotlin/auxiliary/components/dialogs/EditWorkingSetSubDialog.kt new file mode 100644 index 000000000..412dc2e72 --- /dev/null +++ b/src/uiTest/kotlin/auxiliary/components/dialogs/EditWorkingSetSubDialog.kt @@ -0,0 +1,33 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset.auxiliary.components.dialogs + +import auxiliary.containers.editWorkingSetDialog +import auxiliary.containers.ideFrameImpl +import com.intellij.remoterobot.RemoteRobot +import com.intellij.remoterobot.search.locators.Locator +import workingset.EDIT_WORKING_SET +import workingset.PROJECT_NAME +import workingset.REMOTE_ROBOT_URL + +class EditWorkingSetSubDialog(fixtureStack: MutableList, remoteRobot: RemoteRobot) :AbstractDialog(fixtureStack, remoteRobot) { + + override val dialogTitle: String = EDIT_WORKING_SET + + constructor() : this(mutableListOf(), RemoteRobot(REMOTE_ROBOT_URL)) + + fun renameWorkingSet(alreadyExistsWorkingSetName: String) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + editWorkingSetDialog(fixtureStack) { + renameWorkingSet(alreadyExistsWorkingSetName) + } + } + } +} diff --git a/src/uiTest/kotlin/auxiliary/components/dialogs/RenameDatasetMaskDialog.kt b/src/uiTest/kotlin/auxiliary/components/dialogs/RenameDatasetMaskDialog.kt new file mode 100644 index 000000000..0481e9afe --- /dev/null +++ b/src/uiTest/kotlin/auxiliary/components/dialogs/RenameDatasetMaskDialog.kt @@ -0,0 +1,40 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset.auxiliary.components.dialogs + +import auxiliary.containers.dialog +import auxiliary.containers.editWorkingSetDialog +import auxiliary.containers.ideFrameImpl +import com.intellij.remoterobot.RemoteRobot +import com.intellij.remoterobot.fixtures.HeavyWeightWindowFixture +import com.intellij.remoterobot.fixtures.JTextFieldFixture +import com.intellij.remoterobot.search.locators.Locator +import com.intellij.remoterobot.search.locators.byXpath +import io.kotest.matchers.string.shouldContain +import org.junit.jupiter.api.Assertions +import workingset.* +import java.time.Duration + +class RenameDatasetMaskDialog(fixtureStack: MutableList, remoteRobot: RemoteRobot) :AbstractDialog(fixtureStack, remoteRobot) { + + override val dialogTitle: String = RENAME_DATASET_MASK_DIALOG + + constructor() : this(mutableListOf(), RemoteRobot(REMOTE_ROBOT_URL)) + + fun renameMaskFromContextMenu( + fieldText: String, + remoteRobot: RemoteRobot + ) = + with(remoteRobot) { + fillFirstFilld(fieldText) + + } +} + diff --git a/src/uiTest/kotlin/auxiliary/components/elements/ButtonElement.kt b/src/uiTest/kotlin/auxiliary/components/elements/ButtonElement.kt new file mode 100644 index 000000000..e661b6f41 --- /dev/null +++ b/src/uiTest/kotlin/auxiliary/components/elements/ButtonElement.kt @@ -0,0 +1,46 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset.auxiliary.components.elements + +import auxiliary.RemoteRobotExtension +import auxiliary.containers.ideFrameImpl +import com.intellij.remoterobot.RemoteRobot +import com.intellij.remoterobot.search.locators.Locator +import com.intellij.remoterobot.utils.waitFor +import org.junit.jupiter.api.TestInstance +import org.junit.jupiter.api.extension.ExtendWith +import workingset.PROJECT_NAME +import workingset.REMOTE_ROBOT_URL +import java.time.Duration + +@TestInstance(TestInstance.Lifecycle.PER_CLASS) +@ExtendWith(RemoteRobotExtension::class) +open class ButtonElement(val buttonText: String, val fixtureStack: MutableList, val remoteRobot: RemoteRobot){ + constructor() : this("", mutableListOf(), RemoteRobot(REMOTE_ROBOT_URL)) + + internal fun isEnabled(): Boolean = with(remoteRobot) { + var status = false + ideFrameImpl(PROJECT_NAME, fixtureStack) { + status = button(buttonText).isEnabled() + } + return status + } + + internal fun click(waitTime: Long = 60) = with(remoteRobot){ + ideFrameImpl(PROJECT_NAME, fixtureStack) { + val button = button(buttonText) + waitFor(Duration.ofSeconds(waitTime)) { + button.isEnabled() + } + button.click() + } + + } +} \ No newline at end of file diff --git a/src/uiTest/kotlin/auxiliary/containers/AddWorkingSetDialog.kt b/src/uiTest/kotlin/auxiliary/containers/AddWorkingSetDialog.kt index fc0c7e773..72d6688f0 100644 --- a/src/uiTest/kotlin/auxiliary/containers/AddWorkingSetDialog.kt +++ b/src/uiTest/kotlin/auxiliary/containers/AddWorkingSetDialog.kt @@ -19,6 +19,7 @@ import com.intellij.remoterobot.fixtures.* import com.intellij.remoterobot.search.locators.Locator import com.intellij.remoterobot.search.locators.byXpath import com.intellij.remoterobot.utils.keyboard +import workingset.removeEMaskButtonLoc import java.awt.event.KeyEvent import java.time.Duration @@ -82,7 +83,7 @@ open class AddWorkingSetDialog( findText("${maskName.substring(0, 46)}...").moveMouse() } findText(maskName).click() - clickActionButton(byXpath("//div[contains(@myvisibleactions, 'Down')]//div[@myaction.key='button.text.remove']")) + clickActionButton(removeEMaskButtonLoc) } /** diff --git a/src/uiTest/kotlin/auxiliary/containers/AllocateDatasetDialog.kt b/src/uiTest/kotlin/auxiliary/containers/AllocateDatasetDialog.kt index ef4bd2d24..d4f9156d0 100644 --- a/src/uiTest/kotlin/auxiliary/containers/AllocateDatasetDialog.kt +++ b/src/uiTest/kotlin/auxiliary/containers/AllocateDatasetDialog.kt @@ -17,6 +17,7 @@ import com.intellij.remoterobot.data.RemoteComponent import com.intellij.remoterobot.fixtures.* import com.intellij.remoterobot.search.locators.Locator import com.intellij.remoterobot.search.locators.byXpath +import workingset.* import java.time.Duration /** @@ -43,18 +44,18 @@ open class AllocateDatasetDialog( blockSize: Int, averageBlockLength: Int = 0 ) { - findAll(byXpath("//div[@class='JBTextField']"))[0].text = datasetName - findAll(byXpath("//div[@class='ComboBox']"))[1].selectItem(datasetOrganization) + findAll(datasetNameInputLoc)[0].text = datasetName + findAll(datasetOrgDropDownLoc)[1].selectItem(datasetOrganization) - val datasetTextParams = findAll(byXpath("//div[@class='JBTextField']")) - val datasetComboBoxParams = findAll(byXpath("//div[@class='ComboBox']")) + val datasetTextParams = findAll(inputFieldLoc) + val datasetComboBoxParams = findAll(dropdownsLoc) datasetComboBoxParams[2].selectItem(allocationUnit) datasetTextParams[1].text = primaryAllocation.toString() datasetTextParams[2].text = secondaryAllocation.toString() datasetComboBoxParams[3].selectItem(recordFormat) - if (datasetOrganization == "Sequential (PS)") { + if (datasetOrganization == SEQUENTIAL_ORG_FULL) { datasetTextParams[3].text = recordLength.toString() datasetTextParams[4].text = blockSize.toString() datasetTextParams[5].text = averageBlockLength.toString() diff --git a/src/uiTest/kotlin/auxiliary/containers/EditWorkingSetDialog.kt b/src/uiTest/kotlin/auxiliary/containers/EditWorkingSetDialog.kt index 7d895136d..2d288bab4 100644 --- a/src/uiTest/kotlin/auxiliary/containers/EditWorkingSetDialog.kt +++ b/src/uiTest/kotlin/auxiliary/containers/EditWorkingSetDialog.kt @@ -17,6 +17,7 @@ import com.intellij.remoterobot.fixtures.ContainerFixture import com.intellij.remoterobot.fixtures.JTextFieldFixture import com.intellij.remoterobot.search.locators.Locator import com.intellij.remoterobot.search.locators.byXpath +import workingset.dropdownsLoc import java.time.Duration /** @@ -40,7 +41,7 @@ class EditWorkingSetDialog( */ fun changeConnection(newConnectionName: String) { if (newConnectionName.isEmpty().not()) { - find(byXpath("//div[@class='ComboBox']")).selectItem(newConnectionName) + find(dropdownsLoc).selectItem(newConnectionName) } } diff --git a/src/uiTest/kotlin/auxiliary/containers/IdeFrameImpl.kt b/src/uiTest/kotlin/auxiliary/containers/IdeFrameImpl.kt index fdc5a2cbf..6e43b54bf 100644 --- a/src/uiTest/kotlin/auxiliary/containers/IdeFrameImpl.kt +++ b/src/uiTest/kotlin/auxiliary/containers/IdeFrameImpl.kt @@ -19,6 +19,7 @@ import com.intellij.remoterobot.data.RemoteComponent import com.intellij.remoterobot.fixtures.FixtureName import com.intellij.remoterobot.search.locators.Locator import com.intellij.remoterobot.search.locators.byXpath +import workingset.PROJECT_NAME import java.time.Duration /** @@ -55,7 +56,7 @@ class IdeFrameImpl(remoteRobot: RemoteRobot, remoteComponent: RemoteComponent) : * Finds the Ide Frame with a specific name and modifies the fixtureStack. The frame needs to be called from the * RemoteRobot as there is no ContainerFixture containing it. */ -fun RemoteRobot.ideFrameImpl(name: String, +fun RemoteRobot.ideFrameImpl(name: String = PROJECT_NAME, fixtureStack: MutableList, function: IdeFrameImpl.() -> Unit) { find(IdeFrameImpl.xPath(name), Duration.ofSeconds(60)).apply { diff --git a/src/uiTest/kotlin/auxiliary/containers/MemberPropertiesDialog.kt b/src/uiTest/kotlin/auxiliary/containers/MemberPropertiesDialog.kt index 2cec71053..f5253d987 100644 --- a/src/uiTest/kotlin/auxiliary/containers/MemberPropertiesDialog.kt +++ b/src/uiTest/kotlin/auxiliary/containers/MemberPropertiesDialog.kt @@ -18,6 +18,8 @@ import com.intellij.remoterobot.data.RemoteComponent import com.intellij.remoterobot.fixtures.* import com.intellij.remoterobot.search.locators.Locator import com.intellij.remoterobot.search.locators.byXpath +import workingset.dataTabLoc +import workingset.inputFieldLoc import java.time.Duration /** @@ -64,22 +66,27 @@ open class MemberPropertiesDialog( curRecNum: String, begRecNum: String, changedRecNum: String): Boolean { - var result = true - val memberGeneralParams = findAll(byXpath("//div[@class='JBTextField']")) + + val memberGeneralParams = findAll(inputFieldLoc) + if(memberGeneralParams[0].text != memName || memberGeneralParams[1].text != version || memberGeneralParams[2].text != createDate || memberGeneralParams[3].text != modDate || memberGeneralParams[4].text != modTime || memberGeneralParams[5].text != userId){ - result = false + return false } - Thread.sleep(1000) - find(byXpath("//div[@text='Data']")).click() - Thread.sleep(1000) - val memberDataParams = findAll(byXpath("//div[@class='JBTextField']")) - if(memberDataParams[0].text != curRecNum || memberDataParams[1].text != begRecNum || memberDataParams[2].text != changedRecNum){ - result = false + + find(dataTabLoc).click() + + val memberDataParams = findAll(inputFieldLoc) + + if(memberDataParams[0].text != curRecNum || + memberDataParams[1].text != begRecNum || + memberDataParams[2].text != changedRecNum) + { + return false } - return result + return true } /** diff --git a/src/uiTest/kotlin/auxiliary/containers/WelcomeFrame.kt b/src/uiTest/kotlin/auxiliary/containers/WelcomeFrame.kt index cd6fac535..ec6e3901a 100644 --- a/src/uiTest/kotlin/auxiliary/containers/WelcomeFrame.kt +++ b/src/uiTest/kotlin/auxiliary/containers/WelcomeFrame.kt @@ -17,6 +17,7 @@ import com.intellij.remoterobot.fixtures.CommonContainerFixture import com.intellij.remoterobot.fixtures.DefaultXpath import com.intellij.remoterobot.fixtures.FixtureName import com.intellij.remoterobot.search.locators.byXpath +import workingset.PROJECT_NAME import java.time.Duration /** @@ -37,12 +38,12 @@ class WelcomeFrame(remoteRobot: RemoteRobot, remoteComponent: RemoteComponent) : /** * Opens project with projectName, which is located in the resources of the uiTest source set. */ - fun open(projectName: String) { + fun open() { openProject.click() Thread.sleep(1000) dialog("Open File or Project") { textField(byXpath("//div[@class='BorderlessTextField']")).text = - System.getProperty("user.dir") + "/src/uiTest/resources/$projectName" + System.getProperty("user.dir") + "/src/uiTest/resources/$PROJECT_NAME" Thread.sleep(1000) clickButton("OK") } diff --git a/src/uiTest/kotlin/auxiliary/utils.kt b/src/uiTest/kotlin/auxiliary/utils.kt index 52bed6ee1..1e21c4fe8 100644 --- a/src/uiTest/kotlin/auxiliary/utils.kt +++ b/src/uiTest/kotlin/auxiliary/utils.kt @@ -33,6 +33,8 @@ import okhttp3.tls.HandshakeCertificates import okhttp3.tls.HeldCertificate import org.junit.jupiter.api.TestInfo import testutils.MockResponseDispatcher +import workingset.* +//import workingset.testutils.InjectDispatcher import java.awt.event.KeyEvent import java.net.InetAddress import java.time.Duration @@ -40,80 +42,64 @@ import java.util.concurrent.TimeUnit lateinit var mockServer: MockWebServer lateinit var responseDispatcher: MockResponseDispatcher +//lateinit var injectDispatcher: InjectDispatcher enum class JobStatus { ACTIVE, INPUT, OUTPUT } //Change ZOS_USERID, ZOS_PWD, CONNECTION_URL with valid values before UI tests execution -const val ZOS_USERID = "user" +const val ZOS_USERID = "User" const val ZOS_PWD = "changeme" const val CONNECTION_URL = "changeme" const val ENTER_VALID_DS_MASK_MESSAGE = "Enter valid dataset mask" -const val IDENTICAL_MASKS_MESSAGE = "You cannot add several identical masks to table" -const val EMPTY_DATASET_MESSAGE = "You are going to create a Working Set that doesn't fetch anything" + + const val TEXT_FIELD_LENGTH_MESSAGE = "Text field must not exceed 8 characters." const val MEMBER_NAME_LENGTH_MESSAGE = "Member name must not exceed 8 characters." const val FILE_NAME_LENGTH_MESSAGE = "Filename must not exceed 255 characters." -const val MEMBER_EMPTY_NAME_MESSAGE = "This field must not be blank" -const val INVALID_MEMBER_NAME_MESSAGE = "Member name should contain only A-Z a-z 0-9 or national characters" -const val INVALID_MEMBER_NAME_BEGINNING_MESSAGE = "Member name should start with A-Z a-z or national characters" const val JOB_ID_LENGTH_MESSAGE = "Job ID length must be 8 characters." const val TEXT_FIELD_CONTAIN_MESSAGE = "Text field should contain only A-Z, a-z, 0-9, *, %." const val JOBID_CONTAIN_MESSAGE = "Text field should contain only A-Z, a-z, 0-9" const val FILE_RESRVED_SYMBOL_MESSAGE = "Filename must not contain reserved '/' symbol." const val PREFIX_OWNER_JOBID_MESSAGE = "You must provide either an owner and a prefix or a job ID." const val IDENTICAL_FILTERS_MESSAGE = "You cannot add several identical job filters to table" -const val DATASET_NAME_LENGTH_MESSAGE = "Dataset name cannot exceed 44 characters" -const val DATASET_INVALID_SECTION_MESSAGE = - "Each name segment (qualifier) is 1 to 8 characters, the first of which must be alphabetic (A to Z) or national (# @ \$). The remaining seven characters are either alphabetic, numeric (0 - 9), national, a hyphen (-). Name segments are separated by a period (.)" - +const val QUALIFIER_ONE_TO_EIGHT = "Qualifier must be in 1 to 8 characters" val maskWithLength44 = - "$ZOS_USERID." + "A2345678.".repeat((44 - (ZOS_USERID.length + 1)) / 9) + "A".repeat((44 - (ZOS_USERID.length + 1)) % 9) + ZOS_USERID + ".A2345678".repeat((44 - (ZOS_USERID.length + 1)) / 9) + "A".repeat((44 - (ZOS_USERID.length + 1)) % 9) val maskWithLength45 = - "$ZOS_USERID." + "A2345678.".repeat((45 - (ZOS_USERID.length + 1)) / 9) + "A".repeat((45 - (ZOS_USERID.length + 1)) % 9) - -//todo add mask *.* when bug is fixed -val maskMessageMap = mapOf( - "1$ZOS_USERID.*" to ENTER_VALID_DS_MASK_MESSAGE, - "$ZOS_USERID.{!" to ENTER_VALID_DS_MASK_MESSAGE, - "$ZOS_USERID.A23456789.*" to "Qualifier must be in 1 to 8 characters", - "$ZOS_USERID." to ENTER_VALID_DS_MASK_MESSAGE, - maskWithLength45 to "Dataset mask length must not exceed 44 characters", - "$ZOS_USERID.***" to "Invalid asterisks in the qualifier" -) + "$ZOS_USERID." + "A2345678.".repeat((45 - (ZOS_USERID.length + 1)) / 9) + "A".repeat((45 - (ZOS_USERID.length + 1)) % 9) + + + + -val validZOSMasks = listOf( - "$ZOS_USERID.*", "$ZOS_USERID.**", "$ZOS_USERID.@#%", "$ZOS_USERID.@#%.*", "Q.*", "WWW.*", maskWithLength44, - ZOS_USERID -) -val validUSSMasks = listOf("/u", "/etc/ssh", "/u/$ZOS_USERID") val validJobsFilters = listOf( - Triple("*", ZOS_USERID, ""), - Triple("TEST1", ZOS_USERID, ""), - Triple("", "", "JOB01234"), - Triple("TEST*", ZOS_USERID, ""), - Triple("TEST**", ZOS_USERID, ""), - Triple("TEST***", ZOS_USERID, ""), - Triple("TEST1", "$ZOS_USERID*", ""), - Triple("TEST1", "$ZOS_USERID**", ""), - Triple("TEST1", "$ZOS_USERID***", ""), - Triple("TEST***", "$ZOS_USERID***", "") + Triple("*", ZOS_USERID, ""), + Triple("TEST1", ZOS_USERID, ""), + Triple("", "", "JOB01234"), + Triple("TEST*", ZOS_USERID, ""), + Triple("TEST**", ZOS_USERID, ""), + Triple("TEST***", ZOS_USERID, ""), + Triple("TEST1", "$ZOS_USERID*", ""), + Triple("TEST1", "$ZOS_USERID**", ""), + Triple("TEST1", "$ZOS_USERID***", ""), + Triple("TEST***", "$ZOS_USERID***", "") ) val invalidJobsFiltersMap = mapOf( - Pair(Triple("123456789", ZOS_USERID, ""), 1) to TEXT_FIELD_LENGTH_MESSAGE, - Pair(Triple("123456789", "A23456789", ""), 1) to TEXT_FIELD_LENGTH_MESSAGE, - Pair(Triple("*", "A23456789", ""), 2) to TEXT_FIELD_LENGTH_MESSAGE, - Pair(Triple("", "", "A23456789"), 3) to JOB_ID_LENGTH_MESSAGE, - Pair(Triple("", "", "A2"), 3) to JOB_ID_LENGTH_MESSAGE, - Pair(Triple("@", ZOS_USERID, ""), 1) to TEXT_FIELD_CONTAIN_MESSAGE, - Pair(Triple("*", "@", ""), 2) to TEXT_FIELD_CONTAIN_MESSAGE, - Pair(Triple("", "", "@@@@@@@@"), 3) to JOBID_CONTAIN_MESSAGE, - Pair(Triple("*", ZOS_USERID, "JOB45678"), 1) to PREFIX_OWNER_JOBID_MESSAGE, - Pair(Triple("*", "", "JOB45678"), 1) to PREFIX_OWNER_JOBID_MESSAGE, - Pair(Triple("", ZOS_USERID, "JOB45678"), 2) to PREFIX_OWNER_JOBID_MESSAGE + Pair(Triple("123456789", ZOS_USERID, ""), 1) to TEXT_FIELD_LENGTH_MESSAGE, + Pair(Triple("123456789", "A23456789", ""), 1) to TEXT_FIELD_LENGTH_MESSAGE, + Pair(Triple("*", "A23456789", ""), 2) to TEXT_FIELD_LENGTH_MESSAGE, + Pair(Triple("", "", "A23456789"), 3) to JOB_ID_LENGTH_MESSAGE, + Pair(Triple("", "", "A2"), 3) to JOB_ID_LENGTH_MESSAGE, + Pair(Triple("@", ZOS_USERID, ""), 1) to TEXT_FIELD_CONTAIN_MESSAGE, + Pair(Triple("*", "@", ""), 2) to TEXT_FIELD_CONTAIN_MESSAGE, + Pair(Triple("", "", "@@@@@@@@"), 3) to JOBID_CONTAIN_MESSAGE, + Pair(Triple("*", ZOS_USERID, "JOB45678"), 1) to PREFIX_OWNER_JOBID_MESSAGE, + Pair(Triple("*", "", "JOB45678"), 1) to PREFIX_OWNER_JOBID_MESSAGE, + Pair(Triple("", ZOS_USERID, "JOB45678"), 2) to PREFIX_OWNER_JOBID_MESSAGE ) val viewTree = byXpath("//div[@class='JBViewport'][.//div[@class='DnDAwareTree']]") @@ -124,33 +110,33 @@ enum class UssFileType { File, Directory } * Waits 60 seconds for the button to be enabled and then clicks on it. */ fun CommonContainerFixture.clickButton(text: String) { - val button = button(text) - waitFor(Duration.ofSeconds(60)) { - button.isEnabled() - } - button.click() + val button = button(text) + waitFor(Duration.ofSeconds(60)) { + button.isEnabled() + } + button.click() } /** * Waits a specific amount of time for the button to be enabled and then clicks on it. */ fun CommonContainerFixture.clickButton(locator: Locator, duration: Duration = Duration.ofSeconds(60)) { - val button = button(locator) - waitFor(duration) { - button.isEnabled() - } - button.click() + val button = button(locator) + waitFor(duration) { + button.isEnabled() + } + button.click() } /** * Waits 60 seconds for the action button to be enabled and then clicks on it. */ fun CommonContainerFixture.clickActionButton(locator: Locator) { - val button = actionButton(locator) - waitFor(Duration.ofSeconds(60)) { - button.isEnabled() - } - button.click() + val button = actionButton(locator) + waitFor(Duration.ofSeconds(60)) { + button.isEnabled() + } + button.click() } /** @@ -158,23 +144,23 @@ fun CommonContainerFixture.clickActionButton(locator: Locator) { */ fun ContainerFixture.createWSFromContextMenu( fixtureStack: MutableList, - closableFixtureCollector: ClosableFixtureCollector + closableFixtureCollector: ClosableFixtureCollector, ) { - explorer { - fileExplorer.click() - find(viewTree).rightClick() - } - actionMenu(remoteRobot, "New").click() - - //workaround when an action menu contains more than 2 action menu items, and you need to choose the 3d item - runJs( - """ + explorer { + fileExplorer.click() + find(viewTree).rightClick() + } + actionMenu(remoteRobot, NEW_POINT_TEXT).click() + + //workaround when an action menu contains more than 2 action menu items, and you need to choose the 3d item + runJs( + """ const point = new java.awt.Point(${locationOnScreen.x}, ${locationOnScreen.y}); robot.moveMouse(component, point); """ - ) - actionMenuItem(remoteRobot, "Working Set").click() - closableFixtureCollector.add(AddWorkingSetDialog.xPath(), fixtureStack) + ) + actionMenuItem(remoteRobot, WORKING_SET).click() + closableFixtureCollector.add(AddWorkingSetDialog.xPath(), fixtureStack) } /** @@ -182,16 +168,16 @@ fun ContainerFixture.createWSFromContextMenu( */ fun ContainerFixture.editWSFromContextMenu( wsName: String, fixtureStack: MutableList, - closableFixtureCollector: ClosableFixtureCollector + closableFixtureCollector: ClosableFixtureCollector, ) { - explorer { - fileExplorer.click() - find(viewTree).findText(wsName) - .rightClick() - Thread.sleep(3000) - } - actionMenuItem(remoteRobot, "Edit").click() - closableFixtureCollector.add(EditWorkingSetDialog.xPath(), fixtureStack) + explorer { + fileExplorer.click() + find(viewTree).findText(wsName) + .rightClick() + Thread.sleep(3000) + } + actionMenuItem(remoteRobot, EDIT_POINT_TEXT).click() + closableFixtureCollector.add(EditWorkingSetDialog.xPath(), fixtureStack) } /** @@ -199,45 +185,43 @@ fun ContainerFixture.editWSFromContextMenu( */ fun ContainerFixture.createMask( wsName: String, fixtureStack: MutableList, - closableFixtureCollector: ClosableFixtureCollector + closableFixtureCollector: ClosableFixtureCollector, ) { - explorer { - fileExplorer.click() - find(viewTree).findText(wsName) - .rightClick() - Thread.sleep(3000) - } - actionMenu(remoteRobot, "New").click() - actionMenuItem(remoteRobot, "Mask").click() - closableFixtureCollector.add(CreateMaskDialog.xPath(), fixtureStack) + explorer { + fileExplorer.click() + find(viewTree).findText(wsName) + .rightClick() + } + actionMenu(remoteRobot, NEW_POINT_TEXT).click() + actionMenuItem(remoteRobot, MASK_POINT_TEXT).click() + closableFixtureCollector.add(CreateMaskDialog.xPath(), fixtureStack) } /** * Deletes a working set via context menu from explorer. */ fun ContainerFixture.deleteWSFromContextMenu(wsName: String) { - explorer { - fileExplorer.click() - find(viewTree).findText(wsName) - .rightClick() - Thread.sleep(3000) - } - actionMenuItem(remoteRobot, "Delete").click() - find(byXpath("//div[@class='MyDialog' and @title='Deletion of Working Set $wsName']")) + explorer { + fileExplorer.click() + find(viewTree).findText(wsName) + .rightClick() + Thread.sleep(3000) + } + actionMenuItem(remoteRobot, DELETE_TEXT).click() } /** * Deletes a JES working set via context menu from explorer. */ fun ContainerFixture.deleteJWSFromContextMenu(jwsName: String) { - explorer { - jesExplorer.click() - find(viewTree).findText(jwsName) - .rightClick() - Thread.sleep(3000) - } - actionMenuItem(remoteRobot, "Delete").click() - find(byXpath("//div[@class='MyDialog' and @title='Deletion of JES Working Set $jwsName']")) + explorer { + jesExplorer.click() + find(viewTree).findText(jwsName) + .rightClick() + Thread.sleep(3000) + } + actionMenuItem(remoteRobot, DELETE_TEXT).click() + find(byXpath("//div[@class='MyDialog' and @title='Deletion of JES Working Set $jwsName']")) } /** @@ -245,16 +229,16 @@ fun ContainerFixture.deleteJWSFromContextMenu(jwsName: String) { */ fun ContainerFixture.createJWSFromContextMenu( fixtureStack: MutableList, - closableFixtureCollector: ClosableFixtureCollector + closableFixtureCollector: ClosableFixtureCollector, ) { - explorer { - jesExplorer.click() - find(viewTree).rightClick() - Thread.sleep(3000) - } - actionMenu(remoteRobot, "New").click() - actionMenuItem(remoteRobot, "JES Working Set").click() - closableFixtureCollector.add(AddJesWorkingSetDialog.xPath(), fixtureStack) + explorer { + jesExplorer.click() + find(viewTree).rightClick() + Thread.sleep(3000) + } + actionMenu(remoteRobot, NEW_POINT_TEXT).click() + actionMenuItem(remoteRobot, "JES Working Set").click() + closableFixtureCollector.add(AddJesWorkingSetDialog.xPath(), fixtureStack) } /** @@ -262,17 +246,17 @@ fun ContainerFixture.createJWSFromContextMenu( */ fun ContainerFixture.createJobsFilter( jwsName: String, fixtureStack: MutableList, - closableFixtureCollector: ClosableFixtureCollector + closableFixtureCollector: ClosableFixtureCollector, ) { - explorer { - jesExplorer.click() - find(viewTree).findText(jwsName) - .rightClick() - Thread.sleep(3000) - } - actionMenu(remoteRobot, "New").click() - actionMenuItem(remoteRobot, "Jobs Filter").click() - closableFixtureCollector.add(CreateJobsFilterDialog.xPath(), fixtureStack) + explorer { + jesExplorer.click() + find(viewTree).findText(jwsName) + .rightClick() + Thread.sleep(3000) + } + actionMenu(remoteRobot, NEW_POINT_TEXT).click() + actionMenuItem(remoteRobot, "Jobs Filter").click() + closableFixtureCollector.add(CreateJobsFilterDialog.xPath(), fixtureStack) } /** @@ -280,38 +264,38 @@ fun ContainerFixture.createJobsFilter( */ fun ContainerFixture.editJWSFromContextMenu( jwsName: String, fixtureStack: MutableList, - closableFixtureCollector: ClosableFixtureCollector + closableFixtureCollector: ClosableFixtureCollector, ) { - explorer { - jesExplorer.click() - find(viewTree).findText(jwsName) - .rightClick() - Thread.sleep(3000) - } - actionMenuItem(remoteRobot, "Edit").click() - closableFixtureCollector.add(EditJesWorkingSetDialog.xPath(), fixtureStack) + explorer { + jesExplorer.click() + find(viewTree).findText(jwsName) + .rightClick() + Thread.sleep(3000) + } + actionMenuItem(remoteRobot, EDIT_POINT_TEXT).click() + closableFixtureCollector.add(EditJesWorkingSetDialog.xPath(), fixtureStack) } /** * Creates a working set via action button. */ -fun ContainerFixture.createWorkingSetFromActionButton( +fun ContainerFixture.callCreateWorkingSetFromActionButton( closableFixtureCollector: ClosableFixtureCollector, - fixtureStack: MutableList + fixtureStack: MutableList, ) { - explorer { - fileExplorer.click() - createConfigItem() - } - find( - byXpath("//div[@class='HeavyWeightWindow']"), - Duration.ofSeconds(30) - ).findAllText().forEach { - if (it.text == "Working Set") { - it.click() - closableFixtureCollector.add(AddWorkingSetDialog.xPath(), fixtureStack) + explorer { + fileExplorer.click() + createConfigItem() + } + find( + byXpath("//div[@class='HeavyWeightWindow']"), + Duration.ofSeconds(30) + ).findAllText().forEach { + if (it.text == WORKING_SET) { + it.click() + closableFixtureCollector.add(AddWorkingSetDialog.xPath(), fixtureStack) + } } - } } /** @@ -319,21 +303,21 @@ fun ContainerFixture.createWorkingSetFromActionButton( */ fun ContainerFixture.createJesWorkingSetFromActionButton( closableFixtureCollector: ClosableFixtureCollector, - fixtureStack: MutableList + fixtureStack: MutableList, ) { - explorer { - jesExplorer.click() - createConfigItem() - } - find( - byXpath("//div[@class='HeavyWeightWindow']"), - Duration.ofSeconds(30) - ).findAllText().forEach { - if (it.text == "JES Working Set") { - it.click() - closableFixtureCollector.add(AddJesWorkingSetDialog.xPath(), fixtureStack) + explorer { + jesExplorer.click() + createConfigItem() + } + find( + messageLoc, + Duration.ofSeconds(30) + ).findAllText().forEach { + if (it.text == "JES Working Set") { + it.click() + closableFixtureCollector.add(AddJesWorkingSetDialog.xPath(), fixtureStack) + } } - } } /** @@ -341,28 +325,27 @@ fun ContainerFixture.createJesWorkingSetFromActionButton( */ fun ContainerFixture.createConnectionFromActionButton( closableFixtureCollector: ClosableFixtureCollector, - fixtureStack: MutableList + fixtureStack: MutableList, ) { - explorer { - jesExplorer.click() - createConfigItem() - } - find( - byXpath("//div[@class='HeavyWeightWindow']"), - Duration.ofSeconds(30) - ).findAllText().forEach { - if (it.text == "Connection") { - it.click() - closableFixtureCollector.add(AddConnectionDialog.xPath(), fixtureStack) + explorer { + jesExplorer.click() + createConfigItem() + } + find( + messageLoc, + Duration.ofSeconds(30) + ).findAllText().forEach { + if (it.text == "Connection") { + it.click() + closableFixtureCollector.add(AddConnectionDialog.xPath(), fixtureStack) + } } - } } /** * Steps to create a connection(valid or invalid) from settings . */ fun createConnection( - projectName: String, fixtureStack: MutableList, closableFixtureCollector: ClosableFixtureCollector, connectionName: String, @@ -370,350 +353,347 @@ fun createConnection( remoteRobot: RemoteRobot, url: String = CONNECTION_URL, user: String = ZOS_USERID, - password: String = ZOS_PWD + password: String = ZOS_PWD, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - settings(closableFixtureCollector, fixtureStack) - } - settingsDialog(fixtureStack) { - configurableEditor { - conTab.click() - add(closableFixtureCollector, fixtureStack) - } - addConnectionDialog(fixtureStack) { - if (isValidConnection) { - addConnection(connectionName, url, user, password, true) - } else { - addConnection(connectionName, "${url}1", user, password, true) + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + settings(closableFixtureCollector, fixtureStack) } - clickButton("OK") - Thread.sleep(3000) - } - closableFixtureCollector.closeOnceIfExists(AddConnectionDialog.name) - if (isValidConnection.not()) { - errorCreatingConnectionDialog(closableFixtureCollector, fixtureStack) { - clickButton("Yes") + settingsDialog(fixtureStack) { + configurableEditor { + conTab.click() + add(closableFixtureCollector, fixtureStack) + } + addConnectionDialog(fixtureStack) { + if (isValidConnection) { + addConnection(connectionName, url, user, password, true) + } else { + addConnection(connectionName, "${url}1", user, password, true) + } + clickButton(OK_TEXT) + Thread.sleep(3000) + } + closableFixtureCollector.closeOnceIfExists(AddConnectionDialog.name) + if (isValidConnection.not()) { + errorCreatingConnectionDialog(closableFixtureCollector, fixtureStack) { + clickButton(YES_TEXT) + } + closableFixtureCollector.closeOnceIfExists(ErrorCreatingConnectionDialog.name) + } + clickButton("OK") } - closableFixtureCollector.closeOnceIfExists(ErrorCreatingConnectionDialog.name) - } - clickButton("OK") + closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) - } } /** * Deletes all JES working sets, working sets and connections. To be used in BeforeAll and AfterAll tests methods. */ fun clearEnvironment( - projectName: String, fixtureStack: MutableList, closableFixtureCollector: ClosableFixtureCollector, - remoteRobot: RemoteRobot + remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - settings(closableFixtureCollector, fixtureStack) - } - settingsDialog(fixtureStack) { - configurableEditor { - workingSetsTab.click() - deleteAllItems() - jesWorkingSetsTab.click() - deleteAllItems() - conTab.click() - deleteAllItems() - } - clickButton("OK") + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + settings(closableFixtureCollector, fixtureStack) + } + settingsDialog(fixtureStack) { + configurableEditor { + workingSetsTab.click() + deleteAllItems() + jesWorkingSetsTab.click() + deleteAllItems() + conTab.click() + deleteAllItems() + } + clickButton(OK_TEXT) + } + closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) - } } /** * Opens a project and an explorer, clears test environment before tests execution. */ fun setUpTestEnvironment( - projectName: String, fixtureStack: MutableList, closableFixtureCollector: ClosableFixtureCollector, - remoteRobot: RemoteRobot + remoteRobot: RemoteRobot, ) = with(remoteRobot) { - welcomeFrame { - open(projectName) - } - Thread.sleep(10000) - - ideFrameImpl(projectName, fixtureStack) { - try { - if (dialog("For Mainframe Plugin Privacy Policy and Terms and Conditions").isShowing) { - clickButton("Dismiss") - } - } catch (e: WaitForConditionTimeoutException) { - e.message.shouldContain("Failed to find 'Dialog' by 'title For Mainframe Plugin Privacy Policy and Terms and Conditions'") + welcomeFrame { + open() } - try { - find(byXpath("//div[@class='ProjectViewTree']")) - stripeButton(byXpath("//div[@accessiblename='Project' and @class='StripeButton' and @text='Project']")) - .click() - } catch (e: WaitForConditionTimeoutException) { - //do nothing if ProjectViewTree is hidden + Thread.sleep(10000) + + ideFrameImpl(PROJECT_NAME, fixtureStack) { + try { + if (dialog("For Mainframe Plugin Privacy Policy and Terms and Conditions").isShowing) { + clickButton("Dismiss") + } + } catch (e: WaitForConditionTimeoutException) { + e.message.shouldContain("Failed to find 'Dialog' by 'title For Mainframe Plugin Privacy Policy and Terms and Conditions'") + } + try { + find(byXpath("//div[@class='ProjectViewTree']")) + stripeButton(byXpath("//div[@accessiblename='Project' and @class='StripeButton' and @text='Project']")) + .click() + } catch (e: WaitForConditionTimeoutException) { + //do nothing if ProjectViewTree is hidden + } + forMainframe() } - forMainframe() - } - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) } /** * Opens a mask in the working set in explorer. */ fun openWSOpenMaskInExplorer( - wsName: String, maskName: String, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + wsName: String, maskName: String, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = - with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(wsName).doubleClick() - Thread.sleep(3000) - find(viewTree).findText(maskName).doubleClick() - Thread.sleep(2000) - waitFor(Duration.ofSeconds(20)) { find(viewTree).hasText("loading…").not() } - find(viewTree).findAllText().shouldNotContain("Error") - find(viewTree).findText(wsName).doubleClick() - } + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(wsName).doubleClick() + Thread.sleep(3000) + find(viewTree).findText(maskName).doubleClick() + Thread.sleep(2000) + waitFor(Duration.ofSeconds(20)) { find(viewTree).hasText("loading…").not() } + find(viewTree).findAllText().shouldNotContain("Error") + find(viewTree).findText(wsName).doubleClick() + } + } } - } /** * Double-clicks on the working set to open or close it in explorer. */ fun openOrCloseWorkingSetInExplorer( - wsName: String, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + wsName: String, fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(wsName).doubleClick() - Thread.sleep(1000) + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(wsName).doubleClick() + Thread.sleep(1000) + } } - } } /** * Double-clicks on the jes working set to open or close it in explorer. */ fun openOrCloseJesWorkingSetInExplorer( - jwsName: String, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + jwsName: String, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - jesExplorer.click() - find(viewTree).findText(jwsName).doubleClick() - Thread.sleep(3000) + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + jesExplorer.click() + find(viewTree).findText(jwsName).doubleClick() + Thread.sleep(3000) + } } - } } /** * Double-clicks on the mask in explorer to open it and checks the message if required. */ fun openMaskInExplorer( - maskName: String, expectedError: String, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + maskName: String, expectedError: String, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = - with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(maskName).doubleClick() - Thread.sleep(20000) - var allText = "" - find(viewTree).findAllText().forEach { allText += it.text } - if (expectedError.isEmpty().not()) { - allText.shouldContain(expectedError) - } else { - allText.shouldNotContain("Error") + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(maskName).doubleClick() + Thread.sleep(20000) + var allText = "" + find(viewTree).findAllText().forEach { allText += it.text } + if (expectedError.isEmpty().not()) { + allText.shouldContain(expectedError) + } else { + allText.shouldNotContain("Error") + } + } } - } } - } /** * Double-clicks on the job filter in explorer to open it and checks the message if required. */ fun openJobFilterInExplorer( - filter: Triple, expectedError: String, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + filter: Triple, expectedError: String, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = with(remoteRobot) { - val textToFind = if (filter.third == "") { - "PREFIX=${filter.first} OWNER=${filter.second}".uppercase() - } else { - "JobID=${filter.third}" - } - ideFrameImpl(projectName, fixtureStack) { - explorer { - jesExplorer.click() - find(viewTree).findText(textToFind).doubleClick() - Thread.sleep(2000) - waitFor(Duration.ofSeconds(20)) { find(viewTree).hasText("loading…").not() } - if (expectedError.isEmpty().not()) { - find(viewTree).findText(expectedError) - } else { - find(viewTree).findAllText().shouldNotContain("Error") - } + val textToFind = if (filter.third == "") { + "PREFIX=${filter.first} OWNER=${filter.second}".uppercase() + } else { + "JobID=${filter.third}" + } + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + jesExplorer.click() + find(viewTree).findText(textToFind).doubleClick() + Thread.sleep(2000) + waitFor(Duration.ofSeconds(20)) { find(viewTree).hasText("loading…").not() } + if (expectedError.isEmpty().not()) { + find(viewTree).findText(expectedError) + } else { + find(viewTree).findAllText().shouldNotContain("Error") + } + } } - } } /** * Double-clicks on job filter to close it in explorer. */ fun closeFilterInExplorer( - filter: Triple, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + filter: Triple, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = - with(remoteRobot) { - val textToFind = if (filter.third == "") { - "PREFIX=${filter.first} OWNER=${filter.second}".uppercase() - } else { - "JobID=${filter.third}" - } - ideFrameImpl(projectName, fixtureStack) { - explorer { - jesExplorer.click() - find(viewTree).findText(textToFind).doubleClick() - } + with(remoteRobot) { + val textToFind = if (filter.third == "") { + "PREFIX=${filter.first} OWNER=${filter.second}".uppercase() + } else { + "JobID=${filter.third}" + } + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + jesExplorer.click() + find(viewTree).findText(textToFind).doubleClick() + } + } } - } /** * Double-clicks on mask to close it in explorer. */ fun closeMaskInExplorer( - maskName: String, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + maskName: String, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = - with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(maskName).doubleClick() - } + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(maskName).doubleClick() + } + } } - } /** * Checks that the mask or the working set is not displayed in explorer. */ fun checkItemWasDeletedWSRefreshed( - deletedItem: String, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + deletedItem: String, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - shouldThrow { - find(viewTree).findText(deletedItem) - } + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + shouldThrow { + find(viewTree).findText(deletedItem) + } + } } - } } /** * Checks that the job filter is not displayed in explorer. */ fun checkFilterWasDeletedJWSRefreshed( - deletedFilter: Triple, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + deletedFilter: Triple, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = with(remoteRobot) { - val textToFind = if (deletedFilter.third == "") { - "PREFIX=${deletedFilter.first} OWNER=${deletedFilter.second}" - } else { - "JobID=${deletedFilter.third}" - } - ideFrameImpl(projectName, fixtureStack) { - explorer { - jesExplorer.click() - shouldThrow { - find(viewTree).findText(textToFind) - } + val textToFind = if (deletedFilter.third == "") { + "PREFIX=${deletedFilter.first} OWNER=${deletedFilter.second}" + } else { + "JobID=${deletedFilter.third}" + } + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + jesExplorer.click() + shouldThrow { + find(viewTree).findText(textToFind) + } + } } - } } /** * Deletes dataset via context menu. */ fun deleteDataset( - datasetName: String, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + datasetName: String, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findAllText(datasetName).last().rightClick() - } - actionMenuItem(remoteRobot, "Delete").click() - dialog("Confirm Files Deletion") { - clickButton("Yes") + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findAllText(datasetName).last().rightClick() + } + actionMenuItem(remoteRobot, "Delete").click() + dialog("Confirm Files Deletion") { + clickButton(YES_TEXT) + } +// Thread.sleep(3000) } - Thread.sleep(3000) - } } /** * Opens the file and copies it's content. */ fun openLocalFileAndCopyContent( - filePath: String, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + filePath: String, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - actionMenu(remoteRobot, "File").click() - runJs( - """ + ideFrameImpl(PROJECT_NAME, fixtureStack) { + actionMenu(remoteRobot, "File").click() + runJs( + """ const point = new java.awt.Point(${locationOnScreen.x}, ${locationOnScreen.y}); robot.moveMouse(component, point); """ - ) - actionMenuItem(remoteRobot, "Open...").click() - Thread.sleep(3000) - dialog("Open File or Project") { - textField(byXpath("//div[@class='BorderlessTextField']")).text = - filePath - Thread.sleep(5000) - clickButton("OK") - } - with(textEditor()) { - keyboard { - hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_A) - hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_C) - hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_F4) - } + ) + actionMenuItem(remoteRobot, "Open...").click() + Thread.sleep(3000) + dialog("Open File or Project") { + textField(byXpath("//div[@class='BorderlessTextField']")).text = + filePath + Thread.sleep(5000) + clickButton("OK") + } + with(textEditor()) { + keyboard { + hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_A) + hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_C) + hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_F4) + } + } } - } } /** * Submits the job via context menu. */ fun submitJob( - jobName: String, projectName: String, - fixtureStack: MutableList, remoteRobot: RemoteRobot + jobName: String, + fixtureStack: MutableList, remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(jobName).rightClick() + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(jobName).rightClick() + } + actionMenuItem(remoteRobot, "Submit Job").click() } - actionMenuItem(remoteRobot, "Submit Job").click() - } } /** @@ -721,37 +701,37 @@ fun submitJob( */ fun createMemberAndPasteContent( datasetName: String, - memberName: String, projectName: String, + memberName: String, fixtureStack: MutableList, - remoteRobot: RemoteRobot + remoteRobot: RemoteRobot, ) = - with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findAllText(datasetName).last().rightClick() - } - actionMenu(remoteRobot, "New").click() - actionMenuItem(remoteRobot, "Member").click() - dialog("Create Member") { - find(byXpath("//div[@class='JBTextField']")).text = memberName - } - clickButton("OK") - Thread.sleep(5000) - explorer { - find(viewTree).findAllText(memberName).last().doubleClick() - } - with(textEditor()) { - keyboard { - hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_V) - Thread.sleep(2000) - hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_SHIFT, KeyEvent.VK_S) - Thread.sleep(2000) - hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_F4) + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findAllText(datasetName).last().rightClick() + } + actionMenu(remoteRobot, NEW_POINT_TEXT).click() + actionMenuItem(remoteRobot, "Member").click() + dialog("Create Member") { + find(datasetNameInputLoc).text = memberName + } + clickButton(OK_TEXT) + Thread.sleep(5000) + explorer { + find(viewTree).findAllText(memberName).last().doubleClick() + } + with(textEditor()) { + keyboard { + hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_V) + Thread.sleep(2000) + hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_SHIFT, KeyEvent.VK_S) + Thread.sleep(2000) + hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_F4) + } + } } - } } - } /** * Allocates a PDS dataset and creates a mask for it. @@ -760,81 +740,93 @@ fun createMemberAndPasteContent( fun allocatePDSAndCreateMask( wsName: String, datasetName: String, - projectName: String, fixtureStack: MutableList, closableFixtureCollector: ClosableFixtureCollector, remoteRobot: RemoteRobot, maskName: String? = null, directory: Int = 1, - openWs: Boolean = true + openWs: Boolean = true, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - if (maskName != null) { - createMask(wsName, fixtureStack, closableFixtureCollector) - createMaskDialog(fixtureStack) { - createMask(Pair(maskName, "z/OS")) - clickButton("OK") - } - } + ideFrameImpl(PROJECT_NAME, fixtureStack) { + if (maskName != null) { + createMask(wsName, fixtureStack, closableFixtureCollector) + createMaskDialog(fixtureStack) { + createMask(Pair(maskName, ZOS_MASK)) + clickButton(OK_TEXT) + } + } - explorer { - fileExplorer.click() - find(viewTree).findText(wsName).rightClick() - } - actionMenu(remoteRobot, "New").click() - actionMenuItem(remoteRobot, "Dataset").click() - allocateDatasetDialog(fixtureStack) { - allocateDataset(datasetName, "PO", "TRK", 10, 1, directory, "VB", 255, 6120) - clickButton("OK") - Thread.sleep(500) - } + explorer { + fileExplorer.click() + find(viewTree).findText(wsName).rightClick() + } + actionMenu(remoteRobot, NEW_POINT_TEXT).click() + actionMenuItem(remoteRobot, DATASET_POINT_TEXT).click() + allocateDatasetDialog(fixtureStack) { + allocateDataset(datasetName, PO_ORG_FULL, "TRK", 10, 1, directory, "VB", 255, 6120) + clickButton(OK_TEXT) + Thread.sleep(500) + } - val textToFind = "Dataset ${datasetName.uppercase()} Has Been Created" - val dialog = find(byXpath("//div[@class='MyDialog']")) - val dialogContents = dialog.findAllText().map(RemoteText::text).joinToString("") - val hasText = dialogContents.contains(textToFind) - if (!hasText) { - throw Exception("Text is not found in dialog: $textToFind") +// val textToFind = "Dataset ${datasetName.uppercase()} Has Been Created" +// val dialog = find(byXpath("//div[@class='MyDialog']")) +// val dialogContents = dialog.findAllText().map(RemoteText::text).joinToString("") +// val hasText = dialogContents.contains(textToFind) +// if (!hasText) { +// throw Exception("Text is not found in dialog: $textToFind") +// } + +// if (maskName != null) { +// clickButton("No") +// Thread.sleep(200) +// } else { +// clickButton("Yes") +// Thread.sleep(200) +// } } - - if (maskName != null) { - clickButton("No") - Thread.sleep(200) - } else { - clickButton("Yes") - Thread.sleep(200) + if (openWs) { + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) } - } - if (openWs) { - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - } } /** * Creates working set without masks. */ fun createWsWithoutMask( - projectName: String, wsName: String, connectionName: String, fixtureStack: MutableList, closableFixtureCollector: ClosableFixtureCollector, - remoteRobot: RemoteRobot + remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - createWSFromContextMenu(fixtureStack, closableFixtureCollector) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName) - clickButton("OK") - Thread.sleep(3000) - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - EMPTY_DATASET_MESSAGE - ) - clickButton("OK") - Thread.sleep(3000) + ideFrameImpl(PROJECT_NAME, fixtureStack) { + createWSFromContextMenu(fixtureStack, closableFixtureCollector) + addWorkingSetDialog(fixtureStack) { + addWorkingSet(wsName, connectionName) + clickButton(OK_TEXT) + } + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + } +} + +/** + * Creates working set without masks from action button. + */ +fun createWsWithoutMaskActionButton( + wsName: String, + connectionName: String, + fixtureStack: MutableList, + closableFixtureCollector: ClosableFixtureCollector, + remoteRobot: RemoteRobot, +) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + callCreateWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) + addWorkingSetDialog(fixtureStack) { + addWorkingSet(wsName, connectionName) + clickButton(OK_TEXT) + } + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } } /** @@ -843,57 +835,69 @@ fun createWsWithoutMask( fun allocateMemberForPDS( datasetName: String, memberName: String, - projectName: String, fixtureStack: MutableList, - remoteRobot: RemoteRobot + remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(datasetName).rightClick() - } + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(datasetName).rightClick() + } + + actionMenu(remoteRobot, "New").click() + actionMenuItem(remoteRobot, "Member").click() + createMemberDialog(fixtureStack) { + createMember(memberName) + clickButton("OK") +// Thread.sleep(5000) - actionMenu(remoteRobot, "New").click() - actionMenuItem(remoteRobot, "Member").click() - createMemberDialog(fixtureStack) { - createMember(memberName) - clickButton("OK") - Thread.sleep(5000) + } } - } } +fun closeNotificztion(fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot){ + ideFrameImpl(PROJECT_NAME, fixtureStack) { + find(closeNotificationLoc).click() + } +} /** * Allocates a sequential dataset. */ fun allocateDataSet( wsName: String, datasetName: String, - projectName: String, fixtureStack: MutableList, - remoteRobot: RemoteRobot + remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(wsName).rightClick() - } - actionMenu(remoteRobot, "New").click() - actionMenuItem(remoteRobot, "Dataset").click() - allocateDatasetDialog(fixtureStack) { - allocateDataset(datasetName, "PS", "TRK", 10, 1, 0, "VB", 255, 6120) - clickButton("OK") - Thread.sleep(10000) - } - find(byXpath("//div[@class='MyDialog']")).findText("Dataset $datasetName Has Been Created") - clickButton("No") - explorer { - fileExplorer.click() - find(viewTree).findText(wsName).rightClick() + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(wsName).rightClick() + } + actionMenu(remoteRobot, "New").click() + actionMenuItem(remoteRobot, "Dataset").click() + allocateDatasetDialog(fixtureStack) { + allocateDataset(datasetName, POE_ORG_FULL, "TRK", 10, 1, 1, "VB", 255, 6120) + clickButton(OK_TEXT) + Thread.sleep(3000) + } +// try { +// find(myDialogXpathLoc).findText("Dataset $datasetName Has Been ") +// } +// catch (e: NoSuchElementException) { +// find(myDialogXpathLoc).findText("Dataset $datasetName Has Been Created") +// } +// finally{ +// clickButton(NO_TEXT)} +// find(myDialogXpathLoc).findText("Dataset $datasetName Has Been Created") +// clickButton(NO_TEXT) + explorer { + fileExplorer.click() + find(viewTree).findText(wsName).rightClick() + } + actionMenuItem(remoteRobot, REFRESH_POINT_TEXT).click() +// Thread.sleep(3000) } - actionMenuItem(remoteRobot, "Refresh").click() - Thread.sleep(3000) - } } /** @@ -903,40 +907,39 @@ fun checkErrorNotification( errorHeader: String, errorType: String, errorDetail: String, - projectName: String, fixtureStack: MutableList, - remoteRobot: RemoteRobot + remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - var errorMessage = "" - try { - find(byXpath("//div[@class='LinkLabel']")).click() - } catch (e: WaitForConditionTimeoutException) { - e.message.shouldContain("Failed to find 'ComponentFixture' by '//div[@class='LinkLabel']'") - } - find(byXpath("//div[@javaclass='javax.swing.JLabel']")).findText(errorHeader) - find(byXpath("//div[@class='JEditorPane']")).findAllText().forEach { - errorMessage += it.text - } - find(byXpath("//div[@tooltiptext.key='tooltip.close.notification']")).click() - if (!(errorMessage.contains(errorType) && errorMessage.contains(errorDetail))) { - throw Exception("Error message is different from expected") + ideFrameImpl(PROJECT_NAME, fixtureStack) { + var errorMessage = "" + try { + find(linkLoc).click() + } catch (e: WaitForConditionTimeoutException) { + e.message.shouldContain("Failed to find 'ComponentFixture' by '//div[@class='LinkLabel']'") + } + find(errorDetailHeaderLoc).findText(errorHeader) + find(errorDetailBodyLoc).findAllText().forEach { + errorMessage += it.text + } + find(closeDialogLoc).click() + find(closeDialogLoc).click() + if (!(errorMessage.contains(errorType) && errorMessage.contains(errorDetail))) { + throw Exception("Error message is different from expected") + } } - } } /** * Creates uss file or directory for provided mask name */ fun createUssFile( - ussMaskName: String, - fileName: String, - fileType: UssFileType, - projectName: String, - fixtureStack: MutableList, - remoteRobot: RemoteRobot + ussMaskName: String, + fileName: String, + fileType: UssFileType, + fixtureStack: MutableList, + remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(ussMaskName).rightClick() @@ -951,6 +954,22 @@ fun createUssFile( } } +fun buildFinalListDatasetJson(mapListDatasets: MutableMap): String { + var result = "{}" + if (mapListDatasets.isNotEmpty()) { + var listDatasetsJson = "{\"items\":[" + mapListDatasets.forEach { + listDatasetsJson += it.value + } + result = listDatasetsJson.dropLast(1) + "],\n" + + " \"returnedRows\": ${mapListDatasets.size},\n" + + " \"totalRows\": ${mapListDatasets.size},\n" + + " \"JSONversion\": 1\n" + + "}" + } + return result +} + /** * Builds json list of objects based on provided map */ @@ -986,177 +1005,204 @@ fun buildResponseListJson( * Starts mock server for UI tests. */ fun startMockServer() { - val localhost = InetAddress.getByName("localhost").canonicalHostName - val localhostCertificate = HeldCertificate.Builder() - .addSubjectAlternativeName(localhost) - .duration(10, TimeUnit.MINUTES) - .build() - val serverCertificates = HandshakeCertificates.Builder() - .heldCertificate(localhostCertificate) - .build() - mockServer = MockWebServer() - responseDispatcher = MockResponseDispatcher() - mockServer.dispatcher = responseDispatcher - mockServer.useHttps(serverCertificates.sslSocketFactory(), false) - mockServer.start() + val localhost = InetAddress.getByName("localhost").canonicalHostName + val localhostCertificate = HeldCertificate.Builder() + .addSubjectAlternativeName(localhost) + .duration(10, TimeUnit.MINUTES) + .build() + val serverCertificates = HandshakeCertificates.Builder() + .heldCertificate(localhostCertificate) + .build() + mockServer = MockWebServer() + responseDispatcher = MockResponseDispatcher() +// injectDispatcher = InjectDispatcher() + mockServer.dispatcher = responseDispatcher + mockServer.useHttps(serverCertificates.sslSocketFactory(), false) + mockServer.start() } /** * Creates working set and a mask. */ fun createWsAndMask( - projectName: String, wsName: String, masks: List>, connectionName: String, fixtureStack: MutableList, closableFixtureCollector: ClosableFixtureCollector, - remoteRobot: RemoteRobot + remoteRobot: RemoteRobot, ) = with(remoteRobot) { - createWsWithoutMask(projectName, wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - masks.forEach { mask -> - createMask(wsName, fixtureStack, closableFixtureCollector) - createMaskDialog(fixtureStack) { - createMask(mask) - Thread.sleep(3000) - clickButton("OK") - } - closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + createWsWithoutMaskActionButton(wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { + masks.forEach { mask -> + createMask(wsName, fixtureStack, closableFixtureCollector) + createMaskDialog(fixtureStack) { + createMask(mask) +// Thread.sleep(3000) + clickButton(OK_TEXT) + } + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + } } - } } /** * Creates json to list dataset. */ fun listDS(dsName: String, dsNtp: String, dsOrg: String): String { - return "{\n" + - " \"dsname\": \"${dsName}\",\n" + - " \"blksz\": \"3200\",\n" + - " \"catnm\": \"TEST.CATALOG.MASTER\",\n" + - " \"cdate\": \"2021/11/15\",\n" + - " \"dev\": \"3390\",\n" + - " \"dsntp\": \"${dsNtp}\",\n" + - " \"dsorg\": \"${dsOrg}\",\n" + - " \"edate\": \"***None***\",\n" + - " \"extx\": \"1\",\n" + - " \"lrecl\": \"255\",\n" + - " \"migr\": \"NO\",\n" + - " \"mvol\": \"N\",\n" + - " \"ovf\": \"NO\",\n" + - " \"rdate\": \"2021/11/17\",\n" + - " \"recfm\": \"VB\",\n" + - " \"sizex\": \"10\",\n" + - " \"spacu\": \"TRACKS\",\n" + - " \"used\": \"1\",\n" + - " \"vol\": \"TESTVOL\",\n" + - " \"vols\": \"TESTVOL\"\n" + - " }," + return "{\n" + + " \"dsname\": \"${dsName}\",\n" + + " \"blksz\": \"3200\",\n" + + " \"catnm\": \"TEST.CATALOG.MASTER\",\n" + + " \"cdate\": \"2021/11/15\",\n" + + " \"dev\": \"3390\",\n" + + " \"dsntp\": \"${dsNtp}\",\n" + + " \"dsorg\": \"${dsOrg}\",\n" + + " \"edate\": \"***None***\",\n" + + " \"extx\": \"1\",\n" + + " \"lrecl\": \"255\",\n" + + " \"migr\": \"NO\",\n" + + " \"mvol\": \"N\",\n" + + " \"ovf\": \"NO\",\n" + + " \"rdate\": \"2021/11/17\",\n" + + " \"recfm\": \"VB\",\n" + + " \"sizex\": \"10\",\n" + + " \"spacu\": \"TRACKS\",\n" + + " \"used\": \"1\",\n" + + " \"vol\": \"TESTVOL\",\n" + + " \"vols\": \"TESTVOL\"\n" + + " }," } /** * Creates valid connection to mock server. */ fun createValidConnectionWithMock( - testInfo: TestInfo, connectionName: String, projectName: String, fixtureStack: MutableList, - closableFixtureCollector: ClosableFixtureCollector, remoteRobot: RemoteRobot + testInfo: TestInfo, connectionName: String, fixtureStack: MutableList, + closableFixtureCollector: ClosableFixtureCollector, remoteRobot: RemoteRobot, ) = with(remoteRobot) { - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_info", - { it?.requestLine?.contains("zosmf/info") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_resttopology", - { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - createConnection( - projectName, - fixtureStack, - closableFixtureCollector, - connectionName, - true, - remoteRobot, - "https://${mockServer.hostName}:${mockServer.port}" - ) + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_info", + { it?.requestLine?.contains("zosmf/info") ?: false }, + { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } + ) + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_resttopology", + { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, + { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } + ) + createConnection( + fixtureStack, + closableFixtureCollector, + connectionName, + true, + remoteRobot, + "https://${mockServer.hostName}:${mockServer.port}" + ) } fun createEmptyDatasetMember( datasetName: String, memberName: String, - projectName: String, fixtureStack: MutableList, - remoteRobot: RemoteRobot + fixtureStack: MutableList, + remoteRobot: RemoteRobot, ) = - with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findAllText(datasetName).last().rightClick() - } - actionMenu(remoteRobot, "New").click() - actionMenuItem(remoteRobot, "Member").click() - dialog("Create Member") { - find(byXpath("//div[@class='JBTextField']")).text = memberName - } - clickButton("OK") + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findAllText(datasetName).last().rightClick() + } + actionMenu(remoteRobot, "New").click() + actionMenuItem(remoteRobot, "Member").click() + dialog("Create Member") { + find(byXpath("//div[@class='JBTextField']")).text = memberName + } + clickButton("OK") + } } - } /** * Pastes content to dataset member from buffer. */ fun pasteContent( memberName: String, - projectName: String, fixtureStack: MutableList, - remoteRobot: RemoteRobot + fixtureStack: MutableList, + remoteRobot: RemoteRobot, ) = - with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - find(viewTree).findAllText(memberName).last().doubleClick() - Thread.sleep(2000) - } - with(textEditor()) { - keyboard { - hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_V) - Thread.sleep(2000) - hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_SHIFT, KeyEvent.VK_S) - Thread.sleep(2000) - hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_F4) + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + find(viewTree).findAllText(memberName).last().doubleClick() + Thread.sleep(2000) + } + with(textEditor()) { + keyboard { + hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_V) + Thread.sleep(2000) + hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_SHIFT, KeyEvent.VK_S) + Thread.sleep(2000) + hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_F4) + } + } } - } } - } /** * Creates json for submitted job. */ fun setBodyJobSubmit(jobName: String, jobStatus: JobStatus): String { - return "{\n" + - " \"owner\": \"${ZOS_USERID.uppercase()}\",\n" + - " \"phase\": 14,\n" + - " \"subsystem\": \"JES2\",\n" + - " \"phase-name\": \"Job is actively executing\",\n" + - " \"job-correlator\": \"J0007380S0W1....DB23523B.......:\",\n" + - " \"type\": \"JOB\",\n" + - " \"url\": \"https://${mockServer.hostName}:${mockServer.port}/zosmf/restjobs/jobs/J0007380S0W1....DB23523B.......%3A\",\n" + - " \"jobid\": \"JOB07380\",\n" + - " \"class\": \"B\",\n" + - " \"files-url\": \"https://${mockServer.hostName}:${mockServer.port}/zosmf/restjobs/jobs/J0007380S0W1....DB23523B.......%3A/files\",\n" + - " \"jobname\": \"${jobName}\",\n" + - " \"status\": \"${jobStatus}\",\n" + - " \"retcode\": null\n" + - "}\n" + return "{\n" + + " \"owner\": \"${ZOS_USERID.uppercase()}\",\n" + + " \"phase\": 14,\n" + + " \"subsystem\": \"JES2\",\n" + + " \"phase-name\": \"Job is actively executing\",\n" + + " \"job-correlator\": \"J0007380S0W1....DB23523B.......:\",\n" + + " \"type\": \"JOB\",\n" + + " \"url\": \"https://${mockServer.hostName}:${mockServer.port}/zosmf/restjobs/jobs/J0007380S0W1....DB23523B.......%3A\",\n" + + " \"jobid\": \"JOB07380\",\n" + + " \"class\": \"B\",\n" + + " \"files-url\": \"https://${mockServer.hostName}:${mockServer.port}/zosmf/restjobs/jobs/J0007380S0W1....DB23523B.......%3A/files\",\n" + + " \"jobname\": \"${jobName}\",\n" + + " \"status\": \"${jobStatus}\",\n" + + " \"retcode\": null\n" + + "}\n" +} +/** + * Click by button with text. + */ +fun clickByText(buttonText: String,fixtureStack: MutableList,remoteRobot: RemoteRobot)= with(remoteRobot){ + ideFrameImpl(PROJECT_NAME, fixtureStack) { + clickButton(buttonText) + } } fun replaceInJson(fileName: String, valuesMap: Map): String { var sourceJson = responseDispatcher.readMockJson(fileName) ?: "" - valuesMap.forEach{ entry -> + valuesMap.forEach { entry -> sourceJson = sourceJson.replace(entry.key, entry.value) } return sourceJson -} \ No newline at end of file +} + +fun createMask(wsName: String, maskName: String, fixtureStack: MutableList,closableFixtureCollector: ClosableFixtureCollector, mask_type: String, remoteRobot: RemoteRobot)= with(remoteRobot){ + ideFrameImpl(PROJECT_NAME, fixtureStack) { + createMask(wsName, fixtureStack, closableFixtureCollector) + createMaskDialog(fixtureStack) { + createMask(Pair(maskName, mask_type)) + clickButton(OK_TEXT) + } + } +} + +fun buildListMembersJson(listMembersInDataset: MutableList): String { + var members = "[ " + if (listMembersInDataset.isNotEmpty()) { + listMembersInDataset.forEach { members += "{\"member\": \"${it}\"}," } + } + members = members.dropLast(1) + "]" + return "{\"items\":$members,\"returnedRows\": ${listMembersInDataset.size},\"JSONversion\": 1}" +} + diff --git a/src/uiTest/kotlin/jes/CancelHoldReleaseJobTest.kt b/src/uiTest/kotlin/jes/CancelHoldReleaseJobTest.kt index 12605c4df..00d3da8d0 100644 --- a/src/uiTest/kotlin/jes/CancelHoldReleaseJobTest.kt +++ b/src/uiTest/kotlin/jes/CancelHoldReleaseJobTest.kt @@ -22,6 +22,8 @@ import com.intellij.remoterobot.utils.keyboard import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.extension.ExtendWith +import workingset.EMPTY_DATASET_MESSAGE +import workingset.PROJECT_NAME import java.awt.event.KeyEvent import java.io.File @@ -58,11 +60,10 @@ class CancelHoldReleaseJobTest { @BeforeAll fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) createValidConnectionWithMock( testInfo, connectionName, - projectName, fixtureStack, closableFixtureCollector, remoteRobot @@ -89,7 +90,7 @@ class CancelHoldReleaseJobTest { MockResponse().setBody(buildListMembersJson()) } ) - allocatePDSAndCreateMask(wsName, datasetName, projectName, fixtureStack, closableFixtureCollector, remoteRobot) + allocatePDSAndCreateMask(wsName, datasetName, fixtureStack, closableFixtureCollector, remoteRobot) createJob(testInfo, remoteRobot) } @@ -111,10 +112,10 @@ class CancelHoldReleaseJobTest { { it?.requestLine?.contains("DELETE /zosmf/restfiles/ds/${datasetName.uppercase()}") ?: false }, { MockResponse().setBody("{}") } ) - deleteDataset(datasetName, projectName, fixtureStack, remoteRobot) + deleteDataset(datasetName, fixtureStack, remoteRobot) mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @@ -210,7 +211,7 @@ class CancelHoldReleaseJobTest { * Closes notifications and jobs tabs in jobs panel if exists. */ private fun closeNotificationsAndJobsTabsIfExist(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { try { find(byXpath("//div[@javaclass='javax.swing.JLabel']")) .click() @@ -241,7 +242,7 @@ class CancelHoldReleaseJobTest { */ private fun getIdSubmittedJob(remoteRobot: RemoteRobot): String = with(remoteRobot) { var jobId = "" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { jobId = find(byXpath("//div[@class='Tree']")).findAllText()[2].text.trim() } return jobId @@ -257,7 +258,7 @@ class CancelHoldReleaseJobTest { JobAction.SUBMIT -> "CC 0000" else -> throw Exception("Unknown action") } - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { find(byXpath("//div[contains(@accessiblename.key, 'editor.accessible.name')]")).findText( "JOB $jobName($jobId) EXECUTED" ) @@ -274,7 +275,7 @@ class CancelHoldReleaseJobTest { * Closes tab in jobs panel. */ private fun closeJobTabInJobsPanel(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { findAll(byXpath("//div[@class='TabPanel'][.//div[@text='Jobs:']]//div[@class='ContentTabLabel']")).last() .findText( "//'$datasetName($jobName)'" @@ -407,7 +408,7 @@ class CancelHoldReleaseJobTest { else -> throw Exception("Unknown action") } - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { clickActionButton(byXpath("//div[@class='ActionButton' and @myaction='$myAction']")) } } @@ -426,7 +427,7 @@ class CancelHoldReleaseJobTest { JobAction.RELEASE -> "$jobName: $jobId has been released" JobAction.SUBMIT -> "Job $jobName has been submitted" } - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { find(byXpath("//div[@javaclass='javax.swing.JLabel']")).findText(textToFind) .click() find(byXpath("//div[@tooltiptext.key='tooltip.close.notification']")).click() @@ -437,7 +438,7 @@ class CancelHoldReleaseJobTest { * Creates empty working set. */ private fun createWS(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createWSFromContextMenu(fixtureStack, closableFixtureCollector) addWorkingSetDialog(fixtureStack) { addWorkingSet(wsName, connectionName) @@ -457,7 +458,7 @@ class CancelHoldReleaseJobTest { * Creates job in dataset. */ private fun createJob(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { - openLocalFileAndCopyContent(filePath + fileName, projectName, fixtureStack, remoteRobot) + openLocalFileAndCopyContent(filePath + fileName, fixtureStack, remoteRobot) Thread.sleep(3000) createMemberAndPasteContentWithMock(testInfo, datasetName, jobName, fileName, remoteRobot) } @@ -493,10 +494,10 @@ class CancelHoldReleaseJobTest { ) createEmptyDatasetMember( - datasetName, memberName, projectName, fixtureStack, remoteRobot + datasetName, memberName, fixtureStack, remoteRobot ) isFirstRequest = false - pasteContent(memberName, projectName, fixtureStack, remoteRobot) + pasteContent(memberName, fixtureStack, remoteRobot) Thread.sleep(3000) isFirst = false } @@ -576,7 +577,7 @@ class CancelHoldReleaseJobTest { Pair("port", mockServer.port.toString()), Pair("jobName", jobName), Pair("retCode", rc), Pair("jobStatus", jobStatus.name)))) } ) - submitJob(jobName, projectName, fixtureStack, remoteRobot) + submitJob(jobName, fixtureStack, remoteRobot) responseDispatcher.removeAllEndpoints() } diff --git a/src/uiTest/kotlin/jes/JesWorkingSetViaActionButtonTest.kt b/src/uiTest/kotlin/jes/JesWorkingSetViaActionButtonTest.kt index f45108d61..4864d12c8 100644 --- a/src/uiTest/kotlin/jes/JesWorkingSetViaActionButtonTest.kt +++ b/src/uiTest/kotlin/jes/JesWorkingSetViaActionButtonTest.kt @@ -26,6 +26,8 @@ import org.junit.jupiter.api.* import org.junit.jupiter.api.Assertions.assertFalse import org.junit.jupiter.api.Assertions.assertTrue import org.junit.jupiter.api.extension.ExtendWith +import workingset.EMPTY_DATASET_MESSAGE +import workingset.PROJECT_NAME import java.time.Duration @@ -49,7 +51,7 @@ class JesWorkingSetViaActionButtonTest { @BeforeAll fun setUpAll(remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) } /** @@ -58,8 +60,8 @@ class JesWorkingSetViaActionButtonTest { @AfterAll fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @@ -91,7 +93,7 @@ class JesWorkingSetViaActionButtonTest { { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJesWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) try { @@ -138,7 +140,7 @@ class JesWorkingSetViaActionButtonTest { @Order(2) fun testAddEmptyJesWorkingSetWithVeryLongNameViaActionButton(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName: String = "B".repeat(200) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJesWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName) @@ -163,7 +165,7 @@ class JesWorkingSetViaActionButtonTest { fun testAddJesWorkingSetWithOneValidFilterViaActionButton(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS1" val filter = Triple("*", ZOS_USERID, "") - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJesWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName, ZOS_USERID, filter) @@ -188,7 +190,7 @@ class JesWorkingSetViaActionButtonTest { { it?.requestLine?.contains("/zosmf/restjobs/jobs") ?: false }, { MockResponse().setBody("[]") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJesWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName, ZOS_USERID, validJobsFilters) @@ -197,12 +199,12 @@ class JesWorkingSetViaActionButtonTest { } closableFixtureCollector.closeOnceIfExists(AddJesWorkingSetDialog.name) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) validJobsFilters.forEach { - openJobFilterInExplorer(it, "", projectName, fixtureStack, remoteRobot) - closeFilterInExplorer(it, projectName, fixtureStack, remoteRobot) + openJobFilterInExplorer(it, "", fixtureStack, remoteRobot) + closeFilterInExplorer(it, fixtureStack, remoteRobot) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } // TODO: eliminate ZOS_USERID @@ -213,7 +215,7 @@ class JesWorkingSetViaActionButtonTest { @Order(5) fun testAddJesWorkingSetWithInvalidFiltersViaActionButton(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS3" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJesWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName) @@ -254,7 +256,7 @@ class JesWorkingSetViaActionButtonTest { @Order(6) fun testAddJesWorkingSetWithTheSameFiltersViaActionButton(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS3" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJesWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName, ZOS_USERID, Triple("*", ZOS_USERID.lowercase(), "")) @@ -291,7 +293,6 @@ class JesWorkingSetViaActionButtonTest { { MockResponse().setBody("[]") } ) createConnection( - projectName, fixtureStack, closableFixtureCollector, "invalid_connection", @@ -300,7 +301,7 @@ class JesWorkingSetViaActionButtonTest { "https://${mockServer.hostName}:$testPort" ) val jwsName = "JWS3" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJesWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, "invalid_connection", ZOS_USERID, Triple("*", ZOS_USERID, "")) @@ -309,13 +310,13 @@ class JesWorkingSetViaActionButtonTest { } closableFixtureCollector.closeOnceIfExists(AddJesWorkingSetDialog.name) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) findAll(byXpath("//div[@class='MyComponent'][.//div[@accessiblename='Invalid URL port: \"104431\"' and @class='JEditorPane']]")).forEach { it.click() findAll( byXpath("//div[@class='ActionButton' and @myicon= 'close.svg']") ).first().click() } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } } diff --git a/src/uiTest/kotlin/jes/JesWorkingSetViaContextMenuTest.kt b/src/uiTest/kotlin/jes/JesWorkingSetViaContextMenuTest.kt index 75268e6d5..8345bd081 100644 --- a/src/uiTest/kotlin/jes/JesWorkingSetViaContextMenuTest.kt +++ b/src/uiTest/kotlin/jes/JesWorkingSetViaContextMenuTest.kt @@ -28,6 +28,8 @@ import org.junit.jupiter.api.* import org.junit.jupiter.api.Assertions.assertFalse import org.junit.jupiter.api.Assertions.assertTrue import org.junit.jupiter.api.extension.ExtendWith +import workingset.EMPTY_DATASET_MESSAGE +import workingset.PROJECT_NAME import java.awt.event.KeyEvent import java.time.Duration @@ -53,7 +55,7 @@ class JesWorkingSetViaContextMenuTest { @BeforeAll fun setUpAll(remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) } /** @@ -62,8 +64,8 @@ class JesWorkingSetViaContextMenuTest { @AfterAll fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @@ -95,7 +97,7 @@ class JesWorkingSetViaContextMenuTest { { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJWSFromContextMenu(fixtureStack, closableFixtureCollector) try { if (dialog("Add JES Working Set Dialog").isShowing) { @@ -154,7 +156,7 @@ class JesWorkingSetViaContextMenuTest { fun testAddJesWorkingSetWithOneValidFilterViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS1" val filter = Triple("*", ZOS_USERID, "") - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJWSFromContextMenu(fixtureStack, closableFixtureCollector) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName, ZOS_USERID, filter) @@ -182,7 +184,7 @@ class JesWorkingSetViaContextMenuTest { @Order(5) fun testAddJWSWithInvalidFiltersViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS2" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJWSFromContextMenu(fixtureStack, closableFixtureCollector) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName) @@ -228,7 +230,7 @@ class JesWorkingSetViaContextMenuTest { { it?.requestLine?.contains("/zosmf/restjobs/jobs") ?: false }, { MockResponse().setBody("[]") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJWSFromContextMenu(fixtureStack, closableFixtureCollector) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName, ZOS_USERID, validJobsFilters) @@ -237,12 +239,12 @@ class JesWorkingSetViaContextMenuTest { } closableFixtureCollector.closeOnceIfExists(AddJesWorkingSetDialog.name) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) validJobsFilters.forEach { - openJobFilterInExplorer(it, "", projectName, fixtureStack, remoteRobot) - closeFilterInExplorer(it, projectName, fixtureStack, remoteRobot) + openJobFilterInExplorer(it, "", fixtureStack, remoteRobot) + closeFilterInExplorer(it, fixtureStack, remoteRobot) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } // TODO: eliminate ZOS_USERID @@ -266,7 +268,6 @@ class JesWorkingSetViaContextMenuTest { { MockResponse().setBody("[]") } ) createConnection( - projectName, fixtureStack, closableFixtureCollector, "invalid connection", @@ -274,7 +275,7 @@ class JesWorkingSetViaContextMenuTest { remoteRobot, "https://${mockServer.hostName}:$testPort" ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJWSFromContextMenu(fixtureStack, closableFixtureCollector) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, "invalid connection", ZOS_USERID, Triple("*", ZOS_USERID, "")) @@ -283,14 +284,14 @@ class JesWorkingSetViaContextMenuTest { } closableFixtureCollector.closeOnceIfExists(AddJesWorkingSetDialog.name) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) findAll(byXpath("//div[@class='MyComponent'][.//div[@accessiblename='Invalid URL port: \"104431\"' and @class='JEditorPane']]")).forEach { it.click() findAll( byXpath("//div[@class='ActionButton' and @myicon= 'close.svg']") ).first().click() } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } // TODO: eliminate ZOS_USERID @@ -301,7 +302,7 @@ class JesWorkingSetViaContextMenuTest { @Order(8) fun testAddJWSWithTheSameFiltersViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS4" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJWSFromContextMenu(fixtureStack, closableFixtureCollector) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName, ZOS_USERID, Triple("*", ZOS_USERID.lowercase(), "")) @@ -325,7 +326,7 @@ class JesWorkingSetViaContextMenuTest { @Order(9) fun testCreateInvalidFiltersViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "first jws" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJobsFilter(jwsName, fixtureStack, closableFixtureCollector) createJobsFilterDialog(fixtureStack) { invalidJobsFiltersMap.forEach { @@ -367,12 +368,12 @@ class JesWorkingSetViaContextMenuTest { validJobsFilters.forEach { createFilterFromContextMenu(jwsName, it, remoteRobot) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) validJobsFilters.forEach { - openJobFilterInExplorer(it, "", projectName, fixtureStack, remoteRobot) - closeFilterInExplorer(it, projectName, fixtureStack, remoteRobot) + openJobFilterInExplorer(it, "", fixtureStack, remoteRobot) + closeFilterInExplorer(it, fixtureStack, remoteRobot) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } /** @@ -382,7 +383,7 @@ class JesWorkingSetViaContextMenuTest { @Order(11) fun testCreateAlreadyExistsFilterViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "first jws" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJobsFilter(jwsName, fixtureStack, closableFixtureCollector) createJobsFilterDialog(fixtureStack) { createJobsFilter(Triple("*", ZOS_USERID, "")) @@ -413,9 +414,9 @@ class JesWorkingSetViaContextMenuTest { { it?.requestLine?.contains("/zosmf/restjobs/jobs") ?: false }, { MockResponse().setBody("[]") } ) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) - closeFilterInExplorer(Triple("*", ZOS_USERID, ""), projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) + closeFilterInExplorer(Triple("*", ZOS_USERID, ""), fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { editJWSFromContextMenu(jwsName, fixtureStack, closableFixtureCollector) editJesWorkingSetDialog(fixtureStack) { addFilter(ZOS_USERID, Triple("*", "$ZOS_USERID*", "")) @@ -424,8 +425,8 @@ class JesWorkingSetViaContextMenuTest { } closableFixtureCollector.closeOnceIfExists(EditJesWorkingSetDialog.name) } - openJobFilterInExplorer(Triple("*", ZOS_USERID, ""), "", projectName, fixtureStack, remoteRobot) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openJobFilterInExplorer(Triple("*", ZOS_USERID, ""), "", fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } /** @@ -437,8 +438,8 @@ class JesWorkingSetViaContextMenuTest { val jwsName = "first jws" val jobsFilter = Triple("*", ZOS_USERID, "") - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { editJWSFromContextMenu(jwsName, fixtureStack, closableFixtureCollector) editJesWorkingSetDialog(fixtureStack) { deleteFilter(jobsFilter) @@ -447,8 +448,8 @@ class JesWorkingSetViaContextMenuTest { } closableFixtureCollector.closeOnceIfExists(EditJesWorkingSetDialog.name) } - checkFilterWasDeletedJWSRefreshed(jobsFilter, projectName, fixtureStack, remoteRobot) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + checkFilterWasDeletedJWSRefreshed(jobsFilter, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } /** @@ -469,8 +470,8 @@ class JesWorkingSetViaContextMenuTest { Triple("TEST1", ZOS_USERID, ""), Triple("TEST***", "$ZOS_USERID***", "") ) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { editJWSFromContextMenu(jwsName, fixtureStack, closableFixtureCollector) editJesWorkingSetDialog(fixtureStack) { deleteAllFilters() @@ -483,8 +484,8 @@ class JesWorkingSetViaContextMenuTest { } closableFixtureCollector.closeOnceIfExists(EditJesWorkingSetDialog.name) } - deletedFilters.forEach { checkFilterWasDeletedJWSRefreshed(it, projectName, fixtureStack, remoteRobot) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + deletedFilters.forEach { checkFilterWasDeletedJWSRefreshed(it, fixtureStack, remoteRobot) } + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } /** @@ -495,8 +496,8 @@ class JesWorkingSetViaContextMenuTest { fun testEditJWSChangeConnectionToInvalidViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { val newConnectionName = "invalid connection" val jwsName = "JWS1" - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { editJWSFromContextMenu(jwsName, fixtureStack, closableFixtureCollector) editJesWorkingSetDialog(fixtureStack) { changeConnection(newConnectionName) @@ -514,7 +515,6 @@ class JesWorkingSetViaContextMenuTest { openJobFilterInExplorer( Triple("*", ZOS_USERID, ""), "Invalid URL port: \"104431\"", - projectName, fixtureStack, remoteRobot ) @@ -544,7 +544,6 @@ class JesWorkingSetViaContextMenuTest { { MockResponse().setBody("[]") } ) createConnection( - projectName, fixtureStack, closableFixtureCollector, newConnectionName, @@ -553,7 +552,7 @@ class JesWorkingSetViaContextMenuTest { "https://${mockServer.hostName}:${mockServer.port}" ) val jwsName = "JWS1" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { editJWSFromContextMenu(jwsName, fixtureStack, closableFixtureCollector) editJesWorkingSetDialog(fixtureStack) { changeConnection(newConnectionName) @@ -562,8 +561,8 @@ class JesWorkingSetViaContextMenuTest { } closableFixtureCollector.closeOnceIfExists(EditJesWorkingSetDialog.name) } - checkItemWasDeletedWSRefreshed("Invalid URL port: \"104431\"", projectName, fixtureStack, remoteRobot) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed("Invalid URL port: \"104431\"", fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } /** @@ -575,8 +574,8 @@ class JesWorkingSetViaContextMenuTest { val newJesWorkingSetName = "new jws name" val oldJesWorkingSetName = "JWS1" val alreadyExistsJesWorkingSetName = "JWS2" - openOrCloseJesWorkingSetInExplorer(oldJesWorkingSetName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseJesWorkingSetInExplorer(oldJesWorkingSetName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { editJWSFromContextMenu(oldJesWorkingSetName, fixtureStack, closableFixtureCollector) editJesWorkingSetDialog(fixtureStack) { renameJesWorkingSet(alreadyExistsJesWorkingSetName) @@ -592,8 +591,8 @@ class JesWorkingSetViaContextMenuTest { } closableFixtureCollector.closeOnceIfExists(EditJesWorkingSetDialog.name) } - checkItemWasDeletedWSRefreshed(oldJesWorkingSetName, projectName, fixtureStack, remoteRobot) - openOrCloseJesWorkingSetInExplorer(newJesWorkingSetName, projectName, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed(oldJesWorkingSetName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(newJesWorkingSetName, fixtureStack, remoteRobot) } /** @@ -610,11 +609,11 @@ class JesWorkingSetViaContextMenuTest { { it?.requestLine?.contains("/zosmf/restjobs/jobs") ?: false }, { MockResponse().setBody("[]") } ) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) editJobFilter(oldFilter, newFilter, remoteRobot) - checkFilterWasDeletedJWSRefreshed(oldFilter, projectName, fixtureStack, remoteRobot) - openJobFilterInExplorer(newFilter, "", projectName, fixtureStack, remoteRobot) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + checkFilterWasDeletedJWSRefreshed(oldFilter, fixtureStack, remoteRobot) + openJobFilterInExplorer(newFilter, "", fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } /** @@ -633,11 +632,11 @@ class JesWorkingSetViaContextMenuTest { @Order(20) fun testDeleteJWSViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS2" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { deleteJWSFromContextMenu(jwsName) clickButton("Yes") } - checkItemWasDeletedWSRefreshed(jwsName, projectName, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed(jwsName, fixtureStack, remoteRobot) } /** @@ -647,7 +646,7 @@ class JesWorkingSetViaContextMenuTest { @Order(21) fun testDeleteAllJWSViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsList = listOf("first jws", "A".repeat(200), "B12#$%^&*", "new jws name", "JWS3") - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { jesExplorer.click() find(viewTree).click() @@ -674,7 +673,7 @@ class JesWorkingSetViaContextMenuTest { remoteRobot: RemoteRobot ) = with(remoteRobot) { val textToFind = convertJobFilterToString(jobFilter) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { jesExplorer.click() find(viewTree).findText(jwsName).doubleClick() @@ -686,7 +685,7 @@ class JesWorkingSetViaContextMenuTest { dialog("Deletion Of Jobs Filter") { clickButton("Yes") } - checkFilterWasDeletedJWSRefreshed(jobFilter, projectName, fixtureStack, remoteRobot) + checkFilterWasDeletedJWSRefreshed(jobFilter, fixtureStack, remoteRobot) explorer { find(viewTree).findText(jwsName).doubleClick() } @@ -711,7 +710,7 @@ class JesWorkingSetViaContextMenuTest { remoteRobot: RemoteRobot ) = with(remoteRobot) { val textToFind = convertJobFilterToString(oldFilter) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { jesExplorer.click() find(viewTree).findText(textToFind).rightClick() @@ -731,7 +730,7 @@ class JesWorkingSetViaContextMenuTest { * Creates empty JES working set from context menu. */ private fun createJWS(jwsName: String, isUniqueName: Boolean, remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJWSFromContextMenu(fixtureStack, closableFixtureCollector) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName) @@ -766,7 +765,7 @@ class JesWorkingSetViaContextMenuTest { remoteRobot: RemoteRobot ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJobsFilter(jwsName, fixtureStack, closableFixtureCollector) createJobsFilterDialog(fixtureStack) { createJobsFilter(filter) diff --git a/src/uiTest/kotlin/jes/JesWorkingSetViaSettingsTest.kt b/src/uiTest/kotlin/jes/JesWorkingSetViaSettingsTest.kt index e904a1585..e102f99c3 100644 --- a/src/uiTest/kotlin/jes/JesWorkingSetViaSettingsTest.kt +++ b/src/uiTest/kotlin/jes/JesWorkingSetViaSettingsTest.kt @@ -24,6 +24,8 @@ import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.Assertions.assertFalse import org.junit.jupiter.api.extension.ExtendWith +import workingset.EMPTY_DATASET_MESSAGE +import workingset.PROJECT_NAME import java.time.Duration /** @@ -38,7 +40,6 @@ class JesWorkingSetViaSettingsTest { private var wantToClose = mutableListOf( "Settings Dialog", "Add JES Working Set Dialog", "Edit JES Working Set Dialog" ) - private val projectName = "untitled" private val connectionName = "valid connection" @@ -48,7 +49,7 @@ class JesWorkingSetViaSettingsTest { @BeforeAll fun setUpAll(remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) } /** @@ -57,8 +58,8 @@ class JesWorkingSetViaSettingsTest { @AfterAll fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @@ -78,7 +79,7 @@ class JesWorkingSetViaSettingsTest { @Test @Order(1) fun testAddJesWorkingSetWithoutConnectionViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -112,7 +113,6 @@ class JesWorkingSetViaSettingsTest { createValidConnectionWithMock( testInfo, connectionName, - projectName, fixtureStack, closableFixtureCollector, remoteRobot @@ -130,7 +130,7 @@ class JesWorkingSetViaSettingsTest { fun testAddJesWorkingSetWithOneValidFilterViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS1" val filter = Triple("*", ZOS_USERID, "") - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -168,7 +168,7 @@ class JesWorkingSetViaSettingsTest { @Order(5) fun testAddJWSWithInvalidFiltersViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS2" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { jesExplorer.click() settings(closableFixtureCollector, fixtureStack) @@ -225,7 +225,7 @@ class JesWorkingSetViaSettingsTest { { it?.requestLine?.contains("/zosmf/restjobs/jobs") ?: false }, { MockResponse().setBody("[]").setResponseCode(200) } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { jesExplorer.click() settings(closableFixtureCollector, fixtureStack) @@ -245,12 +245,12 @@ class JesWorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) validJobsFilters.forEach { - openJobFilterInExplorer(it, "", projectName, fixtureStack, remoteRobot) - closeFilterInExplorer(it, projectName, fixtureStack, remoteRobot) + openJobFilterInExplorer(it, "", fixtureStack, remoteRobot) + closeFilterInExplorer(it, fixtureStack, remoteRobot) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } // TODO: eliminate ZOS_USERID @@ -273,7 +273,6 @@ class JesWorkingSetViaSettingsTest { ) createConnection( - projectName, fixtureStack, closableFixtureCollector, "invalid_connection", @@ -282,7 +281,7 @@ class JesWorkingSetViaSettingsTest { "https://${mockServer.hostName}:$testPort" ) val jwsName = "JWS3" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -301,14 +300,14 @@ class JesWorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) findAll(byXpath("//div[@class='MyComponent'][.//div[@accessiblename='Invalid URL port: \"104431\"' and @class='JEditorPane']]")).forEach { it.click() findAll( byXpath("//div[@class='ActionButton' and @myicon= 'close.svg']") ).first().click() } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } // TODO: eliminate ZOS_USERID @@ -319,7 +318,7 @@ class JesWorkingSetViaSettingsTest { @Order(8) fun testAddJWSWithTheSameFiltersViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS4" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -360,9 +359,9 @@ class JesWorkingSetViaSettingsTest { { it?.requestLine?.contains("/zosmf/restjobs/jobs") ?: false }, { MockResponse().setBody("[]") } ) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) - closeFilterInExplorer(Triple("*", ZOS_USERID, ""), projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) + closeFilterInExplorer(Triple("*", ZOS_USERID, ""), fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -381,8 +380,8 @@ class JesWorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - openJobFilterInExplorer(newFilter, "", projectName, fixtureStack, remoteRobot) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + openJobFilterInExplorer(newFilter, "", fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } /** @@ -394,8 +393,8 @@ class JesWorkingSetViaSettingsTest { val jwsName = "JWS2" val filtersToBeDeleted = listOf(Triple("*", ZOS_USERID, ""), Triple("TEST**", ZOS_USERID, ""), Triple("TEST***", ZOS_USERID, "")) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -414,8 +413,8 @@ class JesWorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - filtersToBeDeleted.forEach { checkFilterWasDeletedJWSRefreshed(it, projectName, fixtureStack, remoteRobot) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + filtersToBeDeleted.forEach { checkFilterWasDeletedJWSRefreshed(it, fixtureStack, remoteRobot) } + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } /** @@ -434,8 +433,8 @@ class JesWorkingSetViaSettingsTest { Triple("TEST1", "$ZOS_USERID***", ""), Triple("TEST***", "$ZOS_USERID***", "") ) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -458,8 +457,8 @@ class JesWorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - filtersToBeDeleted.forEach { checkFilterWasDeletedJWSRefreshed(it, projectName, fixtureStack, remoteRobot) } - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + filtersToBeDeleted.forEach { checkFilterWasDeletedJWSRefreshed(it, fixtureStack, remoteRobot) } + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } /** @@ -469,8 +468,8 @@ class JesWorkingSetViaSettingsTest { @Order(12) fun testEditJWSChangeConnectionToInvalidViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS1" - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -498,7 +497,6 @@ class JesWorkingSetViaSettingsTest { openJobFilterInExplorer( Triple("*", ZOS_USERID, ""), "Invalid URL port: \"104431\"", - projectName, fixtureStack, remoteRobot ) @@ -529,7 +527,6 @@ class JesWorkingSetViaSettingsTest { { MockResponse().setBody("[]") } ) createConnection( - projectName, fixtureStack, closableFixtureCollector, newConnectionName, @@ -537,7 +534,7 @@ class JesWorkingSetViaSettingsTest { remoteRobot, "https://${mockServer.hostName}:${mockServer.port}" ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -556,8 +553,8 @@ class JesWorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - checkItemWasDeletedWSRefreshed("Invalid URL port: \"104431\"", projectName, fixtureStack, remoteRobot) - openOrCloseJesWorkingSetInExplorer(jwsName, projectName, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed("Invalid URL port: \"104431\"", fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(jwsName, fixtureStack, remoteRobot) } /** @@ -569,8 +566,8 @@ class JesWorkingSetViaSettingsTest { val newJesWorkingSetName = "new jws name" val oldJesWorkingSetName = "JWS1" val alreadyExistsJesWorkingSetName = "JWS2" - openOrCloseJesWorkingSetInExplorer(oldJesWorkingSetName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseJesWorkingSetInExplorer(oldJesWorkingSetName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -596,8 +593,8 @@ class JesWorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - checkItemWasDeletedWSRefreshed(oldJesWorkingSetName, projectName, fixtureStack, remoteRobot) - openOrCloseJesWorkingSetInExplorer(newJesWorkingSetName, projectName, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed(oldJesWorkingSetName, fixtureStack, remoteRobot) + openOrCloseJesWorkingSetInExplorer(newJesWorkingSetName, fixtureStack, remoteRobot) } /** @@ -607,7 +604,7 @@ class JesWorkingSetViaSettingsTest { @Order(15) fun testDeleteJWSViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val jwsName = "JWS2" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -620,7 +617,7 @@ class JesWorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - checkItemWasDeletedWSRefreshed(jwsName, projectName, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed(jwsName, fixtureStack, remoteRobot) } /** @@ -629,7 +626,7 @@ class JesWorkingSetViaSettingsTest { @Test @Order(16) fun testDeleteAllJWSViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -649,7 +646,7 @@ class JesWorkingSetViaSettingsTest { * Creates empty JES working set via settings. */ private fun createJWS(jwsName: String, isUniqueName: Boolean, remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } diff --git a/src/uiTest/kotlin/jes/PurgeJobTest.kt b/src/uiTest/kotlin/jes/PurgeJobTest.kt index 74441c2cc..50690f1de 100644 --- a/src/uiTest/kotlin/jes/PurgeJobTest.kt +++ b/src/uiTest/kotlin/jes/PurgeJobTest.kt @@ -23,6 +23,8 @@ import io.kotest.assertions.throwables.shouldThrow import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.extension.ExtendWith +import workingset.EMPTY_DATASET_MESSAGE +import workingset.PROJECT_NAME import java.awt.event.KeyEvent import java.io.File @@ -38,7 +40,6 @@ class PurgeJobTest { private var wantToClose = mutableListOf( "Allocate Dataset Dialog" ) - private val projectName = "untitled" private val connectionName = "valid connection" private val wsName = "WS1" @@ -56,11 +57,10 @@ class PurgeJobTest { @BeforeAll fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) createValidConnectionWithMock( testInfo, connectionName, - projectName, fixtureStack, closableFixtureCollector, remoteRobot @@ -87,7 +87,7 @@ class PurgeJobTest { MockResponse().setBody(buildListMembersJson()) } ) - allocatePDSAndCreateMask(wsName, datasetName, projectName, fixtureStack, closableFixtureCollector, remoteRobot) + allocatePDSAndCreateMask(wsName, datasetName, fixtureStack, closableFixtureCollector, remoteRobot) createJobs(testInfo, remoteRobot) createJWS(remoteRobot) } @@ -110,10 +110,10 @@ class PurgeJobTest { { it?.requestLine?.contains("DELETE /zosmf/restfiles/ds/${datasetName.uppercase()}") ?: false }, { MockResponse().setBody("{}") } ) - deleteDataset(datasetName, projectName, fixtureStack, remoteRobot) + deleteDataset(datasetName, fixtureStack, remoteRobot) mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @@ -162,7 +162,7 @@ class PurgeJobTest { Triple("TEST2", ZOS_USERID, ""), Triple("TEST3", ZOS_USERID, "") ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJesWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName, ZOS_USERID, filters) @@ -220,7 +220,7 @@ class PurgeJobTest { Thread.sleep(3000) checkNotificationJobPurged(jobName, jobId, true, remoteRobot) checkJobsOutputWasDeletedJWSRefreshed(jobName, jobId, remoteRobot) - closeFilterInExplorer(Triple(jobName, ZOS_USERID, ""), projectName, fixtureStack, remoteRobot) + closeFilterInExplorer(Triple(jobName, ZOS_USERID, ""), fixtureStack, remoteRobot) responseDispatcher.removeAllEndpoints() } @@ -230,7 +230,7 @@ class PurgeJobTest { private fun checkJobsOutputWasDeletedJWSRefreshed( jobName: String, jobId: String, remoteRobot: RemoteRobot ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { shouldThrow { find(viewTree).findText("$jobName ($jobId)") @@ -243,7 +243,7 @@ class PurgeJobTest { * Closes tab for job in jobs panel. */ private fun closeTabInJobsPanel(jobName: String, remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { find(byXpath("//div[@class='TabPanel'][.//div[@text='Jobs:']]//div[@class='ContentTabLabel']")).findText( "//'$datasetName($jobName)'" ).click() @@ -259,7 +259,7 @@ class PurgeJobTest { private fun createJobs(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { var n = 1 filesList.forEach { - openLocalFileAndCopyContent(filePath + it, projectName, fixtureStack, remoteRobot) + openLocalFileAndCopyContent(filePath + it, fixtureStack, remoteRobot) Thread.sleep(3000) createMemberAndPasteContentWithMock(testInfo, datasetName, "TEST$n", it, remoteRobot) n++ @@ -270,7 +270,7 @@ class PurgeJobTest { * Creates working set. */ private fun createWS(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createWSFromContextMenu(fixtureStack, closableFixtureCollector) addWorkingSetDialog(fixtureStack) { addWorkingSet(wsName, connectionName) @@ -290,7 +290,7 @@ class PurgeJobTest { * Purges job via action button. */ private fun purgeJobFromTerminal(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { clickActionButton(byXpath("//div[@class='ActionButton' and @myaction='Purge Job ()']")) } } @@ -344,14 +344,14 @@ class PurgeJobTest { Pair("port", mockServer.port.toString()), Pair("jobName", jobName), Pair("retCode", rc), Pair("jobStatus", "")))).setResponseCode(200) } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { jesExplorer.click() } } - openJobFilterInExplorer(Triple(jobName, ZOS_USERID, ""), "", projectName, fixtureStack, remoteRobot) + openJobFilterInExplorer(Triple(jobName, ZOS_USERID, ""), "", fixtureStack, remoteRobot) isFirstRequest = false - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { find(viewTree).findAllText { it.text.startsWith("$jobName ($jobId)") }.first() .rightClick() @@ -367,7 +367,7 @@ class PurgeJobTest { */ private fun getJobIdFromPanel(remoteRobot: RemoteRobot): String = with(remoteRobot) { var jobId = "" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { jobId = find(byXpath("//div[@class='Tree']")).findAllText()[2].text.trim() } return jobId @@ -377,7 +377,7 @@ class PurgeJobTest { * Checks notification that correct info is returned for submitted job. */ private fun checkNotificationJobSubmitted(jobName: String, remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { find(byXpath("//div[@javaclass='javax.swing.JLabel']")).findText("Job $jobName has been submitted") .click() find(byXpath("//div[@tooltiptext.key='tooltip.close.notification']")).click() @@ -398,7 +398,7 @@ class PurgeJobTest { } else { "Error purging $jobName: $jobId" } - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { find(byXpath("//div[@javaclass='javax.swing.JLabel']")).findText(textToFind) .click() find(byXpath("//div[@tooltiptext.key='tooltip.close.notification']")).click() @@ -457,9 +457,9 @@ class PurgeJobTest { { MockResponse().setBody("") } ) - createEmptyDatasetMember(datasetName, memberName, projectName, fixtureStack, remoteRobot) + createEmptyDatasetMember(datasetName, memberName, fixtureStack, remoteRobot) isFirstRequest = false - pasteContent(memberName, projectName, fixtureStack, remoteRobot) + pasteContent(memberName, fixtureStack, remoteRobot) Thread.sleep(3000) } @@ -517,7 +517,7 @@ class PurgeJobTest { Pair("port", mockServer.port.toString()), Pair("jobName", jobName), Pair("retCode", rc), Pair("jobStatus", "OUTPUT")))) } ) - submitJob(jobName, projectName, fixtureStack, remoteRobot) + submitJob(jobName, fixtureStack, remoteRobot) } private fun setBodyJobSubmit(jobName: String): String { diff --git a/src/uiTest/kotlin/jes/SubmitJobTest.kt b/src/uiTest/kotlin/jes/SubmitJobTest.kt index f88b291a8..468ea1554 100644 --- a/src/uiTest/kotlin/jes/SubmitJobTest.kt +++ b/src/uiTest/kotlin/jes/SubmitJobTest.kt @@ -21,6 +21,7 @@ import com.intellij.remoterobot.utils.keyboard import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.extension.ExtendWith +import workingset.PROJECT_NAME import java.awt.event.KeyEvent import java.io.File @@ -36,7 +37,6 @@ class SubmitJobTest { private var wantToClose = mutableListOf( "Allocate Dataset Dialog" ) - private val projectName = "untitled" private val connectionName = "valid connection" private val wsName = "WS1" @@ -51,16 +51,15 @@ class SubmitJobTest { @BeforeAll fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) createValidConnectionWithMock( testInfo, connectionName, - projectName, fixtureStack, closableFixtureCollector, remoteRobot ) - createWsWithoutMask(projectName, wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + createWsWithoutMask(wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) responseDispatcher.injectEndpoint( "${testInfo.displayName}_${datasetName}_restfiles", { it?.requestLine?.contains("POST /zosmf/restfiles/ds/${datasetName}") ?: false }, @@ -69,7 +68,6 @@ class SubmitJobTest { allocatePDSAndCreateMask( wsName, datasetName, - projectName, fixtureStack, closableFixtureCollector, remoteRobot, @@ -99,10 +97,10 @@ class SubmitJobTest { { it?.requestLine?.contains("DELETE /zosmf/restfiles/ds/${datasetName.uppercase()}") ?: false }, { MockResponse().setBody("{}") } ) - deleteDataset(datasetName, projectName, fixtureStack, remoteRobot) + deleteDataset(datasetName, fixtureStack, remoteRobot) mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @@ -170,7 +168,7 @@ class SubmitJobTest { Pair("port", mockServer.port.toString()), Pair("jobName", jobName), Pair("retCode", "CC 0000"), Pair("jobStatus", "OUTPUT")))).setResponseCode(200) } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { jesExplorer.click() settings(closableFixtureCollector, fixtureStack) @@ -204,7 +202,7 @@ class SubmitJobTest { { it?.requestLine?.contains("GET /zosmf/restjobs/jobs/${jobName}/JOB07380/files/") ?: false }, { MockResponse().setBody("").setResponseCode(200) } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { if (find(viewTree).findAllText { it.text.startsWith(jobName) }.size > 1) { find(viewTree).findAllText { it.text.startsWith(jobName) }.first().doubleClick() } @@ -233,7 +231,7 @@ class SubmitJobTest { remoteRobot: RemoteRobot ) = with(remoteRobot) { - openLocalFileAndCopyContent(filePath + fileName, projectName, fixtureStack, remoteRobot) + openLocalFileAndCopyContent(filePath + fileName, fixtureStack, remoteRobot) Thread.sleep(3000) createMemberAndPasteContentWithMock(testInfo, datasetName, jobName, fileName, remoteRobot) submitJobWithMock(testInfo, datasetName, jobName, rc, remoteRobot) @@ -290,7 +288,7 @@ class SubmitJobTest { Pair("port", mockServer.port.toString()), Pair("jobName", jobName), Pair("retCode", rc), Pair("jobStatus", "OUTPUT")))) } ) - submitJob(jobName, projectName, fixtureStack, remoteRobot) + submitJob(jobName, fixtureStack, remoteRobot) } private fun createMemberAndPasteContentWithMock( @@ -320,9 +318,9 @@ class SubmitJobTest { { MockResponse().setBody("") } ) - createEmptyDatasetMember(datasetName, memberName, projectName, fixtureStack, remoteRobot) + createEmptyDatasetMember(datasetName, memberName, fixtureStack, remoteRobot) isFirstRequest = false - pasteContent(memberName, projectName, fixtureStack, remoteRobot) + pasteContent(memberName, fixtureStack, remoteRobot) } @@ -331,7 +329,7 @@ class SubmitJobTest { */ private fun checkTabPanelAndConsole(jobName: String, jobId: String, rc: String, remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { find(byXpath("//div[contains(@accessiblename.key, 'editor.accessible.name')]")).findText( "JOB $jobName($jobId) EXECUTED" ) @@ -355,7 +353,7 @@ class SubmitJobTest { */ private fun getJobIdFromPanel(remoteRobot: RemoteRobot): String = with(remoteRobot) { var jobId = "" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { jobId = find(byXpath("//div[@class='Tree']")).findAllText()[2].text.trim() } return jobId @@ -365,7 +363,7 @@ class SubmitJobTest { * Checks notification that correct info is returned. */ private fun checkNotification(jobName: String, remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { find(byXpath("//div[@javaclass='javax.swing.JLabel']")).findText("Job $jobName has been submitted") .click() find(byXpath("//div[@tooltiptext.key='tooltip.close.notification']")).click() @@ -413,6 +411,6 @@ class SubmitJobTest { MockResponse().setBody(buildListMembersJson()) } ) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) } } \ No newline at end of file diff --git a/src/uiTest/kotlin/jes/ViewJwsPropertyTest.kt b/src/uiTest/kotlin/jes/ViewJwsPropertyTest.kt index 5d9db6368..1c62a0939 100644 --- a/src/uiTest/kotlin/jes/ViewJwsPropertyTest.kt +++ b/src/uiTest/kotlin/jes/ViewJwsPropertyTest.kt @@ -22,6 +22,8 @@ import com.intellij.remoterobot.search.locators.byXpath import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.extension.ExtendWith +import workingset.EMPTY_DATASET_MESSAGE +import workingset.PROJECT_NAME /** * Tests viewing job and spool file properties. @@ -33,7 +35,6 @@ class ViewJwsPropertyTest { private val closableFixtureCollector = ClosableFixtureCollector() private val fixtureStack = mutableListOf() - private val projectName = "untitled" private val connectionName = "con1" private val jwsName = "JWS name" private val jobsFilterName = Triple("*", ZOS_USERID.uppercase(), "") @@ -56,9 +57,8 @@ class ViewJwsPropertyTest { { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) createConnection( - projectName, fixtureStack, closableFixtureCollector, connectionName, @@ -66,7 +66,7 @@ class ViewJwsPropertyTest { remoteRobot, "https://${mockServer.hostName}:${mockServer.port}" ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createJWSFromContextMenu(fixtureStack, closableFixtureCollector) addJesWorkingSetDialog(fixtureStack) { addJesWorkingSet(jwsName, connectionName) @@ -94,7 +94,7 @@ class ViewJwsPropertyTest { "listSpoolFiles_restfiles", { it?.requestLine?.contains("GET /zosmf/restjobs/jobs/") ?: false }, { MockResponse().setBody(replaceInJson("getSingleSpoolFile", mapOf(Pair("jobName", jobName), Pair("spoolFileName", spoolFileName)))) }) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { jesExplorer.click() find(viewTree).findText(jwsName).doubleClick() @@ -110,8 +110,8 @@ class ViewJwsPropertyTest { fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @@ -122,7 +122,7 @@ class ViewJwsPropertyTest { @Test @Order(1) fun testViewJobProperties(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { find(viewTree).findAllText{ it.text.startsWith("$jobName (JOB07380)") }.first() .rightClick() @@ -146,7 +146,7 @@ class ViewJwsPropertyTest { @Test @Order(2) fun testViewSpoolFileProperties(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { jesExplorer.click() Thread.sleep(1000) diff --git a/src/uiTest/kotlin/settings/connection/ConnectionManager.kt b/src/uiTest/kotlin/settings/connection/ConnectionManager.kt index cc69a5cf6..084aebce9 100644 --- a/src/uiTest/kotlin/settings/connection/ConnectionManager.kt +++ b/src/uiTest/kotlin/settings/connection/ConnectionManager.kt @@ -23,6 +23,7 @@ import org.junit.jupiter.api.* import org.junit.jupiter.api.Assertions.assertFalse import org.junit.jupiter.api.Assertions.assertTrue import org.junit.jupiter.api.extension.ExtendWith +import workingset.PROJECT_NAME import java.time.Duration /** @@ -47,7 +48,7 @@ class ConnectionManager { @BeforeAll fun setUpAll(remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) } /** @@ -57,8 +58,8 @@ class ConnectionManager { fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @@ -80,7 +81,7 @@ class ConnectionManager { fun testAddWrongConnection(remoteRobot: RemoteRobot) = with(remoteRobot) { val host = "a.com" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -108,7 +109,7 @@ class ConnectionManager { @Order(2) fun testAddTwoConnectionsWithTheSameName(remoteRobot: RemoteRobot) = with(remoteRobot) { val connectionName = "a" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -164,7 +165,6 @@ class ConnectionManager { { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) createConnection( - projectName, fixtureStack, closableFixtureCollector, "valid connection1", @@ -190,7 +190,7 @@ class ConnectionManager { { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -228,7 +228,7 @@ class ConnectionManager { { it?.requestLine?.contains("zosmf/info") ?: false }, { MockResponse().setResponseCode(401) } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -270,7 +270,7 @@ class ConnectionManager { { it?.requestLine?.contains("zosmf/info") ?: false }, { MockResponse().setBody("Unable to find valid certification path to requested target") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -318,7 +318,6 @@ class ConnectionManager { { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) createConnection( - projectName, fixtureStack, closableFixtureCollector, "A".repeat(200), @@ -339,7 +338,7 @@ class ConnectionManager { { it?.requestLine?.contains("zosmf/info") ?: false }, { MockResponse().setBody("Please provide a valid URL to z/OSMF. Example: https://myhost.com:10443") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -375,7 +374,7 @@ class ConnectionManager { @Order(9) fun testEditConnectionFromValidToInvalid(remoteRobot: RemoteRobot) = with(remoteRobot) { val testPort = "104431" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -424,7 +423,7 @@ class ConnectionManager { { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -474,7 +473,7 @@ class ConnectionManager { { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -564,7 +563,6 @@ class ConnectionManager { { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) createConnection( - projectName, fixtureStack, closableFixtureCollector, connectionName, @@ -583,7 +581,7 @@ class ConnectionManager { */ private fun deleteConnection(connectionName: String, wsName: String, jwsName: String, remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -631,7 +629,7 @@ class ConnectionManager { { it?.requestLine?.contains("/zosmf/restjobs/jobs") ?: false }, { MockResponse().setBody("[]") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -661,7 +659,7 @@ class ConnectionManager { { it?.requestLine?.contains("zosmf/restfiles/ds?dslevel=") ?: false }, { MockResponse().setBody("{}") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } diff --git a/src/uiTest/kotlin/testutils/InjectDispatcher.kt.kt b/src/uiTest/kotlin/testutils/InjectDispatcher.kt.kt new file mode 100644 index 000000000..f3e0f9701 --- /dev/null +++ b/src/uiTest/kotlin/testutils/InjectDispatcher.kt.kt @@ -0,0 +1,209 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package workingset.testutils + +import okhttp3.mockwebserver.MockResponse +import auxiliary.* +import org.junit.jupiter.api.TestInfo +import workingset.DATA_SET_RENAME_FAILED + +//import testutils.MockResponseDispatcher + +fun injectRenameDataset(testInfo: TestInfo, dsFinalName: String, dsName: String){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles_rename_ds", + { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${dsFinalName}") ?: false }, + { MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${dsName}\"}}") } + ) +} + +fun injectRenameMember(testInfo: TestInfo, datasetName: String, memberNameNew: String, memberNameOld: String, handler: Boolean=false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles_rename_member", + { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${datasetName}(${memberNameNew})") ?: handler }, + { MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${datasetName}\",\"member\":\"${memberNameOld}\"}}") } + ) +} + +fun injectRenameMemberUnsuccessful(testInfo: TestInfo, datasetName: String, memberNameNew: String, memberNameOld: String, code: Int, rcCode: String, errorDetail:String, handler: Boolean=false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles_rename_member", + { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${datasetName}(${memberNameOld})") ?: handler }, + { + MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${datasetName}\",\"member\":\"${memberNameNew}\"}}") + .setResponseCode(code) + .setBody("{\"category\":\"4.0\",\"message\":\"Rename member failed\",\"rc\":$rcCode,\"details\":[\"ISRZ002 $errorDetail - Directory already contains the specified member name.\"],\"reason\":\"0.0\"}") + } + ) +} + +fun injectRenameDatasetUnsuccessful(testInfo: TestInfo, dsFinalName: String, anotherDsName: String, code: Int, rcCode: String, errorDetail:String, handler: Boolean=false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles_rename_ds", + { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${anotherDsName}") ?: handler }, + { + MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${dsFinalName}\"}}") + .setResponseCode(code) + .setBody("{\"category\":\"1.0\",\"message\":\"$DATA_SET_RENAME_FAILED\",\"rc\":\"$rcCode\",\"details\":[\"$errorDetail\"],\"reason\":\"6.0\"}") + } + ) +} + + +fun injectMemberList(testInfo: TestInfo, datasetName: String, listElem: List, listName: String? = null, isFirstRequest: Boolean= true){//= true){ + val requestName = if (listName != null) { + "${testInfo.displayName}_restfiles_listmembers$listName" + } else { + "${testInfo.displayName}_restfiles_listmembers" + } + + val body_headr = "{\"items\":[" + val body_tell = "],\"returnedRows\": ${listElem.size},\"JSONversion\": 1}" + val memebersString = listElem.joinToString(separator = ",") { "{\"member\": \"$it\"}" } + + responseDispatcher.injectEndpoint( + requestName, + { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${datasetName}/member") ?: false && isFirstRequest}, + { MockResponse().setBody(body_headr + memebersString + body_tell)} + ) +} +fun injectSingleMember(testInfo:TestInfo, datasetName: String, listMembersInDataset: MutableList){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles_listmembers", + { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${datasetName}/member") ?: false }, + { + MockResponse().setBody(buildListMembersJson(listMembersInDataset)) + } + ) +} +fun injectSingleSpecificMember(pdsName: String, memberName: String, responseCode: Int=204){ + responseDispatcher.injectEndpoint( + "allocateDatasetMember_restfiles", + { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/$pdsName($memberName)") ?: false }, + { MockResponse().setResponseCode(responseCode) } + ) +} + +fun injectListAllAllocatedDatasets(datasetMask: String, mapListDatasets: MutableMap, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "listAllAllocatedDatasets_restfiles", + { it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${datasetMask}*") ?: handler }, + { MockResponse().setBody(buildFinalListDatasetJson(mapListDatasets)) } + ) +} + +fun injectListEmptyData(testInfo:TestInfo, isDataset: Boolean=true, handler: Boolean = false){ + val pathType = if (isDataset) "ds?dslevel=" else "fs?path" + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles", + { it?.requestLine?.contains("GET /zosmf/restfiles/${pathType}") ?: handler }, + { MockResponse().setBody("{}") } + ) +} + +fun injectListAllAllocatedDatasetsWithContents(testInfo:TestInfo,datasetName: String, mapListDatasets: MutableMap, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles", + { + it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${datasetName}") + ?: false + }, + { MockResponse().setBody(buildResponseListJson(mapListDatasets, handler)) } + ) +} + +fun injectMemberContent(testInfo: TestInfo,datasetName:String, memberName:String, memberContent: String="", handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles_getmember", + { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${datasetName}($memberName)") ?: handler }, + { MockResponse().setBody(memberContent) } + ) +} + +fun injectPsDatasetContent(testInfo: TestInfo, datasetName:String, psContent:String="", handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles_getcontent", + { it?.requestLine?.contains("GET /zosmf/restfiles/ds/-(TESTVOL)/$datasetName") ?: handler }, + { MockResponse().setBody(psContent) } + ) +} + +fun injectAllocateUssFile(ussMaskName:String, ussFileName:String, responseCode: Int = 201, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "allocateUssFile_restfiles", + { it?.requestLine?.contains("POST /zosmf/restfiles/fs$ussMaskName/$ussFileName") ?: handler }, + { MockResponse().setResponseCode(responseCode) } + ) +} +fun injectListAllUssFiles(mapListUssFiles: Map, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "listAllUssFiles_restfiles", + { it?.requestLine?.contains("GET /zosmf/restfiles/fs?path") ?: handler }, + { MockResponse().setBody(buildResponseListJson(mapListUssFiles, true)) } + ) +} + +fun injectTestInfoRestTopology(testInfo: TestInfo, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_resttopology", + { it?.requestLine?.contains("zosmf/resttopology/systems") ?: handler }, + { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } + ) +} + + +fun injectTestInfo(testInfo: TestInfo,handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_info", + { it?.requestLine?.contains("zosmf/info") ?: handler }, + { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } + ) +} + +fun injectInvalidUrlPortInfo(testInfo: TestInfo,testPort:String, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_info", + { it?.requestLine?.contains("zosmf/info") ?: handler }, + { MockResponse().setBody("Invalid URL port: \"${testPort}\"") } + ) +} + +fun injectEmptyZosmfRestfilesPath(testInfo: TestInfo, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_second", + { it?.requestLine?.contains("zosmf/restfiles/") ?: handler }, + { MockResponse().setBody("{}") } + ) +} +//class InjectDispatcher: MockResponseDispatcher() { +// +// +// fun injectAllocationResultPo( +// recordFormatShort: String, +// dsName:String, +// dsOrganisationShort:String, +// recordLength:Int, +// handler: Boolean = false){ +// +// responseDispatcher.injectEndpoint( +// "testAllocateValid${dsOrganisationShort}Datasets_${recordFormatShort}_restfiles", +// { it?.requestLine?.contains("POST /zosmf/restfiles/ds/${dsName}") ?: handler }, +// { MockResponse().setBody("{\"dsorg\":\"${dsOrganisationShort}\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":1,\"recfm\":\"${recordFormatShort}\",\"blksize\":3200,\"lrecl\":${recordLength}}") } +// ) +// +// } +// fun injectRenameMember(testInfo: TestInfo, datasetName: String, memberNameNew: String, memberNameOld: String, handler: Boolean=false){ +// responseDispatcher.injectEndpoint( +// "${testInfo.displayName}_restfiles_rename_member", +// { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${datasetName}(${memberNameNew})") ?: false }, +// { MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${datasetName}\",\"member\":\"${memberNameOld}\"}}") } +// ) +// } +//} \ No newline at end of file diff --git a/src/uiTest/kotlin/testutils/MockResponseDispatcher.kt b/src/uiTest/kotlin/testutils/MockResponseDispatcher.kt index 797c0c4df..027b31d8b 100644 --- a/src/uiTest/kotlin/testutils/MockResponseDispatcher.kt +++ b/src/uiTest/kotlin/testutils/MockResponseDispatcher.kt @@ -9,10 +9,13 @@ */ package testutils - +//import auxiliary.buildFinalListDatasetJson +import auxiliary.responseDispatcher import okhttp3.mockwebserver.Dispatcher import okhttp3.mockwebserver.MockResponse import okhttp3.mockwebserver.RecordedRequest +import org.junit.jupiter.api.TestInfo +import java.io.File data class ValidationListElem( val name: String, @@ -20,7 +23,7 @@ data class ValidationListElem( val handler: (RecordedRequest?) -> MockResponse ) -class MockResponseDispatcher : Dispatcher() { +open class MockResponseDispatcher : Dispatcher() { private var validationList = mutableListOf() @@ -48,11 +51,153 @@ class MockResponseDispatcher : Dispatcher() { validationList.clear() } + fun injectAllocationResultPo( + datasetOrganization: String, + recordFormatShort: String, + dsName:String, + dsOrganisationShort:String, + recordLength:Int, + handler: Boolean = false){ + + responseDispatcher.injectEndpoint( + "testAllocateValid${datasetOrganization}Datasets_${recordFormatShort}_restfiles", + { it?.requestLine?.contains("POST /zosmf/restfiles/ds/${dsName}") ?: handler }, + { MockResponse().setBody("{\"dsorg\":\"${dsOrganisationShort}\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":1,\"recfm\":\"${recordFormatShort}\",\"blksize\":3200,\"lrecl\":${recordLength}}") } + ) + + } + + fun injectAllocationResultPds(pdsName: String, handler: Boolean = false){ + + responseDispatcher.injectEndpoint( + "allocatePds_restfiles", + { it?.requestLine?.contains("POST /zosmf/restfiles/ds/$pdsName") ?: handler }, + { MockResponse().setBody("{\"dsorg\":\"PDS\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":2,\"recfm\":\"VB\",\"blksize\":6120,\"lrecl\":255}") } + ) + + } + + fun injectAllocatedDatasets(datasetMask: String, body: String, datasetName:String, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "listAllocatedDatasets_restfiles_${datasetMask}", + { it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${datasetName}*") ?: handler }, + { MockResponse().setBody(body) } + ) + } + fun injectTestInfoForPdsDataset(testInfoDisplayedName: String, body: String, pdsName:String, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfoDisplayedName}_${pdsName}_restfiles", + { it?.requestLine?.contains("POST /zosmf/restfiles/ds/${pdsName}") ?: handler }, + { MockResponse().setBody(body) } + ) + } + + fun injectListMembers(body: String, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "listMembers_restfiles", + { it?.requestLine?.contains("/member") ?: handler }, + { MockResponse().setBody(body) } + ) + } + + fun injectListAllDatasetMembersRestfiles(pdsName: String, body: String, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "listAllDatasetMembers_restfiles", + { it?.requestLine?.contains("GET /zosmf/restfiles/ds/$pdsName/member") ?: handler }, + { MockResponse().setBody(body) } + ) + } + + fun injectDeleteDataset(datasetName: String, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "deleteDataset_restfiles_${datasetName}", + { it?.requestLine?.contains("DELETE /zosmf/restfiles/ds/${datasetName}") ?: handler }, + { MockResponse().setBody("{}") } + ) + } + + fun injectTestInfo(testInfo: TestInfo,handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_info", + { it?.requestLine?.contains("zosmf/info") ?: handler }, + { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } + ) + } + + fun injectTestInfoRestTopology(testInfo: TestInfo, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_resttopology", + { it?.requestLine?.contains("zosmf/resttopology/systems") ?: handler }, + { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } + ) + } + + fun injectMigratePdsRestFiles(pdsName:String, contains: String, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "migratePds_restfiles", + { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/$pdsName") ?: false && + it?.body?.toString()?.contains(contains) ?: handler }, + { MockResponse().setResponseCode(200) } + ) + } + + fun injectRecallPds(pdsName: String, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "migratePds_restfiles", + { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/$pdsName") ?: false && + it?.body?.toString()?.contains("hrecall") ?: handler }, + { MockResponse().setResponseCode(200) } + ) + } + + internal open fun injectRenameMember(testInfo: TestInfo, datasetName: String, memberNameNew: String, memberNameOld: String, handler: Boolean = false){ + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles_rename_member", + { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${datasetName}(${memberNameNew})") ?: false }, + { MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${datasetName}\",\"member\":\"${memberNameOld}\"}}") } + ) + } + fun injectMemberList(testInfo: TestInfo, datasetName: String, listElem: List, listName: String? = null){//= true){ + val requestName = if (listName != null) { + "${testInfo.displayName}_restfiles_listmembers$listName" + } else { + "${testInfo.displayName}_restfiles_listmembers" + } + + val body_headr = "{\"items\":[" + val body_tell = "],\"returnedRows\": ${listElem.size},\"JSONversion\": 1}" + val memebersString = listElem.joinToString(separator = ",") { "{\"member\": \"$it\"}" } + + injectEndpoint( + requestName, + { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${datasetName}/member") ?: false}, + { MockResponse().setBody(body_headr + memebersString + body_tell)} + ) + } + + + override fun dispatch(request: RecordedRequest): MockResponse { -// val fileName = System.getProperty("user.dir") + "/src/uiTest/resources/request_received.txt" -// File(fileName).writeText("${request.requestLine}\n${request.requestUrl}\n${request.path}\n${request.body}\n") - return validationList + val fileName = System.getProperty("user.dir") + "/src/uiTest/resources/request_received.txt" + + + + val x = validationList .firstOrNull { it.validator(request) } + +// if (x?.handler != null && x != null){ +// val kClass = x.handler::class +// val properties = kClass.memberProperties +// +// for (property in properties) { +// +// File(fileName).appendText("=================${property.name}===========") +//// println("${property.name}: ${property.get(x.handler)}") +// } +// } + + File(fileName).appendText("${x?.name}\n${x?.validator}${x?.handler}\n${x?.handler}\n") + return x ?.handler ?.let { it(request) } ?: MockResponse().setBody("Response is not implemented").setResponseCode(404) diff --git a/src/uiTest/kotlin/workingset/AllocateDatasetTest.kt b/src/uiTest/kotlin/workingset/AllocateDatasetTest.kt index 3b968801a..eb79ba8ed 100644 --- a/src/uiTest/kotlin/workingset/AllocateDatasetTest.kt +++ b/src/uiTest/kotlin/workingset/AllocateDatasetTest.kt @@ -17,71 +17,72 @@ import auxiliary.components.actionMenuItem import auxiliary.containers.* import com.intellij.remoterobot.RemoteRobot import com.intellij.remoterobot.fixtures.ComponentFixture -import com.intellij.remoterobot.fixtures.ContainerFixture import com.intellij.remoterobot.fixtures.HeavyWeightWindowFixture import com.intellij.remoterobot.search.locators.Locator -import com.intellij.remoterobot.search.locators.byXpath import io.kotest.matchers.string.shouldContain -import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.Assertions.assertFalse import org.junit.jupiter.api.extension.ExtendWith +import org.junit.jupiter.params.ParameterizedTest +import org.junit.jupiter.params.provider.Arguments +import org.junit.jupiter.params.provider.MethodSource +import org.junit.jupiter.params.provider.ValueSource +import workingset.testutils.injectListAllAllocatedDatasets import java.time.Duration +import java.util.stream.Stream /** * Tests allocating datasets with valid and invalid inputs. */ @TestInstance(TestInstance.Lifecycle.PER_CLASS) @ExtendWith(RemoteRobotExtension::class) -class AllocateDatasetTest { +class AllocateDatasetTest : WorkingSetBase() { private var closableFixtureCollector = ClosableFixtureCollector() private var fixtureStack = mutableListOf() private var wantToClose = mutableListOf( "Allocate Dataset Dialog" ) - private val projectName = "untitled" - private val connectionName = "valid connection" - private val wsName = "WS1" - private val datasetName = "$ZOS_USERID.ALLOC." + override val datasetName = "$ZOS_USERID.ALLOC." + override val wsName = "WS1" + private val datasetMask = "$ZOS_USERID.ALLOC." private val recordFormats = mutableListOf("F", "FB", "V", "VA", "VB") private var datasetsToBeDeleted = mutableListOf() - //TODO change message when ijmp-907 is fixed - private val numberGreaterThanOneMsg = "Enter a number greater than or equal to 1" - private val enterPositiveNumberMsg = "Enter a positive number" + companion object { + @JvmStatic + fun valuesProvider(): Stream { + return invalidAllocateScenarios.entries.stream().map { entry -> + Arguments.of(entry.key, entry.value) + } + } + } + private var mapListDatasets = mutableMapOf() - /** - * Opens the project and Explorer, clears test environment, creates working set and mask. - */ @BeforeAll - fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { + fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) createValidConnectionWithMock( testInfo, connectionName, - projectName, fixtureStack, closableFixtureCollector, remoteRobot ) - createWsAndMask(remoteRobot) - + return createWsAndMask(remoteRobot) } /** * Closes the project and clears test environment, deletes created datasets. */ @AfterAll - fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { - deleteDatasets(remoteRobot) + fun tearDownAll(remoteRobot: RemoteRobot) { +// deleteDatasets(remoteRobot) mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - close() - } + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + closeIntelligentProject(fixtureStack, remoteRobot) } /** @@ -89,6 +90,7 @@ class AllocateDatasetTest { */ @AfterEach fun tearDown(remoteRobot: RemoteRobot) { + deleteDatasets(remoteRobot) responseDispatcher.removeAllEndpoints() closableFixtureCollector.closeWantedClosables(wantToClose, remoteRobot) } @@ -96,158 +98,66 @@ class AllocateDatasetTest { //todo add tests with advanced parameters when switched to mock /** - * Tests to allocate PO datasets with valid parameters. - */ - @Test - fun testAllocateValidPODatasets(remoteRobot: RemoteRobot) { - doValidTest("PO", remoteRobot) - } - - /** - * Tests to allocate PS datasets with valid parameters. - */ - @Test - fun testAllocateValidPSDatasets(remoteRobot: RemoteRobot) { - doValidTest("PS", remoteRobot) - } - - /** - * Tests to allocate POE datasets with valid parameters. - */ - @Test - fun testAllocateValidPOEDatasets(remoteRobot: RemoteRobot) { - doValidTest("PO-E", remoteRobot) - } - - /** - * Tests to allocate dataset with invalid dataset name, checks that correct message is returned. - */ - @Test - fun testInvalidDatasetName(remoteRobot: RemoteRobot) = with(remoteRobot) { - val message = - "Each name segment (qualifier) is 1 to 8 characters, the first of which must be alphabetic (A to Z) or " + - "national (# @ $). The remaining seven characters are either alphabetic, numeric (0 - 9), national, " + - "a hyphen (-). Name segments are separated by a period (.)" - allocateDataSet( - wsName, "A23456789.A", "PO", "TRK", 10, 1, 1, - "FB", 80, 3200, 0, remoteRobot, false, message - ) - } - - /** - * Tests to allocate dataset with invalid primary allocation, checks that correct message is returned. - */ - @Test - fun testInvalidPrimaryAllocation(remoteRobot: RemoteRobot) { - allocateDataSet( - wsName, "A23.A23", "PO", "TRK", -2, 0, 1, - "FB", 80, 3200, 0, remoteRobot, false, numberGreaterThanOneMsg - ) - } - - /** - * Tests to allocate dataset with invalid directory, checks that correct message is returned. - */ - @Test - fun testInvalidDirectory(remoteRobot: RemoteRobot) { - allocateDataSet( - wsName, "A23.A23", "PO", "TRK", 10, 0, 0, - "FB", 80, 3200, 0, remoteRobot, false, numberGreaterThanOneMsg - ) - } - - /** - * Tests to allocate dataset with invalid record length, checks that correct message is returned. + * Tests to allocate PO, PS, POE datasets with valid parameters. */ - @Test - fun testInvalidRecordLength(remoteRobot: RemoteRobot) { - allocateDataSet( - wsName, "A23.A23", "PO", "TRK", 10, 0, 1, - "FB", 0, 3200, 0, remoteRobot, false, numberGreaterThanOneMsg - ) + @ParameterizedTest + @ValueSource(strings = [SEQUENTIAL_ORG_FULL, PO_ORG_FULL, POE_ORG_FULL]) + fun testAllocateValidPODatasets(input: String, remoteRobot: RemoteRobot) { + doValidTest(input, remoteRobot) } /** - * Tests to allocate dataset with invalid secondary allocation, checks that correct message is returned. + * Tests to allocate dataset with invalid dataset params, checks that correct message is returned. */ - @Test - fun testInvalidSecondaryAllocation(remoteRobot: RemoteRobot) { - allocateDataSet( - wsName, "A23.A23", "PO", "TRK", 10, -10, 1, - "FB", 80, 3200, 0, remoteRobot, false, enterPositiveNumberMsg + @ParameterizedTest + @MethodSource("valuesProvider") + fun testInvalidDatasetName(scenarioName: String, value: InvalidAllocate, remoteRobot: RemoteRobot) = + invalidAllocateDataSet( + wsName, + value.datasetName, + value.datasetOrganization, + value.allocationUnit, + value.primaryAllocation, + value.secondaryAllocation, + value.directory, + value.recordFormat, + value.recordLength, + value.blockSize, + value.averageBlockLength, + remoteRobot, + value.message ) - } - - /** - * Tests to allocate dataset with invalid block size, checks that correct message is returned. - */ - @Test - fun testInvalidBlockSize(remoteRobot: RemoteRobot) { - allocateDataSet( - wsName, "A23.A23", "PO", "TRK", 10, 0, 1, - "FB", 80, -1, 0, remoteRobot, false, enterPositiveNumberMsg - ) - } - - /** - * Tests to allocate dataset with invalid average block length, checks that correct message is returned. - */ - @Test - fun testInvalidAverageBlockLength(remoteRobot: RemoteRobot) { - allocateDataSet( - wsName, "A23.A23", "PO", "TRK", 10, 0, 1, - "FB", 80, 3200, -1, remoteRobot, false, enterPositiveNumberMsg - ) - } /** * Allocates dataset with different record formats. */ private fun doValidTest(datasetOrganization: String, remoteRobot: RemoteRobot) { recordFormats.forEach { s -> - val recordLength = if (s == "F") { + val recordLength = if (s == F_RECORD_FORMAT_SHORT) { 3200 } else { 80 } - val dsName = "${datasetName}${datasetOrganization.replace("-", "")}.${s}".uppercase() - responseDispatcher.injectEndpoint( - "testAllocateValid${datasetOrganization}Datasets_${s}_restfiles", - { it?.requestLine?.contains("POST /zosmf/restfiles/ds/${dsName}") ?: false }, - { MockResponse().setBody("{\"dsorg\":\"${datasetOrganization}\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":1,\"recfm\":\"${s}\",\"blksize\":3200,\"lrecl\":${recordLength}}") } - ) + val dsOrganisationShort = "\\((.*?)\\)".toRegex().find(datasetOrganization)?.groupValues?.get(1) + val dsName = "${datasetMask}${dsOrganisationShort}.${s}".uppercase().replace("-", "") + + if (dsOrganisationShort != null) { + responseDispatcher.injectAllocationResultPo( + datasetOrganization, s, dsName, dsOrganisationShort, recordLength + ) + } allocateDataSet( - wsName, dsName, datasetOrganization, "TRK", 10, 1, 1, - s, recordLength, 3200, 0, remoteRobot, true, "" + wsName, dsName, datasetOrganization, TRACKS_ALLOCATION_UNIT_SHORT, 10, 1, 1, + s, recordLength, 3200, 0, remoteRobot ) - val dsntp = if (datasetOrganization == "PS") { + + val dsntp = if (dsOrganisationShort == SEQUENTIAL_ORG_SHORT) { "" } else { - "PDS" + PDS_TYPE } - val listDs = "{\n" + - " \"dsname\": \"${dsName}\",\n" + - " \"blksz\": \"3200\",\n" + - " \"catnm\": \"TEST.CATALOG.MASTER\",\n" + - " \"cdate\": \"2021/11/15\",\n" + - " \"dev\": \"3390\",\n" + - " \"dsntp\": \"${dsntp}\",\n" + - " \"dsorg\": \"${datasetOrganization}\",\n" + - " \"edate\": \"***None***\",\n" + - " \"extx\": \"1\",\n" + - " \"lrecl\": \"${recordLength}\",\n" + - " \"migr\": \"NO\",\n" + - " \"mvol\": \"N\",\n" + - " \"ovf\": \"NO\",\n" + - " \"rdate\": \"2021/11/17\",\n" + - " \"recfm\": \"${s}\",\n" + - " \"sizex\": \"10\",\n" + - " \"spacu\": \"TRACKS\",\n" + - " \"used\": \"1\",\n" + - " \"vol\": \"TESTVOL\",\n" + - " \"vols\": \"TESTVOL\"\n" + - " }," + val listDs = buildDatasetConfigString(dsName, dsntp, datasetOrganization, recordLength, s) mapListDatasets[dsName] = listDs datasetsToBeDeleted.add(dsName) } @@ -256,7 +166,7 @@ class AllocateDatasetTest { /** * Allocates dataset. */ - private fun allocateDataSet( + private fun invalidAllocateDataSet( wsName: String, datasetName: String, datasetOrganization: String, @@ -269,16 +179,15 @@ class AllocateDatasetTest { blockSize: Int, averageBlockLength: Int, remoteRobot: RemoteRobot, - isValid: Boolean, - message: String + message: String, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(wsName).rightClick() } - actionMenu(remoteRobot, "New").click() - actionMenuItem(remoteRobot, "Dataset").click() + actionMenu(remoteRobot, NEW_POINT_TEXT).click() + actionMenuItem(remoteRobot, DATASET_POINT_TEXT).click() allocateDatasetDialog(fixtureStack) { allocateDataset( datasetName, @@ -292,95 +201,41 @@ class AllocateDatasetTest { blockSize, averageBlockLength ) - clickButton("OK") - Thread.sleep(5000) - } - if (isValid) { - closableFixtureCollector.closeOnceIfExists(AllocateDatasetDialog.name) - find(byXpath("//div[@class='MyDialog']")).findText("Dataset $datasetName Has Been Created") - clickButton("No") - } else { - val msgAll = find( - byXpath("//div[@class='HeavyWeightWindow']"), - Duration.ofSeconds(30) - ).findAllText() - var msg = "" - msgAll.forEach { msg += it.text } - msg.shouldContain(message) - assertFalse(button("OK").isEnabled()) - clickButton("Cancel") - closableFixtureCollector.closeOnceIfExists(AllocateDatasetDialog.name) + clickButton(OK_TEXT) } - } - } - /** - * Deletes created datasets. - */ - private fun deleteDatasets(remoteRobot: RemoteRobot) = with(remoteRobot) { - responseDispatcher.injectEndpoint( - "listAllAllocatedDatasets_restfiles", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${datasetName.uppercase()}*") ?: false }, - { MockResponse().setBody(buildFinalListDatasetJson()) } - ) - responseDispatcher.injectEndpoint( - "listMembers_restfiles", - { it?.requestLine?.contains("/member") ?: false }, - { MockResponse().setBody("{\"items\":[],\"returnedRows\":0,\"totalRows\":0,\"moreRows\":null,\"JSONversion\":1}") } - ) - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findAllText(wsName).last().doubleClick() - Thread.sleep(10000) - } - } - datasetsToBeDeleted.forEach { s -> - mapListDatasets.remove(s) - responseDispatcher.injectEndpoint( - "listAllocatedDatasets_restfiles_${s}", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${datasetName.uppercase()}*") ?: false }, - { MockResponse().setBody(buildFinalListDatasetJson()) } - ) - responseDispatcher.injectEndpoint( - "deleteDataset_restfiles_${s}", - { it?.requestLine?.contains("DELETE /zosmf/restfiles/ds/${s}") ?: false }, - { MockResponse().setBody("{}") } - ) - deleteDataset(s, projectName, fixtureStack, remoteRobot) - } - } + val msgAll = find( + messageLoc, + Duration.ofSeconds(30) + ).findAllText() + var msg = "" + msgAll.forEach { msg += it.text } + msg.shouldContain(message) + assertFalse(button(OK_TEXT).isEnabled()) + clickButton(CANCEL_TEXT) + closableFixtureCollector.closeOnceIfExists(AllocateDatasetDialog.name) - private fun buildFinalListDatasetJson(): String { - var result = "{}" - if (mapListDatasets.isNotEmpty()) { - var listDatasetsJson = "{\"items\":[" - mapListDatasets.forEach { - listDatasetsJson += it.value - } - result = listDatasetsJson.dropLast(1) + "],\n" + - " \"returnedRows\": ${mapListDatasets.size},\n" + - " \"totalRows\": ${mapListDatasets.size},\n" + - " \"JSONversion\": 1\n" + - "}" } - return result } +// private val datasetMask = "$ZOS_USERID.ALLOC." /** - * Creates working set and z/OS mask. + * Deletes created datasets. */ - private fun createWsAndMask(remoteRobot: RemoteRobot) = with(remoteRobot) { - createWsWithoutMask(projectName, wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - createMask(wsName, fixtureStack, closableFixtureCollector) - createMaskDialog(fixtureStack) { - createMask(Pair("$datasetName*", "z/OS")) - Thread.sleep(3000) - clickButton("OK") - } - closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + private fun deleteDatasets(remoteRobot: RemoteRobot) { + injectListAllAllocatedDatasets(datasetName.uppercase(), mapListDatasets) + responseDispatcher.injectListMembers(NO_MEMBERS) + refreshWorkSpace(wsName, fixtureStack, remoteRobot) + compressAndDecompressTree(wsName, fixtureStack, remoteRobot) + + datasetsToBeDeleted.forEach { s -> + mapListDatasets.remove(s) + responseDispatcher.injectAllocatedDatasets(s, buildFinalListDatasetJson(mapListDatasets), s) + responseDispatcher.injectDeleteDataset(s) + deleteDataset(s, fixtureStack, remoteRobot) } + compressAndDecompressTree(wsName, fixtureStack, remoteRobot) + return datasetsToBeDeleted.clear() } } \ No newline at end of file diff --git a/src/uiTest/kotlin/workingset/CreateUssFileAndDirTest.kt b/src/uiTest/kotlin/workingset/CreateUssFileAndDirTest.kt index 1b435782c..d7d0a153f 100644 --- a/src/uiTest/kotlin/workingset/CreateUssFileAndDirTest.kt +++ b/src/uiTest/kotlin/workingset/CreateUssFileAndDirTest.kt @@ -73,9 +73,8 @@ class CreateUssFileAndDirTest { { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) createConnection( - projectName, fixtureStack, closableFixtureCollector, connectionName, @@ -83,7 +82,7 @@ class CreateUssFileAndDirTest { remoteRobot, "https://${mockServer.hostName}:${mockServer.port}" ) - createWsWithoutMask(projectName, wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + createWsWithoutMask(wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) mapListUssFiles["."] = dirHereList mapListUssFiles[".."] = dirParentList responseDispatcher.injectEndpoint( @@ -91,7 +90,7 @@ class CreateUssFileAndDirTest { { it?.requestLine?.contains("GET /zosmf/restfiles/fs?path") ?: false }, { MockResponse().setBody(buildResponseListJson(mapListUssFiles, true)) } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { createMask(wsName, fixtureStack, closableFixtureCollector) createMaskDialog(fixtureStack) { createMask(Pair(ussMaskName, "USS")) @@ -113,8 +112,8 @@ class CreateUssFileAndDirTest { fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @@ -140,8 +139,8 @@ class CreateUssFileAndDirTest { { it?.requestLine?.contains("GET /zosmf/restfiles/fs?path") ?: false }, { MockResponse().setBody(buildResponseListJson(mapListUssFiles, true)) } ) - ideFrameImpl(projectName, fixtureStack) { - createUssFile(ussMaskName, ussFileName, UssFileType.File, projectName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { + createUssFile(ussMaskName, ussFileName, UssFileType.File, fixtureStack, remoteRobot) mapListUssFiles[ussFileName] = fileList explorer { fileExplorer.click() @@ -162,7 +161,7 @@ class CreateUssFileAndDirTest { { it?.requestLine?.contains("GET /zosmf/restfiles/fs?path") ?: false }, { MockResponse().setBody(buildResponseListJson(mapListUssFiles, true)) } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(ussMaskName).rightClick() @@ -192,7 +191,7 @@ class CreateUssFileAndDirTest { { it?.requestLine?.contains("GET /zosmf/restfiles/fs?path") ?: false }, { MockResponse().setBody(buildResponseListJson(mapListUssFiles, true)) } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(ussMaskName).rightClick() @@ -230,7 +229,7 @@ class CreateUssFileAndDirTest { .setBody("{\"category\":\"1.0\",\"message\":\"The specified file already exists\",\"rc\":\"4.0\",\"reason\":\"19.0\"}") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(ussMaskName).rightClick() @@ -242,7 +241,8 @@ class CreateUssFileAndDirTest { clickButton("OK") } clickButton("Cancel") - checkErrorNotification(errorHeader, errorType, fileErrorDetail, projectName, fixtureStack, remoteRobot) + checkErrorNotification(fileErrorDetail, errorType, fileErrorDetail, fixtureStack, remoteRobot) + closeNotificztion(fixtureStack, remoteRobot) } } @@ -262,8 +262,8 @@ class CreateUssFileAndDirTest { { it?.requestLine?.contains("POST /zosmf/restfiles/fs$ussMaskName/$ussDirName") ?: false }, { MockResponse().setResponseCode(201) } ) - ideFrameImpl(projectName, fixtureStack) { - createUssFile(ussMaskName, ussDirName, UssFileType.Directory, projectName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { + createUssFile(ussMaskName, ussDirName, UssFileType.Directory, fixtureStack, remoteRobot) mapListUssFiles[ussDirName] = dirList explorer { fileExplorer.click() @@ -292,7 +292,7 @@ class CreateUssFileAndDirTest { .setBody("{\"category\":\"1.0\",\"message\":\"The specified file already exists\",\"rc\":\"4.0\",\"reason\":\"19.0\"}") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(ussMaskName).rightClick() @@ -304,7 +304,8 @@ class CreateUssFileAndDirTest { clickButton("OK") } clickButton("Cancel") - checkErrorNotification(errorHeader, errorType, fileErrorDetail, projectName, fixtureStack, remoteRobot) + checkErrorNotification(fileErrorDetail, errorType, fileErrorDetail, fixtureStack, remoteRobot) + closeNotificztion(fixtureStack, remoteRobot) } } @@ -319,7 +320,7 @@ class CreateUssFileAndDirTest { { it?.requestLine?.contains("GET /zosmf/restfiles/fs?path") ?: false }, { MockResponse().setBody(buildResponseListJson(mapListUssFiles, true)) } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(ussMaskName).rightClick() @@ -349,7 +350,7 @@ class CreateUssFileAndDirTest { { it?.requestLine?.contains("GET /zosmf/restfiles/fs?path") ?: false }, { MockResponse().setBody(buildResponseListJson(mapListUssFiles, true)) } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(ussMaskName).rightClick() @@ -387,7 +388,7 @@ class CreateUssFileAndDirTest { .setBody("{\"category\":\"1.0\",\"message\":\"The specified directory already exists\",\"rc\":\"4.0\",\"reason\":\"19.0\"}") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(ussMaskName).rightClick() @@ -399,7 +400,8 @@ class CreateUssFileAndDirTest { clickButton("OK") } clickButton("Cancel") - checkErrorNotification(errorHeader, errorType, dirErrorDetail, projectName, fixtureStack, remoteRobot) + checkErrorNotification(dirErrorDetail, errorType, dirErrorDetail, fixtureStack, remoteRobot) + closeNotificztion(fixtureStack, remoteRobot) } } @@ -422,7 +424,7 @@ class CreateUssFileAndDirTest { .setBody("{\"category\":\"1.0\",\"message\":\"The specified directory already exists\",\"rc\":\"4.0\",\"reason\":\"19.0\"}") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(ussMaskName).rightClick() @@ -434,7 +436,8 @@ class CreateUssFileAndDirTest { clickButton("OK") } clickButton("Cancel") - checkErrorNotification(errorHeader, errorType, dirErrorDetail, projectName, fixtureStack, remoteRobot) + checkErrorNotification(dirErrorDetail, errorType, dirErrorDetail, fixtureStack, remoteRobot) + closeNotificztion(fixtureStack, remoteRobot) } } } \ No newline at end of file diff --git a/src/uiTest/kotlin/workingset/Data.kt b/src/uiTest/kotlin/workingset/Data.kt deleted file mode 100644 index ef6a17441..000000000 --- a/src/uiTest/kotlin/workingset/Data.kt +++ /dev/null @@ -1,102 +0,0 @@ -package workingset - -//BUTTON TEXT -const val NO_TEXT = "No" -const val OK_TEXT = "OK" -const val CANCEL_TEXT = "Cancel" - -// Datasets types -const val PARTITION_NAME_FULL = "Partitioned (PO)" -const val PARTITION_NAME_SHORT = "PO" -const val SEQUENTIAL_NAME_FULL = "Sequential (PS)" -const val SEQUENTIAL_NAME_SHORT = "PS" -const val PARTITIONED_EXTENDED_NAME_FULL = "Partitioned Extended (PO-E)" -const val PARTITIONED_EXTENDED_NAME_SHORT = "POE" - -//rename dataset -const val DATASET_FOR_RENAME_PROPERTY = "{\"dsorg\":\"PO\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":2,\"recfm\":\"VB\",\"blksize\":6120,\"lrecl\":255}" - -//record formats -const val F_RECORD_FORMAT_SHORT = "F" -const val FB_RECORD_FORMAT_SHORT = "FB" -const val V_RECORD_FORMAT_SHORT = "V" -const val VA_RECORD_FORMAT_SHORT = "VA" -const val VB_RECORD_FORMAT_SHORT = "VB" - -//action menu points -const val REFRESH_POINT_TEXT = "Refresh" -const val NEW_POINT_TEXT = "New" -const val DATASET_POINT_TEXT = "Dataset" -const val MIGRATE_POINT_TEXT="Migrate" - -//Errors messages -val invalidDatasetNameConstant = "Each name segment (qualifier) is 1 to 8 characters, the first of which must be alphabetic (A to Z) or " + - "national (# @ $). The remaining seven characters are either alphabetic, numeric (0 - 9), national, " + - "a hyphen (-). Name segments are separated by a period (.)" -val numberGreaterThanOneMsg = "Enter a number greater than or equal to 1" -val enterPositiveNumberMsg = "Enter a positive number" - -//Migrate options: -const val HMIGRATE_MIGRATE_OPTIONS = "hmigrate" -const val HRECALL_MIGRATE_OPTIONS = "hrecall" - - -//dialog text\patterns - -val datasetHasBeenCreated = "Dataset %s Has Been Created " - - -//bad alloc params cases -data class InvalidAllocate( - val wsName: String, - val datasetName: String, - val datasetOrganization: String, - val allocationUnit: String, - val primaryAllocation: Int, - val secondaryAllocation: Int, - val directory: Int, - val recordFormat: String, - val recordLength: Int, - val blockSize: Int, - val averageBlockLength: Int, - val message: String -) - -val invalidDatasetNameParams = InvalidAllocate( - "", "A23456789.A", PARTITION_NAME_FULL, "TRK", 10, 1, - 1, "FB", 80, 3200, 0, invalidDatasetNameConstant -) -val invalidPrimaryAllocationParams = InvalidAllocate( - "", "A23.A23", PARTITION_NAME_FULL, "TRK", -2, 0, 1, - "FB", 80, 3200, 0, numberGreaterThanOneMsg) - -val invalidDirectoryParams = InvalidAllocate( - "", "A23.A23", PARTITION_NAME_FULL, "TRK", 10, 0, 0, - "FB", 80, 3200, 0,numberGreaterThanOneMsg -) -val invalidRecordLengthParams = InvalidAllocate( - "", "A23.A23", PARTITION_NAME_FULL, "TRK", 10, 0, 1, - "FB", 0, 3200, 0,numberGreaterThanOneMsg -) -val invalidSecondaryAllocationParams = InvalidAllocate( - "", "A23.A23", PARTITION_NAME_FULL, "TRK", 10, -10, 1, - "FB", 80, 3200, 0, enterPositiveNumberMsg -) -val invalidBlockSizeParams = InvalidAllocate( - "", "A23.A23", PARTITION_NAME_FULL, "TRK", 10, 0, 1, - "FB", 80, -1, 0, enterPositiveNumberMsg -) -val invalidAverageBlockLengthParams = InvalidAllocate( - "", "A23.A23", PARTITION_NAME_FULL, "TRK", 10, 0, 1, - "FB", 80, 3200, -1, enterPositiveNumberMsg -) - -val invalidAllocateScenarios = mapOf( - Pair("invalidDatasetNameParams", invalidDatasetNameParams), - Pair("invalidPrimaryAllocationParams", invalidPrimaryAllocationParams), - Pair("invalidDirectoryParams", invalidDirectoryParams), - Pair("invalidRecordLengthParams", invalidRecordLengthParams), - Pair("invalidSecondaryAllocationParams", invalidSecondaryAllocationParams), - Pair("invalidBlockSizeParams", invalidBlockSizeParams), - Pair("invalidAverageBlockLengthParams", invalidAverageBlockLengthParams), -) diff --git a/src/uiTest/kotlin/workingset/DeleteDatasetTest.kt b/src/uiTest/kotlin/workingset/DeleteDatasetTest.kt index e3a7efa72..c9af91075 100644 --- a/src/uiTest/kotlin/workingset/DeleteDatasetTest.kt +++ b/src/uiTest/kotlin/workingset/DeleteDatasetTest.kt @@ -10,73 +10,58 @@ package workingset + import auxiliary.* import auxiliary.closable.ClosableFixtureCollector -import auxiliary.components.actionMenu -import auxiliary.components.actionMenuItem -import auxiliary.containers.* import com.intellij.remoterobot.RemoteRobot -import com.intellij.remoterobot.fixtures.ComponentFixture -import com.intellij.remoterobot.fixtures.ContainerFixture import com.intellij.remoterobot.search.locators.Locator -import com.intellij.remoterobot.search.locators.byXpath -import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.extension.ExtendWith import org.junit.jupiter.params.ParameterizedTest import org.junit.jupiter.params.provider.ValueSource +import workingset.testutils.injectListAllAllocatedDatasets + /** * Tests allocating datasets with valid and invalid inputs. */ @TestInstance(TestInstance.Lifecycle.PER_CLASS) @ExtendWith(RemoteRobotExtension::class) -class DeleteDatasetTest:WorkingSetBase() { +class DeleteDatasetTest : WorkingSetBase() { private var closableFixtureCollector = ClosableFixtureCollector() private var fixtureStack = mutableListOf() - private var wantToClose = mutableListOf( - "Allocate Dataset Dialog" - ) - private val projectName = "untitled" - override val connectionName = "valid connection" - - val wsName = "WS1" + private var wantToClose = mutableListOf(ALLOCATE_DATASET_DIALOG) override val datasetName = "$ZOS_USERID.ALLOC." - private val recordFormats = mutableListOf("F", "FB", "V", "VA", "VB") + private val recordFormats = mutableListOf(F_RECORD_FORMAT_SHORT, FB_RECORD_FORMAT_SHORT, V_RECORD_FORMAT_SHORT, VA_RECORD_FORMAT_SHORT, VB_RECORD_FORMAT_SHORT) private var datasetsToBeDeleted = mutableListOf() - - private var mapListDatasets = mutableMapOf() + private var mapListDatasets = mutableMapOf() /** * Opens the project and Explorer, clears test environment, creates working set and mask. */ @BeforeAll - fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { + fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) createValidConnectionWithMock( testInfo, connectionName, - projectName, fixtureStack, closableFixtureCollector, remoteRobot ) createWsAndMask(remoteRobot) - } /** * Closes the project and clears test environment, deletes created datasets. */ @AfterAll - fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { + fun tearDownAll(remoteRobot: RemoteRobot) { deleteDatasets(remoteRobot) mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - close() - } + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + closeIntelligentProject(fixtureStack, remoteRobot) } /** @@ -94,7 +79,7 @@ class DeleteDatasetTest:WorkingSetBase() { * Tests to allocate PO datasets with valid parameters. */ @ParameterizedTest - @ValueSource(strings = ["Sequential (PS)", "Partitioned (PO)", "Partitioned Extended (PO-E)"]) + @ValueSource(strings = [SEQUENTIAL_ORG_FULL, PO_ORG_FULL, POE_ORG_FULL]) fun testDeleteDatasets(input: String, remoteRobot: RemoteRobot) { doValidTest(input, remoteRobot) deleteDatasets(remoteRobot) @@ -105,53 +90,35 @@ class DeleteDatasetTest:WorkingSetBase() { */ private fun doValidTest(datasetOrganization: String, remoteRobot: RemoteRobot) { recordFormats.forEach { s -> - val recordLength = if (s == "F") { + val recordLength = if (s == F_RECORD_FORMAT_SHORT) { 3200 } else { 80 } -// val dsName = "${datasetName}${datasetOrganization.replace("-", "")}.${s}".uppercase() val dsOrganisationShort = "\\((.*?)\\)".toRegex().find(datasetOrganization)?.groupValues?.get(1) val dsName = "${datasetName}${dsOrganisationShort}.${s}".uppercase().replace("-", "") - responseDispatcher.injectEndpoint( - "testAllocateValid${datasetOrganization}Datasets_${s}_restfiles", - { it?.requestLine?.contains("POST /zosmf/restfiles/ds/${dsName}") ?: false }, - { MockResponse().setBody("{\"dsorg\":\"${dsOrganisationShort}\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":1,\"recfm\":\"${s}\",\"blksize\":3200,\"lrecl\":${recordLength}}") } - ) -// Thread.sleep(3000) + + if (dsOrganisationShort != null) { + responseDispatcher.injectAllocationResultPo( + datasetOrganization, + s, + dsName, + dsOrganisationShort, + recordLength + ) + } allocateDataSet( - wsName, dsName, datasetOrganization, "TRK", 10, 1, 1, + wsName, dsName, datasetOrganization, TRACKS_ALLOCATION_UNIT_SHORT, 10, 1, 1, s, recordLength, 3200, 0, remoteRobot ) - val dsntp = if (dsOrganisationShort == "PS") { + val dsntp = if (dsOrganisationShort == SEQUENTIAL_ORG_SHORT) { "" } else { - "PDS" + PDS_TYPE } - val listDs = "{\n" + - " \"dsname\": \"${dsName}\",\n" + - " \"blksz\": \"3200\",\n" + - " \"catnm\": \"TEST.CATALOG.MASTER\",\n" + - " \"cdate\": \"2021/11/15\",\n" + - " \"dev\": \"3390\",\n" + - " \"dsntp\": \"${dsntp}\",\n" + - " \"dsorg\": \"${datasetOrganization}\",\n" + - " \"edate\": \"***None***\",\n" + - " \"extx\": \"1\",\n" + - " \"lrecl\": \"${recordLength}\",\n" + - " \"migr\": \"NO\",\n" + - " \"mvol\": \"N\",\n" + - " \"ovf\": \"NO\",\n" + - " \"rdate\": \"2021/11/17\",\n" + - " \"recfm\": \"${s}\",\n" + - " \"sizex\": \"10\",\n" + - " \"spacu\": \"TRACKS\",\n" + - " \"used\": \"1\",\n" + - " \"vol\": \"TESTVOL\",\n" + - " \"vols\": \"TESTVOL\"\n" + - " }," + val listDs = buildDatasetConfigString(dsName, dsntp, datasetOrganization, recordLength, s) mapListDatasets[dsName] = listDs datasetsToBeDeleted.add(dsName) } @@ -160,68 +127,21 @@ class DeleteDatasetTest:WorkingSetBase() { /** * Deletes created datasets. */ - private fun deleteDatasets(remoteRobot: RemoteRobot) = with(remoteRobot) { - responseDispatcher.injectEndpoint( - "listAllAllocatedDatasets_restfiles", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${datasetName.uppercase()}*") ?: false }, - { MockResponse().setBody(buildFinalListDatasetJson()) } - ) - responseDispatcher.injectEndpoint( - "listMembers_restfiles", - { it?.requestLine?.contains("/member") ?: false }, - { MockResponse().setBody("{\"items\":[],\"returnedRows\":0,\"totalRows\":0,\"moreRows\":null,\"JSONversion\":1}") } - ) - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(wsName).rightClick() - } - actionMenuItem(remoteRobot, "Refresh").click() - explorer { - find(viewTree).findAllText(wsName).last().doubleClick() - } - } -// Thread.sleep(1000) -// Thread.sleep(3000) + private fun deleteDatasets(remoteRobot: RemoteRobot) { + injectListAllAllocatedDatasets(datasetName.uppercase(), mapListDatasets) + responseDispatcher.injectListMembers(NO_MEMBERS) + + refreshWorkSpace(wsName, fixtureStack, remoteRobot) + compressAndDecompressTree(wsName, fixtureStack, remoteRobot) + datasetsToBeDeleted.forEach { s -> mapListDatasets.remove(s) - responseDispatcher.injectEndpoint( - "listAllocatedDatasets_restfiles_${s}", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${datasetName.uppercase()}*") ?: false }, - { MockResponse().setBody(buildFinalListDatasetJson()) } - ) - responseDispatcher.injectEndpoint( - "deleteDataset_restfiles_${s}", - { it?.requestLine?.contains("DELETE /zosmf/restfiles/ds/${s}") ?: false }, - { MockResponse().setBody("{}") } - ) - deleteDataset(s, projectName, fixtureStack, remoteRobot) - - } - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findAllText(wsName).last().doubleClick() -// Thread.sleep(1000) - } + responseDispatcher.injectAllocatedDatasets(s, buildFinalListDatasetJson(mapListDatasets), s) + responseDispatcher.injectDeleteDataset(s) + deleteDataset(s, fixtureStack, remoteRobot) } + compressAndDecompressTree(wsName, fixtureStack, remoteRobot) + datasetsToBeDeleted.clear() } - - - -// /** -// * Creates working set and z/OS mask. -// */ -// private fun createWsAndMask(remoteRobot: RemoteRobot) = with(remoteRobot) { -// createWsWithoutMask(projectName, wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) -// ideFrameImpl(projectName, fixtureStack) { -// createMask(wsName, fixtureStack, closableFixtureCollector) -// createMaskDialog(fixtureStack) { -// createMask(Pair("$datasetName*", "z/OS")) -// clickButton("OK") -// } -// closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) -// } -// } } \ No newline at end of file diff --git a/src/uiTest/kotlin/workingset/Locators.kt b/src/uiTest/kotlin/workingset/Locators.kt deleted file mode 100644 index 1b3639ec9..000000000 --- a/src/uiTest/kotlin/workingset/Locators.kt +++ /dev/null @@ -1,13 +0,0 @@ -package workingset - -import com.intellij.remoterobot.search.locators.byXpath - -//common dialogs locators -val myDialogXpathLoc = byXpath("//div[@class='MyDialog']") - - -//allocate dialog locators -val datasetNameInputLoc = byXpath("//div[@class='JBTextField']") -val datasetOrgDropDownLoc = byXpath("//div[@class='ComboBox']") -val inputFieldLoc = byXpath("//div[@class='JBTextField']") -val dropdownsLoc = byXpath("//div[@class='ComboBox']") \ No newline at end of file diff --git a/src/uiTest/kotlin/workingset/MigrateDatasetTest.kt b/src/uiTest/kotlin/workingset/MigrateDatasetTest.kt index 2110af430..a5532fa60 100644 --- a/src/uiTest/kotlin/workingset/MigrateDatasetTest.kt +++ b/src/uiTest/kotlin/workingset/MigrateDatasetTest.kt @@ -12,21 +12,12 @@ package workingset import auxiliary.* import auxiliary.closable.ClosableFixtureCollector -import auxiliary.components.actionMenu -import auxiliary.components.actionMenuItem import auxiliary.containers.* import com.intellij.remoterobot.RemoteRobot -import com.intellij.remoterobot.fixtures.ComponentFixture -import com.intellij.remoterobot.fixtures.ContainerFixture -import com.intellij.remoterobot.fixtures.HeavyWeightWindowFixture import com.intellij.remoterobot.search.locators.Locator -import com.intellij.remoterobot.search.locators.byXpath -import io.kotest.matchers.string.shouldContain -import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* -import org.junit.jupiter.api.Assertions.assertFalse import org.junit.jupiter.api.extension.ExtendWith -import java.time.Duration +import workingset.testutils.injectListAllAllocatedDatasets /** * Tests migrating and recalling dataset. @@ -34,13 +25,13 @@ import java.time.Duration @TestMethodOrder(MethodOrderer.OrderAnnotation::class) @TestInstance(TestInstance.Lifecycle.PER_CLASS) @ExtendWith(RemoteRobotExtension::class) -class MigrateDatasetTest { +class MigrateDatasetTest:WorkingSetBase() { private var closableFixtureCollector = ClosableFixtureCollector() private var fixtureStack = mutableListOf() private val projectName = "untitled" - private val connectionName = "con1" - private val wsName = "WS name" + override val connectionName = "con1" + override val wsName = "WS name" private val maskName = "${ZOS_USERID.uppercase()}.UI.TEST*" private val pdsName = "${ZOS_USERID.uppercase()}.UI.TEST" @@ -55,112 +46,43 @@ class MigrateDatasetTest { @BeforeAll fun setUpAll(testInfo: TestInfo,remoteRobot: RemoteRobot) = with(remoteRobot) { startMockServer() - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_info", - { it?.requestLine?.contains("zosmf/info") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_resttopology", - { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - createConnection( - projectName, - fixtureStack, - closableFixtureCollector, - connectionName, - true, - remoteRobot, - "https://${mockServer.hostName}:${mockServer.port}" - ) - responseDispatcher.injectEndpoint( - "listAllAllocatedDatasets_restfiles", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${maskName}") ?: false }, - { MockResponse().setBody(buildResponseListJson(mapListDatasets, true)) } - ) - responseDispatcher.injectEndpoint( - "listAllDatasetMembers_restfiles", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds/$pdsName/member") ?: false }, - { MockResponse().setBody(buildResponseListJson(mapListDatasetMembers, false)) } - ) - responseDispatcher.injectEndpoint( - "allocatePds_restfiles", - { it?.requestLine?.contains("POST /zosmf/restfiles/ds/$pdsName") ?: false }, - { MockResponse().setBody("{\"dsorg\":\"PDS\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":2,\"recfm\":\"VB\",\"blksize\":6120,\"lrecl\":255}") } - ) - createWsWithoutMask(projectName, wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) - mapListDatasets[pdsName] = listDS(pdsName, "PDS", "PO") - allocatePDSAndCreateMask( - wsName, - pdsName, - projectName, - fixtureStack, - closableFixtureCollector, - remoteRobot, - maskName, - directory = 2 + responseDispatcher.injectTestInfo(testInfo) + responseDispatcher.injectTestInfoRestTopology(testInfo) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + createConnection(fixtureStack, closableFixtureCollector, connectionName, true, remoteRobot, "https://${mockServer.hostName}:${mockServer.port}") + mapListDatasets[pdsName] = auxiliary.listDS(pdsName, PDS_TYPE, PO_ORG_SHORT) + injectListAllAllocatedDatasets(pdsName, mapListDatasets, true) + responseDispatcher.injectListAllDatasetMembersRestfiles(pdsName, buildResponseListJson(mapListDatasetMembers, false)) + responseDispatcher.injectAllocationResultPds(pdsName) + + createWsWithoutMask(wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + + allocatePDSAndCreateMask(wsName, pdsName,fixtureStack, closableFixtureCollector, remoteRobot, maskName, directory = 2 ) } + /** * Closes the project and clears test environment. */ @AfterAll fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } + responseDispatcher.removeAllEndpoints() } /** * Test to migrate and recall dataset */ @Test - @Order(1) fun testViewJobProperties(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - Thread.sleep(1000) - find(viewTree).findText(pdsName).rightClick() - } - responseDispatcher.injectEndpoint( - "migratePds_restfiles", - { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/$pdsName") ?: false && - it?.body?.toString()?.contains("hmigrate") ?: false }, - { MockResponse().setResponseCode(200) } - ) - mapListDatasets[pdsName] = migratedDs - actionMenuItem(remoteRobot, "Migrate").click() - explorer { - fileExplorer.click() - Thread.sleep(1000) - find(viewTree).findText(pdsName).rightClick() - } - actionMenuItem(remoteRobot, "Properties").click() - datasetPropertiesDialog(fixtureStack) { - if (isDatasetMigrated()) { - throw Exception("Dataset is not migrated") - } - clickButton("OK") - } - responseDispatcher.injectEndpoint( - "migratePds_restfiles", - { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/$pdsName") ?: false && - it?.body?.toString()?.contains("hrecall") ?: false }, - { MockResponse().setResponseCode(200) } - ) - explorer { - fileExplorer.click() - Thread.sleep(1000) - find(viewTree).findText(pdsName).rightClick() - } - mapListDatasets[pdsName] = listDS(pdsName, "PDS", "PO") - actionMenuItem(remoteRobot, "Recall").click() + ideFrameImpl(PROJECT_NAME, fixtureStack) { + migrateDataset(pdsName, migratedDs, mapListDatasets, fixtureStack, remoteRobot) + recallDataset(pdsName, fixtureStack, mapListDatasets, remoteRobot) } } } \ No newline at end of file diff --git a/src/uiTest/kotlin/workingset/RenameDatasetTest.kt b/src/uiTest/kotlin/workingset/RenameDatasetTest.kt index aab37ede2..b4fb54d1c 100644 --- a/src/uiTest/kotlin/workingset/RenameDatasetTest.kt +++ b/src/uiTest/kotlin/workingset/RenameDatasetTest.kt @@ -12,15 +12,20 @@ package workingset import auxiliary.* import auxiliary.closable.ClosableFixtureCollector -import auxiliary.components.actionMenuItem -import auxiliary.containers.* import com.intellij.remoterobot.RemoteRobot -import com.intellij.remoterobot.fixtures.* import com.intellij.remoterobot.search.locators.Locator -import com.intellij.remoterobot.search.locators.byXpath +import com.intellij.remoterobot.fixtures.HeavyWeightWindowFixture import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.extension.ExtendWith +import org.junit.jupiter.params.ParameterizedTest +import org.junit.jupiter.params.provider.Arguments +import org.junit.jupiter.params.provider.MethodSource +import workingset.auxiliary.components.elements.ButtonElement +import workingset.testutils.injectMemberList +import workingset.testutils.* +import java.time.Duration +import java.util.stream.Stream /** * Tests creating, editing and deleting working sets and masks from context menu. @@ -28,26 +33,36 @@ import org.junit.jupiter.api.extension.ExtendWith @TestMethodOrder(MethodOrderer.OrderAnnotation::class) @TestInstance(TestInstance.Lifecycle.PER_CLASS) @ExtendWith(RemoteRobotExtension::class) -class RenameDatasetTest { +class RenameDatasetTest : WorkingSetBase() { private var closableFixtureCollector = ClosableFixtureCollector() private var fixtureStack = mutableListOf() private var wantToClose = mutableListOf("Allocate DataSet Dialog", "Allocate Member Dialog") + override val connectionName = "con1" + override val wsName = "WS name" - private val projectName = "untitled" - private val connectionName = "con1" - private val wsName = "WS name" - private val errorHeader = "Error in plugin For Mainframe" private val errorType = "Unable to rename" private val pdsName = "$ZOS_USERID.UI.TEST".uppercase() + private val pdsMaskName = "$ZOS_USERID.UI.TEST*".uppercase() private val memberName = "TESTM" private val memberFinalName = "TESTMF" private val anotherMemberName = "TESTMA" - private val dsName = "$ZOS_USERID.UI.TESTD".uppercase() - private val dsFinalName = "$ZOS_USERID.UI.TESTDF".uppercase() + private val dsName = "$ZOS_USERID.UI.TESTD".uppercase() + private val dsFinalName = "$ZOS_USERID.UI.TESTDF".uppercase() private val anotherDsName = "$ZOS_USERID.UI.TESTA".uppercase() private var mapListDatasets = mutableMapOf() + private val dsFinalNameLong = "$ZOS_USERID.UI.TESTDF.123456789".uppercase() + private val tooLongString45 = "A".repeat(45) + + companion object { + @JvmStatic + fun valuesProviderMembers(): Stream { + return incorrectRenameMember.entries.stream().map { entry -> + Arguments.of(entry.key, entry.value) + } + } + } /** * Opens the project and Explorer, clears test environment. @@ -55,52 +70,36 @@ class RenameDatasetTest { @BeforeAll fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - createValidConnectionWithMock( - testInfo, - connectionName, - projectName, - fixtureStack, - closableFixtureCollector, - remoteRobot - ) - createWsWithoutMask(projectName, wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + createValidConnectionWithMock(testInfo, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + createWsWithoutMask(wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + createMask(wsName, pdsMaskName, fixtureStack,closableFixtureCollector, ZOS_MASK, remoteRobot) + + okButton = ButtonElement(OK_TEXT, fixtureStack, remoteRobot) + } + + @BeforeEach + fun setUp(testInfo: TestInfo, remoteRobot: RemoteRobot){ + responseDispatcher.injectTestInfoForPdsDataset(testInfo.displayName, DATASET_FOR_RENAME_PROPERTY, pdsName) + + responseDispatcher.injectAllocationResultPo(PO_ORG_FULL, VB_RECORD_FORMAT_SHORT, dsName, PO_ORG_SHORT, 255) + responseDispatcher.injectAllocationResultPo(PO_ORG_FULL, VB_RECORD_FORMAT_SHORT, anotherDsName, PO_ORG_SHORT, 255) + + mapListDatasets[dsName] = listDS(dsName, PDS_TYPE, PO_ORG_SHORT) + mapListDatasets[anotherDsName] = listDS(anotherDsName, PDS_TYPE, PO_ORG_SHORT) + mapListDatasets[pdsName] = listDS(pdsName, PDS_TYPE, PO_ORG_SHORT) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_${pdsName}_restfiles", - { it?.requestLine?.contains("POST /zosmf/restfiles/ds/${pdsName}") ?: false }, - { MockResponse().setBody("{\"dsorg\":\"PO\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":2,\"recfm\":\"VB\",\"blksize\":6120,\"lrecl\":255}") } - ) - allocatePDSAndCreateMask( - wsName, - pdsName, - projectName, - fixtureStack, - closableFixtureCollector, - remoteRobot, - "$ZOS_USERID.UI.TEST*", - directory = 2, - false - ) - mapListDatasets[pdsName] = listDS(pdsName, "PDS", "PO") - allocateDSWithMock(testInfo, dsName, remoteRobot) - allocateDSWithMock(testInfo, anotherDsName, remoteRobot) openWSAndListDatasets(testInfo, remoteRobot) - allocateMemberWithMock(testInfo, pdsName, memberName, remoteRobot) - allocateMemberWithMock(testInfo, pdsName, anotherMemberName, remoteRobot) } /** * Closes the project and clears test environment. */ @AfterAll - fun tearDownAll(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { - deleteDatasetsWithMock(testInfo, remoteRobot) + fun tearDownAll(testInfo: TestInfo, remoteRobot: RemoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - close() - } + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + return closeIntelligentProject(fixtureStack, remoteRobot) } /** @@ -108,6 +107,8 @@ class RenameDatasetTest { */ @AfterEach fun tearDown(remoteRobot: RemoteRobot) { + compressAndDecompressTree(wsName, fixtureStack, remoteRobot) + mapListDatasets.clear() responseDispatcher.removeAllEndpoints() closableFixtureCollector.closeWantedClosables(wantToClose, remoteRobot) } @@ -116,402 +117,170 @@ class RenameDatasetTest { * Tests renaming member when valid member name is provided. */ @Test - @Order(1) - fun testRenameMemberWithCorrectNameViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - var isFirstRequest = true - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_listmembers1", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${pdsName}/member") ?: false && isFirstRequest }, - { MockResponse().setBody("{\"items\":[{\"member\": \"${memberName}\"},{\"member\": \"${anotherMemberName}\"}],\"returnedRows\": 2,\"JSONversion\": 1}") } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_listmembers2", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${pdsName}/member") ?: false && !isFirstRequest }, - { MockResponse().setBody("{\"items\":[{\"member\": \"${memberFinalName}\"},{\"member\": \"${anotherMemberName}\"}],\"returnedRows\": 2,\"JSONversion\": 1}") } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_rename_member", - { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${pdsName}(${memberFinalName})") ?: false }, - { MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${pdsName}\",\"member\":\"${memberName}\"}}") } - ) - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - Thread.sleep(3000) - find(viewTree).findText(pdsName).doubleClick() - Thread.sleep(5000) - find(viewTree).findText(memberName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Member") { - find(byXpath("//div[@class='JBTextField']")).text = memberFinalName - isFirstRequest = false - } - clickButton("OK") - } - } + fun testRenameMemberWithCorrectNameViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + var isFirstRequest = true + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles_listmembers1", + { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${pdsName}/member") ?: false && isFirstRequest }, + { MockResponse().setBody("{\"items\":[{\"member\": \"${memberName}\"},{\"member\": \"${anotherMemberName}\"}],\"returnedRows\": 2,\"JSONversion\": 1}") } + ) + responseDispatcher.injectEndpoint( + "${testInfo.displayName}_restfiles_listmembers2", + { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${pdsName}/member") ?: false && !isFirstRequest }, + { MockResponse().setBody("{\"items\":[{\"member\": \"${memberFinalName}\"},{\"member\": \"${anotherMemberName}\"}],\"returnedRows\": 2,\"JSONversion\": 1}") } + ) + injectRenameMember(testInfo,pdsName,memberFinalName,memberName) + callRenameMemberPoint(fixtureStack,pdsName,memberName, remoteRobot) + newMemberNameInput(memberFinalName, fixtureStack, remoteRobot) + isFirstRequest = false + return clickByText(OK_TEXT, fixtureStack, remoteRobot) + } + // todo: 3 tests with the same scenario, create 1 test with parametrisation /** * Tests renaming member to name of another member in the same PDS and validates error pop-up notification. */ @Test - @Order(2) - fun testRenameMemberWithNameOfAnotherMemberViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val errorDetail = "Member already exists" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_rename_member", - { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${pdsName}(${anotherMemberName})") ?: false }, - { - MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${pdsName}\",\"member\":\"${memberFinalName}\"}}") - .setResponseCode(500) - .setBody("{\"category\":\"4.0\",\"message\":\"Rename member failed\",\"rc\":\"4.0\",\"details\":[\"ISRZ002 $errorDetail - Directory already contains the specified member name.\"],\"reason\":\"0.0\"}") - } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_listmembers", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${pdsName}/member") ?: false }, - { MockResponse().setBody("{\"items\":[{\"member\": \"${memberFinalName}\"},{\"member\": \"${anotherMemberName}\"}],\"returnedRows\": 2,\"JSONversion\": 1}") } - ) - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(5000) - find(viewTree).findText(memberFinalName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Member") { - find(byXpath("//div[@class='JBTextField']")).text = anotherMemberName - } - clickButton("OK") - Thread.sleep(3000) - checkErrorNotification(errorHeader, errorType, errorDetail, projectName, fixtureStack, remoteRobot) - } - } - - /** - * Tests renaming member to the same name and validates error pop-up notification. - */ - @Test - @Order(3) - fun testRenameMemberWithTheSameNameViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val errorDetail = "Member in use" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_rename_member", - { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${pdsName}(${memberFinalName})") ?: false }, - { - MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${pdsName}\",\"member\":\"${memberFinalName}\"}}") - .setResponseCode(500) - .setBody("{\"category\":\"4.0\",\"message\":\"Rename member failed\",\"rc\":\"12.0\",\"details\":[\"ISRZ002 $errorDetail - Member is being updated by you or another user. Enter HELP for a list of users using the data set.\"],\"reason\":\"0.0\"}") - } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_listmembers", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${pdsName}/member") ?: false }, - { MockResponse().setBody("{\"items\":[{\"member\": \"${memberFinalName}\"},{\"member\": \"${anotherMemberName}\"}],\"returnedRows\": 2,\"JSONversion\": 1}") } - ) - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(5000) - find(viewTree).findText(memberFinalName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Member") { - find(byXpath("//div[@class='JBTextField']")).text = memberFinalName - } - clickButton("OK") - Thread.sleep(3000) - checkErrorNotification(errorHeader, errorType, errorDetail, projectName, fixtureStack, remoteRobot) - } - } - - /** - * Tests renaming member to very long and validates error notification. - */ - @Test - @Order(4) - fun testRenameMemberWithTooLongNameViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(5000) - find(viewTree).findText(memberFinalName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Member") { - find(byXpath("//div[@class='JBTextField']")).text = "123456789" - } - clickButton("OK") - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - MEMBER_NAME_LENGTH_MESSAGE - ) - Thread.sleep(3000) - clickButton("Cancel") - } + fun testRenameMemberWithNameOfAnotherMemberViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + injectRenameMemberUnsuccessful(testInfo, pdsName, memberFinalName, anotherMemberName, 500, "4.0", MEMBER_ALREADY_EXISTS) + injectMemberList(testInfo, pdsName, listOf(memberFinalName, anotherMemberName)) + callRenameMemberPoint(fixtureStack,pdsName,memberFinalName, remoteRobot) + newMemberNameInput(anotherMemberName, fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + checkErrorNotification(RENAME_MEMBER_FAILED, RENAME_MEMBER_FAILED, MEMBER_ALREADY_EXISTS, fixtureStack, remoteRobot) + closeNotificztion(fixtureStack, remoteRobot) } /** - * Tests renaming member to invalid name and validates error notification. + * Tests renaming DataSet to name of another DataSet and validates error pop-up notification. */ @Test - @Order(5) - fun testRenameMemberWithInvalidNameViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(5000) - find(viewTree).findText(memberFinalName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Member") { - find(byXpath("//div[@class='JBTextField']")).text = "@*" - } - clickButton("OK") - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - INVALID_MEMBER_NAME_MESSAGE - ) - Thread.sleep(3000) - clickButton("Cancel") - } + fun testRenameDatasetWithNameOfAnotherDatasetViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + injectRenameDatasetUnsuccessful(testInfo, dsFinalName, anotherDsName, 500, RC_8, RC_8_TEXT) + injectListAllAllocatedDatasets("$ZOS_USERID.UI.TEST*".uppercase(),mapListDatasets) + callRenameDatasetPoint(fixtureStack, dsName, remoteRobot) + newDatasetNameInput(anotherDsName,fixtureStack,remoteRobot) + clickByText(OK_TEXT,fixtureStack,remoteRobot) + checkErrorNotification(DATA_SET_RENAME_FAILED_MSG, DATA_SET_RENAME_FAILED_MSG, DATA_SET_RENAME_FAILED, fixtureStack, remoteRobot) + closeNotificztion(fixtureStack, remoteRobot) } /** - * Tests renaming member to name with the invalid first symbol and validates error notification. + * Tests renaming member to the same name and validates error pop-up notification. */ @Test - @Order(6) - fun testRenameMemberWithInvalidFirstSymbolViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(5000) - find(viewTree).findText(memberFinalName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Member") { - find(byXpath("//div[@class='JBTextField']")).text = "**" - } - clickButton("OK") - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - INVALID_MEMBER_NAME_BEGINNING_MESSAGE - ) - Thread.sleep(3000) - clickButton("Cancel") - } + fun testRenameMemberWithTheSameNameViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + injectRenameMemberUnsuccessful(testInfo, pdsName, memberFinalName, memberFinalName, 500, "12.0", MEMBER_IN_USE) + injectMemberList(testInfo, pdsName, listOf(memberFinalName, anotherMemberName)) + callRenameMemberPoint(fixtureStack,pdsName,memberFinalName, remoteRobot) + newMemberNameInput(memberFinalName, fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + checkErrorNotification(RENAME_MEMBER_FAILED, RENAME_MEMBER_FAILED, MEMBER_IN_USE, fixtureStack, remoteRobot) + closeNotificztion(fixtureStack, remoteRobot) } /** - * Tests renaming member to empty name and validates error notification. + * Tests renaming member to the incorrect name and validates error pop-up notification. */ - @Test - @Order(7) - fun testRenameMemberWithEmptyNameViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(5000) - find(viewTree).findText(memberFinalName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Member") { - find(byXpath("//div[@class='JBTextField']")).text = "" - } - clickButton("OK") - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - MEMBER_EMPTY_NAME_MESSAGE - ) - Thread.sleep(3000) - clickButton("Cancel") - } + @ParameterizedTest + @MethodSource("valuesProviderMembers") + fun testIncorrectRename(invalidName: String, errorMsg: String, testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { + injectRenameMemberUnsuccessful(testInfo, pdsName, memberFinalName, memberFinalName, 500, "12.0", MEMBER_IN_USE) + injectMemberList(testInfo, pdsName, listOf(memberFinalName, anotherMemberName)) + callRenameMemberPoint(fixtureStack,pdsName,memberFinalName,remoteRobot) + newMemberNameInput(invalidName, fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + + + val msgAll = find(messageLoc, Duration.ofSeconds(30) + ).findAllText() + var msg = "" + msgAll.forEach { msg += it.text } + + clickByText(CANCEL_TEXT, fixtureStack, remoteRobot) + Assertions.assertEquals(errorMsg, msg) } /** * Tests renaming DataSet when valid member name is provided. */ @Test - @Order(8) - fun testRenameDataSetWithCorrectNameViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_listmembers", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${pdsName}/member") ?: false }, - { MockResponse().setBody("{\"items\":[{\"member\": \"${memberFinalName}\"},{\"member\": \"${anotherMemberName}\"}],\"returnedRows\": 2,\"JSONversion\": 1}") } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_rename_ds", - { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${dsFinalName}") ?: false }, - { MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${dsName}\"}}") } - ) - mapListDatasets.remove(dsName) - mapListDatasets[dsFinalName] = listDS(dsFinalName, "PDS", "PO") - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles", - { - it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${"$ZOS_USERID.UI.TEST*".uppercase()}") - ?: false - }, - { MockResponse().setBody(buildFinalListDatasetJson()) } - ) - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(3000) - find(viewTree).findText(dsName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Dataset") { - find(byXpath("//div[@class='JBTextField']")).text = dsFinalName - } - clickButton("OK") - } - } - - /** - * Tests renaming DataSet to name of another DataSet and validates error pop-up notification. - */ - @Test - @Order(9) - fun testRenameDatasetWithNameOfAnotherDatasetViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val errorDetail = "data set rename failed" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_rename_ds", - { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${anotherDsName}") ?: false }, - { - MockResponse().setBody("{\"request\":\"rename\",\"from-dataset\":{\"dsn\":\"${dsFinalName}\"}}") - .setResponseCode(500) - .setBody("{\"category\":\"1.0\",\"message\":\"$errorDetail\",\"rc\":\"8.0\",\"details\":[\"EDC5051I An error occurred when renaming a file.\"],\"reason\":\"6.0\"}") - } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_list_ds", - { - it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${"$ZOS_USERID.UI.TEST*".uppercase()}") - ?: false - }, - { MockResponse().setBody(buildFinalListDatasetJson()) } - ) - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(5000) - find(viewTree).findText(dsFinalName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Dataset") { - find(byXpath("//div[@class='JBTextField']")).text = anotherDsName - } - clickButton("OK") - Thread.sleep(3000) - checkErrorNotification(errorHeader, errorType, errorDetail, projectName, fixtureStack, remoteRobot) - } - } + fun testRenameDataSetWithCorrectNameViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + injectRenameMember(testInfo, pdsName, memberFinalName, anotherMemberName) + injectMemberList(testInfo, pdsName, listOf(memberFinalName, anotherMemberName)) + injectRenameDataset(testInfo, dsFinalName, dsName) + mapListDatasets.remove(dsName) + mapListDatasets[dsFinalName] = listDS(dsFinalName, PDS_TYPE, PO_ORG_SHORT) + injectListAllAllocatedDatasets("$ZOS_USERID.UI.TEST*".uppercase(),mapListDatasets) + callRenameDatasetPoint(fixtureStack, dsName, remoteRobot) + newDatasetNameInput(dsFinalName,fixtureStack,remoteRobot) + okButton.click() + } /** * Tests renaming DataSet to name with invalid section and validates error notification. */ @Test - @Order(10) fun testRenameDatasetWithInvalidSectionViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(5000) - find(viewTree).findText(dsFinalName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Dataset") { - find(byXpath("//div[@class='JBTextField']")).text = "$dsFinalName.123456789" - } - clickButton("OK") - var message = "" - find(byXpath("//div[@class='HeavyWeightWindow']")).findAllText().forEach { - message += it.text - } - Thread.sleep(3000) - clickButton("Cancel") - if (!message.contains(DATASET_INVALID_SECTION_MESSAGE)) { - throw Exception("Error message is different from expected") - } - } + callRenameDatasetPoint(fixtureStack, anotherDsName, remoteRobot) + newDatasetNameInput(dsFinalNameLong,fixtureStack,remoteRobot) + clickByText(OK_TEXT,fixtureStack,remoteRobot) + + val msgAll = find(messageLoc, Duration.ofSeconds(30) + ).findAllText() + var msg = "" + msgAll.forEach { msg += it.text } + + clickByText(CANCEL_TEXT, fixtureStack, remoteRobot) + Assertions.assertEquals(DATASET_INVALID_SECTION_MESSAGE, msg) } /** * Tests renaming DataSet to very long name and validates error notification. */ @Test - @Order(11) fun testRenameDatasetWithTooLongNameViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(5000) - find(viewTree).findText(dsFinalName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Dataset") { - find(byXpath("//div[@class='JBTextField']")).text = "A".repeat(45) - } - clickButton("OK") - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - DATASET_NAME_LENGTH_MESSAGE - ) - Thread.sleep(3000) - clickButton("Cancel") - } + callRenameDatasetPoint(fixtureStack, anotherDsName, remoteRobot) + newDatasetNameInput(tooLongString45,fixtureStack,remoteRobot) + clickByText(OK_TEXT,fixtureStack,remoteRobot) + + val msgAll = find(messageLoc, Duration.ofSeconds(30) + ).findAllText() + var msg = "" + msgAll.forEach { msg += it.text } + + clickByText(CANCEL_TEXT, fixtureStack, remoteRobot) + Assertions.assertEquals(DATASET_NAME_LENGTH_MESSAGE, msg) } /** * Tests renaming DataSet to empty name and validates error notification. */ @Test - @Order(12) fun testRenameDatasetWithEmptyNameViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - Thread.sleep(5000) - find(viewTree).findText(dsFinalName).rightClick() - } - actionMenuItem(remoteRobot, "Rename").click() - dialog("Rename Dataset") { - find(byXpath("//div[@class='JBTextField']")).text = "" - } - clickButton("OK") - Thread.sleep(3000) - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - MEMBER_EMPTY_NAME_MESSAGE - ) - clickButton("Cancel") - } - } + callRenameDatasetPoint(fixtureStack, anotherDsName, remoteRobot) + newDatasetNameInput(EMPTY_STRING,fixtureStack,remoteRobot) + clickByText(OK_TEXT,fixtureStack,remoteRobot) - private fun allocateDSWithMock(testInfo: TestInfo, dsName: String, remoteRobot: RemoteRobot) { - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_${dsName}_restfiles", - { it?.requestLine?.contains("POST /zosmf/restfiles/ds/${dsName}") ?: false }, - { MockResponse().setBody("{\"dsorg\":\"PO\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":1,\"recfm\":\"VB\",\"blksize\":6120,\"lrecl\":255}") } - ) - allocateDataSet(wsName, dsName, projectName, fixtureStack, remoteRobot) - mapListDatasets[dsName] = listDS(dsName, "PDS", "PO") + val msgAll = find(messageLoc, Duration.ofSeconds(30) + ).findAllText() + var msg = "" + msgAll.forEach { msg += it.text } + + clickByText(CANCEL_TEXT, fixtureStack, remoteRobot) + Assertions.assertEquals(MEMBER_EMPTY_NAME_MESSAGE, msg) } private fun openWSAndListDatasets(testInfo: TestInfo, remoteRobot: RemoteRobot) { responseDispatcher.injectEndpoint( "${testInfo.displayName}_restfiles", { - it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${"$ZOS_USERID.UI.TEST*".uppercase()}") + it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=$pdsMaskName") ?: false }, { MockResponse().setBody(buildFinalListDatasetJson()) } ) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - } - - private fun allocateMemberWithMock( - testInfo: TestInfo, - dsName: String, - memberName: String, - remoteRobot: RemoteRobot - ) { - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_${memberName}in${dsName}_restfiles", - { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/${dsName}(${memberName})") ?: false }, - { MockResponse().setBody("{}") } - ) - allocateMemberForPDS(pdsName, memberName, projectName, fixtureStack, remoteRobot) + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) } private fun buildFinalListDatasetJson(): String { @@ -529,31 +298,4 @@ class RenameDatasetTest { } return result } - - private fun deleteDatasetsWithMock(testInfo: TestInfo, remoteRobot: RemoteRobot) { - val datasetsToBeDeleted = mutableListOf() - mapListDatasets.forEach { datasetsToBeDeleted.add(it.key) } - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_members", - { it?.requestLine?.contains("/member") ?: false }, - { MockResponse().setBody("{\"items\":[],\"returnedRows\":0,\"totalRows\":0,\"moreRows\":null,\"JSONversion\":1}") } - ) - datasetsToBeDeleted.forEach { entry -> - mapListDatasets.remove(entry) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_${entry}", - { - it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${"$ZOS_USERID.UI.TEST*".uppercase()}") - ?: false - }, - { MockResponse().setBody(buildFinalListDatasetJson()) } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_delete_${entry}", - { it?.requestLine?.contains("DELETE /zosmf/restfiles/ds/${entry}") ?: false }, - { MockResponse().setBody("{}") } - ) - deleteDataset(entry, projectName, fixtureStack, remoteRobot) - } - } } diff --git a/src/uiTest/kotlin/workingset/ViewDatasetTest.kt b/src/uiTest/kotlin/workingset/ViewDatasetTest.kt index cd282a881..ac46e9928 100644 --- a/src/uiTest/kotlin/workingset/ViewDatasetTest.kt +++ b/src/uiTest/kotlin/workingset/ViewDatasetTest.kt @@ -12,33 +12,47 @@ package workingset import auxiliary.* import auxiliary.closable.ClosableFixtureCollector -import auxiliary.containers.* import com.intellij.remoterobot.RemoteRobot -import com.intellij.remoterobot.fixtures.ComponentFixture -import com.intellij.remoterobot.fixtures.dataExtractor.RemoteText import com.intellij.remoterobot.search.locators.Locator -import com.intellij.remoterobot.utils.keyboard -import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.extension.ExtendWith -import java.awt.event.KeyEvent +import org.junit.jupiter.params.ParameterizedTest +import org.junit.jupiter.params.provider.MethodSource +import workingset.testutils.injectMemberContent +import org.junit.jupiter.params.provider.Arguments +import workingset.testutils.injectListAllAllocatedDatasetsWithContents +import workingset.testutils.injectPsDatasetContent +import workingset.testutils.injectSingleMember +import java.util.stream.Stream /** * Tests viewing dataset and members. */ @TestInstance(TestInstance.Lifecycle.PER_CLASS) @ExtendWith(RemoteRobotExtension::class) -class ViewDatasetTest { +class ViewDatasetTest :WorkingSetBase(){ private var closableFixtureCollector = ClosableFixtureCollector() private var fixtureStack = mutableListOf() private var wantToClose = mutableListOf("Add Working Set Dialog", "Create Mask Dialog") - private val projectName = "untitled" - private val connectionName = "valid connection" private var mapListDatasets = mutableMapOf() private var listMembersInDataset = mutableListOf() private val noItemsFoundMsg = "No items found" + private val wsNameWs1 = "WS1" + private val userIdPrefix = ZOS_USERID.uppercase() + private val emptyPdsDatasetNameTell = ".EMPTY.PDS".uppercase() + private val pdsDatasetName = "$ZOS_USERID.NONEMPTY.PDS".uppercase() + + companion object { + @JvmStatic + fun pairProvider(): Stream { + return Stream.of( + Arguments.of(WS_NAME_WS_2, EMPTY_MEMBER_CONTENT, ".EMPTY.PS"), + Arguments.of(WS_NAME_WS_3, SHORT_MEMBER_CONTENT, ".NONEMPTY.PS"), + ) + } + } /** * Opens the project and Explorer, clears test environment, creates valid connection. @@ -46,11 +60,10 @@ class ViewDatasetTest { @BeforeAll fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) createValidConnectionWithMock( testInfo, connectionName, - projectName, fixtureStack, closableFixtureCollector, remoteRobot @@ -62,19 +75,17 @@ class ViewDatasetTest { * Closes the project and clears test environment. */ @AfterAll - fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { + fun tearDownAll(remoteRobot: RemoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - close() - } + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + closeIntelligentProject(fixtureStack, remoteRobot) } /** * Closes all unclosed closable fixtures that we want to close. */ @AfterEach - fun tearDown(remoteRobot: RemoteRobot) = with(remoteRobot) { + fun tearDown(remoteRobot: RemoteRobot) { closableFixtureCollector.closeWantedClosables(wantToClose, remoteRobot) responseDispatcher.removeAllEndpoints() mapListDatasets.clear() @@ -85,130 +96,44 @@ class ViewDatasetTest { * Test to view empty PDS dataset and check that File Explorer contains correct message. */ @Test - fun testViewEmptyPDS(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS1" - val datasetName = "$ZOS_USERID.EMPTY.PDS".uppercase() - createWsAndMaskWithMock(wsName, datasetName, true, "PDS", "PO", testInfo, remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) + fun testViewEmptyPDS(testInfo: TestInfo, remoteRobot: RemoteRobot) { + createWsAndMaskWithMock(wsNameWs1, userIdPrefix+emptyPdsDatasetNameTell, EMPTY_MEMBER_CONTENT, PDS_TYPE, PO_ORG_SHORT, testInfo, remoteRobot) + openOrCloseWorkingSetInExplorer(wsNameWs1, fixtureStack, remoteRobot) findMessageInExplorer(noItemsFoundMsg, remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) } /** - * Test to view empty PS dataset. + * Views empty/non-empty PS dataset. */ - @Test - fun testViewEmptyPS(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS2" - val datasetName = "$ZOS_USERID.EMPTY.PS".uppercase() - viewPSDataset(wsName, datasetName, true, testInfo, remoteRobot) + @ParameterizedTest + @MethodSource("pairProvider") + fun testViewPsDataset(wsName: String, memberContent:String, tell: String, testInfo: TestInfo, remoteRobot: RemoteRobot){ + val datasetName = userIdPrefix + tell + createWsAndMaskWithMock(wsName, datasetName, memberContent, EMPTY_STRING, SEQUENTIAL_ORG_SHORT, testInfo, remoteRobot) + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) + + injectPsDatasetContent(testInfo, datasetName, memberContent) + openTreesElement(datasetName, remoteRobot) + closeMemberOrDataset(remoteRobot) } - /** - * Test to view non-empty PS dataset. - */ - @Test - fun testViewNonEmptyPS(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS3" - val datasetName = "$ZOS_USERID.NONEMPTY.PS".uppercase() - viewPSDataset(wsName, datasetName, false, testInfo, remoteRobot) - } /** * Test to view non-empty PDS dataset and view empty/non-empty members. */ @Test - fun testViewNonEmptyPDS(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS4" - val datasetName = "$ZOS_USERID.NONEMPTY.PDS".uppercase() - createWsAndMaskWithMock(wsName, datasetName, false, "PDS", "PO", testInfo, remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - openMember(datasetName, "MEMBER1", true, testInfo, remoteRobot) - closeMemberOrDataset(remoteRobot) - openMember(datasetName, "MEMBER2", false, testInfo, remoteRobot) - closeMemberOrDataset(remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - } + fun testViewNonEmptyPDS(testInfo: TestInfo, remoteRobot: RemoteRobot) { + createWsAndMaskWithMock(WS_NAME_WS_4, pdsDatasetName, SHORT_MEMBER_CONTENT, PDS_TYPE, PO_ORG_SHORT, testInfo, remoteRobot) + openOrCloseWorkingSetInExplorer(WS_NAME_WS_4, fixtureStack, remoteRobot) - /** - * Views empty/non-empty PS dataset. - */ - private fun viewPSDataset( - wsName: String, - datasetName: String, - isEmpty: Boolean, - testInfo: TestInfo, - remoteRobot: RemoteRobot - ) = with(remoteRobot) { - createWsAndMaskWithMock(wsName, datasetName, isEmpty, "", "PS", testInfo, remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - openPSDataset(datasetName, isEmpty, testInfo, remoteRobot) + injectMemberContent(testInfo,pdsDatasetName, MEMBER_NAME_1) + openTreesElement(MEMBER_NAME_1, remoteRobot) closeMemberOrDataset(remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - } - - /** - * Closes opened dataset member or PS dataset. - */ - private fun closeMemberOrDataset(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - with(textEditor()) { - keyboard { - hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_F4) - } - } - } - } - /** - * Opens PS dataset. - */ - private fun openPSDataset(datasetName: String, isEmpty: Boolean, testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val psContent = if (isEmpty) { - "" - } else { - "content" - } - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_getcontent", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds/-(TESTVOL)/$datasetName") ?: false }, - { MockResponse().setBody(psContent) } - ) - ideFrameImpl(projectName, fixtureStack) { - explorer { - find(viewTree).findAllText(datasetName).last().doubleClick() - Thread.sleep(2000) - } - } - } + injectMemberContent(testInfo,pdsDatasetName, MEMBER_NAME_2, "content") + openTreesElement(MEMBER_NAME_2, remoteRobot) + closeMemberOrDataset(remoteRobot) - /** - * Opens dataset member. - */ - private fun openMember( - datasetName: String, - memberName: String, - isEmpty: Boolean, - testInfo: TestInfo, - remoteRobot: RemoteRobot - ) = with(remoteRobot) { - val memberContent = if (isEmpty) { - "" - } else { - "content" - } - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_getmember", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${datasetName}($memberName)") ?: false }, - { MockResponse().setBody(memberContent) } - ) - ideFrameImpl(projectName, fixtureStack) { - explorer { - find(viewTree).findAllText(memberName).last().doubleClick() - Thread.sleep(2000) - } - } } /** @@ -217,54 +142,24 @@ class ViewDatasetTest { private fun createWsAndMaskWithMock( wsName: String, datasetName: String, - isEmpty: Boolean, + memberContent: String, dsNtp: String, dsOrg: String, testInfo: TestInfo, remoteRobot: RemoteRobot - ) = with(remoteRobot) { - val maskList = listOf(Pair(datasetName, "z/OS")) + ) { + val maskList = listOf(Pair(datasetName, ZOS_MASK)) mapListDatasets[datasetName] = listDS(datasetName, dsNtp, dsOrg) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles", - { - it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${datasetName}") - ?: false - }, - { MockResponse().setBody(buildResponseListJson(mapListDatasets, true)) } - ) - if (!isEmpty && dsNtp == "PDS") { + + injectListAllAllocatedDatasetsWithContents(testInfo, datasetName, mapListDatasets) + + if (memberContent != EMPTY_MEMBER_CONTENT && dsNtp == PDS_TYPE) { for (i in 1..5) { - listMembersInDataset.add("MEMBER$i") + listMembersInDataset.add(MEMBER_NAME_PATTERN+"$i") } - } - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles_listmembers", - { it?.requestLine?.contains("GET /zosmf/restfiles/ds/${datasetName}/member") ?: false }, - { - MockResponse().setBody(buildListMembersJson()) } - ) - createWsAndMask(projectName, wsName, maskList, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + injectSingleMember(testInfo, datasetName, listMembersInDataset) + createWsAndMask(wsName, maskList, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) } - private fun buildListMembersJson(): String { - var members = "[ " - if (listMembersInDataset.isNotEmpty()) { - listMembersInDataset.forEach { members += "{\"member\": \"${it}\"}," } - } - members = members.dropLast(1) + "]" - return "{\"items\":$members,\"returnedRows\": ${listMembersInDataset.size},\"JSONversion\": 1}" - } - - /** - * Checks if File Explorer contains expected message. - */ - private fun findMessageInExplorer(msg: String, remoteRobot: RemoteRobot) = with(remoteRobot) { - if (!find(viewTree).findAllText().map(RemoteText::text).joinToString("") - .contains(msg) - ) { - throw Exception("Expected message is not found") - } - } } \ No newline at end of file diff --git a/src/uiTest/kotlin/workingset/ViewWsPropertyTest.kt b/src/uiTest/kotlin/workingset/ViewWsPropertyTest.kt index 82b0344eb..07c97f130 100644 --- a/src/uiTest/kotlin/workingset/ViewWsPropertyTest.kt +++ b/src/uiTest/kotlin/workingset/ViewWsPropertyTest.kt @@ -12,14 +12,15 @@ package workingset import auxiliary.* import auxiliary.closable.ClosableFixtureCollector -import auxiliary.components.actionMenuItem import auxiliary.containers.* import com.intellij.remoterobot.RemoteRobot -import com.intellij.remoterobot.fixtures.ComponentFixture import com.intellij.remoterobot.search.locators.Locator import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.extension.ExtendWith +import workingset.testutils.injectAllocateUssFile +import workingset.testutils.injectListAllUssFiles +import workingset.testutils.injectSingleSpecificMember /** * Tests viewing dataset and uss file properties. @@ -27,7 +28,7 @@ import org.junit.jupiter.api.extension.ExtendWith @TestMethodOrder(MethodOrderer.OrderAnnotation::class) @TestInstance(TestInstance.Lifecycle.PER_CLASS) @ExtendWith(RemoteRobotExtension::class) -class ViewWsPropertyTest { +class ViewWsPropertyTest : WorkingSetBase() { private val closableFixtureCollector = ClosableFixtureCollector() private val fixtureStack = mutableListOf() private val mapListDatasets = mutableMapOf() @@ -35,8 +36,8 @@ class ViewWsPropertyTest { private val mapListUssFiles = mutableMapOf() private val projectName = "untitled" - private val connectionName = "con1" - private val wsName = "WS name" + override val connectionName = "con1" + override val wsName = "WS name" private val pdsName = "${ZOS_USERID.uppercase()}.UI.TEST" private val memberName = "TESTM" private val dsName = "${ZOS_USERID.uppercase()}.UI.TESTD" @@ -48,35 +49,30 @@ class ViewWsPropertyTest { private val memList = "{\"member\":\"$memberName\",\"vers\":1,\"mod\":0,\"c4date\":\"2015/08/12\",\"m4date\":\"2015/08/12\",\"cnorc\":22,\n" + "\"inorc\":22,\"mnorc\":0,\"mtime\":\"05:48\",\"msec\":\"43\",\"user\":\"IBMUSER\",\"sclm\":\"N\"}," - private val fileList = "{\"name\":\"$ussFileName\",\"mode\":\"-rwxr--rw-\",\"size\":20,\"uid\":0,\"user\":\"${ZOS_USERID.uppercase()}\",\"gid\":1,\n" + - "\"group\":\"OMVSGRP\",\"mtime\":\"2015-11-24T02:12:04\"}," - private val dirList = "{\"name\":\"$ussDirName\",\"mode\":\"drwxr--rw-\",\"size\":888, \"uid\":0, \"user\":\"${ZOS_USERID.uppercase()}\",\"gid\":1,\n" + - "\"group\":\"OMVSGRP\",\"mtime\":\"2013-05-07T11:23:08\"}," - private val dirHereList = "{\"name\":\".\", \"mode\":\"drwxrwxrwx\", \"size\":8192, \"uid\":0, \"user\":\"${ZOS_USERID.uppercase()}\", \"gid\":1, \n" + - "\"group\":\"OMVSGRP\", \"mtime\":\"2015-11-24T02:12:04\"}," - private val dirParentList = "{\"name\":\"..\", \"mode\":\"drwxr-xr-x\", \"size\":8192, \"uid\":0, \"user\":\"${ZOS_USERID.uppercase()}\", \"gid\":1, \n" + - "\"group\":\"OMVSGRP\", \"mtime\":\"2015-09-15T02:38:29\"}," + private val fileList = + "{\"name\":\"$ussFileName\",\"mode\":\"-rwxr--rw-\",\"size\":20,\"uid\":0,\"user\":\"${ZOS_USERID.uppercase()}\",\"gid\":1,\n" + + "\"group\":\"OMVSGRP\",\"mtime\":\"2015-11-24T02:12:04\"}," + private val dirList = + "{\"name\":\"$ussDirName\",\"mode\":\"drwxr--rw-\",\"size\":888, \"uid\":0, \"user\":\"${ZOS_USERID.uppercase()}\",\"gid\":1,\n" + + "\"group\":\"OMVSGRP\",\"mtime\":\"2013-05-07T11:23:08\"}," + private val dirHereList = + "{\"name\":\".\", \"mode\":\"drwxrwxrwx\", \"size\":8192, \"uid\":0, \"user\":\"${ZOS_USERID.uppercase()}\", \"gid\":1, \n" + + "\"group\":\"OMVSGRP\", \"mtime\":\"2015-11-24T02:12:04\"}," + private val dirParentList = + "{\"name\":\"..\", \"mode\":\"drwxr-xr-x\", \"size\":8192, \"uid\":0, \"user\":\"${ZOS_USERID.uppercase()}\", \"gid\":1, \n" + + "\"group\":\"OMVSGRP\", \"mtime\":\"2015-09-15T02:38:29\"}," /** * Opens the project and Explorer, clears test environment, creates working set with dataset and uss masks, * allocates dataset and pds with member, creates uss file and directory. */ @BeforeAll - fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { + fun setUpAll(testInfo: TestInfo, remoteRobot: RemoteRobot) { startMockServer() - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_info", - { it?.requestLine?.contains("zosmf/info") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_resttopology", - { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + responseDispatcher.injectTestInfo(testInfo) + responseDispatcher.injectTestInfoRestTopology(testInfo) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) createConnection( - projectName, fixtureStack, closableFixtureCollector, connectionName, @@ -84,7 +80,7 @@ class ViewWsPropertyTest { remoteRobot, "https://${mockServer.hostName}:${mockServer.port}" ) - createWsWithoutMask(projectName, wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + createWsWithoutMask(wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) responseDispatcher.injectEndpoint( "listAllAllocatedDatasets_restfiles", { it?.requestLine?.contains("GET /zosmf/restfiles/ds?dslevel=${maskName}") ?: false }, @@ -97,201 +93,187 @@ class ViewWsPropertyTest { ) mapListUssFiles["."] = dirHereList mapListUssFiles[".."] = dirParentList - responseDispatcher.injectEndpoint( - "listAllUssFiles_restfiles", - { it?.requestLine?.contains("GET /zosmf/restfiles/fs?path") ?: false }, - { MockResponse().setBody(buildResponseListJson(mapListUssFiles, true)) } - ) - responseDispatcher.injectEndpoint( - "allocateDatasetMember_restfiles", - { it?.requestLine?.contains("PUT /zosmf/restfiles/ds/$pdsName($memberName)") ?: false }, - { MockResponse().setResponseCode(204) } + injectListAllUssFiles(mapListUssFiles) + injectSingleSpecificMember(pdsName, memberName) + responseDispatcher.injectAllocationResultPds(pdsName) + responseDispatcher.injectAllocationResultPo( + SEQUENTIAL_ORG_SHORT, + TRACKS_ALLOCATION_UNIT_SHORT, + dsName, + VB_RECORD_FORMAT_SHORT, + 255 ) - responseDispatcher.injectEndpoint( - "allocatePds_restfiles", - { it?.requestLine?.contains("POST /zosmf/restfiles/ds/$pdsName") ?: false }, - { MockResponse().setBody("{\"dsorg\":\"PDS\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":2,\"recfm\":\"VB\",\"blksize\":6120,\"lrecl\":255}") } - ) - responseDispatcher.injectEndpoint( - "allocateDs_restfiles", - { it?.requestLine?.contains("POST /zosmf/restfiles/ds/$dsName") ?: false }, - { MockResponse().setBody("{\"dsorg\":\"PS\",\"alcunit\":\"TRK\",\"primary\":10,\"secondary\":1,\"dirblk\":0,\"recfm\":\"VB\",\"blksize\":6120,\"lrecl\":255}") } - ) - responseDispatcher.injectEndpoint( - "allocateUssFile_restfiles", - { it?.requestLine?.contains("POST /zosmf/restfiles/fs$ussMaskName/$ussFileName") ?: false }, - { MockResponse().setResponseCode(201) } - ) - responseDispatcher.injectEndpoint( - "allocateUssDir_restfiles", - { it?.requestLine?.contains("POST /zosmf/restfiles/fs$ussMaskName/$ussDirName") ?: false }, - { MockResponse().setResponseCode(201) } - ) - mapListDatasets[pdsName] = listDS(pdsName, "PDS", "PO") + injectAllocateUssFile(ussMaskName, ussFileName) + injectAllocateUssFile(ussMaskName, ussDirName) + mapListDatasets[pdsName] = listDS(pdsName, PDS_TYPE, PO_ORG_SHORT) allocatePDSAndCreateMask( wsName, pdsName, - projectName, fixtureStack, closableFixtureCollector, remoteRobot, maskName, directory = 2 ) - ideFrameImpl(projectName, fixtureStack) { - createMask(wsName, fixtureStack, closableFixtureCollector) - createMaskDialog(fixtureStack) { - createMask(Pair(ussMaskName, "USS")) - clickButton("OK") - Thread.sleep(3000) - } - } - mapListDatasets[dsName] = listDS(dsName, "", "PS") - allocateDataSet(wsName, dsName, projectName, fixtureStack, remoteRobot) + createMask(wsName, ussMaskName, fixtureStack, closableFixtureCollector, USS_MASK, remoteRobot) + mapListDatasets[dsName] = listDS(dsName, "", SEQUENTIAL_ORG_SHORT) + allocateDataSet(wsName, dsName, fixtureStack, remoteRobot) mapListDatasetMembers[memberName] = memList - allocateMemberForPDS(pdsName, memberName, projectName, fixtureStack, remoteRobot) - createUssFile(ussMaskName, ussFileName, UssFileType.File, projectName, fixtureStack, remoteRobot) + allocateMemberForPDS(pdsName, memberName, fixtureStack, remoteRobot) mapListUssFiles[ussFileName] = fileList - createUssFile(ussMaskName, ussDirName, UssFileType.Directory, projectName, fixtureStack, remoteRobot) mapListUssFiles[ussDirName] = dirList - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(pdsName).doubleClick() - Thread.sleep(1000) - find(viewTree).findText(ussMaskName).doubleClick() - Thread.sleep(1000) - } - } + openTreesElement(pdsName, remoteRobot) + openTreesElement(ussMaskName, remoteRobot) } /** * Closes the project and clears test environment. */ @AfterAll - fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { + fun tearDownAll(remoteRobot: RemoteRobot) { mockServer.shutdown() - - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - close() - } + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + closeIntelligentProject(fixtureStack, remoteRobot) } /** * Test to check if member properties in opened dialog and expected values are matching */ @Test - @Order(1) - fun testViewMemberProperties(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - Thread.sleep(1000) - find(viewTree).findText(memberName).rightClick() - } - actionMenuItem(remoteRobot, "Properties").click() - memberPropertiesDialog(fixtureStack) { - if (!areMemberPropertiesValid(memberName, "1.0", "2015/08/12", "2015/08/12", - "05:48", "IBMUSER", "22", "22", "0")) { - throw Exception("Properties in opened 'Member Properties' dialog are different from expected") - } - clickButton("OK") - } - } + fun testViewMemberProperties(remoteRobot: RemoteRobot) { + callTreesElementProperty(memberName, fixtureStack, remoteRobot) + val memberPropertiesValid = isMemberPropertyValid( + memberName, + "1.0", + "2015/08/12", + "2015/08/12", + "05:48", + "IBMUSER", + "22", + "22", + "0", + remoteRobot + ) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + Assertions.assertTrue(memberPropertiesValid) } /** * Test to check if dataset properties in opened dialog and expected values are matching */ @Test - @Order(2) - fun testViewDatasetProperties(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - Thread.sleep(1000) - find(viewTree).findText(dsName).rightClick() - } - actionMenuItem(remoteRobot, "Properties").click() - datasetPropertiesDialog(fixtureStack) { - if (!areDatasetPropertiesValid(dsName, "", "TEST.CATALOG.MASTER", "TESTVOL", - "3390", "PS", "VB", "255", "3200", "10", - "TRACKS", "1", "1", "2021/11/15", "2021/11/17", "***None***")) { - throw Exception("Properties in opened 'Dataset Properties' dialog are different from expected") - } - clickButton("OK") - } - } + fun testViewDatasetProperties(remoteRobot: RemoteRobot) { + openPropertyDatasetName(dsName, fixtureStack, remoteRobot) + val datasetPropertyValid = isDatasetPropertyValid( + dsName, "", "TEST.CATALOG.MASTER", "TESTVOL", + "3390", "Sequential (PS)", "VB", "255", "3200", "10", + "TRACKS", "1", "1", "2021/11/15", "2021/11/17", "***None***", remoteRobot + ) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + Assertions.assertTrue(datasetPropertyValid) } /** * Test to check if pds properties in opened dialog and expected values are matching */ @Test - @Order(3) - fun testViewPdsProperties(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - Thread.sleep(1000) - find(viewTree).findText(pdsName).rightClick() - } - actionMenuItem(remoteRobot, "Properties").click() - datasetPropertiesDialog(fixtureStack) { - if (!areDatasetPropertiesValid(pdsName, "PDS", "TEST.CATALOG.MASTER", "TESTVOL", - "3390", "PO", "VB", "255", "3200", "10", - "TRACKS", "1", "1", "2021/11/15", "2021/11/17", "***None***")) { - throw Exception("Properties in opened 'Dataset Properties' dialog are different from expected") - } - clickButton("OK") - } - } + fun testViewPdsProperties(remoteRobot: RemoteRobot) { + openPropertyDatasetName(pdsName, fixtureStack, remoteRobot) + val datasetPropertyValid = isDatasetPropertyValid( + pdsName, "PDS", "TEST.CATALOG.MASTER", "TESTVOL", + "3390", PO_ORG_FULL, "VB", "255", "3200", "10", + "TRACKS", "1", "1", "2021/11/15", "2021/11/17", "***None***", remoteRobot + ) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + Assertions.assertTrue(datasetPropertyValid) } /** * Test to check if uss file properties in opened dialog and expected values are matching */ @Test - @Order(4) - fun testViewUssFileProperties(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - Thread.sleep(1000) - find(viewTree).findText(ussFileName).rightClick() - } - actionMenuItem(remoteRobot, "Properties").click() - ussFilePropertiesDialog(fixtureStack) { - if (!areUssFilePropertiesValid(ussFileName, ussMaskName.drop(1), "$ussMaskName/$ussFileName", "20 bytes", - "2015-11-24T02:12:04", "USER", "OMVSGRP", "1", "READ_WRITE_EXECUTE", "READ", "READ_WRITE")) { - throw Exception("Properties in opened 'File Properties' dialog are different from expected") - } - clickButton("OK") - } - } + fun testViewUssFileProperties(remoteRobot: RemoteRobot) { + openPropertyDatasetName(ussFileName, fixtureStack, remoteRobot) + val validPropertyStatus = isUssFilePropertyValid( + ussFileName, ussMaskName.drop(1), "$ussMaskName/$ussFileName", "20 bytes", + "2015-11-24T02:12:04", ZOS_USERID.uppercase(), "OMVSGRP", "1", + "READ_WRITE_EXECUTE", "READ", "READ_WRITE", remoteRobot + ) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + Assertions.assertTrue(validPropertyStatus) } /** * Test to check if uss directory properties in opened dialog and expected values are matching */ @Test - @Order(5) - fun testViewUssDirProperties(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - Thread.sleep(1000) - find(viewTree).findText(ussDirName).rightClick() + fun testViewUssDirProperties(remoteRobot: RemoteRobot) { + openPropertyDatasetName(ussDirName, fixtureStack, remoteRobot) + val validPropertyStatus = isUssFilePropertyValid( + ussDirName, ussMaskName.drop(1), + "$ussMaskName/$ussDirName", "888 bytes", "2013-05-07T11:23:08", ZOS_USERID.uppercase(), + "OMVSGRP", "1", "READ_WRITE_EXECUTE", "READ", "READ_WRITE", + remoteRobot + ) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + Assertions.assertTrue(validPropertyStatus) + } + + private fun isMemberPropertyValid( + memName: String, + version: String, + createDate: String, + modDate: String, + modTime: String, + userId: String, + curRecNum: String, + begRecNum: String, + changedRecNum: String, + remoteRobot: RemoteRobot, + ): Boolean = with(remoteRobot) { + var isValid = false + ideFrameImpl(PROJECT_NAME, fixtureStack) { + memberPropertiesDialog(fixtureStack) { + isValid = areMemberPropertiesValid( + memName, version, createDate, modDate, modTime, userId, curRecNum, begRecNum, changedRecNum + ) } - actionMenuItem(remoteRobot, "Properties").click() + } + return isValid + } + + private fun isDatasetPropertyValid( + dsName: String, dsNameType: String, catalogName: String, volumeSerials: String, deviceType: String, + organization: String, recordFormat: String, recordLength: String, blockSize: String, sizeInTracks: String, + spaceUnits: String, usedTracks: String, usedExtents: String, createDate: String, modDate: String, + expirationDate: String, remoteRobot: RemoteRobot, + ): Boolean = with(remoteRobot) { + var isValid = false + ideFrameImpl(PROJECT_NAME, fixtureStack) { + datasetPropertiesDialog(fixtureStack) { + isValid = areDatasetPropertiesValid( + dsName, dsNameType, catalogName, volumeSerials, deviceType, organization, recordFormat, + recordLength, blockSize, sizeInTracks, spaceUnits, usedTracks, usedExtents, createDate, modDate, + expirationDate + ) + } + } + return isValid + } + + private fun isUssFilePropertyValid( + fileName: String, location: String, path: String, size: String, modDate: String, + owner: String, group: String, groupId: String, ownerPerm: String, groupPerm: String, + allPerm: String, remoteRobot: RemoteRobot, + ): Boolean = with(remoteRobot) { + var isValid = false + ideFrameImpl(PROJECT_NAME, fixtureStack) { ussFilePropertiesDialog(fixtureStack) { - if (!areUssFilePropertiesValid(ussDirName, ussMaskName.drop(1), "$ussMaskName/$ussDirName", "888 bytes", - "2013-05-07T11:23:08", "USER", "OMVSGRP", "1", "READ_WRITE_EXECUTE", "READ", "READ_WRITE")) { - throw Exception("Properties in opened 'Directory Properties' dialog are different from expected") - } - clickButton("OK") + isValid = areUssFilePropertiesValid( + fileName, location, path, size, modDate, owner, group, groupId, ownerPerm, groupPerm, allPerm + ) } } + return isValid } } \ No newline at end of file diff --git a/src/uiTest/kotlin/workingset/WorkingSetBase.kt b/src/uiTest/kotlin/workingset/WorkingSetBase.kt index 554e9aa7f..1d34d7bde 100644 --- a/src/uiTest/kotlin/workingset/WorkingSetBase.kt +++ b/src/uiTest/kotlin/workingset/WorkingSetBase.kt @@ -1,5 +1,3 @@ -package workingset - /* * This program and the accompanying materials are made available under the terms of the * Eclipse Public License v2.0 which accompanies this distribution, and is available at @@ -9,6 +7,9 @@ package workingset * * Copyright IBA Group 2020 */ +package workingset + + import auxiliary.* import auxiliary.closable.ClosableFixtureCollector @@ -18,10 +19,16 @@ import auxiliary.containers.* import com.intellij.remoterobot.RemoteRobot import com.intellij.remoterobot.fixtures.ComponentFixture import com.intellij.remoterobot.fixtures.ContainerFixture +import com.intellij.remoterobot.fixtures.HeavyWeightWindowFixture +import com.intellij.remoterobot.fixtures.JTextFieldFixture +import com.intellij.remoterobot.fixtures.dataExtractor.RemoteText import com.intellij.remoterobot.search.locators.Locator -import com.intellij.remoterobot.search.locators.byXpath +import com.intellij.remoterobot.utils.keyboard import org.junit.jupiter.api.* import org.junit.jupiter.api.extension.ExtendWith +import workingset.auxiliary.components.dialogs.* +import workingset.auxiliary.components.elements.ButtonElement +import java.awt.event.KeyEvent /** * Tests allocating datasets with valid and invalid inputs. @@ -31,15 +38,29 @@ import org.junit.jupiter.api.extension.ExtendWith open class WorkingSetBase { private var closableFixtureCollector = ClosableFixtureCollector() private var fixtureStack = mutableListOf() - private var mapListDatasets = mutableMapOf() - private val projectName = "untitled" open val connectionName = "valid connection" - // - private val wsName = "WS1" + var okButton: ButtonElement = ButtonElement() + var yesButton: ButtonElement = ButtonElement() + var cancelButton: ButtonElement = ButtonElement() + + open val wsName = "WS1" internal open val datasetName = "$ZOS_USERID.ALLOC." - internal fun buildDatasetConfigString(dsName: String, dsntp: String, datasetOrganization:String, recordLength:Int, recordFormatShort:String): String{ + var editWorkingSetSubDialog: EditWorkingSetSubDialog = EditWorkingSetSubDialog() + var createMaskSubDialog: CreateMaskSubDialog = CreateMaskSubDialog() + var renameDatasetMaskDialog: RenameDatasetMaskDialog = RenameDatasetMaskDialog() + var deletionOfDSMask: DeletionOfDSMask = DeletionOfDSMask() + var deletionOfUssPathRoot: DeletionOfUssPathRoot = DeletionOfUssPathRoot() + var addWorkingSetDialog = AddWorkingSetSubDialog() + + internal fun buildDatasetConfigString( + dsName: String, + dsntp: String, + datasetOrganization: String, + recordLength: Int, + recordFormatShort: String + ): String { return "{\n" + " \"dsname\": \"${dsName}\",\n" + " \"blksz\": \"3200\",\n" + @@ -64,36 +85,6 @@ open class WorkingSetBase { " }," } - internal fun allocateDataSetWithDefaultConfig( - wsName: String, - datasetName: String, - projectName: String, - fixtureStack: MutableList, - remoteRobot: RemoteRobot - ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(wsName).rightClick() - } - actionMenu(remoteRobot, NEW_POINT_TEXT).click() - actionMenuItem(remoteRobot, DATASET_POINT_TEXT).click() - allocateDatasetDialog(fixtureStack) { - allocateDataset(datasetName, PARTITIONED_EXTENDED_NAME_FULL, "TRK", 10, 1, 0, "VB", 255, 6120) - clickButton(OK_TEXT) -// Thread.sleep(10000) - } - find(myDialogXpathLoc).findText(datasetHasBeenCreated.format(datasetName)) - clickButton(NO_TEXT) - explorer { - fileExplorer.click() - find(viewTree).findText(wsName).rightClick() - } - actionMenuItem(remoteRobot, REFRESH_POINT_TEXT).click() -// Thread.sleep(3000) - } - } - fun listDS(dsName: String, dsNtp: String, dsOrg: String): String { return "{\n" + " \"dsname\": \"${dsName}\",\n" + @@ -118,46 +109,49 @@ open class WorkingSetBase { " \"vols\": \"TESTVOL\"\n" + " }," } + internal fun allocateDataSet( - wsName: String, - datasetName: String, - datasetOrganization: String, - allocationUnit: String, - primaryAllocation: Int, - secondaryAllocation: Int, - directory: Int, - recordFormat: String, - recordLength: Int, - blockSize: Int, - averageBlockLength: Int, - remoteRobot: RemoteRobot, + wsName: String, + datasetName: String, + datasetOrganization: String, + allocationUnit: String, + primaryAllocation: Int, + secondaryAllocation: Int, + directory: Int, + recordFormat: String, + recordLength: Int, + blockSize: Int, + averageBlockLength: Int, + remoteRobot: RemoteRobot, ) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { fileExplorer.click() find(viewTree).findText(wsName).rightClick() } - actionMenu(remoteRobot, "New").click() - actionMenuItem(remoteRobot, "Dataset").click() + actionMenu(remoteRobot, NEW_POINT_TEXT).click() + actionMenuItem(remoteRobot, DATASET_POINT_TEXT).click() allocateDatasetDialog(fixtureStack) { allocateDataset( - datasetName, - datasetOrganization, - allocationUnit, - primaryAllocation, - secondaryAllocation, - directory, - recordFormat, - recordLength, - blockSize, - averageBlockLength + datasetName, + datasetOrganization, + allocationUnit, + primaryAllocation, + secondaryAllocation, + directory, + recordFormat, + recordLength, + blockSize, + averageBlockLength ) - clickButton("OK") - find(myDialogXpathLoc).findText("Dataset $datasetName Has Been ") - clickButton("No") -// closableFixtureCollector.closeOnceIfExists(AllocateDatasetDialog.name) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + + closableFixtureCollector.closeOnceIfExists(AllocateDatasetDialog.name) } +// Thread.sleep(3000) +// find(myDialogXpathLoc).findText("Dataset $datasetName Has Been ") +// clickButton("No") } } @@ -166,18 +160,18 @@ open class WorkingSetBase { * Creates working set and z/OS mask. */ internal fun createWsAndMask(remoteRobot: RemoteRobot) = with(remoteRobot) { - createWsWithoutMask(projectName, wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + createWsWithoutMask(wsName, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { createMask(wsName, fixtureStack, closableFixtureCollector) createMaskDialog(fixtureStack) { - createMask(Pair("$datasetName*", "z/OS")) - clickButton("OK") + createMask(Pair("$datasetName*", ZOS_MASK)) + clickByText(OK_TEXT, fixtureStack, remoteRobot) } closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) } } - internal fun buildFinalListDatasetJson(): String { + internal fun buildFinalListDatasetJson(mapListDatasets: MutableMap): String { var result = "{}" if (mapListDatasets.isNotEmpty()) { var listDatasetsJson = "{\"items\":[" @@ -192,4 +186,369 @@ open class WorkingSetBase { } return result } -} \ No newline at end of file + + private fun openDatasetProperty(datasetName: String, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(datasetName).rightClick() + } + } + } + + internal fun migrateDataset( + datasetName: String, + migratedDs: String, + mapListDatasets: MutableMap, + fixtureStack: MutableList, + remoteRobot: RemoteRobot + ) = with(remoteRobot) { + openDatasetProperty(datasetName, remoteRobot) + responseDispatcher.injectMigratePdsRestFiles(datasetName, HMIGRATE_MIGRATE_OPTIONS) + mapListDatasets[datasetName] = migratedDs + ideFrameImpl(PROJECT_NAME, fixtureStack) { + actionMenuItem(remoteRobot, MIGRATE_POINT_TEXT).click() + } + } + + internal fun recallDataset( + datasetName: String, + fixtureStack: MutableList, + mapListDatasets: MutableMap, + remoteRobot: RemoteRobot + ) = with(remoteRobot) { + openDatasetProperty(datasetName, remoteRobot) + responseDispatcher.injectRecallPds(datasetName) + mapListDatasets[datasetName] = listDS(datasetName, PDS_TYPE, PO_ORG_SHORT) + ideFrameImpl(PROJECT_NAME, fixtureStack) { + actionMenuItem(remoteRobot, RECALL_POINT_TEXT).click() + } + } + + internal fun callRenameDatasetPoint( + fixtureStack: MutableList, + datasetName: String, + remoteRobot: RemoteRobot + ) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + Thread.sleep(3000) + find(viewTree).findText(datasetName).rightClick() + } + actionMenuItem(remoteRobot, RENAME_POINT_TEXT).click() + } + } + + internal fun callRenameMemberPoint( + fixtureStack: MutableList, + datasetName: String, + memberName: String, + remoteRobot: RemoteRobot + ) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(datasetName).doubleClick() + find(viewTree).findText(memberName).rightClick() + } + actionMenuItem(remoteRobot, RENAME_POINT_TEXT).click() + } + } + + internal fun callEditWSFromContextMenu( + datasetName: String, + fixtureStack: MutableList, + remoteRobot: RemoteRobot + ) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(datasetName).doubleClick() + find(viewTree).findText(datasetName).rightClick() + } + actionMenuItem(remoteRobot, EDIT_POINT_TEXT).click() + } + } + + internal fun callTreesElementProperty( + treesElement: String, + fixtureStack: MutableList, + remoteRobot: RemoteRobot + ) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + find(viewTree).findText(treesElement).rightClick() + } + actionMenuItem(remoteRobot, PROPERTIES_POINT_TEXT).click() + } + } + + internal fun newMemberNameInput(newName: String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + dialog(RENAME_MEMBER_NAME) { + find(inputFieldLoc).text = newName + } + } + } + + internal fun newDatasetNameInput(newName: String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + dialog(RENAME_DATASET_NAME) { + find(inputFieldLoc).text = newName + } + } + } + + /** + * Checks if File Explorer contains expected message. + */ + internal fun findMessageInExplorer(msg: String, remoteRobot: RemoteRobot) = with(remoteRobot) { + if (!find(viewTree).findAllText().map(RemoteText::text).joinToString("") + .contains(msg) + ) { + throw Exception("Expected message is not found") + } + } + + internal fun closeIntelligentProject(fixtureStack: MutableList, remoteRobot: RemoteRobot) = + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + close() + } + } + + internal fun refreshWorkSpace(wsName:String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) + { + explorer { + fileExplorer.click() + find(viewTree).findText(wsName).rightClick() + } + actionMenuItem(remoteRobot, REFRESH_POINT_TEXT).click() + } + } + + internal fun compressAndDecompressTree(wsName:String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + find(viewTree).findAllText(wsName).last().doubleClick() + } + } + } + + /** + * Closes opened dataset member or PS dataset. + */ + internal fun closeMemberOrDataset(remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + with(textEditor()) { + keyboard { + hotKey(KeyEvent.VK_CONTROL, KeyEvent.VK_F4) + } + } + } + } + + /** + * Open property + */ + internal fun openPropertyDatasetName(dsName: String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + fileExplorer.click() + Thread.sleep(1000) + find(viewTree).findText(dsName).rightClick() + } + actionMenuItem(remoteRobot, PROPERTIES_POINT_TEXT).click() + } + } + + fun openTreesElement(treesElement: String, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + explorer { + find(viewTree).findAllText(treesElement).last().doubleClick() + } + } + } + + fun callCreateWorkingSetFromActionButton(fixtureStack: MutableList, remoteRobot: RemoteRobot) = + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + callCreateWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) + } + } + + fun callCreateWSFromContextMenu(fixtureStack: MutableList, remoteRobot: RemoteRobot) = + with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + createWSFromContextMenu(fixtureStack, closableFixtureCollector) + } + } + +// fun createWsWithConnection(connectionName: String, wsName: String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = +// with(remoteRobot) { +// callCreateWSFromContextMenu(fixtureStack, remoteRobot) +// addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName, fixtureStack, remoteRobot) +// okButton.click() +// } +// fun createWsWithConnection(connectionName: String, wsName: String, mask: Pair, fixtureStack: MutableList, remoteRobot: RemoteRobot) = +// with(remoteRobot) { +// callCreateWSFromContextMenu(fixtureStack, remoteRobot) +// addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName, mask, fixtureStack, remoteRobot) +// okButton.click() +// } +// fun callCreateConnectionFromActionButton(fixtureStack: MutableList, remoteRobot: RemoteRobot) = +// with(remoteRobot) { +// ideFrameImpl(PROJECT_NAME, fixtureStack) { +// createConnectionFromActionButton(closableFixtureCollector, fixtureStack) +// } +// } + + fun createWsWithConnectionFromAction( + connectionName: String, + wsName: String, + fixtureStack: MutableList, + closableFixtureCollector: ClosableFixtureCollector, + remoteRobot: RemoteRobot, + ) { + callCreateWorkingSetFromActionButton(fixtureStack, remoteRobot) + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName, fixtureStack, remoteRobot) + return addWorkingSetDialog.clickButtonByName(OK_TEXT) + } + + fun createWsWithConnectionFromAction(connectionName: String, wsName: String, mask: Pair, fixtureStack: MutableList, remoteRobot: RemoteRobot) { + callCreateWorkingSetFromActionButton(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName, mask, fixtureStack, remoteRobot) + return addWorkingSetDialog.clickButtonByName(OK_TEXT) + } + + fun fillConnectionFilds( + connectionName: String = PROJECT_NAME, hostName: String = mockServer.hostName, port: Int = mockServer.port, + userId: String = ZOS_USERID, userPwd: String=ZOS_PWD, ssl: Boolean=true, protocol: String = "https", + fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + addConnectionDialog(fixtureStack) + { + addConnection(connectionName, "${protocol}://${hostName}:${port}", userId, userPwd, ssl) + } + } + } + + + + + fun deleteInEditWorkingSet(masks: List, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + editWorkingSetDialog(fixtureStack) { + deleteMasks(masks) + } + } + } + + fun changeConnectionInEditWorkingSet(newConnectionName: String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + editWorkingSetDialog(fixtureStack) { + changeConnection(newConnectionName) + } + } + } + + + + fun fillEditWorkingSet(connectionName:String, wsName:String, mask: Pair, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + editWorkingSetDialog(fixtureStack) { + addWorkingSet(wsName, connectionName, mask) + } + } + } + + + fun isButtonEnableByTextAddWorkingSet(buttonText:String, fixtureStack: MutableList, remoteRobot: RemoteRobot): Boolean = with(remoteRobot) { + var status = false + ideFrameImpl(PROJECT_NAME, fixtureStack) { + addWorkingSetDialog(fixtureStack) { + status = button(buttonText).isEnabled() + } + } + return status + } + + + fun hoverToByTextAddWorkingSet(text:String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + addWorkingSetDialog(fixtureStack) { + findText(text).moveMouse() + } + } + } + + fun clickToByTextAddWorkingSet(text:String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + addWorkingSetDialog(fixtureStack) { + findText(text).click() + } + } + } + + fun clickActionButtonByXpath(xpath: Locator, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + addWorkingSetDialog(fixtureStack) { + clickActionButton(xpath) + } + } + } + + fun setInComboBox(newConnectionName: String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + editWorkingSetDialog(fixtureStack) { + changeConnection(newConnectionName) + } + } + } + + fun deleteWSFromContextMenu(wsName:String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + deleteWSFromContextMenu(wsName) + } + yesButton.click() + } + + fun callCreateMask(wsName:String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { + createMask(wsName, fixtureStack, closableFixtureCollector) + } + } + + fun decompressWsIfCompressed(wsName1: String, compressedId: String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + try { + find(treesLoc).findText(compressedId) + }catch(_: NoSuchElementException){ + compressAndDecompressTree(wsName1, fixtureStack, remoteRobot) + } + } + + fun compressWsIfcDecompressed(wsName1: String, compressedId: String, fixtureStack: MutableList, remoteRobot: RemoteRobot) = with(remoteRobot) { + try { + find(treesLoc).findText(compressedId) + compressAndDecompressTree(wsName1, fixtureStack, remoteRobot) + }catch(_: NoSuchElementException){} + } +} + + + + + + + + + + + diff --git a/src/uiTest/kotlin/workingset/WorkingSetViaActionButtonTest.kt b/src/uiTest/kotlin/workingset/WorkingSetViaActionButtonTest.kt index c9618124d..6e6a6c468 100644 --- a/src/uiTest/kotlin/workingset/WorkingSetViaActionButtonTest.kt +++ b/src/uiTest/kotlin/workingset/WorkingSetViaActionButtonTest.kt @@ -16,14 +16,13 @@ import auxiliary.containers.* import com.intellij.remoterobot.RemoteRobot import com.intellij.remoterobot.fixtures.HeavyWeightWindowFixture import com.intellij.remoterobot.search.locators.Locator -import com.intellij.remoterobot.search.locators.byXpath import com.intellij.remoterobot.utils.WaitForConditionTimeoutException import io.kotest.matchers.string.shouldContain -import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.Assertions.assertFalse import org.junit.jupiter.api.extension.ExtendWith -import java.time.Duration +import workingset.auxiliary.components.dialogs.AddWorkingSetSubDialog +import workingset.testutils.injectListEmptyData /** @@ -32,33 +31,35 @@ import java.time.Duration @TestMethodOrder(MethodOrderer.OrderAnnotation::class) @TestInstance(TestInstance.Lifecycle.PER_CLASS) @ExtendWith(RemoteRobotExtension::class) -class WorkingSetViaActionButtonTest { +class WorkingSetViaActionButtonTest : WorkingSetBase() { private var closableFixtureCollector = ClosableFixtureCollector() private var fixtureStack = mutableListOf() private var wantToClose = mutableListOf("Add Working Set Dialog") - - private val projectName = "untitled" - private val connectionName = "valid connection" + private val wsName1 = "WS1" + private val wsName2 = "WS2" + private val wsName3 = "WS3" + private val wsName4 = "WS4" + private val wsName5 = "WS5" /** * Opens the project and Explorer, clears test environment. */ @BeforeAll - fun setUpAll(remoteRobot: RemoteRobot) { + fun setUpAll(remoteRobot: RemoteRobot, testInfo:TestInfo) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + createValidConnectionWithMock(testInfo, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + addWorkingSetDialog = AddWorkingSetSubDialog(fixtureStack, remoteRobot) } /** * Closes the project and clears test environment. */ @AfterAll - fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { + fun tearDownAll(remoteRobot: RemoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - close() - } + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + closeIntelligentProject(fixtureStack, remoteRobot) } /** @@ -73,233 +74,190 @@ class WorkingSetViaActionButtonTest { /** * Tests to add new working set without connection, checks that correct message is returned. */ - @Test - @Order(1) - fun testAddWorkingSetWithoutConnectionViaActionButton(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val wsName = "first ws" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_info", - { it?.requestLine?.contains("zosmf/info") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_resttopology", - { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - ideFrameImpl(projectName, fixtureStack) { - createWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) - try { - if (dialog("Add Working Set Dialog").isShowing) { - Assertions.assertTrue(false) - } - } catch (e: WaitForConditionTimeoutException) { - e.message.shouldContain("Failed to find 'Dialog' by 'title Add Working Set Dialog'") - } finally { - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } - createConnectionFromActionButton(closableFixtureCollector, fixtureStack) - addConnectionDialog(fixtureStack) { - addConnection( - connectionName, - "https://${mockServer.hostName}:${mockServer.port}", - ZOS_USERID, - ZOS_PWD, - true - ) - clickButton("OK") - Thread.sleep(2000) - } - createWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName) - clickButton("OK") - Thread.sleep(2000) - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - EMPTY_DATASET_MESSAGE - ) - clickButton("OK") - Thread.sleep(2000) - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } - } + + + /** * Tests to add new empty working set with very long name, checks that correct message is returned. */ @Test - @Order(2) + @Disabled("https://jira.ibagroup.eu/browse/IJMP-977") fun testAddEmptyWorkingSetWithVeryLongNameViaActionButton(remoteRobot: RemoteRobot) = with(remoteRobot) { val wsName: String = "B".repeat(200) - ideFrameImpl(projectName, fixtureStack) { - createWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName) - clickButton("OK") - Thread.sleep(2000) - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - EMPTY_DATASET_MESSAGE - ) - clickButton("OK") - Thread.sleep(2000) - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } + callCreateWorkingSetFromActionButton(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName, fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + find(messageLoc).findText(EMPTY_DATASET_MESSAGE) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) } /** * Tests to add new working set with one valid mask. */ @Test - @Order(3) - fun testAddWorkingSetWithOneValidMaskViaActionButton(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS1" + fun testAddWorkingSetWithOneValidMaskViaActionButton(remoteRobot: RemoteRobot){ val mask = Pair("$ZOS_USERID.*", "z/OS") - ideFrameImpl(projectName, fixtureStack) { - createWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName, mask) - clickButton("OK") - Thread.sleep(2000) - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } + callCreateWorkingSetFromActionButton(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName1, mask, fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) } /** * Tests to add new working set with several valid z/OS masks, opens masks in explorer. */ @Test - @Order(4) - fun testAddWorkingSetWithValidZOSMasksViaActionButton(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val wsName = "WS2" - val masks: ArrayList> = ArrayList() - //todo allocate dataset with 44 length when 'Allocate Dataset Dialog' implemented + fun testAddWorkingSetWithValidZOSMasksViaActionButton(testInfo: TestInfo, remoteRobot: RemoteRobot){ - validZOSMasks.forEach { - masks.add(Pair(it, "z/OS")) - } - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles", - { it?.requestLine?.contains("zosmf/restfiles/ds?dslevel") ?: false }, - { MockResponse().setBody("{}") } - ) + val masks: ArrayList> = ArrayList() + //todo allocate dataset with 44 length when 'Allocate Dataset Dialog' implemented - ideFrameImpl(projectName, fixtureStack) { - createWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName, masks) - clickButton("OK") - Thread.sleep(2000) - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } - validZOSMasks.forEach { - openWSOpenMaskInExplorer( - wsName, - it.uppercase(), - projectName, - fixtureStack, - remoteRobot - ) - } + validZOSMasks.forEach { + masks.add(Pair(it, ZOS_MASK)) + } + injectListEmptyData(testInfo) + callCreateWorkingSetFromActionButton(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName2, masks, fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + Thread.sleep(3000) + validZOSMasks.forEach { + openWSOpenMaskInExplorer( + wsName2, + it.uppercase(), + fixtureStack, + remoteRobot + ) } + } /** * Tests to add new working set with several valid USS masks, opens masks in explorer. */ @Test - @Order(5) - fun testAddWorkingSetWithValidUSSMasksViaActionButton(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val wsName = "WS3" - val masks: ArrayList> = ArrayList() + fun testAddWorkingSetWithValidUSSMasksViaActionButton(testInfo: TestInfo, remoteRobot: RemoteRobot){ - validUSSMasks.forEach { - masks.add(Pair(it, "USS")) - } - - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles", - { it?.requestLine?.contains("zosmf/restfiles/fs?path") ?: false }, - { MockResponse().setBody("{}") } - ) + val masks: ArrayList> = ArrayList() - ideFrameImpl(projectName, fixtureStack) { - createWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName, masks) - clickButton("OK") - Thread.sleep(2000) - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } - validUSSMasks.forEach { openWSOpenMaskInExplorer(wsName, it, projectName, fixtureStack, remoteRobot) } + validUSSMasks.forEach { + masks.add(Pair(it, USS_MASK)) } + injectListEmptyData(testInfo, false) + callCreateWorkingSetFromActionButton(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName3, masks, fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + compressAndDecompressTree(wsName3, fixtureStack, remoteRobot) + validUSSMasks.forEach { openWSOpenMaskInExplorer(wsName3, it, fixtureStack, remoteRobot) } + } /** * Tests to add new working set with invalid masks, checks that correct messages are returned. */ @Test - @Order(6) fun testAddWorkingSetWithInvalidMasksViaActionButton(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS4" - ideFrameImpl(projectName, fixtureStack) { - createWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName) - maskMessageMap.forEach { - addMask(Pair(it.key, "z/OS")) - if (button("OK").isEnabled()) { - clickButton("OK") - } else { - findText("OK").moveMouse() - } - if (it.key.length < 49) { - findText(it.key.uppercase()).moveMouse() - } else { - findText("${it.key.uppercase().substring(0, 46)}...").moveMouse() - } - Thread.sleep(2000) - find(byXpath("//div[@class='HeavyWeightWindow'][.//div[@class='Header']]")).findText( - it.value - ) - assertFalse(button("OK").isEnabled()) - findText(it.key.uppercase()).click() - clickActionButton(byXpath("//div[contains(@myvisibleactions, 'Down')]//div[@myaction.key='button.text.remove']")) - } - clickButton("Cancel") + + callCreateWorkingSetFromActionButton(fixtureStack, remoteRobot) + maskMessageMap.forEach { + val mask = Pair(it.key, ZOS_MASK) + addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName4, mask, fixtureStack, remoteRobot) + if(isButtonEnableByTextAddWorkingSet(OK_TEXT, fixtureStack, remoteRobot)){ + clickByText(OK_TEXT, fixtureStack,remoteRobot) + } else { + hoverToByTextAddWorkingSet(OK_TEXT, fixtureStack, remoteRobot) } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + if (it.key.length < 49) { + hoverToByTextAddWorkingSet(it.key.uppercase(), fixtureStack, remoteRobot) + } else { + hoverToByTextAddWorkingSet("${it.key.uppercase().substring(0, 46)}...", fixtureStack, remoteRobot) + } + Thread.sleep(1000) + find(helpLoc).findText(it.value) + assertFalse(isButtonEnableByTextAddWorkingSet(OK_TEXT, fixtureStack, remoteRobot)) + clickToByTextAddWorkingSet(it.key.uppercase(), fixtureStack,remoteRobot) + Thread.sleep(3000) + clickActionButtonByXpath(removeEMaskButtonLoc, fixtureStack, remoteRobot) } + clickByText(CANCEL_TEXT, fixtureStack,remoteRobot) + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) } /** * Tests to add working set with the same masks, checks that correct message is returned. */ @Test - @Order(7) fun testAddWorkingSetWithTheSameMasksViaActionButton(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS4" - ideFrameImpl(projectName, fixtureStack) { - createWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName) - addMask(Pair("$ZOS_USERID.*", "z/OS")) - addMask(Pair("$ZOS_USERID.*", "z/OS")) - clickButton("OK") - find( - byXpath("//div[@class='HeavyWeightWindow']"), - Duration.ofSeconds(30) - ).findText(IDENTICAL_MASKS_MESSAGE) - assertFalse(button("OK").isEnabled()) - clickButton("Cancel") + callCreateWorkingSetFromActionButton(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName5, Pair("$ZOS_USERID.*", "z/OS"), fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, wsName5, Pair("$ZOS_USERID.*", "z/OS"), fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack,remoteRobot) + Thread.sleep(2000) + find(messageLoc).findText(IDENTICAL_MASKS_MESSAGE) + assertFalse(isButtonEnableByTextAddWorkingSet(OK_TEXT, fixtureStack, remoteRobot)) + clickByText(CANCEL_TEXT, fixtureStack,remoteRobot) + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + } +} + +/** + * Tests creating working sets and masks via action button without creating connection + */ +@TestMethodOrder(MethodOrderer.OrderAnnotation::class) +@TestInstance(TestInstance.Lifecycle.PER_CLASS) +@ExtendWith(RemoteRobotExtension::class) +class WorkingSetViaActionButtonNoConnectionTest : WorkingSetBase(){ + private var closableFixtureCollector = ClosableFixtureCollector() + private var fixtureStack = mutableListOf() + @BeforeAll + fun setUpAll(remoteRobot: RemoteRobot, testInfo:TestInfo){ + startMockServer() + Thread.sleep(3000) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + } + + @AfterAll + fun tearDownAll(remoteRobot: RemoteRobot){ + mockServer.shutdown() + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + closeIntelligentProject(fixtureStack, remoteRobot) + } + @Test + @Disabled("https://jira.ibagroup.eu/browse/IJMP-977") + fun testAddWorkingSetWithoutConnectionViaActionButton(testInfo: TestInfo, remoteRobot: RemoteRobot) = + with(remoteRobot) { + val wsName = "first ws" + responseDispatcher.injectTestInfo(testInfo) + responseDispatcher.injectTestInfoRestTopology(testInfo) + callCreateWorkingSetFromActionButton(fixtureStack,remoteRobot) + + ideFrameImpl(PROJECT_NAME, fixtureStack) { + try { + if (dialog("Add Working Set Dialog").isShowing) { + Assertions.assertTrue(false) + } + } catch (e: WaitForConditionTimeoutException) { + e.message.shouldContain("Failed to find 'Dialog' by 'dialogTitle Add Working Set Dialog'") + } finally { + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + } + + createConnectionFromActionButton(closableFixtureCollector, fixtureStack) + fillConnectionFilds(fixtureStack=fixtureStack, remoteRobot=remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + callCreateWorkingSetFromActionButton(closableFixtureCollector, fixtureStack) + addWorkingSetDialog(fixtureStack) { + addWorkingSet(wsName, connectionName) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + if (dialog("Add Working Set Dialog").isShowing) { + Assertions.assertTrue(false) + } + clickByText(OK_TEXT, fixtureStack, remoteRobot) + } + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) } - } + } diff --git a/src/uiTest/kotlin/workingset/WorkingSetViaContextMenuTest.kt b/src/uiTest/kotlin/workingset/WorkingSetViaContextMenuTest.kt index d06ceeef4..c3a8c953a 100644 --- a/src/uiTest/kotlin/workingset/WorkingSetViaContextMenuTest.kt +++ b/src/uiTest/kotlin/workingset/WorkingSetViaContextMenuTest.kt @@ -12,21 +12,19 @@ package workingset import auxiliary.* import auxiliary.closable.ClosableFixtureCollector -import auxiliary.components.actionMenuItem import auxiliary.containers.* import com.intellij.remoterobot.RemoteRobot -import com.intellij.remoterobot.fixtures.ActionButtonFixture -import com.intellij.remoterobot.fixtures.ComponentFixture import com.intellij.remoterobot.fixtures.HeavyWeightWindowFixture -import com.intellij.remoterobot.fixtures.JTextFieldFixture import com.intellij.remoterobot.search.locators.Locator import com.intellij.remoterobot.search.locators.byXpath import com.intellij.remoterobot.utils.WaitForConditionTimeoutException import io.kotest.matchers.string.shouldContain -import okhttp3.mockwebserver.MockResponse import org.junit.jupiter.api.* import org.junit.jupiter.api.Assertions.assertFalse import org.junit.jupiter.api.extension.ExtendWith +import workingset.auxiliary.components.dialogs.* +import workingset.auxiliary.components.elements.ButtonElement +import workingset.testutils.* import java.time.Duration /** @@ -35,34 +33,56 @@ import java.time.Duration @TestMethodOrder(MethodOrderer.OrderAnnotation::class) @TestInstance(TestInstance.Lifecycle.PER_CLASS) @ExtendWith(RemoteRobotExtension::class) -class WorkingSetViaContextMenuTest { +class WorkingSetViaContextMenuTest : WorkingSetBase() { private var closableFixtureCollector = ClosableFixtureCollector() private var fixtureStack = mutableListOf() - private var wantToClose = mutableListOf("Add Working Set Dialog", "Edit Working Set Dialog", "Create Mask Dialog") - - private val projectName = "untitled" - private val connectionName = "valid connection" - - - /** + private var wantToClose = mutableListOf(EDIT_WORKING_SET, CREATE_MASK_DIALOG, ADD_WORKING_SET_DIALOG) + + override val connectionName: String = "valid connection" + private val wsNameA: String = "A".repeat(200) + private val newInvalidName: String = "invalid connection" + + private val newWorkingSetName = "new ws name" + private val oldWorkingSetName = "old name" + private val alreadyExistsWorkingSetName = "already exists" + private val alreadyExistsWorkingSetNameRename = "already exists for rename" + private val wsForDelete = "ws for delete" + private val wsForInvalidMask = "ws for invalid mask" + private val wsWithUssAndDataset = "ws with uss and dataset" + private val wsWithAlreadyExistsMask = "ws with already exists mask" + private val wsWithMaskForRename = "ws with mask for rename" + private val wsWithMaskForDelete = "ws with mask for delete" + + /** * Opens the project and Explorer, clears test environment. */ @BeforeAll - fun setUpAll(remoteRobot: RemoteRobot) { + fun setUpAll(remoteRobot: RemoteRobot, testInfo:TestInfo) { + addWorkingSetDialog = AddWorkingSetSubDialog(fixtureStack, remoteRobot) startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + createValidConnectionWithMock(testInfo, connectionName, fixtureStack, closableFixtureCollector, remoteRobot) + + editWorkingSetSubDialog = EditWorkingSetSubDialog(fixtureStack=fixtureStack, remoteRobot=remoteRobot) + createMaskSubDialog = CreateMaskSubDialog(fixtureStack=fixtureStack, remoteRobot=remoteRobot) + renameDatasetMaskDialog = RenameDatasetMaskDialog(fixtureStack=fixtureStack, remoteRobot=remoteRobot) + deletionOfUssPathRoot = DeletionOfUssPathRoot(fixtureStack=fixtureStack, remoteRobot=remoteRobot) + deletionOfDSMask = DeletionOfDSMask(fixtureStack=fixtureStack, remoteRobot=remoteRobot) + + okButton = ButtonElement(OK_TEXT, fixtureStack, remoteRobot) + cancelButton = ButtonElement(CANCEL_TEXT, fixtureStack, remoteRobot) + yesButton = ButtonElement(YES_TEXT, fixtureStack, remoteRobot) + } /** * Closes the project and clears test environment. */ @AfterAll - fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { + fun tearDownAll(remoteRobot: RemoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - close() - } + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + return closeIntelligentProject(fixtureStack, remoteRobot) } /** @@ -74,175 +94,75 @@ class WorkingSetViaContextMenuTest { closableFixtureCollector.closeWantedClosables(wantToClose, remoteRobot) } - /** - * Tests to add new working set without connection, checks that correct message is returned. - */ - @Test - @Order(1) - fun testAddWorkingSetWithoutConnectionViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val wsName = "first ws" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_info", - { it?.requestLine?.contains("zosmf/info") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_resttopology", - { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - - ideFrameImpl(projectName, fixtureStack) { - createWSFromContextMenu(fixtureStack, closableFixtureCollector) - try { - if (dialog("Add Working Set Dialog").isShowing) { - Assertions.assertTrue(false) - } - } catch (e: WaitForConditionTimeoutException) { - e.message.shouldContain("Failed to find 'Dialog' by 'title Add Working Set Dialog'") - } finally { - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } - - createConnectionFromActionButton(closableFixtureCollector, fixtureStack) - addConnectionDialog(fixtureStack) { - addConnection( - connectionName, - "https://${mockServer.hostName}:${mockServer.port}", - ZOS_USERID, - ZOS_PWD, - true - ) - clickButton("OK") - } - closableFixtureCollector.closeOnceIfExists(AddConnectionDialog.name) - - createWSFromContextMenu(fixtureStack, closableFixtureCollector) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName) - clickButton("OK") - Thread.sleep(3000) - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - EMPTY_DATASET_MESSAGE - ) - clickButton("OK") - Thread.sleep(3000) - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } - } - /** * Tests to add new empty working set with very long name, checks that correct message is returned. */ @Test - @Order(2) + @Disabled("https://jira.ibagroup.eu/browse/IJMP-977") fun testAddEmptyWorkingSetWithVeryLongNameViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName: String = "A".repeat(200) - ideFrameImpl(projectName, fixtureStack) { - createWSFromContextMenu(fixtureStack, closableFixtureCollector) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName) - clickButton("OK") - Thread.sleep(3000) - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - EMPTY_DATASET_MESSAGE - ) - clickButton("OK") - Thread.sleep(3000) - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, wsNameA, fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + find(messageLoc).findText(EMPTY_DATASET_MESSAGE) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + } /** * Tests to add new working set with one valid mask. */ @Test - @Order(3) - fun testAddWorkingSetWithOneValidMaskViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS1" - val mask = Pair("$ZOS_USERID.*", "z/OS") - ideFrameImpl(projectName, fixtureStack) { - createWSFromContextMenu(fixtureStack, closableFixtureCollector) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName, mask) - clickButton("OK") - Thread.sleep(3000) - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } + fun testAddWorkingSetWithOneValidMaskViaContextMenu(remoteRobot: RemoteRobot) { + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, WS_NAME_WS_1, singleMask, fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + return deleteWSFromContextMenu(WS_NAME_WS_1, fixtureStack, remoteRobot) } /** * Tests to add new working set with several valid z/OS masks, opens masks in explorer. */ @Test - @Order(4) - fun testAddWorkingSetWithValidZOSMasksViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val wsName = "WS2" - val masks: ArrayList> = ArrayList() - //todo allocate dataset with 44 length when 'Allocate Dataset Dialog' implemented - - validZOSMasks.forEach { - masks.add(Pair(it, "z/OS")) - } - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles", - { it?.requestLine?.contains("zosmf/restfiles/ds?dslevel=") ?: false }, - { MockResponse().setBody("{}") } + fun testAddWorkingSetWithValidZOSMasksViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + //todo allocate dataset with 44 length when 'Allocate Dataset Dialog' implemented + val masks: ArrayList> = ArrayList() + validZOSMasks.forEach {masks.add(Pair(it, ZOS_MASK))} + injectListEmptyData(testInfo) + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, WS_NAME_WS_2, masks, fixtureStack, remoteRobot) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + compressWsIfcDecompressed(WS_NAME_WS_2, validZOSMasks[0],fixtureStack,remoteRobot) + validZOSMasks.forEach { + openWSOpenMaskInExplorer( + WS_NAME_WS_2, + it.uppercase(), + fixtureStack, + remoteRobot ) - - ideFrameImpl(projectName, fixtureStack) { - createWSFromContextMenu(fixtureStack, closableFixtureCollector) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName, masks) - clickButton("OK") - Thread.sleep(3000) - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } - validZOSMasks.forEach { - openWSOpenMaskInExplorer( - wsName, - it.uppercase(), - projectName, - fixtureStack, - remoteRobot - ) - } } + return deleteWSFromContextMenu(WS_NAME_WS_2, fixtureStack, remoteRobot) + } /** * Tests to add new working set with several valid USS masks, opens masks in explorer. */ @Test - @Order(5) - fun testAddWorkingSetWithValidUSSMasksViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val wsName = "WS3" - val masks: ArrayList> = ArrayList() - validUSSMasks.forEach { - masks.add(Pair(it, "USS")) - } - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles", - { it?.requestLine?.contains("zosmf/restfiles/fs?path") ?: false }, - { MockResponse().setBody("{}") } - ) - - ideFrameImpl(projectName, fixtureStack) { - createWSFromContextMenu(fixtureStack, closableFixtureCollector) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName, masks) - clickButton("OK") - Thread.sleep(3000) - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } - validUSSMasks.forEach { openWSOpenMaskInExplorer(wsName, it, projectName, fixtureStack, remoteRobot) } + fun testAddWorkingSetWithValidUSSMasksViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot){ + val masks: ArrayList> = ArrayList() + validUSSMasks.forEach { + masks.add(Pair(it, USS_MASK)) + } + injectListEmptyData(testInfo, false) + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, WS_NAME_WS_3, masks, fixtureStack, remoteRobot) + addWorkingSetDialog.okButton.click() + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + compressAndDecompressTree(WS_NAME_WS_3, fixtureStack, remoteRobot) + validUSSMasks.forEach { openWSOpenMaskInExplorer(WS_NAME_WS_3, it, fixtureStack, remoteRobot) } + deleteWSFromContextMenu(WS_NAME_WS_3, fixtureStack, remoteRobot) } @@ -250,504 +170,455 @@ class WorkingSetViaContextMenuTest { * Tests to add new working set with invalid masks, checks that correct messages are returned. */ @Test - @Order(6) fun testAddWorkingSetWithInvalidMasksViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS4" - ideFrameImpl(projectName, fixtureStack) { - createWSFromContextMenu(fixtureStack, closableFixtureCollector) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName) - maskMessageMap.forEach { - addMask(Pair(it.key, "z/OS")) - if (button("OK").isEnabled()) { - clickButton("OK") - } else { - findText("OK").moveMouse() - } - if (it.key.length < 49) { - findText(it.key.uppercase()).moveMouse() - } else { - findText("${it.key.uppercase().substring(0, 46)}...").moveMouse() - } - Thread.sleep(3000) - find(byXpath("//div[@class='HeavyWeightWindow'][.//div[@class='Header']]")).findText( - it.value - ) - assertFalse(button("OK").isEnabled()) - findText(it.key.uppercase()).click() - clickActionButton(byXpath("//div[contains(@myvisibleactions, 'Down')]//div[@myaction.key='button.text.remove']")) - } - clickButton("Cancel") + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + maskMessageMap.forEach { + val mask = Pair(it.key, ZOS_MASK) + addWorkingSetDialog.fillAddWorkingSet(connectionName, WS_NAME_WS_4, mask, fixtureStack, remoteRobot) + if(addWorkingSetDialog.okButton.isEnabled()){ + addWorkingSetDialog.okButton.click() + } else { + hoverToByTextAddWorkingSet(OK_TEXT, fixtureStack, remoteRobot) } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + if (it.key.length < 49) { + hoverToByTextAddWorkingSet(it.key.uppercase(), fixtureStack, remoteRobot) + } else { + hoverToByTextAddWorkingSet("${it.key.uppercase().substring(0, 46)}...", fixtureStack, remoteRobot) + } + Thread.sleep(1000) + find(helpLoc).findText(it.value) + assertFalse(addWorkingSetDialog.okButton.isEnabled()) + clickToByTextAddWorkingSet(it.key.uppercase(), fixtureStack,remoteRobot) + Thread.sleep(3000) + clickActionButtonByXpath(removeEMaskButtonLoc, fixtureStack, remoteRobot) } + cancelButton.click() + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) } /** * Tests to add working set with the same masks, checks that correct message is returned. */ @Test - @Order(7) fun testAddWorkingSetWithTheSameMasksViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS4" - ideFrameImpl(projectName, fixtureStack) { - createWSFromContextMenu(fixtureStack, closableFixtureCollector) - addWorkingSetDialog(fixtureStack) { - addWorkingSet(wsName, connectionName) - addMask(Pair("$ZOS_USERID.*", "z/OS")) - addMask(Pair("$ZOS_USERID.*", "z/OS")) - clickButton("OK") - find( - byXpath("//div[@class='HeavyWeightWindow']"), - Duration.ofSeconds(30) - ).findText(IDENTICAL_MASKS_MESSAGE) - assertFalse(button("OK").isEnabled()) - clickButton("Cancel") - } - closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) - } + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, WS_NAME_WS_5, singleMask, fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, WS_NAME_WS_5, singleMask, fixtureStack, remoteRobot) + addWorkingSetDialog.okButton.click() + Thread.sleep(2000) + find(messageLoc).findText(IDENTICAL_MASKS_MESSAGE) + assertFalse(isButtonEnableByTextAddWorkingSet(OK_TEXT, fixtureStack, remoteRobot)) + addWorkingSetDialog.cancelButton.click() + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) } /** * Tests to edit working set by adding one mask, checks that ws is refreshed in explorer, opens new mask. */ @Test - @Order(8) - fun testEditWorkingSetAddOneMaskViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS1" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles", - { it?.requestLine?.contains("zosmf/restfiles/") ?: false }, - { MockResponse().setBody("{}") } - ) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - closeMaskInExplorer("$ZOS_USERID.*".uppercase(), projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - editWSFromContextMenu(wsName, fixtureStack, closableFixtureCollector) - editWorkingSetDialog(fixtureStack) { - addMask(Pair("/u/$ZOS_USERID", "USS")) - clickButton("OK") - Thread.sleep(3000) - } - closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) - } - openMaskInExplorer("/u/$ZOS_USERID", "", projectName, fixtureStack, remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) + fun testEditWorkingSetAddOneMaskViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + injectListEmptyData(testInfo) + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + addWorkingSetDialog.fillAddWorkingSet(connectionName, WS_NAME_WS_6, singleMask, fixtureStack, remoteRobot) + okButton.click() + callEditWSFromContextMenu(WS_NAME_WS_6, fixtureStack, remoteRobot) + fillEditWorkingSet(connectionName,WS_NAME_WS_6,singleUssMask,fixtureStack, remoteRobot) + okButton.click() + closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) + decompressWsIfCompressed(WS_NAME_WS_6, ussMask, fixtureStack, remoteRobot) + openMaskInExplorer(ussMask,"", fixtureStack, remoteRobot) + openOrCloseWorkingSetInExplorer(WS_NAME_WS_6, fixtureStack, remoteRobot) + return deleteWSFromContextMenu(WS_NAME_WS_6, fixtureStack, remoteRobot) } /** * Tests to edit working set by deleting several masks, checks that ws is refreshed in explorer and masks were deleted. */ @Test - @Order(9) - fun testEditWorkingSetDeleteMasksViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS2" - val masks = listOf("$ZOS_USERID.*".uppercase(), "Q.*", ZOS_USERID.uppercase()) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - editWSFromContextMenu(wsName, fixtureStack, closableFixtureCollector) - editWorkingSetDialog(fixtureStack) { - deleteMasks(masks) - clickButton("OK") - Thread.sleep(5000) - } - closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) - } - masks.forEach { checkItemWasDeletedWSRefreshed(it.uppercase(), projectName, fixtureStack, remoteRobot) } - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) + fun testEditWorkingSetDeleteMasksViaContextMenu(remoteRobot: RemoteRobot) { + val masks = listOf(zosUserDatasetMask, "Q.*", ZOS_USERID.uppercase()) + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + masks.forEach {addWorkingSetDialog.fillAddWorkingSet(connectionName, WS_NAME_WS_7, Pair(it, ZOS_MASK), fixtureStack, remoteRobot)} + okButton.click() + callEditWSFromContextMenu(WS_NAME_WS_7, fixtureStack, remoteRobot) + deleteInEditWorkingSet(masks, fixtureStack, remoteRobot) + okButton.click() + + masks.forEach { checkItemWasDeletedWSRefreshed(it.uppercase(), fixtureStack, remoteRobot) } + openOrCloseWorkingSetInExplorer(WS_NAME_WS_7, fixtureStack, remoteRobot) + deleteWSFromContextMenu(WS_NAME_WS_7, fixtureStack, remoteRobot) + return closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) } /** * Tests to edit working set by deleting all masks, checks that ws is refreshed in explorer and masks were deleted. */ @Test - @Order(10) + @Disabled("https://jira.ibagroup.eu/browse/IJMP-977") fun testEditWorkingSetDeleteAllMasksViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS2" - val deletedMasks = listOf("$ZOS_USERID.**", "$ZOS_USERID.@#%", "$ZOS_USERID.@#%.*", "WWW.*", maskWithLength44) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - editWSFromContextMenu(wsName, fixtureStack, closableFixtureCollector) + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + validZOSMasks.forEach {addWorkingSetDialog.fillAddWorkingSet(connectionName, WS_NAME_WS_8, Pair(it, ZOS_MASK), fixtureStack, remoteRobot)} + addWorkingSetDialog.okButton.click() + openOrCloseWorkingSetInExplorer(WS_NAME_WS_8, fixtureStack, remoteRobot) + callEditWSFromContextMenu(WS_NAME_WS_8, fixtureStack, remoteRobot) + deleteInEditWorkingSet(validZOSMasks, fixtureStack, remoteRobot) + okButton.click() + ideFrameImpl(PROJECT_NAME, fixtureStack) { editWorkingSetDialog(fixtureStack) { - deleteAllMasks() - clickButton("OK") - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - EMPTY_DATASET_MESSAGE - ) - clickButton("OK") + find(messageLoc).findText(EMPTY_DATASET_MESSAGE) + okButton.click() Thread.sleep(5000) } closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) } - deletedMasks.forEach { checkItemWasDeletedWSRefreshed(it.uppercase(), projectName, fixtureStack, remoteRobot) } - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) + validZOSMasks.forEach { checkItemWasDeletedWSRefreshed(it.uppercase(), fixtureStack, remoteRobot) } + openOrCloseWorkingSetInExplorer(WS_NAME_WS_8, fixtureStack, remoteRobot) } /** * Tests to edit working set by changing connection to invalid, checks that correct message is returned. */ @Test - @Order(11) - fun testEditWorkingSetChangeConnectionToInvalidViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val newConnectionName = "invalid connection" - val testPort = "10443" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_info", - { it?.requestLine?.contains("zosmf/info") ?: false }, - { MockResponse().setBody("Invalid URL port: \"${testPort}1\"") } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_second", - { it?.requestLine?.contains("zosmf/restfiles/") ?: false }, - { MockResponse().setBody("{}") } - ) - createConnection( - projectName, - fixtureStack, - closableFixtureCollector, - newConnectionName, - false, - remoteRobot, - "https://${mockServer.hostName}:$testPort" - ) - val wsName = "WS1" - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - editWSFromContextMenu(wsName, fixtureStack, closableFixtureCollector) - editWorkingSetDialog(fixtureStack) { - changeConnection(newConnectionName) - clickButton("OK") - Thread.sleep(5000) - } - closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) - } - findAll(byXpath("//div[@class='MyComponent'][.//div[@accessiblename='Invalid URL port: \"104431\"' and @class='JEditorPane']]")).forEach { - it.click() - findAll( - byXpath("//div[@class='ActionButton' and @myicon= 'close.svg']") - ).first().click() - } - openMaskInExplorer( - "$ZOS_USERID.*".uppercase(), - "Invalid URL port: \"104431\"", - projectName, - fixtureStack, - remoteRobot - ) - } + fun testEditWorkingSetChangeConnectionToInvalidViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + injectInvalidUrlPortInfo(testInfo,PORT_104431_AND_1) + injectEmptyZosmfRestfilesPath(testInfo) + createConnection(fixtureStack, closableFixtureCollector, newInvalidName, false, remoteRobot, "https://${mockServer.hostName}:$PORT_104431") + + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + validZOSMasks.forEach {addWorkingSetDialog.fillAddWorkingSet(connectionName, WS_NAME_WS_9, Pair(it, ZOS_MASK), fixtureStack, remoteRobot)} + addWorkingSetDialog.okButton.click() + + openOrCloseWorkingSetInExplorer(WS_NAME_WS_9, fixtureStack, remoteRobot) + callEditWSFromContextMenu(WS_NAME_WS_9, fixtureStack, remoteRobot) + setInComboBox(newInvalidName,fixtureStack,remoteRobot) + okButton.click() + closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) + + openMaskInExplorer( + zosUserDatasetMask, INVALID_URL_PORT.format(PORT_104431_AND_1), fixtureStack, remoteRobot + ) + return deleteWSFromContextMenu(WS_NAME_WS_9, fixtureStack, remoteRobot) + } /** * Tests to edit working set by changing connection from invalid to valid, checks that ws is refreshed in explorer and error message disappeared. */ @Test - @Order(12) - fun testEditWorkingSetChangeConnectionToValidViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = - with(remoteRobot) { - val newConnectionName = "new $connectionName" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_info", - { it?.requestLine?.contains("zosmf/info") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_resttopology", - { it?.requestLine?.contains("zosmf/resttopology/systems") ?: false }, - { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } - ) - createConnection( - projectName, - fixtureStack, - closableFixtureCollector, - newConnectionName, - true, - remoteRobot, - "https://${mockServer.hostName}:${mockServer.port}" - ) - val wsName = "WS1" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles", - { it?.requestLine?.contains("zosmf/restfiles") ?: false }, - { MockResponse().setBody("{}") } - ) - ideFrameImpl(projectName, fixtureStack) { - editWSFromContextMenu(wsName, fixtureStack, closableFixtureCollector) - editWorkingSetDialog(fixtureStack) { - changeConnection(newConnectionName) - clickButton("OK") - Thread.sleep(5000) - } - closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) - } - checkItemWasDeletedWSRefreshed("Invalid URL port: \"104431\"", projectName, fixtureStack, remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - } + fun testEditWorkingSetChangeConnectionToValidViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + injectInvalidUrlPortInfo(testInfo,PORT_104431_AND_1) + injectTestInfoRestTopology(testInfo) + createConnection(fixtureStack, closableFixtureCollector, WS_NAME_WS_10, false, remoteRobot, "https://${mockServer.hostName}:$PORT_104431") + + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + validZOSMasks.forEach {addWorkingSetDialog.fillAddWorkingSet(WS_NAME_WS_10, WS_NAME_WS_10, Pair(it, ZOS_MASK), fixtureStack, remoteRobot)} + addWorkingSetDialog.okButton.click() + + injectEmptyZosmfRestfilesPath(testInfo) + callEditWSFromContextMenu(WS_NAME_WS_10, fixtureStack, remoteRobot) + changeConnectionInEditWorkingSet(connectionName, fixtureStack, remoteRobot) + okButton.click() + + openOrCloseWorkingSetInExplorer(WS_NAME_WS_10, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed(INVALID_URL_PORT.format(PORT_104431_AND_1), fixtureStack, remoteRobot) + closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) + return deleteWSFromContextMenu(WS_NAME_WS_10, fixtureStack, remoteRobot) + } /** * Tests to edit working set by renaming it, checks that ws is refreshed in explorer. */ @Test - @Order(13) fun testEditWorkingSetRenameViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val newWorkingSetName = "new ws name" - val oldWorkingSetName = "WS1" - val alreadyExistsWorkingSetName = "WS2" - openOrCloseWorkingSetInExplorer(oldWorkingSetName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { - editWSFromContextMenu(oldWorkingSetName, fixtureStack, closableFixtureCollector) - editWorkingSetDialog(fixtureStack) { - renameWorkingSet(alreadyExistsWorkingSetName) - clickButton("OK") - val message = find( - byXpath("//div[@class='HeavyWeightWindow']"), - Duration.ofSeconds(30) - ).findAllText() - (message[0].text + message[1].text).shouldContain("You must provide unique working set name. Working Set $alreadyExistsWorkingSetName already exists.") - renameWorkingSet(newWorkingSetName) - clickButton("OK") - Thread.sleep(5000) - } - closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) - } - checkItemWasDeletedWSRefreshed(oldWorkingSetName, projectName, fixtureStack, remoteRobot) - openOrCloseWorkingSetInExplorer(newWorkingSetName, projectName, fixtureStack, remoteRobot) + createWsWithConnectionFromAction(connectionName, oldWorkingSetName, singleMask, fixtureStack, remoteRobot) + + createWsWithConnectionFromAction( + connectionName, alreadyExistsWorkingSetName, singleMask, fixtureStack, remoteRobot) + + callEditWSFromContextMenu(oldWorkingSetName, fixtureStack, remoteRobot) + + editWorkingSetSubDialog.renameWorkingSet(alreadyExistsWorkingSetName) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + + val message = find(messageLoc,Duration.ofSeconds(30)).findAllText() + (message[0].text + message[1].text).shouldContain(UNIQUE_WORKING_SET_NAME.format(alreadyExistsWorkingSetName)) + + editWorkingSetSubDialog.renameWorkingSet(newWorkingSetName) + clickByText(OK_TEXT, fixtureStack, remoteRobot) + + closableFixtureCollector.closeOnceIfExists(EditWorkingSetDialog.name) + + checkItemWasDeletedWSRefreshed(oldWorkingSetName, fixtureStack, remoteRobot) + openOrCloseWorkingSetInExplorer(newWorkingSetName, fixtureStack, remoteRobot) + deleteWSFromContextMenu(alreadyExistsWorkingSetName, fixtureStack, remoteRobot) + deleteWSFromContextMenu(newWorkingSetName, fixtureStack, remoteRobot) } /** * Tests to delete working set, checks that explorer info is refreshed. */ @Test - @Order(14) - fun testDeleteWorkingSetViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "WS2" - ideFrameImpl(projectName, fixtureStack) { - deleteWSFromContextMenu(wsName) - clickButton("Yes") - } - checkItemWasDeletedWSRefreshed(wsName, projectName, fixtureStack, remoteRobot) + fun testDeleteWorkingSetViaContextMenu(remoteRobot: RemoteRobot) { + createWsWithConnectionFromAction(connectionName, wsForDelete, fixtureStack, closableFixtureCollector, remoteRobot) + + deleteWSFromContextMenu(wsForDelete, fixtureStack,remoteRobot) + + return checkItemWasDeletedWSRefreshed(wsForDelete, fixtureStack, remoteRobot) } /** * Tests to create invalid masks, checks that correct messages are returned. */ @Test - @Order(15) fun testCreateInvalidMasksViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "first ws" - ideFrameImpl(projectName, fixtureStack) { - createMask(wsName, fixtureStack, closableFixtureCollector) - createMaskDialog(fixtureStack) { - maskMessageMap.forEach { - createMask(Pair(it.key, "z/OS")) - Thread.sleep(3000) - if (button("OK").isEnabled()) { - clickButton("OK") - } - find(byXpath("//div[@class='HeavyWeightWindow']")).findText( - it.value - ) - assertFalse(button("OK").isEnabled()) - } - clickButton("Cancel") + createWsWithConnectionFromAction(connectionName, wsForInvalidMask, fixtureStack, closableFixtureCollector, remoteRobot) + + callCreateMask(wsForInvalidMask, fixtureStack,remoteRobot) + maskMessageMap.forEach { + createMaskSubDialog.setMask(Pair(it.key, ZOS_MASK)) + Thread.sleep(3000) + if (okButton.isEnabled()) { + okButton.click() } - closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + find(messageLoc).findText( + it.value + ) + assertFalse(okButton.isEnabled()) } + cancelButton.click() + closableFixtureCollector.closeOnceIfExists(createMaskSubDialog.dialogTitle) + deleteWSFromContextMenu(wsForInvalidMask, fixtureStack, remoteRobot) } /** * Tests to create valid USS and z/OS masks from context menu. */ @Test - @Order(16) - fun testCreateValidMasksViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "first ws" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles", - { it?.requestLine?.contains("zosmf/restfiles/") ?: false }, - { MockResponse().setBody("{}") } - ) + fun testCreateValidMasksViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + createWsWithConnectionFromAction(connectionName, wsWithUssAndDataset, fixtureStack, closableFixtureCollector, remoteRobot) + + injectEmptyZosmfRestfilesPath(testInfo) + validZOSMasks.forEach { - createMaskFromContextMenu(wsName, it, "z/OS", remoteRobot) - openWSOpenMaskInExplorer(wsName, it.uppercase(), projectName, fixtureStack, remoteRobot) + createMask(wsWithUssAndDataset, it, fixtureStack,closableFixtureCollector, ZOS_MASK, remoteRobot) + openWSOpenMaskInExplorer(wsWithUssAndDataset, it.uppercase(), fixtureStack, remoteRobot) + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) } validUSSMasks.forEach { - createMaskFromContextMenu(wsName, it, "USS", remoteRobot) + createMask(wsWithUssAndDataset, it, fixtureStack,closableFixtureCollector, USS_MASK, remoteRobot) + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) } + return deleteWSFromContextMenu(wsWithUssAndDataset, fixtureStack, remoteRobot) } /** * Tests to create already exists mask in working set, checks that correct message is returned. */ @Test - @Order(17) fun testCreateAlreadyExistsMaskViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "first ws" - val maskName = "$ZOS_USERID.*" - ideFrameImpl(projectName, fixtureStack) { - createMask(wsName, fixtureStack, closableFixtureCollector) - createMaskDialog(fixtureStack) { - createMask(Pair(maskName, "z/OS")) - clickButton("OK") - assertFalse(button("OK").isEnabled()) - val message = find( - byXpath("//div[@class='HeavyWeightWindow']"), - Duration.ofSeconds(30) - ).findAllText() - (message[0].text + message[1].text).shouldContain("You must provide unique mask in working set. Working Set \"$wsName\" already has mask - $maskName") - clickButton("Cancel") - } - closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) - } + createWsWithConnectionFromAction(connectionName, wsWithAlreadyExistsMask, fixtureStack, closableFixtureCollector, remoteRobot) + createMask(wsWithAlreadyExistsMask, zosUserDatasetMask, fixtureStack,closableFixtureCollector, ZOS_MASK, remoteRobot) + + callCreateMask(wsWithAlreadyExistsMask, fixtureStack,remoteRobot) + createMaskSubDialog.setMask(singleMask) + okButton.click() + assertFalse(okButton.isEnabled()) + + val message = find(messageLoc,Duration.ofSeconds(30)).findAllText() + (message[0].text + message[1].text).shouldContain(UNIQUE_MASK.format(wsWithAlreadyExistsMask, zosUserDatasetMask)) + cancelButton.click() + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + deleteWSFromContextMenu(wsWithAlreadyExistsMask, fixtureStack, remoteRobot) + } /** * Tests to rename mask, checks that info is refreshed in explorer. */ @Test - @Order(18) - fun testRenameMasksViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "new ws name" - responseDispatcher.injectEndpoint( - "${testInfo.displayName}_restfiles", - { it?.requestLine?.contains("zosmf/restfiles/") ?: false }, - { MockResponse().setBody("{}") } - ) - renameMaskFromContextMenu( - wsName, - "$ZOS_USERID.*".uppercase(), - "$ZOS_USERID.**", - true, - "Rename Dataset Mask", - remoteRobot - ) - renameMaskFromContextMenu(wsName, "/u/$ZOS_USERID", "/etc/ssh", true, "Rename USS Mask", remoteRobot) - openWSOpenMaskInExplorer(wsName, "$ZOS_USERID.**".uppercase(), projectName, fixtureStack, remoteRobot) - openWSOpenMaskInExplorer(wsName, "/etc/ssh", projectName, fixtureStack, remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - checkItemWasDeletedWSRefreshed("$ZOS_USERID.*".uppercase(), projectName, fixtureStack, remoteRobot) - checkItemWasDeletedWSRefreshed("/u/$ZOS_USERID", projectName, fixtureStack, remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) + fun testRenameMasksViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) { + createWsWithConnectionFromAction(connectionName, wsWithMaskForRename, fixtureStack, closableFixtureCollector, remoteRobot) + + createMask(wsWithMaskForRename, zosUserDatasetMask, fixtureStack,closableFixtureCollector, ZOS_MASK, remoteRobot) + createMask(wsWithMaskForRename, ussMask, fixtureStack,closableFixtureCollector, USS_MASK, remoteRobot) + + injectEmptyZosmfRestfilesPath(testInfo) + compressAndDecompressTree(wsWithMaskForRename, fixtureStack, remoteRobot) + + callEditWSFromContextMenu(zosUserDatasetMask, fixtureStack, remoteRobot) + renameDatasetMaskDialog.renameMaskFromContextMenu(zosUserDatasetMaskDoubleStar, remoteRobot) + okButton.click() + + callEditWSFromContextMenu(ussMask, fixtureStack, remoteRobot) + renameDatasetMaskDialog.renameMaskFromContextMenu(defaultNewUssMask, remoteRobot) + okButton.click() + + refreshWorkSpace(wsWithMaskForRename, fixtureStack,remoteRobot) + + compressAndDecompressTree(zosUserDatasetMaskDoubleStar, fixtureStack, remoteRobot) + compressAndDecompressTree(defaultNewUssMask, fixtureStack, remoteRobot) + + checkItemWasDeletedWSRefreshed(zosUserDatasetMask, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed(ussMask, fixtureStack, remoteRobot) + + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + return deleteWSFromContextMenu(wsWithMaskForRename, fixtureStack, remoteRobot) } /** * Tests to rename mask to already exists, checks that correct message is returned. */ @Test - @Order(19) fun testRenameMaskToAlreadyExistsViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "first ws" - renameMaskFromContextMenu( - wsName, - "$ZOS_USERID.*".uppercase(), - "$ZOS_USERID.**", - false, - "Rename Dataset Mask", - remoteRobot - ) - renameMaskFromContextMenu(wsName, "/u", "/etc/ssh", false, "Rename USS Mask", remoteRobot) + // before test preparation + createWsWithConnectionFromAction(connectionName, alreadyExistsWorkingSetNameRename, fixtureStack, closableFixtureCollector, remoteRobot) + + createMask(alreadyExistsWorkingSetNameRename, zosUserDatasetMask, fixtureStack,closableFixtureCollector, ZOS_MASK, remoteRobot) + createMask(alreadyExistsWorkingSetNameRename, zosUserDatasetMaskDoubleStar, fixtureStack,closableFixtureCollector, ZOS_MASK, remoteRobot) + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + createMask(alreadyExistsWorkingSetNameRename, ussMask, fixtureStack,closableFixtureCollector, USS_MASK, remoteRobot) + createMask(alreadyExistsWorkingSetNameRename, defaultNewUssMask, fixtureStack,closableFixtureCollector, USS_MASK, remoteRobot) + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + + compressAndDecompressTree(alreadyExistsWorkingSetNameRename, fixtureStack, remoteRobot) + + //test + callEditWSFromContextMenu(zosUserDatasetMask, fixtureStack, remoteRobot) + renameDatasetMaskDialog.renameMaskFromContextMenu(zosUserDatasetMaskDoubleStar, remoteRobot) + okButton.click() + + assertFalse(okButton.isEnabled()) + val message = find(messageLoc,Duration.ofSeconds(30)).findAllText() + (message[0].text + message[1].text).shouldContain(UNIQUE_MASK.format(alreadyExistsWorkingSetNameRename, zosUserDatasetMask)) + cancelButton.click() + + callEditWSFromContextMenu(ussMask, fixtureStack, remoteRobot) + renameDatasetMaskDialog.renameMaskFromContextMenu(defaultNewUssMask, remoteRobot) + okButton.click() + + assertFalse(okButton.isEnabled()) + val messageUss = find(messageLoc,Duration.ofSeconds(30)).findAllText() + (messageUss[0].text + messageUss[1].text).shouldContain(UNIQUE_MASK.format(alreadyExistsWorkingSetNameRename, defaultNewUssMask)) + cancelButton.click() + + deleteWSFromContextMenu(alreadyExistsWorkingSetNameRename, fixtureStack, remoteRobot) + } /** * Tests to delete masks, checks that ws is refreshed in explorer and masks were deleted. */ @Test - @Order(20) - fun testDeleteMaskViaContextMenu(remoteRobot: RemoteRobot) = with(remoteRobot) { - val wsName = "new ws name" - deleteMaskFromContextMenu(wsName, "$ZOS_USERID.**".uppercase(), true, remoteRobot) - deleteMaskFromContextMenu(wsName, "/etc/ssh", false, remoteRobot) + fun testDeleteMaskViaContextMenu(remoteRobot: RemoteRobot) { + // before test preparation + createWsWithConnectionFromAction(connectionName, wsWithMaskForDelete, fixtureStack, closableFixtureCollector, remoteRobot) + + createMask(wsWithMaskForDelete, zosUserDatasetMask, fixtureStack,closableFixtureCollector, ZOS_MASK, remoteRobot) + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + createMask(wsWithMaskForDelete, ussMask, fixtureStack,closableFixtureCollector, USS_MASK, remoteRobot) + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + + compressAndDecompressTree(wsWithMaskForDelete, fixtureStack, remoteRobot) + + deletionOfUssPathRoot.dialogTitle = deletionOfUssPathRoot.dialogTitle.format(ussMask) + deletionOfDSMask.dialogTitle = deletionOfDSMask.dialogTitle.format(zosUserDatasetMask) + //test + + deleteWSFromContextMenu(ussMask, fixtureStack,remoteRobot) + deleteWSFromContextMenu(zosUserDatasetMask, fixtureStack,remoteRobot) + closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) + return deleteWSFromContextMenu(wsWithMaskForDelete, fixtureStack, remoteRobot) } +} + + +@TestMethodOrder(MethodOrderer.OrderAnnotation::class) +@TestInstance(TestInstance.Lifecycle.PER_CLASS) +@ExtendWith(RemoteRobotExtension::class) +class WorkingSetViaContextMenuNoConnectionTest : WorkingSetBase(){ + private var closableFixtureCollector = ClosableFixtureCollector() + private var fixtureStack = mutableListOf() + private var wantToClose = mutableListOf("Add Working Set Dialog", "Edit Working Set Dialog", "Create Mask Dialog") +// private var addWorkingSetDialog = AddWorkingSetSubDialog() + override val wsName = "first ws" /** - * Deletes mask from working set via context menu. Checks that info is refreshed in explorer. + * Opens the project and Explorer, clears test environment. */ - private fun deleteMaskFromContextMenu( - wsName: String, - maskName: String, - isZOSMaskType: Boolean, - remoteRobot: RemoteRobot - ) = with(remoteRobot) { - val dialogTitle = if (isZOSMaskType) { - "Deletion of DS Mask $maskName" - } else { - "Deletion of Uss Path Root $maskName" - } - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(wsName).doubleClick() - Thread.sleep(3000) - find(viewTree).findText(maskName).rightClick() - } - actionMenuItem(remoteRobot, "Delete").click() - dialog(dialogTitle) { - clickButton("Yes") - } - checkItemWasDeletedWSRefreshed(maskName, projectName, fixtureStack, remoteRobot) - explorer { - find(viewTree).findText(wsName).doubleClick() - } - } + @BeforeAll + fun setUpAll(remoteRobot: RemoteRobot) { + addWorkingSetDialog = AddWorkingSetSubDialog(fixtureStack, remoteRobot) + startMockServer() + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + } + /** + * Closes the project and clears test environment. + */ + @AfterAll + fun tearDownAll(remoteRobot: RemoteRobot) { + mockServer.shutdown() + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + return closeIntelligentProject(fixtureStack, remoteRobot) } /** - * Creates mask in working set via context menu. + * Closes all unclosed closable fixtures that we want to close. */ - private fun createMaskFromContextMenu( - wsName: String, - maskName: String, - maskType: String, - remoteRobot: RemoteRobot - ) = - with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - createMask(wsName, fixtureStack, closableFixtureCollector) - createMaskDialog(fixtureStack) { - createMask(Pair(maskName, maskType)) - Thread.sleep(3000) - clickButton("OK") - } - closableFixtureCollector.closeOnceIfExists(CreateMaskDialog.name) - } - } + @AfterEach + fun tearDown(remoteRobot: RemoteRobot) { + responseDispatcher.removeAllEndpoints() + closableFixtureCollector.closeWantedClosables(wantToClose, remoteRobot) + } /** - * Renames mask in working set via context menu. + * Tests to add new working set without connection, checks that correct message is returned. */ - private fun renameMaskFromContextMenu( - wsName: String, - oldMaskName: String, - newMaskName: String, - isNewMaskUnique: Boolean, - dialogTitle: String, - remoteRobot: RemoteRobot - ) = + @Test + @Disabled("https://jira.ibagroup.eu/browse/IJMP-977") + fun testAddWorkingSetWithoutConnectionViaContextMenu(testInfo: TestInfo, remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { - explorer { - fileExplorer.click() - find(viewTree).findText(wsName).doubleClick() - Thread.sleep(3000) - find(viewTree).findText(oldMaskName).rightClick() - } - actionMenuItem(remoteRobot, "Edit").click() - dialog(dialogTitle) { - find(byXpath("//div[@class='JBTextField']")).text = newMaskName + injectTestInfo(testInfo) + injectTestInfoRestTopology(testInfo) + + callCreateWSFromContextMenu(fixtureStack, remoteRobot) + try { + if (addWorkingSetDialog.isShown()) { + Assertions.assertTrue(false) } - clickButton("OK") - if (!isNewMaskUnique) { - assertFalse(button("OK").isEnabled()) - val message = find( - byXpath("//div[@class='HeavyWeightWindow']"), - Duration.ofSeconds(30) - ).findAllText() - (message[0].text + message[1].text).shouldContain("You must provide unique mask in working set. Working Set \"$wsName\" already has mask - $newMaskName") - clickButton("Cancel") + } catch (e: WaitForConditionTimeoutException) { + e.message.shouldContain("Failed to find 'Dialog' by 'Title Add Working Set Dialog'") + } finally { + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) + } + + ideFrameImpl(PROJECT_NAME, fixtureStack) { + createConnectionFromActionButton(closableFixtureCollector, fixtureStack) + addConnectionDialog(fixtureStack) { + addConnection( + connectionName, + "https://${mockServer.hostName}:${mockServer.port}", + ZOS_USERID, + ZOS_PWD, + true + ) + clickButton("OK") } - explorer { - find(viewTree).findText(wsName).doubleClick() + closableFixtureCollector.closeOnceIfExists(AddConnectionDialog.name) + + createWSFromContextMenu(fixtureStack, closableFixtureCollector) + addWorkingSetDialog(fixtureStack) { + addWorkingSet(wsName, connectionName) + clickButton("OK") + Thread.sleep(3000) + find(byXpath("//div[@class='HeavyWeightWindow']")).findText( + EMPTY_DATASET_MESSAGE + ) + clickButton("OK") + Thread.sleep(3000) } + closableFixtureCollector.closeOnceIfExists(AddWorkingSetDialog.name) } } + } diff --git a/src/uiTest/kotlin/workingset/WorkingSetViaSettingsTest.kt b/src/uiTest/kotlin/workingset/WorkingSetViaSettingsTest.kt index 6b761dee4..c307ca82c 100644 --- a/src/uiTest/kotlin/workingset/WorkingSetViaSettingsTest.kt +++ b/src/uiTest/kotlin/workingset/WorkingSetViaSettingsTest.kt @@ -49,7 +49,7 @@ class WorkingSetViaSettingsTest { @BeforeAll fun setUpAll(remoteRobot: RemoteRobot) { startMockServer() - setUpTestEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) + setUpTestEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) } /** @@ -58,8 +58,8 @@ class WorkingSetViaSettingsTest { @AfterAll fun tearDownAll(remoteRobot: RemoteRobot) = with(remoteRobot) { mockServer.shutdown() - clearEnvironment(projectName, fixtureStack, closableFixtureCollector, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + clearEnvironment(fixtureStack, closableFixtureCollector, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { close() } } @@ -79,7 +79,7 @@ class WorkingSetViaSettingsTest { @Test @Order(1) fun testAddWorkingSetWithoutConnectionViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -121,7 +121,6 @@ class WorkingSetViaSettingsTest { { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) createConnection( - projectName, fixtureStack, closableFixtureCollector, connectionName, @@ -130,7 +129,7 @@ class WorkingSetViaSettingsTest { "https://${mockServer.hostName}:${mockServer.port}" ) val wsName: String = "A".repeat(200) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -164,7 +163,7 @@ class WorkingSetViaSettingsTest { fun testAddWorkingSetWithOneValidMaskViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val wsName = "WS1" val mask = Pair("$ZOS_USERID.*", "z/OS") - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -206,7 +205,7 @@ class WorkingSetViaSettingsTest { { MockResponse().setBody("{}") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -229,7 +228,6 @@ class WorkingSetViaSettingsTest { openWSOpenMaskInExplorer( wsName, it.uppercase(), - projectName, fixtureStack, remoteRobot ) @@ -255,7 +253,7 @@ class WorkingSetViaSettingsTest { { MockResponse().setBody("{}") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -274,7 +272,7 @@ class WorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - validUSSMasks.forEach { openWSOpenMaskInExplorer(wsName, it, projectName, fixtureStack, remoteRobot) } + validUSSMasks.forEach { openWSOpenMaskInExplorer(wsName, it, fixtureStack, remoteRobot) } } @@ -285,7 +283,7 @@ class WorkingSetViaSettingsTest { @Order(6) fun testAddWorkingSetWithInvalidMasksViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val wsName = "WS4" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -332,7 +330,7 @@ class WorkingSetViaSettingsTest { @Order(7) fun testAddWorkingSetWithTheSameMasksViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val wsName = "WS4" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -372,9 +370,9 @@ class WorkingSetViaSettingsTest { { it?.requestLine?.contains("zosmf/restfiles/") ?: false }, { MockResponse().setBody("{}") } ) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - closeMaskInExplorer("$ZOS_USERID.*".uppercase(), projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) + closeMaskInExplorer("$ZOS_USERID.*".uppercase(), fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -393,8 +391,8 @@ class WorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - openMaskInExplorer("/u/$ZOS_USERID", "", projectName, fixtureStack, remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) + openMaskInExplorer("/u/$ZOS_USERID", "", fixtureStack, remoteRobot) + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) } /** @@ -405,8 +403,8 @@ class WorkingSetViaSettingsTest { fun testEditWorkingSetDeleteMasksViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val wsName = "WS2" val masks = listOf("$ZOS_USERID.*".uppercase(), "Q.*", ZOS_USERID.uppercase()) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -425,8 +423,8 @@ class WorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - masks.forEach { checkItemWasDeletedWSRefreshed(it.uppercase(), projectName, fixtureStack, remoteRobot) } - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) + masks.forEach { checkItemWasDeletedWSRefreshed(it.uppercase(), fixtureStack, remoteRobot) } + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) } /** @@ -437,8 +435,8 @@ class WorkingSetViaSettingsTest { fun testEditWorkingSetDeleteAllMasksViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val wsName = "WS2" val deletedMasks = listOf("$ZOS_USERID.**", "$ZOS_USERID.@#%", "$ZOS_USERID.@#%.*", "WWW.*", maskWithLength44) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -459,8 +457,8 @@ class WorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - deletedMasks.forEach { checkItemWasDeletedWSRefreshed(it.uppercase(), projectName, fixtureStack, remoteRobot) } - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) + deletedMasks.forEach { checkItemWasDeletedWSRefreshed(it.uppercase(), fixtureStack, remoteRobot) } + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) } /** @@ -484,7 +482,6 @@ class WorkingSetViaSettingsTest { { MockResponse().setBody("{}") } ) createConnection( - projectName, fixtureStack, closableFixtureCollector, newConnectionName, @@ -492,8 +489,8 @@ class WorkingSetViaSettingsTest { remoteRobot, "https://${mockServer.hostName}:$testPort" ) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -521,7 +518,6 @@ class WorkingSetViaSettingsTest { openMaskInExplorer( "$ZOS_USERID.*".uppercase(), "Invalid URL port: \"${testPort}1\"", - projectName, fixtureStack, remoteRobot ) @@ -546,7 +542,6 @@ class WorkingSetViaSettingsTest { { MockResponse().setBody(responseDispatcher.readMockJson("infoResponse") ?: "") } ) createConnection( - projectName, fixtureStack, closableFixtureCollector, newConnectionName, @@ -560,7 +555,7 @@ class WorkingSetViaSettingsTest { { it?.requestLine?.contains("zosmf/restfiles") ?: false }, { MockResponse().setBody("{}") } ) - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -579,8 +574,8 @@ class WorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - checkItemWasDeletedWSRefreshed("Invalid URL port: \"104431\"", projectName, fixtureStack, remoteRobot) - openOrCloseWorkingSetInExplorer(wsName, projectName, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed("Invalid URL port: \"104431\"", fixtureStack, remoteRobot) + openOrCloseWorkingSetInExplorer(wsName, fixtureStack, remoteRobot) } /** @@ -592,8 +587,8 @@ class WorkingSetViaSettingsTest { val newWorkingSetName = "new ws name" val oldWorkingSetName = "WS1" val alreadyExistsWorkingSetName = "WS2" - openOrCloseWorkingSetInExplorer(oldWorkingSetName, projectName, fixtureStack, remoteRobot) - ideFrameImpl(projectName, fixtureStack) { + openOrCloseWorkingSetInExplorer(oldWorkingSetName, fixtureStack, remoteRobot) + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -619,8 +614,8 @@ class WorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - checkItemWasDeletedWSRefreshed(oldWorkingSetName, projectName, fixtureStack, remoteRobot) - openOrCloseWorkingSetInExplorer(newWorkingSetName, projectName, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed(oldWorkingSetName, fixtureStack, remoteRobot) + openOrCloseWorkingSetInExplorer(newWorkingSetName, fixtureStack, remoteRobot) } /** @@ -630,7 +625,7 @@ class WorkingSetViaSettingsTest { @Order(14) fun testDeleteWorkingSetViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { val wsName = "WS2" - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } @@ -643,7 +638,7 @@ class WorkingSetViaSettingsTest { } closableFixtureCollector.closeOnceIfExists(SettingsDialog.name) } - checkItemWasDeletedWSRefreshed(wsName, projectName, fixtureStack, remoteRobot) + checkItemWasDeletedWSRefreshed(wsName, fixtureStack, remoteRobot) } /** @@ -652,7 +647,7 @@ class WorkingSetViaSettingsTest { @Test @Order(15) fun testDeleteAllWorkingSetsViaSettings(remoteRobot: RemoteRobot) = with(remoteRobot) { - ideFrameImpl(projectName, fixtureStack) { + ideFrameImpl(PROJECT_NAME, fixtureStack) { explorer { settings(closableFixtureCollector, fixtureStack) } From b4e4c37bdd25a0e4fcb03154463444f99365ee0c Mon Sep 17 00:00:00 2001 From: Arseni Tsikhamirau Date: Tue, 16 Jan 2024 14:48:07 +0100 Subject: [PATCH 26/34] IJMP-1361-Filters-and-mask-refresh-date-and-time --- .../dataops/fetch/FileFetchProvider.kt | 21 +++ .../fetch/RemoteFileFetchProviderBase.kt | 14 +- .../formainframe/explorer/ui/DSMaskNode.kt | 1 + .../formainframe/explorer/ui/FileFetchNode.kt | 29 +++- .../formainframe/explorer/ui/JesFilterNode.kt | 4 +- .../formainframe/explorer/ui/UssDirNode.kt | 6 +- .../ibagroup/formainframe/utils/miscUtils.kt | 14 ++ .../RemoteFileFetchProviderBaseTestSpec.kt | 126 ++++++++++++++++ .../actions/UssSortActionHolderTestSpec.kt | 5 +- .../explorer/ui/FileFetchNodeTestSpec.kt | 141 ++++++++++++++++++ .../formainframe/utils/UtilsTestSpec.kt | 15 ++ 11 files changed, 370 insertions(+), 6 deletions(-) create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/dataops/RemoteFileFetchProviderBaseTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNodeTestSpec.kt diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/FileFetchProvider.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/FileFetchProvider.kt index 50ddab8e4..72e2420fb 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/FileFetchProvider.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/FileFetchProvider.kt @@ -10,12 +10,14 @@ package eu.ibagroup.formainframe.dataops.fetch +import com.intellij.ide.util.treeView.AbstractTreeNode import com.intellij.openapi.extensions.ExtensionPointName import com.intellij.openapi.progress.DumbProgressIndicator import com.intellij.openapi.progress.ProgressIndicator import com.intellij.openapi.vfs.VirtualFile import com.intellij.util.messages.Topic import eu.ibagroup.formainframe.dataops.Query +import java.time.LocalDateTime /** Interface that represents the file fetch provider and operations available for it */ interface FileFetchProvider, File : VirtualFile> { @@ -41,6 +43,25 @@ interface FileFetchProvider, File : VirtualFile> { /** Function for "load more" nodes */ fun loadMode(query: Q, progressIndicator: ProgressIndicator = DumbProgressIndicator.INSTANCE) + /** + * Function adds (node,query) pair with @param lastRefresh into the corresponding fetch provider refreshCacheState map + * + * @param query + * @param node + * @param lastRefresh + * @return Void + */ + fun applyRefreshCacheDate(query: Q, node: AbstractTreeNode<*>, lastRefresh: LocalDateTime) + + /** + * Function finds the lastRefresh date by query in refreshCacheSate map and returns it. + * If date was not found then returns null + * + * @param query + * @return LocalDateTime instance or null + */ + fun findCacheRefreshDateIfPresent(query: Q): LocalDateTime? + /** * File fetch provider contains all list of queries inside. * If the query was created with default parameters - it will find query with real parameters. diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt index 8f79cb581..7bfa691af 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/fetch/RemoteFileFetchProviderBase.kt @@ -10,10 +10,12 @@ package eu.ibagroup.formainframe.dataops.fetch +import com.intellij.ide.util.treeView.AbstractTreeNode import com.intellij.openapi.components.service import com.intellij.openapi.progress.ProcessCanceledException import com.intellij.openapi.progress.ProgressIndicator import com.intellij.openapi.vfs.VirtualFile +import com.jetbrains.rd.util.firstOrNull import eu.ibagroup.formainframe.config.connect.ConnectionConfigBase import eu.ibagroup.formainframe.dataops.* import eu.ibagroup.formainframe.dataops.exceptions.CallException @@ -22,6 +24,7 @@ import eu.ibagroup.formainframe.utils.castOrNull import eu.ibagroup.formainframe.utils.runIfTrue import eu.ibagroup.formainframe.utils.runWriteActionOnWriteThread import eu.ibagroup.formainframe.utils.sendTopic +import java.time.LocalDateTime import java.util.concurrent.locks.ReentrantLock import kotlin.collections.set import kotlin.concurrent.withLock @@ -42,7 +45,8 @@ abstract class RemoteFileFetchProviderBase, Collection>() private val cacheState = mutableMapOf, CacheState>() - protected var errorMessages = mutableMapOf, String>() + private val refreshCacheState = mutableMapOf, RemoteQuery>, LocalDateTime>() + private var errorMessages = mutableMapOf, String>() /** * Returns successfully cached files. @@ -207,6 +211,13 @@ abstract class RemoteFileFetchProviderBase, node: AbstractTreeNode<*>, lastRefresh: LocalDateTime) { + lock.withLock { refreshCacheState.computeIfAbsent(Pair(node, query)) { _ -> lastRefresh } } + } + override fun findCacheRefreshDateIfPresent(query: RemoteQuery): LocalDateTime? { + return refreshCacheState.filter { it.key.second == query }.firstOrNull()?.value + } + /** * Clears the cache with sending the cache change topic. * @param query query that identifies the cache. @@ -230,6 +241,7 @@ abstract class RemoteFileFetchProviderBase, sendTopic: Boolean) { cacheState.remove(query) if (sendTopic) { + lock.withLock { refreshCacheState.keys.removeIf { it.second == query } } sendTopic(FileFetchProvider.CACHE_CHANGES, dataOpsManager.componentManager).onCacheCleaned(query) } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNode.kt index 1f83234b9..d9659a7e1 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNode.kt @@ -40,6 +40,7 @@ class DSMaskNode( presentation.addText(value.mask, SimpleTextAttributes.REGULAR_ATTRIBUTES) presentation.addText(" ${value.volser}", SimpleTextAttributes.GRAYED_ATTRIBUTES) presentation.setIcon(ForMainframeIcons.DatasetMask) + updateRefreshDateAndTime(presentation) } override val query: RemoteQuery? diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNode.kt index 748ba91cc..1e7b9a860 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNode.kt @@ -10,10 +10,12 @@ package eu.ibagroup.formainframe.explorer.ui +import com.intellij.ide.projectView.PresentationData import com.intellij.ide.util.treeView.AbstractTreeNode import com.intellij.openapi.progress.runBackgroundableTask import com.intellij.openapi.project.Project import com.intellij.openapi.vfs.VirtualFile +import com.intellij.ui.SimpleTextAttributes import com.intellij.ui.tree.LeafState import com.intellij.util.containers.toMutableSmartList import eu.ibagroup.formainframe.common.message @@ -23,6 +25,8 @@ import eu.ibagroup.formainframe.explorer.ExplorerUnit import eu.ibagroup.formainframe.utils.castOrNull import eu.ibagroup.formainframe.utils.locked import eu.ibagroup.formainframe.utils.service +import eu.ibagroup.formainframe.utils.toHumanReadableFormat +import java.time.LocalDateTime import java.util.concurrent.atomic.AtomicBoolean import java.util.concurrent.locks.ReentrantLock import kotlin.concurrent.withLock @@ -44,6 +48,8 @@ abstract class FileFetchNode> getSelectedNodesWorkingSets(view: Expl fun String.removeTrailingSlashes(): String { return this.replace(Regex("/+$"), "/") } + +/** + * Utility function which transforms LocalDateTime timestamp to human-readable format (without nanos) + * + * @receiver LocalDateTime instance + * @return String representation of LocalDateTime in human-readable format + */ +fun LocalDateTime.toHumanReadableFormat(): String { + return "$dayOfMonth ${month.name} ${toLocalTime().truncatedTo(ChronoUnit.SECONDS).format( + DateTimeFormatter.ISO_LOCAL_TIME)}" +} diff --git a/src/test/kotlin/eu/ibagroup/formainframe/dataops/RemoteFileFetchProviderBaseTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/dataops/RemoteFileFetchProviderBaseTestSpec.kt new file mode 100644 index 000000000..4b0f32734 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/dataops/RemoteFileFetchProviderBaseTestSpec.kt @@ -0,0 +1,126 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.dataops + +import com.intellij.ide.util.treeView.AbstractTreeNode +import com.intellij.openapi.application.ApplicationManager +import com.intellij.testFramework.LightProjectDescriptor +import com.intellij.testFramework.fixtures.IdeaTestFixtureFactory +import com.intellij.testFramework.fixtures.impl.LightTempDirTestFixtureImpl +import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.config.ws.DSMask +import eu.ibagroup.formainframe.dataops.fetch.DatasetFileFetchProvider +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.utils.service +import io.kotest.assertions.assertSoftly +import io.kotest.core.spec.style.ShouldSpec +import io.kotest.matchers.maps.shouldContainExactly +import io.kotest.matchers.maps.shouldHaveSize +import io.kotest.matchers.shouldBe +import io.mockk.clearAllMocks +import io.mockk.mockk +import io.mockk.unmockkAll +import io.mockk.verify +import org.junit.platform.commons.util.ReflectionUtils +import java.lang.reflect.Field +import java.time.LocalDateTime +import java.util.concurrent.locks.ReentrantLock + +class RemoteFileFetchProviderBaseTestSpec: WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + } + + context("refresh cache test spec") { + + val dataOpsManagerService = + ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + val classUnderTest = DatasetFileFetchProvider(dataOpsManagerService) + + val queryMock = mockk>() + val nodeMock = mockk>() + val queryOtherMock = mockk>() + val nodeOtherMock = mockk>() + val lastRefreshDate = LocalDateTime.now() + val lastRefreshDateOther = LocalDateTime.of(2023, 12, 30, 10, 0, 0) + + val refreshCacheStateField = ReflectionUtils + .findFields( + DatasetFileFetchProvider::class.java, { f -> f.name.equals("refreshCacheState") }, + ReflectionUtils.HierarchyTraversalMode.TOP_DOWN + )[0] + refreshCacheStateField.isAccessible = true + + context("applyRefreshCacheDate") { + should("should add new entry with given node and query and last refreshDate") { + //given + val actualRefreshCacheMap = mutableMapOf, RemoteQuery>, LocalDateTime>() + val expectedRefreshCacheMap = mutableMapOf(Pair(Pair(nodeMock,queryMock), lastRefreshDate)) + refreshCacheStateField.set(classUnderTest, actualRefreshCacheMap) + + //when + classUnderTest.applyRefreshCacheDate(queryMock, nodeMock, lastRefreshDate) + + //then + assertSoftly { + actualRefreshCacheMap shouldContainExactly expectedRefreshCacheMap + } + } + + should("should not add new entry if entry already present") { + //given + val actualRefreshCacheMap = mutableMapOf(Pair(Pair(nodeMock,queryMock), lastRefreshDate)) + val expectedRefreshCacheMap = mutableMapOf(Pair(Pair(nodeMock,queryMock), lastRefreshDate)) + refreshCacheStateField.set(classUnderTest, actualRefreshCacheMap) + + //when + classUnderTest.applyRefreshCacheDate(queryMock, nodeMock, lastRefreshDate) + + //then + assertSoftly { + actualRefreshCacheMap shouldContainExactly expectedRefreshCacheMap + } + } + } + context("findCacheRefreshDateIfPresent") { + should("should find the last refreshDate for the node for the given query") { + //given + val refreshCacheMapForTest = mutableMapOf(Pair(Pair(nodeMock,queryMock), lastRefreshDate), Pair(Pair(nodeOtherMock,queryOtherMock), lastRefreshDateOther)) + refreshCacheStateField.set(classUnderTest, refreshCacheMapForTest) + + //when + val actualRefreshDate = classUnderTest.findCacheRefreshDateIfPresent(queryMock) + + //then + assertSoftly { + actualRefreshDate shouldBe lastRefreshDate + } + } + should("should not find the last refreshDate and return null for the node for the given query") { + //given + val refreshCacheMapForTest = mutableMapOf(Pair(Pair(nodeMock,queryMock), lastRefreshDate), Pair(Pair(nodeOtherMock,queryOtherMock), lastRefreshDateOther)) + refreshCacheStateField.set(classUnderTest, refreshCacheMapForTest) + val queryMockForTest = mockk>() + + //when + val actualRefreshDate = classUnderTest.findCacheRefreshDateIfPresent(queryMockForTest) + + //then + assertSoftly { + actualRefreshDate shouldBe null + } + } + } + unmockkAll() + } +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt index db17cde3a..d4e640204 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt @@ -12,6 +12,7 @@ package eu.ibagroup.formainframe.explorer.actions import com.intellij.openapi.actionSystem.AnActionEvent import com.intellij.openapi.application.ApplicationManager +import com.intellij.openapi.util.Key import com.intellij.openapi.vfs.VirtualFile import com.intellij.testFramework.LightProjectDescriptor import com.intellij.testFramework.fixtures.IdeaTestFixtureFactory @@ -81,9 +82,9 @@ class UssSortActionHolderTestSpec : ShouldSpec({ } val mockedActionEvent = mockk() every { mockedActionEvent.presentation } returns mockk() - every { mockedActionEvent.presentation.putClientProperty(any() as String, any() as Boolean) } just Runs + every { mockedActionEvent.presentation.putClientProperty(any() as Key, any() as Boolean) } just Runs // isFromContextMenu is marked as deprecated - every { mockedActionEvent.isFromContextMenu } returns false + every { mockedActionEvent.isFromContextMenu() } returns false every { mockedActionEvent.presentation.isEnabledAndVisible = true } just Runs every { mockedActionEvent.presentation.isEnabledAndVisible = false } just Runs diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNodeTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNodeTestSpec.kt new file mode 100644 index 000000000..04933c0d6 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/FileFetchNodeTestSpec.kt @@ -0,0 +1,141 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.ui + +import com.intellij.ide.projectView.PresentationData +import com.intellij.openapi.application.ApplicationManager +import com.intellij.openapi.project.Project +import com.intellij.openapi.vfs.VirtualFile +import com.intellij.testFramework.LightProjectDescriptor +import com.intellij.testFramework.fixtures.IdeaTestFixtureFactory +import com.intellij.testFramework.fixtures.impl.LightTempDirTestFixtureImpl +import com.intellij.ui.SimpleTextAttributes +import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.config.ws.DSMask +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.Query +import eu.ibagroup.formainframe.dataops.RemoteQuery +import eu.ibagroup.formainframe.dataops.fetch.DatasetFileFetchProvider +import eu.ibagroup.formainframe.dataops.fetch.FileFetchProvider +import eu.ibagroup.formainframe.explorer.FileExplorer +import eu.ibagroup.formainframe.explorer.FilesWorkingSetImpl +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.utils.service +import io.kotest.core.spec.style.ShouldSpec +import io.mockk.* +import java.time.LocalDateTime + +class FileFetchNodeTestSpec: WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + } + + context("refresh date test spec") { + + val dataOpsManagerService = + ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + val componentManager = dataOpsManagerService.componentManager + val datasetFileFetchProvider = mockk() + + val queryMock = mockk>() + val lastRefreshDate = LocalDateTime.of(2023, 12, 30, 10, 0, 0) + + val presentationMock = mockk() + val mockedMask = mockk() + val mockedProject = mockk() + val mockedExplorerTreeNodeParent = mockk() + val mockedWorkingSet = mockk() + val mockedExplorer = mockk() + val mockedExplorerTreeStructure = mockk() + + every { mockedWorkingSet.explorer } returns mockedExplorer + every { mockedExplorer.componentManager } returns mockk() + every { mockedExplorer.componentManager.service() } returns dataOpsManagerService + every { mockedExplorerTreeStructure.registerNode(any()) } just Runs + every { mockedWorkingSet.connectionConfig } returns mockk() + + dataOpsManagerService.testInstance = object: TestDataOpsManagerImpl(componentManager) { + override fun , File : VirtualFile> getFileFetchProvider( + requestClass: Class, + queryClass: Class>, + vFileClass: Class + ): FileFetchProvider { + return datasetFileFetchProvider as FileFetchProvider + } + } + + val classUnderTest = DSMaskNode(mockedMask, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure) + + context("updateRefreshDateAndTime") { + should("should update node presentation with correct refresh date and time given valid query") { + //given + val text = " latest refresh: 30 DECEMBER 10:00:00" + every { presentationMock.addText(any(), any()) } just Runs + every { datasetFileFetchProvider.getRealQueryInstance(any()) } returns queryMock + every { datasetFileFetchProvider.findCacheRefreshDateIfPresent(any()) } returns lastRefreshDate + + //when + classUnderTest.updateRefreshDateAndTime(presentationMock) + + //then + verify(exactly = 1) { datasetFileFetchProvider.findCacheRefreshDateIfPresent(queryMock) } + verify(exactly = 1) { presentationMock.addText(text, SimpleTextAttributes.GRAYED_ATTRIBUTES) } + clearMocks(presentationMock, verificationMarks = true) + + } + + should("should update node presentation with correct refresh date and time given valid query if no real instance found") { + //given + val text = " latest refresh: 30 DECEMBER 10:00:00" + every { presentationMock.addText(any(), any()) } just Runs + every { datasetFileFetchProvider.getRealQueryInstance(any()) } returns null + every { datasetFileFetchProvider.findCacheRefreshDateIfPresent(any()) } returns lastRefreshDate + + //when + classUnderTest.updateRefreshDateAndTime(presentationMock) + + //then + verify(exactly = 1) { presentationMock.addText(text, SimpleTextAttributes.GRAYED_ATTRIBUTES) } + clearMocks(presentationMock, verificationMarks = true) + } + + should("should not update presentation for node if no refresh date found") { + //given + every { datasetFileFetchProvider.getRealQueryInstance(any()) } returns null + every { datasetFileFetchProvider.findCacheRefreshDateIfPresent(any()) } returns null + + //when + classUnderTest.updateRefreshDateAndTime(presentationMock) + + //then + verify { presentationMock wasNot Called } + clearMocks(presentationMock, verificationMarks = true) + } + + should("should update node presentation with Out-Of-Sync text if no valid query") { + //given + val text = " Out of sync" + every { mockedWorkingSet.connectionConfig } returns null + every { presentationMock.addText(any(), any()) } just Runs + every { datasetFileFetchProvider.getRealQueryInstance(any()) } returns null + + //when + classUnderTest.updateRefreshDateAndTime(presentationMock) + + //then + verify(exactly = 1) { presentationMock.addText(text, SimpleTextAttributes.GRAYED_ATTRIBUTES) } + } + } + unmockkAll() + } +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/utils/UtilsTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/utils/UtilsTestSpec.kt index c964ae803..d8ef18a70 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/utils/UtilsTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/utils/UtilsTestSpec.kt @@ -13,6 +13,7 @@ package eu.ibagroup.formainframe.utils import com.intellij.openapi.progress.ProgressIndicator import com.intellij.openapi.progress.TaskInfo import com.intellij.openapi.ui.ValidationInfo +import com.intellij.openapi.util.text.StringUtil import com.intellij.openapi.wm.ex.ProgressIndicatorEx import com.intellij.ui.components.JBTextField import eu.ibagroup.formainframe.config.ConfigStateV2 @@ -41,11 +42,13 @@ import io.mockk.spyk import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.withContext import org.junit.jupiter.api.Assertions.* +import org.junit.platform.commons.util.StringUtils import org.zowe.kotlinsdk.annotations.ZVersion import retrofit2.Call import retrofit2.Response import java.time.Duration import java.time.Instant.now +import java.time.LocalDateTime import java.util.stream.Stream import javax.swing.JTextField @@ -883,5 +886,17 @@ class UtilsTestSpec : ShouldSpec({ duration.toMillis() shouldBeGreaterThanOrEqual 500 } } + + should("return a human readable date format given valid LocalDateTime instance") { + //given + val actualLocalDate = LocalDateTime.of(2023, 12, 30, 10, 0, 0) + val expectedString = "30 DECEMBER 10:00:00" + //when + val actualString = actualLocalDate.toHumanReadableFormat() + //then + assertSoftly { + actualString shouldBe expectedString + } + } } }) From e5e37437b26ed216011a7a1bf7149300c3f5d19f Mon Sep 17 00:00:00 2001 From: Arseni Tsikhamirau Date: Tue, 6 Feb 2024 12:49:25 +0100 Subject: [PATCH 27/34] IJMP-1496-Jobs-Sort --- .../formainframe/dataops/SortableQuery.kt | 20 + .../dataops/UnitRemoteQueryImpl.kt | 33 +- .../dataops/sort/SortQueryKeys.kt | 39 + .../explorer/actions/UssSortActionHolder.kt | 334 ------- .../explorer/actions/sort/SortAction.kt | 37 + .../actions/sort/jobs/JobsSortAction.kt | 75 ++ .../actions/sort/jobs/JobsSortActionGroup.kt | 49 + .../actions/sort/uss/UssSortAction.kt | 75 ++ .../actions/sort/uss/UssSortActionGroup.kt | 49 + .../formainframe/explorer/ui/JesFilterNode.kt | 67 +- .../formainframe/explorer/ui/JobNode.kt | 2 - .../formainframe/explorer/ui/SortableNode.kt | 37 + .../formainframe/explorer/ui/UssDirNode.kt | 122 +-- .../formainframe/explorer/ui/UssFileNode.kt | 5 +- .../formainframe/explorer/ui/UssNode.kt | 4 +- .../explorer/ui/UssSortableNode.kt | 21 - .../ibagroup/formainframe/utils/miscUtils.kt | 36 + src/main/resources/META-INF/plugin.xml | 90 +- .../actions/UssSortActionHolderTestSpec.kt | 884 ------------------ .../sort/jobs/JobsSortActionGroupTestSpec.kt | 145 +++ .../sort/jobs/JobsSortActionTestSpec.kt | 226 +++++ .../sort/uss/UssSortActionGroupTestSpec.kt | 145 +++ .../actions/sort/uss/UssSortActionTestSpec.kt | 229 +++++ .../explorer/ui/JesFilterNodeTestSpec.kt | 276 ++++++ .../explorer/ui/UssDirNodeTestSpec.kt | 57 +- .../formainframe/utils/UtilsTestSpec.kt | 54 ++ 26 files changed, 1727 insertions(+), 1384 deletions(-) create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/dataops/SortableQuery.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/dataops/sort/SortQueryKeys.kt delete mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortAction.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortAction.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroup.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortAction.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroup.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/SortableNode.kt delete mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssSortableNode.kt delete mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroupTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroupTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/JesFilterNodeTestSpec.kt diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/SortableQuery.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/SortableQuery.kt new file mode 100644 index 000000000..b9832b4a2 --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/SortableQuery.kt @@ -0,0 +1,20 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.dataops + +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys + +/** + * Interface identifies what query is SortableQuery + */ +interface SortableQuery { + val sortKeys : List +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt index 66c617634..23d215ddf 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt @@ -11,6 +11,9 @@ package eu.ibagroup.formainframe.dataops import eu.ibagroup.formainframe.config.connect.ConnectionConfigBase +import eu.ibagroup.formainframe.config.ws.JobsFilter +import eu.ibagroup.formainframe.dataops.fetch.UssQuery +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys import eu.ibagroup.formainframe.explorer.ui.UssNode import eu.ibagroup.formainframe.utils.UNIT_CLASS @@ -19,27 +22,15 @@ import eu.ibagroup.formainframe.utils.UNIT_CLASS */ data class UnitRemoteQueryImpl( override val request: R, - override val connectionConfig: Connection -) : RemoteQuery { - val sortKeys = mutableListOf(SortQueryKeys.DATE, SortQueryKeys.ASCENDING) - var requester: UssNode? = null + override val connectionConfig: Connection, +) : RemoteQuery, SortableQuery { override val resultClass: Class get() = UNIT_CLASS -} - -/** - * Enum class represents the sorting keys which is currently enabled for particular Node - */ -enum class SortQueryKeys(private val sortType: String) { - NAME("name"), - TYPE("type"), - DATE("date"), - NONE("none"), - ASCENDING("ascending"), - DESCENDING("descending"); - - override fun toString(): String { - return sortType - } - + override val sortKeys: List + get() = when(request) { + is UssQuery -> mutableListOf(SortQueryKeys.FILE_MODIFICATION_DATE, SortQueryKeys.ASCENDING) + is JobsFilter -> mutableListOf(SortQueryKeys.JOB_CREATION_DATE, SortQueryKeys.ASCENDING) + // TODO: Add sort query keys for other queries when implemented + else -> mutableListOf() + } } \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/sort/SortQueryKeys.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/sort/SortQueryKeys.kt new file mode 100644 index 000000000..181011592 --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/sort/SortQueryKeys.kt @@ -0,0 +1,39 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ +package eu.ibagroup.formainframe.dataops.sort + +/** + * Enum class represents the sorting keys which is currently enabled for particular Node + */ +enum class SortQueryKeys(private val sortType: String) { + FILE_NAME("uss_file_name"), + FILE_TYPE("uss_type"), + FILE_MODIFICATION_DATE("uss_modification_date"), + JOB_NAME("Job Name"), + JOB_CREATION_DATE("Job Creation Date"), + JOB_COMPLETION_DATE("Job Completion Date"), + JOB_STATUS("Job Status"), + JOB_OWNER("Job Owner"), + JOB_ID("Job ID"), + ASCENDING("Ascending"), + DESCENDING("Descending"); + + override fun toString(): String { + return sortType + } +} + +val typedSortKeys : List by lazy { + return@lazy listOf(SortQueryKeys.FILE_NAME, SortQueryKeys.FILE_TYPE, SortQueryKeys.FILE_MODIFICATION_DATE, SortQueryKeys.JOB_NAME, SortQueryKeys.JOB_ID, SortQueryKeys.JOB_OWNER, SortQueryKeys.JOB_STATUS, SortQueryKeys.JOB_CREATION_DATE, SortQueryKeys.JOB_COMPLETION_DATE) +} + +val orderingSortKeys : List by lazy { + return@lazy listOf(SortQueryKeys.ASCENDING, SortQueryKeys.DESCENDING) +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt deleted file mode 100644 index a256ae750..000000000 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolder.kt +++ /dev/null @@ -1,334 +0,0 @@ -/* - * This program and the accompanying materials are made available under the terms of the - * Eclipse Public License v2.0 which accompanies this distribution, and is available at - * https://www.eclipse.org/legal/epl-v20.html - * - * SPDX-License-Identifier: EPL-2.0 - * - * Copyright IBA Group 2020 - */ - -// TODO: too much boilerplate -package eu.ibagroup.formainframe.explorer.actions - -import com.intellij.openapi.actionSystem.AnActionEvent -import com.intellij.openapi.actionSystem.DefaultActionGroup -import com.intellij.openapi.actionSystem.ToggleAction -import com.intellij.openapi.components.service -import eu.ibagroup.formainframe.dataops.DataOpsManager -import eu.ibagroup.formainframe.dataops.RemoteQuery -import eu.ibagroup.formainframe.dataops.SortQueryKeys -import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl -import eu.ibagroup.formainframe.dataops.fetch.UssQuery -import eu.ibagroup.formainframe.explorer.ui.FileExplorerView -import eu.ibagroup.formainframe.explorer.ui.UssDirNode -import eu.ibagroup.formainframe.explorer.ui.getExplorerView -import eu.ibagroup.formainframe.vfs.MFVirtualFile - -/** - * Represents internal USS fetch provider to be able to update the query for each USS dir node whose sorting is enabled - */ -internal val fetchProvider = service() - .getFileFetchProvider( - UssQuery::class.java, - RemoteQuery::class.java, - MFVirtualFile::class.java - ) - -/** - * Represents the custom sort action group in the FileExplorerView context menu - */ -class SortActionGroup : DefaultActionGroup() { - - /** - * Update method to determine if sorting is possible for particular item in the tree - */ - override fun update(e: AnActionEvent) { - val view = e.getExplorerView() - view ?: let { - e.presentation.isEnabledAndVisible = false - return@let - } - val selectedNode = view?.mySelectedNodesData ?: return - val treePathFromModel = view.myTree.selectionPath - e.presentation.isEnabledAndVisible = selectedNode.any { - it.node is UssDirNode && view.myTree.isExpanded(treePathFromModel) - } - } -} - -/** - * Custom action for sorting by item Name (Dir or File name in case of USS) - */ -class SortByNameAction : ToggleAction() { - - /** - * Update method to determine if sorting by Name is possible for particular item in the tree - */ - override fun update(e: AnActionEvent) { - super.update(e) - e.presentation.isEnabledAndVisible = true - } - - /** - * Action performed method to register the custom behavior when By Name sorting is clicked - */ - override fun setSelected(e: AnActionEvent, state: Boolean) { - val view = e.getExplorerView() ?: return - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) - if (selectedNode.currentSortQueryKeysList.isEmpty()) { - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) - queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) - } else { - selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.TYPE) - selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DATE) - if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.NAME)) - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.NAME) - queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) - } - queryToUpdate.requester = selectedNode - selectedNode.cleanCache(false) - fetchProvider.reload(queryToUpdate) - } - } - - /** - * Custom isSelected method determines if the sorting By Name is currently enabled or not. Updates UI by 'tick' mark - */ - override fun isSelected(e: AnActionEvent): Boolean { - val view = e.getExplorerView() ?: return false - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - return selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.NAME) - } - return false - } - - /** - * If action is dumb aware or not - */ - override fun isDumbAware(): Boolean { - return true - } -} - -/** - * Custom action for sorting by item Type (Dir or File type+name in case of USS) - */ -class SortByTypeAction : ToggleAction() { - - override fun update(e: AnActionEvent) { - super.update(e) - e.presentation.isEnabledAndVisible = true - } - - /** - * Action performed method to register the custom behavior when By Type sorting is clicked - */ - override fun setSelected(e: AnActionEvent, state: Boolean) { - val view = e.getExplorerView() ?: return - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) - if (selectedNode.currentSortQueryKeysList.isEmpty()) { - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) - queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) - } else { - selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.NAME) - selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DATE) - if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.TYPE)) - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.TYPE) - queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) - } - queryToUpdate.requester = selectedNode - selectedNode.cleanCache(false) - fetchProvider.reload(queryToUpdate) - } - } - - /** - * Custom isSelected method determines if the sorting By Type is currently enabled or not. Updates UI by 'tick' mark - */ - override fun isSelected(e: AnActionEvent): Boolean { - val view = e.getExplorerView() ?: return false - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - return selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.TYPE) - } - return false - } - - /** - * If action is dumb aware or not - */ - override fun isDumbAware(): Boolean { - return true - } -} - -/** - * Custom action for sorting by item Modification Date (Dir or File modify date in case of USS) - */ -class SortByModificationDateAction : ToggleAction() { - - override fun update(e: AnActionEvent) { - super.update(e) - e.presentation.isEnabledAndVisible = true - } - - /** - * Action performed method to register the custom behavior when By Modification Date sorting is clicked - */ - override fun setSelected(e: AnActionEvent, state: Boolean) { - val view = e.getExplorerView() ?: return - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) - if (selectedNode.currentSortQueryKeysList.isEmpty()) { - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) - queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) - } else { - selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.NAME) - selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.TYPE) - if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DATE)) - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) - queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) - } - queryToUpdate.requester = selectedNode - selectedNode.cleanCache(false) - fetchProvider.reload(queryToUpdate) - } - } - - /** - * Custom isSelected method determines if the sorting By Modification Date is currently enabled or not. Updates UI by 'tick' mark - */ - override fun isSelected(e: AnActionEvent): Boolean { - val view = e.getExplorerView() ?: return false - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - return selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DATE) - } - return false - } - - /** - * If action is dumb aware or not - */ - override fun isDumbAware(): Boolean { - return true - } -} - -/** - * Custom action for sorting by item Name/Type/Modify Date in ascending order (Dir or File name in case of USS) - */ -class SortByAscendingOrderAction : ToggleAction() { - - override fun update(e: AnActionEvent) { - super.update(e) - e.presentation.isEnabledAndVisible = true - } - - /** - * Action performed method to register the custom behavior when Ascending sorting is clicked - */ - override fun setSelected(e: AnActionEvent, state: Boolean) { - val view = e.getExplorerView() ?: return - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) - if (selectedNode.currentSortQueryKeysList.isEmpty()) { - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) - queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) - } else { - selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.DESCENDING) - if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.ASCENDING)) - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.ASCENDING) - queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) - } - queryToUpdate.requester = selectedNode - selectedNode.cleanCache(false) - fetchProvider.reload(queryToUpdate) - } - } - - /** - * Custom isSelected method determines if the ascending sorting is currently enabled or not. Updates UI by 'tick' mark - */ - override fun isSelected(e: AnActionEvent): Boolean { - val view = e.getExplorerView() ?: return false - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - return selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.ASCENDING) - } - return false - } - - /** - * If action is dumb aware or not - */ - override fun isDumbAware(): Boolean { - return true - } -} - -/** - * Custom action for sorting by item Name/Type/Modify Date in descending order (Dir or File name in case of USS) - */ -class SortByDescendingOrderAction : ToggleAction() { - - override fun update(e: AnActionEvent) { - super.update(e) - e.presentation.isEnabledAndVisible = true - } - - /** - * Action performed method to register the custom behavior when Descending sorting is clicked - */ - override fun setSelected(e: AnActionEvent, state: Boolean) { - val view = e.getExplorerView() ?: return - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - val queryToUpdate = (selectedNode.query as UnitRemoteQueryImpl) - if (selectedNode.currentSortQueryKeysList.isEmpty()) { - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DATE) - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DESCENDING) - queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) - } else { - selectedNode.currentSortQueryKeysList.remove(SortQueryKeys.ASCENDING) - if (!selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DESCENDING)) - selectedNode.currentSortQueryKeysList.add(SortQueryKeys.DESCENDING) - queryToUpdate.sortKeys.addAll(selectedNode.currentSortQueryKeysList) - } - queryToUpdate.requester = selectedNode - selectedNode.cleanCache(false) - fetchProvider.reload(queryToUpdate) - } - } - - /** - * Custom isSelected method determines if the descending sorting is currently enabled or not. Updates UI by 'tick' mark - */ - override fun isSelected(e: AnActionEvent): Boolean { - val view = e.getExplorerView() ?: return false - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - return selectedNode.currentSortQueryKeysList.contains(SortQueryKeys.DESCENDING) - } - return false - } - - /** - * If action is dumb aware or not - */ - override fun isDumbAware(): Boolean { - return true - } -} diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortAction.kt new file mode 100644 index 000000000..ebd3db6bb --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortAction.kt @@ -0,0 +1,37 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort + +import com.intellij.openapi.actionSystem.ActionUpdateThread +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.ToggleAction + +abstract class SortAction : ToggleAction() { + + override fun update(e: AnActionEvent) { + super.update(e) + e.presentation.isEnabledAndVisible = true + } + + /** + * If action is dumb aware or not + */ + override fun isDumbAware(): Boolean { + return true + } + + /** + * Tells that only UI component is affected + */ + override fun getActionUpdateThread(): ActionUpdateThread { + return ActionUpdateThread.EDT + } +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortAction.kt new file mode 100644 index 000000000..3e8cf83dd --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortAction.kt @@ -0,0 +1,75 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.jobs + +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.components.service +import eu.ibagroup.formainframe.config.ws.JobsFilter +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.actions.sort.SortAction +import eu.ibagroup.formainframe.explorer.ui.JesExplorerView +import eu.ibagroup.formainframe.explorer.ui.JesFilterNode +import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.utils.clearAndMergeWith +import eu.ibagroup.formainframe.utils.clearOldKeysAndAddNew +import eu.ibagroup.formainframe.utils.runWriteActionInEdt +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import java.time.LocalDateTime + +/** + * Represents internal Jobs fetch provider to be able to update the query for each Job Filter node whose sorting is enabled + */ +internal val fetchJobsProvider = service() + .getFileFetchProvider( + JobsFilter::class.java, + UnitRemoteQueryImpl::class.java, + MFVirtualFile::class.java + ) +class JobsSortAction : SortAction() { + + /** + * Action performed method to register the custom behavior when any Jobs Sort Key was clicked in UI + */ + override fun setSelected(e: AnActionEvent, state: Boolean) { + val view = e.getExplorerView() ?: return + val sortJobKey = this.templateText?.uppercase()?.replace(" ", "_")?.let { SortQueryKeys.valueOf(it) } + ?: throw Exception("Sort key for the selected action was not found.") + if (isSelected(e)) return + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is JesFilterNode) { + val queryToUpdate = selectedNode.query as UnitRemoteQueryImpl + selectedNode.currentSortQueryKeysList.clearOldKeysAndAddNew(sortJobKey) + queryToUpdate.sortKeys.clearAndMergeWith(selectedNode.currentSortQueryKeysList) + runWriteActionInEdt { + selectedNode.cleanCache(false) + fetchJobsProvider.apply { + reload(queryToUpdate) + applyRefreshCacheDate(queryToUpdate, selectedNode, LocalDateTime.now()) + } + } + } + } + + /** + * Custom isSelected method determines if the Jobs Sort Key is currently enabled or not. Updates UI by 'tick' mark + */ + override fun isSelected(e: AnActionEvent): Boolean { + val view = e.getExplorerView() ?: return false + val sortJobKey = this.templateText?.uppercase()?.replace(" ", "_")?.let { SortQueryKeys.valueOf(it) } ?: return false + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is JesFilterNode) { + return selectedNode.currentSortQueryKeysList.contains(sortJobKey) + } + return false + } +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroup.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroup.kt new file mode 100644 index 000000000..8de610c11 --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroup.kt @@ -0,0 +1,49 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.jobs + +import com.intellij.openapi.actionSystem.ActionUpdateThread +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DefaultActionGroup +import eu.ibagroup.formainframe.explorer.ui.JesExplorerView +import eu.ibagroup.formainframe.explorer.ui.JesFilterNode +import eu.ibagroup.formainframe.explorer.ui.getExplorerView + +/** + * Represents the custom Jobs sort action group in the JesExplorerView context menu + */ +class JobsSortActionGroup : DefaultActionGroup() { + + /** + * Update method to determine if sorting is possible for particular item in the tree + */ + override fun update(e: AnActionEvent) { + val view = e.getExplorerView() + view ?: let { + e.presentation.isEnabledAndVisible = false + return + } + val selectedNodes = view.mySelectedNodesData + val treePathFromModel = view.myTree.selectionPath + e.presentation.apply { + isEnabledAndVisible = selectedNodes.size == 1 && selectedNodes.any { + it.node is JesFilterNode && view.myTree.isExpanded(treePathFromModel) + } + } + } + + /** + * Tells that only UI component is affected + */ + override fun getActionUpdateThread(): ActionUpdateThread { + return ActionUpdateThread.EDT + } +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortAction.kt new file mode 100644 index 000000000..6b66e6db2 --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortAction.kt @@ -0,0 +1,75 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.uss + +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.components.service +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.fetch.UssQuery +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.actions.sort.SortAction +import eu.ibagroup.formainframe.explorer.ui.FileExplorerView +import eu.ibagroup.formainframe.explorer.ui.UssDirNode +import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.utils.clearAndMergeWith +import eu.ibagroup.formainframe.utils.clearOldKeysAndAddNew +import eu.ibagroup.formainframe.utils.runWriteActionInEdt +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import java.time.LocalDateTime + +/** + * Represents internal Jobs fetch provider to be able to update the query for each Job Filter node whose sorting is enabled + */ +internal val fetchUssProvider = service() + .getFileFetchProvider( + UssQuery::class.java, + UnitRemoteQueryImpl::class.java, + MFVirtualFile::class.java + ) +class UssSortAction : SortAction() { + + /** + * Action performed method to register the custom behavior when any USS Sort Key was clicked in UI + */ + override fun setSelected(e: AnActionEvent, state: Boolean) { + val view = e.getExplorerView() ?: return + val sortUssKey = this.templateText?.uppercase()?.replace(" ", "_")?.let { SortQueryKeys.valueOf(it) } + ?: throw Exception("Sort key for the selected action was not found.") + if (isSelected(e)) return + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + val queryToUpdate = selectedNode.query as UnitRemoteQueryImpl + selectedNode.currentSortQueryKeysList.clearOldKeysAndAddNew(sortUssKey) + queryToUpdate.sortKeys.clearAndMergeWith(selectedNode.currentSortQueryKeysList) + runWriteActionInEdt { + selectedNode.cleanCache(false) + fetchUssProvider.apply { + reload(queryToUpdate) + applyRefreshCacheDate(queryToUpdate, selectedNode, LocalDateTime.now()) + } + } + } + } + + /** + * Custom isSelected method determines if the USS Sort Key is currently enabled or not. Updates UI by 'tick' mark + */ + override fun isSelected(e: AnActionEvent): Boolean { + val view = e.getExplorerView() ?: return false + val sortUssKey = this.templateText?.uppercase()?.replace(" ", "_")?.let { SortQueryKeys.valueOf(it) } ?: return false + val selectedNode = view.mySelectedNodesData[0].node + if (selectedNode is UssDirNode) { + return selectedNode.currentSortQueryKeysList.contains(sortUssKey) + } + return false + } +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroup.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroup.kt new file mode 100644 index 000000000..f3a63962b --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroup.kt @@ -0,0 +1,49 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.uss + +import com.intellij.openapi.actionSystem.ActionUpdateThread +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DefaultActionGroup +import eu.ibagroup.formainframe.explorer.ui.FileExplorerView +import eu.ibagroup.formainframe.explorer.ui.UssDirNode +import eu.ibagroup.formainframe.explorer.ui.getExplorerView + +/** + * Represents the custom USS files sort action group in the FileExplorerView context menu + */ +class UssSortActionGroup : DefaultActionGroup() { + + /** + * Update method to determine if sorting is possible for particular item in the tree + */ + override fun update(e: AnActionEvent) { + val view = e.getExplorerView() + view ?: let { + e.presentation.isEnabledAndVisible = false + return + } + val selectedNodes = view.mySelectedNodesData + val treePathFromModel = view.myTree.selectionPath + e.presentation.apply { + isEnabledAndVisible = selectedNodes.size == 1 && selectedNodes.any { + it.node is UssDirNode && view.myTree.isExpanded(treePathFromModel) + } + } + } + + /** + * Tells that only UI component is affected + */ + override fun getActionUpdateThread(): ActionUpdateThread { + return ActionUpdateThread.EDT + } +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JesFilterNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JesFilterNode.kt index c119aae11..da0a6eaa6 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JesFilterNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JesFilterNode.kt @@ -13,14 +13,19 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.ide.projectView.PresentationData import com.intellij.ide.util.treeView.AbstractTreeNode +import com.intellij.openapi.components.service import com.intellij.openapi.project.Project import com.intellij.ui.SimpleTextAttributes -import com.intellij.util.containers.toMutableSmartList import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.ws.JobsFilter +import eu.ibagroup.formainframe.dataops.DataOpsManager import eu.ibagroup.formainframe.dataops.RemoteQuery import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.dataops.sort.typedSortKeys import eu.ibagroup.formainframe.explorer.JesWorkingSet +import eu.ibagroup.formainframe.utils.clearAndMergeWith import eu.ibagroup.formainframe.vfs.MFVirtualFile import icons.ForMainframeIcons @@ -32,10 +37,12 @@ class JesFilterNode( project: Project, parent: ExplorerTreeNode, workingSet: JesWorkingSet, - treeStructure: ExplorerTreeStructureBase + treeStructure: ExplorerTreeStructureBase, + override val currentSortQueryKeysList: List = mutableListOf(SortQueryKeys.JOB_CREATION_DATE, SortQueryKeys.ASCENDING), + override val sortedNodes: List> = mutableListOf() ) : RemoteMFFileFetchNode( jobsFilter, project, parent, workingSet, treeStructure -), RefreshableNode { +), SortableNode, RefreshableNode { override val query: RemoteQuery? get() { @@ -45,10 +52,10 @@ class JesFilterNode( } else null } - override fun Collection.toChildrenNodes(): MutableList> { + override fun Collection.toChildrenNodes(): List> { return map { JobNode(it, notNullProject, this@JesFilterNode, unit, treeStructure) - }.toMutableSmartList() + }.let { sortChildrenNodes(it, currentSortQueryKeysList) } } override val requestClass = JobsFilter::class.java @@ -65,4 +72,54 @@ class JesFilterNode( return "Fetching jobs for ${query.request}" } + override fun > sortChildrenNodes(childrenNodes: List, sortKeys: List): List { + val listToReturn : List = mutableListOf() + val foundSortKey = sortKeys.firstOrNull { typedSortKeys.contains(it) } + if (foundSortKey != null) { + listToReturn.clearAndMergeWith(performJobsSorting(childrenNodes, this@JesFilterNode, foundSortKey)) + } else { + listToReturn.clearAndMergeWith(childrenNodes) + } + return listToReturn + } + + /** + * Function sorts the children nodes by specified sorting key + * @param nodes + * @param jesFilter + * @param sortKey + * @return sorted nodes by specified key + */ + private fun performJobsSorting(nodes: List>, jesFilter: JesFilterNode, sortKey: SortQueryKeys) : List> { + val sortedNodesInternal: List> = if (jesFilter.currentSortQueryKeysList.contains(SortQueryKeys.ASCENDING)) { + nodes.sortedBy { + selector(sortKey).invoke(it) + } + } else { + nodes.sortedByDescending { + selector(sortKey).invoke(it) + } + } + return sortedNodesInternal.also { sortedNodes.clearAndMergeWith(it) } + } + + /** + * Selector which extracts the job info by specified sort key + * @param key - sort key + * @return String representation of the extracted job info attribute of the virtual file + */ + private fun selector(key: SortQueryKeys) : (AbstractTreeNode<*>) -> String? { + return { + val jobInfo = (service().tryToGetAttributes((it as JobNode).value) as RemoteJobAttributes).jobInfo + when(key) { + SortQueryKeys.JOB_NAME -> jobInfo.jobName + SortQueryKeys.JOB_OWNER -> jobInfo.owner + SortQueryKeys.JOB_STATUS -> jobInfo.status?.value + SortQueryKeys.JOB_ID -> jobInfo.jobId + SortQueryKeys.JOB_CREATION_DATE -> jobInfo.execStarted + SortQueryKeys.JOB_COMPLETION_DATE -> jobInfo.execEnded + else -> null + } + } + } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JobNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JobNode.kt index 488b3567d..a5feb746d 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JobNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JobNode.kt @@ -51,8 +51,6 @@ class JobNode( override val query: RemoteQuery? get() { val connectionConfig = unit.connectionConfig - - return if (connectionConfig != null) { UnitRemoteQueryImpl(JobQuery(value), connectionConfig) } else null diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/SortableNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/SortableNode.kt new file mode 100644 index 000000000..c9ee6dead --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/SortableNode.kt @@ -0,0 +1,37 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.ui + +import com.intellij.ide.util.treeView.AbstractTreeNode +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys + +/** + * Interface which represents any sortable Node + */ +interface SortableNode { + /** + * Field which holds and identifies the current sort keys for particular Node + */ + val currentSortQueryKeysList : List + + /** + * Stores the sorted nodes for particular SortableNode + */ + val sortedNodes: List> + + /** + * Method sorts the children nodes regarding the sort keys are currently enabled + * @param sortKeys - Sort keys to check + * @return list of sorted children nodes + */ + fun > sortChildrenNodes(childrenNodes: List, sortKeys: List): List = mutableListOf() + +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNode.kt index 3f0b6c765..525b579f5 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNode.kt @@ -21,7 +21,9 @@ import eu.ibagroup.formainframe.config.ws.UssPath import eu.ibagroup.formainframe.dataops.* import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes import eu.ibagroup.formainframe.dataops.fetch.UssQuery +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys import eu.ibagroup.formainframe.explorer.FilesWorkingSet +import eu.ibagroup.formainframe.utils.clearAndMergeWith import eu.ibagroup.formainframe.utils.service import eu.ibagroup.formainframe.vfs.MFVirtualFile @@ -45,7 +47,9 @@ class UssDirNode( workingSet: FilesWorkingSet, treeStructure: ExplorerTreeStructureBase, private var vFile: MFVirtualFile? = null, - private val isRootNode: Boolean = false + private val isRootNode: Boolean = false, + override val currentSortQueryKeysList: List = mutableListOf(SortQueryKeys.FILE_MODIFICATION_DATE, SortQueryKeys.ASCENDING), + override val sortedNodes: List> = mutableListOf() ) : RemoteMFFileFetchNode( ussPath, project, parent, workingSet, treeStructure ), UssNode, RefreshableNode { @@ -56,12 +60,6 @@ class UssDirNode( super.init() } - /** Field which holds and identifies the current sort keys for particular Node, be default Nodes are sorted by Data in Ascending order */ - var currentSortQueryKeysList = mutableListOf(SortQueryKeys.DATE, SortQueryKeys.ASCENDING) - - /** Stores the sorted nodes for particular Node */ - private var sortedNodes: List> = mutableListOf() - val isUssMask = vFile == null override val query: RemoteQuery? @@ -79,10 +77,7 @@ class UssDirNode( /** Transform the collection of mainframe virtual files to the list of USS children nodes */ override fun Collection.toChildrenNodes(): List> { return find { attributesService.getAttributes(it)?.path == value.path } - ?.also { - vFile = it - treeStructure.registerNode(this@UssDirNode) - } + ?.also { vFile = it } ?.children ?.map { if (it.isDirectory) { @@ -97,7 +92,7 @@ class UssDirNode( } else { UssFileNode(it, notNullProject, this@UssDirNode, unit, treeStructure) } - }?.sortChildrenNodes(currentSortQueryKeysList) ?: listOf() + }?.let { sortChildrenNodes(it, currentSortQueryKeysList) } ?: listOf() } override val requestClass = UssQuery::class.java @@ -152,85 +147,92 @@ class UssDirNode( return vFile } - /** - * Method sorts the children nodes regarding the sort keys are currently enabled - * @param sortKeys - Sort keys to check - * @return list of sorted nodes - */ - override fun List>.sortChildrenNodes(sortKeys: List): List> { - if (sortKeys.contains(SortQueryKeys.NAME)) { + override fun > sortChildrenNodes(childrenNodes: List, sortKeys: List): List { + if (sortKeys.contains(SortQueryKeys.FILE_NAME)) { if (sortKeys.contains(SortQueryKeys.ASCENDING)) { - return this.sortedBy { + return childrenNodes.sortedBy { when (it) { is UssDirNode -> it.vFile?.filenameInternal is UssFileNode -> it.virtualFile.filenameInternal else -> null } - }.also { sortedNodes = it } + }.also { + sortedNodes.clearAndMergeWith(it) + } } else { - return this.sortedByDescending { + return childrenNodes.sortedByDescending { when (it) { is UssDirNode -> it.vFile?.filenameInternal is UssFileNode -> it.virtualFile.filenameInternal else -> null } - }.also { sortedNodes = it } + }.also { + sortedNodes.clearAndMergeWith(it) + } } - } else if (sortKeys.contains(SortQueryKeys.TYPE)) { - val listToReturn = mutableListOf>() - val dirs = mutableListOf() - val files = mutableListOf() - val sortedDirs: List - val sortedFiles: List - this.forEach { + } else if (sortKeys.contains(SortQueryKeys.FILE_TYPE)) { + val listToReturn = mutableListOf() + val dirs = mutableListOf() + val files = mutableListOf() + val sortedDirs: List + val sortedFiles: List + childrenNodes.forEach { when (it) { is UssDirNode -> dirs.add(it) is UssFileNode -> files.add(it) } } return if (sortKeys.contains(SortQueryKeys.ASCENDING)) { - sortedDirs = dirs.sortedBy { it.vFile?.filenameInternal } - sortedFiles = files.sortedBy { it.virtualFile.filenameInternal } + sortedDirs = dirs.sortedBy { (it as UssDirNode).vFile?.filenameInternal } + sortedFiles = files.sortedBy { (it as UssFileNode).virtualFile.filenameInternal } listToReturn.addAll(sortedDirs) listToReturn.addAll(sortedFiles) - listToReturn.also { sortedNodes = it } + listToReturn.also { + sortedNodes.clearAndMergeWith(it) + } } else { - sortedDirs = dirs.sortedByDescending { it.vFile?.filenameInternal } - sortedFiles = files.sortedByDescending { it.virtualFile.filenameInternal } + sortedDirs = dirs.sortedByDescending { (it as UssDirNode).vFile?.filenameInternal } + sortedFiles = files.sortedByDescending { (it as UssFileNode).virtualFile.filenameInternal } listToReturn.addAll(sortedDirs) listToReturn.addAll(sortedFiles) - listToReturn.also { sortedNodes = it } - } - } else if (sortKeys.contains(SortQueryKeys.DATE)) { - - fun select(node: AbstractTreeNode<*>) : String? { - return when (node) { - is UssDirNode -> { - node.virtualFile?.let { vFile -> attributesService.getAttributes(vFile) }?.modificationTime - } - is UssFileNode -> { - attributesService.getAttributes(node.virtualFile)?.modificationTime - } - else -> null + listToReturn.also { + sortedNodes.clearAndMergeWith(it) } } - + } else if (sortKeys.contains(SortQueryKeys.FILE_MODIFICATION_DATE)) { return if (sortKeys.contains(SortQueryKeys.ASCENDING)) { - this.sortedBy { - return@sortedBy select(it) - }.also { sortedNodes = it } + childrenNodes.sortedBy { + modificationTimeSelector().invoke(it) + }.also { + sortedNodes.clearAndMergeWith(it) + } } else { - this.sortedByDescending { - return@sortedByDescending select(it) - }.also { sortedNodes = it } + childrenNodes.sortedByDescending { + modificationTimeSelector().invoke(it) + }.also { + sortedNodes.clearAndMergeWith(it) + } } } else { - return this + return childrenNodes } } - fun sortChildrenForTestInternal(sortKeys: List): (List>) -> List> = { it.sortChildrenNodes(sortKeys) } + /** + * Selector which extracts modification time for the every child node + * @return String representation of the modification time of the virtual file + */ + private fun modificationTimeSelector(): (AbstractTreeNode<*>) -> String? { + return { + when(it) { + is UssDirNode -> { + it.virtualFile?.let { vFile -> attributesService.getAttributes(vFile) }?.modificationTime + } + is UssFileNode -> { + attributesService.getAttributes(it.virtualFile)?.modificationTime + } + else -> null + } + } + } } - -fun sortChildrenForTest(listToSort: List>, sortKeys: List): UssDirNode.() -> List> - = { sortChildrenForTestInternal(sortKeys).invoke(listToSort) } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssFileNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssFileNode.kt index 8464e2b73..c20b2619b 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssFileNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssFileNode.kt @@ -17,6 +17,7 @@ import com.intellij.openapi.util.Iconable import com.intellij.ui.AnimatedIcon import com.intellij.util.IconUtil import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys import eu.ibagroup.formainframe.explorer.ExplorerUnit import eu.ibagroup.formainframe.vfs.MFVirtualFile @@ -26,7 +27,9 @@ class UssFileNode( project: Project, parent: ExplorerTreeNode, unit: ExplorerUnit, - treeStructure: ExplorerTreeStructureBase + treeStructure: ExplorerTreeStructureBase, + override val currentSortQueryKeysList: List = mutableListOf(), + override val sortedNodes: List> = mutableListOf() ) : ExplorerUnitTreeNodeBase>( file, project, parent, unit, treeStructure ), UssNode { diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssNode.kt index 7bb375d66..0b169a120 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssNode.kt @@ -13,6 +13,6 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.ide.util.treeView.AbstractTreeNode /** - * interface which represents any USS Node. Extents USS sortable Node which implements children nodes sorting method + * interface which represents any USS Node. Extends SortableNode which implements children nodes sorting method */ -interface UssNode : UssSortableNode> \ No newline at end of file +interface UssNode : SortableNode \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssSortableNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssSortableNode.kt deleted file mode 100644 index a7d90e6ca..000000000 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssSortableNode.kt +++ /dev/null @@ -1,21 +0,0 @@ -/* - * This program and the accompanying materials are made available under the terms of the - * Eclipse Public License v2.0 which accompanies this distribution, and is available at - * https://www.eclipse.org/legal/epl-v20.html - * - * SPDX-License-Identifier: EPL-2.0 - * - * Copyright IBA Group 2020 - */ - -package eu.ibagroup.formainframe.explorer.ui - -import eu.ibagroup.formainframe.dataops.SortQueryKeys - -/** - * Interface which represents any USS sortable Node - * @param Node - Nodes type to sort - */ -interface UssSortableNode { - fun List.sortChildrenNodes(sortKeys: List): List = mutableListOf() -} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/utils/miscUtils.kt b/src/main/kotlin/eu/ibagroup/formainframe/utils/miscUtils.kt index 560e5a45f..53b8b5066 100755 --- a/src/main/kotlin/eu/ibagroup/formainframe/utils/miscUtils.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/utils/miscUtils.kt @@ -15,6 +15,9 @@ import com.intellij.util.containers.minimalElements import com.intellij.util.containers.toArray import eu.ibagroup.formainframe.config.ConfigDeclaration import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.dataops.sort.orderingSortKeys +import eu.ibagroup.formainframe.dataops.sort.typedSortKeys import eu.ibagroup.formainframe.explorer.WorkingSet import eu.ibagroup.formainframe.explorer.ui.ExplorerTreeView import eu.ibagroup.formainframe.explorer.ui.ExplorerUnitTreeNodeBase @@ -181,6 +184,39 @@ fun List.mergeWith(another: List): MutableList { return this.plus(another).toSet().toMutableList() } +/** + * Function clears the input list and adds another list elements to the end of this list + * @receiver any kind of MutableList + * @param another + */ +fun List.clearAndMergeWith(another: List) { + (this as MutableList).apply { + clear() + addAll(another) + } +} + +/** + * Function clears the input list and adds the new sortKey to this list or does nothing if sortKey is null + * @receiver Any kind of MutableList of the current sortKeys + * @param toAdd + */ +fun List.clearOldKeysAndAddNew(toAdd: SortQueryKeys?) { + if (toAdd != null) { + if (typedSortKeys.contains(toAdd)) { + (this as MutableList).apply { + removeAll(typedSortKeys.toSet()) + add(toAdd) + } + } else { + (this as MutableList).apply { + removeAll(orderingSortKeys.toSet()) + add(toAdd) + } + } + } +} + val UNIT_CLASS = Unit::class.java inline fun T.applyIfNotNull(v: V?, block: T.(V) -> T): T { diff --git a/src/main/resources/META-INF/plugin.xml b/src/main/resources/META-INF/plugin.xml index cb755887e..097796361 100755 --- a/src/main/resources/META-INF/plugin.xml +++ b/src/main/resources/META-INF/plugin.xml @@ -436,25 +436,59 @@ Example of how to see the output:
class="eu.ibagroup.formainframe.explorer.actions.CreateUssDirectoryAction" text="Directory" icon="AllIcons.Nodes.Folder"/> - + - + + class="eu.ibagroup.formainframe.explorer.actions.sort.uss.UssSortAction" + text="File Modification Date"/> - + + + + + + + + + + + + - + + + + @@ -600,17 +634,33 @@ Example of how to see the output:
- - - + + - - + + + + + + + + + + + + + + - + @@ -671,6 +721,8 @@ Example of how to see the output:
+ + diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt deleted file mode 100644 index d4e640204..000000000 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/UssSortActionHolderTestSpec.kt +++ /dev/null @@ -1,884 +0,0 @@ -/* - * This program and the accompanying materials are made available under the terms of the - * Eclipse Public License v2.0 which accompanies this distribution, and is available at - * https://www.eclipse.org/legal/epl-v20.html - * - * SPDX-License-Identifier: EPL-2.0 - * - * Copyright IBA Group 2020 - */ - -package eu.ibagroup.formainframe.explorer.actions - -import com.intellij.openapi.actionSystem.AnActionEvent -import com.intellij.openapi.application.ApplicationManager -import com.intellij.openapi.util.Key -import com.intellij.openapi.vfs.VirtualFile -import com.intellij.testFramework.LightProjectDescriptor -import com.intellij.testFramework.fixtures.IdeaTestFixtureFactory -import com.intellij.testFramework.fixtures.impl.LightTempDirTestFixtureImpl -import eu.ibagroup.formainframe.config.connect.ConnectionConfig -import eu.ibagroup.formainframe.dataops.DataOpsManager -import eu.ibagroup.formainframe.dataops.Query -import eu.ibagroup.formainframe.dataops.SortQueryKeys -import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl -import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes -import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes -import eu.ibagroup.formainframe.dataops.fetch.FileFetchProvider -import eu.ibagroup.formainframe.dataops.fetch.UssFileFetchProvider -import eu.ibagroup.formainframe.dataops.fetch.UssQuery -import eu.ibagroup.formainframe.explorer.ui.FileExplorerView -import eu.ibagroup.formainframe.explorer.ui.LibraryNode -import eu.ibagroup.formainframe.explorer.ui.NodeData -import eu.ibagroup.formainframe.explorer.ui.UssDirNode -import eu.ibagroup.formainframe.explorer.ui.getExplorerView -import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl -import eu.ibagroup.formainframe.utils.service -import eu.ibagroup.formainframe.vfs.MFVirtualFile -import io.kotest.assertions.assertSoftly -import io.kotest.core.spec.style.ShouldSpec -import io.kotest.matchers.shouldBe -import io.mockk.Runs -import io.mockk.clearAllMocks -import io.mockk.every -import io.mockk.just -import io.mockk.mockk -import io.mockk.mockkObject -import io.mockk.spyk -import javax.swing.tree.TreePath - -class UssSortActionHolderTestSpec : ShouldSpec({ - beforeSpec { - // FIXTURE SETUP TO HAVE ACCESS TO APPLICATION INSTANCE - val factory = IdeaTestFixtureFactory.getFixtureFactory() - val projectDescriptor = LightProjectDescriptor.EMPTY_PROJECT_DESCRIPTOR - val fixtureBuilder = factory.createLightFixtureBuilder(projectDescriptor, "for-mainframe") - val fixture = fixtureBuilder.fixture - val myFixture = IdeaTestFixtureFactory.getFixtureFactory().createCodeInsightFixture( - fixture, - LightTempDirTestFixtureImpl(true) - ) - myFixture.setUp() - } - - afterSpec { - clearAllMocks() - } - - context("sort actions") { - - val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl - val mockedFileFetchProvider = mockk() - dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(ApplicationManager.getApplication()) { - @Suppress("UNCHECKED_CAST") - override fun , File : VirtualFile> getFileFetchProvider( - requestClass: Class, - queryClass: Class>, - vFileClass: Class - ): FileFetchProvider { - return mockedFileFetchProvider as FileFetchProvider - } - - } - val mockedActionEvent = mockk() - every { mockedActionEvent.presentation } returns mockk() - every { mockedActionEvent.presentation.putClientProperty(any() as Key, any() as Boolean) } just Runs - // isFromContextMenu is marked as deprecated - every { mockedActionEvent.isFromContextMenu() } returns false - every { mockedActionEvent.presentation.isEnabledAndVisible = true } just Runs - every { mockedActionEvent.presentation.isEnabledAndVisible = false } just Runs - - context("common spec") { - - val mockedFileExplorerView = mockk() - - // Target UssDirNode + Query for test - val mockedMFVirtualFile = mockk() - val mockedUssDirNode = mockk() - val mockedUssRemoteAttributes = mockk() - val mockedUssQuery = mockk>() - every { mockedUssDirNode.virtualFile } returns mockedMFVirtualFile - every { mockedUssDirNode.query } returns mockedUssQuery - every { mockedUssDirNode.cleanCache(false) } just Runs - every { mockedUssQuery.setProperty("requester").value(mockedUssDirNode) } just Runs - every { mockedUssQuery.sortKeys } returns mutableListOf() - - // NodeData for test - val mockedNodeDataForTest = NodeData(mockedUssDirNode, mockedMFVirtualFile, mockedUssRemoteAttributes) - mockkObject(mockedNodeDataForTest) - - // Common config for test - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - - context("sort action group spec") { - - // group action to spy - val mockedActionGroup = spyk(SortActionGroup()) - - val selectionPath = mockk() - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - every { mockedFileExplorerView.myTree } returns mockk() - every { mockedFileExplorerView.myTree.selectionPath } returns selectionPath - - should("is visible from context menu if file explorer view is null") { - var isVisible = true - every { mockedActionEvent.getExplorerView() } answers { - isVisible = false - null - } - - mockedActionGroup.update(mockedActionEvent) - assertSoftly { - isVisible shouldBe false - } - } - - should("is visible from context menu if file explorer view is not null and selected node is not UssDirNode") { - var isVisible = true - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - isVisible = false - listOf(mockedNodeDataNotUssForTest) - } - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - - mockedActionGroup.update(mockedActionEvent) - assertSoftly { - isVisible shouldBe false - } - } - - should("is visible from context menu if file explorer view is not null and selected node is UssDirNode and path is expanded") { - var isVisible = false - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - every { mockedFileExplorerView.myTree.isExpanded(selectionPath) } answers { - isVisible = true - true - } - - mockedActionGroup.update(mockedActionEvent) - assertSoftly { - isVisible shouldBe true - } - } - - should("is visible from context menu if file explorer view is not null and selected node is UssDirNode and path is not expanded") { - var isVisible = true - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - every { mockedFileExplorerView.myTree.isExpanded(selectionPath) } answers { - isVisible = false - false - } - - mockedActionGroup.update(mockedActionEvent) - assertSoftly { - isVisible shouldBe false - } - } - } - - context("sort by name action spec") { - - // action to spy - val mockedSortActionByName = spyk(SortByNameAction()) - - should("sort by name action performed if file explorer is null") { - var actionPerformed = false - every { mockedActionEvent.getExplorerView() } answers { - actionPerformed = true - null - } - mockedSortActionByName.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by name action isSelected if file explorer is null") { - var isSelected = true - every { mockedActionEvent.getExplorerView() } answers { - isSelected = false - null - } - mockedSortActionByName.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort by name action performed if selected node is not UssDirNode") { - var actionPerformed = false - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - actionPerformed = true - listOf(mockedNodeDataNotUssForTest) - } - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - mockedSortActionByName.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by name action isSelected if selected node is not UssDirNode") { - var isSelected = true - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - isSelected = false - listOf(mockedNodeDataNotUssForTest) - } - mockedSortActionByName.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort by name action performed if sort query keys is empty") { - var actionPerformed = false - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf() - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionByName.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by name action isSelected if current sort query keys does not contain NAME key") { - var isSelected = true - every { mockedUssDirNode.currentSortQueryKeysList } answers { - isSelected = false - mutableListOf() - } - mockedSortActionByName.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort by name action performed if sort query keys is not empty") { - var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( - SortQueryKeys.DATE, - SortQueryKeys.ASCENDING - ) - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionByName.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by name action performed if sort query keys is not empty and contains desired key") { - var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( - SortQueryKeys.DATE, - SortQueryKeys.NAME, - SortQueryKeys.ASCENDING - ) - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionByName.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by name action isSelected if current sort query keys contain NAME key") { - var isSelected = false - every { mockedUssDirNode.currentSortQueryKeysList } answers { - isSelected = true - mutableListOf(SortQueryKeys.NAME) - } - mockedSortActionByName.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe true - } - } - - should("sort by name action miscellaneous") { - mockedSortActionByName.setSelected(mockedActionEvent, true) - mockedSortActionByName.setSelected(mockedActionEvent, false) - mockedSortActionByName.update(mockedActionEvent) - val aware = mockedSortActionByName.isDumbAware - assertSoftly { - aware shouldBe true - } - } - } - - context("sort by type action spec") { - - // action to spy - val mockedSortActionByType = spyk(SortByTypeAction()) - - should("sort by type action performed if file explorer is null") { - var actionPerformed = false - every { mockedActionEvent.getExplorerView() } answers { - actionPerformed = true - null - } - mockedSortActionByType.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by type action isSelected if file explorer is null") { - var isSelected = true - every { mockedActionEvent.getExplorerView() } answers { - isSelected = false - null - } - mockedSortActionByType.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort by type action performed if selected node is not UssDirNode") { - var actionPerformed = false - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - actionPerformed = true - listOf(mockedNodeDataNotUssForTest) - } - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - mockedSortActionByType.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by type action isSelected if selected node is not UssDirNode") { - var isSelected = true - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - isSelected = false - listOf(mockedNodeDataNotUssForTest) - } - mockedSortActionByType.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort by type action performed if sort query keys is empty") { - var actionPerformed = false - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf() - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionByType.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by type action isSelected if current sort query keys does not contain TYPE key") { - var isSelected = true - every { mockedUssDirNode.currentSortQueryKeysList } answers { - isSelected = false - mutableListOf() - } - mockedSortActionByType.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort by type action performed if sort query keys is not empty") { - var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( - SortQueryKeys.DATE, - SortQueryKeys.ASCENDING - ) - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionByType.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by type action performed if sort query keys is not empty and contains desired key") { - var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( - SortQueryKeys.DATE, - SortQueryKeys.TYPE, - SortQueryKeys.ASCENDING - ) - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionByType.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by type action isSelected if current sort query keys contain TYPE key") { - var isSelected = false - every { mockedUssDirNode.currentSortQueryKeysList } answers { - isSelected = true - mutableListOf(SortQueryKeys.TYPE) - } - mockedSortActionByType.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe true - } - } - - should("sort by type action miscellaneous") { - mockedSortActionByType.setSelected(mockedActionEvent, true) - mockedSortActionByType.setSelected(mockedActionEvent, false) - mockedSortActionByType.update(mockedActionEvent) - val aware = mockedSortActionByType.isDumbAware - assertSoftly { - aware shouldBe true - } - } - } - - context("sort by date action spec") { - - // action to spy - val mockedSortActionByDate = spyk(SortByModificationDateAction()) - - should("sort by date action performed if file explorer is null") { - var actionPerformed = false - every { mockedActionEvent.getExplorerView() } answers { - actionPerformed = true - null - } - mockedSortActionByDate.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by date action isSelected if file explorer is null") { - var isSelected = true - every { mockedActionEvent.getExplorerView() } answers { - isSelected = false - null - } - mockedSortActionByDate.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort by date action performed if selected node is not UssDirNode") { - var actionPerformed = false - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - actionPerformed = true - listOf(mockedNodeDataNotUssForTest) - } - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - mockedSortActionByDate.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by date action isSelected if selected node is not UssDirNode") { - var isSelected = true - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - isSelected = false - listOf(mockedNodeDataNotUssForTest) - } - mockedSortActionByDate.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort by date action performed if sort query keys is empty") { - var actionPerformed = false - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf() - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionByDate.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by date action isSelected if current sort query keys does not contain DATE key") { - var isSelected = true - every { mockedUssDirNode.currentSortQueryKeysList } answers { - isSelected = false - mutableListOf() - } - mockedSortActionByDate.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort by date action performed if sort query keys is not empty") { - var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( - SortQueryKeys.NAME, - SortQueryKeys.ASCENDING - ) - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionByDate.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by date action performed if sort query keys is not empty and contains desired key") { - var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( - SortQueryKeys.DATE, - SortQueryKeys.NAME, - SortQueryKeys.ASCENDING - ) - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionByDate.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort by date action isSelected if current sort query keys contain DATE key") { - var isSelected = false - every { mockedUssDirNode.currentSortQueryKeysList } answers { - isSelected = true - mutableListOf(SortQueryKeys.DATE) - } - mockedSortActionByDate.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe true - } - } - - should("sort by date action miscellaneous") { - mockedSortActionByDate.setSelected(mockedActionEvent, true) - mockedSortActionByDate.setSelected(mockedActionEvent, false) - mockedSortActionByDate.update(mockedActionEvent) - val aware = mockedSortActionByDate.isDumbAware - assertSoftly { - aware shouldBe true - } - } - } - - context("sort ascending action spec") { - - // action to spy - val mockedSortActionAscending = spyk(SortByAscendingOrderAction()) - - should("sort Ascending action performed if file explorer is null") { - var actionPerformed = false - every { mockedActionEvent.getExplorerView() } answers { - actionPerformed = true - null - } - mockedSortActionAscending.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort Ascending action isSelected if file explorer is null") { - var isSelected = true - every { mockedActionEvent.getExplorerView() } answers { - isSelected = false - null - } - mockedSortActionAscending.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort Ascending action performed if selected node is not UssDirNode") { - var actionPerformed = false - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - actionPerformed = true - listOf(mockedNodeDataNotUssForTest) - } - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - mockedSortActionAscending.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort Ascending action isSelected if selected node is not UssDirNode") { - var isSelected = true - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - isSelected = false - listOf(mockedNodeDataNotUssForTest) - } - mockedSortActionAscending.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort Ascending action performed if sort query keys is empty") { - var actionPerformed = false - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf() - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionAscending.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort Ascending action isSelected if current sort query keys does not contain ASCENDING key") { - var isSelected = true - every { mockedUssDirNode.currentSortQueryKeysList } answers { - isSelected = false - mutableListOf() - } - mockedSortActionAscending.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort Ascending action performed if sort query keys is not empty") { - var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( - SortQueryKeys.NAME, - SortQueryKeys.DESCENDING - ) - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionAscending.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort Ascending action performed if sort query keys is not empty and contains desired key") { - var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( - SortQueryKeys.DESCENDING, - SortQueryKeys.NAME, - SortQueryKeys.ASCENDING - ) - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionAscending.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort Ascending action isSelected if current sort query keys contain ASCENDING key") { - var isSelected = false - every { mockedUssDirNode.currentSortQueryKeysList } answers { - isSelected = true - mutableListOf(SortQueryKeys.ASCENDING) - } - mockedSortActionAscending.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe true - } - } - - should("sort Ascending action miscellaneous") { - mockedSortActionAscending.setSelected(mockedActionEvent, true) - mockedSortActionAscending.setSelected(mockedActionEvent, false) - mockedSortActionAscending.update(mockedActionEvent) - val aware = mockedSortActionAscending.isDumbAware - assertSoftly { - aware shouldBe true - } - } - } - - context("sort descending action spec") { - - // action to spy - val mockedSortActionDescending = spyk(SortByDescendingOrderAction()) - - should("sort Descending action performed if file explorer is null") { - var actionPerformed = false - every { mockedActionEvent.getExplorerView() } answers { - actionPerformed = true - null - } - mockedSortActionDescending.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort Descending action isSelected if file explorer is null") { - var isSelected = true - every { mockedActionEvent.getExplorerView() } answers { - isSelected = false - null - } - mockedSortActionDescending.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort Descending action performed if selected node is not UssDirNode") { - var actionPerformed = false - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - actionPerformed = true - listOf(mockedNodeDataNotUssForTest) - } - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - mockedSortActionDescending.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort Descending action isSelected if selected node is not UssDirNode") { - var isSelected = true - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - isSelected = false - listOf(mockedNodeDataNotUssForTest) - } - mockedSortActionDescending.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort Descending action performed if sort query keys is empty") { - var actionPerformed = false - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf() - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionDescending.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort Descending action isSelected if current sort query keys does not contain DESCENDING key") { - var isSelected = true - every { mockedUssDirNode.currentSortQueryKeysList } answers { - isSelected = false - mutableListOf() - } - mockedSortActionDescending.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("sort Descending action performed if sort query keys is not empty") { - var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( - SortQueryKeys.NAME, - SortQueryKeys.ASCENDING - ) - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionDescending.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort Descending action performed if sort query keys is not empty and contains desired key") { - var actionPerformed = false - every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf( - SortQueryKeys.DESCENDING, - SortQueryKeys.NAME, - SortQueryKeys.ASCENDING - ) - every { mockedFileFetchProvider.reload(mockedUssQuery) } answers { - actionPerformed = true - } - - mockedSortActionDescending.actionPerformed(mockedActionEvent) - assertSoftly { - actionPerformed shouldBe true - } - } - - should("sort Descending action isSelected if current sort query keys contain DESCENDING key") { - var isSelected = false - every { mockedUssDirNode.currentSortQueryKeysList } answers { - isSelected = true - mutableListOf(SortQueryKeys.DESCENDING) - } - mockedSortActionDescending.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe true - } - } - - should("sort Descending action miscellaneous") { - mockedSortActionDescending.setSelected(mockedActionEvent, true) - mockedSortActionDescending.setSelected(mockedActionEvent, false) - mockedSortActionDescending.update(mockedActionEvent) - val aware = mockedSortActionDescending.isDumbAware - assertSoftly { - aware shouldBe true - } - } - } - } - } - -}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroupTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroupTestSpec.kt new file mode 100644 index 000000000..115976f38 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroupTestSpec.kt @@ -0,0 +1,145 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.jobs + +import com.intellij.openapi.actionSystem.ActionUpdateThread +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.Presentation +import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.config.ws.JobsFilter +import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes +import eu.ibagroup.formainframe.explorer.ui.* +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.matchers.shouldBe +import io.mockk.* +import javax.swing.tree.TreePath + +class JobsSortActionGroupTestSpec : WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("jobs sort action group spec") { + + // group action to spy + val classUnderTest = spyk(JobsSortActionGroup()) + + val mockedActionEvent = mockk() + val selectionPath = mockk() + val mockedFileExplorerView = mockk() + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { mockedFileExplorerView.myTree } returns mockk() + every { mockedFileExplorerView.myTree.selectionPath } returns selectionPath + + // Presentation + val presentation = Presentation() + every { mockedActionEvent.presentation } returns presentation + + // Target UssDirNode + Query for test + val mockedMFVirtualFile = mockk() + val mockedJesFilterNode = mockk() + val mockedJobRemoteAttributes = mockk() + val mockedJobQuery = mockk>() + every { mockedJesFilterNode.virtualFile } returns mockedMFVirtualFile + every { mockedJesFilterNode.query } returns mockedJobQuery + + // NodeData for test + val mockedNodeDataForTest = NodeData(mockedJesFilterNode, mockedMFVirtualFile, mockedJobRemoteAttributes) + mockkObject(mockedNodeDataForTest) + + should("is visible from context menu if file explorer view is null") { + var isVisible = true + every { mockedActionEvent.getExplorerView() } answers { + isVisible = false + null + } + + classUnderTest.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + + should("is visible from context menu if file explorer view is not null and selected node is not UssDirNode") { + var isVisible = true + val mockedNodeDataNotJesFilterForTest = + NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + isVisible = false + listOf(mockedNodeDataNotJesFilterForTest) + } + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + + classUnderTest.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + + should("is visible from context menu if file explorer view is not null and selected node is UssDirNode and path is expanded") { + var isVisible = false + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedFileExplorerView.myTree.isExpanded(selectionPath) } answers { + isVisible = true + true + } + + classUnderTest.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe true + } + } + + should("is visible from context menu if file explorer view is not null and selected node is UssDirNode and path is not expanded") { + var isVisible = true + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedFileExplorerView.myTree.isExpanded(selectionPath) } answers { + isVisible = false + false + } + + classUnderTest.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + + should("is visible from context menu if file explorer view is not null and selectedNodes size > 1") { + var isVisible = true + every { mockedFileExplorerView.mySelectedNodesData } answers { + isVisible = false + listOf(mockedNodeDataForTest, mockedNodeDataForTest) + } + + classUnderTest.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + + should("return EDT thread_whenGetActionUpdateThread_givenNothing") { + //given + + //when + val thread = classUnderTest.actionUpdateThread + //then + assertSoftly { + thread shouldBe ActionUpdateThread.EDT + } + } + } +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionTestSpec.kt new file mode 100644 index 000000000..75fb2e2ed --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionTestSpec.kt @@ -0,0 +1,226 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.jobs + +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.application.ApplicationManager +import com.intellij.openapi.vfs.VirtualFile +import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.config.ws.JobsFilter +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.Query +import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes +import eu.ibagroup.formainframe.dataops.fetch.FileFetchProvider +import eu.ibagroup.formainframe.dataops.fetch.JobFetchProvider +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.ui.* +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.utils.service +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.assertions.throwables.shouldThrowExactly +import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe +import io.mockk.* + +class JobsSortActionTestSpec : WithApplicationShouldSpec ({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("Jobs sort action") { + + // action to spy + val classUnderTest = spyk(JobsSortAction()) + + val mockedActionEvent = mockk() + val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + val mockedFileFetchProvider = mockk() + dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(ApplicationManager.getApplication()) { + @Suppress("UNCHECKED_CAST") + override fun , File : VirtualFile> getFileFetchProvider( + requestClass: Class, + queryClass: Class>, + vFileClass: Class + ): FileFetchProvider { + return mockedFileFetchProvider as FileFetchProvider + } + + } + + context("common spec") { + + val mockedFileExplorerView = mockk() + + // Target UssDirNode + Query for test + val mockedMFVirtualFile = mockk() + val mockedJesFilterNode = mockk() + val mockedJobRemoteAttributes = mockk() + val mockedJobQuery = mockk>() + every { mockedJesFilterNode.virtualFile } returns mockedMFVirtualFile + every { mockedJesFilterNode.query } returns mockedJobQuery + every { mockedJobQuery.sortKeys } returns mutableListOf() + + // NodeData for test + val mockedNodeDataForTest = NodeData(mockedJesFilterNode, mockedMFVirtualFile, mockedJobRemoteAttributes) + mockkObject(mockedNodeDataForTest) + + // Common config for test + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { mockedJesFilterNode.cleanCache(false) } just Runs + every { mockedFileFetchProvider.reload(any()) } just Runs + every { mockedFileFetchProvider.applyRefreshCacheDate(any(), any(), any()) } just Runs + + context("isSelected") { + + should("returnFalse_whenIsSelected_givenExplorerNull") { + every { mockedActionEvent.getExplorerView() } returns null + val isSelected = classUnderTest.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("returnFalse_whenIsSelected_givenNullTemplateText") { + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { classUnderTest.templateText } returns null + val isSelected = classUnderTest.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("returnFalse_whenIsSelected_givenNotUssDirNode") { + every { classUnderTest.templateText } returns "Job Name" + val mockedNodeDataNotJesFilterForTest = + NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + listOf(mockedNodeDataNotJesFilterForTest) + } + val isSelected = classUnderTest.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("returnFalse_whenIsSelected_givenNodeWithNoKeysSpecified") { + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedJesFilterNode.currentSortQueryKeysList } answers { + mutableListOf() + } + val isSelected = classUnderTest.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("returnTrue_whenIsSelected_givenValidDataAndSortKey") { + every { mockedJesFilterNode.currentSortQueryKeysList } answers { + mutableListOf(SortQueryKeys.JOB_NAME, SortQueryKeys.ASCENDING) + } + val isSelected = classUnderTest.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe true + } + } + } + + context("setSelected") { + + should("return_whenSetSelected_givenExplorerNull") { + // given + clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + every { mockedActionEvent.getExplorerView() } returns null + + // when + classUnderTest.setSelected(mockedActionEvent, true) + + // then + verify { mockedFileFetchProvider wasNot Called } + } + + should("throwException_whenSetSelected_givenNullTemplateText") { + // given + clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { classUnderTest.templateText } returns null + + // when + val exception = shouldThrowExactly { classUnderTest.setSelected(mockedActionEvent, true) } + + // then + assertSoftly { + exception shouldNotBe null + exception.message shouldBe "Sort key for the selected action was not found." + } + } + + should("return_whenSetSelected_givenAlreadySelectedSortKey") { + // + clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + every { classUnderTest.templateText } returns "Job Name" + every { classUnderTest.isSelected(any()) } returns true + + // when + classUnderTest.setSelected(mockedActionEvent, true) + + // then + verify(exactly = 1) { classUnderTest.isSelected(mockedActionEvent) } + verify { mockedFileFetchProvider wasNot Called } + } + + should("return_whenSetSelected_givenNotJesFilterNode") { + // given + clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + every { classUnderTest.isSelected(any()) } returns false + val mockedNodeDataNotJesFilterForTest = + NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataNotJesFilterForTest) + + // when + classUnderTest.setSelected(mockedActionEvent, true) + + // then + verify { mockedFileFetchProvider wasNot Called } + } + + should("callFetchProvider_whenSetSelected_givenValidSortKey") { + // given + clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + every { classUnderTest.isSelected(any()) } returns false + every { classUnderTest.templateText } returns "Job Name" + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedJesFilterNode.currentSortQueryKeysList } answers { + mutableListOf(SortQueryKeys.JOB_ID, SortQueryKeys.ASCENDING) + } + + // when + classUnderTest.setSelected(mockedActionEvent, true) + + // then + verify(exactly = 1) { mockedJesFilterNode.cleanCache(false) } + verify(exactly = 1) { mockedFileFetchProvider.reload(mockedJobQuery) } + verify(exactly = 1) { mockedFileFetchProvider.applyRefreshCacheDate(mockedJobQuery, mockedJesFilterNode, any()) } + } + } + } + } +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroupTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroupTestSpec.kt new file mode 100644 index 000000000..ca23d6c37 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroupTestSpec.kt @@ -0,0 +1,145 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.uss + +import com.intellij.openapi.actionSystem.ActionUpdateThread +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.Presentation +import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes +import eu.ibagroup.formainframe.dataops.fetch.UssQuery +import eu.ibagroup.formainframe.explorer.ui.* +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.matchers.shouldBe +import io.mockk.* +import javax.swing.tree.TreePath + +class UssSortActionGroupTestSpec : WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("uss sort action group spec") { + + // group action to spy + val classUnderTest = spyk(UssSortActionGroup()) + + val mockedActionEvent = mockk() + val selectionPath = mockk() + val mockedFileExplorerView = mockk() + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { mockedFileExplorerView.myTree } returns mockk() + every { mockedFileExplorerView.myTree.selectionPath } returns selectionPath + + // Presentation + val presentation = Presentation() + every { mockedActionEvent.presentation } returns presentation + + // Target UssDirNode + Query for test + val mockedMFVirtualFile = mockk() + val mockedUssDirNode = mockk() + val mockedUssRemoteAttributes = mockk() + val mockedUssQuery = mockk>() + every { mockedUssDirNode.virtualFile } returns mockedMFVirtualFile + every { mockedUssDirNode.query } returns mockedUssQuery + + // NodeData for test + val mockedNodeDataForTest = NodeData(mockedUssDirNode, mockedMFVirtualFile, mockedUssRemoteAttributes) + mockkObject(mockedNodeDataForTest) + + should("is visible from context menu if file explorer view is null") { + var isVisible = true + every { mockedActionEvent.getExplorerView() } answers { + isVisible = false + null + } + + classUnderTest.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + + should("is visible from context menu if file explorer view is not null and selected node is not UssDirNode") { + var isVisible = true + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + isVisible = false + listOf(mockedNodeDataNotUssForTest) + } + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + + classUnderTest.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + + should("is visible from context menu if file explorer view is not null and selected node is UssDirNode and path is expanded") { + var isVisible = false + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedFileExplorerView.myTree.isExpanded(selectionPath) } answers { + isVisible = true + true + } + + classUnderTest.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe true + } + } + + should("is visible from context menu if file explorer view is not null and selected node is UssDirNode and path is not expanded") { + var isVisible = true + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedFileExplorerView.myTree.isExpanded(selectionPath) } answers { + isVisible = false + false + } + + classUnderTest.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + + should("is visible from context menu if file explorer view is not null and selectedNodes size > 1") { + var isVisible = true + every { mockedFileExplorerView.mySelectedNodesData } answers { + isVisible = false + listOf(mockedNodeDataForTest, mockedNodeDataForTest) + } + + classUnderTest.update(mockedActionEvent) + assertSoftly { + isVisible shouldBe false + } + } + + should("return EDT thread_whenGetActionUpdateThread_givenNothing") { + //given + + //when + val thread = classUnderTest.actionUpdateThread + //then + assertSoftly { + thread shouldBe ActionUpdateThread.EDT + } + } + } +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionTestSpec.kt new file mode 100644 index 000000000..b8d3d0e9d --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionTestSpec.kt @@ -0,0 +1,229 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.uss + +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.application.ApplicationManager +import com.intellij.openapi.vfs.VirtualFile +import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.Query +import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes +import eu.ibagroup.formainframe.dataops.fetch.FileFetchProvider +import eu.ibagroup.formainframe.dataops.fetch.UssFileFetchProvider +import eu.ibagroup.formainframe.dataops.fetch.UssQuery +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.ui.FileExplorerView +import eu.ibagroup.formainframe.explorer.ui.LibraryNode +import eu.ibagroup.formainframe.explorer.ui.NodeData +import eu.ibagroup.formainframe.explorer.ui.UssDirNode +import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.utils.service +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.assertions.throwables.shouldThrowExactly +import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe +import io.mockk.* + +class UssSortActionTestSpec : WithApplicationShouldSpec ({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("Jobs sort action") { + + // action to spy + val classUnderTest = spyk(UssSortAction()) + + val mockedActionEvent = mockk() + val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + val mockedFileFetchProvider = mockk() + dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(ApplicationManager.getApplication()) { + @Suppress("UNCHECKED_CAST") + override fun , File : VirtualFile> getFileFetchProvider( + requestClass: Class, + queryClass: Class>, + vFileClass: Class + ): FileFetchProvider { + return mockedFileFetchProvider as FileFetchProvider + } + + } + + context("common spec") { + + val mockedFileExplorerView = mockk() + + // Target UssDirNode + Query for test + val mockedMFVirtualFile = mockk() + val mockedUssDirNode = mockk() + val mockedUssRemoteAttributes = mockk() + val mockedUssQuery = mockk>() + every { mockedUssDirNode.virtualFile } returns mockedMFVirtualFile + every { mockedUssDirNode.query } returns mockedUssQuery + every { mockedUssQuery.sortKeys } returns mutableListOf() + + // NodeData for test + val mockedNodeDataForTest = NodeData(mockedUssDirNode, mockedMFVirtualFile, mockedUssRemoteAttributes) + mockkObject(mockedNodeDataForTest) + + // Common config for test + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { mockedUssDirNode.cleanCache(false) } just Runs + every { mockedFileFetchProvider.reload(any()) } just Runs + every { mockedFileFetchProvider.applyRefreshCacheDate(any(), any(), any()) } just Runs + + context("isSelected") { + + should("returnFalse_whenIsSelected_givenExplorerNull") { + every { mockedActionEvent.getExplorerView() } returns null + val isSelected = classUnderTest.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("returnFalse_whenIsSelected_givenNullTemplateText") { + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { classUnderTest.templateText } returns null + val isSelected = classUnderTest.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("returnFalse_whenIsSelected_givenNotUssDirNode") { + every { classUnderTest.templateText } returns "File Name" + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + listOf(mockedNodeDataNotUssForTest) + } + val isSelected = classUnderTest.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("returnFalse_whenIsSelected_givenNodeWithNoKeysSpecified") { + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedUssDirNode.currentSortQueryKeysList } answers { + mutableListOf() + } + val isSelected = classUnderTest.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe false + } + } + + should("returnTrue_whenIsSelected_givenValidDataAndSortKey") { + every { mockedUssDirNode.currentSortQueryKeysList } answers { + mutableListOf(SortQueryKeys.FILE_NAME, SortQueryKeys.ASCENDING) + } + val isSelected = classUnderTest.isSelected(mockedActionEvent) + assertSoftly { + isSelected shouldBe true + } + } + } + + context("setSelected") { + + should("return_whenSetSelected_givenExplorerNull") { + // given + clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + every { mockedActionEvent.getExplorerView() } returns null + + // when + classUnderTest.setSelected(mockedActionEvent, true) + + // then + verify { mockedFileFetchProvider wasNot Called } + } + + should("throwException_whenSetSelected_givenNullTemplateText") { + // given + clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { classUnderTest.templateText } returns null + + // when + val exception = shouldThrowExactly { classUnderTest.setSelected(mockedActionEvent, true) } + + // then + assertSoftly { + exception shouldNotBe null + exception.message shouldBe "Sort key for the selected action was not found." + } + } + + should("return_whenSetSelected_givenAlreadySelectedSortKey") { + // + clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + every { classUnderTest.templateText } returns "File Name" + every { classUnderTest.isSelected(any()) } returns true + + // when + classUnderTest.setSelected(mockedActionEvent, true) + + // then + verify(exactly = 1) { classUnderTest.isSelected(mockedActionEvent) } + verify { mockedFileFetchProvider wasNot Called } + } + + should("return_whenSetSelected_givenNotUssDirNode") { + // given + clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + every { classUnderTest.isSelected(any()) } returns false + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataNotUssForTest) + + // when + classUnderTest.setSelected(mockedActionEvent, true) + + // then + verify { mockedFileFetchProvider wasNot Called } + } + + should("callFetchProvider_whenSetSelected_givenValidSortKey") { + // given + clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + every { classUnderTest.isSelected(any()) } returns false + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedUssDirNode.currentSortQueryKeysList } answers { + mutableListOf(SortQueryKeys.FILE_TYPE, SortQueryKeys.ASCENDING) + } + + // when + classUnderTest.setSelected(mockedActionEvent, true) + + // then + verify(exactly = 1) { mockedUssDirNode.cleanCache(false) } + verify(exactly = 1) { mockedFileFetchProvider.reload(mockedUssQuery) } + verify(exactly = 1) { mockedFileFetchProvider.applyRefreshCacheDate(mockedUssQuery, mockedUssDirNode, any()) } + } + } + } + } +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/JesFilterNodeTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/JesFilterNodeTestSpec.kt new file mode 100644 index 000000000..780057e43 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/JesFilterNodeTestSpec.kt @@ -0,0 +1,276 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.ui + +import com.intellij.openapi.application.ApplicationManager +import com.intellij.openapi.project.Project +import com.intellij.openapi.vfs.VirtualFile +import eu.ibagroup.formainframe.config.ws.JobsFilter +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.attributes.* +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.JesExplorer +import eu.ibagroup.formainframe.explorer.JesWorkingSetImpl +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.utils.clearOldKeysAndAddNew +import eu.ibagroup.formainframe.utils.service +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.matchers.collections.shouldContainExactly +import io.mockk.* +import org.zowe.kotlinsdk.Job + +class JesFilterNodeTestSpec : WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("explorer module: ui/JesFilterNode") { + + val mockedJobsFilter = mockk() + val mockedProject = mockk() + val mockedParent = mockk() + val mockedWorkingSet = mockk() + val mockedExplorer = mockk() + val mockedExplorerTreeStructure = mockk() + + every { mockedWorkingSet.explorer } returns mockedExplorer + every { mockedExplorerTreeStructure.registerNode(any()) } just Runs + + val classUnderTest = spyk(JesFilterNode(mockedJobsFilter, mockedProject, mockedParent, mockedWorkingSet, mockedExplorerTreeStructure)) + + context("sort children nodes") { + + val virtualFile1 = mockk() + val virtualFile2 = mockk() + val virtualFile3 = mockk() + val jobAttributes1 = mockk() + val jobAttributes2 = mockk() + val jobAttributes3 = mockk() + val jobInfo1 = Job( + "id1", + "name1", + null, + "ARST", + Job.Status.OUTPUT, + Job.JobType.JOB, + null, + null, + "url1", + "filesUrl1", + null, + 1, + "phase1", + listOf(), + null, + null, + null, + null, + "2024-01-02", + "2024-01-02" + ) + val jobInfo2 = Job( + "id2", + "name2", + null, + "DLIS", + Job.Status.ACTIVE, + Job.JobType.JOB, + null, + null, + "url2", + "filesUrl2", + null, + 2, + "phase2", + listOf(), + null, + null, + null, + null, + "2024-01-04", + "2024-01-04" + ) + val jobInfo3 = Job( + "id3", + "name3", + null, + "ZOSMFAD", + Job.Status.INPUT, + Job.JobType.JOB, + null, + null, + "url3", + "filesUrl3", + null, + 3, + "phase3", + listOf(), + null, + null, + null, + null, + "2024-01-01", + "2024-01-06" + ) + val jobNode1 = mockk() + val jobNode2 = mockk() + val jobNode3 = mockk() + every { jobNode1.value } returns virtualFile1 + every { jobNode2.value } returns virtualFile2 + every { jobNode3.value } returns virtualFile3 + every { jobAttributes1.jobInfo } returns jobInfo1 + every { jobAttributes2.jobInfo } returns jobInfo2 + every { jobAttributes3.jobInfo } returns jobInfo3 + + val nodeToAttributesMap = mutableMapOf(Pair(virtualFile1, jobAttributes1), Pair(virtualFile2, jobAttributes2), Pair(virtualFile3, jobAttributes3)) + + val listToSort = listOf(jobNode1, jobNode2, jobNode3) + + val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(ApplicationManager.getApplication()) { + override fun tryToGetAttributes(file: VirtualFile): FileAttributes? { + return nodeToAttributesMap[file] + } + } + + should("return children nodes sorted by JOB NAME in ascending order") { + // given + val sortKeys = listOf(SortQueryKeys.JOB_NAME, SortQueryKeys.ASCENDING) + val expected = listOf(jobNode1, jobNode2, jobNode3) + + // when + val actual = classUnderTest.sortChildrenNodes(listToSort, sortKeys) + + //then + assertSoftly { + actual shouldContainExactly expected + } + } + + should("return children nodes sorted by JOB OWNER in ascending order") { + // given + val sortKeys = listOf(SortQueryKeys.JOB_OWNER, SortQueryKeys.ASCENDING) + val expected = listOf(jobNode1, jobNode2, jobNode3) + + // when + val actual = classUnderTest.sortChildrenNodes(listToSort, sortKeys) + + //then + assertSoftly { + actual shouldContainExactly expected + } + } + + should("return children nodes sorted by JOB STATUS in ascending order") { + // given + val sortKeys = listOf(SortQueryKeys.JOB_STATUS, SortQueryKeys.ASCENDING) + val expected = listOf(jobNode2, jobNode3, jobNode1) + + // when + val actual = classUnderTest.sortChildrenNodes(listToSort, sortKeys) + + //then + assertSoftly { + actual shouldContainExactly expected + } + } + + should("return children nodes sorted by JOB ID in ascending order") { + // given + val sortKeys = listOf(SortQueryKeys.JOB_ID, SortQueryKeys.ASCENDING) + val expected = listOf(jobNode1, jobNode2, jobNode3) + + // when + val actual = classUnderTest.sortChildrenNodes(listToSort, sortKeys) + + //then + assertSoftly { + actual shouldContainExactly expected + } + } + + should("return children nodes sorted by JOB CREATION DATE in ascending order") { + // given + val sortKeys = listOf(SortQueryKeys.JOB_CREATION_DATE, SortQueryKeys.ASCENDING) + val expected = listOf(jobNode3, jobNode1, jobNode2) + + // when + val actual = classUnderTest.sortChildrenNodes(listToSort, sortKeys) + + //then + assertSoftly { + actual shouldContainExactly expected + } + } + + should("return children nodes sorted by JOB COMPLETION DATE in ascending order") { + // given + val sortKeys = listOf(SortQueryKeys.JOB_COMPLETION_DATE, SortQueryKeys.ASCENDING) + val expected = listOf(jobNode1, jobNode2, jobNode3) + + // when + val actual = classUnderTest.sortChildrenNodes(listToSort, sortKeys) + + //then + assertSoftly { + actual shouldContainExactly expected + } + } + + should("return children nodes sorted by invalid sort key") { + // given + val sortKeys = listOf(SortQueryKeys.FILE_NAME, SortQueryKeys.ASCENDING) + val expected = listOf(jobNode1, jobNode2, jobNode3) + + // when + val actual = classUnderTest.sortChildrenNodes(listToSort, sortKeys) + + //then + assertSoftly { + actual shouldContainExactly expected + } + } + + should("return children nodes without sorting if sort keys are null") { + // given + val sortKeys = listOf() + val expected = listOf(jobNode1, jobNode2, jobNode3) + + // when + val actual = classUnderTest.sortChildrenNodes(listToSort, sortKeys) + + //then + assertSoftly { + actual shouldContainExactly expected + } + } + + should("return children nodes sorted by JOB NAME in descending order") { + // given + classUnderTest.currentSortQueryKeysList.clearOldKeysAndAddNew(SortQueryKeys.DESCENDING) + val sortKeys = listOf(SortQueryKeys.JOB_NAME) + val expected = listOf(jobNode3, jobNode2, jobNode1) + + // when + val actual = classUnderTest.sortChildrenNodes(listToSort, sortKeys) + + //then + assertSoftly { + actual shouldContainExactly expected + } + } + } + } +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNodeTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNodeTestSpec.kt index 7348eefb9..2c0b7c802 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNodeTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/UssDirNodeTestSpec.kt @@ -12,40 +12,28 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.ide.util.treeView.AbstractTreeNode import com.intellij.openapi.project.Project -import com.intellij.testFramework.LightProjectDescriptor -import com.intellij.testFramework.fixtures.IdeaTestFixtureFactory -import com.intellij.testFramework.fixtures.impl.LightTempDirTestFixtureImpl import eu.ibagroup.formainframe.config.ws.UssPath import eu.ibagroup.formainframe.dataops.DataOpsManager -import eu.ibagroup.formainframe.dataops.SortQueryKeys import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributesService import eu.ibagroup.formainframe.dataops.getAttributesService +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys import eu.ibagroup.formainframe.explorer.FileExplorer import eu.ibagroup.formainframe.explorer.FilesWorkingSetImpl +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec import eu.ibagroup.formainframe.utils.service import eu.ibagroup.formainframe.vfs.MFVirtualFile import io.kotest.assertions.assertSoftly -import io.kotest.core.spec.style.ShouldSpec import io.kotest.matchers.shouldBe import io.mockk.* -class UssDirNodeTestSpec : ShouldSpec({ - beforeSpec { - // FIXTURE SETUP TO HAVE ACCESS TO APPLICATION INSTANCE - val factory = IdeaTestFixtureFactory.getFixtureFactory() - val projectDescriptor = LightProjectDescriptor.EMPTY_PROJECT_DESCRIPTOR - val fixtureBuilder = factory.createLightFixtureBuilder(projectDescriptor, "for-mainframe") - val fixture = fixtureBuilder.fixture - val myFixture = IdeaTestFixtureFactory.getFixtureFactory().createCodeInsightFixture( - fixture, - LightTempDirTestFixtureImpl(true) - ) - myFixture.setUp() - } +class UssDirNodeTestSpec : WithApplicationShouldSpec({ + afterSpec { clearAllMocks() + unmockkAll() } + context("explorer module: ui/UssDirNode") { val mockedPath = mockk("USS//test_path/ZOSMFAD/test_dir") val mockedProject = mockk() @@ -109,11 +97,11 @@ class UssDirNodeTestSpec : ShouldSpec({ should("sort by name ascending") { - val sortQueryKeys = listOf(SortQueryKeys.NAME, SortQueryKeys.ASCENDING) + val sortQueryKeys = listOf(SortQueryKeys.FILE_NAME, SortQueryKeys.ASCENDING) mockkObject(sortQueryKeys) val expected = listOf(unexpectedNode, mockedUssDirNodeChild1, childNode3, mockedUssDirNodeChild2, childNode4) - val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + val actual = mockedUssDirNode.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) assertSoftly { actual shouldBe expected @@ -122,11 +110,11 @@ class UssDirNodeTestSpec : ShouldSpec({ should("sort by name descending") { - val sortQueryKeys = listOf(SortQueryKeys.NAME, SortQueryKeys.DESCENDING) + val sortQueryKeys = listOf(SortQueryKeys.FILE_NAME, SortQueryKeys.DESCENDING) mockkObject(sortQueryKeys) val expected = listOf(childNode4, mockedUssDirNodeChild2, childNode3, mockedUssDirNodeChild1, unexpectedNode) - val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + val actual = mockedUssDirNode.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) assertSoftly { actual shouldBe expected @@ -135,11 +123,11 @@ class UssDirNodeTestSpec : ShouldSpec({ should("sort by type ascending") { - val sortQueryKeys = listOf(SortQueryKeys.TYPE, SortQueryKeys.ASCENDING) + val sortQueryKeys = listOf(SortQueryKeys.FILE_TYPE, SortQueryKeys.ASCENDING) mockkObject(sortQueryKeys) val expected = listOf(mockedUssDirNodeChild1, mockedUssDirNodeChild2, childNode3, childNode4) - val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + val actual = mockedUssDirNode.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) assertSoftly { actual shouldBe expected @@ -148,11 +136,11 @@ class UssDirNodeTestSpec : ShouldSpec({ should("sort by type descending") { - val sortQueryKeys = listOf(SortQueryKeys.TYPE, SortQueryKeys.DESCENDING) + val sortQueryKeys = listOf(SortQueryKeys.FILE_TYPE, SortQueryKeys.DESCENDING) mockkObject(sortQueryKeys) val expected = listOf(mockedUssDirNodeChild2, mockedUssDirNodeChild1, childNode4, childNode3) - val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + val actual = mockedUssDirNode.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) assertSoftly { actual shouldBe expected @@ -161,11 +149,11 @@ class UssDirNodeTestSpec : ShouldSpec({ should("sort by date ascending") { - val sortQueryKeys = listOf(SortQueryKeys.DATE, SortQueryKeys.ASCENDING) + val sortQueryKeys = listOf(SortQueryKeys.FILE_MODIFICATION_DATE, SortQueryKeys.ASCENDING) mockkObject(sortQueryKeys) val expected = listOf(unexpectedNode, childNode4, childNode3, mockedUssDirNodeChild2, mockedUssDirNodeChild1) - val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + val actual = mockedUssDirNode.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) assertSoftly { actual shouldBe expected @@ -174,27 +162,26 @@ class UssDirNodeTestSpec : ShouldSpec({ should("sort by date descending") { - val sortQueryKeys = listOf(SortQueryKeys.DATE, SortQueryKeys.DESCENDING) + val sortQueryKeys = listOf(SortQueryKeys.FILE_MODIFICATION_DATE, SortQueryKeys.DESCENDING) mockkObject(sortQueryKeys) val expected = listOf(mockedUssDirNodeChild1, mockedUssDirNodeChild2, childNode3, childNode4, unexpectedNode) - val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + val actual = mockedUssDirNode.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) assertSoftly { actual shouldBe expected } } - should("sort by none key") { + should("return unsorted nodes when passing invalid sort key") { - val sortQueryKeys = listOf(SortQueryKeys.NONE) + val sortQueryKeys = listOf(SortQueryKeys.JOB_NAME, SortQueryKeys.DESCENDING) mockkObject(sortQueryKeys) - val expected = listOf(mockedUssDirNodeChild1, mockedUssDirNodeChild2, childNode3, childNode4, unexpectedNode) - val actual = sortChildrenForTest(mockedChildrenNodes, sortQueryKeys).invoke(mockedUssDirNode) + val actual = mockedUssDirNode.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) assertSoftly { - actual shouldBe expected + actual shouldBe mockedChildrenNodes } } } diff --git a/src/test/kotlin/eu/ibagroup/formainframe/utils/UtilsTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/utils/UtilsTestSpec.kt index d8ef18a70..4d8205e2d 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/utils/UtilsTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/utils/UtilsTestSpec.kt @@ -25,6 +25,7 @@ import eu.ibagroup.formainframe.config.ws.JobsFilter import eu.ibagroup.formainframe.config.ws.MaskStateWithWS import eu.ibagroup.formainframe.config.ws.UssPath import eu.ibagroup.formainframe.config.ws.WorkingSetConfig +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys import eu.ibagroup.formainframe.explorer.FilesWorkingSet import eu.ibagroup.formainframe.explorer.ui.NodeData import eu.ibagroup.formainframe.explorer.ui.UssDirNode @@ -33,6 +34,7 @@ import eu.ibagroup.formainframe.vfs.MFVirtualFile import eu.ibagroup.formainframe.vfs.MFVirtualFileSystem import io.kotest.assertions.assertSoftly import io.kotest.core.spec.style.ShouldSpec +import io.kotest.matchers.collections.shouldContainExactly import io.kotest.matchers.longs.shouldBeGreaterThanOrEqual import io.kotest.matchers.shouldBe import io.mockk.every @@ -898,5 +900,57 @@ class UtilsTestSpec : ShouldSpec({ actualString shouldBe expectedString } } + + should("return this list cleared and one new element added, given the input list and another list") { + //given + val receiver = mutableListOf("AAA", "BBB", "CCC") + val another = listOf("DDD") + val expectedList = listOf("DDD") + //when + receiver.clearAndMergeWith(another) + //then + assertSoftly { + receiver shouldContainExactly expectedList + } + } + + should("return this list cleared and new sort key added, given the input list and TYPED sort key to add") { + //given + val receiver = mutableListOf(SortQueryKeys.JOB_NAME, SortQueryKeys.DESCENDING) + val toAdd = SortQueryKeys.JOB_STATUS + val expectedList = listOf(SortQueryKeys.DESCENDING, SortQueryKeys.JOB_STATUS) + //when + receiver.clearOldKeysAndAddNew(toAdd) + //then + assertSoftly { + receiver shouldContainExactly expectedList + } + } + + should("return this list cleared and new sort key added, given the input list and ORDERING sort key to add") { + //given + val receiver = mutableListOf(SortQueryKeys.JOB_NAME, SortQueryKeys.DESCENDING) + val toAdd = SortQueryKeys.ASCENDING + val expectedList = listOf(SortQueryKeys.JOB_NAME, SortQueryKeys.ASCENDING) + //when + receiver.clearOldKeysAndAddNew(toAdd) + //then + assertSoftly { + receiver shouldContainExactly expectedList + } + } + + should("return this without modifications, given the input list and null key to add") { + //given + val receiver = mutableListOf(SortQueryKeys.JOB_NAME, SortQueryKeys.DESCENDING) + val toAdd = null + val expectedList = listOf(SortQueryKeys.JOB_NAME, SortQueryKeys.DESCENDING) + //when + receiver.clearOldKeysAndAddNew(toAdd) + //then + assertSoftly { + receiver shouldContainExactly expectedList + } + } } }) From 1458ac2c132f6ece967fff12293c6419fab4d77c Mon Sep 17 00:00:00 2001 From: Uladzislau Date: Fri, 9 Feb 2024 17:02:23 +0100 Subject: [PATCH 28/34] Added downloadSources property for the IntelliJ Platform SDK Signed-off-by: Uladzislau --- build.gradle.kts | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/build.gradle.kts b/build.gradle.kts index 237f5aa22..18a4edad0 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -77,6 +77,10 @@ dependencies { intellij { version.set("2023.1") + // !Development only! + // downloadSources.set(true) + // In Settings | Advanced Settings enable option Download sources in section Build Tools. Gradle. + // Then invoke Reload All Gradle Projects action from the Gradle tool window. } tasks { From 62fc8de4c0e0d92627e8538756e0c76f90c93d8b Mon Sep 17 00:00:00 2001 From: Arseni Tsikhamirau Date: Tue, 20 Feb 2024 12:03:14 +0100 Subject: [PATCH 29/34] IJMP-1502-Make-unsecured-connections-slightly-prohibited --- .../connect/ui/zosmf/ConnectionDialog.kt | 97 ++++++++++++++++--- 1 file changed, 85 insertions(+), 12 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt index 764fb9910..fbfdc8de0 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/zosmf/ConnectionDialog.kt @@ -17,11 +17,13 @@ import com.intellij.openapi.progress.runBackgroundableTask import com.intellij.openapi.project.Project import com.intellij.openapi.ui.MessageDialogBuilder import com.intellij.openapi.ui.MessageType +import com.intellij.openapi.ui.Messages import com.intellij.openapi.ui.popup.Balloon import com.intellij.openapi.ui.popup.JBPopupFactory import com.intellij.ui.awt.RelativePoint +import com.intellij.ui.components.JBCheckBox +import com.intellij.ui.components.JBTextField import com.intellij.ui.dsl.builder.* -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.message import eu.ibagroup.formainframe.common.ui.DialogMode import eu.ibagroup.formainframe.common.ui.StatefulDialog @@ -34,16 +36,14 @@ import eu.ibagroup.formainframe.dataops.DataOpsManager import eu.ibagroup.formainframe.dataops.operations.* import eu.ibagroup.formainframe.utils.* import eu.ibagroup.formainframe.utils.crudable.Crudable +import eu.ibagroup.formainframe.utils.crudable.find import eu.ibagroup.formainframe.utils.crudable.getAll import org.zowe.kotlinsdk.ChangePassword import org.zowe.kotlinsdk.annotations.ZVersion import java.awt.Component import java.awt.Point import java.util.* -import javax.swing.JCheckBox -import javax.swing.JComponent -import javax.swing.JPasswordField -import javax.swing.JTextField +import javax.swing.* /** Dialog to add a new connection */ class ConnectionDialog( @@ -56,7 +56,11 @@ class ConnectionDialog( * Private field * In case of DialogMode.UPDATE takes the last successful state from crudable, takes default state otherwise */ - private val lastSuccessfulState: ConnectionDialogState = if(state.mode == DialogMode.UPDATE) state.connectionConfig.toDialogState(crudable) else ConnectionDialogState() + private val lastSuccessfulState: ConnectionDialogState = + if(state.mode == DialogMode.UPDATE) crudable.find { it.uuid == state.connectionUuid } + .findAny() + .orElseGet { state.connectionConfig } + .toDialogState(crudable) else ConnectionDialogState() companion object { /** @@ -70,6 +74,20 @@ class ConnectionDialog( initialState.isAllowSsl == state.isAllowSsl } + /** + * Companion function which takes the instance of ConnectionDialog and checks its current state after OK button is pressed. + * @param dialog + * @return returns true if there are violations found, if no violations were found then returns false as a result of validation + */ + private fun validateSecureConnectionUsage(dialog : ConnectionDialog) : Boolean { + val urlToCheck = dialog.urlTextField.text.trim().startsWith("http://", true) + val sslEnabledCheck = dialog.sslCheckbox.isSelected + if (urlToCheck || sslEnabledCheck) { + return dialog.showSelfSignedUsageWarningDialog(dialog.urlTextField, dialog.sslCheckbox) + } + return false + } + /** Show Test connection dialog and test the connection regarding the dialog state. * First the method checks whether connection succeeds for specified user/password. * If connection succeeds then the method automatically fill in z/OS version for this connection. @@ -82,11 +100,20 @@ class ConnectionDialog( project: Project? = null, initialState: ConnectionDialogState ): ConnectionDialogState? { + var connectionDialog = ConnectionDialog(crudable, initialState, project) val initState = initialState.clone() return showUntilDone( initialState = initialState, - factory = { ConnectionDialog(crudable, initialState, project) }, + factory = { connectionDialog }, test = { state -> + + if (validateSecureConnectionUsage(connectionDialog)) { + state.connectionUrl = connectionDialog.urlTextField.text + state.isAllowSsl = connectionDialog.sslCheckbox.isSelected + connectionDialog = ConnectionDialog(crudable, state, project) + return@showUntilDone false + } + val newTestedConnConfig : ConnectionConfig if (initialState.mode == DialogMode.UPDATE) { if (isOnlyConnectionNameChanged(initState, state)) { @@ -162,6 +189,7 @@ class ConnectionDialog( ask(project) } } + connectionDialog = ConnectionDialog(crudable, state, project) addAnyway } else { true @@ -201,7 +229,7 @@ class ConnectionDialog( ) } .focused() - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Connection URL: ") @@ -213,7 +241,7 @@ class ConnectionDialog( validateForBlank(it) ?: validateZosmfUrl(it) } .also { urlTextField = it.component } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Username: ") @@ -226,7 +254,7 @@ class ConnectionDialog( }.onApply { state.username = state.username.uppercase() } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Password: ") @@ -234,13 +262,22 @@ class ConnectionDialog( passField = cell(JPasswordField()) .bindText(state::password) .validationOnApply { validateForBlank(it) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } indent { row { checkBox("Accept self-signed SSL certificates") .bindSelected(state::isAllowSsl) - .also { sslCheckbox = it.component } + .also { + it.component.apply { + addActionListener { + if (this.isSelected) { + showSelfSignedUsageWarningDialog(this) + } + } + sslCheckbox = this + } + } } } if (state.mode == DialogMode.UPDATE) { @@ -378,4 +415,40 @@ class ConnectionDialog( } } + /** + * Function shows the warning dialog if any violations found and resolves them in case "Back to safety" was clicked + * @param components - dialog components which have to be resolved to valid values + * @return result of the pressed button + */ + private fun showSelfSignedUsageWarningDialog(vararg components : Component) : Boolean { + // default return backToSafety + val backToSafety = true + val choice = Messages.showDialog( + project, + "Creating an unsecure connection (HTTP instead of HTTP(s) and/or using self-signed certificates) is not recommended.\n" + + "You do this at your own peril and risk, and we do not bear any responsibility for the possible consequences of using this type of connection.\n" + + "Please contact your system administrator to configure your system to be able to create a secure connection.\n\n" + + "Do you want to proceed anyway?", + "Attempt to create an unsecured connection", + arrayOf( + "Back to safety", + "Proceed" + ), + 0, + AllIcons.General.WarningDialog, + null + ) + return when (choice) { + 0 -> { + components.forEach { + if (it is JBCheckBox) it.isSelected = false + if (it is JBTextField) it.text = it.text.replace("http", "https", true) + } + backToSafety + } + 1 -> !backToSafety + else -> backToSafety + } + } + } From e8f564db5d36d93f1fed3ed619d31bf3f8f017e2 Mon Sep 17 00:00:00 2001 From: Arseni Tsikhamirau Date: Wed, 6 Mar 2024 15:24:27 +0100 Subject: [PATCH 30/34] IJMP-1529-Datasets-sorting --- .../dataops/BatchedRemoteQuery.kt | 14 +- .../dataops/UnitRemoteQueryImpl.kt | 1 - .../dataops/sort/SortQueryKeys.kt | 9 +- .../explorer/actions/sort/SortAction.kt | 71 ++++- .../explorer/actions/sort/SortActionGroup.kt | 59 ++++ .../sort/datasets/DatasetsSortAction.kt | 40 +++ .../sort/datasets/DatasetsSortActionGroup.kt | 29 ++ .../actions/sort/jobs/JobsSortAction.kt | 65 ++--- .../actions/sort/jobs/JobsSortActionGroup.kt | 36 +-- .../actions/sort/members/MembersSortAction.kt | 43 +++ .../sort/members/MembersSortActionGroup.kt | 29 ++ .../actions/sort/uss/UssSortAction.kt | 65 ++--- .../actions/sort/uss/UssSortActionGroup.kt | 35 +-- .../formainframe/explorer/ui/DSMaskNode.kt | 77 ++++- .../formainframe/explorer/ui/LibraryNode.kt | 64 ++++- src/main/resources/META-INF/plugin.xml | 127 +++++++-- .../DatasetsSortActionGroupTestSpec.kt | 68 +++++ .../datasets/DatasetsSortActionTestSpec.kt | 126 +++++++++ .../sort/jobs/JobsSortActionGroupTestSpec.kt | 40 +++ .../sort/jobs/JobsSortActionTestSpec.kt | 263 ++++++------------ .../members/MembersSortActionGroupTestSpec.kt | 66 +++++ .../sort/members/MembersSortActionTestSpec.kt | 125 +++++++++ .../sort/uss/UssSortActionGroupTestSpec.kt | 40 +++ .../actions/sort/uss/UssSortActionTestSpec.kt | 210 +++++++++----- .../explorer/ui/DSMaskNodeTestSpec.kt | 205 ++++++++++++++ .../explorer/ui/LibraryNodeTestSpec.kt | 175 ++++++++++++ 26 files changed, 1644 insertions(+), 438 deletions(-) create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortActionGroup.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortAction.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionGroup.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortAction.kt create mode 100644 src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionGroup.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionGroupTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionGroupTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNodeTestSpec.kt create mode 100644 src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/LibraryNodeTestSpec.kt diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/BatchedRemoteQuery.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/BatchedRemoteQuery.kt index b0cd7e6c2..a6e0dc2fe 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/BatchedRemoteQuery.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/BatchedRemoteQuery.kt @@ -11,6 +11,11 @@ package eu.ibagroup.formainframe.dataops import eu.ibagroup.formainframe.config.connect.ConnectionConfig +import eu.ibagroup.formainframe.config.ws.DSMask +import eu.ibagroup.formainframe.config.ws.JobsFilter +import eu.ibagroup.formainframe.dataops.fetch.LibraryQuery +import eu.ibagroup.formainframe.dataops.fetch.UssQuery +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys import eu.ibagroup.formainframe.utils.UNIT_CLASS /** @@ -30,10 +35,17 @@ class BatchedRemoteQuery( var alreadyFetched: Int = 0, var start: String? = null, var fetchNeeded: Boolean = true -): RemoteQuery { +): RemoteQuery, SortableQuery { override val resultClass: Class get() = UNIT_CLASS + override val sortKeys: List + get() = when(request) { + is DSMask -> mutableListOf(SortQueryKeys.DATASET_MODIFICATION_DATE, SortQueryKeys.ASCENDING) + is LibraryQuery -> mutableListOf(SortQueryKeys.MEMBER_MODIFICATION_DATE, SortQueryKeys.ASCENDING) + else -> mutableListOf() + } + /** * Sets default values for params that identify current state of fetching. */ diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt index 23d215ddf..172969fb0 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/UnitRemoteQueryImpl.kt @@ -30,7 +30,6 @@ data class UnitRemoteQueryImpl( get() = when(request) { is UssQuery -> mutableListOf(SortQueryKeys.FILE_MODIFICATION_DATE, SortQueryKeys.ASCENDING) is JobsFilter -> mutableListOf(SortQueryKeys.JOB_CREATION_DATE, SortQueryKeys.ASCENDING) - // TODO: Add sort query keys for other queries when implemented else -> mutableListOf() } } \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/dataops/sort/SortQueryKeys.kt b/src/main/kotlin/eu/ibagroup/formainframe/dataops/sort/SortQueryKeys.kt index 181011592..7a10b1fdf 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/dataops/sort/SortQueryKeys.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/dataops/sort/SortQueryKeys.kt @@ -16,6 +16,11 @@ enum class SortQueryKeys(private val sortType: String) { FILE_NAME("uss_file_name"), FILE_TYPE("uss_type"), FILE_MODIFICATION_DATE("uss_modification_date"), + DATASET_NAME("dataset_name"), + DATASET_TYPE("dataset_type"), + DATASET_MODIFICATION_DATE("dataset_modification_date"), + MEMBER_NAME("member_name"), + MEMBER_MODIFICATION_DATE("member_modification_date"), JOB_NAME("Job Name"), JOB_CREATION_DATE("Job Creation Date"), JOB_COMPLETION_DATE("Job Completion Date"), @@ -31,7 +36,9 @@ enum class SortQueryKeys(private val sortType: String) { } val typedSortKeys : List by lazy { - return@lazy listOf(SortQueryKeys.FILE_NAME, SortQueryKeys.FILE_TYPE, SortQueryKeys.FILE_MODIFICATION_DATE, SortQueryKeys.JOB_NAME, SortQueryKeys.JOB_ID, SortQueryKeys.JOB_OWNER, SortQueryKeys.JOB_STATUS, SortQueryKeys.JOB_CREATION_DATE, SortQueryKeys.JOB_COMPLETION_DATE) + return@lazy listOf(SortQueryKeys.FILE_NAME, SortQueryKeys.FILE_TYPE, SortQueryKeys.FILE_MODIFICATION_DATE, SortQueryKeys.JOB_NAME, + SortQueryKeys.JOB_ID, SortQueryKeys.JOB_OWNER, SortQueryKeys.JOB_STATUS, SortQueryKeys.JOB_CREATION_DATE, SortQueryKeys.JOB_COMPLETION_DATE, + SortQueryKeys.DATASET_NAME, SortQueryKeys.DATASET_TYPE, SortQueryKeys.DATASET_MODIFICATION_DATE, SortQueryKeys.MEMBER_NAME, SortQueryKeys.MEMBER_MODIFICATION_DATE) } val orderingSortKeys : List by lazy { diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortAction.kt index ebd3db6bb..5d51fe4d2 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortAction.kt @@ -10,11 +10,47 @@ package eu.ibagroup.formainframe.explorer.actions.sort -import com.intellij.openapi.actionSystem.ActionUpdateThread -import com.intellij.openapi.actionSystem.AnActionEvent -import com.intellij.openapi.actionSystem.ToggleAction +import com.intellij.openapi.actionSystem.* +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.ui.* -abstract class SortAction : ToggleAction() { +abstract class SortAction> : ToggleAction() { + + companion object { + fun runRefreshAction(e: AnActionEvent) { + val refreshActionInstance: AnAction = ActionManager.getInstance().getAction("eu.ibagroup.formainframe.explorer.actions.RefreshNodeAction") + refreshActionInstance.actionPerformed(e) + } + } + + /** + * Function gets the source view where the action is triggered + * @param e + * @return an instance of ExplorerTreeView + */ + abstract fun getSourceView(e: AnActionEvent) : ExplorerTreeView<*, *, *>? + + /** + * Function gets the source node on which the action is triggered + * @param view + * @return an instance of Node + */ + abstract fun getSourceNode(view: ExplorerTreeView<*, *, *>) : Node? + + /** + * Function performs the query update for the selected node and adds/removes the selected sort key + * @param selectedNode + * @param sortKey + */ + abstract fun performQueryUpdateForNode(selectedNode: Node, sortKey: SortQueryKeys) + + /** + * Function checks if the sort key is currently enabled for the particular node + * @param selectedNode + * @param sortKey + * @return true if the sort key is currently enabled, false otherwise + */ + abstract fun shouldEnableSortKeyForNode(selectedNode: Node, sortKey: SortQueryKeys) : Boolean override fun update(e: AnActionEvent) { super.update(e) @@ -34,4 +70,31 @@ abstract class SortAction : ToggleAction() { override fun getActionUpdateThread(): ActionUpdateThread { return ActionUpdateThread.EDT } + + /** + * Action performed method to register the custom behavior when any Sort Key was clicked in UI + */ + override fun setSelected(e: AnActionEvent, state: Boolean) { + val view = getSourceView(e) ?: return + val selectedNode = getSourceNode(view) ?: return + val sortKey = this.templateText?.uppercase()?.replace(" ", "_")?.let { SortQueryKeys.valueOf(it) } + ?: throw Exception("Sort key for the selected action was not found.") + if (isSelected(e)) return + performQueryUpdateForNode(selectedNode, sortKey) + + // Create an instance of ActionEvent from the received sort action and trigger refresh action + val actionEvent = AnActionEvent.createFromDataContext(e.place, null, e.dataContext) + runRefreshAction(actionEvent) + } + + /** + * Custom isSelected method determines if the Sort Key is currently enabled or not. Updates UI by 'tick' mark + */ + override fun isSelected(e: AnActionEvent): Boolean { + val view = getSourceView(e) ?: return false + val selectedNode = getSourceNode(view) ?: return false + val sortKey = this.templateText?.uppercase()?.replace(" ", "_")?.let { SortQueryKeys.valueOf(it) } ?: return false + return shouldEnableSortKeyForNode(selectedNode, sortKey) + } + } \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortActionGroup.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortActionGroup.kt new file mode 100644 index 000000000..83d6ad0fd --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/SortActionGroup.kt @@ -0,0 +1,59 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort + +import com.intellij.openapi.actionSystem.ActionUpdateThread +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DefaultActionGroup +import eu.ibagroup.formainframe.explorer.ui.* + +/** + * Abstract class represents the custom sort action group in the FileExplorerView/JesExplorerView context menu + */ +abstract class SortActionGroup : DefaultActionGroup() { + + /** + * Function gets the source view where the action is triggered + * @param e + * @return an instance of ExplorerTreeView + */ + abstract fun getSourceView(e: AnActionEvent) : ExplorerTreeView<*, *, *>? + + /** + * Function checks if the selected node is suitable to display the Sort actions in the ContextMenu group + * @param node + * @returns true if node is a candidate, false otherwise + */ + abstract fun checkNode(node: ExplorerTreeNode<*, *>) : Boolean + + /** + * Update method to determine if sorting is possible for particular item in the tree + */ + override fun update(e: AnActionEvent) { + val view = getSourceView(e) + view ?: let { + e.presentation.isEnabledAndVisible = false + return + } + val selectedNodes = view.mySelectedNodesData + val treePathFromModel = view.myTree.selectionPath + e.presentation.apply { + isEnabledAndVisible = selectedNodes.size == 1 && view.myTree.isExpanded(treePathFromModel) && checkNode(selectedNodes[0].node) + } + } + + /** + * Tells that only UI component is affected + */ + override fun getActionUpdateThread(): ActionUpdateThread { + return ActionUpdateThread.EDT + } +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortAction.kt new file mode 100644 index 000000000..ce195080a --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortAction.kt @@ -0,0 +1,40 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.datasets + +import com.intellij.openapi.actionSystem.AnActionEvent +import eu.ibagroup.formainframe.dataops.BatchedRemoteQuery +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.actions.sort.SortAction +import eu.ibagroup.formainframe.explorer.ui.* +import eu.ibagroup.formainframe.utils.castOrNull +import eu.ibagroup.formainframe.utils.clearAndMergeWith +import eu.ibagroup.formainframe.utils.clearOldKeysAndAddNew + +class DatasetsSortAction : SortAction() { + override fun getSourceView(e: AnActionEvent): FileExplorerView? { + return e.getExplorerView() + } + + override fun getSourceNode(view: ExplorerTreeView<*, *, *>): DSMaskNode? { + return view.mySelectedNodesData[0].node.castOrNull() + } + + override fun shouldEnableSortKeyForNode(selectedNode: DSMaskNode, sortKey: SortQueryKeys): Boolean { + return selectedNode.currentSortQueryKeysList.contains(sortKey) + } + + override fun performQueryUpdateForNode(selectedNode: DSMaskNode, sortKey: SortQueryKeys) { + val queryToUpdate = selectedNode.query as BatchedRemoteQuery + selectedNode.currentSortQueryKeysList.clearOldKeysAndAddNew(sortKey) + queryToUpdate.sortKeys.clearAndMergeWith(selectedNode.currentSortQueryKeysList) + } +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionGroup.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionGroup.kt new file mode 100644 index 000000000..c30dbea20 --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionGroup.kt @@ -0,0 +1,29 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.datasets + +import com.intellij.openapi.actionSystem.AnActionEvent +import eu.ibagroup.formainframe.explorer.actions.sort.SortActionGroup +import eu.ibagroup.formainframe.explorer.ui.* + +/** + * Represents the custom PS/PDS/PDSe/GDG files sort action group in the FileExplorerView context menu + */ +class DatasetsSortActionGroup : SortActionGroup() { + override fun getSourceView(e: AnActionEvent): FileExplorerView? { + return e.getExplorerView() + } + + override fun checkNode(node: ExplorerTreeNode<*, *>): Boolean { + return node is DSMaskNode + } + +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortAction.kt index 3e8cf83dd..40aad0cda 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortAction.kt @@ -11,65 +11,34 @@ package eu.ibagroup.formainframe.explorer.actions.sort.jobs import com.intellij.openapi.actionSystem.AnActionEvent -import com.intellij.openapi.components.service -import eu.ibagroup.formainframe.config.ws.JobsFilter -import eu.ibagroup.formainframe.dataops.DataOpsManager import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys import eu.ibagroup.formainframe.explorer.actions.sort.SortAction +import eu.ibagroup.formainframe.explorer.ui.ExplorerTreeView import eu.ibagroup.formainframe.explorer.ui.JesExplorerView import eu.ibagroup.formainframe.explorer.ui.JesFilterNode import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.utils.castOrNull import eu.ibagroup.formainframe.utils.clearAndMergeWith import eu.ibagroup.formainframe.utils.clearOldKeysAndAddNew -import eu.ibagroup.formainframe.utils.runWriteActionInEdt -import eu.ibagroup.formainframe.vfs.MFVirtualFile -import java.time.LocalDateTime -/** - * Represents internal Jobs fetch provider to be able to update the query for each Job Filter node whose sorting is enabled - */ -internal val fetchJobsProvider = service() - .getFileFetchProvider( - JobsFilter::class.java, - UnitRemoteQueryImpl::class.java, - MFVirtualFile::class.java - ) -class JobsSortAction : SortAction() { +class JobsSortAction : SortAction() { + override fun getSourceView(e: AnActionEvent): JesExplorerView? { + return e.getExplorerView() + } - /** - * Action performed method to register the custom behavior when any Jobs Sort Key was clicked in UI - */ - override fun setSelected(e: AnActionEvent, state: Boolean) { - val view = e.getExplorerView() ?: return - val sortJobKey = this.templateText?.uppercase()?.replace(" ", "_")?.let { SortQueryKeys.valueOf(it) } - ?: throw Exception("Sort key for the selected action was not found.") - if (isSelected(e)) return - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is JesFilterNode) { - val queryToUpdate = selectedNode.query as UnitRemoteQueryImpl - selectedNode.currentSortQueryKeysList.clearOldKeysAndAddNew(sortJobKey) - queryToUpdate.sortKeys.clearAndMergeWith(selectedNode.currentSortQueryKeysList) - runWriteActionInEdt { - selectedNode.cleanCache(false) - fetchJobsProvider.apply { - reload(queryToUpdate) - applyRefreshCacheDate(queryToUpdate, selectedNode, LocalDateTime.now()) - } - } - } + override fun getSourceNode(view: ExplorerTreeView<*, *, *>): JesFilterNode? { + return view.mySelectedNodesData[0].node.castOrNull() } - /** - * Custom isSelected method determines if the Jobs Sort Key is currently enabled or not. Updates UI by 'tick' mark - */ - override fun isSelected(e: AnActionEvent): Boolean { - val view = e.getExplorerView() ?: return false - val sortJobKey = this.templateText?.uppercase()?.replace(" ", "_")?.let { SortQueryKeys.valueOf(it) } ?: return false - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is JesFilterNode) { - return selectedNode.currentSortQueryKeysList.contains(sortJobKey) - } - return false + override fun shouldEnableSortKeyForNode(selectedNode: JesFilterNode, sortKey: SortQueryKeys): Boolean { + return selectedNode.currentSortQueryKeysList.contains(sortKey) } + + override fun performQueryUpdateForNode(selectedNode: JesFilterNode, sortKey: SortQueryKeys) { + val queryToUpdate = selectedNode.query as UnitRemoteQueryImpl + selectedNode.currentSortQueryKeysList.clearOldKeysAndAddNew(sortKey) + queryToUpdate.sortKeys.clearAndMergeWith(selectedNode.currentSortQueryKeysList) + } + } \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroup.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroup.kt index 8de610c11..aca4e9856 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroup.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroup.kt @@ -10,40 +10,20 @@ package eu.ibagroup.formainframe.explorer.actions.sort.jobs -import com.intellij.openapi.actionSystem.ActionUpdateThread import com.intellij.openapi.actionSystem.AnActionEvent -import com.intellij.openapi.actionSystem.DefaultActionGroup -import eu.ibagroup.formainframe.explorer.ui.JesExplorerView -import eu.ibagroup.formainframe.explorer.ui.JesFilterNode -import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.explorer.actions.sort.SortActionGroup +import eu.ibagroup.formainframe.explorer.ui.* /** * Represents the custom Jobs sort action group in the JesExplorerView context menu */ -class JobsSortActionGroup : DefaultActionGroup() { - - /** - * Update method to determine if sorting is possible for particular item in the tree - */ - override fun update(e: AnActionEvent) { - val view = e.getExplorerView() - view ?: let { - e.presentation.isEnabledAndVisible = false - return - } - val selectedNodes = view.mySelectedNodesData - val treePathFromModel = view.myTree.selectionPath - e.presentation.apply { - isEnabledAndVisible = selectedNodes.size == 1 && selectedNodes.any { - it.node is JesFilterNode && view.myTree.isExpanded(treePathFromModel) - } - } +class JobsSortActionGroup : SortActionGroup() { + override fun getSourceView(e: AnActionEvent): JesExplorerView? { + return e.getExplorerView() } - /** - * Tells that only UI component is affected - */ - override fun getActionUpdateThread(): ActionUpdateThread { - return ActionUpdateThread.EDT + override fun checkNode(node: ExplorerTreeNode<*, *>): Boolean { + return node is JesFilterNode } + } \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortAction.kt new file mode 100644 index 000000000..b1d0f0ade --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortAction.kt @@ -0,0 +1,43 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.members + +import com.intellij.openapi.actionSystem.AnActionEvent +import eu.ibagroup.formainframe.dataops.BatchedRemoteQuery +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.actions.sort.SortAction +import eu.ibagroup.formainframe.explorer.ui.ExplorerTreeView +import eu.ibagroup.formainframe.explorer.ui.FileExplorerView +import eu.ibagroup.formainframe.explorer.ui.LibraryNode +import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.utils.castOrNull +import eu.ibagroup.formainframe.utils.clearAndMergeWith +import eu.ibagroup.formainframe.utils.clearOldKeysAndAddNew + +class MembersSortAction: SortAction() { + override fun getSourceView(e: AnActionEvent): FileExplorerView? { + return e.getExplorerView() + } + + override fun getSourceNode(view: ExplorerTreeView<*, *, *>): LibraryNode? { + return view.mySelectedNodesData[0].node.castOrNull() + } + + override fun shouldEnableSortKeyForNode(selectedNode: LibraryNode, sortKey: SortQueryKeys): Boolean { + return selectedNode.currentSortQueryKeysList.contains(sortKey) + } + + override fun performQueryUpdateForNode(selectedNode: LibraryNode, sortKey: SortQueryKeys) { + val queryToUpdate = selectedNode.query as BatchedRemoteQuery + selectedNode.currentSortQueryKeysList.clearOldKeysAndAddNew(sortKey) + queryToUpdate.sortKeys.clearAndMergeWith(selectedNode.currentSortQueryKeysList) + } +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionGroup.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionGroup.kt new file mode 100644 index 000000000..e2c9bd222 --- /dev/null +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionGroup.kt @@ -0,0 +1,29 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.actions.sort.members + +import com.intellij.openapi.actionSystem.AnActionEvent +import eu.ibagroup.formainframe.explorer.actions.sort.SortActionGroup +import eu.ibagroup.formainframe.explorer.ui.* + +/** + * Represents the custom members sort action group in the FileExplorerView context menu + */ +class MembersSortActionGroup : SortActionGroup() { + override fun getSourceView(e: AnActionEvent): FileExplorerView? { + return e.getExplorerView() + } + + override fun checkNode(node: ExplorerTreeNode<*, *>): Boolean { + return node is LibraryNode + } + +} \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortAction.kt index 6b66e6db2..496201770 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortAction.kt @@ -11,65 +11,34 @@ package eu.ibagroup.formainframe.explorer.actions.sort.uss import com.intellij.openapi.actionSystem.AnActionEvent -import com.intellij.openapi.components.service -import eu.ibagroup.formainframe.dataops.DataOpsManager import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl -import eu.ibagroup.formainframe.dataops.fetch.UssQuery import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys import eu.ibagroup.formainframe.explorer.actions.sort.SortAction +import eu.ibagroup.formainframe.explorer.ui.ExplorerTreeView import eu.ibagroup.formainframe.explorer.ui.FileExplorerView import eu.ibagroup.formainframe.explorer.ui.UssDirNode import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.utils.castOrNull import eu.ibagroup.formainframe.utils.clearAndMergeWith import eu.ibagroup.formainframe.utils.clearOldKeysAndAddNew -import eu.ibagroup.formainframe.utils.runWriteActionInEdt -import eu.ibagroup.formainframe.vfs.MFVirtualFile -import java.time.LocalDateTime -/** - * Represents internal Jobs fetch provider to be able to update the query for each Job Filter node whose sorting is enabled - */ -internal val fetchUssProvider = service() - .getFileFetchProvider( - UssQuery::class.java, - UnitRemoteQueryImpl::class.java, - MFVirtualFile::class.java - ) -class UssSortAction : SortAction() { +class UssSortAction : SortAction() { + override fun getSourceView(e: AnActionEvent): FileExplorerView? { + return e.getExplorerView() + } - /** - * Action performed method to register the custom behavior when any USS Sort Key was clicked in UI - */ - override fun setSelected(e: AnActionEvent, state: Boolean) { - val view = e.getExplorerView() ?: return - val sortUssKey = this.templateText?.uppercase()?.replace(" ", "_")?.let { SortQueryKeys.valueOf(it) } - ?: throw Exception("Sort key for the selected action was not found.") - if (isSelected(e)) return - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - val queryToUpdate = selectedNode.query as UnitRemoteQueryImpl - selectedNode.currentSortQueryKeysList.clearOldKeysAndAddNew(sortUssKey) - queryToUpdate.sortKeys.clearAndMergeWith(selectedNode.currentSortQueryKeysList) - runWriteActionInEdt { - selectedNode.cleanCache(false) - fetchUssProvider.apply { - reload(queryToUpdate) - applyRefreshCacheDate(queryToUpdate, selectedNode, LocalDateTime.now()) - } - } - } + override fun getSourceNode(view: ExplorerTreeView<*, *, *>): UssDirNode? { + return view.mySelectedNodesData[0].node.castOrNull() } - /** - * Custom isSelected method determines if the USS Sort Key is currently enabled or not. Updates UI by 'tick' mark - */ - override fun isSelected(e: AnActionEvent): Boolean { - val view = e.getExplorerView() ?: return false - val sortUssKey = this.templateText?.uppercase()?.replace(" ", "_")?.let { SortQueryKeys.valueOf(it) } ?: return false - val selectedNode = view.mySelectedNodesData[0].node - if (selectedNode is UssDirNode) { - return selectedNode.currentSortQueryKeysList.contains(sortUssKey) - } - return false + override fun shouldEnableSortKeyForNode(selectedNode: UssDirNode, sortKey: SortQueryKeys): Boolean { + return selectedNode.currentSortQueryKeysList.contains(sortKey) } + + override fun performQueryUpdateForNode(selectedNode: UssDirNode, sortKey: SortQueryKeys) { + val queryToUpdate = selectedNode.query as UnitRemoteQueryImpl + selectedNode.currentSortQueryKeysList.clearOldKeysAndAddNew(sortKey) + queryToUpdate.sortKeys.clearAndMergeWith(selectedNode.currentSortQueryKeysList) + } + } \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroup.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroup.kt index f3a63962b..849064519 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroup.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroup.kt @@ -10,40 +10,19 @@ package eu.ibagroup.formainframe.explorer.actions.sort.uss -import com.intellij.openapi.actionSystem.ActionUpdateThread import com.intellij.openapi.actionSystem.AnActionEvent -import com.intellij.openapi.actionSystem.DefaultActionGroup -import eu.ibagroup.formainframe.explorer.ui.FileExplorerView -import eu.ibagroup.formainframe.explorer.ui.UssDirNode -import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.explorer.actions.sort.SortActionGroup +import eu.ibagroup.formainframe.explorer.ui.* /** * Represents the custom USS files sort action group in the FileExplorerView context menu */ -class UssSortActionGroup : DefaultActionGroup() { - - /** - * Update method to determine if sorting is possible for particular item in the tree - */ - override fun update(e: AnActionEvent) { - val view = e.getExplorerView() - view ?: let { - e.presentation.isEnabledAndVisible = false - return - } - val selectedNodes = view.mySelectedNodesData - val treePathFromModel = view.myTree.selectionPath - e.presentation.apply { - isEnabledAndVisible = selectedNodes.size == 1 && selectedNodes.any { - it.node is UssDirNode && view.myTree.isExpanded(treePathFromModel) - } - } +class UssSortActionGroup : SortActionGroup() { + override fun getSourceView(e: AnActionEvent): FileExplorerView? { + return e.getExplorerView() } - /** - * Tells that only UI component is affected - */ - override fun getActionUpdateThread(): ActionUpdateThread { - return ActionUpdateThread.EDT + override fun checkNode(node: ExplorerTreeNode<*, *>): Boolean { + return node is UssDirNode } } \ No newline at end of file diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNode.kt index d9659a7e1..be6b487e2 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNode.kt @@ -12,14 +12,22 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.ide.projectView.PresentationData import com.intellij.ide.util.treeView.AbstractTreeNode +import com.intellij.openapi.components.service import com.intellij.openapi.project.Project import com.intellij.ui.SimpleTextAttributes import com.intellij.util.containers.toMutableSmartList import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.ws.DSMask import eu.ibagroup.formainframe.dataops.BatchedRemoteQuery +import eu.ibagroup.formainframe.dataops.DataOpsManager import eu.ibagroup.formainframe.dataops.RemoteQuery +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteMemberAttributes +import eu.ibagroup.formainframe.dataops.getAttributesService +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.dataops.sort.typedSortKeys import eu.ibagroup.formainframe.explorer.FilesWorkingSet +import eu.ibagroup.formainframe.utils.clearAndMergeWith import eu.ibagroup.formainframe.vfs.MFVirtualFile import icons.ForMainframeIcons @@ -31,10 +39,12 @@ class DSMaskNode( project: Project, parent: ExplorerTreeNode, workingSet: FilesWorkingSet, - treeStructure: ExplorerTreeStructureBase + treeStructure: ExplorerTreeStructureBase, + override val currentSortQueryKeysList: List = mutableListOf(SortQueryKeys.DATASET_MODIFICATION_DATE, SortQueryKeys.ASCENDING), + override val sortedNodes: List> = mutableListOf() ) : RemoteMFFileFetchNode( dsMask, project, parent, workingSet, treeStructure -), RefreshableNode { +), RefreshableNode, SortableNode { override fun update(presentation: PresentationData) { presentation.addText(value.mask, SimpleTextAttributes.REGULAR_ATTRIBUTES) @@ -54,14 +64,14 @@ class DSMaskNode( /** * Returns map of children nodes (datasets and uss files). */ - override fun Collection.toChildrenNodes(): MutableList> { + override fun Collection.toChildrenNodes(): List> { return map { if (it.isDirectory) { LibraryNode(it, notNullProject, this@DSMaskNode, unit, treeStructure) } else { FileLikeDatasetNode(it, notNullProject, this@DSMaskNode, unit, treeStructure) } - }.toMutableSmartList() + }.let { sortChildrenNodes(it, currentSortQueryKeysList) } } override val requestClass = DSMask::class.java @@ -73,4 +83,63 @@ class DSMaskNode( return "Fetching listings for ${query.request.mask}" } + override fun > sortChildrenNodes(childrenNodes: List, sortKeys: List): List { + val listToReturn = mutableListOf() + val psFiles = childrenNodes.filter { it is FileLikeDatasetNode } + val libraries = childrenNodes.filter { it is LibraryNode } + val foundSortKey = sortKeys.firstOrNull { typedSortKeys.contains(it) } + if (foundSortKey != null && foundSortKey == SortQueryKeys.DATASET_TYPE) { + val sortedPSFiles = performDatasetsSorting(psFiles, this@DSMaskNode, SortQueryKeys.DATASET_NAME) + val sortedLibraries = performDatasetsSorting(libraries, this@DSMaskNode, SortQueryKeys.DATASET_NAME) + listToReturn.addAll(sortedLibraries) + listToReturn.addAll(sortedPSFiles) + also { sortedNodes.clearAndMergeWith(listToReturn) } + } else if (foundSortKey != null) { + listToReturn.clearAndMergeWith(performDatasetsSorting(childrenNodes, this@DSMaskNode, foundSortKey)) + } else { + listToReturn.clearAndMergeWith(childrenNodes) + } + return listToReturn + } + + /** + * Function sorts the children nodes by specified sorting key + * @param nodes + * @param mask + * @param sortKey + * @return sorted nodes by specified key + */ + private fun > performDatasetsSorting(nodes: List, mask: DSMaskNode, sortKey: SortQueryKeys) : List { + val sortedNodesInternal: List = if (mask.currentSortQueryKeysList.contains(SortQueryKeys.ASCENDING)) { + nodes.sortedBy { + selector(sortKey).invoke(it) + } + } else { + nodes.sortedByDescending { + selector(sortKey).invoke(it) + } + } + return sortedNodesInternal.also { sortedNodes.clearAndMergeWith(it) } + } + + /** + * Selector which extracts the dataset attributes by specified sort key + * @param key - sort key + * @return String representation of the extracted dataset info of the virtual file + */ + private fun selector(key: SortQueryKeys) : (AbstractTreeNode<*>) -> String? { + return { + val datasetAttributes = when (it) { + is FileLikeDatasetNode -> service().tryToGetAttributes(it.virtualFile) as RemoteDatasetAttributes + is LibraryNode -> service().tryToGetAttributes(it.virtualFile) as RemoteDatasetAttributes + else -> null + } + when (key) { + SortQueryKeys.DATASET_NAME -> datasetAttributes?.datasetInfo?.name + SortQueryKeys.DATASET_MODIFICATION_DATE -> datasetAttributes?.datasetInfo?.lastReferenceDate + else -> null + } + } + } + } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/LibraryNode.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/LibraryNode.kt index d4d12271c..ca2e2ccd0 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/LibraryNode.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/LibraryNode.kt @@ -13,6 +13,7 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.icons.AllIcons import com.intellij.ide.projectView.PresentationData import com.intellij.ide.util.treeView.AbstractTreeNode +import com.intellij.openapi.components.service import com.intellij.openapi.project.Project import com.intellij.ui.SimpleTextAttributes import eu.ibagroup.formainframe.config.connect.ConnectionConfig @@ -20,9 +21,14 @@ import eu.ibagroup.formainframe.dataops.BatchedRemoteQuery import eu.ibagroup.formainframe.dataops.DataOpsManager import eu.ibagroup.formainframe.dataops.RemoteQuery import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteMemberAttributes import eu.ibagroup.formainframe.dataops.fetch.LibraryQuery import eu.ibagroup.formainframe.dataops.getAttributesService +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.dataops.sort.typedSortKeys import eu.ibagroup.formainframe.explorer.FilesWorkingSet +import eu.ibagroup.formainframe.utils.clearAndMergeWith import eu.ibagroup.formainframe.utils.service import eu.ibagroup.formainframe.vfs.MFVirtualFile import icons.ForMainframeIcons @@ -45,10 +51,12 @@ class LibraryNode( project: Project, parent: ExplorerTreeNode, workingSet: FilesWorkingSet, - treeStructure: ExplorerTreeStructureBase + treeStructure: ExplorerTreeStructureBase, + override val currentSortQueryKeysList: List = mutableListOf(SortQueryKeys.MEMBER_MODIFICATION_DATE, SortQueryKeys.ASCENDING), + override val sortedNodes: List> = mutableListOf() ) : RemoteMFFileFetchNode( library, project, parent, workingSet, treeStructure -), RefreshableNode { +), RefreshableNode, SortableNode { override val query: RemoteQuery? get() { @@ -60,7 +68,7 @@ class LibraryNode( } override fun Collection.toChildrenNodes(): List> { - return map { FileLikeDatasetNode(it, notNullProject, this@LibraryNode, unit, treeStructure) } + return sortChildrenNodes(map { FileLikeDatasetNode(it, notNullProject, this@LibraryNode, unit, treeStructure) }, currentSortQueryKeysList) } override val requestClass = LibraryQuery::class.java @@ -68,7 +76,7 @@ class LibraryNode( override fun update(presentation: PresentationData) { presentation.setIcon(if (value.isDirectory) ForMainframeIcons.DatasetMask else AllIcons.FileTypes.Any_type) updateNodeTitleUsingCutBuffer(value.presentableName, presentation) - val dataOpsManager = explorer.componentManager.service() + val dataOpsManager = service() getVolserIfPresent(dataOpsManager, value) ?.let { presentation.addText(it, SimpleTextAttributes.GRAYED_ITALIC_ATTRIBUTES) } } @@ -80,4 +88,52 @@ class LibraryNode( override fun makeFetchTaskTitle(query: RemoteQuery): String { return "Fetching members for ${query.request.library.name}" } + + override fun > sortChildrenNodes(childrenNodes: List, sortKeys: List): List { + val listToReturn : List = mutableListOf() + val foundSortKey = sortKeys.firstOrNull { typedSortKeys.contains(it) } + if (foundSortKey != null) { + listToReturn.clearAndMergeWith(performMembersSorting(childrenNodes, this@LibraryNode, foundSortKey)) + } else { + listToReturn.clearAndMergeWith(childrenNodes) + } + return listToReturn + } + + /** + * Function sorts the children nodes by specified sorting key + * @param nodes + * @param dataset + * @param sortKey + * @return sorted nodes by specified key + */ + private fun performMembersSorting(nodes: List>, dataset: LibraryNode, sortKey: SortQueryKeys) : List> { + val sortedNodesInternal: List> = if (dataset.currentSortQueryKeysList.contains(SortQueryKeys.ASCENDING)) { + nodes.sortedBy { + selector(sortKey).invoke(it) + } + } else { + nodes.sortedByDescending { + selector(sortKey).invoke(it) + } + } + return sortedNodesInternal.also { sortedNodes.clearAndMergeWith(it) } + } + + /** + * Selector which extracts the member info by specified sort key + * @param key - sort key + * @return String representation of the extracted member info attribute of the virtual file + */ + private fun selector(key: SortQueryKeys) : (AbstractTreeNode<*>) -> String? { + return { + val memberInfo = + (service().tryToGetAttributes((it as FileLikeDatasetNode).virtualFile) as RemoteMemberAttributes).info + when (key) { + SortQueryKeys.MEMBER_NAME -> memberInfo.name + SortQueryKeys.MEMBER_MODIFICATION_DATE -> memberInfo.modificationDate + else -> null + } + } + } } diff --git a/src/main/resources/META-INF/plugin.xml b/src/main/resources/META-INF/plugin.xml index 097796361..c95805eb6 100755 --- a/src/main/resources/META-INF/plugin.xml +++ b/src/main/resources/META-INF/plugin.xml @@ -436,58 +436,98 @@ Example of how to see the output:
class="eu.ibagroup.formainframe.explorer.actions.CreateUssDirectoryAction" text="Directory" icon="AllIcons.Nodes.Folder"/> - - - - + + + + + + + + + + - - - - - - - - + + + + + + + + - @@ -634,33 +674,58 @@ Example of how to see the output:
- - - - + + + + + + + + + + + + + + + + + + + + - - + + - - - - - - - + + + + + + - - + + - + + + @@ -721,7 +788,7 @@ Example of how to see the output:
- + diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionGroupTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionGroupTestSpec.kt new file mode 100644 index 000000000..b4c9f06a9 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionGroupTestSpec.kt @@ -0,0 +1,68 @@ +package eu.ibagroup.formainframe.explorer.actions.sort.datasets + +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DataKey +import eu.ibagroup.formainframe.explorer.actions.sort.members.MembersSortActionGroup +import eu.ibagroup.formainframe.explorer.ui.DSMaskNode +import eu.ibagroup.formainframe.explorer.ui.FileExplorerView +import eu.ibagroup.formainframe.explorer.ui.FileLikeDatasetNode +import eu.ibagroup.formainframe.explorer.ui.LibraryNode +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import io.kotest.assertions.assertSoftly +import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe +import io.mockk.* + +class DatasetsSortActionGroupTestSpec : WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("datasets sort action group spec") { + + val actionEventMock = mockk() + val explorerViewMock = mockk() + // group action to spy + val classUnderTest = spyk(DatasetsSortActionGroup()) + + should("shouldReturnExplorerView_whenGetExplorerView_givenActionEvent") { + every { actionEventMock.getData(any() as DataKey) } returns explorerViewMock + val actualExplorer = classUnderTest.getSourceView(actionEventMock) + + assertSoftly { + actualExplorer shouldNotBe null + actualExplorer is FileExplorerView + } + } + + should("shouldReturnTrue_whenCheckNode_givenDSMaskNode") { + val nodeMock = mockk() + val checkNode = classUnderTest.checkNode(nodeMock) + + assertSoftly { + checkNode shouldBe true + } + } + + should("shouldReturnNull_whenGetExplorerView_givenActionEvent") { + every { actionEventMock.getData(any() as DataKey) } returns null + val actualExplorer = classUnderTest.getSourceView(actionEventMock) + + assertSoftly { + actualExplorer shouldBe null + } + } + + should("shouldReturnFalse_whenCheckNode_givenWrongNode") { + val nodeMock = mockk() + val checkNode = classUnderTest.checkNode(nodeMock) + + assertSoftly { + checkNode shouldBe false + } + } + } + +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionTestSpec.kt new file mode 100644 index 000000000..cb8445e26 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/datasets/DatasetsSortActionTestSpec.kt @@ -0,0 +1,126 @@ +package eu.ibagroup.formainframe.explorer.actions.sort.datasets + +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DataKey +import eu.ibagroup.formainframe.config.ws.DSMask +import eu.ibagroup.formainframe.dataops.BatchedRemoteQuery +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.ui.* +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.matchers.collections.shouldContainExactly +import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe +import io.mockk.* + +class DatasetsSortActionTestSpec : WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("Datasets sort action") { + + val actionEventMock = mockk() + val explorerViewMock = mockk() + // action to spy + val classUnderTest = spyk(DatasetsSortAction()) + + should("returnSourceView_whenGetSourceView_givenActionEvent") { + every { actionEventMock.getData(any() as DataKey) } returns explorerViewMock + + val actualExplorerView = classUnderTest.getSourceView(actionEventMock) + + assertSoftly { + actualExplorerView shouldNotBe null + actualExplorerView is FileExplorerView + } + } + + should("returnNull_whenGetSourceView_givenActionEvent") { + every { actionEventMock.getData(any() as DataKey) } returns null + + val actualExplorerView = classUnderTest.getSourceView(actionEventMock) + + assertSoftly { + actualExplorerView shouldBe null + } + } + + should("returnSourceNode_whenGetSourceNode_givenView") { + val nodeMock = mockk() + val fileMock = mockk() + val attributesMock = mockk() + val myNodesData = mutableListOf(NodeData(nodeMock, fileMock, attributesMock)) + mockkObject(myNodesData) + every { explorerViewMock.mySelectedNodesData } returns myNodesData + + val actualNode = classUnderTest.getSourceNode(explorerViewMock) + + assertSoftly { + actualNode shouldBe nodeMock + } + } + + should("returnNull_whenGetSourceNode_givenView") { + val nodeMock = mockk() + val fileMock = mockk() + val attributesMock = mockk() + val myNodesData = mutableListOf(NodeData(nodeMock, fileMock, attributesMock)) + mockkObject(myNodesData) + every { explorerViewMock.mySelectedNodesData } returns myNodesData + + val actualNode = classUnderTest.getSourceNode(explorerViewMock) + + assertSoftly { + actualNode shouldBe null + } + } + + should("returnTrue_whenShouldEnableSortKeyForNode_givenSelectedNodeAndSortKey") { + val nodeMock = mockk() + val sortKey = SortQueryKeys.DATASET_NAME + every { nodeMock.currentSortQueryKeysList } returns listOf(SortQueryKeys.DATASET_NAME, SortQueryKeys.ASCENDING) + + val shouldEnableSortKey = classUnderTest.shouldEnableSortKeyForNode(nodeMock, sortKey) + + assertSoftly { + shouldEnableSortKey shouldBe true + } + } + + should("returnFalse_whenShouldEnableSortKeyForNode_givenSelectedNodeAndSortKey") { + val nodeMock = mockk() + val sortKey = SortQueryKeys.DATASET_NAME + every { nodeMock.currentSortQueryKeysList } returns listOf() + + val shouldEnableSortKey = classUnderTest.shouldEnableSortKeyForNode(nodeMock, sortKey) + + assertSoftly { + shouldEnableSortKey shouldBe false + } + } + + should("updateQuery_whenPerformQueryUpdateForNode_givenSelectedNodeAndSortKey") { + val batchedQueryMock = mockk>() + val nodeMock = mockk() + val sortKey = SortQueryKeys.DATASET_NAME + val expectedSortKeys = listOf(SortQueryKeys.ASCENDING, SortQueryKeys.DATASET_NAME) + every { nodeMock.query } returns batchedQueryMock + every { batchedQueryMock.sortKeys } returns mutableListOf(SortQueryKeys.ASCENDING, SortQueryKeys.DATASET_MODIFICATION_DATE) + every { nodeMock.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.ASCENDING, SortQueryKeys.DATASET_MODIFICATION_DATE) + + classUnderTest.performQueryUpdateForNode(nodeMock, sortKey) + + assertSoftly { + batchedQueryMock.sortKeys shouldContainExactly expectedSortKeys + } + } + + } + +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroupTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroupTestSpec.kt index 115976f38..3bf664996 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroupTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionGroupTestSpec.kt @@ -12,6 +12,7 @@ package eu.ibagroup.formainframe.explorer.actions.sort.jobs import com.intellij.openapi.actionSystem.ActionUpdateThread import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DataKey import com.intellij.openapi.actionSystem.Presentation import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.ws.JobsFilter @@ -23,6 +24,7 @@ import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec import eu.ibagroup.formainframe.vfs.MFVirtualFile import io.kotest.assertions.assertSoftly import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe import io.mockk.* import javax.swing.tree.TreePath @@ -61,6 +63,43 @@ class JobsSortActionGroupTestSpec : WithApplicationShouldSpec({ val mockedNodeDataForTest = NodeData(mockedJesFilterNode, mockedMFVirtualFile, mockedJobRemoteAttributes) mockkObject(mockedNodeDataForTest) + should("shouldReturnExplorerView_whenGetExplorerView_givenActionEvent") { + every { mockedActionEvent.getData(any() as DataKey) } returns mockedFileExplorerView + val actualExplorer = classUnderTest.getSourceView(mockedActionEvent) + + assertSoftly { + actualExplorer shouldNotBe null + actualExplorer is JesExplorerView + } + } + + should("shouldReturnTrue_whenCheckNode_givenJesFilterNode") { + val nodeMock = mockk() + val checkNode = classUnderTest.checkNode(nodeMock) + + assertSoftly { + checkNode shouldBe true + } + } + + should("shouldReturnNull_whenGetExplorerView_givenActionEvent") { + every { mockedActionEvent.getData(any() as DataKey) } returns null + val actualExplorer = classUnderTest.getSourceView(mockedActionEvent) + + assertSoftly { + actualExplorer shouldBe null + } + } + + should("shouldReturnFalse_whenCheckNode_givenWrongNode") { + val nodeMock = mockk() + val checkNode = classUnderTest.checkNode(nodeMock) + + assertSoftly { + checkNode shouldBe false + } + } + should("is visible from context menu if file explorer view is null") { var isVisible = true every { mockedActionEvent.getExplorerView() } answers { @@ -82,6 +121,7 @@ class JobsSortActionGroupTestSpec : WithApplicationShouldSpec({ isVisible = false listOf(mockedNodeDataNotJesFilterForTest) } + every { mockedFileExplorerView.myTree.isExpanded(selectionPath) } returns true every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView classUnderTest.update(mockedActionEvent) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionTestSpec.kt index 75fb2e2ed..4d0827985 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/jobs/JobsSortActionTestSpec.kt @@ -11,25 +11,18 @@ package eu.ibagroup.formainframe.explorer.actions.sort.jobs import com.intellij.openapi.actionSystem.AnActionEvent -import com.intellij.openapi.application.ApplicationManager -import com.intellij.openapi.vfs.VirtualFile +import com.intellij.openapi.actionSystem.DataKey import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.ws.JobsFilter -import eu.ibagroup.formainframe.dataops.DataOpsManager -import eu.ibagroup.formainframe.dataops.Query import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes -import eu.ibagroup.formainframe.dataops.fetch.FileFetchProvider -import eu.ibagroup.formainframe.dataops.fetch.JobFetchProvider import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys import eu.ibagroup.formainframe.explorer.ui.* import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec -import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl -import eu.ibagroup.formainframe.utils.service import eu.ibagroup.formainframe.vfs.MFVirtualFile import io.kotest.assertions.assertSoftly -import io.kotest.assertions.throwables.shouldThrowExactly +import io.kotest.matchers.collections.shouldContainExactly import io.kotest.matchers.shouldBe import io.kotest.matchers.shouldNotBe import io.mockk.* @@ -43,183 +36,99 @@ class JobsSortActionTestSpec : WithApplicationShouldSpec ({ context("Jobs sort action") { + val actionEventMock = mockk() + val explorerViewMock = mockk() // action to spy val classUnderTest = spyk(JobsSortAction()) - val mockedActionEvent = mockk() - val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl - val mockedFileFetchProvider = mockk() - dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(ApplicationManager.getApplication()) { - @Suppress("UNCHECKED_CAST") - override fun , File : VirtualFile> getFileFetchProvider( - requestClass: Class, - queryClass: Class>, - vFileClass: Class - ): FileFetchProvider { - return mockedFileFetchProvider as FileFetchProvider + should("returnSourceView_whenGetSourceView_givenActionEvent") { + every { actionEventMock.getData(any() as DataKey) } returns explorerViewMock + + val actualExplorerView = classUnderTest.getSourceView(actionEventMock) + + assertSoftly { + actualExplorerView shouldNotBe null + actualExplorerView is JesExplorerView + } + } + + should("returnNull_whenGetSourceView_givenActionEvent") { + every { actionEventMock.getData(any() as DataKey) } returns null + + val actualExplorerView = classUnderTest.getSourceView(actionEventMock) + + assertSoftly { + actualExplorerView shouldBe null + } + } + + should("returnSourceNode_whenGetSourceNode_givenView") { + val nodeMock = mockk() + val fileMock = mockk() + val attributesMock = mockk() + val myNodesData = mutableListOf(NodeData(nodeMock, fileMock, attributesMock)) + mockkObject(myNodesData) + every { explorerViewMock.mySelectedNodesData } returns myNodesData + + val actualNode = classUnderTest.getSourceNode(explorerViewMock) + + assertSoftly { + actualNode shouldBe nodeMock + } + } + + should("returnNull_whenGetSourceNode_givenView") { + val nodeMock = mockk() + val fileMock = mockk() + val attributesMock = mockk() + val myNodesData = mutableListOf(NodeData(nodeMock, fileMock, attributesMock)) + mockkObject(myNodesData) + every { explorerViewMock.mySelectedNodesData } returns myNodesData + + val actualNode = classUnderTest.getSourceNode(explorerViewMock) + + assertSoftly { + actualNode shouldBe null } + } + + should("returnTrue_whenShouldEnableSortKeyForNode_givenSelectedNodeAndSortKey") { + val nodeMock = mockk() + val sortKey = SortQueryKeys.JOB_NAME + every { nodeMock.currentSortQueryKeysList } returns listOf(SortQueryKeys.JOB_NAME, SortQueryKeys.ASCENDING) + + val shouldEnableSortKey = classUnderTest.shouldEnableSortKeyForNode(nodeMock, sortKey) + assertSoftly { + shouldEnableSortKey shouldBe true + } } - context("common spec") { - - val mockedFileExplorerView = mockk() - - // Target UssDirNode + Query for test - val mockedMFVirtualFile = mockk() - val mockedJesFilterNode = mockk() - val mockedJobRemoteAttributes = mockk() - val mockedJobQuery = mockk>() - every { mockedJesFilterNode.virtualFile } returns mockedMFVirtualFile - every { mockedJesFilterNode.query } returns mockedJobQuery - every { mockedJobQuery.sortKeys } returns mutableListOf() - - // NodeData for test - val mockedNodeDataForTest = NodeData(mockedJesFilterNode, mockedMFVirtualFile, mockedJobRemoteAttributes) - mockkObject(mockedNodeDataForTest) - - // Common config for test - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - every { mockedJesFilterNode.cleanCache(false) } just Runs - every { mockedFileFetchProvider.reload(any()) } just Runs - every { mockedFileFetchProvider.applyRefreshCacheDate(any(), any(), any()) } just Runs - - context("isSelected") { - - should("returnFalse_whenIsSelected_givenExplorerNull") { - every { mockedActionEvent.getExplorerView() } returns null - val isSelected = classUnderTest.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("returnFalse_whenIsSelected_givenNullTemplateText") { - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - every { classUnderTest.templateText } returns null - val isSelected = classUnderTest.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("returnFalse_whenIsSelected_givenNotUssDirNode") { - every { classUnderTest.templateText } returns "Job Name" - val mockedNodeDataNotJesFilterForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } answers { - listOf(mockedNodeDataNotJesFilterForTest) - } - val isSelected = classUnderTest.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("returnFalse_whenIsSelected_givenNodeWithNoKeysSpecified") { - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - every { mockedJesFilterNode.currentSortQueryKeysList } answers { - mutableListOf() - } - val isSelected = classUnderTest.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe false - } - } - - should("returnTrue_whenIsSelected_givenValidDataAndSortKey") { - every { mockedJesFilterNode.currentSortQueryKeysList } answers { - mutableListOf(SortQueryKeys.JOB_NAME, SortQueryKeys.ASCENDING) - } - val isSelected = classUnderTest.isSelected(mockedActionEvent) - assertSoftly { - isSelected shouldBe true - } - } + should("returnFalse_whenShouldEnableSortKeyForNode_givenSelectedNodeAndSortKey") { + val nodeMock = mockk() + val sortKey = SortQueryKeys.JOB_NAME + every { nodeMock.currentSortQueryKeysList } returns listOf() + + val shouldEnableSortKey = classUnderTest.shouldEnableSortKeyForNode(nodeMock, sortKey) + + assertSoftly { + shouldEnableSortKey shouldBe false } + } + + should("updateQuery_whenPerformQueryUpdateForNode_givenSelectedNodeAndSortKey") { + val jobQueryMock = mockk>() + val nodeMock = mockk() + val sortKey = SortQueryKeys.JOB_NAME + val expectedSortKeys = listOf(SortQueryKeys.ASCENDING, SortQueryKeys.JOB_NAME) + every { nodeMock.query } returns jobQueryMock + every { jobQueryMock.sortKeys } returns mutableListOf(SortQueryKeys.ASCENDING, SortQueryKeys.JOB_COMPLETION_DATE) + every { nodeMock.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.ASCENDING, SortQueryKeys.JOB_COMPLETION_DATE) + + classUnderTest.performQueryUpdateForNode(nodeMock, sortKey) - context("setSelected") { - - should("return_whenSetSelected_givenExplorerNull") { - // given - clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, - answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) - every { mockedActionEvent.getExplorerView() } returns null - - // when - classUnderTest.setSelected(mockedActionEvent, true) - - // then - verify { mockedFileFetchProvider wasNot Called } - } - - should("throwException_whenSetSelected_givenNullTemplateText") { - // given - clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, - answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - every { classUnderTest.templateText } returns null - - // when - val exception = shouldThrowExactly { classUnderTest.setSelected(mockedActionEvent, true) } - - // then - assertSoftly { - exception shouldNotBe null - exception.message shouldBe "Sort key for the selected action was not found." - } - } - - should("return_whenSetSelected_givenAlreadySelectedSortKey") { - // - clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, - answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) - every { classUnderTest.templateText } returns "Job Name" - every { classUnderTest.isSelected(any()) } returns true - - // when - classUnderTest.setSelected(mockedActionEvent, true) - - // then - verify(exactly = 1) { classUnderTest.isSelected(mockedActionEvent) } - verify { mockedFileFetchProvider wasNot Called } - } - - should("return_whenSetSelected_givenNotJesFilterNode") { - // given - clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, - answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) - every { classUnderTest.isSelected(any()) } returns false - val mockedNodeDataNotJesFilterForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataNotJesFilterForTest) - - // when - classUnderTest.setSelected(mockedActionEvent, true) - - // then - verify { mockedFileFetchProvider wasNot Called } - } - - should("callFetchProvider_whenSetSelected_givenValidSortKey") { - // given - clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, - answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) - every { classUnderTest.isSelected(any()) } returns false - every { classUnderTest.templateText } returns "Job Name" - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - every { mockedJesFilterNode.currentSortQueryKeysList } answers { - mutableListOf(SortQueryKeys.JOB_ID, SortQueryKeys.ASCENDING) - } - - // when - classUnderTest.setSelected(mockedActionEvent, true) - - // then - verify(exactly = 1) { mockedJesFilterNode.cleanCache(false) } - verify(exactly = 1) { mockedFileFetchProvider.reload(mockedJobQuery) } - verify(exactly = 1) { mockedFileFetchProvider.applyRefreshCacheDate(mockedJobQuery, mockedJesFilterNode, any()) } - } + assertSoftly { + jobQueryMock.sortKeys shouldContainExactly expectedSortKeys } } } diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionGroupTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionGroupTestSpec.kt new file mode 100644 index 000000000..95df95f0d --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionGroupTestSpec.kt @@ -0,0 +1,66 @@ +package eu.ibagroup.formainframe.explorer.actions.sort.members + +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DataKey +import eu.ibagroup.formainframe.explorer.ui.DSMaskNode +import eu.ibagroup.formainframe.explorer.ui.FileExplorerView +import eu.ibagroup.formainframe.explorer.ui.LibraryNode +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import io.kotest.assertions.assertSoftly +import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe +import io.mockk.* + +class MembersSortActionGroupTestSpec : WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("members sort action group spec") { + + val actionEventMock = mockk() + val explorerViewMock = mockk() + // group action to spy + val classUnderTest = spyk(MembersSortActionGroup()) + + should("shouldReturnExplorerView_whenGetExplorerView_givenActionEvent") { + every { actionEventMock.getData(any() as DataKey) } returns explorerViewMock + val actualExplorer = classUnderTest.getSourceView(actionEventMock) + + assertSoftly { + actualExplorer shouldNotBe null + actualExplorer is FileExplorerView + } + } + + should("shouldReturnTrue_whenCheckNode_givenLibraryNode") { + val nodeMock = mockk() + val checkNode = classUnderTest.checkNode(nodeMock) + + assertSoftly { + checkNode shouldBe true + } + } + + should("shouldReturnNull_whenGetExplorerView_givenActionEvent") { + every { actionEventMock.getData(any() as DataKey) } returns null + val actualExplorer = classUnderTest.getSourceView(actionEventMock) + + assertSoftly { + actualExplorer shouldBe null + } + } + + should("shouldReturnFalse_whenCheckNode_givenWrongNode") { + val nodeMock = mockk() + val checkNode = classUnderTest.checkNode(nodeMock) + + assertSoftly { + checkNode shouldBe false + } + } + } + +}) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionTestSpec.kt new file mode 100644 index 000000000..fb3879bc9 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/members/MembersSortActionTestSpec.kt @@ -0,0 +1,125 @@ +package eu.ibagroup.formainframe.explorer.actions.sort.members + +import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DataKey +import eu.ibagroup.formainframe.dataops.BatchedRemoteQuery +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.fetch.LibraryQuery +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.ui.* +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.matchers.collections.shouldContainExactly +import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe +import io.mockk.* + +class MembersSortActionTestSpec : WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("Members sort action") { + + val actionEventMock = mockk() + val explorerViewMock = mockk() + // action to spy + val classUnderTest = spyk(MembersSortAction()) + + should("returnSourceView_whenGetSourceView_givenActionEvent") { + every { actionEventMock.getData(any() as DataKey) } returns explorerViewMock + + val actualExplorerView = classUnderTest.getSourceView(actionEventMock) + + assertSoftly { + actualExplorerView shouldNotBe null + actualExplorerView is FileExplorerView + } + } + + should("returnNull_whenGetSourceView_givenActionEvent") { + every { actionEventMock.getData(any() as DataKey) } returns null + + val actualExplorerView = classUnderTest.getSourceView(actionEventMock) + + assertSoftly { + actualExplorerView shouldBe null + } + } + + should("returnSourceNode_whenGetSourceNode_givenView") { + val nodeMock = mockk() + val fileMock = mockk() + val attributesMock = mockk() + val myNodesData = mutableListOf(NodeData(nodeMock, fileMock, attributesMock)) + mockkObject(myNodesData) + every { explorerViewMock.mySelectedNodesData } returns myNodesData + + val actualNode = classUnderTest.getSourceNode(explorerViewMock) + + assertSoftly { + actualNode shouldBe nodeMock + } + } + + should("returnNull_whenGetSourceNode_givenView") { + val nodeMock = mockk() + val fileMock = mockk() + val attributesMock = mockk() + val myNodesData = mutableListOf(NodeData(nodeMock, fileMock, attributesMock)) + mockkObject(myNodesData) + every { explorerViewMock.mySelectedNodesData } returns myNodesData + + val actualNode = classUnderTest.getSourceNode(explorerViewMock) + + assertSoftly { + actualNode shouldBe null + } + } + + should("returnTrue_whenShouldEnableSortKeyForNode_givenSelectedNodeAndSortKey") { + val nodeMock = mockk() + val sortKey = SortQueryKeys.MEMBER_NAME + every { nodeMock.currentSortQueryKeysList } returns listOf(SortQueryKeys.MEMBER_NAME, SortQueryKeys.ASCENDING) + + val shouldEnableSortKey = classUnderTest.shouldEnableSortKeyForNode(nodeMock, sortKey) + + assertSoftly { + shouldEnableSortKey shouldBe true + } + } + + should("returnFalse_whenShouldEnableSortKeyForNode_givenSelectedNodeAndSortKey") { + val nodeMock = mockk() + val sortKey = SortQueryKeys.MEMBER_NAME + every { nodeMock.currentSortQueryKeysList } returns listOf() + + val shouldEnableSortKey = classUnderTest.shouldEnableSortKeyForNode(nodeMock, sortKey) + + assertSoftly { + shouldEnableSortKey shouldBe false + } + } + + should("updateQuery_whenPerformQueryUpdateForNode_givenSelectedNodeAndSortKey") { + val batchedQueryMock = mockk>() + val nodeMock = mockk() + val sortKey = SortQueryKeys.MEMBER_NAME + val expectedSortKeys = listOf(SortQueryKeys.ASCENDING, SortQueryKeys.MEMBER_NAME) + every { nodeMock.query } returns batchedQueryMock + every { batchedQueryMock.sortKeys } returns mutableListOf(SortQueryKeys.ASCENDING, SortQueryKeys.MEMBER_MODIFICATION_DATE) + every { nodeMock.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.ASCENDING, SortQueryKeys.MEMBER_MODIFICATION_DATE) + + classUnderTest.performQueryUpdateForNode(nodeMock, sortKey) + + assertSoftly { + batchedQueryMock.sortKeys shouldContainExactly expectedSortKeys + } + } + + } + +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroupTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroupTestSpec.kt index ca23d6c37..39a0549cc 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroupTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionGroupTestSpec.kt @@ -12,6 +12,7 @@ package eu.ibagroup.formainframe.explorer.actions.sort.uss import com.intellij.openapi.actionSystem.ActionUpdateThread import com.intellij.openapi.actionSystem.AnActionEvent +import com.intellij.openapi.actionSystem.DataKey import com.intellij.openapi.actionSystem.Presentation import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl @@ -23,6 +24,7 @@ import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec import eu.ibagroup.formainframe.vfs.MFVirtualFile import io.kotest.assertions.assertSoftly import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe import io.mockk.* import javax.swing.tree.TreePath @@ -61,6 +63,43 @@ class UssSortActionGroupTestSpec : WithApplicationShouldSpec({ val mockedNodeDataForTest = NodeData(mockedUssDirNode, mockedMFVirtualFile, mockedUssRemoteAttributes) mockkObject(mockedNodeDataForTest) + should("shouldReturnExplorerView_whenGetExplorerView_givenActionEvent") { + every { mockedActionEvent.getData(any() as DataKey) } returns mockedFileExplorerView + val actualExplorer = classUnderTest.getSourceView(mockedActionEvent) + + assertSoftly { + actualExplorer shouldNotBe null + actualExplorer is FileExplorerView + } + } + + should("shouldReturnTrue_whenCheckNode_givenUssDirNode") { + val nodeMock = mockk() + val checkNode = classUnderTest.checkNode(nodeMock) + + assertSoftly { + checkNode shouldBe true + } + } + + should("shouldReturnNull_whenGetExplorerView_givenActionEvent") { + every { mockedActionEvent.getData(any() as DataKey) } returns null + val actualExplorer = classUnderTest.getSourceView(mockedActionEvent) + + assertSoftly { + actualExplorer shouldBe null + } + } + + should("shouldReturnFalse_whenCheckNode_givenWrongNode") { + val nodeMock = mockk() + val checkNode = classUnderTest.checkNode(nodeMock) + + assertSoftly { + checkNode shouldBe false + } + } + should("is visible from context menu if file explorer view is null") { var isVisible = true every { mockedActionEvent.getExplorerView() } answers { @@ -82,6 +121,7 @@ class UssSortActionGroupTestSpec : WithApplicationShouldSpec({ isVisible = false listOf(mockedNodeDataNotUssForTest) } + every { mockedFileExplorerView.myTree.isExpanded(selectionPath) } returns true every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView classUnderTest.update(mockedActionEvent) diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionTestSpec.kt index b8d3d0e9d..8e21824f2 100644 --- a/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionTestSpec.kt +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/actions/sort/uss/UssSortActionTestSpec.kt @@ -11,29 +11,21 @@ package eu.ibagroup.formainframe.explorer.actions.sort.uss import com.intellij.openapi.actionSystem.AnActionEvent -import com.intellij.openapi.application.ApplicationManager -import com.intellij.openapi.vfs.VirtualFile +import com.intellij.openapi.actionSystem.DataKey +import com.intellij.openapi.actionSystem.impl.SimpleDataContext import eu.ibagroup.formainframe.config.connect.ConnectionConfig -import eu.ibagroup.formainframe.dataops.DataOpsManager -import eu.ibagroup.formainframe.dataops.Query import eu.ibagroup.formainframe.dataops.UnitRemoteQueryImpl import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes -import eu.ibagroup.formainframe.dataops.fetch.FileFetchProvider -import eu.ibagroup.formainframe.dataops.fetch.UssFileFetchProvider import eu.ibagroup.formainframe.dataops.fetch.UssQuery import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys -import eu.ibagroup.formainframe.explorer.ui.FileExplorerView -import eu.ibagroup.formainframe.explorer.ui.LibraryNode -import eu.ibagroup.formainframe.explorer.ui.NodeData -import eu.ibagroup.formainframe.explorer.ui.UssDirNode -import eu.ibagroup.formainframe.explorer.ui.getExplorerView +import eu.ibagroup.formainframe.explorer.actions.sort.SortAction +import eu.ibagroup.formainframe.explorer.ui.* import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec -import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl -import eu.ibagroup.formainframe.utils.service import eu.ibagroup.formainframe.vfs.MFVirtualFile import io.kotest.assertions.assertSoftly import io.kotest.assertions.throwables.shouldThrowExactly +import io.kotest.matchers.collections.shouldContainExactly import io.kotest.matchers.shouldBe import io.kotest.matchers.shouldNotBe import io.mockk.* @@ -45,25 +37,15 @@ class UssSortActionTestSpec : WithApplicationShouldSpec ({ unmockkAll() } - context("Jobs sort action") { + context("USS sort action") { + + mockkObject(SortAction.Companion) + every { SortAction.runRefreshAction(any()) } just Runs // action to spy val classUnderTest = spyk(UssSortAction()) val mockedActionEvent = mockk() - val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl - val mockedFileFetchProvider = mockk() - dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(ApplicationManager.getApplication()) { - @Suppress("UNCHECKED_CAST") - override fun , File : VirtualFile> getFileFetchProvider( - requestClass: Class, - queryClass: Class>, - vFileClass: Class - ): FileFetchProvider { - return mockedFileFetchProvider as FileFetchProvider - } - - } context("common spec") { @@ -84,14 +66,105 @@ class UssSortActionTestSpec : WithApplicationShouldSpec ({ // Common config for test every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView - every { mockedUssDirNode.cleanCache(false) } just Runs - every { mockedFileFetchProvider.reload(any()) } just Runs - every { mockedFileFetchProvider.applyRefreshCacheDate(any(), any(), any()) } just Runs + + context("misc") { + + should("returnSourceView_whenGetSourceView_givenActionEvent") { + every { mockedActionEvent.getData(any() as DataKey) } returns mockedFileExplorerView + + val actualExplorerView = classUnderTest.getSourceView(mockedActionEvent) + + assertSoftly { + actualExplorerView shouldNotBe null + actualExplorerView is FileExplorerView + } + } + + should("returnNull_whenGetSourceView_givenActionEvent") { + every { mockedActionEvent.getData(any() as DataKey) } returns null + + val actualExplorerView = classUnderTest.getSourceView(mockedActionEvent) + + assertSoftly { + actualExplorerView shouldBe null + } + } + + should("returnSourceNode_whenGetSourceNode_givenView") { + val nodeMock = mockk() + val fileMock = mockk() + val attributesMock = mockk() + val myNodesData = mutableListOf(NodeData(nodeMock, fileMock, attributesMock)) + mockkObject(myNodesData) + every { mockedFileExplorerView.mySelectedNodesData } returns myNodesData + + val actualNode = classUnderTest.getSourceNode(mockedFileExplorerView) + + assertSoftly { + actualNode shouldBe nodeMock + } + } + + should("returnNull_whenGetSourceNode_givenView") { + val nodeMock = mockk() + val fileMock = mockk() + val attributesMock = mockk() + val myNodesData = mutableListOf(NodeData(nodeMock, fileMock, attributesMock)) + mockkObject(myNodesData) + every { mockedFileExplorerView.mySelectedNodesData } returns myNodesData + + val actualNode = classUnderTest.getSourceNode(mockedFileExplorerView) + + assertSoftly { + actualNode shouldBe null + } + } + + should("returnTrue_whenShouldEnableSortKeyForNode_givenSelectedNodeAndSortKey") { + val nodeMock = mockk() + val sortKey = SortQueryKeys.FILE_NAME + every { nodeMock.currentSortQueryKeysList } returns listOf(SortQueryKeys.FILE_NAME, SortQueryKeys.ASCENDING) + + val shouldEnableSortKey = classUnderTest.shouldEnableSortKeyForNode(nodeMock, sortKey) + + assertSoftly { + shouldEnableSortKey shouldBe true + } + } + + should("returnFalse_whenShouldEnableSortKeyForNode_givenSelectedNodeAndSortKey") { + val nodeMock = mockk() + val sortKey = SortQueryKeys.FILE_NAME + every { nodeMock.currentSortQueryKeysList } returns listOf() + + val shouldEnableSortKey = classUnderTest.shouldEnableSortKeyForNode(nodeMock, sortKey) + + assertSoftly { + shouldEnableSortKey shouldBe false + } + } + + should("updateQuery_whenPerformQueryUpdateForNode_givenSelectedNodeAndSortKey") { + val ussQueryMock = mockk>() + val nodeMock = mockk() + val sortKey = SortQueryKeys.FILE_NAME + val expectedSortKeys = listOf(SortQueryKeys.ASCENDING, SortQueryKeys.FILE_NAME) + every { nodeMock.query } returns ussQueryMock + every { ussQueryMock.sortKeys } returns mutableListOf(SortQueryKeys.ASCENDING, SortQueryKeys.FILE_MODIFICATION_DATE) + every { nodeMock.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.ASCENDING, SortQueryKeys.FILE_MODIFICATION_DATE) + + classUnderTest.performQueryUpdateForNode(nodeMock, sortKey) + + assertSoftly { + ussQueryMock.sortKeys shouldContainExactly expectedSortKeys + } + } + } context("isSelected") { should("returnFalse_whenIsSelected_givenExplorerNull") { - every { mockedActionEvent.getExplorerView() } returns null + every { mockedActionEvent.getData(any() as DataKey) } returns null val isSelected = classUnderTest.isSelected(mockedActionEvent) assertSoftly { isSelected shouldBe false @@ -99,7 +172,7 @@ class UssSortActionTestSpec : WithApplicationShouldSpec ({ } should("returnFalse_whenIsSelected_givenNullTemplateText") { - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { mockedActionEvent.getData(any() as DataKey) } returns mockedFileExplorerView every { classUnderTest.templateText } returns null val isSelected = classUnderTest.isSelected(mockedActionEvent) assertSoftly { @@ -146,22 +219,22 @@ class UssSortActionTestSpec : WithApplicationShouldSpec ({ should("return_whenSetSelected_givenExplorerNull") { // given - clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, - answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) - every { mockedActionEvent.getExplorerView() } returns null + var setSelected = true + every { mockedActionEvent.getData(any() as DataKey) } answers { + setSelected = false + null + } // when classUnderTest.setSelected(mockedActionEvent, true) // then - verify { mockedFileFetchProvider wasNot Called } + assertSoftly { setSelected shouldBe false } } should("throwException_whenSetSelected_givenNullTemplateText") { // given - clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, - answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) - every { mockedActionEvent.getExplorerView() } returns mockedFileExplorerView + every { mockedActionEvent.getData(any() as DataKey) } returns mockedFileExplorerView every { classUnderTest.templateText } returns null // when @@ -174,54 +247,63 @@ class UssSortActionTestSpec : WithApplicationShouldSpec ({ } } - should("return_whenSetSelected_givenAlreadySelectedSortKey") { - // - clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, - answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) - every { classUnderTest.templateText } returns "File Name" - every { classUnderTest.isSelected(any()) } returns true + should("return_whenSetSelected_givenNotUssDirNode") { + // given + var setSelected = true + every { classUnderTest.isSelected(any()) } returns false + val mockedNodeDataNotUssForTest = + NodeData(mockk(), mockk(), mockk()) + every { mockedFileExplorerView.mySelectedNodesData } answers { + setSelected = false + listOf(mockedNodeDataNotUssForTest) + } // when classUnderTest.setSelected(mockedActionEvent, true) // then - verify(exactly = 1) { classUnderTest.isSelected(mockedActionEvent) } - verify { mockedFileFetchProvider wasNot Called } + assertSoftly { + setSelected shouldBe false + } } - should("return_whenSetSelected_givenNotUssDirNode") { + should("shouldSetSelected_whenSetSelected_givenValidSortKey") { // given - clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, - answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) + val dataContext = mockk() + val expectedSortKeys = listOf(SortQueryKeys.ASCENDING, SortQueryKeys.FILE_TYPE) + every { classUnderTest.templateText } returns "File Type" every { classUnderTest.isSelected(any()) } returns false - val mockedNodeDataNotUssForTest = - NodeData(mockk(), mockk(), mockk()) - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataNotUssForTest) + every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) + every { mockedUssDirNode.currentSortQueryKeysList } returns mutableListOf(SortQueryKeys.FILE_NAME, SortQueryKeys.ASCENDING) + every { mockedActionEvent.place } returns "Place" + every { mockedActionEvent.dataContext } returns dataContext // when classUnderTest.setSelected(mockedActionEvent, true) // then - verify { mockedFileFetchProvider wasNot Called } + assertSoftly { + mockedUssQuery.sortKeys shouldContainExactly expectedSortKeys + } } - should("callFetchProvider_whenSetSelected_givenValidSortKey") { - // given - clearMocks(classUnderTest, mockedActionEvent, mockedFileFetchProvider, + should("return_whenSetSelected_givenAlreadySelectedSortKey") { + // + clearMocks(classUnderTest, mockedActionEvent, answers = false, recordedCalls = true, childMocks = false, verificationMarks = true, exclusionRules = false) - every { classUnderTest.isSelected(any()) } returns false - every { mockedFileExplorerView.mySelectedNodesData } returns listOf(mockedNodeDataForTest) - every { mockedUssDirNode.currentSortQueryKeysList } answers { - mutableListOf(SortQueryKeys.FILE_TYPE, SortQueryKeys.ASCENDING) + var setSelected = true + every { classUnderTest.templateText } returns "File Name" + every { classUnderTest.isSelected(any()) } answers { + setSelected = false + true } // when classUnderTest.setSelected(mockedActionEvent, true) // then - verify(exactly = 1) { mockedUssDirNode.cleanCache(false) } - verify(exactly = 1) { mockedFileFetchProvider.reload(mockedUssQuery) } - verify(exactly = 1) { mockedFileFetchProvider.applyRefreshCacheDate(mockedUssQuery, mockedUssDirNode, any()) } + verify(exactly = 1) { classUnderTest.isSelected(mockedActionEvent) } + assertSoftly { setSelected shouldBe false } } } } diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNodeTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNodeTestSpec.kt new file mode 100644 index 000000000..4f11b2e38 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/DSMaskNodeTestSpec.kt @@ -0,0 +1,205 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.ui + +import com.intellij.ide.util.treeView.AbstractTreeNode +import com.intellij.openapi.application.ApplicationManager +import com.intellij.openapi.project.Project +import com.intellij.openapi.vfs.VirtualFile +import eu.ibagroup.formainframe.config.ws.DSMask +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.attributes.FileAttributes +import eu.ibagroup.formainframe.dataops.attributes.RemoteDatasetAttributes +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.FileExplorer +import eu.ibagroup.formainframe.explorer.FilesWorkingSetImpl +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.utils.service +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.matchers.shouldBe +import io.mockk.* +import org.zowe.kotlinsdk.Dataset + +class DSMaskNodeTestSpec : WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("explorer module: ui/DSMaskNode") { + val mockedDSMask = mockk() + val mockedProject = mockk() + val mockedExplorerTreeNodeParent = mockk() + val mockedWorkingSet = mockk() + val mockedExplorer = mockk() + val mockedExplorerTreeStructure = mockk() + + every { mockedWorkingSet.explorer } returns mockedExplorer + every { mockedExplorerTreeStructure.registerNode(any()) } just Runs + + val classUnderTest = spyk(DSMaskNode(mockedDSMask, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure)) + + context("sort children nodes") { + val unexpectedNode = mockk() + val mockedVFileChild1 = mockk() + val mockedVFileChild2 = mockk() + val mockedVFileChild3 = mockk() + val mockedAttributes1 = mockk() + val mockedAttributes2 = mockk() + val mockedAttributes3 = mockk() + val datasetInfo1 = mockk() + val datasetInfo2 = mockk() + val datasetInfo3 = mockk() + + val nodeToAttributesMap = mutableMapOf(Pair(mockedVFileChild1, mockedAttributes1), Pair(mockedVFileChild2, mockedAttributes2), Pair(mockedVFileChild3, mockedAttributes3)) + + val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(ApplicationManager.getApplication()) { + override fun tryToGetAttributes(file: VirtualFile): FileAttributes? { + return nodeToAttributesMap[file] + } + } + + beforeEach { + every { mockedAttributes1.datasetInfo } returns datasetInfo1 + every { mockedAttributes2.datasetInfo } returns datasetInfo2 + every { mockedAttributes3.datasetInfo } returns datasetInfo3 + every { mockedAttributes1.datasetInfo.name } returns "AAAA" + every { mockedAttributes2.datasetInfo.name } returns "BBBB" + every { mockedAttributes3.datasetInfo.name } returns "CCCC" + every { mockedAttributes1.datasetInfo.lastReferenceDate } returns "2024/01/10" + every { mockedAttributes2.datasetInfo.lastReferenceDate } returns "2024/01/09" + every { mockedAttributes3.datasetInfo.lastReferenceDate } returns "2024/02/02" + } + + val mockedDataset1 = spyk( + FileLikeDatasetNode(mockedVFileChild1, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure) + ) + val mockedDataset2 = spyk( + FileLikeDatasetNode(mockedVFileChild2, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure) + ) + val mockedDataset3 = spyk( + LibraryNode(mockedVFileChild3, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure) + ) + + every { mockedDataset1.virtualFile } returns mockedVFileChild1 + every { mockedDataset2.virtualFile } returns mockedVFileChild2 + every { mockedDataset3.virtualFile } returns mockedVFileChild3 + + val mockedChildrenNodes = listOf>(mockedDataset1, mockedDataset2, mockedDataset3) + + should("sort by name ascending") { + + val sortQueryKeys = listOf(SortQueryKeys.DATASET_NAME) + every { classUnderTest.currentSortQueryKeysList } returns listOf(SortQueryKeys.DATASET_NAME, SortQueryKeys.ASCENDING) + + val expected = listOf(mockedDataset1, mockedDataset2, mockedDataset3) + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by name descending") { + + val sortQueryKeys = listOf(SortQueryKeys.DATASET_NAME) + every { classUnderTest.currentSortQueryKeysList } returns listOf(SortQueryKeys.DATASET_NAME, SortQueryKeys.DESCENDING) + + val expected = listOf(mockedDataset3, mockedDataset2, mockedDataset1) + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by date ascending") { + + val sortQueryKeys = listOf(SortQueryKeys.DATASET_MODIFICATION_DATE) + every { classUnderTest.currentSortQueryKeysList } returns listOf(SortQueryKeys.DATASET_MODIFICATION_DATE, SortQueryKeys.ASCENDING) + + val expected = listOf(mockedDataset2, mockedDataset1, mockedDataset3) + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by date descending") { + + val sortQueryKeys = listOf(SortQueryKeys.DATASET_MODIFICATION_DATE) + every { classUnderTest.currentSortQueryKeysList } returns listOf(SortQueryKeys.DATASET_MODIFICATION_DATE, SortQueryKeys.DESCENDING) + + val expected = listOf(mockedDataset3, mockedDataset1, mockedDataset2) + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by type ascending") { + + val sortQueryKeys = listOf(SortQueryKeys.DATASET_TYPE) + every { classUnderTest.currentSortQueryKeysList } returns listOf(SortQueryKeys.DATASET_TYPE, SortQueryKeys.ASCENDING) + + val expected = listOf(mockedDataset3, mockedDataset1, mockedDataset2) + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by type descending") { + + val sortQueryKeys = listOf(SortQueryKeys.DATASET_TYPE) + every { classUnderTest.currentSortQueryKeysList } returns listOf(SortQueryKeys.DATASET_TYPE, SortQueryKeys.DESCENDING) + + val expected = listOf(mockedDataset3, mockedDataset2, mockedDataset1) + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe expected + } + } + + should("return unsorted nodes when passing null sort key") { + + val sortQueryKeys = listOf() + + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe mockedChildrenNodes + } + } + + should("return unsorted nodes when passing invalid sort key") { + + val mockedChildrenNodesForThisTest = listOf>(mockedDataset1, mockedDataset2, mockedDataset3, unexpectedNode) + val sortQueryKeys = listOf(SortQueryKeys.JOB_NAME) + + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodesForThisTest, sortQueryKeys) + + assertSoftly { + actual shouldBe mockedChildrenNodesForThisTest + } + } + } + } + +}) \ No newline at end of file diff --git a/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/LibraryNodeTestSpec.kt b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/LibraryNodeTestSpec.kt new file mode 100644 index 000000000..64ce76af1 --- /dev/null +++ b/src/test/kotlin/eu/ibagroup/formainframe/explorer/ui/LibraryNodeTestSpec.kt @@ -0,0 +1,175 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright IBA Group 2020 + */ + +package eu.ibagroup.formainframe.explorer.ui + +import com.intellij.ide.util.treeView.AbstractTreeNode +import com.intellij.openapi.application.ApplicationManager +import com.intellij.openapi.project.Project +import com.intellij.openapi.vfs.VirtualFile +import eu.ibagroup.formainframe.dataops.DataOpsManager +import eu.ibagroup.formainframe.dataops.attributes.* +import eu.ibagroup.formainframe.dataops.sort.SortQueryKeys +import eu.ibagroup.formainframe.explorer.FileExplorer +import eu.ibagroup.formainframe.explorer.FilesWorkingSetImpl +import eu.ibagroup.formainframe.testutils.WithApplicationShouldSpec +import eu.ibagroup.formainframe.testutils.testServiceImpl.TestDataOpsManagerImpl +import eu.ibagroup.formainframe.utils.service +import eu.ibagroup.formainframe.vfs.MFVirtualFile +import io.kotest.assertions.assertSoftly +import io.kotest.matchers.shouldBe +import io.mockk.* +import org.zowe.kotlinsdk.Member + +class LibraryNodeTestSpec : WithApplicationShouldSpec({ + + afterSpec { + clearAllMocks() + unmockkAll() + } + + context("explorer module: ui/LibraryNode") { + val mockedLibrary = mockk() + val mockedProject = mockk() + val mockedExplorerTreeNodeParent = mockk() + val mockedWorkingSet = mockk() + val mockedExplorer = mockk() + val mockedExplorerTreeStructure = mockk() + + every { mockedWorkingSet.explorer } returns mockedExplorer + every { mockedExplorerTreeStructure.registerNode(any()) } just Runs + + val classUnderTest = spyk(LibraryNode(mockedLibrary, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure)) + + context("sort children nodes") { + val mockedVFileChild1 = mockk() + val mockedVFileChild2 = mockk() + val mockedVFileChild3 = mockk() + val mockedAttributes1 = mockk() + val mockedAttributes2 = mockk() + val mockedAttributes3 = mockk() + val memberInfo1 = mockk() + val memberInfo2 = mockk() + val memberInfo3 = mockk() + + val nodeToAttributesMap = mutableMapOf(Pair(mockedVFileChild1, mockedAttributes1), Pair(mockedVFileChild2, mockedAttributes2), Pair(mockedVFileChild3, mockedAttributes3)) + + val dataOpsManagerService = ApplicationManager.getApplication().service() as TestDataOpsManagerImpl + dataOpsManagerService.testInstance = object : TestDataOpsManagerImpl(ApplicationManager.getApplication()) { + override fun tryToGetAttributes(file: VirtualFile): FileAttributes? { + return nodeToAttributesMap[file] + } + } + + beforeEach { + every { mockedAttributes1.info } returns memberInfo1 + every { mockedAttributes2.info } returns memberInfo2 + every { mockedAttributes3.info } returns memberInfo3 + every { mockedAttributes1.info.name } returns "AAAA" + every { mockedAttributes2.info.name } returns "BBBB" + every { mockedAttributes3.info.name } returns "CCCC" + every { mockedAttributes1.info.modificationDate } returns "2024/01/10" + every { mockedAttributes2.info.modificationDate } returns "2024/01/09" + every { mockedAttributes3.info.modificationDate } returns "2024/02/02" + } + + val mockedMember1 = spyk( + FileLikeDatasetNode(mockedVFileChild1, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure) + ) + val mockedMember2 = spyk( + FileLikeDatasetNode(mockedVFileChild2, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure) + ) + val mockedMember3 = spyk( + FileLikeDatasetNode(mockedVFileChild3, mockedProject, mockedExplorerTreeNodeParent, mockedWorkingSet, mockedExplorerTreeStructure) + ) + + every { mockedMember1.virtualFile } returns mockedVFileChild1 + every { mockedMember2.virtualFile } returns mockedVFileChild2 + every { mockedMember3.virtualFile } returns mockedVFileChild3 + + val mockedChildrenNodes = listOf>(mockedMember1, mockedMember2, mockedMember3) + + should("sort by name ascending") { + + val sortQueryKeys = listOf(SortQueryKeys.MEMBER_NAME) + every { classUnderTest.currentSortQueryKeysList } returns listOf(SortQueryKeys.MEMBER_NAME, SortQueryKeys.ASCENDING) + + val expected = listOf(mockedMember1, mockedMember2, mockedMember3) + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by name descending") { + + val sortQueryKeys = listOf(SortQueryKeys.MEMBER_NAME) + every { classUnderTest.currentSortQueryKeysList } returns listOf(SortQueryKeys.MEMBER_NAME, SortQueryKeys.DESCENDING) + + val expected = listOf(mockedMember3, mockedMember2, mockedMember1) + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by date ascending") { + + val sortQueryKeys = listOf(SortQueryKeys.MEMBER_MODIFICATION_DATE) + every { classUnderTest.currentSortQueryKeysList } returns listOf(SortQueryKeys.MEMBER_MODIFICATION_DATE, SortQueryKeys.ASCENDING) + + val expected = listOf(mockedMember2, mockedMember1, mockedMember3) + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe expected + } + } + + should("sort by date descending") { + + val sortQueryKeys = listOf(SortQueryKeys.MEMBER_MODIFICATION_DATE) + every { classUnderTest.currentSortQueryKeysList } returns listOf(SortQueryKeys.MEMBER_MODIFICATION_DATE, SortQueryKeys.DESCENDING) + + val expected = listOf(mockedMember3, mockedMember1, mockedMember2) + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe expected + } + } + + should("return unsorted nodes when passing null sort key") { + + val sortQueryKeys = listOf() + + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe mockedChildrenNodes + } + } + + should("return unsorted nodes when passing invalid sort key") { + + val sortQueryKeys = listOf(SortQueryKeys.JOB_NAME) + + val actual = classUnderTest.sortChildrenNodes(mockedChildrenNodes, sortQueryKeys) + + assertSoftly { + actual shouldBe mockedChildrenNodes + } + } + } + } + +}) \ No newline at end of file From cc34a8d36d2bbb442ff3dcfc681af3cf82681c72 Mon Sep 17 00:00:00 2001 From: Arseni Tsikhamirau Date: Tue, 19 Mar 2024 12:39:34 +0100 Subject: [PATCH 31/34] IJMP-1507-Rewrite-horizontal-and-vertical-align --- .../analytics/ui/AnalyticsPolicyDialog.kt | 11 +++--- .../ibagroup/formainframe/common/ui/Table.kt | 7 +--- .../config/connect/ui/ChangePasswordDialog.kt | 9 ++--- .../explorer/ui/AddJobsFilterDialog.kt | 8 ++-- .../explorer/ui/AddMemberDialog.kt | 4 +- .../explorer/ui/AddOrEditMaskDialog.kt | 8 +--- .../explorer/ui/AllocationDialog.kt | 32 +++++++--------- .../explorer/ui/CreateFileDialog.kt | 4 +- .../explorer/ui/DatasetPropertiesDialog.kt | 37 +++++++++--------- .../explorer/ui/EditJobsFilterDialog.kt | 8 ++-- .../explorer/ui/JobPropertiesDialog.kt | 38 +++++++++---------- .../explorer/ui/MemberPropertiesDialog.kt | 36 +++++++++--------- .../explorer/ui/SpoolFilePropertiesDialog.kt | 30 +++++++-------- .../explorer/ui/UssFilePropertiesDialog.kt | 27 +++++++------ .../ui/build/tso/ui/TSOSessionDialog.kt | 34 +++++------------ 15 files changed, 131 insertions(+), 162 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/analytics/ui/AnalyticsPolicyDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/analytics/ui/AnalyticsPolicyDialog.kt index 8bb754425..9ae26164f 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/analytics/ui/AnalyticsPolicyDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/analytics/ui/AnalyticsPolicyDialog.kt @@ -13,9 +13,10 @@ package eu.ibagroup.formainframe.analytics.ui import com.intellij.openapi.project.Project import com.intellij.openapi.ui.DialogWrapper import com.intellij.ui.components.JBTextArea +import com.intellij.ui.dsl.builder.AlignX +import com.intellij.ui.dsl.builder.AlignY import com.intellij.ui.dsl.builder.panel -import com.intellij.ui.dsl.gridLayout.HorizontalAlign -import com.intellij.ui.dsl.gridLayout.VerticalAlign +import com.intellij.ui.dsl.builder.plus import com.intellij.util.ui.UIUtil import eu.ibagroup.formainframe.analytics.AnalyticsService import eu.ibagroup.formainframe.analytics.PolicyProvider @@ -63,8 +64,7 @@ class AnalyticsPolicyDialog( caretPosition = 0 } scrollCell(licenseTextArea) - .horizontalAlign(HorizontalAlign.FILL) - .verticalAlign(VerticalAlign.FILL) + .align(AlignX.FILL.plus(AlignY.FILL)) }.resizableRow() row { text(policyProvider.buildAgreementText("clicking")) @@ -72,8 +72,7 @@ class AnalyticsPolicyDialog( component.font = UIUtil.getFont(UIUtil.FontSize.NORMAL, component.font) component.preferredSize = Dimension(660, 20) } - .horizontalAlign(HorizontalAlign.FILL) - .verticalAlign(VerticalAlign.FILL) + .align(AlignX.FILL.plus(AlignY.FILL)) } } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/common/ui/Table.kt b/src/main/kotlin/eu/ibagroup/formainframe/common/ui/Table.kt index a68f595e3..81bef2857 100755 --- a/src/main/kotlin/eu/ibagroup/formainframe/common/ui/Table.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/common/ui/Table.kt @@ -14,9 +14,7 @@ import com.intellij.openapi.actionSystem.ActionToolbarPosition import com.intellij.openapi.actionSystem.AnActionEvent import com.intellij.ui.AnActionButton import com.intellij.ui.ToolbarDecorator -import com.intellij.ui.dsl.builder.Row -import com.intellij.ui.dsl.gridLayout.HorizontalAlign -import com.intellij.ui.dsl.gridLayout.VerticalAlign +import com.intellij.ui.dsl.builder.* import com.intellij.ui.table.TableView import javax.swing.JPanel import javax.swing.JTable @@ -152,8 +150,7 @@ fun Row.tableWithToolbar( .apply { toolbarTableBuilder() } .createPanel() return cell(tableComponent) - .horizontalAlign(HorizontalAlign.FILL) - .verticalAlign(VerticalAlign.FILL) + .align(AlignX.FILL.plus(AlignY.FILL)) } val TableModelEvent.rows: IntRange diff --git a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/ChangePasswordDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/ChangePasswordDialog.kt index d3f20df55..3fe0b1a2f 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/ChangePasswordDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/ui/ChangePasswordDialog.kt @@ -13,7 +13,6 @@ package eu.ibagroup.formainframe.config.connect.ui import com.intellij.icons.AllIcons import com.intellij.openapi.project.Project import com.intellij.ui.dsl.builder.* -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.ui.StatefulDialog import eu.ibagroup.formainframe.utils.validateForBlank import eu.ibagroup.formainframe.utils.validateForPassword @@ -46,7 +45,7 @@ class ChangePasswordDialog( it.text = it.text.trim() validateForBlank(it) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Old password: ") @@ -58,7 +57,7 @@ class ChangePasswordDialog( addShowHidePasswordIcon(this) } .validationOnApply { validateForBlank(it) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("New password: ") @@ -75,7 +74,7 @@ class ChangePasswordDialog( }) } .validationOnApply { validateForBlank(it) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Confirm password: ") @@ -87,7 +86,7 @@ class ChangePasswordDialog( addShowHidePasswordIcon(this) } .validationOnApply { validateForBlank(it) ?: validateForPassword(state.newPassword, it) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddJobsFilterDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddJobsFilterDialog.kt index 9442d5330..e008ad785 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddJobsFilterDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddJobsFilterDialog.kt @@ -13,9 +13,9 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.openapi.project.Project import com.intellij.openapi.ui.DialogWrapper import com.intellij.ui.components.JBTextField +import com.intellij.ui.dsl.builder.AlignX import com.intellij.ui.dsl.builder.bindText import com.intellij.ui.dsl.builder.panel -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.ui.StatefulComponent import eu.ibagroup.formainframe.config.ws.JobFilterStateWithWS import eu.ibagroup.formainframe.utils.validateJobFilter @@ -52,7 +52,7 @@ class AddJobsFilterDialog( .validationOnApply { validateJobFilter(it.text, ownerField.text, jobIdField.text, state.ws.masks, it, false) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Owner: ") @@ -63,7 +63,7 @@ class AddJobsFilterDialog( .validationOnApply { validateJobFilter(prefixField.text, it.text, jobIdField.text, state.ws.masks, it, false) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Job ID: ") @@ -74,7 +74,7 @@ class AddJobsFilterDialog( .validationOnApply { validateJobFilter(prefixField.text, ownerField.text, it.text, state.ws.masks, it, true) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddMemberDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddMemberDialog.kt index 561e9781e..ffb88aaf0 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddMemberDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddMemberDialog.kt @@ -12,9 +12,9 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.openapi.project.Project import com.intellij.openapi.ui.DialogWrapper +import com.intellij.ui.dsl.builder.AlignX import com.intellij.ui.dsl.builder.bindText import com.intellij.ui.dsl.builder.panel -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.ui.StatefulComponent import eu.ibagroup.formainframe.dataops.operations.MemberAllocationParams import eu.ibagroup.formainframe.utils.validateForBlank @@ -33,7 +33,7 @@ class AddMemberDialog(project: Project?, override var state: MemberAllocationPar .bindText(state::memberName) .validationOnApply { validateForBlank(it) ?: validateMemberName(it) } .apply { focused() } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddOrEditMaskDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddOrEditMaskDialog.kt index baab707c1..a59da2cef 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddOrEditMaskDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AddOrEditMaskDialog.kt @@ -13,11 +13,7 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.openapi.project.Project import com.intellij.openapi.ui.ComboBox import com.intellij.openapi.ui.DialogWrapper -import com.intellij.ui.dsl.builder.Cell -import com.intellij.ui.dsl.builder.bindItem -import com.intellij.ui.dsl.builder.bindText -import com.intellij.ui.dsl.builder.panel -import com.intellij.ui.dsl.gridLayout.HorizontalAlign +import com.intellij.ui.dsl.builder.* import eu.ibagroup.formainframe.common.ui.StatefulComponent import eu.ibagroup.formainframe.config.connect.ConnectionConfig import eu.ibagroup.formainframe.config.connect.getUsername @@ -125,7 +121,7 @@ class AddOrEditMaskDialog( .apply { component.minimumSize = Dimension(10, component.height) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AllocationDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AllocationDialog.kt index 052d3e028..4a2c4d446 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AllocationDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/AllocationDialog.kt @@ -16,11 +16,7 @@ import com.intellij.openapi.ui.ComboBox import com.intellij.openapi.ui.ValidationInfo import com.intellij.ui.SimpleListCellRenderer import com.intellij.ui.components.JBScrollPane -import com.intellij.ui.dsl.builder.bindItem -import com.intellij.ui.dsl.builder.bindText -import com.intellij.ui.dsl.builder.panel -import com.intellij.ui.dsl.builder.toNullableProperty -import com.intellij.ui.dsl.gridLayout.HorizontalAlign +import com.intellij.ui.dsl.builder.* import com.intellij.ui.layout.selectedValueMatches import eu.ibagroup.formainframe.common.message import eu.ibagroup.formainframe.common.ui.StatefulDialog @@ -89,7 +85,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var datasetNameField.text = "${HLQ}." } .onApply { state.datasetName = state.datasetName.uppercase() } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) .focused() } row { @@ -102,7 +98,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var memberNameField.text = "SAMPLE" } .onApply { state.memberName = state.memberName.uppercase() } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } .visibleIf(presetsBox.selectedValueMatches { it == Presets.PDS_WITH_EMPTY_MEMBER || it == Presets.PDS_WITH_SAMPLE_JCL_MEMBER }) row { @@ -148,7 +144,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.primaryAllocation = it.toIntOrNull() ?: 0 } ) .also { primaryAllocationField = it.component } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Secondary allocation: ") @@ -159,7 +155,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.secondaryAllocation = it.toIntOrNull() ?: 0 } ) .also { secondaryAllocationField = it.component } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Directory: ") @@ -176,7 +172,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.directoryBlocks = it.toIntOrNull() ?: 0 } ) .also { directoryBlocksField = it.component } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } .visibleIf(datasetOrganizationBox.selectedValueMatches { it != DatasetOrganization.PS }) row { @@ -205,7 +201,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.recordLength = it.toIntOrNull() } ) .also { recordLengthField = it.component } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Block size: ") @@ -216,7 +212,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.blockSize = it.toIntOrNull() } ) .also { blockSizeField = it.component } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Average Block Length: ") @@ -227,7 +223,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.averageBlockLength = it.toIntOrNull() } ) .also { averageBlockLengthField = it.component } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } collapsibleGroup("Advanced Parameters", false) { row { @@ -239,7 +235,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.volumeSerial = it } ) .also { advancedParametersField = it.component } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Device Type: ") @@ -249,7 +245,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.deviceType ?: "" }, { state.allocationParameters.deviceType = it } ) - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Storage class: ") @@ -259,7 +255,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.storageClass ?: "" }, { state.allocationParameters.storageClass = it } ) - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Management class: ") @@ -269,7 +265,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.managementClass ?: "" }, { state.allocationParameters.managementClass = it } ) - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Data class: ") @@ -279,7 +275,7 @@ class AllocationDialog(project: Project?, config: ConnectionConfig, override var { state.allocationParameters.dataClass ?: "" }, { state.allocationParameters.dataClass = it } ) - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/CreateFileDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/CreateFileDialog.kt index 816698654..76f08a695 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/CreateFileDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/CreateFileDialog.kt @@ -11,10 +11,10 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.openapi.project.Project +import com.intellij.ui.dsl.builder.AlignX import com.intellij.ui.dsl.builder.bindItem import com.intellij.ui.dsl.builder.bindText import com.intellij.ui.dsl.builder.panel -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.ui.StatefulDialog import eu.ibagroup.formainframe.dataops.operations.UssAllocationParams import eu.ibagroup.formainframe.utils.validateForBlank @@ -71,7 +71,7 @@ class CreateFileDialog(project: Project?, override var state: CreateFileDialogSt textField() .bindText(state::fileName) .validationOnApply { validateForBlank(it) ?: validateUssFileName(it) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) .focused() } row { diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DatasetPropertiesDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DatasetPropertiesDialog.kt index 8a885fcb6..c24b8f100 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DatasetPropertiesDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/DatasetPropertiesDialog.kt @@ -13,9 +13,9 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.openapi.project.Project import com.intellij.openapi.ui.DialogWrapper import com.intellij.ui.components.JBTabbedPane +import com.intellij.ui.dsl.builder.AlignX import com.intellij.ui.dsl.builder.panel import com.intellij.ui.dsl.builder.text -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.ui.DialogMode import eu.ibagroup.formainframe.common.ui.DialogState import eu.ibagroup.formainframe.common.ui.StatefulComponent @@ -45,7 +45,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.name) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Dataset name type: ") @@ -53,7 +53,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.dsnameType?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Catalog name: ") @@ -61,7 +61,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.catalogName ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Volume serials: ") @@ -69,7 +69,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.volumeSerials ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Device type: ") @@ -77,7 +77,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.deviceType ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } if (dataset.migrated?.equals(HasMigrated.YES) == true) { row { @@ -87,7 +87,6 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset } ) - tabbedPanel.add( "Data", panel { @@ -104,7 +103,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset } ) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Record format: ") @@ -112,7 +111,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.recordFormat?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Record length: ") @@ -120,7 +119,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.recordLength?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Block size: ") @@ -128,7 +127,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.blockSize?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Size in tracks: ") @@ -136,7 +135,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.sizeInTracks?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Space units: ") @@ -144,7 +143,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.spaceUnits?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } if ("YES" == dataset.spaceOverflowIndicator) { row { @@ -154,8 +153,6 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset } ) - - tabbedPanel.add("Extended", panel { row { label("Current Utilization") @@ -167,7 +164,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.usedTracksOrBlocks?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Used extents: ") @@ -175,7 +172,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.extendsUsed?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Dates") @@ -187,7 +184,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.creationDate ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Referenced date: ") @@ -195,7 +192,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.lastReferenceDate ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Expiration date: ") @@ -203,7 +200,7 @@ class DatasetPropertiesDialog(val project: Project?, override var state: Dataset textField() .text(dataset.expirationDate ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } }) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/EditJobsFilterDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/EditJobsFilterDialog.kt index 0d389417e..380311c7c 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/EditJobsFilterDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/EditJobsFilterDialog.kt @@ -13,9 +13,9 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.openapi.project.Project import com.intellij.openapi.ui.DialogWrapper import com.intellij.ui.components.JBTextField +import com.intellij.ui.dsl.builder.AlignX import com.intellij.ui.dsl.builder.bindText import com.intellij.ui.dsl.builder.panel -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.ui.StatefulComponent import eu.ibagroup.formainframe.config.ws.JobFilterStateWithWS import eu.ibagroup.formainframe.config.ws.JobsFilter @@ -57,7 +57,7 @@ class EditJobsFilterDialog( .validationOnApply { validateJobFilter(initJobFilter, it.text, ownerField.text, jobIdField.text, state.ws.masks, it, false) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Owner: ") @@ -68,7 +68,7 @@ class EditJobsFilterDialog( .validationOnApply { validateJobFilter(initJobFilter, prefixField.text, it.text, jobIdField.text, state.ws.masks, it, false) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Job ID: ") @@ -79,7 +79,7 @@ class EditJobsFilterDialog( .validationOnApply { validateJobFilter(initJobFilter, prefixField.text, ownerField.text, it.text, state.ws.masks, it, true) } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JobPropertiesDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JobPropertiesDialog.kt index f7f3a70ec..5a09060bb 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JobPropertiesDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/JobPropertiesDialog.kt @@ -13,9 +13,9 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.openapi.project.Project import com.intellij.openapi.ui.DialogWrapper import com.intellij.ui.components.JBTabbedPane +import com.intellij.ui.dsl.builder.AlignX import com.intellij.ui.dsl.builder.panel import com.intellij.ui.dsl.builder.text -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.ui.DialogMode import eu.ibagroup.formainframe.common.ui.DialogState import eu.ibagroup.formainframe.common.ui.StatefulComponent @@ -53,7 +53,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.jobId) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Job name: ") @@ -61,7 +61,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.jobName) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Subsystem: ") @@ -69,7 +69,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.subSystem ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Owner: ") @@ -77,7 +77,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.owner) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Status: ") @@ -85,7 +85,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.status?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Job type: ") @@ -93,7 +93,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.type.toString()) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Job class: ") @@ -101,7 +101,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.jobClass ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Return code: ") @@ -109,7 +109,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.returnedCode ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Job correlator: ") @@ -117,7 +117,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.jobCorrelator ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } }) @@ -130,7 +130,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.phase.toString()) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Phase name: ") @@ -138,7 +138,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.phaseName) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("URL: ") @@ -146,7 +146,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.url) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Files URL: ") @@ -154,7 +154,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.filesUrl) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("System executor: ") @@ -162,7 +162,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.execSystem ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Reason not running: ") @@ -170,7 +170,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.reasonNotRunning ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Run info") @@ -183,7 +183,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.execSubmitted ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Job start time: ") @@ -191,7 +191,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.execStarted ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Time ended: ") @@ -199,7 +199,7 @@ class JobPropertiesDialog(val project: Project?, override var state: JobState) : textField() .text(job.execEnded ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } }) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/MemberPropertiesDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/MemberPropertiesDialog.kt index bc46c5ed6..8fc7f484e 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/MemberPropertiesDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/MemberPropertiesDialog.kt @@ -13,9 +13,9 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.openapi.project.Project import com.intellij.openapi.ui.DialogWrapper import com.intellij.ui.components.JBTabbedPane +import com.intellij.ui.dsl.builder.AlignX import com.intellij.ui.dsl.builder.panel import com.intellij.ui.dsl.builder.text -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.ui.DialogMode import eu.ibagroup.formainframe.common.ui.DialogState import eu.ibagroup.formainframe.common.ui.StatefulComponent @@ -44,7 +44,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.name) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Version.Modification: ") @@ -58,7 +58,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt } ) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Create Date: ") @@ -66,7 +66,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.creationDate ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Modification Date: ") @@ -74,7 +74,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.modificationDate ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Modification Time: ") @@ -82,7 +82,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.lastChangeTime ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Userid that Created/Modified: ") @@ -90,7 +90,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.user ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } } ) @@ -104,7 +104,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.currentNumberOfRecords?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Beginning number of records: ") @@ -112,7 +112,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.beginningNumberOfRecords?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Number of changed records: ") @@ -120,7 +120,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.numberOfChangedRecords?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { val updatePlace = if ("Y" == member.sclm) { @@ -146,7 +146,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.authorizationCode ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Current Member is alias of: ") @@ -154,7 +154,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.aliasOf ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Load module attributes: ") @@ -162,7 +162,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.loadModuleAttributes ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Member AMODE: ") @@ -170,7 +170,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.amode ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Member RMODE: ") @@ -178,7 +178,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.rmode ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Size: ") @@ -186,7 +186,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.size ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Member TTR: ") @@ -194,7 +194,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.ttr ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("SSI information for a load module: ") @@ -202,7 +202,7 @@ class MemberPropertiesDialog(var project: Project?, override var state: MemberSt textField() .text(member.ssi ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } } ) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/SpoolFilePropertiesDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/SpoolFilePropertiesDialog.kt index 247c91557..28181df12 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/SpoolFilePropertiesDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/SpoolFilePropertiesDialog.kt @@ -13,9 +13,9 @@ package eu.ibagroup.formainframe.explorer.ui import com.intellij.openapi.project.Project import com.intellij.openapi.ui.DialogWrapper import com.intellij.ui.components.JBTabbedPane +import com.intellij.ui.dsl.builder.AlignX import com.intellij.ui.dsl.builder.panel import com.intellij.ui.dsl.builder.text -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.ui.DialogMode import eu.ibagroup.formainframe.common.ui.DialogState import eu.ibagroup.formainframe.common.ui.StatefulComponent @@ -55,7 +55,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.jobId) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Job name: ") @@ -63,7 +63,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.jobname) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Job correlator: ") @@ -71,7 +71,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.jobCorrelator ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Class: ") @@ -79,7 +79,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.fileClass) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("ID: ") @@ -87,7 +87,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.id.toString()) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("DD name: ") @@ -95,7 +95,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.ddName) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Step name: ") @@ -103,7 +103,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.stepName ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Process step: ") @@ -111,7 +111,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.procStep ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } }) tabbedPanel.add( @@ -123,7 +123,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.recfm) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Byte content: ") @@ -131,7 +131,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.byteCount.toString()) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Record count: ") @@ -139,7 +139,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.recordCount.toString()) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Record URL: ") @@ -147,7 +147,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.recordsUrl) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Record length: ") @@ -155,7 +155,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.recordLength.toString()) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Subsystem: ") @@ -163,7 +163,7 @@ class SpoolFilePropertiesDialog(val project: Project?, override var state: Spool textField() .text(spoolFile.subsystem ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } }) return tabbedPanel diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssFilePropertiesDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssFilePropertiesDialog.kt index 618c4d2cb..1414a2b42 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssFilePropertiesDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/ui/UssFilePropertiesDialog.kt @@ -18,7 +18,6 @@ import com.intellij.ui.components.JBTabbedPane import com.intellij.ui.dsl.builder.bindItem import com.intellij.ui.dsl.builder.panel import com.intellij.ui.dsl.builder.text -import com.intellij.ui.dsl.gridLayout.HorizontalAlign import eu.ibagroup.formainframe.common.ui.StatefulComponent import eu.ibagroup.formainframe.dataops.attributes.RemoteUssAttributes import javax.swing.JComponent @@ -71,7 +70,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat textField() .text(state.ussAttributes.name) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Location: ") @@ -79,7 +78,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat textField() .text(state.ussAttributes.parentDirPath) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Path: ") @@ -87,7 +86,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat textField() .text(state.ussAttributes.path) .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("$fileTypeName size: ") @@ -95,7 +94,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat textField() .text("${state.ussAttributes.length} bytes") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Last modified: ") @@ -103,14 +102,14 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat textField() .text(state.ussAttributes.modificationTime ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } if (!state.ussAttributes.isDirectory && state.fileIsBeingEditingNow) { row { label("File encoding: ").widthGroup(sameWidthGroup) comboBox = comboBox(getSupportedEncodings()) .bindItem(state.ussAttributes::charset.toNullableProperty()) - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { button("Reset Default Encoding") { @@ -126,7 +125,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat textField() .text(state.ussAttributes.symlinkTarget ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } } } @@ -138,7 +137,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat textField() .text(state.ussAttributes.owner ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Group: ") @@ -146,7 +145,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat textField() .text(state.ussAttributes.groupId ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("The numeric group ID (GID): ") @@ -154,7 +153,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat textField() .text(state.ussAttributes.gid?.toString() ?: "") .applyToComponent { isEditable = false } - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Owner permissions: ") @@ -164,7 +163,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat { state.ussAttributes.fileMode?.owner?.toFileModeValue() }, { state.ussAttributes.fileMode?.owner = it?.mode ?: 0 } ) - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Group permissions: ") @@ -174,7 +173,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat { state.ussAttributes.fileMode?.group?.toFileModeValue() }, { state.ussAttributes.fileMode?.group = it?.mode ?: 0 } ) - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } row { label("Permissions for all users: ") @@ -184,7 +183,7 @@ class UssFilePropertiesDialog(project: Project?, override var state: UssFileStat { state.ussAttributes.fileMode?.all?.toFileModeValue() }, { state.ussAttributes.fileMode?.all = it?.mode ?: 0 } ) - .horizontalAlign(HorizontalAlign.FILL) + .align(AlignX.FILL) } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOSessionDialog.kt b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOSessionDialog.kt index 5e7a5ee4f..1921593af 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOSessionDialog.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOSessionDialog.kt @@ -13,12 +13,7 @@ package eu.ibagroup.formainframe.ui.build.tso.ui import com.intellij.openapi.project.Project import com.intellij.ui.CollectionComboBoxModel import com.intellij.ui.SimpleListCellRenderer -import com.intellij.ui.dsl.builder.bindItem -import com.intellij.ui.dsl.builder.bindText -import com.intellij.ui.dsl.builder.panel -import com.intellij.ui.dsl.builder.toNullableProperty -import com.intellij.ui.dsl.gridLayout.HorizontalAlign -import com.intellij.ui.dsl.gridLayout.VerticalAlign +import com.intellij.ui.dsl.builder.* import eu.ibagroup.formainframe.common.ui.StatefulDialog import eu.ibagroup.formainframe.config.configCrudable import eu.ibagroup.formainframe.config.connect.ConnectionConfig @@ -68,8 +63,7 @@ class TSOSessionDialog(project: Project?, override var state: TSOSessionParams) .also { connectionBox = it.component resizableRow() - it.verticalAlign(VerticalAlign.FILL) - it.horizontalAlign(HorizontalAlign.FILL) + it.align(AlignX.FILL.plus(AlignY.FILL)) } } row { @@ -82,8 +76,7 @@ class TSOSessionDialog(project: Project?, override var state: TSOSessionParams) }.also { logonProcField = it.component resizableRow() - it.verticalAlign(VerticalAlign.FILL) - it.horizontalAlign(HorizontalAlign.FILL) + it.align(AlignX.FILL.plus(AlignY.FILL)) }.validationOnInput { validateForBlank(it) } } row { @@ -94,8 +87,7 @@ class TSOSessionDialog(project: Project?, override var state: TSOSessionParams) .also { charsetField = it.component resizableRow() - it.verticalAlign(VerticalAlign.FILL) - it.horizontalAlign(HorizontalAlign.FILL) + it.align(AlignX.FILL.plus(AlignY.FILL)) }.validationOnInput { validateForBlank(it) ?: validateForPositiveInteger(it) } } row { @@ -108,8 +100,7 @@ class TSOSessionDialog(project: Project?, override var state: TSOSessionParams) .also { codepageField = it.component resizableRow() - it.verticalAlign(VerticalAlign.FILL) - it.horizontalAlign(HorizontalAlign.FILL) + it.align(AlignX.FILL.plus(AlignY.FILL)) } } row { @@ -120,8 +111,7 @@ class TSOSessionDialog(project: Project?, override var state: TSOSessionParams) .also { rowsField = it.component resizableRow() - it.verticalAlign(VerticalAlign.FILL) - it.horizontalAlign(HorizontalAlign.FILL) + it.align(AlignX.FILL.plus(AlignY.FILL)) }.validationOnInput { validateForBlank(it) ?: validateForPositiveInteger(it) } } row { @@ -132,8 +122,7 @@ class TSOSessionDialog(project: Project?, override var state: TSOSessionParams) .also { colsField = it.component resizableRow() - it.verticalAlign(VerticalAlign.FILL) - it.horizontalAlign(HorizontalAlign.FILL) + it.align(AlignX.FILL.plus(AlignY.FILL)) }.validationOnInput { validateForBlank(it) ?: validateForPositiveInteger(it) } } row { @@ -144,8 +133,7 @@ class TSOSessionDialog(project: Project?, override var state: TSOSessionParams) .also { acctField = it.component resizableRow() - it.verticalAlign(VerticalAlign.FILL) - it.horizontalAlign(HorizontalAlign.FILL) + it.align(AlignX.FILL.plus(AlignY.FILL)) }.validationOnInput { validateForBlank(it) } } row { @@ -156,8 +144,7 @@ class TSOSessionDialog(project: Project?, override var state: TSOSessionParams) .also { userGroupField = it.component resizableRow() - it.verticalAlign(VerticalAlign.FILL) - it.horizontalAlign(HorizontalAlign.FILL) + it.align(AlignX.FILL.plus(AlignY.FILL)) }.validationOnInput { validateForBlank(it) } } row { @@ -168,8 +155,7 @@ class TSOSessionDialog(project: Project?, override var state: TSOSessionParams) .also { regionField = it.component resizableRow() - it.verticalAlign(VerticalAlign.FILL) - it.horizontalAlign(HorizontalAlign.FILL) + it.align(AlignX.FILL.plus(AlignY.FILL)) }.validationOnInput { validateForBlank(it) ?: validateForPositiveInteger(it) } } row { From 0b8816f7f8d539d00a0d477db3f720e902ee3a1b Mon Sep 17 00:00:00 2001 From: Katsiaryna Tsytsenia Date: Wed, 3 Apr 2024 18:05:54 +0200 Subject: [PATCH 32/34] IJMP-1324 The minimum column size has been set for the settings fields --- .../formainframe/common/ui/ValidatingTableView.kt | 5 +++++ .../eu/ibagroup/formainframe/config/ws/ui/UrlColumn.kt | 8 -------- .../eu/ibagroup/formainframe/config/ws/ui/WSNameColumn.kt | 4 ---- 3 files changed, 5 insertions(+), 12 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/common/ui/ValidatingTableView.kt b/src/main/kotlin/eu/ibagroup/formainframe/common/ui/ValidatingTableView.kt index 433b0ee3c..893308404 100755 --- a/src/main/kotlin/eu/ibagroup/formainframe/common/ui/ValidatingTableView.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/common/ui/ValidatingTableView.kt @@ -19,6 +19,11 @@ class ValidatingTableView( model: ValidatingListTableModel, val disposable: Disposable ) : TableView(model) { + init { + for (column in columnModel.columns) { + column.minWidth = 150 + } + } // /** // * Get cell renderer with the changed cell size for the cells with default cell editor diff --git a/src/main/kotlin/eu/ibagroup/formainframe/config/ws/ui/UrlColumn.kt b/src/main/kotlin/eu/ibagroup/formainframe/config/ws/ui/UrlColumn.kt index 7e0756ee3..b6d65741a 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/config/ws/ui/UrlColumn.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/config/ws/ui/UrlColumn.kt @@ -43,14 +43,6 @@ class UrlColumn( return false } - /** Returns width of url column - * @param table column of table in GUI - * @return width of column in pixels - */ - override fun getWidth(table: JTable?): Int { - return 270 - } - /** * Returns instance of renderer object * @param item instance of working set config diff --git a/src/main/kotlin/eu/ibagroup/formainframe/config/ws/ui/WSNameColumn.kt b/src/main/kotlin/eu/ibagroup/formainframe/config/ws/ui/WSNameColumn.kt index c31c95942..3fa90e99b 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/config/ws/ui/WSNameColumn.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/config/ws/ui/WSNameColumn.kt @@ -61,10 +61,6 @@ class WSNameColumn(private val wsProvider: () -> Li return false } - override fun getWidth(table: JTable?): Int { - return 200 - } - /** * Overloaded setter method. Sets a new name in the working set config. * @param item working set config [WorkingSetConfig]. From e3e32dd3f31f7e7ceda070dbbfc4256558ce808c Mon Sep 17 00:00:00 2001 From: Arseni Tsikhamirau Date: Mon, 15 Apr 2024 15:27:37 +0200 Subject: [PATCH 33/34] IJMP-1644-TSO-session-reopen-button --- .../ui/build/tso/TSOWindowFactory.kt | 226 +++++++++--------- .../ui/build/tso/ui/TSOConsoleView.kt | 17 +- 2 files changed, 129 insertions(+), 114 deletions(-) diff --git a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/TSOWindowFactory.kt b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/TSOWindowFactory.kt index a2a1a6b4f..377fccd7e 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/TSOWindowFactory.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/TSOWindowFactory.kt @@ -35,24 +35,35 @@ import org.zowe.kotlinsdk.TsoResponse import java.net.ConnectException /** - * Interface class which represents callable methods for topics + * Interface class which represents create topic handler */ -interface TSOSessionHandler { +interface TSOSessionCreateHandler { /** * Function which is called when TSO session is created * @param project - a root project * @param newSession - an instance of config wrapper for new TSO session created */ - fun create (project: Project, newSession: TSOConfigWrapper) + fun create(project: Project, newSession: TSOConfigWrapper) +} +/** + * Interface class which represents reconnect topic handler + */ +interface TSOSessionReconnectHandler { /** * Function which is called when we need to reconnect to the disconnected TSO session * @param project - a root project * @param console - an instance of TSO console view * @param oldSession - an instance of TSO session which we want to reconnect to */ - fun reconnect (project: Project, console: TSOConsoleView, oldSession: TSOConfigWrapper) + fun reconnect(project: Project, console: TSOConsoleView, oldSession: TSOConfigWrapper) +} + +/** + * Interface class which represents process command topic handler + */ +interface TSOSessionProcessCommandHandler { /** * Function which is called when we want to process an entered command. @@ -65,7 +76,7 @@ interface TSOSessionHandler { * @param messageData - message data we want to execute * @param processHandler - process handler to display the result of response message in tool window */ - fun processCommand ( + fun processCommand( project: Project, console: TSOConsoleView, session: TSOConfigWrapper, @@ -74,6 +85,12 @@ interface TSOSessionHandler { messageData: MessageData, processHandler: ProcessHandler ) +} + +/** + * Interface class which represents close topic handler + */ +interface TSOSessionCloseHandler { /** * Function which is called when we want to close currently running TSO session @@ -85,17 +102,32 @@ interface TSOSessionHandler { fun close (project: Project, session: TSOConfigWrapper) } +/** + * Interface class which represents reopen topic handler + */ +interface TSOSessionReopenHandler { + /** + * Function which is called when reopen session event is triggered + * @param project + * @param console + */ + fun reopen(project: Project, console: TSOConsoleView) +} + @JvmField -val SESSION_ADDED_TOPIC = Topic.create("tsoSessionAdded", TSOSessionHandler::class.java) +val SESSION_ADDED_TOPIC = Topic.create("tsoSessionAdded", TSOSessionCreateHandler::class.java) @JvmField -val SESSION_RECONNECT_TOPIC = Topic.create("tsoSessionReconnect", TSOSessionHandler::class.java) +val SESSION_RECONNECT_TOPIC = Topic.create("tsoSessionReconnect", TSOSessionReconnectHandler::class.java) @JvmField -val SESSION_COMMAND_ENTERED = Topic.create("tsoSessionCommandTyped", TSOSessionHandler::class.java) +val SESSION_COMMAND_ENTERED = Topic.create("tsoSessionCommandTyped", TSOSessionProcessCommandHandler::class.java) @JvmField -val SESSION_CLOSED_TOPIC = Topic.create("tsoSessionClosed", TSOSessionHandler::class.java) +val SESSION_CLOSED_TOPIC = Topic.create("tsoSessionClosed", TSOSessionCloseHandler::class.java) + +@JvmField +val SESSION_REOPEN_TOPIC = Topic.create("tsoSessionReopen", TSOSessionReopenHandler::class.java) /** * Factory class for building an instance of TSO tool window when TSO session is created @@ -110,20 +142,6 @@ class TSOWindowFactory : ToolWindowFactory { private var currentTsoSession: TSOConfigWrapper? = null private val tsoSessionToConfigMap = mutableMapOf() - /** - * Getter for TSO session wrapper class for each TSO session created - */ - fun getTsoSession(): TSOConfigWrapper? { - return currentTsoSession - } - - /** - * Getter for TSO config map which contains all the TSO session currently created and active - */ - fun getTsoSessionConfigMap(): Map { - return tsoSessionToConfigMap - } - /** * Method is used to parse response for every TSO session created * @param tsoResponse - response from TSO session @@ -184,16 +202,7 @@ class TSOWindowFactory : ToolWindowFactory { contentManager.addContent(content) contentManager.setSelectedContent(content) - val component = toolWindow.contentManager.selectedContent?.component as TSOConsoleView - val processHandler = component.getProcessHandler() - processHandler.notifyTextAvailable(parseTSODataResponse(tsoSession.getTSOResponse()), ProcessOutputType.STDOUT) - - while (tsoSession.getTSOResponseMessageQueue().last().tsoPrompt == null) { - val response = getTsoMessageQueue(tsoSession) - processHandler.notifyTextAvailable(parseTSODataResponse(response), ProcessOutputType.STDOUT) - tsoSession.setTSOResponseMessageQueue(response.tsoData) - } - processHandler.notifyTextAvailable("> ", ProcessOutputType.STDOUT) + fetchNewSessionResponseMessages(tsoContent, tsoSession) } } @@ -227,92 +236,46 @@ class TSOWindowFactory : ToolWindowFactory { subscribe( project = project, topic = SESSION_ADDED_TOPIC, - handler = object : TSOSessionHandler { + handler = object : TSOSessionCreateHandler { override fun create(project: Project, newSession: TSOConfigWrapper) { - val servletKey = newSession.getTSOResponse().servletKey - if (servletKey != null) { - tsoSessionToConfigMap[servletKey] = newSession.getTSOSessionParams() - addToolWindowContent(project, toolWindow, newSession) - } + val servletKey = newSession.getTSOResponse().servletKey ?: throw Exception("Cannot create a new session, because new session ID was not recognized.") + addToolWindowContent(project, toolWindow, newSession) + tsoSessionToConfigMap[servletKey] = newSession.getTSOSessionParams() } - - override fun reconnect(project: Project, console: TSOConsoleView, oldSession: TSOConfigWrapper) {} - - override fun processCommand( - project: Project, - console: TSOConsoleView, - session: TSOConfigWrapper, - command: String, - messageType: MessageType, - messageData: MessageData, - processHandler: ProcessHandler - ) {} - - override fun close(project: Project, session: TSOConfigWrapper) {} } ) + subscribe( project = project, topic = SESSION_RECONNECT_TOPIC, - handler = object : TSOSessionHandler { - - override fun create(project: Project, newSession: TSOConfigWrapper) {} + handler = object : TSOSessionReconnectHandler { override fun reconnect(project: Project, console: TSOConsoleView, oldSession: TSOConfigWrapper) { val oldServletKey = oldSession.getTSOResponse().servletKey - val params = oldSession.getTSOSessionParams() if (tsoSessionToConfigMap[oldServletKey] != null) { - tsoSessionToConfigMap.remove(oldServletKey) - val tsoResponse = service().performOperation( - TsoOperation( - params, - TsoOperationMode.START - ) - ) - val servletKey = tsoResponse.servletKey - if (servletKey != null) { - val config = TSOConfigWrapper(params, tsoResponse) - currentTsoSession = config - console.setTsoSession(config) - tsoSessionToConfigMap[servletKey] = config.getTSOSessionParams() - console.getProcessHandler() - .notifyTextAvailable(parseTSODataResponse(config.getTSOResponse()), ProcessOutputType.STDOUT) - while (config.getTSOResponseMessageQueue().last().tsoPrompt == null) { - val response = getTsoMessageQueue(config) - console.getProcessHandler() - .notifyTextAvailable(parseTSODataResponse(response), ProcessOutputType.STDOUT) - config.setTSOResponseMessageQueue(response.tsoData) - } + val newSessionConfig = createNewSessionFromOldConfig(oldSession) + if (newSessionConfig != null) { + val sessionResponse = newSessionConfig.getTSOResponse() + val newServletKey = sessionResponse.servletKey ?: throw Exception("Cannot reconnect to the session, because new session ID was not recognized.") + currentTsoSession = newSessionConfig + console.setTsoSession(newSessionConfig) + fetchNewSessionResponseMessages(console, newSessionConfig) + tsoSessionToConfigMap.remove(oldServletKey) + tsoSessionToConfigMap[newServletKey] = newSessionConfig.getTSOSessionParams() } else { - throw Exception("Cannot reconnect to the session, because new session ID was not recognized.") + throw Exception("Cannot reconnect to the session, because new session config is missing.") } } else { throw Exception("Cannot reconnect to the session, because session ID was not found.") } } - - override fun processCommand( - project: Project, - console: TSOConsoleView, - session: TSOConfigWrapper, - command: String, - messageType: MessageType, - messageData: MessageData, - processHandler: ProcessHandler - ) {} - - override fun close(project: Project, session: TSOConfigWrapper) {} } ) subscribe( project = project, topic = SESSION_COMMAND_ENTERED, - handler = object : TSOSessionHandler { - - override fun create(project: Project, newSession: TSOConfigWrapper) {} - - override fun reconnect(project: Project, console: TSOConsoleView, oldSession: TSOConfigWrapper) {} + handler = object : TSOSessionProcessCommandHandler { override fun processCommand( project: Project, @@ -358,27 +321,12 @@ class TSOWindowFactory : ToolWindowFactory { } } } - - override fun close(project: Project, session: TSOConfigWrapper) {} } ) subscribe( project = project, topic = SESSION_CLOSED_TOPIC, - handler = object : TSOSessionHandler { - override fun create(project: Project, newSession: TSOConfigWrapper) {} - - override fun reconnect(project: Project, console: TSOConsoleView, oldSession: TSOConfigWrapper) {} - - override fun processCommand( - project: Project, - console: TSOConsoleView, - session: TSOConfigWrapper, - command: String, - messageType: MessageType, - messageData: MessageData, - processHandler: ProcessHandler - ) {} + handler = object : TSOSessionCloseHandler { override fun close(project: Project, session: TSOConfigWrapper) { /** @@ -407,5 +355,59 @@ class TSOWindowFactory : ToolWindowFactory { } } ) + + subscribe( + project = project, + topic = SESSION_REOPEN_TOPIC, + handler = object : TSOSessionReopenHandler { + override fun reopen(project: Project, console: TSOConsoleView) { + val oldConfig = console.getTsoSession() + runInEdt { + toolWindow.contentManager.apply { + selectedContent?.let { removeContent(it, true) } + } + } + val newConfig = createNewSessionFromOldConfig(oldConfig) ?: throw Exception("Unable to establish a new TSO session with parameters: ${oldConfig.getTSOSessionParams()}") + sendTopic(SESSION_ADDED_TOPIC).create(project, newConfig) + } + } + ) + } + + /** + * Method is used to create a new session config from the parameters of the old session config + * @param oldConfig + * @return a new instance of TSOConfigWrapper if a new session was successfully created, null otherwise + */ + private fun createNewSessionFromOldConfig(oldConfig: TSOConfigWrapper) : TSOConfigWrapper? { + val params = oldConfig.getTSOSessionParams() + val tsoResponse = service().performOperation( + TsoOperation( + params, + TsoOperationMode.START + ) + ) + val servletKey = tsoResponse.servletKey + return if (servletKey != null) TSOConfigWrapper(params, tsoResponse) else null } + + /** + * Method is used to fetch and display the welcome messages when a new session is created and a tool window content is added + * @param console + * @param newConfig + */ + private fun fetchNewSessionResponseMessages(console: TSOConsoleView, newConfig: TSOConfigWrapper) { + val sessionResponse = newConfig.getTSOResponse() + val processHandler = console.getProcessHandler() + processHandler + .notifyTextAvailable(parseTSODataResponse(sessionResponse), ProcessOutputType.STDOUT) + while (newConfig.getTSOResponseMessageQueue().last().tsoPrompt == null) { + val response = getTsoMessageQueue(newConfig) + processHandler + .notifyTextAvailable(parseTSODataResponse(response), ProcessOutputType.STDOUT) + newConfig.setTSOResponseMessageQueue(response.tsoData) + } + processHandler.notifyTextAvailable("> ", ProcessOutputType.STDOUT) + } + } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt index 83c903374..a5f081f65 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt @@ -10,11 +10,11 @@ package eu.ibagroup.formainframe.ui.build.tso.ui -//import com.intellij.ui.layout.cellPanel import com.intellij.execution.process.ProcessEvent import com.intellij.execution.process.ProcessHandler import com.intellij.execution.process.ProcessListener import com.intellij.execution.ui.ExecutionConsole +import com.intellij.openapi.progress.runBackgroundableTask import com.intellij.openapi.project.Project import com.intellij.openapi.util.Disposer import com.intellij.openapi.util.Key @@ -28,7 +28,7 @@ import eu.ibagroup.formainframe.common.isDebugModeEnabled import eu.ibagroup.formainframe.dataops.operations.MessageData import eu.ibagroup.formainframe.dataops.operations.MessageType import eu.ibagroup.formainframe.ui.build.TerminalCommandReceiver -import eu.ibagroup.formainframe.ui.build.tso.SESSION_COMMAND_ENTERED +import eu.ibagroup.formainframe.ui.build.tso.* import eu.ibagroup.formainframe.ui.build.tso.config.TSOConfigWrapper import eu.ibagroup.formainframe.ui.build.tso.utils.InputRecognizer import eu.ibagroup.formainframe.utils.log @@ -50,6 +50,7 @@ class TSOConsoleView( private lateinit var tsoMessageType: MessageType private lateinit var tsoDataType: MessageData private lateinit var cancelCommandButton: JButton + private lateinit var reopenSessionButton: JButton private val tsoWidthGroup: String = "TSO_WIDTH_GROUP" private val tsoMessageTypes: List = @@ -93,6 +94,17 @@ class TSOConsoleView( tsoDataType = it.component.item } }.visible(debugMode) + row { + button("Reopen Session") { + runBackgroundableTask("Re-opening TSO session", project) { + sendTopic(SESSION_REOPEN_TOPIC).reopen(project, this@TSOConsoleView) + } + }.also { + reopenSessionButton = it.component + reopenSessionButton.apply { toolTipText = "The server tries to re-open the current session in case of some troubles (for example console hangs)" } + } + .widthGroup(tsoWidthGroup) + } row { button("Cancel Command (PA1)") { log.info("CANCEL COMMAND (PA1)") @@ -107,6 +119,7 @@ class TSOConsoleView( }.also { cancelCommandButton = it.component } + .widthGroup(tsoWidthGroup) } }.also { it.border = JBEmptyBorder(10, 15, 10, 15) From f483ccef0264f59c602d45f813dae5479f6b4b9c Mon Sep 17 00:00:00 2001 From: Uladzislau Date: Tue, 16 Apr 2024 22:15:45 +0200 Subject: [PATCH 34/34] Release v2.0.0 branch is prepared and fixed Signed-off-by: Uladzislau --- build.gradle.kts | 31 ++++++++++++------- .../config/connect/Credentials.kt | 10 +++--- .../editor/FileEditorEventsListener.kt | 11 +++++-- .../editor/MFPastePreprocessor.kt | 1 - .../explorer/actions/PurgeJobAction.kt | 9 ++++-- .../ui/build/tso/ui/TSOConsoleView.kt | 23 +++++++------- 6 files changed, 50 insertions(+), 35 deletions(-) diff --git a/build.gradle.kts b/build.gradle.kts index 2873ea13a..67aeac0b4 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -37,6 +37,13 @@ version = properties("pluginVersion").get() val remoteRobotVersion = "0.11.22" val okHttp3Version = "4.12.0" val kotestVersion = "5.8.1" +val retrofit2Vertion = "2.11.0" +val junitVersion = "5.10.2" +val mockkVersion = "1.13.10" +val ibmMqVersion = "9.3.5.0" +val jGraphTVersion = "1.5.2" +val zoweKotlinSdkVersion = "0.4.0" +val javaAnalyticsVersion = "3.5.1" repositories { mavenCentral() @@ -62,25 +69,25 @@ java { } dependencies { - implementation(group = "com.squareup.retrofit2", name = "retrofit", version = "2.9.0") - implementation("com.squareup.retrofit2:converter-gson:2.9.0") - implementation("com.squareup.retrofit2:converter-scalars:2.9.0") + implementation(group = "com.squareup.retrofit2", name = "retrofit", version = retrofit2Vertion) + implementation("com.squareup.retrofit2:converter-gson:$retrofit2Vertion") + implementation("com.squareup.retrofit2:converter-scalars:$retrofit2Vertion") implementation("com.squareup.okhttp3:okhttp:$okHttp3Version") - implementation("org.jgrapht:jgrapht-core:1.5.2") - implementation("org.zowe.sdk:zowe-kotlin-sdk:0.4.0") - implementation("com.segment.analytics.java:analytics:3.5.0") - implementation("com.ibm.mq:com.ibm.mq.allclient:9.3.4.1") - implementation("org.junit.jupiter:junit-jupiter-params:5.9.2") - testImplementation("org.junit.jupiter:junit-jupiter-api:5.10.1") - testImplementation("io.mockk:mockk:1.13.9") + implementation("org.jgrapht:jgrapht-core:$jGraphTVersion") + implementation("org.zowe.sdk:zowe-kotlin-sdk:$zoweKotlinSdkVersion") + implementation("com.segment.analytics.java:analytics:$javaAnalyticsVersion") + implementation("com.ibm.mq:com.ibm.mq.allclient:$ibmMqVersion") + implementation("org.junit.jupiter:junit-jupiter-params:$junitVersion") + testImplementation("org.junit.jupiter:junit-jupiter-api:$junitVersion") + testImplementation("io.mockk:mockk:$mockkVersion") testImplementation("io.kotest:kotest-assertions-core:$kotestVersion") testImplementation("io.kotest:kotest-runner-junit5:$kotestVersion") testImplementation("com.intellij.remoterobot:remote-robot:$remoteRobotVersion") testImplementation("com.intellij.remoterobot:remote-fixtures:$remoteRobotVersion") testImplementation("com.squareup.okhttp3:mockwebserver:$okHttp3Version") testImplementation("com.squareup.okhttp3:okhttp-tls:$okHttp3Version") - testRuntimeOnly("org.junit.jupiter:junit-jupiter-engine:5.10.1") - testRuntimeOnly("org.junit.vintage:junit-vintage-engine:5.10.1") + testRuntimeOnly("org.junit.jupiter:junit-jupiter-engine:$junitVersion") + testRuntimeOnly("org.junit.vintage:junit-vintage-engine:$junitVersion") } intellij { diff --git a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/Credentials.kt b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/Credentials.kt index 815bbe949..e57018345 100755 --- a/src/main/kotlin/eu/ibagroup/formainframe/config/connect/Credentials.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/config/connect/Credentials.kt @@ -38,7 +38,7 @@ class Credentials { if (other == null || javaClass != other.javaClass) return false val that = other as Credentials if (configUuid != that.configUuid) return false - return if (username != that.username) false else password == that.password + return username == that.username && password == that.password } override fun hashCode(): Int { @@ -51,9 +51,9 @@ class Credentials { override fun toString(): String { return "Credentials{" + - "connectionConfigUuid='" + configUuid + '\'' + - ", username='" + username + '\'' + - ", password='" + password + '\'' + - '}' + "connectionConfigUuid='" + configUuid + '\'' + + ", username='" + username + '\'' + + ", password='" + password + '\'' + + '}' } } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/editor/FileEditorEventsListener.kt b/src/main/kotlin/eu/ibagroup/formainframe/editor/FileEditorEventsListener.kt index 8d3829e6a..1112be42e 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/editor/FileEditorEventsListener.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/editor/FileEditorEventsListener.kt @@ -10,7 +10,6 @@ package eu.ibagroup.formainframe.editor -import com.intellij.openapi.actionSystem.ex.ActionUtil import com.intellij.openapi.components.service import com.intellij.openapi.editor.ex.EditorEx import com.intellij.openapi.fileEditor.FileEditorManager @@ -19,8 +18,14 @@ import com.intellij.openapi.progress.runModalTask import com.intellij.openapi.vfs.VirtualFile import eu.ibagroup.formainframe.config.ConfigService import eu.ibagroup.formainframe.dataops.DataOpsManager -import eu.ibagroup.formainframe.dataops.content.synchronizer.* -import eu.ibagroup.formainframe.utils.* +import eu.ibagroup.formainframe.dataops.content.synchronizer.AutoSyncFileListener +import eu.ibagroup.formainframe.dataops.content.synchronizer.DocumentedSyncProvider +import eu.ibagroup.formainframe.dataops.content.synchronizer.SaveStrategy +import eu.ibagroup.formainframe.utils.checkEncodingCompatibility +import eu.ibagroup.formainframe.utils.runReadActionInEdtAndWait +import eu.ibagroup.formainframe.utils.runWriteActionInEdtAndWait +import eu.ibagroup.formainframe.utils.sendTopic +import eu.ibagroup.formainframe.utils.showSaveAnywayDialog import eu.ibagroup.formainframe.vfs.MFVirtualFile /** diff --git a/src/main/kotlin/eu/ibagroup/formainframe/editor/MFPastePreprocessor.kt b/src/main/kotlin/eu/ibagroup/formainframe/editor/MFPastePreprocessor.kt index d4439aca9..9efe12adb 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/editor/MFPastePreprocessor.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/editor/MFPastePreprocessor.kt @@ -20,7 +20,6 @@ import com.intellij.openapi.util.Ref import com.intellij.psi.PsiFile import eu.ibagroup.formainframe.utils.`is` import eu.ibagroup.formainframe.vfs.MFVirtualFile -import org.jetbrains.annotations.ApiStatus import java.awt.datatransfer.DataFlavor import java.awt.datatransfer.Transferable diff --git a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt index 434022c4f..8dfbf6ec3 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/explorer/actions/PurgeJobAction.kt @@ -25,11 +25,11 @@ import eu.ibagroup.formainframe.dataops.DataOpsManager import eu.ibagroup.formainframe.dataops.attributes.RemoteJobAttributes import eu.ibagroup.formainframe.dataops.operations.jobs.BasicPurgeJobParams import eu.ibagroup.formainframe.dataops.operations.jobs.PurgeJobOperation -import eu.ibagroup.formainframe.explorer.ui.ExplorerTreeNode -import eu.ibagroup.formainframe.explorer.ui.FetchNode import eu.ibagroup.formainframe.explorer.ui.JesExplorerView import eu.ibagroup.formainframe.explorer.ui.JesFilterNode +import eu.ibagroup.formainframe.explorer.ui.JesWsNode import eu.ibagroup.formainframe.explorer.ui.JobNode +import eu.ibagroup.formainframe.explorer.ui.NodeData import eu.ibagroup.formainframe.explorer.ui.getExplorerView import eu.ibagroup.formainframe.ui.build.jobs.JOBS_LOG_VIEW import eu.ibagroup.formainframe.ui.build.jobs.JobBuildTreeView @@ -134,7 +134,10 @@ class PurgeJobAction : AnAction() { if (foundJobsWaitingInPurgeQueue.isNotEmpty()) { foundJobsWaitingInPurgeQueue.clear() val filterRefreshSize = jobsByFilterWaitingPurgeMap[filterNode]!!.size - runRefreshByFilter(filterNode, if (filterRefreshSize == 1) (filterRefreshSize.toLong() * 1000) else (filterRefreshSize / 2).toLong() * 1000) + runRefreshByFilter( + filterNode, + if (filterRefreshSize == 1) (filterRefreshSize.toLong() * 1000) else (filterRefreshSize / 2).toLong() * 1000 + ) } else { filterNode.cleanCache() } diff --git a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt index 6efecfdfb..766855a61 100644 --- a/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt +++ b/src/main/kotlin/eu/ibagroup/formainframe/ui/build/tso/ui/TSOConsoleView.kt @@ -51,6 +51,7 @@ class TSOConsoleView( private lateinit var tsoMessageTypeBox: ComboBox private lateinit var tsoDataTypeBox: ComboBox private lateinit var cancelCommandButton: JButton + private lateinit var reopenSessionButton: JButton private val tsoWidthGroup: String = "TSO_WIDTH_GROUP" private val tsoMessageTypes: List = @@ -93,6 +94,17 @@ class TSOConsoleView( tsoDataTypeBox = it.component } }.visible(debugMode) + row { + button("Reopen Session") { + runBackgroundableTask("Re-opening TSO session", project) { + sendTopic(SESSION_REOPEN_TOPIC).reopen(project, this@TSOConsoleView) + } + }.also { + reopenSessionButton = it.component + reopenSessionButton.apply { toolTipText = "The server tries to re-open the current session in case of some troubles (for example console hangs)" } + } + .widthGroup(tsoWidthGroup) + } row { button("Cancel Command (PA1)") { log.info("CANCEL COMMAND (PA1)") @@ -108,17 +120,6 @@ class TSOConsoleView( cancelCommandButton = it.component } .widthGroup(tsoWidthGroup) - }.visible(debugMode) - row { - button("Reopen Session") { - runBackgroundableTask("Re-opening TSO session", project) { - sendTopic(SESSION_REOPEN_TOPIC).reopen(project, this@TSOConsoleView) - } - }.also { - reopenSessionButton = it.component - reopenSessionButton.apply { toolTipText = "The server tries to re-open the current session in case of some troubles (for example console hangs)" } - } - .widthGroup(tsoWidthGroup) } }.also { it.border = JBEmptyBorder(10, 15, 10, 15)