Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 1 | #!/bin/bash |
| 2 | # |
| 3 | # **inc/python** - Python-related functions |
| 4 | # |
| 5 | # Support for pip/setuptools interfaces and virtual environments |
| 6 | # |
| 7 | # External functions used: |
| 8 | # - GetOSVersion |
| 9 | # - is_fedora |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 10 | # - safe_chown |
| 11 | |
| 12 | # Save trace setting |
| 13 | INC_PY_TRACE=$(set +o | grep xtrace) |
| 14 | set +o xtrace |
| 15 | |
| 16 | |
Dean Troyer | 8c2ce6e | 2015-02-18 14:47:54 -0600 | [diff] [blame] | 17 | # Global Config Variables |
| 18 | |
Atsushi SAKAI | 5509ed5 | 2015-11-30 20:20:21 +0900 | [diff] [blame] | 19 | # PROJECT_VENV contains the name of the virtual environment for each |
Dean Troyer | 8c2ce6e | 2015-02-18 14:47:54 -0600 | [diff] [blame] | 20 | # project. A null value installs to the system Python directories. |
Sean Dague | afef8bf | 2017-03-06 14:07:23 -0500 | [diff] [blame] | 21 | declare -A -g PROJECT_VENV |
Dean Troyer | 8c2ce6e | 2015-02-18 14:47:54 -0600 | [diff] [blame] | 22 | |
Radosław Piliszek | be26306 | 2020-03-30 09:56:53 +0200 | [diff] [blame] | 23 | # Utility Functions |
| 24 | # ================= |
| 25 | |
| 26 | # Joins bash array of extras with commas as expected by other functions |
| 27 | function join_extras { |
| 28 | local IFS="," |
| 29 | echo "$*" |
| 30 | } |
Dean Troyer | 8c2ce6e | 2015-02-18 14:47:54 -0600 | [diff] [blame] | 31 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 32 | # Python Functions |
| 33 | # ================ |
| 34 | |
Clark Boylan | a40f9cb | 2018-04-04 14:02:30 -0700 | [diff] [blame] | 35 | # Setup the global devstack virtualenvs and the associated environment |
| 36 | # updates. |
| 37 | function setup_devstack_virtualenv { |
| 38 | # We run devstack out of a global virtualenv. |
| 39 | if [[ ! -d $DEVSTACK_VENV ]] ; then |
| 40 | # Using system site packages to enable nova to use libguestfs. |
| 41 | # This package is currently installed via the distro and not |
| 42 | # available on pypi. |
| 43 | python$PYTHON3_VERSION -m venv --system-site-packages $DEVSTACK_VENV |
| 44 | pip_install -U pip |
| 45 | fi |
| 46 | if [[ ":$PATH:" != *":$DEVSTACK_VENV/bin:"* ]] ; then |
| 47 | export PATH="$DEVSTACK_VENV/bin:$PATH" |
| 48 | export PYTHON="$DEVSTACK_VENV/bin/python3" |
| 49 | fi |
| 50 | } |
| 51 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 52 | # Get the path to the pip command. |
| 53 | # get_pip_command |
| 54 | function get_pip_command { |
Doug Hellmann | ddc3839 | 2015-05-07 21:06:24 +0000 | [diff] [blame] | 55 | local version="$1" |
Tom Barron | 4db9d56 | 2019-01-09 08:43:52 -0500 | [diff] [blame] | 56 | if [ -z "$version" ]; then |
| 57 | die $LINENO "pip python version is not set." |
| 58 | fi |
| 59 | |
Doug Hellmann | ddc3839 | 2015-05-07 21:06:24 +0000 | [diff] [blame] | 60 | # NOTE(dhellmann): I don't know if we actually get a pip3.4-python |
| 61 | # under any circumstances. |
| 62 | which pip${version} || which pip${version}-python |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 63 | |
| 64 | if [ $? -ne 0 ]; then |
Doug Hellmann | ddc3839 | 2015-05-07 21:06:24 +0000 | [diff] [blame] | 65 | die $LINENO "Unable to find pip${version}; cannot continue" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 66 | fi |
| 67 | } |
| 68 | |
Atsushi SAKAI | 5509ed5 | 2015-11-30 20:20:21 +0900 | [diff] [blame] | 69 | # Get the path to the directory where python executables are installed. |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 70 | # get_python_exec_prefix |
| 71 | function get_python_exec_prefix { |
Ian Wienand | 433a9b1 | 2015-10-07 13:29:31 +1100 | [diff] [blame] | 72 | local xtrace |
| 73 | xtrace=$(set +o | grep xtrace) |
Dean Troyer | 2b56476 | 2015-02-11 17:01:02 -0600 | [diff] [blame] | 74 | set +o xtrace |
| 75 | if [[ -z "$os_PACKAGE" ]]; then |
| 76 | GetOSVersion |
| 77 | fi |
| 78 | $xtrace |
| 79 | |
Clark Boylan | a40f9cb | 2018-04-04 14:02:30 -0700 | [diff] [blame] | 80 | if [[ "$GLOBAL_VENV" == "True" ]] ; then |
| 81 | echo "$DEVSTACK_VENV/bin" |
| 82 | else |
| 83 | echo "/usr/local/bin" |
| 84 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 85 | } |
| 86 | |
Sean Dague | 60996b1 | 2015-04-08 09:06:49 -0400 | [diff] [blame] | 87 | # Wrapper for ``pip install`` that only installs versions of libraries |
| 88 | # from the global-requirements specification. |
| 89 | # |
| 90 | # Uses globals ``REQUIREMENTS_DIR`` |
| 91 | # |
| 92 | # pip_install_gr packagename |
| 93 | function pip_install_gr { |
| 94 | local name=$1 |
Ian Wienand | ada886d | 2015-10-07 14:06:26 +1100 | [diff] [blame] | 95 | local clean_name |
| 96 | clean_name=$(get_from_global_requirements $name) |
Sean Dague | 60996b1 | 2015-04-08 09:06:49 -0400 | [diff] [blame] | 97 | pip_install $clean_name |
| 98 | } |
| 99 | |
Mehdi Abaakouk | 52b1074 | 2016-12-01 16:11:17 +0100 | [diff] [blame] | 100 | # Wrapper for ``pip install`` that only installs versions of libraries |
| 101 | # from the global-requirements specification with extras. |
| 102 | # |
| 103 | # Uses globals ``REQUIREMENTS_DIR`` |
| 104 | # |
| 105 | # pip_install_gr_extras packagename extra1,extra2,... |
| 106 | function pip_install_gr_extras { |
| 107 | local name=$1 |
| 108 | local extras=$2 |
Radosław Piliszek | be26306 | 2020-03-30 09:56:53 +0200 | [diff] [blame] | 109 | local version_constraints |
| 110 | version_constraints=$(get_version_constraints_from_global_requirements $name) |
| 111 | pip_install $name[$extras]$version_constraints |
Mehdi Abaakouk | 52b1074 | 2016-12-01 16:11:17 +0100 | [diff] [blame] | 112 | } |
| 113 | |
Doug Hellmann | 36377f6 | 2018-12-04 11:33:03 -0500 | [diff] [blame] | 114 | # enable_python3_package() -- no-op for backwards compatibility |
Doug Hellmann | 94129c7 | 2017-01-09 21:24:24 +0000 | [diff] [blame] | 115 | # |
Doug Hellmann | 94129c7 | 2017-01-09 21:24:24 +0000 | [diff] [blame] | 116 | # enable_python3_package dir [dir ...] |
| 117 | function enable_python3_package { |
| 118 | local xtrace |
| 119 | xtrace=$(set +o | grep xtrace) |
| 120 | set +o xtrace |
| 121 | |
Doug Hellmann | 36377f6 | 2018-12-04 11:33:03 -0500 | [diff] [blame] | 122 | echo "It is no longer necessary to call enable_python3_package()." |
Doug Hellmann | 94129c7 | 2017-01-09 21:24:24 +0000 | [diff] [blame] | 123 | |
| 124 | $xtrace |
| 125 | } |
| 126 | |
Stephen Finucane | 6b6bdc7 | 2019-10-09 16:13:20 +0100 | [diff] [blame] | 127 | # disable_python3_package() -- no-op for backwards compatibility |
Doug Hellmann | 94129c7 | 2017-01-09 21:24:24 +0000 | [diff] [blame] | 128 | # |
Doug Hellmann | 94129c7 | 2017-01-09 21:24:24 +0000 | [diff] [blame] | 129 | # disable_python3_package dir [dir ...] |
| 130 | function disable_python3_package { |
| 131 | local xtrace |
| 132 | xtrace=$(set +o | grep xtrace) |
| 133 | set +o xtrace |
| 134 | |
Stephen Finucane | 6b6bdc7 | 2019-10-09 16:13:20 +0100 | [diff] [blame] | 135 | echo "It is no longer possible to call disable_python3_package()." |
Doug Hellmann | 94129c7 | 2017-01-09 21:24:24 +0000 | [diff] [blame] | 136 | |
| 137 | $xtrace |
| 138 | } |
| 139 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 140 | # Wrapper for ``pip install`` to set cache and proxy environment variables |
Dean Troyer | 41d6f85 | 2015-03-25 22:42:46 -0500 | [diff] [blame] | 141 | # Uses globals ``OFFLINE``, ``PIP_VIRTUAL_ENV``, |
Ian Wienand | bcb2c30 | 2020-01-13 16:31:20 +1100 | [diff] [blame] | 142 | # ``PIP_UPGRADE``, ``*_proxy``, |
Zane Bitter | 9e7ead9 | 2017-10-05 16:51:09 -0400 | [diff] [blame] | 143 | # Usage: |
| 144 | # pip_install pip_arguments |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 145 | function pip_install { |
Federico Ressi | e208d06 | 2015-11-21 11:15:39 +0000 | [diff] [blame] | 146 | local xtrace result |
Ian Wienand | 433a9b1 | 2015-10-07 13:29:31 +1100 | [diff] [blame] | 147 | xtrace=$(set +o | grep xtrace) |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 148 | set +o xtrace |
Chris Dent | ebdd9ac | 2015-03-04 12:35:14 +0000 | [diff] [blame] | 149 | local upgrade="" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 150 | local offline=${OFFLINE:-False} |
| 151 | if [[ "$offline" == "True" || -z "$@" ]]; then |
| 152 | $xtrace |
| 153 | return |
| 154 | fi |
| 155 | |
Sean Dague | cb658fa | 2015-10-08 17:12:03 -0400 | [diff] [blame] | 156 | time_start "pip_install" |
| 157 | |
Chris Dent | ebdd9ac | 2015-03-04 12:35:14 +0000 | [diff] [blame] | 158 | PIP_UPGRADE=$(trueorfalse False PIP_UPGRADE) |
| 159 | if [[ "$PIP_UPGRADE" = "True" ]] ; then |
| 160 | upgrade="--upgrade" |
| 161 | fi |
| 162 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 163 | if [[ -z "$os_PACKAGE" ]]; then |
| 164 | GetOSVersion |
| 165 | fi |
Zane Bitter | 9e7ead9 | 2017-10-05 16:51:09 -0400 | [diff] [blame] | 166 | |
| 167 | # Try to extract the path of the package we are installing into |
| 168 | # package_dir. We need this to check for test-requirements.txt, |
| 169 | # at least. |
| 170 | # |
| 171 | # ${!#} expands to the last positional argument to this function. |
| 172 | # With "extras" syntax included, our arguments might be something |
| 173 | # like: |
| 174 | # -e /path/to/fooproject[extra] |
| 175 | # Thus this magic line grabs just the path without extras |
| 176 | # |
| 177 | # Note that this makes no sense if this is a pypi (rather than |
| 178 | # local path) install; ergo you must check this path exists before |
| 179 | # use. Also, if we had multiple or mixed installs, we would also |
| 180 | # likely break. But for historical reasons, it's basically only |
| 181 | # the other wrapper functions in here calling this to install |
| 182 | # local packages, and they do so with single call per install. So |
| 183 | # this works (for now...) |
| 184 | local package_dir=${!#%\[*\]} |
| 185 | |
Ian Wienand | bcb2c30 | 2020-01-13 16:31:20 +1100 | [diff] [blame] | 186 | if [[ -n ${PIP_VIRTUAL_ENV:=} && -d ${PIP_VIRTUAL_ENV} ]]; then |
| 187 | local cmd_pip=$PIP_VIRTUAL_ENV/bin/pip |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 188 | local sudo_pip="env" |
Clark Boylan | a40f9cb | 2018-04-04 14:02:30 -0700 | [diff] [blame] | 189 | elif [[ "${GLOBAL_VENV}" == "True" && -d ${DEVSTACK_VENV} ]] ; then |
| 190 | # We have to check that the DEVSTACK_VENV exists because early |
| 191 | # devstack boostrapping needs to operate in a system context |
| 192 | # too bootstrap pip. Once pip is bootstrapped we create the |
| 193 | # global venv and can start to use it. |
| 194 | local cmd_pip=$DEVSTACK_VENV/bin/pip |
| 195 | local sudo_pip="env" |
| 196 | echo "Using python $PYTHON3_VERSION to install $package_dir" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 197 | else |
Jens Harbott | d7a82f4 | 2020-06-23 10:21:09 +0200 | [diff] [blame] | 198 | local cmd_pip="python$PYTHON3_VERSION -m pip" |
Ian Wienand | 18b4251 | 2020-08-31 16:22:57 +1000 | [diff] [blame] | 199 | # See |
| 200 | # https://github.com/pypa/setuptools/issues/2232 |
| 201 | # http://lists.openstack.org/pipermail/openstack-discuss/2020-August/016905.html |
| 202 | # this makes setuptools >=50 use the platform distutils. |
| 203 | # We only want to do this on global pip installs, not if |
| 204 | # installing in a virtualenv |
| 205 | local sudo_pip="sudo -H LC_ALL=en_US.UTF-8 SETUPTOOLS_USE_DISTUTILS=stdlib " |
Jens Harbott | d7a82f4 | 2020-06-23 10:21:09 +0200 | [diff] [blame] | 206 | echo "Using python $PYTHON3_VERSION to install $package_dir" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 207 | fi |
| 208 | |
Robert Collins | 635a5ba | 2015-06-10 08:48:06 +1200 | [diff] [blame] | 209 | cmd_pip="$cmd_pip install" |
Clark Boylan | 05aa384 | 2015-08-03 11:14:13 -0700 | [diff] [blame] | 210 | # Always apply constraints |
| 211 | cmd_pip="$cmd_pip -c $REQUIREMENTS_DIR/upper-constraints.txt" |
Robert Collins | 635a5ba | 2015-06-10 08:48:06 +1200 | [diff] [blame] | 212 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 213 | $xtrace |
Clark Boylan | f266a2d | 2017-06-12 14:57:59 -0700 | [diff] [blame] | 214 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 215 | $sudo_pip \ |
Eli Qiao | 6a83c42 | 2015-03-17 16:54:16 +0800 | [diff] [blame] | 216 | http_proxy="${http_proxy:-}" \ |
| 217 | https_proxy="${https_proxy:-}" \ |
| 218 | no_proxy="${no_proxy:-}" \ |
Joe Gordon | cd8824a | 2015-03-04 16:40:19 -0800 | [diff] [blame] | 219 | PIP_FIND_LINKS=$PIP_FIND_LINKS \ |
Monty Taylor | 09b5b05 | 2020-03-27 11:22:39 -0500 | [diff] [blame] | 220 | $cmd_pip $upgrade \ |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 221 | $@ |
Federico Ressi | e208d06 | 2015-11-21 11:15:39 +0000 | [diff] [blame] | 222 | result=$? |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 223 | |
Sean Dague | cb658fa | 2015-10-08 17:12:03 -0400 | [diff] [blame] | 224 | time_stop "pip_install" |
Federico Ressi | e208d06 | 2015-11-21 11:15:39 +0000 | [diff] [blame] | 225 | return $result |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 226 | } |
| 227 | |
Sean Dague | f28e7ef | 2017-05-07 22:02:10 -0400 | [diff] [blame] | 228 | function pip_uninstall { |
Sampath Priyankara | 87d2396 | 2017-08-03 16:12:40 +0900 | [diff] [blame] | 229 | # Skip uninstall if offline |
| 230 | [[ "${OFFLINE}" = "True" ]] && return |
| 231 | |
Sean Dague | f28e7ef | 2017-05-07 22:02:10 -0400 | [diff] [blame] | 232 | local name=$1 |
| 233 | if [[ -n ${PIP_VIRTUAL_ENV:=} && -d ${PIP_VIRTUAL_ENV} ]]; then |
| 234 | local cmd_pip=$PIP_VIRTUAL_ENV/bin/pip |
| 235 | local sudo_pip="env" |
| 236 | else |
Jens Harbott | d7a82f4 | 2020-06-23 10:21:09 +0200 | [diff] [blame] | 237 | local cmd_pip="python$PYTHON3_VERSION -m pip" |
| 238 | local sudo_pip="sudo -H LC_ALL=en_US.UTF-8" |
Sean Dague | f28e7ef | 2017-05-07 22:02:10 -0400 | [diff] [blame] | 239 | fi |
| 240 | # don't error if we can't uninstall, it might not be there |
Brian Haley | 954fd1b | 2017-05-16 12:24:45 -0400 | [diff] [blame] | 241 | $sudo_pip $cmd_pip uninstall -y $name || /bin/true |
Sean Dague | f28e7ef | 2017-05-07 22:02:10 -0400 | [diff] [blame] | 242 | } |
| 243 | |
Joe Gordon | d5ac785 | 2015-02-06 19:29:23 -0800 | [diff] [blame] | 244 | # get version of a package from global requirements file |
| 245 | # get_from_global_requirements <package> |
| 246 | function get_from_global_requirements { |
| 247 | local package=$1 |
Ian Wienand | ada886d | 2015-10-07 14:06:26 +1100 | [diff] [blame] | 248 | local required_pkg |
| 249 | required_pkg=$(grep -i -h ^${package} $REQUIREMENTS_DIR/global-requirements.txt | cut -d\# -f1) |
Joe Gordon | d5ac785 | 2015-02-06 19:29:23 -0800 | [diff] [blame] | 250 | if [[ $required_pkg == "" ]]; then |
| 251 | die $LINENO "Can't find package $package in requirements" |
| 252 | fi |
| 253 | echo $required_pkg |
| 254 | } |
| 255 | |
Radosław Piliszek | be26306 | 2020-03-30 09:56:53 +0200 | [diff] [blame] | 256 | # get only version constraints of a package from global requirements file |
| 257 | # get_version_constraints_from_global_requirements <package> |
| 258 | function get_version_constraints_from_global_requirements { |
| 259 | local package=$1 |
| 260 | local required_pkg_version_constraint |
| 261 | # drop the package name from output (\K) |
| 262 | required_pkg_version_constraint=$(grep -i -h -o -P "^${package}\K.*" $REQUIREMENTS_DIR/global-requirements.txt | cut -d\# -f1) |
| 263 | if [[ $required_pkg_version_constraint == "" ]]; then |
| 264 | die $LINENO "Can't find package $package in requirements" |
| 265 | fi |
| 266 | echo $required_pkg_version_constraint |
| 267 | } |
| 268 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 269 | # should we use this library from their git repo, or should we let it |
| 270 | # get pulled in via pip dependencies. |
| 271 | function use_library_from_git { |
| 272 | local name=$1 |
| 273 | local enabled=1 |
Marc Koderer | 46f8cb7 | 2016-05-13 09:08:16 +0200 | [diff] [blame] | 274 | [[ ${LIBS_FROM_GIT} = 'ALL' ]] || [[ ,${LIBS_FROM_GIT}, =~ ,${name}, ]] && enabled=0 |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 275 | return $enabled |
| 276 | } |
| 277 | |
Sean Dague | c71973e | 2015-09-08 07:12:48 -0400 | [diff] [blame] | 278 | # determine if a package was installed from git |
| 279 | function lib_installed_from_git { |
| 280 | local name=$1 |
DamonLi | 007f588 | 2017-11-23 10:05:46 +0800 | [diff] [blame] | 281 | local safe_name |
| 282 | safe_name=$(python -c "from pkg_resources import safe_name; \ |
| 283 | print(safe_name('${name}'))") |
Ian Wienand | ae9c6ab | 2017-09-29 10:16:47 +1000 | [diff] [blame] | 284 | # Note "pip freeze" doesn't always work here, because it tries to |
| 285 | # be smart about finding the remote of the git repo the package |
| 286 | # was installed from. This doesn't work with zuul which clones |
| 287 | # repos with no remote. |
| 288 | # |
| 289 | # The best option seems to be to use "pip list" which will tell |
| 290 | # you the path an editable install was installed from; for example |
| 291 | # in response to something like |
Matt Riedemann | 9b6d2f2 | 2019-06-18 10:43:16 -0400 | [diff] [blame] | 292 | # pip install -e 'git+https://opendev.org/openstack/bashate#egg=bashate' |
Monty Taylor | f0cd9a8 | 2017-10-06 13:11:48 -0500 | [diff] [blame] | 293 | # pip list --format columns shows |
| 294 | # bashate 0.5.2.dev19 /tmp/env/src/bashate |
| 295 | # Thus we check the third column to see if we're installed from |
| 296 | # some local place. |
DamonLi | 007f588 | 2017-11-23 10:05:46 +0800 | [diff] [blame] | 297 | [[ -n $(pip list --format=columns 2>/dev/null | awk "/^$safe_name/ {print \$3}") ]] |
Sean Dague | c71973e | 2015-09-08 07:12:48 -0400 | [diff] [blame] | 298 | } |
| 299 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 300 | # setup a library by name. If we are trying to use the library from |
| 301 | # git, we'll do a git based install, otherwise we'll punt and the |
| 302 | # library should be installed by a requirements pull from another |
| 303 | # project. |
| 304 | function setup_lib { |
| 305 | local name=$1 |
| 306 | local dir=${GITDIR[$name]} |
| 307 | setup_install $dir |
| 308 | } |
| 309 | |
Atsushi SAKAI | 5509ed5 | 2015-11-30 20:20:21 +0900 | [diff] [blame] | 310 | # setup a library by name in editable mode. If we are trying to use |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 311 | # the library from git, we'll do a git based install, otherwise we'll |
| 312 | # punt and the library should be installed by a requirements pull from |
| 313 | # another project. |
| 314 | # |
| 315 | # use this for non namespaced libraries |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 316 | # |
Radosław Piliszek | be26306 | 2020-03-30 09:56:53 +0200 | [diff] [blame] | 317 | # setup_dev_lib [-bindep] <name> [<extras>] |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 318 | function setup_dev_lib { |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 319 | local bindep |
| 320 | if [[ $1 == -bindep* ]]; then |
| 321 | bindep="${1}" |
| 322 | shift |
| 323 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 324 | local name=$1 |
| 325 | local dir=${GITDIR[$name]} |
Radosław Piliszek | be26306 | 2020-03-30 09:56:53 +0200 | [diff] [blame] | 326 | local extras=$2 |
| 327 | setup_develop $bindep $dir $extras |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 328 | } |
| 329 | |
| 330 | # this should be used if you want to install globally, all libraries should |
| 331 | # use this, especially *oslo* ones |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 332 | # |
| 333 | # setup_install project_dir [extras] |
| 334 | # project_dir: directory of project repo (e.g., /opt/stack/keystone) |
| 335 | # extras: comma-separated list of optional dependencies to install |
| 336 | # (e.g., ldap,memcache). |
Takashi NATSUME | fa00777 | 2017-07-22 08:59:43 +0900 | [diff] [blame] | 337 | # See https://docs.openstack.org/pbr/latest/user/using.html#extra-requirements |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 338 | # bindep: Set "-bindep" as first argument to install bindep.txt packages |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 339 | # The command is like "pip install <project_dir>[<extras>]" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 340 | function setup_install { |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 341 | local bindep |
| 342 | if [[ $1 == -bindep* ]]; then |
| 343 | bindep="${1}" |
| 344 | shift |
| 345 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 346 | local project_dir=$1 |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 347 | local extras=$2 |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 348 | _setup_package_with_constraints_edit $bindep $project_dir "" $extras |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 349 | } |
| 350 | |
| 351 | # this should be used for projects which run services, like all services |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 352 | # |
| 353 | # setup_develop project_dir [extras] |
| 354 | # project_dir: directory of project repo (e.g., /opt/stack/keystone) |
| 355 | # extras: comma-separated list of optional dependencies to install |
| 356 | # (e.g., ldap,memcache). |
Takashi NATSUME | fa00777 | 2017-07-22 08:59:43 +0900 | [diff] [blame] | 357 | # See https://docs.openstack.org/pbr/latest/user/using.html#extra-requirements |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 358 | # The command is like "pip install -e <project_dir>[<extras>]" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 359 | function setup_develop { |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 360 | local bindep |
| 361 | if [[ $1 == -bindep* ]]; then |
| 362 | bindep="${1}" |
| 363 | shift |
| 364 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 365 | local project_dir=$1 |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 366 | local extras=$2 |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 367 | _setup_package_with_constraints_edit $bindep $project_dir -e $extras |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 368 | } |
| 369 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 370 | # ``pip install -e`` the package, which processes the dependencies |
| 371 | # using pip before running `setup.py develop` |
| 372 | # |
Clark Boylan | 05aa384 | 2015-08-03 11:14:13 -0700 | [diff] [blame] | 373 | # Updates the constraints from REQUIREMENTS_DIR to reflect the |
| 374 | # future installed state of this package. This ensures when we |
| 375 | # install this package we get the from source version. |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 376 | # |
Clark Boylan | 05aa384 | 2015-08-03 11:14:13 -0700 | [diff] [blame] | 377 | # Uses globals ``REQUIREMENTS_DIR`` |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 378 | # _setup_package_with_constraints_edit project_dir flags [extras] |
| 379 | # project_dir: directory of project repo (e.g., /opt/stack/keystone) |
| 380 | # flags: pip CLI options/flags |
| 381 | # extras: comma-separated list of optional dependencies to install |
| 382 | # (e.g., ldap,memcache). |
Takashi NATSUME | fa00777 | 2017-07-22 08:59:43 +0900 | [diff] [blame] | 383 | # See https://docs.openstack.org/pbr/latest/user/using.html#extra-requirements |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 384 | # The command is like "pip install <flags> <project_dir>[<extras>]" |
| 385 | function _setup_package_with_constraints_edit { |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 386 | local bindep |
| 387 | if [[ $1 == -bindep* ]]; then |
| 388 | bindep="${1}" |
| 389 | shift |
| 390 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 391 | local project_dir=$1 |
| 392 | local flags=$2 |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 393 | local extras=$3 |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 394 | |
YAMAMOTO Takashi | c8c1c61 | 2016-03-22 14:29:47 +0900 | [diff] [blame] | 395 | # Normalize the directory name to avoid |
| 396 | # "installation from path or url cannot be constrained to a version" |
| 397 | # error. |
| 398 | # REVISIT(yamamoto): Remove this when fixed in pip. |
| 399 | # https://github.com/pypa/pip/pull/3582 |
| 400 | project_dir=$(cd $project_dir && pwd) |
| 401 | |
Robert Collins | 635a5ba | 2015-06-10 08:48:06 +1200 | [diff] [blame] | 402 | if [ -n "$REQUIREMENTS_DIR" ]; then |
Ian Wienand | 6b9a564 | 2021-07-28 11:19:57 +1000 | [diff] [blame] | 403 | # Remove this package from constraints before we install it. |
| 404 | # That way, later installs won't "downgrade" the install from |
| 405 | # source we are about to do. |
Ian Wienand | ada886d | 2015-10-07 14:06:26 +1100 | [diff] [blame] | 406 | local name |
| 407 | name=$(awk '/^name.*=/ {print $3}' $project_dir/setup.cfg) |
Robert Collins | 7c83861 | 2015-07-03 13:28:09 +1200 | [diff] [blame] | 408 | $REQUIREMENTS_DIR/.venv/bin/edit-constraints \ |
Ian Wienand | 6b9a564 | 2021-07-28 11:19:57 +1000 | [diff] [blame] | 409 | $REQUIREMENTS_DIR/upper-constraints.txt -- $name |
Robert Collins | 635a5ba | 2015-06-10 08:48:06 +1200 | [diff] [blame] | 410 | fi |
| 411 | |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 412 | setup_package $bindep $project_dir "$flags" $extras |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 413 | |
James E. Blair | e1edde3 | 2018-03-02 15:05:14 +0000 | [diff] [blame] | 414 | # If this project is in LIBS_FROM_GIT, verify it was actually installed |
| 415 | # correctly. This helps catch errors caused by constraints mismatches. |
| 416 | if use_library_from_git "$project_dir"; then |
| 417 | if ! lib_installed_from_git "$project_dir"; then |
| 418 | die $LINENO "The following LIBS_FROM_GIT was not installed correctly: $project_dir" |
| 419 | fi |
| 420 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 421 | } |
| 422 | |
| 423 | # ``pip install -e`` the package, which processes the dependencies |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 424 | # using pip before running `setup.py develop`. The command is like |
| 425 | # "pip install <flags> <project_dir>[<extras>]" |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 426 | # |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 427 | # Uses globals ``STACK_USER`` |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 428 | # |
| 429 | # Usage: |
| 430 | # setup_package [-bindep[=profile,profile]] <project_dir> <flags> [extras] |
| 431 | # |
| 432 | # -bindep : Use bindep to install dependencies; select extra profiles |
| 433 | # as comma separated arguments after "=" |
| 434 | # project_dir : directory of project repo (e.g., /opt/stack/keystone) |
| 435 | # flags : pip CLI options/flags |
| 436 | # extras : comma-separated list of optional dependencies to install |
| 437 | # (e.g., ldap,memcache). |
| 438 | # See https://docs.openstack.org/pbr/latest/user/using.html#extra-requirements |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 439 | function setup_package { |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 440 | local bindep=0 |
| 441 | local bindep_flag="" |
| 442 | local bindep_profiles="" |
| 443 | if [[ $1 == -bindep* ]]; then |
| 444 | bindep=1 |
| 445 | IFS="=" read bindep_flag bindep_profiles <<< ${1} |
| 446 | shift |
| 447 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 448 | local project_dir=$1 |
| 449 | local flags=$2 |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 450 | local extras=$3 |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 451 | |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 452 | # if the flags variable exists, and it doesn't look like a flag, |
| 453 | # assume it's actually the extras list. |
| 454 | if [[ -n "$flags" && -z "$extras" && ! "$flags" =~ ^-.* ]]; then |
| 455 | extras=$flags |
| 456 | flags="" |
| 457 | fi |
| 458 | |
| 459 | if [[ ! -z "$extras" ]]; then |
| 460 | extras="[$extras]" |
| 461 | fi |
| 462 | |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 463 | # install any bindep packages |
| 464 | if [[ $bindep == 1 ]]; then |
| 465 | install_bindep $project_dir/bindep.txt $bindep_profiles |
| 466 | fi |
| 467 | |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 468 | pip_install $flags "$project_dir$extras" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 469 | # ensure that further actions can do things like setup.py sdist |
Clark Boylan | a40f9cb | 2018-04-04 14:02:30 -0700 | [diff] [blame] | 470 | if [[ "$flags" == "-e" && "$GLOBAL_VENV" == "False" ]]; then |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 471 | safe_chown -R $STACK_USER $1/*.egg-info |
| 472 | fi |
| 473 | } |
| 474 | |
Doug Hellmann | ddc3839 | 2015-05-07 21:06:24 +0000 | [diff] [blame] | 475 | # Report whether python 3 should be used |
Jens Harbott | d7a82f4 | 2020-06-23 10:21:09 +0200 | [diff] [blame] | 476 | # TODO(frickler): drop this once all legacy uses are removed |
Doug Hellmann | ddc3839 | 2015-05-07 21:06:24 +0000 | [diff] [blame] | 477 | function python3_enabled { |
Radosław Piliszek | 3cbb33e | 2020-06-30 17:52:10 +0200 | [diff] [blame] | 478 | return 0 |
Doug Hellmann | ddc3839 | 2015-05-07 21:06:24 +0000 | [diff] [blame] | 479 | } |
| 480 | |
Federico Ressi | 21a10d3 | 2020-01-31 07:43:30 +0100 | [diff] [blame] | 481 | # Provide requested python version and sets PYTHON variable |
| 482 | function install_python { |
Jens Harbott | d7a82f4 | 2020-06-23 10:21:09 +0200 | [diff] [blame] | 483 | install_python3 |
| 484 | export PYTHON=$(which python${PYTHON3_VERSION} 2>/dev/null) |
Federico Ressi | 21a10d3 | 2020-01-31 07:43:30 +0100 | [diff] [blame] | 485 | } |
| 486 | |
Doug Hellmann | ddc3839 | 2015-05-07 21:06:24 +0000 | [diff] [blame] | 487 | # Install python3 packages |
| 488 | function install_python3 { |
| 489 | if is_ubuntu; then |
Lubosz "diltram" Kosnik | 0a09976 | 2016-08-03 10:21:41 -0500 | [diff] [blame] | 490 | apt_get install python${PYTHON3_VERSION} python${PYTHON3_VERSION}-dev |
Terry Wilson | 78cf6f6 | 2019-10-15 19:45:09 +0000 | [diff] [blame] | 491 | elif is_fedora; then |
Federico Ressi | 2dcbc28 | 2020-02-05 11:29:51 +0100 | [diff] [blame] | 492 | if [ "$os_VENDOR" = "Fedora" ]; then |
| 493 | install_package python${PYTHON3_VERSION//.} |
| 494 | else |
| 495 | install_package python${PYTHON3_VERSION//.} python${PYTHON3_VERSION//.}-devel |
| 496 | fi |
Doug Hellmann | ddc3839 | 2015-05-07 21:06:24 +0000 | [diff] [blame] | 497 | fi |
| 498 | } |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 499 | |
Sean Dague | f80e2cf | 2017-01-18 15:42:32 -0500 | [diff] [blame] | 500 | function install_devstack_tools { |
| 501 | # intentionally old to ensure devstack-gate has control |
| 502 | local dstools_version=${DSTOOLS_VERSION:-0.1.2} |
| 503 | install_python3 |
| 504 | sudo pip3 install -U devstack-tools==${dstools_version} |
| 505 | } |
| 506 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 507 | # Restore xtrace |
| 508 | $INC_PY_TRACE |
| 509 | |
| 510 | # Local variables: |
| 511 | # mode: shell-script |
| 512 | # End: |