From time-to-time, the templates in civix may change. If you want to update your module to match the newer templates, then use this procedure:
- Make sure you have a backup of your code. If you use version-control (git/svn), then you should be good to go.
- In the shell, navigate to the extension base directory. (If the extension is "org.example.myext" and it lives in "/var/www/extensions/org.example.myext", then navigate to "/var/www/extensions".)
- Re-run the "civix generate:module" command (e.g. "civix generate:module org.example.myext"). This will regenerate the *.civix.php file (e.g. "/var/www/extensions/org.example.myext/myext.civix.php").
- Compare the new code with the old code (e.g. "git diff" or "svn diff").
- Look for additional, version-specific upgrade steps (below).
Sometimes new versions introduce new hook stubs. These generally are not
mandatory. However, in civix documentation and online support, we will
assume that they have been properly configured, so it's recommended that you
update your extension's main PHP file. For example, if the main PHP file
for the extension is "/var/www/extensions/org.example.myext/myext.php", the
snippets mentioned below (adjusting myext
to match your extension).
Hook stubs are documented below as special tasks.
Sometimes new versions introduce changes to the Upgrader
classes (e.g.,
CRM_Myext_Upgrader_Base
and its child). These generally are not mandatory --
CiviCRM is largely agnostic as to how modules manage schema upgrades -- but
civix
suggests a reasonable approach to doing so, and documentation and online
support will assume this approach.
The steps for upgrading the Upgrader
are as follows:
- Make sure you have a backup of your code. If you use version-control (git/svn), then you should be good to go.
- In the shell, navigate to your extension's root directory (e.g., "/var/www/extensions/org.example.myext").
- Re-run the civix generate:upgrader command. This will regenerate the upgrader base class (e.g. "/var/www/extensions/org.example.myext/CRM/Myext/Upgrader/Base.php").
- Compare the new code with the old code (e.g. "git diff" or "svn diff").
- Look for additional, version-specific upgrade steps (below).
The PHPUnit bootstrap file (tests/phpunit/bootstrap.php
) has been updated to support autoloading of utility classes within your extensions tests
folder. To follow this revised convention, update bootstrap.php
. After the the call to eval(...);
, say:
$loader = new \Composer\Autoload\ClassLoader();
$loader->add('CRM_', __DIR__);
$loader->add('Civi\\', __DIR__);
$loader->register();
civix v17.08.1 makes corrections to the behavior of the new helpers, E::path()
and E::url()
. They are now
more consistent in that:
E::path()
andE::url()
(without arguments) both return the folder without a trailing/
.E::path($file)
andE::path($url)
(with an argument) both return the folder plus/
plus filename.
Suggestion: search your codebase for instances of E::path
or E::url
to ensure consistent path construction.
civix v17.08.0+ introduces a new helper class. You can generate it by following the "General Tasks" (above). No other changes are required.
Optionally, if you want to use this helper class, then add a line like this to your other *.php
files:
use CRM_Myextension_ExtensionUtil as E;
(See also: "General Tasks: Upgrader Class")
In version 16.10.0, hook_civicrm_postInstall was implemented in the extension's main PHP file and delegated to the Upgrader base class. If you wish to run your own code post-install, you should copy the following snippet (or something like it) into the Upgrader class (e.g. "/var/www/extensions/org.example.myext/CRM/Myext/Upgrader.php"):
/**
* Example: Work with entities usually not available during the install step.
*
* This method can be used for any post-install tasks. For example, if a step
* of your installation depends on accessing an entity that is itself
* created during the installation (e.g., a setting or a managed entity), do
* so here to avoid order of operation problems.
*/
public function postInstall() {
$customFieldId = civicrm_api3('CustomField', 'getvalue', array(
'return' => array("id"),
'name' => "customFieldCreatedViaManagedHook",
));
civicrm_api3('Setting', 'create', array(
'myWeirdFieldSetting' => array('id' => $customFieldId, 'weirdness' => 1),
));
}
Prior to v16.10.0, extension schema versions were stored in the civicrm_settings
table under the namespace org.example.myext:version
. This storage
mechanism proved problematic for multisites utilizing more than one domain (see
CRM-19252). civix
now
utilizes hook_civicrm_postInstall
and an updated Upgrader to
store schema versions in the civicrm_extension
table.
/**
* Implements hook_civicrm_postInstall().
*
* @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_postInstall
*/
function myext_civicrm_postInstall() {
_myext_civix_civicrm_postInstall();
}
Prior to civix v16.03, civix included the commands civix generate:test
and civix test
. Beginning with v16.03, civix templates now
comply with the Testapalooza PHPUnit Template. The key changes:
- Tests are now executed directly with standalone
phpunit
. There is no need forcivix test
or for Civi's embeddedphpunit
. This should enable better support for IDEs and other tools. - The code-generator creates two additional files,
phpunit.xml.dist
andtests/phpunit/bootstrap.php
. These are requirements for using standalonephpunit
. - Tests use
PHPUnit_Framework_TestCase
withCivi\Test
instead ofCiviUnitTestCase
. This gives you more control over the test environment. - You must have
cv
installed when running tests.
Given that there isn't a very large body of existing extension tests, we haven't thoroughly tested the migration path, but here are some expectations:
- The command
civix test
hasn't changed. If you used it before to run your existing tests, then you should still be able to use it now. However, it's deprecated. - The civicrm-core repo still has
tools/scripts/phpunit
. If you used it before run your existing tests, then you should still be able to use it now. - If you want to start using standalone
phpunit
, then:- You need to create
phpunit.xml.dist
andtests/phpunit/bootstrap.php
. These files will be autogenerated the next time you usecivix generate:test
. - You should update the existing tests to implement the
HeadlessInterface
, to define functionsetupHeadless()
, and to declare@group headless
. This will ensure that the headless system boots correctly when running your test. - Try creating a new test using the
legacy
template, e.g.civix generate:test --template=legacy CRM_Myextension_DummyTest
. This will generatephpunit.xml.dist
andtests/phpunit/bootstrap.php
, and it will create an example of usingCiviUnitTestCase
. - Note: Legacy tests executed this way may reset key variables (e.g.
CRM_Core_Config::singleton()
) extra times. However, the pool of existing extension tests is fairly small, so we don't expect this to have a big real-world impact.
- You need to create
Prior to v4.7, the hook for manipulating the navigation menu required that the
extension author compute a navID
and parentID
for each new menu entry, but the
common examples for doing this were error-prone. In v4.7, the navID
and parentID
may be omitted and computed automatically.
For backward compatibility, civix
provides an adapter -- simply declare the menu
item (without navID
or parentID
; as you would in v4.7) and then delegate to
the helper function for navigationMenu
.
/**
* Implements hook_civicrm_navigationMenu().
*
* @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_navigationMenu
*/
function myext_civicrm_navigationMenu(&$menu) {
_myext_civix_insert_navigation_menu($menu, NULL, array(
'label' => ts('The Page', array('domain' => 'org.example.myext')),
'name' => 'the_page',
'url' => 'civicrm/the-page',
'permission' => 'access CiviReport,access CiviContribute',
'operator' => 'OR',
'separator' => 0,
));
_myext_civix_navigationMenu($menu);
}
Civix-based modules should scan for Angular modules names in ang/*.ang.php
and auto-register them with the Civi-Angular base app (civicrm/a/#
).
/**
* Implements hook_civicrm_angularModules().
*
* Generate a list of Angular modules.
*
* Note: This hook only runs in CiviCRM 4.5+. It may
* use features only available in v4.6+.
*
* @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_caseTypes
*/
function myext_civicrm_angularModules(&$angularModules) {
_myext_civix_civicrm_angularModules($angularModules);
}
Civix-based modules should scan for any CiviCase XML files in
xml/case/*.xml
and automatically register these.
/**
* Implementation of hook_civicrm_caseTypes
*
* Generate a list of case-types
*
* Note: This hook only runs in CiviCRM 4.4+.
*
* @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_caseTypes
*/
function myext_civicrm_caseTypes(&$caseTypes) {
_myext_civix_civicrm_caseTypes($caseTypes);
}
Civix-based modules should scan for any settings files in
settings/*.setting.php
and automatically register these.
/**
* Implementation of hook_civicrm_alterSettingsFolders
*
* @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_alterSettingsFolders
*/
function myext_civicrm_alterSettingsFolders(&$metaDataFolders = NULL) {
_myext_civix_civicrm_alterSettingsFolders($metaDataFolders);
}