2020-03-11 20:29:08 +00:00
|
|
|
<?php
|
2020-03-15 21:21:11 +00:00
|
|
|
|
2020-03-21 20:18:05 +00:00
|
|
|
/*
|
|
|
|
* This file is part of GNU social - https://www.gnu.org/software/social
|
|
|
|
*
|
|
|
|
* GNU social 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.
|
|
|
|
*
|
|
|
|
* GNU social 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 GNU social. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Common utility functions
|
|
|
|
*
|
|
|
|
* @package GNUsocial
|
|
|
|
* @category Util
|
|
|
|
*
|
|
|
|
* @author Hugo Sales <hugo@fc.up.pt>
|
|
|
|
* @copyright 2020 Free Software Foundation, Inc http://www.fsf.org
|
|
|
|
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
|
|
|
|
*/
|
2020-03-11 20:29:08 +00:00
|
|
|
|
|
|
|
namespace App\Util;
|
|
|
|
|
2020-03-20 22:10:01 +00:00
|
|
|
use Functional as F;
|
2020-05-10 21:43:15 +01:00
|
|
|
use InvalidArgumentException;
|
2020-03-20 22:10:01 +00:00
|
|
|
use Symfony\Component\Config\Definition\Exception\Exception;
|
2020-05-10 21:43:15 +01:00
|
|
|
use const DIRECTORY_SEPARATOR;
|
2020-03-20 22:10:01 +00:00
|
|
|
|
2020-03-13 22:31:56 +00:00
|
|
|
abstract class Common
|
2020-03-11 20:29:08 +00:00
|
|
|
{
|
2020-05-10 21:43:15 +01:00
|
|
|
/**
|
|
|
|
* Access sysadmin's configuration preferences for GNU social
|
|
|
|
*
|
|
|
|
* @param string $section
|
|
|
|
* @param string $field
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2020-03-13 22:31:56 +00:00
|
|
|
public static function config(string $section, string $field)
|
|
|
|
{
|
2020-05-10 21:43:15 +01:00
|
|
|
// TODO: implement it x)
|
|
|
|
return [];
|
2020-03-13 22:31:56 +00:00
|
|
|
}
|
|
|
|
|
2020-03-21 20:18:05 +00:00
|
|
|
/**
|
|
|
|
* Normalize path by converting \ to /
|
2020-05-10 21:43:15 +01:00
|
|
|
* @param string $path
|
|
|
|
* @return string
|
2020-03-21 20:18:05 +00:00
|
|
|
*/
|
2020-03-11 20:29:08 +00:00
|
|
|
public static function normalizePath(string $path): string
|
|
|
|
{
|
2020-05-10 21:43:15 +01:00
|
|
|
if (DIRECTORY_SEPARATOR !== '/') {
|
2020-03-11 20:29:08 +00:00
|
|
|
$path = strtr($path, DIRECTORY_SEPARATOR, '/');
|
|
|
|
}
|
|
|
|
return $path;
|
|
|
|
}
|
|
|
|
|
2020-03-21 20:18:05 +00:00
|
|
|
/**
|
|
|
|
* Get plugin name from it's path, or null if not a plugin
|
2020-05-10 21:43:15 +01:00
|
|
|
* @param string $path
|
|
|
|
* @return string|null
|
2020-03-21 20:18:05 +00:00
|
|
|
*/
|
2020-03-11 20:29:08 +00:00
|
|
|
public static function pluginFromPath(string $path): ?string
|
|
|
|
{
|
|
|
|
$plug = strpos($path, '/plugins/');
|
|
|
|
if ($plug === false) {
|
|
|
|
return null;
|
2020-03-15 21:21:11 +00:00
|
|
|
}
|
|
|
|
$cut = $plug + strlen('/plugins/');
|
|
|
|
$cut2 = strpos($path, '/', $cut);
|
|
|
|
if ($cut2) {
|
|
|
|
$final = substr($path, $cut, $cut2 - $cut);
|
2020-03-11 20:29:08 +00:00
|
|
|
} else {
|
2020-03-15 21:21:11 +00:00
|
|
|
// We might be running directly from the plugins dir?
|
|
|
|
// If so, there's no place to store locale info.
|
2020-03-20 22:10:01 +00:00
|
|
|
throw new Exception('The GNU social install dir seems to contain a piece named plugin');
|
2020-03-11 20:29:08 +00:00
|
|
|
}
|
|
|
|
return $final;
|
|
|
|
}
|
2020-03-20 22:10:01 +00:00
|
|
|
|
2020-03-21 20:18:05 +00:00
|
|
|
/**
|
|
|
|
* Check whether $haystack starts with $needle
|
|
|
|
*
|
|
|
|
* @param array|string $haystack if array, check that all strings start with $needle
|
2020-05-10 21:43:15 +01:00
|
|
|
* @param string $needle
|
|
|
|
* @return bool
|
2020-03-21 20:18:05 +00:00
|
|
|
*/
|
2020-03-20 22:10:01 +00:00
|
|
|
public static function startsWith($haystack, string $needle): bool
|
|
|
|
{
|
|
|
|
if (is_string($haystack)) {
|
|
|
|
$length = strlen($needle);
|
|
|
|
return substr($haystack, 0, $length) === $needle;
|
|
|
|
}
|
|
|
|
return F\every($haystack,
|
|
|
|
function ($haystack) use ($needle) {
|
|
|
|
return self::startsWith($haystack, $needle);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-03-21 20:18:05 +00:00
|
|
|
/**
|
|
|
|
* Check whether $haystack ends with $needle
|
|
|
|
*
|
|
|
|
* @param array|string $haystack if array, check that all strings end with $needle
|
2020-05-10 21:43:15 +01:00
|
|
|
* @param string $needle
|
|
|
|
* @return bool
|
2020-03-21 20:18:05 +00:00
|
|
|
*/
|
2020-03-20 22:10:01 +00:00
|
|
|
public static function endsWith($haystack, string $needle)
|
|
|
|
{
|
|
|
|
if (is_string($haystack)) {
|
|
|
|
$length = strlen($needle);
|
|
|
|
if ($length == 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return substr($haystack, -$length) === $needle;
|
|
|
|
}
|
|
|
|
return F\every($haystack,
|
|
|
|
function ($haystack) use ($needle) {
|
|
|
|
return self::endsWith($haystack, $needle);
|
|
|
|
});
|
|
|
|
}
|
2020-03-29 19:05:48 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Call $func with only abs($count) arguments, taken either from the
|
|
|
|
* left or right depending on the sign
|
2020-05-10 21:43:15 +01:00
|
|
|
* @param callable $func
|
|
|
|
* @param int $count
|
|
|
|
* @return callable
|
2020-03-29 19:05:48 +01:00
|
|
|
*/
|
|
|
|
public static function arity(callable $func, int $count): callable
|
|
|
|
{
|
|
|
|
return function (...$args) use ($func, $count) {
|
|
|
|
if ($count > 0) {
|
|
|
|
return call_user_func_array($func, F\take_left($args, $count));
|
|
|
|
}
|
|
|
|
return call_user_func_array($func, F\take_right($args, -$count));
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-05-10 21:43:15 +01:00
|
|
|
/**
|
|
|
|
* @param string $str
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function camelCaseToSnakeCase(string $str): string
|
2020-03-29 19:05:48 +01:00
|
|
|
{
|
|
|
|
return strtolower(preg_replace('/([a-z])([A-Z])/', '$1_$2', $str));
|
|
|
|
}
|
|
|
|
|
2020-05-10 21:43:15 +01:00
|
|
|
/**
|
|
|
|
* @param string $str
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function snakeCaseToCamelCase(string $str): string
|
2020-03-29 19:05:48 +01:00
|
|
|
{
|
|
|
|
return implode('', F\map(preg_split('/[\b_]/', $str), self::arity('ucfirst', 1)));
|
|
|
|
}
|
2020-04-20 15:51:53 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Indent $in, a string or array, $level levels
|
|
|
|
*
|
|
|
|
* @param array|string $in
|
2020-05-10 21:43:15 +01:00
|
|
|
* @param int $level
|
|
|
|
* @param int $count
|
|
|
|
* @return string
|
2020-04-20 15:51:53 +01:00
|
|
|
*/
|
|
|
|
public static function indent($in, int $level = 1, int $count = 2): string
|
|
|
|
{
|
|
|
|
if (is_string($in)) {
|
|
|
|
return self::indent(explode("\n", $in), $level, $count);
|
|
|
|
} elseif (is_array($in)) {
|
|
|
|
$indent = str_repeat(' ', $count * $level);
|
|
|
|
return implode("\n", F\map(F\select($in,
|
|
|
|
self::arity(function ($s) { return $s != ''; }, 1)),
|
|
|
|
function ($val) use ($indent) {
|
|
|
|
return F\concat($indent . $val);
|
|
|
|
}));
|
|
|
|
}
|
2020-05-10 21:43:15 +01:00
|
|
|
throw new InvalidArgumentException('Common:indent first parameter must be either an array or a string. Input was: '.$in);
|
2020-04-20 15:51:53 +01:00
|
|
|
}
|
2020-03-11 20:29:08 +00:00
|
|
|
}
|