diff --git a/common.py b/common.py index 7fbc86f57c646dea203c3d3549324bb6588b23e4..f93f5e1e95f661db8ab81417b0113236b75b2280 100755 --- a/common.py +++ b/common.py @@ -254,7 +254,7 @@ def get_repository_file_obj(project: Project, filename, ref=None): repository_tree = project.repository_tree( ref=ref, all=True, retry_transient_errors=True ) - logging.debug(repository_tree) + # logging.debug(repository_tree) fileobj = [f for f in repository_tree if f["name"] == filename] if len(fileobj) == 0: diff --git a/deploy_gitlab_ci.py b/deploy_gitlab_ci.py index f2caf7247fd4ab4c14fa2c18f70978185fbb1876..75938a540326d2ffd2f320e53e58b57398a505c4 100755 --- a/deploy_gitlab_ci.py +++ b/deploy_gitlab_ci.py @@ -46,7 +46,7 @@ def integrate_submodule_into( new_revision, branch, commit_and_push=True, - force_replace_of_existing_branch=False, + force_clone=False, ): gitlab_project = common.get_project(gitlab, project_name) @@ -62,10 +62,8 @@ def integrate_submodule_into( new_revision, branch, commit_and_push=commit_and_push, - force_replace_of_existing_branch=force_replace_of_existing_branch, + force_clone=force_clone, ) - if integration_branch_name is None: - return None # ====================================== # Store the references for creating the integration # commit in the manifest later @@ -192,7 +190,9 @@ def main(): res = integrate_submodule_into( gitlab, p, args.submodule, args.revision, args.branch ) - if res is not None: + # Store in the list if commit is set (meaning there was an update or + # an exising integration branch) + if res["commit"] is not None: project_integration[p] = res # Update submodule in manifest project @@ -203,27 +203,19 @@ def main(): args.revision, args.branch, commit_and_push=False, - # We need the checkout and integration - # branch in the manifest commit - # TODO develop a way to allow amend to - # this branch also - force_replace_of_existing_branch=True, + force_clone=True, ) - if manifest_project is not None: - project_integration[args.project] = manifest_project - - # If submodule is already at specified revision in all projects, exit successfully - if len(project_integration) == 0: - print("No integration done, changes are already included in all projects.") - sys.exit(0) + branch = args.branch + if branch is None: + branch = manifest_project["project"].default_branch # ======================================================= # Create and merge merge_requests if needed # ======================================================= if args.merge: # Get source merge request ( the one in the gitlab-ci repo) submodule_project_path, _ = get_submodule_project_path_and_revision( - manifest_project["project"], args.submodule, args.branch + manifest_project["project"], args.submodule, branch ) submodule_project = common.get_project(gitlab, submodule_project_path) mrs = get_merge_requests( @@ -330,6 +322,13 @@ def main(): manifest_project["repo"].git.add(args.srcrev_file) logging.debug(srcrev) + # ======================================================== + # Squash all commits on the integration branch to one + # ======================================================== + manifest_project["repo"].remotes.origin.fetch(branch) + manifest_master = manifest_project["project"].branches.get(branch) + manifest_project["repo"].git.reset("--soft", manifest_master.commit["id"]) + # ======================================================== # Now commit and push the changes to the manifest repo # ======================================================== diff --git a/update_submodule.py b/update_submodule.py index 6dbf07924b0dc97c226d1c0150117011883b2aa1..c5a422f1bf3e89d6b2319e6ef38fd7ed287a5b3d 100755 --- a/update_submodule.py +++ b/update_submodule.py @@ -121,7 +121,13 @@ def clone_project_and_submodule(project: Project, submodule_name, branch=None): submodule_relative_url = submodule.url with submodule.config_writer() as writer: writer.set("url", submodule_clone_url.url) - submodule.update(init=True) + try: + submodule.update(init=True) + except GitCommandError: + # This seems to happen when a not existing commit is referenced + logging.error("Failed to initialize submodule %s", submodule_name) + pass + with submodule.config_writer() as writer: writer.set("url", submodule_relative_url) @@ -181,7 +187,7 @@ def update_submodule_and_include_ref( new_revision, branch=None, commit_and_push=True, - force_replace_of_existing_branch=False, + force_clone=False, ): """Update the submodule and include refs to the submodule in the given project. Create mergerequest if needed. @@ -192,6 +198,7 @@ def update_submodule_and_include_ref( new_revision (hex string): The sha hash of the commit to update the submodule to branch (string): branch to update, if None, the projects default branch is used commit_and_push: Set to false if no commit should be created. Changes are left in staging. + force_clone: Checkout repo and setup integration branch even if no update is needed Returns: tuple of: project_repo (Repo): GitPython repo with the cloned project @@ -201,6 +208,10 @@ def update_submodule_and_include_ref( """ gitlab = project.manager.gitlab + submodule_update_needed = True + project_repo = None + integration_commit = None + if branch is None: branch = project.default_branch logging.debug("Branch: %s", branch) @@ -210,11 +221,6 @@ def update_submodule_and_include_ref( submodule_current_rev, ) = get_submodule_project_path_and_revision(project, submodule_name, branch) - # Check if revisions are different - if submodule_current_rev == new_revision and not force_replace_of_existing_branch: - print("Submodule is already at %s" % new_revision) - return (None, None, None, None) - # Get submodule project submodule_project = common.get_project(gitlab, submodule_project_path) @@ -250,6 +256,11 @@ def update_submodule_and_include_ref( common.list_commits(commits), ) + # Check if revisions are different + if submodule_current_rev == new_revision: + print("Submodule is already at %s" % new_revision) + submodule_update_needed = False + # Check if we already have an integration branch (before we actually do the checkout) # Check if integration branch already exists and if it is up to date # This is needed for one use case: @@ -263,8 +274,8 @@ def update_submodule_and_include_ref( # To get this change 'atomic' the updated include statement should be # in the same commit as the update of the submodule - existing_branch = None - if not force_replace_of_existing_branch: + if submodule_update_needed or force_clone: + existing_branch = None try: existing_branch = project.branches.get(integration_branch_name) except GitlabGetError: @@ -290,62 +301,76 @@ def update_submodule_and_include_ref( % (new_revision, integration_branch_name) ) integration_commit = existing_branch.commit["id"] - return None, integration_branch_name, integration_commit, message + submodule_update_needed = False # Clone the project, we need to do changes - project_repo, submodule_project = clone_project_and_submodule( - project, submodule_name, branch - ) + if submodule_update_needed or force_clone: + clone_branch = branch + if existing_branch: + clone_branch = integration_branch_name - if existing_branch: - print("Replacing outdated integration branch %s" % integration_branch_name) - else: - print("Creating integration branch %s" % integration_branch_name) + # Actually clone + project_repo, submodule_project = clone_project_and_submodule( + project, submodule_name, clone_branch + ) - # Create branch - project_repo.head.set_reference(project_repo.create_head(integration_branch_name)) + if existing_branch: + print("Using existing integration branch %s" % integration_branch_name) + else: + # Create branch + print("Creating integration branch %s" % integration_branch_name) + project_repo.head.set_reference( + project_repo.create_head(integration_branch_name) + ) - # Update submodule to new revision - submodule_repo = common.get_submodule(project_repo, submodule_name) - update_submodule_in_repo(project_repo, submodule_repo, new_revision) + if submodule_update_needed: + # Update submodule to new revision + submodule_repo = common.get_submodule(project_repo, submodule_name) + update_submodule_in_repo(project_repo, submodule_repo, new_revision) - # Update the gitlab-ci.yml file to the new revision - # Now also update the project '.gitlab-ci.yml' file - gitlab_ci_yml_filename = os.path.join( - project_repo.working_tree_dir, ".gitlab-ci.yml" - ) - with open(gitlab_ci_yml_filename, "r", encoding="utf8") as fp: - gitlab_ci_yml = fp.read() - logging.debug(gitlab_ci_yml) - - new_gitlab_ci_yml = update_gitlab_ci_include( - gitlab_ci_yml, - submodule_project.web_url.split("//")[1].split("/", 1)[1], - new_revision, - ) - if new_gitlab_ci_yml is None: - print("Failed to update the include revision in '.gitlab-ci.yml'") - else: - logging.debug(new_gitlab_ci_yml) - with open(gitlab_ci_yml_filename, "w", encoding="utf8") as fp: - fp.write(new_gitlab_ci_yml) - project_repo.git.add(os.path.basename(gitlab_ci_yml_filename)) - - # Commit the changes - if commit_and_push: - # Make an API request to create the gitlab.user object - gitlab.auth() - integration_commit = common.commit_and_push( - project, - project_repo, - integration_branch_name, - message, - gitlab.user.username, - gitlab.user.email, - less_verbose=True, + # Update the gitlab-ci.yml file to the new revision + # Now also update the project '.gitlab-ci.yml' file + gitlab_ci_yml_filename = os.path.join( + project_repo.working_tree_dir, ".gitlab-ci.yml" ) - else: - integration_commit = None + with open(gitlab_ci_yml_filename, "r", encoding="utf8") as fp: + gitlab_ci_yml = fp.read() + logging.debug(gitlab_ci_yml) + + new_gitlab_ci_yml = update_gitlab_ci_include( + gitlab_ci_yml, + submodule_project.web_url.split("//")[1].split("/", 1)[1], + new_revision, + ) + if new_gitlab_ci_yml is None: + print("Failed to update the include revision in '.gitlab-ci.yml'") + else: + logging.debug(new_gitlab_ci_yml) + with open(gitlab_ci_yml_filename, "w", encoding="utf8") as fp: + fp.write(new_gitlab_ci_yml) + project_repo.git.add(os.path.basename(gitlab_ci_yml_filename)) + + # Commit the changes + if commit_and_push: + # ======================================================== + # Squash all commits on the integration branch to one + # ======================================================== + project_repo.remotes.origin.fetch(branch) + gitlab_branch = project.branches.get(branch) + project_repo.git.reset("--soft", gitlab_branch.commit["id"]) + + # Make an API request to create the gitlab.user object + gitlab.auth() + # Push the changes + integration_commit = common.commit_and_push( + project, + project_repo, + integration_branch_name, + message, + gitlab.user.username, + gitlab.user.email, + less_verbose=True, + ) return project_repo, integration_branch_name, integration_commit, message