2018-10-21 07:11:19 +02:00
|
|
|
<?php
|
|
|
|
/**
|
2020-02-09 15:45:36 +01:00
|
|
|
* @copyright Copyright (C) 2020, Friendica
|
|
|
|
*
|
|
|
|
* @license GNU AGPL version 3 or any later version
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*
|
2018-10-21 07:11:19 +02:00
|
|
|
*/
|
2020-02-09 15:45:36 +01:00
|
|
|
|
2018-10-21 07:11:19 +02:00
|
|
|
namespace Friendica\Core;
|
|
|
|
|
|
|
|
use Friendica\App;
|
|
|
|
use Friendica\Database\DBA;
|
2019-12-15 22:34:11 +01:00
|
|
|
use Friendica\DI;
|
2019-04-01 03:53:08 +02:00
|
|
|
use Friendica\Util\Strings;
|
2018-10-21 07:11:19 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Some functions to handle hooks
|
|
|
|
*/
|
2019-12-15 23:28:01 +01:00
|
|
|
class Hook
|
2018-10-21 07:11:19 +02:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Array of registered hooks
|
|
|
|
*
|
|
|
|
* Format:
|
|
|
|
* [
|
|
|
|
* ["<hook name>"] => [
|
|
|
|
* 0 => "<hook file>",
|
|
|
|
* 1 => "<hook function name>"
|
|
|
|
* ],
|
|
|
|
* ...
|
|
|
|
* ]
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private static $hooks = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Load hooks
|
|
|
|
*/
|
|
|
|
public static function loadHooks()
|
|
|
|
{
|
|
|
|
self::$hooks = [];
|
|
|
|
$stmt = DBA::select('hook', ['hook', 'file', 'function'], [], ['order' => ['priority' => 'desc', 'file']]);
|
|
|
|
|
|
|
|
while ($hook = DBA::fetch($stmt)) {
|
2018-10-24 08:07:44 +02:00
|
|
|
self::add($hook['hook'], $hook['file'], $hook['function']);
|
2018-10-21 07:11:19 +02:00
|
|
|
}
|
|
|
|
DBA::close($stmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-01-19 07:05:23 +01:00
|
|
|
* Adds a new hook to the hooks array.
|
2018-10-24 08:07:44 +02:00
|
|
|
*
|
|
|
|
* This function is meant to be called by modules on each page load as it works after loadHooks has been called.
|
|
|
|
*
|
2019-01-06 22:06:53 +01:00
|
|
|
* @param string $hook
|
|
|
|
* @param string $file
|
|
|
|
* @param string $function
|
2018-10-24 08:07:44 +02:00
|
|
|
*/
|
|
|
|
public static function add($hook, $file, $function)
|
|
|
|
{
|
|
|
|
if (!array_key_exists($hook, self::$hooks)) {
|
|
|
|
self::$hooks[$hook] = [];
|
|
|
|
}
|
|
|
|
self::$hooks[$hook][] = [$file, $function];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-01-19 07:05:23 +01:00
|
|
|
* Registers a hook.
|
2018-10-24 08:07:44 +02:00
|
|
|
*
|
|
|
|
* This function is meant to be called once when an addon is enabled for example as it doesn't add to the current hooks.
|
2018-10-21 07:11:19 +02:00
|
|
|
*
|
|
|
|
* @param string $hook the name of the hook
|
|
|
|
* @param string $file the name of the file that hooks into
|
|
|
|
* @param string $function the name of the function that the hook will call
|
|
|
|
* @param int $priority A priority (defaults to 0)
|
|
|
|
* @return mixed|bool
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws \Friendica\Network\HTTPException\InternalServerErrorException
|
2018-10-21 07:11:19 +02:00
|
|
|
*/
|
|
|
|
public static function register($hook, $file, $function, $priority = 0)
|
|
|
|
{
|
2019-12-15 22:34:11 +01:00
|
|
|
$file = str_replace(DI::app()->getBasePath() . DIRECTORY_SEPARATOR, '', $file);
|
2018-10-21 07:11:19 +02:00
|
|
|
|
|
|
|
$condition = ['hook' => $hook, 'file' => $file, 'function' => $function];
|
|
|
|
if (DBA::exists('hook', $condition)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-07-27 07:33:24 +02:00
|
|
|
return self::insert(['hook' => $hook, 'file' => $file, 'function' => $function, 'priority' => $priority]);
|
2018-10-21 07:11:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unregisters a hook.
|
|
|
|
*
|
|
|
|
* @param string $hook the name of the hook
|
|
|
|
* @param string $file the name of the file that hooks into
|
|
|
|
* @param string $function the name of the function that the hook called
|
|
|
|
* @return boolean
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws \Friendica\Network\HTTPException\InternalServerErrorException
|
2018-10-21 07:11:19 +02:00
|
|
|
*/
|
|
|
|
public static function unregister($hook, $file, $function)
|
|
|
|
{
|
2019-12-15 22:34:11 +01:00
|
|
|
$relative_file = str_replace(DI::app()->getBasePath() . DIRECTORY_SEPARATOR, '', $file);
|
2018-10-21 07:11:19 +02:00
|
|
|
|
|
|
|
// This here is only needed for fixing a problem that existed on the develop branch
|
|
|
|
$condition = ['hook' => $hook, 'file' => $file, 'function' => $function];
|
2020-07-27 07:33:24 +02:00
|
|
|
self::delete($condition);
|
2018-10-21 07:11:19 +02:00
|
|
|
|
|
|
|
$condition = ['hook' => $hook, 'file' => $relative_file, 'function' => $function];
|
2020-07-27 07:33:24 +02:00
|
|
|
$result = self::delete($condition);
|
2018-10-21 07:11:19 +02:00
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the list of callbacks for a single hook
|
|
|
|
*
|
2018-10-21 13:53:16 +02:00
|
|
|
* @param string $name Name of the hook
|
2018-10-21 07:11:19 +02:00
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public static function getByName($name)
|
|
|
|
{
|
|
|
|
$return = [];
|
|
|
|
|
|
|
|
if (isset(self::$hooks[$name])) {
|
|
|
|
$return = self::$hooks[$name];
|
|
|
|
}
|
|
|
|
|
|
|
|
return $return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-01-19 07:05:23 +01:00
|
|
|
* Forks a hook.
|
2018-10-21 07:11:19 +02:00
|
|
|
*
|
|
|
|
* Use this function when you want to fork a hook via the worker.
|
|
|
|
*
|
2018-10-21 13:53:16 +02:00
|
|
|
* @param integer $priority of the hook
|
|
|
|
* @param string $name of the hook to call
|
|
|
|
* @param mixed $data to transmit to the callback handler
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws \Friendica\Network\HTTPException\InternalServerErrorException
|
2018-10-21 07:11:19 +02:00
|
|
|
*/
|
|
|
|
public static function fork($priority, $name, $data = null)
|
|
|
|
{
|
|
|
|
if (array_key_exists($name, self::$hooks)) {
|
|
|
|
foreach (self::$hooks[$name] as $hook) {
|
2018-11-10 16:15:46 +01:00
|
|
|
// Call a hook to check if this hook call needs to be forked
|
|
|
|
if (array_key_exists('hook_fork', self::$hooks)) {
|
2018-11-10 16:20:23 +01:00
|
|
|
$hookdata = ['name' => $name, 'data' => $data, 'execute' => true];
|
|
|
|
|
2018-11-10 16:15:46 +01:00
|
|
|
foreach (self::$hooks['hook_fork'] as $fork_hook) {
|
|
|
|
if ($hook[0] != $fork_hook[0]) {
|
|
|
|
continue;
|
|
|
|
}
|
2019-12-15 22:34:11 +01:00
|
|
|
self::callSingle(DI::app(), 'hook_fork', $fork_hook, $hookdata);
|
2018-11-10 16:15:46 +01:00
|
|
|
}
|
|
|
|
|
2018-11-10 16:20:23 +01:00
|
|
|
if (!$hookdata['execute']) {
|
|
|
|
continue;
|
|
|
|
}
|
2018-11-10 16:15:46 +01:00
|
|
|
}
|
|
|
|
|
2018-10-21 07:11:19 +02:00
|
|
|
Worker::add($priority, 'ForkHook', $name, $hook, $data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-01-19 07:05:23 +01:00
|
|
|
* Calls a hook.
|
2018-10-21 07:11:19 +02:00
|
|
|
*
|
|
|
|
* Use this function when you want to be able to allow a hook to manipulate
|
|
|
|
* the provided data.
|
|
|
|
*
|
2019-01-06 22:06:53 +01:00
|
|
|
* @param string $name of the hook to call
|
2018-10-21 07:11:19 +02:00
|
|
|
* @param string|array &$data to transmit to the callback handler
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws \Friendica\Network\HTTPException\InternalServerErrorException
|
2018-10-21 07:11:19 +02:00
|
|
|
*/
|
|
|
|
public static function callAll($name, &$data = null)
|
|
|
|
{
|
|
|
|
if (array_key_exists($name, self::$hooks)) {
|
|
|
|
foreach (self::$hooks[$name] as $hook) {
|
2019-12-15 22:34:11 +01:00
|
|
|
self::callSingle(DI::app(), $name, $hook, $data);
|
2018-10-21 07:11:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-01-19 07:05:23 +01:00
|
|
|
* Calls a single hook.
|
2018-10-21 07:11:19 +02:00
|
|
|
*
|
2019-01-06 22:06:53 +01:00
|
|
|
* @param App $a
|
|
|
|
* @param string $name of the hook to call
|
|
|
|
* @param array $hook Hook data
|
2018-10-21 07:11:19 +02:00
|
|
|
* @param string|array &$data to transmit to the callback handler
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws \Friendica\Network\HTTPException\InternalServerErrorException
|
2018-10-21 07:11:19 +02:00
|
|
|
*/
|
|
|
|
public static function callSingle(App $a, $name, $hook, &$data = null)
|
|
|
|
{
|
|
|
|
// Don't run a theme's hook if the user isn't using the theme
|
|
|
|
if (strpos($hook[0], 'view/theme/') !== false && strpos($hook[0], 'view/theme/' . $a->getCurrentTheme()) === false) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
@include_once($hook[0]);
|
|
|
|
if (function_exists($hook[1])) {
|
|
|
|
$func = $hook[1];
|
|
|
|
$func($a, $data);
|
|
|
|
} else {
|
|
|
|
// remove orphan hooks
|
|
|
|
$condition = ['hook' => $name, 'file' => $hook[0], 'function' => $hook[1]];
|
2020-07-27 07:33:24 +02:00
|
|
|
self::delete($condition, ['cascade' => false]);
|
2018-10-21 07:11:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-10-21 13:53:16 +02:00
|
|
|
* Checks if an app_menu hook exist for the provided addon name.
|
2018-10-21 07:11:19 +02:00
|
|
|
* Return true if the addon is an app
|
2018-10-21 13:53:16 +02:00
|
|
|
*
|
|
|
|
* @param string $name Name of the addon
|
|
|
|
* @return boolean
|
2018-10-21 07:11:19 +02:00
|
|
|
*/
|
|
|
|
public static function isAddonApp($name)
|
|
|
|
{
|
2019-04-01 03:53:08 +02:00
|
|
|
$name = Strings::sanitizeFilePathItem($name);
|
|
|
|
|
2018-10-21 07:11:19 +02:00
|
|
|
if (array_key_exists('app_menu', self::$hooks)) {
|
|
|
|
foreach (self::$hooks['app_menu'] as $hook) {
|
|
|
|
if ($hook[0] == 'addon/' . $name . '/' . $name . '.php') {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2020-07-27 07:33:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Deletes one or more hook records
|
|
|
|
*
|
2020-07-27 07:57:44 +02:00
|
|
|
* We have to clear the cached routerDispatchData because addons can provide routes
|
|
|
|
*
|
2020-07-27 07:33:24 +02:00
|
|
|
* @param array $condition
|
|
|
|
* @param array $options
|
|
|
|
* @return bool
|
|
|
|
* @throws \Exception
|
|
|
|
*/
|
|
|
|
public static function delete(array $condition, array $options = [])
|
|
|
|
{
|
|
|
|
$result = DBA::delete('hook', $condition, $options);
|
|
|
|
|
2020-07-27 07:57:44 +02:00
|
|
|
if ($result) {
|
|
|
|
DI::cache()->delete('routerDispatchData');
|
|
|
|
}
|
|
|
|
|
2020-07-27 07:33:24 +02:00
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Inserts a hook record
|
|
|
|
*
|
2020-07-27 07:57:44 +02:00
|
|
|
* We have to clear the cached routerDispatchData because addons can provide routes
|
|
|
|
*
|
2020-07-27 07:33:24 +02:00
|
|
|
* @param array $condition
|
|
|
|
* @return bool
|
|
|
|
* @throws \Exception
|
|
|
|
*/
|
|
|
|
private static function insert(array $condition)
|
|
|
|
{
|
|
|
|
$result = DBA::insert('hook', $condition);
|
|
|
|
|
2020-07-27 07:57:44 +02:00
|
|
|
if ($result) {
|
|
|
|
DI::cache()->delete('routerDispatchData');
|
|
|
|
}
|
|
|
|
|
2020-07-27 07:33:24 +02:00
|
|
|
return $result;
|
|
|
|
}
|
2018-10-21 07:11:19 +02:00
|
|
|
}
|