Code Review Quality Best Practices
This document supplements our Code of Conduct and Best Practices. It provides technical and in-depth guidelines for specific cases discussed during code quality reviews by the YMCA Website Services team. All code should be reviewed by 1-2 developers before being merged into the YMCA Website Services codebase.
General Rules
Components in YMCA Website Services (modules, themes, or other code structures) should be reusable and atomic whenever possible. Bundle all features, content types, settings, and styles to create cohesive components.
- General naming conventions
- Features module naming
- openy_${entity_type|abbr}_${entity_bundle|abbr}_${feature|optional}
- Example: openy_node_blog_feature
- openy_prgf_sc_feature -> OpenY Paragraph Simple Content (name within yml)
- openy_${entity_type|abbr}_${entity_bundle|abbr}_${feature|optional}
- Fields naming (<=20 chars)
- field_${entity_type|abbr}_${entity_bundle|abbr}_{name|abbr}
- Example: field_prgf_sc_body
- field_${entity_type|abbr}_${entity_bundle|abbr}_{name|abbr}
- All descriptions are mandatory!
- Features module naming
- Module naming conventions - Choose the name from this list, depending on the context:
- ${project_name|abbr}_${business_name|abbr} - when the code looks like legacy and has specifics that are not ready to be open-sourced.
- openy_${business_name|abbr} - when the code is ready to be extracted to an OpenY package.
- ${business_name} - when the code is so abstract that it has no connection to OpenY and is ready to be hosted on Drupal.org as an independent project.
Code Sharing
To support community reuse, the MODULE-NAME
should relate to the module’s business logic. Avoid creating modules by abstracting them out of the business context. Modules shared to Drupal.org from past projects were shareable because they represented a feature tied to a business need. For example:
- personify - module for SOAP related methods for working with Personify API
- acrypt - Asymmetric crypt algorithm
and so on.
PHP
Return Early Pattern
To improve readability in functions and methods, return early if simple conditions can be checked at the beginning of a method:
<?php
function foo($bar, $baz)
{
if ($foo) {
// logic goes here
return $calculated_value;
} else {
return null;
}
}
?>
Returning early reduces indentation and cognitive load.
<?php
function foo($bar, $baz)
{
if (!$foo) {
return null;
}
// logic goes here
return $calculated_value;
}
?>
Define Early Pattern
When a condition aims to change a variable’s value without additional logic, define the variable early and change it based on conditions, avoiding if else elseif
structures.
Before:
<?php
if ($a == 'hello') {
$text = 'Welcome to site';
}
else {
$text = 'Register please';
}
?>
After:
<?php
$text = 'Register please';
if ($a == 'hello') {
$text = 'Welcome to site';
}
?>
Null Checks with isset()
isset()
verifies if a variable is set and not null. Additional verification against NULL
is unnecessary.
Before:
<?php
...
'video' => (isset($feed['profile_media_videos']) || $feed['profile_media_videos'] != NULL) ? $feed['profile_media_videos'] : '',
...
?>
After:
<?php
...
'video' => (isset($feed['profile_media_videos'])) ? $feed['profile_media_videos'] : '',
...
?>
Dependency Injection
Use dependency injection instead of calling methods from services statically for decoupled and easier-to-upgrade code in Drupal 8+.
See the Drupal API Overview of the Dependency Injection Container and Services.
Before:
<?php
...
$node = Drupal::entityTypeManager()->getStorage('node')->load($result->getField('nid')->getValues()[0]);
...
?>
After:
<?php
...
$node = $this->entityTypeManager->getStorage('node')->load($result->getField('nid')->getValues()[0]);
...
?>
Creating Meaningful Log Messages
Write meaningful log messages with proper context to provide useful logging for site managers.
Before:
<?php
...
if($type == 'program') {
if ($feed['profile_media_videos'] != NULL || $feed['profile_media_images'] != NULL) {
\Drupal::logger('272')->notice($type);
...
?>
After:
<?php
...
if($type == 'program') {
if ($feed['profile_media_videos'] != NULL || $feed['profile_media_images'] != NULL) {
\Drupal::logger('form_import')->notice("FORM IMPORT: type is $type");
...
?>
Maintaining an Upgrade Path
Add all configuration changes to appropriate hook_update_N
functions to update existing environments. We suggest using the
Config Importer and Tools package for working with hook_update_N
.
Install Files
openy.install
in Profile
Place updates related to the distribution in general and not specific to any feature in this file.
- Enable/Disable module
- General configs
openy_*.install
in Modules
If you update a configuration for a specific feature, put the updates in the appropriate module’s install file.
Config Management
Revert Only Specific Property from Config
This is the preferred method for updating configurations, resulting in fewer conflicts when upgrading customized YMCA Website Services instances.
You can update only part of the full config with the Config Importer module.
Use the openy_upgrade_tool.param_updater
service to update a specific property in config:
- Find the module where your config lives.
- Create a new
hook_update_N
in theopeny_*.install
file. - Add the update code to that hook (for example):
<?php
$config = drupal_get_path('module', 'openy_media_image') . '/config/install/views.view.images_library.yml';
$config_importer = \Drupal::service('openy_upgrade_tool.param_updater');
$config_importer->update($config, 'views.view.images_library', 'display.default.display_options.pager');
?>
Where:
$config
variable contains the path to the config file.views.view.images_library
- config namedisplay.default.display_options.pager
- config specific property (you can set value from a nested array with variable depth)
Revert Full Configs
This method involves extensive config file manipulation and increases upgrade time.
Use the openy_upgrade_tool.importer
service to update a full config or several configs from a directory:
<?php
$config_dir = drupal_get_path('module', 'openy_media_image') . '/config/install';
$config_importer = \Drupal::service('openy_upgrade_tool.importer');
$config_importer->setDirectory($config_dir);
$config_importer->importConfigs(['views.view.images_library']);
?>
Where:
$config_dir
- path to directory with config filesviews.view.images_library
- config name
You can also update several configs from a directory:
<?php
$config_importer->importConfigs([
'views.view.images_library',
'views.view.example_view',
]);
?>