| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142 | # SPDX-License-Identifier: AGPL-3.0-or-later# lint: pylint# pylint: disable=,missing-module-docstring,missing-class-docstringimport reimport osimport shleximport subprocessimport logging# fallback values# if there is searx.version_frozen module, and it is not possible to get the git tagVERSION_STRING = "1.0.0"VERSION_TAG = "1.0.0"GIT_URL = "unknow"GIT_BRANCH = "unknow"logger = logging.getLogger("searx")SUBPROCESS_RUN_ENV = {    "PATH": os.environ["PATH"],    "LC_ALL": "C",    "LANGUAGE": "",}def subprocess_run(args, **kwargs):    """Call :py:func:`subprocess.run` and return (striped) stdout.  If returncode is    non-zero, raise a :py:func:`subprocess.CalledProcessError`.    """    if not isinstance(args, (list, tuple)):        args = shlex.split(args)    kwargs["env"] = kwargs.get("env", SUBPROCESS_RUN_ENV)    kwargs["encoding"] = kwargs.get("encoding", "utf-8")    kwargs["stdout"] = subprocess.PIPE    kwargs["stderr"] = subprocess.PIPE    # raise CalledProcessError if returncode is non-zero    kwargs["check"] = True    proc = subprocess.run(args, **kwargs)  # pylint: disable=subprocess-run-check    return proc.stdout.strip()def get_git_url_and_branch():    try:        ref = subprocess_run("git rev-parse --abbrev-ref @{upstream}")    except subprocess.CalledProcessError:        ref = subprocess_run("git rev-parse --abbrev-ref master@{upstream}")    origin, git_branch = ref.split("/", 1)    git_url = subprocess_run(["git", "remote", "get-url", origin])    # get https:// url from git@ url    if git_url.startswith("git@"):        git_url = git_url.replace(":", "/", 2).replace("git@", "https://", 1)    if git_url.endswith(".git"):        git_url = git_url.replace(".git", "", 1)    return git_url, git_branchdef get_git_version():    try:        tag = subprocess_run("git describe HEAD")        # a. HEAD is on tag name, example: tag = "v1.0.1"        # b. HEAD is not a tag name, example "<tag>-<distance>-g<commit>"        tag_version, tag_distance, tag_commit = (tag.split("-") + ["", ""])[:3]        if re.match(r"v[0-9]+\.[0-9]+\.[0-9]+", tag_version):            # tag_version "v1.0.0" becomes "1.0.0" (without the v)            # other patterns are kept untouched            tag_version = tag_version[1:]        # remove "g" prefix from tag_commit        if tag_commit and tag_commit[0] == "g":            tag_commit = tag_commit[1:]        # set git_version to "1.0.0-590-0686e274" or '1.0.0'        git_version = "-".join(filter(bool, [tag_version, tag_distance, tag_commit]))    except subprocess.CalledProcessError:        # fall back to "YYYY.MM.DD.Hash" if there is no tag at all        git_version = subprocess_run(r"git show -s --format='%as-%h'")        # PEP 440: replace - with .        tag_version = git_version = git_version.replace("-", ".")    # add "-dirty" suffix if there are uncommited changes except searx/settings.yml    try:        subprocess_run(            "git diff --quiet -- . ':!searx/settings.yml' ':!utils/brand.env'"        )    except subprocess.CalledProcessError as e:        if e.returncode == 1:            git_version += "-dirty"        else:            logger.warning(                '"%s" returns an unexpected return code %i', e.returncode, e.cmd            )    return git_version, tag_versiontry:    from searx.version_frozen import VERSION_STRING, VERSION_TAG, GIT_URL, GIT_BRANCHexcept ImportError:    try:        try:            VERSION_STRING, VERSION_TAG = get_git_version()        except subprocess.CalledProcessError as ex:            logger.error("Error while getting the version: %s", ex.stderr)        try:            GIT_URL, GIT_BRANCH = get_git_url_and_branch()        except subprocess.CalledProcessError as ex:            logger.error("Error while getting the git URL & branch: %s", ex.stderr)    except FileNotFoundError as ex:        logger.error("%s is not found, fallback to the default version", ex.filename)logger.info("version: %s", VERSION_STRING)if __name__ == "__main__":    import sys    if len(sys.argv) >= 2 and sys.argv[1] == "freeze":        # freeze the version (to create an archive outside a git repository)        python_code = f"""# SPDX-License-Identifier: AGPL-3.0-or-later# this file is generated automatically by searx/version.pyVERSION_STRING = "{VERSION_STRING}"VERSION_TAG = "{VERSION_TAG}"GIT_URL = "{GIT_URL}"GIT_BRANCH = "{GIT_BRANCH}""""        with open(                os.path.join(os.path.dirname(__file__), "version_frozen.py"),                "w", encoding="utf8") as f:            f.write(python_code)            print(f"{f.name} created")    else:        # output shell code to set the variables        # usage: eval "$(python -m searx.version)"        shell_code = f"""VERSION_STRING="{VERSION_STRING}"VERSION_TAG="{VERSION_TAG}"GIT_URL="{GIT_URL}"GIT_BRANCH="{GIT_BRANCH}""""        print(shell_code)
 |