#tags #comments #inline #annotations #parser #php #documentation

mago-docblock

Analyzes PHP docblocks to extract annotations, tags, and documentation comments, aiding tools that rely on inline documentation

18 releases

new 0.0.19 Jan 9, 2025
0.0.18 Dec 29, 2024

#161 in Text processing

Download history 376/week @ 2024-12-04 439/week @ 2024-12-11 508/week @ 2024-12-18 331/week @ 2024-12-25 51/week @ 2025-01-01

1,475 downloads per month
Used in 4 crates (via mago-linter)

MIT/Apache

410KB
9K SLoC

Mago PHPDoc Parser

This crate provides a parser for PHPDoc comments, offering a standardized approach to interpreting PHPDoc blocks. Given the lack of a strict standard for PHPDoc formatting, we've established our own conventions to ensure consistent parsing and to facilitate tooling such as linters or documentation generators.

Table of Contents

Features

  • Parses PHPDoc comments into a structured AST (Abstract Syntax Tree).
  • Supports text, code blocks, tags, annotations, inline code, and inline tags.
  • Enforces a consistent formatting standard for PHPDoc comments.
  • Provides detailed error handling with helpful messages and suggestions.
  • Supports custom annotations with optional argument lists.

Standardization of PHPDoc Comments

Due to the absence of an official standard for PHPDoc formatting, we've established conventions to ensure consistent parsing. This standardization helps in maintaining uniform documentation and facilitates tooling.

Comment Structure

PHPDoc comments must start with /** and end with */. Comments can be single-line or multi-line.

Single-line Comments

For single-line comments:

  • There must be a space after the opening /**.
  • There must be a space before the closing */.
  • The entire content is placed on the same line.
/** This is a single-line PHPDoc comment */

Multi-line Comments

For multi-line comments:

  • Each line (except the first and last) must start with an asterisk *.
  • There must be a space after the asterisk *.
  • All lines must be indented consistently.
/**
 * This is a multi-line PHPDoc comment.
 * Each line starts with an asterisk.
 */

Indentation Rules

  • All lines after the opening /** must have the same level of indentation.
  • Inconsistent indentation levels will result in a parsing error.
  • This ensures that the comment is visually consistent and properly formatted.

Parsing Elements

The parser recognizes various elements within PHPDoc comments:

Text

Plain text within the comment is parsed and can include inline code or inline tags.

Inline Code

  • Inline code is enclosed within backticks `.
  • Can be enclosed with one or more backticks (e.g., `code`, ``code`` ).
  • There must be whitespace before the opening backtick unless it's at the start of a line.
Example
/**
 * This is an example of inline code: `echo "Hello, World!";`.
 */

Inline Tags

  • Inline tags are enclosed within {@ and }.
  • Used to reference other elements or provide metadata.
  • There must be whitespace before the opening {@ unless it's at the start of a line.
Example
/**
 * For more information, see {@see \Some\Class}.
 */

Code Blocks

The parser supports two types of code blocks: fenced code blocks and indented code blocks.

Fenced Code Blocks

  • Enclosed within triple backticks ```.
  • Optionally, you can specify one or more directives after the opening backticks separated by a comma (e.g., ```php,no-run).
  • The code block ends with closing triple backticks.
Example
/**
 * This is a fenced code block:
 *
 * ```php
 * echo "Hello, World!";
 * ```
 */

Indented Code Blocks

  • Lines that are indented with spaces or tabs are treated as code blocks.
  • The indentation level must be consistent for all lines in the code block.
Example
/**
 * This is an indented code block:
 *
 *     echo "Hello, World!";
 */

Tags

Tags provide metadata and additional information about the code. They start with @ and are followed by a tag name.

Tag Syntax

  • Begins with @ followed by the tag name.
  • The tag name must only contain letters, numbers, and hyphens.
  • After the tag name, a description can follow, which can span multiple lines until an empty line or another tag/code block is encountered.
Example
/**
 * @param string $name The name of the user.
 */

Tag Parsing

  • The parser extracts the tag name.
  • Everything following the tag name is considered the description.
  • The parser does not interpret the contents of the description; it's treated as plain text.

Note: This approach simplifies parsing and delegates the responsibility of interpreting tag descriptions to other tools if necessary.

Annotations

Annotations are similar to tags but have distinct syntax and usage.

Annotation Syntax

  • Annotations start with @ followed by a name that must start with an uppercase letter, underscore _, or backslash \.
  • The name can contain letters, numbers, underscores, backslashes, or Unicode characters.
  • Optionally, annotations can have an argument list enclosed in parentheses ( and ).
  • The argument list can span multiple lines and can contain nested parentheses.
Example
/**
 * @Route("/home", name="homepage")
 */

Argument List Parsing

  • The parser keeps track of opening and closing parentheses to handle nested or multi-line argument lists.
  • The argument list is captured as a single string and is not further parsed.
  • Unclosed parentheses will result in a parsing error.

Error Handling

The parser provides detailed error messages and suggestions to help users correct formatting issues. Errors include:

  • Invalid Trivia: The comment is not recognized as a PHPDoc block.
  • Unclosed Inline Tag: An inline tag is missing a closing }.
  • Unclosed Inline Code: Inline code is missing a closing backtick `.
  • Unclosed Code Block: A code block is missing a closing delimiter ```.
  • Invalid Tag Name: The tag name contains invalid characters.
  • Invalid Annotation Name: The annotation name is invalid.
  • Unclosed Annotation Arguments: Annotation arguments are missing a closing ).
  • Inconsistent Indentation: Indentation levels are not consistent across lines.
  • Missing Asterisk: A line in a multi-line comment is missing the leading \*.
  • Missing Whitespace After Asterisk: No space after the * in a line.
  • Missing Whitespace After Opening Asterisk: No space after the opening /** in a single-line comment.
  • Missing Whitespace Before Closing Asterisk: No space before the closing */ in a single-line comment.

Each error includes:

  • A description of the problem.
  • A note explaining why it is a problem.
  • A help message suggesting how to fix it.

Usage

To use the parser, include the crate in your project and utilize the public API provided.

use mago_interner::ThreadedInterner;
use mago_span::Span;
use mago_docblock::parse_phpdoc_with_span;

const PHPDOC: &str = r#"/**
 * This is a simple description.
 *
 * @param string $name The name of the user.
 * @return void
 */
"#;

pub fn main() {
    let interner = ThreadedInterner::new();

    let span = Span::new(Position::dummy(0), Position::dummy(phpdoc.len()));

    let result = parse_phpdoc_with_span(&interner, phpdoc, span);

    match result {
        Ok(document) => {
            println!("Document: {:#?}", document);
        }
        Err(err) => {
            eprintln!("Error: {:#?}", err);
        }
    }
}

Conclusion

This parser aims to provide a consistent and reliable way to parse PHPDoc comments, adhering to a standardized format. By following the conventions outlined above, you can ensure that your PHPDoc comments are well-structured and easily parsed by tooling, enhancing code readability and maintainability.

[!IMPORTANT]

This parser focuses on the structural aspects of PHPDoc comments and does not interpret the semantics of tag descriptions or annotation arguments. The descriptions are treated as plain text, allowing other tools to process them as needed.

Dependencies

~3.5–10MB
~82K SLoC