osx_utils.sh 13.3 KB
Newer Older
Matthew Brett's avatar
Matthew Brett committed
1
2
3
4
#!/bin/bash
# Use with ``source osx_utils.sh``
set -e

5
6
7
8
# Get our own location on this filesystem, load common utils
MULTIBUILD_DIR=$(dirname "${BASH_SOURCE[0]}")
source $MULTIBUILD_DIR/common_utils.sh

Matthew Brett's avatar
Matthew Brett committed
9
10
11
12
13
MACPYTHON_URL=https://www.python.org/ftp/python
MACPYTHON_PY_PREFIX=/Library/Frameworks/Python.framework/Versions
GET_PIP_URL=https://bootstrap.pypa.io/get-pip.py
WORKING_SDIR=working

14
# As of 8 December 2020 - latest Python of each version with binary download
Matthew Brett's avatar
Matthew Brett committed
15
# available.
16
# See: https://www.python.org/downloads/mac-osx/
robbuckley's avatar
robbuckley committed
17
LATEST_2p7=2.7.18
Andrew Murray's avatar
Andrew Murray committed
18
LATEST_3p5=3.5.4
Andrew Murray's avatar
Andrew Murray committed
19
LATEST_3p6=3.6.8
Andrew Murray's avatar
Andrew Murray committed
20
LATEST_3p7=3.7.9
Andrew Murray's avatar
Andrew Murray committed
21
LATEST_3p8=3.8.6
22
LATEST_3p9=3.9.1
Matthew Brett's avatar
Matthew Brett committed
23

Matthew Brett's avatar
Matthew Brett committed
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

function check_python {
    if [ -z "$PYTHON_EXE" ]; then
        echo "PYTHON_EXE variable not defined"
        exit 1
    fi
}

function check_pip {
    if [ -z "$PIP_CMD" ]; then
        echo "PIP_CMD variable not defined"
        exit 1
    fi
}

function check_var {
    if [ -z "$1" ]; then
        echo "required variable not defined"
        exit 1
    fi
}

function get_py_digit {
    check_python
    $PYTHON_EXE -c "import sys; print(sys.version_info[0])"
}

function get_py_mm {
    check_python
    $PYTHON_EXE -c "import sys; print('{0}.{1}'.format(*sys.version_info[0:2]))"
}

function get_py_mm_nodot {
    check_python
    $PYTHON_EXE -c "import sys; print('{0}{1}'.format(*sys.version_info[0:2]))"
}

function get_py_prefix {
    check_python
    $PYTHON_EXE -c "import sys; print(sys.prefix)"
}

function fill_pyver {
    # Convert major or major.minor format to major.minor.micro
    #
    # Hence:
    # 2 -> 2.7.11  (depending on LATEST_2p7 value)
    # 2.7 -> 2.7.11  (depending on LATEST_2p7 value)
    local ver=$1
    check_var $ver
    if [[ $ver =~ [0-9]+\.[0-9]+\.[0-9]+ ]]; then
        # Major.minor.micro format already
        echo $ver
    elif [ $ver == 2 ] || [ $ver == "2.7" ]; then
        echo $LATEST_2p7
Andrew Murray's avatar
Andrew Murray committed
79
80
81
    elif [ $ver == 3 ] || [ $ver == "3.9" ]; then
        echo $LATEST_3p9
    elif [ $ver == "3.8" ]; then
82
83
        echo $LATEST_3p8
    elif [ $ver == "3.7" ]; then
Andrew Murray's avatar
Andrew Murray committed
84
85
        echo $LATEST_3p7
    elif [ $ver == "3.6" ]; then
Matthew Brett's avatar
Matthew Brett committed
86
87
        echo $LATEST_3p6
    elif [ $ver == "3.5" ]; then
Matthew Brett's avatar
Matthew Brett committed
88
89
        echo $LATEST_3p5
    else
90
        echo "Can't fill version $ver" 1>&2
Matthew Brett's avatar
Matthew Brett committed
91
92
93
94
        exit 1
    fi
}

95
96
97
98
99
100
101
102
103
104
105
106
function macpython_sdk_list_for_version {
    # return a list of SDK targets supported for a given CPython version
    # Parameters
    #   $py_version (python version in major.minor.extra format)
    # eg
    #  macpython_sdks_for_version 2.7.15
    #  >> 10.6 10.9
    local _ver=$(fill_pyver $1)
    local _major=${_ver%%.*}
    local _return

    if [ "$_major" -eq "2" ]; then
robbuckley's avatar
robbuckley committed
107
        [ $(lex_ver $_ver) -lt $(lex_ver 2.7.18) ] && _return="10.6"
108
109
110
111
112
113
114
115
116
117
118
119
120
        [ $(lex_ver $_ver) -ge $(lex_ver 2.7.15) ] && _return="$_return 10.9"
    elif [ "$_major" -eq "3" ]; then
        [ $(lex_ver $_ver) -lt $(lex_ver 3.8)    ] && _return="10.6"
        [ $(lex_ver $_ver) -ge $(lex_ver 3.6.5)  ] && _return="$_return 10.9"
    else
        echo "Error version=${_ver}, expecting 2.x or 3.x" 1>&2
        exit 1
    fi
    echo $_return
}

function macpython_sdk_for_version {
    # assumes the output of macpython_sdk_list_for_version is a list
Andrew Murray's avatar
Andrew Murray committed
121
    # of SDK versions XX.Y in sorted order, eg "10.6 10.9" or "10.9"
122
123
124
    echo $(macpython_sdk_list_for_version $1) | awk -F' ' '{print $NF}'
}

Matthew Brett's avatar
Matthew Brett committed
125
126
127
128
129
130
131
132
function pyinst_ext_for_version {
    # echo "pkg" or "dmg" depending on the passed Python version
    # Parameters
    #   $py_version (python version in major.minor.extra format)
    #
    # Earlier Python installers are .dmg, later are .pkg.
    local py_version=$1
    check_var $py_version
133
    py_version=$(fill_pyver $py_version)
Matthew Brett's avatar
Matthew Brett committed
134
135
    local py_0=${py_version:0:1}
    if [ $py_0 -eq 2 ]; then
136
        if [ "$(lex_ver $py_version)" -ge "$(lex_ver 2.7.9)" ]; then
Matthew Brett's avatar
Matthew Brett committed
137
138
139
140
141
            echo "pkg"
        else
            echo "dmg"
        fi
    elif [ $py_0 -ge 3 ]; then
Andrew Murray's avatar
Andrew Murray committed
142
		echo "pkg"
Matthew Brett's avatar
Matthew Brett committed
143
144
145
    fi
}

146
function pyinst_fname_for_version {
147
    # echo filename for OSX installer file given Python and minimum
148
    # macOS versions
149
    # Parameters
150
    #   $py_version (Python version in major.minor.extra format)
151
    #   $py_osx_ver: {major.minor | not defined}
152
153
154
    #       if defined, the minimum macOS SDK version that Python is
    #       built for, eg: "10.6" or "10.9", if not defined, infers
    #       this from $py_version using macpython_sdk_for_version
155
    local py_version=$1
156
    local py_osx_ver=${2:-$(macpython_sdk_for_version $py_version)}
157
    local inst_ext=$(pyinst_ext_for_version $py_version)
158
    echo "python-${py_version}-macosx${py_osx_ver}.${inst_ext}"
159
160
}

161
function get_macpython_arch {
robbuckley's avatar
robbuckley committed
162
163
164
165
166
167
    # echo arch (e.g. intel or x86_64), extracted from the distutils platform tag
    # Parameters
    #   $distutils_plat   PEP425 style platform tag, or if not provided, calls
    #                       the function get_distutils_platform, provided by
    #                       common_utils.sh. Fails if this is not a mac platform
    #
Andrew Murray's avatar
Andrew Murray committed
168
    # Note: MUST only be called after the version of Python used to build the
robbuckley's avatar
robbuckley committed
169
170
    # target wheel has been installed and is on the path
    local distutils_plat=${1:-$(get_distutils_platform)}
171
172
173
    if [[ $distutils_plat =~ macosx-(10\.[0-9]+)-(.*) ]]; then
        echo ${BASH_REMATCH[2]}
    else
robbuckley's avatar
robbuckley committed
174
        echo "Error parsing macOS distutils platform '$distutils_plat'"
175
176
177
178
179
        exit 1
    fi
}

function get_macpython_osx_ver {
robbuckley's avatar
robbuckley committed
180
181
182
183
184
185
    # echo minimum macOS version (e.g. 10.9) from the distutils platform tag
    # Parameters
    #   $distutils_plat   PEP425 style platform tag, or if not provided, calls
    #                       the function get_distutils_platform, provided by
    #                       common_utils.sh. Fails if this is not a mac platform
    #
Andrew Murray's avatar
Andrew Murray committed
186
    # Note: MUST only be called after the version of Python used to build the
robbuckley's avatar
robbuckley committed
187
188
    # target wheel has been installed and is on the path
    local distutils_plat=${1:-$(get_distutils_platform)}
189
190
191
    if [[ $distutils_plat =~ macosx-(10\.[0-9]+)-(.*) ]]; then
        echo ${BASH_REMATCH[1]}
    else
robbuckley's avatar
robbuckley committed
192
        echo "Error parsing macOS distutils platform '$distutils_plat'"
193
194
195
        exit 1
    fi
}
196

197
function macpython_arch_for_version {
198
199
    # echo arch (intel or x86_64) that a version of Python is expected
    # to be built for
200
    # Parameters
201
202
    #   $py_ver     Python version, in the format (major.minor.patch) for
    #               CPython, or pypy-(major.minor) for PyPy
203
    #   $py_osx_ver minimum macOS version the target Python is built for
204
    #               (major.minor)
205
206
207
208
209
210
211
212
213
    local py_ver=$1
    local py_osx_ver=${2:-$MB_PYTHON_OSX_VER}
    check_var $1
    if [[ $(macpython_impl_for_version $py_ver) == "cp" ]]; then
        if [[ "$py_osx_ver" == "10.6" ]]; then
            echo "intel"
        elif [[ "$py_osx_ver" == "10.9" ]]; then
            echo "x86_64"
        else
214
            echo "Unexpected CPython macOS version: ${py_osx_ver}, supported values: 10.6 and 10.9"
215
216
217
            exit 1
        fi
    else
Rob Buckley's avatar
Rob Buckley committed
218
        echo "x86_64"
219
220
221
222
223
224
    fi
}

function macpython_impl_for_version {
    # echo Python implementation (cp for CPython, pp for PyPy) given a
    # suitably formatted version string
225
226
    # Parameters:
    #     $version : [implementation-]major[.minor[.patch]]
227
228
    #         Python implementation, e.g. "3.6" for CPython or
    #         "pypy-5.4" for PyPy
229
    local version=$1
230
    check_var $1
mattip's avatar
mattip committed
231
    if [[ "$version" =~ ^pypy ]]; then
232
        echo pp
233
    elif [[ "$version" =~ ([0-9\.]+) ]]; then
234
        echo cp
Kyle Stewart's avatar
Kyle Stewart committed
235
    else
Matt Bachmann's avatar
Matt Bachmann committed
236
        echo "config error: Issue parsing this implementation in install_python:"
237
        echo "    version=$version"
Kyle Stewart's avatar
Kyle Stewart committed
238
239
240
241
        exit 1
    fi
}

242
function strip_macpython_ver_prefix {
243
    # strip any implementation prefix from a Python version string
244
245
246
247
248
249
250
251
252
253
254
    # Parameters:
    #     $version : [implementation-]major[.minor[.patch]]
    #         Python implementation, e.g. "3.6" for CPython or
    #         "pypy-5.4" for PyPy
    local version=$1
    check_var $1
    if [[ "$version" =~ (pypy-)?([0-9\.]+) ]]; then
        echo ${BASH_REMATCH[2]}
    fi
}

255
256
257
258
function install_macpython {
    # Install Python and set $PYTHON_EXE to the installed executable
    # Parameters:
    #     $version : [implementation-]major[.minor[.patch]]
mattip's avatar
mattip committed
259
    #         The Python implementation to install, e.g. "3.6", "pypy-5.4" or "pypy3.6-7.2"
260
    #     $py_osx_ver: {major.minor | not defined}
261
262
    #       if defined, the macOS version that CPython is built for, e.g.
    #       "10.6" or "10.9". Ignored for PyPy
263
    local version=$1
264
    local py_osx_ver=$2
265
266
    local impl=$(macpython_impl_for_version $version)
    if [[ "$impl" == "pp" ]]; then
mattip's avatar
mattip committed
267
        install_pypy $version
268
    elif [[ "$impl" == "cp" ]]; then
269
        local stripped_ver=$(strip_macpython_ver_prefix $version)
270
        install_mac_cpython $stripped_ver $py_osx_ver
Kyle Stewart's avatar
Kyle Stewart committed
271
    else
272
        echo "Unexpected Python impl: ${impl}"
Kyle Stewart's avatar
Kyle Stewart committed
273
274
275
276
        exit 1
    fi
}

277
function install_mac_cpython {
Matthew Brett's avatar
Matthew Brett committed
278
    # Installs Python.org Python
279
280
281
    # Parameters
    #   $py_version
    #       Version given in major or major.minor or major.minor.micro e.g
Andrew Murray's avatar
Andrew Murray committed
282
    #       "3" or "3.7" or "3.7.1".
283
    #   $py_osx_ver
284
    #       {major.minor | not defined}
285
    #       if defined, the macOS version that Python is built for, e.g.
286
    #        "10.6" or "10.9"
287
    # sets $PYTHON_EXE variable to Python executable
Matthew Brett's avatar
Matthew Brett committed
288
    local py_version=$(fill_pyver $1)
Rob Buckley's avatar
Rob Buckley committed
289
    local py_osx_ver=$2
290
    local py_stripped=$(strip_ver_suffix $py_version)
291
    local py_inst=$(pyinst_fname_for_version $py_version $py_osx_ver)
Matthew Brett's avatar
Matthew Brett committed
292
293
    local inst_path=$DOWNLOADS_SDIR/$py_inst
    mkdir -p $DOWNLOADS_SDIR
294
    curl $MACPYTHON_URL/$py_stripped/${py_inst} > $inst_path
295
    if [ "${py_inst: -3}" == "dmg" ]; then
Matthew Brett's avatar
Matthew Brett committed
296
297
298
299
300
301
        hdiutil attach $inst_path -mountpoint /Volumes/Python
        inst_path=/Volumes/Python/Python.mpkg
    fi
    sudo installer -pkg $inst_path -target /
    local py_mm=${py_version:0:3}
    PYTHON_EXE=$MACPYTHON_PY_PREFIX/$py_mm/bin/python$py_mm
302
303
304
305
306
    # Install certificates for Python 3.6
    local inst_cmd="/Applications/Python ${py_mm}/Install Certificates.command"
    if [ -e "$inst_cmd" ]; then
        sh "$inst_cmd"
    fi
Matthew Brett's avatar
Matthew Brett committed
307
308
309
310
311
312
313
314
315
316
317
}

function install_virtualenv {
    # Generic install of virtualenv
    # Installs virtualenv into python given by $PYTHON_EXE
    # Assumes virtualenv will be installed into same directory as $PYTHON_EXE
    check_pip
    # Travis VMS install virtualenv for system python by default - force
    # install even if installed already
    $PIP_CMD install virtualenv --ignore-installed
    check_python
318
    VIRTUALENV_CMD="$(dirname $PYTHON_EXE)/virtualenv"
Matthew Brett's avatar
Matthew Brett committed
319
320
321
322
323
324
325
326
327
328
329
330
331
}

function make_workon_venv {
    # Make a virtualenv in given directory ('venv' default)
    # Set $PYTHON_EXE, $PIP_CMD to virtualenv versions
    # Parameter $venv_dir
    #    directory for virtualenv
    local venv_dir=$1
    if [ -z "$venv_dir" ]; then
        venv_dir="venv"
    fi
    venv_dir=`abspath $venv_dir`
    check_python
xoviat's avatar
xoviat committed
332
    $PYTHON_EXE -m virtualenv $venv_dir
Matthew Brett's avatar
Matthew Brett committed
333
334
335
336
337
338
    PYTHON_EXE=$venv_dir/bin/python
    PIP_CMD=$venv_dir/bin/pip
}

function remove_travis_ve_pip {
    # Remove travis installs of virtualenv and pip
xoviat's avatar
xoviat committed
339
340
    # FIXME: What if virtualenv is installed but pip is not?
    if [ "$(sudo which virtualenv)" == /usr/local/bin/virtualenv ] && [ "$(sudo which pip)" == /usr/local/bin/pip ]; then
Matthew Brett's avatar
Matthew Brett committed
341
342
343
344
345
346
347
348
        sudo pip uninstall -y virtualenv;
    fi
    if [ "$(sudo which pip)" == /usr/local/bin/pip ]; then
        sudo pip uninstall -y pip;
    fi
}

function set_py_vars {
349
    # Used by terryfy project; left here for back-compatibility
Matthew Brett's avatar
Matthew Brett committed
350
351
352
353
    export PATH="`dirname $PYTHON_EXE`:$PATH"
    export PYTHON_EXE PIP_CMD
}

354
function get_macpython_environment {
Matthew Brett's avatar
Matthew Brett committed
355
356
    # Set up MacPython environment
    # Parameters:
357
358
    #     $version : [implementation-]major[.minor[.patch]]
    #         The Python implementation to install, e.g. "3.6" or "pypy-5.4"
Matthew Brett's avatar
Matthew Brett committed
359
360
361
    #     $venv_dir : {directory_name|not defined}
    #         If defined - make virtualenv in this directory, set python / pip
    #         commands accordingly
362
    #     $py_osx_ver: {major.minor | not defined}
363
364
    #         if defined, the macOS version that Python is built for, e.g.
    #         "10.6" or "10.9", if not defined, use the version from MB_PYTHON_OSX_VER
Matthew Brett's avatar
Matthew Brett committed
365
366
367
368
369
370
    #
    # Installs Python
    # Sets $PYTHON_EXE to path to Python executable
    # Sets $PIP_CMD to full command for pip (including sudo if necessary)
    # If $venv_dir defined, Sets $VIRTUALENV_CMD to virtualenv executable
    # Puts directory of $PYTHON_EXE on $PATH
371
372
    local version=$1
    local venv_dir=$2
373
    local py_osx_ver=${3:-$MB_PYTHON_OSX_VER}
xoviat's avatar
xoviat committed
374
375
376
377

    if [ "$USE_CCACHE" == "1" ]; then
        activate_ccache
    fi
378

Matthew Brett's avatar
Matthew Brett committed
379
    remove_travis_ve_pip
380
    install_macpython $version $py_osx_ver
Matthew Brett's avatar
Matthew Brett committed
381
    install_pip
xoviat's avatar
xoviat committed
382

Matthew Brett's avatar
Matthew Brett committed
383
384
385
    if [ -n "$venv_dir" ]; then
        install_virtualenv
        make_workon_venv $venv_dir
386
387
388
        source $venv_dir/bin/activate
    else
        export PATH="`dirname $PYTHON_EXE`:$PATH"
Matthew Brett's avatar
Matthew Brett committed
389
    fi
390
    export PYTHON_EXE PIP_CMD
Matthew Brett's avatar
Matthew Brett committed
391
}
392

393
function install_delocate {
394
395
    check_pip
    $PIP_CMD install delocate
396
397
398
399
400
}

function repair_wheelhouse {
    local wheelhouse=$1
    install_delocate
401
402
    delocate-wheel $wheelhouse/*.whl # copies library dependencies into wheel
}
403
404
405
406
407
408
409

function install_pkg_config {
    # Install pkg-config avoiding error from homebrew
    # See :
    # https://github.com/matthew-brett/multibuild/issues/24#issue-221951587
    command -v pkg-config > /dev/null 2>&1 || brew install pkg-config
}
xoviat's avatar
xoviat committed
410
411

function activate_ccache {
xoviat's avatar
xoviat committed
412

xoviat's avatar
xoviat committed
413
414
415
416
417
    brew install ccache
    export PATH=/usr/local/opt/ccache/libexec:$PATH
    export CCACHE_CPP2=1

    # Prove to the developer that ccache is activated
xoviat's avatar
xoviat committed
418
    echo "Using C compiler: $(which clang)"
xoviat's avatar
xoviat committed
419
}