Skip to content

jamesabel/pyship

Repository files navigation

PyShip

CI codecov

Enables shipping a python application to end users.

PyShip's Major Features

  • Freeze practically any Python application
  • Code Signed (avoid Windows "Unknown Publisher" warning)
  • Installs via Microsoft Windows App Store or self-hosted installers (entirely outside the app store)
  • Automatic application updating in the background (no user intervention)
  • OS native application (e.g. .exe for Windows)
  • Run on OS startup option

Currently only for Windows. May be extended to other Operating Systems in the future.

Configuration

pyship settings are configured in your project's pyproject.toml under [tool.pyship].

pyproject.toml

[tool.pyship]
# App metadata
is_gui = false           # true if the app is a GUI application (default: false)
run_on_startup = false   # true to run the app on OS startup (default: false)

# Cloud upload settings
profile = "default"      # AWS IAM profile for S3 uploads
upload = true            # upload installer and clip to S3 (default: true)
public_readable = false  # make uploaded S3 objects publicly readable (default: false)

AWS Keys in CLI Arguments

The user may optionally provide their AWS keys as CLI arguments. They may also be provided in the ~/.aws/credentials file (typical for AWS CLI and boto3). These are not in pyproject.toml since it is usually checked into the code repo.

Argument Description
-i, --id AWS Access Key ID
-s, --secret AWS Secret Access Key

Microsoft Windows Code Signing (Optional)

Signing your executables suppresses the Windows SmartScreen "Unknown Publisher" warning. pyship signs two files: the launcher stub ({app_name}.exe) and the NSIS installer ({app_name}_installer_*.exe). The launcher is signed before NSIS packages it, so the signed binary ends up inside the installer.

You need an Authenticode certificate in PFX format and signtool.exe from the Windows SDK.

Getting signtool.exe

Install the Windows SDK and select Windows SDK Signing Tools for Desktop Apps. pyship auto-discovers the newest version under C:\Program Files (x86)\Windows Kits\10\bin\.

Basic usage

from pathlib import Path
from pyship import PyShip

ps = PyShip(
    project_dir=Path("path/to/your/app"),
    pfx_path=Path("path/to/certificate.pfx"),
    certificate_password="your-pfx-password",
)
ps.ship()

CI / environment variable password

Avoid storing the PFX password in source code. Pass it via an environment variable instead:

ps = PyShip(
    project_dir=Path("path/to/your/app"),
    pfx_path=Path("path/to/certificate.pfx"),
    certificate_password_env_var="PFX_PASSWORD",  # reads os.environ["PFX_PASSWORD"] at ship() time
)
ps.ship()

Set the secret in your CI system (e.g. GitHub Actions → Settings → Secrets) and expose it in your workflow:

- name: Ship
  env:
    PFX_PASSWORD: ${{ secrets.PFX_PASSWORD }}
  run: python ship.py

Explicit signtool path

If signtool.exe is not in the default Windows SDK location, point pyship directly at it:

ps = PyShip(
    ...
signtool_path = Path(r"C:\custom\path\signtool.exe"),
)

Timestamp server

By default pyship uses DigiCert's RFC 3161 server (http://timestamp.digicert.com). Override it with the timestamp_url field:

ps = PyShip(
    ...
timestamp_url = "http://timestamp.sectigo.com",
)

Verifying a signed executable

signtool verify /pa /v YourApp.exe

Skipping signing

Leave pfx_path and certificate_password / certificate_password_env_var unset (the defaults). pyship will build and package the executables without signing them.

Microsoft Store Distribution (MSIX) (Optional)

The Microsoft Store is an alternative distribution channel that gives users a trusted, one-click install experience and eliminates SmartScreen warnings entirely. pyship can produce both a traditional NSIS installer and an MSIX package in a single ship() call.

Prerequisites

Requirement Notes
Windows SDK Provides makeappx.exe and signtool.exe. Install from developer.microsoft.com/windows/downloads/windows-sdk and select Windows SDK Signing Tools for Desktop Apps. pyship auto-discovers both tools.
Authenticode certificate Same PFX used for code signing. The Publisher field in the MSIX manifest must exactly match the certificate subject. Self-signed certificates work for sideloading and testing but not for Store submission.
Microsoft Partner Center account Required for Store submission only. Register at partner.microsoft.com. One-time $19 fee for individuals.

Basic usage

Set msix=True and supply the certificate subject DN as msix_publisher:

from pathlib import Path
from pyship import PyShip

ps = PyShip(
    project_dir=Path("path/to/your/app"),
    pfx_path=Path("certificate.pfx"),
    certificate_password_env_var="PFX_PASSWORD",
    msix=True,
    msix_publisher="CN=My Company, O=My Company LLC, C=US",  # must match cert subject exactly
)
ps.ship()

This produces two files in installers/:

  • {app_name}_installer_win.exe — the NSIS installer (direct distribution, auto-updates via pyshipupdate)
  • {app_name}_installer_win.msix — the MSIX package (Store or sideloading, updates managed by the Store)

Both are signed with the same certificate. The NSIS installer is built first; the MSIX is packed from the same app directory afterwards, so they do not interfere with each other.

Finding the msix_publisher string

msix_publisher must be the exact Distinguished Name of your signing certificate's subject. To read it from your PFX:

certutil -dump certificate.pfx

Look for the Subject: line, e.g. CN=My Company, O=My Company LLC, C=US. Copy it verbatim — any difference will cause installation to fail.

Store logo assets

MSIX packages require three PNG logo files. pyship generates 1×1 white placeholder PNGs automatically, which is sufficient for testing and sideloading. For Store submission, provide real assets at the correct sizes:

File Size
StoreLogo.png 50×50
Square44x44Logo.png 44×44
Square150x150Logo.png 150×150

Point pyship at a directory containing these files with store_assets_dir:

ps = PyShip(
    ...
msix = True,
msix_publisher = "CN=My Company, O=My Company LLC, C=US",
store_assets_dir = Path("store_assets"),
)

Any asset not found in that directory falls back to the placeholder PNG automatically.

Explicit makeappx path

If makeappx.exe is not in the default Windows SDK location:

ps = PyShip(
    ...
makeappx_path = Path(r"C:\custom\path\makeappx.exe"),
)

Sideloading (without the Store)

Signed MSIX packages can be installed directly without going through the Store — double-click the .msix file or use:

Add-AppxPackage -Path "{app_name}_installer_win.msix"

This is useful for enterprise deployments and beta distribution.

Submitting to the Microsoft Store

  1. Log in to Partner Center.
  2. Go to Windows & XboxOverviewCreate a new app and reserve your app name.
  3. Under SubmissionsNew submission, complete the store listing (description, screenshots, age rating, pricing).
  4. Upload your signed .msix from the installers/ directory under Packages.
  5. Submit for certification. Review typically takes 1–3 business days.

Limitations

  • run_on_startup = true is not supported in MSIX builds. The MSIX runtime requires a StartupTask extension in the manifest; pyship logs a warning and continues, but the app will not auto-start. Configure startup behaviour manually in a custom manifest if needed.
  • MSIX packages installed via the Store are updated by the Store, not by pyshipupdate. The self-update logic in pyshipupdate is silently inactive when running inside an MSIX container.

NSIS vs. MSIX comparison

NSIS installer MSIX
Distribution Your own URL / S3 Microsoft Store or sideload
SmartScreen Suppressed with EV cert Not shown (implicitly trusted)
Auto-update pyshipupdate Store manages updates
Review required No Yes (Store only; ~1–3 days)
Revenue share None 15% (>$1M/yr: 12%)
Startup support Yes Requires manifest extension

Testing

Run tests with:

venv\Scripts\python.exe -m pytest test_pyship/ -v

Environment Variables

Variable Description Default
AWSIMPLE_USE_MOTO_MOCK Set to 0 to use real AWS instead of moto mock. Required for test_f_update which tests cross-process S3 updates. Requires valid AWS credentials configured. 1 (use moto)
MAKE_NSIS_PATH Path to the NSIS makensis.exe executable. C:\Program Files (x86)\NSIS\makensis.exe

Test Modes

  • Default (moto mock): All tests run with mocked AWS S3. No credentials needed. test_f_update is skipped.
  • Real AWS (AWSIMPLE_USE_MOTO_MOCK=0): All tests run against real AWS S3. test_f_update runs and tests cross-process updates.

About

pyship - ship Python desktop apps to end users

Topics

Resources

License

Code of conduct

Stars

Watchers

Forks

Releases

No releases published

Packages

 
 
 

Contributors