2018-02-06 19:35:10 +00:00
|
|
|
<?php
|
|
|
|
/**
|
2021-03-29 08:40:20 +02:00
|
|
|
* @copyright Copyright (C) 2010-2021, the Friendica project
|
2020-02-09 15:45:36 +01:00
|
|
|
*
|
|
|
|
* @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-02-06 19:35:10 +00:00
|
|
|
*/
|
2019-04-26 09:11:01 -04:00
|
|
|
|
2018-02-06 19:35:10 +00:00
|
|
|
namespace Friendica\Core;
|
|
|
|
|
2018-07-20 08:19:26 -04:00
|
|
|
use Friendica\Database\DBA;
|
2019-12-15 22:34:11 +01:00
|
|
|
use Friendica\DI;
|
2021-10-01 13:25:00 +00:00
|
|
|
use Friendica\Model\Contact;
|
2019-03-31 21:53:08 -04:00
|
|
|
use Friendica\Util\Strings;
|
2018-02-06 19:35:10 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Some functions to handle addons
|
|
|
|
*/
|
2019-12-15 23:28:01 +01:00
|
|
|
class Addon
|
2018-02-06 19:35:10 +00: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 01:15:02 -04:00
|
|
|
* List of the names of enabled addons
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private static $addons = [];
|
|
|
|
|
2019-04-27 22:19:54 -04: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 12:20:04 -04: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);
|
|
|
|
|
2020-01-19 21:21:13 +01:00
|
|
|
if (DI::config()->get('system', 'show_unsupported_addons')
|
2019-04-21 12:20:04 -04:00
|
|
|
|| strtolower($info['status']) != 'unsupported'
|
|
|
|
|| self::isEnabled($addon)
|
|
|
|
) {
|
|
|
|
$addons[] = [$addon, (self::isEnabled($addon) ? 'on' : 'off'), $info];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $addons;
|
|
|
|
}
|
|
|
|
|
2019-04-27 22:19:54 -04: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 01:15:02 -04:00
|
|
|
/**
|
2020-01-19 06:05:23 +00:00
|
|
|
* 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 01:15:02 -04:00
|
|
|
public static function loadAddons()
|
2018-10-06 01:13:29 +02:00
|
|
|
{
|
2020-03-09 23:28:37 +00:00
|
|
|
$installed_addons = DBA::selectToArray('addon', ['name'], ['installed' => true]);
|
2020-03-10 13:13:43 +00:00
|
|
|
self::$addons = array_column($installed_addons, 'name');
|
2018-10-06 01:13:29 +02:00
|
|
|
}
|
|
|
|
|
2018-02-06 19:35:10 +00:00
|
|
|
/**
|
2020-01-19 06:05:23 +00:00
|
|
|
* uninstalls an addon.
|
2018-02-06 19:35:10 +00:00
|
|
|
*
|
|
|
|
* @param string $addon name of the addon
|
2019-01-06 16:06:53 -05:00
|
|
|
* @return void
|
|
|
|
* @throws \Exception
|
2018-02-06 19:35:10 +00:00
|
|
|
*/
|
|
|
|
public static function uninstall($addon)
|
|
|
|
{
|
2019-03-31 21:53:08 -04: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 08:19:26 -04:00
|
|
|
DBA::delete('addon', ['name' => $addon]);
|
2018-02-06 19:35:10 +00:00
|
|
|
|
|
|
|
@include_once('addon/' . $addon . '/' . $addon . '.php');
|
|
|
|
if (function_exists($addon . '_uninstall')) {
|
|
|
|
$func = $addon . '_uninstall';
|
|
|
|
$func();
|
|
|
|
}
|
2018-10-21 01:15:02 -04:00
|
|
|
|
2020-07-27 01:33:24 -04:00
|
|
|
Hook::delete(['file' => 'addon/' . $addon . '/' . $addon . '.php']);
|
2019-05-08 00:46:13 -04:00
|
|
|
|
2018-10-24 20:44:05 -04:00
|
|
|
unset(self::$addons[array_search($addon, self::$addons)]);
|
2018-02-06 19:35:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-01-19 06:05:23 +00:00
|
|
|
* installs an addon.
|
2018-02-06 19:35:10 +00:00
|
|
|
*
|
|
|
|
* @param string $addon name of the addon
|
|
|
|
* @return bool
|
2019-01-06 16:06:53 -05:00
|
|
|
* @throws \Exception
|
2018-02-06 19:35:10 +00:00
|
|
|
*/
|
|
|
|
public static function install($addon)
|
|
|
|
{
|
2019-03-31 21:53:08 -04:00
|
|
|
$addon = Strings::sanitizeFilePathItem($addon);
|
2018-02-06 19:35:10 +00:00
|
|
|
|
2021-10-09 00:11:08 -04:00
|
|
|
$addon_file_path = 'addon/' . $addon . '/' . $addon . '.php';
|
|
|
|
|
2019-03-31 21:53:08 -04:00
|
|
|
// silently fail if addon was removed of if $addon is funky
|
2021-10-09 00:11:08 -04:00
|
|
|
if (!file_exists($addon_file_path)) {
|
2018-02-06 19:35:10 +00:00
|
|
|
return false;
|
|
|
|
}
|
2019-03-31 21:53:08 -04:00
|
|
|
|
2018-12-30 21:42:56 +01:00
|
|
|
Logger::notice("Addon {addon}: {action}", ['action' => 'install', 'addon' => $addon]);
|
2021-10-09 00:11:08 -04:00
|
|
|
$t = @filemtime($addon_file_path);
|
|
|
|
@include_once($addon_file_path);
|
2018-02-06 19:35:10 +00:00
|
|
|
if (function_exists($addon . '_install')) {
|
|
|
|
$func = $addon . '_install';
|
2019-12-15 22:34:11 +01:00
|
|
|
$func(DI::app());
|
2020-11-21 23:19:03 -05:00
|
|
|
}
|
2018-02-06 19:35:10 +00:00
|
|
|
|
2020-11-21 23:19:03 -05:00
|
|
|
DBA::insert('addon', [
|
|
|
|
'name' => $addon,
|
|
|
|
'installed' => true,
|
|
|
|
'timestamp' => $t,
|
|
|
|
'plugin_admin' => function_exists($addon . '_addon_admin'),
|
|
|
|
'hidden' => file_exists('addon/' . $addon . '/.hidden')
|
|
|
|
]);
|
2019-03-31 21:53:08 -04:00
|
|
|
|
2020-11-21 23:19:03 -05:00
|
|
|
if (!self::isEnabled($addon)) {
|
|
|
|
self::$addons[] = $addon;
|
2018-02-06 19:35:10 +00:00
|
|
|
}
|
2020-11-21 23:19:03 -05:00
|
|
|
|
|
|
|
return true;
|
2018-02-06 19:35:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* reload all updated addons
|
|
|
|
*/
|
|
|
|
public static function reload()
|
|
|
|
{
|
2020-03-10 13:13:43 +00:00
|
|
|
$addons = DBA::selectToArray('addon', [], ['installed' => true]);
|
|
|
|
|
|
|
|
foreach ($addons as $addon) {
|
|
|
|
$addonname = Strings::sanitizeFilePathItem(trim($addon['name']));
|
2021-10-09 00:11:08 -04:00
|
|
|
$addon_file_path = 'addon/' . $addonname . '/' . $addonname . '.php';
|
|
|
|
if (file_exists($addon_file_path) && $addon['timestamp'] == filemtime($addon_file_path)) {
|
|
|
|
// Addon unmodified, skipping
|
2020-03-10 13:13:43 +00:00
|
|
|
continue;
|
|
|
|
}
|
2020-03-09 23:28:37 +00:00
|
|
|
|
2020-03-10 13:13:43 +00:00
|
|
|
Logger::notice("Addon {addon}: {action}", ['action' => 'reload', 'addon' => $addon['name']]);
|
2020-03-09 23:28:37 +00:00
|
|
|
|
2021-10-09 00:11:08 -04:00
|
|
|
self::uninstall($addon['name']);
|
|
|
|
self::install($addon['name']);
|
2018-02-06 19:35:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-01-19 06:05:23 +00:00
|
|
|
* Parse addon comment in search of addon infos.
|
2018-02-06 19:35:10 +00:00
|
|
|
*
|
|
|
|
* 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 16:06:53 -05:00
|
|
|
* @throws \Exception
|
2018-02-06 19:35:10 +00:00
|
|
|
*/
|
|
|
|
public static function getInfo($addon)
|
|
|
|
{
|
2019-03-31 21:53:08 -04:00
|
|
|
$addon = Strings::sanitizeFilePathItem($addon);
|
|
|
|
|
2018-02-06 19:35:10 +00:00
|
|
|
$info = [
|
|
|
|
'name' => $addon,
|
|
|
|
'description' => "",
|
|
|
|
'author' => [],
|
|
|
|
'maintainer' => [],
|
|
|
|
'version' => "",
|
|
|
|
'status' => ""
|
|
|
|
];
|
|
|
|
|
|
|
|
if (!is_file("addon/$addon/$addon.php")) {
|
|
|
|
return $info;
|
|
|
|
}
|
|
|
|
|
2021-07-27 04:57:29 +00:00
|
|
|
DI::profiler()->startRecording('file');
|
2018-02-06 19:35:10 +00:00
|
|
|
$f = file_get_contents("addon/$addon/$addon.php");
|
2021-07-27 04:57:29 +00:00
|
|
|
DI::profiler()->stopRecording();
|
2018-02-06 19:35:10 +00: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 19:35:10 +00:00
|
|
|
$type = strtolower($type);
|
|
|
|
if ($type == "author" || $type == "maintainer") {
|
|
|
|
$r = preg_match("|([^<]+)<([^>]+)>|", $v, $m);
|
|
|
|
if ($r) {
|
2021-10-01 13:25:00 +00:00
|
|
|
if (!empty($m[2]) && empty(parse_url($m[2], PHP_URL_SCHEME))) {
|
|
|
|
$contact = Contact::getByURL($m[2], false);
|
|
|
|
if (!empty($contact['url'])) {
|
|
|
|
$m[2] = $contact['url'];
|
|
|
|
}
|
|
|
|
}
|
2018-02-06 19:35:10 +00:00
|
|
|
$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 01:15:02 -04: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 07:53:16 -04:00
|
|
|
/**
|
|
|
|
* Returns a list of the enabled addon names
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
2018-10-21 01:15:02 -04:00
|
|
|
public static function getEnabledList()
|
|
|
|
{
|
|
|
|
return self::$addons;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the list of non-hidden enabled addon names
|
|
|
|
*
|
|
|
|
* @return array
|
2019-01-06 16:06:53 -05:00
|
|
|
* @throws \Exception
|
2018-10-21 01:15:02 -04: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;
|
|
|
|
}
|
2018-02-06 19:35:10 +00:00
|
|
|
}
|