There are a few rules to help us maintain Phing targets in Shopsys Framework.
Names of targets should be in lower-case, using a dash (-
) as a word separator.
Names of properties should be also in lower-case, using a dot (.
) to separate levels of hierarchy and a dash (-
) as a word separator.
When naming targets, it's usually best to start with a general subject and then mention the action (typically a verb) so that related commands are next to each other.
For example, a phing target used to generate DB migrations is named db-migrations-generate
.
It's similar to naming Symfony commands where you also start with a more general namespace (eg. shopsys:migrations:generate
).
When a target enforces some application environment, it should have the environment as a prefix (eg. test-db-create
or prod-warmup
).
Targets without such prefix should use the current environment.
There are two ways how to combine phing targets together: using depends="..."
attribute or the <phingcall .../>
task.
You can set a dependency between targets using the depends
property like this:
<target name="my-target" depends="first-target,second-target">
<!-- implementation -->
</target>
This will mean that before the execution of my-target
, the targets first-target
and second-target
will be executed as well.
This is also useful when you want to implement a target that runs a sequence of subtargets.
Typically, targets starting with a build-
prefix are such sequences.
On the other hand, you can use a <phingcall .../>
task inside your implementation to run a specific target:
<target name="my-target">
<!-- implementation -->
<phingcall target="sub-target"/>
<!-- implementation -->
</target>
This will execute the sub-target
task during the execution of my-target
.
All dependencies of my-target
will be called as well, even if they were already called during the execution.
Because of this, phingcall
is not suitable for implementation of sequences as the dependencies get called many times, making the build run longer.
Still, it can be very useful when you need to control the exact time when it is called, or you want to run it with some properties overwritten.
Phing targets that use application dependency injection container (i.e. targets that execute Symfony commands) must be run after composer install
is run.
This means that inside targets that include a dependency on composer-dev
or composer-prod
, these DIC-dependent tasks must not be executed before Composer is run.
When implementing a phing target inside the shopsys/framework
package, please remember that the end users may have different needs.
Try to avoid hardcoding values that can be replaced by a property. Such properties can be overwritten in a project repository without the need for overwriting the whole target.
For example, all paths to executables should be used via path.*.executable
property.
Sometimes, it can be even useful to include an argument such as <arg line="${my-target.flags}"/>
in the exec
task of your target along with an empty my-target.flags
property (defined on the root level).
This will allow the end users to provide their own flags, modifying the command's behavior without the need for overwriting the whole target.
Removing DRY violations in the phing target definitions by extracting common tasks and sequences into their own (possibly hidden) targets also helps with extensibility. Instead if having to overwrite a few targets to change the definition, in such case users could overwrite just the extracted part.
Targets in the monorepo should have the same behavior as when called from the project repository, increasing the scope to the whole monorepo.
For example, the tests-unit
target is overwritten in the monorepo's build.xml
to execute the original target via <phingcall target="shopsys-framework.unit-tests"/>
and then execute the unit tests in all packages and utils repositories.
When this is not possible, please overwrite the target and display a message instead, explaining the actual outcome to avoid misunderstandings.
When you want to rename or delete a phing target you should hide the old target and display a deprecation message instead.
When renaming, call the new target via <phingcall target="..."/>
.
<target name="old-target" hidden="true">
<phingcall target="new-target"/>
<echo level="warning" message="This phing target is deprecated since Shopsys Framework X.Y.Z, use 'new-target' instead."/>
</target>
This will allow the users to gradually get used to the new targets.
All definitions of targets should be alphabetically sorted in the build.xml
.
It helps developers to maintain them more easily and prevents merge conflicts.