| 1: | <?php declare(strict_types = 1); |
| 2: | |
| 3: | namespace PHPStan\PhpDocParser\Ast\ConstExpr; |
| 4: | |
| 5: | use PHPStan\PhpDocParser\Ast\NodeAttributes; |
| 6: | use function addcslashes; |
| 7: | use function assert; |
| 8: | use function dechex; |
| 9: | use function ord; |
| 10: | use function preg_replace_callback; |
| 11: | use function sprintf; |
| 12: | use function str_pad; |
| 13: | use function strlen; |
| 14: | use const STR_PAD_LEFT; |
| 15: | |
| 16: | class ConstExprStringNode implements ConstExprNode |
| 17: | { |
| 18: | |
| 19: | public const SINGLE_QUOTED = 1; |
| 20: | public const DOUBLE_QUOTED = 2; |
| 21: | |
| 22: | use NodeAttributes; |
| 23: | |
| 24: | public string $value; |
| 25: | |
| 26: | |
| 27: | public $quoteType; |
| 28: | |
| 29: | |
| 30: | |
| 31: | |
| 32: | public function __construct(string $value, int $quoteType) |
| 33: | { |
| 34: | $this->value = $value; |
| 35: | $this->quoteType = $quoteType; |
| 36: | } |
| 37: | |
| 38: | |
| 39: | public function __toString(): string |
| 40: | { |
| 41: | if ($this->quoteType === self::SINGLE_QUOTED) { |
| 42: | |
| 43: | return sprintf("'%s'", addcslashes($this->value, '\'\\')); |
| 44: | } |
| 45: | |
| 46: | |
| 47: | return sprintf('"%s"', $this->escapeDoubleQuotedString()); |
| 48: | } |
| 49: | |
| 50: | private function escapeDoubleQuotedString(): string |
| 51: | { |
| 52: | $quote = '"'; |
| 53: | $escaped = addcslashes($this->value, "\n\r\t\f\v$" . $quote . '\\'); |
| 54: | |
| 55: | |
| 56: | |
| 57: | $regex = '/( |
| 58: | [\x00-\x08\x0E-\x1F] # Control characters |
| 59: | | [\xC0-\xC1] # Invalid UTF-8 Bytes |
| 60: | | [\xF5-\xFF] # Invalid UTF-8 Bytes |
| 61: | | \xE0(?=[\x80-\x9F]) # Overlong encoding of prior code point |
| 62: | | \xF0(?=[\x80-\x8F]) # Overlong encoding of prior code point |
| 63: | | [\xC2-\xDF](?![\x80-\xBF]) # Invalid UTF-8 Sequence Start |
| 64: | | [\xE0-\xEF](?![\x80-\xBF]{2}) # Invalid UTF-8 Sequence Start |
| 65: | | [\xF0-\xF4](?![\x80-\xBF]{3}) # Invalid UTF-8 Sequence Start |
| 66: | | (?<=[\x00-\x7F\xF5-\xFF])[\x80-\xBF] # Invalid UTF-8 Sequence Middle |
| 67: | | (?<![\xC2-\xDF]|[\xE0-\xEF]|[\xE0-\xEF][\x80-\xBF]|[\xF0-\xF4]|[\xF0-\xF4][\x80-\xBF]|[\xF0-\xF4][\x80-\xBF]{2})[\x80-\xBF] # Overlong Sequence |
| 68: | | (?<=[\xE0-\xEF])[\x80-\xBF](?![\x80-\xBF]) # Short 3 byte sequence |
| 69: | | (?<=[\xF0-\xF4])[\x80-\xBF](?![\x80-\xBF]{2}) # Short 4 byte sequence |
| 70: | | (?<=[\xF0-\xF4][\x80-\xBF])[\x80-\xBF](?![\x80-\xBF]) # Short 4 byte sequence (2) |
| 71: | )/x'; |
| 72: | return preg_replace_callback($regex, static function ($matches) { |
| 73: | assert(strlen($matches[0]) === 1); |
| 74: | $hex = dechex(ord($matches[0])); |
| 75: | |
| 76: | return '\\x' . str_pad($hex, 2, '0', STR_PAD_LEFT); |
| 77: | }, $escaped); |
| 78: | } |
| 79: | |
| 80: | } |
| 81: | |