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

3
4
5
set -e -E -o -u pipefail

# defaults
6
CONDA_ENV="test-env"
7
8
9
10
IN_UBUNTU_BASE_CONTAINER=${IN_UBUNTU_BASE_CONTAINER:-"false"}
METHOD=${METHOD:-""}
PRODUCES_ARTIFACTS=${PRODUCES_ARTIFACTS:-"false"}
SANITIZERS=${SANITIZERS:-""}
11

12
13
ARCH=$(uname -m)

14
LGB_VER=$(head -n 1 "${BUILD_DIRECTORY}/VERSION.txt")
15

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

27
28
29
30
31
if [[ $IN_UBUNTU_BASE_CONTAINER == "true" ]]; then
    export LANG="en_US.UTF-8"
    export LC_ALL="en_US.UTF-8"
fi

32
33
34
35
36
37
38
39
40
41
42
43
44
# Setting MACOSX_DEPLOYMENT_TARGET prevents CMake from building against too-new
# macOS features, and helps tools like Python build tools determine the appropriate
# wheel compatibility tags.
#
# ref:
#   * https://cmake.org/cmake/help/latest/envvar/MACOSX_DEPLOYMENT_TARGET.html
#   * https://github.com/scikit-build/scikit-build-core/blob/acb7d0346e4a05bcb47a4ea3939c705ab71e3145/src/scikit_build_core/builder/macos.py#L36
if [[ $ARCH == "x86_64" ]]; then
    export MACOSX_DEPLOYMENT_TARGET=10.15
else
    export MACOSX_DEPLOYMENT_TARGET=12.0
fi

45
if [[ "${TASK}" == "r-package" ]]; then
46
    bash "${BUILD_DIRECTORY}/.ci/test-r-package.sh" || exit 1
47
48
49
    exit 0
fi

50
if [[ "$TASK" == "cpp-tests" ]]; then
51
52
53
54
    cmake_args=(
        -DBUILD_CPP_TEST=ON
        -DUSE_DEBUG=ON
    )
55
    if [[ $METHOD == "with-sanitizers" ]]; then
56
        cmake_args+=("-DUSE_SANITIZER=ON")
57
        if [[ -n $SANITIZERS ]]; then
58
            cmake_args+=("-DENABLED_SANITIZERS=$SANITIZERS")
59
60
        fi
    fi
61
    cmake -B build -S . "${cmake_args[@]}"
62
63
    cmake --build build --target testlightgbm -j4 || exit 1
    ./testlightgbm || exit 1
64
65
66
    exit 0
fi

67
68
69
# including python=version=[build=*_cp*] to ensure that conda prefers CPython and doesn't fall back to
# other implementations like pypy
CONDA_PYTHON_REQUIREMENT="python=${PYTHON_VERSION}[build=*_cp*]"
70
71

if [[ $TASK == "if-else" ]]; then
72
73
74
    conda create -q -y -n "${CONDA_ENV}" "${CONDA_PYTHON_REQUIREMENT}" numpy
    # shellcheck disable=SC1091
    source activate "${CONDA_ENV}"
75
76
    cmake -B build -S . || exit 1
    cmake --build build --target lightgbm -j4 || exit 1
77
78
79
80
81
    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
    ../../lightgbm config=predict.conf output_result=ifelse.pred
    python test.py
82
83
84
    exit 0
fi

85
86
cd "${BUILD_DIRECTORY}"

87
if [[ $TASK == "swig" ]]; then
88
    cmake -B build -S . -DUSE_SWIG=ON
89
    cmake --build build -j4 || exit 1
90
    if [[ $OS_NAME == "linux" ]] && [[ $COMPILER == "gcc" ]]; then
91
92
        objdump -T ./lib_lightgbm.so > ./objdump.log || exit 1
        objdump -T ./lib_lightgbm_swig.so >> ./objdump.log || exit 1
93
        python ./.ci/check-dynamic-dependencies.py ./objdump.log || exit 1
94
95
    fi
    if [[ $PRODUCES_ARTIFACTS == "true" ]]; then
96
        cp ./build/lightgbmlib.jar "${BUILD_ARTIFACTSTAGINGDIRECTORY}/lightgbmlib_${OS_NAME}.jar"
97
98
99
100
101
    fi
    exit 0
fi

if [[ $TASK == "lint" ]]; then
102
103
    pwsh -command "Install-Module -Name PSScriptAnalyzer -Scope CurrentUser -SkipPublisherCheck"
    echo "Linting PowerShell code"
104
    pwsh -file ./.ci/lint-powershell.ps1 || exit 1
105
106
    conda create -q -y -n "${CONDA_ENV}" \
        "${CONDA_PYTHON_REQUIREMENT}" \
107
        'biome>=1.9.3' \
108
        'cpplint>=1.6.0' \
109
110
111
112
        'matplotlib-base>=3.9.1' \
        'mypy>=1.11.1' \
        'pre-commit>=3.8.0' \
        'pyarrow-core>=17.0' \
113
        'scikit-learn>=1.5.2' \
114
        'r-lintr>=3.1.2'
115
116
    # shellcheck disable=SC1091
    source activate "${CONDA_ENV}"
117
118
    echo "Linting Python and bash code"
    bash ./.ci/lint-python-bash.sh || exit 1
119
    echo "Linting R code"
120
    Rscript ./.ci/lint-r-code.R "${BUILD_DIRECTORY}" || exit 1
121
    echo "Linting C++ code"
122
    bash ./.ci/lint-cpp.sh || exit 1
123
124
    echo "Linting JavaScript code"
    bash ./.ci/lint-js.sh || exit 1
125
126
127
    exit 0
fi

128
if [[ $TASK == "check-docs" ]] || [[ $TASK == "check-links" ]]; then
129
    conda env create \
130
131
        -n "${CONDA_ENV}" \
        --file ./docs/env.yml || exit 1
132
    conda install \
133
134
        -q \
        -y \
135
        -n "${CONDA_ENV}" \
136
            'doxygen>=1.10.0' \
137
            'rstcheck>=6.2.4' || exit 1
138
139
    # shellcheck disable=SC1091
    source activate "${CONDA_ENV}"
140
    # check reStructuredText formatting
141
142
143
144
    find "${BUILD_DIRECTORY}/python-package" -type f -name "*.rst" \
        -exec rstcheck --report-level warning {} \+ || exit 1
    find "${BUILD_DIRECTORY}/docs" -type f -name "*.rst" \
        -exec rstcheck --report-level warning --ignore-directives=autoclass,autofunction,autosummary,doxygenfile {} \+ || exit 1
145
    # build docs
146
    make -C docs html || exit 1
147
148
    if [[ $TASK == "check-links" ]]; then
        # check docs for broken links
James Lamb's avatar
James Lamb committed
149
150
        pip install 'linkchecker>=10.5.0'
        linkchecker --config=./docs/.linkcheckerrc ./docs/_build/html/*.html || exit 1
151
152
        exit 0
    fi
153
    # check the consistency of parameters' descriptions and other stuff
154
155
    cp ./docs/Parameters.rst ./docs/Parameters-backup.rst
    cp ./src/io/config_auto.cpp ./src/io/config_auto-backup.cpp
156
    python ./.ci/parameter-generator.py || exit 1
157
158
    diff ./docs/Parameters-backup.rst ./docs/Parameters.rst || exit 1
    diff ./src/io/config_auto-backup.cpp ./src/io/config_auto.cpp || exit 1
159
160
    exit 0
fi
161

162
163
if [[ $PYTHON_VERSION == "3.9" ]]; then
    CONDA_REQUIREMENT_FILE="${BUILD_DIRECTORY}/.ci/conda-envs/ci-core-py39.txt"
164
else
165
    CONDA_REQUIREMENT_FILE="${BUILD_DIRECTORY}/.ci/conda-envs/ci-core.txt"
166
167
fi

168
conda create \
169
    -y \
170
171
172
    -n "${CONDA_ENV}" \
    --file "${CONDA_REQUIREMENT_FILE}" \
    "${CONDA_PYTHON_REQUIREMENT}" \
173
|| exit 1
174

175
176
177
178
# print output of 'conda list', to help in submitting bug reports
echo "conda list:"
conda list -n ${CONDA_ENV}

179
# shellcheck disable=SC1091
180
181
source activate $CONDA_ENV

182
cd "${BUILD_DIRECTORY}"
183

Guolin Ke's avatar
Guolin Ke committed
184
if [[ $TASK == "sdist" ]]; then
185
    sh ./build-python.sh sdist || exit 1
186
    sh .ci/check-python-dists.sh ./dist || exit 1
187
    pip install "./dist/lightgbm-${LGB_VER}.tar.gz" -v || exit 1
188
    if [[ $PRODUCES_ARTIFACTS == "true" ]]; then
189
        cp "./dist/lightgbm-${LGB_VER}.tar.gz" "${BUILD_ARTIFACTSTAGINGDIRECTORY}" || exit 1
190
    fi
191
    pytest ./tests/python_package_test || exit 1
Guolin Ke's avatar
Guolin Ke committed
192
193
    exit 0
elif [[ $TASK == "bdist" ]]; then
194
    if [[ $OS_NAME == "macos" ]]; then
195
        sh ./build-python.sh bdist_wheel || exit 1
196
        sh .ci/check-python-dists.sh ./dist || exit 1
197
        if [[ $PRODUCES_ARTIFACTS == "true" ]]; then
198
            cp "$(echo "dist/lightgbm-${LGB_VER}-py3-none-macosx"*.whl)" "${BUILD_ARTIFACTSTAGINGDIRECTORY}" || exit 1
199
        fi
Guolin Ke's avatar
Guolin Ke committed
200
    else
201
        if [[ $ARCH == "x86_64" ]]; then
202
            PLATFORM="manylinux_2_28_x86_64"
203
204
205
        else
            PLATFORM="manylinux2014_$ARCH"
        fi
206
        sh ./build-python.sh bdist_wheel --integrated-opencl || exit 1
207
208
        # rename wheel, to fix scikit-build-core choosing the platform 'linux_aarch64' instead of
        # a manylinux tag
209
210
        mv \
            ./dist/*.whl \
211
            ./dist/tmp.whl || exit 1
212
213
        mv \
            ./dist/tmp.whl \
214
            "./dist/lightgbm-${LGB_VER}-py3-none-${PLATFORM}.whl" || exit 1
215
        sh .ci/check-python-dists.sh ./dist || exit 1
216
        if [[ $PRODUCES_ARTIFACTS == "true" ]]; then
217
            cp "dist/lightgbm-${LGB_VER}-py3-none-${PLATFORM}.whl" "${BUILD_ARTIFACTSTAGINGDIRECTORY}" || exit 1
218
        fi
219
220
        # 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
221
    fi
222
223
    pip install -v ./dist/*.whl || exit 1
    pytest ./tests || exit 1
Guolin Ke's avatar
Guolin Ke committed
224
225
226
227
    exit 0
fi

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

298
cmake --build build --target _lightgbm -j4 || exit 1
Guolin Ke's avatar
Guolin Ke committed
299

300
301
sh ./build-python.sh install --precompile || exit 1
pytest ./tests || exit 1
Guolin Ke's avatar
Guolin Ke committed
302
303

if [[ $TASK == "regular" ]]; then
304
    if [[ $PRODUCES_ARTIFACTS == "true" ]]; then
305
        if [[ $OS_NAME == "macos" ]]; then
306
            cp ./lib_lightgbm.dylib "${BUILD_ARTIFACTSTAGINGDIRECTORY}/lib_lightgbm.dylib"
307
        else
308
            if [[ $COMPILER == "gcc" ]]; then
309
                objdump -T ./lib_lightgbm.so > ./objdump.log || exit 1
310
                python ./.ci/check-dynamic-dependencies.py ./objdump.log || exit 1
311
            fi
312
            cp ./lib_lightgbm.so "${BUILD_ARTIFACTSTAGINGDIRECTORY}/lib_lightgbm.so"
313
        fi
Guolin Ke's avatar
Guolin Ke committed
314
    fi
315
    cd "$BUILD_DIRECTORY/examples/python-guide"
Guolin Ke's avatar
Guolin Ke committed
316
317
318
319
    sed -i'.bak' '/import lightgbm as lgb/a\
import matplotlib\
matplotlib.use\(\"Agg\"\)\
' plot_example.py  # prevent interactive window mode
320
    sed -i'.bak' 's/graph.render(view=True)/graph.render(view=False)/' plot_example.py
321
    # requirements for examples
322
    conda install -y -n $CONDA_ENV \
323
324
325
        'h5py>=3.10' \
        'ipywidgets>=8.1.2' \
        'notebook>=7.1.2'
326
    for f in *.py **/*.py; do python "${f}" || exit 1; done  # run all examples
327
    cd "$BUILD_DIRECTORY/examples/python-guide/notebooks"
328
    sed -i'.bak' 's/INTERACTIVE = False/assert False, \\"Interactive mode disabled\\"/' interactive_plot_example.ipynb
329
    jupyter nbconvert --ExecutePreprocessor.timeout=180 --to notebook --execute --inplace ./*.ipynb || exit 1  # run all notebooks
330
331

    # importing the library should succeed even if all optional dependencies are not present
332
    conda uninstall -n $CONDA_ENV --force --yes \
333
        cffi \
334
        dask \
335
336
        distributed \
        joblib \
337
        matplotlib-base \
338
        pandas \
339
        psutil \
340
        pyarrow \
341
        python-graphviz \
342
343
        scikit-learn || exit 1
    python -c "import lightgbm" || exit 1
Guolin Ke's avatar
Guolin Ke committed
344
fi