DO_NOT_MERGE: test #68
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
name: Coverage | |
permissions: | |
contents: write | |
pull-requests: write | |
on: | |
push: | |
branches: | |
- main | |
pull_request: | |
types: | |
- opened | |
- synchronize | |
- reopened | |
jobs: | |
Linux: | |
runs-on: ubuntu-latest | |
strategy: | |
matrix: | |
version: ['8.3'] | |
type: ['cli', 'zts'] | |
distro: ['bookworm'] | |
outputs: | |
matrix: ${{ toJson(matrix) }} | |
steps: | |
- name: Checkout | |
uses: actions/checkout@v4 | |
- name: Setup QEMU | |
uses: docker/setup-qemu-action@v3 | |
- name: Setup buildx | |
uses: docker/setup-buildx-action@v3 | |
- name: Build container | |
run: | | |
docker compose build --pull --no-cache --build-arg PLATFORM="linux/amd64" --build-arg IMAGE="php" --build-arg TAG="${{ matrix.version }}-${{ matrix.type }}-${{ matrix.distro }}" | |
- name: Test with gcov | |
run: | | |
docker compose run -v "$(pwd)/ext:/ext" --rm shell /bin/sh -c 'pskel test gcov && lcov --capture --directory "/ext" --output-file "/ext/lcov.info" --exclude "/usr/local/include/*" --exclude "third_party/*" && lcov --list "/ext/lcov.info"' | |
- name: Upload coverage to artifact | |
uses: actions/upload-artifact@v4 | |
with: | |
name: coverage-${{ matrix.version }}-${{ matrix.type }}-${{ matrix.distro }} | |
path: ${{ github.workspace }}/ext/lcov.info | |
Coverage: | |
needs: [Linux] | |
runs-on: ubuntu-latest | |
steps: | |
- name: Checkout | |
uses: actions/checkout@v4 | |
- name: Download coverage artifacts | |
uses: actions/download-artifact@v4 | |
- name: Merge coverages | |
run: | | |
sudo apt-get install -y "lcov" | |
LCOV_FILES="$(find . -name "lcov.info")" | |
CMD="$(which "lcov")" | |
for LCOV_FILE in ${LCOV_FILES}; do | |
CMD+=" -a ${LCOV_FILE}" | |
done | |
CMD+=" -o lcov.info" | |
echo "Merging coverages: ${LCOV_FILES}" | |
${CMD} | |
- name: Report coverage | |
uses: k1LoW/octocov-action@v1 | |
with: | |
github-token: ${{ secrets.GITHUB_TOKEN }} | |
config: .github/octocov.yml | |
- name: Upload coverage to artifact | |
uses: actions/upload-artifact@v4 | |
with: | |
name: coverage-unified | |
path: ${{ github.workspace }}/lcov.info | |
PullRequestReviewForUncoveredLines: | |
needs: [Coverage] | |
if: github.event_name == 'pull_request' | |
runs-on: ubuntu-latest | |
steps: | |
- name: Download coverage artifact | |
uses: actions/download-artifact@v4 | |
with: | |
name: coverage-unified | |
- name: Report comment for PR | |
uses: actions/github-script@v7 | |
with: | |
github-token: ${{ secrets.GITHUB_TOKEN }} | |
script: | | |
const fs = require('fs'); | |
const path = require('path'); | |
function parseLcovInfo(filename) { | |
const content = fs.readFileSync(filename, 'utf8'); | |
const lines = content.split('\n'); | |
const uncoveredLines = {}; | |
let currentFile = ''; | |
for (const line of lines) { | |
if (line.startsWith('SF:')) { | |
currentFile = line.substring(3).trim(); | |
} else if (line.startsWith('DA:')) { | |
const [lineNo, hits] = line.substring(3).trim().split(',').map(Number); | |
if (hits === 0) { | |
if (!uncoveredLines[currentFile]) { | |
uncoveredLines[currentFile] = []; | |
} | |
uncoveredLines[currentFile].push(lineNo); | |
} | |
} | |
} | |
return uncoveredLines; | |
} | |
function groupConsecutiveLines(lines) { | |
if (lines.length === 0) { | |
return []; | |
} | |
lines.sort((a, b) => a - b); | |
const groups = []; | |
let currentGroup = [lines[0]]; | |
for (let i = 1; i < lines.length; i++) { | |
if (lines[i] === lines[i-1] + 1) { | |
currentGroup.push(lines[i]); | |
} else { | |
groups.push(currentGroup); | |
currentGroup = [lines[i]]; | |
} | |
} | |
groups.push(currentGroup); | |
return groups; | |
} | |
async function createReviewForUncoveredLines(uncoveredLines) { | |
const { owner, repo } = context.repo; | |
const pull_number = context.issue.number; | |
const { data: files } = await github.rest.pulls.listFiles({ | |
owner, | |
repo, | |
pull_number, | |
}); | |
const comments = []; | |
for (const file of files) { | |
const filename = file.filename; | |
if (uncoveredLines[filename]) { | |
const groups = groupConsecutiveLines(uncoveredLines[filename]); | |
for (const group of groups) { | |
const startLine = group[0]; | |
const endLine = group[group.length - 1]; | |
const commentBody = `These lines are not covered by tests.`; | |
const comment = { | |
path: filename, | |
body: commentBody, | |
line: endLine, | |
side: 'RIGHT', | |
}; | |
if (startLine !== endLine) { | |
comment.start_line = startLine; | |
} | |
comments.push(comment); | |
} | |
} | |
} | |
if (comments.length > 0) { | |
try { | |
await github.rest.pulls.createReview({ | |
owner, | |
repo, | |
pull_number, | |
event: 'COMMENT', | |
body: 'Review for uncovered lines', | |
comments, | |
}); | |
} catch (error) { | |
console.error('Error creating review:', error); | |
if (error.response) { | |
console.error('Response status:', error.response.status); | |
console.error('Response data:', error.response.data); | |
} | |
} | |
} | |
} | |
const uncoveredLines = parseLcovInfo('lcov.info'); | |
await createReviewForUncoveredLines(uncoveredLines); |