2018-09-21 16:55:31 +02:00
|
|
|
import static com.sap.piper.Prerequisites.checkScript
|
|
|
|
|
2018-08-09 11:35:33 +02:00
|
|
|
import com.sap.piper.Utils
|
2018-06-20 11:46:28 +02:00
|
|
|
import groovy.transform.Field
|
|
|
|
|
2019-03-28 11:10:01 +02:00
|
|
|
import com.sap.piper.GenerateDocumentation
|
2018-07-06 09:05:26 +02:00
|
|
|
import com.sap.piper.ConfigurationHelper
|
2018-06-20 11:46:28 +02:00
|
|
|
import com.sap.piper.cm.ChangeManagement
|
2018-09-24 14:06:48 +02:00
|
|
|
import com.sap.piper.cm.BackendType
|
2018-06-20 11:46:28 +02:00
|
|
|
import com.sap.piper.cm.ChangeManagementException
|
|
|
|
|
|
|
|
import hudson.AbortException
|
|
|
|
|
2018-09-28 15:32:58 +02:00
|
|
|
import static com.sap.piper.cm.StepHelpers.getTransportRequestId
|
2018-09-28 15:54:20 +02:00
|
|
|
import static com.sap.piper.cm.StepHelpers.getChangeDocumentId
|
2018-09-28 14:43:51 +02:00
|
|
|
import static com.sap.piper.cm.StepHelpers.getBackendTypeAndLogInfoIfCMIntegrationDisabled
|
2018-06-20 11:46:28 +02:00
|
|
|
|
2018-11-29 10:54:05 +02:00
|
|
|
@Field def STEP_NAME = getClass().getName()
|
2018-06-20 11:46:28 +02:00
|
|
|
|
2018-10-25 14:22:04 +02:00
|
|
|
@Field Set GENERAL_CONFIG_KEYS = [
|
2019-03-28 11:10:01 +02:00
|
|
|
'changeManagement',
|
|
|
|
/**
|
|
|
|
* @see checkChangeInDevelopment
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'changeDocumentLabel',
|
|
|
|
/**
|
|
|
|
* Defines where the transport request is created, e.g. SAP Solution Manager, ABAP System.
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
* @possibleValues `SOLMAN`, `CTS`, `RFC`
|
|
|
|
*/
|
|
|
|
'type',
|
|
|
|
/**
|
|
|
|
* @see checkChangeInDevelopment
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'clientOpts',
|
|
|
|
/**
|
|
|
|
* @see checkChangeInDevelopment
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'credentialsId',
|
|
|
|
/**
|
|
|
|
* @see checkChangeInDevelopment
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'endpoint',
|
|
|
|
/**
|
|
|
|
* @see checkChangeInDevelopment
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'git/from',
|
|
|
|
/**
|
|
|
|
* @see checkChangeInDevelopment
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'git/to',
|
|
|
|
/**
|
|
|
|
* @see checkChangeInDevelopment
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'git/format',
|
|
|
|
/**
|
|
|
|
* @see transportRequestCreate
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'rfc/developmentInstance',
|
|
|
|
/**
|
|
|
|
* @see transportRequestCreate
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'rfc/developmentClient',
|
|
|
|
/**
|
|
|
|
* A pattern used for identifying lines holding the transport request id.
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
2020-10-21 12:04:11 +02:00
|
|
|
'transportRequestLabel',
|
|
|
|
/**
|
|
|
|
* Some CTS related transport related steps are cm_client based, others are node based.
|
|
|
|
* For the node based steps the docker image is specified here.
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'cts/nodeDocker/image',
|
|
|
|
/**
|
|
|
|
* The ABAP client. Only for `CTS`
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'client',
|
|
|
|
/**
|
|
|
|
* By default we use a standard node docker image and prepare some fiori related packages
|
|
|
|
* before performing the deployment. For that we need to launch the image with root privileges.
|
|
|
|
* After that, before actually performing the deployment we swith to a non root user. This user
|
|
|
|
* can be specified here.
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'cts/osDeployUser',
|
|
|
|
/**
|
|
|
|
* By default we use a standard node docker iamge and prepare some fiori related packages
|
|
|
|
* performing the deployment. The additional dependencies can be provided here. In case you
|
|
|
|
* use an already prepared docker image which contains the required dependencies, the empty
|
2021-01-12 16:15:30 +02:00
|
|
|
* list can be provided here. Caused hereby installing additional dependencies will be skipped.
|
2020-10-21 12:04:11 +02:00
|
|
|
*
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'cts/deployToolDependencies',
|
|
|
|
/**
|
|
|
|
* A list containing additional options for the npm install call. `-g`, `--global` is always assumed.
|
|
|
|
* Can be used for e.g. providing custom registries (`--registry https://your.registry.com`) or
|
|
|
|
* for providing the verbose flag (`--verbose`) for troubleshooting.
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'cts/npmInstallOpts',
|
|
|
|
/**
|
|
|
|
* The file handed over to `fiori deploy` with flag `-c --config`.
|
|
|
|
* @parentConfigKey changeManagement
|
|
|
|
*/
|
|
|
|
'cts/deployConfigFile',
|
2018-06-20 11:46:28 +02:00
|
|
|
]
|
|
|
|
|
2018-10-25 14:22:04 +02:00
|
|
|
@Field Set STEP_CONFIG_KEYS = GENERAL_CONFIG_KEYS.plus([
|
2020-10-21 12:04:11 +02:00
|
|
|
/** The name of the application. `RFC` and `CTS` only. */
|
|
|
|
'applicationName', // RFC, CTS
|
2019-03-28 11:10:01 +02:00
|
|
|
/** The id of the application. Only for `SOLMAN`.*/
|
2019-02-11 11:52:47 +02:00
|
|
|
'applicationId', // SOLMAN
|
2020-10-21 12:04:11 +02:00
|
|
|
/** The application description, `RFC` and `CTS` only. For `CTS`: the desription is only
|
|
|
|
taken into account for a new upload. In case of an update the description will not be
|
|
|
|
updated.
|
|
|
|
*/
|
2019-02-11 11:52:47 +02:00
|
|
|
'applicationDescription',
|
2020-10-21 12:04:11 +02:00
|
|
|
/** The path of the file to upload, Only for `SOLMAN`.*/
|
|
|
|
'filePath', // SOLMAN
|
2019-03-28 11:10:01 +02:00
|
|
|
/** The URL where to find the UI5 package to upload to the transport request. Only for `RFC`. */
|
2019-02-19 10:53:11 +02:00
|
|
|
'applicationUrl', // RFC
|
2019-03-28 11:10:01 +02:00
|
|
|
/** The ABAP package name of your application. */
|
2019-02-11 11:52:47 +02:00
|
|
|
'abapPackage',
|
2019-03-28 11:10:01 +02:00
|
|
|
/** The code page of your ABAP system. E.g. UTF-8. */
|
2019-02-14 10:36:51 +02:00
|
|
|
'codePage', //RFC
|
2021-09-27 15:14:35 +02:00
|
|
|
/** If unix style line endings should be accepted. Only for `RFC`.*/
|
2019-02-14 14:51:24 +02:00
|
|
|
'acceptUnixStyleLineEndings', // RFC
|
2019-03-28 11:10:01 +02:00
|
|
|
/** @see transportRequestCreate */
|
2019-02-15 13:52:44 +02:00
|
|
|
'verbose', // RFC
|
2018-08-03 12:41:50 +02:00
|
|
|
])
|
|
|
|
|
2018-10-25 14:22:04 +02:00
|
|
|
@Field Set PARAMETER_KEYS = STEP_CONFIG_KEYS.plus([
|
2019-03-28 11:10:01 +02:00
|
|
|
/** @see transportRequestCreate */
|
2018-07-12 09:49:18 +02:00
|
|
|
'changeDocumentId',
|
2020-09-30 14:29:46 +02:00
|
|
|
/** The id of the transport request to upload the file. This parameter is only taken into account
|
|
|
|
* when provided via signature to the step.
|
|
|
|
*/
|
2018-07-12 09:49:18 +02:00
|
|
|
'transportRequestId'])
|
|
|
|
|
2019-03-28 11:10:01 +02:00
|
|
|
/** Uploads a file to a Transport Request. */
|
|
|
|
@GenerateDocumentation
|
2020-08-26 15:32:58 +02:00
|
|
|
void call(Map parameters = [:]) {
|
2018-06-20 11:46:28 +02:00
|
|
|
|
|
|
|
handlePipelineStepErrors (stepName: STEP_NAME, stepParameters: parameters) {
|
|
|
|
|
2018-10-31 09:40:12 +02:00
|
|
|
def script = checkScript(this, parameters) ?: this
|
2020-08-26 15:32:58 +02:00
|
|
|
String stageName = parameters.stageName ?: env.STAGE_NAME
|
2018-06-20 11:46:28 +02:00
|
|
|
|
2018-06-29 10:11:46 +02:00
|
|
|
ChangeManagement cm = parameters.cmUtils ?: new ChangeManagement(script)
|
2018-06-20 11:46:28 +02:00
|
|
|
|
2018-10-17 11:05:20 +02:00
|
|
|
ConfigurationHelper configHelper = ConfigurationHelper.newInstance(this)
|
2020-08-26 15:32:58 +02:00
|
|
|
.loadStepDefaults([:], stageName)
|
2018-10-25 14:22:04 +02:00
|
|
|
.mixinGeneralConfig(script.commonPipelineEnvironment, GENERAL_CONFIG_KEYS)
|
|
|
|
.mixinStepConfig(script.commonPipelineEnvironment, STEP_CONFIG_KEYS)
|
2020-08-26 15:32:58 +02:00
|
|
|
.mixinStageConfig(script.commonPipelineEnvironment, stageName, STEP_CONFIG_KEYS)
|
2018-10-25 14:22:04 +02:00
|
|
|
.mixin(parameters, PARAMETER_KEYS)
|
2018-08-15 10:37:34 +02:00
|
|
|
.addIfEmpty('filePath', script.commonPipelineEnvironment.getMtarFilePath())
|
2018-07-13 09:28:41 +02:00
|
|
|
|
|
|
|
Map configuration = configHelper.use()
|
2018-06-20 11:46:28 +02:00
|
|
|
|
2018-09-28 14:43:51 +02:00
|
|
|
BackendType backendType = getBackendTypeAndLogInfoIfCMIntegrationDisabled(this, configuration)
|
|
|
|
if(backendType == BackendType.NONE) return
|
2018-09-28 13:45:26 +02:00
|
|
|
|
|
|
|
configHelper
|
2018-12-14 16:24:03 +02:00
|
|
|
.collectValidationFailures()
|
2018-08-15 10:37:34 +02:00
|
|
|
.withMandatoryProperty('changeManagement/changeDocumentLabel')
|
|
|
|
.withMandatoryProperty('changeManagement/clientOpts')
|
|
|
|
.withMandatoryProperty('changeManagement/credentialsId')
|
|
|
|
.withMandatoryProperty('changeManagement/endpoint')
|
2020-10-21 12:04:11 +02:00
|
|
|
.withMandatoryProperty('changeManagement/client', null, {backendType == BackendType.CTS})
|
2018-09-28 13:45:26 +02:00
|
|
|
.withMandatoryProperty('changeManagement/type')
|
2018-08-15 10:37:34 +02:00
|
|
|
.withMandatoryProperty('changeManagement/git/from')
|
|
|
|
.withMandatoryProperty('changeManagement/git/to')
|
|
|
|
.withMandatoryProperty('changeManagement/git/format')
|
2020-10-21 12:04:11 +02:00
|
|
|
.withMandatoryProperty('filePath', null, { backendType == BackendType.SOLMAN })
|
2019-01-30 11:16:38 +02:00
|
|
|
.withMandatoryProperty('applicationUrl', null, { backendType == BackendType.RFC })
|
2019-02-14 10:36:51 +02:00
|
|
|
.withMandatoryProperty('codePage', null, { backendType == BackendType.RFC })
|
2019-02-14 14:51:24 +02:00
|
|
|
.withMandatoryProperty('acceptUnixStyleLineEndings', null, { backendType == BackendType.RFC })
|
2019-02-07 12:46:45 +02:00
|
|
|
.withMandatoryProperty('changeManagement/rfc/developmentInstance', null, { backendType == BackendType.RFC })
|
2019-02-07 11:11:22 +02:00
|
|
|
.withMandatoryProperty('changeManagement/rfc/developmentClient', null, { backendType == BackendType.RFC })
|
2019-02-12 14:44:09 +02:00
|
|
|
.withMandatoryProperty('changeManagement/rfc/docker/image', null, {backendType == BackendType.RFC})
|
|
|
|
.withMandatoryProperty('changeManagement/rfc/docker/options', null, {backendType == BackendType.RFC})
|
|
|
|
.withMandatoryProperty('changeManagement/rfc/docker/envVars', null, {backendType == BackendType.RFC})
|
2019-02-18 17:59:44 +02:00
|
|
|
.withMandatoryProperty('changeManagement/rfc/docker/pullImage', null, {backendType == BackendType.RFC})
|
2018-12-14 16:24:03 +02:00
|
|
|
.withMandatoryProperty('applicationDescription', null, { backendType == BackendType.RFC })
|
2020-10-21 12:04:11 +02:00
|
|
|
.withMandatoryProperty('abapPackage', null, { backendType in [BackendType.RFC, BackendType.CTS] })
|
2019-02-07 09:26:31 +02:00
|
|
|
.withMandatoryProperty('applicationId', null, {backendType == BackendType.SOLMAN})
|
2020-10-21 12:04:11 +02:00
|
|
|
.withMandatoryProperty('applicationName', null, {backendType in [BackendType.RFC, BackendType.CTS]})
|
2019-02-15 12:05:21 +02:00
|
|
|
.withMandatoryProperty('failOnWarning', null, {backendType == BackendType.RFC})
|
2019-02-15 13:52:44 +02:00
|
|
|
.withMandatoryProperty('verbose', null, {backendType == BackendType.RFC})
|
2018-07-13 09:28:41 +02:00
|
|
|
|
2019-01-21 09:47:34 +02:00
|
|
|
new Utils().pushToSWA([
|
|
|
|
step: STEP_NAME,
|
|
|
|
stepParamKey1: 'changeManagementType',
|
|
|
|
stepParam1: configuration.changeManagement.type,
|
|
|
|
stepParamKey2: 'scriptMissing',
|
|
|
|
stepParam2: parameters?.script == null
|
|
|
|
], configuration)
|
2018-09-28 13:45:26 +02:00
|
|
|
|
2018-09-18 14:49:06 +02:00
|
|
|
def changeDocumentId = null
|
2018-06-20 11:46:28 +02:00
|
|
|
|
2018-09-24 14:06:48 +02:00
|
|
|
if(backendType == BackendType.SOLMAN) {
|
2021-09-27 11:57:20 +02:00
|
|
|
changeDocumentId = getChangeDocumentId(script, configuration)
|
2018-07-10 15:15:54 +02:00
|
|
|
}
|
2018-08-09 11:35:33 +02:00
|
|
|
|
2021-09-27 11:57:20 +02:00
|
|
|
def transportRequestId = getTransportRequestId(script, configuration)
|
2018-07-12 13:55:01 +02:00
|
|
|
|
2018-09-18 14:49:06 +02:00
|
|
|
configHelper
|
|
|
|
.mixin([changeDocumentId: changeDocumentId?.trim() ?: null,
|
|
|
|
transportRequestId: transportRequestId?.trim() ?: null], ['changeDocumentId', 'transportRequestId'] as Set)
|
|
|
|
|
2018-09-24 14:06:48 +02:00
|
|
|
if(backendType == BackendType.SOLMAN) {
|
2018-09-18 14:49:06 +02:00
|
|
|
configHelper
|
|
|
|
.withMandatoryProperty('changeDocumentId',
|
2020-09-30 14:29:46 +02:00
|
|
|
"Change document id not provided (parameter: \'changeDocumentId\' provided to the step call or via commit history).")
|
2018-09-18 14:49:06 +02:00
|
|
|
}
|
2018-07-13 09:28:41 +02:00
|
|
|
configuration = configHelper
|
2019-05-17 13:20:13 +02:00
|
|
|
.withMandatoryProperty('transportRequestId',
|
2020-09-30 14:29:46 +02:00
|
|
|
"Transport request id not provided (parameter: \'transportRequestId\' provided to the step call or via commit history).")
|
2019-05-17 13:20:13 +02:00
|
|
|
.use()
|
2018-06-20 11:46:28 +02:00
|
|
|
|
|
|
|
try {
|
2018-07-16 15:41:46 +02:00
|
|
|
|
2018-12-14 16:24:03 +02:00
|
|
|
switch(backendType) {
|
|
|
|
|
|
|
|
case BackendType.SOLMAN:
|
2020-10-21 12:04:11 +02:00
|
|
|
|
|
|
|
echo "[INFO] Uploading file '${configuration.filePath}' to transport request '${configuration.transportRequestId}'" +
|
|
|
|
" of change document '${configuration.changeDocumentId}'."
|
|
|
|
|
2021-09-21 10:29:43 +02:00
|
|
|
Map paramsUpload = [
|
|
|
|
script: script,
|
2021-09-13 13:31:43 +02:00
|
|
|
filePath: configuration.filePath,
|
|
|
|
uploadCredentialsId: configuration.changeManagement.credentialsId,
|
|
|
|
endpoint: configuration.changeManagement.endpoint,
|
|
|
|
applicationId: configuration.applicationId,
|
|
|
|
changeDocumentId: configuration.changeDocumentId,
|
2021-09-21 10:29:43 +02:00
|
|
|
transportRequestId: configuration.transportRequestId
|
|
|
|
]
|
|
|
|
|
2021-09-29 17:50:45 +02:00
|
|
|
if(configuration.changeManagement.clientOpts) {
|
|
|
|
paramsUpload.cmClientOpts = configuration.changeManagement.clientOpts
|
|
|
|
}
|
2021-09-21 10:29:43 +02:00
|
|
|
paramsUpload = addDockerParams(script, paramsUpload, configuration.changeManagement.solman?.docker)
|
|
|
|
|
|
|
|
transportRequestUploadSOLMAN(paramsUpload)
|
2020-10-21 12:04:11 +02:00
|
|
|
|
|
|
|
echo "[INFO] File '${configuration.filePath}' has been successfully uploaded to transport request '${configuration.transportRequestId}'" +
|
|
|
|
" of change document '${configuration.changeDocumentId}'."
|
|
|
|
|
2018-12-14 16:24:03 +02:00
|
|
|
break
|
|
|
|
case BackendType.CTS:
|
2020-10-21 12:04:11 +02:00
|
|
|
|
|
|
|
echo "[INFO] Uploading application '${configuration.applicationName}' to transport request '${configuration.transportRequestId}'."
|
|
|
|
|
2021-09-21 10:29:43 +02:00
|
|
|
Map paramsUpload = [
|
|
|
|
script: script,
|
2021-09-16 13:18:03 +02:00
|
|
|
transportRequestId: configuration.transportRequestId,
|
|
|
|
endpoint: configuration.changeManagement.endpoint,
|
|
|
|
client: configuration.changeManagement.client,
|
|
|
|
applicationName: configuration.applicationName,
|
|
|
|
description: configuration.applicationDescription,
|
|
|
|
abapPackage: configuration.abapPackage,
|
|
|
|
osDeployUser: configuration.changeManagement.cts.osDeployUser,
|
|
|
|
deployToolDependencies: configuration.changeManagement.cts.deployToolDependencies,
|
|
|
|
npmInstallOpts: configuration.changeManagement.cts.npmInstallOpts,
|
|
|
|
deployConfigFile: configuration.changeManagement.cts.deployConfigFile,
|
2021-09-21 10:29:43 +02:00
|
|
|
uploadCredentialsId: configuration.changeManagement.credentialsId
|
|
|
|
]
|
|
|
|
|
|
|
|
paramsUpload = addDockerParams(script, paramsUpload, configuration.changeManagement.cts?.nodeDocker)
|
|
|
|
|
|
|
|
transportRequestUploadCTS(paramsUpload)
|
2020-10-21 12:04:11 +02:00
|
|
|
|
|
|
|
echo "[INFO] Application '${configuration.applicationName}' has been successfully uploaded to transport request '${configuration.transportRequestId}'."
|
|
|
|
|
2018-12-14 16:24:03 +02:00
|
|
|
break
|
|
|
|
case BackendType.RFC:
|
2019-02-07 11:11:22 +02:00
|
|
|
|
2020-10-21 12:04:11 +02:00
|
|
|
echo "[INFO] Uploading file '${configuration.applicationUrl}' to transport request '${configuration.transportRequestId}'."
|
2021-09-21 10:29:43 +02:00
|
|
|
|
|
|
|
Map paramsUpload = [script: script,
|
2021-09-13 14:14:38 +02:00
|
|
|
transportRequestId: configuration.transportRequestId,
|
|
|
|
applicationName: configuration.applicationName,
|
|
|
|
applicationUrl: configuration.applicationUrl,
|
|
|
|
endpoint: configuration.changeManagement.endpoint,
|
|
|
|
uploadCredentialsId: configuration.changeManagement.credentialsId,
|
|
|
|
instance: configuration.changeManagement.rfc.developmentInstance,
|
|
|
|
client: configuration.changeManagement.rfc.developmentClient,
|
|
|
|
applicationDescription: configuration.applicationDescription,
|
|
|
|
abapPackage: configuration.abapPackage,
|
|
|
|
codePage: configuration.codePage,
|
|
|
|
acceptUnixStyleLineEndings: configuration.acceptUnixStyleLineEndings,
|
|
|
|
failUploadOnWarning: configuration.failOnWarning,
|
|
|
|
verbose: configuration.verbose
|
2021-09-21 10:29:43 +02:00
|
|
|
]
|
|
|
|
|
|
|
|
paramsUpload = addDockerParams(script, paramsUpload, configuration.changeManagement.rfc?.docker)
|
|
|
|
|
|
|
|
transportRequestUploadRFC(paramsUpload)
|
2019-02-15 12:05:21 +02:00
|
|
|
|
2020-10-21 12:04:11 +02:00
|
|
|
echo "[INFO] File 'configuration.applicationUrl' has been successfully uploaded to transport request '${configuration.transportRequestId}'."
|
2018-12-14 16:24:03 +02:00
|
|
|
|
2020-10-21 12:04:11 +02:00
|
|
|
break
|
2018-12-14 16:24:03 +02:00
|
|
|
}
|
2018-07-16 15:41:46 +02:00
|
|
|
|
2018-06-20 11:46:28 +02:00
|
|
|
} catch(ChangeManagementException ex) {
|
|
|
|
throw new AbortException(ex.getMessage())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-09-21 10:29:43 +02:00
|
|
|
|
|
|
|
Map addDockerParams(Script script, Map parameters, Map docker) {
|
|
|
|
|
|
|
|
if(docker) {
|
|
|
|
if(docker.image) {
|
|
|
|
parameters.dockerImage = docker.image
|
|
|
|
}
|
|
|
|
if(docker.options) {
|
|
|
|
parameters.dockerOptions = docker.options
|
|
|
|
}
|
|
|
|
if(docker.envVars) {
|
|
|
|
parameters.dockerEnvVars = docker.envVars
|
|
|
|
}
|
|
|
|
if(docker.pullImage != null) {
|
|
|
|
parameters.put('dockerPullImage', docker.pullImage)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return parameters
|
|
|
|
}
|