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

io.quarkus.devui.runtime.build.BuildMetricsDevUIController Maven / Gradle / Ivy

The newest version!
package io.quarkus.devui.runtime.build;

import java.io.IOException;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.time.Duration;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.format.DateTimeFormatter;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;

import org.jboss.logging.Logger;

import io.quarkus.devui.runtime.build.BuildMetricsDevUIController.DependencyGraph.Link;
import io.quarkus.devui.runtime.build.BuildMetricsDevUIController.DependencyGraph.Node;
import io.vertx.core.json.JsonArray;
import io.vertx.core.json.JsonObject;

public class BuildMetricsDevUIController {

    private static final Logger LOG = Logger.getLogger(BuildMetricsDevUIController.class.getName());

    private static final BuildMetricsDevUIController INSTANCE = new BuildMetricsDevUIController();

    public static BuildMetricsDevUIController get() {
        return INSTANCE;
    }

    private Path buildMetricsPath;

    private volatile Map buildStepsMetrics;

    private BuildMetricsDevUIController() {
    }

    void setBuildMetricsPath(Path buildMetricsPath) {
        this.buildMetricsPath = buildMetricsPath;
        // Reread the data after reload
        this.buildStepsMetrics = null;
    }

    Map getBuildStepsMetrics() {
        if (buildStepsMetrics != null) {
            return buildStepsMetrics;
        }

        synchronized (this) {
            if (buildStepsMetrics != null) {
                return buildStepsMetrics;
            }

            buildStepsMetrics = prepareBuildStepsMetrics();
            return buildStepsMetrics;
        }
    }

    public Map prepareBuildStepsMetrics() {
        Map metrics = new HashMap<>();
        Map stepIdToRecord = new HashMap<>();
        Map recordIdToRecord = new HashMap<>();
        Map> threadToRecords = new HashMap<>();
        long buildDuration = 0;
        LocalTime buildStarted = null;

        if (Files.isReadable(buildMetricsPath)) {
            try {
                JsonObject data = new JsonObject(Files.readString(buildMetricsPath));
                buildDuration = data.getLong("duration");
                buildStarted = LocalDateTime
                        .parse(data.getString("started"), DateTimeFormatter.ISO_LOCAL_DATE_TIME).toLocalTime();

                JsonArray records = data.getJsonArray("records");
                for (Object record : records) {
                    JsonObject recordObj = (JsonObject) record;
                    recordObj.put("encodedStepId", URLEncoder.encode(recordObj.getString("stepId"),
                            StandardCharsets.UTF_8.toString()));
                    String thread = recordObj.getString("thread");
                    stepIdToRecord.put(recordObj.getString("stepId"), recordObj);
                    recordIdToRecord.put(recordObj.getInteger("id"), recordObj);
                    List steps = threadToRecords.get(thread);
                    if (steps == null) {
                        steps = new ArrayList<>();
                        threadToRecords.put(thread, steps);
                    }
                    steps.add(recordObj);
                }

                metrics.put("records", records);
                metrics.put("items", data.getJsonArray("items"));
                metrics.put("itemsCount", data.getInteger("itemsCount"));
                metrics.put("duration", buildDuration);
            } catch (IOException e) {
                LOG.error(e);
            }
        }

        // Build dependency graphs
        Map dependencyGraphs = new HashMap<>();
        for (Map.Entry e : stepIdToRecord.entrySet()) {
            dependencyGraphs.put(e.getKey(),
                    buildDependencyGraph(e.getValue(), stepIdToRecord, recordIdToRecord).toJson());
        }
        metrics.put("dependencyGraphs", dependencyGraphs);

        // Time slots
        long slotDuration = Math.max(10, buildDuration / 100);
        List slots = new ArrayList<>();
        long currentSlot = slotDuration;
        while (currentSlot < buildDuration) {
            slots.add(currentSlot);
            currentSlot += slotDuration;
        }
        if (currentSlot != buildDuration) {
            slots.add(buildDuration);
        }
        metrics.put("slots", slots);

        Map>> threadToSlotRecords = new HashMap<>();
        DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss.SSS");

        for (Map.Entry> entry : threadToRecords.entrySet()) {
            String thread = entry.getKey();
            List records = entry.getValue();
            List> threadSlots = new ArrayList<>();

            for (Long slot : slots) {
                List slotRecords = new ArrayList<>();
                for (JsonObject record : records) {
                    LocalTime started = LocalTime.parse(record.getString("started"), formatter);
                    long startAt = Duration.between(buildStarted, started).toMillis();
                    if (startAt < slot && (slot - slotDuration) < (startAt + record.getLong("duration"))) {
                        slotRecords.add(record.getString("stepId"));
                    }
                }
                threadSlots.add(slotRecords);
            }
            threadToSlotRecords.put(thread, threadSlots);
        }
        metrics.put("threadSlotRecords", threadToSlotRecords);

        return metrics;
    }

    DependencyGraph buildDependencyGraph(JsonObject step, Map stepIdToRecord,
            Map recordIdToRecord) {
        Set nodes = new HashSet<>();
        Set links = new HashSet<>();

        addNodesDependents(step, nodes, links, step, stepIdToRecord, recordIdToRecord);
        addNodeDependencies(step, nodes, links, step, stepIdToRecord, recordIdToRecord);
        return new DependencyGraph(nodes, links);
    }

    void addNodesDependents(JsonObject root, Set nodes, Set links, JsonObject record,
            Map stepIdToRecord, Map recordIdToRecord) {
        String stepId = record.getString("stepId");
        nodes.add(new Node(stepId, record.getString("encodedStepId")));
        for (Object dependentRecordId : record.getJsonArray("dependents")) {
            int recordId = (int) dependentRecordId;
            if (recordId != record.getInteger("id")) {
                JsonObject dependentRecord = recordIdToRecord.get(recordId);
                String dependentStepId = dependentRecord.getString("stepId");
                links.add(Link.dependent(root.equals(record), dependentStepId, stepId));
                nodes.add(new Node(dependentStepId, dependentRecord.getString("encodedStepId")));
                // NOTE: we do not fetch transient dependencies yet because the UI is not ready to show so many nodes
                // if (added) {
                // addNodesDependents(root, nodes, links, dependentRecord, stepIdToRecord, recordIdToRecord);
                // }
            }
        }
    }

    void addNodeDependencies(JsonObject root, Set nodes, Set links, JsonObject record,
            Map stepIdToRecord, Map recordIdToRecord) {
        for (Map.Entry entry : stepIdToRecord.entrySet()) {
            for (Object dependentRecordId : entry.getValue().getJsonArray("dependents")) {
                int recordId = (int) dependentRecordId;
                if (record.getInteger("id") == recordId) {
                    links.add(Link.dependency(root.equals(record),
                            record.getString("stepId"), entry.getValue().getString("stepId")));
                    nodes.add(new Node(entry.getValue().getString("stepId"), entry.getValue().getString("encodedStepId")));
                    // NOTE: we do not fetch transient dependencies yet because the UI is not ready to show so many nodes
                    // if (added) {
                    // addNodeDependencies(root, nodes, links, entry.getValue(), stepIdToRecord, recordIdToRecord);
                    // }
                }
            }
        }
    }

    public static class DependencyGraph {

        public final Set nodes;
        public final Set links;

        public DependencyGraph(Set nodes, Set links) {
            this.nodes = nodes;
            this.links = links;
        }

        public JsonObject toJson() {
            JsonObject dependencyGraph = new JsonObject();
            JsonArray nodes = new JsonArray();
            JsonArray links = new JsonArray();
            for (Node node : this.nodes) {
                nodes.add(node.toJson());
            }
            for (Link link : this.links) {
                links.add(link.toJson());
            }
            dependencyGraph.put("nodes", nodes);
            dependencyGraph.put("links", links);

            return dependencyGraph;
        }

        public static class Node {

            public final String stepId;
            public final String simpleName;
            public final String encodedStepId;

            public Node(String stepId, String encodedStepId) {
                this.stepId = stepId;
                this.encodedStepId = encodedStepId;
                int lastDot = stepId.lastIndexOf('.');
                String simple = lastDot > 0 ? stepId.substring(lastDot + 1) : stepId;
                int hash = simple.indexOf('#');
                if (hash > 0) {
                    StringBuilder sb = new StringBuilder();
                    char[] chars = simple.substring(0, hash).toCharArray();
                    for (char c : chars) {
                        if (Character.isUpperCase(c)) {
                            sb.append(c);
                        }
                    }
                    simple = sb + simple.substring(hash);
                }
                this.simpleName = simple;
            }

            public JsonObject toJson() {
                JsonObject node = new JsonObject();
                node.put("stepId", stepId);
                node.put("simpleName", simpleName);
                node.put("encodedStepId", encodedStepId);
                return node;
            }

            @Override
            public int hashCode() {
                return Objects.hash(stepId);
            }

            @Override
            public boolean equals(Object obj) {
                if (this == obj) {
                    return true;
                }
                if (obj == null) {
                    return false;
                }
                if (getClass() != obj.getClass()) {
                    return false;
                }
                Node other = (Node) obj;
                return Objects.equals(stepId, other.stepId);
            }

        }

        public static class Link {

            static Link dependent(boolean direct, String source, String target) {
                return new Link(source, target, direct ? "directDependent" : "dependency");
            }

            static Link dependency(boolean direct, String source, String target) {
                return new Link(source, target, direct ? "directDependency" : "dependency");
            }

            public final String source;
            public final String target;
            public final String type;

            public Link(String source, String target, String type) {
                this.source = source;
                this.target = target;
                this.type = type;
            }

            public JsonObject toJson() {
                JsonObject link = new JsonObject();
                link.put("source", source);
                link.put("target", target);
                link.put("type", type);
                return link;
            }

            @Override
            public int hashCode() {
                return Objects.hash(source, target);
            }

            @Override
            public boolean equals(Object obj) {
                if (this == obj) {
                    return true;
                }
                if (obj == null) {
                    return false;
                }
                if (getClass() != obj.getClass()) {
                    return false;
                }
                Link other = (Link) obj;
                return Objects.equals(source, other.source) && Objects.equals(target, other.target);
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy