mirror of
				https://github.com/containers/podman.git
				synced 2025-10-25 18:25:59 +08:00 
			
		
		
		
	Github-Actions: Send e-mail on Cirrus cron failure
This repository has a number of automaticly triggered branch-level testing enabled. However, other than remembering to go look at a specific WebUI, there is no way for anybody to notice if/when these jobs fail. This commit introduces a github-action workflow which runs periodically, checking for failed cron-triggered Cirrus-CI jobs. When it finds any, it formats a simple report for e-mail delivery. The list of destination addresses is configurable at any time by merging changes to a simple CSV file. Signed-off-by: Chris Evich <cevich@redhat.com>
This commit is contained in:
		
							
								
								
									
										116
									
								
								.github/actions/check_cirrus_cron/cron_failures.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										116
									
								
								.github/actions/check_cirrus_cron/cron_failures.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @ -0,0 +1,116 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| set -eo pipefail | ||||
|  | ||||
| # Intended to be executed from a github action workflow step. | ||||
| # Outputs the Cirrus cron names and IDs of any failed builds | ||||
|  | ||||
| err() { | ||||
|     # Ref: https://docs.github.com/en/free-pro-team@latest/actions/reference/workflow-commands-for-github-actions | ||||
|     echo "::error file=${BASH_SOURCE[0]},line=${BASH_LINENO[0]}::${1:-No error message given}" | ||||
|     exit 1 | ||||
| } | ||||
|  | ||||
| _errfmt="Expecting %s value to not be empty" | ||||
| if [[ -z "$GITHUB_REPOSITORY" ]]; then | ||||
|     err $(printf "$_errfmt" "\$GITHUB_REPOSITORY") | ||||
| elif [[ -z "$NAME_ID_FILEPATH" ]]; then | ||||
|     err $(printf "$_errfmt" "\$NAME_ID_FILEPATH") | ||||
| fi | ||||
|  | ||||
| mkdir -p artifacts | ||||
| cat > ./artifacts/query_raw.json << "EOF" | ||||
| {"query":" | ||||
|   query CronNameStatus($owner: String!, $repo: String!) { | ||||
|     githubRepository(owner: $owner, name: $repo) { | ||||
|       cronSettings { | ||||
|         name | ||||
|         lastInvocationBuild { | ||||
|           id | ||||
|           status | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| ", | ||||
| "variables":"{ | ||||
|   \"owner\": \"@@OWNER@@\", | ||||
|   \"repo\": \"@@REPO@@\" | ||||
| }"} | ||||
| EOF | ||||
| # Makes for easier copy/pasting query to/from | ||||
| # https://cirrus-ci.com/explorer | ||||
| owner=$(cut -d '/' -f 1 <<<"$GITHUB_REPOSITORY") | ||||
| repo=$(cut -d '/' -f 2 <<<"$GITHUB_REPOSITORY") | ||||
| sed -i -r -e "s/@@OWNER@@/$owner/g" -e "s/@@REPO@@/$repo/g" ./artifacts/query_raw.json | ||||
|  | ||||
| echo "::group::Posting GraphQL Query" | ||||
| # Easier to debug in error-reply when query is compacted | ||||
| tr -d '\n' < ./artifacts/query_raw.json | tr -s ' ' | tee ./artifacts/query.json | \ | ||||
|     jq --indent 4 --color-output . | ||||
|  | ||||
| if grep -q '@@' ./artifacts/query.json; then | ||||
|     err "Found unreplaced substitution token in raw query JSON" | ||||
| fi | ||||
| curl \ | ||||
|   --request POST \ | ||||
|   --silent \ | ||||
|   --location \ | ||||
|   --header 'content-type: application/json' \ | ||||
|   --url 'https://api.cirrus-ci.com/graphql' \ | ||||
|   --data @./artifacts/query.json \ | ||||
|   --output ./artifacts/reply.json | ||||
| echo "::endgroup::" | ||||
|  | ||||
| echo "::group::Received GraphQL Reply" | ||||
| jq --indent 4 --color-output . <./artifacts/reply.json || \ | ||||
|     cat ./artifacts/reply.json | ||||
| echo "::endgroup::" | ||||
|  | ||||
| # Desireable to catch non-JSON encoded errors in reply. | ||||
| if grep -qi 'error' ./artifacts/reply.json; then | ||||
|     err "Found the word 'error' in reply" | ||||
| fi | ||||
|  | ||||
| # e.x. reply.json | ||||
| # { | ||||
| #   "data": { | ||||
| #     "githubRepository": { | ||||
| #       "cronSettings": [ | ||||
| #         { | ||||
| #           "name": "Keepalive_v2.0", | ||||
| #           "lastInvocationBuild": { | ||||
| #             "id": "5776050544181248", | ||||
| #             "status": "EXECUTING" | ||||
| #           } | ||||
| #         }, | ||||
| #         { | ||||
| #           "name": "Keepalive_v1.9", | ||||
| #           "lastInvocationBuild": { | ||||
| #             "id": "5962921081569280", | ||||
| #             "status": "COMPLETED" | ||||
| #           } | ||||
| #         }, | ||||
| #         { | ||||
| #           "name": "Keepalive_v2.0.5-rhel", | ||||
| #           "lastInvocationBuild": { | ||||
| #             "id": "5003065549914112", | ||||
| #             "status": "FAILED" | ||||
| #           } | ||||
| #         } | ||||
| #       ] | ||||
| #     } | ||||
| #   } | ||||
| # } | ||||
| _filt='.data.githubRepository.cronSettings | map(select(.lastInvocationBuild.status=="FAILED") | { name:.name, id:.lastInvocationBuild.id} | join(" ")) | join("\n")' | ||||
| jq --raw-output "$_filt" ./artifacts/reply.json > "$NAME_ID_FILEPATH" | ||||
|  | ||||
| echo "<Cron Name> <Failed Build ID>" | ||||
| cat "$NAME_ID_FILEPATH" | ||||
|  | ||||
| # Don't rely on a newline present for zero/one output line, always count words | ||||
| records=$(wc --words "$NAME_ID_FILEPATH" | cut -d ' ' -f 1) | ||||
| # Always two words per record | ||||
| failures=$((records/2)) | ||||
| echo "::set-output name=failures::$failures" | ||||
| echo "Total failed Cirrus-CI cron builds: $failures" | ||||
							
								
								
									
										81
									
								
								.github/workflows/check_cirrus_cron.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								.github/workflows/check_cirrus_cron.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,81 @@ | ||||
| --- | ||||
|  | ||||
| # Format Ref: https://docs.github.com/en/free-pro-team@latest/actions/reference/workflow-syntax-for-github-actions | ||||
|  | ||||
| # Required to un-FUBAR default ${{github.workflow}} value | ||||
| name: check_cirrus_cron | ||||
|  | ||||
| on: | ||||
|     schedule: | ||||
|         # Assume cirrus cron jobs runs at least once per day | ||||
|         - cron:  '59 23 * * *' | ||||
|     # Debug: Allow triggering job manually in github-actions WebUI | ||||
|     workflow_dispatch: {} | ||||
|  | ||||
| env: | ||||
|     # Debug-mode can reveal secrets, only enable by a secret value. | ||||
|     # Ref: https://help.github.com/en/actions/configuring-and-managing-workflows/managing-a-workflow-run#enabling-step-debug-logging | ||||
|     ACTIONS_STEP_DEBUG: '${{ secrets.ACTIONS_STEP_DEBUG }}' | ||||
|     # File with CSV listing of zero or more e-mail addresses for delivery | ||||
|     # of daily failure notice e-mails. | ||||
|     FAILMAILCSV: './contrib/cirrus/cron-fail_addrs.csv' | ||||
|     # Filename for table of cron-name to build-id data | ||||
|     # (must be in $GITHUB_WORKSPACE/artifacts/) | ||||
|     NAME_ID_FILEPATH: './artifacts/name_id.txt' | ||||
|  | ||||
| jobs: | ||||
|     cron_failures: | ||||
|         runs-on: ubuntu-latest | ||||
|         steps: | ||||
|             - uses: actions/checkout@v2 | ||||
|               with: | ||||
|                   ref: master | ||||
|                   persist-credentials: false | ||||
|  | ||||
|             - name: Get failed cron names and Build IDs | ||||
|               id: cron | ||||
|               run: './.github/actions/${{ github.workflow }}/${{ github.job }}.sh' | ||||
|  | ||||
|             - if: steps.cron.outputs.failures > 0 | ||||
|               shell: bash | ||||
|               # Must be inline, since context expressions are used. | ||||
|               # Ref: https://docs.github.com/en/free-pro-team@latest/actions/reference/context-and-expression-syntax-for-github-actions | ||||
|               run: | | ||||
|                 set -eo pipefail | ||||
|                 ( | ||||
|                 echo "Detected one or more Cirrus-CI cron-triggered jobs have failed recently:" | ||||
|                 echo "" | ||||
|  | ||||
|                 while read -r NAME BID; do | ||||
|                     echo "Cron build '$NAME' Failed: https://cirrus-ci.com/build/$BID" | ||||
|                 done < "$NAME_ID_FILEPATH" | ||||
|  | ||||
|                 echo "" | ||||
|                 echo "# Source: ${{ github.workflow }} workflow on ${{ github.repository }}." | ||||
|                 # Separate content from sendgrid.com automatic footer. | ||||
|                 echo "" | ||||
|                 ) > ./artifacts/email_body.txt | ||||
|  | ||||
|             - if: steps.cron.outputs.failures > 0 | ||||
|               id: mailto | ||||
|               run: printf "::set-output name=csv::%s\n" $(cat "$FAILMAILCSV") | ||||
|  | ||||
|             - if: steps.mailto.outputs.csv != '' | ||||
|               name: Send failure notification e-mail | ||||
|               # Ref: https://github.com/dawidd6/action-send-mail | ||||
|               uses: dawidd6/action-send-mail@v2.2.2 | ||||
|               with: | ||||
|                 server_address: ${{secrets.ACTION_MAIL_SERVER}} | ||||
|                 server_port: 465 | ||||
|                 username: ${{secrets.ACTION_MAIL_USERNAME}} | ||||
|                 password: ${{secrets.ACTION_MAIL_PASSWORD}} | ||||
|                 subject: Cirrus-CI cron build failures on ${{github.repository}} | ||||
|                 to: ${{steps.mailto.outputs.csv}} | ||||
|                 from: ${{secrets.ACTION_MAIL_SENDER}} | ||||
|                 body: file://./artifacts/email_body.txt | ||||
|  | ||||
|             - if: always() | ||||
|               uses: actions/upload-artifact@v2 | ||||
|               with: | ||||
|                   name: ${{ github.job }}_artifacts | ||||
|                   path: artifacts/* | ||||
							
								
								
									
										1
									
								
								contrib/cirrus/cron-fail_addrs.csv
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								contrib/cirrus/cron-fail_addrs.csv
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | ||||
| rh.container.bot@gmail.com | ||||
| 
 | 
		Reference in New Issue
	
	Block a user
	 Chris Evich
					Chris Evich