Skip to content

twas-base CICD

twas-base CICD #71

# Copyright (c) IBM Corporation.
# Copyright (c) Microsoft Corporation.
name: twas-base CICD
# Controls when the action will run.
on:
# Allows you to run this workflow manually from the Actions tab
workflow_dispatch:
inputs:
imageVersionNumber:
description: 'Provide image version number'
required: false
updateOfferArtifact:
description: 'Update offer artifact'
required: true
type: boolean
default: true
skipIntegrationTests:
description: 'If true, do not call integration tests in other repositories.'
required: true
type: boolean
default: false
retainTestingVMs:
description: 'If true, retain testing VMs that are created with the image.'
required: true
type: boolean
default: false
# Allows you to run this workflow using GitHub APIs
# PERSONAL_ACCESS_TOKEN=<GITHUB_PERSONAL_ACCESS_TOKEN>
# REPO_NAME=WASdev/azure.websphere-traditional.image
# curl --verbose -X POST -u "WASdev:${PERSONAL_ACCESS_TOKEN}" -H "Accept: application/vnd.github.everest-preview+json" -H "Content-Type: application/json" https://api.github.com/repos/${REPO_NAME}/actions/workflows/twas-baseBuild.yml/dispatches --data '{"ref": "main", "inputs": {"imageVersionNumber": "9.0.20230428"}}'
repository_dispatch:
types: [integration-test-twasbase, integration-test-all]
# sample request
# PERSONAL_ACCESS_TOKEN=<GITHUB_PERSONAL_ACCESS_TOKEN>
# REPO_NAME=WASdev/azure.websphere-traditional.image
# curl --verbose -X POST https://api.github.com/repos/${REPO_NAME}/dispatches -H "Accept: application/vnd.github.everest-preview+json" -H "Authorization: token ${PERSONAL_ACCESS_TOKEN}" --data '{"event_type": "integration-test-twasbase", "client_payload": {"imageVersionNumber": "9.0.20230428"}}'
# The integration-test-all event is used to run all integration tests (twas-base, twas-nd & ihs) in the repository with the same image version number.
# curl --verbose -X POST https://api.github.com/repos/${REPO_NAME}/dispatches -H "Accept: application/vnd.github.everest-preview+json" -H "Authorization: token ${PERSONAL_ACCESS_TOKEN}" --data '{"event_type": "integration-test-all", "client_payload": {"imageVersionNumber": "9.0.20230428"}}'
env:
repoName: azure.websphere-traditional.image
userName: ${{ secrets.USER_NAME }}
azureCredentials: ${{ secrets.AZURE_CREDENTIALS }}
entitledIbmUserId: ${{ secrets.ENTITLED_IBM_USER_ID }}
entitledIbmPassword: ${{ secrets.ENTITLED_IBM_USER_PWD }}
unEntitledIbmUserId: ${{ secrets.UNENTITLED_IBM_USER_ID }}
unEntitledIbmPassword: ${{ secrets.UNENTITLED_IBM_USER_PWD }}
vmAdminId: ${{ secrets.VM_ADMIN_ID }}
vmAdminPassword: ${{ secrets.VM_ADMIN_PASSWORD }}
testResourceGroup: imageTest-${{ github.repository_owner }}-${{ github.run_id }}-${{ github.run_number }}-twasBase
vmName: vm-${{ github.run_id }}-${{ github.run_number }}
vhdStorageAccountName: storage${{ github.run_id }}${{ github.run_number }}
location: eastus
scriptLocation: https://raw.githubusercontent.com/${{ secrets.USER_NAME }}/azure.websphere-traditional.image/$GITHUB_REF_NAME/twas-base/test/
utilitiesLocation: https://raw.githubusercontent.com/${{ secrets.USER_NAME }}/azure.websphere-traditional.image/$GITHUB_REF_NAME/utilities/
accessToken: ${{ secrets.PERSONAL_ACCESS_TOKEN }}
twasSingleRepo: ${{ secrets.USER_NAME }}/azure.websphere-traditional.singleserver
offerId: "2023-03-27-twas-single-server-base-image"
planId: "2023-03-27-twas-single-server-base-image"
offerType: "vm_image_offer"
clientId: ${{ secrets.CLIENT_ID }}
secretValue: ${{ secrets.SECRET_VALUE }}
tenantId: ${{ secrets.TENANT_ID }}
imageType: "x64Gen1"
operatingSystemFamily: "linux"
operatingSystemType: "redHat"
# A workflow run is made up of one or more jobs that can run sequentially or in parallel
jobs:
build:
# The type of runner that the job will run on
runs-on: ubuntu-latest
outputs:
imageVersionNumber: ${{ steps.setup-env-variables-based-on-dispatch-event.outputs.imageVersionNumber }}
# Steps represent a sequence of tasks that will be executed as part of the job
steps:
- name: Get versions of external dependencies
run: |
curl -Lo external-deps-versions.properties https://raw.githubusercontent.com/Azure/azure-javaee-iaas/main/external-deps-versions.properties
source external-deps-versions.properties
echo "azCliVersion=${AZ_CLI_VERSION}" >> $GITHUB_ENV
- name: Setup environment variables
id: setup-env-variables-based-on-dispatch-event
run: |
if [ ${{ github.event_name }} == 'workflow_dispatch' ]; then
imageVersionNumber=${{ github.event.inputs.imageVersionNumber }}
else
imageVersionNumber=${{ github.event.client_payload.imageVersionNumber }}
fi
echo "imageVersionNumber=${imageVersionNumber}" >> $GITHUB_OUTPUT
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
distribution: 'microsoft'
java-version: '11'
server-id: github # Value of the distributionManagement/repository/id field of the pom.xml
server-username: MAVEN_USERNAME # env variable for username
server-password: MAVEN_TOKEN # env variable for token
- name: Set Maven env
env:
MAVEN_USERNAME: github
MAVEN_TOKEN: ${{ secrets.GITHUB_TOKEN }}
shell: bash
run: |
echo "MAVEN_USERNAME=${MAVEN_USERNAME}" >> "$GITHUB_ENV"
echo "MAVEN_TOKEN=${MAVEN_TOKEN}" >> "$GITHUB_ENV"
- name: Checkout arm-ttk
uses: actions/checkout@v3
with:
repository: Azure/arm-ttk
path: arm-ttk
- name: Checkout ${{ env.repoName }}
uses: actions/checkout@v3
with:
path: ${{ env.repoName }}
ref: ${{ github.event.inputs.ref }}
- name: Azure login
uses: azure/login@v1
with:
creds: ${{ env.azureCredentials }}
- name: Build ${{ env.repoName }}
run: |
echo "Branch name is $GITHUB_REF_NAME"
mvn -Dgit.repo=${{ env.userName }} -Dgit.tag=$GITHUB_REF_NAME \
-DibmUserId=${{ env.entitledIbmUserId }} -DibmUserPwd=${{ env.entitledIbmPassword }} \
-DstorageAccount=${{ env.vhdStorageAccountName }} -DvmName=${{ env.vmName }} \
-DvmAdminId=${{ env.vmAdminId }} -DvmAdminPwd=${{ env.vmAdminPassword }} \
-Dtest.args="-Test All" -Ptemplate-validation-tests -Dtemplate.validation.tests.directory=../../arm-ttk/arm-ttk \
clean install --file ${{ env.repoName }}/twas-base/pom.xml
- name: Deploy a RHEL 9.x VM and install twas Base server
run: |
cd ${{ env.repoName }}/twas-base/target/cli
chmod a+x deploy.azcli
./deploy.azcli -n testDeployment -g ${{ env.testResourceGroup }} -l ${{ env.location }}
- name: Query public IP of VM
uses: azure/CLI@v1
with:
azcliversion: ${{ env.azCliVersion }}
inlineScript: |
echo "query public ip"
publicIP=$(az vm show \
--resource-group ${{ env.testResourceGroup }} \
--name ${{ env.vmName }} -d \
--query publicIps -o tsv)
echo "publicIP=${publicIP}" >> $GITHUB_ENV
- name: Output installation log
run: |
sudo apt-get install -y sshpass
timeout 1m sh -c 'until nc -zv $0 $1; do echo "nc rc: $?"; sleep 5; done' ${publicIP} 22
echo "Output stdout:"
result=$(sshpass -p ${{ env.vmAdminPassword }} -v ssh -p 22 -o StrictHostKeyChecking=no -o TCPKeepAlive=yes -o ServerAliveCountMax=20 -o ServerAliveInterval=15 -o ConnectTimeout=100 -v -tt ${{ env.vmAdminId }}@${publicIP} 'echo "${{ env.vmAdminPassword }}" | sudo -S cat /var/lib/waagent/custom-script/download/0/stdout')
echo "$result"
- name: Update the packages
run: |
sudo apt-get install -y sshpass
timeout 1m sh -c 'until nc -zv $0 $1; do echo "nc rc: $?"; sleep 5; done' ${publicIP} 22
sshpass -p ${{ env.vmAdminPassword }} -v ssh -p 22 -o StrictHostKeyChecking=no -o TCPKeepAlive=yes -o ServerAliveCountMax=20 -o ServerAliveInterval=15 -o ConnectTimeout=100 -v -tt ${{ env.vmAdminId }}@${publicIP} 'echo "${{ env.vmAdminPassword }}" | sudo -S yum update -y'
- name: Harden the VM using OpenSCAP tool
run: |
az vm extension set --name CustomScript \
--extension-instance-name apply-oscap-rules \
--resource-group ${{ env.testResourceGroup }} --vm-name ${{ env.vmName }} \
--publisher Microsoft.Azure.Extensions --version 2.0 \
--settings "{\"fileUris\": [\"${{ env.utilitiesLocation }}oscap.sh\"]}" \
--protected-settings "{\"commandToExecute\":\"bash oscap.sh ${{ env.vmAdminId }}\"}"
- name: Copy openscap scanning reports
run: |
sudo apt-get install -y sshpass
timeout 1m sh -c 'until nc -zv $0 $1; do echo "nc rc: $?"; sleep 5; done' ${publicIP} 22
sshpass -p ${vmAdminPassword} scp ${vmAdminId}@${publicIP}:/home/${vmAdminId}/scan_report_* .
- name: Upload openscap scanning report before remidation
uses: actions/upload-artifact@v4
with:
name: scan-report-before
path: scan_report_before.html
- name: Upload openscap scanning report after remidation
uses: actions/upload-artifact@v4
with:
name: scan-report-after
path: scan_report_after.html
- name: Deprovision the VM
run: |
sudo apt-get install -y sshpass
timeout 1m sh -c 'until nc -zv $0 $1; do echo "nc rc: $?"; sleep 5; done' ${publicIP} 22
sshpass -p ${{ env.vmAdminPassword }} -v ssh -p 22 -o StrictHostKeyChecking=no -o TCPKeepAlive=yes -o ServerAliveCountMax=20 -o ServerAliveInterval=15 -o ConnectTimeout=100 -v -tt ${{ env.vmAdminId }}@${publicIP} 'echo "${{ env.vmAdminPassword }}" | sudo -S waagent -deprovision+user -force'
- name: Generate the image
run: |
az vm deallocate --resource-group ${{ env.testResourceGroup }} --name ${{ env.vmName }}
az vm generalize --resource-group ${{ env.testResourceGroup }} --name ${{ env.vmName }}
az image create --resource-group ${{ env.testResourceGroup }} --name ${{ env.vmName }} --source ${{ env.vmName }}
verify_the_image:
needs: build
runs-on: ubuntu-latest
steps:
- name: Checkout ${{ env.repoName }}
uses: actions/checkout@v3
with:
path: ${{ env.repoName }}
ref: ${{ github.event.inputs.ref }}
- name: Azure login
uses: azure/login@v1
with:
creds: ${{ env.azureCredentials }}
- name: Verify the image
run: |
imageResourceId=$(az image show --name ${{ env.vmName }} --resource-group ${{ env.testResourceGroup }} --query id -o tsv)
echo "imageResourceId=${imageResourceId}" >> $GITHUB_ENV
# Deploy VMs using different IBMids and verify installation
vmGroups=( ${{ env.testResourceGroup }}-entitled ${{ env.testResourceGroup }}-unentitled ${{ env.testResourceGroup }}-evaluation )
deploymentModes=( Entitled Unentitled Evaluation )
ibmUserIds=( ${{ env.entitledIbmUserId }} ${{ env.unEntitledIbmUserId }} "" )
ibmUserPwds=( ${{ env.entitledIbmPassword }} ${{ env.unEntitledIbmPassword }} "" )
for (( i=0; i<${#vmGroups[@]}; i++ )); do
rgName=${vmGroups[$i]}
az group create -n $rgName -l ${{ env.location }}
az deployment group create --resource-group $rgName --name testDeployment \
--template-file ${{ env.repoName }}/utilities/verifyTemplate.json \
--parameters deploymentMode=${deploymentModes[$i]} ibmUserId=${ibmUserIds[$i]} ibmUserPwd=${ibmUserPwds[$i]} \
imageResourceId=$imageResourceId vmAdminId=${{ env.vmAdminId }} vmAdminPwd=${{ env.vmAdminPassword }} \
scriptLocation=${{ env.scriptLocation }}
# Remove VM if inputs.retainTestingVMs is false
if [ ${{ inputs.retainTestingVMs }} == 'false' ]; then
az group delete -n $rgName --yes --no-wait
fi
done
- name: Verify the image with twas-single integration-test pipeline
if: ${{ !inputs.skipIntegrationTests }}
run: |
# Trigger the workflow run
requestData={\"ref\":\"${GITHUB_REF_NAME}\",\"inputs\":{\"databaseType\":\"none\",\"imageResourceId\":\"${imageResourceId}\",\"location\":\"${{ env.location }}\"}}
curl --fail --verbose -L -X POST \
-H "Accept: application/vnd.github+json" \
-H "Authorization: Bearer ${accessToken}"\
-H "X-GitHub-Api-Version: 2022-11-28" \
https://api.github.com/repos/${{ env.twasSingleRepo }}/actions/workflows/integration-test.yaml/dispatches \
-d $(echo $requestData | jq -c)
# Wait until the workflow run starts
idAndStatus=$(curl -L \
-H "Accept: application/vnd.github+json" \
-H "Authorization: Bearer ${accessToken}"\
-H "X-GitHub-Api-Version: 2022-11-28" \
https://api.github.com/repos/${{ env.twasSingleRepo }}/actions/workflows/integration-test.yaml/runs \
| jq -r '.workflow_runs[0] | {id, status}')
status=$(echo $idAndStatus | jq -r '.status')
while [[ $status != queued ]]
do
echo "Wait until the workflow run starts..."
idAndStatus=$(curl -L \
-H "Accept: application/vnd.github+json" \
-H "Authorization: Bearer ${accessToken}"\
-H "X-GitHub-Api-Version: 2022-11-28" \
https://api.github.com/repos/${{ env.twasSingleRepo }}/actions/workflows/integration-test.yaml/runs \
| jq -r '.workflow_runs[0] | {id, status}')
status=$(echo $idAndStatus | jq -r '.status')
echo "The latest workflow run is ${idAndStatus}"
done
workflowRunId=$(echo $idAndStatus | jq '.id')
# Wait until the workflow run completes
status=$(curl -L \
-H "Accept: application/vnd.github+json" \
-H "Authorization: Bearer ${accessToken}"\
-H "X-GitHub-Api-Version: 2022-11-28" \
https://api.github.com/repos/${{ env.twasSingleRepo }}/actions/runs/$workflowRunId \
| jq -r '.status')
while [[ $status != completed ]]
do
sleep 60
echo "Wait until the workflow run ${workflowRunId} completes..."
status=$(curl -L \
-H "Accept: application/vnd.github+json" \
-H "Authorization: Bearer ${accessToken}"\
-H "X-GitHub-Api-Version: 2022-11-28" \
https://api.github.com/repos/${{ env.twasSingleRepo }}/actions/runs/$workflowRunId \
| jq -r '.status')
echo "Workflow run ${workflowRunId} is ${status}"
done
# Verify the workflow run result
jobs=$(curl -L \
-H "Accept: application/vnd.github+json" \
-H "Authorization: Bearer ${accessToken}"\
-H "X-GitHub-Api-Version: 2022-11-28" \
https://api.github.com/repos/${{ env.twasSingleRepo }}/actions/runs/$workflowRunId/jobs)
successIntegTestJobCnt=$(echo $jobs | jq 'select(.jobs != null) | .jobs | map(select(.name=="integration-test" and .conclusion=="success")) | length')
if [[ $successIntegTestJobCnt != 1 ]]; then
echo "twas-single integration test workflow run ${workflowRunId} failed."
exit 1
else
echo "twas-single integration test workflow run ${workflowRunId} succeeded."
fi
- name: Clean up all resources except for vhd storage account
run: |
az image delete --ids \
$(az image show -g ${{ env.testResourceGroup }} -n ${{ env.vmName }} --query id -o tsv)
az vm delete --yes --ids \
$(az vm show -g ${{ env.testResourceGroup }} -n ${{ env.vmName }} --query id -o tsv)
az network nic delete --ids \
$(az network nic list -g ${{ env.testResourceGroup }} --query [0].id -o tsv)
az network vnet delete --ids \
$(az network vnet list -g ${{ env.testResourceGroup }} --query [0].id -o tsv)
az network public-ip delete --ids \
$(az network public-ip list -g ${{ env.testResourceGroup }} --query [0].id -o tsv)
az network nsg delete --ids \
$(az network nsg list -g ${{ env.testResourceGroup }} --query [0].id -o tsv)
update_sas_url:
needs: [ build, verify_the_image ]
runs-on: ubuntu-latest
steps:
- name: Azure login
uses: azure/login@v1
with:
creds: ${{ env.azureCredentials }}
- name: Generate SAS url
id: sas_url
run: |
# Get a minus-24-hour date and a plus-30-day date for the SAS token
minus24HoursUtc=$(date -u --date "$dte -24 hour" +%Y-%m-%dT%H:%MZ)
plus30DaysUtc=$(date -u --date "$dte 30 day" +%Y-%m-%dT%H:%MZ)
vhdStorageAccountAccessKey=$(az storage account keys list --account-name ${{ env.vhdStorageAccountName }} --query "[?keyName=='key1'].value" -o tsv)
sasToken=$(az storage container generate-sas \
--connection-string "DefaultEndpointsProtocol=https;AccountName=${{ env.vhdStorageAccountName }};AccountKey=${vhdStorageAccountAccessKey};EndpointSuffix=core.windows.net" \
--name vhds --permissions rl --start "${minus24HoursUtc}" --expiry "${plus30DaysUtc}" -o tsv)
blobStorageEndpoint=$( az storage account show -n ${{ env.vhdStorageAccountName }} -g ${{ env.testResourceGroup }} -o json | jq -r '.primaryEndpoints.blob' )
osDiskSasUrl=${blobStorageEndpoint}vhds/${{ env.vmName }}.vhd?$sasToken
dataDiskSasUrl=${blobStorageEndpoint}vhds/${{ env.vmName }}datadisk1.vhd?$sasToken
echo "osDiskSasUrl: ${osDiskSasUrl}, dataDiskSasUrl: ${dataDiskSasUrl}" > sas-url-twasbase.txt
echo "##[set-output name=osDiskSasUrl;]${osDiskSasUrl}"
echo "##[set-output name=dataDiskSasUrl;]${dataDiskSasUrl}"
- name: Upload SAS url
uses: actions/upload-artifact@v4
with:
name: sasurl-twasbase
path: sas-url-twasbase.txt
- name: Update offer sas url and version
uses: microsoft/[email protected]
if: ${{ inputs.updateOfferArtifact == true && needs.build.outputs.imageVersionNumber != '' }}
with:
offerId: ${{ env.offerId }}
planId: ${{ env.planId }}
clientId: ${{ env.clientId }}
tenantId: ${{ env.tenantId }}
secretValue: ${{ env.secretValue }}
imageType: ${{ env.imageType }}
imageVersionNumber: ${{ needs.build.outputs.imageVersionNumber }}
offerType: ${{ env.offerType }}
operatingSystemFamily: ${{ env.operatingSystemFamily }}
operatingSystemType: ${{ env.operatingSystemType }}
osDiskSasUrl: ${{steps.sas_url.outputs.osDiskSasUrl}}
dataDiskSasUrl: ${{steps.sas_url.outputs.dataDiskSasUrl}}
verbose: "false"
summary:
needs: [build, verify_the_image, update_sas_url]
runs-on: ubuntu-latest
steps:
- name: Output inputs from workflow_dispatch
run: echo "${{ toJSON(github.event.inputs) }}"
- name: Output client_payload from repository_dispatch
run: echo "${{ toJSON(github.event.client_payload) }}"
delete_resource:
needs: [ build, verify_the_image, update_sas_url,summary ]
runs-on: ubuntu-latest
if: always()
steps:
- name: Azure login
uses: azure/login@v1
with:
creds: ${{ env.azureCredentials }}
- name: Delete resource groups during verification
continue-on-error: true
if: ${{ !inputs.retainTestingVMs }}
run: |
vmGroups=( ${{ env.testResourceGroup }}-entitled ${{ env.testResourceGroup }}-unentitled ${{ env.testResourceGroup }}-evaluation )
for (( i=0; i<${#vmGroups[@]}; i++ )); do
rgName=${vmGroups[$i]}
if $(az group exists --name $rgName); then
echo "Resource group $rgName exists. Deleting..."
az group delete --name $rgName --yes --no-wait
echo "Resource group $rgName has been scheduled for deletion."
else
echo "Resource group $rgName does not exist."
fi
done
- name: Delete testResourceGroup
if: ${{ needs.verify_the_image.result != 'success' || needs.build.outputs.imageVersionNumber == '' }}
run: |
echo "The verify_the_image result is ${{ needs.verify_the_image.result }}, delete the testResourceGroup."
az group delete -n ${{ env.testResourceGroup }} --yes --no-wait