name: Get Conan Recipe Version on: workflow_call: inputs: project_name: required: true type: string user: required: false default: ultimaker type: string additional_buildmetadata: required: false default: "" type: string outputs: recipe_id_full: description: "The full Conan recipe id: /@/" value: ${{ jobs.get-semver.outputs.recipe_id_full }} recipe_id_latest: description: "The full Conan recipe aliased (latest) id: /(latest)@/" value: ${{ jobs.get-semver.outputs.recipe_id_latest }} recipe_semver_full: description: "The full semver ..-+" value: ${{ jobs.get-semver.outputs.semver_full }} is_release_branch: description: "is current branch a release branch?" value: ${{ jobs.get-semver.outputs.release_branch }} user: description: "The conan user" value: ${{ jobs.get-semver.outputs.user }} channel: description: "The conan channel" value: ${{ jobs.get-semver.outputs.channel }} project_name: description: "The conan projectname" value: ${{ inputs.project_name }} jobs: get-semver: runs-on: ubuntu-latest outputs: recipe_id_full: ${{ steps.get-conan-broadcast-data.outputs.recipe_id_full }} recipe_id_latest: ${{ steps.get-conan-broadcast-data.outputs.recipe_id_latest }} semver_full: ${{ steps.get-conan-broadcast-data.outputs.semver_full }} is_release_branch: ${{ steps.get-conan-broadcast-data.outputs.is_release_branch }} user: ${{ steps.get-conan-broadcast-data.outputs.user }} channel: ${{ steps.get-conan-broadcast-data.outputs.channel }} steps: - name: Checkout repo uses: actions/checkout@v3 if: ${{ github.event.pull_request.head.repo.full_name == github.repository }} with: fetch-depth: 0 ref: ${{ github.head_ref }} - name: Checkout repo PR uses: actions/checkout@v3 if: ${{ github.event.pull_request.head.repo.full_name != github.repository }} with: fetch-depth: 0 ref: ${{ github.base_ref }} - name: Setup Python and pip uses: actions/setup-python@v4 with: python-version: "3.10.x" cache: 'pip' cache-dependency-path: .github/workflows/requirements-conan-package.txt - name: Install Python requirements and Create default Conan profile run: | pip install -r .github/workflows/requirements-conan-package.txt pip install gitpython - id: get-conan-broadcast-data name: Get Conan broadcast data run: | import subprocess import os from conan.tools.scm import Version from conan.errors import ConanException from git import Repo repo = Repo('.') user = "${{ inputs.user }}".lower() project_name = "${{ inputs.project_name }}" event_name = "${{ github.event_name }}" issue_number = "${{ github.ref }}".split('/')[2] is_tag = "${{ github.ref_type }}" == "tag" is_release_branch = False ref_name = "${{ github.base_ref }}" if event_name == "pull_request" else "${{ github.ref_name }}" buildmetadata = "" if "${{ inputs.additional_buildmetadata }}" == "" else "${{ inputs.additional_buildmetadata }}_" # FIXME: for when we push a tag (such as an release) channel = "testing" if is_tag: branch_version = Version(ref_name) is_release_branch = True channel = "_" user = "_" actual_version = f"{branch_version}" else: try: branch_version = Version(repo.active_branch.name) except ConanException: branch_version = Version('0.0.0') if ref_name == f"{branch_version.major}.{branch_version.minor}": channel = 'stable' is_release_branch = True elif ref_name in ("main", "master"): channel = 'testing' else: channel = "_".join(repo.active_branch.name.replace("-", "_").split("_")[:2]).lower() if "pull_request" in event_name: channel = f"pr_{issue_number}" # %% Get the actual version latest_branch_version = Version("0.0.0") latest_branch_tag = None for tag in repo.active_branch.repo.tags: if str(tag).startswith("firmware") or str(tag).startswith("master"): continue # Quick-fix for the versioning scheme name of the embedded team in fdm_materials(_private) repo try: version = Version(tag) except ConanException: continue if version > latest_branch_version and version < Version("6.0.0"): # FIXME: stupid old Cura tags 13.04 etc. keep popping up, als the fdm_material tag for firmware are messing with this latest_branch_version = version latest_branch_tag = repo.tag(tag) if latest_branch_tag: # %% Get the actual version no_commits = 0 for commit in repo.iter_commits("HEAD"): if commit == latest_branch_tag.commit: break no_commits += 1 latest_branch_version_prerelease = latest_branch_version.pre if latest_branch_version.pre and not "." in str(latest_branch_version.pre): # The prerealese did not contain a version number, default it to 1 latest_branch_version_prerelease = f"{latest_branch_version.pre}.1" if event_name == "pull_request": actual_version = f"{latest_branch_version.major}.{latest_branch_version.minor}.{latest_branch_version.patch}-{str(latest_branch_version_prerelease).lower()}+{buildmetadata}pr_{issue_number}_{no_commits}" channel_metadata = f"{channel}_{no_commits}" else: if channel in ("stable", "_", ""): channel_metadata = f"{no_commits}" else: channel_metadata = f"{channel}_{no_commits}" if is_release_branch: if latest_branch_version.pre == "" and branch_version > latest_branch_version: actual_version = f"{branch_version.major}.{branch_version.minor}.0-beta.1+{buildmetadata}{channel_metadata}" elif latest_branch_version.pre == "": # An actual full release has been created, we are working on patch bump_up_patch = int(str(latest_branch_version.patch)) + 1 actual_version = f"{latest_branch_version.major}.{latest_branch_version.minor}.{bump_up_patch}-beta.1+{buildmetadata}{channel_metadata}" elif latest_branch_version.pre is None: actual_version = f"{latest_branch_version.major}.{latest_branch_version.minor}.{int(latest_branch_version.patch.value) + 1}-beta.1+{buildmetadata}{channel_metadata}" else: # An beta release has been created we are working toward a next beta or full release bump_up_release_tag = int(str(latest_branch_version.pre).split('.')[1]) + 1 actual_version = f"{latest_branch_version.major}.{latest_branch_version.minor}.{latest_branch_version.patch}-{str(latest_branch_version.pre).split('.')[0]}.{bump_up_release_tag}+{buildmetadata}{channel_metadata}" else: max_branches_version = Version("0.0.0") branches_no_commits = no_commits for branch in repo.references: try: if "remotes/origin" in branch.abspath: b_version = Version(branch.name.split("/")[-1]) if b_version < Version("10.0.0") and b_version > max_branches_version: max_branches_version = b_version branches_no_commits = repo.commit().count() - branch.commit.count() except: pass if max_branches_version > latest_branch_version: actual_version = f"{max_branches_version.major}.{int(str(max_branches_version.minor)) + 1}.0-alpha+{buildmetadata}{channel}_{branches_no_commits}" else: actual_version = f"{latest_branch_version.major}.{int(str(latest_branch_version.minor)) + 1}.0-alpha+{buildmetadata}{channel_metadata}" # %% Set the environment output output_env = os.environ["GITHUB_OUTPUT"] content = "" if os.path.exists(output_env): with open(output_env, "r") as f: content = f.read() with open(output_env, "w") as f: f.write(content) f.writelines(f"name={project_name}\n") f.writelines(f"version={actual_version}\n") f.writelines(f"channel={channel}\n") f.writelines(f"recipe_id_full={project_name}/{actual_version}@{user}/{channel}\n") f.writelines(f"recipe_id_latest={project_name}/latest@{user}/{channel}\n") f.writelines(f"semver_full={actual_version}\n") f.writelines(f"is_release_branch={str(is_release_branch).lower()}\n") summary_env = os.environ["GITHUB_STEP_SUMMARY"] with open(summary_env, "w") as f: f.writelines(f"# {project_name}\n") f.writelines(f"name={project_name}\n") f.writelines(f"version={actual_version}\n") f.writelines(f"channel={channel}\n") f.writelines(f"recipe_id_full={project_name}/{actual_version}@{user}/{channel}\n") f.writelines(f"recipe_id_latest={project_name}/latest@{user}/{channel}\n") f.writelines(f"semver_full={actual_version}\n") f.writelines(f"is_release_branch={str(is_release_branch).lower()}\n") shell: python