Skip to content

Build and Publish (Slicer/Slicer@main, preview: false) #486

Build and Publish (Slicer/Slicer@main, preview: false)

Build and Publish (Slicer/Slicer@main, preview: false) #486

name: Build and Publish
on:
# Allows running this workflow manually from the Actions tab
workflow_dispatch:
inputs:
slicer_repository:
description: "Slicer Repository for which to build and publish the documentation"
default: Slicer/Slicer
slicer_ref:
description: "Slicer Branch or tag for which to build and publish the documentation"
default: main
preview:
description: "Publish at https://preview.apidocs.slicer.org"
default: false
type: boolean
run-name: "Build and Publish (${{ inputs.slicer_repository }}@${{ inputs.slicer_ref }}, preview: ${{ github.event.inputs.preview }})"
permissions:
# Needed in the publish step to update gh-pages branch
contents: write
# Needed to update the status associated with the Slicer commit
statuses: write
jobs:
build-and-publish:
runs-on: ubuntu-latest
steps:
- name: Collect Inputs
id: collect_inputs
run: |
echo "EVENT_NAME [$EVENT_NAME]"
if [[ "$EVENT_NAME" == "workflow_dispatch" ]]; then
slicer_repository=${{ github.event.inputs.slicer_repository }}
slicer_ref=${{ github.event.inputs.slicer_ref }}
preview=${{ github.event.inputs.preview }}
echo "preview [$preview]"
else
echo "::error ::Unsupported EVENT_NAME [$EVENT_NAME]"
exit 1
fi
if [[ ! $preview =~ ^(false|true)$ ]]; then
echo "::error ::Invalid skip value 'preview'. Supported values are 'true' or 'false'"
exit 1
fi
if [[ $preview == "false" ]]; then
publish_repository="apidocs.slicer.org"
else
publish_repository="preview.apidocs.slicer.org"
fi
echo "publish_repository [$publish_repository]"
echo "slicer_repository=$slicer_repository" >> $GITHUB_OUTPUT
echo "slicer_ref=$slicer_ref" >> $GITHUB_OUTPUT
echo "preview=$preview" >> $GITHUB_OUTPUT
echo "publish_repository=$publish_repository" >> $GITHUB_OUTPUT
env:
EVENT_NAME: ${{ github.event_name }}
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
with:
repository: ${{ steps.collect_inputs.outputs.slicer_repository }}
path: Slicer
ref: ${{ steps.collect_inputs.outputs.slicer_ref }}
- name: Slicer Checkout Details
id: slicer-checkout-details
run: |
cd Slicer
sha=$(git rev-parse HEAD)
echo "sha [$sha]"
echo "sha=$sha" >> $GITHUB_OUTPUT
- uses: actions/create-github-app-token@c1a285145b9d317df6ced56c09f525b5c2b6f755 # v1.11.1
id: app-token
with:
app-id: ${{ vars.SLICER_APP_ID }}
private-key: ${{ secrets.SLICER_APP_PRIVATE_KEY }}
owner: Slicer
repositories: |
apidocs.slicer.org
preview.apidocs.slicer.org
Slicer
Slicer-CI-Testing
- name: Set GitHub status to 'pending'
run: |
gh api \
--method POST \
-H "Accept: application/vnd.github+json" \
/repos/${SLICER_REPO_NAME}/statuses/${SLICER_REPO_REVISION} \
-f state='pending' \
-f target_url='https://github.com/${{ github.repository }}/actions/runs/${{ github.run_id }}' \
-f description='API documentation is being generated' \
-f context='slicer/apidocs'
env:
GH_TOKEN: ${{ steps.app-token.outputs.token }}
SLICER_REPO_NAME: ${{ steps.collect_inputs.outputs.slicer_repository }}
SLICER_REPO_REVISION: ${{ steps.slicer-checkout-details.outputs.sha }}
- name: Determine if ref is branch or tag
id: determine
run: |
cd Slicer
if git show-ref -q --verify "refs/heads/$SLICER_REPO_REF" 2>/dev/null; then
slicer_repo_branch=$SLICER_REPO_REF
echo "slicer_repo_branch [$slicer_repo_branch]"
echo "slicer_repo_branch=$slicer_repo_branch" >> $GITHUB_OUTPUT
exit 0
elif git show-ref -q --verify "refs/tags/$SLICER_REPO_REF" 2>/dev/null; then
slicer_repo_tag=$SLICER_REPO_REF
echo "slicer_repo_tag [$slicer_repo_tag]"
echo "slicer_repo_tag=$slicer_repo_tag" >> $GITHUB_OUTPUT
exit 0
elif git show-ref -q --verify "refs/remote/$SLICER_REPO_REF" 2>/dev/null; then
echo "::error ::Specifying reference as remote [$SLICER_REPO_REF] is not supported"
exit 1
elif git rev-parse --verify "$SLICER_REPO_REF^{commit}" >/dev/null 2>&1; then
echo "::error ::Specifying reference as remote [$SLICER_REPO_REF] is not supported"
exit 1
else
echo "unknown"
exit 1
fi
env:
SLICER_REPO_REF: ${{ steps.collect_inputs.outputs.slicer_ref }}
- uses: ssciwr/doxygen-install@527824132256e685f03ec80c0851fe79937eb1d6 # v1.6.3
with:
version: "1.13.0"
# The "dot" binary is provided by Graphviz
- uses: ts-graphviz/setup-graphviz@b1de5da23ed0a6d14e0aeee8ed52fdd87af2363c # v2.0.2
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
with:
repository: Slicer/slicer-apidocs-builder
path: slicer-apidocs-builder
ref: 0af7ea0c1d8ed80ffa904f65dfdc7e822d73e487
- uses: actions/setup-python@0b93645e9fea7318ecaed2b359559ac225c90a2b # v5.3.0
with:
python-version: '3.13'
cache: 'pip'
- name: Install slicer-apidocs-builder
run: |
pip install ./slicer-apidocs-builder
- name: Generate Documentation
id: generate
run: |
WARNING_LOG_FILE=/tmp/${SLICER_REPO_NAME//\//-}-$SLICER_REPO_BRANCH-build/Utilities/Doxygen/UserDoxygenWarnings.txt
echo "WARNING_LOG_FILE [$WARNING_LOG_FILE]"
mkdir -p $(dirname $WARNING_LOG_FILE)
slicer-apidocs-builder \
--skip-publish \
--slicer-repo-dir $(pwd)/Slicer \
--slicer-repo-name ${SLICER_REPO_NAME} \
--slicer-repo-branch "$SLICER_REPO_BRANCH" \
--slicer-repo-tag "${SLICER_REPO_TAG}" 2> >(tee $WARNING_LOG_FILE >&2)
echo "warning_log_file=$WARNING_LOG_FILE" >> $GITHUB_OUTPUT
env:
SLICER_REPO_NAME: ${{ steps.collect_inputs.outputs.slicer_repository }}
SLICER_REPO_BRANCH: ${{ steps.determine.outputs.slicer_repo_branch }}
SLICER_REPO_TAG: ${{ steps.determine.outputs.slicer_repo_tag }}
- name: Parse and annotate Doxygen warnings
run: |
echo "WARNING_LOG_FILE [$WARNING_LOG_FILE]"
if [[ -f $WARNING_LOG_FILE ]]; then
buffer=""
while IFS= read -r line || [[ -n "$line" ]]; do
# If a buffer exists, prepend the buffered line to this line
if [[ -n "$buffer" ]]; then
line="$buffer$line"
buffer=""
fi
# Extract file, line number, and warning message
FILE=$(echo "$line" | grep -oP "^[^:]+" || true)
LINE=$(echo "$line" | grep -oP ":(\d+):" | tr -d ":" || true)
MESSAGE=$(echo "$line" | grep -oP "warning:.*" || true)
# If MESSAGE is found, process further
if [[ -n "$MESSAGE" ]]; then
MESSAGE=${MESSAGE#warning: } # Strip "warning: " prefix
# Aggregate all subsequent lines starting with at least a space
while IFS= read -r continuation || [[ -n "$continuation" ]]; do
if [[ "$continuation" =~ ^[[:space:]] ]]; then
MESSAGE="${MESSAGE}%0A${continuation}"
else
# Buffer the line to be processed in the main loop
buffer="$continuation"
break
fi
done
fi
# Annotate in GitHub Actions
if [[ -n "$FILE" && -n "$LINE" && -n "$MESSAGE" ]]; then
echo "::warning file=$FILE,line=$LINE::$MESSAGE"
elif [[ -n "$MESSAGE" ]]; then
echo "::warning ::$MESSAGE"
else
echo "Skipped unmatched line: $line"
fi
done < "$WARNING_LOG_FILE"
else
echo "No Doxygen warnings log found."
fi
env:
WARNING_LOG_FILE: ${{ steps.generate.outputs.warning_log_file }}
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
with:
repository: Slicer/${{ steps.collect_inputs.outputs.publish_repository }}
ref: gh-pages
path: gh-pages
token: ${{ steps.app-token.outputs.token }}
- name: GitHub App Committer Details
id: committer
run: |
committer_name=${APP_SLUG}[bot]
echo "committer_name [$committer_name]"
user_id=$(gh api "/users/$committer_name" --jq .id)
echo "user_id [$user_id]"
committer_email=${user_id}+${committer_name}@users.noreply.github.com
echo "committer_email [$committer_email]"
echo "committer-name=$committer_name" >> "$GITHUB_OUTPUT"
echo "committer-email=$committer_email" >> "$GITHUB_OUTPUT"
env:
GH_TOKEN: ${{ steps.app-token.outputs.token }}
APP_SLUG: ${{ steps.app-token.outputs.app-slug }}
- name: Publish documentation
run: |
slicer-apidocs-builder \
--skip-build \
--slicer-repo-dir $(pwd)/Slicer \
--slicer-repo-name ${SLICER_REPO_NAME} \
--slicer-repo-branch "${SLICER_REPO_BRANCH}" \
--slicer-repo-tag "${SLICER_REPO_TAG}" \
--publish-github-username "${PUBLISH_GITHUB_USERNAME}" \
--publish-github-useremail "${PUBLISH_GITHUB_USERMAIL}" \
--publish-github-repo-dir "$(pwd)/gh-pages" \
--publish-github-repo-branch gh-pages \
--publish-github-skip-auth
env:
PUBLISH_GITHUB_USERNAME: ${{ steps.committer.outputs.committer-name }}
PUBLISH_GITHUB_USERMAIL: ${{ steps.committer.outputs.committer-email }}
SLICER_REPO_NAME: ${{ steps.collect_inputs.outputs.slicer_repository }}
SLICER_REPO_BRANCH: ${{ steps.determine.outputs.slicer_repo_branch }}
SLICER_REPO_TAG: ${{ steps.determine.outputs.slicer_repo_tag }}
- name: Set GitHub status to 'success'
run: |
target_url_path="$SLICER_REPO_BRANCH"
if [[ "$SLICER_REPO_TAG" != "" ]]; then
target_url_path=$(echo "$SLICER_REPO_TAG" | grep -oE '^v[0-9]+\.[0-9]+')
fi
echo "target_url_path [$target_url_path]"
target_url="https://${{ steps.collect_inputs.outputs.publish_repository }}/${target_url_path}"
echo "target_url [$target_url]"
gh api \
--method POST \
-H "Accept: application/vnd.github+json" \
/repos/${SLICER_REPO_NAME}/statuses/${SLICER_REPO_REVISION} \
-f state='success' \
-f target_url="$target_url" \
-f description='API documentation published' \
-f context='slicer/apidocs'
env:
GH_TOKEN: ${{ steps.app-token.outputs.token }}
SLICER_REPO_NAME: ${{ steps.collect_inputs.outputs.slicer_repository }}
SLICER_REPO_REVISION: ${{ steps.slicer-checkout-details.outputs.sha }}
SLICER_REPO_BRANCH: ${{ steps.determine.outputs.slicer_repo_branch }}
SLICER_REPO_TAG: ${{ steps.determine.outputs.slicer_repo_tag }}
- name: Set GitHub status to 'failure'
if: ${{ failure() }}
run: |
gh api \
--method POST \
-H "Accept: application/vnd.github+json" \
/repos/${SLICER_REPO_NAME}/statuses/${SLICER_REPO_REVISION} \
-f state='failure' \
-f target_url='https://github.com/${{ github.repository }}/actions/runs/${{ github.run_id }}' \
-f description='API documentation failed to be generated' \
-f context='slicer/apidocs'
env:
GH_TOKEN: ${{ steps.app-token.outputs.token }}
SLICER_REPO_NAME: ${{ steps.collect_inputs.outputs.slicer_repository }}
SLICER_REPO_REVISION: ${{ steps.slicer-checkout-details.outputs.sha }}