diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index 0fcdf03e67f2ce0e23404d1d8347331e1d1eb181..81216c79ef1deeab1279274212dbd3fa1bfb2d46 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -9,6 +9,7 @@ variables: CI_IMAGES_REV: latest CI_IMAGE_PYTHON: "${CI_IMAGES_PATH}/infrastructure/python3.9:${CI_IMAGES_REV}" CI_IMAGE_YOCTO: "secodocker/edgehog-builder:${CI_IMAGES_REV}" + PROJECT_INTEGRATION_GROUPS: "arm/nxp/imx/develop/yocto/5.x" #arm/rockchip image: "${CI_IMAGE_PYTHON}" @@ -75,14 +76,26 @@ executable: MERGE: "" script: - cd ${CI_PROJECT_DIR} - - echo - - echo ${CI_SERVER_URL} - echo ${GITBOT_TOKEN} - echo ${MANIFEST_PROJECT} - echo ${MANIFEST_BRANCH} - echo ${CI_COMMIT_SHA} - echo ${PROJECT_GROUP} - echo ${MERGE} + - | + for PROJECT_INTEGRATION in ${PROJECT_INTEGRATION_GROUPS}; do + echo ${PROJECT_INTEGRATION} + scripts/deploy_gitlab_ci_projects.py \ + --gitlab-url=${CI_SERVER_URL} \ + --token=${GITBOT_TOKEN} \ + --manifest-project=${MANIFEST_PROJECT} \ + --manifest-branch=${MANIFEST_BRANCH} \ + --submodule=.gitlab-ci \ + --revision=${CI_COMMIT_SHA} \ + --group="${PROJECT_INTEGRATION}" \ + --verbose \ + ${MERGE} + done - scripts/deploy_gitlab_ci.py --gitlab-url=${CI_SERVER_URL} --token=${GITBOT_TOKEN} @@ -116,7 +129,7 @@ executable: # extends: .integrate-ci-test # variables: # MANIFEST_BRANCH: primary -# + #integrate-ci-test:secondary: # extends: .integrate-ci-test # variables: @@ -166,7 +179,7 @@ integrate-yocto:kirkstone: # needs: ["integrate-ci-test:primary"] # variables: # MANIFEST_BRANCH: primary -# + #build-ci-test:secondary: # extends: .build-ci-test # needs: ["integrate-ci-test:secondary"] diff --git a/layers-integration.yml b/layers-integration.yml new file mode 100644 index 0000000000000000000000000000000000000000..ca522a0a60e7fc145aaf915e0c9fd1d930a43bf1 --- /dev/null +++ b/layers-integration.yml @@ -0,0 +1,186 @@ +--- +# -------------------------------------------------------------------------------------- +# Global +# -------------------------------------------------------------------------------------- + +include: + - local: common.yml + +stages: + - infrastructure + - integrate-into-layer + - create-merge + - check + - accept-merge + +variables: + # The BB_RECIPE_NAME is used for projects referenced in the SRCREV file + # to match the repository and the bitbake recipe name. + # We set it here to none, as every project needing it + # has to specify it in its own gitlab-ci.yml file. + # The BB_RECIPE_NAME is passed to the python scripts below anyway, but not + # used for projects referenced in the manifest file. + # FIXME: This is only necessary due to the following GitLab limitation: + # https://gitlab.com/gitlab-org/gitlab/-/issues/209904 + # As soon as this gets fixed upstream, the hard-coded branch name should be removed. + MANIFEST_PROJECT: yocto_ng/seco-manifest + MASTER_BRANCH_LAYER: kirkstone/develop + BB_RECIPE_NAME: none + SRCREV_FILE: conf/SRCREV.conf + + DEPLOYPATH_TEST: "/artifacts/${CI_JOB_ID}/" + GITLAB_SERVER: "${CI_SERVER_HOST}:${CI_SERVER_SSH_PORT}" + GIT_BASE_URL: "ssh://git@${GITLAB_SERVER}/${CI_PROJECT_ROOT_NAMESPACE}" + TESTS_GIT_URL: "${GIT_BASE_URL}/yocto/tests.git" + +# -------------------------------------------------------------------------------------- +# Stage: infrastructure +# -------------------------------------------------------------------------------------- +integrate-into-layer: + extends: .infrastructure + rules: + # Do not integration pipeline for merge requests for integrate/gitlab-ci/ branches + # The integration is done from the pipeline in gitlab-ci already + - if: $CI_COMMIT_REF_NAME =~ /^integrate\/gitlab-ci\/.*/ + when: never + # We have to make sure that the pipeline runs for the current manifest + # master at the time a merge request is created. Otherwise we cannot + # guarantee a green master after merging. + - if: $CI_MERGE_REQUEST_IID + # Explicitly allow externally triggered pipelines in every case + - if: $CI_PIPELINE_SOURCE == "pipeline" || $CI_PIPELINE_SOURCE == "api" + cache: + policy: push + script: + - cd ${CI_PROJECT_DIR} + - if [ -n "${CI_MERGE_REQUEST_IID}" ];then + MERGE_REQUEST="${CI_MERGE_REQUEST_IID}"; + else + MERGE_REQUEST="${CI_OPEN_MERGE_REQUESTS%%,*}"; + fi + - .gitlab-ci/scripts/integrate_into_layer.py + --gitlab-url=${CI_SERVER_URL} + --token=${GITBOT_TOKEN} + --layer-project=${LAYER_PROJECT_PATH} + --layer-branch=${MASTER_BRANCH_LAYER} + --project=${CI_PROJECT_PATH} + --srcrev-file=${SRCREV_FILE} + --merge-request=${MERGE_REQUEST} + --save-revision-to=srcrev_revision + --recipe-name=${BB_RECIPE_NAME} + --verbose + artifacts: + paths: + - srcrev_revision + - integration_branch_file + +# -------------------------------------------------------------------------------------- +# Stage: create-merge-request +# -------------------------------------------------------------------------------------- +create-merge-request: + stage: create-merge + extends: .infrastructure + rules: + # Do not integration pipeline for merge requests for integrate/gitlab-ci/ branches + # The integration is done from the pipeline in gitlab-ci already + - if: $CI_COMMIT_REF_NAME =~ /^integrate\/gitlab-ci\/.*/ + when: never + # We have to make sure that the pipeline runs for the current manifest + # master at the time a merge request is created. Otherwise we cannot + # guarantee a green master after merging. + - if: $CI_MERGE_REQUEST_IID + - if: $CI_PIPELINE_SOURCE == "pipeline" || $CI_PIPELINE_SOURCE == "api" + cache: + policy: push + script: + - cd ${CI_PROJECT_DIR} + - INTEGRATION_BRANCH=$(cat integration_branch_file) + - .gitlab-ci/scripts/create_merge_request.py + --gitlab-url=${CI_SERVER_URL} + --token=${GITBOT_TOKEN} + --project=${LAYER_PROJECT_PATH} + --source-branch=${INTEGRATION_BRANCH} + --target-branch=${MASTER_BRANCH_LAYER} + needs: + - job: integrate-into-layer + artifacts: true + +#yamllint: +# extends: .yamllint + +# -------------------------------------------------------------------------------------- +# Stage: check +# -------------------------------------------------------------------------------------- +check: + extends: .infrastructure + stage: check + rules: + # Do not run check if the "skip build" label is set on the merge request + - if: $CI_MERGE_REQUEST_LABELS =~ /skip build/ + when: never + # Do not integration pipeline for merge requests for integrate/gitlab-ci/ branches + # The integration is done from the pipeline in gitlab-ci already + - if: $CI_COMMIT_REF_NAME =~ /^integrate\/gitlab-ci\/.*/ + when: never + - if: $CI_MERGE_REQUEST_IID + # Explicitly allow externally triggered pipelines in every case + - if: $CI_PIPELINE_SOURCE == "pipeline" || $CI_PIPELINE_SOURCE == "api" + needs: ["integrate-into-layer"] + allow_failure: true + script: + - cd ${CI_PROJECT_DIR} + # When running in a trigger pipeline the CII_MERGE_REQUEST_IID is not set + # but CI_OPEN_MERGE_REQUESTS. We use the first of this comma separated list + # in this case + - if [ -n "${CI_MERGE_REQUEST_IID}" ];then + MERGE_REQUEST="${CI_MERGE_REQUEST_IID}"; + else + MERGE_REQUEST="${CI_OPEN_MERGE_REQUESTS%%,*}"; + fi + # The 'parent_merge_request' is passed from the trigger + # in case this check job is part of a gitlab-ci integration + # pipeline. It is only used to display the correct MR to run again + # in a failed check + - if [ -n "${parent_merge_request}" ];then + PARENT_MR="--parent-merge-request=${parent_merge_request}"; + fi + - while read -r integration; do + SOURCE_BRANCH=$(echo $integration | cut -d':' -f1); + TARGET_PROJECT=${LAYER_PROJECT_PATH}; + TARGET_BRANCH=$(echo $integration | cut -d':' -f3); + if [[ "$SOURCE_BRANCH" == "$CI_MERGE_REQUEST_TARGET_BRANCH_NAME" ]]; then + .gitlab-ci/scripts/check_if_layer_branch_is_up_to_date.py + --gitlab-url=${CI_SERVER_URL} + --token=${GITBOT_TOKEN} + --target-project=${TARGET_PROJECT} + --target-branch=${TARGET_BRANCH} + --source-project=${CI_PROJECT_PATH} + --recipe-name=${BB_RECIPE_NAME} + --merge-request=${CI_MERGE_REQUEST_IID} + ; + fi; + done <<< "$INTEGRATION" + +merge-into-layer: + extends: .infrastructure + stage: accept-merge + timeout: 4h + rules: + # Do not integration pipeline for merge requests for integrate/gitlab-ci/ branches + # The integration is done from the pipeline in gitlab-ci already + - if: $CI_COMMIT_REF_NAME =~ /^integrate\/gitlab-ci\/.*/ + when: never + - if: $CI_COMMIT_BRANCH == $MASTER_BRANCH + when: always + script: + - cd ${CI_PROJECT_DIR} + - echo ${CI_COMMIT_REF_NAME} + - .gitlab-ci/scripts/accept_layer_merge_request.py + --gitlab-url=${CI_SERVER_URL} + --token=${GITBOT_TOKEN} + --project=${CI_PROJECT_PATH} + --layer-project=${LAYER_PROJECT_PATH} + --target-branch=${MASTER_BRANCH} + --layer-target-branch=${MASTER_BRANCH_LAYER} + --recipe-name=${BB_RECIPE_NAME} + --rebase \ No newline at end of file diff --git a/manifest-integration.yml b/manifest-integration.yml index 605a2c3a7df82ac28ad95e1c873d91fc13ab27fa..70f3c463d0a1095176511c076b023acc09dd059b 100644 --- a/manifest-integration.yml +++ b/manifest-integration.yml @@ -24,6 +24,7 @@ variables: # https://gitlab.com/gitlab-org/gitlab/-/issues/209904 # As soon as this gets fixed upstream, the hard-coded branch name should be removed. MANIFEST_PROJECT: seco-manifest + MANIFEST_GROUP: yocto_ng MASTER_BRANCH_MANIFEST: kirkstone/develop BB_RECIPE_NAME: none @@ -61,7 +62,7 @@ integrate: - .gitlab-ci/scripts/integrate_into_manifest.py --gitlab-url=${CI_SERVER_URL} --token=${GITBOT_TOKEN} - --manifest-project=${MANIFEST_PROJECT} + --manifest-project=${MANIFEST_GROUP}/${MANIFEST_PROJECT} --manifest-file=${MANIFEST_FILE} --manifest-branch=${MASTER_BRANCH_MANIFEST} --project=${CI_PROJECT_PATH} @@ -73,7 +74,6 @@ integrate: paths: - manifest_revision - #yamllint: # extends: .yamllint @@ -92,16 +92,25 @@ merge: when: always script: - cd ${CI_PROJECT_DIR} + - echo ${CI_SERVER_URL} + - echo ${GITBOT_TOKEN} + - echo ${MANIFEST_PROJECT} + - echo ${MASTER_BRANCH_MANIFEST} + - echo ${CI_PROJECT_PATH} + - echo ${MASTER_BRANCH} + - echo ${CI_COMMIT_SOURCE} + - echo ${BB_RECIPE_NAME} - .gitlab-ci/scripts/merge_into_manifest.py --gitlab-url=${CI_SERVER_URL} --token=${GITBOT_TOKEN} - --manifest-project=${MANIFEST_PROJECT} + --manifest-project=${MANIFEST_GROUP}/${MANIFEST_PROJECT} --manifest-branch=${MASTER_BRANCH_MANIFEST} --project=${CI_PROJECT_PATH} --project-branch=${MASTER_BRANCH} --commit=${CI_COMMIT_SOURCE} --save-revision-to=manifest_revision --recipe-name=${BB_RECIPE_NAME} + --verbose artifacts: paths: - manifest_revision diff --git a/manifest-pipeline-yocto.yml b/manifest-pipeline-yocto.yml index c0d96905f407ba209b9a50a864964a525218b814..46c34f37ccbb7a3b80a2c3c810f9979c2761b26d 100644 --- a/manifest-pipeline-yocto.yml +++ b/manifest-pipeline-yocto.yml @@ -191,7 +191,10 @@ retrigger: if [ ${CI_PIPELINE_SOURCE} == "pipeline" ]; then echo "This is the manifest branch used for the build ${CI_COMMIT_REF_NAME}" BUILD_BRANCH="${CI_COMMIT_REF_NAME}" +<<<<<<< HEAD fi +======= +>>>>>>> c0c25b3b80578877a4c95177a8158f4c2adf5245 - | su secous -c " repo init -u ${CI_REPOSITORY_URL} -b ${BUILD_BRANCH}; diff --git a/scripts/accept_layer_merge_request.py b/scripts/accept_layer_merge_request.py new file mode 100755 index 0000000000000000000000000000000000000000..ecc0453b9c6b792df645078a30396fac44fc944b --- /dev/null +++ b/scripts/accept_layer_merge_request.py @@ -0,0 +1,242 @@ +#!/usr/bin/env python3 +import common + +import argparse +import logging +import sys +import time +from gitlab import ( + Gitlab, + GitlabGetError, + GitlabMRClosedError, +) +from get_merge_requests import get_merge_requests + +critical_error = ( + "This is a critical error! Please make sure to:\n" + " 1. merge the above-mentioned merge request by hand as soon as possible\n" + " (this is important, otherwise following merge requests will get stuck, too)\n" + " 2. examine why this has happened and fix it in the CI pipeline" +) + + +def get_source_integration_requests( + project, state=None, target_branch=None, commit=None +): + """Get merge request by target branch and optionally commit sha""" + merge_requests = [] + try: + all_merge_requests = project.mergerequests.list( + target_branch=target_branch, + state=state if state else "all", + all=True, + retry_transient_errors=True, + order_by='updated_at' + ) + except GitlabGetError as e: + sys.exit( + "ERROR: could not list merge requests for project '%s': %s" + % (project.name, e) + ) + if commit: + for mr in all_merge_requests: + if mr.sha == commit or mr.squash_commit_sha == commit: + merge_requests.append(mr) + elif all_merge_requests: + merge_requests = all_merge_requests + + # Get complete objects + full_merge_requests = [] + for mr in merge_requests: + mr = project.mergerequests.get(mr.iid, retry_transient_errors=True) + full_merge_requests.append(mr) + + return full_merge_requests[0].source_branch + +def accept_merge_request(project, mr, rebase=False, should_remove_source_branch=True): + """Attempt to merge a merge request, rebase if necessary""" + merged = False + pipeline_pending = False + + while not merged: + # Update merge request before trying to merge it in order to get the latest + # pipeline status + try: + updated_mr = project.mergerequests.get( + id=mr.iid, retry_transient_errors=True + ) + mr = updated_mr + except GitlabGetError as e: + print("WARNING: Could not update merge request object: %s" % e) + + # Try to merge the merge request + try: + mr.merge(should_remove_source_branch=should_remove_source_branch) + if pipeline_pending: + print("") + if mr.state == "merged": + merged = True + else: + if mr.merge_error: + print("Merge error: %s" % mr.merge_error) + else: + print("Merge reported success, but MR state is '%s'" % mr.state) + return False, mr.sha + + except GitlabMRClosedError as e: + # See HTTP error codes for merge requests here: + # https://docs.gitlab.com/ce/api/merge_requests.html#accept-mr + logging.debug("Error from gitlab: %d", e.response_code) + + if e.response_code == 405: + # Not allowed (draft, closed, pipeline pending or failed) + # Contrary to the documentation, this response is also issued when the + # merge failed due to a merge conflict. See GitLab issue: + # https://gitlab.com/gitlab-org/gitlab/-/issues/364102 + + if mr.has_conflicts: + # Merge conflict, automatic rebase not possible + if pipeline_pending: + print("") + print("Merge not possible, has to be rebased manually") + return False, mr.sha + + # If pipeline is running, wait for completion + if not mr.head_pipeline: + # No pipeline created yet + print("No pipeline created yet") + time.sleep(1) + elif mr.head_pipeline["status"] in common.pending_states: + # Pipeline pending + if not pipeline_pending: + print("Waiting for pending pipeline", end="", flush=True) + pipeline_pending = True + print(".", end="", flush=True) + time.sleep(1) + else: + # Merge failed due to some other reason + if pipeline_pending: + print("") + print("Merge not possible for unkown reason") + return False, mr.sha + + elif e.response_code == 406: + # Merge conflict, automatic rebase is possible + if pipeline_pending: + print("") + pipeline_pending = False + print("Merge not possible, but branch can be automatically rebased") + if not rebase: + return False, mr.sha + print("Trying to rebase...") + mr = common.rebase_merge_request(project, mr) + if mr.merge_error: + print("ERROR: rebase not possible\n'%s'" % mr.merge_error) + sys.exit(critical_error) + print("Sucessfully rebased") + + else: + if pipeline_pending: + print("") + print("ERROR: merge not possible: %s" % e) + sys.exit(critical_error) + + return True, mr.sha + + +def main(): + parser = argparse.ArgumentParser() + parser.add_argument( + "--gitlab-url", + help="""URL to the GitLab instance""", + dest="gitlab_url", + required=True, + ) + parser.add_argument( + "--token", + help="""GitLab REST API private access token""", + dest="token", + required=True, + ) + parser.add_argument( + "--project", + help="""name of the GitLab project""", + dest="project", + required=True, + ) + parser.add_argument( + "--layer-project", + help="""name of the GitLab layer project""", + dest="layer_project", + required=True, + ) + parser.add_argument( + "--target-branch", + help="""target branch of the merge request""", + dest="target_branch", + required=True, + ) + parser.add_argument( + "--layer-target-branch", + help="""target branch of the layer merge request""", + dest="layer_target_branch", + required=True, + ) + parser.add_argument( + "--recipe-name", + help="""recipe name of the merge request""", + dest="recipe_name", + required=True, + ) + parser.add_argument( + "--rebase", + help="""attempt to automatically rebase merge request if necessary""", + dest="rebase", + action="store_true", + required=False, + ) + + args, _ = parser.parse_known_args() + + gitlab = Gitlab(args.gitlab_url, private_token=args.token) + project = common.get_project(gitlab, args.project) + layer_project = common.get_project(gitlab, args.layer_project) + + #Retrieving the source branch from the last MR merged in the project + try: + integration_branch_name = get_source_integration_requests( + project, + target_branch=args.target_branch, + state="merged", + ) + print("This is the source branch of the latest MR merged: ", integration_branch_name) + + #If the last merged branch is the gitlab-ci integration do nothing + if "integrate/gitlab-ci" in integration_branch_name: + print("The job was triggered by a merge of the gitlab-ci projects into the master branch, doing nothing!") + return "" + + except GitlabGetError as e: + sys.exit("Could not get integration branch name for latest project MR: %s" % e) + + integration_branch_name = args.recipe_name + "/" + integration_branch_name + + try: + merge_request = get_merge_requests( + layer_project, + source_branch=integration_branch_name, + target_branch=args.layer_target_branch, + state="opened", + ) + except GitlabGetError as e: + sys.exit("Could not get merge request: %s" % e) + + print("This job is going to merge MR %s in project %s", merge_request[0], layer_project) + if accept_merge_request(layer_project, merge_request[0], rebase=args.rebase): + print("Successfully merged") + else: + sys.exit(1) + + +if __name__ == "__main__": + main() diff --git a/scripts/check_if_layer_branch_is_up_to_date.py b/scripts/check_if_layer_branch_is_up_to_date.py new file mode 100755 index 0000000000000000000000000000000000000000..e3c302478834676fea97e684c46fb1b12f0dc9f4 --- /dev/null +++ b/scripts/check_if_layer_branch_is_up_to_date.py @@ -0,0 +1,140 @@ +#!/usr/bin/env python3 +import common + +import argparse +import sys +import logging +from gitlab import Gitlab, GitlabGetError +from gitlab.v4.objects import Project + + +def check_if_layer_branch_is_up_to_date( + target_project: Project, + target_branch_name: str, + integration_branch_name: str, +): + try: + integration_branch = target_project.branches.get( + integration_branch_name, retry_transient_errors=True + ) + except GitlabGetError: + sys.exit( + "ERROR: could not find integration branch {} in {}.".format( + integration_branch_name, target_project.name + ) + ) + + try: + target_branch = target_project.branches.get( + target_branch_name, retry_transient_errors=True + ) + except GitlabGetError: + sys.exit( + "ERROR: could not find target branch {} in {}.".format( + target_branch_name, target_project.name + ) + ) + + # Loop over the commits until the integration_branch head id is found + return common.is_commit_parent_of_project_commit( + target_project, + integration_branch.commit["id"], + target_branch.commit["id"], + limit=10, + ) + + +def main(): + parser = argparse.ArgumentParser() + parser.add_argument( + "--gitlab-url", + help="""URL to the GitLab instance""", + dest="gitlab_url", + required=True, + ) + parser.add_argument( + "--token", + help="""GitLab REST API private access token""", + dest="token", + required=True, + ) + parser.add_argument( + "--target-project", + help="""name of the target project""", + dest="target_project", + required=True, + ) + parser.add_argument( + "--target-branch", + help="""target branch to integrate into""", + dest="target_branch", + required=True, + ) + parser.add_argument( + "--source-project", + help="""name of the source project""", + dest="source_project", + required=True, + ) + parser.add_argument( + "--recipe-name", + help="""name of the BB recipe (for integration branch name)""", + dest="recipe_name", + required=True, + ) + parser.add_argument( + "--merge-request", + help="""source project merge request IID containing the changes to be integrated""", + dest="merge_request", + required=True, + ) + parser.add_argument( + "-v", + "--verbose", + action="store_true", + help="""Increase verbosity.""", + ) + + args, _ = parser.parse_known_args() + if args.verbose: + logging.basicConfig(level=logging.DEBUG) + + gitlab = Gitlab(args.gitlab_url, private_token=args.token) + + logging.debug(args) + target_project = common.get_project(gitlab, args.target_project) + source_project = common.get_project(gitlab, args.source_project) + merge_request = common.get_merge_request(source_project, args.merge_request) + if merge_request is None: + sys.exit( + "ERROR: could not get %s %s" % (source_project.name, args.merge_request) + ) + + integration_branch_name = args.recipe_name + "/" + merge_request.source_branch + + if check_if_layer_branch_is_up_to_date( + target_project=target_project, + target_branch_name=args.target_branch, + integration_branch_name=integration_branch_name, + ): + print( + "Integration branch {} in {} is up to date.".format( + integration_branch_name, target_project.name + ) + ) + else: + sys.exit( + "Integration branch {} in {} is not up to date.\n" + "Please re-run the MR pipeline:\n" + " 1. Open the MR pipelines page:\n" + " {}\n" + " 2. Click 'Run Pipeline'".format( + integration_branch_name, + target_project.name, + merge_request.web_url + "/pipelines", + ) + ) + + +if __name__ == "__main__": + main() diff --git a/scripts/deploy_gitlab_ci_projects.py b/scripts/deploy_gitlab_ci_projects.py new file mode 100755 index 0000000000000000000000000000000000000000..25d5f9652ed4c253972cc5c4b28a3bf5c8eedbb6 --- /dev/null +++ b/scripts/deploy_gitlab_ci_projects.py @@ -0,0 +1,307 @@ +#!/usr/bin/env python3 +import common + +import argparse +import logging +import sys +import os +from gitlab import Gitlab +from gitlab.v4.objects import Project, MergeRequest + +from accept_merge_request import accept_merge_request +from create_merge_request import create_merge_request +from get_integration_sources import get_integration_sources +from get_merge_requests import get_merge_requests +from update_submodule import update_submodule_and_include_ref +from integrate_into_manifest import update_manifest, update_srcrev + +from ruamel.yaml import YAML + + +def read_keys_from_gitlab_ci_yml(gitlab_ci_yml): + + # Read values from existing file + yaml = YAML() + data = yaml.load(gitlab_ci_yml) + logging.debug("Yaml: %s", data) + + try: + recipe = data["variables"]["BB_RECIPE_NAME"] + logging.debug("Recipe %s", recipe) + except KeyError: + recipe = None + return {"recipe": recipe} + + +def integrate_submodule_into( + gitlab, + project_name, + submodule_name, + new_revision, + branch, + commit_and_push=True, + force_clone=False, +): + + gitlab_project = common.get_project(gitlab, project_name) + + ( + project_repo, + project_dir, + integration_branch_name, + integration_commit, + message, + ) = update_submodule_and_include_ref( + gitlab_project, + submodule_name, + new_revision, + branch, + commit_and_push=commit_and_push, + force_clone=force_clone, + ) + # ====================================== + # Store the references for creating the integration + # commit in the manifest later + # ====================================== + ret = { + "project": gitlab_project, + "repo": project_repo, + "dir": project_dir, + "integration_branch": integration_branch_name, + "master_branch": branch, + "commit": integration_commit, + "message": message, + } + logging.debug( + "Integration branch: %s (%s)", + integration_branch_name, + integration_commit, + ) + return ret + + +def create_integration_merge_request( + project: Project, + integration_branch: str, + target_branch: str, + source_mr: MergeRequest = None, +) -> MergeRequest: + # Create merge request + # This should be optional + mr, created = create_merge_request(project, integration_branch, target_branch) + if created: + if source_mr is not None: + common.crosslink_merge_requests(source_mr, mr) + print("Created new merge request:\n%s" % mr.web_url) + else: + print("Existing integration merge request:\n%s" % mr.web_url) + return mr + + +def main(): + parser = argparse.ArgumentParser() + parser.add_argument( + "--gitlab-url", + help="""URL to the GitLab instance""", + dest="gitlab_url", + required=True, + ) + parser.add_argument( + "--token", + help="""GitLab REST API private access token""", + dest="token", + required=True, + ) + parser.add_argument( + "--manifest-project", + help="""name of the manifest project""", + dest="manifest_project", + required=True, + ) + parser.add_argument( + "--manifest-branch", + help="""manifest branch to integrate changes into (can be a comma-separated list)""", + dest="manifest_branch", + required=True, + ) + parser.add_argument( + "--submodule", + help="""submodule to update""", + dest="submodule", + required=True, + ) + parser.add_argument( + "--revision", + help="""new revision for submodule""", + dest="revision", + required=True, + ) + parser.add_argument( + "--merge", + help="""if set, perform merge after integration""", + dest="merge", + action="store_true", + required=False, + default=False, + ) + parser.add_argument( + "--project", + help="""gitlab-ci project path or id""", + dest="project", + default=os.environ.get("CI_PROJECT_PATH"), + required=False, + ) + parser.add_argument( + "--branch", + help="""gitlab-ci branch that we're merging into""", + dest="branch", + default="master", + required=False, + ) + parser.add_argument( + "--manifest-file", + help="""manifest file name (default: 'default.xml')""", + dest="manifest_file", + default=common.manifest_file, + required=False, + ) + parser.add_argument( + "--srcrev-file", + help="""source revision file name (default: 'SRCREV.conf')""", + dest="srcrev_file", + default=common.srcrev_file, + required=False, + ) + parser.add_argument( + "--group", + help="""group path or id to limit search scope to""", + dest="group", + required=True, + ) + parser.add_argument( + "-v", + "--verbose", + action="store_true", + help="""Increase verbosity.""", + ) + + args, _ = parser.parse_known_args() + if args.verbose: + logging.basicConfig( + level=logging.DEBUG, + format="%(asctime)s %(levelname)-8s %(message)s", + datefmt="%H:%M:%S", + ) + + manifest_branches = args.manifest_branch.split(",") + + gitlab = Gitlab(args.gitlab_url, private_token=args.token) + group = gitlab.groups.get(args.group) + + # ======================================================= + # Create integration branches and commits with updates + # submodule in all projects + # ======================================================= + integration_sources = {} + all_integration_sources = [] + for manifest_branch in manifest_branches: + print( + "Searching for projects in %s that are configured for automatic integration into %s:%s" + % (args.group, args.manifest_project, manifest_branch) + ) + integration_sources[manifest_branch] = get_integration_sources( + args.manifest_project, manifest_branch, group + ) + for s in integration_sources[manifest_branch]: + if s not in all_integration_sources: + all_integration_sources.append(s) + + # Update submodule in all integration sources + project_integrations = [] + for s in all_integration_sources: + print("Create integration commit in %s:%s" % (s["project"], s["branch"])) + + integration = integrate_submodule_into( + gitlab, s["project"], args.submodule, args.revision, s["branch"] + ) + # Store in the list if commit is set (meaning there was an update or + # an exising integration branch) + if integration["commit"] is not None: + project_integrations.append(integration) + + # Update submodule in all manifest branches + manifest_integrations = [] + for manifest_branch in manifest_branches: + print( + "Create integration commit in %s:%s" + % (args.manifest_project, manifest_branch), + ) + manifest_integrations.append( + integrate_submodule_into( + gitlab, + args.manifest_project, + args.submodule, + args.revision, + manifest_branch, + commit_and_push=False, + force_clone=True, + ) + ) + + # ======================================================= + # Create and merge merge_requests if needed + # ======================================================= + if args.merge: + # Get source merge request (the one in the gitlab-ci repo) + gitlab_ci_project = common.get_project(gitlab, args.project) + mrs = get_merge_requests( + project=gitlab_ci_project, + target_branch=args.branch, + state="merged", + commit=args.revision, + ) + if not mrs: + sys.exit( + "ERROR: could not determine source merge request for commit %s" + % args.revision + ) + source_mr = mrs[0] + + for project_integration in project_integrations: + logging.debug("Create MR in %s", project_integration["project"].name) + mr = create_integration_merge_request( + project_integration["project"], + project_integration["integration_branch"], + project_integration["master_branch"], + source_mr, + ) + # Now merge + logging.debug("Merge %s!%s", project_integration["project"], mr.iid) + + # Wait until GitLab has checked merge status + common.wait_until_merge_status_is_set(project_integration["project"], mr) + + # Attempt to merge + merged, integration_commit = accept_merge_request( + project_integration["project"], mr, rebase=True + ) + # if this has rebased the integration commit needs to be adapted: + project_integration["commit"] = integration_commit + # Save the target branch here, as the source branch gets deleted + # during merge + project_integration["integration_branch"] = mr.target_branch + + if not merged: + sys.exit( + "Integration MR could not be merged:\n" + "%s\n" + "This can probably be resolved by creating a new commit in " + "gitlab-ci and merging it. The above MR can be closed then." + % mr.web_url + ) + + if not args.merge: + sys.exit(0) + +if __name__ == "__main__": + main() diff --git a/scripts/integrate_into_layer.py b/scripts/integrate_into_layer.py new file mode 100755 index 0000000000000000000000000000000000000000..03e4e82c48e64387fb2e4593a3cb4b5fc2c309a5 --- /dev/null +++ b/scripts/integrate_into_layer.py @@ -0,0 +1,233 @@ +#!/usr/bin/env python3 +import common + +import argparse +import logging +import sys +import tempfile +import re +from pathlib import Path +from furl import furl +from git import GitCommandError, Repo +from gitlab import Gitlab +from gitlab.v4.objects import Project +from lxml import etree + +def update_srcrev(srcrev, recipe_name, new_revision): + # Check if project is referenced in SRCREV.conf + # Match "...RECIPE_NAME =" + pattern = re.compile("{}[ ,\t]{{0,}}?=".format(recipe_name)) + + project_line = None + for line in srcrev.splitlines(): + if pattern.search(line): + project_line = line + break + if project_line is None: + return None + + # Get current project revision from SRCREV file + # Assuming notation: <project> = "<hash>" + old_revision = project_line.split('"')[1] + + # Update SRCREV file + srcrev = srcrev.replace(old_revision, new_revision) + return srcrev + + +def integrate_into_layer( + layer_project: Project, + layer_branch, + srcrev_file, + recipe_name, + project: Project, + merge_request, +): + gitlab = layer_project.manager.gitlab + + with tempfile.TemporaryDirectory() as layer_dir: + srcrev_filepath = Path(layer_dir) / srcrev_file + + # Construct clone url containing access token + clone_url = furl(layer_project.http_url_to_repo) + print("This is the url to clone", clone_url.url) + clone_url.username = "gitlab-ci" + clone_url.password = gitlab.private_token + + # Checkout layer + # TODO replace checkout with gitlab api access + print("Cloning layer repo: %s" % layer_project.http_url_to_repo) + try: + layer_repo = Repo.clone_from( + clone_url.url, layer_dir, branch=layer_branch + ) + except GitCommandError as e: + sys.exit("ERROR: could not clone layer repository\n" + str(e)) + except IndexError: + sys.exit("ERROR: branch '%s' not found" % layer_branch) + + # Special handling for the gitlab-ci integration + # When the branch 'merge_request.source_branch' already starts with + # integrate/gitlab-ci we add our new commit to this branch + # Otherwise (normal behaviour) a new integration branch is created + integration_branch = common.find_gitlab_ci_integration_branch( + layer_repo, merge_request.source_branch + ) + + if integration_branch is not None: + layer_repo.git.checkout( + "-b", integration_branch, "origin/{}".format(integration_branch) + ) + logging.debug("Heads: %s", layer_repo.heads) + layer_repo.heads[integration_branch].checkout() + logging.debug(layer_repo.git.log("--oneline", "-n", "5")) + print("Using existing integration branch: %s" % integration_branch) + else: + # Create integration branch (delete former one if already exists) + integration_branch = recipe_name + "/" + merge_request.source_branch + for ref in layer_repo.references: + if integration_branch == ref.name: + layer_repo.delete_head(ref) + + print("Creating integration branch: %s" % integration_branch) + layer_repo.head.set_reference( + layer_repo.create_head(integration_branch) + ) + with open('integration_branch_file', "w", encoding="utf-8") as file: + file.write(integration_branch) + # Get new project revision from merge request + new_revision = merge_request.sha + + with open(srcrev_filepath, "r", encoding="utf8") as fp: + srcrev = fp.read() + new_srcrev = update_srcrev(srcrev, recipe_name, new_revision) + # write file + if new_srcrev is None: + sys.exit( + "ERROR: project '%s' not found in layer and " + "no recipe name is specified" % project.path + ) + with open(srcrev_filepath.as_posix(), "w", encoding="utf8") as fp: + fp.write(new_srcrev) + layer_repo.index.add([srcrev_file]) + + # Make an API request to create the gitlab.user object + gitlab.auth() + + # Construct commit message and commit the change + message = "Integrate %s/%s\n%s" % ( + project.path, + merge_request.source_branch, + common.list_commits(merge_request.commits()), + ) + layer_revision = common.commit_and_push( + layer_project, + layer_repo, + message, + gitlab.user.username, + gitlab.user.email, + ) + logging.debug("New revision in layer: %s", layer_revision) + + return layer_revision + + +def main(): + parser = argparse.ArgumentParser() + parser.add_argument( + "--gitlab-url", + help="""URL to the GitLab instance""", + dest="gitlab_url", + required=True, + ) + parser.add_argument( + "--token", + help="""GitLab REST API private access token for the layer group""", + dest="token", + required=True, + ) + parser.add_argument( + "--layer-project", + help="""name of the yocto layer project""", + dest="layer_project", + required=True, + ) + parser.add_argument( + "--layer-branch", + help="""yocto layer branch to branch off from""", + dest="layer_branch", + required=True, + ) + parser.add_argument( + "--srcrev-file", + help="""source revision file name (default: 'SRCREV.conf')""", + dest="srcrev_file", + default=common.srcrev_file, + required=False, + ) + parser.add_argument( + "--recipe-name", + help="""recipe name to resolve project in 'SRCREV.conf'""", + dest="recipe_name", + default=None, + required=False, + ) + parser.add_argument( + "--project", + help="""name of the project, as specified in the layer""", + dest="project", + required=True, + ) + parser.add_argument( + "--merge-request", + help="""project merge request IID or link containing the + changes to be integrated""", + dest="merge_request", + required=True, + ) + parser.add_argument( + "--save-revision-to", + help="""path to a file where the new layer revision is stored""", + dest="revision_file", + required=False, + ) + parser.add_argument( + "-v", + "--verbose", + action="store_true", + help="""Increase verbosity.""", + ) + + args, _ = parser.parse_known_args() + if args.verbose: + logging.basicConfig(level=logging.DEBUG) + + logging.debug(args) + gitlab = Gitlab(args.gitlab_url, private_token=args.token) + layer_project = common.get_project(gitlab, args.layer_project) + + project = common.get_project(gitlab, args.project) + logging.debug("Project: %s", project.name) + logging.debug("Merge Request: %s", args.merge_request) + + gitlab = Gitlab(args.gitlab_url, private_token=args.token) + + merge_request = common.get_merge_request(project, args.merge_request) + if merge_request is None: + sys.exit("ERROR: could not get %s %s" % (project.name, args.merge_request)) + + layer_revision = integrate_into_layer( + layer_project=layer_project, + layer_branch=args.layer_branch, + srcrev_file=args.srcrev_file, + recipe_name=args.recipe_name, + project=project, + merge_request=merge_request, + ) + + if args.revision_file: + with open(args.revision_file, "w", encoding="utf-8") as file: + file.write(layer_revision) + +if __name__ == "__main__": + main()