>} * @psalm-suppress PossiblyUnusedMethod * * @deprecated use parsePreservingLength instead * * @psalm-pure */ public static function parse($docblock, $line_number = null, $preserve_format = false) { // Strip off comments. $docblock = trim($docblock); $docblock = preg_replace('@^/\*\*@', '', $docblock); $docblock = preg_replace('@\*/$@', '', $docblock); $docblock = preg_replace('@^[ \t]*\*@m', '', $docblock); // Normalize multi-line @specials. $lines = explode("\n", $docblock); $line_map = []; $last = false; foreach ($lines as $k => $line) { if (preg_match('/^\s?@\w/i', $line)) { $last = $k; } elseif (preg_match('/^\s*$/', $line)) { $last = false; } elseif ($last !== false) { $old_last_line = $lines[$last]; $lines[$last] = rtrim($old_last_line) . ($preserve_format || trim($old_last_line) === '@return' ? "\n" . $line : ' ' . trim($line)); if ($line_number) { $old_line_number = $line_map[$old_last_line]; unset($line_map[$old_last_line]); $line_map[$lines[$last]] = $old_line_number; } unset($lines[$k]); } if ($line_number) { $line_map[$line] = $line_number++; } } $special = []; if ($preserve_format) { foreach ($lines as $m => $line) { if (preg_match('/^\s?@([\w\-:]+)[\t ]*(.*)$/sm', $line, $matches)) { list($full_match, $type, $data) = $matches; $docblock = str_replace($full_match, '', $docblock); if (empty($special[$type])) { $special[$type] = []; } $line_number = $line_map && isset($line_map[$full_match]) ? $line_map[$full_match] : (int)$m; $special[$type][$line_number] = rtrim($data); } } } else { $docblock = implode("\n", $lines); // Parse @specials. if (preg_match_all('/^\s?@([\w\-:]+)[\t ]*([^\n]*)/m', $docblock, $matches, PREG_SET_ORDER)) { $docblock = preg_replace('/^\s?@([\w\-:]+)\s*([^\n]*)/m', '', $docblock); /** @var string[] $match */ foreach ($matches as $m => $match) { list($_, $type, $data) = $match; if (empty($special[$type])) { $special[$type] = []; } $line_number = $line_map && isset($line_map[$_]) ? $line_map[$_] : (int)$m; $special[$type][$line_number] = $data; } } } $docblock = str_replace("\t", ' ', $docblock); // Smush the whole docblock to the left edge. $min_indent = 80; $indent = 0; foreach (array_filter(explode("\n", $docblock)) as $line) { for ($ii = 0; $ii < strlen($line); ++$ii) { if ($line[$ii] != ' ') { break; } ++$indent; } $min_indent = min($indent, $min_indent); } $docblock = preg_replace('/^' . str_repeat(' ', $min_indent) . '/m', '', $docblock); $docblock = rtrim($docblock); // Trim any empty lines off the front, but leave the indent level if there // is one. $docblock = preg_replace('/^\s*\n/', '', $docblock); foreach ($special as $special_key => $_) { if (substr($special_key, 0, 6) === 'psalm-') { $special_key = substr($special_key, 6); if (!in_array( $special_key, self::PSALM_ANNOTATIONS, true )) { throw new DocblockParseException('Unrecognised annotation @psalm-' . $special_key); } } } return [ 'description' => $docblock, 'specials' => $special, ]; } /** * Parse a docblock comment into its parts. * * @param \PhpParser\Comment\Doc $docblock * @param bool $preserve_format */ public static function parsePreservingLength(\PhpParser\Comment\Doc $docblock) : ParsedDocblock { $parsed_docblock = \Psalm\Internal\Scanner\DocblockParser::parse($docblock->getText()); foreach ($parsed_docblock->tags as $special_key => $_) { if (substr($special_key, 0, 6) === 'psalm-') { $special_key = substr($special_key, 6); if (!in_array( $special_key, self::PSALM_ANNOTATIONS, true )) { throw new DocblockParseException('Unrecognised annotation @psalm-' . $special_key); } } } return $parsed_docblock; } }