summaryrefslogtreecommitdiff
path: root/searx/version.py
blob: ac42834d978fc8df77d39ec3479510f6fa2da232 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
# SPDX-License-Identifier: AGPL-3.0-or-later
# lint: pylint
# pylint: disable=,missing-module-docstring,missing-class-docstring

import re
import os
import shlex
import subprocess
import logging

# fallback values
# if there is searx.version_frozen module, and it is not possible to get the git tag
VERSION_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_branch


def 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_version


try:
    from searx.version_frozen import VERSION_STRING, VERSION_TAG, GIT_URL, GIT_BRANCH
except 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.py

VERSION_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)