diff --git a/.pax/pre-packaging.sh b/.pax/pre-packaging.sh index 819d4818ea..85f92603a7 100755 --- a/.pax/pre-packaging.sh +++ b/.pax/pre-packaging.sh @@ -26,6 +26,51 @@ # ./content/templates/ # ./content/ +# --------------------------------------------------------------------- +# --- convert files to ascii +# $1: (input) pattern to convert. +# Files will be determined by 'find -type f' +# $2: (input) optional output directory. +# If unset, conversion happens in-place +# If set, conversion will mirror directory structure in output +# (output) converted files or directory following $2 +# TODO: is this replacable with autoconv? +# --------------------------------------------------------------------- +function _convertEbcdicToAscii { + input=$1 + output_dir=$2 + using_output_dir="no" + + if [ -z "$output_dir" ]; then + echo "[$SCRIPT_NAME] converting $input to ascii in-place" + else + if [ -f "$output_dir" ]; then + echo "[$SCRIPT_NAME] $output_dir already exists and is a file, aborting _convertEbcdicToAscii" + return 1 + elif [ ! -d "$output_dir" ]; then + mkdir -p "$output_dir" + fi + using_output_dir="yes" + echo "[$SCRIPT_NAME] will convert $input to ascii, results in $output_dir" + fi + + files_to_convert=$(find $input -type f) # processes all files + for ebcdic_file in $files_to_convert; do + echo "[$SCRIPT_NAME] converting $ebcdic_file to ascii..." + tmpfile="$(basename $ebcdic_file).tmp" + iconv -f IBM-1047 -t ISO8859-1 "${ebcdic_file}" >${tmpfile} + if [[ "$using_output_dir" == "yes" ]]; then + dir_path=$(dirname $ebcdic_file) + mkdir -p ${output_dir}/${dir_path} + mv "${tmpfile}" "${output_dir}/${ebcdic_file}" + else + mv "${tmpfile}" "${ebcdic_file}" + fi + done + + return 0 +} # _convertEbcdicToAscii + # --------------------------------------------------------------------- # --- create JCL files # $1: (input) location of .vtl & .properties files @@ -35,33 +80,31 @@ # --------------------------------------------------------------------- function _createJCL { - VTLCLI_PATH="/ZOWE/vtl-cli" # tools, path must be absolute + VTLCLI_PATH="/ZOWE/vtl-cli" # tools, path must be absolute # vtl-cli source: https://github.com/plavjanik/vtl-cli if [ -f "$1/$2.vtl" ]; then - vtlList="$2.vtl" # process just this file + vtlList="$2.vtl" # process just this file vtlPath="$1" elif [ -d "$1/$2" ]; then - vtlList="$(ls $1/$2/)" # process all if directory passed in + vtlList="$(ls $1/$2/)" # process all if directory passed in vtlPath="$1/${2:-.}" else echo "[$SCRIPT_NAME] $1/$2.vtl not found" exit 1 fi - for vtlEntry in $vtlList - do - if [ "${vtlEntry##*.}" = "vtl" ] # keep from last . (exclusive) - then - vtlBase="${vtlEntry%.*}" # keep up to last . (exclusive) + for vtlEntry in $vtlList; do + if [ "${vtlEntry##*.}" = "vtl" ]; then # keep from last . (exclusive) + vtlBase="${vtlEntry%.*}" # keep up to last . (exclusive) JCL="${JCL_PATH}/${vtlBase}.jcl" VTL="${vtlPath}/${vtlEntry}" if [ -f ${vtlPath}/${vtlBase}.properties ]; then YAML="${vtlPath}/${vtlBase}.properties" - # elif [ -f ${vtlPath}/${vtlBase}.yaml ]; then - # YAML="${vtlPath}/${vtlBase}.yaml" - # elif [ -f ${vtlPath}/${vtlBase}.yml ]; then - # YAML="${vtlPath}/${vtlBase}.yml" + # elif [ -f ${vtlPath}/${vtlBase}.yaml ]; then + # YAML="${vtlPath}/${vtlBase}.yaml" + # elif [ -f ${vtlPath}/${vtlBase}.yml ]; then + # YAML="${vtlPath}/${vtlBase}.yml" else echo "[$SCRIPT_NAME] ${vtlPath}/${vtlBase}.properties not found" exit 1 @@ -74,9 +117,9 @@ function _createJCL # assumes java is in $PATH java -jar ${VTLCLI_PATH}/vtl-cli.jar \ -ie Cp1140 --yaml-context ${YAML} ${VTL} -o ${JCL} -oe Cp1140 - fi # vtl found + fi # vtl found done -} # _createJCL +} # _createJCL # --------------------------------------------------------------------- # --- create workflow & JCL files @@ -89,62 +132,63 @@ function _createJCL function _createWorkflow { here=$(pwd) - CREAXML_PATH="${here}/templates" # tools, path must be absolute + CREAXML_PATH="${here}/templates" # tools, path must be absolute if [ -f "$1/$2.xml" ]; then - xmlList="$2.xml" # process just this file + xmlList="$2.xml" # process just this file xmlPath="$1" elif [ -d "$1/$2" ]; then - xmlList="$(ls $1/$2/)" # process all if directory passed in + xmlList="$(ls $1/$2/)" # process all if directory passed in xmlPath="$1/${2:-.}" else echo "[$SCRIPT_NAME] $1/$2.xml not found" exit 1 fi - for xmlEntry in $xmlList - do - if [ "${xmlEntry##*.}" = "xml" ] # keep from last . (exclusive) - then - xmlBase="${xmlEntry%.*}" # keep up to last . (exclusive) - XML="${here}/${WORKFLOW_PATH}/${xmlBase}.xml" # use absolute path + for xmlEntry in $xmlList; do + if [ "${xmlEntry##*.}" = "xml" ]; then # keep from last . (exclusive) + xmlBase="${xmlEntry%.*}" # keep up to last . (exclusive) + XML="${here}/${WORKFLOW_PATH}/${xmlBase}.xml" # use absolute path if [ -d ${xmlBase} ]; then # TODO ensure workflow yaml has all variables of JCL yamls fi - + # create JCL related to this workflow - _createJCL ${xmlPath} ${xmlBase} # ${xmlBase} can be a directory + _createJCL ${xmlPath} ${xmlBase} # ${xmlBase} can be a directory # create workflow echo "[$SCRIPT_NAME] creating $XML" # inlineTemplate definition in xml expects us to be in $xmlPath cd "${xmlPath}" ${CREAXML_PATH}/build-workflow.rex -d -i ${xmlEntry} -o ${XML} - rm -f ${xmlEntry} # remove to avoid processing twice - cd - # return to previous directory + rm -f ${xmlEntry} # remove to avoid processing twice + cd - # return to previous directory # copy default variable definitions to ${WORKFLOW_PATH} if [ -f ${xmlPath}/${xmlBase}.properties ]; then YAML="${xmlPath}/${xmlBase}.properties" - # elif [ -f ${xmlPath}/${xmlBase}.yaml ]; then - # YAML="${xmlPath}/${xmlBase}.yaml" - # elif [ -f ${xmlPath}/${xmlBase}.yml ]; then - # YAML="${xmlPath}/${xmlBase}.yml" + # elif [ -f ${xmlPath}/${xmlBase}.yaml ]; then + # YAML="${xmlPath}/${xmlBase}.yaml" + # elif [ -f ${xmlPath}/${xmlBase}.yml ]; then + # YAML="${xmlPath}/${xmlBase}.yml" else echo "[$SCRIPT_NAME] ${xmlPath}/${xmlBase}.properties not found" exit 1 fi cp "${YAML}" "${WORKFLOW_PATH}/${xmlBase}.properties" - fi # xml found + fi # xml found done -} # _createWorkflow +} # _createWorkflow # --------------------------------------------------------------------- # --- main --- main --- main --- main --- main --- main --- main --- # --------------------------------------------------------------------- -SCRIPT_NAME=$(basename "$0") # $0=./pre-packaging.sh -BASE_DIR=$(cd $(dirname "$0"); pwd) # /.pax +SCRIPT_NAME=$(basename "$0") # $0=./pre-packaging.sh +BASE_DIR=$( + cd $(dirname "$0") + pwd +) # /.pax # use node v14 to build export NODE_HOME=/ZOWE/node/node-v14.21.3.1-os390-s390x @@ -196,15 +240,14 @@ cd "${BASE_DIR}" # prepare for SMPE echo "[$SCRIPT_NAME] smpe is not part of zowe.pax, moving it out ..." -mv ./content/smpe . +mv ./content/smpe . # workflow customization # >>> echo "[$SCRIPT_NAME] templates is not part of zowe.pax, moving it out ..." -mv ./content/templates . +mv ./content/templates . chmod +x templates/*.rex - mkdir -p "${ZOWE_ROOT_DIR}/bin/utils" configmgr=$(find "${ZOWE_ROOT_DIR}/files" -type f \( -name "configmgr-2*.pax" \) | head -n 1) echo "[$SCRIPT_NAME] extract configmgr $configmgr" @@ -221,9 +264,8 @@ pax -ppx -rf "${configmgr_rexx}" rm "${configmgr_rexx}" cd "${BASE_DIR}" - echo "[$SCRIPT_NAME] create dummy zowe.yaml for install" -cat <> "${BASE_DIR}/zowe.yaml" +cat <>"${BASE_DIR}/zowe.yaml" zowe: extensionDirectory: "${ZOWE_ROOT_DIR}/components" useConfigmgr: false @@ -250,9 +292,9 @@ for component in app-server; do echo "[$SCRIPT_NAME] - ${component}" # FIXME: these environment variables are changed in v2 ZOWE_ROOT_DIR=${ZOWE_ROOT_DIR} \ - ZWED_INSTALL_DIR=${ZOWE_ROOT_DIR} \ - LOG_FILE="${BASE_DIR}/logs/zwe-components-install-process-hook.log" \ - "${ZOWE_ROOT_DIR}/bin/zwe" \ + ZWED_INSTALL_DIR=${ZOWE_ROOT_DIR} \ + LOG_FILE="${BASE_DIR}/logs/zwe-components-install-process-hook.log" \ + "${ZOWE_ROOT_DIR}/bin/zwe" \ components install process-hook \ --component-name "${component}" \ --config "${BASE_DIR}/zowe.yaml" \ @@ -318,7 +360,10 @@ else exit 1 fi -#3. clean up working files +#3. Convert z/OSMF workflows and templates to ASCII in-place +_convertEbcdicToAscii "${WORKFLOW_PATH}" + +#4. clean up working files echo "[$SCRIPT_NAME] clean up working files" rm -rf "./templates" @@ -332,11 +377,11 @@ echo "[$SCRIPT_NAME] generate fingerprints" mkdir -p "${BASE_DIR}/fingerprints" mkdir -p "${ZOWE_ROOT_DIR}/fingerprint" cd "${ZOWE_ROOT_DIR}" -find . -name ./SMPE -prune \ - -o -name "./ZWE*" -prune \ - -o -name ./fingerprint -prune \ - -o -type f -print > "${BASE_DIR}/fingerprints/files.in" -java -cp "${ZOWE_ROOT_DIR}/bin/utils" HashFiles "${BASE_DIR}/fingerprints/files.in" | sort > "${ZOWE_ROOT_DIR}/fingerprint/RefRuntimeHash-${ZOWE_VERSION}.txt" +find . -name ./SMPE -prune \ + -o -name "./ZWE*" -prune \ + -o -name ./fingerprint -prune \ + -o -type f -print >"${BASE_DIR}/fingerprints/files.in" +java -cp "${ZOWE_ROOT_DIR}/bin/utils" HashFiles "${BASE_DIR}/fingerprints/files.in" | sort >"${ZOWE_ROOT_DIR}/fingerprint/RefRuntimeHash-${ZOWE_VERSION}.txt" echo "[$SCRIPT_NAME] cleanup fingerprints code" rm -fr "${BASE_DIR}/fingerprints" diff --git a/CHANGELOG.md b/CHANGELOG.md index c76257168b..f0dfd72d56 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,18 @@ All notable changes to the Zowe Installer will be documented in this file. +## `2.13.0` + +### New features and enhancements + +#### Minor enhancements/defect fixes +- Bugfix: Workflow files in the Zowe PAX are now ASCII-encoded. Fixes [#3591](https://github.com/zowe/zowe-install-packaging/issues/3591). + +## `2.12.0` + +### New features and enhancements + +#### Minor enhancements/defect fixes ## `2.13.0` #### Minor enhancements/defect fixes diff --git a/bin/libs/json.ts b/bin/libs/json.ts index c4f5f77bb1..70243b8901 100644 --- a/bin/libs/json.ts +++ b/bin/libs/json.ts @@ -73,7 +73,7 @@ export function readYaml(file: string, key: string) { const jq=`${utils_dir}/njq/src/index.js`; const fconv=`${utils_dir}/fconv/src/index.js`; - common.printTrace(`- read_yaml load content from ${file}`); + common.printTrace(`- readYaml load content from ${file}`); if (std.getenv('ZWE_CLI_PARAMETER_CONFIG') == file) { return fakejq.jqget(ZOWE_CONFIG, key); } else { @@ -86,7 +86,7 @@ export function readYaml(file: string, key: string) { return; } - common.printTrace(`- read_yaml ${key} from yaml content`); + common.printTrace(`- readYaml ${key} from yaml content`); const result=shell.execOutSync('sh', '-c', `echo "${ZWE_PRIVATE_YAML_CACHE}" | node "${jq}" -r "${key}" 2>&1`); code=result.rc; common.printTrace(` * Exit code: ${code}`); @@ -104,7 +104,7 @@ export function readJson(file: string, key: string):any { const utils_dir=`${ZOWE_CONFIG.zowe.runtimeDirectory}/bin/utils`; const jq=`${utils_dir}/njq/src/index.js`; - common.printTrace(`- read_json ${key} from ${file}`); + common.printTrace(`- readJson ${key} from ${file}`); let result=shell.execOutSync('sh', '-c', `cat "${file}" | node "${jq}" -r "${key}" 2>&1`); const code = result.rc; common.printTrace(` * Exit code: ${code}`); diff --git a/playbooks/roles/configfmid/defaults/main.yml b/playbooks/roles/configfmid/defaults/main.yml index 65d34fcae1..322e90b06d 100644 --- a/playbooks/roles/configfmid/defaults/main.yml +++ b/playbooks/roles/configfmid/defaults/main.yml @@ -70,7 +70,7 @@ zowe_apiml_security_oidc_enabled: false zowe_apiml_security_oidc_client_id: zowe_apiml_security_oidc_client_secret: zowe_apiml_security_oidc_registry: -zowe_apiml_security_oidc_introspect_url: +zowe_apiml_security_oidc_jwks_uri: # explorer APIs/plugins ports zowe_jobs_api_port: 7558 zowe_mvs_api_port: 7559 diff --git a/playbooks/roles/configfmid/tasks/main.yml b/playbooks/roles/configfmid/tasks/main.yml index bd049e7cf9..7b2ad2c6c5 100644 --- a/playbooks/roles/configfmid/tasks/main.yml +++ b/playbooks/roles/configfmid/tasks/main.yml @@ -39,7 +39,7 @@ - zowe_apiml_security_oidc_client_id - zowe_apiml_security_oidc_client_secret - zowe_apiml_security_oidc_registry - - zowe_apiml_security_oidc_introspect_url + - zowe_apiml_security_oidc_jwks_uri - zowe_jobs_api_port - zowe_mvs_api_port - zowe_jobs_explorer_port @@ -299,7 +299,7 @@ "components.gateway.apiml.security.oidc.clientId": "{{ zowe_apiml_security_oidc_client_id|string }}" "components.gateway.apiml.security.oidc.clientSecret": "{{ zowe_apiml_security_oidc_client_secret|string }}" "components.gateway.apiml.security.oidc.registry": "{{ zowe_apiml_security_oidc_registry|string }}" - "components.gateway.apiml.security.oidc.introspectUrl": "{{ zowe_apiml_security_oidc_introspect_url|string }}" + "components.gateway.apiml.security.oidc.jwks.uri": "{{ zowe_apiml_security_oidc_jwks_uri|string }}" # desktop customizations "zowe.environments.ZWED_SSH_PORT": "{{ zowe_zlux_terminal_ssh_port }}" "zowe.environments.ZWED_TN3270_PORT": "{{ zowe_zlux_terminal_telnet_port }}" diff --git a/playbooks/roles/configure/defaults/main.yml b/playbooks/roles/configure/defaults/main.yml index 5e0cd717d4..24ff9b6446 100644 --- a/playbooks/roles/configure/defaults/main.yml +++ b/playbooks/roles/configure/defaults/main.yml @@ -71,7 +71,7 @@ zowe_apiml_security_oidc_enabled: false zowe_apiml_security_oidc_client_id: zowe_apiml_security_oidc_client_secret: zowe_apiml_security_oidc_registry: -zowe_apiml_security_oidc_introspect_url: +zowe_apiml_security_oidc_jwks_uri: # explorer APIs/plugins ports zowe_jobs_api_port: 7558 zowe_mvs_api_port: 7559 diff --git a/playbooks/roles/configure/tasks/main.yml b/playbooks/roles/configure/tasks/main.yml index e8fbfa267e..0cc6b1f017 100644 --- a/playbooks/roles/configure/tasks/main.yml +++ b/playbooks/roles/configure/tasks/main.yml @@ -39,7 +39,7 @@ - zowe_apiml_security_oidc_client_id - zowe_apiml_security_oidc_client_secret - zowe_apiml_security_oidc_registry - - zowe_apiml_security_oidc_introspect_url + - zowe_apiml_security_oidc_jwks_uri - zowe_jobs_api_port - zowe_mvs_api_port - zowe_jobs_explorer_port @@ -301,7 +301,7 @@ "components.gateway.apiml.security.oidc.clientId": "{{ zowe_apiml_security_oidc_client_id|string }}" "components.gateway.apiml.security.oidc.clientSecret": "{{ zowe_apiml_security_oidc_client_secret|string }}" "components.gateway.apiml.security.oidc.registry": "{{ zowe_apiml_security_oidc_registry|string }}" - "components.gateway.apiml.security.oidc.introspectUrl": "{{ zowe_apiml_security_oidc_introspect_url|string }}" + "components.gateway.apiml.security.oidc.jwks.uri": "{{ zowe_apiml_security_oidc_jwks_uri|string }}" # desktop customizations "zowe.environments.ZWED_SSH_PORT": "{{ zowe_zlux_terminal_ssh_port }}" "zowe.environments.ZWED_TN3270_PORT": "{{ zowe_zlux_terminal_telnet_port }}" diff --git a/playbooks/roles/custom_for_test/defaults/main.yml b/playbooks/roles/custom_for_test/defaults/main.yml index 9dbd3c90c8..0f223bc20e 100644 --- a/playbooks/roles/custom_for_test/defaults/main.yml +++ b/playbooks/roles/custom_for_test/defaults/main.yml @@ -68,7 +68,7 @@ zowe_apiml_security_oidc_enabled: false zowe_apiml_security_oidc_client_id: zowe_apiml_security_oidc_client_secret: zowe_apiml_security_oidc_registry: -zowe_apiml_security_oidc_introspect_url: +zowe_apiml_security_oidc_jwks_uri: zowe_apiml_security_zosmf_applid: IZUDFLT zowe_apiml_security_auth_provider: zosmf zowe_apiml_security_zosmf_jwt_autoconfiguration_mode: auto diff --git a/pswi/03_create.sh b/pswi/03_create.sh index a300646fab..99d4ab7687 100644 --- a/pswi/03_create.sh +++ b/pswi/03_create.sh @@ -131,28 +131,34 @@ echo "SH set -x;set -e;" >> JCL echo "cd ${WORK_MOUNT};" >> JCL echo "source=\"${ZOWE_MOUNT}files/workflows/ZWEWRF02.xml\";" >> JCL echo "target=\"//'${WORKFLOW_DSN}(ZWEWRF02)'\";" >> JCL -echo "sed 's|UTF-8|IBM-1047|g' \$source > _ZWEWRF02;" >> JCL -echo "cp -T _ZWEWRF02 \$target;" >> JCL +echo "iconv -f ISO8859-1 -t IBM-1047 \$source > _ZWEWRF02;" >> JCL +echo "sed 's|UTF-8|IBM-1047|g' _ZWEWRF02 > ZWEWRF02;" >> JCL +echo "cp -T ZWEWRF02 \$target;" >> JCL echo "source=\"${ZOWE_MOUNT}files/workflows/ZWECRECR.xml\";" >> JCL echo "target=\"//'${WORKFLOW_DSN}(ZWECRECR)'\";" >> JCL -echo "sed 's|UTF-8|IBM-1047|g' \$source > _ZWECRECR;" >> JCL -echo "cp -T _ZWECRECR \$target;" >> JCL +echo "iconv -f ISO8859-1 -t IBM-1047 \$source > _ZWECRECR;" >> JCL +echo "sed 's|UTF-8|IBM-1047|g' _ZWECRECR > ZWECRECR;" >> JCL +echo "cp -T ZWECRECR \$target;" >> JCL echo "source=\"${ZOWE_MOUNT}files/workflows/ZWEKRING.xml\";" >> JCL echo "target=\"//'${WORKFLOW_DSN}(ZWEKRING)'\";" >> JCL -echo "sed 's|UTF-8|IBM-1047|g' \$source > _ZWEKRING;" >> JCL -echo "cp -T _ZWEKRING \$target;" >> JCL +echo "iconv -f ISO8859-1 -t IBM-1047 \$source > _ZWEKRING;" >> JCL +echo "sed 's|UTF-8|IBM-1047|g' _ZWEKRING > ZWEKRING;" >> JCL +echo "cp -T ZWEKRING \$target;" >> JCL echo "source=\"${ZOWE_MOUNT}files/workflows/ZWELOADC.xml\";" >> JCL echo "target=\"//'${WORKFLOW_DSN}(ZWELOADC)'\";" >> JCL -echo "sed 's|UTF-8|IBM-1047|g' \$source > _ZWELOADC;" >> JCL -echo "cp -T _ZWELOADC \$target;" >> JCL +echo "iconv -f ISO8859-1 -t IBM-1047 \$source > _ZWELOADC;" >> JCL +echo "sed 's|UTF-8|IBM-1047|g' _ZWELOADC > ZWELOADC;" >> JCL +echo "cp -T ZWELOADC \$target;" >> JCL echo "source=\"${ZOWE_MOUNT}files/workflows/ZWESIGNC.xml\";" >> JCL echo "target=\"//'${WORKFLOW_DSN}(ZWESIGNC)'\";" >> JCL -echo "sed 's|UTF-8|IBM-1047|g' \$source > _ZWESIGNC;" >> JCL -echo "cp -T _ZWESIGNC \$target;" >> JCL +echo "iconv -f ISO8859-1 -t IBM-1047 \$source > _ZWESIGNC;" >> JCL +echo "sed 's|UTF-8|IBM-1047|g' _ZWESIGNC > ZWESIGNC;" >> JCL +echo "cp -T ZWESIGNC \$target;" >> JCL echo "source=\"${ZOWE_MOUNT}files/workflows/ZWECONF.xml\";" >> JCL echo "target=\"//'${WORKFLOW_DSN}(ZWECONF)'\";" >> JCL -echo "sed 's|UTF-8|IBM-1047|g' \$source > _ZWECONF;" >> JCL -echo "cp -T _ZWECONF \$target;" >> JCL +echo "iconv -f ISO8859-1 -t IBM-1047 \$source > _ZWECONF;" >> JCL +echo "sed 's|UTF-8|IBM-1047|g' _ZWECONF > ZWECONF;" >> JCL +echo "cp -T ZWECONF \$target;" >> JCL echo "/*" >> JCL sh scripts/submit_jcl.sh "`cat JCL`" diff --git a/tests/installation/src/constants.ts b/tests/installation/src/constants.ts index 74ecdddce8..c60aa86853 100644 --- a/tests/installation/src/constants.ts +++ b/tests/installation/src/constants.ts @@ -43,7 +43,7 @@ export const APIML_OIDC_VARS = { 'zowe_apiml_security_oidc_client_id': process.env['OKTA_CLIENT_ID'] || 'dummy_id_from_constants_ts', 'zowe_apiml_security_oidc_client_secret': process.env['OKTA_CLIENT_SECRET'] || 'dummy_secret_from_constants_ts', 'zowe_apiml_security_oidc_registry': process.env['OIDC_REGISTRY'] || 'dummy_registry_from_constants_ts', - 'zowe_apiml_security_oidc_introspect_url': `https://${process.env['OKTA_HOSTNAME']}/oauth2/default/v1/introspect`, + 'zowe_apiml_security_oidc_jwks_uri': `https://${process.env['OKTA_HOSTNAME']}/oauth2/default/v1/keys`, }; // debug(`process.env >>>>>>>>>>>>>>>>>>>>>>>>>>\n${JSON.stringify(process.env)}\n<<<<<<<<<<<<<<<<<<<<<<<`);