| #!/usr/bin/env python |
| |
| # This is a variant of the `workspace_status.py` script that in addition to |
| # plain `git describe` implements a few heuristics to arrive at more to the |
| # point stamps for directories. But due to the implemented heuristics, it will |
| # typically take longer to run (especially if you use lots of plugins that |
| # come without tags) and might slow down your development cycle when used |
| # as default. |
| # |
| # To use it, simply add |
| # |
| # --workspace_status_command="python ./tools/workspace_status_release.py" |
| # |
| # to your bazel command. So for example instead of |
| # |
| # bazel build release.war |
| # |
| # use |
| # |
| # bazel build --workspace_status_command="python ./tools/workspace_status_release.py" release.war |
| # |
| # Alternatively, you can add |
| # |
| # build --workspace_status_command="python ./tools/workspace_status_release.py" |
| # |
| # to `.bazelrc` in your home directory. |
| # |
| # If the script exits with non-zero code, it's considered as a failure |
| # and the output will be discarded. |
| |
| from __future__ import print_function |
| import os |
| import subprocess |
| import sys |
| import re |
| |
| ROOT = os.path.abspath(__file__) |
| while not os.path.exists(os.path.join(ROOT, 'WORKSPACE')): |
| ROOT = os.path.dirname(ROOT) |
| REVISION_CMD = ['git', 'describe', '--always', '--dirty'] |
| |
| |
| def run(command): |
| try: |
| return subprocess.check_output(command).strip().decode("utf-8") |
| except OSError as err: |
| print('could not invoke %s: %s' % (command[0], err), file=sys.stderr) |
| sys.exit(1) |
| except subprocess.CalledProcessError: |
| # ignore "not a git repository error" to report unknown version |
| return None |
| |
| |
| def revision_with_match(pattern=None, prefix=False, all_refs=False, |
| return_unmatched=False): |
| """Return a description of the current commit |
| |
| Keyword arguments: |
| pattern -- (Default: None) Use only refs that match this pattern. |
| prefix -- (Default: False) If True, the pattern is considered a prefix |
| and does not require an exact match. |
| all_refs -- (Default: False) If True, consider all refs, not just tags. |
| return_unmatched -- (Default: False) If False and a pattern is given that |
| cannot be matched, return the empty string. If True, return |
| the unmatched description nonetheless. |
| """ |
| |
| command = REVISION_CMD[:] |
| if pattern: |
| command += ['--match', pattern + ('*' if prefix else '')] |
| if all_refs: |
| command += ['--all', '--long'] |
| |
| description = run(command) |
| |
| if pattern and not return_unmatched and not description.startswith(pattern): |
| return '' |
| return description |
| |
| |
| def branch_with_match(pattern): |
| for ref_kind in ['origin/', 'gerrit/', '']: |
| description = revision_with_match(ref_kind + pattern, all_refs=True, |
| return_unmatched=True) |
| for cutoff in ['heads/', 'remotes/', ref_kind]: |
| if description.startswith(cutoff): |
| description = description[len(cutoff):] |
| if description.startswith(pattern): |
| return description |
| return '' |
| |
| |
| def revision(template=None): |
| if template: |
| # We use the version `v2.16.19-1-gec686a6352` as running example for the |
| # below comments. First, we split into ['v2', '16', '19'] |
| parts = template.split('-')[0].split('.') |
| |
| # Although we have releases with version tags containing 4 numbers, we |
| # treat only the first three numbers for simplicity. See discussion on |
| # Ib1681b2730cf2c443a3cb55fe6e282f6484e18de. |
| |
| if len(parts) >= 3: |
| # Match for v2.16.19 |
| version_part = '.'.join(parts[0:3]) |
| description = revision_with_match(version_part) |
| if description: |
| return description |
| |
| if len(parts) >= 2: |
| version_part = '.'.join(parts[0:2]) |
| |
| # Match for v2.16.* |
| description = revision_with_match(version_part + '.', prefix=True) |
| if description: |
| return description |
| |
| # Match for v2.16 |
| description = revision_with_match(version_part) |
| if description.startswith(version_part): |
| return description |
| |
| if template.startswith('v'): |
| # Match for stable-2.16 branches |
| branch = 'stable-' + version_part[1:] |
| description = branch_with_match(branch) |
| if description: |
| return description |
| |
| # None of the template based methods worked out, so we're falling back to |
| # generic matches. |
| |
| # Match for master branch |
| description = branch_with_match('master') |
| if description: |
| return description |
| |
| # Match for anything that looks like a version tag |
| description = revision_with_match('v[0-9].', return_unmatched=True) |
| if description.startswith('v'): |
| return description |
| |
| # Still no good tag, so we re-try without any matching |
| return revision_with_match() |
| |
| |
| # prints the stamps for the current working directory |
| def print_stamps_for_cwd(name, template): |
| workspace_status_script = os.path.join( |
| 'tools', 'workspace_status_release.py') |
| if os.path.isfile(workspace_status_script): |
| # directory has own workspace_status_command, so we use stamps from that |
| for line in run(["python", workspace_status_script]).split('\n'): |
| if re.search("^STABLE_[a-zA-Z0-9().:@/_ -]*$", line): |
| print(line) |
| else: |
| # directory lacks own workspace_status_command, so we create a default |
| # stamp |
| v = revision(template) |
| print('STABLE_BUILD_%s_LABEL %s' % (name.upper(), |
| v if v else 'unknown')) |
| |
| |
| # os.chdir is different from plain `cd` in shells in that it follows symlinks |
| # and does not update the PWD environment. So when using os.chdir to change into |
| # a symlinked directory from gerrit's `plugins` or `modules` directory, we |
| # cannot recover gerrit's directory. This prevents the plugins'/modules' |
| # `workspace_status_release.py` scripts to detect the name they were symlinked |
| # as (E.g.: it-* plugins sometimes get linked in more than once under different |
| # names) and to detect gerrit's root directory. To work around this problem, we |
| # mimic the `cd` of ordinary shells. By using this function, symlink information |
| # is preserved in the `PWD` environment variable (as it is for example also done |
| # in bash) and plugin/module `workspace_status_release.py` scripts can pick up |
| # the needed information from there. |
| def cd(absolute_path): |
| os.environ['PWD'] = absolute_path |
| os.chdir(absolute_path) |
| |
| |
| def print_stamps(): |
| cd(ROOT) |
| gerrit_version = revision() |
| print("STABLE_BUILD_GERRIT_LABEL %s" % gerrit_version) |
| for kind in ['modules', 'plugins']: |
| kind_dir = os.path.join(ROOT, kind) |
| for d in os.listdir(kind_dir) if os.path.isdir(kind_dir) else []: |
| p = os.path.join(kind_dir, d) |
| if os.path.isdir(p): |
| cd(p) |
| name = os.path.basename(p) |
| print_stamps_for_cwd(name, gerrit_version) |
| |
| |
| if __name__ == '__main__': |
| print_stamps() |