build-python.sh 12.3 KB
Newer Older
1
2
3
4
5
#!/bin/sh

# [description]
#
#     Prepare a source distribution (sdist) or built distribution (wheel)
6
#     of the Python-package, and optionally install it.
7
8
9
10
11
12
13
14
15
#
# [usage]
#
#     # build sdist and put it in dist/
#     sh ./build-python.sh sdist
#
#     # build wheel and put it in dist/
#     sh ./build-python.sh bdist_wheel [OPTIONS]
#
16
#     # compile lib_lightgbm and install the Python-package wrapping it
17
18
#     sh ./build-python.sh install [OPTIONS]
#
19
#     # install the Python-package using a pre-compiled lib_lightgbm
20
21
22
23
24
#     # (assumes lib_lightgbm.{dll,so} is located at the root of the repo)
#     sh ./build-python.sh install --precompile
#
# [options]
#
25
26
#     --boost-dir=FILEPATH
#                                   Directory with Boost package configuration file.
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
#     --boost-include-dir=FILEPATH
#                                   Directory containing Boost headers.
#     --boost-librarydir=FILEPATH
#                                   Preferred Boost library directory.
#     --boost-root=FILEPATH
#                                   Boost preferred installation prefix.
#     --opencl-include-dir=FILEPATH
#                                   OpenCL include directory.
#     --opencl-library=FILEPATH
#                                   Path to OpenCL library.
#     --bit32
#                                   Compile 32-bit version.
#     --cuda
#                                   Compile CUDA version.
#     --gpu
#                                   Compile GPU version.
#     --integrated-opencl
#                                   Compile integrated OpenCL version.
#     --mingw
#                                   Compile with MinGW.
#     --mpi
#                                   Compile MPI version.
49
50
51
#     --no-isolation
#                                   Assume all build and install dependencies are already installed,
#                                   don't go to the internet to get them.
52
53
54
55
56
57
58
59
60
61
62
#     --nomp
#                                   Compile version without OpenMP support.
#     --precompile
#                                   Use precompiled library.
#                                   Only used with 'install' command.
#     --time-costs
#                                   Output time costs for different internal routines.
#     --user
#                                   Install into user-specific instead of global site-packages directory.
#                                   Only used with 'install' command.

63
set -e -u
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

echo "building lightgbm"

# Default values of arguments
INSTALL="false"
BUILD_SDIST="false"
BUILD_WHEEL="false"

PIP_INSTALL_ARGS=""
BUILD_ARGS=""
PRECOMPILE="false"

while [ $# -gt 0 ]; do
  case "$1" in
    ############################
    # sub-commands of setup.py #
    ############################
    install)
      INSTALL="true"
      ;;
    sdist)
      BUILD_SDIST="true"
      ;;
    bdist_wheel)
      BUILD_WHEEL="true"
      ;;
    ############################
    # customized library paths #
    ############################
93
    --boost-dir|--boost-dir=*)
94
        if echo "$1" | grep -q '^*=*$';
95
96
97
            then shift;
        fi
        BOOST_DIR="${1#*=}"
98
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.Boost_DIR='${BOOST_DIR}'"
99
        ;;
100
    --boost-include-dir|--boost-include-dir=*)
101
        if echo "$1" | grep -q '^*=*$';
102
103
104
            then shift;
        fi
        BOOST_INCLUDE_DIR="${1#*=}"
105
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.Boost_INCLUDE_DIR='${BOOST_INCLUDE_DIR}'"
106
107
        ;;
    --boost-librarydir|--boost-librarydir=*)
108
        if echo "$1" | grep -q '^*=*$';
109
110
111
            then shift;
        fi
        BOOST_LIBRARY_DIR="${1#*=}"
112
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.BOOST_LIBRARYDIR='${BOOST_LIBRARY_DIR}'"
113
114
        ;;
    --boost-root|--boost-root=*)
115
        if echo "$1" | grep -q '^*=*$';
116
117
118
            then shift;
        fi
        BOOST_ROOT="${1#*=}"
119
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.Boost_ROOT='${BOOST_ROOT}'"
120
121
        ;;
    --opencl-include-dir|--opencl-include-dir=*)
122
        if echo "$1" | grep -q '^*=*$';
123
124
125
            then shift;
        fi
        OPENCL_INCLUDE_DIR="${1#*=}"
126
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.OpenCL_INCLUDE_DIR='${OPENCL_INCLUDE_DIR}'"
127
128
        ;;
    --opencl-library|--opencl-library=*)
129
        if echo "$1" | grep -q '^*=*$';
130
131
132
            then shift;
        fi
        OPENCL_LIBRARY="${1#*=}"
133
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.OpenCL_LIBRARY='${OPENCL_LIBRARY}'"
134
135
136
137
138
        ;;
    #########
    # flags #
    #########
    --bit32)
139
140
141
        export CMAKE_GENERATOR="Visual Studio 17 2022"
        export CMAKE_GENERATOR_PLATFORM="Win32"
        echo "[INFO] Attempting to build 32-bit version of LightGBM, which is only supported on Windows with generator '${CMAKE_GENERATOR}'."
142
143
        ;;
    --cuda)
144
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.USE_CUDA=ON"
145
146
        ;;
    --gpu)
147
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.USE_GPU=ON"
148
149
        ;;
    --integrated-opencl)
150
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.__INTEGRATE_OPENCL=ON"
151
152
        ;;
    --mingw)
153
154
155
        export CMAKE_GENERATOR='MinGW Makefiles'
        # ref: https://stackoverflow.com/a/45104058/3986677
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.CMAKE_SH=CMAKE_SH-NOTFOUND"
156
157
        ;;
    --mpi)
158
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.USE_MPI=ON"
159
        ;;
160
161
162
163
    --no-isolation)
        BUILD_ARGS="${BUILD_ARGS} --no-isolation"
        PIP_INSTALL_ARGS="${PIP_INSTALL_ARGS} --no-build-isolation"
        ;;
164
    --nomp)
165
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.USE_OPENMP=OFF"
166
167
168
169
170
        ;;
    --precompile)
        PRECOMPILE="true"
        ;;
    --time-costs)
171
        BUILD_ARGS="${BUILD_ARGS} --config-setting=cmake.define.USE_TIMETAG=ON"
172
173
174
175
176
177
        ;;
    --user)
        PIP_INSTALL_ARGS="${PIP_INSTALL_ARGS} --user"
        ;;
    *)
        echo "invalid argument '${1}'"
178
        exit 1
179
180
181
182
183
        ;;
  esac
  shift
done

184
185
pip install --prefer-binary 'build>=0.10.0'

186
# create a new directory that just contains the files needed
187
# to build the Python-package
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
create_isolated_source_dir() {
    rm -rf \
        ./lightgbm-python \
        ./lightgbm \
        ./python-package/build \
        ./python-package/build_cpp \
        ./python-package/compile \
        ./python-package/dist \
        ./python-package/lightgbm.egg-info

    cp -R ./python-package ./lightgbm-python

    cp LICENSE ./lightgbm-python/
    cp VERSION.txt ./lightgbm-python/lightgbm/VERSION.txt

203
204
205
206
207
    cp -R ./cmake ./lightgbm-python
    cp CMakeLists.txt ./lightgbm-python
    cp -R ./include ./lightgbm-python
    cp -R ./src ./lightgbm-python
    cp -R ./swig ./lightgbm-python
208
209
210
211
212
213
214
215

    # include only specific files from external_libs, to keep the package
    # small and avoid redistributing code with licenses incompatible with
    # LightGBM's license

    ######################
    # fast_double_parser #
    ######################
216
    mkdir -p ./lightgbm-python/external_libs/fast_double_parser
217
218
    cp \
        external_libs/fast_double_parser/CMakeLists.txt \
219
        ./lightgbm-python/external_libs/fast_double_parser/CMakeLists.txt
220
221
    cp \
        external_libs/fast_double_parser/LICENSE* \
222
        ./lightgbm-python/external_libs/fast_double_parser/
223

224
    mkdir -p ./lightgbm-python/external_libs/fast_double_parser/include/
225
226
    cp \
        external_libs/fast_double_parser/include/fast_double_parser.h \
227
        ./lightgbm-python/external_libs/fast_double_parser/include/
228
229
230
231

    #######
    # fmt #
    #######
232
    mkdir -p ./lightgbm-python/external_libs/fmt
233
234
    cp \
        external_libs/fast_double_parser/CMakeLists.txt \
235
        ./lightgbm-python/external_libs/fmt/CMakeLists.txt
236
237
    cp \
        external_libs/fmt/LICENSE* \
238
        ./lightgbm-python/external_libs/fmt/
239

240
    mkdir -p ./lightgbm-python/external_libs/fmt/include/fmt
241
242
    cp \
        external_libs/fmt/include/fmt/*.h \
243
        ./lightgbm-python/external_libs/fmt/include/fmt/
244
245
246
247

    #########
    # Eigen #
    #########
248
    mkdir -p ./lightgbm-python/external_libs/eigen/Eigen
249
250
    cp \
        external_libs/eigen/CMakeLists.txt \
251
        ./lightgbm-python/external_libs/eigen/CMakeLists.txt
252
253
254
255

    modules="Cholesky Core Dense Eigenvalues Geometry Householder Jacobi LU QR SVD"
    for eigen_module in ${modules}; do
        cp \
256
257
258
259
            "external_libs/eigen/Eigen/${eigen_module}" \
            "./lightgbm-python/external_libs/eigen/Eigen/${eigen_module}"
        if [ "${eigen_module}" != "Dense" ]; then
            mkdir -p "./lightgbm-python/external_libs/eigen/Eigen/src/${eigen_module}/"
260
261
            cp \
                -R \
262
263
                "external_libs/eigen/Eigen/src/${eigen_module}"/* \
                "./lightgbm-python/external_libs/eigen/Eigen/src/${eigen_module}/"
264
265
266
        fi
    done

267
    mkdir -p ./lightgbm-python/external_libs/eigen/Eigen/misc
268
269
270
    cp \
        -R \
        external_libs/eigen/Eigen/src/misc \
271
        ./lightgbm-python/external_libs/eigen/Eigen/src/misc/
272

273
    mkdir -p ./lightgbm-python/external_libs/eigen/Eigen/plugins
274
275
276
    cp \
        -R \
        external_libs/eigen/Eigen/src/plugins \
277
        ./lightgbm-python/external_libs/eigen/Eigen/src/plugins/
278
279
280
281

    ###################
    # compute (Boost) #
    ###################
282
    mkdir -p ./lightgbm-python/external_libs/compute
283
284
285
    cp \
        -R \
        external_libs/compute/include \
286
        ./lightgbm-python/external_libs/compute/include/
287
288
289
290
291
292
293
294
295
}

create_isolated_source_dir

cd ./lightgbm-python

# installation involves building the wheel + `pip install`-ing it
if test "${INSTALL}" = true; then
    if test "${PRECOMPILE}" = true; then
296
297
298
299
300
301
302
303
304
        BUILD_SDIST=true
        BUILD_WHEEL=false
        BUILD_ARGS=""
        rm -rf \
            ./cmake \
            ./CMakeLists.txt \
            ./external_libs \
            ./include \
            ./src \
305
            ./swig
306
307
308
        # use regular-old setuptools for these builds, to avoid
        # trying to recompile the shared library
        sed -i.bak -e '/start:build-system/,/end:build-system/d' pyproject.toml
309
        # shellcheck disable=SC2129
310
311
312
313
        echo '[build-system]' >> ./pyproject.toml
        echo 'requires = ["setuptools"]' >> ./pyproject.toml
        echo 'build-backend = "setuptools.build_meta"' >> ./pyproject.toml
        echo "" >> ./pyproject.toml
314
        echo "recursive-include lightgbm *.dll *.dylib *.so" > ./MANIFEST.in
315
316
317
318
319
        echo "" >> ./MANIFEST.in
        mkdir -p ./lightgbm/lib
        if test -f ../lib_lightgbm.so; then
            echo "found pre-compiled lib_lightgbm.so"
            cp ../lib_lightgbm.so ./lightgbm/lib/lib_lightgbm.so
320
321
322
        elif test -f ../lib_lightgbm.dylib; then
            echo "found pre-compiled lib_lightgbm.dylib"
            cp ../lib_lightgbm.dylib ./lightgbm/lib/lib_lightgbm.dylib
323
324
325
326
327
328
329
330
331
        elif test -f ../Release/lib_lightgbm.dll; then
            echo "found pre-compiled Release/lib_lightgbm.dll"
            cp ../Release/lib_lightgbm.dll ./lightgbm/lib/lib_lightgbm.dll
        elif test -f ../windows/x64/DLL/lib_lightgbm.dll; then
            echo "found pre-compiled windows/x64/DLL/lib_lightgbm.dll"
            cp ../windows/x64/DLL/lib_lightgbm.dll ./lightgbm/lib/lib_lightgbm.dll
            cp ../windows/x64/DLL/lib_lightgbm.lib ./lightgbm/lib/lib_lightgbm.lib
        fi
        rm -f ./*.bak
332
333
334
335
336
337
338
339
340
    else
        BUILD_SDIST="false"
        BUILD_WHEEL="true"
    fi
fi

if test "${BUILD_SDIST}" = true; then
    echo "--- building sdist ---"
    rm -f ../dist/*.tar.gz
341
    # shellcheck disable=SC2086
342
343
344
    python -m build \
        --sdist \
        --outdir ../dist \
345
        ${BUILD_ARGS} \
346
        .
347
348
349
fi

if test "${BUILD_WHEEL}" = true; then
350
    echo "--- building wheel ---"
351
    rm -f ../dist/*.whl || true
352
    # shellcheck disable=SC2086
353
354
355
356
357
    python -m build \
        --wheel \
        --outdir ../dist \
        ${BUILD_ARGS} \
        .
358
359
360
361
362
fi

if test "${INSTALL}" = true; then
    echo "--- installing lightgbm ---"
    cd ../dist
363
    if test "${BUILD_WHEEL}" = true; then
364
        PACKAGE_NAME="$(echo lightgbm*.whl)"
365
    else
366
        PACKAGE_NAME="$(echo lightgbm*.tar.gz)"
367
    fi
368
    # ref for use of '--find-links': https://stackoverflow.com/a/52481267/3986677
369
    # shellcheck disable=SC2086
370
371
    pip install \
        ${PIP_INSTALL_ARGS} \
372
        --force-reinstall \
373
        --no-cache-dir \
374
        --no-deps \
375
        --find-links=. \
376
        "${PACKAGE_NAME}"
377
378
379
380
381
    cd ../
fi

echo "cleaning up"
rm -rf ./lightgbm-python