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. |
Stephen Finucane | 320c2bf | 2024-12-03 17:04:39 +0000 | [diff] [blame] | 43 | $PYTHON -m venv --system-site-packages "${DEVSTACK_VENV}" |
| 44 | pip_install -U pip setuptools[core] |
Clark Boylan | a40f9cb | 2018-04-04 14:02:30 -0700 | [diff] [blame] | 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" |
Joel Capitao | c6c5e12 | 2024-11-13 10:33:28 +0100 | [diff] [blame] | 199 | local sudo_pip="sudo -H LC_ALL=en_US.UTF-8" |
Jens Harbott | d7a82f4 | 2020-06-23 10:21:09 +0200 | [diff] [blame] | 200 | echo "Using python $PYTHON3_VERSION to install $package_dir" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 201 | fi |
| 202 | |
Robert Collins | 635a5ba | 2015-06-10 08:48:06 +1200 | [diff] [blame] | 203 | cmd_pip="$cmd_pip install" |
Clark Boylan | 05aa384 | 2015-08-03 11:14:13 -0700 | [diff] [blame] | 204 | # Always apply constraints |
| 205 | cmd_pip="$cmd_pip -c $REQUIREMENTS_DIR/upper-constraints.txt" |
Robert Collins | 635a5ba | 2015-06-10 08:48:06 +1200 | [diff] [blame] | 206 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 207 | $xtrace |
Clark Boylan | f266a2d | 2017-06-12 14:57:59 -0700 | [diff] [blame] | 208 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 209 | $sudo_pip \ |
Eli Qiao | 6a83c42 | 2015-03-17 16:54:16 +0800 | [diff] [blame] | 210 | http_proxy="${http_proxy:-}" \ |
| 211 | https_proxy="${https_proxy:-}" \ |
| 212 | no_proxy="${no_proxy:-}" \ |
Joe Gordon | cd8824a | 2015-03-04 16:40:19 -0800 | [diff] [blame] | 213 | PIP_FIND_LINKS=$PIP_FIND_LINKS \ |
Monty Taylor | 09b5b05 | 2020-03-27 11:22:39 -0500 | [diff] [blame] | 214 | $cmd_pip $upgrade \ |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 215 | $@ |
Federico Ressi | e208d06 | 2015-11-21 11:15:39 +0000 | [diff] [blame] | 216 | result=$? |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 217 | |
Sean Dague | cb658fa | 2015-10-08 17:12:03 -0400 | [diff] [blame] | 218 | time_stop "pip_install" |
Federico Ressi | e208d06 | 2015-11-21 11:15:39 +0000 | [diff] [blame] | 219 | return $result |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 220 | } |
| 221 | |
Sean Dague | f28e7ef | 2017-05-07 22:02:10 -0400 | [diff] [blame] | 222 | function pip_uninstall { |
Sampath Priyankara | 87d2396 | 2017-08-03 16:12:40 +0900 | [diff] [blame] | 223 | # Skip uninstall if offline |
| 224 | [[ "${OFFLINE}" = "True" ]] && return |
| 225 | |
Sean Dague | f28e7ef | 2017-05-07 22:02:10 -0400 | [diff] [blame] | 226 | local name=$1 |
| 227 | if [[ -n ${PIP_VIRTUAL_ENV:=} && -d ${PIP_VIRTUAL_ENV} ]]; then |
| 228 | local cmd_pip=$PIP_VIRTUAL_ENV/bin/pip |
| 229 | local sudo_pip="env" |
| 230 | else |
Jens Harbott | d7a82f4 | 2020-06-23 10:21:09 +0200 | [diff] [blame] | 231 | local cmd_pip="python$PYTHON3_VERSION -m pip" |
| 232 | local sudo_pip="sudo -H LC_ALL=en_US.UTF-8" |
Sean Dague | f28e7ef | 2017-05-07 22:02:10 -0400 | [diff] [blame] | 233 | fi |
| 234 | # 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] | 235 | $sudo_pip $cmd_pip uninstall -y $name || /bin/true |
Sean Dague | f28e7ef | 2017-05-07 22:02:10 -0400 | [diff] [blame] | 236 | } |
| 237 | |
Joe Gordon | d5ac785 | 2015-02-06 19:29:23 -0800 | [diff] [blame] | 238 | # get version of a package from global requirements file |
| 239 | # get_from_global_requirements <package> |
| 240 | function get_from_global_requirements { |
| 241 | local package=$1 |
Ian Wienand | ada886d | 2015-10-07 14:06:26 +1100 | [diff] [blame] | 242 | local required_pkg |
| 243 | 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] | 244 | if [[ $required_pkg == "" ]]; then |
| 245 | die $LINENO "Can't find package $package in requirements" |
| 246 | fi |
| 247 | echo $required_pkg |
| 248 | } |
| 249 | |
Radosław Piliszek | be26306 | 2020-03-30 09:56:53 +0200 | [diff] [blame] | 250 | # get only version constraints of a package from global requirements file |
| 251 | # get_version_constraints_from_global_requirements <package> |
| 252 | function get_version_constraints_from_global_requirements { |
| 253 | local package=$1 |
| 254 | local required_pkg_version_constraint |
| 255 | # drop the package name from output (\K) |
| 256 | required_pkg_version_constraint=$(grep -i -h -o -P "^${package}\K.*" $REQUIREMENTS_DIR/global-requirements.txt | cut -d\# -f1) |
| 257 | if [[ $required_pkg_version_constraint == "" ]]; then |
| 258 | die $LINENO "Can't find package $package in requirements" |
| 259 | fi |
| 260 | echo $required_pkg_version_constraint |
| 261 | } |
| 262 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 263 | # should we use this library from their git repo, or should we let it |
| 264 | # get pulled in via pip dependencies. |
| 265 | function use_library_from_git { |
| 266 | local name=$1 |
| 267 | local enabled=1 |
Marc Koderer | 46f8cb7 | 2016-05-13 09:08:16 +0200 | [diff] [blame] | 268 | [[ ${LIBS_FROM_GIT} = 'ALL' ]] || [[ ,${LIBS_FROM_GIT}, =~ ,${name}, ]] && enabled=0 |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 269 | return $enabled |
| 270 | } |
| 271 | |
Sean Dague | c71973e | 2015-09-08 07:12:48 -0400 | [diff] [blame] | 272 | # determine if a package was installed from git |
| 273 | function lib_installed_from_git { |
| 274 | local name=$1 |
DamonLi | 007f588 | 2017-11-23 10:05:46 +0800 | [diff] [blame] | 275 | local safe_name |
| 276 | safe_name=$(python -c "from pkg_resources import safe_name; \ |
| 277 | print(safe_name('${name}'))") |
Ian Wienand | ae9c6ab | 2017-09-29 10:16:47 +1000 | [diff] [blame] | 278 | # Note "pip freeze" doesn't always work here, because it tries to |
| 279 | # be smart about finding the remote of the git repo the package |
| 280 | # was installed from. This doesn't work with zuul which clones |
| 281 | # repos with no remote. |
| 282 | # |
| 283 | # The best option seems to be to use "pip list" which will tell |
| 284 | # you the path an editable install was installed from; for example |
| 285 | # in response to something like |
Matt Riedemann | 9b6d2f2 | 2019-06-18 10:43:16 -0400 | [diff] [blame] | 286 | # pip install -e 'git+https://opendev.org/openstack/bashate#egg=bashate' |
Monty Taylor | f0cd9a8 | 2017-10-06 13:11:48 -0500 | [diff] [blame] | 287 | # pip list --format columns shows |
| 288 | # bashate 0.5.2.dev19 /tmp/env/src/bashate |
| 289 | # Thus we check the third column to see if we're installed from |
| 290 | # some local place. |
DamonLi | 007f588 | 2017-11-23 10:05:46 +0800 | [diff] [blame] | 291 | [[ -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] | 292 | } |
| 293 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 294 | # setup a library by name. If we are trying to use the library from |
| 295 | # git, we'll do a git based install, otherwise we'll punt and the |
| 296 | # library should be installed by a requirements pull from another |
| 297 | # project. |
| 298 | function setup_lib { |
| 299 | local name=$1 |
| 300 | local dir=${GITDIR[$name]} |
| 301 | setup_install $dir |
| 302 | } |
| 303 | |
Atsushi SAKAI | 5509ed5 | 2015-11-30 20:20:21 +0900 | [diff] [blame] | 304 | # 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] | 305 | # the library from git, we'll do a git based install, otherwise we'll |
| 306 | # punt and the library should be installed by a requirements pull from |
| 307 | # another project. |
| 308 | # |
| 309 | # use this for non namespaced libraries |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 310 | # |
Radosław Piliszek | be26306 | 2020-03-30 09:56:53 +0200 | [diff] [blame] | 311 | # setup_dev_lib [-bindep] <name> [<extras>] |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 312 | function setup_dev_lib { |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 313 | local bindep |
| 314 | if [[ $1 == -bindep* ]]; then |
| 315 | bindep="${1}" |
| 316 | shift |
| 317 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 318 | local name=$1 |
| 319 | local dir=${GITDIR[$name]} |
Radosław Piliszek | be26306 | 2020-03-30 09:56:53 +0200 | [diff] [blame] | 320 | local extras=$2 |
| 321 | setup_develop $bindep $dir $extras |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 322 | } |
| 323 | |
| 324 | # this should be used if you want to install globally, all libraries should |
| 325 | # use this, especially *oslo* ones |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 326 | # |
| 327 | # setup_install project_dir [extras] |
| 328 | # project_dir: directory of project repo (e.g., /opt/stack/keystone) |
| 329 | # extras: comma-separated list of optional dependencies to install |
| 330 | # (e.g., ldap,memcache). |
Takashi NATSUME | fa00777 | 2017-07-22 08:59:43 +0900 | [diff] [blame] | 331 | # See https://docs.openstack.org/pbr/latest/user/using.html#extra-requirements |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 332 | # bindep: Set "-bindep" as first argument to install bindep.txt packages |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 333 | # The command is like "pip install <project_dir>[<extras>]" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 334 | function setup_install { |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 335 | local bindep |
| 336 | if [[ $1 == -bindep* ]]; then |
| 337 | bindep="${1}" |
| 338 | shift |
| 339 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 340 | local project_dir=$1 |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 341 | local extras=$2 |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 342 | _setup_package_with_constraints_edit $bindep $project_dir "" $extras |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 343 | } |
| 344 | |
| 345 | # this should be used for projects which run services, like all services |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 346 | # |
| 347 | # setup_develop project_dir [extras] |
| 348 | # project_dir: directory of project repo (e.g., /opt/stack/keystone) |
| 349 | # extras: comma-separated list of optional dependencies to install |
| 350 | # (e.g., ldap,memcache). |
Takashi NATSUME | fa00777 | 2017-07-22 08:59:43 +0900 | [diff] [blame] | 351 | # See https://docs.openstack.org/pbr/latest/user/using.html#extra-requirements |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 352 | # The command is like "pip install -e <project_dir>[<extras>]" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 353 | function setup_develop { |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 354 | local bindep |
| 355 | if [[ $1 == -bindep* ]]; then |
| 356 | bindep="${1}" |
| 357 | shift |
| 358 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 359 | local project_dir=$1 |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 360 | local extras=$2 |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 361 | _setup_package_with_constraints_edit $bindep $project_dir -e $extras |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 362 | } |
| 363 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 364 | # ``pip install -e`` the package, which processes the dependencies |
| 365 | # using pip before running `setup.py develop` |
| 366 | # |
Clark Boylan | 05aa384 | 2015-08-03 11:14:13 -0700 | [diff] [blame] | 367 | # Updates the constraints from REQUIREMENTS_DIR to reflect the |
| 368 | # future installed state of this package. This ensures when we |
| 369 | # install this package we get the from source version. |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 370 | # |
Clark Boylan | 05aa384 | 2015-08-03 11:14:13 -0700 | [diff] [blame] | 371 | # Uses globals ``REQUIREMENTS_DIR`` |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 372 | # _setup_package_with_constraints_edit project_dir flags [extras] |
| 373 | # project_dir: directory of project repo (e.g., /opt/stack/keystone) |
| 374 | # flags: pip CLI options/flags |
| 375 | # extras: comma-separated list of optional dependencies to install |
| 376 | # (e.g., ldap,memcache). |
Takashi NATSUME | fa00777 | 2017-07-22 08:59:43 +0900 | [diff] [blame] | 377 | # See https://docs.openstack.org/pbr/latest/user/using.html#extra-requirements |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 378 | # The command is like "pip install <flags> <project_dir>[<extras>]" |
| 379 | function _setup_package_with_constraints_edit { |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 380 | local bindep |
| 381 | if [[ $1 == -bindep* ]]; then |
| 382 | bindep="${1}" |
| 383 | shift |
| 384 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 385 | local project_dir=$1 |
| 386 | local flags=$2 |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 387 | local extras=$3 |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 388 | |
YAMAMOTO Takashi | c8c1c61 | 2016-03-22 14:29:47 +0900 | [diff] [blame] | 389 | # Normalize the directory name to avoid |
| 390 | # "installation from path or url cannot be constrained to a version" |
| 391 | # error. |
| 392 | # REVISIT(yamamoto): Remove this when fixed in pip. |
| 393 | # https://github.com/pypa/pip/pull/3582 |
| 394 | project_dir=$(cd $project_dir && pwd) |
| 395 | |
Robert Collins | 635a5ba | 2015-06-10 08:48:06 +1200 | [diff] [blame] | 396 | if [ -n "$REQUIREMENTS_DIR" ]; then |
Ian Wienand | 6b9a564 | 2021-07-28 11:19:57 +1000 | [diff] [blame] | 397 | # Remove this package from constraints before we install it. |
| 398 | # That way, later installs won't "downgrade" the install from |
| 399 | # source we are about to do. |
Ian Wienand | ada886d | 2015-10-07 14:06:26 +1100 | [diff] [blame] | 400 | local name |
| 401 | name=$(awk '/^name.*=/ {print $3}' $project_dir/setup.cfg) |
Slawek Kaplonski | 4ddd456 | 2024-02-09 14:11:44 +0100 | [diff] [blame] | 402 | if [ -z $name ]; then |
| 403 | name=$(awk '/^name =/ {gsub(/"/, "", $3); print $3}' $project_dir/pyproject.toml) |
| 404 | fi |
Robert Collins | 7c83861 | 2015-07-03 13:28:09 +1200 | [diff] [blame] | 405 | $REQUIREMENTS_DIR/.venv/bin/edit-constraints \ |
Ian Wienand | 6b9a564 | 2021-07-28 11:19:57 +1000 | [diff] [blame] | 406 | $REQUIREMENTS_DIR/upper-constraints.txt -- $name |
Robert Collins | 635a5ba | 2015-06-10 08:48:06 +1200 | [diff] [blame] | 407 | fi |
| 408 | |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 409 | setup_package $bindep $project_dir "$flags" $extras |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 410 | |
James E. Blair | e1edde3 | 2018-03-02 15:05:14 +0000 | [diff] [blame] | 411 | # If this project is in LIBS_FROM_GIT, verify it was actually installed |
| 412 | # correctly. This helps catch errors caused by constraints mismatches. |
| 413 | if use_library_from_git "$project_dir"; then |
| 414 | if ! lib_installed_from_git "$project_dir"; then |
| 415 | die $LINENO "The following LIBS_FROM_GIT was not installed correctly: $project_dir" |
| 416 | fi |
| 417 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 418 | } |
| 419 | |
| 420 | # ``pip install -e`` the package, which processes the dependencies |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 421 | # using pip before running `setup.py develop`. The command is like |
| 422 | # "pip install <flags> <project_dir>[<extras>]" |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 423 | # |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 424 | # Uses globals ``STACK_USER`` |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 425 | # |
| 426 | # Usage: |
| 427 | # setup_package [-bindep[=profile,profile]] <project_dir> <flags> [extras] |
| 428 | # |
| 429 | # -bindep : Use bindep to install dependencies; select extra profiles |
| 430 | # as comma separated arguments after "=" |
| 431 | # project_dir : directory of project repo (e.g., /opt/stack/keystone) |
| 432 | # flags : pip CLI options/flags |
| 433 | # extras : comma-separated list of optional dependencies to install |
| 434 | # (e.g., ldap,memcache). |
| 435 | # See https://docs.openstack.org/pbr/latest/user/using.html#extra-requirements |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 436 | function setup_package { |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 437 | local bindep=0 |
| 438 | local bindep_flag="" |
| 439 | local bindep_profiles="" |
| 440 | if [[ $1 == -bindep* ]]; then |
| 441 | bindep=1 |
| 442 | IFS="=" read bindep_flag bindep_profiles <<< ${1} |
| 443 | shift |
| 444 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 445 | local project_dir=$1 |
| 446 | local flags=$2 |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 447 | local extras=$3 |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 448 | |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 449 | # if the flags variable exists, and it doesn't look like a flag, |
| 450 | # assume it's actually the extras list. |
| 451 | if [[ -n "$flags" && -z "$extras" && ! "$flags" =~ ^-.* ]]; then |
| 452 | extras=$flags |
| 453 | flags="" |
| 454 | fi |
| 455 | |
| 456 | if [[ ! -z "$extras" ]]; then |
| 457 | extras="[$extras]" |
| 458 | fi |
| 459 | |
Ian Wienand | 58243f6 | 2018-12-13 14:05:53 +1100 | [diff] [blame] | 460 | # install any bindep packages |
| 461 | if [[ $bindep == 1 ]]; then |
| 462 | install_bindep $project_dir/bindep.txt $bindep_profiles |
| 463 | fi |
| 464 | |
Brant Knudson | 0842b81 | 2015-08-03 13:31:25 -0500 | [diff] [blame] | 465 | pip_install $flags "$project_dir$extras" |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 466 | # ensure that further actions can do things like setup.py sdist |
Clark Boylan | a40f9cb | 2018-04-04 14:02:30 -0700 | [diff] [blame] | 467 | if [[ "$flags" == "-e" && "$GLOBAL_VENV" == "False" ]]; then |
yatinkarel | 0ff6272 | 2024-09-02 17:29:55 +0530 | [diff] [blame] | 468 | # egg-info is not created when project have pyproject.toml |
| 469 | if [ -d $1/*.egg-info ]; then |
| 470 | safe_chown -R $STACK_USER $1/*.egg-info |
| 471 | fi |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 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 |
yatinkarel | f41a16c | 2025-04-09 18:14:00 +0530 | [diff] [blame] | 492 | install_package python${PYTHON3_VERSION}-devel python${PYTHON3_VERSION}-pip |
Doug Hellmann | ddc3839 | 2015-05-07 21:06:24 +0000 | [diff] [blame] | 493 | fi |
| 494 | } |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 495 | |
Sean Dague | f80e2cf | 2017-01-18 15:42:32 -0500 | [diff] [blame] | 496 | function install_devstack_tools { |
| 497 | # intentionally old to ensure devstack-gate has control |
| 498 | local dstools_version=${DSTOOLS_VERSION:-0.1.2} |
| 499 | install_python3 |
| 500 | sudo pip3 install -U devstack-tools==${dstools_version} |
| 501 | } |
| 502 | |
Dean Troyer | 490430d | 2015-01-30 14:38:35 -0600 | [diff] [blame] | 503 | # Restore xtrace |
| 504 | $INC_PY_TRACE |
| 505 | |
| 506 | # Local variables: |
| 507 | # mode: shell-script |
| 508 | # End: |