codewithkyrian / platform-package-installer
A Composer plugin that provides fine-grained control over platform-specific package distribution for PHP projects
Installs: 366
Dependents: 1
Suggesters: 0
Security: 0
Stars: 7
Watchers: 1
Forks: 0
Open Issues: 0
Type:composer-plugin
Requires
- php: ^8.1
- composer-plugin-api: ^1.1 || ^2.0
- composer-runtime-api: *
- symfony/yaml: ^6.4 || ^7.2
Requires (Dev)
- composer/composer: ~1.0 || ~2.0
- pestphp/pest: ^2.36.0|^3.5.0
- symfony/var-dumper: ^6.0|^7.0
README
The Composer Platform Package Installer is a powerful Composer plugin that provides fine-grained control over package distribution across different platforms and architectures. Unlike traditional Composer package management, this plugin allows you to:
- Exclude specific folders and files for different platforms
- Generate platform-specific distribution URLs
- Customize package installation for various operating systems and architectures
Requirements
- PHP 8.1 or higher
- Composer 2.0 or higher
Installation
Install the plugin using Composer:
composer require codewithkyrian/platform-package-installer
Usage Guide
Step 1: Package Configuration
Change your package type to platform-package
in composer.json
:
{ "name": "org/library", "type": "platform-package", "require": { "codewithkyrian/platform-package-installer": "^1.0" } }
Step 2: Create Platforms Configuration
Create a platforms.yml
file in the root of your project and list the platforms you want to support:
linux-x86_64: linux-arm64: darwin-arm64: darwin-x86_64: windows-32: windows-64:
You can also specify exclusion rules for each platform (this will come in handy later when generating distribution archives).
linux-x86_64: exclude: - libs/linux-arm64 - libs/darwin-* - libs/windows-* - "*.exe" - "*.dylib" - "**/*.windows.*" darwin-x86_64: exclude: - libs/darwin-arm64 - libs/windows-* - libs/linux-* - "*.exe" - "*.so" - "**/*.linux.*" # Add configurations for other platforms similarly
Exclusion Rule Patterns:
- Use standard glob patterns to specify exclusions
*
matches any number of characters**
matches nested directories- Platform-specific libraries, binaries, and incompatible files can be easily filtered
Step 3: Generate Distribution URLs
Use the platform:generate-urls
command to create platform-specific download URLs and add them to composer.json
. This
is very important because Composer will use these URLs to download the appropriate platform-specific package.
composer platform:generate-urls --dist-type=github --repo-path=vendor/repo // Or using a custom template composer platform:generate-urls --dist-type=https://cdn.example.com/vendor/repo/release-{version}-{platform}.{ext}
Command Options
--platforms-file
: Path to the platforms.yml file (optional, defaults toplatforms.yml
)--dist-type
: Distribution source type (github
,gitlab
,huggingface
, or a custom URL template)--repo-path
: Repository path (optional when using a custom URL template)--extension
: File extension (optional. Uses .zip for Windows and .tar.gz for others)
For the example above, the generated config added to composer.json
would look like:
{ "extra": { "platform-urls": { "linux-x86_64": "https://github.com/Codewithkyrian/example/releases/download/{version}/dist-linux-x86_64.tar.gz", "darwin-arm64": "https://github.com/Codewithkyrian/example/releases/download/{version}/dist-darwin-arm64.tar.gz" } } }
Now the next time someone installs your package, Composer will use the generated URLs to download the appropriate platform-specific package. The good thing is that if for any reason the URLs are wrong or not working, Composer falls back to downloading the original source (which means more things to download and slower installation but hey, it's better than nothing).
Platform Identifiers
Platform Identifiers are used to specify the platform-specific package URL. Platform identifiers can be:
- Base platform names: The supported base platform names are
linux
,darwin
,windows
andraspberrypi
. This means that the distribution archive can be used on any architecture of these platforms. - Specific architectures: You can be more specific by specifying the architecture as well. The identifier is formed
by joining the platform identifier with the architecture identifier (
-
), e.g.darwin-arm64
,darwin-x86_64
,windows-32
,windows-64
,linux-aarch64
, etc. - Universal: Use
all
to cover every platform
Distribution Archives
For regular packages, GitHub and GitLab provide an automatic distribution archive (tarball or zip) for each release. Using this package however, means you can no longer rely on those automatic distribution archives.
The composer platform:generate-urls
only generates urls for the platforms specified in the platforms.yml
file using a
template, but you still need to manually generate the archive for each platform and make them available at those
locations.
GitHub
The generated GitHub distribution archive is available at the following URL:
https://github.com/vendor/package/releases/download/{version}/dist-{platform}.{ext}
{platform}
and {ext}
will be replaced with the platform identifier and the file extension when generating the URL
while {version}
will be replaced with the package version at installation time.
The package provides a convenient example GitHub Actions workflow that uses the
platforms.yml
file to generate the distribution archives every release and upload them as release assets. Feel free to
modify the workflow to suit your needs.
GitLab
The GitLab distribution archive is available at the following URL:
https://gitlab.com/vendor/package/-/releases/{version}/downloads/dist-{platform}.{ext}
At the moment, there's no example workflow for GitLab. Contributions are welcome for a similar workflow.
Always test the workflow thoroughly to ensure it generates the correct distribution archives for each platform.
Runtime Utility
The package provides a powerful utility method findBestMatch()
to easily handle platform-specific resources at
runtime. It finds the most appropriate match for the current platform from a given set of platform-specific entries.
use Codewithkyrian\PlatformPackageInstaller\Platform; // Example with directory paths $libraryPaths = [ 'linux-x86_64' => '/path/to/linux/x86_64/libs', 'darwin-arm64' => '/path/to/mac/arm64/libs', 'windows-x86_64' => '/path/to/windows/x86_64/libs' ]; $bestLibraryPath = Platform::findBestMatch($libraryPaths); // Example with configuration arrays $platformConfigs = [ 'linux-x86_64' => [ 'library_path' => '/path/to/linux/libs', 'additional_config' => ['key' => 'value'] ], 'darwin-arm64' => [ 'library_path' => '/path/to/mac/libs', 'additional_config' => ['key' => 'another_value'] ] ]; $currentPlatformConfig = Platform::findBestMatch($platformConfigs);
Tests
The tests
folder contains a suite of tests that verify the behavior of the plugin. To run the tests, you need
to install the development dependencies using the composer install --dev
command. Then, run the tests using either
the composer test
command or the ./vendor/bin/pest
command.
License
This project is licensed under the MIT License. See the LICENSE file for more information.
Contributing
Pull requests are welcome. For major changes, please open an issue first to discuss what you would like to change.