Skip to content

24.3 Update report action and smarter pr number fetching for grype and report #848

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 5 commits into from
Jun 13, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
41 changes: 41 additions & 0 deletions .github/actions/create_workflow_report/action.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
name: Create and Upload Combined Report
description: Create and upload a combined CI report
inputs:
final:
description: "Control whether the report is final or a preview"
required: false
default: "false"
runs:
using: "composite"
steps:
- name: Create and upload workflow report
env:
PR_NUMBER: ${{ github.event.pull_request.number || 0 }}
COMMIT_SHA: ${{ github.event_name == 'pull_request' && github.event.pull_request.head.sha || github.sha }}
ACTIONS_RUN_URL: ${{ github.event.repository.html_url }}/actions/runs/${{ github.run_id }}
FINAL: ${{ inputs.final }}
shell: bash
run: |
pip install clickhouse-driver==0.2.8 numpy==1.26.4 pandas==2.0.3 jinja2==3.1.5

CMD="python3 .github/actions/create_workflow_report/create_workflow_report.py"
ARGS="--pr-number $PR_NUMBER --commit-sha $COMMIT_SHA --actions-run-url $ACTIONS_RUN_URL --known-fails tests/broken_tests.json --cves"

set +e
if [[ "$FINAL" == "false" ]]; then
REPORT_LINK=$($CMD $ARGS --mark-preview)
else
REPORT_LINK=$($CMD $ARGS)
fi

echo $REPORT_LINK

if [[ "$FINAL" == "true" ]]; then
IS_VALID_URL=$(echo $REPORT_LINK | grep -E '^https?://')
if [[ -n $IS_VALID_URL ]]; then
echo "Workflow Run Report: [View Report]($REPORT_LINK)" >> $GITHUB_STEP_SUMMARY
else
echo "Error: $REPORT_LINK" >> $GITHUB_STEP_SUMMARY
exit 1
fi
fi
269 changes: 269 additions & 0 deletions .github/actions/create_workflow_report/ci_run_report.html.jinja
Original file line number Diff line number Diff line change
@@ -0,0 +1,269 @@
<!DOCTYPE html>
<html lang="en" data-lt-installed="true">

<head>
<meta http-equiv="content-type" content="text/html; charset=UTF-8">
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<style>
/* Base colors for Altinity */
:root {
--altinity-background: #000D45;
--altinity-accent: #189DCF;
--altinity-highlight: #FFC600;
--altinity-gray: #6c757d;
--altinity-light-gray: #f8f9fa;
--altinity-white: #ffffff;
}

/* Body and heading fonts */
body {
font-family: Arimo, "Proxima Nova", "Helvetica Neue", Helvetica, Arial, sans-serif;
font-size: 1rem;
background-color: var(--altinity-background);
color: var(--altinity-light-gray);
padding: 2rem;
}

h1,
h2,
h3,
h4,
h5,
h6 {
font-family: Figtree, "Proxima Nova", "Helvetica Neue", Helvetica, Arial, sans-serif;
color: var(--altinity-white);
}

.logo {
width: auto;
height: 5em;
}

/* General table styling */
table {
min-width: min(900px, 98vw);
margin: 1rem 0;
border-collapse: collapse;
background-color: var(--altinity-white);
border: 1px solid var(--altinity-accent);
box-shadow: 0 0 8px rgba(0, 0, 0, 0.05);
color: var(--altinity-background);
}

/* Table header styling */
th {
background-color: var(--altinity-accent);
color: var(--altinity-white);
padding: 10px 16px;
text-align: left;
border: none;
border-bottom: 2px solid var(--altinity-background);
white-space: nowrap;
}

th.hth {
border-bottom: 1px solid var(--altinity-accent);
border-right: 2px solid var(--altinity-background);
}

/* Table header sorting styling */
th {
cursor: pointer;
}

th.no-sort {
pointer-events: none;
}

th::after,
th::before {
transition: color 0.2s ease-in-out;
font-size: 1.2em;
color: transparent;
}

th::after {
margin-left: 3px;
content: '\025B8';
}

th:hover::after {
color: inherit;
}

th.dir-d::after {
color: inherit;
content: '\025BE';
}

th.dir-u::after {
color: inherit;
content: '\025B4';
}

/* Table body row styling */
tr:hover {
background-color: var(--altinity-light-gray);
}

/* Table cell styling */
td {
padding: 8px 8px;
border: 1px solid var(--altinity-accent);
}

/* Link styling */
a {
color: var(--altinity-accent);
text-decoration: none;
}

a:hover {
color: var(--altinity-highlight);
text-decoration: underline;
}
</style>
<title>{{ title }}</title>
<link rel="icon" type="image/svg+xml"
href=''
/>
</head>

<body>
<p><img class="logo"
src=""
alt="logo"></p>

<h1>{{ title }}</h1>
<table>
<tbody>
<tr>
<th class="hth no-sort">Pull Request</th>
<td>{{ pr_info_html }}</td>
</tr>
<tr>
<th class="hth no-sort">Workflow Run</th>
<td><a href="https://github.com/{{ github_repo }}/actions/runs/{{ workflow_id }}">{{ workflow_id }}</a></td>
</tr>
<tr>
<th class="hth no-sort">Commit</th>
<td><a href="https://github.com/{{ github_repo }}/commit/{{ commit_sha }}">{{ commit_sha }}</a></td>
</tr>
<tr>
<th class="hth no-sort">Build Report</th>
<td><a href="https://s3.amazonaws.com/{{ s3_bucket }}/{{ pr_number }}/{{ commit_sha }}/builds/report.html">Build Report</a></td>
</tr>
<tr>
<th class="hth no-sort">Date</th>
<td> {{ date }}</td>
</tr>
</tbody>
</table>
{% if is_preview %}
<p style="font-weight: bold;color: red;">This is a preview. The workflow is not yet finished.</p>
{% endif %}
<h2>Table of Contents</h2>
<ul>
{%- if pr_number != 0 %}<li><a href="#new-fails-pr">New Fails in PR</a> ({{ counts.pr_new_fails }})</li>{% endif %}
<li><a href="#ci-jobs-status">CI Jobs Status</a> ({{ counts.jobs_status }})</li>
<li><a href="#checks-errors">Checks Errors</a> ({{ counts.checks_errors }})</li>
<li><a href="#checks-fails">Checks New Fails</a> ({{ counts.checks_new_fails }})</li>
<li><a href="#regression-fails">Regression New Fails</a> ({{ counts.regression_new_fails }})</li>
<li><a href="#docker-images-cves">Docker Images CVEs</a> ({{ counts.cves }})</li>
<li><a href="#checks-known-fails">Checks Known Fails</a> ({{ counts.checks_known_fails }})</li>
</ul>

{%- if pr_number != 0 -%}
<h2 id="new-fails-pr">New Fails in PR</h2>
<p> Compared with base sha {{ base_sha }} </p>
{{ new_fails_html }}
{%- endif %}

<h2 id="ci-jobs-status">CI Jobs Status</h2>
{{ ci_jobs_status_html }}

<h2 id="checks-errors">Checks Errors</h2>
{{ checks_errors_html }}

<h2 id="checks-fails">Checks New Fails</h2>
{{ checks_fails_html }}

<h2 id="regression-fails">Regression New Fails</h2>
{{ regression_fails_html }}

<h2 id="docker-images-cves">Docker Images CVEs</h2>
{{ docker_images_cves_html }}

<h2 id="checks-known-fails">Checks Known Fails</h2>
<p>
Fail reason conventions:<br/>
KNOWN - Accepted fail and fix is not planned<br/>
INVESTIGATE - We don't know why it fails<br/>
NEEDSFIX - Investigation done and a fix is needed to make it pass<br/>
</p>
{{ checks_known_fails_html }}

<script>
document.addEventListener('click', function (e) {
try {
function findElementRecursive(element, tag) {
return element.nodeName === tag ? element :
findElementRecursive(element.parentNode, tag)
}
var descending_th_class = ' dir-d '
var ascending_th_class = ' dir-u '
var ascending_table_sort_class = 'asc'
var regex_dir = / dir-(u|d) /
var alt_sort = e.shiftKey || e.altKey
var element = findElementRecursive(e.target, 'TH')
var tr = findElementRecursive(element, 'TR')
var table = findElementRecursive(tr, 'TABLE')
function reClassify(element, dir) {
element.className = element.className.replace(regex_dir, '') + dir
}
function getValue(element) {
return (
(alt_sort && element.getAttribute('data-sort-alt')) ||
element.getAttribute('data-sort') || element.innerText
)
}
if (true) {
var column_index
var nodes = tr.cells
for (var i = 0; i < nodes.length; i++) {
if (nodes[i] === element) {
column_index = element.getAttribute('data-sort-col') || i
} else {
reClassify(nodes[i], '')
}
}
var dir = descending_th_class
if (
element.className.indexOf(descending_th_class) !== -1 ||
(table.className.indexOf(ascending_table_sort_class) !== -1 &&
element.className.indexOf(ascending_th_class) == -1)
) {
dir = ascending_th_class
}
reClassify(element, dir)
var org_tbody = table.tBodies[0]
var rows = [].slice.call(org_tbody.rows, 0)
var reverse = dir === ascending_th_class
rows.sort(function (a, b) {
var x = getValue((reverse ? a : b).cells[column_index])
var y = getValue((reverse ? b : a).cells[column_index])
return isNaN(x - y) ? x.localeCompare(y) : x - y
})
var clone_tbody = org_tbody.cloneNode()
while (rows.length) {
clone_tbody.appendChild(rows.splice(0, 1)[0])
}
table.replaceChild(clone_tbody, org_tbody)
}
} catch (error) {
}
});
</script>
</body>
</html>
Loading
Loading