2020-05-11 19:15:08 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
// {{{ License
|
|
|
|
// 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/>.
|
|
|
|
// }}}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* String formatting utilities
|
|
|
|
*
|
|
|
|
* @package GNUsocial
|
|
|
|
* @category Util
|
|
|
|
*
|
2021-02-19 23:29:43 +00:00
|
|
|
* @author Hugo Sales <hugo@hsal.es>
|
|
|
|
* @copyright 2020-2021 Free Software Foundation, Inc http://www.fsf.org
|
2020-05-11 19:15:08 +01:00
|
|
|
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace App\Util;
|
|
|
|
|
2021-04-10 22:57:00 +01:00
|
|
|
use App\Core\Log;
|
|
|
|
use App\Util\Exception\ServerException;
|
2020-05-11 19:15:08 +01:00
|
|
|
use Functional as F;
|
|
|
|
use InvalidArgumentException;
|
|
|
|
|
|
|
|
abstract class Formatting
|
|
|
|
{
|
2021-04-15 01:59:30 +01:00
|
|
|
private static ?\Twig\Environment $twig;
|
|
|
|
public static function setTwig(\Twig\Environment $twig)
|
|
|
|
{
|
|
|
|
self::$twig = $twig;
|
|
|
|
}
|
|
|
|
|
2021-04-30 22:29:42 +01:00
|
|
|
public static function twigRenderString(string $template, array $context): string
|
2021-04-15 01:59:30 +01:00
|
|
|
{
|
|
|
|
return self::$twig->createTemplate($template, null)->render($context);
|
|
|
|
}
|
|
|
|
|
2021-04-30 22:29:42 +01:00
|
|
|
public static function twigRenderFile(string $template, array $context): string
|
|
|
|
{
|
|
|
|
return self::$twig->render($template, $context);
|
|
|
|
}
|
|
|
|
|
2020-05-11 19:15:08 +01:00
|
|
|
/**
|
|
|
|
* Normalize path by converting \ to /
|
|
|
|
*
|
|
|
|
* @param string $path
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function normalizePath(string $path): string
|
|
|
|
{
|
2021-04-10 22:57:00 +01:00
|
|
|
return str_replace(['/', '\\'], ['/', '/'], $path);
|
2020-05-11 19:15:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get plugin name from it's path, or null if not a plugin
|
|
|
|
*
|
|
|
|
* @param string $path
|
|
|
|
*
|
|
|
|
* @return null|string
|
|
|
|
*/
|
2021-04-10 22:57:00 +01:00
|
|
|
public static function moduleFromPath(string $path): ?string
|
2020-05-11 19:15:08 +01:00
|
|
|
{
|
2021-04-10 22:57:00 +01:00
|
|
|
foreach (['/plugins/', '/components/'] as $mod_p) {
|
|
|
|
$module = strpos($path, $mod_p);
|
|
|
|
if ($module === false) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$cut = $module + strlen($mod_p);
|
|
|
|
$cut2 = strpos($path, '/', $cut);
|
|
|
|
if ($cut2) {
|
|
|
|
$final = substr($path, $cut, $cut2 - $cut);
|
|
|
|
} else {
|
|
|
|
// We might be running directly from the plugins dir?
|
|
|
|
// If so, there's no place to store locale info.
|
|
|
|
$m = 'The GNU social install dir seems to contain a piece named \'plugin\' or \'component\'';
|
|
|
|
Log::critical($m);
|
|
|
|
throw new ServerException($m);
|
|
|
|
}
|
|
|
|
return $final;
|
2020-05-11 19:15:08 +01:00
|
|
|
}
|
2021-04-10 22:57:00 +01:00
|
|
|
return null;
|
2020-05-11 19:15:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check whether $haystack starts with $needle
|
|
|
|
*
|
|
|
|
* @param array|string $haystack if array, check that all strings start with $needle
|
|
|
|
* @param string $needle
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
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);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check whether $haystack ends with $needle
|
|
|
|
*
|
|
|
|
* @param array|string $haystack if array, check that all strings end with $needle
|
|
|
|
* @param string $needle
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
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);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-04-28 16:00:04 +01:00
|
|
|
/**
|
|
|
|
* If $haystack starts with $needle, remove it from the beginning
|
|
|
|
*/
|
2021-04-27 21:52:12 +01:00
|
|
|
public static function removePrefix(string $haystack, string $needle)
|
|
|
|
{
|
2021-04-28 16:00:04 +01:00
|
|
|
return self::startsWith($haystack, $needle) ? substr($haystack, strlen($needle)) : $haystack;
|
2021-04-27 21:52:12 +01:00
|
|
|
}
|
|
|
|
|
2021-04-28 16:00:04 +01:00
|
|
|
/**
|
|
|
|
* If $haystack ends with $needle, remove it from the end
|
|
|
|
*/
|
2021-04-27 21:52:12 +01:00
|
|
|
public static function removeSuffix(string $haystack, string $needle)
|
|
|
|
{
|
2021-04-28 16:00:04 +01:00
|
|
|
return self::startsWith($haystack, $needle) ? substr($haystack, -strlen($needle)) : $haystack;
|
2021-04-27 21:52:12 +01:00
|
|
|
}
|
|
|
|
|
2020-05-11 19:15:08 +01:00
|
|
|
public static function camelCaseToSnakeCase(string $str): string
|
|
|
|
{
|
|
|
|
return strtolower(preg_replace('/([a-z])([A-Z])/', '$1_$2', $str));
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function snakeCaseToCamelCase(string $str): string
|
|
|
|
{
|
2020-07-22 02:57:03 +01:00
|
|
|
return implode('', F\map(preg_split('/[\b_]/', $str), F\ary('ucfirst', 1)));
|
2020-05-11 19:15:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indent $in, a string or array, $level levels
|
|
|
|
*
|
|
|
|
* @param array|string $in
|
2020-07-25 15:42:00 +01:00
|
|
|
* @param int $level How many levels of indentation
|
|
|
|
* @param int $count How many spaces per indentation
|
2020-05-11 19:15:08 +01:00
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
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,
|
2020-07-22 02:57:03 +01:00
|
|
|
F\ary(function ($s) {
|
2020-05-11 19:15:08 +01:00
|
|
|
return $s != '';
|
|
|
|
}, 1)),
|
|
|
|
function ($val) use ($indent) {
|
|
|
|
return F\concat($indent . $val);
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
throw new InvalidArgumentException('Formatting::indent\'s first parameter must be either an array or a string. Input was: ' . $in);
|
|
|
|
}
|
2020-07-25 15:42:00 +01:00
|
|
|
|
2020-07-25 18:48:44 +01:00
|
|
|
const SPLIT_BY_SPACE = ' ';
|
2021-04-10 22:57:00 +01:00
|
|
|
const JOIN_BY_SPACE = ' ';
|
2020-07-25 18:48:44 +01:00
|
|
|
const SPLIT_BY_COMMA = ', ';
|
2021-04-10 22:57:00 +01:00
|
|
|
const JOIN_BY_COMMA = ', ';
|
2020-07-27 04:44:25 +01:00
|
|
|
const SPLIT_BY_BOTH = '/[, ]/';
|
2020-07-25 18:48:44 +01:00
|
|
|
|
2020-07-25 15:42:00 +01:00
|
|
|
/**
|
|
|
|
* Convert scalars, objects implementing __toString or arrays to strings
|
|
|
|
*
|
|
|
|
* @param mixed $value
|
|
|
|
*/
|
2021-04-10 22:57:00 +01:00
|
|
|
public static function toString($value, string $join_type = self::JOIN_BY_COMMA): string
|
2020-07-25 15:42:00 +01:00
|
|
|
{
|
2021-04-10 22:57:00 +01:00
|
|
|
if (!in_array($join_type, [static::JOIN_BY_SPACE, static::JOIN_BY_COMMA])) {
|
|
|
|
throw new \Exception('Formatting::toString received invalid join option');
|
2020-07-25 18:48:44 +01:00
|
|
|
} else {
|
2021-04-10 22:57:00 +01:00
|
|
|
if (!is_array($value)) {
|
|
|
|
return (string) $value;
|
|
|
|
} else {
|
|
|
|
return implode($join_type, $value);
|
|
|
|
}
|
2020-07-25 18:48:44 +01:00
|
|
|
}
|
2020-07-25 15:42:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert a user supplied string to array and return whether the conversion was successfull
|
|
|
|
*
|
|
|
|
* @param mixed $output
|
|
|
|
*/
|
2020-07-27 04:44:25 +01:00
|
|
|
public static function toArray(string $input, &$output, string $split_type = self::SPLIT_BY_COMMA): bool
|
2020-07-25 15:42:00 +01:00
|
|
|
{
|
2021-04-10 22:57:00 +01:00
|
|
|
if (!in_array($split_type, [static::SPLIT_BY_SPACE, static::SPLIT_BY_COMMA, static::SPLIT_BY_BOTH])) {
|
|
|
|
throw new \Exception('Formatting::toArray received invalid split option');
|
|
|
|
}
|
|
|
|
if ($input == '') {
|
|
|
|
$output = [];
|
|
|
|
return true;
|
|
|
|
}
|
2020-07-25 15:42:00 +01:00
|
|
|
$matches = [];
|
2020-07-25 18:48:44 +01:00
|
|
|
if (preg_match('/^ *\[?([^,]+(, ?[^,]+)*)\]? *$/', $input, $matches)) {
|
2020-07-27 04:44:25 +01:00
|
|
|
switch ($split_type) {
|
|
|
|
case self::SPLIT_BY_BOTH:
|
|
|
|
$arr = preg_split($split_type, $matches[1], 0, PREG_SPLIT_NO_EMPTY);
|
|
|
|
break;
|
|
|
|
case self::SPLIT_BY_COMMA:
|
|
|
|
$arr = preg_split('/, ?/', $matches[1]);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
$arr = explode($split_type[0], $matches[1]);
|
|
|
|
}
|
|
|
|
$output = str_replace([' \'', '\'', ' "', '"'], '', $arr);
|
2020-07-25 18:48:44 +01:00
|
|
|
$output = F\map($output, F\ary('trim', 1));
|
2020-07-25 15:42:00 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2020-05-11 19:15:08 +01:00
|
|
|
}
|