|  | package com.mirantis.mk | 
|  |  | 
|  | /** | 
|  | * | 
|  | * Git functions | 
|  | * | 
|  | */ | 
|  |  | 
|  | /** | 
|  | * Checkout single git repository | 
|  | * | 
|  | * @param path            Directory to checkout repository to | 
|  | * @param url             Source Git repository URL | 
|  | * @param branch          Source Git repository branch | 
|  | * @param credentialsId   Credentials ID to use for source Git | 
|  | * @param poll            Enable git polling (default true) | 
|  | * @param timeout         Set checkout timeout (default 10) | 
|  | * @param depth           Git depth param (default 0 means no depth) | 
|  | */ | 
|  | def checkoutGitRepository(path, url, branch, credentialsId = null, poll = true, timeout = 10, depth = 0){ | 
|  | dir(path) { | 
|  | checkout( | 
|  | changelog:true, | 
|  | poll: poll, | 
|  | scm: [ | 
|  | $class: 'GitSCM', | 
|  | branches: [[name: "*/${branch}"]], | 
|  | doGenerateSubmoduleConfigurations: false, | 
|  | extensions: [ | 
|  | [$class: 'CheckoutOption', timeout: timeout], | 
|  | [$class: 'CloneOption', depth: depth, noTags: false, reference: '', shallow: depth > 0, timeout: timeout]], | 
|  | submoduleCfg: [], | 
|  | userRemoteConfigs: [[url: url, credentialsId: credentialsId]]] | 
|  | ) | 
|  | sh(returnStdout: true, script: 'git rev-parse HEAD').trim() | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Parse HEAD of current directory and return commit hash | 
|  | */ | 
|  | def getGitCommit() { | 
|  | git_commit = sh ( | 
|  | script: 'git rev-parse HEAD', | 
|  | returnStdout: true | 
|  | ).trim() | 
|  | return git_commit | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Change actual working branch of repo | 
|  | * | 
|  | * @param path            Path to the git repository | 
|  | * @param branch          Branch desired to switch to | 
|  | */ | 
|  | def changeGitBranch(path, branch) { | 
|  | dir(path) { | 
|  | git_cmd = sh ( | 
|  | script: "git checkout ${branch}", | 
|  | returnStdout: true | 
|  | ).trim() | 
|  | } | 
|  | return git_cmd | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Get remote URL | 
|  | * | 
|  | * @param name  Name of remote (default any) | 
|  | * @param type  Type (fetch or push, default fetch) | 
|  | */ | 
|  | def getGitRemote(name = '', type = 'fetch') { | 
|  | gitRemote = sh ( | 
|  | script: "git remote -v | grep '${name}' | grep ${type} | awk '{print \$2}' | head -1", | 
|  | returnStdout: true | 
|  | ).trim() | 
|  | return gitRemote | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Create new working branch for repo | 
|  | * | 
|  | * @param path            Path to the git repository | 
|  | * @param branch          Branch desired to switch to | 
|  | */ | 
|  | def createGitBranch(path, branch) { | 
|  | def git_cmd | 
|  | dir(path) { | 
|  | git_cmd = sh ( | 
|  | script: "git checkout -b ${branch}", | 
|  | returnStdout: true | 
|  | ).trim() | 
|  | } | 
|  | return git_cmd | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Commit changes to the git repo | 
|  | * | 
|  | * @param path            Path to the git repository | 
|  | * @param message         A commit message | 
|  | * @param global          Use global config | 
|  | * @param amend           Whether to use "--amend" in commit command | 
|  | */ | 
|  | def commitGitChanges(path, message, gitEmail='jenkins@localhost', gitName='jenkins-slave', global=false, amend=false) { | 
|  | def git_cmd | 
|  | def gitOpts | 
|  | def global_arg = '' | 
|  | if (global) { | 
|  | global_arg = '--global' | 
|  | } | 
|  | if (amend) { | 
|  | gitOpts = '--amend' | 
|  | } else { | 
|  | gitOpts = '' | 
|  | } | 
|  | dir(path) { | 
|  | sh "git config ${global_arg} user.email '${gitEmail}'" | 
|  | sh "git config ${global_arg} user.name '${gitName}'" | 
|  |  | 
|  | sh( | 
|  | script: 'git add -A', | 
|  | returnStdout: true | 
|  | ).trim() | 
|  | git_cmd = sh( | 
|  | script: "git commit ${gitOpts} -m '${message}'", | 
|  | returnStdout: true | 
|  | ).trim() | 
|  | } | 
|  | return git_cmd | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Push git changes to remote repo | 
|  | * | 
|  | * @param path            Path to the local git repository | 
|  | * @param branch          Branch on the remote git repository | 
|  | * @param remote          Name of the remote repository | 
|  | * @param credentialsId   Credentials with write permissions | 
|  | */ | 
|  | def pushGitChanges(path, branch = 'master', remote = 'origin', credentialsId = null) { | 
|  | def ssh = new com.mirantis.mk.Ssh() | 
|  | dir(path) { | 
|  | if (credentialsId == null) { | 
|  | sh script: "git push ${remote} ${branch}" | 
|  | } | 
|  | else { | 
|  | ssh.prepareSshAgentKey(credentialsId) | 
|  | ssh.runSshAgentCommand("git push ${remote} ${branch}") | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Mirror git repository, merge target changes (downstream) on top of source | 
|  | * (upstream) and push target or both if pushSource is true | 
|  | * | 
|  | * @param sourceUrl      Source git repository | 
|  | * @param targetUrl      Target git repository | 
|  | * @param credentialsId  Credentials id to use for accessing source/target | 
|  | *                       repositories | 
|  | * @param branches       List or comma-separated string of branches to sync | 
|  | * @param followTags     Mirror tags | 
|  | * @param pushSource     Push back into source branch, resulting in 2-way sync | 
|  | * @param pushSourceTags Push target tags into source or skip pushing tags | 
|  | * @param gitEmail       Email for creation of merge commits | 
|  | * @param gitName        Name for creation of merge commits | 
|  | */ | 
|  | def mirrorGit(sourceUrl, targetUrl, credentialsId, branches, followTags = false, pushSource = false, pushSourceTags = false, gitEmail = 'jenkins@localhost', gitName = 'Jenkins') { | 
|  | def common = new com.mirantis.mk.Common() | 
|  | def ssh = new com.mirantis.mk.Ssh() | 
|  | if (branches instanceof String) { | 
|  | branches = branches.tokenize(',') | 
|  | } | 
|  |  | 
|  | ssh.prepareSshAgentKey(credentialsId) | 
|  | ssh.ensureKnownHosts(targetUrl) | 
|  | sh "git config user.email '${gitEmail}'" | 
|  | sh "git config user.name '${gitName}'" | 
|  |  | 
|  | def remoteExistence = sh(script: "git remote -v | grep ${TARGET_URL} | grep target", returnStatus: true) | 
|  | if(remoteExistence != 0){ | 
|  | // silently try to remove target | 
|  | sh(script:"git remote remove target", returnStatus: true) | 
|  | sh("git remote add target ${TARGET_URL}") | 
|  | } | 
|  | ssh.agentSh "git remote update --prune" | 
|  |  | 
|  | for (i=0; i < branches.size; i++) { | 
|  | branch = branches[i] | 
|  | sh "git branch | grep ${branch} || git checkout -b ${branch}" | 
|  | def resetResult = sh(script: "git checkout ${branch} && git reset --hard origin/${branch}", returnStatus: true) | 
|  | if(resetResult != 0){ | 
|  | common.warningMsg("Cannot reset to origin/${branch} for perform git mirror, trying to reset from target/${branch}") | 
|  | resetResult = sh(script: "git checkout ${branch} && git reset --hard target/${branch}", returnStatus: true) | 
|  | if(resetResult != 0){ | 
|  | throw new Exception("Cannot reset even to target/${branch}, git mirroring failed!") | 
|  | } | 
|  | } | 
|  |  | 
|  | sh "git ls-tree target/${branch} && git merge --no-edit --ff target/${branch} || echo 'Target repository is empty, skipping merge'" | 
|  | followTagsArg = followTags ? "--follow-tags" : "" | 
|  | ssh.agentSh "git push ${followTagsArg} target HEAD:${branch}" | 
|  |  | 
|  | if (pushSource == true) { | 
|  | followTagsArg = followTags && pushSourceTags ? "--follow-tags" : "" | 
|  | ssh.agentSh "git push ${followTagsArg} origin HEAD:${branch}" | 
|  | } | 
|  | } | 
|  | if (followTags == true) { | 
|  | ssh.agentSh "git push -f target --tags" | 
|  |  | 
|  | if (pushSourceTags == true) { | 
|  | ssh.agentSh "git push -f origin --tags" | 
|  | } | 
|  | } | 
|  | sh "git remote rm target" | 
|  | } | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Return all branches for the defined git repository that match the matcher. | 
|  | * | 
|  | * @param repoUrl        URL of git repository | 
|  | * @param branchMatcher  matcher to filter out the branches (If '' or '*', returns all branches without filtering) | 
|  | * @return branchesList  list of branches | 
|  | */ | 
|  |  | 
|  | def getBranchesForGitRepo(repoUrl, branchMatcher = ''){ | 
|  |  | 
|  | if (branchMatcher.equals("*")) { | 
|  | branchMatcher = '' | 
|  | } | 
|  | branchesList = sh ( | 
|  | script: "git ls-remote --heads ${repoUrl} | cut -f2 | grep -e '${branchMatcher}' | sed 's/refs\\/heads\\///g'", | 
|  | returnStdout: true | 
|  | ).trim() | 
|  | return branchesList.tokenize('\n') | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Method for preparing a tag to be SemVer 2 compatible, and can handle next cases: | 
|  | * - length of tag splitted by dots is more than 3 | 
|  | * - first part of splitted tag starts not from digit | 
|  | * - length of tag is lower than 3 | 
|  | * | 
|  | * @param   tag       String which contains a git tag from repository | 
|  | * @return  HashMap   HashMap in the form: ['version': 'x.x.x', 'extra': 'x.x.x'], extra | 
|  | *                    is added only if size of original tag splitted by dots is more than 3 | 
|  | */ | 
|  |  | 
|  | def prepareTag(tag){ | 
|  | def parts = tag.tokenize('.') | 
|  | def res = [:] | 
|  | // Handle case with tags like v1.1.1 | 
|  | parts[0] = parts[0].replaceFirst("[^\\d.]", '') | 
|  | // handle different sizes of tags - 1.1.1.1 or 1.1.1.1rc1 | 
|  | if (parts.size() > 3){ | 
|  | res['extra'] = parts[3..-1].join('.') | 
|  | } else if (parts.size() < 3){ | 
|  | (parts.size()..2).each { | 
|  | parts[it] = '0' | 
|  | } | 
|  | } | 
|  | res['version'] = "${parts[0]}.${parts[1]}.${parts[2]}" | 
|  | return res | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Method for incrementing SemVer 2 compatible version | 
|  | * | 
|  | * @param version  String which contains main part of SemVer2 version - '2.1.0' | 
|  | * @return string  String conaining version with Patch part of version incremented by 1 | 
|  | */ | 
|  |  | 
|  | def incrementVersion(version){ | 
|  | def parts = checkVersion(version) | 
|  | return "${parts[0]}.${parts[1]}.${parts[2].toInteger() + 1}" | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Method for checking whether version is compatible with Sem Ver 2 | 
|  | * | 
|  | * @param version  String which contains main part of SemVer2 version - '2.1.0' | 
|  | * @return list    With 3 strings as result of splitting version by dots | 
|  | */ | 
|  |  | 
|  | def checkVersion(version) { | 
|  | def parts = version.tokenize('.') | 
|  | if (parts.size() != 3 || !(parts[0] ==~ /^\d+/)) { | 
|  | error "Bad version ${version}" | 
|  | } | 
|  | return parts | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Method for constructing SemVer2 compatible version from tag in Git repository: | 
|  | * - if current commit matches the last tag, last tag will be returned as version | 
|  | * - if no tag found assuming no release was done, version will be 0.0.1 with pre release metadata | 
|  | * - if tag found - patch part of version will be incremented and pre-release metadata will be added | 
|  | * | 
|  | * | 
|  | * @param repoDir          String which contains path to directory with git repository | 
|  | * @param allowNonSemVer2  Bool   whether to allow working with tags which aren't compatible | 
|  | *                                with Sem Ver 2 (not in form X.Y.Z). if set to true tag will be | 
|  | *                                 converted to Sem Ver 2 version e.g tag 1.1.1.1rc1 -> version 1.1.1-1rc1 | 
|  | * @return version  String | 
|  | */ | 
|  | def getVersion(repoDir, allowNonSemVer2 = false) { | 
|  | def common = new com.mirantis.mk.Common() | 
|  | dir(repoDir){ | 
|  | def cmd = common.shCmdStatus('git describe --tags --first-parent --abbrev=0') | 
|  | def tag_data = [:] | 
|  | def last_tag = cmd['stdout'].trim() | 
|  | def commits_since_tag | 
|  | if (cmd['status'] != 0){ | 
|  | if (cmd['stderr'].contains('fatal: No names found, cannot describe anything')){ | 
|  | common.warningMsg('No parent tag found, using initial version 0.0.0') | 
|  | tag_data['version'] = '0.0.0' | 
|  | commits_since_tag = sh(script: 'git rev-list --count HEAD', returnStdout: true).trim() | 
|  | } else { | 
|  | error("Something went wrong, cannot find git information ${cmd['stderr']}") | 
|  | } | 
|  | } else { | 
|  | tag_data['version'] = last_tag | 
|  | commits_since_tag = sh(script: "git rev-list --count ${last_tag}..HEAD", returnStdout: true).trim() | 
|  | } | 
|  | try { | 
|  | checkVersion(tag_data['version']) | 
|  | } catch (Exception e) { | 
|  | if (allowNonSemVer2){ | 
|  | common.errorMsg( | 
|  | """Git tag isn't compatible with SemVer2, but allowNonSemVer2 is set. | 
|  | Trying to convert git tag to Sem Ver 2 compatible version | 
|  | ${e.message}""") | 
|  | tag_data = prepareTag(tag_data['version']) | 
|  | } else { | 
|  | error("Git tag isn't compatible with SemVer2\n${e.message}") | 
|  | } | 
|  | } | 
|  | // If current commit is exact match to the first parent tag than return it | 
|  | def pre_release_meta = [] | 
|  | if (tag_data.get('extra')){ | 
|  | pre_release_meta.add(tag_data['extra']) | 
|  | } | 
|  | if (common.shCmdStatus('git describe --tags --first-parent --exact-match')['status'] == 0){ | 
|  | if (pre_release_meta){ | 
|  | return "${tag_data['version']}-${pre_release_meta[0]}" | 
|  | } else { | 
|  | return tag_data['version'] | 
|  | } | 
|  | } | 
|  | // If we away from last tag for some number of commits - add additional metadata and increment version | 
|  | pre_release_meta.add(commits_since_tag) | 
|  | def next_version = incrementVersion(tag_data['version']) | 
|  | def commit_sha = sh(script: 'git rev-parse --short=7 HEAD', returnStdout: true).trim() | 
|  | return "${next_version}-${pre_release_meta.join('.')}-${commit_sha}" | 
|  | } | 
|  | } |