2018-02-06 20:35:10 +01:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* @file src/Core/Addon.php
|
|
|
|
*/
|
2019-04-26 15:11:01 +02:00
|
|
|
|
2018-02-06 20:35:10 +01:00
|
|
|
namespace Friendica\Core;
|
|
|
|
|
2018-07-20 14:19:26 +02:00
|
|
|
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-02-06 20:35:10 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Some functions to handle addons
|
|
|
|
*/
|
2019-12-15 23:28:01 +01:00
|
|
|
class Addon
|
2018-02-06 20:35:10 +01:00
|
|
|
{
|
2019-02-04 09:33:55 +01:00
|
|
|
/**
|
|
|
|
* The addon sub-directory
|
|
|
|
* @var string
|
|
|
|
*/
|
2019-02-05 23:36:01 +01:00
|
|
|
const DIRECTORY = 'addon';
|
2019-02-04 09:33:55 +01:00
|
|
|
|
2018-10-06 01:13:29 +02:00
|
|
|
/**
|
2018-10-21 07:15:02 +02:00
|
|
|
* List of the names of enabled addons
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private static $addons = [];
|
|
|
|
|
2019-04-28 04:19:54 +02:00
|
|
|
/**
|
|
|
|
* Returns the list of available addons with their current status and info.
|
|
|
|
* This list is made from scanning the addon/ folder.
|
|
|
|
* Unsupported addons are excluded unless they already are enabled or system.show_unsupported_addon is set.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* @throws \Exception
|
|
|
|
*/
|
2019-04-21 18:20:04 +02:00
|
|
|
public static function getAvailableList()
|
|
|
|
{
|
|
|
|
$addons = [];
|
|
|
|
$files = glob('addon/*/');
|
|
|
|
if (is_array($files)) {
|
|
|
|
foreach ($files as $file) {
|
|
|
|
if (is_dir($file)) {
|
|
|
|
list($tmp, $addon) = array_map('trim', explode('/', $file));
|
|
|
|
$info = self::getInfo($addon);
|
|
|
|
|
|
|
|
if (Config::get('system', 'show_unsupported_addons')
|
|
|
|
|| strtolower($info['status']) != 'unsupported'
|
|
|
|
|| self::isEnabled($addon)
|
|
|
|
) {
|
|
|
|
$addons[] = [$addon, (self::isEnabled($addon) ? 'on' : 'off'), $info];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $addons;
|
|
|
|
}
|
|
|
|
|
2019-04-28 04:19:54 +02:00
|
|
|
/**
|
|
|
|
* Returns a list of addons that can be configured at the node level.
|
|
|
|
* The list is formatted for display in the admin panel aside.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* @throws \Exception
|
|
|
|
*/
|
|
|
|
public static function getAdminList()
|
|
|
|
{
|
|
|
|
$addons_admin = [];
|
|
|
|
$addonsAdminStmt = DBA::select('addon', ['name'], ['plugin_admin' => 1], ['order' => ['name']]);
|
|
|
|
while ($addon = DBA::fetch($addonsAdminStmt)) {
|
|
|
|
$addons_admin[$addon['name']] = [
|
|
|
|
'url' => 'admin/addons/' . $addon['name'],
|
|
|
|
'name' => $addon['name'],
|
|
|
|
'class' => 'addon'
|
|
|
|
];
|
|
|
|
}
|
|
|
|
DBA::close($addonsAdminStmt);
|
|
|
|
|
|
|
|
return $addons_admin;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-10-21 07:15:02 +02:00
|
|
|
/**
|
|
|
|
* @brief Synchronize addons:
|
2018-10-06 01:13:29 +02:00
|
|
|
*
|
|
|
|
* system.addon contains a comma-separated list of names
|
|
|
|
* of addons which are used on this system.
|
|
|
|
* Go through the database list of already installed addons, and if we have
|
|
|
|
* an entry, but it isn't in the config list, call the uninstall procedure
|
|
|
|
* and mark it uninstalled in the database (for now we'll remove it).
|
|
|
|
* Then go through the config list and if we have a addon that isn't installed,
|
|
|
|
* call the install procedure and add it to the database.
|
|
|
|
*
|
|
|
|
*/
|
2018-10-21 07:15:02 +02:00
|
|
|
public static function loadAddons()
|
2018-10-06 01:13:29 +02:00
|
|
|
{
|
2018-10-21 07:15:02 +02:00
|
|
|
$installed_addons = [];
|
2018-10-06 01:13:29 +02:00
|
|
|
|
2018-10-06 01:21:05 +02:00
|
|
|
$r = DBA::select('addon', [], ['installed' => 1]);
|
2018-10-06 01:13:29 +02:00
|
|
|
if (DBA::isResult($r)) {
|
2018-10-21 07:15:02 +02:00
|
|
|
$installed_addons = DBA::toArray($r);
|
2018-10-06 01:13:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$addons = Config::get('system', 'addon');
|
|
|
|
$addons_arr = [];
|
|
|
|
|
|
|
|
if ($addons) {
|
|
|
|
$addons_arr = explode(',', str_replace(' ', '', $addons));
|
|
|
|
}
|
|
|
|
|
2018-10-21 07:15:02 +02:00
|
|
|
self::$addons = $addons_arr;
|
2018-10-06 01:13:29 +02:00
|
|
|
|
|
|
|
$installed_arr = [];
|
|
|
|
|
2018-10-21 07:15:02 +02:00
|
|
|
foreach ($installed_addons as $addon) {
|
|
|
|
if (!self::isEnabled($addon['name'])) {
|
|
|
|
self::uninstall($addon['name']);
|
|
|
|
} else {
|
|
|
|
$installed_arr[] = $addon['name'];
|
2018-10-06 01:13:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-21 07:15:02 +02:00
|
|
|
foreach (self::$addons as $p) {
|
|
|
|
if (!in_array($p, $installed_arr)) {
|
|
|
|
self::install($p);
|
2018-10-06 01:13:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-06 20:35:10 +01:00
|
|
|
/**
|
|
|
|
* @brief uninstalls an addon.
|
|
|
|
*
|
|
|
|
* @param string $addon name of the addon
|
2019-01-06 22:06:53 +01:00
|
|
|
* @return void
|
|
|
|
* @throws \Exception
|
2018-02-06 20:35:10 +01:00
|
|
|
*/
|
|
|
|
public static function uninstall($addon)
|
|
|
|
{
|
2019-04-01 03:53:08 +02:00
|
|
|
$addon = Strings::sanitizeFilePathItem($addon);
|
|
|
|
|
2018-12-30 21:42:56 +01:00
|
|
|
Logger::notice("Addon {addon}: {action}", ['action' => 'uninstall', 'addon' => $addon]);
|
2018-07-20 14:19:26 +02:00
|
|
|
DBA::delete('addon', ['name' => $addon]);
|
2018-02-06 20:35:10 +01:00
|
|
|
|
|
|
|
@include_once('addon/' . $addon . '/' . $addon . '.php');
|
|
|
|
if (function_exists($addon . '_uninstall')) {
|
|
|
|
$func = $addon . '_uninstall';
|
|
|
|
$func();
|
|
|
|
}
|
2018-10-21 07:15:02 +02:00
|
|
|
|
2019-05-08 06:46:13 +02:00
|
|
|
DBA::delete('hook', ['file' => 'addon/' . $addon . '/' . $addon . '.php']);
|
|
|
|
|
2018-10-25 02:44:05 +02:00
|
|
|
unset(self::$addons[array_search($addon, self::$addons)]);
|
2019-04-21 18:20:04 +02:00
|
|
|
|
|
|
|
Addon::saveEnabledList();
|
2018-02-06 20:35:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief installs an addon.
|
|
|
|
*
|
|
|
|
* @param string $addon name of the addon
|
|
|
|
* @return bool
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws \Exception
|
2018-02-06 20:35:10 +01:00
|
|
|
*/
|
|
|
|
public static function install($addon)
|
|
|
|
{
|
2019-04-01 03:53:08 +02:00
|
|
|
$addon = Strings::sanitizeFilePathItem($addon);
|
2018-02-06 20:35:10 +01:00
|
|
|
|
2019-04-01 03:53:08 +02:00
|
|
|
// silently fail if addon was removed of if $addon is funky
|
2018-02-06 20:35:10 +01:00
|
|
|
if (!file_exists('addon/' . $addon . '/' . $addon . '.php')) {
|
|
|
|
return false;
|
|
|
|
}
|
2019-04-01 03:53:08 +02:00
|
|
|
|
2018-12-30 21:42:56 +01:00
|
|
|
Logger::notice("Addon {addon}: {action}", ['action' => 'install', 'addon' => $addon]);
|
2018-02-06 20:35:10 +01:00
|
|
|
$t = @filemtime('addon/' . $addon . '/' . $addon . '.php');
|
|
|
|
@include_once('addon/' . $addon . '/' . $addon . '.php');
|
|
|
|
if (function_exists($addon . '_install')) {
|
|
|
|
$func = $addon . '_install';
|
2019-12-15 22:34:11 +01:00
|
|
|
$func(DI::app());
|
2018-02-06 20:35:10 +01:00
|
|
|
|
2018-10-21 07:15:02 +02:00
|
|
|
$addon_admin = (function_exists($addon . "_addon_admin") ? 1 : 0);
|
2018-02-06 20:35:10 +01:00
|
|
|
|
2018-07-20 14:19:26 +02:00
|
|
|
DBA::insert('addon', ['name' => $addon, 'installed' => true,
|
2018-10-21 07:15:02 +02:00
|
|
|
'timestamp' => $t, 'plugin_admin' => $addon_admin]);
|
2018-02-06 20:35:10 +01:00
|
|
|
|
|
|
|
// we can add the following with the previous SQL
|
|
|
|
// once most site tables have been updated.
|
|
|
|
// This way the system won't fall over dead during the update.
|
|
|
|
|
|
|
|
if (file_exists('addon/' . $addon . '/.hidden')) {
|
2018-07-20 14:19:26 +02:00
|
|
|
DBA::update('addon', ['hidden' => true], ['name' => $addon]);
|
2018-02-06 20:35:10 +01:00
|
|
|
}
|
2018-10-21 07:15:02 +02:00
|
|
|
|
|
|
|
if (!self::isEnabled($addon)) {
|
|
|
|
self::$addons[] = $addon;
|
|
|
|
}
|
2019-04-01 03:53:08 +02:00
|
|
|
|
2019-04-21 18:20:04 +02:00
|
|
|
Addon::saveEnabledList();
|
|
|
|
|
2018-02-06 20:35:10 +01:00
|
|
|
return true;
|
|
|
|
} else {
|
2019-04-21 18:20:04 +02:00
|
|
|
Logger::error("Addon {addon}: {action} failed", ['action' => 'install', 'addon' => $addon]);
|
2018-02-06 20:35:10 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* reload all updated addons
|
|
|
|
*/
|
|
|
|
public static function reload()
|
|
|
|
{
|
|
|
|
$addons = Config::get('system', 'addon');
|
|
|
|
if (strlen($addons)) {
|
2018-07-20 14:19:26 +02:00
|
|
|
$r = DBA::select('addon', [], ['installed' => 1]);
|
2018-07-21 14:46:04 +02:00
|
|
|
if (DBA::isResult($r)) {
|
2018-07-21 04:03:40 +02:00
|
|
|
$installed = DBA::toArray($r);
|
2018-02-06 20:35:10 +01:00
|
|
|
} else {
|
|
|
|
$installed = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
$addon_list = explode(',', $addons);
|
|
|
|
|
2019-04-01 03:53:08 +02:00
|
|
|
foreach ($addon_list as $addon) {
|
|
|
|
$addon = Strings::sanitizeFilePathItem(trim($addon));
|
|
|
|
$fname = 'addon/' . $addon . '/' . $addon . '.php';
|
|
|
|
if (file_exists($fname)) {
|
|
|
|
$t = @filemtime($fname);
|
|
|
|
foreach ($installed as $i) {
|
|
|
|
if (($i['name'] == $addon) && ($i['timestamp'] != $t)) {
|
|
|
|
|
|
|
|
Logger::notice("Addon {addon}: {action}", ['action' => 'reload', 'addon' => $i['name']]);
|
|
|
|
@include_once($fname);
|
|
|
|
|
|
|
|
if (function_exists($addon . '_uninstall')) {
|
|
|
|
$func = $addon . '_uninstall';
|
2019-12-15 22:34:11 +01:00
|
|
|
$func(DI::app());
|
2019-04-01 03:53:08 +02:00
|
|
|
}
|
|
|
|
if (function_exists($addon . '_install')) {
|
|
|
|
$func = $addon . '_install';
|
2019-12-15 22:34:11 +01:00
|
|
|
$func(DI::app());
|
2018-02-06 20:35:10 +01:00
|
|
|
}
|
2019-04-01 03:53:08 +02:00
|
|
|
DBA::update('addon', ['timestamp' => $t], ['id' => $i['id']]);
|
2018-02-06 20:35:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Parse addon comment in search of addon infos.
|
|
|
|
*
|
|
|
|
* like
|
|
|
|
* \code
|
|
|
|
* * Name: addon
|
|
|
|
* * Description: An addon which plugs in
|
|
|
|
* . * Version: 1.2.3
|
|
|
|
* * Author: John <profile url>
|
|
|
|
* * Author: Jane <email>
|
|
|
|
* * Maintainer: Jess <email>
|
|
|
|
* *
|
|
|
|
* *\endcode
|
|
|
|
* @param string $addon the name of the addon
|
|
|
|
* @return array with the addon information
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws \Exception
|
2018-02-06 20:35:10 +01:00
|
|
|
*/
|
|
|
|
public static function getInfo($addon)
|
|
|
|
{
|
2019-12-15 22:34:11 +01:00
|
|
|
$a = DI::app();
|
2018-02-06 20:35:10 +01:00
|
|
|
|
2019-04-01 03:53:08 +02:00
|
|
|
$addon = Strings::sanitizeFilePathItem($addon);
|
|
|
|
|
2018-02-06 20:35:10 +01:00
|
|
|
$info = [
|
|
|
|
'name' => $addon,
|
|
|
|
'description' => "",
|
|
|
|
'author' => [],
|
|
|
|
'maintainer' => [],
|
|
|
|
'version' => "",
|
|
|
|
'status' => ""
|
|
|
|
];
|
|
|
|
|
|
|
|
if (!is_file("addon/$addon/$addon.php")) {
|
|
|
|
return $info;
|
|
|
|
}
|
|
|
|
|
|
|
|
$stamp1 = microtime(true);
|
|
|
|
$f = file_get_contents("addon/$addon/$addon.php");
|
2019-02-16 23:17:10 +01:00
|
|
|
$a->getProfiler()->saveTimestamp($stamp1, "file", System::callstack());
|
2018-02-06 20:35:10 +01:00
|
|
|
|
|
|
|
$r = preg_match("|/\*.*\*/|msU", $f, $m);
|
|
|
|
|
|
|
|
if ($r) {
|
|
|
|
$ll = explode("\n", $m[0]);
|
|
|
|
foreach ($ll as $l) {
|
|
|
|
$l = trim($l, "\t\n\r */");
|
|
|
|
if ($l != "") {
|
2018-07-30 06:41:20 +02:00
|
|
|
$addon_info = array_map("trim", explode(":", $l, 2));
|
|
|
|
if (count($addon_info) < 2) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
list($type, $v) = $addon_info;
|
2018-02-06 20:35:10 +01:00
|
|
|
$type = strtolower($type);
|
|
|
|
if ($type == "author" || $type == "maintainer") {
|
|
|
|
$r = preg_match("|([^<]+)<([^>]+)>|", $v, $m);
|
|
|
|
if ($r) {
|
|
|
|
$info[$type][] = ['name' => $m[1], 'link' => $m[2]];
|
|
|
|
} else {
|
|
|
|
$info[$type][] = ['name' => $v];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (array_key_exists($type, $info)) {
|
|
|
|
$info[$type] = $v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $info;
|
|
|
|
}
|
2018-10-21 07:15:02 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if the provided addon is enabled
|
|
|
|
*
|
|
|
|
* @param string $addon
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public static function isEnabled($addon)
|
|
|
|
{
|
|
|
|
return in_array($addon, self::$addons);
|
|
|
|
}
|
|
|
|
|
2018-10-21 13:53:16 +02:00
|
|
|
/**
|
|
|
|
* Returns a list of the enabled addon names
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
2018-10-21 07:15:02 +02:00
|
|
|
public static function getEnabledList()
|
|
|
|
{
|
|
|
|
return self::$addons;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Saves the current enabled addon list in the system.addon config key
|
|
|
|
*
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public static function saveEnabledList()
|
|
|
|
{
|
2019-04-26 15:11:01 +02:00
|
|
|
return Config::set('system', 'addon', implode(',', self::$addons));
|
2018-10-21 07:15:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the list of non-hidden enabled addon names
|
|
|
|
*
|
|
|
|
* @return array
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws \Exception
|
2018-10-21 07:15:02 +02:00
|
|
|
*/
|
|
|
|
public static function getVisibleList()
|
|
|
|
{
|
|
|
|
$visible_addons = [];
|
|
|
|
$stmt = DBA::select('addon', ['name'], ['hidden' => false, 'installed' => true]);
|
|
|
|
if (DBA::isResult($stmt)) {
|
|
|
|
foreach (DBA::toArray($stmt) as $addon) {
|
|
|
|
$visible_addons[] = $addon['name'];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $visible_addons;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shim of Hook::register left for backward compatibility purpose.
|
|
|
|
*
|
2019-01-06 22:06:53 +01:00
|
|
|
* @see Hook::register
|
2018-10-21 07:15:02 +02:00
|
|
|
* @deprecated since version 2018.12
|
|
|
|
* @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:15:02 +02:00
|
|
|
*/
|
|
|
|
public static function registerHook($hook, $file, $function, $priority = 0)
|
|
|
|
{
|
|
|
|
return Hook::register($hook, $file, $function, $priority);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shim of Hook::unregister left for backward compatibility purpose.
|
|
|
|
*
|
2019-01-06 22:06:53 +01:00
|
|
|
* @see Hook::unregister
|
2018-10-21 07:15:02 +02:00
|
|
|
* @deprecated since version 2018.12
|
|
|
|
* @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:15:02 +02:00
|
|
|
*/
|
|
|
|
public static function unregisterHook($hook, $file, $function)
|
|
|
|
{
|
|
|
|
return Hook::unregister($hook, $file, $function);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shim of Hook::callAll left for backward-compatibility purpose.
|
|
|
|
*
|
2019-01-06 22:06:53 +01:00
|
|
|
* @see Hook::callAll
|
2018-10-21 07:15:02 +02:00
|
|
|
* @deprecated since version 2018.12
|
2019-01-06 22:06:53 +01:00
|
|
|
* @param string $name of the hook to call
|
2018-10-21 07:15:02 +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:15:02 +02:00
|
|
|
*/
|
|
|
|
public static function callHooks($name, &$data = null)
|
|
|
|
{
|
|
|
|
Hook::callAll($name, $data);
|
|
|
|
}
|
2018-02-06 20:35:10 +01:00
|
|
|
}
|