fivefifteen / fetcher
A package manager written in PHP that supports installing dependencies from GitHub, npm, custom URLs, and local file paths.
Installs: 57
Dependents: 1
Suggesters: 0
Security: 0
Stars: 14
Watchers: 2
Forks: 1
Open Issues: 2
pkg:composer/fivefifteen/fetcher
Requires
- php: ^8.1
- adhocore/cli: ^1.9
- composer/semver: ^3.4
- splitbrain/php-archive: ^1.2
Requires (Dev)
- mockery/mockery: ^1.6
- phpunit/phpunit: ^10.5
README
Fetcher
A package manager written in PHP that supports installing dependencies from GitHub, npm, custom URLs, and local file paths.
Table of Contents
Why?
Fetcher was primarily built for downloading frontend packages without the need for Node.js/npm. It pairs nicely with Piler to handle all of your frontend dependency needs using only PHP.
Requirements
- PHP 8.1 or above
- Composer
Installation
To a package (local)
composer require-dev fivefifteen/fetcher
./vendor/bin/fetcher --version
To your system (global)
composer global require fivefifteen/fetcher fetcher --version
Usage
Commands
Install
fetcher install [packages...] fetcher i [packages...]
Installs the defined packages. If no packages are defined, fetcher will attempt to locate a fetch.json
file and install packages located in it's dependencies
section under the fetcher
key.
Note: Fetcher only installs defined dependencies. Child dependencies such as ones found in composer.json
or package.json
are ignored.
Options
[-a|--auth]
- Path to an auth file for installing from private repositories (See: Installing from Private Repositories) [default:auth.json
][-c|--config]
- Path to the config file [default:fetch.json
][-d|--install-directory]
- Path where packages should be installed to [default:fetched
][-e|--extensions]
- A comma separated list of extensions to extract from the packages[-f|--fresh-start]
- Deletes the entire fetched directory before running installation[-h|--header]
- The headers to use in any requests[-i|--ignore-errors]
- Ignore any errors that occur and continue installing the rest of the packages[-p|--providers]
- The providers to search for packages from and their order [default:github,npm
][-q|--quiet]
- Run but don't output anything in the terminal[-s|--save]
- Save the newly installed packages to the config file'sdependencies
section (config file will be created if it doesn't exist)[-w|--working-directory]
- Sets the working directory that all paths will be relative to [default:.
][-x|--no-extract]
- Don't extract packages after they are downloaded
Examples
# Install packages defined in a custom config file fetcher install --config content/themes/my-theme/compile.json # Install the latest version of the `include-media` package from npm fetcher install --providers npm include-media fetcher install npm:include-media # same as above fetcher install npm:include-media@latest # same as above # Install a specific version of the `include-media` package from npm fetcher install --providers npm include-media@1.4.10 fetcher install npm:include-media@1.4.10 # same as above # Install the latest version of the `include-media` package from GitHub fetcher install --providers github eduardoboucas/include-media fetcher install github:eduardoboucas/include-media # same as above fetcher install github:eduardoboucas/include-media@latest # same as above # Install the `include-media` package from a specific commit on GitHub but don't extract it, just download it save it to a custom config file fetcher i -c content/themes/my-theme/compile.json -s -p github -x eduardoboucas/include-media@"#fb3ab8e"
Directory Structure
Inside of the fetched
directory, packages with an author in the name (such as GitHub packages and scoped npm packages) will be installed in fetched/author/package
. If a package has no author in it's name (such as non-scoped npm packages), it will be installed in fetched/package
.
When installing a package via a URL or local file path, the basename of the filename will be used as the package name. For example, if I install package.zip
, it will be installed to fetched/package
. The exception to this rule is if the package being installed contains a single file with the same extension as the package name. For example if I install package.css.zip
and that zip file contains a single css file, you will end up with a single file at fetched/package.css
.
See the Package Aliases section for info about how to customize package names.
Uninstall
fetcher uninstall [packages...] fetcher u [packages...]
Uninstalls the defined packages. If no packages are defined, a confirmation message will ask you if you wish to delete all packages inside the fetched
directory (and the directory itself).
Options
[-c|--config]
- Path to the config file [default:fetch.json
][-d|--install-directory]
- Path where packages are installed to [default:fetched
][-f|--fresh-start]
- Deletes the entire fetched directory and optionally deletes all dependencies from fetch.json[-q|--quiet]
- Run but don't output anything in the terminal (implies--skip-prompts
)[-s|--save]
- Remove the uninstalled packages from the config file'sdependencies
section[-w|--working-directory]
- Sets the working directory that all paths will be relative to [default:.
][-y|--skip-prompts]
- Skips the confirmation prompt and continues with deletion
Examples
# Uninstall the `include-media` package fetcher uninstall include-media # Uninstall the `include-media` package and remove it from a custom config file fetcher u -c content/themes/my-theme/compile.json -s include-media
Info
fetcher info <package> fetcher n <package>
Displays information about a package such as author, homepage, latest version, time of last update, and more.
Options
[-p|--providers]
- The providers to search for packages from and their order [default:github,npm
][-y|--skip-prompts]
- Skips any prompts and displays information for the first available package found
Examples
# Get info about the `include-media` package fetcher info include-media # Get info about the `include-media` package from GitHub fetcher n -p github eduardoboucas/include-media
Versions
fetcher versions <package> fetcher v <package>
Displays the available versions of a package.
Options
[-l|--limit]
- The maximum number of package versions to show (newest versions are display first) [default:20
][-p|--providers]
- The providers to search for packages from and their order [default:github,npm
]
Examples
# Get the latest 20 versions of the `include-media` package fetcher versions include-media # Get the latest 5 versions of the `include-media` package available on GitHub (this includes non-tagged commits) fetcher v -l 5 -p github eduardoboucas/include-media
Global Options
These options can be used with all commands:
[-h|--help]
- Displays helpful information about a command[-v|--verbosity]
- Sets the verbosity level [default:0
][-V|--version]
- Displays the current Fetcher version
Package & Version Parsing
Because Fetcher supports multiple package providers and those providers have their own unique ways of naming and structuring packages, Fetcher has it's own unique but familiar syntax for package names and versions.
Fetcher uses Composer's semver module for passing version constraints.
Examples
# latest version of include-media package from no specific provider (will end up being npm because GitHub requires an author) include-media include-media@latest # same as above # latest version of eduardoboucas/include-media package from no specific provider (will end up being GitHub because the include-media package is not scoped on npm [it's name on npm is just include-media rather than @eduardoboucas/include-media]) eduardoboucas/include-media eduardoboucas/include-media@latest # same as above # any version of the include-media package specifically from npm that is v1.3.2 or above and below v1.4.0 npm:include-media@~1.3.2 # latest version of the include-media package from npm but give it an alias so it's folder structure is like GitHub's [eduardoboucas/include-media]npm:include-media # whatever is currently in the master branch of the eduardoboucas/include-media repository on GitHub github:eduardoboucas/include-media@dev-master # a specific commit from the eduardoboucas/include-media repository on GitHub github:eduardoboucas/include-media@#e620564 # a specific tag from the eduardoboucas/include-media repository on GitHub github:eduardoboucas/include-media@feature-test github:eduardoboucas/include-media@tag-feature-test # same as above # a package from a URL that has been given an alias (in this case we know the archive has a single css file in it so that file will be named wordpress.css) [wordpress/wordpress.css]https://gist.github.com/kodie/d7da9f3db934adea8e44ee38d1885bf8/archive/aaf369827720c564ec3b6c43cba8b00748dbd73d.zip # a package from a local file that has been given an alias [some-package]file:/home/user/Downloads/my-package.zip
Package Aliases
A package can be given an alias name and alias author. An alias author is only used if the alias name is used. This makes it possible to install multiple versions of the same package as well as name packages that are downloaded via URL and have a hash for a name.
For example, running fetcher i [scss-helpers/media-queries]npm:include-media
will install the include-media
package to fetched/scss-helpers/media-queries
.
Keys in the dependencies
section are considered aliases if the package name is defined in the version string (see the scss-helpers/media-queries
example below).
Configuration
While Fetcher can be used out of the box without any configuration, a config file allows for better customization and easier package management.
Example
{ "fetcher": { "config": { "extensions": ["js", "css", "scss", "md"], "working_directory": "content/themes/my-theme" }, "dependencies": { "kenwheeler/slick": "github:~1.8.0", "scss-helpers/media-queries": "npm:include-media@^2.0", "wordpress/wordpress-core.css": "https://gist.github.com/kodie/d7da9f3db934adea8e44ee38d1885bf8/archive/aaf369827720c564ec3b6c43cba8b00748dbd73d.zip" } } }
Configuration via Composer
Fetcher also supports loading configuration options from a composer.json
file, except for in this case Fetcher checks for it's key under the extra
section like so:
{ "name": "username/package", "version": "0.0.1", "require-dev": { "fivefifteen/fetcher": "*" }, "scripts": { "fetch": "./vendor/bin/fetcher install" }, "extra": { "fetcher": { "config": { "extensions": ["js", "css", "scss", "md"], "working_directory": "content/themes/my-theme" }, "dependencies": { "kenwheeler/slick": "github:~1.8.0", "scss-helpers/media-queries": "npm:include-media@^2.0", "wordpress/wordpress-core.css": "https://gist.github.com/kodie/d7da9f3db934adea8e44ee38d1885bf8/archive/aaf369827720c564ec3b6c43cba8b00748dbd73d.zip" } } } }
Installing from Private Repositories
Fetcher will check for the existence of an auth.json
file (or whatever --auth
is set to) and that file is expected to be formatted exactly the way Composer would expect it to be. Currently only the http-basic
method of authentication is supported.
For example, if you would like to install from private GitHub repositories, you would want to create a GitHub Personal Access Token (giving it read permissions to whatever repositories you would like to give Fetcher access to), and then create an auth.json
file with the access token like so:
{ "http-basic": { "api.github.com": { "username": "token", "password": "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2mB" } } }
Related Projects
- Piler - A CLI tool written in PHP that compiles and minifies JavaScript and CSS/SCSS files.
License Information
MIT. See the license.md file for more info.