File: ci.yml

package info (click to toggle)
pyinstaller 6.18.0%2Bds-3
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 11,824 kB
  • sloc: python: 41,828; ansic: 12,123; makefile: 171; sh: 131; xml: 19
file content (560 lines) | stat: -rw-r--r-- 21,534 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
name: CI

on:
  workflow_dispatch:  # Allow workflow to be manually triggered
  pull_request:  # Trigger on PRs opened against develop branch
    branches:
      - develop

env:
  # Colored pytest output on CI despite not having a tty
  FORCE_COLOR: 1
  # Enable strict unpack mode to catch file duplication problems in onefile builds (at executable run-time).
  PYINSTALLER_STRICT_UNPACK_MODE: 1
  # Enable strict collect mode to catch file duplication problems in PKG/Carchive (onefile builds) or COLLECT
  # (onedir builds) at build time.
  PYINSTALLER_STRICT_COLLECT_MODE: 1
  # Enable strict handling of codesign errors for macOS bundles.
  PYINSTALLER_STRICT_BUNDLE_CODESIGN_ERROR: 1
  # Enable strict verification of macOS bundles w.r.t. the code-signing requirements.
  PYINSTALLER_VERIFY_BUNDLE_SIGNATURE: 1
  # Enable PEP 597 EncodingWarnings
  PYTHONWARNDEFAULTENCODING: true

permissions:
  contents: read # to fetch code (actions/checkout)

jobs:
  tests:
    runs-on: ${{ matrix.os }}
    strategy:
      matrix:
        python-version: ['3.8', '3.9', '3.10', '3.11', '3.12', '3.13', '3.14']
        os: ['windows-latest', 'ubuntu-24.04', 'macos-15-intel', 'macos-14']
        # Split macOS workflows between macos-15-intel (x86_64) and macos-14 (arm64)
        # runners to cover both architectures without running all combinations.
        exclude:
          - python-version: '3.8'
            os: 'macos-14'
          - python-version: '3.10'
            os: 'macos-14'
          - python-version: '3.12'
            os: 'macos-14'
          - python-version: '3.14'
            os: 'macos-14'
          - python-version: '3.9'
            os: 'macos-15-intel'
          - python-version: '3.11'
            os: 'macos-15-intel'
          - python-version: '3.13'
            os: 'macos-15-intel'
      fail-fast: false
    steps:
      - name: Checkout PyInstaller code
        uses: actions/checkout@v4

      - name: Set up Python ${{ matrix.python-version }}
        uses: actions/setup-python@v5
        with:
          python-version: ${{ matrix.python-version }}

      - name: Install apt packages
        if: startsWith(matrix.os, 'ubuntu')
        run: |
          sudo apt-get update -qq
          sudo apt-get upgrade -qq
          sudo apt-get install -qq --no-install-recommends \
            libcmocka-dev \
            libxml2-dev libxslt1-dev gfortran libatlas-base-dev \
            libespeak1 libxcb-image0 libxcb-keysyms1 libxcb-render-util0 \
            libxkbcommon-x11-0 libxcb-icccm4 libxcb1 openssl \
            libxcb-randr0-dev libxcb-xtest0-dev libxcb-xinerama0-dev \
            libxcb-shape0-dev libxcb-xkb-dev libxcb-cursor0 xvfb \
            libopengl0 libegl1 \
            libpulse0 libpulse-mainloop-glib0 \
            gstreamer1.0-plugins-base libgstreamer-gl1.0-0 \
            libgirepository1.0-dev libgirepository-2.0-dev libcairo2-dev \
            gir1.2-girepository-2.0 gir1.2-girepository-3.0 gir1.2-gtk-3.0 \
            libfuse2

      # The following locales are required by test_basic::test_user_preferred_locale
      #  - en_US.UTF-8
      #  - en_US.ISO8859-1
      #  - sl_SI.UTF-8
      #  - sl_SI.ISO8859-2
      #
      # The following locale is used by test_basic::test_time_module_localized
      #  - cs_CZ.UTF-8
      - name: Install and enable additional locales
        if: startsWith(matrix.os, 'ubuntu')
        run: |
          sudo apt-get update -qq
          sudo apt-get install -qq --no-install-recommends locales
          sudo locale-gen \
            en_US.UTF-8 \
            en_US \
            sl_SI.UTF-8 \
            sl_SI \
            cs_CZ.UTF-8
          locale -a

      - name: Download AppImage tool
        if: startsWith(matrix.os, 'ubuntu')
        run: |
          wget \
            https://github.com/AppImage/AppImageKit/releases/download/12/appimagetool-x86_64.AppImage \
            -O $HOME/appimagetool-x86_64.AppImage
          chmod a+x $HOME/appimagetool-x86_64.AppImage

      - name: Set cache dir
        run: echo "pip_cache_dir=$(pip cache dir)" >> $GITHUB_ENV
        shell: bash

      - name: Fetch cache
        id: cache-target
        uses: actions/cache@v4
        with:
          path: ${{ env.pip_cache_dir }}
          key: ${{ runner.os }}-${{ matrix.python-version }}

      - name: Check if bootloader code conforms to gnu90 C standard
        if: startsWith(matrix.os, 'ubuntu')
        run: |
          # Compile bootloader
          cd bootloader
          CC="gcc -std=gnu90" python waf --tests all

      - name: Check if bootloader code conforms to c99 ISO C standard (pedantic mode)
        if: startsWith(matrix.os, 'ubuntu')
        run: |
          # Compile bootloader
          cd bootloader
          CC="gcc -std=c99 -pedantic" python waf --tests all

      - name: Check if bootloader is buildable with --static-zlib option
        if: startsWith(matrix.os, 'ubuntu')
        run: cd bootloader && python waf --static-zlib --tests all

      - name: Check if bootloader is buildable for Windows ARM
        if: startsWith(matrix.os, 'windows')
        run: |
            cd bootloader
            python waf --tests --target-arch=64bit-arm all
            ls ../PyInstaller/bootloader/Windows-64bit-arm

      - name: Update pip
        run: python -m pip install --upgrade pip hatchling

      - name: Compile bootloader
        run: cd bootloader && python waf --tests all

      - name: Download dependencies
        shell: bash
        run: python -m pip download --dest=dist .[completion] && rm -f dist/pyinstaller-*.whl

      - name: Build wheels
        run: sh release/build-wheels

      - name: Install PyInstaller
        run: python -m pip install --no-index --find-links=dist pyinstaller[completion]

      - name: Check pyinstaller --help
        run: python -m PyInstaller -h

      - name: Install test dependencies (base tools)
        run: |
          python -m pip install --progress-bar=off --upgrade --requirement tests/requirements-base.txt

      # Give python 3.13t same treatment as development version and avoid trying to install packges
      # from requirements-libraries.txt. This is necessary because not all packages provide free-threaded
      # builds of their wheels yet, and we have no way of filtering them in the requirements file.
      # This should be remedied in python 3.14 with the new `sys_abi_feature` environment marker (PEP 780).
      - name: Install test dependencies (tools and libraries)
        if: ${{ !endsWith(matrix.python-version, '-dev') && matrix.python-version != '3.13t' }}
        run: |
          python -m pip install --progress-bar=off --upgrade --requirement tests/requirements-libraries.txt

            # On (macos-14, python 3.10), (macos-13, python 3.11) and (macos-14, python 3.11)
      # combinations the initial `setuptools` installation seems to contain `.opt-1.pyc`
      # files in `__pycache__` directories. During the upgrade, `pip` fails to remove
      # those directories (https://github.com/pypa/pip/issues/11835).
      #
      # `setuptools` 75.4 removed its vendored copy of `importlib_resources`, but due to
      # the above issue, the effectively empty directory remains, turning `importlib_resources`
      # into defunct namespace package, which causes some of our tests to fail.
      - name: Work around potentially broken setuptools upgrade
        shell: python
        run: |
          import sys
          import pathlib
          import shutil
          import importlib.util

          try:
            spec = importlib.util.find_spec('setuptools._vendor.importlib_resources')
          except ImportError:
            spec = None  # setuptools or setuptools._vendor does not exist
          if spec is None:
            print("Did not find setuptools-vendored copy of importlib_resources.")
            sys.exit(0)
          elif spec.loader is not None:
            print("Found a valid setuptools-vendored copy of importlib_resources.")
            sys.exit(0)

          print("Found a defunct setuptools-vendored copy of importlib_resources!")

          # List the contents of importlib_resources package directory for debug purposes
          def list_directory(path, pad=""):
            for child in path.iterdir():
              if child.is_dir():
                print(f"{pad} + {child.name}")
                list_directory(child, pad + " ")
              else:
                print(f"{pad} - {child.name} ({child.stat().st_size} bytes)")

          for path in spec.submodule_search_locations:
            print(f"Listing contents of {path}")
            list_directory(pathlib.Path(path))

          # Remove
          for path in spec.submodule_search_locations:
            print(f"Removing {path}...")
            shutil.rmtree(path)

      - name: Start display server
        if: startsWith(matrix.os, 'ubuntu')
        run: |
          Xvfb :99 &
          echo "DISPLAY=:99" >> $GITHUB_ENV

      # Required on macOS >= 11 by tests that register custom URL schema. This could also be achieved by passing
      # --basetemp to pytest, but using environment variable allows us to have a unified "Run test" step for
      # all OSes.
      #
      # We now relocate the temporary directory to a fixed location on all OSes, in order to be able to generate
      # artifacts out of failed tests.
      - name: Relocate temporary dir
        shell: bash
        run: |
          echo "PYTEST_DEBUG_TEMPROOT=$RUNNER_TEMP" >> $GITHUB_ENV

      - name: Run tests
        id: run-tests
        run: >
            pytest
            -n 5 --maxfail 3 --durations 10 tests/unit tests/functional

      # On all platforms, create a tarball to ensure that symlinks are preserved. Avoid using compression here,
      # as the tarball will end up collected into artifact zip archive.
      # To simplify this across platform, run this step in python and use python's tarfile module.
      - name: Archive failed tests
        if: ${{ failure() && steps.run-tests.outcome == 'failure' }}
        shell: python
        run: |
          import os
          import sys
          import tarfile

          try:
            import getpass
            user = getpass.getuser() or "unknown"
          except Exception:
            user = "unknown"

          temproot = os.environ['PYTEST_DEBUG_TEMPROOT']

          pytest_name = f'pytest-of-{user}'
          pytest_fullpath = os.path.join(temproot, pytest_name)
          print(f"Input directory: {pytest_fullpath}!", file=sys.stderr)

          output_file = os.path.join(temproot, 'archived-failed-tests.tar')
          print(f"Output file: {output_file}!", file=sys.stderr)

          assert os.path.isdir(pytest_fullpath)
          assert not os.path.exists(output_file)

          with tarfile.open(output_file, "w") as tf:
            tf.add(pytest_fullpath, arcname=pytest_name, recursive=True)

          print(f"Created {output_file}!", file=sys.stderr)

      - name: Create artifact out of archived failed tests
        if: ${{ failure() && steps.run-tests.outcome == 'failure' }}
        uses: actions/upload-artifact@v4
        with:
          name: failed-tests-${{ matrix.os }}-python-${{ matrix.python-version }}
          path: '${{ env.PYTEST_DEBUG_TEMPROOT }}/archived-failed-tests.tar'

      # Install and test PyInstaller Hook Sample, to ensure that tests declared in
      # entry-points are discovered.
      - name: Install hooksample
        run: python -m pip install "https://github.com/pyinstaller/hooksample/archive/v4.0rc1.zip"

      # Augment _pyinstaller_hooks_contrib with bogus hooks that conflict with the hooks provided by hooksample.
      # Due to (implicit) hook priority, the 3rd-party hooks should be chosen over _pyinstaller_hooks_contrib
      # ones, and so the bogus hooks should never be ran. This applies to standard module hooks, as well as
      # the pre-find-module-path and pre-safe-import-module hooks.
      - name: Inject bogus hooks
        shell: python
        run: |
          import os
          from _pyinstaller_hooks_contrib import (
              stdhooks,
              pre_safe_import_module,
              pre_find_module_path,
          )
          with open(os.path.join(stdhooks.__path__[0], "hook-pyi_hooksample.py"), "w", encoding="utf-8") as f:
              f.write('raise Exception("Wrong hook! Use the pyi_hooksample copy instead!")\n')
          with open(os.path.join(pre_safe_import_module.__path__[0], "hook-pyi_hooksample.py"), "w", encoding="utf-8") as f:
              f.write('raise Exception("Wrong hook! Use the pyi_hooksample copy instead!")\n')
          with open(os.path.join(pre_find_module_path.__path__[0], "hook-pyi_hooksample.py"), "w", encoding="utf-8") as f:
              f.write('raise Exception("Wrong hook! Use the pyi_hooksample copy instead!")\n')

      - name: Run hooksample tests
        run: |
          # The ``run_tests`` script is invoked somewhere outside of the pyinstaller git clone to prevent pytest from
          # loading PyInstaller's pytest.ini.
          cd ~
          python -m PyInstaller.utils.run_tests --include_only=pyi_hooksample.

  test-freethreaded:
    runs-on: ${{ matrix.os }}
    strategy:
      matrix:
        python-version: ['3.13t', '3.14t']
        os: ['windows-latest', 'ubuntu-24.04', 'macos-15-intel', 'macos-14']
      fail-fast: false
    steps:
      - name: Checkout PyInstaller code
        uses: actions/checkout@v4

      - name: Set up Python ${{ matrix.python-version }}
        uses: actions/setup-python@v5
        with:
          python-version: ${{ matrix.python-version }}

      - name: Update pip
        run: python -m pip install --upgrade pip

      - name: Install PyInstaller
        run: python -m pip install . --requirement tests/requirements-base.txt

      - name: Run tests
        id: run-tests
        run: pytest tests/functional/test_freethreading.py

  test-alpine:
    runs-on: ubuntu-latest
    steps:
      - name: Checkout PyInstaller code
        uses: actions/checkout@v4

      - name: Build container
        run: docker build -f alpine.dockerfile -t foo .

      - name: Run tests in container
        run: >
          docker run
          foo
          pytest
          -n 5 --maxfail 3 --durations 10 tests/unit tests/functional

      # Verify that installing from sdist works. This is mostly just a verification that the MANIFEST.in contains
      # all the extras needed to compile the bootloader.
      - name: Setup Python for creating sdist
        uses: actions/setup-python@v5
        with:
          python-version: 3.x

      - name: Create sdist
        run: |
          git clean -xfdq .
          python -m pip install build
          python -m build --sdist --outdir dist
          docker build --target=wheel-factory -t bar -f alpine.dockerfile .

      - name: Install and test PyInstaller installed from sdist
        run: |
          sdist="$(ls dist)"
          docker run -v "$PWD/dist:/io" bar ash -c "
            python -m pip install /io/$sdist
            echo 'print(1 + 1)' > test.py
            pyinstaller test.py
            ./dist/test/test
          "

  test-msys2:
    runs-on: windows-latest
    strategy:
      matrix:
        include:
          - { sys: mingw64, env: x86_64 }
          - { sys: mingw32, env: i686 }
          - { sys: ucrt64,  env: ucrt-x86_64 }
      fail-fast: false
    defaults:
      run:
        shell: msys2 {0}
    steps:
      - name: Setup msys2 environment
        uses: msys2/setup-msys2@v2
        with:
          update: true
          msystem: ${{matrix.sys}}
          install: >-
            mingw-w64-${{matrix.env}}-gcc
            mingw-w64-${{matrix.env}}-python
            mingw-w64-${{matrix.env}}-python-pip
            mingw-w64-${{matrix.env}}-python-setuptools
            mingw-w64-${{matrix.env}}-python-pywin32-ctypes
            mingw-w64-${{matrix.env}}-python-packaging
            mingw-w64-${{matrix.env}}-python-pytest
            mingw-w64-${{matrix.env}}-python-pytest-xdist
            mingw-w64-${{matrix.env}}-python-pytest-timeout
            mingw-w64-${{matrix.env}}-python-pywin32
            mingw-w64-${{matrix.env}}-python-gobject

      # Some msys2 python packages are not available for i686 anymore:
      #  - numpy
      #  - pefile
      #  - psutil (removed with update to psutil 6.1.1; see https://github.com/msys2/MINGW-packages/commit/7f1c75b33a5aebb89899f99f5fa656c623eeb9ed)
      #  - pillow (see https://github.com/msys2/MINGW-packages/commit/a7d88c34aa36adce78ba71633711412a24d8792c)
      # If running in 64-bit environment, install the said packages here.
      # Otherwise, we will run tests without these packages (note that
      # if not installed here, `pefile` will end up installed via `pip`
      # as part of PyInstaller's dependencies).
      - name: Install extra msys2 packages (64-bit only)
        if: matrix.env != 'i686'
        run: >
          pacman -S --noconfirm
          mingw-w64-${{matrix.env}}-python-numpy
          mingw-w64-${{matrix.env}}-python-pefile
          mingw-w64-${{matrix.env}}-python-psutil
          mingw-w64-${{matrix.env}}-python-pillow

      - name: Checkout PyInstaller code
        uses: actions/checkout@v4

      - name: Show system and python information
        run: |
          uname -a
          python --version

      - name: Install PyInstaller
        run: |
          # Compile bootloader
          cd bootloader
          python waf --gcc --tests all
          cd ..

          # Install PyInstaller.
          python -m pip install --progress-bar=off .[completion]

          # Make sure the help options print.
          python -m PyInstaller -h

      - name: Run tests
        run: >
            pytest
            -n 5 --maxfail 3 --durations 10 tests/unit tests/functional

  test-cygwin:
    runs-on: windows-latest
    defaults:
      run:
        shell: bash -o igncr -eo pipefail '{0}'
    steps:
      - name: Setup Cygwin
        uses: cygwin/cygwin-install-action@master
        with:
          platform: x86_64
          # NOTE: some of the provided packages (python39-setuptools,
          # python39-packaging, python39-pytest) are out-of-date, and we instead
          # install PyPI wheels.
          #
          # NOTE: for some reason, python39-pip depends on python39-sphinx,
          # which is an old sphinx v4.4.0 package. This version contains
          # an issue caused by open() without specified encoding, which
          # was fixed in sphinx >= v5.0. Therefore, we install an up-to-date
          # version of sphinx from PyPI in the "Update pip and base test tools"
          # step.
          packages: >-
            gcc-core
            zlib
            zlib-devel
            xorg-server-extra
            python39
            python39-devel
            python39-pip
            python39-tkinter


      - name: Disable CRLF line endings in git checkout
        run: git config --global core.autocrlf input

      - name: Checkout PyInstaller code
        uses: actions/checkout@v4

      - name: Show system and python information
        run: |
          uname -a
          python --version

      - name: Update pip and base test tools
        run: python -m pip install --progress-bar=off --upgrade pip setuptools --requirement tests/requirements-base.txt sphinx

      - name: Install PyInstaller
        run: |
          # Compile bootloader
          cd bootloader
          python waf --gcc --tests all
          cd ..

          # Install PyInstaller.
          python -m pip install --progress-bar=off .[completion]

          # Make sure the help options print.
          python -m PyInstaller -h

      - name: Start display server
        run: |
          Xvfb :99 &
          echo "DISPLAY=:99" >> $GITHUB_ENV

      - name: Run tests
        # Keep the number of pytest workers below the number of available
        # CPU cores to avoid CPU contention scenarios, which seem to trigger
        # a race condition in Cygwin `tkinter` (seemingly in both unfrozen
        # and frozen programs).
        run: >
            python -m pytest
            -n 3 --maxfail 3 --durations 10 tests/unit tests/functional

  test-termux:
    runs-on: ubuntu-latest
    steps:
      - name: Checkout PyInstaller code
        uses: actions/checkout@v4

      # Explicitly set CC=clang, to prevent `pip` from trying to build `psutil` extensions with `x86_64-linux-android-clang`,
      # which (in contrast to `clang` and `gcc`) does not seem to set `__ANDROID_API__` and `__ANDROID_MIN_SDK_VERSION__`;
      # consequently, `/usr/include/ifaddrs.h` header does not provide `getifaddrs()` and `freeifaddrs()`, leading to build error.
      - name: Run tests in container
        run: |
          docker run -v"$PWD:/io" termux/termux-docker:x86_64 bash -ec '
            echo "*** Installing system dependencies ***"
            pkg install -y python ldd binutils
            export CC=clang
            echo "*** Installing PyInstaller ***"
            cp -fr /io $HOME/PyInstaller
            cd $HOME/PyInstaller
            cd bootloader
            python waf all
            cd ..
            python -m pip install .
            echo "*** Installing test requirements ***"
            python -m pip install -r tests/requirements-base.txt
            echo "*** Running tests ***"
            export FORCE_COLOR=1
            python -m pytest -n 4 --maxfail 3 --durations 10 tests/unit tests/functional
          '