If you wish to contribute to this project, please be sure to read/subscribe to the following resources:
If you are working on new features or refactoring for an existing project item (e.g. adding a new filter or validator or rewriting an existing class), create a Request For Comments (RFC) in the correspondig repository via a new issue report.
If you want to propose a new component/package, create a RFC in the Technical Steering Committee repository via a new issue report.
To run tests:
-
Clone the repository. Click the "Clone or download" button on the repository to find both the URL and instructions on cloning.
-
Install dependencies via composer:
$ composer install
If you don't have
composer
installed, please download it from https://getcomposer.org/download/ -
Run the tests using the "test" command shipped in the
composer.json
:$ composer test
You can turn on conditional tests with the phpunit.xml
file. To do so:
- Copy the
phpunit.xml.dist
file tophpunit.xml
- Edit the
phpunit.xml
file to enable any specific functionality you want to test, as well as to provide test values to utilize.
First, ensure you've installed dependencies via composer, per the previous section on running tests.
To run CS checks only:
$ composer cs-check
To attempt to automatically fix common CS issues:
$ composer cs-fix
If the above fixes any CS issues, please re-run the tests to ensure they pass, and make sure you add and commit the changes after verification.
Each repository uses release branches named after the minor release series they represent; e.g., "2.11.x", "3.1.x", "0.2.x", etc. Each branch is under a "security window", and can and will receive security updates as long as the minimum supported version of PHP in that branch is still supported by php.net. Generally speaking, the default branch is targeting the next minor or major version, and does not yet have releases against it; it represents new features or changes to the package. The release branch with the next most recent release series is the one for the currently active release, and will receive bugfixes and security fixes, but no new features. Since each component has its own lifecycle, the number and names of releases branches will vary from component to component.
Your first step is to establish a public repository from which we can pull your work into the canonical repository. We recommend using GitHub, as that is where the component is already hosted.
-
Setup a GitHub account, if you haven't yet
-
Fork the repository using the "Fork" button at the top right of the repository landing page.
-
Clone the canonical repository locally. Use the "Clone or download" button above the code listing on the repository landing pages to obtain the URL and instructions.
-
Navigate to the directory where you have cloned the repository.
-
Add a remote to your fork; substitute your GitHub username and the repository name in the commands below.
$ git remote add fork [email protected]:{username}/{repository}.git $ git fetch fork
Alternately, you can use the GitHub CLI tool to accomplish these steps:
$ gh repo clone {org}/{repo}
$ cd {repo}
$ gh repo fork
Periodically, you should update your fork or personal repository to match the canonical repository. Assuming you have setup your local repository per the instructions above, you can do the following:
$ git fetch origin
$ git switch {branch to update}
$ git pull --rebase --autostash
# OPTIONALLY, to keep your remote up-to-date -
$ git push fork {branch}:{branch}
If you're tracking other release branches, you'll want to do the same operations for each branch.
We recommend you do each new feature or bugfix in a new branch. This simplifies the task of code review as well as the task of merging your changes into the canonical repository.
A typical workflow will then consist of the following:
- Create a new local branch based off the appropriate release branch.
- Switch to your new local branch.
(This step can be combined with the previous step with the use of
git switch -c {new branch} {original branch}
, or, if the original branch is the current one,git switch -c {new branch}
.) - Do some work, commit, repeat as necessary.
- Push the local branch to your remote repository.
- Send a pull request.
The mechanics of this process are actually quite trivial. Below, we will create a branch for fixing an issue in the tracker.
$ git switch -c hotfix/9295
Switched to a new branch 'hotfix/9295'
... do some work ...
$ git commit -s
See the section on commit signoffs below for more details on the
-s
option togit commit
and why we require it.
... write your log message ...
$ git push fork hotfix/9295:hotfix/9295
Counting objects: 38, done.
Delta compression using up to 2 threads.
Compression objects: 100% (18/18), done.
Writing objects: 100% (20/20), 8.19KiB, done.
Total 20 (delta 12), reused 0 (delta 0)
To ssh://[email protected]/{username}/laminas-zendframework-bridge.git
b5583aa..4f51698 HEAD -> hotfix/9295
To send a pull request, you have several options.
If using GitHub, you can do the pull request from there. Navigate to your repository, select the branch you just created, and then select the "Pull Request" button in the upper right. Select the user/organization "laminas" (or whatever the upstream organization is) as the recipient.
You can also perform the same steps via the GitHub CLI tool.
Execute gh pr create
, and step through the dialog to create the pull request.
If the branch you will submit against is not the default branch, use the -B {branch}
option to specify the branch to create the patch against.
We distinguish assertions in two categories:
- Expectations of something that is always supposed to be true
- Expectations of some input to follow some rule
The former is a development-only concern, aiming to aid tools (e.g. IDEs, static analysis) to perform a more accurate type detection.
In this scenario, you should use the assert()
function, which executes the expression when zend.assertions
is enabled.
That will make sure that the condition is valid, while easing future refactoring.
The latter is something that must be executed regardless of configuration or environment.
These expectations should be configured using the class Webmozart\Assert\Assert
, which provides an extensive list of helpful methods for input validation.
Please make sure to add a helpful message to be used when the assertion fails.
Example:
<?php
declare(strict_types=true);
namespace Laminas\Examples;
use Psr\Container\ContainerInterface;
use Webmozart\Assert\Assert;
$container = require __DIR__ . '/../config/di-container.php';
// we know that the file above ALWAYS returns an instance of this type
\assert($container instanceof ContainerInterface);
// now we can call `ContainerInterface#get()`, which will be understood by psalm and even autocompleted by PHPStorm
$app = $container->get(MyApp::class);
// we cannot always rely that the `MyApp::class` service is indeed an instance of that class, hence an assertion
// that guards it despite the environment
Assert::isInstanceOf(
$app,
MyApp::class,
'It looks like the DI container is misconfigured, please verify the service ' . MyApp::class
);
$app->run();
Which branch should you issue a pull request against?
- For fixes against the stable release, issue the pull request against the release branch matching the minor release you want to fix.
- For new features, or fixes that introduce new elements to the public API (such as new public methods or properties), issue the pull request against the default branch.
As you might imagine, if you are a frequent contributor, you'll start to get a ton of branches both locally and on your remote.
Once you know that your changes have been accepted to the canonical repository, we suggest doing some cleanup of these branches.
-
Local branch cleanup
$ git branch -d <branchname>
-
Remote branch removal
$ git push fork :<branchname>
The documentation for each repository is available in the "docs/" directory of that repository, is written in Markdown format, and is built using MkDocs. To learn more about how to contribute to the documentation for a repository, including how to setup the documentation locally, please refer to https://github.com/laminas/documentation.
In order for us to accept your patches, you must provide a signoff in all commits; this is done by using the -s
or --signoff
option when calling git commit
.
The signoff is used to certify that you have rights to submit your patch under the project's license, and that you agree to the Developer Certificate of Origin.
You can automate adding your signoff by using a commit template that includes the line:
Signed-off-by: Your Name <[email protected]>
Put that line into a file, and then run the command:
git config commit.template {path to file}
If you want to use this template everywhere, pass the --global
option to git config
.
You can add a signoff manually when committing from the GitHub website by adding the line Signed-off-by: Your Name <[email protected]>
by itself in the commit message.
If you have forgotten to signoff your commits, you have two options. For both, the first step is determining the number of commits you have made in your patch. On Unix-like systems (Linux, BSD, OSX, WSL, etc.), this is generally accomplished via:
git log --oneline {original branch}..HEAD | wc -l
From there, choose either to only signoff, or perform an interactive rebase:
-
Signoff only: run
git rebase --signoff HEAD~{number you discovered}
-
Full interactive rebase: run
git rebase -i HEAD~{number you discovered} -x "git commit --amend --signoff --no-edit"
. This will present the standard interactive rebase screen, but include the lineexec git commit --amend --signoff --no-edit
between each commit. Leave those lines after each commit you will be keeping.
If you run into issues during a rebase operation, you can generally execute git rebase --abort
to return to the original state.
When done, execute git push -f
, specifying the correct remote and branch, in order to force-push your amendments.
Small contributions, such as fixing spelling errors, where the content is small enough to not be considered intellectual property, can be submitted without signing the contribution for the DCO.
As a rule of thumb, changes are obvious fixes if they do not introduce any new functionality or creative thinking. Assuming the change does not affect functionality, some common obvious fix examples include the following:
- Spelling / grammar fixes.
- Typo correction, white space, and formatting changes.
- Comment clean up.
- Bug fixes that change default return values or error codes stored in constants.
- Adding or changing exception messages.
- Changes to 'metadata' files like
.gitignore
,composer.json
, etc. - Moving source files from one directory to another.
Whenever you invoke the "obvious fix" rule, please say so in your commit message:
commit 87b503f5e190e13359d4abb3e2fccd04e949d8df
Author: I.M. Developer <[email protected]>
Date: Wed Apr 01 14:46:40 2020 -0500
Fix typo in the README.
Obvious fix.
Occasionally, a patch or set of changes may need to be propagated to multiple repositories/packages. As some past examples:
- Rolling out the GitHub Actions automatic-releases workflow.
- Rolling out the GitHub Actions continuous-integration workflow.
- Updating repositories to laminas-coding-standard (or a later revision of laminas-coding-standard).
In all such cases, contributors MUST get approval from the Technical Steering Committee PRIOR to submitting the patches. The reasons for this include:
- Ensuring no duplication of effort occurs.
- Ensuring the changes have been discussed and approved by the TSC.
- Ensuring there will be maintainers available to review and merge the patches.
We do not want to waste either maintainer or contributor time, so coordinating such changes, while it may seem like an extra step, can save everybody time and effort in the long run.
If you are unsure whether to start pushing patches to all repositories, please open an RFC with the Technical Steering Committee. If you need assistance doing so, ask in the #contributors channel.