test.sh 13.7 KB
Newer Older
Guolin Ke's avatar
Guolin Ke committed
1
2
#!/bin/bash

3
4
5
6
7
8
9
set -e -E -o -u pipefail

# defaults
IN_UBUNTU_BASE_CONTAINER=${IN_UBUNTU_BASE_CONTAINER:-"false"}
METHOD=${METHOD:-""}
PRODUCES_ARTIFACTS=${PRODUCES_ARTIFACTS:-"false"}
SANITIZERS=${SANITIZERS:-""}
10

11
12
ARCH=$(uname -m)

13
if [[ $OS_NAME == "macos" ]] && [[ $COMPILER == "gcc" ]]; then
14
15
    export CXX=g++-11
    export CC=gcc-11
16
elif [[ $OS_NAME == "linux" ]] && [[ $COMPILER == "clang" ]]; then
17
18
    export CXX=clang++
    export CC=clang
19
20
21
elif [[ $OS_NAME == "linux" ]] && [[ $COMPILER == "clang-17" ]]; then
    export CXX=clang++-17
    export CC=clang-17
Guolin Ke's avatar
Guolin Ke committed
22
23
fi

24
25
26
27
28
if [[ $IN_UBUNTU_BASE_CONTAINER == "true" ]]; then
    export LANG="en_US.UTF-8"
    export LC_ALL="en_US.UTF-8"
fi

29
if [[ "${TASK}" == "r-package" ]] || [[ "${TASK}" == "r-rchk" ]]; then
30
    bash ${BUILD_DIRECTORY}/.ci/test_r_package.sh || exit 1
31
32
33
    exit 0
fi

34
if [[ "$TASK" == "cpp-tests" ]]; then
35
36
37
38
39
40
41
42
    if [[ $METHOD == "with-sanitizers" ]]; then
        extra_cmake_opts="-DUSE_SANITIZER=ON"
        if [[ -n $SANITIZERS ]]; then
            extra_cmake_opts="$extra_cmake_opts -DENABLED_SANITIZERS=$SANITIZERS"
        fi
    else
        extra_cmake_opts=""
    fi
43
44
45
    cmake -B build -S . -DBUILD_CPP_TEST=ON -DUSE_OPENMP=OFF -DUSE_DEBUG=ON $extra_cmake_opts
    cmake --build build --target testlightgbm -j4 || exit 1
    ./testlightgbm || exit 1
46
47
48
    exit 0
fi

49
50
51
CONDA_PYTHON_REQUIREMENT="python=$PYTHON_VERSION[build=*cpython]"

if [[ $TASK == "if-else" ]]; then
52
    mamba create -q -y -n $CONDA_ENV ${CONDA_PYTHON_REQUIREMENT} numpy
53
    source activate $CONDA_ENV
54
55
    cmake -B build -S . || exit 1
    cmake --build build --target lightgbm -j4 || exit 1
56
57
    cd $BUILD_DIRECTORY/tests/cpp_tests && ../../lightgbm config=train.conf convert_model_language=cpp convert_model=../../src/boosting/gbdt_prediction.cpp && ../../lightgbm config=predict.conf output_result=origin.pred || exit 1
    cd $BUILD_DIRECTORY/tests/cpp_tests && ../../lightgbm config=predict.conf output_result=ifelse.pred && python test.py || exit 1
58
59
60
61
    exit 0
fi

if [[ $TASK == "swig" ]]; then
62
    cmake -B build -S . -DUSE_SWIG=ON
63
    cmake --build build -j4 || exit 1
64
    if [[ $OS_NAME == "linux" ]] && [[ $COMPILER == "gcc" ]]; then
65
66
67
        objdump -T $BUILD_DIRECTORY/lib_lightgbm.so > $BUILD_DIRECTORY/objdump.log || exit 1
        objdump -T $BUILD_DIRECTORY/lib_lightgbm_swig.so >> $BUILD_DIRECTORY/objdump.log || exit 1
        python $BUILD_DIRECTORY/helpers/check_dynamic_dependencies.py $BUILD_DIRECTORY/objdump.log || exit 1
68
69
70
71
72
73
74
75
    fi
    if [[ $PRODUCES_ARTIFACTS == "true" ]]; then
        cp $BUILD_DIRECTORY/build/lightgbmlib.jar $BUILD_ARTIFACTSTAGINGDIRECTORY/lightgbmlib_$OS_NAME.jar
    fi
    exit 0
fi

if [[ $TASK == "lint" ]]; then
76
    cd ${BUILD_DIRECTORY}
77
    mamba create -q -y -n $CONDA_ENV \
78
        ${CONDA_PYTHON_REQUIREMENT} \
79
80
        'cmakelint>=1.4.2' \
        'cpplint>=1.6.0' \
81
        'matplotlib>=3.8.3' \
82
        'mypy>=1.8.0' \
83
        'pre-commit>=3.6.0' \
84
        'pyarrow>=14.0' \
85
        'r-lintr>=3.1'
86
87
    source activate $CONDA_ENV
    echo "Linting Python code"
88
    bash ${BUILD_DIRECTORY}/.ci/lint-python.sh || exit 1
89
    echo "Linting R code"
90
    Rscript ${BUILD_DIRECTORY}/.ci/lint_r_code.R ${BUILD_DIRECTORY} || exit 1
91
    echo "Linting C++ code"
92
    bash ${BUILD_DIRECTORY}/.ci/lint-cpp.sh || exit 1
93
94
95
    exit 0
fi

96
if [[ $TASK == "check-docs" ]] || [[ $TASK == "check-links" ]]; then
97
    cd $BUILD_DIRECTORY/docs
98
    mamba env create \
99
        -n $CONDA_ENV \
100
        --file ./env.yml || exit 1
101
    mamba install \
102
103
104
        -q \
        -y \
        -n $CONDA_ENV \
105
106
            'doxygen>=1.10.0' \
            'rstcheck>=6.2.0' || exit 1
107
    source activate $CONDA_ENV
108
109
    # check reStructuredText formatting
    cd $BUILD_DIRECTORY/python-package
110
    rstcheck --report-level warning $(find . -type f -name "*.rst") || exit 1
111
    cd $BUILD_DIRECTORY/docs
112
    rstcheck --report-level warning --ignore-directives=autoclass,autofunction,autosummary,doxygenfile $(find . -type f -name "*.rst") || exit 1
113
    # build docs
114
    make html || exit 1
115
116
    if [[ $TASK == "check-links" ]]; then
        # check docs for broken links
117
        pip install --user linkchecker
118
        linkchecker --config=.linkcheckerrc ./_build/html/*.html || exit 1
119
120
        exit 0
    fi
121
122
123
    # check the consistency of parameters' descriptions and other stuff
    cp $BUILD_DIRECTORY/docs/Parameters.rst $BUILD_DIRECTORY/docs/Parameters-backup.rst
    cp $BUILD_DIRECTORY/src/io/config_auto.cpp $BUILD_DIRECTORY/src/io/config_auto-backup.cpp
124
125
126
    python $BUILD_DIRECTORY/helpers/parameter_generator.py || exit 1
    diff $BUILD_DIRECTORY/docs/Parameters-backup.rst $BUILD_DIRECTORY/docs/Parameters.rst || exit 1
    diff $BUILD_DIRECTORY/src/io/config_auto-backup.cpp $BUILD_DIRECTORY/src/io/config_auto.cpp || exit 1
127
128
    exit 0
fi
129

130
131
132
# older versions of Dask are incompatible with pandas>=2.0, but not all conda packages' metadata accurately reflects that
#
# ref: https://github.com/microsoft/LightGBM/issues/6030
133
CONSTRAINED_DEPENDENCIES="'dask>=2023.5.0' 'distributed>=2023.5.0' 'pandas>=2.0' python-graphviz"
134
if [[ $PYTHON_VERSION == "3.7" ]]; then
135
    CONSTRAINED_DEPENDENCIES="'dask' 'distributed' 'python-graphviz<0.20.2' 'pandas<2.0'"
136
137
fi

138
# including python=version[build=*cpython] to ensure that conda doesn't fall back to pypy
139
mamba create -q -y -n $CONDA_ENV \
140
    ${CONSTRAINED_DEPENDENCIES} \
141
    cffi \
142
143
144
145
146
    cloudpickle \
    joblib \
    matplotlib \
    numpy \
    psutil \
147
    pyarrow \
148
    pytest \
149
    ${CONDA_PYTHON_REQUIREMENT} \
150
    scikit-learn \
151
    scipy || exit 1
152

153
154
155
156
source activate $CONDA_ENV

cd $BUILD_DIRECTORY

157
if [[ $OS_NAME == "macos" ]] && [[ $COMPILER == "clang" ]]; then
158
    # fix "OMP: Error #15: Initializing libiomp5.dylib, but found libomp.dylib already initialized." (OpenMP library conflict due to conda's MKL)
159
    for LIBOMP_ALIAS in libgomp.dylib libiomp5.dylib libomp.dylib; do sudo ln -sf "$(brew --cellar libomp)"/*/lib/libomp.dylib $CONDA_PREFIX/lib/$LIBOMP_ALIAS || exit 1; done
160
161
fi

Guolin Ke's avatar
Guolin Ke committed
162
if [[ $TASK == "sdist" ]]; then
163
164
165
    cd $BUILD_DIRECTORY && sh ./build-python.sh sdist || exit 1
    sh $BUILD_DIRECTORY/.ci/check_python_dists.sh $BUILD_DIRECTORY/dist || exit 1
    pip install --user $BUILD_DIRECTORY/dist/lightgbm-$LGB_VER.tar.gz -v || exit 1
166
    if [[ $PRODUCES_ARTIFACTS == "true" ]]; then
167
        cp $BUILD_DIRECTORY/dist/lightgbm-$LGB_VER.tar.gz $BUILD_ARTIFACTSTAGINGDIRECTORY || exit 1
168
    fi
169
    pytest $BUILD_DIRECTORY/tests/python_package_test || exit 1
Guolin Ke's avatar
Guolin Ke committed
170
171
    exit 0
elif [[ $TASK == "bdist" ]]; then
172
    if [[ $OS_NAME == "macos" ]]; then
173
174
        cd $BUILD_DIRECTORY && sh ./build-python.sh bdist_wheel || exit 1
        sh $BUILD_DIRECTORY/.ci/check_python_dists.sh $BUILD_DIRECTORY/dist || exit 1
175
        mv \
176
            ./dist/*.whl \
177
            ./dist/tmp.whl || exit 1
178
179
180
181
182
183
        if [[ $ARCH == "x86_64" ]]; then
            PLATFORM="macosx_10_15_x86_64.macosx_11_6_x86_64.macosx_12_5_x86_64"
        else
            echo "ERROR: macos wheels not supported yet on architecture '${ARCH}'"
            exit 1
        fi
184
185
        mv \
            ./dist/tmp.whl \
186
            dist/lightgbm-$LGB_VER-py3-none-$PLATFORM.whl || exit 1
187
        if [[ $PRODUCES_ARTIFACTS == "true" ]]; then
188
            cp dist/lightgbm-$LGB_VER-py3-none-macosx*.whl $BUILD_ARTIFACTSTAGINGDIRECTORY || exit 1
189
        fi
Guolin Ke's avatar
Guolin Ke committed
190
    else
191
        if [[ $ARCH == "x86_64" ]]; then
192
            PLATFORM="manylinux_2_28_x86_64"
193
194
195
        else
            PLATFORM="manylinux2014_$ARCH"
        fi
196
        cd $BUILD_DIRECTORY && sh ./build-python.sh bdist_wheel --integrated-opencl || exit 1
197
198
        mv \
            ./dist/*.whl \
199
            ./dist/tmp.whl || exit 1
200
201
        mv \
            ./dist/tmp.whl \
202
203
            ./dist/lightgbm-$LGB_VER-py3-none-$PLATFORM.whl || exit 1
        sh $BUILD_DIRECTORY/.ci/check_python_dists.sh $BUILD_DIRECTORY/dist || exit 1
204
        if [[ $PRODUCES_ARTIFACTS == "true" ]]; then
205
            cp dist/lightgbm-$LGB_VER-py3-none-$PLATFORM.whl $BUILD_ARTIFACTSTAGINGDIRECTORY || exit 1
206
        fi
207
208
        # Make sure we can do both CPU and GPU; see tests/python_package_test/test_dual.py
        export LIGHTGBM_TEST_DUAL_CPU_GPU=1
Guolin Ke's avatar
Guolin Ke committed
209
    fi
210
211
    pip install --user $BUILD_DIRECTORY/dist/*.whl || exit 1
    pytest $BUILD_DIRECTORY/tests || exit 1
Guolin Ke's avatar
Guolin Ke committed
212
213
214
215
    exit 0
fi

if [[ $TASK == "gpu" ]]; then
216
    sed -i'.bak' 's/std::string device_type = "cpu";/std::string device_type = "gpu";/' $BUILD_DIRECTORY/include/LightGBM/config.h
217
    grep -q 'std::string device_type = "gpu"' $BUILD_DIRECTORY/include/LightGBM/config.h || exit 1  # make sure that changes were really done
Guolin Ke's avatar
Guolin Ke committed
218
    if [[ $METHOD == "pip" ]]; then
219
220
        cd $BUILD_DIRECTORY && sh ./build-python.sh sdist || exit 1
        sh $BUILD_DIRECTORY/.ci/check_python_dists.sh $BUILD_DIRECTORY/dist || exit 1
221
222
223
        pip install \
            --user \
            -v \
224
            --config-settings=cmake.define.USE_GPU=ON \
225
            $BUILD_DIRECTORY/dist/lightgbm-$LGB_VER.tar.gz \
226
227
        || exit 1
        pytest $BUILD_DIRECTORY/tests/python_package_test || exit 1
Guolin Ke's avatar
Guolin Ke committed
228
        exit 0
229
    elif [[ $METHOD == "wheel" ]]; then
230
231
232
233
        cd $BUILD_DIRECTORY && sh ./build-python.sh bdist_wheel --gpu || exit 1
        sh $BUILD_DIRECTORY/.ci/check_python_dists.sh $BUILD_DIRECTORY/dist || exit 1
        pip install --user $BUILD_DIRECTORY/dist/lightgbm-$LGB_VER*.whl -v || exit 1
        pytest $BUILD_DIRECTORY/tests || exit 1
234
235
        exit 0
    elif [[ $METHOD == "source" ]]; then
236
        cmake -B build -S . -DUSE_GPU=ON
Guolin Ke's avatar
Guolin Ke committed
237
    fi
238
239
elif [[ $TASK == "cuda" ]]; then
    sed -i'.bak' 's/std::string device_type = "cpu";/std::string device_type = "cuda";/' $BUILD_DIRECTORY/include/LightGBM/config.h
240
    grep -q 'std::string device_type = "cuda"' $BUILD_DIRECTORY/include/LightGBM/config.h || exit 1  # make sure that changes were really done
241
242
    # by default ``gpu_use_dp=false`` for efficiency. change to ``true`` here for exact results in ci tests
    sed -i'.bak' 's/gpu_use_dp = false;/gpu_use_dp = true;/' $BUILD_DIRECTORY/include/LightGBM/config.h
243
    grep -q 'gpu_use_dp = true' $BUILD_DIRECTORY/include/LightGBM/config.h || exit 1  # make sure that changes were really done
244
    if [[ $METHOD == "pip" ]]; then
245
246
        cd $BUILD_DIRECTORY && sh ./build-python.sh sdist || exit 1
        sh $BUILD_DIRECTORY/.ci/check_python_dists.sh $BUILD_DIRECTORY/dist || exit 1
247
248
249
        pip install \
            --user \
            -v \
250
            --config-settings=cmake.define.USE_CUDA=ON \
251
            $BUILD_DIRECTORY/dist/lightgbm-$LGB_VER.tar.gz \
252
253
        || exit 1
        pytest $BUILD_DIRECTORY/tests/python_package_test || exit 1
254
        exit 0
255
    elif [[ $METHOD == "wheel" ]]; then
256
257
258
259
        cd $BUILD_DIRECTORY && sh ./build-python.sh bdist_wheel --cuda || exit 1
        sh $BUILD_DIRECTORY/.ci/check_python_dists.sh $BUILD_DIRECTORY/dist || exit 1
        pip install --user $BUILD_DIRECTORY/dist/lightgbm-$LGB_VER*.whl -v || exit 1
        pytest $BUILD_DIRECTORY/tests || exit 1
260
261
        exit 0
    elif [[ $METHOD == "source" ]]; then
262
        cmake -B build -S . -DUSE_CUDA=ON
263
    fi
264
elif [[ $TASK == "mpi" ]]; then
265
    if [[ $METHOD == "pip" ]]; then
266
267
        cd $BUILD_DIRECTORY && sh ./build-python.sh sdist || exit 1
        sh $BUILD_DIRECTORY/.ci/check_python_dists.sh $BUILD_DIRECTORY/dist || exit 1
268
269
270
        pip install \
            --user \
            -v \
271
            --config-settings=cmake.define.USE_MPI=ON \
272
            $BUILD_DIRECTORY/dist/lightgbm-$LGB_VER.tar.gz \
273
274
        || exit 1
        pytest $BUILD_DIRECTORY/tests/python_package_test || exit 1
275
        exit 0
276
    elif [[ $METHOD == "wheel" ]]; then
277
278
279
280
        cd $BUILD_DIRECTORY && sh ./build-python.sh bdist_wheel --mpi || exit 1
        sh $BUILD_DIRECTORY/.ci/check_python_dists.sh $BUILD_DIRECTORY/dist || exit 1
        pip install --user $BUILD_DIRECTORY/dist/lightgbm-$LGB_VER*.whl -v || exit 1
        pytest $BUILD_DIRECTORY/tests || exit 1
281
282
        exit 0
    elif [[ $METHOD == "source" ]]; then
283
        cmake -B build -S . -DUSE_MPI=ON -DUSE_DEBUG=ON
284
    fi
Guolin Ke's avatar
Guolin Ke committed
285
else
286
    cmake -B build -S .
Guolin Ke's avatar
Guolin Ke committed
287
288
fi

289
cmake --build build --target _lightgbm -j4 || exit 1
Guolin Ke's avatar
Guolin Ke committed
290

291
292
cd $BUILD_DIRECTORY && sh ./build-python.sh install --precompile --user || exit 1
pytest $BUILD_DIRECTORY/tests || exit 1
Guolin Ke's avatar
Guolin Ke committed
293
294

if [[ $TASK == "regular" ]]; then
295
    if [[ $PRODUCES_ARTIFACTS == "true" ]]; then
296
        if [[ $OS_NAME == "macos" ]]; then
297
            cp $BUILD_DIRECTORY/lib_lightgbm.dylib $BUILD_ARTIFACTSTAGINGDIRECTORY/lib_lightgbm.dylib
298
        else
299
            if [[ $COMPILER == "gcc" ]]; then
300
301
                objdump -T $BUILD_DIRECTORY/lib_lightgbm.so > $BUILD_DIRECTORY/objdump.log || exit 1
                python $BUILD_DIRECTORY/helpers/check_dynamic_dependencies.py $BUILD_DIRECTORY/objdump.log || exit 1
302
            fi
303
304
            cp $BUILD_DIRECTORY/lib_lightgbm.so $BUILD_ARTIFACTSTAGINGDIRECTORY/lib_lightgbm.so
        fi
Guolin Ke's avatar
Guolin Ke committed
305
    fi
306
    cd $BUILD_DIRECTORY/examples/python-guide
Guolin Ke's avatar
Guolin Ke committed
307
308
309
310
    sed -i'.bak' '/import lightgbm as lgb/a\
import matplotlib\
matplotlib.use\(\"Agg\"\)\
' plot_example.py  # prevent interactive window mode
311
    sed -i'.bak' 's/graph.render(view=True)/graph.render(view=False)/' plot_example.py
312
    # requirements for examples
313
    mamba install -q -y -n $CONDA_ENV \
314
315
316
        h5py \
        ipywidgets \
        notebook
317
    for f in *.py **/*.py; do python $f || exit 1; done  # run all examples
318
    cd $BUILD_DIRECTORY/examples/python-guide/notebooks
319
    sed -i'.bak' 's/INTERACTIVE = False/assert False, \\"Interactive mode disabled\\"/' interactive_plot_example.ipynb
320
    jupyter nbconvert --ExecutePreprocessor.timeout=180 --to notebook --execute --inplace *.ipynb || exit 1  # run all notebooks
321
322

    # importing the library should succeed even if all optional dependencies are not present
323
    conda uninstall -n $CONDA_ENV --force --yes \
324
        cffi \
325
        dask \
326
327
328
329
        distributed \
        joblib \
        matplotlib \
        psutil \
330
        pyarrow \
331
        python-graphviz \
332
333
        scikit-learn || exit 1
    python -c "import lightgbm" || exit 1
Guolin Ke's avatar
Guolin Ke committed
334
fi