mirror of
https://github.com/ppy/osu
synced 2025-01-11 00:29:30 +00:00
9681e3ac46
In https://github.com/ppy/osu/actions/runs/11311858931/job/31458581002, I cancelled the run during the download from `data.ppy.sh`. In https://github.com/ppy/osu/actions/runs/11313128285/job/31461534857, `wget` skipped downloading the file due to the `-nc` option (no-clobber), i.e.: if the file exists, don't re-download. The only way I'm aware of to resolve this with wget is to either use `-c` (continue), which may lead to broken files, or to explicitly specify the output file via `-O`. Thought I'd clean up a few pieces in the process. Why not curl? Mostly historical - some distros don't come with curl. It may be okay now but there's probably no point changing this at the moment...
410 lines
16 KiB
YAML
410 lines
16 KiB
YAML
# ## Description
|
|
#
|
|
# Uses [diffcalc-sheet-generator](https://github.com/smoogipoo/diffcalc-sheet-generator) to run two builds of osu and generate an SR/PP/Score comparison spreadsheet.
|
|
#
|
|
# ## Requirements
|
|
#
|
|
# Self-hosted runner with installed:
|
|
# - `docker >= 20.10.16`
|
|
# - `docker-compose >= 2.5.1`
|
|
# - `lbzip2`
|
|
# - `jq`
|
|
#
|
|
# ## Usage
|
|
#
|
|
# The workflow can be run in two ways:
|
|
# 1. Via workflow dispatch.
|
|
# 2. By an owner of the repository posting a pull request or issue comment containing `!diffcalc`.
|
|
# For pull requests, the workflow will assume the pull request as the target to compare against (i.e. the `OSU_B` variable).
|
|
# Any lines in the comment of the form `KEY=VALUE` are treated as variables for the generator.
|
|
#
|
|
# ## Google Service Account
|
|
#
|
|
# Spreadsheets are uploaded to a Google Service Account, and exposed with read-only permissions to the wider audience.
|
|
#
|
|
# 1. Create a project at https://console.cloud.google.com
|
|
# 2. Enable the `Google Sheets` and `Google Drive` APIs.
|
|
# 3. Create a Service Account
|
|
# 4. Generate a key in the JSON format.
|
|
# 5. Encode the key as base64 and store as an **actions secret** with name **`DIFFCALC_GOOGLE_CREDENTIALS`**
|
|
#
|
|
# ## Environment variables
|
|
#
|
|
# The default environment may be configured via **actions variables**.
|
|
#
|
|
# Refer to [the sample environment](https://github.com/smoogipoo/diffcalc-sheet-generator/blob/master/.env.sample), and prefix each variable with `DIFFCALC_` (e.g. `DIFFCALC_THREADS`, `DIFFCALC_INNODB_BUFFER_SIZE`, etc...).
|
|
|
|
name: Run difficulty calculation comparison
|
|
|
|
run-name: "${{ github.event_name == 'workflow_dispatch' && format('Manual run: {0}', inputs.osu-b) || 'Automatic comment trigger' }}"
|
|
|
|
on:
|
|
issue_comment:
|
|
types: [ created ]
|
|
workflow_dispatch:
|
|
inputs:
|
|
osu-b:
|
|
description: "The target build of ppy/osu"
|
|
type: string
|
|
required: true
|
|
ruleset:
|
|
description: "The ruleset to process"
|
|
type: choice
|
|
required: true
|
|
options:
|
|
- osu
|
|
- taiko
|
|
- catch
|
|
- mania
|
|
converts:
|
|
description: "Include converted beatmaps"
|
|
type: boolean
|
|
required: false
|
|
default: true
|
|
ranked-only:
|
|
description: "Only ranked beatmaps"
|
|
type: boolean
|
|
required: false
|
|
default: true
|
|
generators:
|
|
description: "Comma-separated list of generators (available: [sr, pp, score])"
|
|
type: string
|
|
required: false
|
|
default: 'pp,sr'
|
|
osu-a:
|
|
description: "The source build of ppy/osu"
|
|
type: string
|
|
required: false
|
|
default: 'latest'
|
|
difficulty-calculator-a:
|
|
description: "The source build of ppy/osu-difficulty-calculator"
|
|
type: string
|
|
required: false
|
|
default: 'latest'
|
|
difficulty-calculator-b:
|
|
description: "The target build of ppy/osu-difficulty-calculator"
|
|
type: string
|
|
required: false
|
|
default: 'latest'
|
|
score-processor-a:
|
|
description: "The source build of ppy/osu-queue-score-statistics"
|
|
type: string
|
|
required: false
|
|
default: 'latest'
|
|
score-processor-b:
|
|
description: "The target build of ppy/osu-queue-score-statistics"
|
|
type: string
|
|
required: false
|
|
default: 'latest'
|
|
|
|
permissions:
|
|
pull-requests: write
|
|
|
|
env:
|
|
EXECUTION_ID: execution-${{ github.run_id }}-${{ github.run_number }}-${{ github.run_attempt }}
|
|
|
|
jobs:
|
|
master-environment:
|
|
name: Save master environment
|
|
runs-on: ubuntu-latest
|
|
outputs:
|
|
HEAD: ${{ steps.get-head.outputs.HEAD }}
|
|
steps:
|
|
- name: Checkout osu
|
|
uses: actions/checkout@v4
|
|
with:
|
|
ref: master
|
|
sparse-checkout: |
|
|
README.md
|
|
|
|
- name: Get HEAD ref
|
|
id: get-head
|
|
run: |
|
|
ref=$(git log -1 --format='%H')
|
|
echo "HEAD=https://github.com/${{ github.repository }}/commit/${ref}" >> "${GITHUB_OUTPUT}"
|
|
|
|
check-permissions:
|
|
name: Check permissions
|
|
runs-on: ubuntu-latest
|
|
if: ${{ github.event_name == 'workflow_dispatch' || contains(github.event.comment.body, '!diffcalc') }}
|
|
steps:
|
|
- name: Check permissions
|
|
run: |
|
|
ALLOWED_USERS=(smoogipoo peppy bdach frenzibyte)
|
|
for i in "${ALLOWED_USERS[@]}"; do
|
|
if [[ "${{ github.actor }}" == "$i" ]]; then
|
|
exit 0
|
|
fi
|
|
done
|
|
exit 1
|
|
|
|
create-comment:
|
|
name: Create PR comment
|
|
needs: [ master-environment, check-permissions ]
|
|
runs-on: ubuntu-latest
|
|
if: ${{ github.event_name == 'issue_comment' && github.event.issue.pull_request }}
|
|
steps:
|
|
- name: Create comment
|
|
uses: thollander/actions-comment-pull-request@fabd468d3a1a0b97feee5f6b9e499eab0dd903f6 # v2.5.0
|
|
with:
|
|
comment_tag: ${{ env.EXECUTION_ID }}
|
|
message: |
|
|
Difficulty calculation queued -- please wait! (${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }})
|
|
|
|
*This comment will update on completion*
|
|
|
|
directory:
|
|
name: Prepare directory
|
|
needs: check-permissions
|
|
runs-on: self-hosted
|
|
outputs:
|
|
GENERATOR_DIR: ${{ steps.set-outputs.outputs.GENERATOR_DIR }}
|
|
GENERATOR_ENV: ${{ steps.set-outputs.outputs.GENERATOR_ENV }}
|
|
GOOGLE_CREDS_FILE: ${{ steps.set-outputs.outputs.GOOGLE_CREDS_FILE }}
|
|
steps:
|
|
- name: Checkout diffcalc-sheet-generator
|
|
uses: actions/checkout@v4
|
|
with:
|
|
path: ${{ env.EXECUTION_ID }}
|
|
repository: 'smoogipoo/diffcalc-sheet-generator'
|
|
|
|
- name: Set outputs
|
|
id: set-outputs
|
|
run: |
|
|
echo "GENERATOR_DIR=${{ github.workspace }}/${{ env.EXECUTION_ID }}" >> "${GITHUB_OUTPUT}"
|
|
echo "GENERATOR_ENV=${{ github.workspace }}/${{ env.EXECUTION_ID }}/.env" >> "${GITHUB_OUTPUT}"
|
|
echo "GOOGLE_CREDS_FILE=${{ github.workspace }}/${{ env.EXECUTION_ID }}/google-credentials.json" >> "${GITHUB_OUTPUT}"
|
|
|
|
environment:
|
|
name: Setup environment
|
|
needs: [ master-environment, directory ]
|
|
runs-on: self-hosted
|
|
env:
|
|
VARS_JSON: ${{ toJSON(vars) }}
|
|
steps:
|
|
- name: Add base environment
|
|
run: |
|
|
# Required by diffcalc-sheet-generator
|
|
cp '${{ needs.directory.outputs.GENERATOR_DIR }}/.env.sample' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
|
|
# Add Google credentials
|
|
echo '${{ secrets.DIFFCALC_GOOGLE_CREDENTIALS }}' | base64 -d > "${{ needs.directory.outputs.GOOGLE_CREDS_FILE }}"
|
|
|
|
# Add repository variables
|
|
echo "${VARS_JSON}" | jq -c '. | to_entries | .[]' | while read -r line; do
|
|
opt=$(jq -r '.key' <<< ${line})
|
|
val=$(jq -r '.value' <<< ${line})
|
|
|
|
if [[ "${opt}" =~ ^DIFFCALC_ ]]; then
|
|
optNoPrefix=$(echo "${opt}" | cut -d '_' -f2-)
|
|
sed -i "s;^${optNoPrefix}=.*$;${optNoPrefix}=${val};" "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
fi
|
|
done
|
|
|
|
- name: Add master environment
|
|
run: |
|
|
sed -i "s;^OSU_A=.*$;OSU_A=${{ needs.master-environment.outputs.HEAD }};" "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
|
|
- name: Add pull-request environment
|
|
if: ${{ github.event_name == 'issue_comment' && github.event.issue.pull_request }}
|
|
run: |
|
|
sed -i "s;^OSU_B=.*$;OSU_B=${{ github.event.issue.pull_request.html_url }};" "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
|
|
- name: Add comment environment
|
|
if: ${{ github.event_name == 'issue_comment' }}
|
|
env:
|
|
COMMENT_BODY: ${{ github.event.comment.body }}
|
|
run: |
|
|
# Add comment environment
|
|
echo "$COMMENT_BODY" | sed -r 's/\r$//' | grep -E '^\w+=' | while read -r line; do
|
|
opt=$(echo "${line}" | cut -d '=' -f1)
|
|
sed -i "s;^${opt}=.*$;${line};" "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
done
|
|
|
|
- name: Add dispatch environment
|
|
if: ${{ github.event_name == 'workflow_dispatch' }}
|
|
run: |
|
|
sed -i 's;^OSU_B=.*$;OSU_B=${{ inputs.osu-b }};' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
sed -i 's/^RULESET=.*$/RULESET=${{ inputs.ruleset }}/' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
sed -i 's/^GENERATORS=.*$/GENERATORS=${{ inputs.generators }}/' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
|
|
if [[ '${{ inputs.osu-a }}' != 'latest' ]]; then
|
|
sed -i 's;^OSU_A=.*$;OSU_A=${{ inputs.osu-a }};' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
fi
|
|
|
|
if [[ '${{ inputs.difficulty-calculator-a }}' != 'latest' ]]; then
|
|
sed -i 's;^DIFFICULTY_CALCULATOR_A=.*$;DIFFICULTY_CALCULATOR_A=${{ inputs.difficulty-calculator-a }};' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
fi
|
|
|
|
if [[ '${{ inputs.difficulty-calculator-b }}' != 'latest' ]]; then
|
|
sed -i 's;^DIFFICULTY_CALCULATOR_B=.*$;DIFFICULTY_CALCULATOR_B=${{ inputs.difficulty-calculator-b }};' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
fi
|
|
|
|
if [[ '${{ inputs.score-processor-a }}' != 'latest' ]]; then
|
|
sed -i 's;^SCORE_PROCESSOR_A=.*$;SCORE_PROCESSOR_A=${{ inputs.score-processor-a }};' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
fi
|
|
|
|
if [[ '${{ inputs.score-processor-b }}' != 'latest' ]]; then
|
|
sed -i 's;^SCORE_PROCESSOR_B=.*$;SCORE_PROCESSOR_B=${{ inputs.score-processor-b }};' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
fi
|
|
|
|
if [[ '${{ inputs.converts }}' == 'true' ]]; then
|
|
sed -i 's/^NO_CONVERTS=.*$/NO_CONVERTS=0/' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
else
|
|
sed -i 's/^NO_CONVERTS=.*$/NO_CONVERTS=1/' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
fi
|
|
|
|
if [[ '${{ inputs.ranked-only }}' == 'true' ]]; then
|
|
sed -i 's/^RANKED_ONLY=.*$/RANKED_ONLY=1/' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
else
|
|
sed -i 's/^RANKED_ONLY=.*$/RANKED_ONLY=0/' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
fi
|
|
|
|
scores:
|
|
name: Setup scores
|
|
needs: [ directory, environment ]
|
|
runs-on: self-hosted
|
|
steps:
|
|
- name: Query latest data
|
|
id: query
|
|
run: |
|
|
ruleset=$(cat ${{ needs.directory.outputs.GENERATOR_ENV }} | grep -E '^RULESET=' | cut -d '=' -f2-)
|
|
performance_data_name=$(curl -s "https://data.ppy.sh/" | grep "performance_${ruleset}_top_1000\b" | tail -1 | awk -F "'" '{print $2}' | sed 's/\.tar\.bz2//g')
|
|
|
|
echo "TARGET_DIR=${{ needs.directory.outputs.GENERATOR_DIR }}/sql/${ruleset}" >> "${GITHUB_OUTPUT}"
|
|
echo "DATA_NAME=${performance_data_name}" >> "${GITHUB_OUTPUT}"
|
|
echo "DATA_PKG=${performance_data_name}.tar.bz2" >> "${GITHUB_OUTPUT}"
|
|
|
|
- name: Restore cache
|
|
id: restore-cache
|
|
uses: maxnowack/local-cache@720e69c948191660a90aa1cf6a42fc4d2dacdf30 # v2
|
|
with:
|
|
path: ${{ steps.query.outputs.DATA_PKG }}
|
|
key: ${{ steps.query.outputs.DATA_NAME }}
|
|
|
|
- name: Download
|
|
if: steps.restore-cache.outputs.cache-hit != 'true'
|
|
run: |
|
|
wget -q -O "${{ steps.query.outputs.DATA_PKG }}" "https://data.ppy.sh/${{ steps.query.outputs.DATA_PKG }}"
|
|
|
|
- name: Extract
|
|
run: |
|
|
tar -I lbzip2 -xf "${{ steps.query.outputs.DATA_PKG }}"
|
|
rm -r "${{ steps.query.outputs.TARGET_DIR }}"
|
|
mv "${{ steps.query.outputs.DATA_NAME }}" "${{ steps.query.outputs.TARGET_DIR }}"
|
|
|
|
beatmaps:
|
|
name: Setup beatmaps
|
|
needs: directory
|
|
runs-on: self-hosted
|
|
steps:
|
|
- name: Query latest data
|
|
id: query
|
|
run: |
|
|
beatmaps_data_name=$(curl -s "https://data.ppy.sh/" | grep "osu_files" | tail -1 | awk -F "'" '{print $2}' | sed 's/\.tar\.bz2//g')
|
|
|
|
echo "TARGET_DIR=${{ needs.directory.outputs.GENERATOR_DIR }}/beatmaps" >> "${GITHUB_OUTPUT}"
|
|
echo "DATA_NAME=${beatmaps_data_name}" >> "${GITHUB_OUTPUT}"
|
|
echo "DATA_PKG=${beatmaps_data_name}.tar.bz2" >> "${GITHUB_OUTPUT}"
|
|
|
|
- name: Restore cache
|
|
id: restore-cache
|
|
uses: maxnowack/local-cache@720e69c948191660a90aa1cf6a42fc4d2dacdf30 # v2
|
|
with:
|
|
path: ${{ steps.query.outputs.DATA_PKG }}
|
|
key: ${{ steps.query.outputs.DATA_NAME }}
|
|
|
|
- name: Download
|
|
if: steps.restore-cache.outputs.cache-hit != 'true'
|
|
run: |
|
|
wget -q -O "${{ steps.query.outputs.DATA_PKG }}" "https://data.ppy.sh/${{ steps.query.outputs.DATA_PKG }}"
|
|
|
|
- name: Extract
|
|
run: |
|
|
tar -I lbzip2 -xf "${{ steps.query.outputs.DATA_PKG }}"
|
|
rm -r "${{ steps.query.outputs.TARGET_DIR }}"
|
|
mv "${{ steps.query.outputs.DATA_NAME }}" "${{ steps.query.outputs.TARGET_DIR }}"
|
|
|
|
generator:
|
|
name: Run generator
|
|
needs: [ directory, environment, scores, beatmaps ]
|
|
runs-on: self-hosted
|
|
timeout-minutes: 720
|
|
outputs:
|
|
TARGET: ${{ steps.run.outputs.TARGET }}
|
|
SPREADSHEET_LINK: ${{ steps.run.outputs.SPREADSHEET_LINK }}
|
|
steps:
|
|
- name: Run
|
|
id: run
|
|
run: |
|
|
# Add the GitHub token. This needs to be done here because it's unique per-job.
|
|
sed -i 's/^GH_TOKEN=.*$/GH_TOKEN=${{ github.token }}/' "${{ needs.directory.outputs.GENERATOR_ENV }}"
|
|
|
|
cd "${{ needs.directory.outputs.GENERATOR_DIR }}"
|
|
docker-compose up --build generator
|
|
|
|
link=$(docker-compose logs generator -n 10 | grep 'http' | sed -E 's/^.*(http.*)$/\1/')
|
|
target=$(cat "${{ needs.directory.outputs.GENERATOR_ENV }}" | grep -E '^OSU_B=' | cut -d '=' -f2-)
|
|
|
|
echo "TARGET=${target}" >> "${GITHUB_OUTPUT}"
|
|
echo "SPREADSHEET_LINK=${link}" >> "${GITHUB_OUTPUT}"
|
|
|
|
- name: Shutdown
|
|
if: ${{ always() }}
|
|
run: |
|
|
cd "${{ needs.directory.outputs.GENERATOR_DIR }}"
|
|
docker-compose down -v
|
|
|
|
output-cli:
|
|
name: Output info
|
|
needs: generator
|
|
runs-on: ubuntu-latest
|
|
steps:
|
|
- name: Output info
|
|
run: |
|
|
echo "Target: ${{ needs.generator.outputs.TARGET }}"
|
|
echo "Spreadsheet: ${{ needs.generator.outputs.SPREADSHEET_LINK }}"
|
|
|
|
cleanup:
|
|
name: Cleanup
|
|
needs: [ directory, generator ]
|
|
if: ${{ always() && needs.directory.result == 'success' }}
|
|
runs-on: self-hosted
|
|
steps:
|
|
- name: Cleanup
|
|
run: |
|
|
rm -rf "${{ needs.directory.outputs.GENERATOR_DIR }}"
|
|
|
|
update-comment:
|
|
name: Update PR comment
|
|
needs: [ create-comment, generator ]
|
|
runs-on: ubuntu-latest
|
|
if: ${{ always() && needs.create-comment.result == 'success' }}
|
|
steps:
|
|
- name: Update comment on success
|
|
if: ${{ needs.generator.result == 'success' }}
|
|
uses: thollander/actions-comment-pull-request@fabd468d3a1a0b97feee5f6b9e499eab0dd903f6 # v2.5.0
|
|
with:
|
|
comment_tag: ${{ env.EXECUTION_ID }}
|
|
mode: recreate
|
|
message: |
|
|
Target: ${{ needs.generator.outputs.TARGET }}
|
|
Spreadsheet: ${{ needs.generator.outputs.SPREADSHEET_LINK }}
|
|
|
|
- name: Update comment on failure
|
|
if: ${{ needs.generator.result == 'failure' }}
|
|
uses: thollander/actions-comment-pull-request@fabd468d3a1a0b97feee5f6b9e499eab0dd903f6 # v2.5.0
|
|
with:
|
|
comment_tag: ${{ env.EXECUTION_ID }}
|
|
mode: recreate
|
|
message: |
|
|
Difficulty calculation failed: ${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }}
|
|
|
|
- name: Update comment on cancellation
|
|
if: ${{ needs.generator.result == 'cancelled' }}
|
|
uses: thollander/actions-comment-pull-request@fabd468d3a1a0b97feee5f6b9e499eab0dd903f6 # v2.5.0
|
|
with:
|
|
comment_tag: ${{ env.EXECUTION_ID }}
|
|
mode: delete
|
|
message: '.' # Appears to be required by this action for non-error status code.
|