2016-05-19 19:50:51 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace SilverStripe\Core\Manifest;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class ClassContentRemover
|
|
|
|
* @package SilverStripe\Core\Manifest
|
|
|
|
*
|
2022-07-19 18:02:20 +02:00
|
|
|
* A utility class to clean the contents of a PHP file containing classes/interfaces/traits/enums.
|
2016-05-19 19:50:51 +02:00
|
|
|
*
|
|
|
|
* It removes any code with in `$cut_off_depth` number of curly braces.
|
|
|
|
*/
|
|
|
|
class ClassContentRemover
|
|
|
|
{
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $filePath
|
|
|
|
* @param int $cutOffDepth The number of levels of curly braces to go before ignoring the content
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function remove_class_content($filePath, $cutOffDepth = 1)
|
|
|
|
{
|
|
|
|
|
|
|
|
// Use PHP's built in method to strip comments and whitespace
|
2022-04-14 03:12:59 +02:00
|
|
|
$contents = php_strip_whitespace($filePath ?? '');
|
2016-05-19 19:50:51 +02:00
|
|
|
|
2022-04-14 03:12:59 +02:00
|
|
|
if (!trim($contents ?? '')) {
|
2016-05-19 19:50:51 +02:00
|
|
|
return $contents;
|
|
|
|
}
|
|
|
|
|
2022-07-19 18:02:20 +02:00
|
|
|
if (!preg_match('/\b(?:class|interface|trait|enum)/i', $contents ?? '')) {
|
2016-05-19 19:50:51 +02:00
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
// tokenize the file contents
|
2022-04-14 03:12:59 +02:00
|
|
|
$tokens = token_get_all($contents ?? '');
|
2016-05-19 19:50:51 +02:00
|
|
|
$cleanContents = '';
|
|
|
|
$depth = 0;
|
|
|
|
$startCounting = false;
|
|
|
|
// iterate over all the tokens and only store the tokens that are outside $cutOffDepth
|
|
|
|
foreach ($tokens as $token) {
|
|
|
|
// only store the string literal of the token, that's all we need
|
|
|
|
if (!is_array($token)) {
|
|
|
|
$token = [
|
|
|
|
T_STRING,
|
|
|
|
$token,
|
|
|
|
null
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
// only count if we see a class/interface/trait keyword
|
2022-07-19 18:02:20 +02:00
|
|
|
$targetTokens = [T_CLASS, T_INTERFACE, T_TRAIT];
|
|
|
|
if (version_compare(phpversion(), '8.1.0', '>')) {
|
|
|
|
$targetTokens[] = T_ENUM;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$startCounting && in_array($token[0], $targetTokens)) {
|
2016-05-19 19:50:51 +02:00
|
|
|
$startCounting = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// use curly braces as a sign of depth
|
|
|
|
if ($token[1] === '{') {
|
|
|
|
if ($depth < $cutOffDepth) {
|
|
|
|
$cleanContents .= $token[1];
|
|
|
|
}
|
|
|
|
if ($startCounting) {
|
|
|
|
++$depth;
|
|
|
|
}
|
|
|
|
} elseif ($token[1] === '}') {
|
|
|
|
if ($startCounting) {
|
|
|
|
--$depth;
|
|
|
|
|
|
|
|
// stop counting if we've just come out of the
|
|
|
|
// class/interface/trait declaration
|
|
|
|
if ($depth <= 0) {
|
|
|
|
$startCounting = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($depth < $cutOffDepth) {
|
|
|
|
$cleanContents .= $token[1];
|
|
|
|
}
|
|
|
|
} elseif ($depth < $cutOffDepth) {
|
|
|
|
$cleanContents .= $token[1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// return cleaned class
|
2022-04-14 03:12:59 +02:00
|
|
|
return trim($cleanContents ?? '');
|
2016-05-19 19:50:51 +02:00
|
|
|
}
|
|
|
|
}
|