All Downloads are FREE. Search and download functionalities are using the official Maven repository.

graphql.parser.CommentParser Maven / Gradle / Ivy

package graphql.parser;

import com.google.common.collect.ImmutableList;
import graphql.Internal;
import graphql.language.AbstractDescribedNode;
import graphql.language.Comment;
import graphql.language.Document;
import graphql.language.Node;
import org.antlr.v4.runtime.CommonTokenStream;
import org.antlr.v4.runtime.ParserRuleContext;
import org.antlr.v4.runtime.Token;

import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.function.Predicate;

/**
 * Contains methods for extracting {@link Comment} in various positions within and around {@link Node}s
 */
@Internal
public class CommentParser {
    private final Map, ParserRuleContext> nodeToRuleMap;
    private CommonTokenStream tokens;
    private static final int CHANNEL_COMMENTS = 2;

    public CommentParser(NodeToRuleCapturingParser.ParserContext parserContext) {
        nodeToRuleMap = parserContext.getNodeToRuleMap();
        tokens = parserContext.getTokens();
    }

    /*
     *  type MyType { # beginning of type block
     *    field( # beginning of field args block
     *      arg1: String
     *      arg2: String
     *    )
     *  }
     */
    public Optional getBeginningOfBlockComment(Node node, String prefix) {
        final ParserRuleContext ctx = nodeToRuleMap.get(node);

        final Token start = ctx.start;

        if (start != null) {
            return this.tokens.getTokens(start.getTokenIndex(), ctx.stop.getTokenIndex()).stream()
                    .filter(token -> token.getText().equals(prefix))
                    .findFirst()
                    .map(token -> tokens.getHiddenTokensToRight(token.getTokenIndex(), CHANNEL_COMMENTS))
                    .map(commentTokens -> getCommentOnChannel(commentTokens, isNotPrecededByLineBreak))
                    .flatMap(comments -> comments.stream().findFirst());
        }

        return Optional.empty();
    }

    /*
     * type MyType {
     *   a(
     *     arg1: A
     *     arg2: B
     *     # end of field args block comment
     *   ): A
     *   # end of type block comment #1
     *   # end of type block comment #2
     * }
     */
    public List getEndOfBlockComments(Node node, String blockSuffix) {
        final ParserRuleContext ctx = nodeToRuleMap.get(node);

        return searchTokenToLeft(ctx.stop, blockSuffix)
                .map(suffixToken -> tokens.getHiddenTokensToLeft(suffixToken.getTokenIndex(), CHANNEL_COMMENTS))
                .map(commentTokens -> getCommentOnChannel(commentTokens, isPrecededByLineBreak))
                .orElse(Collections.emptyList());
    }

    /*
     * type MyType {
     *   a: A # field trailing comment
     * } # type trailing comment
     */
    public Optional getTrailingComment(Node node) {
        // Only nodes that can hold descriptions can have trailing comments
        if (!(node instanceof AbstractDescribedNode)) {
            return Optional.empty();
        }
        final ParserRuleContext ctx = nodeToRuleMap.get(node);

        List rightRefChannel = this.tokens.getHiddenTokensToRight(ctx.stop.getTokenIndex(), CHANNEL_COMMENTS);

        if (rightRefChannel != null) {
            List comments = getCommentOnChannel(rightRefChannel, isNotPrecededByLineBreak);

            return comments.stream().findFirst();
        }

        return Optional.empty();
    }

    /*
     * # type leading comment #1
     * # type leading comment #2
     * type MyType {
     *   # field leading comment #1
     *   # field leading comment #2
     *   a: A
     * }
     */
    public List getLeadingComments(Node node) {
        final ParserRuleContext ctx = nodeToRuleMap.get(node);

        final Token start = ctx.start;
        if (start != null) {
            int tokPos = start.getTokenIndex();
            List leftRefChannel = this.tokens.getHiddenTokensToLeft(tokPos, CHANNEL_COMMENTS);
            if (leftRefChannel != null) {
                return getCommentOnChannel(leftRefChannel, isPrecededByLineBreak);
            }
        }

        return Collections.emptyList();
    }

    /*
     * """ Description """
     * # comment after description #1
     * # comment after description #2
     * type MyType {
     *   a: A
     * }
     */
    public List getCommentsAfterDescription(Node node) {
        // Early return if node doesn't have a description
        if (!(node instanceof AbstractDescribedNode) ||
                (node instanceof AbstractDescribedNode && ((AbstractDescribedNode) node).getDescription() == null)
        ) {
            return Collections.emptyList();
        }

        final ParserRuleContext ctx = nodeToRuleMap.get(node);

        final Token start = ctx.start;
        if (start != null) {
            List commentTokens = tokens.getHiddenTokensToRight(start.getTokenIndex(), CHANNEL_COMMENTS);

            if (commentTokens != null) {
                return getCommentOnChannel(commentTokens, isPrecededByLineBreak);
            }
        }

        return Collections.emptyList();
    }

    public Optional getCommentOnFirstLineOfDocument(Document node) {
        final ParserRuleContext ctx = nodeToRuleMap.get(node);

        final Token start = ctx.start;
        if (start != null) {
            int tokPos = start.getTokenIndex();
            List leftRefChannel = this.tokens.getHiddenTokensToLeft(tokPos, CHANNEL_COMMENTS);
            if (leftRefChannel != null) {
                List comments = getCommentOnChannel(leftRefChannel, isFirstToken.or(isPrecededOnlyBySpaces));

                return comments.stream().findFirst();
            }
        }

        return Optional.empty();
    }

    public List getCommentsAfterAllDefinitions(Document node) {
        final ParserRuleContext ctx = nodeToRuleMap.get(node);

        final Token start = ctx.start;
        if (start != null) {
            List leftRefChannel = this.tokens.getHiddenTokensToRight(ctx.stop.getTokenIndex(), CHANNEL_COMMENTS);
            if (leftRefChannel != null) {
                return getCommentOnChannel(leftRefChannel,
                        refToken -> Optional.ofNullable(this.tokens.getHiddenTokensToLeft(refToken.getTokenIndex(), -1))
                                .map(hiddenTokens -> hiddenTokens.stream().anyMatch(token -> token.getText().equals("\n")))
                                .orElse(false)
                );
            }
        }

        return Collections.emptyList();
    }

    protected List getCommentOnChannel(List refChannel, Predicate shouldIncludePredicate) {
        ImmutableList.Builder comments = ImmutableList.builder();
        for (Token refTok : refChannel) {
            String text = refTok.getText();
            // we strip the leading hash # character but we don't trim because we don't
            // know the "comment markup".  Maybe it's space sensitive, maybe it's not.  So
            // consumers can decide that
            if (text == null) {
                continue;
            }

            boolean shouldIncludeComment = shouldIncludePredicate.test(refTok);

            if (!shouldIncludeComment) {
                continue;
            }

            text = text.replaceFirst("^#", "");

            comments.add(new Comment(text, null));
        }
        return comments.build();
    }

    private Optional searchTokenToLeft(Token token, String text) {
        int i = token.getTokenIndex();

        while (i > 0) {
            Token t = tokens.get(i);
            if (t.getText().equals(text)) {
                return Optional.of(t);
            }
            i--;
        }

        return Optional.empty();
    }

    private final Predicate alwaysTrue = token -> true;

    private final Predicate isNotPrecededByLineBreak = refToken ->
            Optional.ofNullable(tokens.getHiddenTokensToLeft(refToken.getTokenIndex(), -1))
                    .map(hiddenTokens -> hiddenTokens.stream().noneMatch(token -> token.getText().equals("\n")))
                    .orElse(false);

    private final Predicate isPrecededByLineBreak = refToken ->
            Optional.ofNullable(this.tokens.getHiddenTokensToLeft(refToken.getTokenIndex(), -1))
                    .map(hiddenTokens -> hiddenTokens.stream().anyMatch(token -> token.getText().equals("\n")))
                    .orElse(false);

    private final Predicate isFirstToken = refToken -> refToken.getTokenIndex() == 0;

    private final Predicate isPrecededOnlyBySpaces = refToken ->
            Optional.ofNullable(this.tokens.getTokens(0, refToken.getTokenIndex() - 1))
                    .map(beforeTokens -> beforeTokens.stream().allMatch(token -> token.getText().equals(" ")))
                    .orElse(false);

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy