Skip to content

Commit ed59c8f

Browse files
committed
CvPlugins - Add a CiviCRM event to let extensions load plugins
1 parent f557b68 commit ed59c8f

File tree

2 files changed

+60
-2
lines changed

2 files changed

+60
-2
lines changed

lib/src/CvPlugins.php

+33-2
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,8 @@ class CvPlugins {
3333
* Ex: ['appName' => 'cv', 'appVersion' => '0.3.50']
3434
*/
3535
public function init(array $pluginEnv) {
36+
require_once __DIR__ . '/cvplugin_loader.php';
37+
3638
$this->pluginEnv = $pluginEnv;
3739
if (getenv('CV_PLUGIN_PATH')) {
3840
$this->paths = explode(PATH_SEPARATOR, getenv('CV_PLUGIN_PATH'));
@@ -68,11 +70,28 @@ public function init(array $pluginEnv) {
6870
$this->loadAll($plugins);
6971
}
7072

73+
/**
74+
* Like CvPlugins::init(), this searches for and loads plugins. This is effectively
75+
* the second phase of plugin-loading. It focuses on CiviCRM extensions
76+
* which embed extra plugins.
77+
*/
78+
public function initExtensions(): void {
79+
$plugins = [];
80+
$event = GenericHookEvent::create([
81+
'plugins' => &$plugins,
82+
'pluginEnv' => $this->pluginEnv + ['protocol' => self::PROTOCOL_VERSION],
83+
]);
84+
\Civi::dispatcher()->dispatch('civi.cv-lib.plugins', $event);
85+
86+
$this->loadAll($plugins);
87+
}
88+
7189
/**
7290
* @param array $plugins
7391
* Ex: ['helloworld' => '/etc/cv/plugin/helloworld.php']
92+
* @internal
7493
*/
75-
protected function loadAll(array $plugins): void {
94+
public function loadAll(array $plugins): void {
7695
ksort($plugins);
7796
foreach ($plugins as $pluginName => $pluginFile) {
7897
$this->load($this->pluginEnv + [
@@ -90,9 +109,21 @@ protected function loadAll(array $plugins): void {
90109
* - version: Protocol version (ex: "1")
91110
* - name: Basenemae of the plugin (eg `hello.php`)
92111
* - file: Logic filename (eg `/etc/cv/plugin/hello.php`)
112+
*
93113
* @return void
114+
* @internal
94115
*/
95-
protected function load(array $CV_PLUGIN) {
116+
public function load(array $CV_PLUGIN) {
117+
if (isset($this->plugins[$CV_PLUGIN['name']])) {
118+
if ($this->plugins[$CV_PLUGIN['name']] === $CV_PLUGIN['file']) {
119+
return;
120+
}
121+
else {
122+
fprintf(STDERR, "WARNING: Plugin %s has already been loaded from %s. Ignore duplicate %s.\n",
123+
$CV_PLUGIN['name'], $this->plugins[$CV_PLUGIN['name']], $CV_PLUGIN['file']);
124+
return;
125+
}
126+
}
96127
$this->plugins[$CV_PLUGIN['name']] = $CV_PLUGIN['file'];
97128
include $CV_PLUGIN['file'];
98129
}

lib/src/cvplugin_loader.php

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
<?php
2+
3+
use Civi\Cv\Cv;
4+
5+
if (defined('CIVICRM_DSN')) {
6+
fprintf(STDERR, "WARNING: Cv plugins initialized after CiviCRM booted. Second stage loading may not work as expected.\n");
7+
}
8+
9+
// Define the hook listener before Civi boots.
10+
$GLOBALS['CIVICRM_FORCE_MODULES'][] = 'cvplugin_loader';
11+
12+
/**
13+
* @param $config
14+
* @param array|NULL $flags
15+
* Only defined on 5.65+.
16+
* @return void
17+
* @see \CRM_Utils_Hook::config()
18+
*/
19+
function cvplugin_loader_civicrm_config($config, $flags = NULL): void {
20+
static $loaded = FALSE;
21+
if (!$loaded) {
22+
if ($flags === NULL || !empty($flags['civicrm'])) {
23+
$loaded = TRUE;
24+
Cv::plugins()->initExtensions();
25+
}
26+
}
27+
}

0 commit comments

Comments
 (0)