Skip to content

Commit

Permalink
IJMP-1522 Added the ability to create a Zote Team Configuration file
Browse files Browse the repository at this point in the history
Zowe Team Configuration file can be created for the current project directly from the plugin

New:
      src/main/kotlin/org/zowe/explorer/config/connect/ui/zosmf/ZoweTeamConfigDialog.kt
      src/main/kotlin/org/zowe/explorer/explorer/actions/AddZoweTeamConfigAction.kt
      src/main/resources/files/zowe.config.json
      src/main/resources/files/zowe.schema.json
      src/main/kotlin/org/zowe/explorer/config/connect/ui/zosmf/CommonConnectionDialog.kt
Modified:
      build.gradle.kts
      src/main/kotlin/org/zowe/explorer/config/connect/ui/zosmf/ConnectionDialog.kt
      src/main/kotlin/org/zowe/explorer/config/connect/ui/zosmf/ZOSMFConnectionConfigurable.kt
      src/main/kotlin/org/zowe/explorer/zowe/ZoweStartupActivity.kt
      src/main/kotlin/org/zowe/explorer/zowe/service/ZoweConfigService.kt
      src/main/kotlin/org/zowe/explorer/zowe/service/ZoweConfigServiceImpl.kt
      src/main/kotlin/org/zowe/explorer/zowe/service/ZoweFileListener.kt
      src/main/resources/META-INF/plugin.xml

Signed-off-by: Katsiaryna Tsytsenia <[email protected]>
  • Loading branch information
Katsiaryna Tsytsenia authored and Katsiaryna Tsytsenia committed Apr 23, 2024
1 parent 08f24b2 commit ac1feee
Show file tree
Hide file tree
Showing 13 changed files with 1,151 additions and 246 deletions.
2 changes: 1 addition & 1 deletion build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ 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 zoweKotlinSdkVersion = "0.5.0"
val javaKeytarVersion = "1.0.0"

repositories {
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,236 @@
package org.zowe.explorer.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.Messages
import com.intellij.ui.components.JBCheckBox
import com.intellij.ui.components.JBTextField
import org.zowe.explorer.common.message
import org.zowe.explorer.common.ui.DialogMode
import org.zowe.explorer.common.ui.StatefulDialog
import org.zowe.explorer.common.ui.showUntilDone
import org.zowe.explorer.config.connect.ConnectionConfig
import org.zowe.explorer.config.connect.CredentialService
import org.zowe.explorer.config.connect.whoAmI
import org.zowe.explorer.dataops.DataOpsManager
import org.zowe.explorer.dataops.operations.InfoOperation
import org.zowe.explorer.dataops.operations.ZOSInfoOperation
import org.zowe.explorer.utils.crudable.Crudable
import org.zowe.explorer.utils.runTask
import org.zowe.kotlinsdk.annotations.ZVersion
import java.awt.Component
import java.util.*
import javax.swing.JCheckBox
import javax.swing.JTextField

abstract class CommonConnectionDialog(
protected val crudable: Crudable,
override var state: ConnectionDialogState = ConnectionDialogState(),
val project: Project? = null
) : StatefulDialog<ConnectionDialogState>(project) {


protected lateinit var urlTextField: JTextField

protected lateinit var sslCheckbox: JCheckBox

/**
* 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
}

/**
* 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: CommonConnectionDialog): 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
}

/**
* 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
*/
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
}
}

abstract fun createConnectionDialog(
crudable: Crudable,
state: ConnectionDialogState = ConnectionDialogState(),
project: Project? = null
): CommonConnectionDialog

/** 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.
* We do not need to worry about choosing z/OS version manually from combo box.
* */
fun showAndTestConnectionCommon(
crudable: Crudable,
parentComponent: Component? = null,
project: Project? = null,
initialState: ConnectionDialogState
): ConnectionDialogState? {
var connectionDialog = this
val initState = initialState.clone()
return showUntilDone(
initialState = initialState,
factory = { connectionDialog },
test = { state ->

if (validateSecureConnectionUsage(connectionDialog)) {
state.connectionUrl = connectionDialog.urlTextField.text
state.isAllowSsl = connectionDialog.sslCheckbox.isSelected
connectionDialog = createConnectionDialog(
crudable,
initialState,
project
)
return@showUntilDone false
}

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
)
CredentialService.instance.setCredentials(
connectionConfigUuid = newState.connectionUuid,
username = newState.username,
password = newState.password
)
} else {
state.initEmptyUuids(crudable)
newTestedConnConfig = state.connectionConfig
CredentialService.instance.setCredentials(
connectionConfigUuid = state.connectionUuid,
username = state.username,
password = state.password
)
}
val throwable = runTask(title = "Testing Connection to ${newTestedConnConfig.url}", project = project) {
return@runTask try {
runCatching {
service<DataOpsManager>().performOperation(InfoOperation(newTestedConnConfig), it)
}.onSuccess {
state.owner = whoAmI(newTestedConnConfig) ?: ""
val systemInfo =
service<DataOpsManager>().performOperation(ZOSInfoOperation(newTestedConnConfig))
state.zVersion = when (systemInfo.zosVersion) {
"04.25.00" -> ZVersion.ZOS_2_2
"04.26.00" -> ZVersion.ZOS_2_3
"04.27.00" -> ZVersion.ZOS_2_4
"04.28.00" -> ZVersion.ZOS_2_5
else -> ZVersion.ZOS_2_1
}
}.onFailure {
throw it
}
null
} catch (t: Throwable) {
t
}
}
if (throwable != null) {
state.mode = DialogMode.UPDATE
val confirmMessage = "Do you want to add it anyway?"
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) {
"$tMessage\n\n$confirmMessage"
} else {
confirmMessage
}
val addAnyway = MessageDialogBuilder
.yesNo(
title = "Error Creating Connection",
message = message
).icon(AllIcons.General.ErrorDialog)
.run {
if (parentComponent != null) {
ask(parentComponent)
} else {
ask(project)
}
}
connectionDialog = createConnectionDialog(
crudable,
initialState,
project
)
addAnyway
} else {
true
}
}
)
}

}
Loading

0 comments on commit ac1feee

Please sign in to comment.