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

graphql.language.PrettyAstPrinter Maven / Gradle / Ivy

package graphql.language;

import graphql.ExperimentalApi;
import graphql.collect.ImmutableKit;
import graphql.parser.CommentParser;
import graphql.parser.NodeToRuleCapturingParser;
import graphql.parser.ParserEnvironment;

import java.util.Collections;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.function.Function;
import java.util.stream.Collectors;

import static graphql.Assert.assertTrue;
import static graphql.parser.ParserEnvironment.newParserEnvironment;

/**
 * A printer that acts as a code formatter.
 *
 * This printer will preserve pretty much all elements from the source text, even those that are not part of the AST
 * (and are thus discarded by the {@link AstPrinter}), like comments.
 *
 * @see AstPrinter
 */
@ExperimentalApi
public class PrettyAstPrinter extends AstPrinter {
    private final CommentParser commentParser;
    private final PrettyPrinterOptions options;

    public PrettyAstPrinter(NodeToRuleCapturingParser.ParserContext parserContext) {
        this(parserContext, PrettyPrinterOptions.defaultOptions);
    }

    public PrettyAstPrinter(NodeToRuleCapturingParser.ParserContext parserContext, PrettyPrinterOptions options) {
        super(false);
        this.commentParser = new CommentParser(parserContext);
        this.options = options;

        this.replacePrinter(DirectiveDefinition.class, directiveDefinition());
        this.replacePrinter(Document.class, document());
        this.replacePrinter(EnumTypeDefinition.class, enumTypeDefinition("enum"));
        this.replacePrinter(EnumTypeExtensionDefinition.class, enumTypeDefinition("extend enum"));
        this.replacePrinter(EnumValueDefinition.class, enumValueDefinition());
        this.replacePrinter(FieldDefinition.class, fieldDefinition());
        this.replacePrinter(InputObjectTypeDefinition.class, inputObjectTypeDefinition("input"));
        this.replacePrinter(InputObjectTypeExtensionDefinition.class, inputObjectTypeDefinition("extend input"));
        this.replacePrinter(InputValueDefinition.class, inputValueDefinition());
        this.replacePrinter(InterfaceTypeDefinition.class, implementingTypeDefinition("interface"));
        this.replacePrinter(InterfaceTypeExtensionDefinition.class, implementingTypeDefinition("extend interface"));
        this.replacePrinter(ObjectTypeDefinition.class, implementingTypeDefinition("type"));
        this.replacePrinter(ObjectTypeExtensionDefinition.class, implementingTypeDefinition("extend type"));
        this.replacePrinter(ScalarTypeDefinition.class, scalarTypeDefinition("scalar"));
        this.replacePrinter(ScalarTypeExtensionDefinition.class, scalarTypeDefinition("extend scalar"));
        this.replacePrinter(UnionTypeDefinition.class, unionTypeDefinition("union"));
        this.replacePrinter(UnionTypeExtensionDefinition.class, unionTypeDefinition("extend union"));
    }

    public String print(Node node) {
        StringBuilder builder = new StringBuilder();

        NodePrinter nodePrinter = this._findPrinter(node);
        nodePrinter.print(builder, node);

        return builder.toString();
    }

    public static String print(String schemaDefinition, PrettyPrinterOptions options) {
        NodeToRuleCapturingParser parser = new NodeToRuleCapturingParser();
        ParserEnvironment parserEnvironment = newParserEnvironment().document(schemaDefinition).build();
        Document document = parser.parseDocument(parserEnvironment);

        return new PrettyAstPrinter(parser.getParserContext(), options).print(document);
    }

    private NodePrinter document() {
        return (out, node) -> {
            String firstLineComment = commentParser.getCommentOnFirstLineOfDocument(node)
                    .map(this::comment)
                    .map(append("\n"))
                    .orElse("");

            out.append(firstLineComment);
            out.append(join(node.getDefinitions(), "\n\n")).append("\n");

            String endComments = comments(commentParser.getCommentsAfterAllDefinitions(node), "\n");

            out.append(endComments);
        };
    }

    private NodePrinter directiveDefinition() {
        return (out, node) -> {
            out.append(outset(node));
            String locations = join(node.getDirectiveLocations(), " | ");
            String repeatable = node.isRepeatable() ? "repeatable " : "";
            out.append("directive @")
                    .append(node.getName())
                    .append(block(node.getInputValueDefinitions(), node, "(", ")", "\n", ", ", ""))
                    .append(" ")
                    .append(repeatable)
                    .append("on ")
                    .append(locations);
        };
    }

    private NodePrinter enumTypeDefinition(String nodeName) {
        return (out, node) -> {
            out.append(outset(node));
            out.append(spaced(
                    nodeName,
                    node.getName(),
                    directives(node.getDirectives()),
                    block(node.getEnumValueDefinitions(), node, "{", "}", "\n", null, null)
            ));
        };
    }

    private NodePrinter enumValueDefinition() {
        return (out, node) -> {
            out.append(outset(node));
            out.append(spaced(
                    node.getName(),
                    directives(node.getDirectives())
            ));
        };
    }

    private NodePrinter fieldDefinition() {
        return (out, node) -> {
            out.append(outset(node));
            out.append(node.getName())
                    .append(block(node.getInputValueDefinitions(), node, "(", ")", "\n", ", ", ""))
                    .append(": ")
                    .append(spaced(
                            type(node.getType()),
                            directives(node.getDirectives())
                    ));
        };
    }

    private String type(Type type) {
        if (type instanceof NonNullType) {
            NonNullType inner = (NonNullType) type;
            return wrap("", type(inner.getType()), "!");
        } else if (type instanceof ListType) {
            ListType inner = (ListType) type;
            return wrap("[", type(inner.getType()), "]");
        } else {
            TypeName inner = (TypeName) type;
            return inner.getName();
        }
    }

    private NodePrinter inputObjectTypeDefinition(String nodeName) {
        return (out, node) -> {
            out.append(outset(node));
            out.append(spaced(
                    nodeName,
                    node.getName(),
                    directives(node.getDirectives()),
                    block(node.getInputValueDefinitions(), node, "{", "}", "\n", null, null)
            ));
        };
    }

    private NodePrinter inputValueDefinition() {
        String nameTypeSep = ": ";
        String defaultValueEquals = "= ";
        return (out, node) -> {
            Value defaultValue = node.getDefaultValue();
            out.append(outset(node));
            out.append(spaced(
                    node.getName() + nameTypeSep + type(node.getType()),
                    wrap(defaultValueEquals, defaultValue, ""),
                    directives(node.getDirectives())
            ));
        };
    }


    private > NodePrinter implementingTypeDefinition(String nodeName) {
        return (out, node) -> {
            out.append(outset(node));
            out.append(spaced(
                    nodeName,
                    node.getName(),
                    wrap("implements ", block(node.getImplements(), node, "", "", " &\n", " & ", ""), ""),
                    directives(node.getDirectives()),
                    block(node.getFieldDefinitions(), node, "{", "}", "\n", null, null)
            ));
        };
    }

    private NodePrinter scalarTypeDefinition(String nodeName) {
        return (out, node) -> {
            out.append(outset(node));
            out.append(spaced(
                    nodeName,
                    node.getName(),
                    directives(node.getDirectives())));
        };
    }

    private NodePrinter unionTypeDefinition(String nodeName) {
        String barSep = " | ";
        String equals = "= ";
        return (out, node) -> {
            out.append(outset(node));
            out.append(spaced(
                    nodeName,
                    node.getName(),
                    directives(node.getDirectives()),
                    equals + join(node.getMemberTypes(), barSep)
            ));
        };
    }

    private String node(Node node, Class startClass) {
        if (startClass != null) {
            assertTrue(startClass.isInstance(node), () -> "The starting class must be in the inherit tree");
        }
        StringBuilder builder = new StringBuilder();

        String comments = comments(commentParser.getLeadingComments(node), "\n");
        builder.append(comments);

        NodePrinter printer = _findPrinter(node, startClass);
        printer.print(builder, node);

        commentParser.getTrailingComment(node)
                .map(this::comment)
                .map(prepend(" "))
                .ifPresent(builder::append);

        return builder.toString();
    }

    private  boolean isEmpty(List list) {
        return list == null || list.isEmpty();
    }

    private boolean isEmpty(String s) {
        return s == null || s.trim().length() == 0;
    }

    private  List nvl(List list) {
        return list != null ? list : ImmutableKit.emptyList();
    }

    // Description and comments positioned before the node
    private String outset(Node node) {
        String description = description(node);
        String commentsAfter = comments(commentParser.getCommentsAfterDescription(node), "\n");

        return description + commentsAfter;
    }

    private String description(Node node) {
        Description description = ((AbstractDescribedNode) node).getDescription();
        if (description == null || description.getContent() == null) {
            return "";
        }
        String s;
        boolean startNewLine = description.getContent().length() > 0 && description.getContent().charAt(0) == '\n';
        if (description.isMultiLine()) {
            s = "\"\"\"" + (startNewLine ? "" : "\n") + description.getContent() + "\n\"\"\"\n";
        } else {
            s = "\"" + description.getContent() + "\"\n";
        }
        return s;
    }

    private String comment(Comment comment) {
        return comments(Collections.singletonList(comment));
    }

    private String comments(List comments) {
        return comments(comments, "");
    }

    private String comments(List comments, String suffix) {
        return comments(comments, "", suffix);
    }

    private String comments(List comments, String prefix, String suffix) {
        if (comments.isEmpty()) {
            return "";
        }

        return comments.stream()
                .map(Comment::getContent)
                .map(content -> "#" + content)
                .collect(Collectors.joining("\n", prefix, suffix));
    }

    private String directives(List directives) {
        return join(nvl(directives), " ");
    }

    private  String join(List nodes, String delim) {
        return join(nodes, delim, "", "");
    }

    private  String join(List nodes, String delim, String prefix, String suffix) {
        StringBuilder joined = new StringBuilder();

        joined.append(prefix);
        boolean first = true;
        for (T node : nodes) {
            if (first) {
                first = false;
            } else {
                joined.append(delim);
            }
            joined.append(node(node));
        }

        joined.append(suffix);
        return joined.toString();
    }

    private String node(Node node) {
        return node(node, null);
    }

    private String spaced(String... args) {
        return join(" ", args);
    }

    private Function prepend(String prefix) {
        return text -> prefix + text;
    }

    private Function append(String suffix) {
        return text -> text + suffix;
    }

    private String join(String delim, String... args) {
        StringBuilder builder = new StringBuilder();

        boolean first = true;
        for (final String arg : args) {
            if (isEmpty(arg)) {
                continue;
            }
            if (first) {
                first = false;
            } else {
                builder.append(delim);
            }
            builder.append(arg);
        }

        return builder.toString();
    }

    private  String block(List nodes, Node parentNode, String prefix, String suffix, String separatorMultiline, String separatorSingleLine, String whenEmpty) {
        if (isEmpty(nodes)) {
            return whenEmpty != null ? whenEmpty : prefix + suffix;
        }

        boolean hasDescriptions = nodes.stream()
                .filter(node -> node instanceof AbstractDescribedNode)
                .map(node -> (AbstractDescribedNode) node)
                .map(AbstractDescribedNode::getDescription)
                .anyMatch(Objects::nonNull);

        boolean hasTrailingComments = nodes.stream()
                .map(commentParser::getTrailingComment)
                .anyMatch(Optional::isPresent);

        boolean hasLeadingComments = nodes.stream()
                .mapToLong(node -> commentParser.getLeadingComments(node).size())
                .sum() > 0;

        boolean isMultiline = hasDescriptions || hasTrailingComments || hasLeadingComments || separatorSingleLine == null;

        String appliedSeparator = isMultiline ? separatorMultiline : separatorSingleLine;

        String blockStart = commentParser.getBeginningOfBlockComment(parentNode, prefix)
                .map(this::comment)
                .map(commentText -> String.format("%s %s\n", prefix, commentText))
                .orElse(String.format("%s%s", prefix, (isMultiline ? "\n" : "")));

        String blockEndComments = comments(commentParser.getEndOfBlockComments(parentNode, suffix), "\n", "");
        String blockEnd = (isMultiline ? "\n" : "") + suffix;

        String content = nodes.stream().map(this::node).collect(Collectors.joining(appliedSeparator));
        String possiblyIndentedContent = isMultiline ? indent(content + blockEndComments) : content + blockEndComments;

        return blockStart + possiblyIndentedContent + blockEnd;
    }

    private String indent(String text) {
        return indent(new StringBuilder(text)).toString();
    }

    private StringBuilder indent(StringBuilder stringBuilder) {
        final String indentText = options.indentText;

        for (int i = 0; i < stringBuilder.length(); i++) {
            char c = stringBuilder.charAt(i);
            if (i == 0) {
                stringBuilder.replace(i, i, indentText);
                i += 2;
            }
            if (c == '\n') {
                stringBuilder.replace(i, i + 1, "\n" + indentText);
                i += 3;
            }
        }
        return stringBuilder;
    }

    /**
     * Contains options that modify how a document is printed.
     */
    public static class PrettyPrinterOptions {
        private final String indentText;
        private static final PrettyPrinterOptions defaultOptions = new PrettyPrinterOptions(IndentType.SPACE, 2);

        private PrettyPrinterOptions(IndentType indentType, int indentWidth) {
            this.indentText =  String.join("", Collections.nCopies(indentWidth, indentType.character));
        }

        public static PrettyPrinterOptions defaultOptions()  {
            return defaultOptions;
        }

        public static Builder builder() {
            return new Builder();
        }

        public enum IndentType {
            TAB("\t"), SPACE(" ");

            private final String character;

            IndentType(String character) {
                this.character = character;
            }
        }

        public static class Builder {
            private IndentType indentType;
            private int indentWidth = 1;

            public Builder indentType(IndentType indentType) {
                this.indentType = indentType;
                return this;
            }

            public Builder indentWith(int indentWidth) {
                this.indentWidth = indentWidth;
                return this;
            }

            public PrettyPrinterOptions build() {
                return new PrettyPrinterOptions(indentType, indentWidth);
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy