001/*
002 * Licensed to the Apache Software Foundation (ASF) under one
003 * or more contributor license agreements.  See the NOTICE file
004 * distributed with this work for additional information
005 * regarding copyright ownership.  The ASF licenses this file
006 * to you under the Apache License, Version 2.0 (the
007 * "License"); you may not use this file except in compliance
008 * with the License.  You may obtain a copy of the License at
009 *
010 *   http://www.apache.org/licenses/LICENSE-2.0
011 *
012 * Unless required by applicable law or agreed to in writing,
013 * software distributed under the License is distributed on an
014 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
015 * KIND, either express or implied.  See the License for the
016 * specific language governing permissions and limitations
017 * under the License.
018 */
019package org.eclipse.aether.util.graph.visitor;
020
021import java.util.ArrayList;
022import java.util.Iterator;
023import java.util.List;
024import java.util.function.Consumer;
025
026import org.eclipse.aether.artifact.Artifact;
027import org.eclipse.aether.graph.Dependency;
028import org.eclipse.aether.graph.DependencyNode;
029import org.eclipse.aether.graph.DependencyVisitor;
030import org.eclipse.aether.util.artifact.ArtifactIdUtils;
031import org.eclipse.aether.util.graph.manager.DependencyManagerUtils;
032import org.eclipse.aether.util.graph.transformer.ConflictResolver;
033
034import static java.util.Objects.requireNonNull;
035
036/**
037 * A dependency visitor that dumps the graph to any {@link Consumer}{@code <String>}. Meant for diagnostic and testing, as
038 * it may output the graph to standard output, error or even some logging interface.
039 *
040 * @since 1.9.8
041 */
042public class DependencyGraphDumper implements DependencyVisitor {
043
044    private final Consumer<String> consumer;
045
046    private final List<ChildInfo> childInfos = new ArrayList<>();
047
048    public DependencyGraphDumper(Consumer<String> consumer) {
049        this.consumer = requireNonNull(consumer);
050    }
051
052    @Override
053    public boolean visitEnter(DependencyNode node) {
054        consumer.accept(formatIndentation() + formatNode(node));
055        childInfos.add(new ChildInfo(node.getChildren().size()));
056        return true;
057    }
058
059    private String formatIndentation() {
060        StringBuilder buffer = new StringBuilder(128);
061        for (Iterator<ChildInfo> it = childInfos.iterator(); it.hasNext(); ) {
062            buffer.append(it.next().formatIndentation(!it.hasNext()));
063        }
064        return buffer.toString();
065    }
066
067    private String formatNode(DependencyNode node) {
068        StringBuilder buffer = new StringBuilder(128);
069        Artifact a = node.getArtifact();
070        Dependency d = node.getDependency();
071        buffer.append(a);
072        if (d != null && !d.getScope().isEmpty()) {
073            buffer.append(" [").append(d.getScope());
074            if (d.isOptional()) {
075                buffer.append(", optional");
076            }
077            buffer.append("]");
078        }
079        String premanaged = DependencyManagerUtils.getPremanagedVersion(node);
080        if (premanaged != null && !premanaged.equals(a.getBaseVersion())) {
081            buffer.append(" (version managed from ").append(premanaged).append(")");
082        }
083
084        premanaged = DependencyManagerUtils.getPremanagedScope(node);
085        if (premanaged != null && d != null && !premanaged.equals(d.getScope())) {
086            buffer.append(" (scope managed from ").append(premanaged).append(")");
087        }
088        DependencyNode winner = (DependencyNode) node.getData().get(ConflictResolver.NODE_DATA_WINNER);
089        if (winner != null) {
090            if (ArtifactIdUtils.equalsId(a, winner.getArtifact())) {
091                buffer.append(" (nearer exists)");
092            } else {
093                Artifact w = winner.getArtifact();
094                buffer.append(" (conflicts with ");
095                if (ArtifactIdUtils.toVersionlessId(a).equals(ArtifactIdUtils.toVersionlessId(w))) {
096                    buffer.append(w.getVersion());
097                } else {
098                    buffer.append(w);
099                }
100                buffer.append(")");
101            }
102        }
103        return buffer.toString();
104    }
105
106    @Override
107    public boolean visitLeave(DependencyNode node) {
108        if (!childInfos.isEmpty()) {
109            childInfos.remove(childInfos.size() - 1);
110        }
111        if (!childInfos.isEmpty()) {
112            childInfos.get(childInfos.size() - 1).index++;
113        }
114        return true;
115    }
116
117    private static class ChildInfo {
118
119        final int count;
120
121        int index;
122
123        ChildInfo(int count) {
124            this.count = count;
125        }
126
127        public String formatIndentation(boolean end) {
128            boolean last = index + 1 >= count;
129            if (end) {
130                return last ? "\\- " : "+- ";
131            }
132            return last ? "   " : "|  ";
133        }
134    }
135}